class J1 { constructor() { this.rootNodes = [], this.cameras = [], this.lights = [], this.meshes = [], this.skeletons = [], this.particleSystems = [], this.animations = [], this.animationGroups = [], this.multiMaterials = [], this.materials = [], this.morphTargetManagers = [], this.geometries = [], this.transformNodes = [], this.actionManagers = [], this.textures = [], this._environmentTexture = null, this.postProcesses = []; } /** * Adds a parser in the list of available ones * @param name Defines the name of the parser * @param parser Defines the parser to add */ static AddParser(e, t) { this._BabylonFileParsers[e] = t; } /** * Gets a general parser from the list of available ones * @param name Defines the name of the parser * @returns the requested parser or null */ static GetParser(e) { return this._BabylonFileParsers[e] ? this._BabylonFileParsers[e] : null; } /** * Adds n individual parser in the list of available ones * @param name Defines the name of the parser * @param parser Defines the parser to add */ static AddIndividualParser(e, t) { this._IndividualBabylonFileParsers[e] = t; } /** * Gets an individual parser from the list of available ones * @param name Defines the name of the parser * @returns the requested parser or null */ static GetIndividualParser(e) { return this._IndividualBabylonFileParsers[e] ? this._IndividualBabylonFileParsers[e] : null; } /** * Parser json data and populate both a scene and its associated container object * @param jsonData Defines the data to parse * @param scene Defines the scene to parse the data for * @param container Defines the container attached to the parsing sequence * @param rootUrl Defines the root url of the data */ static Parse(e, t, r, n) { for (const i in this._BabylonFileParsers) Object.prototype.hasOwnProperty.call(this._BabylonFileParsers, i) && this._BabylonFileParsers[i](e, t, r, n); } /** * Texture used in all pbr material as the reflection texture. * As in the majority of the scene they are the same (exception for multi room and so on), * this is easier to reference from here than from all the materials. */ get environmentTexture() { return this._environmentTexture; } set environmentTexture(e) { this._environmentTexture = e; } /** * @returns all meshes, lights, cameras, transformNodes and bones */ getNodes() { let e = []; return e = e.concat(this.meshes), e = e.concat(this.lights), e = e.concat(this.cameras), e = e.concat(this.transformNodes), this.skeletons.forEach((t) => e = e.concat(t.bones)), e; } } J1._BabylonFileParsers = {}; J1._IndividualBabylonFileParsers = {}; class Ml { constructor() { this.hoverCursor = "", this.actions = [], this.isRecursive = !1; } /** * Does exist one action manager with at least one trigger **/ static get HasTriggers() { for (const e in Ml.Triggers) if (Object.prototype.hasOwnProperty.call(Ml.Triggers, e)) return !0; return !1; } /** * Does exist one action manager with at least one pick trigger **/ static get HasPickTriggers() { for (const e in Ml.Triggers) if (Object.prototype.hasOwnProperty.call(Ml.Triggers, e)) { const t = parseInt(e); if (t >= 1 && t <= 7) return !0; } return !1; } /** * Does exist one action manager that handles actions of a given trigger * @param trigger defines the trigger to be tested * @returns a boolean indicating whether the trigger is handled by at least one action manager **/ static HasSpecificTrigger(e) { for (const t in Ml.Triggers) if (Object.prototype.hasOwnProperty.call(Ml.Triggers, t) && parseInt(t) === e) return !0; return !1; } } Ml.Triggers = {}; class IN { /** * Create a new EventState * @param mask defines the mask associated with this state * @param skipNextObservers defines a flag which will instruct the observable to skip following observers when set to true * @param target defines the original target of the state * @param currentTarget defines the current target of the state */ constructor(e, t = !1, r, n) { this.initialize(e, t, r, n); } /** * Initialize the current event state * @param mask defines the mask associated with this state * @param skipNextObservers defines a flag which will instruct the observable to skip following observers when set to true * @param target defines the original target of the state * @param currentTarget defines the current target of the state * @returns the current event state */ initialize(e, t = !1, r, n) { return this.mask = e, this.skipNextObservers = t, this.target = r, this.currentTarget = n, this; } } class c$ { /** * Creates a new observer * @param callback defines the callback to call when the observer is notified * @param mask defines the mask of the observer (used to filter notifications) * @param scope defines the current scope used to restore the JS context */ constructor(e, t, r = null) { this.callback = e, this.mask = t, this.scope = r, this._willBeUnregistered = !1, this.unregisterOnNextCall = !1, this._remove = null; } /** * Remove the observer from its observable * This can be used instead of using the observable's remove function. */ remove() { this._remove && this._remove(); } } class Oe { /** * Create an observable from a Promise. * @param promise a promise to observe for fulfillment. * @param onErrorObservable an observable to notify if a promise was rejected. * @returns the new Observable */ static FromPromise(e, t) { const r = new Oe(); return e.then((n) => { r.notifyObservers(n); }).catch((n) => { if (t) t.notifyObservers(n); else throw n; }), r; } /** * Gets the list of observers * Note that observers that were recently deleted may still be present in the list because they are only really deleted on the next javascript tick! */ get observers() { return this._observers; } /** * Creates a new observable * @param onObserverAdded defines a callback to call when a new observer is added * @param notifyIfTriggered If set to true the observable will notify when an observer was added if the observable was already triggered. */ constructor(e, t = !1) { this.notifyIfTriggered = t, this._observers = new Array(), this._numObserversMarkedAsDeleted = 0, this._hasNotified = !1, this._eventState = new IN(0), e && (this._onObserverAdded = e); } add(e, t = -1, r = !1, n = null, i = !1) { if (!e) return null; const s = new c$(e, t, n); return s.unregisterOnNextCall = i, r ? this._observers.unshift(s) : this._observers.push(s), this._onObserverAdded && this._onObserverAdded(s), this._hasNotified && this.notifyIfTriggered && this._lastNotifiedValue !== void 0 && this.notifyObserver(s, this._lastNotifiedValue), s._remove = () => { this.remove(s); }, s; } addOnce(e) { return this.add(e, void 0, void 0, void 0, !0); } /** * Remove an Observer from the Observable object * @param observer the instance of the Observer to remove * @returns false if it doesn't belong to this Observable */ remove(e) { return e ? (e._remove = null, this._observers.indexOf(e) !== -1 ? (this._deferUnregister(e), !0) : !1) : !1; } /** * Remove a callback from the Observable object * @param callback the callback to remove * @param scope optional scope. If used only the callbacks with this scope will be removed * @returns false if it doesn't belong to this Observable */ removeCallback(e, t) { for (let r = 0; r < this._observers.length; r++) { const n = this._observers[r]; if (!n._willBeUnregistered && n.callback === e && (!t || t === n.scope)) return this._deferUnregister(n), !0; } return !1; } /** * @internal */ _deferUnregister(e) { e._willBeUnregistered || (this._numObserversMarkedAsDeleted++, e.unregisterOnNextCall = !1, e._willBeUnregistered = !0, setTimeout(() => { this._remove(e); }, 0)); } // This should only be called when not iterating over _observers to avoid callback skipping. // Removes an observer from the _observer Array. _remove(e, t = !0) { if (!e) return !1; const r = this._observers.indexOf(e); return r !== -1 ? (t && this._numObserversMarkedAsDeleted--, this._observers.splice(r, 1), !0) : !1; } /** * Moves the observable to the top of the observer list making it get called first when notified * @param observer the observer to move */ makeObserverTopPriority(e) { this._remove(e, !1), this._observers.unshift(e); } /** * Moves the observable to the bottom of the observer list making it get called last when notified * @param observer the observer to move */ makeObserverBottomPriority(e) { this._remove(e, !1), this._observers.push(e); } /** * Notify all Observers by calling their respective callback with the given data * Will return true if all observers were executed, false if an observer set skipNextObservers to true, then prevent the subsequent ones to execute * @param eventData defines the data to send to all observers * @param mask defines the mask of the current notification (observers with incompatible mask (ie mask & observer.mask === 0) will not be notified) * @param target defines the original target of the state * @param currentTarget defines the current target of the state * @param userInfo defines any user info to send to observers * @returns false if the complete observer chain was not processed (because one observer set the skipNextObservers to true) */ notifyObservers(e, t = -1, r, n, i) { if (this.notifyIfTriggered && (this._hasNotified = !0, this._lastNotifiedValue = e), !this._observers.length) return !0; const s = this._eventState; s.mask = t, s.target = r, s.currentTarget = n, s.skipNextObservers = !1, s.lastReturnValue = e, s.userInfo = i; for (const a of this._observers) if (!a._willBeUnregistered && (a.mask & t && (a.unregisterOnNextCall && this._deferUnregister(a), a.scope ? s.lastReturnValue = a.callback.apply(a.scope, [e, s]) : s.lastReturnValue = a.callback(e, s)), s.skipNextObservers)) return !1; return !0; } /** * Notify a specific observer * @param observer defines the observer to notify * @param eventData defines the data to be sent to each callback * @param mask is used to filter observers defaults to -1 */ notifyObserver(e, t, r = -1) { if (this.notifyIfTriggered && (this._hasNotified = !0, this._lastNotifiedValue = t), e._willBeUnregistered) return; const n = this._eventState; n.mask = r, n.skipNextObservers = !1, e.unregisterOnNextCall && this._deferUnregister(e), e.callback(t, n); } /** * Gets a boolean indicating if the observable has at least one observer * @returns true is the Observable has at least one Observer registered */ hasObservers() { return this._observers.length - this._numObserversMarkedAsDeleted > 0; } /** * Clear the list of observers */ clear() { for (; this._observers.length; ) { const e = this._observers.pop(); e && (e._remove = null); } this._onObserverAdded = null, this._numObserversMarkedAsDeleted = 0, this.cleanLastNotifiedState(); } /** * Clean the last notified state - both the internal last value and the has-notified flag */ cleanLastNotifiedState() { this._hasNotified = !1, this._lastNotifiedValue = void 0; } /** * Clone the current observable * @returns a new observable */ clone() { const e = new Oe(); return e._observers = this._observers.slice(0), e; } /** * Does this observable handles observer registered with a given mask * @param mask defines the mask to be tested * @returns whether or not one observer registered with the given mask is handled **/ hasSpecificMask(e = -1) { for (const t of this._observers) if (t.mask & e || t.mask === e) return !0; return !1; } } class Xt { /** * Boolean : true if the absolute difference between a and b is lower than epsilon (default = 1.401298E-45) * @param a number * @param b number * @param epsilon (default = 1.401298E-45) * @returns true if the absolute difference between a and b is lower than epsilon (default = 1.401298E-45) */ static WithinEpsilon(e, t, r = 1401298e-51) { return Math.abs(e - t) <= r; } /** * Returns a string : the upper case translation of the number i to hexadecimal. * @param i number * @returns the upper case translation of the number i to hexadecimal. */ static ToHex(e) { const t = e.toString(16); return e <= 15 ? ("0" + t).toUpperCase() : t.toUpperCase(); } /** * Returns -1 if value is negative and +1 is value is positive. * @param value the value * @returns the value itself if it's equal to zero. */ static Sign(e) { return e = +e, e === 0 || isNaN(e) ? e : e > 0 ? 1 : -1; } /** * Returns the value itself if it's between min and max. * Returns min if the value is lower than min. * Returns max if the value is greater than max. * @param value the value to clmap * @param min the min value to clamp to (default: 0) * @param max the max value to clamp to (default: 1) * @returns the clamped value */ static Clamp(e, t = 0, r = 1) { return Math.min(r, Math.max(t, e)); } /** * the log2 of value. * @param value the value to compute log2 of * @returns the log2 of value. */ static Log2(e) { return Math.log(e) * Math.LOG2E; } /** * the floor part of a log2 value. * @param value the value to compute log2 of * @returns the log2 of value. */ static ILog2(e) { if (Math.log2) return Math.floor(Math.log2(e)); if (e < 0) return NaN; if (e === 0) return -1 / 0; let t = 0; if (e < 1) { for (; e < 1; ) t++, e = e * 2; t = -t; } else if (e > 1) for (; e > 1; ) t++, e = Math.floor(e / 2); return t; } /** * Loops the value, so that it is never larger than length and never smaller than 0. * * This is similar to the modulo operator but it works with floating point numbers. * For example, using 3.0 for t and 2.5 for length, the result would be 0.5. * With t = 5 and length = 2.5, the result would be 0.0. * Note, however, that the behaviour is not defined for negative numbers as it is for the modulo operator * @param value the value * @param length the length * @returns the looped value */ static Repeat(e, t) { return e - Math.floor(e / t) * t; } /** * Normalize the value between 0.0 and 1.0 using min and max values * @param value value to normalize * @param min max to normalize between * @param max min to normalize between * @returns the normalized value */ static Normalize(e, t, r) { return (e - t) / (r - t); } /** * Denormalize the value from 0.0 and 1.0 using min and max values * @param normalized value to denormalize * @param min max to denormalize between * @param max min to denormalize between * @returns the denormalized value */ static Denormalize(e, t, r) { return e * (r - t) + t; } /** * Calculates the shortest difference between two given angles given in degrees. * @param current current angle in degrees * @param target target angle in degrees * @returns the delta */ static DeltaAngle(e, t) { let r = Xt.Repeat(t - e, 360); return r > 180 && (r -= 360), r; } /** * PingPongs the value t, so that it is never larger than length and never smaller than 0. * @param tx value * @param length length * @returns The returned value will move back and forth between 0 and length */ static PingPong(e, t) { const r = Xt.Repeat(e, t * 2); return t - Math.abs(r - t); } /** * Interpolates between min and max with smoothing at the limits. * * This function interpolates between min and max in a similar way to Lerp. However, the interpolation will gradually speed up * from the start and slow down toward the end. This is useful for creating natural-looking animation, fading and other transitions. * @param from from * @param to to * @param tx value * @returns the smooth stepped value */ static SmoothStep(e, t, r) { let n = Xt.Clamp(r); return n = -2 * n * n * n + 3 * n * n, t * n + e * (1 - n); } /** * Moves a value current towards target. * * This is essentially the same as Mathf.Lerp but instead the function will ensure that the speed never exceeds maxDelta. * Negative values of maxDelta pushes the value away from target. * @param current current value * @param target target value * @param maxDelta max distance to move * @returns resulting value */ static MoveTowards(e, t, r) { let n = 0; return Math.abs(t - e) <= r ? n = t : n = e + Xt.Sign(t - e) * r, n; } /** * Same as MoveTowards but makes sure the values interpolate correctly when they wrap around 360 degrees. * * Variables current and target are assumed to be in degrees. For optimization reasons, negative values of maxDelta * are not supported and may cause oscillation. To push current away from a target angle, add 180 to that angle instead. * @param current current value * @param target target value * @param maxDelta max distance to move * @returns resulting angle */ static MoveTowardsAngle(e, t, r) { const n = Xt.DeltaAngle(e, t); let i = 0; return -r < n && n < r ? i = t : (t = e + n, i = Xt.MoveTowards(e, t, r)), i; } /** * Creates a new scalar with values linearly interpolated of "amount" between the start scalar and the end scalar. * @param start start value * @param end target value * @param amount amount to lerp between * @returns the lerped value */ static Lerp(e, t, r) { return e + (t - e) * r; } /** * Same as Lerp but makes sure the values interpolate correctly when they wrap around 360 degrees. * The parameter t is clamped to the range [0, 1]. Variables a and b are assumed to be in degrees. * @param start start value * @param end target value * @param amount amount to lerp between * @returns the lerped value */ static LerpAngle(e, t, r) { let n = Xt.Repeat(t - e, 360); return n > 180 && (n -= 360), e + n * Xt.Clamp(r); } /** * Calculates the linear parameter t that produces the interpolant value within the range [a, b]. * @param a start value * @param b target value * @param value value between a and b * @returns the inverseLerp value */ static InverseLerp(e, t, r) { let n = 0; return e != t ? n = Xt.Clamp((r - e) / (t - e)) : n = 0, n; } /** * Returns a new scalar located for "amount" (float) on the Hermite spline defined by the scalars "value1", "value3", "tangent1", "tangent2". * @see http://mathworld.wolfram.com/HermitePolynomial.html * @param value1 defines the first control point * @param tangent1 defines the first tangent * @param value2 defines the second control point * @param tangent2 defines the second tangent * @param amount defines the amount on the interpolation spline (between 0 and 1) * @returns hermite result */ static Hermite(e, t, r, n, i) { const s = i * i, a = i * s, f = 2 * a - 3 * s + 1, o = -2 * a + 3 * s, d = a - 2 * s + i, v = a - s; return e * f + r * o + t * d + n * v; } /** * Returns a new scalar which is the 1st derivative of the Hermite spline defined by the scalars "value1", "value2", "tangent1", "tangent2". * @param value1 defines the first control point * @param tangent1 defines the first tangent * @param value2 defines the second control point * @param tangent2 defines the second tangent * @param time define where the derivative must be done * @returns 1st derivative */ static Hermite1stDerivative(e, t, r, n, i) { const s = i * i; return (s - i) * 6 * e + (3 * s - 4 * i + 1) * t + (-s + i) * 6 * r + (3 * s - 2 * i) * n; } /** * Returns a random float number between and min and max values * @param min min value of random * @param max max value of random * @returns random value */ static RandomRange(e, t) { return e === t ? e : Math.random() * (t - e) + e; } /** * This function returns percentage of a number in a given range. * * RangeToPercent(40,20,60) will return 0.5 (50%) * RangeToPercent(34,0,100) will return 0.34 (34%) * @param number to convert to percentage * @param min min range * @param max max range * @returns the percentage */ static RangeToPercent(e, t, r) { return (e - t) / (r - t); } /** * This function returns number that corresponds to the percentage in a given range. * * PercentToRange(0.34,0,100) will return 34. * @param percent to convert to number * @param min min range * @param max max range * @returns the number */ static PercentToRange(e, t, r) { return (r - t) * e + t; } /** * Returns the angle converted to equivalent value between -Math.PI and Math.PI radians. * @param angle The angle to normalize in radian. * @returns The converted angle. */ static NormalizeRadians(e) { return e -= Xt.TwoPi * Math.floor((e + Math.PI) / Xt.TwoPi), e; } /** * Returns the highest common factor of two integers. * @param a first parameter * @param b second parameter * @returns HCF of a and b */ static HCF(e, t) { const r = e % t; return r === 0 ? t : Xt.HCF(t, r); } } Xt.TwoPi = Math.PI * 2; const JW = 1 / 2.2, xI = 2.2, kf = (1 + Math.sqrt(5)) / 2, Dn = 1e-3; class Nf { /** * Returns an array of the given size filled with elements built from the given constructor and the parameters. * @param size the number of element to construct and put in the array. * @param itemBuilder a callback responsible for creating new instance of item. Called once per array entry. * @returns a new array filled with new objects. */ static BuildArray(e, t) { const r = []; for (let n = 0; n < e; ++n) r.push(t()); return r; } /** * Returns a tuple of the given size filled with elements built from the given constructor and the parameters. * @param size he number of element to construct and put in the tuple. * @param itemBuilder a callback responsible for creating new instance of item. Called once per tuple entry. * @returns a new tuple filled with new objects. */ static BuildTuple(e, t) { return Nf.BuildArray(e, t); } } function j9e(A, e, t) { const r = A[e]; if (typeof r != "function") return null; const n = function() { const i = A.length, s = n.previous.apply(A, arguments); return t(e, i), s; }; return r.next = n, n.previous = r, A[e] = n, () => { const i = n.previous; if (!i) return; const s = n.next; s ? (i.next = s, s.previous = i) : (i.next = void 0, A[e] = i), n.next = void 0, n.previous = void 0; }; } const w9e = ["push", "splice", "pop", "shift", "unshift"]; function p$(A, e) { const t = w9e.map((r) => j9e(A, r, e)); return () => { t.forEach((r) => { r == null || r(); }); }; } const h$ = {}; function Ue(A, e) { h$[A] = e; } function Jo(A) { return h$[A]; } class k9 { /** * @internal */ static SetMatrixPrecision(e) { if (k9.MatrixTrackPrecisionChange = !1, e && !k9.MatrixUse64Bits && k9.MatrixTrackedMatrices) for (let t = 0; t < k9.MatrixTrackedMatrices.length; ++t) { const r = k9.MatrixTrackedMatrices[t], n = r._m; r._m = new Array(16); for (let i = 0; i < 16; ++i) r._m[i] = n[i]; } k9.MatrixUse64Bits = e, k9.MatrixCurrentType = k9.MatrixUse64Bits ? Array : Float32Array, k9.MatrixTrackedMatrices = null; } } k9.MatrixUse64Bits = !1; k9.MatrixTrackPrecisionChange = !0; k9.MatrixCurrentType = Float32Array; k9.MatrixTrackedMatrices = []; class gr { /** * Gets the latest created engine */ static get LastCreatedEngine() { return this.Instances.length === 0 ? null : this.Instances[this.Instances.length - 1]; } /** * Gets the latest created scene */ static get LastCreatedScene() { return this._LastCreatedScene; } } gr.Instances = []; gr.OnEnginesDisposedObservable = new Oe(); gr._LastCreatedScene = null; gr.UseFallbackTexture = !0; gr.FallbackTexture = ""; const Kl = (A) => parseInt(A.toString().replace(/\W/g, "")); class at { /** * Creates a new Vector2 from the given x and y coordinates * @param x defines the first coordinate * @param y defines the second coordinate */ constructor(e = 0, t = 0) { this.x = e, this.y = t; } /** * Gets a string with the Vector2 coordinates * @returns a string with the Vector2 coordinates */ toString() { return `{X: ${this.x} Y: ${this.y}}`; } /** * Gets class name * @returns the string "Vector2" */ getClassName() { return "Vector2"; } /** * Gets current vector hash code * @returns the Vector2 hash code as a number */ getHashCode() { const e = Kl(this.x), t = Kl(this.y); let r = e; return r = r * 397 ^ t, r; } // Operators /** * Sets the Vector2 coordinates in the given array or Float32Array from the given index. * Example Playground https://playground.babylonjs.com/#QYBWV4#15 * @param array defines the source array * @param index defines the offset in source array * @returns the current Vector2 */ toArray(e, t = 0) { return e[t] = this.x, e[t + 1] = this.y, this; } /** * Update the current vector from an array * Example Playground https://playground.babylonjs.com/#QYBWV4#39 * @param array defines the destination array * @param index defines the offset in the destination array * @returns the current Vector2 */ fromArray(e, t = 0) { return at.FromArrayToRef(e, t, this), this; } /** * Copy the current vector to an array * Example Playground https://playground.babylonjs.com/#QYBWV4#40 * @returns a new array with 2 elements: the Vector2 coordinates. */ asArray() { const e = []; return this.toArray(e, 0), e; } /** * Sets the Vector2 coordinates with the given Vector2 coordinates * Example Playground https://playground.babylonjs.com/#QYBWV4#24 * @param source defines the source Vector2 * @returns the current updated Vector2 */ copyFrom(e) { return this.x = e.x, this.y = e.y, this; } /** * Sets the Vector2 coordinates with the given floats * Example Playground https://playground.babylonjs.com/#QYBWV4#25 * @param x defines the first coordinate * @param y defines the second coordinate * @returns the current updated Vector2 */ copyFromFloats(e, t) { return this.x = e, this.y = t, this; } /** * Sets the Vector2 coordinates with the given floats * Example Playground https://playground.babylonjs.com/#QYBWV4#62 * @param x defines the first coordinate * @param y defines the second coordinate * @returns the current updated Vector2 */ set(e, t) { return this.copyFromFloats(e, t); } /** * Add another vector with the current one * Example Playground https://playground.babylonjs.com/#QYBWV4#11 * @param otherVector defines the other vector * @returns a new Vector2 set with the addition of the current Vector2 and the given one coordinates */ add(e) { return new this.constructor(this.x + e.x, this.y + e.y); } /** * Sets the "result" coordinates with the addition of the current Vector2 and the given one coordinates * Example Playground https://playground.babylonjs.com/#QYBWV4#12 * @param otherVector defines the other vector * @param result defines the target vector * @returns result input */ addToRef(e, t) { return t.x = this.x + e.x, t.y = this.y + e.y, t; } /** * Set the Vector2 coordinates by adding the given Vector2 coordinates * Example Playground https://playground.babylonjs.com/#QYBWV4#13 * @param otherVector defines the other vector * @returns the current updated Vector2 */ addInPlace(e) { return this.x += e.x, this.y += e.y, this; } /** * Gets a new Vector2 by adding the current Vector2 coordinates to the given Vector3 x, y coordinates * Example Playground https://playground.babylonjs.com/#QYBWV4#14 * @param otherVector defines the other vector * @returns a new Vector2 */ addVector3(e) { return new this.constructor(this.x + e.x, this.y + e.y); } /** * Gets a new Vector2 set with the subtracted coordinates of the given one from the current Vector2 * Example Playground https://playground.babylonjs.com/#QYBWV4#61 * @param otherVector defines the other vector * @returns a new Vector2 */ subtract(e) { return new this.constructor(this.x - e.x, this.y - e.y); } /** * Sets the "result" coordinates with the subtraction of the given one from the current Vector2 coordinates. * Example Playground https://playground.babylonjs.com/#QYBWV4#63 * @param otherVector defines the other vector * @param result defines the target vector * @returns result input */ subtractToRef(e, t) { return t.x = this.x - e.x, t.y = this.y - e.y, t; } /** * Sets the current Vector2 coordinates by subtracting from it the given one coordinates * Example Playground https://playground.babylonjs.com/#QYBWV4#88 * @param otherVector defines the other vector * @returns the current updated Vector2 */ subtractInPlace(e) { return this.x -= e.x, this.y -= e.y, this; } /** * Multiplies in place the current Vector2 coordinates by the given ones * Example Playground https://playground.babylonjs.com/#QYBWV4#43 * @param otherVector defines the other vector * @returns the current updated Vector2 */ multiplyInPlace(e) { return this.x *= e.x, this.y *= e.y, this; } /** * Returns a new Vector2 set with the multiplication of the current Vector2 and the given one coordinates * Example Playground https://playground.babylonjs.com/#QYBWV4#42 * @param otherVector defines the other vector * @returns a new Vector2 */ multiply(e) { return new this.constructor(this.x * e.x, this.y * e.y); } /** * Sets "result" coordinates with the multiplication of the current Vector2 and the given one coordinates * Example Playground https://playground.babylonjs.com/#QYBWV4#44 * @param otherVector defines the other vector * @param result defines the target vector * @returns result input */ multiplyToRef(e, t) { return t.x = this.x * e.x, t.y = this.y * e.y, t; } /** * Gets a new Vector2 set with the Vector2 coordinates multiplied by the given floats * Example Playground https://playground.babylonjs.com/#QYBWV4#89 * @param x defines the first coordinate * @param y defines the second coordinate * @returns a new Vector2 */ multiplyByFloats(e, t) { return new this.constructor(this.x * e, this.y * t); } /** * Returns a new Vector2 set with the Vector2 coordinates divided by the given one coordinates * Example Playground https://playground.babylonjs.com/#QYBWV4#27 * @param otherVector defines the other vector * @returns a new Vector2 */ divide(e) { return new this.constructor(this.x / e.x, this.y / e.y); } /** * Sets the "result" coordinates with the Vector2 divided by the given one coordinates * Example Playground https://playground.babylonjs.com/#QYBWV4#30 * @param otherVector defines the other vector * @param result defines the target vector * @returns result input */ divideToRef(e, t) { return t.x = this.x / e.x, t.y = this.y / e.y, t; } /** * Divides the current Vector2 coordinates by the given ones * Example Playground https://playground.babylonjs.com/#QYBWV4#28 * @param otherVector defines the other vector * @returns the current updated Vector2 */ divideInPlace(e) { return this.divideToRef(e, this); } /** * Gets a new Vector2 with current Vector2 negated coordinates * Example Playground https://playground.babylonjs.com/#QYBWV4#22 * @returns a new Vector2 */ negate() { return new this.constructor(-this.x, -this.y); } /** * Negate this vector in place * Example Playground https://playground.babylonjs.com/#QYBWV4#23 * @returns this */ negateInPlace() { return this.x *= -1, this.y *= -1, this; } /** * Negate the current Vector2 and stores the result in the given vector "result" coordinates * Example Playground https://playground.babylonjs.com/#QYBWV4#41 * @param result defines the Vector3 object where to store the result * @returns the result */ negateToRef(e) { return e.copyFromFloats(this.x * -1, this.y * -1); } /** * Multiply the Vector2 coordinates by * Example Playground https://playground.babylonjs.com/#QYBWV4#59 * @param scale defines the scaling factor * @returns the current updated Vector2 */ scaleInPlace(e) { return this.x *= e, this.y *= e, this; } /** * Returns a new Vector2 scaled by "scale" from the current Vector2 * Example Playground https://playground.babylonjs.com/#QYBWV4#52 * @param scale defines the scaling factor * @returns a new Vector2 */ scale(e) { const t = new this.constructor(0, 0); return this.scaleToRef(e, t), t; } /** * Scale the current Vector2 values by a factor to a given Vector2 * Example Playground https://playground.babylonjs.com/#QYBWV4#57 * @param scale defines the scale factor * @param result defines the Vector2 object where to store the result * @returns result input */ scaleToRef(e, t) { return t.x = this.x * e, t.y = this.y * e, t; } /** * Scale the current Vector2 values by a factor and add the result to a given Vector2 * Example Playground https://playground.babylonjs.com/#QYBWV4#58 * @param scale defines the scale factor * @param result defines the Vector2 object where to store the result * @returns result input */ scaleAndAddToRef(e, t) { return t.x += this.x * e, t.y += this.y * e, t; } /** * Gets a boolean if two vectors are equals * Example Playground https://playground.babylonjs.com/#QYBWV4#31 * @param otherVector defines the other vector * @returns true if the given vector coordinates strictly equal the current Vector2 ones */ equals(e) { return e && this.x === e.x && this.y === e.y; } /** * Gets a boolean if two vectors are equals (using an epsilon value) * Example Playground https://playground.babylonjs.com/#QYBWV4#32 * @param otherVector defines the other vector * @param epsilon defines the minimal distance to consider equality * @returns true if the given vector coordinates are close to the current ones by a distance of epsilon. */ equalsWithEpsilon(e, t = Dn) { return e && Xt.WithinEpsilon(this.x, e.x, t) && Xt.WithinEpsilon(this.y, e.y, t); } /** * Gets a new Vector2 from current Vector2 floored values * Example Playground https://playground.babylonjs.com/#QYBWV4#35 * eg (1.2, 2.31) returns (1, 2) * @returns a new Vector2 */ floor() { return new this.constructor(Math.floor(this.x), Math.floor(this.y)); } /** * Gets a new Vector2 from current Vector2 fractional values * Example Playground https://playground.babylonjs.com/#QYBWV4#34 * eg (1.2, 2.31) returns (0.2, 0.31) * @returns a new Vector2 */ fract() { return new this.constructor(this.x - Math.floor(this.x), this.y - Math.floor(this.y)); } /** * Rotate the current vector into a given result vector * Example Playground https://playground.babylonjs.com/#QYBWV4#49 * @param angle defines the rotation angle * @param result defines the result vector where to store the rotated vector * @returns result input */ rotateToRef(e, t) { const r = Math.cos(e), n = Math.sin(e), i = r * this.x - n * this.y, s = n * this.x + r * this.y; return t.x = i, t.y = s, t; } // Properties /** * Gets the length of the vector * @returns the vector length (float) */ length() { return Math.sqrt(this.x * this.x + this.y * this.y); } /** * Gets the vector squared length * @returns the vector squared length (float) */ lengthSquared() { return this.x * this.x + this.y * this.y; } // Methods /** * Normalize the vector * Example Playground https://playground.babylonjs.com/#QYBWV4#48 * @returns the current updated Vector2 */ normalize() { return this.normalizeFromLength(this.length()); } /** * Normalize the current Vector2 with the given input length. * Please note that this is an in place operation. * @param len the length of the vector * @returns the current updated Vector2 */ normalizeFromLength(e) { return e === 0 || e === 1 ? this : this.scaleInPlace(1 / e); } /** * Normalize the current Vector2 to a new vector * @returns the new Vector2 */ normalizeToNew() { const e = new this.constructor(0, 0); return this.normalizeToRef(e), e; } /** * Normalize the current Vector2 to the reference * @param reference define the Vector2 to update * @returns the updated Vector2 */ normalizeToRef(e) { const t = this.length(); return t === 0 || t === 1 ? e.copyFromFloats(this.x, this.y) : this.scaleToRef(1 / t, e); } /** * Gets a new Vector2 copied from the Vector2 * Example Playground https://playground.babylonjs.com/#QYBWV4#20 * @returns a new Vector2 */ clone() { return new this.constructor(this.x, this.y); } /** * Gets the dot product of the current vector and the vector "otherVector" * @param otherVector defines second vector * @returns the dot product (float) */ dot(e) { return this.x * e.x + this.y * e.y; } // Statics /** * Gets a new Vector2(0, 0) * @returns a new Vector2 */ static Zero() { return new at(0, 0); } /** * Gets a new Vector2(1, 1) * @returns a new Vector2 */ static One() { return new at(1, 1); } /** * Returns a new Vector2 with random values between min and max * @param min the minimum random value * @param max the maximum random value * @returns a Vector2 with random values between min and max */ static Random(e = 0, t = 1) { return new at(Xt.RandomRange(e, t), Xt.RandomRange(e, t)); } /** * Gets a zero Vector2 that must not be updated */ static get ZeroReadOnly() { return at._ZeroReadOnly; } /** * Gets a new Vector2 set from the given index element of the given array * Example Playground https://playground.babylonjs.com/#QYBWV4#79 * @param array defines the data source * @param offset defines the offset in the data source * @returns a new Vector2 */ static FromArray(e, t = 0) { return new at(e[t], e[t + 1]); } /** * Sets "result" from the given index element of the given array * Example Playground https://playground.babylonjs.com/#QYBWV4#80 * @param array defines the data source * @param offset defines the offset in the data source * @param result defines the target vector * @returns result input */ static FromArrayToRef(e, t, r) { return r.x = e[t], r.y = e[t + 1], r; } /** * Gets a new Vector2 located for "amount" (float) on the CatmullRom spline defined by the given four Vector2 * Example Playground https://playground.babylonjs.com/#QYBWV4#65 * @param value1 defines 1st point of control * @param value2 defines 2nd point of control * @param value3 defines 3rd point of control * @param value4 defines 4th point of control * @param amount defines the interpolation factor * @returns a new Vector2 */ static CatmullRom(e, t, r, n, i) { const s = i * i, a = i * s, f = 0.5 * (2 * t.x + (-e.x + r.x) * i + (2 * e.x - 5 * t.x + 4 * r.x - n.x) * s + (-e.x + 3 * t.x - 3 * r.x + n.x) * a), o = 0.5 * (2 * t.y + (-e.y + r.y) * i + (2 * e.y - 5 * t.y + 4 * r.y - n.y) * s + (-e.y + 3 * t.y - 3 * r.y + n.y) * a); return new e.constructor(f, o); } /** * Returns a new Vector2 set with same the coordinates than "value" ones if the vector "value" is in the square defined by "min" and "max". * If a coordinate of "value" is lower than "min" coordinates, the returned Vector2 is given this "min" coordinate. * If a coordinate of "value" is greater than "max" coordinates, the returned Vector2 is given this "max" coordinate * Example Playground https://playground.babylonjs.com/#QYBWV4#76 * @param value defines the value to clamp * @param min defines the lower limit * @param max defines the upper limit * @returns a new Vector2 */ static Clamp(e, t, r) { let n = e.x; n = n > r.x ? r.x : n, n = n < t.x ? t.x : n; let i = e.y; return i = i > r.y ? r.y : i, i = i < t.y ? t.y : i, new e.constructor(n, i); } /** * Returns a new Vector2 located for "amount" (float) on the Hermite spline defined by the vectors "value1", "value2", "tangent1", "tangent2" * Example Playground https://playground.babylonjs.com/#QYBWV4#81 * @param value1 defines the 1st control point * @param tangent1 defines the outgoing tangent * @param value2 defines the 2nd control point * @param tangent2 defines the incoming tangent * @param amount defines the interpolation factor * @returns a new Vector2 */ static Hermite(e, t, r, n, i) { const s = i * i, a = i * s, f = 2 * a - 3 * s + 1, o = -2 * a + 3 * s, d = a - 2 * s + i, v = a - s, u = e.x * f + r.x * o + t.x * d + n.x * v, l = e.y * f + r.y * o + t.y * d + n.y * v; return new e.constructor(u, l); } /** * Returns a new Vector2 which is the 1st derivative of the Hermite spline defined by the vectors "value1", "value2", "tangent1", "tangent2". * Example Playground https://playground.babylonjs.com/#QYBWV4#82 * @param value1 defines the first control point * @param tangent1 defines the first tangent * @param value2 defines the second control point * @param tangent2 defines the second tangent * @param time define where the derivative must be done * @returns 1st derivative */ static Hermite1stDerivative(e, t, r, n, i) { const s = new e.constructor(); return this.Hermite1stDerivativeToRef(e, t, r, n, i, s), s; } /** * Returns a new Vector2 which is the 1st derivative of the Hermite spline defined by the vectors "value1", "value2", "tangent1", "tangent2". * Example Playground https://playground.babylonjs.com/#QYBWV4#83 * @param value1 defines the first control point * @param tangent1 defines the first tangent * @param value2 defines the second control point * @param tangent2 defines the second tangent * @param time define where the derivative must be done * @param result define where the derivative will be stored * @returns result input */ static Hermite1stDerivativeToRef(e, t, r, n, i, s) { const a = i * i; return s.x = (a - i) * 6 * e.x + (3 * a - 4 * i + 1) * t.x + (-a + i) * 6 * r.x + (3 * a - 2 * i) * n.x, s.y = (a - i) * 6 * e.y + (3 * a - 4 * i + 1) * t.y + (-a + i) * 6 * r.y + (3 * a - 2 * i) * n.y, s; } /** * Returns a new Vector2 located for "amount" (float) on the linear interpolation between the vector "start" adn the vector "end". * Example Playground https://playground.babylonjs.com/#QYBWV4#84 * @param start defines the start vector * @param end defines the end vector * @param amount defines the interpolation factor * @returns a new Vector2 */ static Lerp(e, t, r) { const n = e.x + (t.x - e.x) * r, i = e.y + (t.y - e.y) * r; return new e.constructor(n, i); } /** * Gets the dot product of the vector "left" and the vector "right" * Example Playground https://playground.babylonjs.com/#QYBWV4#90 * @param left defines first vector * @param right defines second vector * @returns the dot product (float) */ static Dot(e, t) { return e.x * t.x + e.y * t.y; } /** * Returns a new Vector2 equal to the normalized given vector * Example Playground https://playground.babylonjs.com/#QYBWV4#46 * @param vector defines the vector to normalize * @returns a new Vector2 */ static Normalize(e) { const t = new e.constructor(); return at.NormalizeToRef(e, t), t; } /** * Normalize a given vector into a second one * Example Playground https://playground.babylonjs.com/#QYBWV4#50 * @param vector defines the vector to normalize * @param result defines the vector where to store the result * @returns result input */ static NormalizeToRef(e, t) { return e.normalizeToRef(t), t; } /** * Gets a new Vector2 set with the minimal coordinate values from the "left" and "right" vectors * Example Playground https://playground.babylonjs.com/#QYBWV4#86 * @param left defines 1st vector * @param right defines 2nd vector * @returns a new Vector2 */ static Minimize(e, t) { const r = e.x < t.x ? e.x : t.x, n = e.y < t.y ? e.y : t.y; return new e.constructor(r, n); } /** * Gets a new Vector2 set with the maximal coordinate values from the "left" and "right" vectors * Example Playground https://playground.babylonjs.com/#QYBWV4#86 * @param left defines 1st vector * @param right defines 2nd vector * @returns a new Vector2 */ static Maximize(e, t) { const r = e.x > t.x ? e.x : t.x, n = e.y > t.y ? e.y : t.y; return new e.constructor(r, n); } /** * Gets a new Vector2 set with the transformed coordinates of the given vector by the given transformation matrix * Example Playground https://playground.babylonjs.com/#QYBWV4#17 * @param vector defines the vector to transform * @param transformation defines the matrix to apply * @returns a new Vector2 */ static Transform(e, t) { const r = new e.constructor(); return at.TransformToRef(e, t, r), r; } /** * Transforms the given vector coordinates by the given transformation matrix and stores the result in the vector "result" coordinates * Example Playground https://playground.babylonjs.com/#QYBWV4#19 * @param vector defines the vector to transform * @param transformation defines the matrix to apply * @param result defines the target vector * @returns result input */ static TransformToRef(e, t, r) { const n = t.m, i = e.x * n[0] + e.y * n[4] + n[12], s = e.x * n[1] + e.y * n[5] + n[13]; return r.x = i, r.y = s, r; } /** * Determines if a given vector is included in a triangle * Example Playground https://playground.babylonjs.com/#QYBWV4#87 * @param p defines the vector to test * @param p0 defines 1st triangle point * @param p1 defines 2nd triangle point * @param p2 defines 3rd triangle point * @returns true if the point "p" is in the triangle defined by the vectors "p0", "p1", "p2" */ static PointInTriangle(e, t, r, n) { const i = 0.5 * (-r.y * n.x + t.y * (-r.x + n.x) + t.x * (r.y - n.y) + r.x * n.y), s = i < 0 ? -1 : 1, a = (t.y * n.x - t.x * n.y + (n.y - t.y) * e.x + (t.x - n.x) * e.y) * s, f = (t.x * r.y - t.y * r.x + (t.y - r.y) * e.x + (r.x - t.x) * e.y) * s; return a > 0 && f > 0 && a + f < 2 * i * s; } /** * Gets the distance between the vectors "value1" and "value2" * Example Playground https://playground.babylonjs.com/#QYBWV4#71 * @param value1 defines first vector * @param value2 defines second vector * @returns the distance between vectors */ static Distance(e, t) { return Math.sqrt(at.DistanceSquared(e, t)); } /** * Returns the squared distance between the vectors "value1" and "value2" * Example Playground https://playground.babylonjs.com/#QYBWV4#72 * @param value1 defines first vector * @param value2 defines second vector * @returns the squared distance between vectors */ static DistanceSquared(e, t) { const r = e.x - t.x, n = e.y - t.y; return r * r + n * n; } /** * Gets a new Vector2 located at the center of the vectors "value1" and "value2" * Example Playground https://playground.babylonjs.com/#QYBWV4#86 * Example Playground https://playground.babylonjs.com/#QYBWV4#66 * @param value1 defines first vector * @param value2 defines second vector * @returns a new Vector2 */ static Center(e, t) { const r = new e.constructor(); return at.CenterToRef(e, t, r); } /** * Gets the center of the vectors "value1" and "value2" and stores the result in the vector "ref" * Example Playground https://playground.babylonjs.com/#QYBWV4#66 * @param value1 defines first vector * @param value2 defines second vector * @param ref defines third vector * @returns ref */ static CenterToRef(e, t, r) { return r.copyFromFloats((e.x + t.x) / 2, (e.y + t.y) / 2); } /** * Gets the shortest distance (float) between the point "p" and the segment defined by the two points "segA" and "segB". * Example Playground https://playground.babylonjs.com/#QYBWV4#77 * @param p defines the middle point * @param segA defines one point of the segment * @param segB defines the other point of the segment * @returns the shortest distance */ static DistanceOfPointFromSegment(e, t, r) { const n = at.DistanceSquared(t, r); if (n === 0) return at.Distance(e, t); const i = r.subtract(t), s = Math.max(0, Math.min(1, at.Dot(e.subtract(t), i) / n)), a = t.add(i.multiplyByFloats(s, s)); return at.Distance(e, a); } } at._ZeroReadOnly = at.Zero(); class S { /** Gets or sets the x coordinate */ get x() { return this._x; } set x(e) { this._x = e, this._isDirty = !0; } /** Gets or sets the y coordinate */ get y() { return this._y; } set y(e) { this._y = e, this._isDirty = !0; } /** Gets or sets the z coordinate */ get z() { return this._z; } set z(e) { this._z = e, this._isDirty = !0; } /** * Creates a new Vector3 object from the given x, y, z (floats) coordinates. * @param x defines the first coordinates (on X axis) * @param y defines the second coordinates (on Y axis) * @param z defines the third coordinates (on Z axis) */ constructor(e = 0, t = 0, r = 0) { this._isDirty = !0, this._x = e, this._y = t, this._z = r; } /** * Creates a string representation of the Vector3 * Example Playground https://playground.babylonjs.com/#R1F8YU#67 * @returns a string with the Vector3 coordinates. */ toString() { return `{X: ${this._x} Y: ${this._y} Z: ${this._z}}`; } /** * Gets the class name * @returns the string "Vector3" */ getClassName() { return "Vector3"; } /** * Creates the Vector3 hash code * @returns a number which tends to be unique between Vector3 instances */ getHashCode() { const e = Kl(this._x), t = Kl(this._y), r = Kl(this._z); let n = e; return n = n * 397 ^ t, n = n * 397 ^ r, n; } // Operators /** * Creates an array containing three elements : the coordinates of the Vector3 * Example Playground https://playground.babylonjs.com/#R1F8YU#10 * @returns a new array of numbers */ asArray() { const e = []; return this.toArray(e, 0), e; } /** * Populates the given array or Float32Array from the given index with the successive coordinates of the Vector3 * Example Playground https://playground.babylonjs.com/#R1F8YU#65 * @param array defines the destination array * @param index defines the offset in the destination array * @returns the current Vector3 */ toArray(e, t = 0) { return e[t] = this._x, e[t + 1] = this._y, e[t + 2] = this._z, this; } /** * Update the current vector from an array * Example Playground https://playground.babylonjs.com/#R1F8YU#24 * @param array defines the destination array * @param index defines the offset in the destination array * @returns the current Vector3 */ fromArray(e, t = 0) { return S.FromArrayToRef(e, t, this), this; } /** * Converts the current Vector3 into a quaternion (considering that the Vector3 contains Euler angles representation of a rotation) * Example Playground https://playground.babylonjs.com/#R1F8YU#66 * @returns a new Quaternion object, computed from the Vector3 coordinates */ toQuaternion() { return Ze.RotationYawPitchRoll(this._y, this._x, this._z); } /** * Adds the given vector to the current Vector3 * Example Playground https://playground.babylonjs.com/#R1F8YU#4 * @param otherVector defines the second operand * @returns the current updated Vector3 */ addInPlace(e) { return this.addInPlaceFromFloats(e._x, e._y, e._z); } /** * Adds the given coordinates to the current Vector3 * Example Playground https://playground.babylonjs.com/#R1F8YU#5 * @param x defines the x coordinate of the operand * @param y defines the y coordinate of the operand * @param z defines the z coordinate of the operand * @returns the current updated Vector3 */ addInPlaceFromFloats(e, t, r) { return this._x += e, this._y += t, this._z += r, this._isDirty = !0, this; } /** * Gets a new Vector3, result of the addition the current Vector3 and the given vector * Example Playground https://playground.babylonjs.com/#R1F8YU#3 * @param otherVector defines the second operand * @returns the resulting Vector3 */ add(e) { return new this.constructor(this._x + e._x, this._y + e._y, this._z + e._z); } /** * Adds the current Vector3 to the given one and stores the result in the vector "result" * Example Playground https://playground.babylonjs.com/#R1F8YU#6 * @param otherVector defines the second operand * @param result defines the Vector3 object where to store the result * @returns the result */ addToRef(e, t) { return t.copyFromFloats(this._x + e._x, this._y + e._y, this._z + e._z); } /** * Subtract the given vector from the current Vector3 * Example Playground https://playground.babylonjs.com/#R1F8YU#61 * @param otherVector defines the second operand * @returns the current updated Vector3 */ subtractInPlace(e) { return this._x -= e._x, this._y -= e._y, this._z -= e._z, this._isDirty = !0, this; } /** * Returns a new Vector3, result of the subtraction of the given vector from the current Vector3 * Example Playground https://playground.babylonjs.com/#R1F8YU#60 * @param otherVector defines the second operand * @returns the resulting Vector3 */ subtract(e) { return new this.constructor(this._x - e._x, this._y - e._y, this._z - e._z); } /** * Subtracts the given vector from the current Vector3 and stores the result in the vector "result". * Example Playground https://playground.babylonjs.com/#R1F8YU#63 * @param otherVector defines the second operand * @param result defines the Vector3 object where to store the result * @returns the result */ subtractToRef(e, t) { return this.subtractFromFloatsToRef(e._x, e._y, e._z, t); } /** * Returns a new Vector3 set with the subtraction of the given floats from the current Vector3 coordinates * Example Playground https://playground.babylonjs.com/#R1F8YU#62 * @param x defines the x coordinate of the operand * @param y defines the y coordinate of the operand * @param z defines the z coordinate of the operand * @returns the resulting Vector3 */ subtractFromFloats(e, t, r) { return new this.constructor(this._x - e, this._y - t, this._z - r); } /** * Subtracts the given floats from the current Vector3 coordinates and set the given vector "result" with this result * Example Playground https://playground.babylonjs.com/#R1F8YU#64 * @param x defines the x coordinate of the operand * @param y defines the y coordinate of the operand * @param z defines the z coordinate of the operand * @param result defines the Vector3 object where to store the result * @returns the result */ subtractFromFloatsToRef(e, t, r, n) { return n.copyFromFloats(this._x - e, this._y - t, this._z - r); } /** * Gets a new Vector3 set with the current Vector3 negated coordinates * Example Playground https://playground.babylonjs.com/#R1F8YU#35 * @returns a new Vector3 */ negate() { return new this.constructor(-this._x, -this._y, -this._z); } /** * Negate this vector in place * Example Playground https://playground.babylonjs.com/#R1F8YU#36 * @returns this */ negateInPlace() { return this._x *= -1, this._y *= -1, this._z *= -1, this._isDirty = !0, this; } /** * Negate the current Vector3 and stores the result in the given vector "result" coordinates * Example Playground https://playground.babylonjs.com/#R1F8YU#37 * @param result defines the Vector3 object where to store the result * @returns the result */ negateToRef(e) { return e.copyFromFloats(this._x * -1, this._y * -1, this._z * -1); } /** * Multiplies the Vector3 coordinates by the float "scale" * Example Playground https://playground.babylonjs.com/#R1F8YU#56 * @param scale defines the multiplier factor * @returns the current updated Vector3 */ scaleInPlace(e) { return this._x *= e, this._y *= e, this._z *= e, this._isDirty = !0, this; } /** * Returns a new Vector3 set with the current Vector3 coordinates multiplied by the float "scale" * Example Playground https://playground.babylonjs.com/#R1F8YU#53 * @param scale defines the multiplier factor * @returns a new Vector3 */ scale(e) { return new this.constructor(this._x * e, this._y * e, this._z * e); } /** * Multiplies the current Vector3 coordinates by the float "scale" and stores the result in the given vector "result" coordinates * Example Playground https://playground.babylonjs.com/#R1F8YU#57 * @param scale defines the multiplier factor * @param result defines the Vector3 object where to store the result * @returns the result */ scaleToRef(e, t) { return t.copyFromFloats(this._x * e, this._y * e, this._z * e); } /** * Creates a vector normal (perpendicular) to the current Vector3 and stores the result in the given vector * Out of the infinite possibilities the normal chosen is the one formed by rotating the current vector * 90 degrees about an axis which lies perpendicular to the current vector * and its projection on the xz plane. In the case of a current vector in the xz plane * the normal is calculated to be along the y axis. * Example Playground https://playground.babylonjs.com/#R1F8YU#230 * Example Playground https://playground.babylonjs.com/#R1F8YU#231 * @param result defines the Vector3 object where to store the resultant normal * returns the result */ getNormalToRef(e) { const t = this.length(); let r = Math.acos(this.y / t); const n = Math.atan2(this.z, this.x); r > Math.PI / 2 ? r -= Math.PI / 2 : r += Math.PI / 2; const i = t * Math.sin(r) * Math.cos(n), s = t * Math.cos(r), a = t * Math.sin(r) * Math.sin(n); return e.set(i, s, a), e; } /** * Rotates the vector using the given unit quaternion and stores the new vector in result * Example Playground https://playground.babylonjs.com/#R1F8YU#9 * @param q the unit quaternion representing the rotation * @param result the output vector * @returns the result */ applyRotationQuaternionToRef(e, t) { const r = this._x, n = this._y, i = this._z, s = e._x, a = e._y, f = e._z, o = e._w, d = 2 * (a * i - f * n), v = 2 * (f * r - s * i), u = 2 * (s * n - a * r); return t._x = r + o * d + a * u - f * v, t._y = n + o * v + f * d - s * u, t._z = i + o * u + s * v - a * d, t._isDirty = !0, t; } /** * Rotates the vector in place using the given unit quaternion * Example Playground https://playground.babylonjs.com/#R1F8YU#8 * @param q the unit quaternion representing the rotation * @returns the current updated Vector3 */ applyRotationQuaternionInPlace(e) { return this.applyRotationQuaternionToRef(e, this); } /** * Rotates the vector using the given unit quaternion and returns the new vector * Example Playground https://playground.babylonjs.com/#R1F8YU#7 * @param q the unit quaternion representing the rotation * @returns a new Vector3 */ applyRotationQuaternion(e) { return this.applyRotationQuaternionToRef(e, new this.constructor()); } /** * Scale the current Vector3 values by a factor and add the result to a given Vector3 * Example Playground https://playground.babylonjs.com/#R1F8YU#55 * @param scale defines the scale factor * @param result defines the Vector3 object where to store the result * @returns result input */ scaleAndAddToRef(e, t) { return t.addInPlaceFromFloats(this._x * e, this._y * e, this._z * e); } /** * Projects the current point Vector3 to a plane along a ray starting from a specified origin and passing through the current point Vector3. * Example Playground https://playground.babylonjs.com/#R1F8YU#48 * @param plane defines the plane to project to * @param origin defines the origin of the projection ray * @returns the projected vector3 */ projectOnPlane(e, t) { const r = new this.constructor(); return this.projectOnPlaneToRef(e, t, r), r; } /** * Projects the current point Vector3 to a plane along a ray starting from a specified origin and passing through the current point Vector3. * Example Playground https://playground.babylonjs.com/#R1F8YU#49 * @param plane defines the plane to project to * @param origin defines the origin of the projection ray * @param result defines the Vector3 where to store the result * @returns result input */ projectOnPlaneToRef(e, t, r) { const n = e.normal, i = e.d, s = Kr.Vector3[0]; this.subtractToRef(t, s), s.normalize(); const a = S.Dot(s, n); if (Math.abs(a) < 1e-10) r.setAll(1 / 0); else { const f = -(S.Dot(t, n) + i) / a, o = s.scaleInPlace(f); t.addToRef(o, r); } return r; } /** * Returns true if the current Vector3 and the given vector coordinates are strictly equal * Example Playground https://playground.babylonjs.com/#R1F8YU#19 * @param otherVector defines the second operand * @returns true if both vectors are equals */ equals(e) { return e && this._x === e._x && this._y === e._y && this._z === e._z; } /** * Returns true if the current Vector3 and the given vector coordinates are distant less than epsilon * Example Playground https://playground.babylonjs.com/#R1F8YU#21 * @param otherVector defines the second operand * @param epsilon defines the minimal distance to define values as equals * @returns true if both vectors are distant less than epsilon */ equalsWithEpsilon(e, t = Dn) { return e && Xt.WithinEpsilon(this._x, e._x, t) && Xt.WithinEpsilon(this._y, e._y, t) && Xt.WithinEpsilon(this._z, e._z, t); } /** * Returns true if the current Vector3 coordinates equals the given floats * Example Playground https://playground.babylonjs.com/#R1F8YU#20 * @param x defines the x coordinate of the operand * @param y defines the y coordinate of the operand * @param z defines the z coordinate of the operand * @returns true if both vectors are equal */ equalsToFloats(e, t, r) { return this._x === e && this._y === t && this._z === r; } /** * Multiplies the current Vector3 coordinates by the given ones * Example Playground https://playground.babylonjs.com/#R1F8YU#32 * @param otherVector defines the second operand * @returns the current updated Vector3 */ multiplyInPlace(e) { return this._x *= e._x, this._y *= e._y, this._z *= e._z, this._isDirty = !0, this; } /** * Returns a new Vector3, result of the multiplication of the current Vector3 by the given vector * Example Playground https://playground.babylonjs.com/#R1F8YU#31 * @param otherVector defines the second operand * @returns the new Vector3 */ multiply(e) { return this.multiplyByFloats(e._x, e._y, e._z); } /** * Multiplies the current Vector3 by the given one and stores the result in the given vector "result" * Example Playground https://playground.babylonjs.com/#R1F8YU#33 * @param otherVector defines the second operand * @param result defines the Vector3 object where to store the result * @returns the result */ multiplyToRef(e, t) { return t.copyFromFloats(this._x * e._x, this._y * e._y, this._z * e._z); } /** * Returns a new Vector3 set with the result of the multiplication of the current Vector3 coordinates by the given floats * Example Playground https://playground.babylonjs.com/#R1F8YU#34 * @param x defines the x coordinate of the operand * @param y defines the y coordinate of the operand * @param z defines the z coordinate of the operand * @returns the new Vector3 */ multiplyByFloats(e, t, r) { return new this.constructor(this._x * e, this._y * t, this._z * r); } /** * Returns a new Vector3 set with the result of the division of the current Vector3 coordinates by the given ones * Example Playground https://playground.babylonjs.com/#R1F8YU#16 * @param otherVector defines the second operand * @returns the new Vector3 */ divide(e) { return new this.constructor(this._x / e._x, this._y / e._y, this._z / e._z); } /** * Divides the current Vector3 coordinates by the given ones and stores the result in the given vector "result" * Example Playground https://playground.babylonjs.com/#R1F8YU#18 * @param otherVector defines the second operand * @param result defines the Vector3 object where to store the result * @returns the result */ divideToRef(e, t) { return t.copyFromFloats(this._x / e._x, this._y / e._y, this._z / e._z); } /** * Divides the current Vector3 coordinates by the given ones. * Example Playground https://playground.babylonjs.com/#R1F8YU#17 * @param otherVector defines the second operand * @returns the current updated Vector3 */ divideInPlace(e) { return this.divideToRef(e, this); } /** * Updates the current Vector3 with the minimal coordinate values between its and the given vector ones * Example Playground https://playground.babylonjs.com/#R1F8YU#29 * @param other defines the second operand * @returns the current updated Vector3 */ minimizeInPlace(e) { return this.minimizeInPlaceFromFloats(e._x, e._y, e._z); } /** * Updates the current Vector3 with the maximal coordinate values between its and the given vector ones. * Example Playground https://playground.babylonjs.com/#R1F8YU#27 * @param other defines the second operand * @returns the current updated Vector3 */ maximizeInPlace(e) { return this.maximizeInPlaceFromFloats(e._x, e._y, e._z); } /** * Updates the current Vector3 with the minimal coordinate values between its and the given coordinates * Example Playground https://playground.babylonjs.com/#R1F8YU#30 * @param x defines the x coordinate of the operand * @param y defines the y coordinate of the operand * @param z defines the z coordinate of the operand * @returns the current updated Vector3 */ minimizeInPlaceFromFloats(e, t, r) { return e < this._x && (this.x = e), t < this._y && (this.y = t), r < this._z && (this.z = r), this; } /** * Updates the current Vector3 with the maximal coordinate values between its and the given coordinates. * Example Playground https://playground.babylonjs.com/#R1F8YU#28 * @param x defines the x coordinate of the operand * @param y defines the y coordinate of the operand * @param z defines the z coordinate of the operand * @returns the current updated Vector3 */ maximizeInPlaceFromFloats(e, t, r) { return e > this._x && (this.x = e), t > this._y && (this.y = t), r > this._z && (this.z = r), this; } /** * Due to float precision, scale of a mesh could be uniform but float values are off by a small fraction * Check if is non uniform within a certain amount of decimal places to account for this * @param epsilon the amount the values can differ * @returns if the vector is non uniform to a certain number of decimal places */ isNonUniformWithinEpsilon(e) { const t = Math.abs(this._x), r = Math.abs(this._y); if (!Xt.WithinEpsilon(t, r, e)) return !0; const n = Math.abs(this._z); return !Xt.WithinEpsilon(t, n, e) || !Xt.WithinEpsilon(r, n, e); } /** * Gets a boolean indicating that the vector is non uniform meaning x, y or z are not all the same */ get isNonUniform() { const e = Math.abs(this._x), t = Math.abs(this._y); if (e !== t) return !0; const r = Math.abs(this._z); return e !== r; } /** * Gets a new Vector3 from current Vector3 floored values * Example Playground https://playground.babylonjs.com/#R1F8YU#22 * @returns a new Vector3 */ floor() { return new this.constructor(Math.floor(this._x), Math.floor(this._y), Math.floor(this._z)); } /** * Gets a new Vector3 from current Vector3 fractional values * Example Playground https://playground.babylonjs.com/#R1F8YU#23 * @returns a new Vector3 */ fract() { return new this.constructor(this._x - Math.floor(this._x), this._y - Math.floor(this._y), this._z - Math.floor(this._z)); } // Properties /** * Gets the length of the Vector3 * Example Playground https://playground.babylonjs.com/#R1F8YU#25 * @returns the length of the Vector3 */ length() { return Math.sqrt(this._x * this._x + this._y * this._y + this._z * this._z); } /** * Gets the squared length of the Vector3 * Example Playground https://playground.babylonjs.com/#R1F8YU#26 * @returns squared length of the Vector3 */ lengthSquared() { return this._x * this._x + this._y * this._y + this._z * this._z; } /** * Gets a boolean indicating if the vector contains a zero in one of its components * Example Playground https://playground.babylonjs.com/#R1F8YU#1 */ get hasAZeroComponent() { return this._x * this._y * this._z === 0; } /** * Normalize the current Vector3. * Please note that this is an in place operation. * Example Playground https://playground.babylonjs.com/#R1F8YU#122 * @returns the current updated Vector3 */ normalize() { return this.normalizeFromLength(this.length()); } /** * Reorders the x y z properties of the vector in place * Example Playground https://playground.babylonjs.com/#R1F8YU#44 * @param order new ordering of the properties (eg. for vector 1,2,3 with "ZYX" will produce 3,2,1) * @returns the current updated vector */ reorderInPlace(e) { if (e = e.toLowerCase(), e === "xyz") return this; const t = Kr.Vector3[0].copyFrom(this); return this.x = t[e[0]], this.y = t[e[1]], this.z = t[e[2]], this; } /** * Rotates the vector around 0,0,0 by a quaternion * Example Playground https://playground.babylonjs.com/#R1F8YU#47 * @param quaternion the rotation quaternion * @param result vector to store the result * @returns the resulting vector */ rotateByQuaternionToRef(e, t) { return e.toRotationMatrix(Kr.Matrix[0]), S.TransformCoordinatesToRef(this, Kr.Matrix[0], t), t; } /** * Rotates a vector around a given point * Example Playground https://playground.babylonjs.com/#R1F8YU#46 * @param quaternion the rotation quaternion * @param point the point to rotate around * @param result vector to store the result * @returns the resulting vector */ rotateByQuaternionAroundPointToRef(e, t, r) { return this.subtractToRef(t, Kr.Vector3[0]), Kr.Vector3[0].rotateByQuaternionToRef(e, Kr.Vector3[0]), t.addToRef(Kr.Vector3[0], r), r; } /** * Returns a new Vector3 as the cross product of the current vector and the "other" one * The cross product is then orthogonal to both current and "other" * Example Playground https://playground.babylonjs.com/#R1F8YU#14 * @param other defines the right operand * @returns the cross product */ cross(e) { const t = new this.constructor(); return S.CrossToRef(this, e, t); } /** * Normalize the current Vector3 with the given input length. * Please note that this is an in place operation. * Example Playground https://playground.babylonjs.com/#R1F8YU#123 * @param len the length of the vector * @returns the current updated Vector3 */ normalizeFromLength(e) { return e === 0 || e === 1 ? this : this.scaleInPlace(1 / e); } /** * Normalize the current Vector3 to a new vector * Example Playground https://playground.babylonjs.com/#R1F8YU#124 * @returns the new Vector3 */ normalizeToNew() { const e = new this.constructor(0, 0, 0); return this.normalizeToRef(e), e; } /** * Normalize the current Vector3 to the reference * Example Playground https://playground.babylonjs.com/#R1F8YU#125 * @param reference define the Vector3 to update * @returns the updated Vector3 */ normalizeToRef(e) { const t = this.length(); return t === 0 || t === 1 ? e.copyFromFloats(this._x, this._y, this._z) : this.scaleToRef(1 / t, e); } /** * Creates a new Vector3 copied from the current Vector3 * Example Playground https://playground.babylonjs.com/#R1F8YU#11 * @returns the new Vector3 */ clone() { return new this.constructor(this._x, this._y, this._z); } /** * Copies the given vector coordinates to the current Vector3 ones * Example Playground https://playground.babylonjs.com/#R1F8YU#12 * @param source defines the source Vector3 * @returns the current updated Vector3 */ copyFrom(e) { return this.copyFromFloats(e._x, e._y, e._z); } /** * Copies the given floats to the current Vector3 coordinates * Example Playground https://playground.babylonjs.com/#R1F8YU#13 * @param x defines the x coordinate of the operand * @param y defines the y coordinate of the operand * @param z defines the z coordinate of the operand * @returns the current updated Vector3 */ copyFromFloats(e, t, r) { return this._x = e, this._y = t, this._z = r, this._isDirty = !0, this; } /** * Copies the given floats to the current Vector3 coordinates * Example Playground https://playground.babylonjs.com/#R1F8YU#58 * @param x defines the x coordinate of the operand * @param y defines the y coordinate of the operand * @param z defines the z coordinate of the operand * @returns the current updated Vector3 */ set(e, t, r) { return this.copyFromFloats(e, t, r); } /** * Copies the given float to the current Vector3 coordinates * Example Playground https://playground.babylonjs.com/#R1F8YU#59 * @param v defines the x, y and z coordinates of the operand * @returns the current updated Vector3 */ setAll(e) { return this._x = this._y = this._z = e, this._isDirty = !0, this; } // Statics /** * Get the clip factor between two vectors * Example Playground https://playground.babylonjs.com/#R1F8YU#126 * @param vector0 defines the first operand * @param vector1 defines the second operand * @param axis defines the axis to use * @param size defines the size along the axis * @returns the clip factor */ static GetClipFactor(e, t, r, n) { const i = S.Dot(e, r), s = S.Dot(t, r); return (i - n) / (i - s); } /** * Get angle between two vectors * Example Playground https://playground.babylonjs.com/#R1F8YU#86 * @param vector0 the starting point * @param vector1 the ending point * @param normal direction of the normal * @returns the angle between vector0 and vector1 */ static GetAngleBetweenVectors(e, t, r) { const n = e.normalizeToRef(Kr.Vector3[1]), i = t.normalizeToRef(Kr.Vector3[2]); let s = S.Dot(n, i); s = Xt.Clamp(s, -1, 1); const a = Math.acos(s), f = Kr.Vector3[3]; return S.CrossToRef(n, i, f), S.Dot(f, r) > 0 ? isNaN(a) ? 0 : a : isNaN(a) ? -Math.PI : -Math.acos(s); } /** * Get angle between two vectors projected on a plane * Example Playground https://playground.babylonjs.com/#R1F8YU#87 * Expectation compute time: 0.01 ms (median) and 0.02 ms (percentile 95%) * @param vector0 angle between vector0 and vector1 * @param vector1 angle between vector0 and vector1 * @param normal Normal of the projection plane * @returns the angle in radians (float) between vector0 and vector1 projected on the plane with the specified normal */ static GetAngleBetweenVectorsOnPlane(e, t, r) { Kr.Vector3[0].copyFrom(e); const n = Kr.Vector3[0]; Kr.Vector3[1].copyFrom(t); const i = Kr.Vector3[1]; Kr.Vector3[2].copyFrom(r); const s = Kr.Vector3[2], a = Kr.Vector3[3], f = Kr.Vector3[4]; n.normalize(), i.normalize(), s.normalize(), S.CrossToRef(s, n, a), S.CrossToRef(a, s, f); const o = Math.atan2(S.Dot(i, a), S.Dot(i, f)); return Xt.NormalizeRadians(o); } /** * Gets the rotation that aligns the roll axis (Y) to the line joining the start point to the target point and stores it in the ref Vector3 * Example PG https://playground.babylonjs.com/#R1F8YU#189 * @param start the starting point * @param target the target point * @param ref the vector3 to store the result * @returns ref in the form (pitch, yaw, 0) */ static PitchYawRollToMoveBetweenPointsToRef(e, t, r) { const n = ue.Vector3[0]; return t.subtractToRef(e, n), r._y = Math.atan2(n.x, n.z) || 0, r._x = Math.atan2(Math.sqrt(n.x ** 2 + n.z ** 2), n.y) || 0, r._z = 0, r._isDirty = !0, r; } /** * Gets the rotation that aligns the roll axis (Y) to the line joining the start point to the target point * Example PG https://playground.babylonjs.com/#R1F8YU#188 * @param start the starting point * @param target the target point * @returns the rotation in the form (pitch, yaw, 0) */ static PitchYawRollToMoveBetweenPoints(e, t) { const r = S.Zero(); return S.PitchYawRollToMoveBetweenPointsToRef(e, t, r); } /** * Slerp between two vectors. See also `SmoothToRef` * Slerp is a spherical linear interpolation * giving a slow in and out effect * Example Playground 1 https://playground.babylonjs.com/#R1F8YU#108 * Example Playground 2 https://playground.babylonjs.com/#R1F8YU#109 * @param vector0 Start vector * @param vector1 End vector * @param slerp amount (will be clamped between 0 and 1) * @param result The slerped vector */ static SlerpToRef(e, t, r, n) { r = Xt.Clamp(r, 0, 1); const i = Kr.Vector3[0], s = Kr.Vector3[1]; i.copyFrom(e); const a = i.length(); i.normalizeFromLength(a), s.copyFrom(t); const f = s.length(); s.normalizeFromLength(f); const o = S.Dot(i, s); let d, v; if (o < 1 - Dn) { const u = Math.acos(o), l = 1 / Math.sin(u); d = Math.sin((1 - r) * u) * l, v = Math.sin(r * u) * l; } else d = 1 - r, v = r; return i.scaleInPlace(d), s.scaleInPlace(v), n.copyFrom(i).addInPlace(s), n.scaleInPlace(Xt.Lerp(a, f, r)), n; } /** * Smooth interpolation between two vectors using Slerp * Example Playground https://playground.babylonjs.com/#R1F8YU#110 * @param source source vector * @param goal goal vector * @param deltaTime current interpolation frame * @param lerpTime total interpolation time * @param result the smoothed vector */ static SmoothToRef(e, t, r, n, i) { return S.SlerpToRef(e, t, n === 0 ? 1 : r / n, i), i; } /** * Returns a new Vector3 set from the index "offset" of the given array * Example Playground https://playground.babylonjs.com/#R1F8YU#83 * @param array defines the source array * @param offset defines the offset in the source array * @returns the new Vector3 */ static FromArray(e, t = 0) { return new S(e[t], e[t + 1], e[t + 2]); } /** * Returns a new Vector3 set from the index "offset" of the given Float32Array * @param array defines the source array * @param offset defines the offset in the source array * @returns the new Vector3 * @deprecated Please use FromArray instead. */ static FromFloatArray(e, t) { return S.FromArray(e, t); } /** * Sets the given vector "result" with the element values from the index "offset" of the given array * Example Playground https://playground.babylonjs.com/#R1F8YU#84 * @param array defines the source array * @param offset defines the offset in the source array * @param result defines the Vector3 where to store the result * @returns result input */ static FromArrayToRef(e, t, r) { return r._x = e[t], r._y = e[t + 1], r._z = e[t + 2], r._isDirty = !0, r; } /** * Sets the given vector "result" with the element values from the index "offset" of the given Float32Array * @param array defines the source array * @param offset defines the offset in the source array * @param result defines the Vector3 where to store the result * @deprecated Please use FromArrayToRef instead. */ static FromFloatArrayToRef(e, t, r) { return S.FromArrayToRef(e, t, r); } /** * Sets the given vector "result" with the given floats. * Example Playground https://playground.babylonjs.com/#R1F8YU#85 * @param x defines the x coordinate of the source * @param y defines the y coordinate of the source * @param z defines the z coordinate of the source * @param result defines the Vector3 where to store the result */ static FromFloatsToRef(e, t, r, n) { return n.copyFromFloats(e, t, r), n; } /** * Returns a new Vector3 set to (0.0, 0.0, 0.0) * @returns a new empty Vector3 */ static Zero() { return new S(0, 0, 0); } /** * Returns a new Vector3 set to (1.0, 1.0, 1.0) * @returns a new Vector3 */ static One() { return new S(1, 1, 1); } /** * Returns a new Vector3 set to (0.0, 1.0, 0.0) * Example Playground https://playground.babylonjs.com/#R1F8YU#71 * @returns a new up Vector3 */ static Up() { return new S(0, 1, 0); } /** * Gets an up Vector3 that must not be updated */ static get UpReadOnly() { return S._UpReadOnly; } /** * Gets a down Vector3 that must not be updated */ static get DownReadOnly() { return S._DownReadOnly; } /** * Gets a right Vector3 that must not be updated */ static get RightReadOnly() { return S._RightReadOnly; } /** * Gets a left Vector3 that must not be updated */ static get LeftReadOnly() { return S._LeftReadOnly; } /** * Gets a forward Vector3 that must not be updated */ static get LeftHandedForwardReadOnly() { return S._LeftHandedForwardReadOnly; } /** * Gets a forward Vector3 that must not be updated */ static get RightHandedForwardReadOnly() { return S._RightHandedForwardReadOnly; } /** * Gets a backward Vector3 that must not be updated */ static get LeftHandedBackwardReadOnly() { return S._LeftHandedBackwardReadOnly; } /** * Gets a backward Vector3 that must not be updated */ static get RightHandedBackwardReadOnly() { return S._RightHandedBackwardReadOnly; } /** * Gets a zero Vector3 that must not be updated */ static get ZeroReadOnly() { return S._ZeroReadOnly; } /** * Gets a one Vector3 that must not be updated */ static get OneReadOnly() { return S._OneReadOnly; } /** * Returns a new Vector3 set to (0.0, -1.0, 0.0) * Example Playground https://playground.babylonjs.com/#R1F8YU#71 * @returns a new down Vector3 */ static Down() { return new S(0, -1, 0); } /** * Returns a new Vector3 set to (0.0, 0.0, 1.0) * Example Playground https://playground.babylonjs.com/#R1F8YU#71 * @param rightHandedSystem is the scene right-handed (negative z) * @returns a new forward Vector3 */ static Forward(e = !1) { return new S(0, 0, e ? -1 : 1); } /** * Returns a new Vector3 set to (0.0, 0.0, -1.0) * Example Playground https://playground.babylonjs.com/#R1F8YU#71 * @param rightHandedSystem is the scene right-handed (negative-z) * @returns a new Backward Vector3 */ static Backward(e = !1) { return new S(0, 0, e ? 1 : -1); } /** * Returns a new Vector3 set to (1.0, 0.0, 0.0) * Example Playground https://playground.babylonjs.com/#R1F8YU#71 * @returns a new right Vector3 */ static Right() { return new S(1, 0, 0); } /** * Returns a new Vector3 set to (-1.0, 0.0, 0.0) * Example Playground https://playground.babylonjs.com/#R1F8YU#71 * @returns a new left Vector3 */ static Left() { return new S(-1, 0, 0); } /** * Returns a new Vector3 with random values between min and max * @param min the minimum random value * @param max the maximum random value * @returns a Vector3 with random values between min and max */ static Random(e = 0, t = 1) { return new S(Xt.RandomRange(e, t), Xt.RandomRange(e, t), Xt.RandomRange(e, t)); } /** * Returns a new Vector3 set with the result of the transformation by the given matrix of the given vector. * This method computes transformed coordinates only, not transformed direction vectors (ie. it takes translation in account) * Example Playground https://playground.babylonjs.com/#R1F8YU#111 * @param vector defines the Vector3 to transform * @param transformation defines the transformation matrix * @returns the transformed Vector3 */ static TransformCoordinates(e, t) { const r = S.Zero(); return S.TransformCoordinatesToRef(e, t, r), r; } /** * Sets the given vector "result" coordinates with the result of the transformation by the given matrix of the given vector * This method computes transformed coordinates only, not transformed direction vectors (ie. it takes translation in account) * Example Playground https://playground.babylonjs.com/#R1F8YU#113 * @param vector defines the Vector3 to transform * @param transformation defines the transformation matrix * @param result defines the Vector3 where to store the result * @returns result input */ static TransformCoordinatesToRef(e, t, r) { return S.TransformCoordinatesFromFloatsToRef(e._x, e._y, e._z, t, r), r; } /** * Sets the given vector "result" coordinates with the result of the transformation by the given matrix of the given floats (x, y, z) * This method computes transformed coordinates only, not transformed direction vectors * Example Playground https://playground.babylonjs.com/#R1F8YU#115 * @param x define the x coordinate of the source vector * @param y define the y coordinate of the source vector * @param z define the z coordinate of the source vector * @param transformation defines the transformation matrix * @param result defines the Vector3 where to store the result * @returns result input */ static TransformCoordinatesFromFloatsToRef(e, t, r, n, i) { const s = n.m, a = e * s[0] + t * s[4] + r * s[8] + s[12], f = e * s[1] + t * s[5] + r * s[9] + s[13], o = e * s[2] + t * s[6] + r * s[10] + s[14], d = 1 / (e * s[3] + t * s[7] + r * s[11] + s[15]); return i._x = a * d, i._y = f * d, i._z = o * d, i._isDirty = !0, i; } /** * Returns a new Vector3 set with the result of the normal transformation by the given matrix of the given vector * This methods computes transformed normalized direction vectors only (ie. it does not apply translation) * Example Playground https://playground.babylonjs.com/#R1F8YU#112 * @param vector defines the Vector3 to transform * @param transformation defines the transformation matrix * @returns the new Vector3 */ static TransformNormal(e, t) { const r = S.Zero(); return S.TransformNormalToRef(e, t, r), r; } /** * Sets the given vector "result" with the result of the normal transformation by the given matrix of the given vector * This methods computes transformed normalized direction vectors only (ie. it does not apply translation) * Example Playground https://playground.babylonjs.com/#R1F8YU#114 * @param vector defines the Vector3 to transform * @param transformation defines the transformation matrix * @param result defines the Vector3 where to store the result * @returns result input */ static TransformNormalToRef(e, t, r) { return this.TransformNormalFromFloatsToRef(e._x, e._y, e._z, t, r), r; } /** * Sets the given vector "result" with the result of the normal transformation by the given matrix of the given floats (x, y, z) * This methods computes transformed normalized direction vectors only (ie. it does not apply translation) * Example Playground https://playground.babylonjs.com/#R1F8YU#116 * @param x define the x coordinate of the source vector * @param y define the y coordinate of the source vector * @param z define the z coordinate of the source vector * @param transformation defines the transformation matrix * @param result defines the Vector3 where to store the result * @returns result input */ static TransformNormalFromFloatsToRef(e, t, r, n, i) { const s = n.m; return i._x = e * s[0] + t * s[4] + r * s[8], i._y = e * s[1] + t * s[5] + r * s[9], i._z = e * s[2] + t * s[6] + r * s[10], i._isDirty = !0, i; } /** * Returns a new Vector3 located for "amount" on the CatmullRom interpolation spline defined by the vectors "value1", "value2", "value3", "value4" * Example Playground https://playground.babylonjs.com/#R1F8YU#69 * @param value1 defines the first control point * @param value2 defines the second control point * @param value3 defines the third control point * @param value4 defines the fourth control point * @param amount defines the amount on the spline to use * @returns the new Vector3 */ static CatmullRom(e, t, r, n, i) { const s = i * i, a = i * s, f = 0.5 * (2 * t._x + (-e._x + r._x) * i + (2 * e._x - 5 * t._x + 4 * r._x - n._x) * s + (-e._x + 3 * t._x - 3 * r._x + n._x) * a), o = 0.5 * (2 * t._y + (-e._y + r._y) * i + (2 * e._y - 5 * t._y + 4 * r._y - n._y) * s + (-e._y + 3 * t._y - 3 * r._y + n._y) * a), d = 0.5 * (2 * t._z + (-e._z + r._z) * i + (2 * e._z - 5 * t._z + 4 * r._z - n._z) * s + (-e._z + 3 * t._z - 3 * r._z + n._z) * a); return new e.constructor(f, o, d); } /** * Returns a new Vector3 set with the coordinates of "value", if the vector "value" is in the cube defined by the vectors "min" and "max" * If a coordinate value of "value" is lower than one of the "min" coordinate, then this "value" coordinate is set with the "min" one * If a coordinate value of "value" is greater than one of the "max" coordinate, then this "value" coordinate is set with the "max" one * Example Playground https://playground.babylonjs.com/#R1F8YU#76 * @param value defines the current value * @param min defines the lower range value * @param max defines the upper range value * @returns the new Vector3 */ static Clamp(e, t, r) { const n = new e.constructor(); return S.ClampToRef(e, t, r, n), n; } /** * Sets the given vector "result" with the coordinates of "value", if the vector "value" is in the cube defined by the vectors "min" and "max" * If a coordinate value of "value" is lower than one of the "min" coordinate, then this "value" coordinate is set with the "min" one * If a coordinate value of "value" is greater than one of the "max" coordinate, then this "value" coordinate is set with the "max" one * Example Playground https://playground.babylonjs.com/#R1F8YU#77 * @param value defines the current value * @param min defines the lower range value * @param max defines the upper range value * @param result defines the Vector3 where to store the result * @returns result input */ static ClampToRef(e, t, r, n) { let i = e._x; i = i > r._x ? r._x : i, i = i < t._x ? t._x : i; let s = e._y; s = s > r._y ? r._y : s, s = s < t._y ? t._y : s; let a = e._z; return a = a > r._z ? r._z : a, a = a < t._z ? t._z : a, n.copyFromFloats(i, s, a), n; } /** * Checks if a given vector is inside a specific range * Example Playground https://playground.babylonjs.com/#R1F8YU#75 * @param v defines the vector to test * @param min defines the minimum range * @param max defines the maximum range */ static CheckExtends(e, t, r) { t.minimizeInPlace(e), r.maximizeInPlace(e); } /** * Returns a new Vector3 located for "amount" (float) on the Hermite interpolation spline defined by the vectors "value1", "tangent1", "value2", "tangent2" * Example Playground https://playground.babylonjs.com/#R1F8YU#89 * @param value1 defines the first control point * @param tangent1 defines the first tangent vector * @param value2 defines the second control point * @param tangent2 defines the second tangent vector * @param amount defines the amount on the interpolation spline (between 0 and 1) * @returns the new Vector3 */ static Hermite(e, t, r, n, i) { const s = i * i, a = i * s, f = 2 * a - 3 * s + 1, o = -2 * a + 3 * s, d = a - 2 * s + i, v = a - s, u = e._x * f + r._x * o + t._x * d + n._x * v, l = e._y * f + r._y * o + t._y * d + n._y * v, P = e._z * f + r._z * o + t._z * d + n._z * v; return new e.constructor(u, l, P); } /** * Returns a new Vector3 which is the 1st derivative of the Hermite spline defined by the vectors "value1", "value2", "tangent1", "tangent2". * Example Playground https://playground.babylonjs.com/#R1F8YU#90 * @param value1 defines the first control point * @param tangent1 defines the first tangent * @param value2 defines the second control point * @param tangent2 defines the second tangent * @param time define where the derivative must be done * @returns 1st derivative */ static Hermite1stDerivative(e, t, r, n, i) { const s = new e.constructor(); return this.Hermite1stDerivativeToRef(e, t, r, n, i, s), s; } /** * Update a Vector3 with the 1st derivative of the Hermite spline defined by the vectors "value1", "value2", "tangent1", "tangent2". * Example Playground https://playground.babylonjs.com/#R1F8YU#91 * @param value1 defines the first control point * @param tangent1 defines the first tangent * @param value2 defines the second control point * @param tangent2 defines the second tangent * @param time define where the derivative must be done * @param result define where to store the derivative * @returns result input */ static Hermite1stDerivativeToRef(e, t, r, n, i, s) { const a = i * i; return s._x = (a - i) * 6 * e._x + (3 * a - 4 * i + 1) * t._x + (-a + i) * 6 * r._x + (3 * a - 2 * i) * n._x, s._y = (a - i) * 6 * e._y + (3 * a - 4 * i + 1) * t._y + (-a + i) * 6 * r._y + (3 * a - 2 * i) * n._y, s._z = (a - i) * 6 * e._z + (3 * a - 4 * i + 1) * t._z + (-a + i) * 6 * r._z + (3 * a - 2 * i) * n._z, s._isDirty = !0, s; } /** * Returns a new Vector3 located for "amount" (float) on the linear interpolation between the vectors "start" and "end" * Example Playground https://playground.babylonjs.com/#R1F8YU#95 * @param start defines the start value * @param end defines the end value * @param amount max defines amount between both (between 0 and 1) * @returns the new Vector3 */ static Lerp(e, t, r) { const n = new e.constructor(0, 0, 0); return S.LerpToRef(e, t, r, n), n; } /** * Sets the given vector "result" with the result of the linear interpolation from the vector "start" for "amount" to the vector "end" * Example Playground https://playground.babylonjs.com/#R1F8YU#93 * @param start defines the start value * @param end defines the end value * @param amount max defines amount between both (between 0 and 1) * @param result defines the Vector3 where to store the result * @returns result input */ static LerpToRef(e, t, r, n) { return n._x = e._x + (t._x - e._x) * r, n._y = e._y + (t._y - e._y) * r, n._z = e._z + (t._z - e._z) * r, n._isDirty = !0, n; } /** * Returns the dot product (float) between the vectors "left" and "right" * Example Playground https://playground.babylonjs.com/#R1F8YU#82 * @param left defines the left operand * @param right defines the right operand * @returns the dot product */ static Dot(e, t) { return e._x * t._x + e._y * t._y + e._z * t._z; } /** * Returns the dot product (float) between the current vectors and "otherVector" * @param otherVector defines the right operand * @returns the dot product */ dot(e) { return this._x * e._x + this._y * e._y + this._z * e._z; } /** * Returns a new Vector3 as the cross product of the vectors "left" and "right" * The cross product is then orthogonal to both "left" and "right" * Example Playground https://playground.babylonjs.com/#R1F8YU#15 * @param left defines the left operand * @param right defines the right operand * @returns the cross product */ static Cross(e, t) { const r = new e.constructor(); return S.CrossToRef(e, t, r), r; } /** * Sets the given vector "result" with the cross product of "left" and "right" * The cross product is then orthogonal to both "left" and "right" * Example Playground https://playground.babylonjs.com/#R1F8YU#78 * @param left defines the left operand * @param right defines the right operand * @param result defines the Vector3 where to store the result * @returns result input */ static CrossToRef(e, t, r) { const n = e._y * t._z - e._z * t._y, i = e._z * t._x - e._x * t._z, s = e._x * t._y - e._y * t._x; return r.copyFromFloats(n, i, s), r; } /** * Returns a new Vector3 as the normalization of the given vector * Example Playground https://playground.babylonjs.com/#R1F8YU#98 * @param vector defines the Vector3 to normalize * @returns the new Vector3 */ static Normalize(e) { const t = S.Zero(); return S.NormalizeToRef(e, t), t; } /** * Sets the given vector "result" with the normalization of the given first vector * Example Playground https://playground.babylonjs.com/#R1F8YU#98 * @param vector defines the Vector3 to normalize * @param result defines the Vector3 where to store the result * @returns result input */ static NormalizeToRef(e, t) { return e.normalizeToRef(t), t; } /** * Project a Vector3 onto screen space * Example Playground https://playground.babylonjs.com/#R1F8YU#101 * @param vector defines the Vector3 to project * @param world defines the world matrix to use * @param transform defines the transform (view x projection) matrix to use * @param viewport defines the screen viewport to use * @returns the new Vector3 */ static Project(e, t, r, n) { const i = new e.constructor(); return S.ProjectToRef(e, t, r, n, i), i; } /** * Project a Vector3 onto screen space to reference * Example Playground https://playground.babylonjs.com/#R1F8YU#102 * @param vector defines the Vector3 to project * @param world defines the world matrix to use * @param transform defines the transform (view x projection) matrix to use * @param viewport defines the screen viewport to use * @param result the vector in which the screen space will be stored * @returns result input */ static ProjectToRef(e, t, r, n, i) { const s = n.width, a = n.height, f = n.x, o = n.y, d = Kr.Matrix[1]; he.FromValuesToRef(s / 2, 0, 0, 0, 0, -a / 2, 0, 0, 0, 0, 0.5, 0, f + s / 2, a / 2 + o, 0.5, 1, d); const v = Kr.Matrix[0]; return t.multiplyToRef(r, v), v.multiplyToRef(d, v), S.TransformCoordinatesToRef(e, v, i), i; } /** * Reflects a vector off the plane defined by a normalized normal * @param inDirection defines the vector direction * @param normal defines the normal - Must be normalized * @returns the resulting vector */ static Reflect(e, t) { return this.ReflectToRef(e, t, new S()); } /** * Reflects a vector off the plane defined by a normalized normal to reference * @param inDirection defines the vector direction * @param normal defines the normal - Must be normalized * @param result defines the Vector3 where to store the result * @returns the resulting vector */ static ReflectToRef(e, t, r) { const n = ue.Vector3[0]; return n.copyFrom(t).scaleInPlace(2 * S.Dot(e, t)), r.copyFrom(e).subtractInPlace(n); } /** * @internal */ static _UnprojectFromInvertedMatrixToRef(e, t, r) { S.TransformCoordinatesToRef(e, t, r); const n = t.m, i = e._x * n[3] + e._y * n[7] + e._z * n[11] + n[15]; return Xt.WithinEpsilon(i, 1) && r.scaleInPlace(1 / i), r; } /** * Unproject from screen space to object space * Example Playground https://playground.babylonjs.com/#R1F8YU#121 * @param source defines the screen space Vector3 to use * @param viewportWidth defines the current width of the viewport * @param viewportHeight defines the current height of the viewport * @param world defines the world matrix to use (can be set to Identity to go to world space) * @param transform defines the transform (view x projection) matrix to use * @returns the new Vector3 */ static UnprojectFromTransform(e, t, r, n, i) { return this.Unproject(e, t, r, n, i, he.IdentityReadOnly); } /** * Unproject from screen space to object space * Example Playground https://playground.babylonjs.com/#R1F8YU#117 * @param source defines the screen space Vector3 to use * @param viewportWidth defines the current width of the viewport * @param viewportHeight defines the current height of the viewport * @param world defines the world matrix to use (can be set to Identity to go to world space) * @param view defines the view matrix to use * @param projection defines the projection matrix to use * @returns the new Vector3 */ static Unproject(e, t, r, n, i, s) { const a = new e.constructor(); return S.UnprojectToRef(e, t, r, n, i, s, a), a; } /** * Unproject from screen space to object space * Example Playground https://playground.babylonjs.com/#R1F8YU#119 * @param source defines the screen space Vector3 to use * @param viewportWidth defines the current width of the viewport * @param viewportHeight defines the current height of the viewport * @param world defines the world matrix to use (can be set to Identity to go to world space) * @param view defines the view matrix to use * @param projection defines the projection matrix to use * @param result defines the Vector3 where to store the result * @returns result input */ static UnprojectToRef(e, t, r, n, i, s, a) { return S.UnprojectFloatsToRef(e._x, e._y, e._z, t, r, n, i, s, a), a; } /** * Unproject from screen space to object space * Example Playground https://playground.babylonjs.com/#R1F8YU#120 * @param sourceX defines the screen space x coordinate to use * @param sourceY defines the screen space y coordinate to use * @param sourceZ defines the screen space z coordinate to use * @param viewportWidth defines the current width of the viewport * @param viewportHeight defines the current height of the viewport * @param world defines the world matrix to use (can be set to Identity to go to world space) * @param view defines the view matrix to use * @param projection defines the projection matrix to use * @param result defines the Vector3 where to store the result * @returns result input */ static UnprojectFloatsToRef(e, t, r, n, i, s, a, f, o) { var d; const v = Kr.Matrix[0]; s.multiplyToRef(a, v), v.multiplyToRef(f, v), v.invert(); const u = Kr.Vector3[0]; return u.x = e / n * 2 - 1, u.y = -(t / i * 2 - 1), !((d = gr.LastCreatedEngine) === null || d === void 0) && d.isNDCHalfZRange ? u.z = r : u.z = 2 * r - 1, S._UnprojectFromInvertedMatrixToRef(u, v, o), o; } /** * Gets the minimal coordinate values between two Vector3 * Example Playground https://playground.babylonjs.com/#R1F8YU#97 * @param left defines the first operand * @param right defines the second operand * @returns the new Vector3 */ static Minimize(e, t) { const r = new e.constructor(); return r.copyFrom(e), r.minimizeInPlace(t), r; } /** * Gets the maximal coordinate values between two Vector3 * Example Playground https://playground.babylonjs.com/#R1F8YU#96 * @param left defines the first operand * @param right defines the second operand * @returns the new Vector3 */ static Maximize(e, t) { const r = new e.constructor(); return r.copyFrom(e), r.maximizeInPlace(t), r; } /** * Returns the distance between the vectors "value1" and "value2" * Example Playground https://playground.babylonjs.com/#R1F8YU#81 * @param value1 defines the first operand * @param value2 defines the second operand * @returns the distance */ static Distance(e, t) { return Math.sqrt(S.DistanceSquared(e, t)); } /** * Returns the squared distance between the vectors "value1" and "value2" * Example Playground https://playground.babylonjs.com/#R1F8YU#80 * @param value1 defines the first operand * @param value2 defines the second operand * @returns the squared distance */ static DistanceSquared(e, t) { const r = e._x - t._x, n = e._y - t._y, i = e._z - t._z; return r * r + n * n + i * i; } /** * Projects "vector" on the triangle determined by its extremities "p0", "p1" and "p2", stores the result in "ref" * and returns the distance to the projected point. * Example Playground https://playground.babylonjs.com/#R1F8YU#104 * From http://citeseerx.ist.psu.edu/viewdoc/download?doi=10.1.1.104.4264&rep=rep1&type=pdf * * @param vector the vector to get distance from * @param p0 extremity of the triangle * @param p1 extremity of the triangle * @param p2 extremity of the triangle * @param ref variable to store the result to * @returns The distance between "ref" and "vector" */ static ProjectOnTriangleToRef(e, t, r, n, i) { const s = Kr.Vector3[0], a = Kr.Vector3[1], f = Kr.Vector3[2], o = Kr.Vector3[3], d = Kr.Vector3[4]; r.subtractToRef(t, s), n.subtractToRef(t, a), n.subtractToRef(r, f); const v = s.length(), u = a.length(), l = f.length(); if (v < Dn || u < Dn || l < Dn) return i.copyFrom(t), S.Distance(e, t); e.subtractToRef(t, d), S.CrossToRef(s, a, o); const P = o.length(); if (P < Dn) return i.copyFrom(t), S.Distance(e, t); o.normalizeFromLength(P); let p = d.length(); if (p < Dn) return i.copyFrom(t), 0; d.normalizeFromLength(p); const c = S.Dot(o, d), H = Kr.Vector3[5], T = Kr.Vector3[6]; H.copyFrom(o).scaleInPlace(-p * c), T.copyFrom(e).addInPlace(H); const q = Kr.Vector3[4], b = Kr.Vector3[5], j = Kr.Vector3[7], w = Kr.Vector3[8]; q.copyFrom(s).scaleInPlace(1 / v), w.copyFrom(a).scaleInPlace(1 / u), q.addInPlace(w).scaleInPlace(-1), b.copyFrom(s).scaleInPlace(-1 / v), w.copyFrom(f).scaleInPlace(1 / l), b.addInPlace(w).scaleInPlace(-1), j.copyFrom(f).scaleInPlace(-1 / l), w.copyFrom(a).scaleInPlace(-1 / u), j.addInPlace(w).scaleInPlace(-1); const m = Kr.Vector3[9]; let I; m.copyFrom(T).subtractInPlace(t), S.CrossToRef(q, m, w), I = S.Dot(w, o); const N = I; m.copyFrom(T).subtractInPlace(r), S.CrossToRef(b, m, w), I = S.Dot(w, o); const k = I; m.copyFrom(T).subtractInPlace(n), S.CrossToRef(j, m, w), I = S.Dot(w, o); const R = I, y = Kr.Vector3[10]; let O, Y; N > 0 && k < 0 ? (y.copyFrom(s), O = t, Y = r) : k > 0 && R < 0 ? (y.copyFrom(f), O = r, Y = n) : (y.copyFrom(a).scaleInPlace(-1), O = n, Y = t); const ee = Kr.Vector3[9], Z = Kr.Vector3[4]; if (O.subtractToRef(T, w), Y.subtractToRef(T, ee), S.CrossToRef(w, ee, Z), !(S.Dot(Z, o) < 0)) return i.copyFrom(T), Math.abs(p * c); const fe = Kr.Vector3[5]; S.CrossToRef(y, Z, fe), fe.normalize(); const _ = Kr.Vector3[9]; _.copyFrom(O).subtractInPlace(T); const G = _.length(); if (G < Dn) return i.copyFrom(O), S.Distance(e, O); _.normalizeFromLength(G); const L = S.Dot(fe, _), $ = Kr.Vector3[7]; $.copyFrom(T).addInPlace(fe.scaleInPlace(G * L)), w.copyFrom($).subtractInPlace(O), p = y.length(), y.normalizeFromLength(p); let ae = S.Dot(w, y) / Math.max(p, Dn); return ae = Xt.Clamp(ae, 0, 1), $.copyFrom(O).addInPlace(y.scaleInPlace(ae * p)), i.copyFrom($), S.Distance(e, $); } /** * Returns a new Vector3 located at the center between "value1" and "value2" * Example Playground https://playground.babylonjs.com/#R1F8YU#72 * @param value1 defines the first operand * @param value2 defines the second operand * @returns the new Vector3 */ static Center(e, t) { return S.CenterToRef(e, t, S.Zero()); } /** * Gets the center of the vectors "value1" and "value2" and stores the result in the vector "ref" * Example Playground https://playground.babylonjs.com/#R1F8YU#73 * @param value1 defines first vector * @param value2 defines second vector * @param ref defines third vector * @returns ref */ static CenterToRef(e, t, r) { return r.copyFromFloats((e._x + t._x) / 2, (e._y + t._y) / 2, (e._z + t._z) / 2); } /** * Given three orthogonal normalized left-handed oriented Vector3 axis in space (target system), * RotationFromAxis() returns the rotation Euler angles (ex : rotation.x, rotation.y, rotation.z) to apply * to something in order to rotate it from its local system to the given target system * Note: axis1, axis2 and axis3 are normalized during this operation * Example Playground https://playground.babylonjs.com/#R1F8YU#106 * @param axis1 defines the first axis * @param axis2 defines the second axis * @param axis3 defines the third axis * @returns a new Vector3 * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/target_align */ static RotationFromAxis(e, t, r) { const n = new e.constructor(); return S.RotationFromAxisToRef(e, t, r, n), n; } /** * The same than RotationFromAxis but updates the given ref Vector3 parameter instead of returning a new Vector3 * Example Playground https://playground.babylonjs.com/#R1F8YU#107 * @param axis1 defines the first axis * @param axis2 defines the second axis * @param axis3 defines the third axis * @param ref defines the Vector3 where to store the result * @returns result input */ static RotationFromAxisToRef(e, t, r, n) { const i = Kr.Quaternion[0]; return Ze.RotationQuaternionFromAxisToRef(e, t, r, i), i.toEulerAnglesToRef(n), n; } } S._UpReadOnly = S.Up(); S._DownReadOnly = S.Down(); S._LeftHandedForwardReadOnly = S.Forward(!1); S._RightHandedForwardReadOnly = S.Forward(!0); S._LeftHandedBackwardReadOnly = S.Backward(!1); S._RightHandedBackwardReadOnly = S.Backward(!0); S._RightReadOnly = S.Right(); S._LeftReadOnly = S.Left(); S._ZeroReadOnly = S.Zero(); S._OneReadOnly = S.One(); class Ir { /** * Creates a Vector4 object from the given floats. * @param x x value of the vector * @param y y value of the vector * @param z z value of the vector * @param w w value of the vector */ constructor(e = 0, t = 0, r = 0, n = 0) { this.x = e, this.y = t, this.z = r, this.w = n; } /** * Returns the string with the Vector4 coordinates. * @returns a string containing all the vector values */ toString() { return `{X: ${this.x} Y: ${this.y} Z: ${this.z} W: ${this.w}}`; } /** * Returns the string "Vector4". * @returns "Vector4" */ getClassName() { return "Vector4"; } /** * Returns the Vector4 hash code. * @returns a unique hash code */ getHashCode() { const e = Kl(this.x), t = Kl(this.y), r = Kl(this.z), n = Kl(this.w); let i = e; return i = i * 397 ^ t, i = i * 397 ^ r, i = i * 397 ^ n, i; } // Operators /** * Returns a new array populated with 4 elements : the Vector4 coordinates. * @returns the resulting array */ asArray() { const e = []; return this.toArray(e, 0), e; } /** * Populates the given array from the given index with the Vector4 coordinates. * @param array array to populate * @param index index of the array to start at (default: 0) * @returns the Vector4. */ toArray(e, t) { return t === void 0 && (t = 0), e[t] = this.x, e[t + 1] = this.y, e[t + 2] = this.z, e[t + 3] = this.w, this; } /** * Update the current vector from an array * @param array defines the destination array * @param index defines the offset in the destination array * @returns the current Vector3 */ fromArray(e, t = 0) { return Ir.FromArrayToRef(e, t, this), this; } /** * Adds the given vector to the current Vector4. * @param otherVector the vector to add * @returns the updated Vector4. */ addInPlace(e) { return this.x += e.x, this.y += e.y, this.z += e.z, this.w += e.w, this; } /** * Returns a new Vector4 as the result of the addition of the current Vector4 and the given one. * @param otherVector the vector to add * @returns the resulting vector */ add(e) { return new this.constructor(this.x + e.x, this.y + e.y, this.z + e.z, this.w + e.w); } /** * Updates the given vector "result" with the result of the addition of the current Vector4 and the given one. * @param otherVector the vector to add * @param result the vector to store the result * @returns result input */ addToRef(e, t) { return t.x = this.x + e.x, t.y = this.y + e.y, t.z = this.z + e.z, t.w = this.w + e.w, t; } /** * Subtract in place the given vector from the current Vector4. * @param otherVector the vector to subtract * @returns the updated Vector4. */ subtractInPlace(e) { return this.x -= e.x, this.y -= e.y, this.z -= e.z, this.w -= e.w, this; } /** * Returns a new Vector4 with the result of the subtraction of the given vector from the current Vector4. * @param otherVector the vector to add * @returns the new vector with the result */ subtract(e) { return new this.constructor(this.x - e.x, this.y - e.y, this.z - e.z, this.w - e.w); } /** * Sets the given vector "result" with the result of the subtraction of the given vector from the current Vector4. * @param otherVector the vector to subtract * @param result the vector to store the result * @returns result input */ subtractToRef(e, t) { return t.x = this.x - e.x, t.y = this.y - e.y, t.z = this.z - e.z, t.w = this.w - e.w, t; } /** * Returns a new Vector4 set with the result of the subtraction of the given floats from the current Vector4 coordinates. */ /** * Returns a new Vector4 set with the result of the subtraction of the given floats from the current Vector4 coordinates. * @param x value to subtract * @param y value to subtract * @param z value to subtract * @param w value to subtract * @returns new vector containing the result */ subtractFromFloats(e, t, r, n) { return new this.constructor(this.x - e, this.y - t, this.z - r, this.w - n); } /** * Sets the given vector "result" set with the result of the subtraction of the given floats from the current Vector4 coordinates. * @param x value to subtract * @param y value to subtract * @param z value to subtract * @param w value to subtract * @param result the vector to store the result in * @returns result input */ subtractFromFloatsToRef(e, t, r, n, i) { return i.x = this.x - e, i.y = this.y - t, i.z = this.z - r, i.w = this.w - n, i; } /** * Returns a new Vector4 set with the current Vector4 negated coordinates. * @returns a new vector with the negated values */ negate() { return new this.constructor(-this.x, -this.y, -this.z, -this.w); } /** * Negate this vector in place * @returns this */ negateInPlace() { return this.x *= -1, this.y *= -1, this.z *= -1, this.w *= -1, this; } /** * Negate the current Vector4 and stores the result in the given vector "result" coordinates * @param result defines the Vector3 object where to store the result * @returns the result */ negateToRef(e) { return e.copyFromFloats(this.x * -1, this.y * -1, this.z * -1, this.w * -1); } /** * Multiplies the current Vector4 coordinates by scale (float). * @param scale the number to scale with * @returns the updated Vector4. */ scaleInPlace(e) { return this.x *= e, this.y *= e, this.z *= e, this.w *= e, this; } /** * Returns a new Vector4 set with the current Vector4 coordinates multiplied by scale (float). * @param scale the number to scale with * @returns a new vector with the result */ scale(e) { return new this.constructor(this.x * e, this.y * e, this.z * e, this.w * e); } /** * Sets the given vector "result" with the current Vector4 coordinates multiplied by scale (float). * @param scale the number to scale with * @param result a vector to store the result in * @returns result input */ scaleToRef(e, t) { return t.x = this.x * e, t.y = this.y * e, t.z = this.z * e, t.w = this.w * e, t; } /** * Scale the current Vector4 values by a factor and add the result to a given Vector4 * @param scale defines the scale factor * @param result defines the Vector4 object where to store the result * @returns result input */ scaleAndAddToRef(e, t) { return t.x += this.x * e, t.y += this.y * e, t.z += this.z * e, t.w += this.w * e, t; } /** * Boolean : True if the current Vector4 coordinates are stricly equal to the given ones. * @param otherVector the vector to compare against * @returns true if they are equal */ equals(e) { return e && this.x === e.x && this.y === e.y && this.z === e.z && this.w === e.w; } /** * Boolean : True if the current Vector4 coordinates are each beneath the distance "epsilon" from the given vector ones. * @param otherVector vector to compare against * @param epsilon (Default: very small number) * @returns true if they are equal */ equalsWithEpsilon(e, t = Dn) { return e && Xt.WithinEpsilon(this.x, e.x, t) && Xt.WithinEpsilon(this.y, e.y, t) && Xt.WithinEpsilon(this.z, e.z, t) && Xt.WithinEpsilon(this.w, e.w, t); } /** * Boolean : True if the given floats are strictly equal to the current Vector4 coordinates. * @param x x value to compare against * @param y y value to compare against * @param z z value to compare against * @param w w value to compare against * @returns true if equal */ equalsToFloats(e, t, r, n) { return this.x === e && this.y === t && this.z === r && this.w === n; } /** * Multiplies in place the current Vector4 by the given one. * @param otherVector vector to multiple with * @returns the updated Vector4. */ multiplyInPlace(e) { return this.x *= e.x, this.y *= e.y, this.z *= e.z, this.w *= e.w, this; } /** * Returns a new Vector4 set with the multiplication result of the current Vector4 and the given one. * @param otherVector vector to multiple with * @returns resulting new vector */ multiply(e) { return new this.constructor(this.x * e.x, this.y * e.y, this.z * e.z, this.w * e.w); } /** * Updates the given vector "result" with the multiplication result of the current Vector4 and the given one. * @param otherVector vector to multiple with * @param result vector to store the result * @returns result input */ multiplyToRef(e, t) { return t.x = this.x * e.x, t.y = this.y * e.y, t.z = this.z * e.z, t.w = this.w * e.w, t; } /** * Returns a new Vector4 set with the multiplication result of the given floats and the current Vector4 coordinates. * @param x x value multiply with * @param y y value multiply with * @param z z value multiply with * @param w w value multiply with * @returns resulting new vector */ multiplyByFloats(e, t, r, n) { return new this.constructor(this.x * e, this.y * t, this.z * r, this.w * n); } /** * Returns a new Vector4 set with the division result of the current Vector4 by the given one. * @param otherVector vector to devide with * @returns resulting new vector */ divide(e) { return new this.constructor(this.x / e.x, this.y / e.y, this.z / e.z, this.w / e.w); } /** * Updates the given vector "result" with the division result of the current Vector4 by the given one. * @param otherVector vector to devide with * @param result vector to store the result * @returns result input */ divideToRef(e, t) { return t.x = this.x / e.x, t.y = this.y / e.y, t.z = this.z / e.z, t.w = this.w / e.w, t; } /** * Divides the current Vector3 coordinates by the given ones. * @param otherVector vector to devide with * @returns the updated Vector3. */ divideInPlace(e) { return this.divideToRef(e, this); } /** * Updates the Vector4 coordinates with the minimum values between its own and the given vector ones * @param other defines the second operand * @returns the current updated Vector4 */ minimizeInPlace(e) { return e.x < this.x && (this.x = e.x), e.y < this.y && (this.y = e.y), e.z < this.z && (this.z = e.z), e.w < this.w && (this.w = e.w), this; } /** * Updates the Vector4 coordinates with the maximum values between its own and the given vector ones * @param other defines the second operand * @returns the current updated Vector4 */ maximizeInPlace(e) { return e.x > this.x && (this.x = e.x), e.y > this.y && (this.y = e.y), e.z > this.z && (this.z = e.z), e.w > this.w && (this.w = e.w), this; } /** * Gets a new Vector4 from current Vector4 floored values * @returns a new Vector4 */ floor() { return new this.constructor(Math.floor(this.x), Math.floor(this.y), Math.floor(this.z), Math.floor(this.w)); } /** * Gets a new Vector4 from current Vector4 fractional values * @returns a new Vector4 */ fract() { return new this.constructor(this.x - Math.floor(this.x), this.y - Math.floor(this.y), this.z - Math.floor(this.z), this.w - Math.floor(this.w)); } // Properties /** * Returns the Vector4 length (float). * @returns the length */ length() { return Math.sqrt(this.x * this.x + this.y * this.y + this.z * this.z + this.w * this.w); } /** * Returns the Vector4 squared length (float). * @returns the length squared */ lengthSquared() { return this.x * this.x + this.y * this.y + this.z * this.z + this.w * this.w; } // Methods /** * Normalizes in place the Vector4. * @returns the updated Vector4. */ normalize() { return this.normalizeFromLength(this.length()); } /** * Normalize the current Vector4 with the given input length. * Please note that this is an in place operation. * @param len the length of the vector * @returns the current updated Vector4 */ normalizeFromLength(e) { return e === 0 || e === 1 ? this : this.scaleInPlace(1 / e); } /** * Normalize the current Vector4 to a new vector * @returns the new Vector4 */ normalizeToNew() { const e = new this.constructor(0, 0, 0, 0); return this.normalizeToRef(e), e; } /** * Normalize the current Vector4 to the reference * @param reference define the Vector4 to update * @returns the updated Vector4 */ normalizeToRef(e) { const t = this.length(); return t === 0 || t === 1 ? e.copyFromFloats(this.x, this.y, this.z, this.w) : this.scaleToRef(1 / t, e); } /** * Returns a new Vector3 from the Vector4 (x, y, z) coordinates. * @returns this converted to a new vector3 */ toVector3() { return new S(this.x, this.y, this.z); } /** * Returns a new Vector4 copied from the current one. * @returns the new cloned vector */ clone() { return new this.constructor(this.x, this.y, this.z, this.w); } /** * Updates the current Vector4 with the given one coordinates. * @param source the source vector to copy from * @returns the updated Vector4. */ copyFrom(e) { return this.x = e.x, this.y = e.y, this.z = e.z, this.w = e.w, this; } /** * Updates the current Vector4 coordinates with the given floats. * @param x float to copy from * @param y float to copy from * @param z float to copy from * @param w float to copy from * @returns the updated Vector4. */ copyFromFloats(e, t, r, n) { return this.x = e, this.y = t, this.z = r, this.w = n, this; } /** * Updates the current Vector4 coordinates with the given floats. * @param x float to set from * @param y float to set from * @param z float to set from * @param w float to set from * @returns the updated Vector4. */ set(e, t, r, n) { return this.copyFromFloats(e, t, r, n); } /** * Copies the given float to the current Vector3 coordinates * @param v defines the x, y, z and w coordinates of the operand * @returns the current updated Vector3 */ setAll(e) { return this.x = this.y = this.z = this.w = e, this; } /** * Returns the dot product (float) between the current vectors and "otherVector" * @param otherVector defines the right operand * @returns the dot product */ dot(e) { return this.x * e.x + this.y * e.y + this.z * e.z + this.w * e.w; } // Statics /** * Returns a new Vector4 set from the starting index of the given array. * @param array the array to pull values from * @param offset the offset into the array to start at * @returns the new vector */ static FromArray(e, t) { return t || (t = 0), new Ir(e[t], e[t + 1], e[t + 2], e[t + 3]); } /** * Updates the given vector "result" from the starting index of the given array. * @param array the array to pull values from * @param offset the offset into the array to start at * @param result the vector to store the result in * @returns result input */ static FromArrayToRef(e, t, r) { return r.x = e[t], r.y = e[t + 1], r.z = e[t + 2], r.w = e[t + 3], r; } /** * Updates the given vector "result" from the starting index of the given Float32Array. * @param array the array to pull values from * @param offset the offset into the array to start at * @param result the vector to store the result in * @returns result input */ static FromFloatArrayToRef(e, t, r) { return Ir.FromArrayToRef(e, t, r), r; } /** * Updates the given vector "result" coordinates from the given floats. * @param x float to set from * @param y float to set from * @param z float to set from * @param w float to set from * @param result the vector to the floats in * @returns result input */ static FromFloatsToRef(e, t, r, n, i) { return i.x = e, i.y = t, i.z = r, i.w = n, i; } /** * Returns a new Vector4 set to (0.0, 0.0, 0.0, 0.0) * @returns the new vector */ static Zero() { return new Ir(0, 0, 0, 0); } /** * Returns a new Vector4 set to (1.0, 1.0, 1.0, 1.0) * @returns the new vector */ static One() { return new Ir(1, 1, 1, 1); } /** * Returns a new Vector4 with random values between min and max * @param min the minimum random value * @param max the maximum random value * @returns a Vector4 with random values between min and max */ static Random(e = 0, t = 1) { return new Ir(Xt.RandomRange(e, t), Xt.RandomRange(e, t), Xt.RandomRange(e, t), Xt.RandomRange(e, t)); } /** * Gets a zero Vector4 that must not be updated */ static get ZeroReadOnly() { return Ir._ZeroReadOnly; } /** * Returns a new normalized Vector4 from the given one. * @param vector the vector to normalize * @returns the vector */ static Normalize(e) { const t = Ir.Zero(); return Ir.NormalizeToRef(e, t), t; } /** * Updates the given vector "result" from the normalization of the given one. * @param vector the vector to normalize * @param result the vector to store the result in * @returns result input */ static NormalizeToRef(e, t) { return e.normalizeToRef(t), t; } /** * Returns a vector with the minimum values from the left and right vectors * @param left left vector to minimize * @param right right vector to minimize * @returns a new vector with the minimum of the left and right vector values */ static Minimize(e, t) { const r = new e.constructor(); return r.copyFrom(e), r.minimizeInPlace(t), r; } /** * Returns a vector with the maximum values from the left and right vectors * @param left left vector to maximize * @param right right vector to maximize * @returns a new vector with the maximum of the left and right vector values */ static Maximize(e, t) { const r = new e.constructor(); return r.copyFrom(e), r.maximizeInPlace(t), r; } /** * Returns the distance (float) between the vectors "value1" and "value2". * @param value1 value to calulate the distance between * @param value2 value to calulate the distance between * @returns the distance between the two vectors */ static Distance(e, t) { return Math.sqrt(Ir.DistanceSquared(e, t)); } /** * Returns the squared distance (float) between the vectors "value1" and "value2". * @param value1 value to calulate the distance between * @param value2 value to calulate the distance between * @returns the distance between the two vectors squared */ static DistanceSquared(e, t) { const r = e.x - t.x, n = e.y - t.y, i = e.z - t.z, s = e.w - t.w; return r * r + n * n + i * i + s * s; } /** * Returns a new Vector4 located at the center between the vectors "value1" and "value2". * @param value1 value to calulate the center between * @param value2 value to calulate the center between * @returns the center between the two vectors */ static Center(e, t) { return Ir.CenterToRef(e, t, Ir.Zero()); } /** * Gets the center of the vectors "value1" and "value2" and stores the result in the vector "ref" * @param value1 defines first vector * @param value2 defines second vector * @param ref defines third vector * @returns ref */ static CenterToRef(e, t, r) { return r.copyFromFloats((e.x + t.x) / 2, (e.y + t.y) / 2, (e.z + t.z) / 2, (e.w + t.w) / 2); } /** * Returns a new Vector4 set with the result of the transformation by the given matrix of the given vector. * This method computes tranformed coordinates only, not transformed direction vectors (ie. it takes translation in account) * The difference with Vector3.TransformCoordinates is that the w component is not used to divide the other coordinates but is returned in the w coordinate instead * @param vector defines the Vector3 to transform * @param transformation defines the transformation matrix * @returns the transformed Vector4 */ static TransformCoordinates(e, t) { const r = Ir.Zero(); return Ir.TransformCoordinatesToRef(e, t, r), r; } /** * Sets the given vector "result" coordinates with the result of the transformation by the given matrix of the given vector * This method computes tranformed coordinates only, not transformed direction vectors (ie. it takes translation in account) * The difference with Vector3.TransformCoordinatesToRef is that the w component is not used to divide the other coordinates but is returned in the w coordinate instead * @param vector defines the Vector3 to transform * @param transformation defines the transformation matrix * @param result defines the Vector4 where to store the result * @returns result input */ static TransformCoordinatesToRef(e, t, r) { return Ir.TransformCoordinatesFromFloatsToRef(e._x, e._y, e._z, t, r), r; } /** * Sets the given vector "result" coordinates with the result of the transformation by the given matrix of the given floats (x, y, z) * This method computes tranformed coordinates only, not transformed direction vectors * The difference with Vector3.TransformCoordinatesFromFloatsToRef is that the w component is not used to divide the other coordinates but is returned in the w coordinate instead * @param x define the x coordinate of the source vector * @param y define the y coordinate of the source vector * @param z define the z coordinate of the source vector * @param transformation defines the transformation matrix * @param result defines the Vector4 where to store the result * @returns result input */ static TransformCoordinatesFromFloatsToRef(e, t, r, n, i) { const s = n.m, a = e * s[0] + t * s[4] + r * s[8] + s[12], f = e * s[1] + t * s[5] + r * s[9] + s[13], o = e * s[2] + t * s[6] + r * s[10] + s[14], d = e * s[3] + t * s[7] + r * s[11] + s[15]; return i.x = a, i.y = f, i.z = o, i.w = d, i; } /** * Returns a new Vector4 set with the result of the normal transformation by the given matrix of the given vector. * This methods computes transformed normalized direction vectors only. * @param vector the vector to transform * @param transformation the transformation matrix to apply * @returns the new vector */ static TransformNormal(e, t) { const r = new e.constructor(); return Ir.TransformNormalToRef(e, t, r), r; } /** * Sets the given vector "result" with the result of the normal transformation by the given matrix of the given vector. * This methods computes transformed normalized direction vectors only. * @param vector the vector to transform * @param transformation the transformation matrix to apply * @param result the vector to store the result in * @returns result input */ static TransformNormalToRef(e, t, r) { const n = t.m, i = e.x * n[0] + e.y * n[4] + e.z * n[8], s = e.x * n[1] + e.y * n[5] + e.z * n[9], a = e.x * n[2] + e.y * n[6] + e.z * n[10]; return r.x = i, r.y = s, r.z = a, r.w = e.w, r; } /** * Sets the given vector "result" with the result of the normal transformation by the given matrix of the given floats (x, y, z, w). * This methods computes transformed normalized direction vectors only. * @param x value to transform * @param y value to transform * @param z value to transform * @param w value to transform * @param transformation the transformation matrix to apply * @param result the vector to store the results in * @returns result input */ static TransformNormalFromFloatsToRef(e, t, r, n, i, s) { const a = i.m; return s.x = e * a[0] + t * a[4] + r * a[8], s.y = e * a[1] + t * a[5] + r * a[9], s.z = e * a[2] + t * a[6] + r * a[10], s.w = n, s; } /** * Creates a new Vector4 from a Vector3 * @param source defines the source data * @param w defines the 4th component (default is 0) * @returns a new Vector4 */ static FromVector3(e, t = 0) { return new Ir(e._x, e._y, e._z, t); } /** * Returns the dot product (float) between the vectors "left" and "right" * @param left defines the left operand * @param right defines the right operand * @returns the dot product */ static Dot(e, t) { return e.dot(t); } } Ir._ZeroReadOnly = Ir.Zero(); class Ze { /** Gets or sets the x coordinate */ get x() { return this._x; } set x(e) { this._x = e, this._isDirty = !0; } /** Gets or sets the y coordinate */ get y() { return this._y; } set y(e) { this._y = e, this._isDirty = !0; } /** Gets or sets the z coordinate */ get z() { return this._z; } set z(e) { this._z = e, this._isDirty = !0; } /** Gets or sets the w coordinate */ get w() { return this._w; } set w(e) { this._w = e, this._isDirty = !0; } /** * Creates a new Quaternion from the given floats * @param x defines the first component (0 by default) * @param y defines the second component (0 by default) * @param z defines the third component (0 by default) * @param w defines the fourth component (1.0 by default) */ constructor(e = 0, t = 0, r = 0, n = 1) { this._isDirty = !0, this._x = e, this._y = t, this._z = r, this._w = n; } /** * Gets a string representation for the current quaternion * @returns a string with the Quaternion coordinates */ toString() { return `{X: ${this._x} Y: ${this._y} Z: ${this._z} W: ${this._w}}`; } /** * Gets the class name of the quaternion * @returns the string "Quaternion" */ getClassName() { return "Quaternion"; } /** * Gets a hash code for this quaternion * @returns the quaternion hash code */ getHashCode() { const e = Kl(this._x), t = Kl(this._y), r = Kl(this._z), n = Kl(this._w); let i = e; return i = i * 397 ^ t, i = i * 397 ^ r, i = i * 397 ^ n, i; } /** * Copy the quaternion to an array * Example Playground https://playground.babylonjs.com/#L49EJ7#13 * @returns a new array populated with 4 elements from the quaternion coordinates */ asArray() { return [this._x, this._y, this._z, this._w]; } /** * Stores from the starting index in the given array the Quaternion successive values * Example Playground https://playground.babylonjs.com/#L49EJ7#59 * @param array defines the array where to store the x,y,z,w components * @param index defines an optional index in the target array to define where to start storing values * @returns the current Quaternion object */ toArray(e, t = 0) { return e[t] = this._x, e[t + 1] = this._y, e[t + 2] = this._z, e[t + 3] = this._w, this; } /** * Check if two quaternions are equals * Example Playground https://playground.babylonjs.com/#L49EJ7#38 * @param otherQuaternion defines the second operand * @returns true if the current quaternion and the given one coordinates are strictly equals */ equals(e) { return e && this._x === e._x && this._y === e._y && this._z === e._z && this._w === e._w; } /** * Gets a boolean if two quaternions are equals (using an epsilon value) * Example Playground https://playground.babylonjs.com/#L49EJ7#37 * @param otherQuaternion defines the other quaternion * @param epsilon defines the minimal distance to consider equality * @returns true if the given quaternion coordinates are close to the current ones by a distance of epsilon. */ equalsWithEpsilon(e, t = Dn) { return e && Xt.WithinEpsilon(this._x, e._x, t) && Xt.WithinEpsilon(this._y, e._y, t) && Xt.WithinEpsilon(this._z, e._z, t) && Xt.WithinEpsilon(this._w, e._w, t); } /** * Clone the current quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#12 * @returns a new quaternion copied from the current one */ clone() { return new this.constructor(this._x, this._y, this._z, this._w); } /** * Copy a quaternion to the current one * Example Playground https://playground.babylonjs.com/#L49EJ7#86 * @param other defines the other quaternion * @returns the updated current quaternion */ copyFrom(e) { return this._x = e._x, this._y = e._y, this._z = e._z, this._w = e._w, this._isDirty = !0, this; } /** * Updates the current quaternion with the given float coordinates * Example Playground https://playground.babylonjs.com/#L49EJ7#87 * @param x defines the x coordinate * @param y defines the y coordinate * @param z defines the z coordinate * @param w defines the w coordinate * @returns the updated current quaternion */ copyFromFloats(e, t, r, n) { return this._x = e, this._y = t, this._z = r, this._w = n, this._isDirty = !0, this; } /** * Updates the current quaternion from the given float coordinates * Example Playground https://playground.babylonjs.com/#L49EJ7#56 * @param x defines the x coordinate * @param y defines the y coordinate * @param z defines the z coordinate * @param w defines the w coordinate * @returns the updated current quaternion */ set(e, t, r, n) { return this.copyFromFloats(e, t, r, n); } /** * Adds two quaternions * Example Playground https://playground.babylonjs.com/#L49EJ7#10 * @param other defines the second operand * @returns a new quaternion as the addition result of the given one and the current quaternion */ add(e) { return new this.constructor(this._x + e._x, this._y + e._y, this._z + e._z, this._w + e._w); } /** * Add a quaternion to the current one * Example Playground https://playground.babylonjs.com/#L49EJ7#11 * @param other defines the quaternion to add * @returns the current quaternion */ addInPlace(e) { return this._x += e._x, this._y += e._y, this._z += e._z, this._w += e._w, this._isDirty = !0, this; } /** * Subtract two quaternions * Example Playground https://playground.babylonjs.com/#L49EJ7#57 * @param other defines the second operand * @returns a new quaternion as the subtraction result of the given one from the current one */ subtract(e) { return new this.constructor(this._x - e._x, this._y - e._y, this._z - e._z, this._w - e._w); } /** * Subtract a quaternion to the current one * Example Playground https://playground.babylonjs.com/#L49EJ7#58 * @param other defines the quaternion to subtract * @returns the current quaternion */ subtractInPlace(e) { return this._x -= e._x, this._y -= e._y, this._z -= e._z, this._w -= e._w, this._isDirty = !0, this; } /** * Multiplies the current quaternion by a scale factor * Example Playground https://playground.babylonjs.com/#L49EJ7#88 * @param value defines the scale factor * @returns a new quaternion set by multiplying the current quaternion coordinates by the float "scale" */ scale(e) { return new this.constructor(this._x * e, this._y * e, this._z * e, this._w * e); } /** * Scale the current quaternion values by a factor and stores the result to a given quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#89 * @param scale defines the scale factor * @param result defines the Quaternion object where to store the result * @returns result input */ scaleToRef(e, t) { return t._x = this._x * e, t._y = this._y * e, t._z = this._z * e, t._w = this._w * e, t._isDirty = !0, t; } /** * Multiplies in place the current quaternion by a scale factor * Example Playground https://playground.babylonjs.com/#L49EJ7#90 * @param value defines the scale factor * @returns the current modified quaternion */ scaleInPlace(e) { return this._x *= e, this._y *= e, this._z *= e, this._w *= e, this._isDirty = !0, this; } /** * Scale the current quaternion values by a factor and add the result to a given quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#91 * @param scale defines the scale factor * @param result defines the Quaternion object where to store the result * @returns result input */ scaleAndAddToRef(e, t) { return t._x += this._x * e, t._y += this._y * e, t._z += this._z * e, t._w += this._w * e, t._isDirty = !0, t; } /** * Multiplies two quaternions * Example Playground https://playground.babylonjs.com/#L49EJ7#43 * @param q1 defines the second operand * @returns a new quaternion set as the multiplication result of the current one with the given one "q1" */ multiply(e) { const t = new this.constructor(0, 0, 0, 1); return this.multiplyToRef(e, t), t; } /** * Sets the given "result" as the multiplication result of the current one with the given one "q1" * Example Playground https://playground.babylonjs.com/#L49EJ7#45 * @param q1 defines the second operand * @param result defines the target quaternion * @returns the current quaternion */ multiplyToRef(e, t) { const r = this._x * e._w + this._y * e._z - this._z * e._y + this._w * e._x, n = -this._x * e._z + this._y * e._w + this._z * e._x + this._w * e._y, i = this._x * e._y - this._y * e._x + this._z * e._w + this._w * e._z, s = -this._x * e._x - this._y * e._y - this._z * e._z + this._w * e._w; return t.copyFromFloats(r, n, i, s), t; } /** * Updates the current quaternion with the multiplication of itself with the given one "q1" * Example Playground https://playground.babylonjs.com/#L49EJ7#46 * @param q1 defines the second operand * @returns the currentupdated quaternion */ multiplyInPlace(e) { return this.multiplyToRef(e, this), this; } /** * Conjugates the current quaternion and stores the result in the given quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#81 * @param ref defines the target quaternion * @returns result input */ conjugateToRef(e) { return e.copyFromFloats(-this._x, -this._y, -this._z, this._w), e; } /** * Conjugates in place the current quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#82 * @returns the current updated quaternion */ conjugateInPlace() { return this._x *= -1, this._y *= -1, this._z *= -1, this._isDirty = !0, this; } /** * Conjugates (1-q) the current quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#83 * @returns a new quaternion */ conjugate() { return new this.constructor(-this._x, -this._y, -this._z, this._w); } /** * Returns the inverse of the current quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#84 * @returns a new quaternion */ invert() { const e = this.conjugate(), t = this.lengthSquared(); return t == 0 || t == 1 || e.scaleInPlace(1 / t), e; } /** * Invert in place the current quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#85 * @returns this quaternion */ invertInPlace() { this.conjugateInPlace(); const e = this.lengthSquared(); return e == 0 || e == 1 ? this : (this.scaleInPlace(1 / e), this); } /** * Gets squared length of current quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#29 * @returns the quaternion length (float) */ lengthSquared() { return this._x * this._x + this._y * this._y + this._z * this._z + this._w * this._w; } /** * Gets length of current quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#28 * @returns the quaternion length (float) */ length() { return Math.sqrt(this.lengthSquared()); } /** * Normalize in place the current quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#54 * @returns the current updated quaternion */ normalize() { return this.normalizeFromLength(this.length()); } /** * Normalize the current quaternion with the given input length. * Please note that this is an in place operation. * @param len the length of the quaternion * @returns the current updated Quaternion */ normalizeFromLength(e) { return e === 0 || e === 1 ? this : this.scaleInPlace(1 / e); } /** * Normalize a copy of the current quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#55 * @returns the normalized quaternion */ normalizeToNew() { const e = new this.constructor(0, 0, 0, 1); return this.normalizeToRef(e), e; } /** * Normalize the current Quaternion to the reference * @param reference define the Quaternion to update * @returns the updated Quaternion */ normalizeToRef(e) { const t = this.length(); return t === 0 || t === 1 ? e.copyFromFloats(this._x, this._y, this._z, this._w) : this.scaleToRef(1 / t, e); } /** * Returns a new Vector3 set with the Euler angles translated from the current quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#32 * @returns a new Vector3 containing the Euler angles * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/rotation_conventions */ toEulerAngles() { const e = S.Zero(); return this.toEulerAnglesToRef(e), e; } /** * Sets the given vector3 "result" with the Euler angles translated from the current quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#31 * @param result defines the vector which will be filled with the Euler angles * @returns result input * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/rotation_conventions */ toEulerAnglesToRef(e) { const t = this._z, r = this._x, n = this._y, i = this._w, s = n * t - r * i, a = 0.4999999; if (s < -a) e._y = 2 * Math.atan2(n, i), e._x = Math.PI / 2, e._z = 0, e._isDirty = !0; else if (s > a) e._y = 2 * Math.atan2(n, i), e._x = -Math.PI / 2, e._z = 0, e._isDirty = !0; else { const f = i * i, o = t * t, d = r * r, v = n * n; e._z = Math.atan2(2 * (r * n + t * i), -o - d + v + f), e._x = Math.asin(-2 * s), e._y = Math.atan2(2 * (t * r + n * i), o - d - v + f), e._isDirty = !0; } return e; } /** * Updates the given rotation matrix with the current quaternion values * Example Playground https://playground.babylonjs.com/#L49EJ7#67 * @param result defines the target matrix * @returns the updated matrix with the rotation */ toRotationMatrix(e) { return he.FromQuaternionToRef(this, e), e; } /** * Updates the current quaternion from the given rotation matrix values * Example Playground https://playground.babylonjs.com/#L49EJ7#41 * @param matrix defines the source matrix * @returns the current updated quaternion */ fromRotationMatrix(e) { return Ze.FromRotationMatrixToRef(e, this), this; } /** * Returns the dot product (float) between the current quaternions and "other" * @param other defines the right operand * @returns the dot product */ dot(e) { return this._x * e._x + this._y * e._y + this._z * e._z + this._w * e._w; } // Statics /** * Creates a new quaternion from a rotation matrix * Example Playground https://playground.babylonjs.com/#L49EJ7#101 * @param matrix defines the source matrix * @returns a new quaternion created from the given rotation matrix values */ static FromRotationMatrix(e) { const t = new Ze(); return Ze.FromRotationMatrixToRef(e, t), t; } /** * Updates the given quaternion with the given rotation matrix values * Example Playground https://playground.babylonjs.com/#L49EJ7#102 * @param matrix defines the source matrix * @param result defines the target quaternion * @returns result input */ static FromRotationMatrixToRef(e, t) { const r = e.m, n = r[0], i = r[4], s = r[8], a = r[1], f = r[5], o = r[9], d = r[2], v = r[6], u = r[10], l = n + f + u; let P; return l > 0 ? (P = 0.5 / Math.sqrt(l + 1), t._w = 0.25 / P, t._x = (v - o) * P, t._y = (s - d) * P, t._z = (a - i) * P, t._isDirty = !0) : n > f && n > u ? (P = 2 * Math.sqrt(1 + n - f - u), t._w = (v - o) / P, t._x = 0.25 * P, t._y = (i + a) / P, t._z = (s + d) / P, t._isDirty = !0) : f > u ? (P = 2 * Math.sqrt(1 + f - n - u), t._w = (s - d) / P, t._x = (i + a) / P, t._y = 0.25 * P, t._z = (o + v) / P, t._isDirty = !0) : (P = 2 * Math.sqrt(1 + u - n - f), t._w = (a - i) / P, t._x = (s + d) / P, t._y = (o + v) / P, t._z = 0.25 * P, t._isDirty = !0), t; } /** * Returns the dot product (float) between the quaternions "left" and "right" * Example Playground https://playground.babylonjs.com/#L49EJ7#61 * @param left defines the left operand * @param right defines the right operand * @returns the dot product */ static Dot(e, t) { return e._x * t._x + e._y * t._y + e._z * t._z + e._w * t._w; } /** * Checks if the orientations of two rotation quaternions are close to each other * Example Playground https://playground.babylonjs.com/#L49EJ7#60 * @param quat0 defines the first quaternion to check * @param quat1 defines the second quaternion to check * @param epsilon defines closeness, 0 same orientation, 1 PI apart, default 0.1 * @returns true if the two quaternions are close to each other within epsilon */ static AreClose(e, t, r = 0.1) { const n = Ze.Dot(e, t); return 1 - n * n <= r; } /** * Smooth interpolation between two quaternions using Slerp * Example Playground https://playground.babylonjs.com/#L49EJ7#93 * @param source source quaternion * @param goal goal quaternion * @param deltaTime current interpolation frame * @param lerpTime total interpolation time * @param result the smoothed quaternion */ static SmoothToRef(e, t, r, n, i) { let s = n === 0 ? 1 : r / n; return s = Xt.Clamp(s, 0, 1), Ze.SlerpToRef(e, t, s, i), i; } /** * Creates an empty quaternion * @returns a new quaternion set to (0.0, 0.0, 0.0) */ static Zero() { return new Ze(0, 0, 0, 0); } /** * Inverse a given quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#103 * @param q defines the source quaternion * @returns a new quaternion as the inverted current quaternion */ static Inverse(e) { return new e.constructor(-e._x, -e._y, -e._z, e._w); } /** * Inverse a given quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#104 * @param q defines the source quaternion * @param result the quaternion the result will be stored in * @returns the result quaternion */ static InverseToRef(e, t) { return t.set(-e._x, -e._y, -e._z, e._w), t; } /** * Creates an identity quaternion * @returns the identity quaternion */ static Identity() { return new Ze(0, 0, 0, 1); } /** * Gets a boolean indicating if the given quaternion is identity * @param quaternion defines the quaternion to check * @returns true if the quaternion is identity */ static IsIdentity(e) { return e && e._x === 0 && e._y === 0 && e._z === 0 && e._w === 1; } /** * Creates a quaternion from a rotation around an axis * Example Playground https://playground.babylonjs.com/#L49EJ7#72 * @param axis defines the axis to use * @param angle defines the angle to use * @returns a new quaternion created from the given axis (Vector3) and angle in radians (float) */ static RotationAxis(e, t) { return Ze.RotationAxisToRef(e, t, new Ze()); } /** * Creates a rotation around an axis and stores it into the given quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#73 * @param axis defines the axis to use * @param angle defines the angle to use * @param result defines the target quaternion * @returns the target quaternion */ static RotationAxisToRef(e, t, r) { const n = Math.sin(t / 2); return e.normalize(), r._w = Math.cos(t / 2), r._x = e._x * n, r._y = e._y * n, r._z = e._z * n, r._isDirty = !0, r; } /** * Creates a new quaternion from data stored into an array * Example Playground https://playground.babylonjs.com/#L49EJ7#63 * @param array defines the data source * @param offset defines the offset in the source array where the data starts * @returns a new quaternion */ static FromArray(e, t) { return t || (t = 0), new Ze(e[t], e[t + 1], e[t + 2], e[t + 3]); } /** * Updates the given quaternion "result" from the starting index of the given array. * Example Playground https://playground.babylonjs.com/#L49EJ7#64 * @param array the array to pull values from * @param offset the offset into the array to start at * @param result the quaternion to store the result in * @returns result input */ static FromArrayToRef(e, t, r) { return r._x = e[t], r._y = e[t + 1], r._z = e[t + 2], r._w = e[t + 3], r._isDirty = !0, r; } /** * Create a quaternion from Euler rotation angles * Example Playground https://playground.babylonjs.com/#L49EJ7#33 * @param x Pitch * @param y Yaw * @param z Roll * @returns the new Quaternion */ static FromEulerAngles(e, t, r) { const n = new Ze(); return Ze.RotationYawPitchRollToRef(t, e, r, n), n; } /** * Updates a quaternion from Euler rotation angles * Example Playground https://playground.babylonjs.com/#L49EJ7#34 * @param x Pitch * @param y Yaw * @param z Roll * @param result the quaternion to store the result * @returns the updated quaternion */ static FromEulerAnglesToRef(e, t, r, n) { return Ze.RotationYawPitchRollToRef(t, e, r, n), n; } /** * Create a quaternion from Euler rotation vector * Example Playground https://playground.babylonjs.com/#L49EJ7#35 * @param vec the Euler vector (x Pitch, y Yaw, z Roll) * @returns the new Quaternion */ static FromEulerVector(e) { const t = new Ze(); return Ze.RotationYawPitchRollToRef(e._y, e._x, e._z, t), t; } /** * Updates a quaternion from Euler rotation vector * Example Playground https://playground.babylonjs.com/#L49EJ7#36 * @param vec the Euler vector (x Pitch, y Yaw, z Roll) * @param result the quaternion to store the result * @returns the updated quaternion */ static FromEulerVectorToRef(e, t) { return Ze.RotationYawPitchRollToRef(e._y, e._x, e._z, t), t; } /** * Updates a quaternion so that it rotates vector vecFrom to vector vecTo * Example Playground - https://playground.babylonjs.com/#L49EJ7#70 * @param vecFrom defines the direction vector from which to rotate * @param vecTo defines the direction vector to which to rotate * @param result the quaternion to store the result * @param epsilon defines the minimal dot value to define vecs as opposite. Default: `BABYLON.Epsilon` * @returns the updated quaternion */ static FromUnitVectorsToRef(e, t, r, n = Dn) { const i = S.Dot(e, t) + 1; return i < n ? Math.abs(e.x) > Math.abs(e.z) ? r.set(-e.y, e.x, 0, 0) : r.set(0, -e.z, e.y, 0) : (S.CrossToRef(e, t, ue.Vector3[0]), r.set(ue.Vector3[0].x, ue.Vector3[0].y, ue.Vector3[0].z, i)), r.normalize(); } /** * Creates a new quaternion from the given Euler float angles (y, x, z) * Example Playground https://playground.babylonjs.com/#L49EJ7#77 * @param yaw defines the rotation around Y axis * @param pitch defines the rotation around X axis * @param roll defines the rotation around Z axis * @returns the new quaternion */ static RotationYawPitchRoll(e, t, r) { const n = new Ze(); return Ze.RotationYawPitchRollToRef(e, t, r, n), n; } /** * Creates a new rotation from the given Euler float angles (y, x, z) and stores it in the target quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#78 * @param yaw defines the rotation around Y axis * @param pitch defines the rotation around X axis * @param roll defines the rotation around Z axis * @param result defines the target quaternion * @returns result input */ static RotationYawPitchRollToRef(e, t, r, n) { const i = r * 0.5, s = t * 0.5, a = e * 0.5, f = Math.sin(i), o = Math.cos(i), d = Math.sin(s), v = Math.cos(s), u = Math.sin(a), l = Math.cos(a); return n._x = l * d * o + u * v * f, n._y = u * v * o - l * d * f, n._z = l * v * f - u * d * o, n._w = l * v * o + u * d * f, n._isDirty = !0, n; } /** * Creates a new quaternion from the given Euler float angles expressed in z-x-z orientation * Example Playground https://playground.babylonjs.com/#L49EJ7#68 * @param alpha defines the rotation around first axis * @param beta defines the rotation around second axis * @param gamma defines the rotation around third axis * @returns the new quaternion */ static RotationAlphaBetaGamma(e, t, r) { const n = new Ze(); return Ze.RotationAlphaBetaGammaToRef(e, t, r, n), n; } /** * Creates a new quaternion from the given Euler float angles expressed in z-x-z orientation and stores it in the target quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#69 * @param alpha defines the rotation around first axis * @param beta defines the rotation around second axis * @param gamma defines the rotation around third axis * @param result defines the target quaternion * @returns result input */ static RotationAlphaBetaGammaToRef(e, t, r, n) { const i = (r + e) * 0.5, s = (r - e) * 0.5, a = t * 0.5; return n._x = Math.cos(s) * Math.sin(a), n._y = Math.sin(s) * Math.sin(a), n._z = Math.sin(i) * Math.cos(a), n._w = Math.cos(i) * Math.cos(a), n._isDirty = !0, n; } /** * Creates a new quaternion containing the rotation value to reach the target (axis1, axis2, axis3) orientation as a rotated XYZ system (axis1, axis2 and axis3 are normalized during this operation) * Example Playground https://playground.babylonjs.com/#L49EJ7#75 * @param axis1 defines the first axis * @param axis2 defines the second axis * @param axis3 defines the third axis * @returns the new quaternion */ static RotationQuaternionFromAxis(e, t, r) { const n = new Ze(0, 0, 0, 0); return Ze.RotationQuaternionFromAxisToRef(e, t, r, n), n; } /** * Creates a rotation value to reach the target (axis1, axis2, axis3) orientation as a rotated XYZ system (axis1, axis2 and axis3 are normalized during this operation) and stores it in the target quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#76 * @param axis1 defines the first axis * @param axis2 defines the second axis * @param axis3 defines the third axis * @param ref defines the target quaternion * @returns result input */ static RotationQuaternionFromAxisToRef(e, t, r, n) { const i = Kr.Matrix[0]; return he.FromXYZAxesToRef(e.normalize(), t.normalize(), r.normalize(), i), Ze.FromRotationMatrixToRef(i, n), n; } /** * Creates a new rotation value to orient an object to look towards the given forward direction, the up direction being oriented like "up". * This function works in left handed mode * Example Playground https://playground.babylonjs.com/#L49EJ7#96 * @param forward defines the forward direction - Must be normalized and orthogonal to up. * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward. * @returns A new quaternion oriented toward the specified forward and up. */ static FromLookDirectionLH(e, t) { const r = new Ze(); return Ze.FromLookDirectionLHToRef(e, t, r), r; } /** * Creates a new rotation value to orient an object to look towards the given forward direction with the up direction being oriented like "up", and stores it in the target quaternion. * This function works in left handed mode * Example Playground https://playground.babylonjs.com/#L49EJ7#97 * @param forward defines the forward direction - Must be normalized and orthogonal to up. * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward. * @param ref defines the target quaternion. * @returns result input */ static FromLookDirectionLHToRef(e, t, r) { const n = Kr.Matrix[0]; return he.LookDirectionLHToRef(e, t, n), Ze.FromRotationMatrixToRef(n, r), r; } /** * Creates a new rotation value to orient an object to look towards the given forward direction, the up direction being oriented like "up". * This function works in right handed mode * Example Playground https://playground.babylonjs.com/#L49EJ7#98 * @param forward defines the forward direction - Must be normalized and orthogonal to up. * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward. * @returns A new quaternion oriented toward the specified forward and up. */ static FromLookDirectionRH(e, t) { const r = new Ze(); return Ze.FromLookDirectionRHToRef(e, t, r), r; } /** * Creates a new rotation value to orient an object to look towards the given forward direction with the up direction being oriented like "up", and stores it in the target quaternion. * This function works in right handed mode * Example Playground https://playground.babylonjs.com/#L49EJ7#105 * @param forward defines the forward direction - Must be normalized and orthogonal to up. * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward. * @param ref defines the target quaternion. * @returns result input */ static FromLookDirectionRHToRef(e, t, r) { const n = Kr.Matrix[0]; return he.LookDirectionRHToRef(e, t, n), Ze.FromRotationMatrixToRef(n, r); } /** * Interpolates between two quaternions * Example Playground https://playground.babylonjs.com/#L49EJ7#79 * @param left defines first quaternion * @param right defines second quaternion * @param amount defines the gradient to use * @returns the new interpolated quaternion */ static Slerp(e, t, r) { const n = Ze.Identity(); return Ze.SlerpToRef(e, t, r, n), n; } /** * Interpolates between two quaternions and stores it into a target quaternion * Example Playground https://playground.babylonjs.com/#L49EJ7#92 * @param left defines first quaternion * @param right defines second quaternion * @param amount defines the gradient to use * @param result defines the target quaternion * @returns result input */ static SlerpToRef(e, t, r, n) { let i, s, a = e._x * t._x + e._y * t._y + e._z * t._z + e._w * t._w, f = !1; if (a < 0 && (f = !0, a = -a), a > 0.999999) s = 1 - r, i = f ? -r : r; else { const o = Math.acos(a), d = 1 / Math.sin(o); s = Math.sin((1 - r) * o) * d, i = f ? -Math.sin(r * o) * d : Math.sin(r * o) * d; } return n._x = s * e._x + i * t._x, n._y = s * e._y + i * t._y, n._z = s * e._z + i * t._z, n._w = s * e._w + i * t._w, n._isDirty = !0, n; } /** * Interpolate between two quaternions using Hermite interpolation * Example Playground https://playground.babylonjs.com/#L49EJ7#47 * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves#hermite-quaternion-spline * @param value1 defines first quaternion * @param tangent1 defines the incoming tangent * @param value2 defines second quaternion * @param tangent2 defines the outgoing tangent * @param amount defines the target quaternion * @returns the new interpolated quaternion */ static Hermite(e, t, r, n, i) { const s = i * i, a = i * s, f = 2 * a - 3 * s + 1, o = -2 * a + 3 * s, d = a - 2 * s + i, v = a - s, u = e._x * f + r._x * o + t._x * d + n._x * v, l = e._y * f + r._y * o + t._y * d + n._y * v, P = e._z * f + r._z * o + t._z * d + n._z * v, p = e._w * f + r._w * o + t._w * d + n._w * v; return new e.constructor(u, l, P, p); } /** * Returns a new Quaternion which is the 1st derivative of the Hermite spline defined by the quaternions "value1", "value2", "tangent1", "tangent2". * Example Playground https://playground.babylonjs.com/#L49EJ7#48 * @param value1 defines the first control point * @param tangent1 defines the first tangent * @param value2 defines the second control point * @param tangent2 defines the second tangent * @param time define where the derivative must be done * @returns 1st derivative */ static Hermite1stDerivative(e, t, r, n, i) { const s = new e.constructor(); return this.Hermite1stDerivativeToRef(e, t, r, n, i, s), s; } /** * Update a Quaternion with the 1st derivative of the Hermite spline defined by the quaternions "value1", "value2", "tangent1", "tangent2". * Example Playground https://playground.babylonjs.com/#L49EJ7#49 * @param value1 defines the first control point * @param tangent1 defines the first tangent * @param value2 defines the second control point * @param tangent2 defines the second tangent * @param time define where the derivative must be done * @param result define where to store the derivative * @returns result input */ static Hermite1stDerivativeToRef(e, t, r, n, i, s) { const a = i * i; return s._x = (a - i) * 6 * e._x + (3 * a - 4 * i + 1) * t._x + (-a + i) * 6 * r._x + (3 * a - 2 * i) * n._x, s._y = (a - i) * 6 * e._y + (3 * a - 4 * i + 1) * t._y + (-a + i) * 6 * r._y + (3 * a - 2 * i) * n._y, s._z = (a - i) * 6 * e._z + (3 * a - 4 * i + 1) * t._z + (-a + i) * 6 * r._z + (3 * a - 2 * i) * n._z, s._w = (a - i) * 6 * e._w + (3 * a - 4 * i + 1) * t._w + (-a + i) * 6 * r._w + (3 * a - 2 * i) * n._w, s._isDirty = !0, s; } /** * Returns a new Quaternion as the normalization of the given Quaternion * @param quat defines the Quaternion to normalize * @returns the new Quaternion */ static Normalize(e) { const t = Ze.Zero(); return Ze.NormalizeToRef(e, t), t; } /** * Sets the given Quaternion "result" with the normalization of the given first Quaternion * @param quat defines the Quaternion to normalize * @param result defines the Quaternion where to store the result * @returns result input */ static NormalizeToRef(e, t) { return e.normalizeToRef(t), t; } } class he { /** * Gets the precision of matrix computations */ static get Use64Bits() { return k9.MatrixUse64Bits; } /** * Gets the internal data of the matrix */ get m() { return this._m; } /** * Update the updateFlag to indicate that the matrix has been updated */ markAsUpdated() { this.updateFlag = he._UpdateFlagSeed++, this._isIdentity = !1, this._isIdentity3x2 = !1, this._isIdentityDirty = !0, this._isIdentity3x2Dirty = !0; } _updateIdentityStatus(e, t = !1, r = !1, n = !0) { this._isIdentity = e, this._isIdentity3x2 = e || r, this._isIdentityDirty = this._isIdentity ? !1 : t, this._isIdentity3x2Dirty = this._isIdentity3x2 ? !1 : n; } /** * Creates an empty matrix (filled with zeros) */ constructor() { this._isIdentity = !1, this._isIdentityDirty = !0, this._isIdentity3x2 = !0, this._isIdentity3x2Dirty = !0, this.updateFlag = -1, k9.MatrixTrackPrecisionChange && k9.MatrixTrackedMatrices.push(this), this._m = new k9.MatrixCurrentType(16), this.markAsUpdated(); } // Properties /** * Check if the current matrix is identity * @returns true is the matrix is the identity matrix */ isIdentity() { if (this._isIdentityDirty) { this._isIdentityDirty = !1; const e = this._m; this._isIdentity = e[0] === 1 && e[1] === 0 && e[2] === 0 && e[3] === 0 && e[4] === 0 && e[5] === 1 && e[6] === 0 && e[7] === 0 && e[8] === 0 && e[9] === 0 && e[10] === 1 && e[11] === 0 && e[12] === 0 && e[13] === 0 && e[14] === 0 && e[15] === 1; } return this._isIdentity; } /** * Check if the current matrix is identity as a texture matrix (3x2 store in 4x4) * @returns true is the matrix is the identity matrix */ isIdentityAs3x2() { return this._isIdentity3x2Dirty && (this._isIdentity3x2Dirty = !1, this._m[0] !== 1 || this._m[5] !== 1 || this._m[15] !== 1 ? this._isIdentity3x2 = !1 : this._m[1] !== 0 || this._m[2] !== 0 || this._m[3] !== 0 || this._m[4] !== 0 || this._m[6] !== 0 || this._m[7] !== 0 || this._m[8] !== 0 || this._m[9] !== 0 || this._m[10] !== 0 || this._m[11] !== 0 || this._m[12] !== 0 || this._m[13] !== 0 || this._m[14] !== 0 ? this._isIdentity3x2 = !1 : this._isIdentity3x2 = !0), this._isIdentity3x2; } /** * Gets the determinant of the matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#34 * @returns the matrix determinant */ determinant() { if (this._isIdentity === !0) return 1; const e = this._m, t = e[0], r = e[1], n = e[2], i = e[3], s = e[4], a = e[5], f = e[6], o = e[7], d = e[8], v = e[9], u = e[10], l = e[11], P = e[12], p = e[13], c = e[14], H = e[15], T = u * H - c * l, q = v * H - p * l, b = v * c - p * u, j = d * H - P * l, w = d * c - u * P, m = d * p - P * v, I = +(a * T - f * q + o * b), N = -(s * T - f * j + o * w), k = +(s * q - a * j + o * m), R = -(s * b - a * w + f * m); return t * I + r * N + n * k + i * R; } // Methods /** * Gets a string with the Matrix values * @returns a string with the Matrix values */ toString() { return `{${this.m[0]}, ${this.m[1]}, ${this.m[2]}, ${this.m[3]} ${this.m[4]}, ${this.m[5]}, ${this.m[6]}, ${this.m[7]} ${this.m[8]}, ${this.m[9]}, ${this.m[10]}, ${this.m[11]} ${this.m[12]}, ${this.m[13]}, ${this.m[14]}, ${this.m[15]}}`; } /** * Returns the matrix as a Float32Array or Array * Example Playground - https://playground.babylonjs.com/#AV9X17#49 * @returns the matrix underlying array */ toArray() { return this._m; } /** * Returns the matrix as a Float32Array or Array * Example Playground - https://playground.babylonjs.com/#AV9X17#114 * @returns the matrix underlying array. */ asArray() { return this._m; } /** * Inverts the current matrix in place * Example Playground - https://playground.babylonjs.com/#AV9X17#118 * @returns the current inverted matrix */ invert() { return this.invertToRef(this), this; } /** * Sets all the matrix elements to zero * @returns the current matrix */ reset() { return he.FromValuesToRef(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, this), this._updateIdentityStatus(!1), this; } /** * Adds the current matrix with a second one * Example Playground - https://playground.babylonjs.com/#AV9X17#44 * @param other defines the matrix to add * @returns a new matrix as the addition of the current matrix and the given one */ add(e) { const t = new this.constructor(); return this.addToRef(e, t), t; } /** * Sets the given matrix "result" to the addition of the current matrix and the given one * Example Playground - https://playground.babylonjs.com/#AV9X17#45 * @param other defines the matrix to add * @param result defines the target matrix * @returns result input */ addToRef(e, t) { const r = this._m, n = t._m, i = e.m; for (let s = 0; s < 16; s++) n[s] = r[s] + i[s]; return t.markAsUpdated(), t; } /** * Adds in place the given matrix to the current matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#46 * @param other defines the second operand * @returns the current updated matrix */ addToSelf(e) { const t = this._m, r = e.m; for (let n = 0; n < 16; n++) t[n] += r[n]; return this.markAsUpdated(), this; } /** * Sets the given matrix to the current inverted Matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#119 * @param other defines the target matrix * @returns result input */ invertToRef(e) { if (this._isIdentity === !0) return he.IdentityToRef(e), e; const t = this._m, r = t[0], n = t[1], i = t[2], s = t[3], a = t[4], f = t[5], o = t[6], d = t[7], v = t[8], u = t[9], l = t[10], P = t[11], p = t[12], c = t[13], H = t[14], T = t[15], q = l * T - H * P, b = u * T - c * P, j = u * H - c * l, w = v * T - p * P, m = v * H - l * p, I = v * c - p * u, N = +(f * q - o * b + d * j), k = -(a * q - o * w + d * m), R = +(a * b - f * w + d * I), y = -(a * j - f * m + o * I), O = r * N + n * k + i * R + s * y; if (O === 0) return e.copyFrom(this), e; const Y = 1 / O, ee = o * T - H * d, Z = f * T - c * d, te = f * H - c * o, fe = a * T - p * d, _ = a * H - p * o, G = a * c - p * f, L = o * P - l * d, $ = f * P - u * d, ae = f * l - u * o, Pe = a * P - v * d, ge = a * l - v * o, me = a * u - v * f, Xe = -(n * q - i * b + s * j), De = +(r * q - i * w + s * m), ne = -(r * b - n * w + s * I), re = +(r * j - n * m + i * I), ve = +(n * ee - i * Z + s * te), qe = -(r * ee - i * fe + s * _), ke = +(r * Z - n * fe + s * G), be = -(r * te - n * _ + i * G), Fe = -(n * L - i * $ + s * ae), Ke = +(r * L - i * Pe + s * ge), nt = -(r * $ - n * Pe + s * me), ut = +(r * ae - n * ge + i * me); return he.FromValuesToRef(N * Y, Xe * Y, ve * Y, Fe * Y, k * Y, De * Y, qe * Y, Ke * Y, R * Y, ne * Y, ke * Y, nt * Y, y * Y, re * Y, be * Y, ut * Y, e), e; } /** * add a value at the specified position in the current Matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#47 * @param index the index of the value within the matrix. between 0 and 15. * @param value the value to be added * @returns the current updated matrix */ addAtIndex(e, t) { return this._m[e] += t, this.markAsUpdated(), this; } /** * mutiply the specified position in the current Matrix by a value * @param index the index of the value within the matrix. between 0 and 15. * @param value the value to be added * @returns the current updated matrix */ multiplyAtIndex(e, t) { return this._m[e] *= t, this.markAsUpdated(), this; } /** * Inserts the translation vector (using 3 floats) in the current matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#120 * @param x defines the 1st component of the translation * @param y defines the 2nd component of the translation * @param z defines the 3rd component of the translation * @returns the current updated matrix */ setTranslationFromFloats(e, t, r) { return this._m[12] = e, this._m[13] = t, this._m[14] = r, this.markAsUpdated(), this; } /** * Adds the translation vector (using 3 floats) in the current matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#20 * Example Playground - https://playground.babylonjs.com/#AV9X17#48 * @param x defines the 1st component of the translation * @param y defines the 2nd component of the translation * @param z defines the 3rd component of the translation * @returns the current updated matrix */ addTranslationFromFloats(e, t, r) { return this._m[12] += e, this._m[13] += t, this._m[14] += r, this.markAsUpdated(), this; } /** * Inserts the translation vector in the current matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#121 * @param vector3 defines the translation to insert * @returns the current updated matrix */ setTranslation(e) { return this.setTranslationFromFloats(e._x, e._y, e._z); } /** * Gets the translation value of the current matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#122 * @returns a new Vector3 as the extracted translation from the matrix */ getTranslation() { return new S(this._m[12], this._m[13], this._m[14]); } /** * Fill a Vector3 with the extracted translation from the matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#123 * @param result defines the Vector3 where to store the translation * @returns the current matrix */ getTranslationToRef(e) { return e.x = this._m[12], e.y = this._m[13], e.z = this._m[14], e; } /** * Remove rotation and scaling part from the matrix * @returns the updated matrix */ removeRotationAndScaling() { const e = this.m; return he.FromValuesToRef(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, e[12], e[13], e[14], e[15], this), this._updateIdentityStatus(e[12] === 0 && e[13] === 0 && e[14] === 0 && e[15] === 1), this; } /** * Multiply two matrices * Example Playground - https://playground.babylonjs.com/#AV9X17#15 * A.multiply(B) means apply B to A so result is B x A * @param other defines the second operand * @returns a new matrix set with the multiplication result of the current Matrix and the given one */ multiply(e) { const t = new this.constructor(); return this.multiplyToRef(e, t), t; } /** * Copy the current matrix from the given one * Example Playground - https://playground.babylonjs.com/#AV9X17#21 * @param other defines the source matrix * @returns the current updated matrix */ copyFrom(e) { e.copyToArray(this._m); const t = e; return this.updateFlag = t.updateFlag, this._updateIdentityStatus(t._isIdentity, t._isIdentityDirty, t._isIdentity3x2, t._isIdentity3x2Dirty), this; } /** * Populates the given array from the starting index with the current matrix values * @param array defines the target array * @param offset defines the offset in the target array where to start storing values * @returns the current matrix */ copyToArray(e, t = 0) { const r = this._m; return e[t] = r[0], e[t + 1] = r[1], e[t + 2] = r[2], e[t + 3] = r[3], e[t + 4] = r[4], e[t + 5] = r[5], e[t + 6] = r[6], e[t + 7] = r[7], e[t + 8] = r[8], e[t + 9] = r[9], e[t + 10] = r[10], e[t + 11] = r[11], e[t + 12] = r[12], e[t + 13] = r[13], e[t + 14] = r[14], e[t + 15] = r[15], this; } /** * Sets the given matrix "result" with the multiplication result of the current Matrix and the given one * A.multiplyToRef(B, R) means apply B to A and store in R and R = B x A * Example Playground - https://playground.babylonjs.com/#AV9X17#16 * @param other defines the second operand * @param result defines the matrix where to store the multiplication * @returns result input */ multiplyToRef(e, t) { return this._isIdentity ? (t.copyFrom(e), t) : e._isIdentity ? (t.copyFrom(this), t) : (this.multiplyToArray(e, t._m, 0), t.markAsUpdated(), t); } /** * Sets the Float32Array "result" from the given index "offset" with the multiplication of the current matrix and the given one * @param other defines the second operand * @param result defines the array where to store the multiplication * @param offset defines the offset in the target array where to start storing values * @returns the current matrix */ multiplyToArray(e, t, r) { const n = this._m, i = e.m, s = n[0], a = n[1], f = n[2], o = n[3], d = n[4], v = n[5], u = n[6], l = n[7], P = n[8], p = n[9], c = n[10], H = n[11], T = n[12], q = n[13], b = n[14], j = n[15], w = i[0], m = i[1], I = i[2], N = i[3], k = i[4], R = i[5], y = i[6], O = i[7], Y = i[8], ee = i[9], Z = i[10], te = i[11], fe = i[12], _ = i[13], G = i[14], L = i[15]; return t[r] = s * w + a * k + f * Y + o * fe, t[r + 1] = s * m + a * R + f * ee + o * _, t[r + 2] = s * I + a * y + f * Z + o * G, t[r + 3] = s * N + a * O + f * te + o * L, t[r + 4] = d * w + v * k + u * Y + l * fe, t[r + 5] = d * m + v * R + u * ee + l * _, t[r + 6] = d * I + v * y + u * Z + l * G, t[r + 7] = d * N + v * O + u * te + l * L, t[r + 8] = P * w + p * k + c * Y + H * fe, t[r + 9] = P * m + p * R + c * ee + H * _, t[r + 10] = P * I + p * y + c * Z + H * G, t[r + 11] = P * N + p * O + c * te + H * L, t[r + 12] = T * w + q * k + b * Y + j * fe, t[r + 13] = T * m + q * R + b * ee + j * _, t[r + 14] = T * I + q * y + b * Z + j * G, t[r + 15] = T * N + q * O + b * te + j * L, this; } /** * Check equality between this matrix and a second one * @param value defines the second matrix to compare * @returns true is the current matrix and the given one values are strictly equal */ equals(e) { const t = e; if (!t) return !1; if ((this._isIdentity || t._isIdentity) && !this._isIdentityDirty && !t._isIdentityDirty) return this._isIdentity && t._isIdentity; const r = this.m, n = t.m; return r[0] === n[0] && r[1] === n[1] && r[2] === n[2] && r[3] === n[3] && r[4] === n[4] && r[5] === n[5] && r[6] === n[6] && r[7] === n[7] && r[8] === n[8] && r[9] === n[9] && r[10] === n[10] && r[11] === n[11] && r[12] === n[12] && r[13] === n[13] && r[14] === n[14] && r[15] === n[15]; } /** * Clone the current matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#18 * @returns a new matrix from the current matrix */ clone() { const e = new this.constructor(); return e.copyFrom(this), e; } /** * Returns the name of the current matrix class * @returns the string "Matrix" */ getClassName() { return "Matrix"; } /** * Gets the hash code of the current matrix * @returns the hash code */ getHashCode() { let e = Kl(this._m[0]); for (let t = 1; t < 16; t++) e = e * 397 ^ Kl(this._m[t]); return e; } /** * Decomposes the current Matrix into a translation, rotation and scaling components of the provided node * Example Playground - https://playground.babylonjs.com/#AV9X17#13 * @param node the node to decompose the matrix to * @returns true if operation was successful */ decomposeToTransformNode(e) { return e.rotationQuaternion = e.rotationQuaternion || new Ze(), this.decompose(e.scaling, e.rotationQuaternion, e.position); } /** * Decomposes the current Matrix into a translation, rotation and scaling components * Example Playground - https://playground.babylonjs.com/#AV9X17#12 * @param scale defines the scale vector3 given as a reference to update * @param rotation defines the rotation quaternion given as a reference to update * @param translation defines the translation vector3 given as a reference to update * @param preserveScalingNode Use scaling sign coming from this node. Otherwise scaling sign might change. * @param useAbsoluteScaling Use scaling sign coming from this absoluteScaling when true or scaling otherwise. * @returns true if operation was successful */ decompose(e, t, r, n, i = !0) { if (this._isIdentity) return r && r.setAll(0), e && e.setAll(1), t && t.copyFromFloats(0, 0, 0, 1), !0; const s = this._m; if (r && r.copyFromFloats(s[12], s[13], s[14]), e = e || Kr.Vector3[0], e.x = Math.sqrt(s[0] * s[0] + s[1] * s[1] + s[2] * s[2]), e.y = Math.sqrt(s[4] * s[4] + s[5] * s[5] + s[6] * s[6]), e.z = Math.sqrt(s[8] * s[8] + s[9] * s[9] + s[10] * s[10]), n) { const a = (i ? n.absoluteScaling.x : n.scaling.x) < 0 ? -1 : 1, f = (i ? n.absoluteScaling.y : n.scaling.y) < 0 ? -1 : 1, o = (i ? n.absoluteScaling.z : n.scaling.z) < 0 ? -1 : 1; e.x *= a, e.y *= f, e.z *= o; } else this.determinant() <= 0 && (e.y *= -1); if (e._x === 0 || e._y === 0 || e._z === 0) return t && t.copyFromFloats(0, 0, 0, 1), !1; if (t) { const a = 1 / e._x, f = 1 / e._y, o = 1 / e._z; he.FromValuesToRef(s[0] * a, s[1] * a, s[2] * a, 0, s[4] * f, s[5] * f, s[6] * f, 0, s[8] * o, s[9] * o, s[10] * o, 0, 0, 0, 0, 1, Kr.Matrix[0]), Ze.FromRotationMatrixToRef(Kr.Matrix[0], t); } return !0; } /** * Gets specific row of the matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#36 * @param index defines the number of the row to get * @returns the index-th row of the current matrix as a new Vector4 */ getRow(e) { if (e < 0 || e > 3) return null; const t = e * 4; return new Ir(this._m[t + 0], this._m[t + 1], this._m[t + 2], this._m[t + 3]); } /** * Gets specific row of the matrix to ref * Example Playground - https://playground.babylonjs.com/#AV9X17#36 * @param index defines the number of the row to get * @param rowVector vector to store the index-th row of the current matrix * @returns result input */ getRowToRef(e, t) { if (e >= 0 && e <= 3) { const r = e * 4; t.x = this._m[r + 0], t.y = this._m[r + 1], t.z = this._m[r + 2], t.w = this._m[r + 3]; } return t; } /** * Sets the index-th row of the current matrix to the vector4 values * Example Playground - https://playground.babylonjs.com/#AV9X17#36 * @param index defines the number of the row to set * @param row defines the target vector4 * @returns the updated current matrix */ setRow(e, t) { return this.setRowFromFloats(e, t.x, t.y, t.z, t.w); } /** * Compute the transpose of the matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#40 * @returns the new transposed matrix */ transpose() { const e = new this.constructor(); return he.TransposeToRef(this, e), e; } /** * Compute the transpose of the matrix and store it in a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#41 * @param result defines the target matrix * @returns result input */ transposeToRef(e) { return he.TransposeToRef(this, e), e; } /** * Sets the index-th row of the current matrix with the given 4 x float values * Example Playground - https://playground.babylonjs.com/#AV9X17#36 * @param index defines the row index * @param x defines the x component to set * @param y defines the y component to set * @param z defines the z component to set * @param w defines the w component to set * @returns the updated current matrix */ setRowFromFloats(e, t, r, n, i) { if (e < 0 || e > 3) return this; const s = e * 4; return this._m[s + 0] = t, this._m[s + 1] = r, this._m[s + 2] = n, this._m[s + 3] = i, this.markAsUpdated(), this; } /** * Compute a new matrix set with the current matrix values multiplied by scale (float) * @param scale defines the scale factor * @returns a new matrix */ scale(e) { const t = new this.constructor(); return this.scaleToRef(e, t), t; } /** * Scale the current matrix values by a factor to a given result matrix * @param scale defines the scale factor * @param result defines the matrix to store the result * @returns result input */ scaleToRef(e, t) { for (let r = 0; r < 16; r++) t._m[r] = this._m[r] * e; return t.markAsUpdated(), t; } /** * Scale the current matrix values by a factor and add the result to a given matrix * @param scale defines the scale factor * @param result defines the Matrix to store the result * @returns result input */ scaleAndAddToRef(e, t) { for (let r = 0; r < 16; r++) t._m[r] += this._m[r] * e; return t.markAsUpdated(), t; } /** * Writes to the given matrix a normal matrix, computed from this one (using values from identity matrix for fourth row and column). * Example Playground - https://playground.babylonjs.com/#AV9X17#17 * @param ref matrix to store the result */ toNormalMatrix(e) { const t = Kr.Matrix[0]; this.invertToRef(t), t.transposeToRef(e); const r = e._m; return he.FromValuesToRef(r[0], r[1], r[2], 0, r[4], r[5], r[6], 0, r[8], r[9], r[10], 0, 0, 0, 0, 1, e), e; } /** * Gets only rotation part of the current matrix * @returns a new matrix sets to the extracted rotation matrix from the current one */ getRotationMatrix() { const e = new this.constructor(); return this.getRotationMatrixToRef(e), e; } /** * Extracts the rotation matrix from the current one and sets it as the given "result" * @param result defines the target matrix to store data to * @returns result input */ getRotationMatrixToRef(e) { const t = Kr.Vector3[0]; if (!this.decompose(t)) return he.IdentityToRef(e), e; const r = this._m, n = 1 / t._x, i = 1 / t._y, s = 1 / t._z; return he.FromValuesToRef(r[0] * n, r[1] * n, r[2] * n, 0, r[4] * i, r[5] * i, r[6] * i, 0, r[8] * s, r[9] * s, r[10] * s, 0, 0, 0, 0, 1, e), e; } /** * Toggles model matrix from being right handed to left handed in place and vice versa */ toggleModelMatrixHandInPlace() { const e = this._m; return e[2] *= -1, e[6] *= -1, e[8] *= -1, e[9] *= -1, e[14] *= -1, this.markAsUpdated(), this; } /** * Toggles projection matrix from being right handed to left handed in place and vice versa */ toggleProjectionMatrixHandInPlace() { const e = this._m; return e[8] *= -1, e[9] *= -1, e[10] *= -1, e[11] *= -1, this.markAsUpdated(), this; } // Statics /** * Creates a matrix from an array * Example Playground - https://playground.babylonjs.com/#AV9X17#42 * @param array defines the source array * @param offset defines an offset in the source array * @returns a new Matrix set from the starting index of the given array */ static FromArray(e, t = 0) { const r = new he(); return he.FromArrayToRef(e, t, r), r; } /** * Copy the content of an array into a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#43 * @param array defines the source array * @param offset defines an offset in the source array * @param result defines the target matrix * @returns result input */ static FromArrayToRef(e, t, r) { for (let n = 0; n < 16; n++) r._m[n] = e[n + t]; return r.markAsUpdated(), r; } /** * Stores an array into a matrix after having multiplied each component by a given factor * Example Playground - https://playground.babylonjs.com/#AV9X17#50 * @param array defines the source array * @param offset defines the offset in the source array * @param scale defines the scaling factor * @param result defines the target matrix * @returns result input */ static FromFloat32ArrayToRefScaled(e, t, r, n) { for (let i = 0; i < 16; i++) n._m[i] = e[i + t] * r; return n.markAsUpdated(), n; } /** * Gets an identity matrix that must not be updated */ static get IdentityReadOnly() { return he._IdentityReadOnly; } /** * Stores a list of values (16) inside a given matrix * @param initialM11 defines 1st value of 1st row * @param initialM12 defines 2nd value of 1st row * @param initialM13 defines 3rd value of 1st row * @param initialM14 defines 4th value of 1st row * @param initialM21 defines 1st value of 2nd row * @param initialM22 defines 2nd value of 2nd row * @param initialM23 defines 3rd value of 2nd row * @param initialM24 defines 4th value of 2nd row * @param initialM31 defines 1st value of 3rd row * @param initialM32 defines 2nd value of 3rd row * @param initialM33 defines 3rd value of 3rd row * @param initialM34 defines 4th value of 3rd row * @param initialM41 defines 1st value of 4th row * @param initialM42 defines 2nd value of 4th row * @param initialM43 defines 3rd value of 4th row * @param initialM44 defines 4th value of 4th row * @param result defines the target matrix * @returns result input */ static FromValuesToRef(e, t, r, n, i, s, a, f, o, d, v, u, l, P, p, c, H) { const T = H._m; T[0] = e, T[1] = t, T[2] = r, T[3] = n, T[4] = i, T[5] = s, T[6] = a, T[7] = f, T[8] = o, T[9] = d, T[10] = v, T[11] = u, T[12] = l, T[13] = P, T[14] = p, T[15] = c, H.markAsUpdated(); } /** * Creates new matrix from a list of values (16) * @param initialM11 defines 1st value of 1st row * @param initialM12 defines 2nd value of 1st row * @param initialM13 defines 3rd value of 1st row * @param initialM14 defines 4th value of 1st row * @param initialM21 defines 1st value of 2nd row * @param initialM22 defines 2nd value of 2nd row * @param initialM23 defines 3rd value of 2nd row * @param initialM24 defines 4th value of 2nd row * @param initialM31 defines 1st value of 3rd row * @param initialM32 defines 2nd value of 3rd row * @param initialM33 defines 3rd value of 3rd row * @param initialM34 defines 4th value of 3rd row * @param initialM41 defines 1st value of 4th row * @param initialM42 defines 2nd value of 4th row * @param initialM43 defines 3rd value of 4th row * @param initialM44 defines 4th value of 4th row * @returns the new matrix */ static FromValues(e, t, r, n, i, s, a, f, o, d, v, u, l, P, p, c) { const H = new he(), T = H._m; return T[0] = e, T[1] = t, T[2] = r, T[3] = n, T[4] = i, T[5] = s, T[6] = a, T[7] = f, T[8] = o, T[9] = d, T[10] = v, T[11] = u, T[12] = l, T[13] = P, T[14] = p, T[15] = c, H.markAsUpdated(), H; } /** * Creates a new matrix composed by merging scale (vector3), rotation (quaternion) and translation (vector3) * Example Playground - https://playground.babylonjs.com/#AV9X17#24 * @param scale defines the scale vector3 * @param rotation defines the rotation quaternion * @param translation defines the translation vector3 * @returns a new matrix */ static Compose(e, t, r) { const n = new he(); return he.ComposeToRef(e, t, r, n), n; } /** * Sets a matrix to a value composed by merging scale (vector3), rotation (quaternion) and translation (vector3) * Example Playground - https://playground.babylonjs.com/#AV9X17#25 * @param scale defines the scale vector3 * @param rotation defines the rotation quaternion * @param translation defines the translation vector3 * @param result defines the target matrix * @returns result input */ static ComposeToRef(e, t, r, n) { const i = n._m, s = t._x, a = t._y, f = t._z, o = t._w, d = s + s, v = a + a, u = f + f, l = s * d, P = s * v, p = s * u, c = a * v, H = a * u, T = f * u, q = o * d, b = o * v, j = o * u, w = e._x, m = e._y, I = e._z; return i[0] = (1 - (c + T)) * w, i[1] = (P + j) * w, i[2] = (p - b) * w, i[3] = 0, i[4] = (P - j) * m, i[5] = (1 - (l + T)) * m, i[6] = (H + q) * m, i[7] = 0, i[8] = (p + b) * I, i[9] = (H - q) * I, i[10] = (1 - (l + c)) * I, i[11] = 0, i[12] = r._x, i[13] = r._y, i[14] = r._z, i[15] = 1, n.markAsUpdated(), n; } /** * Creates a new identity matrix * @returns a new identity matrix */ static Identity() { const e = he.FromValues(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1); return e._updateIdentityStatus(!0), e; } /** * Creates a new identity matrix and stores the result in a given matrix * @param result defines the target matrix * @returns result input */ static IdentityToRef(e) { return he.FromValuesToRef(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, e), e._updateIdentityStatus(!0), e; } /** * Creates a new zero matrix * @returns a new zero matrix */ static Zero() { const e = he.FromValues(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0); return e._updateIdentityStatus(!1), e; } /** * Creates a new rotation matrix for "angle" radians around the X axis * Example Playground - https://playground.babylonjs.com/#AV9X17#97 * @param angle defines the angle (in radians) to use * @returns the new matrix */ static RotationX(e) { const t = new he(); return he.RotationXToRef(e, t), t; } /** * Creates a new matrix as the invert of a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#124 * @param source defines the source matrix * @returns the new matrix */ static Invert(e) { const t = new e.constructor(); return e.invertToRef(t), t; } /** * Creates a new rotation matrix for "angle" radians around the X axis and stores it in a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#98 * @param angle defines the angle (in radians) to use * @param result defines the target matrix * @returns result input */ static RotationXToRef(e, t) { const r = Math.sin(e), n = Math.cos(e); return he.FromValuesToRef(1, 0, 0, 0, 0, n, r, 0, 0, -r, n, 0, 0, 0, 0, 1, t), t._updateIdentityStatus(n === 1 && r === 0), t; } /** * Creates a new rotation matrix for "angle" radians around the Y axis * Example Playground - https://playground.babylonjs.com/#AV9X17#99 * @param angle defines the angle (in radians) to use * @returns the new matrix */ static RotationY(e) { const t = new he(); return he.RotationYToRef(e, t), t; } /** * Creates a new rotation matrix for "angle" radians around the Y axis and stores it in a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#100 * @param angle defines the angle (in radians) to use * @param result defines the target matrix * @returns result input */ static RotationYToRef(e, t) { const r = Math.sin(e), n = Math.cos(e); return he.FromValuesToRef(n, 0, -r, 0, 0, 1, 0, 0, r, 0, n, 0, 0, 0, 0, 1, t), t._updateIdentityStatus(n === 1 && r === 0), t; } /** * Creates a new rotation matrix for "angle" radians around the Z axis * Example Playground - https://playground.babylonjs.com/#AV9X17#101 * @param angle defines the angle (in radians) to use * @returns the new matrix */ static RotationZ(e) { const t = new he(); return he.RotationZToRef(e, t), t; } /** * Creates a new rotation matrix for "angle" radians around the Z axis and stores it in a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#102 * @param angle defines the angle (in radians) to use * @param result defines the target matrix * @returns result input */ static RotationZToRef(e, t) { const r = Math.sin(e), n = Math.cos(e); return he.FromValuesToRef(n, r, 0, 0, -r, n, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, t), t._updateIdentityStatus(n === 1 && r === 0), t; } /** * Creates a new rotation matrix for "angle" radians around the given axis * Example Playground - https://playground.babylonjs.com/#AV9X17#96 * @param axis defines the axis to use * @param angle defines the angle (in radians) to use * @returns the new matrix */ static RotationAxis(e, t) { const r = new he(); return he.RotationAxisToRef(e, t, r), r; } /** * Creates a new rotation matrix for "angle" radians around the given axis and stores it in a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#94 * @param axis defines the axis to use * @param angle defines the angle (in radians) to use * @param result defines the target matrix * @returns result input */ static RotationAxisToRef(e, t, r) { const n = Math.sin(-t), i = Math.cos(-t), s = 1 - i; e.normalize(); const a = r._m; return a[0] = e._x * e._x * s + i, a[1] = e._x * e._y * s - e._z * n, a[2] = e._x * e._z * s + e._y * n, a[3] = 0, a[4] = e._y * e._x * s + e._z * n, a[5] = e._y * e._y * s + i, a[6] = e._y * e._z * s - e._x * n, a[7] = 0, a[8] = e._z * e._x * s - e._y * n, a[9] = e._z * e._y * s + e._x * n, a[10] = e._z * e._z * s + i, a[11] = 0, a[12] = 0, a[13] = 0, a[14] = 0, a[15] = 1, r.markAsUpdated(), r; } /** * Takes normalised vectors and returns a rotation matrix to align "from" with "to". * Taken from http://www.iquilezles.org/www/articles/noacos/noacos.htm * Example Playground - https://playground.babylonjs.com/#AV9X17#93 * @param from defines the vector to align * @param to defines the vector to align to * @param result defines the target matrix * @param useYAxisForCoplanar defines a boolean indicating that we should favor Y axis for coplanar vectors (default is false) * @returns result input */ static RotationAlignToRef(e, t, r, n = !1) { const i = S.Dot(t, e), s = r._m; if (i < -1 + Dn) s[0] = -1, s[1] = 0, s[2] = 0, s[3] = 0, s[4] = 0, s[5] = n ? 1 : -1, s[6] = 0, s[7] = 0, s[8] = 0, s[9] = 0, s[10] = n ? -1 : 1, s[11] = 0; else { const a = S.Cross(t, e), f = 1 / (1 + i); s[0] = a._x * a._x * f + i, s[1] = a._y * a._x * f - a._z, s[2] = a._z * a._x * f + a._y, s[3] = 0, s[4] = a._x * a._y * f + a._z, s[5] = a._y * a._y * f + i, s[6] = a._z * a._y * f - a._x, s[7] = 0, s[8] = a._x * a._z * f - a._y, s[9] = a._y * a._z * f + a._x, s[10] = a._z * a._z * f + i, s[11] = 0; } return s[12] = 0, s[13] = 0, s[14] = 0, s[15] = 1, r.markAsUpdated(), r; } /** * Creates a rotation matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#103 * Example Playground - https://playground.babylonjs.com/#AV9X17#105 * @param yaw defines the yaw angle in radians (Y axis) * @param pitch defines the pitch angle in radians (X axis) * @param roll defines the roll angle in radians (Z axis) * @returns the new rotation matrix */ static RotationYawPitchRoll(e, t, r) { const n = new he(); return he.RotationYawPitchRollToRef(e, t, r, n), n; } /** * Creates a rotation matrix and stores it in a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#104 * @param yaw defines the yaw angle in radians (Y axis) * @param pitch defines the pitch angle in radians (X axis) * @param roll defines the roll angle in radians (Z axis) * @param result defines the target matrix * @returns result input */ static RotationYawPitchRollToRef(e, t, r, n) { return Ze.RotationYawPitchRollToRef(e, t, r, Kr.Quaternion[0]), Kr.Quaternion[0].toRotationMatrix(n), n; } /** * Creates a scaling matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#107 * @param x defines the scale factor on X axis * @param y defines the scale factor on Y axis * @param z defines the scale factor on Z axis * @returns the new matrix */ static Scaling(e, t, r) { const n = new he(); return he.ScalingToRef(e, t, r, n), n; } /** * Creates a scaling matrix and stores it in a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#108 * @param x defines the scale factor on X axis * @param y defines the scale factor on Y axis * @param z defines the scale factor on Z axis * @param result defines the target matrix * @returns result input */ static ScalingToRef(e, t, r, n) { return he.FromValuesToRef(e, 0, 0, 0, 0, t, 0, 0, 0, 0, r, 0, 0, 0, 0, 1, n), n._updateIdentityStatus(e === 1 && t === 1 && r === 1), n; } /** * Creates a translation matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#109 * @param x defines the translation on X axis * @param y defines the translation on Y axis * @param z defines the translationon Z axis * @returns the new matrix */ static Translation(e, t, r) { const n = new he(); return he.TranslationToRef(e, t, r, n), n; } /** * Creates a translation matrix and stores it in a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#110 * @param x defines the translation on X axis * @param y defines the translation on Y axis * @param z defines the translationon Z axis * @param result defines the target matrix * @returns result input */ static TranslationToRef(e, t, r, n) { return he.FromValuesToRef(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, e, t, r, 1, n), n._updateIdentityStatus(e === 0 && t === 0 && r === 0), n; } /** * Returns a new Matrix whose values are the interpolated values for "gradient" (float) between the ones of the matrices "startValue" and "endValue". * Example Playground - https://playground.babylonjs.com/#AV9X17#55 * @param startValue defines the start value * @param endValue defines the end value * @param gradient defines the gradient factor * @returns the new matrix */ static Lerp(e, t, r) { const n = new e.constructor(); return he.LerpToRef(e, t, r, n), n; } /** * Set the given matrix "result" as the interpolated values for "gradient" (float) between the ones of the matrices "startValue" and "endValue". * Example Playground - https://playground.babylonjs.com/#AV9X17#54 * @param startValue defines the start value * @param endValue defines the end value * @param gradient defines the gradient factor * @param result defines the Matrix object where to store data * @returns result input */ static LerpToRef(e, t, r, n) { const i = n._m, s = e.m, a = t.m; for (let f = 0; f < 16; f++) i[f] = s[f] * (1 - r) + a[f] * r; return n.markAsUpdated(), n; } /** * Builds a new matrix whose values are computed by: * * decomposing the "startValue" and "endValue" matrices into their respective scale, rotation and translation matrices * * interpolating for "gradient" (float) the values between each of these decomposed matrices between the start and the end * * recomposing a new matrix from these 3 interpolated scale, rotation and translation matrices * Example Playground - https://playground.babylonjs.com/#AV9X17#22 * Example Playground - https://playground.babylonjs.com/#AV9X17#51 * @param startValue defines the first matrix * @param endValue defines the second matrix * @param gradient defines the gradient between the two matrices * @returns the new matrix */ static DecomposeLerp(e, t, r) { const n = new e.constructor(); return he.DecomposeLerpToRef(e, t, r, n), n; } /** * Update a matrix to values which are computed by: * * decomposing the "startValue" and "endValue" matrices into their respective scale, rotation and translation matrices * * interpolating for "gradient" (float) the values between each of these decomposed matrices between the start and the end * * recomposing a new matrix from these 3 interpolated scale, rotation and translation matrices * Example Playground - https://playground.babylonjs.com/#AV9X17#23 * Example Playground - https://playground.babylonjs.com/#AV9X17#53 * @param startValue defines the first matrix * @param endValue defines the second matrix * @param gradient defines the gradient between the two matrices * @param result defines the target matrix * @returns result input */ static DecomposeLerpToRef(e, t, r, n) { const i = Kr.Vector3[0], s = Kr.Quaternion[0], a = Kr.Vector3[1]; e.decompose(i, s, a); const f = Kr.Vector3[2], o = Kr.Quaternion[1], d = Kr.Vector3[3]; t.decompose(f, o, d); const v = Kr.Vector3[4]; S.LerpToRef(i, f, r, v); const u = Kr.Quaternion[2]; Ze.SlerpToRef(s, o, r, u); const l = Kr.Vector3[5]; return S.LerpToRef(a, d, r, l), he.ComposeToRef(v, u, l, n), n; } /** * Creates a new matrix that transforms vertices from world space to camera space. It takes three vectors as arguments that together describe the position and orientation of the camera. * This function generates a matrix suitable for a left handed coordinate system * Example Playground - https://playground.babylonjs.com/#AV9X17#58 * Example Playground - https://playground.babylonjs.com/#AV9X17#59 * @param eye defines the final position of the entity * @param target defines where the entity should look at * @param up defines the up vector for the entity * @returns the new matrix */ static LookAtLH(e, t, r) { const n = new he(); return he.LookAtLHToRef(e, t, r, n), n; } /** * Sets the given "result" Matrix to a matrix that transforms vertices from world space to camera space. It takes three vectors as arguments that together describe the position and orientation of the camera. * This function generates a matrix suitable for a left handed coordinate system * Example Playground - https://playground.babylonjs.com/#AV9X17#60 * Example Playground - https://playground.babylonjs.com/#AV9X17#61 * @param eye defines the final position of the entity * @param target defines where the entity should look at * @param up defines the up vector for the entity * @param result defines the target matrix * @returns result input */ static LookAtLHToRef(e, t, r, n) { const i = Kr.Vector3[0], s = Kr.Vector3[1], a = Kr.Vector3[2]; t.subtractToRef(e, a), a.normalize(), S.CrossToRef(r, a, i); const f = i.lengthSquared(); f === 0 ? i.x = 1 : i.normalizeFromLength(Math.sqrt(f)), S.CrossToRef(a, i, s), s.normalize(); const o = -S.Dot(i, e), d = -S.Dot(s, e), v = -S.Dot(a, e); return he.FromValuesToRef(i._x, s._x, a._x, 0, i._y, s._y, a._y, 0, i._z, s._z, a._z, 0, o, d, v, 1, n), n; } /** * Creates a new matrix that transforms vertices from world space to camera space. It takes three vectors as arguments that together describe the position and orientation of the camera. * This function generates a matrix suitable for a right handed coordinate system * Example Playground - https://playground.babylonjs.com/#AV9X17#62 * Example Playground - https://playground.babylonjs.com/#AV9X17#63 * @param eye defines the final position of the entity * @param target defines where the entity should look at * @param up defines the up vector for the entity * @returns the new matrix */ static LookAtRH(e, t, r) { const n = new he(); return he.LookAtRHToRef(e, t, r, n), n; } /** * Sets the given "result" Matrix to a matrix that transforms vertices from world space to camera space. It takes three vectors as arguments that together describe the position and orientation of the camera. * This function generates a matrix suitable for a right handed coordinate system * Example Playground - https://playground.babylonjs.com/#AV9X17#64 * Example Playground - https://playground.babylonjs.com/#AV9X17#65 * @param eye defines the final position of the entity * @param target defines where the entity should look at * @param up defines the up vector for the entity * @param result defines the target matrix * @returns result input */ static LookAtRHToRef(e, t, r, n) { const i = Kr.Vector3[0], s = Kr.Vector3[1], a = Kr.Vector3[2]; e.subtractToRef(t, a), a.normalize(), S.CrossToRef(r, a, i); const f = i.lengthSquared(); f === 0 ? i.x = 1 : i.normalizeFromLength(Math.sqrt(f)), S.CrossToRef(a, i, s), s.normalize(); const o = -S.Dot(i, e), d = -S.Dot(s, e), v = -S.Dot(a, e); return he.FromValuesToRef(i._x, s._x, a._x, 0, i._y, s._y, a._y, 0, i._z, s._z, a._z, 0, o, d, v, 1, n), n; } /** * Creates a new matrix that transforms vertices from world space to camera space. It takes two vectors as arguments that together describe the orientation of the camera. The position is assumed to be at the origin (0,0,0) * This function generates a matrix suitable for a left handed coordinate system * Example Playground - https://playground.babylonjs.com/#AV9X17#66 * @param forward defines the forward direction - Must be normalized and orthogonal to up. * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward. * @returns the new matrix */ static LookDirectionLH(e, t) { const r = new he(); return he.LookDirectionLHToRef(e, t, r), r; } /** * Sets the given "result" Matrix to a matrix that transforms vertices from world space to camera space. It takes two vectors as arguments that together describe the orientation of the camera. The position is assumed to be at the origin (0,0,0) * This function generates a matrix suitable for a left handed coordinate system * Example Playground - https://playground.babylonjs.com/#AV9X17#67 * @param forward defines the forward direction - Must be normalized and orthogonal to up. * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward. * @param result defines the target matrix * @returns result input */ static LookDirectionLHToRef(e, t, r) { const n = Kr.Vector3[0]; n.copyFrom(e), n.scaleInPlace(-1); const i = Kr.Vector3[1]; return S.CrossToRef(t, n, i), he.FromValuesToRef(i._x, i._y, i._z, 0, t._x, t._y, t._z, 0, n._x, n._y, n._z, 0, 0, 0, 0, 1, r), r; } /** * Creates a new matrix that transforms vertices from world space to camera space. It takes two vectors as arguments that together describe the orientation of the camera. The position is assumed to be at the origin (0,0,0) * This function generates a matrix suitable for a right handed coordinate system * Example Playground - https://playground.babylonjs.com/#AV9X17#68 * @param forward defines the forward direction - Must be normalized and orthogonal to up. * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward. * @returns the new matrix */ static LookDirectionRH(e, t) { const r = new he(); return he.LookDirectionRHToRef(e, t, r), r; } /** * Sets the given "result" Matrix to a matrix that transforms vertices from world space to camera space. It takes two vectors as arguments that together describe the orientation of the camera. The position is assumed to be at the origin (0,0,0) * This function generates a matrix suitable for a right handed coordinate system * Example Playground - https://playground.babylonjs.com/#AV9X17#69 * @param forward defines the forward direction - Must be normalized and orthogonal to up. * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward. * @param result defines the target matrix * @returns result input */ static LookDirectionRHToRef(e, t, r) { const n = Kr.Vector3[2]; return S.CrossToRef(t, e, n), he.FromValuesToRef(n._x, n._y, n._z, 0, t._x, t._y, t._z, 0, e._x, e._y, e._z, 0, 0, 0, 0, 1, r), r; } /** * Create a left-handed orthographic projection matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#70 * @param width defines the viewport width * @param height defines the viewport height * @param znear defines the near clip plane * @param zfar defines the far clip plane * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false) * @returns a new matrix as a left-handed orthographic projection matrix */ static OrthoLH(e, t, r, n, i) { const s = new he(); return he.OrthoLHToRef(e, t, r, n, s, i), s; } /** * Store a left-handed orthographic projection to a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#71 * @param width defines the viewport width * @param height defines the viewport height * @param znear defines the near clip plane * @param zfar defines the far clip plane * @param result defines the target matrix * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false) * @returns result input */ static OrthoLHToRef(e, t, r, n, i, s) { const a = r, f = n, o = 2 / e, d = 2 / t, v = 2 / (f - a), u = -(f + a) / (f - a); return he.FromValuesToRef(o, 0, 0, 0, 0, d, 0, 0, 0, 0, v, 0, 0, 0, u, 1, i), s && i.multiplyToRef(kw, i), i._updateIdentityStatus(o === 1 && d === 1 && v === 1 && u === 0), i; } /** * Create a left-handed orthographic projection matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#72 * @param left defines the viewport left coordinate * @param right defines the viewport right coordinate * @param bottom defines the viewport bottom coordinate * @param top defines the viewport top coordinate * @param znear defines the near clip plane * @param zfar defines the far clip plane * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false) * @returns a new matrix as a left-handed orthographic projection matrix */ static OrthoOffCenterLH(e, t, r, n, i, s, a) { const f = new he(); return he.OrthoOffCenterLHToRef(e, t, r, n, i, s, f, a), f; } /** * Stores a left-handed orthographic projection into a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#73 * @param left defines the viewport left coordinate * @param right defines the viewport right coordinate * @param bottom defines the viewport bottom coordinate * @param top defines the viewport top coordinate * @param znear defines the near clip plane * @param zfar defines the far clip plane * @param result defines the target matrix * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false) * @returns result input */ static OrthoOffCenterLHToRef(e, t, r, n, i, s, a, f) { const o = i, d = s, v = 2 / (t - e), u = 2 / (n - r), l = 2 / (d - o), P = -(d + o) / (d - o), p = (e + t) / (e - t), c = (n + r) / (r - n); return he.FromValuesToRef(v, 0, 0, 0, 0, u, 0, 0, 0, 0, l, 0, p, c, P, 1, a), f && a.multiplyToRef(kw, a), a.markAsUpdated(), a; } /** * Stores a left-handed oblique projection into a given matrix * @param left defines the viewport left coordinate * @param right defines the viewport right coordinate * @param bottom defines the viewport bottom coordinate * @param top defines the viewport top coordinate * @param znear defines the near clip plane * @param zfar defines the far clip plane * @param angle Angle (along X/Y Plane) to apply shear * @param length Length of the shear * @param distance Distance from shear point * @param result defines the target matrix * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false) * @returns result input */ static ObliqueOffCenterLHToRef(e, t, r, n, i, s, a, f, o, d, v) { const u = -a * Math.cos(f), l = -a * Math.sin(f); return he.TranslationToRef(0, 0, -o, Kr.Matrix[1]), he.FromValuesToRef(1, 0, 0, 0, 0, 1, 0, 0, u, l, 1, 0, 0, 0, 0, 1, Kr.Matrix[0]), Kr.Matrix[1].multiplyToRef(Kr.Matrix[0], Kr.Matrix[0]), he.TranslationToRef(0, 0, o, Kr.Matrix[1]), Kr.Matrix[0].multiplyToRef(Kr.Matrix[1], Kr.Matrix[0]), he.OrthoOffCenterLHToRef(e, t, r, n, i, s, d, v), Kr.Matrix[0].multiplyToRef(d, d), d; } /** * Creates a right-handed orthographic projection matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#76 * @param left defines the viewport left coordinate * @param right defines the viewport right coordinate * @param bottom defines the viewport bottom coordinate * @param top defines the viewport top coordinate * @param znear defines the near clip plane * @param zfar defines the far clip plane * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false) * @returns a new matrix as a right-handed orthographic projection matrix */ static OrthoOffCenterRH(e, t, r, n, i, s, a) { const f = new he(); return he.OrthoOffCenterRHToRef(e, t, r, n, i, s, f, a), f; } /** * Stores a right-handed orthographic projection into a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#77 * @param left defines the viewport left coordinate * @param right defines the viewport right coordinate * @param bottom defines the viewport bottom coordinate * @param top defines the viewport top coordinate * @param znear defines the near clip plane * @param zfar defines the far clip plane * @param result defines the target matrix * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false) * @returns result input */ static OrthoOffCenterRHToRef(e, t, r, n, i, s, a, f) { return he.OrthoOffCenterLHToRef(e, t, r, n, i, s, a, f), a._m[10] *= -1, a; } /** * Stores a right-handed oblique projection into a given matrix * @param left defines the viewport left coordinate * @param right defines the viewport right coordinate * @param bottom defines the viewport bottom coordinate * @param top defines the viewport top coordinate * @param znear defines the near clip plane * @param zfar defines the far clip plane * @param angle Angle (along X/Y Plane) to apply shear * @param length Length of the shear * @param distance Distance from shear point * @param result defines the target matrix * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false) * @returns result input */ static ObliqueOffCenterRHToRef(e, t, r, n, i, s, a, f, o, d, v) { const u = a * Math.cos(f), l = a * Math.sin(f); return he.TranslationToRef(0, 0, o, Kr.Matrix[1]), he.FromValuesToRef(1, 0, 0, 0, 0, 1, 0, 0, u, l, 1, 0, 0, 0, 0, 1, Kr.Matrix[0]), Kr.Matrix[1].multiplyToRef(Kr.Matrix[0], Kr.Matrix[0]), he.TranslationToRef(0, 0, -o, Kr.Matrix[1]), Kr.Matrix[0].multiplyToRef(Kr.Matrix[1], Kr.Matrix[0]), he.OrthoOffCenterRHToRef(e, t, r, n, i, s, d, v), Kr.Matrix[0].multiplyToRef(d, d), d; } /** * Creates a left-handed perspective projection matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#85 * @param width defines the viewport width * @param height defines the viewport height * @param znear defines the near clip plane * @param zfar defines the far clip plane * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false) * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal) * @returns a new matrix as a left-handed perspective projection matrix */ static PerspectiveLH(e, t, r, n, i, s = 0) { const a = new he(), f = r, o = n, d = 2 * f / e, v = 2 * f / t, u = (o + f) / (o - f), l = -2 * o * f / (o - f), P = Math.tan(s); return he.FromValuesToRef(d, 0, 0, 0, 0, v, 0, P, 0, 0, u, 1, 0, 0, l, 0, a), i && a.multiplyToRef(kw, a), a._updateIdentityStatus(!1), a; } /** * Creates a left-handed perspective projection matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#78 * @param fov defines the horizontal field of view * @param aspect defines the aspect ratio * @param znear defines the near clip plane * @param zfar defines the far clip plane. If 0, assume we are in "infinite zfar" mode * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false) * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal) * @param reverseDepthBufferMode true to indicate that we are in a reverse depth buffer mode (meaning znear and zfar have been inverted when calling the function) * @returns a new matrix as a left-handed perspective projection matrix */ static PerspectiveFovLH(e, t, r, n, i, s = 0, a = !1) { const f = new he(); return he.PerspectiveFovLHToRef(e, t, r, n, f, !0, i, s, a), f; } /** * Stores a left-handed perspective projection into a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#81 * @param fov defines the horizontal field of view * @param aspect defines the aspect ratio * @param znear defines the near clip plane * @param zfar defines the far clip plane. If 0, assume we are in "infinite zfar" mode * @param result defines the target matrix * @param isVerticalFovFixed defines it the fov is vertically fixed (default) or horizontally * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false) * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal) * @param reverseDepthBufferMode true to indicate that we are in a reverse depth buffer mode (meaning znear and zfar have been inverted when calling the function) * @returns result input */ static PerspectiveFovLHToRef(e, t, r, n, i, s = !0, a, f = 0, o = !1) { const d = r, v = n, u = 1 / Math.tan(e * 0.5), l = s ? u / t : u, P = s ? u : u * t, p = o && d === 0 ? -1 : v !== 0 ? (v + d) / (v - d) : 1, c = o && d === 0 ? 2 * v : v !== 0 ? -2 * v * d / (v - d) : -2 * d, H = Math.tan(f); return he.FromValuesToRef(l, 0, 0, 0, 0, P, 0, H, 0, 0, p, 1, 0, 0, c, 0, i), a && i.multiplyToRef(kw, i), i._updateIdentityStatus(!1), i; } /** * Stores a left-handed perspective projection into a given matrix with depth reversed * Example Playground - https://playground.babylonjs.com/#AV9X17#89 * @param fov defines the horizontal field of view * @param aspect defines the aspect ratio * @param znear defines the near clip plane * @param zfar not used as infinity is used as far clip * @param result defines the target matrix * @param isVerticalFovFixed defines it the fov is vertically fixed (default) or horizontally * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false) * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal) * @returns result input */ static PerspectiveFovReverseLHToRef(e, t, r, n, i, s = !0, a, f = 0) { const o = 1 / Math.tan(e * 0.5), d = s ? o / t : o, v = s ? o : o * t, u = Math.tan(f); return he.FromValuesToRef(d, 0, 0, 0, 0, v, 0, u, 0, 0, -r, 1, 0, 0, 1, 0, i), a && i.multiplyToRef(kw, i), i._updateIdentityStatus(!1), i; } /** * Creates a right-handed perspective projection matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#83 * @param fov defines the horizontal field of view * @param aspect defines the aspect ratio * @param znear defines the near clip plane * @param zfar defines the far clip plane. If 0, assume we are in "infinite zfar" mode * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false) * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal) * @param reverseDepthBufferMode true to indicate that we are in a reverse depth buffer mode (meaning znear and zfar have been inverted when calling the function) * @returns a new matrix as a right-handed perspective projection matrix */ static PerspectiveFovRH(e, t, r, n, i, s = 0, a = !1) { const f = new he(); return he.PerspectiveFovRHToRef(e, t, r, n, f, !0, i, s, a), f; } /** * Stores a right-handed perspective projection into a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#84 * @param fov defines the horizontal field of view * @param aspect defines the aspect ratio * @param znear defines the near clip plane * @param zfar defines the far clip plane. If 0, assume we are in "infinite zfar" mode * @param result defines the target matrix * @param isVerticalFovFixed defines it the fov is vertically fixed (default) or horizontally * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false) * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal) * @param reverseDepthBufferMode true to indicate that we are in a reverse depth buffer mode (meaning znear and zfar have been inverted when calling the function) * @returns result input */ static PerspectiveFovRHToRef(e, t, r, n, i, s = !0, a, f = 0, o = !1) { const d = r, v = n, u = 1 / Math.tan(e * 0.5), l = s ? u / t : u, P = s ? u : u * t, p = o && d === 0 ? 1 : v !== 0 ? -(v + d) / (v - d) : -1, c = o && d === 0 ? 2 * v : v !== 0 ? -2 * v * d / (v - d) : -2 * d, H = Math.tan(f); return he.FromValuesToRef(l, 0, 0, 0, 0, P, 0, H, 0, 0, p, -1, 0, 0, c, 0, i), a && i.multiplyToRef(kw, i), i._updateIdentityStatus(!1), i; } /** * Stores a right-handed perspective projection into a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#90 * @param fov defines the horizontal field of view * @param aspect defines the aspect ratio * @param znear defines the near clip plane * @param zfar not used as infinity is used as far clip * @param result defines the target matrix * @param isVerticalFovFixed defines it the fov is vertically fixed (default) or horizontally * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false) * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal) * @returns result input */ static PerspectiveFovReverseRHToRef(e, t, r, n, i, s = !0, a, f = 0) { const o = 1 / Math.tan(e * 0.5), d = s ? o / t : o, v = s ? o : o * t, u = Math.tan(f); return he.FromValuesToRef(d, 0, 0, 0, 0, v, 0, u, 0, 0, -r, -1, 0, 0, -1, 0, i), a && i.multiplyToRef(kw, i), i._updateIdentityStatus(!1), i; } /** * Computes a complete transformation matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#113 * @param viewport defines the viewport to use * @param world defines the world matrix * @param view defines the view matrix * @param projection defines the projection matrix * @param zmin defines the near clip plane * @param zmax defines the far clip plane * @returns the transformation matrix */ static GetFinalMatrix(e, t, r, n, i, s) { const a = e.width, f = e.height, o = e.x, d = e.y, v = he.FromValues(a / 2, 0, 0, 0, 0, -f / 2, 0, 0, 0, 0, s - i, 0, o + a / 2, f / 2 + d, i, 1), u = new t.constructor(); return t.multiplyToRef(r, u), u.multiplyToRef(n, u), u.multiplyToRef(v, u); } /** * Extracts a 2x2 matrix from a given matrix and store the result in a Float32Array * @param matrix defines the matrix to use * @returns a new Float32Array array with 4 elements : the 2x2 matrix extracted from the given matrix */ static GetAsMatrix2x2(e) { const t = e.m, r = [t[0], t[1], t[4], t[5]]; return k9.MatrixUse64Bits ? r : new Float32Array(r); } /** * Extracts a 3x3 matrix from a given matrix and store the result in a Float32Array * @param matrix defines the matrix to use * @returns a new Float32Array array with 9 elements : the 3x3 matrix extracted from the given matrix */ static GetAsMatrix3x3(e) { const t = e.m, r = [t[0], t[1], t[2], t[4], t[5], t[6], t[8], t[9], t[10]]; return k9.MatrixUse64Bits ? r : new Float32Array(r); } /** * Compute the transpose of a given matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#111 * @param matrix defines the matrix to transpose * @returns the new matrix */ static Transpose(e) { const t = new e.constructor(); return he.TransposeToRef(e, t), t; } /** * Compute the transpose of a matrix and store it in a target matrix * Example Playground - https://playground.babylonjs.com/#AV9X17#112 * @param matrix defines the matrix to transpose * @param result defines the target matrix * @returns result input */ static TransposeToRef(e, t) { const r = e.m, n = r[0], i = r[4], s = r[8], a = r[12], f = r[1], o = r[5], d = r[9], v = r[13], u = r[2], l = r[6], P = r[10], p = r[14], c = r[3], H = r[7], T = r[11], q = r[15], b = t._m; return b[0] = n, b[1] = i, b[2] = s, b[3] = a, b[4] = f, b[5] = o, b[6] = d, b[7] = v, b[8] = u, b[9] = l, b[10] = P, b[11] = p, b[12] = c, b[13] = H, b[14] = T, b[15] = q, t.markAsUpdated(), t._updateIdentityStatus(e._isIdentity, e._isIdentityDirty), t; } /** * Computes a reflection matrix from a plane * Example Playground - https://playground.babylonjs.com/#AV9X17#87 * @param plane defines the reflection plane * @returns a new matrix */ static Reflection(e) { const t = new he(); return he.ReflectionToRef(e, t), t; } /** * Computes a reflection matrix from a plane * Example Playground - https://playground.babylonjs.com/#AV9X17#88 * @param plane defines the reflection plane * @param result defines the target matrix * @returns result input */ static ReflectionToRef(e, t) { e.normalize(); const r = e.normal.x, n = e.normal.y, i = e.normal.z, s = -2 * r, a = -2 * n, f = -2 * i; return he.FromValuesToRef(s * r + 1, a * r, f * r, 0, s * n, a * n + 1, f * n, 0, s * i, a * i, f * i + 1, 0, s * e.d, a * e.d, f * e.d, 1, t), t; } /** * Sets the given matrix as a rotation matrix composed from the 3 left handed axes * @param xaxis defines the value of the 1st axis * @param yaxis defines the value of the 2nd axis * @param zaxis defines the value of the 3rd axis * @param result defines the target matrix * @returns result input */ static FromXYZAxesToRef(e, t, r, n) { return he.FromValuesToRef(e._x, e._y, e._z, 0, t._x, t._y, t._z, 0, r._x, r._y, r._z, 0, 0, 0, 0, 1, n), n; } /** * Creates a rotation matrix from a quaternion and stores it in a target matrix * @param quat defines the quaternion to use * @param result defines the target matrix * @returns result input */ static FromQuaternionToRef(e, t) { const r = e._x * e._x, n = e._y * e._y, i = e._z * e._z, s = e._x * e._y, a = e._z * e._w, f = e._z * e._x, o = e._y * e._w, d = e._y * e._z, v = e._x * e._w; return t._m[0] = 1 - 2 * (n + i), t._m[1] = 2 * (s + a), t._m[2] = 2 * (f - o), t._m[3] = 0, t._m[4] = 2 * (s - a), t._m[5] = 1 - 2 * (i + r), t._m[6] = 2 * (d + v), t._m[7] = 0, t._m[8] = 2 * (f + o), t._m[9] = 2 * (d - v), t._m[10] = 1 - 2 * (n + r), t._m[11] = 0, t._m[12] = 0, t._m[13] = 0, t._m[14] = 0, t._m[15] = 1, t.markAsUpdated(), t; } } he._UpdateFlagSeed = 0; he._IdentityReadOnly = he.Identity(); class Kr { } Kr.Vector3 = Nf.BuildTuple(11, S.Zero); Kr.Matrix = Nf.BuildTuple(2, he.Identity); Kr.Quaternion = Nf.BuildTuple(3, Ze.Zero); class ue { } ue.Vector2 = Nf.BuildTuple(3, at.Zero); ue.Vector3 = Nf.BuildTuple(13, S.Zero); ue.Vector4 = Nf.BuildTuple(3, Ir.Zero); ue.Quaternion = Nf.BuildTuple(2, Ze.Zero); ue.Matrix = Nf.BuildTuple(8, he.Identity); Ue("BABYLON.Vector2", at); Ue("BABYLON.Vector3", S); Ue("BABYLON.Vector4", Ir); Ue("BABYLON.Matrix", he); const kw = he.FromValues(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0.5, 0, 0, 0, 0.5, 1); function zW(A) { return Math.pow(A, xI); } function GW(A) { return A <= 0.04045 ? 0.0773993808 * A : Math.pow(0.947867299 * (A + 0.055), 2.4); } function ZW(A) { return Math.pow(A, JW); } function _W(A) { return A <= 31308e-7 ? 12.92 * A : 1.055 * Math.pow(A, 0.41666) - 0.055; } class Ne { /** * Creates a new Color3 object from red, green, blue values, all between 0 and 1 * @param r defines the red component (between 0 and 1, default is 0) * @param g defines the green component (between 0 and 1, default is 0) * @param b defines the blue component (between 0 and 1, default is 0) */ constructor(e = 0, t = 0, r = 0) { this.r = e, this.g = t, this.b = r; } /** * Creates a string with the Color3 current values * @returns the string representation of the Color3 object */ toString() { return "{R: " + this.r + " G:" + this.g + " B:" + this.b + "}"; } /** * Returns the string "Color3" * @returns "Color3" */ getClassName() { return "Color3"; } /** * Compute the Color3 hash code * @returns an unique number that can be used to hash Color3 objects */ getHashCode() { let e = this.r * 255 | 0; return e = e * 397 ^ (this.g * 255 | 0), e = e * 397 ^ (this.b * 255 | 0), e; } // Operators /** * Stores in the given array from the given starting index the red, green, blue values as successive elements * @param array defines the array where to store the r,g,b components * @param index defines an optional index in the target array to define where to start storing values * @returns the current Color3 object */ toArray(e, t = 0) { return e[t] = this.r, e[t + 1] = this.g, e[t + 2] = this.b, this; } /** * Update the current color with values stored in an array from the starting index of the given array * @param array defines the source array * @param offset defines an offset in the source array * @returns the current Color3 object */ fromArray(e, t = 0) { return Ne.FromArrayToRef(e, t, this), this; } /** * Returns a new Color4 object from the current Color3 and the given alpha * @param alpha defines the alpha component on the new Color4 object (default is 1) * @returns a new Color4 object */ toColor4(e = 1) { return new xt(this.r, this.g, this.b, e); } /** * Returns a new array populated with 3 numeric elements : red, green and blue values * @returns the new array */ asArray() { return [this.r, this.g, this.b]; } /** * Returns the luminance value * @returns a float value */ toLuminance() { return this.r * 0.3 + this.g * 0.59 + this.b * 0.11; } /** * Multiply each Color3 rgb values by the given Color3 rgb values in a new Color3 object * @param otherColor defines the second operand * @returns the new Color3 object */ multiply(e) { return new Ne(this.r * e.r, this.g * e.g, this.b * e.b); } /** * Multiply the rgb values of the Color3 and the given Color3 and stores the result in the object "result" * @param otherColor defines the second operand * @param result defines the Color3 object where to store the result * @returns the current Color3 */ multiplyToRef(e, t) { return t.r = this.r * e.r, t.g = this.g * e.g, t.b = this.b * e.b, this; } /** * Determines equality between Color3 objects * @param otherColor defines the second operand * @returns true if the rgb values are equal to the given ones */ equals(e) { return e && this.r === e.r && this.g === e.g && this.b === e.b; } /** * Determines equality between the current Color3 object and a set of r,b,g values * @param r defines the red component to check * @param g defines the green component to check * @param b defines the blue component to check * @returns true if the rgb values are equal to the given ones */ equalsFloats(e, t, r) { return this.r === e && this.g === t && this.b === r; } /** * Creates a new Color3 with the current Color3 values multiplied by scale * @param scale defines the scaling factor to apply * @returns a new Color3 object */ scale(e) { return new Ne(this.r * e, this.g * e, this.b * e); } /** * Multiplies the Color3 values by the float "scale" * @param scale defines the scaling factor to apply * @returns the current updated Color3 */ scaleInPlace(e) { return this.r *= e, this.g *= e, this.b *= e, this; } /** * Multiplies the rgb values by scale and stores the result into "result" * @param scale defines the scaling factor * @param result defines the Color3 object where to store the result * @returns the unmodified current Color3 */ scaleToRef(e, t) { return t.r = this.r * e, t.g = this.g * e, t.b = this.b * e, this; } /** * Scale the current Color3 values by a factor and add the result to a given Color3 * @param scale defines the scale factor * @param result defines color to store the result into * @returns the unmodified current Color3 */ scaleAndAddToRef(e, t) { return t.r += this.r * e, t.g += this.g * e, t.b += this.b * e, this; } /** * Clamps the rgb values by the min and max values and stores the result into "result" * @param min defines minimum clamping value (default is 0) * @param max defines maximum clamping value (default is 1) * @param result defines color to store the result into * @returns the original Color3 */ clampToRef(e = 0, t = 1, r) { return r.r = Xt.Clamp(this.r, e, t), r.g = Xt.Clamp(this.g, e, t), r.b = Xt.Clamp(this.b, e, t), this; } /** * Creates a new Color3 set with the added values of the current Color3 and of the given one * @param otherColor defines the second operand * @returns the new Color3 */ add(e) { return new Ne(this.r + e.r, this.g + e.g, this.b + e.b); } /** * Stores the result of the addition of the current Color3 and given one rgb values into "result" * @param otherColor defines the second operand * @param result defines Color3 object to store the result into * @returns the unmodified current Color3 */ addToRef(e, t) { return t.r = this.r + e.r, t.g = this.g + e.g, t.b = this.b + e.b, this; } /** * Returns a new Color3 set with the subtracted values of the given one from the current Color3 * @param otherColor defines the second operand * @returns the new Color3 */ subtract(e) { return new Ne(this.r - e.r, this.g - e.g, this.b - e.b); } /** * Stores the result of the subtraction of given one from the current Color3 rgb values into "result" * @param otherColor defines the second operand * @param result defines Color3 object to store the result into * @returns the unmodified current Color3 */ subtractToRef(e, t) { return t.r = this.r - e.r, t.g = this.g - e.g, t.b = this.b - e.b, this; } /** * Copy the current object * @returns a new Color3 copied the current one */ clone() { return new Ne(this.r, this.g, this.b); } /** * Copies the rgb values from the source in the current Color3 * @param source defines the source Color3 object * @returns the updated Color3 object */ copyFrom(e) { return this.r = e.r, this.g = e.g, this.b = e.b, this; } /** * Updates the Color3 rgb values from the given floats * @param r defines the red component to read from * @param g defines the green component to read from * @param b defines the blue component to read from * @returns the current Color3 object */ copyFromFloats(e, t, r) { return this.r = e, this.g = t, this.b = r, this; } /** * Updates the Color3 rgb values from the given floats * @param r defines the red component to read from * @param g defines the green component to read from * @param b defines the blue component to read from * @returns the current Color3 object */ set(e, t, r) { return this.copyFromFloats(e, t, r); } /** * Compute the Color3 hexadecimal code as a string * @returns a string containing the hexadecimal representation of the Color3 object */ toHexString() { const e = Math.round(this.r * 255), t = Math.round(this.g * 255), r = Math.round(this.b * 255); return "#" + Xt.ToHex(e) + Xt.ToHex(t) + Xt.ToHex(r); } /** * Converts current color in rgb space to HSV values * @returns a new color3 representing the HSV values */ toHSV() { const e = new Ne(); return this.toHSVToRef(e), e; } /** * Converts current color in rgb space to HSV values * @param result defines the Color3 where to store the HSV values */ toHSVToRef(e) { const t = this.r, r = this.g, n = this.b, i = Math.max(t, r, n), s = Math.min(t, r, n); let a = 0, f = 0; const o = i, d = i - s; i !== 0 && (f = d / i), i != s && (i == t ? (a = (r - n) / d, r < n && (a += 6)) : i == r ? a = (n - t) / d + 2 : i == n && (a = (t - r) / d + 4), a *= 60), e.r = a, e.g = f, e.b = o; } /** * Computes a new Color3 converted from the current one to linear space * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false) * @returns a new Color3 object */ toLinearSpace(e = !1) { const t = new Ne(); return this.toLinearSpaceToRef(t, e), t; } /** * Converts the Color3 values to linear space and stores the result in "convertedColor" * @param convertedColor defines the Color3 object where to store the linear space version * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false) * @returns the unmodified Color3 */ toLinearSpaceToRef(e, t = !1) { return t ? (e.r = GW(this.r), e.g = GW(this.g), e.b = GW(this.b)) : (e.r = zW(this.r), e.g = zW(this.g), e.b = zW(this.b)), this; } /** * Computes a new Color3 converted from the current one to gamma space * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false) * @returns a new Color3 object */ toGammaSpace(e = !1) { const t = new Ne(); return this.toGammaSpaceToRef(t, e), t; } /** * Converts the Color3 values to gamma space and stores the result in "convertedColor" * @param convertedColor defines the Color3 object where to store the gamma space version * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false) * @returns the unmodified Color3 */ toGammaSpaceToRef(e, t = !1) { return t ? (e.r = _W(this.r), e.g = _W(this.g), e.b = _W(this.b)) : (e.r = ZW(this.r), e.g = ZW(this.g), e.b = ZW(this.b)), this; } /** * Converts Hue, saturation and value to a Color3 (RGB) * @param hue defines the hue (value between 0 and 360) * @param saturation defines the saturation (value between 0 and 1) * @param value defines the value (value between 0 and 1) * @param result defines the Color3 where to store the RGB values */ static HSVtoRGBToRef(e, t, r, n) { const i = r * t, s = e / 60, a = i * (1 - Math.abs(s % 2 - 1)); let f = 0, o = 0, d = 0; s >= 0 && s <= 1 ? (f = i, o = a) : s >= 1 && s <= 2 ? (f = a, o = i) : s >= 2 && s <= 3 ? (o = i, d = a) : s >= 3 && s <= 4 ? (o = a, d = i) : s >= 4 && s <= 5 ? (f = a, d = i) : s >= 5 && s <= 6 && (f = i, d = a); const v = r - i; n.set(f + v, o + v, d + v); } /** * Converts Hue, saturation and value to a new Color3 (RGB) * @param hue defines the hue (value between 0 and 360) * @param saturation defines the saturation (value between 0 and 1) * @param value defines the value (value between 0 and 1) * @returns a new Color3 object */ static FromHSV(e, t, r) { const n = new Ne(0, 0, 0); return Ne.HSVtoRGBToRef(e, t, r, n), n; } /** * Creates a new Color3 from the string containing valid hexadecimal values * @param hex defines a string containing valid hexadecimal values * @returns a new Color3 object */ static FromHexString(e) { if (e.substring(0, 1) !== "#" || e.length !== 7) return new Ne(0, 0, 0); const t = parseInt(e.substring(1, 3), 16), r = parseInt(e.substring(3, 5), 16), n = parseInt(e.substring(5, 7), 16); return Ne.FromInts(t, r, n); } /** * Creates a new Color3 from the starting index of the given array * @param array defines the source array * @param offset defines an offset in the source array * @returns a new Color3 object */ static FromArray(e, t = 0) { return new Ne(e[t], e[t + 1], e[t + 2]); } /** * Creates a new Color3 from the starting index element of the given array * @param array defines the source array to read from * @param offset defines the offset in the source array * @param result defines the target Color3 object */ static FromArrayToRef(e, t = 0, r) { r.r = e[t], r.g = e[t + 1], r.b = e[t + 2]; } /** * Creates a new Color3 from integer values (< 256) * @param r defines the red component to read from (value between 0 and 255) * @param g defines the green component to read from (value between 0 and 255) * @param b defines the blue component to read from (value between 0 and 255) * @returns a new Color3 object */ static FromInts(e, t, r) { return new Ne(e / 255, t / 255, r / 255); } /** * Creates a new Color3 with values linearly interpolated of "amount" between the start Color3 and the end Color3 * @param start defines the start Color3 value * @param end defines the end Color3 value * @param amount defines the gradient value between start and end * @returns a new Color3 object */ static Lerp(e, t, r) { const n = new Ne(0, 0, 0); return Ne.LerpToRef(e, t, r, n), n; } /** * Creates a new Color3 with values linearly interpolated of "amount" between the start Color3 and the end Color3 * @param left defines the start value * @param right defines the end value * @param amount defines the gradient factor * @param result defines the Color3 object where to store the result */ static LerpToRef(e, t, r, n) { n.r = e.r + (t.r - e.r) * r, n.g = e.g + (t.g - e.g) * r, n.b = e.b + (t.b - e.b) * r; } /** * Returns a new Color3 located for "amount" (float) on the Hermite interpolation spline defined by the vectors "value1", "tangent1", "value2", "tangent2" * @param value1 defines the first control point * @param tangent1 defines the first tangent Color3 * @param value2 defines the second control point * @param tangent2 defines the second tangent Color3 * @param amount defines the amount on the interpolation spline (between 0 and 1) * @returns the new Color3 */ static Hermite(e, t, r, n, i) { const s = i * i, a = i * s, f = 2 * a - 3 * s + 1, o = -2 * a + 3 * s, d = a - 2 * s + i, v = a - s, u = e.r * f + r.r * o + t.r * d + n.r * v, l = e.g * f + r.g * o + t.g * d + n.g * v, P = e.b * f + r.b * o + t.b * d + n.b * v; return new Ne(u, l, P); } /** * Returns a new Color3 which is the 1st derivative of the Hermite spline defined by the colors "value1", "value2", "tangent1", "tangent2". * @param value1 defines the first control point * @param tangent1 defines the first tangent * @param value2 defines the second control point * @param tangent2 defines the second tangent * @param time define where the derivative must be done * @returns 1st derivative */ static Hermite1stDerivative(e, t, r, n, i) { const s = Ne.Black(); return this.Hermite1stDerivativeToRef(e, t, r, n, i, s), s; } /** * Returns a new Color3 which is the 1st derivative of the Hermite spline defined by the colors "value1", "value2", "tangent1", "tangent2". * @param value1 defines the first control point * @param tangent1 defines the first tangent * @param value2 defines the second control point * @param tangent2 defines the second tangent * @param time define where the derivative must be done * @param result define where to store the derivative */ static Hermite1stDerivativeToRef(e, t, r, n, i, s) { const a = i * i; s.r = (a - i) * 6 * e.r + (3 * a - 4 * i + 1) * t.r + (-a + i) * 6 * r.r + (3 * a - 2 * i) * n.r, s.g = (a - i) * 6 * e.g + (3 * a - 4 * i + 1) * t.g + (-a + i) * 6 * r.g + (3 * a - 2 * i) * n.g, s.b = (a - i) * 6 * e.b + (3 * a - 4 * i + 1) * t.b + (-a + i) * 6 * r.b + (3 * a - 2 * i) * n.b; } /** * Returns a Color3 value containing a red color * @returns a new Color3 object */ static Red() { return new Ne(1, 0, 0); } /** * Returns a Color3 value containing a green color * @returns a new Color3 object */ static Green() { return new Ne(0, 1, 0); } /** * Returns a Color3 value containing a blue color * @returns a new Color3 object */ static Blue() { return new Ne(0, 0, 1); } /** * Returns a Color3 value containing a black color * @returns a new Color3 object */ static Black() { return new Ne(0, 0, 0); } /** * Gets a Color3 value containing a black color that must not be updated */ static get BlackReadOnly() { return Ne._BlackReadOnly; } /** * Returns a Color3 value containing a white color * @returns a new Color3 object */ static White() { return new Ne(1, 1, 1); } /** * Returns a Color3 value containing a purple color * @returns a new Color3 object */ static Purple() { return new Ne(0.5, 0, 0.5); } /** * Returns a Color3 value containing a magenta color * @returns a new Color3 object */ static Magenta() { return new Ne(1, 0, 1); } /** * Returns a Color3 value containing a yellow color * @returns a new Color3 object */ static Yellow() { return new Ne(1, 1, 0); } /** * Returns a Color3 value containing a gray color * @returns a new Color3 object */ static Gray() { return new Ne(0.5, 0.5, 0.5); } /** * Returns a Color3 value containing a teal color * @returns a new Color3 object */ static Teal() { return new Ne(0, 1, 1); } /** * Returns a Color3 value containing a random color * @returns a new Color3 object */ static Random() { return new Ne(Math.random(), Math.random(), Math.random()); } } Ne._BlackReadOnly = Ne.Black(); class xt { /** * Creates a new Color4 object from red, green, blue values, all between 0 and 1 * @param r defines the red component (between 0 and 1, default is 0) * @param g defines the green component (between 0 and 1, default is 0) * @param b defines the blue component (between 0 and 1, default is 0) * @param a defines the alpha component (between 0 and 1, default is 1) */ constructor(e = 0, t = 0, r = 0, n = 1) { this.r = e, this.g = t, this.b = r, this.a = n; } // Operators /** * Adds in place the given Color4 values to the current Color4 object * @param right defines the second operand * @returns the current updated Color4 object */ addInPlace(e) { return this.r += e.r, this.g += e.g, this.b += e.b, this.a += e.a, this; } /** * Creates a new array populated with 4 numeric elements : red, green, blue, alpha values * @returns the new array */ asArray() { return [this.r, this.g, this.b, this.a]; } /** * Stores from the starting index in the given array the Color4 successive values * @param array defines the array where to store the r,g,b components * @param index defines an optional index in the target array to define where to start storing values * @returns the current Color4 object */ toArray(e, t = 0) { return e[t] = this.r, e[t + 1] = this.g, e[t + 2] = this.b, e[t + 3] = this.a, this; } /** * Update the current color with values stored in an array from the starting index of the given array * @param array defines the source array * @param offset defines an offset in the source array * @returns the current Color4 object */ fromArray(e, t = 0) { return xt.FromArrayToRef(e, t, this), this; } /** * Determines equality between Color4 objects * @param otherColor defines the second operand * @returns true if the rgba values are equal to the given ones */ equals(e) { return e && this.r === e.r && this.g === e.g && this.b === e.b && this.a === e.a; } /** * Creates a new Color4 set with the added values of the current Color4 and of the given one * @param right defines the second operand * @returns a new Color4 object */ add(e) { return new xt(this.r + e.r, this.g + e.g, this.b + e.b, this.a + e.a); } /** * Creates a new Color4 set with the subtracted values of the given one from the current Color4 * @param right defines the second operand * @returns a new Color4 object */ subtract(e) { return new xt(this.r - e.r, this.g - e.g, this.b - e.b, this.a - e.a); } /** * Subtracts the given ones from the current Color4 values and stores the results in "result" * @param right defines the second operand * @param result defines the Color4 object where to store the result * @returns the current Color4 object */ subtractToRef(e, t) { return t.r = this.r - e.r, t.g = this.g - e.g, t.b = this.b - e.b, t.a = this.a - e.a, this; } /** * Creates a new Color4 with the current Color4 values multiplied by scale * @param scale defines the scaling factor to apply * @returns a new Color4 object */ scale(e) { return new xt(this.r * e, this.g * e, this.b * e, this.a * e); } /** * Multiplies the Color4 values by the float "scale" * @param scale defines the scaling factor to apply * @returns the current updated Color4 */ scaleInPlace(e) { return this.r *= e, this.g *= e, this.b *= e, this.a *= e, this; } /** * Multiplies the current Color4 values by scale and stores the result in "result" * @param scale defines the scaling factor to apply * @param result defines the Color4 object where to store the result * @returns the current unmodified Color4 */ scaleToRef(e, t) { return t.r = this.r * e, t.g = this.g * e, t.b = this.b * e, t.a = this.a * e, this; } /** * Scale the current Color4 values by a factor and add the result to a given Color4 * @param scale defines the scale factor * @param result defines the Color4 object where to store the result * @returns the unmodified current Color4 */ scaleAndAddToRef(e, t) { return t.r += this.r * e, t.g += this.g * e, t.b += this.b * e, t.a += this.a * e, this; } /** * Clamps the rgb values by the min and max values and stores the result into "result" * @param min defines minimum clamping value (default is 0) * @param max defines maximum clamping value (default is 1) * @param result defines color to store the result into. * @returns the current Color4 */ clampToRef(e = 0, t = 1, r) { return r.r = Xt.Clamp(this.r, e, t), r.g = Xt.Clamp(this.g, e, t), r.b = Xt.Clamp(this.b, e, t), r.a = Xt.Clamp(this.a, e, t), this; } /** * Multiply an Color4 value by another and return a new Color4 object * @param color defines the Color4 value to multiply by * @returns a new Color4 object */ multiply(e) { return new xt(this.r * e.r, this.g * e.g, this.b * e.b, this.a * e.a); } /** * Multiply a Color4 value by another and push the result in a reference value * @param color defines the Color4 value to multiply by * @param result defines the Color4 to fill the result in * @returns the result Color4 */ multiplyToRef(e, t) { return t.r = this.r * e.r, t.g = this.g * e.g, t.b = this.b * e.b, t.a = this.a * e.a, t; } /** * Creates a string with the Color4 current values * @returns the string representation of the Color4 object */ toString() { return "{R: " + this.r + " G:" + this.g + " B:" + this.b + " A:" + this.a + "}"; } /** * Returns the string "Color4" * @returns "Color4" */ getClassName() { return "Color4"; } /** * Compute the Color4 hash code * @returns an unique number that can be used to hash Color4 objects */ getHashCode() { let e = this.r * 255 | 0; return e = e * 397 ^ (this.g * 255 | 0), e = e * 397 ^ (this.b * 255 | 0), e = e * 397 ^ (this.a * 255 | 0), e; } /** * Creates a new Color4 copied from the current one * @returns a new Color4 object */ clone() { return new xt(this.r, this.g, this.b, this.a); } /** * Copies the given Color4 values into the current one * @param source defines the source Color4 object * @returns the current updated Color4 object */ copyFrom(e) { return this.r = e.r, this.g = e.g, this.b = e.b, this.a = e.a, this; } /** * Copies the given float values into the current one * @param r defines the red component to read from * @param g defines the green component to read from * @param b defines the blue component to read from * @param a defines the alpha component to read from * @returns the current updated Color4 object */ copyFromFloats(e, t, r, n) { return this.r = e, this.g = t, this.b = r, this.a = n, this; } /** * Copies the given float values into the current one * @param r defines the red component to read from * @param g defines the green component to read from * @param b defines the blue component to read from * @param a defines the alpha component to read from * @returns the current updated Color4 object */ set(e, t, r, n) { return this.copyFromFloats(e, t, r, n); } /** * Compute the Color4 hexadecimal code as a string * @param returnAsColor3 defines if the string should only contains RGB values (off by default) * @returns a string containing the hexadecimal representation of the Color4 object */ toHexString(e = !1) { const t = Math.round(this.r * 255), r = Math.round(this.g * 255), n = Math.round(this.b * 255); if (e) return "#" + Xt.ToHex(t) + Xt.ToHex(r) + Xt.ToHex(n); const i = Math.round(this.a * 255); return "#" + Xt.ToHex(t) + Xt.ToHex(r) + Xt.ToHex(n) + Xt.ToHex(i); } /** * Computes a new Color4 converted from the current one to linear space * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false) * @returns a new Color4 object */ toLinearSpace(e = !1) { const t = new xt(); return this.toLinearSpaceToRef(t, e), t; } /** * Converts the Color4 values to linear space and stores the result in "convertedColor" * @param convertedColor defines the Color4 object where to store the linear space version * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false) * @returns the unmodified Color4 */ toLinearSpaceToRef(e, t = !1) { return t ? (e.r = GW(this.r), e.g = GW(this.g), e.b = GW(this.b)) : (e.r = zW(this.r), e.g = zW(this.g), e.b = zW(this.b)), e.a = this.a, this; } /** * Computes a new Color4 converted from the current one to gamma space * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false) * @returns a new Color4 object */ toGammaSpace(e = !1) { const t = new xt(); return this.toGammaSpaceToRef(t, e), t; } /** * Converts the Color4 values to gamma space and stores the result in "convertedColor" * @param convertedColor defines the Color4 object where to store the gamma space version * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false) * @returns the unmodified Color4 */ toGammaSpaceToRef(e, t = !1) { return t ? (e.r = _W(this.r), e.g = _W(this.g), e.b = _W(this.b)) : (e.r = ZW(this.r), e.g = ZW(this.g), e.b = ZW(this.b)), e.a = this.a, this; } // Statics /** * Creates a new Color4 from the string containing valid hexadecimal values. * * A valid hex string is either in the format #RRGGBB or #RRGGBBAA. * * When a hex string without alpha is passed, the resulting Color4 has * its alpha value set to 1.0. * * An invalid string results in a Color with all its channels set to 0.0, * i.e. "transparent black". * * @param hex defines a string containing valid hexadecimal values * @returns a new Color4 object */ static FromHexString(e) { if (e.substring(0, 1) !== "#" || e.length !== 9 && e.length !== 7) return new xt(0, 0, 0, 0); const t = parseInt(e.substring(1, 3), 16), r = parseInt(e.substring(3, 5), 16), n = parseInt(e.substring(5, 7), 16), i = e.length === 9 ? parseInt(e.substring(7, 9), 16) : 255; return xt.FromInts(t, r, n, i); } /** * Creates a new Color4 object set with the linearly interpolated values of "amount" between the left Color4 object and the right Color4 object * @param left defines the start value * @param right defines the end value * @param amount defines the gradient factor * @returns a new Color4 object */ static Lerp(e, t, r) { const n = new xt(0, 0, 0, 0); return xt.LerpToRef(e, t, r, n), n; } /** * Set the given "result" with the linearly interpolated values of "amount" between the left Color4 object and the right Color4 object * @param left defines the start value * @param right defines the end value * @param amount defines the gradient factor * @param result defines the Color4 object where to store data */ static LerpToRef(e, t, r, n) { n.r = e.r + (t.r - e.r) * r, n.g = e.g + (t.g - e.g) * r, n.b = e.b + (t.b - e.b) * r, n.a = e.a + (t.a - e.a) * r; } /** * Interpolate between two Color4 using Hermite interpolation * @param value1 defines first Color4 * @param tangent1 defines the incoming tangent * @param value2 defines second Color4 * @param tangent2 defines the outgoing tangent * @param amount defines the target Color4 * @returns the new interpolated Color4 */ static Hermite(e, t, r, n, i) { const s = i * i, a = i * s, f = 2 * a - 3 * s + 1, o = -2 * a + 3 * s, d = a - 2 * s + i, v = a - s, u = e.r * f + r.r * o + t.r * d + n.r * v, l = e.g * f + r.g * o + t.g * d + n.g * v, P = e.b * f + r.b * o + t.b * d + n.b * v, p = e.a * f + r.a * o + t.a * d + n.a * v; return new xt(u, l, P, p); } /** * Returns a new Color4 which is the 1st derivative of the Hermite spline defined by the colors "value1", "value2", "tangent1", "tangent2". * @param value1 defines the first control point * @param tangent1 defines the first tangent * @param value2 defines the second control point * @param tangent2 defines the second tangent * @param time define where the derivative must be done * @returns 1st derivative */ static Hermite1stDerivative(e, t, r, n, i) { const s = new xt(); return this.Hermite1stDerivativeToRef(e, t, r, n, i, s), s; } /** * Update a Color4 with the 1st derivative of the Hermite spline defined by the colors "value1", "value2", "tangent1", "tangent2". * @param value1 defines the first control point * @param tangent1 defines the first tangent * @param value2 defines the second control point * @param tangent2 defines the second tangent * @param time define where the derivative must be done * @param result define where to store the derivative */ static Hermite1stDerivativeToRef(e, t, r, n, i, s) { const a = i * i; s.r = (a - i) * 6 * e.r + (3 * a - 4 * i + 1) * t.r + (-a + i) * 6 * r.r + (3 * a - 2 * i) * n.r, s.g = (a - i) * 6 * e.g + (3 * a - 4 * i + 1) * t.g + (-a + i) * 6 * r.g + (3 * a - 2 * i) * n.g, s.b = (a - i) * 6 * e.b + (3 * a - 4 * i + 1) * t.b + (-a + i) * 6 * r.b + (3 * a - 2 * i) * n.b, s.a = (a - i) * 6 * e.a + (3 * a - 4 * i + 1) * t.a + (-a + i) * 6 * r.a + (3 * a - 2 * i) * n.a; } /** * Creates a new Color4 from a Color3 and an alpha value * @param color3 defines the source Color3 to read from * @param alpha defines the alpha component (1.0 by default) * @returns a new Color4 object */ static FromColor3(e, t = 1) { return new xt(e.r, e.g, e.b, t); } /** * Creates a new Color4 from the starting index element of the given array * @param array defines the source array to read from * @param offset defines the offset in the source array * @returns a new Color4 object */ static FromArray(e, t = 0) { return new xt(e[t], e[t + 1], e[t + 2], e[t + 3]); } /** * Creates a new Color4 from the starting index element of the given array * @param array defines the source array to read from * @param offset defines the offset in the source array * @param result defines the target Color4 object */ static FromArrayToRef(e, t = 0, r) { r.r = e[t], r.g = e[t + 1], r.b = e[t + 2], r.a = e[t + 3]; } /** * Creates a new Color3 from integer values (< 256) * @param r defines the red component to read from (value between 0 and 255) * @param g defines the green component to read from (value between 0 and 255) * @param b defines the blue component to read from (value between 0 and 255) * @param a defines the alpha component to read from (value between 0 and 255) * @returns a new Color3 object */ static FromInts(e, t, r, n) { return new xt(e / 255, t / 255, r / 255, n / 255); } /** * Check the content of a given array and convert it to an array containing RGBA data * If the original array was already containing count * 4 values then it is returned directly * @param colors defines the array to check * @param count defines the number of RGBA data to expect * @returns an array containing count * 4 values (RGBA) */ static CheckColors4(e, t) { if (e.length === t * 3) { const r = []; for (let n = 0; n < e.length; n += 3) { const i = n / 3 * 4; r[i] = e[n], r[i + 1] = e[n + 1], r[i + 2] = e[n + 2], r[i + 3] = 1; } return r; } return e; } } class Hs { } Hs.Color3 = Nf.BuildArray(3, Ne.Black); Hs.Color4 = Nf.BuildArray(3, () => new xt(0, 0, 0, 0)); Ue("BABYLON.Color3", Ne); Ue("BABYLON.Color4", xt); class wa { /** * Creates a new Action * @param triggerOptions the trigger, with or without parameters, for the action * @param condition an optional determinant of action */ constructor(e, t) { this.triggerOptions = e, this.onBeforeExecuteObservable = new Oe(), e.parameter ? (this.trigger = e.trigger, this._triggerParameter = e.parameter) : e.trigger ? this.trigger = e.trigger : this.trigger = e, this._nextActiveAction = this, this._condition = t; } /** * Internal only * @internal */ _prepare() { } /** * Gets the trigger parameter * @returns the trigger parameter */ getTriggerParameter() { return this._triggerParameter; } /** * Sets the trigger parameter * @param value defines the new trigger parameter */ setTriggerParameter(e) { this._triggerParameter = e; } /** * Internal only - Returns if the current condition allows to run the action * @internal */ _evaluateConditionForCurrentFrame() { const e = this._condition; if (!e) return !0; const t = this._actionManager.getScene().getRenderId(); return e._evaluationId !== t && (e._evaluationId = t, e._currentResult = e.isValid()), e._currentResult; } /** * Internal only - executes current action event * @internal */ _executeCurrent(e) { this._evaluateConditionForCurrentFrame() && (this.onBeforeExecuteObservable.notifyObservers(this), this._nextActiveAction.execute(e), this.skipToNextActiveAction()); } /** * Execute placeholder for child classes * @param evt optional action event */ // eslint-disable-next-line @typescript-eslint/no-unused-vars execute(e) { } /** * Skips to next active action */ skipToNextActiveAction() { this._nextActiveAction._child ? (this._nextActiveAction._child._actionManager || (this._nextActiveAction._child._actionManager = this._actionManager), this._nextActiveAction = this._nextActiveAction._child) : this._nextActiveAction = this; } /** * Adds action to chain of actions, may be a DoNothingAction * @param action defines the next action to execute * @returns The action passed in * @see https://www.babylonjs-playground.com/#1T30HR#0 */ then(e) { return this._child = e, e._actionManager = this._actionManager, e._prepare(), e; } /** * Internal only * @internal */ _getProperty(e) { return this._actionManager._getProperty(e); } /** * @internal */ _getEffectiveTarget(e, t) { return this._actionManager._getEffectiveTarget(e, t); } /** * Serialize placeholder for child classes * @param parent of child * @returns the serialized object */ // eslint-disable-next-line @typescript-eslint/no-unused-vars serialize(e) { } /** * Internal only called by serialize * @internal */ _serialize(e, t) { const r = { type: 1, children: [], name: e.name, properties: e.properties || [] }; if (this._child && this._child.serialize(r), this._condition) { const n = this._condition.serialize(); return n.children.push(r), t && t.children.push(n), n; } return t && t.children.push(r), r; } } wa._SerializeValueAsString = (A) => typeof A == "number" ? A.toString() : typeof A == "boolean" ? A ? "true" : "false" : A instanceof at ? A.x + ", " + A.y : A instanceof S ? A.x + ", " + A.y + ", " + A.z : A instanceof Ne ? A.r + ", " + A.g + ", " + A.b : A instanceof xt ? A.r + ", " + A.g + ", " + A.b + ", " + A.a : A; wa._GetTargetProperty = (A) => ({ name: "target", targetType: A._isMesh ? "MeshProperties" : A._isLight ? "LightProperties" : A._isCamera ? "CameraProperties" : A._isMaterial ? "MaterialProperties" : "SceneProperties", value: A._isScene ? "Scene" : A.name }); Ue("BABYLON.Action", wa); class mo { /** * Creates a new ActionEvent * @param source The mesh or sprite that triggered the action * @param pointerX The X mouse cursor position at the time of the event * @param pointerY The Y mouse cursor position at the time of the event * @param meshUnderPointer The mesh that is currently pointed at (can be null) * @param sourceEvent the original (browser) event that triggered the ActionEvent * @param additionalData additional data for the event */ constructor(e, t, r, n, i, s) { this.source = e, this.pointerX = t, this.pointerY = r, this.meshUnderPointer = n, this.sourceEvent = i, this.additionalData = s; } /** * Helper function to auto-create an ActionEvent from a source mesh. * @param source The source mesh that triggered the event * @param evt The original (browser) event * @param additionalData additional data for the event * @returns the new ActionEvent */ static CreateNew(e, t, r) { const n = e.getScene(); return new mo(e, n.pointerX, n.pointerY, n.meshUnderPointer || e, t, r); } /** * Helper function to auto-create an ActionEvent from a source sprite * @param source The source sprite that triggered the event * @param scene Scene associated with the sprite * @param evt The original (browser) event * @param additionalData additional data for the event * @returns the new ActionEvent */ static CreateNewFromSprite(e, t, r, n) { return new mo(e, t.pointerX, t.pointerY, t.meshUnderPointer, r, n); } /** * Helper function to auto-create an ActionEvent from a scene. If triggered by a mesh use ActionEvent.CreateNew * @param scene the scene where the event occurred * @param evt The original (browser) event * @returns the new ActionEvent */ static CreateNewFromScene(e, t) { return new mo(null, e.pointerX, e.pointerY, e.meshUnderPointer, t); } /** * Helper function to auto-create an ActionEvent from a primitive * @param prim defines the target primitive * @param pointerPos defines the pointer position * @param evt The original (browser) event * @param additionalData additional data for the event * @returns the new ActionEvent */ static CreateNewFromPrimitive(e, t, r, n) { return new mo(e, t.x, t.y, null, r, n); } } class DS { /** * Creates a new Condition * @param actionManager the manager of the action the condition is applied to */ constructor(e) { this._actionManager = e; } /** * Check if the current condition is valid * @returns a boolean */ isValid() { return !0; } /** * @internal */ _getProperty(e) { return this._actionManager._getProperty(e); } /** * @internal */ _getEffectiveTarget(e, t) { return this._actionManager._getEffectiveTarget(e, t); } /** * Serialize placeholder for child classes * @returns the serialized object */ serialize() { } /** * @internal */ _serialize(e) { return { type: 2, children: [], name: e.name, properties: e.properties }; } } class C9 extends DS { /** * returns the number for IsEqual */ static get IsEqual() { return C9._IsEqual; } /** * Returns the number for IsDifferent */ static get IsDifferent() { return C9._IsDifferent; } /** * Returns the number for IsGreater */ static get IsGreater() { return C9._IsGreater; } /** * Returns the number for IsLesser */ static get IsLesser() { return C9._IsLesser; } /** * Creates a new ValueCondition * @param actionManager manager for the action the condition applies to * @param target for the action * @param propertyPath path to specify the property of the target the conditional operator uses * @param value the value compared by the conditional operator against the current value of the property * @param operator the conditional operator, default ValueCondition.IsEqual */ constructor(e, t, r, n, i = C9.IsEqual) { super(e), this.propertyPath = r, this.value = n, this.operator = i, this._target = t, this._effectiveTarget = this._getEffectiveTarget(t, this.propertyPath), this._property = this._getProperty(this.propertyPath); } /** * Compares the given value with the property value for the specified conditional operator * @returns the result of the comparison */ isValid() { switch (this.operator) { case C9.IsGreater: return this._effectiveTarget[this._property] > this.value; case C9.IsLesser: return this._effectiveTarget[this._property] < this.value; case C9.IsEqual: case C9.IsDifferent: { let e; return this.value.equals ? e = this.value.equals(this._effectiveTarget[this._property]) : e = this.value === this._effectiveTarget[this._property], this.operator === C9.IsEqual ? e : !e; } } return !1; } /** * Serialize the ValueCondition into a JSON compatible object * @returns serialization object */ serialize() { return this._serialize({ name: "ValueCondition", properties: [ wa._GetTargetProperty(this._target), { name: "propertyPath", value: this.propertyPath }, { name: "value", value: wa._SerializeValueAsString(this.value) }, { name: "operator", value: C9.GetOperatorName(this.operator) } ] }); } /** * Gets the name of the conditional operator for the ValueCondition * @param operator the conditional operator * @returns the name */ static GetOperatorName(e) { switch (e) { case C9._IsEqual: return "IsEqual"; case C9._IsDifferent: return "IsDifferent"; case C9._IsGreater: return "IsGreater"; case C9._IsLesser: return "IsLesser"; default: return ""; } } } C9._IsEqual = 0; C9._IsDifferent = 1; C9._IsGreater = 2; C9._IsLesser = 3; class H$ extends DS { /** * Creates a new PredicateCondition * @param actionManager manager for the action the condition applies to * @param predicate defines the predicate function used to validate the condition */ constructor(e, t) { super(e), this.predicate = t; } /** * @returns the validity of the predicate condition */ isValid() { return this.predicate(); } } class g$ extends DS { /** * Creates a new StateCondition * @param actionManager manager for the action the condition applies to * @param target of the condition * @param value to compare with target state */ constructor(e, t, r) { super(e), this.value = r, this._target = t; } /** * Gets a boolean indicating if the current condition is met * @returns the validity of the state */ isValid() { return this._target.state === this.value; } /** * Serialize the StateCondition into a JSON compatible object * @returns serialization object */ serialize() { return this._serialize({ name: "StateCondition", properties: [wa._GetTargetProperty(this._target), { name: "value", value: this.value }] }); } } Ue("BABYLON.ValueCondition", C9); Ue("BABYLON.PredicateCondition", H$); Ue("BABYLON.StateCondition", g$); class Se { static _CheckLimit(e, t) { let r = Se._LogLimitOutputs[e]; return r ? r.current++ : (r = { limit: t, current: 1 }, Se._LogLimitOutputs[e] = r), r.current <= r.limit; } static _GenerateLimitMessage(e, t = 1) { var r; const n = Se._LogLimitOutputs[e]; if (!n || !Se.MessageLimitReached) return; const i = this._Levels[t]; n.current === n.limit && Se[i.name](Se.MessageLimitReached.replace(/%LIMIT%/g, "" + n.limit).replace(/%TYPE%/g, (r = i.name) !== null && r !== void 0 ? r : "")); } static _AddLogEntry(e) { Se._LogCache = e + Se._LogCache, Se.OnNewCacheEntry && Se.OnNewCacheEntry(e); } static _FormatMessage(e) { const t = (n) => n < 10 ? "0" + n : "" + n, r = /* @__PURE__ */ new Date(); return "[" + t(r.getHours()) + ":" + t(r.getMinutes()) + ":" + t(r.getSeconds()) + "]: " + e; } // eslint-disable-next-line @typescript-eslint/no-unused-vars static _LogDisabled(e, t) { } static _LogEnabled(e = 1, t, r) { if (r !== void 0 && !Se._CheckLimit(t, r)) return; const n = Se._FormatMessage(t), i = this._Levels[e]; i.logFunc && i.logFunc("BJS - " + n); const s = `
${n}

`; Se._AddLogEntry(s), Se._GenerateLimitMessage(t, e); } /** * Gets current log cache (list of logs) */ static get LogCache() { return Se._LogCache; } /** * Clears the log cache */ static ClearLogCache() { Se._LogCache = "", Se._LogLimitOutputs = {}, Se.errorsCount = 0; } /** * Sets the current log level (MessageLogLevel / WarningLogLevel / ErrorLogLevel) */ static set LogLevels(e) { Se.Log = Se._LogDisabled, Se.Warn = Se._LogDisabled, Se.Error = Se._LogDisabled, [Se.MessageLogLevel, Se.WarningLogLevel, Se.ErrorLogLevel].forEach((t) => { if ((e & t) === t) { const r = this._Levels[t]; Se[r.name] = Se._LogEnabled.bind(Se, t); } }); } } Se.NoneLogLevel = 0; Se.MessageLogLevel = 1; Se.WarningLogLevel = 2; Se.ErrorLogLevel = 4; Se.AllLogLevel = 7; Se.MessageLimitReached = "Too many %TYPE%s (%LIMIT%), no more %TYPE%s will be reported for this message."; Se._LogCache = ""; Se._LogLimitOutputs = {}; Se._Levels = [ {}, { color: "white", logFunc: console.log, name: "Log" }, { color: "orange", logFunc: console.warn, name: "Warn" }, {}, { color: "red", logFunc: console.error, name: "Error" } ]; Se.errorsCount = 0; Se.Log = Se._LogEnabled.bind(Se, Se.MessageLogLevel); Se.Warn = Se._LogEnabled.bind(Se, Se.WarningLogLevel); Se.Error = Se._LogEnabled.bind(Se, Se.ErrorLogLevel); class X$ extends wa { /** * Instantiate the action * @param triggerOptions defines the trigger options * @param target defines the object containing the boolean * @param propertyPath defines the path to the boolean property in the target object * @param condition defines the trigger related conditions */ constructor(e, t, r, n) { super(e, n), this.propertyPath = r, this._target = this._effectiveTarget = t; } /** @internal */ _prepare() { this._effectiveTarget = this._getEffectiveTarget(this._effectiveTarget, this.propertyPath), this._property = this._getProperty(this.propertyPath); } /** * Execute the action toggle the boolean value. */ execute() { this._effectiveTarget[this._property] = !this._effectiveTarget[this._property]; } /** * Serializes the actions and its related information. * @param parent defines the object to serialize in * @returns the serialized object */ serialize(e) { return super._serialize({ name: "SwitchBooleanAction", properties: [wa._GetTargetProperty(this._target), { name: "propertyPath", value: this.propertyPath }] }, e); } } class T$ extends wa { /** * Instantiate the action * @param triggerOptions defines the trigger options * @param target defines the object containing the state property * @param value defines the value to store in the state field * @param condition defines the trigger related conditions */ constructor(e, t, r, n) { super(e, n), this.value = r, this._target = t; } /** * Execute the action and store the value on the target state property. */ execute() { this._target.state = this.value; } /** * Serializes the actions and its related information. * @param parent defines the object to serialize in * @returns the serialized object */ serialize(e) { return super._serialize({ name: "SetStateAction", properties: [wa._GetTargetProperty(this._target), { name: "value", value: this.value }] }, e); } } class q$ extends wa { /** * Instantiate the action * @param triggerOptions defines the trigger options * @param target defines the object containing the property * @param propertyPath defines the path of the property to set in the target * @param value defines the value to set in the property * @param condition defines the trigger related conditions */ constructor(e, t, r, n, i) { super(e, i), this.propertyPath = r, this.value = n, this._target = this._effectiveTarget = t; } /** @internal */ _prepare() { this._effectiveTarget = this._getEffectiveTarget(this._effectiveTarget, this.propertyPath), this._property = this._getProperty(this.propertyPath); } /** * Execute the action and set the targeted property to the desired value. */ execute() { this._effectiveTarget[this._property] = this.value, this._target.markAsDirty && this._target.markAsDirty(this._property); } /** * Serializes the actions and its related information. * @param parent defines the object to serialize in * @returns the serialized object */ serialize(e) { return super._serialize({ name: "SetValueAction", properties: [ wa._GetTargetProperty(this._target), { name: "propertyPath", value: this.propertyPath }, { name: "value", value: wa._SerializeValueAsString(this.value) } ] }, e); } } class b$ extends wa { /** * Instantiate the action * @param triggerOptions defines the trigger options * @param target defines the object containing the property * @param propertyPath defines the path of the property to increment in the target * @param value defines the value value we should increment the property by * @param condition defines the trigger related conditions */ constructor(e, t, r, n, i) { super(e, i), this.propertyPath = r, this.value = n, this._target = this._effectiveTarget = t; } /** @internal */ _prepare() { this._effectiveTarget = this._getEffectiveTarget(this._effectiveTarget, this.propertyPath), this._property = this._getProperty(this.propertyPath), typeof this._effectiveTarget[this._property] != "number" && Se.Warn("Warning: IncrementValueAction can only be used with number values"); } /** * Execute the action and increment the target of the value amount. */ execute() { this._effectiveTarget[this._property] += this.value, this._target.markAsDirty && this._target.markAsDirty(this._property); } /** * Serializes the actions and its related information. * @param parent defines the object to serialize in * @returns the serialized object */ serialize(e) { return super._serialize({ name: "IncrementValueAction", properties: [ wa._GetTargetProperty(this._target), { name: "propertyPath", value: this.propertyPath }, { name: "value", value: wa._SerializeValueAsString(this.value) } ] }, e); } } class x$ extends wa { /** * Instantiate the action * @param triggerOptions defines the trigger options * @param target defines the target animation or animation name * @param from defines from where the animation should start (animation frame) * @param to defines where the animation should stop (animation frame) * @param loop defines if the animation should loop or stop after the first play * @param condition defines the trigger related conditions */ constructor(e, t, r, n, i, s) { super(e, s), this.from = r, this.to = n, this.loop = i, this._target = t; } /** @internal */ _prepare() { } /** * Execute the action and play the animation. */ execute() { this._actionManager.getScene().beginAnimation(this._target, this.from, this.to, this.loop); } /** * Serializes the actions and its related information. * @param parent defines the object to serialize in * @returns the serialized object */ serialize(e) { return super._serialize({ name: "PlayAnimationAction", properties: [ wa._GetTargetProperty(this._target), { name: "from", value: String(this.from) }, { name: "to", value: String(this.to) }, { name: "loop", value: wa._SerializeValueAsString(this.loop) || !1 } ] }, e); } } class D$ extends wa { /** * Instantiate the action * @param triggerOptions defines the trigger options * @param target defines the target animation or animation name * @param condition defines the trigger related conditions */ constructor(e, t, r) { super(e, r), this._target = t; } /** @internal */ _prepare() { } /** * Execute the action and stop the animation. */ execute() { this._actionManager.getScene().stopAnimation(this._target); } /** * Serializes the actions and its related information. * @param parent defines the object to serialize in * @returns the serialized object */ serialize(e) { return super._serialize({ name: "StopAnimationAction", properties: [wa._GetTargetProperty(this._target)] }, e); } } class RN extends wa { /** * Instantiate the action * @param triggerOptions defines the trigger options * @param condition defines the trigger related conditions */ constructor(e = 0, t) { super(e, t); } /** * Execute the action and do nothing. */ execute() { } /** * Serializes the actions and its related information. * @param parent defines the object to serialize in * @returns the serialized object */ serialize(e) { return super._serialize({ name: "DoNothingAction", properties: [] }, e); } } class j$ extends wa { /** * Instantiate the action * @param triggerOptions defines the trigger options * @param children defines the list of aggregated animations to run * @param condition defines the trigger related conditions * @param enableChildrenConditions defines if the children actions conditions should be check before execution */ constructor(e, t, r, n = !0) { super(e, r), this.children = t, this.enableChildrenConditions = n; } /** @internal */ _prepare() { for (let e = 0; e < this.children.length; e++) this.children[e]._actionManager = this._actionManager, this.children[e]._prepare(); } /** * Execute the action and executes all the aggregated actions. * @param evt */ execute(e) { for (const t of this.children) (!this.enableChildrenConditions || t._evaluateConditionForCurrentFrame()) && t.execute(e); } /** * Serializes the actions and its related information. * @param parent defines the object to serialize in * @returns the serialized object */ serialize(e) { const t = super._serialize({ name: "CombineAction", properties: [], combine: [] }, e); for (let r = 0; r < this.children.length; r++) t.combine.push(this.children[r].serialize(null)); return t; } } class w$ extends wa { /** * Instantiate the action * @param triggerOptions defines the trigger options * @param func defines the callback function to run * @param condition defines the trigger related conditions */ constructor(e, t, r) { super(e, r), this.func = t; } /** * Execute the action and run the attached code. * @param evt */ execute(e) { this.func(e); } } class VN extends wa { /** * Instantiate the action * @param triggerOptions defines the trigger options * @param target defines the target containing the parent property * @param parent defines from where the animation should start (animation frame) * @param condition defines the trigger related conditions */ constructor(e, t, r, n) { super(e, n), this._target = t, this._parent = r; } /** @internal */ _prepare() { } /** * Execute the action and set the parent property. */ execute() { if (this._target.parent === this._parent) return; const e = this._parent.getWorldMatrix().clone(); e.invert(), this._target.position = S.TransformCoordinates(this._target.position, e), this._target.parent = this._parent; } /** * Serializes the actions and its related information. * @param parent defines the object to serialize in * @returns the serialized object */ serialize(e) { return super._serialize({ name: "SetParentAction", properties: [wa._GetTargetProperty(this._target), wa._GetTargetProperty(this._parent)] }, e); } } Ue("BABYLON.SetParentAction", VN); Ue("BABYLON.ExecuteCodeAction", w$); Ue("BABYLON.DoNothingAction", RN); Ue("BABYLON.StopAnimationAction", D$); Ue("BABYLON.PlayAnimationAction", x$); Ue("BABYLON.IncrementValueAction", b$); Ue("BABYLON.SetValueAction", q$); Ue("BABYLON.SetStateAction", T$); Ue("BABYLON.SetParentAction", VN); Ue("BABYLON.SwitchBooleanAction", X$); Ue("BABYLON.CombineAction", j$); const iG = (A, e, t) => !A || A.getClassName && A.getClassName() === "Mesh" ? null : A.getClassName && (A.getClassName() === "SubMesh" || A.getClassName() === "PhysicsBody") ? A.clone(e) : A.clone ? A.clone() : Array.isArray(A) ? A.slice() : t && typeof A == "object" ? Object.assign({}, A) : null; function m9e(A) { const e = []; do Object.getOwnPropertyNames(A).forEach(function(t) { e.indexOf(t) === -1 && e.push(t); }); while (A = Object.getPrototypeOf(A)); return e; } class sA { /** * Tries to copy an object by duplicating every property * @param source defines the source object * @param destination defines the target object * @param doNotCopyList defines a list of properties to avoid * @param mustCopyList defines a list of properties to copy (even if they start with _) * @param shallowCopyValues defines wether properties referencing objects (none cloneable) must be shallow copied (false by default) * @remarks shallowCopyValues will not instantite the copied values which makes it only usable for "JSON objects" */ static DeepCopy(e, t, r, n, i = !1) { const s = m9e(e); for (const a of s) { if (a[0] === "_" && (!n || n.indexOf(a) === -1) || a.endsWith("Observable") || r && r.indexOf(a) !== -1) continue; const f = e[a], o = typeof f; if (o !== "function") try { if (o === "object") if (f instanceof Uint8Array) t[a] = Uint8Array.from(f); else if (f instanceof Array) { if (t[a] = [], f.length > 0) if (typeof f[0] == "object") for (let d = 0; d < f.length; d++) { const v = iG(f[d], t, i); t[a].indexOf(v) === -1 && t[a].push(v); } else t[a] = f.slice(0); } else t[a] = iG(f, t, i); else t[a] = f; } catch (d) { Se.Warn(d.message); } } } } class Rs extends Ml { /** * Creates a new action manager * @param scene defines the hosting scene */ constructor(e) { super(), e = e || gr.LastCreatedScene, e && (this._scene = e, e.actionManagers.push(this)); } // Methods /** * Releases all associated resources */ dispose() { const e = this._scene.actionManagers.indexOf(this); for (let r = 0; r < this.actions.length; r++) { const n = this.actions[r]; Rs.Triggers[n.trigger]--, Rs.Triggers[n.trigger] === 0 && delete Rs.Triggers[n.trigger]; } this.actions.length = 0, e > -1 && this._scene.actionManagers.splice(e, 1); const t = this._scene.meshes.filter((r) => r.actionManager === this); for (const r of t) r.actionManager = null; } /** * Gets hosting scene * @returns the hosting scene */ getScene() { return this._scene; } /** * Does this action manager handles actions of any of the given triggers * @param triggers defines the triggers to be tested * @returns a boolean indicating whether one (or more) of the triggers is handled */ hasSpecificTriggers(e) { for (let t = 0; t < this.actions.length; t++) { const r = this.actions[t]; if (e.indexOf(r.trigger) > -1) return !0; } return !1; } /** * Does this action manager handles actions of any of the given triggers. This function takes two arguments for * speed. * @param triggerA defines the trigger to be tested * @param triggerB defines the trigger to be tested * @returns a boolean indicating whether one (or more) of the triggers is handled */ hasSpecificTriggers2(e, t) { for (let r = 0; r < this.actions.length; r++) { const n = this.actions[r]; if (e == n.trigger || t == n.trigger) return !0; } return !1; } /** * Does this action manager handles actions of a given trigger * @param trigger defines the trigger to be tested * @param parameterPredicate defines an optional predicate to filter triggers by parameter * @returns whether the trigger is handled */ hasSpecificTrigger(e, t) { for (let r = 0; r < this.actions.length; r++) { const n = this.actions[r]; if (n.trigger === e) if (t) { if (t(n.getTriggerParameter())) return !0; } else return !0; } return !1; } /** * Does this action manager has pointer triggers */ get hasPointerTriggers() { for (let e = 0; e < this.actions.length; e++) { const t = this.actions[e]; if (t.trigger >= Rs.OnPickTrigger && t.trigger <= Rs.OnPointerOutTrigger) return !0; } return !1; } /** * Does this action manager has pick triggers */ get hasPickTriggers() { for (let e = 0; e < this.actions.length; e++) { const t = this.actions[e]; if (t.trigger >= Rs.OnPickTrigger && t.trigger <= Rs.OnPickUpTrigger) return !0; } return !1; } /** * Registers an action to this action manager * @param action defines the action to be registered * @returns the action amended (prepared) after registration */ registerAction(e) { return e.trigger === Rs.OnEveryFrameTrigger && this.getScene().actionManager !== this ? (Se.Warn("OnEveryFrameTrigger can only be used with scene.actionManager"), null) : (this.actions.push(e), this.getScene()._registeredActions++, Rs.Triggers[e.trigger] ? Rs.Triggers[e.trigger]++ : Rs.Triggers[e.trigger] = 1, e._actionManager = this, e._prepare(), e); } /** * Unregisters an action to this action manager * @param action defines the action to be unregistered * @returns a boolean indicating whether the action has been unregistered */ unregisterAction(e) { const t = this.actions.indexOf(e); return t !== -1 ? (this.actions.splice(t, 1), Rs.Triggers[e.trigger] -= 1, Rs.Triggers[e.trigger] === 0 && delete Rs.Triggers[e.trigger], e._actionManager = null, this.getScene()._registeredActions--, !0) : !1; } /** * Process a specific trigger * @param trigger defines the trigger to process * @param evt defines the event details to be processed */ processTrigger(e, t) { for (let r = 0; r < this.actions.length; r++) { const n = this.actions[r]; if (n.trigger === e) { if (t && (e === Rs.OnKeyUpTrigger || e === Rs.OnKeyDownTrigger)) { const i = n.getTriggerParameter(); if (typeof i == "function") { if (!i(t)) continue; } else if (i && i !== t.sourceEvent.keyCode) { if (!i.toLowerCase) continue; const s = i.toLowerCase(); if (s !== t.sourceEvent.key) { const a = t.sourceEvent.charCode ? t.sourceEvent.charCode : t.sourceEvent.keyCode; if (String.fromCharCode(a).toLowerCase() !== s) continue; } } } n._executeCurrent(t); } } } /** * @internal */ _getEffectiveTarget(e, t) { const r = t.split("."); for (let n = 0; n < r.length - 1; n++) e = e[r[n]]; return e; } /** * @internal */ _getProperty(e) { const t = e.split("."); return t[t.length - 1]; } /** * Serialize this manager to a JSON object * @param name defines the property name to store this manager * @returns a JSON representation of this manager */ serialize(e) { const t = { children: new Array(), name: e, type: 3, properties: new Array() // Empty for root but required }; for (let r = 0; r < this.actions.length; r++) { const n = { type: 0, children: new Array(), name: Rs.GetTriggerName(this.actions[r].trigger), properties: new Array() }, i = this.actions[r].triggerOptions; if (i && typeof i != "number") if (i.parameter instanceof Node) n.properties.push(wa._GetTargetProperty(i.parameter)); else if (typeof i.parameter == "object") { const s = {}; sA.DeepCopy(i.parameter, s, ["mesh"]), i.parameter && i.parameter.mesh && (s._meshId = i.parameter.mesh.id), n.properties.push({ name: "parameter", targetType: null, value: s }); } else n.properties.push({ name: "parameter", targetType: null, value: i.parameter }); this.actions[r].serialize(n), t.children.push(n); } return t; } /** * Creates a new ActionManager from a JSON data * @param parsedActions defines the JSON data to read from * @param object defines the hosting mesh * @param scene defines the hosting scene */ static Parse(e, t, r) { const n = new Rs(r); t === null ? r.actionManager = n : t.actionManager = n; const i = (f, o) => { const d = Jo("BABYLON." + f); return d && new d(...o); }, s = (f, o, d, v) => { if (v === null) { const p = parseFloat(o); return o === "true" || o === "false" ? o === "true" : isNaN(p) ? o : p; } const u = v.split("."), l = o.split(","); for (let p = 0; p < u.length; p++) d = d[u[p]]; if (typeof d == "boolean") return l[0] === "true"; if (typeof d == "string") return l[0]; const P = []; for (let p = 0; p < l.length; p++) P.push(parseFloat(l[p])); return d instanceof S ? S.FromArray(P) : d instanceof Ir ? Ir.FromArray(P) : d instanceof Ne ? Ne.FromArray(P) : d instanceof xt ? xt.FromArray(P) : parseFloat(l[0]); }, a = (f, o, d, v, u = null) => { if (f.detached) return; const l = []; let P = null, p = null; const c = f.combine && f.combine.length > 0; if (f.type === 2 ? l.push(n) : l.push(o), c) { const T = []; for (let q = 0; q < f.combine.length; q++) a(f.combine[q], Rs.NothingTrigger, d, v, T); l.push(T); } else for (let T = 0; T < f.properties.length; T++) { let q = f.properties[T].value; const b = f.properties[T].name, j = f.properties[T].targetType; b === "target" ? j === "SceneProperties" ? q = P = r : j === "MaterialProperties" ? q = P = r.getMaterialByName(q) : q = P = r.getNodeByName(q) : b === "parent" ? q = r.getNodeByName(q) : b === "sound" ? r.getSoundByName && (q = r.getSoundByName(q)) : b !== "propertyPath" ? f.type === 2 && b === "operator" ? q = C9[q] : q = s(b, q, P, b === "value" ? p : null) : p = q, l.push(q); } if (u === null ? l.push(d) : l.push(null), f.name === "InterpolateValueAction") { const T = l[l.length - 2]; l[l.length - 1] = T, l[l.length - 2] = d; } let H = i(f.name, l); if (H instanceof DS && d !== null) { const T = new RN(o, d); v ? v.then(T) : n.registerAction(T), v = T; } u === null ? H instanceof DS ? (d = H, H = v) : (d = null, v ? v.then(H) : n.registerAction(H)) : u.push(H); for (let T = 0; T < f.children.length; T++) a(f.children[T], o, d, H, null); }; for (let f = 0; f < e.children.length; f++) { let o; const d = e.children[f]; if (d.properties.length > 0) { const v = d.properties[0].value, u = d.properties[0].targetType === null ? v : r.getMeshByName(v); u._meshId && (u.mesh = r.getMeshById(u._meshId)), o = { trigger: Rs[d.name], parameter: u }; } else o = Rs[d.name]; for (let v = 0; v < d.children.length; v++) d.detached || a(d.children[v], o, null, null); } } /** * Get a trigger name by index * @param trigger defines the trigger index * @returns a trigger name */ static GetTriggerName(e) { switch (e) { case 0: return "NothingTrigger"; case 1: return "OnPickTrigger"; case 2: return "OnLeftPickTrigger"; case 3: return "OnRightPickTrigger"; case 4: return "OnCenterPickTrigger"; case 5: return "OnPickDownTrigger"; case 6: return "OnDoublePickTrigger"; case 7: return "OnPickUpTrigger"; case 8: return "OnLongPressTrigger"; case 9: return "OnPointerOverTrigger"; case 10: return "OnPointerOutTrigger"; case 11: return "OnEveryFrameTrigger"; case 12: return "OnIntersectionEnterTrigger"; case 13: return "OnIntersectionExitTrigger"; case 14: return "OnKeyDownTrigger"; case 15: return "OnKeyUpTrigger"; case 16: return "OnPickOutTrigger"; default: return ""; } } } Rs.NothingTrigger = 0; Rs.OnPickTrigger = 1; Rs.OnLeftPickTrigger = 2; Rs.OnRightPickTrigger = 3; Rs.OnCenterPickTrigger = 4; Rs.OnPickDownTrigger = 5; Rs.OnDoublePickTrigger = 6; Rs.OnPickUpTrigger = 7; Rs.OnPickOutTrigger = 16; Rs.OnLongPressTrigger = 8; Rs.OnPointerOverTrigger = 9; Rs.OnPointerOutTrigger = 10; Rs.OnEveryFrameTrigger = 11; Rs.OnIntersectionEnterTrigger = 12; Rs.OnIntersectionExitTrigger = 13; Rs.OnKeyDownTrigger = 14; Rs.OnKeyUpTrigger = 15; class m$ extends wa { /** * Instantiate the action * @param triggerOptions defines the trigger options * @param sound defines the sound to play * @param condition defines the trigger related conditions */ constructor(e, t, r) { super(e, r), this._sound = t; } /** @internal */ _prepare() { } /** * Execute the action and play the sound. */ execute() { this._sound !== void 0 && this._sound.play(); } /** * Serializes the actions and its related information. * @param parent defines the object to serialize in * @returns the serialized object */ serialize(e) { return super._serialize({ name: "PlaySoundAction", properties: [{ name: "sound", value: this._sound.name }] }, e); } } class B$ extends wa { /** * Instantiate the action * @param triggerOptions defines the trigger options * @param sound defines the sound to stop * @param condition defines the trigger related conditions */ constructor(e, t, r) { super(e, r), this._sound = t; } /** @internal */ _prepare() { } /** * Execute the action and stop the sound. */ execute() { this._sound !== void 0 && this._sound.stop(); } /** * Serializes the actions and its related information. * @param parent defines the object to serialize in * @returns the serialized object */ serialize(e) { return super._serialize({ name: "StopSoundAction", properties: [{ name: "sound", value: this._sound.name }] }, e); } } Ue("BABYLON.PlaySoundAction", m$); Ue("BABYLON.StopSoundAction", B$); class Mx { /** * Evaluate a query * @param query defines the query to evaluate * @param evaluateCallback defines the callback used to filter result * @returns true if the query matches */ static Eval(e, t) { return e.match(/\([^()]*\)/g) ? e = e.replace(/\([^()]*\)/g, (r) => (r = r.slice(1, r.length - 1), Mx._HandleParenthesisContent(r, t))) : e = Mx._HandleParenthesisContent(e, t), e === "true" ? !0 : e === "false" ? !1 : Mx.Eval(e, t); } static _HandleParenthesisContent(e, t) { t = t || ((i) => i === "true"); let r; const n = e.split("||"); for (const i in n) if (Object.prototype.hasOwnProperty.call(n, i)) { let s = Mx._SimplifyNegation(n[i].trim()); const a = s.split("&&"); if (a.length > 1) for (let f = 0; f < a.length; ++f) { const o = Mx._SimplifyNegation(a[f].trim()); if (o !== "true" && o !== "false" ? o[0] === "!" ? r = !t(o.substring(1)) : r = t(o) : r = o === "true", !r) { s = "false"; break; } } if (r || s === "true") { r = !0; break; } s !== "true" && s !== "false" ? s[0] === "!" ? r = !t(s.substring(1)) : r = t(s) : r = s === "true"; } return r ? "true" : "false"; } static _SimplifyNegation(e) { return e = e.replace(/^[\s!]+/, (t) => (t = t.replace(/[\s]/g, () => ""), t.length % 2 ? "!" : "")), e = e.trim(), e === "!true" ? e = "false" : e === "!false" && (e = "true"), e; } } class Zi { /** * Adds support for tags on the given object * @param obj defines the object to use */ static EnableFor(e) { e._tags = e._tags || {}, e.hasTags = () => Zi.HasTags(e), e.addTags = (t) => Zi.AddTagsTo(e, t), e.removeTags = (t) => Zi.RemoveTagsFrom(e, t), e.matchesTagsQuery = (t) => Zi.MatchesQuery(e, t); } /** * Removes tags support * @param obj defines the object to use */ static DisableFor(e) { delete e._tags, delete e.hasTags, delete e.addTags, delete e.removeTags, delete e.matchesTagsQuery; } /** * Gets a boolean indicating if the given object has tags * @param obj defines the object to use * @returns a boolean */ static HasTags(e) { if (!e._tags) return !1; const t = e._tags; for (const r in t) if (Object.prototype.hasOwnProperty.call(t, r)) return !0; return !1; } /** * Gets the tags available on a given object * @param obj defines the object to use * @param asString defines if the tags must be returned as a string instead of an array of strings * @returns the tags */ static GetTags(e, t = !0) { if (!e._tags) return null; if (t) { const r = []; for (const n in e._tags) Object.prototype.hasOwnProperty.call(e._tags, n) && e._tags[n] === !0 && r.push(n); return r.join(" "); } else return e._tags; } /** * Adds tags to an object * @param obj defines the object to use * @param tagsString defines the tag string. The tags 'true' and 'false' are reserved and cannot be used as tags. * A tag cannot start with '||', '&&', and '!'. It cannot contain whitespaces */ static AddTagsTo(e, t) { if (!t || typeof t != "string") return; t.split(" ").forEach(function(n) { Zi._AddTagTo(e, n); }); } /** * @internal */ static _AddTagTo(e, t) { t = t.trim(), !(t === "" || t === "true" || t === "false") && (t.match(/[\s]/) || t.match(/^([!]|([|]|[&]){2})/) || (Zi.EnableFor(e), e._tags[t] = !0)); } /** * Removes specific tags from a specific object * @param obj defines the object to use * @param tagsString defines the tags to remove */ static RemoveTagsFrom(e, t) { if (!Zi.HasTags(e)) return; const r = t.split(" "); for (const n in r) Zi._RemoveTagFrom(e, r[n]); } /** * @internal */ static _RemoveTagFrom(e, t) { delete e._tags[t]; } /** * Defines if tags hosted on an object match a given query * @param obj defines the object to use * @param tagsQuery defines the tag query * @returns a boolean */ static MatchesQuery(e, t) { return t === void 0 ? !0 : t === "" ? Zi.HasTags(e) : Mx.Eval(t, (r) => Zi.HasTags(e) && e._tags[r]); } } const sG = {}; function qn(A, e = !1) { if (!(e && sG[A])) return sG[A] = !0, `${A} needs to be imported before as it contains a side-effect required by your code.`; } const xC = {}, vC = {}, aG = function(A, e, t, r = {}) { const n = A(); Zi && Zi.HasTags(e) && Zi.AddTagsTo(n, Zi.GetTags(e, !0)); const i = N5(n), s = {}; for (const a in i) { const f = i[a], o = e[a], d = f.type; if (o != null && (a !== "uniqueId" || jt.AllowLoadingUniqueId)) switch (d) { case 0: case 6: case 11: n[a] = o; break; case 1: r.cloneTexturesOnlyOnce && s[o.uniqueId] ? n[a] = s[o.uniqueId] : (n[a] = t || o.isRenderTarget ? o : o.clone(), s[o.uniqueId] = n[a]); break; case 2: case 3: case 4: case 5: case 7: case 10: case 12: n[a] = t ? o : o.clone(); break; } } return n; }; function B9e(A) { const e = A.getClassName(); return xC[e] || (xC[e] = {}), xC[e]; } function N5(A) { const e = A.getClassName(); if (vC[e]) return vC[e]; vC[e] = {}; const t = vC[e]; let r = A, n = e; for (; n; ) { const i = xC[n]; for (const f in i) t[f] = i[f]; let s, a = !1; do { if (s = Object.getPrototypeOf(r), !s.getClassName) { a = !0; break; } if (s.getClassName() !== n) break; r = s; } while (s); if (a) break; n = s.getClassName(), r = s; } return t; } function Dp(A, e) { return (t, r) => { const n = B9e(t); n[r] || (n[r] = { type: A, sourceName: e }); }; } function W9e(A, e = null) { return (t, r) => { const n = e || "_" + r; Object.defineProperty(t, r, { get: function() { return this[n]; }, set: function(i) { typeof this.equals == "function" && this.equals(i) || this[n] !== i && (this[n] = i, t[A].apply(this)); }, enumerable: !0, configurable: !0 }); }; } function At(A, e = null) { return W9e(A, e); } function M(A) { return Dp(0, A); } function en(A) { return Dp(1, A); } function Oi(A) { return Dp(2, A); } function eU(A) { return Dp(3, A); } function HR(A) { return Dp(4, A); } function fo(A) { return Dp(5, A); } function tU(A) { return Dp(6, A); } function W$(A) { return Dp(7, A); } function rU(A) { return Dp(8, A); } function CN(A) { return Dp(9, A); } function S$(A) { return Dp(10, A); } function qO(A) { return Dp(12, A); } function U$(A) { return Dp(11, A); } class jt { /** * Appends the serialized animations from the source animations * @param source Source containing the animations * @param destination Target to store the animations */ static AppendSerializedAnimations(e, t) { if (e.animations) { t.animations = []; for (let r = 0; r < e.animations.length; r++) { const n = e.animations[r]; t.animations.push(n.serialize()); } } } /** * Static function used to serialized a specific entity * @param entity defines the entity to serialize * @param serializationObject defines the optional target object where serialization data will be stored * @returns a JSON compatible object representing the serialization of the entity */ static Serialize(e, t) { t || (t = {}), Zi && (t.tags = Zi.GetTags(e)); const r = N5(e); for (const n in r) { const i = r[n], s = i.sourceName || n, a = i.type, f = e[n]; if (f != null && (n !== "uniqueId" || jt.AllowLoadingUniqueId)) switch (a) { case 0: t[s] = f; break; case 1: t[s] = f.serialize(); break; case 2: t[s] = f.asArray(); break; case 3: t[s] = f.serialize(); break; case 4: t[s] = f.asArray(); break; case 5: t[s] = f.asArray(); break; case 6: t[s] = f.id; break; case 7: t[s] = f.serialize(); break; case 8: t[s] = f.asArray(); break; case 9: t[s] = f.serialize(); break; case 10: t[s] = f.asArray(); break; case 11: t[s] = f.id; break; case 12: t[s] = f.asArray(); break; } } return t; } /** * Given a source json and a destination object in a scene, this function will parse the source and will try to apply its content to the destination object * @param source the source json data * @param destination the destination object * @param scene the scene where the object is * @param rootUrl root url to use to load assets */ static ParseProperties(e, t, r, n) { n || (n = ""); const i = N5(t); for (const s in i) { const a = i[s], f = e[a.sourceName || s], o = a.type; if (f != null && (s !== "uniqueId" || jt.AllowLoadingUniqueId)) { const d = t; switch (o) { case 0: d[s] = f; break; case 1: r && (d[s] = jt._TextureParser(f, r, n)); break; case 2: d[s] = Ne.FromArray(f); break; case 3: d[s] = jt._FresnelParametersParser(f); break; case 4: d[s] = at.FromArray(f); break; case 5: d[s] = S.FromArray(f); break; case 6: r && (d[s] = r.getLastMeshById(f)); break; case 7: d[s] = jt._ColorCurvesParser(f); break; case 8: d[s] = xt.FromArray(f); break; case 9: d[s] = jt._ImageProcessingConfigurationParser(f); break; case 10: d[s] = Ze.FromArray(f); break; case 11: r && (d[s] = r.getCameraById(f)); break; case 12: d[s] = he.FromArray(f); break; } } } } /** * Creates a new entity from a serialization data object * @param creationFunction defines a function used to instanciated the new entity * @param source defines the source serialization data * @param scene defines the hosting scene * @param rootUrl defines the root url for resources * @returns a new entity */ static Parse(e, t, r, n = null) { const i = e(); return Zi && Zi.AddTagsTo(i, t.tags), jt.ParseProperties(t, i, r, n), i; } /** * Clones an object * @param creationFunction defines the function used to instanciate the new object * @param source defines the source object * @returns the cloned object */ static Clone(e, t, r = {}) { return aG(e, t, !1, r); } /** * Instanciates a new object based on a source one (some data will be shared between both object) * @param creationFunction defines the function used to instanciate the new object * @param source defines the source object * @returns the new object */ static Instanciate(e, t) { return aG(e, t, !0); } } jt.AllowLoadingUniqueId = !1; jt._ImageProcessingConfigurationParser = (A) => { throw qn("ImageProcessingConfiguration"); }; jt._FresnelParametersParser = (A) => { throw qn("FresnelParameters"); }; jt._ColorCurvesParser = (A) => { throw qn("ColorCurves"); }; jt._TextureParser = (A, e, t) => { throw qn("Texture"); }; function Hq(A, e, t, r) { const n = t.value; t.value = (...i) => { let s = n; if (typeof _native < "u" && _native[e]) { const a = _native[e]; r ? s = (...f) => r(...f) ? a(...f) : n(...f) : s = a; } return A[e] = s, s(...i); }; } Hq.filter = function(A) { return (e, t, r) => Hq(e, t, r, A); }; var CI; (function(A) { A[A.NONE = 0] = "NONE", A[A.STEP = 1] = "STEP"; })(CI || (CI = {})); class Am { /** * Initializes the range of an animation * @param name The name of the animation range * @param from The starting frame of the animation * @param to The ending frame of the animation */ constructor(e, t, r) { this.name = e, this.from = t, this.to = r; } /** * Makes a copy of the animation range * @returns A copy of the animation range */ clone() { return new Am(this.name, this.from, this.to); } } function C(A, e, t, r) { var n = arguments.length, i = n < 3 ? e : r === null ? r = Object.getOwnPropertyDescriptor(e, t) : r, s; if (typeof Reflect == "object" && typeof Reflect.decorate == "function") i = Reflect.decorate(A, e, t, r); else for (var a = A.length - 1; a >= 0; a--) (s = A[a]) && (i = (n < 3 ? s(i) : n > 3 ? s(e, t, i) : s(e, t)) || i); return n > 3 && i && Object.defineProperty(e, t, i), i; } class S9e { constructor() { this._doNotSerialize = !1, this._isDisposed = !1, this._sceneRootNodesIndex = -1, this._isEnabled = !0, this._isParentEnabled = !0, this._isReady = !0, this._onEnabledStateChangedObservable = new Oe(), this._onClonedObservable = new Oe(); } } let Cs = class Q5 { /** * Add a new node constructor * @param type defines the type name of the node to construct * @param constructorFunc defines the constructor function */ static AddNodeConstructor(e, t) { this._NodeConstructors[e] = t; } /** * Returns a node constructor based on type name * @param type defines the type name * @param name defines the new node name * @param scene defines the hosting scene * @param options defines optional options to transmit to constructors * @returns the new constructor or null */ static Construct(e, t, r, n) { const i = this._NodeConstructors[e]; return i ? i(t, r, n) : null; } /** * Gets or sets the accessibility tag to describe the node for accessibility purpose. */ set accessibilityTag(e) { this._accessibilityTag = e, this.onAccessibilityTagChangedObservable.notifyObservers(e); } get accessibilityTag() { return this._accessibilityTag; } /** * Gets or sets a boolean used to define if the node must be serialized */ get doNotSerialize() { return this._nodeDataStorage._doNotSerialize ? !0 : this._parentNode ? this._parentNode.doNotSerialize : !1; } set doNotSerialize(e) { this._nodeDataStorage._doNotSerialize = e; } /** * Gets a boolean indicating if the node has been disposed * @returns true if the node was disposed */ isDisposed() { return this._nodeDataStorage._isDisposed; } /** * Gets or sets the parent of the node (without keeping the current position in the scene) * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/parent_pivot/parent */ set parent(e) { if (this._parentNode === e) return; const t = this._parentNode; if (this._parentNode && this._parentNode._children !== void 0 && this._parentNode._children !== null) { const r = this._parentNode._children.indexOf(this); r !== -1 && this._parentNode._children.splice(r, 1), !e && !this._nodeDataStorage._isDisposed && this._addToSceneRootNodes(); } this._parentNode = e, this._parentNode && ((this._parentNode._children === void 0 || this._parentNode._children === null) && (this._parentNode._children = new Array()), this._parentNode._children.push(this), t || this._removeFromSceneRootNodes()), this._syncParentEnabledState(); } get parent() { return this._parentNode; } /** * @internal */ _serializeAsParent(e) { e.parentId = this.uniqueId; } /** @internal */ _addToSceneRootNodes() { this._nodeDataStorage._sceneRootNodesIndex === -1 && (this._nodeDataStorage._sceneRootNodesIndex = this._scene.rootNodes.length, this._scene.rootNodes.push(this)); } /** @internal */ _removeFromSceneRootNodes() { if (this._nodeDataStorage._sceneRootNodesIndex !== -1) { const e = this._scene.rootNodes, t = e.length - 1; e[this._nodeDataStorage._sceneRootNodesIndex] = e[t], e[this._nodeDataStorage._sceneRootNodesIndex]._nodeDataStorage._sceneRootNodesIndex = this._nodeDataStorage._sceneRootNodesIndex, this._scene.rootNodes.pop(), this._nodeDataStorage._sceneRootNodesIndex = -1; } } /** * Gets or sets the animation properties override */ get animationPropertiesOverride() { return this._animationPropertiesOverride ? this._animationPropertiesOverride : this._scene.animationPropertiesOverride; } set animationPropertiesOverride(e) { this._animationPropertiesOverride = e; } /** * Gets a string identifying the name of the class * @returns "Node" string */ getClassName() { return "Node"; } /** * Sets a callback that will be raised when the node will be disposed */ set onDispose(e) { this._onDisposeObserver && this.onDisposeObservable.remove(this._onDisposeObserver), this._onDisposeObserver = this.onDisposeObservable.add(e); } /** * An event triggered when the enabled state of the node changes */ get onEnabledStateChangedObservable() { return this._nodeDataStorage._onEnabledStateChangedObservable; } /** * An event triggered when the node is cloned */ get onClonedObservable() { return this._nodeDataStorage._onClonedObservable; } /** * Creates a new Node * @param name the name and id to be given to this node * @param scene the scene this node will be added to */ constructor(e, t = null) { this._isDirty = !1, this._nodeDataStorage = new S9e(), this.state = "", this.metadata = null, this.reservedDataStore = null, this._accessibilityTag = null, this.onAccessibilityTagChangedObservable = new Oe(), this._parentContainer = null, this.animations = [], this._ranges = {}, this.onReady = null, this._currentRenderId = -1, this._parentUpdateId = -1, this._childUpdateId = -1, this._waitingParentId = null, this._waitingParentInstanceIndex = null, this._waitingParsedUniqueId = null, this._cache = {}, this._parentNode = null, this._children = null, this._worldMatrix = he.Identity(), this._worldMatrixDeterminant = 0, this._worldMatrixDeterminantIsDirty = !0, this._animationPropertiesOverride = null, this._isNode = !0, this.onDisposeObservable = new Oe(), this._onDisposeObserver = null, this._behaviors = new Array(), this.name = e, this.id = e, this._scene = t || gr.LastCreatedScene, this.uniqueId = this._scene.getUniqueId(), this._initCache(); } /** * Gets the scene of the node * @returns a scene */ getScene() { return this._scene; } /** * Gets the engine of the node * @returns a Engine */ getEngine() { return this._scene.getEngine(); } /** * Attach a behavior to the node * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors * @param behavior defines the behavior to attach * @param attachImmediately defines that the behavior must be attached even if the scene is still loading * @returns the current Node */ addBehavior(e, t = !1) { return this._behaviors.indexOf(e) !== -1 ? this : (e.init(), this._scene.isLoading && !t ? this._scene.onDataLoadedObservable.addOnce(() => { e.attach(this); }) : e.attach(this), this._behaviors.push(e), this); } /** * Remove an attached behavior * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors * @param behavior defines the behavior to attach * @returns the current Node */ removeBehavior(e) { const t = this._behaviors.indexOf(e); return t === -1 ? this : (this._behaviors[t].detach(), this._behaviors.splice(t, 1), this); } /** * Gets the list of attached behaviors * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors */ get behaviors() { return this._behaviors; } /** * Gets an attached behavior by name * @param name defines the name of the behavior to look for * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors * @returns null if behavior was not found else the requested behavior */ getBehaviorByName(e) { for (const t of this._behaviors) if (t.name === e) return t; return null; } /** * Returns the latest update of the World matrix * @returns a Matrix */ getWorldMatrix() { return this._currentRenderId !== this._scene.getRenderId() && this.computeWorldMatrix(), this._worldMatrix; } /** @internal */ _getWorldMatrixDeterminant() { return this._worldMatrixDeterminantIsDirty && (this._worldMatrixDeterminantIsDirty = !1, this._worldMatrixDeterminant = this._worldMatrix.determinant()), this._worldMatrixDeterminant; } /** * Returns directly the latest state of the mesh World matrix. * A Matrix is returned. */ get worldMatrixFromCache() { return this._worldMatrix; } // override it in derived class if you add new variables to the cache // and call the parent class method /** @internal */ _initCache() { this._cache = {}, this._cache.parent = void 0; } /** * @internal */ updateCache(e) { !e && this.isSynchronized() || (this._cache.parent = this.parent, this._updateCache()); } /** * @internal */ _getActionManagerForTrigger(e, t = !0) { return this.parent ? this.parent._getActionManagerForTrigger(e, !1) : null; } // override it in derived class if you add new variables to the cache // and call the parent class method if !ignoreParentClass /** * @internal */ _updateCache(e) { } // override it in derived class if you add new variables to the cache /** @internal */ _isSynchronized() { return !0; } /** @internal */ _markSyncedWithParent() { this._parentNode && (this._parentUpdateId = this._parentNode._childUpdateId); } /** @internal */ isSynchronizedWithParent() { return this._parentNode ? this._parentNode._isDirty || this._parentUpdateId !== this._parentNode._childUpdateId ? !1 : this._parentNode.isSynchronized() : !0; } /** @internal */ isSynchronized() { return this._cache.parent !== this._parentNode ? (this._cache.parent = this._parentNode, !1) : this._parentNode && !this.isSynchronizedWithParent() ? !1 : this._isSynchronized(); } /** * Is this node ready to be used/rendered * @param _completeCheck defines if a complete check (including materials and lights) has to be done (false by default) * @returns true if the node is ready */ isReady(e = !1) { return this._nodeDataStorage._isReady; } /** * Flag the node as dirty (Forcing it to update everything) * @param _property helps children apply precise "dirtyfication" * @returns this node */ markAsDirty(e) { return this._currentRenderId = Number.MAX_VALUE, this._isDirty = !0, this; } /** * Is this node enabled? * If the node has a parent, all ancestors will be checked and false will be returned if any are false (not enabled), otherwise will return true * @param checkAncestors indicates if this method should check the ancestors. The default is to check the ancestors. If set to false, the method will return the value of this node without checking ancestors * @returns whether this node (and its parent) is enabled */ isEnabled(e = !0) { return e === !1 ? this._nodeDataStorage._isEnabled : this._nodeDataStorage._isEnabled ? this._nodeDataStorage._isParentEnabled : !1; } /** @internal */ _syncParentEnabledState() { this._nodeDataStorage._isParentEnabled = this._parentNode ? this._parentNode.isEnabled() : !0, this._children && this._children.forEach((e) => { e._syncParentEnabledState(); }); } /** * Set the enabled state of this node * @param value defines the new enabled state */ setEnabled(e) { this._nodeDataStorage._isEnabled !== e && (this._nodeDataStorage._isEnabled = e, this._syncParentEnabledState(), this._nodeDataStorage._onEnabledStateChangedObservable.notifyObservers(e)); } /** * Is this node a descendant of the given node? * The function will iterate up the hierarchy until the ancestor was found or no more parents defined * @param ancestor defines the parent node to inspect * @returns a boolean indicating if this node is a descendant of the given node */ isDescendantOf(e) { return this.parent ? this.parent === e ? !0 : this.parent.isDescendantOf(e) : !1; } /** * @internal */ _getDescendants(e, t = !1, r) { if (this._children) for (let n = 0; n < this._children.length; n++) { const i = this._children[n]; (!r || r(i)) && e.push(i), t || i._getDescendants(e, !1, r); } } /** * Will return all nodes that have this node as ascendant * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored * @returns all children nodes of all types */ getDescendants(e, t) { const r = []; return this._getDescendants(r, e, t), r; } /** * Get all child-meshes of this node * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered (Default: false) * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored * @returns an array of AbstractMesh */ getChildMeshes(e, t) { const r = []; return this._getDescendants(r, e, (n) => (!t || t(n)) && n.cullingStrategy !== void 0), r; } /** * Get all direct children of this node * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered (Default: true) * @returns an array of Node */ getChildren(e, t = !0) { return this.getDescendants(t, e); } /** * @internal */ _setReady(e) { if (e !== this._nodeDataStorage._isReady) { if (!e) { this._nodeDataStorage._isReady = !1; return; } this.onReady && this.onReady(this), this._nodeDataStorage._isReady = !0; } } /** * Get an animation by name * @param name defines the name of the animation to look for * @returns null if not found else the requested animation */ getAnimationByName(e) { for (let t = 0; t < this.animations.length; t++) { const r = this.animations[t]; if (r.name === e) return r; } return null; } /** * Creates an animation range for this node * @param name defines the name of the range * @param from defines the starting key * @param to defines the end key */ createAnimationRange(e, t, r) { if (!this._ranges[e]) { this._ranges[e] = Q5._AnimationRangeFactory(e, t, r); for (let n = 0, i = this.animations.length; n < i; n++) this.animations[n] && this.animations[n].createRange(e, t, r); } } /** * Delete a specific animation range * @param name defines the name of the range to delete * @param deleteFrames defines if animation frames from the range must be deleted as well */ deleteAnimationRange(e, t = !0) { for (let r = 0, n = this.animations.length; r < n; r++) this.animations[r] && this.animations[r].deleteRange(e, t); this._ranges[e] = null; } /** * Get an animation range by name * @param name defines the name of the animation range to look for * @returns null if not found else the requested animation range */ getAnimationRange(e) { return this._ranges[e] || null; } /** * Clone the current node * @param name Name of the new clone * @param newParent New parent for the clone * @param doNotCloneChildren Do not clone children hierarchy * @returns the new transform node */ clone(e, t, r) { const n = jt.Clone(() => new Q5(e, this.getScene()), this); if (t && (n.parent = t), !r) { const i = this.getDescendants(!0); for (let s = 0; s < i.length; s++) { const a = i[s]; a.clone(e + "." + a.name, n); } } return n; } /** * Gets the list of all animation ranges defined on this node * @returns an array */ getAnimationRanges() { const e = []; let t; for (t in this._ranges) e.push(this._ranges[t]); return e; } /** * Will start the animation sequence * @param name defines the range frames for animation sequence * @param loop defines if the animation should loop (false by default) * @param speedRatio defines the speed factor in which to run the animation (1 by default) * @param onAnimationEnd defines a function to be executed when the animation ended (undefined by default) * @returns the object created for this animation. If range does not exist, it will return null */ beginAnimation(e, t, r, n) { const i = this.getAnimationRange(e); return i ? this._scene.beginAnimation(this, i.from, i.to, t, r, n) : null; } /** * Serialize animation ranges into a JSON compatible object * @returns serialization object */ serializeAnimationRanges() { const e = []; for (const t in this._ranges) { const r = this._ranges[t]; if (!r) continue; const n = {}; n.name = t, n.from = r.from, n.to = r.to, e.push(n); } return e; } /** * Computes the world matrix of the node * @param _force defines if the cache version should be invalidated forcing the world matrix to be created from scratch * @returns the world matrix */ computeWorldMatrix(e) { return this._worldMatrix || (this._worldMatrix = he.Identity()), this._worldMatrix; } /** * Releases resources associated with this node. * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default) * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default) */ dispose(e, t = !1) { if (this._nodeDataStorage._isDisposed = !0, !e) { const r = this.getDescendants(!0); for (const n of r) n.dispose(e, t); } this.parent ? this.parent = null : this._removeFromSceneRootNodes(), this.onDisposeObservable.notifyObservers(this), this.onDisposeObservable.clear(), this.onEnabledStateChangedObservable.clear(), this.onClonedObservable.clear(); for (const r of this._behaviors) r.detach(); this._behaviors.length = 0, this.metadata = null; } /** * Parse animation range data from a serialization object and store them into a given node * @param node defines where to store the animation ranges * @param parsedNode defines the serialization object to read data from * @param _scene defines the hosting scene */ static ParseAnimationRanges(e, t, r) { if (t.ranges) for (let n = 0; n < t.ranges.length; n++) { const i = t.ranges[n]; e.createAnimationRange(i.name, i.from, i.to); } } /** * Return the minimum and maximum world vectors of the entire hierarchy under current node * @param includeDescendants Include bounding info from descendants as well (true by default) * @param predicate defines a callback function that can be customize to filter what meshes should be included in the list used to compute the bounding vectors * @returns the new bounding vectors */ getHierarchyBoundingVectors(e = !0, t = null) { this.getScene().incrementRenderId(), this.computeWorldMatrix(!0); let r, n; const i = this; if (i.getBoundingInfo && i.subMeshes) { const s = i.getBoundingInfo(); r = s.boundingBox.minimumWorld.clone(), n = s.boundingBox.maximumWorld.clone(); } else r = new S(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE), n = new S(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE); if (e) { const s = this.getDescendants(!1); for (const a of s) { const f = a; if (f.computeWorldMatrix(!0), t && !t(f) || !f.getBoundingInfo || f.getTotalVertices() === 0) continue; const d = f.getBoundingInfo().boundingBox, v = d.minimumWorld, u = d.maximumWorld; S.CheckExtends(v, r, n), S.CheckExtends(u, r, n); } } return { min: r, max: n }; } }; Cs._AnimationRangeFactory = (A, e, t) => { throw qn("AnimationRange"); }; Cs._NodeConstructors = {}; C([ M() ], Cs.prototype, "name", void 0); C([ M() ], Cs.prototype, "id", void 0); C([ M() ], Cs.prototype, "uniqueId", void 0); C([ M() ], Cs.prototype, "state", void 0); C([ M() ], Cs.prototype, "metadata", void 0); class Qd { /** * Creates a Size object from the given width and height (floats). * @param width width of the new size * @param height height of the new size */ constructor(e, t) { this.width = e, this.height = t; } /** * Returns a string with the Size width and height * @returns a string with the Size width and height */ toString() { return `{W: ${this.width}, H: ${this.height}}`; } /** * "Size" * @returns the string "Size" */ getClassName() { return "Size"; } /** * Returns the Size hash code. * @returns a hash code for a unique width and height */ getHashCode() { let e = this.width | 0; return e = e * 397 ^ (this.height | 0), e; } /** * Updates the current size from the given one. * @param src the given size */ copyFrom(e) { this.width = e.width, this.height = e.height; } /** * Updates in place the current Size from the given floats. * @param width width of the new size * @param height height of the new size * @returns the updated Size. */ copyFromFloats(e, t) { return this.width = e, this.height = t, this; } /** * Updates in place the current Size from the given floats. * @param width width to set * @param height height to set * @returns the updated Size. */ set(e, t) { return this.copyFromFloats(e, t); } /** * Multiplies the width and height by numbers * @param w factor to multiple the width by * @param h factor to multiple the height by * @returns a new Size set with the multiplication result of the current Size and the given floats. */ multiplyByFloats(e, t) { return new Qd(this.width * e, this.height * t); } /** * Clones the size * @returns a new Size copied from the given one. */ clone() { return new Qd(this.width, this.height); } /** * True if the current Size and the given one width and height are strictly equal. * @param other the other size to compare against * @returns True if the current Size and the given one width and height are strictly equal. */ equals(e) { return e ? this.width === e.width && this.height === e.height : !1; } /** * The surface of the Size : width * height (float). */ get surface() { return this.width * this.height; } /** * Create a new size of zero * @returns a new Size set to (0.0, 0.0) */ static Zero() { return new Qd(0, 0); } /** * Sums the width and height of two sizes * @param otherSize size to add to this size * @returns a new Size set as the addition result of the current Size and the given one. */ add(e) { return new Qd(this.width + e.width, this.height + e.height); } /** * Subtracts the width and height of two * @param otherSize size to subtract to this size * @returns a new Size set as the subtraction result of the given one from the current Size. */ subtract(e) { return new Qd(this.width - e.width, this.height - e.height); } /** * Scales the width and height * @param scale the scale to multiply the width and height by * @returns a new Size set with the multiplication result of the current Size and the given floats. */ scale(e) { return new Qd(this.width * e, this.height * e); } /** * Creates a new Size set at the linear interpolation "amount" between "start" and "end" * @param start starting size to lerp between * @param end end size to lerp between * @param amount amount to lerp between the start and end values * @returns a new Size set at the linear interpolation "amount" between "start" and "end" */ static Lerp(e, t, r) { const n = e.width + (t.width - e.width) * r, i = e.height + (t.height - e.height) * r; return new Qd(n, i); } } function U9e() { return typeof _native < "u" && _native.XMLHttpRequest ? new _native.XMLHttpRequest() : new XMLHttpRequest(); } class ho { constructor() { this._xhr = U9e(), this._requestURL = ""; } /** * This function can be called to check if there are request modifiers for network requests * @returns true if there are any custom requests available */ static get IsCustomRequestAvailable() { return Object.keys(ho.CustomRequestHeaders).length > 0 || ho.CustomRequestModifiers.length > 0; } _injectCustomRequestHeaders() { if (!this._shouldSkipRequestModifications(this._requestURL)) for (const e in ho.CustomRequestHeaders) { const t = ho.CustomRequestHeaders[e]; t && this._xhr.setRequestHeader(e, t); } } _shouldSkipRequestModifications(e) { return ho.SkipRequestModificationForBabylonCDN && (e.includes("preview.babylonjs.com") || e.includes("cdn.babylonjs.com")); } /** * Gets or sets a function to be called when loading progress changes */ get onprogress() { return this._xhr.onprogress; } set onprogress(e) { this._xhr.onprogress = e; } /** * Returns client's state */ get readyState() { return this._xhr.readyState; } /** * Returns client's status */ get status() { return this._xhr.status; } /** * Returns client's status as a text */ get statusText() { return this._xhr.statusText; } /** * Returns client's response */ get response() { return this._xhr.response; } /** * Returns client's response url */ get responseURL() { return this._xhr.responseURL; } /** * Returns client's response as text */ get responseText() { return this._xhr.responseText; } /** * Gets or sets the expected response type */ get responseType() { return this._xhr.responseType; } set responseType(e) { this._xhr.responseType = e; } /** * Gets or sets the timeout value in milliseconds */ get timeout() { return this._xhr.timeout; } set timeout(e) { this._xhr.timeout = e; } addEventListener(e, t, r) { this._xhr.addEventListener(e, t, r); } removeEventListener(e, t, r) { this._xhr.removeEventListener(e, t, r); } /** * Cancels any network activity */ abort() { this._xhr.abort(); } /** * Initiates the request. The optional argument provides the request body. The argument is ignored if request method is GET or HEAD * @param body defines an optional request body */ send(e) { ho.CustomRequestHeaders && this._injectCustomRequestHeaders(), this._xhr.send(e); } /** * Sets the request method, request URL * @param method defines the method to use (GET, POST, etc..) * @param url defines the url to connect with */ open(e, t) { for (const r of ho.CustomRequestModifiers) { if (this._shouldSkipRequestModifications(t)) return; r(this._xhr, t); } return t = t.replace("file:http:", "http:"), t = t.replace("file:https:", "https:"), this._requestURL = t, this._xhr.open(e, t, !0); } /** * Sets the value of a request header. * @param name The name of the header whose value is to be set * @param value The value to set as the body of the header */ setRequestHeader(e, t) { this._xhr.setRequestHeader(e, t); } /** * Get the string containing the text of a particular header's value. * @param name The name of the header * @returns The string containing the text of the given header name */ getResponseHeader(e) { return this._xhr.getResponseHeader(e); } } ho.CustomRequestHeaders = {}; ho.CustomRequestModifiers = new Array(); ho.SkipRequestModificationForBabylonCDN = !0; const ON = Object.freeze(new Ze(0, 0, 0, 0)), yN = Object.freeze(S.Zero()), kN = Object.freeze(at.Zero()), EN = Object.freeze(Qd.Zero()), FN = Object.freeze(Ne.Black()), NN = Object.freeze(new xt(0, 0, 0, 0)), kH = { key: 0, repeatCount: 0, loopMode: 2 }; class st { /** * @internal Internal use */ static _PrepareAnimation(e, t, r, n, i, s, a, f) { let o; if (!isNaN(parseFloat(i)) && isFinite(i) ? o = st.ANIMATIONTYPE_FLOAT : i instanceof Ze ? o = st.ANIMATIONTYPE_QUATERNION : i instanceof S ? o = st.ANIMATIONTYPE_VECTOR3 : i instanceof at ? o = st.ANIMATIONTYPE_VECTOR2 : i instanceof Ne ? o = st.ANIMATIONTYPE_COLOR3 : i instanceof xt ? o = st.ANIMATIONTYPE_COLOR4 : i instanceof Qd && (o = st.ANIMATIONTYPE_SIZE), o == null) return null; const d = new st(e, t, r, o, a), v = [ { frame: 0, value: i }, { frame: n, value: s } ]; return d.setKeys(v), f !== void 0 && d.setEasingFunction(f), d; } /** * Sets up an animation * @param property The property to animate * @param animationType The animation type to apply * @param framePerSecond The frames per second of the animation * @param easingFunction The easing function used in the animation * @returns The created animation */ static CreateAnimation(e, t, r, n) { const i = new st(e + "Animation", e, r, t, st.ANIMATIONLOOPMODE_CONSTANT); return i.setEasingFunction(n), i; } /** * Create and start an animation on a node * @param name defines the name of the global animation that will be run on all nodes * @param target defines the target where the animation will take place * @param targetProperty defines property to animate * @param framePerSecond defines the number of frame per second yo use * @param totalFrame defines the number of frames in total * @param from defines the initial value * @param to defines the final value * @param loopMode defines which loop mode you want to use (off by default) * @param easingFunction defines the easing function to use (linear by default) * @param onAnimationEnd defines the callback to call when animation end * @param scene defines the hosting scene * @returns the animatable created for this animation */ static CreateAndStartAnimation(e, t, r, n, i, s, a, f, o, d, v) { const u = st._PrepareAnimation(e, r, n, i, s, a, f, o); return !u || (t.getScene && (v = t.getScene()), !v) ? null : v.beginDirectAnimation(t, [u], 0, i, u.loopMode === 1, 1, d); } /** * Create and start an animation on a node and its descendants * @param name defines the name of the global animation that will be run on all nodes * @param node defines the root node where the animation will take place * @param directDescendantsOnly if true only direct descendants will be used, if false direct and also indirect (children of children, an so on in a recursive manner) descendants will be used * @param targetProperty defines property to animate * @param framePerSecond defines the number of frame per second to use * @param totalFrame defines the number of frames in total * @param from defines the initial value * @param to defines the final value * @param loopMode defines which loop mode you want to use (off by default) * @param easingFunction defines the easing function to use (linear by default) * @param onAnimationEnd defines the callback to call when an animation ends (will be called once per node) * @returns the list of animatables created for all nodes * @example https://www.babylonjs-playground.com/#MH0VLI */ static CreateAndStartHierarchyAnimation(e, t, r, n, i, s, a, f, o, d, v) { const u = st._PrepareAnimation(e, n, i, s, a, f, o, d); return u ? t.getScene().beginDirectHierarchyAnimation(t, r, [u], 0, s, u.loopMode === 1, 1, v) : null; } /** * Creates a new animation, merges it with the existing animations and starts it * @param name Name of the animation * @param node Node which contains the scene that begins the animations * @param targetProperty Specifies which property to animate * @param framePerSecond The frames per second of the animation * @param totalFrame The total number of frames * @param from The frame at the beginning of the animation * @param to The frame at the end of the animation * @param loopMode Specifies the loop mode of the animation * @param easingFunction (Optional) The easing function of the animation, which allow custom mathematical formulas for animations * @param onAnimationEnd Callback to run once the animation is complete * @returns Nullable animation */ static CreateMergeAndStartAnimation(e, t, r, n, i, s, a, f, o, d) { const v = st._PrepareAnimation(e, r, n, i, s, a, f, o); return v ? (t.animations.push(v), t.getScene().beginAnimation(t, 0, i, v.loopMode === 1, 1, d)) : null; } /** @internal */ static MakeAnimationAdditive(e, t, r, n = !1, i) { var s, a; let f; typeof t == "object" ? f = t : f = { referenceFrame: t ?? 0, range: r, cloneOriginalAnimation: n, clonedAnimationName: i }; let o = e; if (f.cloneOriginalAnimation && (o = e.clone(), o.name = f.clonedAnimationName || o.name), !o._keys.length) return o; const d = f.referenceFrame && f.referenceFrame >= 0 ? f.referenceFrame : 0; let v = 0; const u = o._keys[0]; let l = o._keys.length - 1; const P = o._keys[l], p = { referenceValue: u.value, referencePosition: ue.Vector3[0], referenceQuaternion: ue.Quaternion[0], referenceScaling: ue.Vector3[1], keyPosition: ue.Vector3[2], keyQuaternion: ue.Quaternion[1], keyScaling: ue.Vector3[3] }; let c = u.frame, H = P.frame; if (f.range) { const b = o.getRange(f.range); b && (c = b.from, H = b.to); } else c = (s = f.fromFrame) !== null && s !== void 0 ? s : c, H = (a = f.toFrame) !== null && a !== void 0 ? a : H; if (c !== u.frame && (v = o.createKeyForFrame(c)), H !== P.frame && (l = o.createKeyForFrame(H)), o._keys.length === 1) { const b = o._getKeyValue(o._keys[0]); p.referenceValue = b.clone ? b.clone() : b; } else if (d <= u.frame) { const b = o._getKeyValue(u.value); p.referenceValue = b.clone ? b.clone() : b; } else if (d >= P.frame) { const b = o._getKeyValue(P.value); p.referenceValue = b.clone ? b.clone() : b; } else { kH.key = 0; const b = o._interpolate(d, kH); p.referenceValue = b.clone ? b.clone() : b; } o.dataType === st.ANIMATIONTYPE_QUATERNION ? p.referenceValue.normalize().conjugateInPlace() : o.dataType === st.ANIMATIONTYPE_MATRIX && (p.referenceValue.decompose(p.referenceScaling, p.referenceQuaternion, p.referencePosition), p.referenceQuaternion.normalize().conjugateInPlace()); let T = Number.MAX_VALUE; const q = f.clipKeys ? [] : null; for (let b = v; b <= l; b++) { let j = o._keys[b]; if (q && (j = { frame: j.frame, value: j.value.clone ? j.value.clone() : j.value, inTangent: j.inTangent, outTangent: j.outTangent, interpolation: j.interpolation, lockedTangent: j.lockedTangent }, T === Number.MAX_VALUE && (T = j.frame), j.frame -= T, q.push(j)), !(b && o.dataType !== st.ANIMATIONTYPE_FLOAT && j.value === u.value)) switch (o.dataType) { case st.ANIMATIONTYPE_MATRIX: j.value.decompose(p.keyScaling, p.keyQuaternion, p.keyPosition), p.keyPosition.subtractInPlace(p.referencePosition), p.keyScaling.divideInPlace(p.referenceScaling), p.referenceQuaternion.multiplyToRef(p.keyQuaternion, p.keyQuaternion), he.ComposeToRef(p.keyScaling, p.keyQuaternion, p.keyPosition, j.value); break; case st.ANIMATIONTYPE_QUATERNION: p.referenceValue.multiplyToRef(j.value, j.value); break; case st.ANIMATIONTYPE_VECTOR2: case st.ANIMATIONTYPE_VECTOR3: case st.ANIMATIONTYPE_COLOR3: case st.ANIMATIONTYPE_COLOR4: j.value.subtractToRef(p.referenceValue, j.value); break; case st.ANIMATIONTYPE_SIZE: j.value.width -= p.referenceValue.width, j.value.height -= p.referenceValue.height; break; default: j.value -= p.referenceValue; } } return q && o.setKeys(q, !0), o; } /** * Transition property of an host to the target Value * @param property The property to transition * @param targetValue The target Value of the property * @param host The object where the property to animate belongs * @param scene Scene used to run the animation * @param frameRate Framerate (in frame/s) to use * @param transition The transition type we want to use * @param duration The duration of the animation, in milliseconds * @param onAnimationEnd Callback trigger at the end of the animation * @returns Nullable animation */ static TransitionTo(e, t, r, n, i, s, a, f = null) { if (a <= 0) return r[e] = t, f && f(), null; const o = i * (a / 1e3); s.setKeys([ { frame: 0, value: r[e].clone ? r[e].clone() : r[e] }, { frame: o, value: t } ]), r.animations || (r.animations = []), r.animations.push(s); const d = n.beginAnimation(r, 0, o, !1); return d.onAnimationEnd = f, d; } /** * Return the array of runtime animations currently using this animation */ get runtimeAnimations() { return this._runtimeAnimations; } /** * Specifies if any of the runtime animations are currently running */ get hasRunningRuntimeAnimations() { for (const e of this._runtimeAnimations) if (!e.isStopped()) return !0; return !1; } /** * Initializes the animation * @param name Name of the animation * @param targetProperty Property to animate * @param framePerSecond The frames per second of the animation * @param dataType The data type of the animation * @param loopMode The loop mode of the animation * @param enableBlending Specifies if blending should be enabled */ constructor(e, t, r, n, i, s) { this.name = e, this.targetProperty = t, this.framePerSecond = r, this.dataType = n, this.loopMode = i, this.enableBlending = s, this._easingFunction = null, this._runtimeAnimations = new Array(), this._events = new Array(), this.blendingSpeed = 0.01, this._ranges = {}, this.targetPropertyPath = t.split("."), this.dataType = n, this.loopMode = i === void 0 ? st.ANIMATIONLOOPMODE_CYCLE : i, this.uniqueId = st._UniqueIdGenerator++; } // Methods /** * Converts the animation to a string * @param fullDetails support for multiple levels of logging within scene loading * @returns String form of the animation */ toString(e) { let t = "Name: " + this.name + ", property: " + this.targetProperty; if (t += ", datatype: " + ["Float", "Vector3", "Quaternion", "Matrix", "Color3", "Vector2"][this.dataType], t += ", nKeys: " + (this._keys ? this._keys.length : "none"), t += ", nRanges: " + (this._ranges ? Object.keys(this._ranges).length : "none"), e) { t += ", Ranges: {"; let r = !0; for (const n in this._ranges) r && (t += ", ", r = !1), t += n; t += "}"; } return t; } /** * Add an event to this animation * @param event Event to add */ addEvent(e) { this._events.push(e), this._events.sort((t, r) => t.frame - r.frame); } /** * Remove all events found at the given frame * @param frame The frame to remove events from */ removeEvents(e) { for (let t = 0; t < this._events.length; t++) this._events[t].frame === e && (this._events.splice(t, 1), t--); } /** * Retrieves all the events from the animation * @returns Events from the animation */ getEvents() { return this._events; } /** * Creates an animation range * @param name Name of the animation range * @param from Starting frame of the animation range * @param to Ending frame of the animation */ createRange(e, t, r) { this._ranges[e] || (this._ranges[e] = new Am(e, t, r)); } /** * Deletes an animation range by name * @param name Name of the animation range to delete * @param deleteFrames Specifies if the key frames for the range should also be deleted (true) or not (false) */ deleteRange(e, t = !0) { const r = this._ranges[e]; if (r) { if (t) { const n = r.from, i = r.to; for (let s = this._keys.length - 1; s >= 0; s--) this._keys[s].frame >= n && this._keys[s].frame <= i && this._keys.splice(s, 1); } this._ranges[e] = null; } } /** * Gets the animation range by name, or null if not defined * @param name Name of the animation range * @returns Nullable animation range */ getRange(e) { return this._ranges[e]; } /** * Gets the key frames from the animation * @returns The key frames of the animation */ getKeys() { return this._keys; } /** * Gets the highest frame rate of the animation * @returns Highest frame rate of the animation */ getHighestFrame() { let e = 0; for (let t = 0, r = this._keys.length; t < r; t++) e < this._keys[t].frame && (e = this._keys[t].frame); return e; } /** * Gets the easing function of the animation * @returns Easing function of the animation */ getEasingFunction() { return this._easingFunction; } /** * Sets the easing function of the animation * @param easingFunction A custom mathematical formula for animation */ setEasingFunction(e) { this._easingFunction = e; } /** * Interpolates a scalar linearly * @param startValue Start value of the animation curve * @param endValue End value of the animation curve * @param gradient Scalar amount to interpolate * @returns Interpolated scalar value */ floatInterpolateFunction(e, t, r) { return Xt.Lerp(e, t, r); } /** * Interpolates a scalar cubically * @param startValue Start value of the animation curve * @param outTangent End tangent of the animation * @param endValue End value of the animation curve * @param inTangent Start tangent of the animation curve * @param gradient Scalar amount to interpolate * @returns Interpolated scalar value */ floatInterpolateFunctionWithTangents(e, t, r, n, i) { return Xt.Hermite(e, t, r, n, i); } /** * Interpolates a quaternion using a spherical linear interpolation * @param startValue Start value of the animation curve * @param endValue End value of the animation curve * @param gradient Scalar amount to interpolate * @returns Interpolated quaternion value */ quaternionInterpolateFunction(e, t, r) { return Ze.Slerp(e, t, r); } /** * Interpolates a quaternion cubically * @param startValue Start value of the animation curve * @param outTangent End tangent of the animation curve * @param endValue End value of the animation curve * @param inTangent Start tangent of the animation curve * @param gradient Scalar amount to interpolate * @returns Interpolated quaternion value */ quaternionInterpolateFunctionWithTangents(e, t, r, n, i) { return Ze.Hermite(e, t, r, n, i).normalize(); } /** * Interpolates a Vector3 linearly * @param startValue Start value of the animation curve * @param endValue End value of the animation curve * @param gradient Scalar amount to interpolate (value between 0 and 1) * @returns Interpolated scalar value */ vector3InterpolateFunction(e, t, r) { return S.Lerp(e, t, r); } /** * Interpolates a Vector3 cubically * @param startValue Start value of the animation curve * @param outTangent End tangent of the animation * @param endValue End value of the animation curve * @param inTangent Start tangent of the animation curve * @param gradient Scalar amount to interpolate (value between 0 and 1) * @returns InterpolatedVector3 value */ vector3InterpolateFunctionWithTangents(e, t, r, n, i) { return S.Hermite(e, t, r, n, i); } /** * Interpolates a Vector2 linearly * @param startValue Start value of the animation curve * @param endValue End value of the animation curve * @param gradient Scalar amount to interpolate (value between 0 and 1) * @returns Interpolated Vector2 value */ vector2InterpolateFunction(e, t, r) { return at.Lerp(e, t, r); } /** * Interpolates a Vector2 cubically * @param startValue Start value of the animation curve * @param outTangent End tangent of the animation * @param endValue End value of the animation curve * @param inTangent Start tangent of the animation curve * @param gradient Scalar amount to interpolate (value between 0 and 1) * @returns Interpolated Vector2 value */ vector2InterpolateFunctionWithTangents(e, t, r, n, i) { return at.Hermite(e, t, r, n, i); } /** * Interpolates a size linearly * @param startValue Start value of the animation curve * @param endValue End value of the animation curve * @param gradient Scalar amount to interpolate * @returns Interpolated Size value */ sizeInterpolateFunction(e, t, r) { return Qd.Lerp(e, t, r); } /** * Interpolates a Color3 linearly * @param startValue Start value of the animation curve * @param endValue End value of the animation curve * @param gradient Scalar amount to interpolate * @returns Interpolated Color3 value */ color3InterpolateFunction(e, t, r) { return Ne.Lerp(e, t, r); } /** * Interpolates a Color3 cubically * @param startValue Start value of the animation curve * @param outTangent End tangent of the animation * @param endValue End value of the animation curve * @param inTangent Start tangent of the animation curve * @param gradient Scalar amount to interpolate * @returns interpolated value */ color3InterpolateFunctionWithTangents(e, t, r, n, i) { return Ne.Hermite(e, t, r, n, i); } /** * Interpolates a Color4 linearly * @param startValue Start value of the animation curve * @param endValue End value of the animation curve * @param gradient Scalar amount to interpolate * @returns Interpolated Color3 value */ color4InterpolateFunction(e, t, r) { return xt.Lerp(e, t, r); } /** * Interpolates a Color4 cubically * @param startValue Start value of the animation curve * @param outTangent End tangent of the animation * @param endValue End value of the animation curve * @param inTangent Start tangent of the animation curve * @param gradient Scalar amount to interpolate * @returns interpolated value */ color4InterpolateFunctionWithTangents(e, t, r, n, i) { return xt.Hermite(e, t, r, n, i); } /** * @internal Internal use only */ _getKeyValue(e) { return typeof e == "function" ? e() : e; } /** * Evaluate the animation value at a given frame * @param currentFrame defines the frame where we want to evaluate the animation * @returns the animation value */ evaluate(e) { return kH.key = 0, this._interpolate(e, kH); } /** * @internal Internal use only */ _interpolate(e, t, r = !1) { var n; if (t.loopMode === st.ANIMATIONLOOPMODE_CONSTANT && t.repeatCount > 0) return t.highLimitValue.clone ? t.highLimitValue.clone() : t.highLimitValue; const i = this._keys, s = i.length; let a = t.key; for (; a >= 0 && e < i[a].frame; ) --a; for (; a + 1 <= s - 1 && e >= i[a + 1].frame; ) ++a; if (t.key = a, a < 0) return r ? void 0 : this._getKeyValue(i[0].value); if (a + 1 > s - 1) return r ? void 0 : this._getKeyValue(i[s - 1].value); const f = i[a], o = i[a + 1]; if (r && (e === f.frame || e === o.frame)) return; const d = this._getKeyValue(f.value), v = this._getKeyValue(o.value); if (f.interpolation === CI.STEP) return o.frame > e ? d : v; const u = f.outTangent !== void 0 && o.inTangent !== void 0, l = o.frame - f.frame; let P = (e - f.frame) / l; const p = f.easingFunction || this.getEasingFunction(); switch (p !== null && (P = p.ease(P)), this.dataType) { case st.ANIMATIONTYPE_FLOAT: { const c = u ? this.floatInterpolateFunctionWithTangents(d, f.outTangent * l, v, o.inTangent * l, P) : this.floatInterpolateFunction(d, v, P); switch (t.loopMode) { case st.ANIMATIONLOOPMODE_CYCLE: case st.ANIMATIONLOOPMODE_CONSTANT: case st.ANIMATIONLOOPMODE_YOYO: return c; case st.ANIMATIONLOOPMODE_RELATIVE: case st.ANIMATIONLOOPMODE_RELATIVE_FROM_CURRENT: return ((n = t.offsetValue) !== null && n !== void 0 ? n : 0) * t.repeatCount + c; } break; } case st.ANIMATIONTYPE_QUATERNION: { const c = u ? this.quaternionInterpolateFunctionWithTangents(d, f.outTangent.scale(l), v, o.inTangent.scale(l), P) : this.quaternionInterpolateFunction(d, v, P); switch (t.loopMode) { case st.ANIMATIONLOOPMODE_CYCLE: case st.ANIMATIONLOOPMODE_CONSTANT: case st.ANIMATIONLOOPMODE_YOYO: return c; case st.ANIMATIONLOOPMODE_RELATIVE: case st.ANIMATIONLOOPMODE_RELATIVE_FROM_CURRENT: return c.addInPlace((t.offsetValue || ON).scale(t.repeatCount)); } return c; } case st.ANIMATIONTYPE_VECTOR3: { const c = u ? this.vector3InterpolateFunctionWithTangents(d, f.outTangent.scale(l), v, o.inTangent.scale(l), P) : this.vector3InterpolateFunction(d, v, P); switch (t.loopMode) { case st.ANIMATIONLOOPMODE_CYCLE: case st.ANIMATIONLOOPMODE_CONSTANT: case st.ANIMATIONLOOPMODE_YOYO: return c; case st.ANIMATIONLOOPMODE_RELATIVE: case st.ANIMATIONLOOPMODE_RELATIVE_FROM_CURRENT: return c.add((t.offsetValue || yN).scale(t.repeatCount)); } break; } case st.ANIMATIONTYPE_VECTOR2: { const c = u ? this.vector2InterpolateFunctionWithTangents(d, f.outTangent.scale(l), v, o.inTangent.scale(l), P) : this.vector2InterpolateFunction(d, v, P); switch (t.loopMode) { case st.ANIMATIONLOOPMODE_CYCLE: case st.ANIMATIONLOOPMODE_CONSTANT: case st.ANIMATIONLOOPMODE_YOYO: return c; case st.ANIMATIONLOOPMODE_RELATIVE: case st.ANIMATIONLOOPMODE_RELATIVE_FROM_CURRENT: return c.add((t.offsetValue || kN).scale(t.repeatCount)); } break; } case st.ANIMATIONTYPE_SIZE: { switch (t.loopMode) { case st.ANIMATIONLOOPMODE_CYCLE: case st.ANIMATIONLOOPMODE_CONSTANT: case st.ANIMATIONLOOPMODE_YOYO: return this.sizeInterpolateFunction(d, v, P); case st.ANIMATIONLOOPMODE_RELATIVE: case st.ANIMATIONLOOPMODE_RELATIVE_FROM_CURRENT: return this.sizeInterpolateFunction(d, v, P).add((t.offsetValue || EN).scale(t.repeatCount)); } break; } case st.ANIMATIONTYPE_COLOR3: { const c = u ? this.color3InterpolateFunctionWithTangents(d, f.outTangent.scale(l), v, o.inTangent.scale(l), P) : this.color3InterpolateFunction(d, v, P); switch (t.loopMode) { case st.ANIMATIONLOOPMODE_CYCLE: case st.ANIMATIONLOOPMODE_CONSTANT: case st.ANIMATIONLOOPMODE_YOYO: return c; case st.ANIMATIONLOOPMODE_RELATIVE: case st.ANIMATIONLOOPMODE_RELATIVE_FROM_CURRENT: return c.add((t.offsetValue || FN).scale(t.repeatCount)); } break; } case st.ANIMATIONTYPE_COLOR4: { const c = u ? this.color4InterpolateFunctionWithTangents(d, f.outTangent.scale(l), v, o.inTangent.scale(l), P) : this.color4InterpolateFunction(d, v, P); switch (t.loopMode) { case st.ANIMATIONLOOPMODE_CYCLE: case st.ANIMATIONLOOPMODE_CONSTANT: case st.ANIMATIONLOOPMODE_YOYO: return c; case st.ANIMATIONLOOPMODE_RELATIVE: case st.ANIMATIONLOOPMODE_RELATIVE_FROM_CURRENT: return c.add((t.offsetValue || NN).scale(t.repeatCount)); } break; } case st.ANIMATIONTYPE_MATRIX: { switch (t.loopMode) { case st.ANIMATIONLOOPMODE_CYCLE: case st.ANIMATIONLOOPMODE_CONSTANT: case st.ANIMATIONLOOPMODE_YOYO: return st.AllowMatricesInterpolation ? this.matrixInterpolateFunction(d, v, P, t.workValue) : d; case st.ANIMATIONLOOPMODE_RELATIVE: case st.ANIMATIONLOOPMODE_RELATIVE_FROM_CURRENT: return d; } break; } } return 0; } /** * Defines the function to use to interpolate matrices * @param startValue defines the start matrix * @param endValue defines the end matrix * @param gradient defines the gradient between both matrices * @param result defines an optional target matrix where to store the interpolation * @returns the interpolated matrix */ matrixInterpolateFunction(e, t, r, n) { return st.AllowMatrixDecomposeForInterpolation ? n ? (he.DecomposeLerpToRef(e, t, r, n), n) : he.DecomposeLerp(e, t, r) : n ? (he.LerpToRef(e, t, r, n), n) : he.Lerp(e, t, r); } /** * Makes a copy of the animation * @returns Cloned animation */ clone() { const e = new st(this.name, this.targetPropertyPath.join("."), this.framePerSecond, this.dataType, this.loopMode); if (e.enableBlending = this.enableBlending, e.blendingSpeed = this.blendingSpeed, this._keys && e.setKeys(this._keys), this._ranges) { e._ranges = {}; for (const t in this._ranges) { const r = this._ranges[t]; r && (e._ranges[t] = r.clone()); } } return e; } /** * Sets the key frames of the animation * @param values The animation key frames to set * @param dontClone Whether to clone the keys or not (default is false, so the array of keys is cloned) */ setKeys(e, t = !1) { this._keys = t ? e : e.slice(0); } /** * Creates a key for the frame passed as a parameter and adds it to the animation IF a key doesn't already exist for that frame * @param frame Frame number * @returns The key index if the key was added or the index of the pre existing key if the frame passed as parameter already has a corresponding key */ createKeyForFrame(e) { kH.key = 0; const t = this._interpolate(e, kH, !0); if (!t) return kH.key === e ? kH.key : kH.key + 1; const r = { frame: e, value: t.clone ? t.clone() : t }; return this._keys.splice(kH.key + 1, 0, r), kH.key + 1; } /** * Serializes the animation to an object * @returns Serialized object */ serialize() { const e = {}; e.name = this.name, e.property = this.targetProperty, e.framePerSecond = this.framePerSecond, e.dataType = this.dataType, e.loopBehavior = this.loopMode, e.enableBlending = this.enableBlending, e.blendingSpeed = this.blendingSpeed; const t = this.dataType; e.keys = []; const r = this.getKeys(); for (let n = 0; n < r.length; n++) { const i = r[n], s = {}; switch (s.frame = i.frame, t) { case st.ANIMATIONTYPE_FLOAT: s.values = [i.value], i.inTangent !== void 0 && s.values.push(i.inTangent), i.outTangent !== void 0 && (i.inTangent === void 0 && s.values.push(void 0), s.values.push(i.outTangent)), i.interpolation !== void 0 && (i.inTangent === void 0 && s.values.push(void 0), i.outTangent === void 0 && s.values.push(void 0), s.values.push(i.interpolation)); break; case st.ANIMATIONTYPE_QUATERNION: case st.ANIMATIONTYPE_MATRIX: case st.ANIMATIONTYPE_VECTOR3: case st.ANIMATIONTYPE_COLOR3: case st.ANIMATIONTYPE_COLOR4: s.values = i.value.asArray(), i.inTangent != null && s.values.push(i.inTangent.asArray()), i.outTangent != null && (i.inTangent === void 0 && s.values.push(void 0), s.values.push(i.outTangent.asArray())), i.interpolation !== void 0 && (i.inTangent === void 0 && s.values.push(void 0), i.outTangent === void 0 && s.values.push(void 0), s.values.push(i.interpolation)); break; } e.keys.push(s); } e.ranges = []; for (const n in this._ranges) { const i = this._ranges[n]; if (!i) continue; const s = {}; s.name = n, s.from = i.from, s.to = i.to, e.ranges.push(s); } return e; } /** * @internal */ static _UniversalLerp(e, t, r) { const n = e.constructor; return n.Lerp ? n.Lerp(e, t, r) : n.Slerp ? n.Slerp(e, t, r) : e.toFixed ? e * (1 - r) + r * t : t; } /** * Parses an animation object and creates an animation * @param parsedAnimation Parsed animation object * @returns Animation object */ static Parse(e) { const t = new st(e.name, e.property, e.framePerSecond, e.dataType, e.loopBehavior), r = e.dataType, n = []; let i, s; for (e.enableBlending && (t.enableBlending = e.enableBlending), e.blendingSpeed && (t.blendingSpeed = e.blendingSpeed), s = 0; s < e.keys.length; s++) { const a = e.keys[s]; let f, o, d; switch (r) { case st.ANIMATIONTYPE_FLOAT: i = a.values[0], a.values.length >= 2 && (f = a.values[1]), a.values.length >= 3 && (o = a.values[2]), a.values.length >= 4 && (d = a.values[3]); break; case st.ANIMATIONTYPE_QUATERNION: if (i = Ze.FromArray(a.values), a.values.length >= 8) { const u = Ze.FromArray(a.values.slice(4, 8)); u.equals(Ze.Zero()) || (f = u); } if (a.values.length >= 12) { const u = Ze.FromArray(a.values.slice(8, 12)); u.equals(Ze.Zero()) || (o = u); } a.values.length >= 13 && (d = a.values[12]); break; case st.ANIMATIONTYPE_MATRIX: i = he.FromArray(a.values), a.values.length >= 17 && (d = a.values[16]); break; case st.ANIMATIONTYPE_COLOR3: i = Ne.FromArray(a.values), a.values[3] && (f = Ne.FromArray(a.values[3])), a.values[4] && (o = Ne.FromArray(a.values[4])), a.values[5] && (d = a.values[5]); break; case st.ANIMATIONTYPE_COLOR4: i = xt.FromArray(a.values), a.values[4] && (f = xt.FromArray(a.values[4])), a.values[5] && (o = xt.FromArray(a.values[5])), a.values[6] && (d = xt.FromArray(a.values[6])); break; case st.ANIMATIONTYPE_VECTOR3: default: i = S.FromArray(a.values), a.values[3] && (f = S.FromArray(a.values[3])), a.values[4] && (o = S.FromArray(a.values[4])), a.values[5] && (d = a.values[5]); break; } const v = {}; v.frame = a.frame, v.value = i, f != null && (v.inTangent = f), o != null && (v.outTangent = o), d != null && (v.interpolation = d), n.push(v); } if (t.setKeys(n), e.ranges) for (s = 0; s < e.ranges.length; s++) i = e.ranges[s], t.createRange(i.name, i.from, i.to); return t; } /** * Appends the serialized animations from the source animations * @param source Source containing the animations * @param destination Target to store the animations */ static AppendSerializedAnimations(e, t) { jt.AppendSerializedAnimations(e, t); } /** * Creates a new animation or an array of animations from a snippet saved in a remote file * @param name defines the name of the animation to create (can be null or empty to use the one from the json data) * @param url defines the url to load from * @returns a promise that will resolve to the new animation or an array of animations */ static ParseFromFileAsync(e, t) { return new Promise((r, n) => { const i = new ho(); i.addEventListener("readystatechange", () => { if (i.readyState == 4) if (i.status == 200) { let s = JSON.parse(i.responseText); if (s.animations && (s = s.animations), s.length) { const a = []; for (const f of s) a.push(this.Parse(f)); r(a); } else { const a = this.Parse(s); e && (a.name = e), r(a); } } else n("Unable to load the animation"); }), i.open("GET", t), i.send(); }); } /** * Creates an animation or an array of animations from a snippet saved by the Inspector * @param snippetId defines the snippet to load * @returns a promise that will resolve to the new animation or a new array of animations */ static ParseFromSnippetAsync(e) { return new Promise((t, r) => { const n = new ho(); n.addEventListener("readystatechange", () => { if (n.readyState == 4) if (n.status == 200) { const i = JSON.parse(JSON.parse(n.responseText).jsonPayload); if (i.animations) { const s = JSON.parse(i.animations), a = []; for (const f of s.animations) { const o = this.Parse(f); o.snippetId = e, a.push(o); } t(a); } else { const s = JSON.parse(i.animation), a = this.Parse(s); a.snippetId = e, t(a); } } else r("Unable to load the snippet " + e); }), n.open("GET", this.SnippetUrl + "/" + e.replace(/#/g, "/")), n.send(); }); } } st._UniqueIdGenerator = 0; st.AllowMatricesInterpolation = !1; st.AllowMatrixDecomposeForInterpolation = !0; st.SnippetUrl = "https://snippet.babylonjs.com"; st.ANIMATIONTYPE_FLOAT = 0; st.ANIMATIONTYPE_VECTOR3 = 1; st.ANIMATIONTYPE_QUATERNION = 2; st.ANIMATIONTYPE_MATRIX = 3; st.ANIMATIONTYPE_COLOR3 = 4; st.ANIMATIONTYPE_COLOR4 = 7; st.ANIMATIONTYPE_VECTOR2 = 5; st.ANIMATIONTYPE_SIZE = 6; st.ANIMATIONLOOPMODE_RELATIVE = 0; st.ANIMATIONLOOPMODE_CYCLE = 1; st.ANIMATIONLOOPMODE_CONSTANT = 2; st.ANIMATIONLOOPMODE_YOYO = 4; st.ANIMATIONLOOPMODE_RELATIVE_FROM_CURRENT = 5; st.CreateFromSnippetAsync = st.ParseFromSnippetAsync; Ue("BABYLON.Animation", st); Cs._AnimationRangeFactory = (A, e, t) => new Am(A, e, t); class I$ extends wa { /** * Instantiate the action * @param triggerOptions defines the trigger options * @param target defines the object containing the value to interpolate * @param propertyPath defines the path to the property in the target object * @param value defines the target value at the end of the interpolation * @param duration defines the time it will take for the property to interpolate to the value. * @param condition defines the trigger related conditions * @param stopOtherAnimations defines if the other scene animations should be stopped when the action has been triggered * @param onInterpolationDone defines a callback raised once the interpolation animation has been done */ constructor(e, t, r, n, i = 1e3, s, a, f) { super(e, s), this.duration = 1e3, this.onInterpolationDoneObservable = new Oe(), this.propertyPath = r, this.value = n, this.duration = i, this.stopOtherAnimations = a, this.onInterpolationDone = f, this._target = this._effectiveTarget = t; } /** @internal */ _prepare() { this._effectiveTarget = this._getEffectiveTarget(this._effectiveTarget, this.propertyPath), this._property = this._getProperty(this.propertyPath); } /** * Execute the action starts the value interpolation. */ execute() { const e = this._actionManager.getScene(), t = [ { frame: 0, value: this._effectiveTarget[this._property] }, { frame: 100, value: this.value } ]; let r; if (typeof this.value == "number") r = st.ANIMATIONTYPE_FLOAT; else if (this.value instanceof Ne) r = st.ANIMATIONTYPE_COLOR3; else if (this.value instanceof S) r = st.ANIMATIONTYPE_VECTOR3; else if (this.value instanceof he) r = st.ANIMATIONTYPE_MATRIX; else if (this.value instanceof Ze) r = st.ANIMATIONTYPE_QUATERNION; else { Se.Warn("InterpolateValueAction: Unsupported type (" + typeof this.value + ")"); return; } const n = new st("InterpolateValueAction", this._property, 100 * (1e3 / this.duration), r, st.ANIMATIONLOOPMODE_CONSTANT); n.setKeys(t), this.stopOtherAnimations && e.stopAnimation(this._effectiveTarget); const i = () => { this.onInterpolationDoneObservable.notifyObservers(this), this.onInterpolationDone && this.onInterpolationDone(); }; e.beginDirectAnimation(this._effectiveTarget, [n], 0, 100, !1, 1, i); } /** * Serializes the actions and its related information. * @param parent defines the object to serialize in * @returns the serialized object */ serialize(e) { return super._serialize({ name: "InterpolateValueAction", properties: [ wa._GetTargetProperty(this._target), { name: "propertyPath", value: this.propertyPath }, { name: "value", value: wa._SerializeValueAsString(this.value) }, { name: "duration", value: wa._SerializeValueAsString(this.duration) }, { name: "stopOtherAnimations", value: wa._SerializeValueAsString(this.stopOtherAnimations) || !1 } ] }, e); } } Ue("BABYLON.InterpolateValueAction", I$); class R$ { /** * Gets the current frame of the runtime animation */ get currentFrame() { return this._currentFrame; } /** * Gets the weight of the runtime animation */ get weight() { return this._weight; } /** * Gets the current value of the runtime animation */ get currentValue() { return this._currentValue; } /** * Gets or sets the target path of the runtime animation */ get targetPath() { return this._targetPath; } /** * Gets the actual target of the runtime animation */ get target() { return this._currentActiveTarget; } /** * Gets the additive state of the runtime animation */ get isAdditive() { return this._host && this._host.isAdditive; } /** * Create a new RuntimeAnimation object * @param target defines the target of the animation * @param animation defines the source animation object * @param scene defines the hosting scene * @param host defines the initiating Animatable */ constructor(e, t, r, n) { if (this._events = new Array(), this._currentFrame = 0, this._originalValue = new Array(), this._originalBlendValue = null, this._offsetsCache = {}, this._highLimitsCache = {}, this._stopped = !1, this._blendingFactor = 0, this._currentValue = null, this._currentActiveTarget = null, this._directTarget = null, this._targetPath = "", this._weight = 1, this._absoluteFrameOffset = 0, this._previousElapsedTime = 0, this._previousAbsoluteFrame = 0, this._targetIsArray = !1, this._animation = t, this._target = e, this._scene = r, this._host = n, this._activeTargets = [], t._runtimeAnimations.push(this), this._animationState = { key: 0, repeatCount: 0, loopMode: this._getCorrectLoopMode() }, this._animation.dataType === st.ANIMATIONTYPE_MATRIX && (this._animationState.workValue = he.Zero()), this._keys = this._animation.getKeys(), this._minFrame = this._keys[0].frame, this._maxFrame = this._keys[this._keys.length - 1].frame, this._minValue = this._keys[0].value, this._maxValue = this._keys[this._keys.length - 1].value, this._minFrame !== 0) { const s = { frame: 0, value: this._minValue }; this._keys.splice(0, 0, s); } if (this._target instanceof Array) { let s = 0; for (const a of this._target) this._preparePath(a, s), this._getOriginalValues(s), s++; this._targetIsArray = !0; } else this._preparePath(this._target), this._getOriginalValues(), this._targetIsArray = !1, this._directTarget = this._activeTargets[0]; const i = t.getEvents(); i && i.length > 0 && i.forEach((s) => { this._events.push(s._clone()); }), this._enableBlending = e && e.animationPropertiesOverride ? e.animationPropertiesOverride.enableBlending : this._animation.enableBlending; } _preparePath(e, t = 0) { const r = this._animation.targetPropertyPath; if (r.length > 1) { let n = e[r[0]]; for (let i = 1; i < r.length - 1; i++) n = n[r[i]]; this._targetPath = r[r.length - 1], this._activeTargets[t] = n; } else this._targetPath = r[0], this._activeTargets[t] = e; } /** * Gets the animation from the runtime animation */ get animation() { return this._animation; } /** * Resets the runtime animation to the beginning * @param restoreOriginal defines whether to restore the target property to the original value */ reset(e = !1) { if (e) if (this._target instanceof Array) { let t = 0; for (const r of this._target) this._originalValue[t] !== void 0 && this._setValue(r, this._activeTargets[t], this._originalValue[t], -1, t), t++; } else this._originalValue[0] !== void 0 && this._setValue(this._target, this._directTarget, this._originalValue[0], -1, 0); this._offsetsCache = {}, this._highLimitsCache = {}, this._currentFrame = 0, this._blendingFactor = 0; for (let t = 0; t < this._events.length; t++) this._events[t].isDone = !1; } /** * Specifies if the runtime animation is stopped * @returns Boolean specifying if the runtime animation is stopped */ isStopped() { return this._stopped; } /** * Disposes of the runtime animation */ dispose() { const e = this._animation.runtimeAnimations.indexOf(this); e > -1 && this._animation.runtimeAnimations.splice(e, 1); } /** * Apply the interpolated value to the target * @param currentValue defines the value computed by the animation * @param weight defines the weight to apply to this value (Defaults to 1.0) */ setValue(e, t) { if (this._targetIsArray) { for (let r = 0; r < this._target.length; r++) { const n = this._target[r]; this._setValue(n, this._activeTargets[r], e, t, r); } return; } this._setValue(this._target, this._directTarget, e, t, 0); } _getOriginalValues(e = 0) { let t; const r = this._activeTargets[e]; r.getLocalMatrix && this._targetPath === "_matrix" ? t = r.getLocalMatrix() : t = r[this._targetPath], t && t.clone ? this._originalValue[e] = t.clone() : this._originalValue[e] = t; } _setValue(e, t, r, n, i) { if (this._currentActiveTarget = t, this._weight = n, this._enableBlending && this._blendingFactor <= 1) { if (!this._originalBlendValue) { const a = t[this._targetPath]; a.clone ? this._originalBlendValue = a.clone() : this._originalBlendValue = a; } this._originalBlendValue.m ? st.AllowMatrixDecomposeForInterpolation ? this._currentValue ? he.DecomposeLerpToRef(this._originalBlendValue, r, this._blendingFactor, this._currentValue) : this._currentValue = he.DecomposeLerp(this._originalBlendValue, r, this._blendingFactor) : this._currentValue ? he.LerpToRef(this._originalBlendValue, r, this._blendingFactor, this._currentValue) : this._currentValue = he.Lerp(this._originalBlendValue, r, this._blendingFactor) : this._currentValue = st._UniversalLerp(this._originalBlendValue, r, this._blendingFactor); const s = e && e.animationPropertiesOverride ? e.animationPropertiesOverride.blendingSpeed : this._animation.blendingSpeed; this._blendingFactor += s; } else this._currentValue ? this._currentValue.copyFrom ? this._currentValue.copyFrom(r) : this._currentValue = r : r != null && r.clone ? this._currentValue = r.clone() : this._currentValue = r; n !== -1 ? this._scene._registerTargetForLateAnimationBinding(this, this._originalValue[i]) : this._animationState.loopMode === st.ANIMATIONLOOPMODE_RELATIVE_FROM_CURRENT ? this._currentValue.addToRef ? this._currentValue.addToRef(this._originalValue[i], t[this._targetPath]) : t[this._targetPath] = this._originalValue[i] + this._currentValue : t[this._targetPath] = this._currentValue, e.markAsDirty && e.markAsDirty(this._animation.targetProperty); } /** * Gets the loop pmode of the runtime animation * @returns Loop Mode */ _getCorrectLoopMode() { return this._target && this._target.animationPropertiesOverride ? this._target.animationPropertiesOverride.loopMode : this._animation.loopMode; } /** * Move the current animation to a given frame * @param frame defines the frame to move to */ goToFrame(e) { const t = this._animation.getKeys(); e < t[0].frame ? e = t[0].frame : e > t[t.length - 1].frame && (e = t[t.length - 1].frame); const r = this._events; if (r.length) for (let i = 0; i < r.length; i++) r[i].onlyOnce || (r[i].isDone = r[i].frame < e); this._currentFrame = e; const n = this._animation._interpolate(e, this._animationState); this.setValue(n, -1); } /** * @internal Internal use only */ _prepareForSpeedRatioChange(e) { const t = this._previousElapsedTime * (this._animation.framePerSecond * e) / 1e3; this._absoluteFrameOffset = this._previousAbsoluteFrame - t; } /** * Execute the current animation * @param elapsedTimeSinceAnimationStart defines the elapsed time (in milliseconds) since the animation was started * @param from defines the lower frame of the animation range * @param to defines the upper frame of the animation range * @param loop defines if the current animation must loop * @param speedRatio defines the current speed ratio * @param weight defines the weight of the animation (default is -1 so no weight) * @returns a boolean indicating if the animation is running */ animate(e, t, r, n, i, s = -1) { const a = this._animation, f = a.targetPropertyPath; if (!f || f.length < 1) return this._stopped = !0, !1; let o = !0; (t < this._minFrame || t > this._maxFrame) && (t = this._minFrame), (r < this._minFrame || r > this._maxFrame) && (r = this._maxFrame); const d = r - t; let v, u = e * (a.framePerSecond * i) / 1e3 + this._absoluteFrameOffset, l = 0; if (n && this._animationState.loopMode === st.ANIMATIONLOOPMODE_YOYO) { const H = (u - t) / d; u = Math.abs(Math.sin(H * Math.PI)) * d + t; } if (this._previousElapsedTime = e, this._previousAbsoluteFrame = u, !n && r >= t && u >= d) o = !1, l = a._getKeyValue(this._maxValue); else if (!n && t >= r && u <= d) o = !1, l = a._getKeyValue(this._minValue); else if (this._animationState.loopMode !== st.ANIMATIONLOOPMODE_CYCLE) { const H = r.toString() + t.toString(); if (!this._offsetsCache[H]) { this._animationState.repeatCount = 0, this._animationState.loopMode = st.ANIMATIONLOOPMODE_CYCLE; const T = a._interpolate(t, this._animationState), q = a._interpolate(r, this._animationState); switch (this._animationState.loopMode = this._getCorrectLoopMode(), a.dataType) { case st.ANIMATIONTYPE_FLOAT: this._offsetsCache[H] = q - T; break; case st.ANIMATIONTYPE_QUATERNION: this._offsetsCache[H] = q.subtract(T); break; case st.ANIMATIONTYPE_VECTOR3: this._offsetsCache[H] = q.subtract(T); break; case st.ANIMATIONTYPE_VECTOR2: this._offsetsCache[H] = q.subtract(T); break; case st.ANIMATIONTYPE_SIZE: this._offsetsCache[H] = q.subtract(T); break; case st.ANIMATIONTYPE_COLOR3: this._offsetsCache[H] = q.subtract(T); break; } this._highLimitsCache[H] = q; } l = this._highLimitsCache[H], v = this._offsetsCache[H]; } if (v === void 0) switch (a.dataType) { case st.ANIMATIONTYPE_FLOAT: v = 0; break; case st.ANIMATIONTYPE_QUATERNION: v = ON; break; case st.ANIMATIONTYPE_VECTOR3: v = yN; break; case st.ANIMATIONTYPE_VECTOR2: v = kN; break; case st.ANIMATIONTYPE_SIZE: v = EN; break; case st.ANIMATIONTYPE_COLOR3: v = FN; break; case st.ANIMATIONTYPE_COLOR4: v = NN; break; } let P; if (this._host && this._host.syncRoot) { const H = this._host.syncRoot, T = (H.masterFrame - H.fromFrame) / (H.toFrame - H.fromFrame); P = t + d * T; } else u > 0 && t > r || u < 0 && t < r ? P = o && d !== 0 ? r + u % d : t : P = o && d !== 0 ? t + u % d : r; const p = this._events; if (i > 0 && this.currentFrame > P || i < 0 && this.currentFrame < P) { this._onLoop(); for (let H = 0; H < p.length; H++) p[H].onlyOnce || (p[H].isDone = !1); this._animationState.key = i > 0 ? 0 : a.getKeys().length - 1; } this._currentFrame = P, this._animationState.repeatCount = d === 0 ? 0 : u / d >> 0, this._animationState.highLimitValue = l, this._animationState.offsetValue = v; const c = a._interpolate(P, this._animationState); if (this.setValue(c, s), p.length) { for (let H = 0; H < p.length; H++) if (d > 0 && P >= p[H].frame && p[H].frame >= t || d < 0 && P <= p[H].frame && p[H].frame <= t) { const T = p[H]; T.isDone || (T.onlyOnce && (p.splice(H, 1), H--), T.isDone = !0, T.action(P)); } } return o || (this._stopped = !0), o; } } function u9() { return typeof window < "u"; } function Mw() { return typeof navigator < "u"; } function $w() { return typeof document < "u"; } function gR(A) { let e = "", t = A.firstChild; for (; t; ) t.nodeType === 3 && (e += t.textContent), t = t.nextSibling; return e; } const I9e = { /** * Checks if the window object exists * @returns true if the window object exists */ IsWindowObjectExist: u9, /** * Checks if the navigator object exists * @returns true if the navigator object exists */ IsNavigatorAvailable: Mw, /** * Check if the document object exists * @returns true if the document object exists */ IsDocumentAvailable: $w, /** * Extracts text content from a DOM element hierarchy * @param element defines the root element * @returns a string */ GetDOMTextContent: gR }; class Yi { /** * Gets either window.performance.now() if supported or Date.now() else */ static get Now() { return u9() && window.performance && window.performance.now ? window.performance.now() : Date.now(); } } class em { } em.FilesToLoad = {}; class V$ { /** * Function used to defines an exponential back off strategy * @param maxRetries defines the maximum number of retries (3 by default) * @param baseInterval defines the interval between retries * @returns the strategy function to use */ static ExponentialBackoff(e = 3, t = 500) { return (r, n, i) => n.status !== 0 || i >= e || r.indexOf("file:") !== -1 ? -1 : Math.pow(2, i) * t; } } class dm extends Error { } dm._setPrototypeOf = Object.setPrototypeOf || ((A, e) => (A.__proto__ = e, A)); const Z2 = { // Mesh errors 0-999 /** Invalid or empty mesh vertex positions. */ MeshInvalidPositionsError: 0, // Texture errors 1000-1999 /** Unsupported texture found. */ UnsupportedTextureError: 1e3, // GLTFLoader errors 2000-2999 /** Unexpected magic number found in GLTF file header. */ GLTFLoaderUnexpectedMagicError: 2e3, // SceneLoader errors 3000-3999 /** SceneLoader generic error code. Ideally wraps the inner exception. */ SceneLoaderError: 3e3, // File related errors 4000-4999 /** Load file error */ LoadFileError: 4e3, /** Request file error */ RequestFileError: 4001, /** Read file error */ ReadFileError: 4002 }; class O0 extends dm { /** * Creates a new RuntimeError * @param message defines the message of the error * @param errorCode the error code * @param innerError the error that caused the outer error */ constructor(e, t, r) { super(e), this.errorCode = t, this.innerError = r, this.name = "RuntimeError", dm._setPrototypeOf(this, O0.prototype); } } const C$ = (A, e) => A.endsWith(e), O$ = (A, e) => A ? A.startsWith(e) : !1, QN = (A) => { if (typeof TextDecoder < "u") return new TextDecoder().decode(A); let e = ""; for (let t = 0; t < A.byteLength; t++) e += String.fromCharCode(A[t]); return e; }, XR = (A) => { const e = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/="; let t = "", r, n, i, s, a, f, o, d = 0; const v = ArrayBuffer.isView(A) ? new Uint8Array(A.buffer, A.byteOffset, A.byteLength) : new Uint8Array(A); for (; d < v.length; ) r = v[d++], n = d < v.length ? v[d++] : Number.NaN, i = d < v.length ? v[d++] : Number.NaN, s = r >> 2, a = (r & 3) << 4 | n >> 4, f = (n & 15) << 2 | i >> 6, o = i & 63, isNaN(n) ? f = o = 64 : isNaN(i) && (o = 64), t += e.charAt(s) + e.charAt(a) + e.charAt(f) + e.charAt(o); return t; }, bO = (A) => atob(A), TR = (A) => { const e = bO(A), t = e.length, r = new Uint8Array(new ArrayBuffer(t)); for (let n = 0; n < t; n++) r[n] = e.charCodeAt(n); return r.buffer; }, y$ = (A, e) => { let t = String(A); for (; t.length < e; ) t = "0" + t; return t; }, R9e = { EndsWith: C$, StartsWith: O$, Decode: QN, EncodeArrayBufferToBase64: XR, DecodeBase64ToString: bO, DecodeBase64ToBinary: TR, PadNumber: y$ }, V9e = "attribute", C9e = "varying"; class DI { constructor() { this.children = []; } // eslint-disable-next-line @typescript-eslint/no-unused-vars isValid(e) { return !0; } process(e, t) { var r, n, i, s, a, f, o; let d = ""; if (this.line) { let v = this.line; const u = t.processor; if (u) { u.lineProcessor && (v = u.lineProcessor(v, t.isFragment, t.processingContext)); const l = (n = (r = t.processor) === null || r === void 0 ? void 0 : r.attributeKeywordName) !== null && n !== void 0 ? n : V9e, P = t.isFragment && (!((i = t.processor) === null || i === void 0) && i.varyingFragmentKeywordName) ? (s = t.processor) === null || s === void 0 ? void 0 : s.varyingFragmentKeywordName : !t.isFragment && (!((a = t.processor) === null || a === void 0) && a.varyingVertexKeywordName) ? (f = t.processor) === null || f === void 0 ? void 0 : f.varyingVertexKeywordName : C9e; !t.isFragment && u.attributeProcessor && this.line.startsWith(l) ? v = u.attributeProcessor(this.line, e, t.processingContext) : u.varyingProcessor && (!((o = u.varyingCheck) === null || o === void 0) && o.call(u, this.line, t.isFragment) || !u.varyingCheck && this.line.startsWith(P)) ? v = u.varyingProcessor(this.line, t.isFragment, e, t.processingContext) : u.uniformProcessor && u.uniformRegexp && u.uniformRegexp.test(this.line) ? t.lookForClosingBracketForUniformBuffer || (v = u.uniformProcessor(this.line, t.isFragment, e, t.processingContext)) : u.uniformBufferProcessor && u.uniformBufferRegexp && u.uniformBufferRegexp.test(this.line) ? t.lookForClosingBracketForUniformBuffer || (v = u.uniformBufferProcessor(this.line, t.isFragment, t.processingContext), t.lookForClosingBracketForUniformBuffer = !0) : u.textureProcessor && u.textureRegexp && u.textureRegexp.test(this.line) ? v = u.textureProcessor(this.line, t.isFragment, e, t.processingContext) : (u.uniformProcessor || u.uniformBufferProcessor) && this.line.startsWith("uniform") && !t.lookForClosingBracketForUniformBuffer && (/uniform\s+(?:(?:highp)?|(?:lowp)?)\s*(\S+)\s+(\S+)\s*;/.test(this.line) ? u.uniformProcessor && (v = u.uniformProcessor(this.line, t.isFragment, e, t.processingContext)) : u.uniformBufferProcessor && (v = u.uniformBufferProcessor(this.line, t.isFragment, t.processingContext), t.lookForClosingBracketForUniformBuffer = !0)), t.lookForClosingBracketForUniformBuffer && this.line.indexOf("}") !== -1 && (t.lookForClosingBracketForUniformBuffer = !1, u.endOfUniformBufferProcessor && (v = u.endOfUniformBufferProcessor(this.line, t.isFragment, t.processingContext))); } d += v + ` `; } return this.children.forEach((v) => { d += v.process(e, t); }), this.additionalDefineKey && (e[this.additionalDefineKey] = this.additionalDefineValue || "true"), d; } } class O9e { constructor() { this._lines = []; } get currentLine() { return this._lines[this.lineIndex]; } get canRead() { return this.lineIndex < this._lines.length - 1; } set lines(e) { this._lines.length = 0; for (const t of e) { if (!t || t === "\r") continue; if (t[0] === "#") { this._lines.push(t); continue; } const r = t.trim(); if (!r) continue; if (r.startsWith("//")) { this._lines.push(t); continue; } const n = r.indexOf(";"); if (n === -1) this._lines.push(r); else if (n === r.length - 1) r.length > 1 && this._lines.push(r); else { const i = t.split(";"); for (let s = 0; s < i.length; s++) { let a = i[s]; a && (a = a.trim(), a && this._lines.push(a + (s !== i.length - 1 ? ";" : ""))); } } } } } class PE extends DI { process(e, t) { for (let r = 0; r < this.children.length; r++) { const n = this.children[r]; if (n.isValid(e)) return n.process(e, t); } return ""; } } class y9e extends DI { isValid(e) { return this.testExpression.isTrue(e); } } class M1 { // eslint-disable-next-line @typescript-eslint/no-unused-vars isTrue(e) { return !0; } static postfixToInfix(e) { const t = []; for (const r of e) if (M1._OperatorPriority[r] === void 0) t.push(r); else { const n = t[t.length - 1], i = t[t.length - 2]; t.length -= 2, t.push(`(${i}${r}${n})`); } return t[t.length - 1]; } /** * Converts an infix expression to a postfix expression. * * This method is used to transform infix expressions, which are more human-readable, * into postfix expressions, also known as Reverse Polish Notation (RPN), that can be * evaluated more efficiently by a computer. The conversion is based on the operator * priority defined in _OperatorPriority. * * The function employs a stack-based algorithm for the conversion and caches the result * to improve performance. The cache keeps track of each converted expression's access time * to manage the cache size and optimize memory usage. When the cache size exceeds a specified * limit, the least recently accessed items in the cache are deleted. * * The cache mechanism is particularly helpful for shader compilation, where the same infix * expressions might be encountered repeatedly, hence the caching can speed up the process. * * @param infix - The infix expression to be converted. * @returns The postfix expression as an array of strings. */ static infixToPostfix(e) { const t = M1._InfixToPostfixCache.get(e); if (t) return t.accessTime = Date.now(), t.result; if (!e.includes("&&") && !e.includes("||") && !e.includes(")") && !e.includes("(")) return [e]; const r = []; let n = -1; const i = () => { d = d.trim(), d !== "" && (r.push(d), d = ""); }, s = (v) => { n < M1._Stack.length - 1 && (M1._Stack[++n] = v); }, a = () => M1._Stack[n], f = () => n === -1 ? "!!INVALID EXPRESSION!!" : M1._Stack[n--]; let o = 0, d = ""; for (; o < e.length; ) { const v = e.charAt(o), u = o < e.length - 1 ? e.substr(o, 2) : ""; if (v === "(") d = "", s(v); else if (v === ")") { for (i(); n !== -1 && a() !== "("; ) r.push(f()); f(); } else if (M1._OperatorPriority[u] > 1) { for (i(); n !== -1 && M1._OperatorPriority[a()] >= M1._OperatorPriority[u]; ) r.push(f()); s(u), o++; } else d += v; o++; } for (i(); n !== -1; ) a() === "(" ? f() : r.push(f()); return M1._InfixToPostfixCache.size >= M1.InfixToPostfixCacheLimitSize && M1.ClearCache(), M1._InfixToPostfixCache.set(e, { result: r, accessTime: Date.now() }), r; } static ClearCache() { const e = Array.from(M1._InfixToPostfixCache.entries()).sort((t, r) => t[1].accessTime - r[1].accessTime); for (let t = 0; t < M1.InfixToPostfixCacheCleanupSize; t++) M1._InfixToPostfixCache.delete(e[t][0]); } } M1.InfixToPostfixCacheLimitSize = 5e4; M1.InfixToPostfixCacheCleanupSize = 25e3; M1._InfixToPostfixCache = /* @__PURE__ */ new Map(); M1._OperatorPriority = { ")": 0, "(": 1, "||": 2, "&&": 3 }; M1._Stack = ["", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", ""]; class uC extends M1 { constructor(e, t = !1) { super(), this.define = e, this.not = t; } isTrue(e) { let t = e[this.define] !== void 0; return this.not && (t = !t), t; } } class k9e extends M1 { isTrue(e) { return this.leftOperand.isTrue(e) || this.rightOperand.isTrue(e); } } class E9e extends M1 { isTrue(e) { return this.leftOperand.isTrue(e) && this.rightOperand.isTrue(e); } } class F9e extends M1 { constructor(e, t, r) { super(), this.define = e, this.operand = t, this.testValue = r; } isTrue(e) { let t = e[this.define]; t === void 0 && (t = this.define); let r = !1; const n = parseInt(t), i = parseInt(this.testValue); switch (this.operand) { case ">": r = n > i; break; case "<": r = n < i; break; case "<=": r = n <= i; break; case ">=": r = n >= i; break; case "==": r = n === i; break; case "!=": r = n !== i; break; } return r; } } var za; (function(A) { A[A.GLSL = 0] = "GLSL", A[A.WGSL = 1] = "WGSL"; })(za || (za = {})); const N9e = /defined\s*?\((.+?)\)/g, cE = /defined\s*?\[(.+?)\]/g, Q9e = /#include\s?<(.+)>(\((.*)\))*(\[(.*)\])*/g, Y9e = /__decl__/, oG = /light\{X\}.(\w*)/g, fG = /\{X\}/g, lC = []; class hp { static Initialize(e) { e.processor && e.processor.initializeShaders && e.processor.initializeShaders(e.processingContext); } static Process(e, t, r, n) { var i; !((i = t.processor) === null || i === void 0) && i.preProcessShaderCode && (e = t.processor.preProcessShaderCode(e, t.isFragment)), this._ProcessIncludes(e, t, (s) => { t.processCodeAfterIncludes && (s = t.processCodeAfterIncludes(t.isFragment ? "fragment" : "vertex", s)); const a = this._ProcessShaderConversion(s, t, n); r(a, s); }); } static PreProcess(e, t, r, n) { var i; !((i = t.processor) === null || i === void 0) && i.preProcessShaderCode && (e = t.processor.preProcessShaderCode(e, t.isFragment)), this._ProcessIncludes(e, t, (s) => { t.processCodeAfterIncludes && (s = t.processCodeAfterIncludes(t.isFragment ? "fragment" : "vertex", s)); const a = this._ApplyPreProcessing(s, t, n); r(a, s); }); } static Finalize(e, t, r) { return !r.processor || !r.processor.finalizeShaders ? { vertexCode: e, fragmentCode: t } : r.processor.finalizeShaders(e, t, r.processingContext); } static _ProcessPrecision(e, t) { var r; if (!((r = t.processor) === null || r === void 0) && r.noPrecision) return e; const n = t.shouldUseHighPrecisionShader; return e.indexOf("precision highp float") === -1 ? n ? e = `precision highp float; ` + e : e = `precision mediump float; ` + e : n || (e = e.replace("precision highp float", "precision mediump float")), e; } static _ExtractOperation(e) { const r = /defined\((.+)\)/.exec(e); if (r && r.length) return new uC(r[1].trim(), e[0] === "!"); const n = ["==", "!=", ">=", "<=", "<", ">"]; let i = "", s = 0; for (i of n) if (s = e.indexOf(i), s > -1) break; if (s === -1) return new uC(e); const a = e.substring(0, s).trim(), f = e.substring(s + i.length).trim(); return new F9e(a, i, f); } static _BuildSubExpression(e) { e = e.replace(N9e, "defined[$1]"); const t = M1.infixToPostfix(e), r = []; for (const i of t) if (i !== "||" && i !== "&&") r.push(i); else if (r.length >= 2) { let s = r[r.length - 1], a = r[r.length - 2]; r.length -= 2; const f = i == "&&" ? new E9e() : new k9e(); typeof s == "string" && (s = s.replace(cE, "defined($1)")), typeof a == "string" && (a = a.replace(cE, "defined($1)")), f.leftOperand = typeof a == "string" ? this._ExtractOperation(a) : a, f.rightOperand = typeof s == "string" ? this._ExtractOperation(s) : s, r.push(f); } let n = r[r.length - 1]; return typeof n == "string" && (n = n.replace(cE, "defined($1)")), typeof n == "string" ? this._ExtractOperation(n) : n; } static _BuildExpression(e, t) { const r = new y9e(), n = e.substring(0, t); let i = e.substring(t); return i = i.substring(0, (i.indexOf("//") + 1 || i.length + 1) - 1).trim(), n === "#ifdef" ? r.testExpression = new uC(i) : n === "#ifndef" ? r.testExpression = new uC(i, !0) : r.testExpression = this._BuildSubExpression(i), r; } static _MoveCursorWithinIf(e, t, r) { let n = e.currentLine; for (; this._MoveCursor(e, r); ) { n = e.currentLine; const i = n.substring(0, 5).toLowerCase(); if (i === "#else") { const s = new DI(); t.children.push(s), this._MoveCursor(e, s); return; } else if (i === "#elif") { const s = this._BuildExpression(n, 5); t.children.push(s), r = s; } } } static _MoveCursor(e, t) { for (; e.canRead; ) { e.lineIndex++; const r = e.currentLine; if (r.indexOf("#") >= 0) { const i = hp._MoveCursorRegex.exec(r); if (i && i.length) { switch (i[0]) { case "#ifdef": { const a = new PE(); t.children.push(a); const f = this._BuildExpression(r, 6); a.children.push(f), this._MoveCursorWithinIf(e, a, f); break; } case "#else": case "#elif": return !0; case "#endif": return !1; case "#ifndef": { const a = new PE(); t.children.push(a); const f = this._BuildExpression(r, 7); a.children.push(f), this._MoveCursorWithinIf(e, a, f); break; } case "#if": { const a = new PE(), f = this._BuildExpression(r, 3); t.children.push(a), a.children.push(f), this._MoveCursorWithinIf(e, a, f); break; } } continue; } } const n = new DI(); if (n.line = r, t.children.push(n), r[0] === "#" && r[1] === "d") { const i = r.replace(";", "").split(" "); n.additionalDefineKey = i[1], i.length === 3 && (n.additionalDefineValue = i[2]); } } return !1; } static _EvaluatePreProcessors(e, t, r) { const n = new DI(), i = new O9e(); return i.lineIndex = -1, i.lines = e.split(` `), this._MoveCursor(i, n), n.process(t, r); } static _PreparePreProcessors(e, t) { var r; const n = e.defines, i = {}; for (const s of n) { const f = s.replace("#define", "").replace(";", "").trim().split(" "); i[f[0]] = f.length > 1 ? f[1] : ""; } return ((r = e.processor) === null || r === void 0 ? void 0 : r.shaderLanguage) === za.GLSL && (i.GL_ES = "true"), i.__VERSION__ = e.version, i[e.platformName] = "true", t._getGlobalDefines(i), i; } static _ProcessShaderConversion(e, t, r) { let n = this._ProcessPrecision(e, t); if (!t.processor || t.processor.shaderLanguage === za.GLSL && n.indexOf("#version 3") !== -1 && (n = n.replace("#version 300 es", ""), !t.processor.parseGLES3)) return n; const i = t.defines, s = this._PreparePreProcessors(t, r); return t.processor.preProcessor && (n = t.processor.preProcessor(n, i, t.isFragment, t.processingContext)), n = this._EvaluatePreProcessors(n, s, t), t.processor.postProcessor && (n = t.processor.postProcessor(n, i, t.isFragment, t.processingContext, r)), r._features.needShaderCodeInlining && (n = r.inlineShaderCode(n)), n; } static _ApplyPreProcessing(e, t, r) { var n, i; let s = e; const a = t.defines, f = this._PreparePreProcessors(t, r); return !((n = t.processor) === null || n === void 0) && n.preProcessor && (s = t.processor.preProcessor(s, a, t.isFragment, t.processingContext)), s = this._EvaluatePreProcessors(s, f, t), !((i = t.processor) === null || i === void 0) && i.postProcessor && (s = t.processor.postProcessor(s, a, t.isFragment, t.processingContext, r)), r._features.needShaderCodeInlining && (s = r.inlineShaderCode(s)), s; } /** @internal */ static _ProcessIncludes(e, t, r) { lC.length = 0; let n; for (; (n = Q9e.exec(e)) !== null; ) lC.push(n); let i = String(e), s = [e], a = !1; for (const f of lC) { let o = f[1]; if (o.indexOf("__decl__") !== -1 && (o = o.replace(Y9e, ""), t.supportsUniformBuffers && (o = o.replace("Vertex", "Ubo").replace("Fragment", "Ubo")), o = o + "Declaration"), t.includesShadersStore[o]) { let d = t.includesShadersStore[o]; if (f[2]) { const u = f[3].split(","); for (let l = 0; l < u.length; l += 2) { const P = new RegExp(u[l], "g"), p = u[l + 1]; d = d.replace(P, p); } } if (f[4]) { const u = f[5]; if (u.indexOf("..") !== -1) { const l = u.split(".."), P = parseInt(l[0]); let p = parseInt(l[1]), c = d.slice(0); d = "", isNaN(p) && (p = t.indexParameters[l[1]]); for (let H = P; H < p; H++) t.supportsUniformBuffers || (c = c.replace(oG, (T, q) => q + "{X}")), d += c.replace(fG, H.toString()) + ` `; } else t.supportsUniformBuffers || (d = d.replace(oG, (l, P) => P + "{X}")), d = d.replace(fG, u); } const v = []; for (const u of s) { const l = u.split(f[0]); for (let P = 0; P < l.length - 1; P++) v.push(l[P]), v.push(d); v.push(l[l.length - 1]); } s = v, a = a || d.indexOf("#include<") >= 0 || d.indexOf("#include <") >= 0; } else { const d = t.shadersRepository + "ShadersInclude/" + o + ".fx"; hp._FileToolsLoadFile(d, (v) => { t.includesShadersStore[o] = v, this._ProcessIncludes(s.join(""), t, r); }); return; } } lC.length = 0, i = s.join(""), a ? this._ProcessIncludes(i.toString(), t, r) : r(i); } /** * Loads a file from a url * @param url url to load * @param onSuccess callback called when the file successfully loads * @param onProgress callback called while file is loading (if the server supports this mode) * @param offlineProvider defines the offline provider for caching * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer * @param onError callback called when the file fails to load * @returns a file request object * @internal */ static _FileToolsLoadFile(e, t, r, n, i, s) { throw qn("FileTools"); } } hp._MoveCursorRegex = /(#ifdef)|(#else)|(#elif)|(#endif)|(#ifndef)|(#if)/; class Le { /** * Gets the shaders repository path for a given shader language * @param shaderLanguage the shader language * @returns the path to the shaders repository */ static GetShadersRepository(e = za.GLSL) { return e === za.GLSL ? Le.ShadersRepository : Le.ShadersRepositoryWGSL; } /** * Gets the shaders store of a given shader language * @param shaderLanguage the shader language * @returns the shaders store */ static GetShadersStore(e = za.GLSL) { return e === za.GLSL ? Le.ShadersStore : Le.ShadersStoreWGSL; } /** * Gets the include shaders store of a given shader language * @param shaderLanguage the shader language * @returns the include shaders store */ static GetIncludesShadersStore(e = za.GLSL) { return e === za.GLSL ? Le.IncludesShadersStore : Le.IncludesShadersStoreWGSL; } } Le.ShadersRepository = "src/Shaders/"; Le.ShadersStore = {}; Le.IncludesShadersStore = {}; Le.ShadersRepositoryWGSL = "src/ShadersWGSL/"; Le.ShadersStoreWGSL = {}; Le.IncludesShadersStoreWGSL = {}; class An { /** * Gets or sets the relative url used to load shaders if using the engine in non-minified mode */ static get ShadersRepository() { return Le.ShadersRepository; } static set ShadersRepository(e) { Le.ShadersRepository = e; } /** * Observable that will be called when effect is bound. */ get onBindObservable() { return this._onBindObservable || (this._onBindObservable = new Oe()), this._onBindObservable; } /** * Instantiates an effect. * An effect can be used to create/manage/execute vertex and fragment shaders. * @param baseName Name of the effect. * @param attributesNamesOrOptions List of attribute names that will be passed to the shader or set of all options to create the effect. * @param uniformsNamesOrEngine List of uniform variable names that will be passed to the shader or the engine that will be used to render effect. * @param samplers List of sampler variables that will be passed to the shader. * @param engine Engine to be used to render the effect * @param defines Define statements to be added to the shader. * @param fallbacks Possible fallbacks for this effect to improve performance when needed. * @param onCompiled Callback that will be called when the shader is compiled. * @param onError Callback that will be called if an error occurs during shader compilation. * @param indexParameters Parameters to be used with Babylons include syntax to iterate over an array (eg. \{lights: 10\}) * @param key Effect Key identifying uniquely compiled shader variants * @param shaderLanguage the language the shader is written in (default: GLSL) */ constructor(e, t, r, n = null, i, s = null, a = null, f = null, o = null, d, v = "", u = za.GLSL) { var l, P, p; if (this.name = null, this.defines = "", this.onCompiled = null, this.onError = null, this.onBind = null, this.uniqueId = 0, this.onCompileObservable = new Oe(), this.onErrorObservable = new Oe(), this._onBindObservable = null, this._wasPreviouslyReady = !1, this._forceRebindOnNextCall = !1, this._wasPreviouslyUsingInstances = null, this._isDisposed = !1, this._bonesComputationForcedToCPU = !1, this._uniformBuffersNames = {}, this._multiTarget = !1, this._samplers = {}, this._isReady = !1, this._compilationError = "", this._allFallbacksProcessed = !1, this._uniforms = {}, this._key = "", this._fallbacks = null, this._vertexSourceCodeOverride = "", this._fragmentSourceCodeOverride = "", this._transformFeedbackVaryings = null, this._pipelineContext = null, this._vertexSourceCode = "", this._fragmentSourceCode = "", this._vertexSourceCodeBeforeMigration = "", this._fragmentSourceCodeBeforeMigration = "", this._rawVertexSourceCode = "", this._rawFragmentSourceCode = "", this._processCodeAfterIncludes = void 0, this._processFinalCode = null, this.name = e, this._key = v, t.attributes) { const c = t; if (this._engine = r, this._attributesNames = c.attributes, this._uniformsNames = c.uniformsNames.concat(c.samplers), this._samplerList = c.samplers.slice(), this.defines = c.defines, this.onError = c.onError, this.onCompiled = c.onCompiled, this._fallbacks = c.fallbacks, this._indexParameters = c.indexParameters, this._transformFeedbackVaryings = c.transformFeedbackVaryings || null, this._multiTarget = !!c.multiTarget, this._shaderLanguage = (l = c.shaderLanguage) !== null && l !== void 0 ? l : za.GLSL, c.uniformBuffersNames) { this._uniformBuffersNamesList = c.uniformBuffersNames.slice(); for (let H = 0; H < c.uniformBuffersNames.length; H++) this._uniformBuffersNames[c.uniformBuffersNames[H]] = H; } this._processFinalCode = (P = c.processFinalCode) !== null && P !== void 0 ? P : null, this._processCodeAfterIncludes = (p = c.processCodeAfterIncludes) !== null && p !== void 0 ? p : void 0; } else this._engine = i, this.defines = s ?? "", this._uniformsNames = r.concat(n), this._samplerList = n ? n.slice() : [], this._attributesNames = t, this._uniformBuffersNamesList = [], this._shaderLanguage = u, this.onError = o, this.onCompiled = f, this._indexParameters = d, this._fallbacks = a; this._attributeLocationByName = {}, this.uniqueId = An._UniqueIdSeed++, this._processShaderCode(); } /** @internal */ _processShaderCode(e = null, t = !1) { let r, n; const i = this.name, s = u9() ? this._engine.getHostDocument() : null; i.vertexSource ? r = "source:" + i.vertexSource : i.vertexElement ? (r = s ? s.getElementById(i.vertexElement) : null, r || (r = i.vertexElement)) : r = i.vertex || i, i.fragmentSource ? n = "source:" + i.fragmentSource : i.fragmentElement ? (n = s ? s.getElementById(i.fragmentElement) : null, n || (n = i.fragmentElement)) : n = i.fragment || i, this._processingContext = this._engine._getShaderProcessingContext(this._shaderLanguage); let a = { defines: this.defines.split(` `), indexParameters: this._indexParameters, isFragment: !1, shouldUseHighPrecisionShader: this._engine._shouldUseHighPrecisionShader, processor: e ?? this._engine._getShaderProcessor(this._shaderLanguage), supportsUniformBuffers: this._engine.supportsUniformBuffers, shadersRepository: Le.GetShadersRepository(this._shaderLanguage), includesShadersStore: Le.GetIncludesShadersStore(this._shaderLanguage), version: (this._engine.version * 100).toString(), platformName: this._engine.shaderPlatformName, processingContext: this._processingContext, isNDCHalfZRange: this._engine.isNDCHalfZRange, useReverseDepthBuffer: this._engine.useReverseDepthBuffer, processCodeAfterIncludes: this._processCodeAfterIncludes }; const f = [void 0, void 0], o = () => { if (f[0] && f[1]) { a.isFragment = !0; const [d, v] = f; hp.Process(v, a, (u, l) => { this._fragmentSourceCodeBeforeMigration = l, this._processFinalCode && (u = this._processFinalCode("fragment", u)); const P = hp.Finalize(d, u, a); a = null, this._useFinalCode(P.vertexCode, P.fragmentCode, i, t); }, this._engine); } }; this._loadShader(r, "Vertex", "", (d) => { hp.Initialize(a), hp.Process(d, a, (v, u) => { this._rawVertexSourceCode = d, this._vertexSourceCodeBeforeMigration = u, this._processFinalCode && (v = this._processFinalCode("vertex", v)), f[0] = v, o(); }, this._engine); }), this._loadShader(n, "Fragment", "Pixel", (d) => { this._rawFragmentSourceCode = d, f[1] = d, o(); }); } _useFinalCode(e, t, r, n = !1) { if (r) { const i = r.vertexElement || r.vertex || r.spectorName || r, s = r.fragmentElement || r.fragment || r.spectorName || r; this._vertexSourceCode = (this._shaderLanguage === za.WGSL ? "//" : "") + "#define SHADER_NAME vertex:" + i + ` ` + e, this._fragmentSourceCode = (this._shaderLanguage === za.WGSL ? "//" : "") + "#define SHADER_NAME fragment:" + s + ` ` + t; } else this._vertexSourceCode = e, this._fragmentSourceCode = t; this._prepareEffect(n); } /** * Unique key for this effect */ get key() { return this._key; } /** * If the effect has been compiled and prepared. * @returns if the effect is compiled and prepared. */ isReady() { try { return this._isReadyInternal(); } catch { return !1; } } _isReadyInternal() { return this._isReady ? !0 : this._pipelineContext ? this._pipelineContext.isReady : !1; } /** * The engine the effect was initialized with. * @returns the engine. */ getEngine() { return this._engine; } /** * The pipeline context for this effect * @returns the associated pipeline context */ getPipelineContext() { return this._pipelineContext; } /** * The set of names of attribute variables for the shader. * @returns An array of attribute names. */ getAttributesNames() { return this._attributesNames; } /** * Returns the attribute at the given index. * @param index The index of the attribute. * @returns The location of the attribute. */ getAttributeLocation(e) { return this._attributes[e]; } /** * Returns the attribute based on the name of the variable. * @param name of the attribute to look up. * @returns the attribute location. */ getAttributeLocationByName(e) { return this._attributeLocationByName[e]; } /** * The number of attributes. * @returns the number of attributes. */ getAttributesCount() { return this._attributes.length; } /** * Gets the index of a uniform variable. * @param uniformName of the uniform to look up. * @returns the index. */ getUniformIndex(e) { return this._uniformsNames.indexOf(e); } /** * Returns the attribute based on the name of the variable. * @param uniformName of the uniform to look up. * @returns the location of the uniform. */ getUniform(e) { return this._uniforms[e]; } /** * Returns an array of sampler variable names * @returns The array of sampler variable names. */ getSamplers() { return this._samplerList; } /** * Returns an array of uniform variable names * @returns The array of uniform variable names. */ getUniformNames() { return this._uniformsNames; } /** * Returns an array of uniform buffer variable names * @returns The array of uniform buffer variable names. */ getUniformBuffersNames() { return this._uniformBuffersNamesList; } /** * Returns the index parameters used to create the effect * @returns The index parameters object */ getIndexParameters() { return this._indexParameters; } /** * The error from the last compilation. * @returns the error string. */ getCompilationError() { return this._compilationError; } /** * Gets a boolean indicating that all fallbacks were used during compilation * @returns true if all fallbacks were used */ allFallbacksProcessed() { return this._allFallbacksProcessed; } /** * Adds a callback to the onCompiled observable and call the callback immediately if already ready. * @param func The callback to be used. */ executeWhenCompiled(e) { if (this.isReady()) { e(this); return; } this.onCompileObservable.add((t) => { e(t); }), (!this._pipelineContext || this._pipelineContext.isAsync) && setTimeout(() => { this._checkIsReady(null); }, 16); } _checkIsReady(e) { try { if (this._isReadyInternal()) return; } catch (t) { this._processCompilationErrors(t, e); return; } this._isDisposed || setTimeout(() => { this._checkIsReady(e); }, 16); } _loadShader(e, t, r, n) { if (typeof HTMLElement < "u" && e instanceof HTMLElement) { const a = gR(e); n(a); return; } if (e.substr(0, 7) === "source:") { n(e.substr(7)); return; } if (e.substr(0, 7) === "base64:") { const a = window.atob(e.substr(7)); n(a); return; } const i = Le.GetShadersStore(this._shaderLanguage); if (i[e + t + "Shader"]) { n(i[e + t + "Shader"]); return; } if (r && i[e + r + "Shader"]) { n(i[e + r + "Shader"]); return; } let s; e[0] === "." || e[0] === "/" || e.indexOf("http") > -1 ? s = e : s = Le.GetShadersRepository(this._shaderLanguage) + e, this._engine._loadFile(s + "." + t.toLowerCase() + ".fx", n); } /** * Gets the vertex shader source code of this effect * This is the final source code that will be compiled, after all the processing has been done (pre-processing applied, code injection/replacement, etc) */ get vertexSourceCode() { var e, t; return this._vertexSourceCodeOverride && this._fragmentSourceCodeOverride ? this._vertexSourceCodeOverride : (t = (e = this._pipelineContext) === null || e === void 0 ? void 0 : e._getVertexShaderCode()) !== null && t !== void 0 ? t : this._vertexSourceCode; } /** * Gets the fragment shader source code of this effect * This is the final source code that will be compiled, after all the processing has been done (pre-processing applied, code injection/replacement, etc) */ get fragmentSourceCode() { var e, t; return this._vertexSourceCodeOverride && this._fragmentSourceCodeOverride ? this._fragmentSourceCodeOverride : (t = (e = this._pipelineContext) === null || e === void 0 ? void 0 : e._getFragmentShaderCode()) !== null && t !== void 0 ? t : this._fragmentSourceCode; } /** * Gets the vertex shader source code before migration. * This is the source code after the include directives have been replaced by their contents but before the code is migrated, i.e. before ShaderProcess._ProcessShaderConversion is executed. * This method is, among other things, responsible for parsing #if/#define directives as well as converting GLES2 syntax to GLES3 (in the case of WebGL). */ get vertexSourceCodeBeforeMigration() { return this._vertexSourceCodeBeforeMigration; } /** * Gets the fragment shader source code before migration. * This is the source code after the include directives have been replaced by their contents but before the code is migrated, i.e. before ShaderProcess._ProcessShaderConversion is executed. * This method is, among other things, responsible for parsing #if/#define directives as well as converting GLES2 syntax to GLES3 (in the case of WebGL). */ get fragmentSourceCodeBeforeMigration() { return this._fragmentSourceCodeBeforeMigration; } /** * Gets the vertex shader source code before it has been modified by any processing */ get rawVertexSourceCode() { return this._rawVertexSourceCode; } /** * Gets the fragment shader source code before it has been modified by any processing */ get rawFragmentSourceCode() { return this._rawFragmentSourceCode; } /** * Recompiles the webGL program * @param vertexSourceCode The source code for the vertex shader. * @param fragmentSourceCode The source code for the fragment shader. * @param onCompiled Callback called when completed. * @param onError Callback called on error. * @internal */ _rebuildProgram(e, t, r, n) { this._isReady = !1, this._vertexSourceCodeOverride = e, this._fragmentSourceCodeOverride = t, this.onError = (i, s) => { n && n(s); }, this.onCompiled = () => { const i = this.getEngine().scenes; if (i) for (let s = 0; s < i.length; s++) i[s].markAllMaterialsAsDirty(63); this._pipelineContext._handlesSpectorRebuildCallback(r); }, this._fallbacks = null, this._prepareEffect(); } /** * Prepares the effect * @internal */ _prepareEffect(e = !1) { var t; const r = this._attributesNames, n = this.defines, i = this._pipelineContext; this._isReady = !1; try { const s = this._engine; this._pipelineContext = (t = e ? i : void 0) !== null && t !== void 0 ? t : s.createPipelineContext(this._processingContext), this._pipelineContext._name = this._key.replace(/\r/g, "").replace(/\n/g, "|"); const a = (f, o, d, v) => this._rebuildProgram(f, o, d, v); this._vertexSourceCodeOverride && this._fragmentSourceCodeOverride ? s._preparePipelineContext(this._pipelineContext, this._vertexSourceCodeOverride, this._fragmentSourceCodeOverride, !0, this._rawVertexSourceCode, this._rawFragmentSourceCode, a, null, this._transformFeedbackVaryings, this._key) : s._preparePipelineContext(this._pipelineContext, this._vertexSourceCode, this._fragmentSourceCode, !1, this._rawVertexSourceCode, this._rawFragmentSourceCode, a, n, this._transformFeedbackVaryings, this._key), s._executeWhenRenderingStateIsCompiled(this._pipelineContext, () => { if (this._attributes = [], this._pipelineContext._fillEffectInformation(this, this._uniformBuffersNames, this._uniformsNames, this._uniforms, this._samplerList, this._samplers, r, this._attributes), r) for (let f = 0; f < r.length; f++) { const o = r[f]; this._attributeLocationByName[o] = this._attributes[f]; } s.bindSamplers(this), this._compilationError = "", this._isReady = !0, this.onCompiled && this.onCompiled(this), this.onCompileObservable.notifyObservers(this), this.onCompileObservable.clear(), this._fallbacks && this._fallbacks.unBindMesh(), i && !e && this.getEngine()._deletePipelineContext(i); }), this._pipelineContext.isAsync && this._checkIsReady(i); } catch (s) { this._processCompilationErrors(s, i); } } _getShaderCodeAndErrorLine(e, t, r) { const n = r ? /FRAGMENT SHADER ERROR: 0:(\d+?):/ : /VERTEX SHADER ERROR: 0:(\d+?):/; let i = null; if (t && e) { const s = t.match(n); if (s && s.length === 2) { const a = parseInt(s[1]), f = e.split(` `, -1); f.length >= a && (i = `Offending line [${a}] in ${r ? "fragment" : "vertex"} code: ${f[a - 1]}`); } } return [e, i]; } _processCompilationErrors(e, t = null) { var r, n, i; this._compilationError = e.message; const s = this._attributesNames, a = this._fallbacks; if (Se.Error("Unable to compile effect:"), Se.Error("Uniforms: " + this._uniformsNames.map(function(o) { return " " + o; })), Se.Error("Attributes: " + s.map(function(o) { return " " + o; })), Se.Error(`Defines: ` + this.defines), An.LogShaderCodeOnCompilationError) { let o = null, d = null, v = null; !((r = this._pipelineContext) === null || r === void 0) && r._getVertexShaderCode() && ([v, o] = this._getShaderCodeAndErrorLine(this._pipelineContext._getVertexShaderCode(), this._compilationError, !1), v && (Se.Error("Vertex code:"), Se.Error(v))), !((n = this._pipelineContext) === null || n === void 0) && n._getFragmentShaderCode() && ([v, d] = this._getShaderCodeAndErrorLine((i = this._pipelineContext) === null || i === void 0 ? void 0 : i._getFragmentShaderCode(), this._compilationError, !0), v && (Se.Error("Fragment code:"), Se.Error(v))), o && Se.Error(o), d && Se.Error(d); } Se.Error("Error: " + this._compilationError); const f = () => { this.onError && this.onError(this, this._compilationError), this.onErrorObservable.notifyObservers(this); }; t && (this._pipelineContext = t, this._isReady = !0, f()), a ? (this._pipelineContext = null, a.hasMoreFallbacks ? (this._allFallbacksProcessed = !1, Se.Error("Trying next fallback."), this.defines = a.reduce(this.defines, this), this._prepareEffect()) : (this._allFallbacksProcessed = !0, f(), this.onErrorObservable.clear(), this._fallbacks && this._fallbacks.unBindMesh())) : (this._allFallbacksProcessed = !0, t || f()); } /** * Checks if the effect is supported. (Must be called after compilation) */ get isSupported() { return this._compilationError === ""; } /** * Binds a texture to the engine to be used as output of the shader. * @param channel Name of the output variable. * @param texture Texture to bind. * @internal */ _bindTexture(e, t) { this._engine._bindTexture(this._samplers[e], t, e); } /** * Sets a texture on the engine to be used in the shader. * @param channel Name of the sampler variable. * @param texture Texture to set. */ setTexture(e, t) { this._engine.setTexture(this._samplers[e], this._uniforms[e], t, e); } /** * Sets a depth stencil texture from a render target on the engine to be used in the shader. * @param channel Name of the sampler variable. * @param texture Texture to set. */ setDepthStencilTexture(e, t) { this._engine.setDepthStencilTexture(this._samplers[e], this._uniforms[e], t, e); } /** * Sets an array of textures on the engine to be used in the shader. * @param channel Name of the variable. * @param textures Textures to set. */ setTextureArray(e, t) { const r = e + "Ex"; if (this._samplerList.indexOf(r + "0") === -1) { const n = this._samplerList.indexOf(e); for (let s = 1; s < t.length; s++) { const a = r + (s - 1).toString(); this._samplerList.splice(n + s, 0, a); } let i = 0; for (const s of this._samplerList) this._samplers[s] = i, i += 1; } this._engine.setTextureArray(this._samplers[e], this._uniforms[e], t, e); } /** * Sets a texture to be the input of the specified post process. (To use the output, pass in the next post process in the pipeline) * @param channel Name of the sampler variable. * @param postProcess Post process to get the input texture from. */ setTextureFromPostProcess(e, t) { this._engine.setTextureFromPostProcess(this._samplers[e], t, e); } /** * (Warning! setTextureFromPostProcessOutput may be desired instead) * Sets the input texture of the passed in post process to be input of this effect. (To use the output of the passed in post process use setTextureFromPostProcessOutput) * @param channel Name of the sampler variable. * @param postProcess Post process to get the output texture from. */ setTextureFromPostProcessOutput(e, t) { this._engine.setTextureFromPostProcessOutput(this._samplers[e], t, e); } /** * Binds a buffer to a uniform. * @param buffer Buffer to bind. * @param name Name of the uniform variable to bind to. */ bindUniformBuffer(e, t) { const r = this._uniformBuffersNames[t]; r === void 0 || An._BaseCache[r] === e && this._engine._features.useUBOBindingCache || (An._BaseCache[r] = e, this._engine.bindUniformBufferBase(e, r, t)); } /** * Binds block to a uniform. * @param blockName Name of the block to bind. * @param index Index to bind. */ bindUniformBlock(e, t) { this._engine.bindUniformBlock(this._pipelineContext, e, t); } /** * Sets an integer value on a uniform variable. * @param uniformName Name of the variable. * @param value Value to be set. * @returns this effect. */ setInt(e, t) { return this._pipelineContext.setInt(e, t), this; } /** * Sets an int2 value on a uniform variable. * @param uniformName Name of the variable. * @param x First int in int2. * @param y Second int in int2. * @returns this effect. */ setInt2(e, t, r) { return this._pipelineContext.setInt2(e, t, r), this; } /** * Sets an int3 value on a uniform variable. * @param uniformName Name of the variable. * @param x First int in int3. * @param y Second int in int3. * @param z Third int in int3. * @returns this effect. */ setInt3(e, t, r, n) { return this._pipelineContext.setInt3(e, t, r, n), this; } /** * Sets an int4 value on a uniform variable. * @param uniformName Name of the variable. * @param x First int in int4. * @param y Second int in int4. * @param z Third int in int4. * @param w Fourth int in int4. * @returns this effect. */ setInt4(e, t, r, n, i) { return this._pipelineContext.setInt4(e, t, r, n, i), this; } /** * Sets an int array on a uniform variable. * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setIntArray(e, t) { return this._pipelineContext.setIntArray(e, t), this; } /** * Sets an int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setIntArray2(e, t) { return this._pipelineContext.setIntArray2(e, t), this; } /** * Sets an int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setIntArray3(e, t) { return this._pipelineContext.setIntArray3(e, t), this; } /** * Sets an int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setIntArray4(e, t) { return this._pipelineContext.setIntArray4(e, t), this; } /** * Sets an unsigned integer value on a uniform variable. * @param uniformName Name of the variable. * @param value Value to be set. * @returns this effect. */ setUInt(e, t) { return this._pipelineContext.setUInt(e, t), this; } /** * Sets an unsigned int2 value on a uniform variable. * @param uniformName Name of the variable. * @param x First unsigned int in uint2. * @param y Second unsigned int in uint2. * @returns this effect. */ setUInt2(e, t, r) { return this._pipelineContext.setUInt2(e, t, r), this; } /** * Sets an unsigned int3 value on a uniform variable. * @param uniformName Name of the variable. * @param x First unsigned int in uint3. * @param y Second unsigned int in uint3. * @param z Third unsigned int in uint3. * @returns this effect. */ setUInt3(e, t, r, n) { return this._pipelineContext.setUInt3(e, t, r, n), this; } /** * Sets an unsigned int4 value on a uniform variable. * @param uniformName Name of the variable. * @param x First unsigned int in uint4. * @param y Second unsigned int in uint4. * @param z Third unsigned int in uint4. * @param w Fourth unsigned int in uint4. * @returns this effect. */ setUInt4(e, t, r, n, i) { return this._pipelineContext.setUInt4(e, t, r, n, i), this; } /** * Sets an unsigned int array on a uniform variable. * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setUIntArray(e, t) { return this._pipelineContext.setUIntArray(e, t), this; } /** * Sets an unsigned int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setUIntArray2(e, t) { return this._pipelineContext.setUIntArray2(e, t), this; } /** * Sets an unsigned int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setUIntArray3(e, t) { return this._pipelineContext.setUIntArray3(e, t), this; } /** * Sets an unsigned int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setUIntArray4(e, t) { return this._pipelineContext.setUIntArray4(e, t), this; } /** * Sets an float array on a uniform variable. * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setFloatArray(e, t) { return this._pipelineContext.setArray(e, t), this; } /** * Sets an float array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setFloatArray2(e, t) { return this._pipelineContext.setArray2(e, t), this; } /** * Sets an float array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setFloatArray3(e, t) { return this._pipelineContext.setArray3(e, t), this; } /** * Sets an float array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setFloatArray4(e, t) { return this._pipelineContext.setArray4(e, t), this; } /** * Sets an array on a uniform variable. * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setArray(e, t) { return this._pipelineContext.setArray(e, t), this; } /** * Sets an array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setArray2(e, t) { return this._pipelineContext.setArray2(e, t), this; } /** * Sets an array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setArray3(e, t) { return this._pipelineContext.setArray3(e, t), this; } /** * Sets an array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setArray4(e, t) { return this._pipelineContext.setArray4(e, t), this; } /** * Sets matrices on a uniform variable. * @param uniformName Name of the variable. * @param matrices matrices to be set. * @returns this effect. */ setMatrices(e, t) { return this._pipelineContext.setMatrices(e, t), this; } /** * Sets matrix on a uniform variable. * @param uniformName Name of the variable. * @param matrix matrix to be set. * @returns this effect. */ setMatrix(e, t) { return this._pipelineContext.setMatrix(e, t), this; } /** * Sets a 3x3 matrix on a uniform variable. (Specified as [1,2,3,4,5,6,7,8,9] will result in [1,2,3][4,5,6][7,8,9] matrix) * @param uniformName Name of the variable. * @param matrix matrix to be set. * @returns this effect. */ setMatrix3x3(e, t) { return this._pipelineContext.setMatrix3x3(e, t), this; } /** * Sets a 2x2 matrix on a uniform variable. (Specified as [1,2,3,4] will result in [1,2][3,4] matrix) * @param uniformName Name of the variable. * @param matrix matrix to be set. * @returns this effect. */ setMatrix2x2(e, t) { return this._pipelineContext.setMatrix2x2(e, t), this; } /** * Sets a float on a uniform variable. * @param uniformName Name of the variable. * @param value value to be set. * @returns this effect. */ setFloat(e, t) { return this._pipelineContext.setFloat(e, t), this; } /** * Sets a boolean on a uniform variable. * @param uniformName Name of the variable. * @param bool value to be set. * @returns this effect. */ setBool(e, t) { return this._pipelineContext.setInt(e, t ? 1 : 0), this; } /** * Sets a Vector2 on a uniform variable. * @param uniformName Name of the variable. * @param vector2 vector2 to be set. * @returns this effect. */ setVector2(e, t) { return this._pipelineContext.setVector2(e, t), this; } /** * Sets a float2 on a uniform variable. * @param uniformName Name of the variable. * @param x First float in float2. * @param y Second float in float2. * @returns this effect. */ setFloat2(e, t, r) { return this._pipelineContext.setFloat2(e, t, r), this; } /** * Sets a Vector3 on a uniform variable. * @param uniformName Name of the variable. * @param vector3 Value to be set. * @returns this effect. */ setVector3(e, t) { return this._pipelineContext.setVector3(e, t), this; } /** * Sets a float3 on a uniform variable. * @param uniformName Name of the variable. * @param x First float in float3. * @param y Second float in float3. * @param z Third float in float3. * @returns this effect. */ setFloat3(e, t, r, n) { return this._pipelineContext.setFloat3(e, t, r, n), this; } /** * Sets a Vector4 on a uniform variable. * @param uniformName Name of the variable. * @param vector4 Value to be set. * @returns this effect. */ setVector4(e, t) { return this._pipelineContext.setVector4(e, t), this; } /** * Sets a Quaternion on a uniform variable. * @param uniformName Name of the variable. * @param quaternion Value to be set. * @returns this effect. */ setQuaternion(e, t) { return this._pipelineContext.setQuaternion(e, t), this; } /** * Sets a float4 on a uniform variable. * @param uniformName Name of the variable. * @param x First float in float4. * @param y Second float in float4. * @param z Third float in float4. * @param w Fourth float in float4. * @returns this effect. */ setFloat4(e, t, r, n, i) { return this._pipelineContext.setFloat4(e, t, r, n, i), this; } /** * Sets a Color3 on a uniform variable. * @param uniformName Name of the variable. * @param color3 Value to be set. * @returns this effect. */ setColor3(e, t) { return this._pipelineContext.setColor3(e, t), this; } /** * Sets a Color4 on a uniform variable. * @param uniformName Name of the variable. * @param color3 Value to be set. * @param alpha Alpha value to be set. * @returns this effect. */ setColor4(e, t, r) { return this._pipelineContext.setColor4(e, t, r), this; } /** * Sets a Color4 on a uniform variable * @param uniformName defines the name of the variable * @param color4 defines the value to be set * @returns this effect. */ setDirectColor4(e, t) { return this._pipelineContext.setDirectColor4(e, t), this; } /** * Release all associated resources. **/ dispose() { this._pipelineContext && this._pipelineContext.dispose(), this._engine._releaseEffect(this), this._isDisposed = !0; } /** * This function will add a new shader to the shader store * @param name the name of the shader * @param pixelShader optional pixel shader content * @param vertexShader optional vertex shader content * @param shaderLanguage the language the shader is written in (default: GLSL) */ static RegisterShader(e, t, r, n = za.GLSL) { t && (Le.GetShadersStore(n)[`${e}PixelShader`] = t), r && (Le.GetShadersStore(n)[`${e}VertexShader`] = r); } /** * Resets the cache of effects. */ static ResetCache() { An._BaseCache = {}; } } An.LogShaderCodeOnCompilationError = !0; An._UniqueIdSeed = 0; An._BaseCache = {}; An.ShadersStore = Le.ShadersStore; An.IncludesShadersStore = Le.IncludesShadersStore; class YN { /** * Initializes the state. * @param reset */ constructor(e = !0) { this._isDepthTestDirty = !1, this._isDepthMaskDirty = !1, this._isDepthFuncDirty = !1, this._isCullFaceDirty = !1, this._isCullDirty = !1, this._isZOffsetDirty = !1, this._isFrontFaceDirty = !1, e && this.reset(); } get isDirty() { return this._isDepthFuncDirty || this._isDepthTestDirty || this._isDepthMaskDirty || this._isCullFaceDirty || this._isCullDirty || this._isZOffsetDirty || this._isFrontFaceDirty; } get zOffset() { return this._zOffset; } set zOffset(e) { this._zOffset !== e && (this._zOffset = e, this._isZOffsetDirty = !0); } get zOffsetUnits() { return this._zOffsetUnits; } set zOffsetUnits(e) { this._zOffsetUnits !== e && (this._zOffsetUnits = e, this._isZOffsetDirty = !0); } get cullFace() { return this._cullFace; } set cullFace(e) { this._cullFace !== e && (this._cullFace = e, this._isCullFaceDirty = !0); } get cull() { return this._cull; } set cull(e) { this._cull !== e && (this._cull = e, this._isCullDirty = !0); } get depthFunc() { return this._depthFunc; } set depthFunc(e) { this._depthFunc !== e && (this._depthFunc = e, this._isDepthFuncDirty = !0); } get depthMask() { return this._depthMask; } set depthMask(e) { this._depthMask !== e && (this._depthMask = e, this._isDepthMaskDirty = !0); } get depthTest() { return this._depthTest; } set depthTest(e) { this._depthTest !== e && (this._depthTest = e, this._isDepthTestDirty = !0); } get frontFace() { return this._frontFace; } set frontFace(e) { this._frontFace !== e && (this._frontFace = e, this._isFrontFaceDirty = !0); } reset() { this._depthMask = !0, this._depthTest = !0, this._depthFunc = null, this._cullFace = null, this._cull = null, this._zOffset = 0, this._zOffsetUnits = 0, this._frontFace = null, this._isDepthTestDirty = !0, this._isDepthMaskDirty = !0, this._isDepthFuncDirty = !1, this._isCullFaceDirty = !1, this._isCullDirty = !1, this._isZOffsetDirty = !0, this._isFrontFaceDirty = !1; } apply(e) { this.isDirty && (this._isCullDirty && (this.cull ? e.enable(e.CULL_FACE) : e.disable(e.CULL_FACE), this._isCullDirty = !1), this._isCullFaceDirty && (e.cullFace(this.cullFace), this._isCullFaceDirty = !1), this._isDepthMaskDirty && (e.depthMask(this.depthMask), this._isDepthMaskDirty = !1), this._isDepthTestDirty && (this.depthTest ? e.enable(e.DEPTH_TEST) : e.disable(e.DEPTH_TEST), this._isDepthTestDirty = !1), this._isDepthFuncDirty && (e.depthFunc(this.depthFunc), this._isDepthFuncDirty = !1), this._isZOffsetDirty && (this.zOffset || this.zOffsetUnits ? (e.enable(e.POLYGON_OFFSET_FILL), e.polygonOffset(this.zOffset, this.zOffsetUnits)) : e.disable(e.POLYGON_OFFSET_FILL), this._isZOffsetDirty = !1), this._isFrontFaceDirty && (e.frontFace(this.frontFace), this._isFrontFaceDirty = !1)); } } class L2 { constructor() { this.reset(); } reset() { this.enabled = !1, this.mask = 255, this.func = L2.ALWAYS, this.funcRef = 1, this.funcMask = 255, this.opStencilFail = L2.KEEP, this.opDepthFail = L2.KEEP, this.opStencilDepthPass = L2.REPLACE; } get stencilFunc() { return this.func; } set stencilFunc(e) { this.func = e; } get stencilFuncRef() { return this.funcRef; } set stencilFuncRef(e) { this.funcRef = e; } get stencilFuncMask() { return this.funcMask; } set stencilFuncMask(e) { this.funcMask = e; } get stencilOpStencilFail() { return this.opStencilFail; } set stencilOpStencilFail(e) { this.opStencilFail = e; } get stencilOpDepthFail() { return this.opDepthFail; } set stencilOpDepthFail(e) { this.opDepthFail = e; } get stencilOpStencilDepthPass() { return this.opStencilDepthPass; } set stencilOpStencilDepthPass(e) { this.opStencilDepthPass = e; } get stencilMask() { return this.mask; } set stencilMask(e) { this.mask = e; } get stencilTest() { return this.enabled; } set stencilTest(e) { this.enabled = e; } } L2.ALWAYS = 519; L2.KEEP = 7680; L2.REPLACE = 7681; class k$ { /** * Initializes the state. */ constructor() { this._blendFunctionParameters = new Array(4), this._blendEquationParameters = new Array(2), this._blendConstants = new Array(4), this._isBlendConstantsDirty = !1, this._alphaBlend = !1, this._isAlphaBlendDirty = !1, this._isBlendFunctionParametersDirty = !1, this._isBlendEquationParametersDirty = !1, this.reset(); } get isDirty() { return this._isAlphaBlendDirty || this._isBlendFunctionParametersDirty || this._isBlendEquationParametersDirty; } get alphaBlend() { return this._alphaBlend; } set alphaBlend(e) { this._alphaBlend !== e && (this._alphaBlend = e, this._isAlphaBlendDirty = !0); } setAlphaBlendConstants(e, t, r, n) { this._blendConstants[0] === e && this._blendConstants[1] === t && this._blendConstants[2] === r && this._blendConstants[3] === n || (this._blendConstants[0] = e, this._blendConstants[1] = t, this._blendConstants[2] = r, this._blendConstants[3] = n, this._isBlendConstantsDirty = !0); } setAlphaBlendFunctionParameters(e, t, r, n) { this._blendFunctionParameters[0] === e && this._blendFunctionParameters[1] === t && this._blendFunctionParameters[2] === r && this._blendFunctionParameters[3] === n || (this._blendFunctionParameters[0] = e, this._blendFunctionParameters[1] = t, this._blendFunctionParameters[2] = r, this._blendFunctionParameters[3] = n, this._isBlendFunctionParametersDirty = !0); } setAlphaEquationParameters(e, t) { this._blendEquationParameters[0] === e && this._blendEquationParameters[1] === t || (this._blendEquationParameters[0] = e, this._blendEquationParameters[1] = t, this._isBlendEquationParametersDirty = !0); } reset() { this._alphaBlend = !1, this._blendFunctionParameters[0] = null, this._blendFunctionParameters[1] = null, this._blendFunctionParameters[2] = null, this._blendFunctionParameters[3] = null, this._blendEquationParameters[0] = null, this._blendEquationParameters[1] = null, this._blendConstants[0] = null, this._blendConstants[1] = null, this._blendConstants[2] = null, this._blendConstants[3] = null, this._isAlphaBlendDirty = !0, this._isBlendFunctionParametersDirty = !1, this._isBlendEquationParametersDirty = !1, this._isBlendConstantsDirty = !1; } apply(e) { this.isDirty && (this._isAlphaBlendDirty && (this._alphaBlend ? e.enable(e.BLEND) : e.disable(e.BLEND), this._isAlphaBlendDirty = !1), this._isBlendFunctionParametersDirty && (e.blendFuncSeparate(this._blendFunctionParameters[0], this._blendFunctionParameters[1], this._blendFunctionParameters[2], this._blendFunctionParameters[3]), this._isBlendFunctionParametersDirty = !1), this._isBlendEquationParametersDirty && (e.blendEquationSeparate(this._blendEquationParameters[0], this._blendEquationParameters[1]), this._isBlendEquationParametersDirty = !1), this._isBlendConstantsDirty && (e.blendColor(this._blendConstants[0], this._blendConstants[1], this._blendConstants[2], this._blendConstants[3]), this._isBlendConstantsDirty = !1)); } } class MN { /** * | Value | Type | Description | * | ----- | ------------------ | ----------- | * | 0 | CLAMP_ADDRESSMODE | | * | 1 | WRAP_ADDRESSMODE | | * | 2 | MIRROR_ADDRESSMODE | | */ get wrapU() { return this._cachedWrapU; } set wrapU(e) { this._cachedWrapU = e; } /** * | Value | Type | Description | * | ----- | ------------------ | ----------- | * | 0 | CLAMP_ADDRESSMODE | | * | 1 | WRAP_ADDRESSMODE | | * | 2 | MIRROR_ADDRESSMODE | | */ get wrapV() { return this._cachedWrapV; } set wrapV(e) { this._cachedWrapV = e; } /** * | Value | Type | Description | * | ----- | ------------------ | ----------- | * | 0 | CLAMP_ADDRESSMODE | | * | 1 | WRAP_ADDRESSMODE | | * | 2 | MIRROR_ADDRESSMODE | | */ get wrapR() { return this._cachedWrapR; } set wrapR(e) { this._cachedWrapR = e; } /** * With compliant hardware and browser (supporting anisotropic filtering) * this defines the level of anisotropic filtering in the texture. * The higher the better but the slower. */ get anisotropicFilteringLevel() { return this._cachedAnisotropicFilteringLevel; } set anisotropicFilteringLevel(e) { this._cachedAnisotropicFilteringLevel = e; } /** * Gets or sets the comparison function (513, 514, etc). Set 0 to not use a comparison function */ get comparisonFunction() { return this._comparisonFunction; } set comparisonFunction(e) { this._comparisonFunction = e; } /** * Indicates to use the mip maps (if available on the texture). * Thanks to this flag, you can instruct the sampler to not sample the mipmaps even if they exist (and if the sampling mode is set to a value that normally samples the mipmaps!) */ get useMipMaps() { return this._useMipMaps; } set useMipMaps(e) { this._useMipMaps = e; } /** * Creates a Sampler instance */ constructor() { this.samplingMode = -1, this._useMipMaps = !0, this._cachedWrapU = null, this._cachedWrapV = null, this._cachedWrapR = null, this._cachedAnisotropicFilteringLevel = null, this._comparisonFunction = 0; } /** * Sets all the parameters of the sampler * @param wrapU u address mode (default: TEXTURE_WRAP_ADDRESSMODE) * @param wrapV v address mode (default: TEXTURE_WRAP_ADDRESSMODE) * @param wrapR r address mode (default: TEXTURE_WRAP_ADDRESSMODE) * @param anisotropicFilteringLevel anisotropic level (default: 1) * @param samplingMode sampling mode (default: 2) * @param comparisonFunction comparison function (default: 0 - no comparison function) * @returns the current sampler instance */ setParameters(e = 1, t = 1, r = 1, n = 1, i = 2, s = 0) { return this._cachedWrapU = e, this._cachedWrapV = t, this._cachedWrapR = r, this._cachedAnisotropicFilteringLevel = n, this.samplingMode = i, this._comparisonFunction = s, this; } /** * Compares this sampler with another one * @param other sampler to compare with * @returns true if the samplers have the same parametres, else false */ compareSampler(e) { return this._cachedWrapU === e._cachedWrapU && this._cachedWrapV === e._cachedWrapV && this._cachedWrapR === e._cachedWrapR && this._cachedAnisotropicFilteringLevel === e._cachedAnisotropicFilteringLevel && this.samplingMode === e.samplingMode && this._comparisonFunction === e._comparisonFunction && this._useMipMaps === e._useMipMaps; } } var ri; (function(A) { A[A.Unknown = 0] = "Unknown", A[A.Url = 1] = "Url", A[A.Temp = 2] = "Temp", A[A.Raw = 3] = "Raw", A[A.Dynamic = 4] = "Dynamic", A[A.RenderTarget = 5] = "RenderTarget", A[A.MultiRenderTarget = 6] = "MultiRenderTarget", A[A.Cube = 7] = "Cube", A[A.CubeRaw = 8] = "CubeRaw", A[A.CubePrefiltered = 9] = "CubePrefiltered", A[A.Raw3D = 10] = "Raw3D", A[A.Raw2DArray = 11] = "Raw2DArray", A[A.DepthStencil = 12] = "DepthStencil", A[A.CubeRawRGBD = 13] = "CubeRawRGBD", A[A.Depth = 14] = "Depth"; })(ri || (ri = {})); class As extends MN { /** * Gets a boolean indicating if the texture uses mipmaps * TODO implements useMipMaps as a separate setting from generateMipMaps */ get useMipMaps() { return this.generateMipMaps; } set useMipMaps(e) { this.generateMipMaps = e; } /** Gets the unique id of the internal texture */ get uniqueId() { return this._uniqueId; } /** @internal */ _setUniqueId(e) { this._uniqueId = e; } /** * Gets the Engine the texture belongs to. * @returns The babylon engine */ getEngine() { return this._engine; } /** * Gets the data source type of the texture */ get source() { return this._source; } /** * Creates a new InternalTexture * @param engine defines the engine to use * @param source defines the type of data that will be used * @param delayAllocation if the texture allocation should be delayed (default: false) */ constructor(e, t, r = !1) { super(), this.isReady = !1, this.isCube = !1, this.is3D = !1, this.is2DArray = !1, this.isMultiview = !1, this.url = "", this.generateMipMaps = !1, this.samples = 0, this.type = -1, this.format = -1, this.onLoadedObservable = new Oe(), this.onErrorObservable = new Oe(), this.onRebuildCallback = null, this.width = 0, this.height = 0, this.depth = 0, this.baseWidth = 0, this.baseHeight = 0, this.baseDepth = 0, this.invertY = !1, this._invertVScale = !1, this._associatedChannel = -1, this._source = ri.Unknown, this._buffer = null, this._bufferView = null, this._bufferViewArray = null, this._bufferViewArrayArray = null, this._size = 0, this._extension = "", this._files = null, this._workingCanvas = null, this._workingContext = null, this._cachedCoordinatesMode = null, this._isDisabled = !1, this._compression = null, this._sphericalPolynomial = null, this._sphericalPolynomialPromise = null, this._sphericalPolynomialComputed = !1, this._lodGenerationScale = 0, this._lodGenerationOffset = 0, this._useSRGBBuffer = !1, this._lodTextureHigh = null, this._lodTextureMid = null, this._lodTextureLow = null, this._isRGBD = !1, this._linearSpecularLOD = !1, this._irradianceTexture = null, this._hardwareTexture = null, this._maxLodLevel = null, this._references = 1, this._gammaSpace = null, this._premulAlpha = !1, this._dynamicTextureSource = null, this._engine = e, this._source = t, this._uniqueId = As._Counter++, r || (this._hardwareTexture = e._createHardwareTexture()); } /** * Increments the number of references (ie. the number of Texture that point to it) */ incrementReferences() { this._references++; } /** * Change the size of the texture (not the size of the content) * @param width defines the new width * @param height defines the new height * @param depth defines the new depth (1 by default) */ updateSize(e, t, r = 1) { this._engine.updateTextureDimensions(this, e, t, r), this.width = e, this.height = t, this.depth = r, this.baseWidth = e, this.baseHeight = t, this.baseDepth = r, this._size = e * t * r; } /** @internal */ _rebuild() { var e; if (this.isReady = !1, this._cachedCoordinatesMode = null, this._cachedWrapU = null, this._cachedWrapV = null, this._cachedWrapR = null, this._cachedAnisotropicFilteringLevel = null, this.onRebuildCallback) { const r = this.onRebuildCallback(this), n = (i) => { i._swapAndDie(this, !1), this.isReady = r.isReady; }; r.isAsync ? r.proxy.then(n) : n(r.proxy); return; } let t; switch (this.source) { case ri.Temp: break; case ri.Url: t = this._engine.createTexture( (e = this._originalUrl) !== null && e !== void 0 ? e : this.url, !this.generateMipMaps, this.invertY, null, this.samplingMode, // Do not use Proxy here as it could be fully synchronous // and proxy would be undefined. (r) => { r._swapAndDie(this, !1), this.isReady = !0; }, null, this._buffer, void 0, this.format, this._extension, void 0, void 0, void 0, this._useSRGBBuffer ); return; case ri.Raw: t = this._engine.createRawTexture(this._bufferView, this.baseWidth, this.baseHeight, this.format, this.generateMipMaps, this.invertY, this.samplingMode, this._compression, this.type, void 0, this._useSRGBBuffer), t._swapAndDie(this, !1), this.isReady = !0; break; case ri.Raw3D: t = this._engine.createRawTexture3D(this._bufferView, this.baseWidth, this.baseHeight, this.baseDepth, this.format, this.generateMipMaps, this.invertY, this.samplingMode, this._compression, this.type), t._swapAndDie(this, !1), this.isReady = !0; break; case ri.Raw2DArray: t = this._engine.createRawTexture2DArray(this._bufferView, this.baseWidth, this.baseHeight, this.baseDepth, this.format, this.generateMipMaps, this.invertY, this.samplingMode, this._compression, this.type), t._swapAndDie(this, !1), this.isReady = !0; break; case ri.Dynamic: t = this._engine.createDynamicTexture(this.baseWidth, this.baseHeight, this.generateMipMaps, this.samplingMode), t._swapAndDie(this, !1), this._dynamicTextureSource && this._engine.updateDynamicTexture(this, this._dynamicTextureSource, this.invertY, this._premulAlpha, this.format, !0); break; case ri.Cube: t = this._engine.createCubeTexture(this.url, null, this._files, !this.generateMipMaps, () => { t._swapAndDie(this, !1), this.isReady = !0; }, null, this.format, this._extension, !1, 0, 0, null, void 0, this._useSRGBBuffer); return; case ri.CubeRaw: t = this._engine.createRawCubeTexture(this._bufferViewArray, this.width, this.format, this.type, this.generateMipMaps, this.invertY, this.samplingMode, this._compression), t._swapAndDie(this, !1), this.isReady = !0; break; case ri.CubeRawRGBD: return; case ri.CubePrefiltered: t = this._engine.createPrefilteredCubeTexture(this.url, null, this._lodGenerationScale, this._lodGenerationOffset, (r) => { r && r._swapAndDie(this, !1), this.isReady = !0; }, null, this.format, this._extension), t._sphericalPolynomial = this._sphericalPolynomial; return; } } /** * @internal */ _swapAndDie(e, t = !0) { var r; (r = this._hardwareTexture) === null || r === void 0 || r.setUsage(e._source, this.generateMipMaps, this.isCube, this.width, this.height), e._hardwareTexture = this._hardwareTexture, t && (e._isRGBD = this._isRGBD), this._lodTextureHigh && (e._lodTextureHigh && e._lodTextureHigh.dispose(), e._lodTextureHigh = this._lodTextureHigh), this._lodTextureMid && (e._lodTextureMid && e._lodTextureMid.dispose(), e._lodTextureMid = this._lodTextureMid), this._lodTextureLow && (e._lodTextureLow && e._lodTextureLow.dispose(), e._lodTextureLow = this._lodTextureLow), this._irradianceTexture && (e._irradianceTexture && e._irradianceTexture.dispose(), e._irradianceTexture = this._irradianceTexture); const n = this._engine.getLoadedTexturesCache(); let i = n.indexOf(this); i !== -1 && n.splice(i, 1), i = n.indexOf(e), i === -1 && n.push(e); } /** * Dispose the current allocated resources */ dispose() { this._references--, this.onLoadedObservable.clear(), this.onErrorObservable.clear(), this._references === 0 && (this._engine._releaseTexture(this), this._hardwareTexture = null, this._dynamicTextureSource = null); } } As._Counter = 0; class M9e { constructor() { this.shaderLanguage = za.GLSL; } postProcessor(e, t, r, n, i) { if (!i.getCaps().drawBuffersExtension) { const s = /#extension.+GL_EXT_draw_buffers.+(enable|require)/g; e = e.replace(s, ""); } return e; } } const L9e = /(flat\s)?\s*varying\s*.*/; class LN { constructor() { this.shaderLanguage = za.GLSL; } attributeProcessor(e) { return e.replace("attribute", "in"); } varyingCheck(e, t) { return L9e.test(e); } varyingProcessor(e, t) { return e.replace("varying", t ? "in" : "out"); } postProcessor(e, t, r) { const n = e.search(/#extension.+GL_EXT_draw_buffers.+require/) !== -1, i = /#extension.+(GL_OVR_multiview2|GL_OES_standard_derivatives|GL_EXT_shader_texture_lod|GL_EXT_frag_depth|GL_EXT_draw_buffers).+(enable|require)/g; if (e = e.replace(i, ""), e = e.replace(/texture2D\s*\(/g, "texture("), r) { const s = e.search(/layout *\(location *= *0\) *out/g) !== -1; e = e.replace(/texture2DLodEXT\s*\(/g, "textureLod("), e = e.replace(/textureCubeLodEXT\s*\(/g, "textureLod("), e = e.replace(/textureCube\s*\(/g, "texture("), e = e.replace(/gl_FragDepthEXT/g, "gl_FragDepth"), e = e.replace(/gl_FragColor/g, "glFragColor"), e = e.replace(/gl_FragData/g, "glFragData"), e = e.replace(/void\s+?main\s*\(/g, (n || s ? "" : `layout(location = 0) out vec4 glFragColor; `) + "void main("); } else if (t.indexOf("#define MULTIVIEW") !== -1) return `#extension GL_OVR_multiview2 : require layout (num_views = 2) in; ` + e; return e; } } class tg { /** * Gets the underlying buffer */ get underlyingResource() { return null; } /** * Constructs the buffer */ constructor() { this.references = 0, this.capacity = 0, this.is32Bits = !1, this.uniqueId = tg._Counter++; } } tg._Counter = 0; class jS extends tg { constructor(e) { super(), this._buffer = e; } get underlyingResource() { return this._buffer; } } class E$ { constructor() { this._valueCache = {}, this.vertexCompilationError = null, this.fragmentCompilationError = null, this.programLinkError = null, this.programValidationError = null, this._isDisposed = !1; } get isAsync() { return this.isParallelCompiled; } get isReady() { return this.program ? this.isParallelCompiled ? this.engine._isRenderingStateCompiled(this) : !0 : !1; } _handlesSpectorRebuildCallback(e) { e && this.program && e(this.program); } _fillEffectInformation(e, t, r, n, i, s, a, f) { const o = this.engine; if (o.supportsUniformBuffers) for (const u in t) e.bindUniformBlock(u, t[u]); this.engine.getUniforms(this, r).forEach((u, l) => { n[r[l]] = u; }), this._uniforms = n; let v; for (v = 0; v < i.length; v++) e.getUniform(i[v]) == null && (i.splice(v, 1), v--); i.forEach((u, l) => { s[u] = l; }); for (const u of o.getAttributes(this, a)) f.push(u); } /** * Release all associated resources. **/ dispose() { this._uniforms = {}, this._isDisposed = !0; } /** * @internal */ _cacheMatrix(e, t) { const r = this._valueCache[e], n = t.updateFlag; return r !== void 0 && r === n ? !1 : (this._valueCache[e] = n, !0); } /** * @internal */ _cacheFloat2(e, t, r) { let n = this._valueCache[e]; if (!n || n.length !== 2) return n = [t, r], this._valueCache[e] = n, !0; let i = !1; return n[0] !== t && (n[0] = t, i = !0), n[1] !== r && (n[1] = r, i = !0), i; } /** * @internal */ _cacheFloat3(e, t, r, n) { let i = this._valueCache[e]; if (!i || i.length !== 3) return i = [t, r, n], this._valueCache[e] = i, !0; let s = !1; return i[0] !== t && (i[0] = t, s = !0), i[1] !== r && (i[1] = r, s = !0), i[2] !== n && (i[2] = n, s = !0), s; } /** * @internal */ _cacheFloat4(e, t, r, n, i) { let s = this._valueCache[e]; if (!s || s.length !== 4) return s = [t, r, n, i], this._valueCache[e] = s, !0; let a = !1; return s[0] !== t && (s[0] = t, a = !0), s[1] !== r && (s[1] = r, a = !0), s[2] !== n && (s[2] = n, a = !0), s[3] !== i && (s[3] = i, a = !0), a; } /** * Sets an integer value on a uniform variable. * @param uniformName Name of the variable. * @param value Value to be set. */ setInt(e, t) { const r = this._valueCache[e]; r !== void 0 && r === t || this.engine.setInt(this._uniforms[e], t) && (this._valueCache[e] = t); } /** * Sets a int2 on a uniform variable. * @param uniformName Name of the variable. * @param x First int in int2. * @param y Second int in int2. */ setInt2(e, t, r) { this._cacheFloat2(e, t, r) && (this.engine.setInt2(this._uniforms[e], t, r) || (this._valueCache[e] = null)); } /** * Sets a int3 on a uniform variable. * @param uniformName Name of the variable. * @param x First int in int3. * @param y Second int in int3. * @param z Third int in int3. */ setInt3(e, t, r, n) { this._cacheFloat3(e, t, r, n) && (this.engine.setInt3(this._uniforms[e], t, r, n) || (this._valueCache[e] = null)); } /** * Sets a int4 on a uniform variable. * @param uniformName Name of the variable. * @param x First int in int4. * @param y Second int in int4. * @param z Third int in int4. * @param w Fourth int in int4. */ setInt4(e, t, r, n, i) { this._cacheFloat4(e, t, r, n, i) && (this.engine.setInt4(this._uniforms[e], t, r, n, i) || (this._valueCache[e] = null)); } /** * Sets an int array on a uniform variable. * @param uniformName Name of the variable. * @param array array to be set. */ setIntArray(e, t) { this._valueCache[e] = null, this.engine.setIntArray(this._uniforms[e], t); } /** * Sets an int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setIntArray2(e, t) { this._valueCache[e] = null, this.engine.setIntArray2(this._uniforms[e], t); } /** * Sets an int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setIntArray3(e, t) { this._valueCache[e] = null, this.engine.setIntArray3(this._uniforms[e], t); } /** * Sets an int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setIntArray4(e, t) { this._valueCache[e] = null, this.engine.setIntArray4(this._uniforms[e], t); } /** * Sets an unsigned integer value on a uniform variable. * @param uniformName Name of the variable. * @param value Value to be set. */ setUInt(e, t) { const r = this._valueCache[e]; r !== void 0 && r === t || this.engine.setUInt(this._uniforms[e], t) && (this._valueCache[e] = t); } /** * Sets an unsigned int2 value on a uniform variable. * @param uniformName Name of the variable. * @param x First unsigned int in uint2. * @param y Second unsigned int in uint2. */ setUInt2(e, t, r) { this._cacheFloat2(e, t, r) && (this.engine.setUInt2(this._uniforms[e], t, r) || (this._valueCache[e] = null)); } /** * Sets an unsigned int3 value on a uniform variable. * @param uniformName Name of the variable. * @param x First unsigned int in uint3. * @param y Second unsigned int in uint3. * @param z Third unsigned int in uint3. */ setUInt3(e, t, r, n) { this._cacheFloat3(e, t, r, n) && (this.engine.setUInt3(this._uniforms[e], t, r, n) || (this._valueCache[e] = null)); } /** * Sets an unsigned int4 value on a uniform variable. * @param uniformName Name of the variable. * @param x First unsigned int in uint4. * @param y Second unsigned int in uint4. * @param z Third unsigned int in uint4. * @param w Fourth unsigned int in uint4. */ setUInt4(e, t, r, n, i) { this._cacheFloat4(e, t, r, n, i) && (this.engine.setUInt4(this._uniforms[e], t, r, n, i) || (this._valueCache[e] = null)); } /** * Sets an unsigned int array on a uniform variable. * @param uniformName Name of the variable. * @param array array to be set. */ setUIntArray(e, t) { this._valueCache[e] = null, this.engine.setUIntArray(this._uniforms[e], t); } /** * Sets an unsigned int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setUIntArray2(e, t) { this._valueCache[e] = null, this.engine.setUIntArray2(this._uniforms[e], t); } /** * Sets an unsigned int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setUIntArray3(e, t) { this._valueCache[e] = null, this.engine.setUIntArray3(this._uniforms[e], t); } /** * Sets an unsigned int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setUIntArray4(e, t) { this._valueCache[e] = null, this.engine.setUIntArray4(this._uniforms[e], t); } /** * Sets an array on a uniform variable. * @param uniformName Name of the variable. * @param array array to be set. */ setArray(e, t) { this._valueCache[e] = null, this.engine.setArray(this._uniforms[e], t); } /** * Sets an array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setArray2(e, t) { this._valueCache[e] = null, this.engine.setArray2(this._uniforms[e], t); } /** * Sets an array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setArray3(e, t) { this._valueCache[e] = null, this.engine.setArray3(this._uniforms[e], t); } /** * Sets an array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setArray4(e, t) { this._valueCache[e] = null, this.engine.setArray4(this._uniforms[e], t); } /** * Sets matrices on a uniform variable. * @param uniformName Name of the variable. * @param matrices matrices to be set. */ setMatrices(e, t) { t && (this._valueCache[e] = null, this.engine.setMatrices(this._uniforms[e], t)); } /** * Sets matrix on a uniform variable. * @param uniformName Name of the variable. * @param matrix matrix to be set. */ setMatrix(e, t) { this._cacheMatrix(e, t) && (this.engine.setMatrices(this._uniforms[e], t.toArray()) || (this._valueCache[e] = null)); } /** * Sets a 3x3 matrix on a uniform variable. (Specified as [1,2,3,4,5,6,7,8,9] will result in [1,2,3][4,5,6][7,8,9] matrix) * @param uniformName Name of the variable. * @param matrix matrix to be set. */ setMatrix3x3(e, t) { this._valueCache[e] = null, this.engine.setMatrix3x3(this._uniforms[e], t); } /** * Sets a 2x2 matrix on a uniform variable. (Specified as [1,2,3,4] will result in [1,2][3,4] matrix) * @param uniformName Name of the variable. * @param matrix matrix to be set. */ setMatrix2x2(e, t) { this._valueCache[e] = null, this.engine.setMatrix2x2(this._uniforms[e], t); } /** * Sets a float on a uniform variable. * @param uniformName Name of the variable. * @param value value to be set. * @returns this effect. */ setFloat(e, t) { const r = this._valueCache[e]; r !== void 0 && r === t || this.engine.setFloat(this._uniforms[e], t) && (this._valueCache[e] = t); } /** * Sets a Vector2 on a uniform variable. * @param uniformName Name of the variable. * @param vector2 vector2 to be set. */ setVector2(e, t) { this._cacheFloat2(e, t.x, t.y) && (this.engine.setFloat2(this._uniforms[e], t.x, t.y) || (this._valueCache[e] = null)); } /** * Sets a float2 on a uniform variable. * @param uniformName Name of the variable. * @param x First float in float2. * @param y Second float in float2. */ setFloat2(e, t, r) { this._cacheFloat2(e, t, r) && (this.engine.setFloat2(this._uniforms[e], t, r) || (this._valueCache[e] = null)); } /** * Sets a Vector3 on a uniform variable. * @param uniformName Name of the variable. * @param vector3 Value to be set. */ setVector3(e, t) { this._cacheFloat3(e, t.x, t.y, t.z) && (this.engine.setFloat3(this._uniforms[e], t.x, t.y, t.z) || (this._valueCache[e] = null)); } /** * Sets a float3 on a uniform variable. * @param uniformName Name of the variable. * @param x First float in float3. * @param y Second float in float3. * @param z Third float in float3. */ setFloat3(e, t, r, n) { this._cacheFloat3(e, t, r, n) && (this.engine.setFloat3(this._uniforms[e], t, r, n) || (this._valueCache[e] = null)); } /** * Sets a Vector4 on a uniform variable. * @param uniformName Name of the variable. * @param vector4 Value to be set. */ setVector4(e, t) { this._cacheFloat4(e, t.x, t.y, t.z, t.w) && (this.engine.setFloat4(this._uniforms[e], t.x, t.y, t.z, t.w) || (this._valueCache[e] = null)); } /** * Sets a Quaternion on a uniform variable. * @param uniformName Name of the variable. * @param quaternion Value to be set. */ setQuaternion(e, t) { this._cacheFloat4(e, t.x, t.y, t.z, t.w) && (this.engine.setFloat4(this._uniforms[e], t.x, t.y, t.z, t.w) || (this._valueCache[e] = null)); } /** * Sets a float4 on a uniform variable. * @param uniformName Name of the variable. * @param x First float in float4. * @param y Second float in float4. * @param z Third float in float4. * @param w Fourth float in float4. * @returns this effect. */ setFloat4(e, t, r, n, i) { this._cacheFloat4(e, t, r, n, i) && (this.engine.setFloat4(this._uniforms[e], t, r, n, i) || (this._valueCache[e] = null)); } /** * Sets a Color3 on a uniform variable. * @param uniformName Name of the variable. * @param color3 Value to be set. */ setColor3(e, t) { this._cacheFloat3(e, t.r, t.g, t.b) && (this.engine.setFloat3(this._uniforms[e], t.r, t.g, t.b) || (this._valueCache[e] = null)); } /** * Sets a Color4 on a uniform variable. * @param uniformName Name of the variable. * @param color3 Value to be set. * @param alpha Alpha value to be set. */ setColor4(e, t, r) { this._cacheFloat4(e, t.r, t.g, t.b, r) && (this.engine.setFloat4(this._uniforms[e], t.r, t.g, t.b, r) || (this._valueCache[e] = null)); } /** * Sets a Color4 on a uniform variable * @param uniformName defines the name of the variable * @param color4 defines the value to be set */ setDirectColor4(e, t) { this._cacheFloat4(e, t.r, t.g, t.b, t.a) && (this.engine.setFloat4(this._uniforms[e], t.r, t.g, t.b, t.a) || (this._valueCache[e] = null)); } _getVertexShaderCode() { return this.vertexShader ? this.engine._getShaderSource(this.vertexShader) : null; } _getFragmentShaderCode() { return this.fragmentShader ? this.engine._getShaderSource(this.fragmentShader) : null; } } class jm { get underlyingResource() { return this._webGLTexture; } constructor(e = null, t) { if (this._MSAARenderBuffers = null, this._context = t, !e && (e = t.createTexture(), !e)) throw new Error("Unable to create webGL texture"); this.set(e); } setUsage() { } set(e) { this._webGLTexture = e; } reset() { this._webGLTexture = null, this._MSAARenderBuffers = null; } addMSAARenderBuffer(e) { this._MSAARenderBuffers || (this._MSAARenderBuffers = []), this._MSAARenderBuffers.push(e); } releaseMSAARenderBuffers() { if (this._MSAARenderBuffers) { for (const e of this._MSAARenderBuffers) this._context.deleteRenderbuffer(e); this._MSAARenderBuffers = null; } } getMSAARenderBuffer(e = 0) { var t, r; return (r = (t = this._MSAARenderBuffers) === null || t === void 0 ? void 0 : t[e]) !== null && r !== void 0 ? r : null; } release() { this.releaseMSAARenderBuffers(), this._webGLTexture && this._context.deleteTexture(this._webGLTexture), this.reset(); } } class zo { static IsWrapper(e) { return e.getPipelineContext === void 0; } static GetEffect(e) { return e.getPipelineContext === void 0 ? e.effect : e; } constructor(e, t = !0) { this.effect = null, this.defines = null, this.drawContext = e.createDrawContext(), t && (this.materialContext = e.createMaterialContext()); } setEffect(e, t, r = !0) { var n; this.effect = e, t !== void 0 && (this.defines = t), r && ((n = this.drawContext) === null || n === void 0 || n.reset()); } dispose() { var e; (e = this.drawContext) === null || e === void 0 || e.dispose(); } } class KN { get isDirty() { return this._isStencilTestDirty || this._isStencilMaskDirty || this._isStencilFuncDirty || this._isStencilOpDirty; } get func() { return this._func; } set func(e) { this._func !== e && (this._func = e, this._isStencilFuncDirty = !0); } get funcRef() { return this._funcRef; } set funcRef(e) { this._funcRef !== e && (this._funcRef = e, this._isStencilFuncDirty = !0); } get funcMask() { return this._funcMask; } set funcMask(e) { this._funcMask !== e && (this._funcMask = e, this._isStencilFuncDirty = !0); } get opStencilFail() { return this._opStencilFail; } set opStencilFail(e) { this._opStencilFail !== e && (this._opStencilFail = e, this._isStencilOpDirty = !0); } get opDepthFail() { return this._opDepthFail; } set opDepthFail(e) { this._opDepthFail !== e && (this._opDepthFail = e, this._isStencilOpDirty = !0); } get opStencilDepthPass() { return this._opStencilDepthPass; } set opStencilDepthPass(e) { this._opStencilDepthPass !== e && (this._opStencilDepthPass = e, this._isStencilOpDirty = !0); } get mask() { return this._mask; } set mask(e) { this._mask !== e && (this._mask = e, this._isStencilMaskDirty = !0); } get enabled() { return this._enabled; } set enabled(e) { this._enabled !== e && (this._enabled = e, this._isStencilTestDirty = !0); } constructor(e = !0) { this._isStencilTestDirty = !1, this._isStencilMaskDirty = !1, this._isStencilFuncDirty = !1, this._isStencilOpDirty = !1, this.useStencilGlobalOnly = !1, e && this.reset(); } reset() { var e; this.stencilMaterial = void 0, (e = this.stencilGlobal) === null || e === void 0 || e.reset(), this._isStencilTestDirty = !0, this._isStencilMaskDirty = !0, this._isStencilFuncDirty = !0, this._isStencilOpDirty = !0; } apply(e) { var t; if (!e) return; const r = !this.useStencilGlobalOnly && !!(!((t = this.stencilMaterial) === null || t === void 0) && t.enabled); this.enabled = r ? this.stencilMaterial.enabled : this.stencilGlobal.enabled, this.func = r ? this.stencilMaterial.func : this.stencilGlobal.func, this.funcRef = r ? this.stencilMaterial.funcRef : this.stencilGlobal.funcRef, this.funcMask = r ? this.stencilMaterial.funcMask : this.stencilGlobal.funcMask, this.opStencilFail = r ? this.stencilMaterial.opStencilFail : this.stencilGlobal.opStencilFail, this.opDepthFail = r ? this.stencilMaterial.opDepthFail : this.stencilGlobal.opDepthFail, this.opStencilDepthPass = r ? this.stencilMaterial.opStencilDepthPass : this.stencilGlobal.opStencilDepthPass, this.mask = r ? this.stencilMaterial.mask : this.stencilGlobal.mask, this.isDirty && (this._isStencilTestDirty && (this.enabled ? e.enable(e.STENCIL_TEST) : e.disable(e.STENCIL_TEST), this._isStencilTestDirty = !1), this._isStencilMaskDirty && (e.stencilMask(this.mask), this._isStencilMaskDirty = !1), this._isStencilFuncDirty && (e.stencilFunc(this.func, this.funcRef, this.funcMask), this._isStencilFuncDirty = !1), this._isStencilOpDirty && (e.stencilOp(this.opStencilFail, this.opDepthFail, this.opStencilDepthPass), this._isStencilOpDirty = !1)); } } class K9e { } class hr { /** * Returns the current npm package of the sdk */ // Not mixed with Version for tooling purpose. static get NpmPackage() { return "babylonjs@6.33.0"; } /** * Returns the current version of the framework */ static get Version() { return "6.33.0"; } /** * Returns a string describing the current engine */ get description() { let e = this.name + this.webGLVersion; return this._caps.parallelShaderCompile && (e += " - Parallel shader compilation"), e; } /** * Gets or sets the name of the engine */ get name() { return this._name; } set name(e) { this._name = e; } /** * Returns the version of the engine */ get version() { return this._webGLVersion; } get isDisposed() { return this._isDisposed; } /** * Gets or sets the relative url used to load shaders if using the engine in non-minified mode */ static get ShadersRepository() { return An.ShadersRepository; } static set ShadersRepository(e) { An.ShadersRepository = e; } /** * @internal */ _getShaderProcessor(e) { return this._shaderProcessor; } /** * Gets or sets a boolean indicating if depth buffer should be reverse, going from far to near. * This can provide greater z depth for distant objects. */ get useReverseDepthBuffer() { return this._useReverseDepthBuffer; } set useReverseDepthBuffer(e) { e !== this._useReverseDepthBuffer && (this._useReverseDepthBuffer = e, e ? this._depthCullingState.depthFunc = 518 : this._depthCullingState.depthFunc = 515); } /** * Gets the current frame id */ get frameId() { return this._frameId; } /** * Gets a boolean indicating that the engine supports uniform buffers * @see https://doc.babylonjs.com/setup/support/webGL2#uniform-buffer-objets */ get supportsUniformBuffers() { return this.webGLVersion > 1 && !this.disableUniformBuffers; } /** * Gets the options used for engine creation * @returns EngineOptions object */ getCreationOptions() { return this._creationOptions; } /** @internal */ get _shouldUseHighPrecisionShader() { return !!(this._caps.highPrecisionShaderSupported && this._highPrecisionShadersAllowed); } /** * Gets a boolean indicating that only power of 2 textures are supported * Please note that you can still use non power of 2 textures but in this case the engine will forcefully convert them */ get needPOTTextures() { return this._webGLVersion < 2 || this.forcePOTTextures; } /** * Gets the list of current active render loop functions * @returns an array with the current render loop functions */ get activeRenderLoops() { return this._activeRenderLoops; } /** * Gets or sets a boolean indicating if resources should be retained to be able to handle context lost events * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#handling-webgl-context-lost */ get doNotHandleContextLost() { return this._doNotHandleContextLost; } set doNotHandleContextLost(e) { this._doNotHandleContextLost = e; } get _supportsHardwareTextureRescaling() { return !1; } /** * sets the object from which width and height will be taken from when getting render width and height * Will fallback to the gl object * @param dimensions the framebuffer width and height that will be used. */ set framebufferDimensionsObject(e) { this._framebufferDimensionsObject = e; } /** * Gets the current viewport */ get currentViewport() { return this._cachedViewport; } /** * Gets the default empty texture */ get emptyTexture() { return this._emptyTexture || (this._emptyTexture = this.createRawTexture(new Uint8Array(4), 1, 1, 5, !1, !1, 1)), this._emptyTexture; } /** * Gets the default empty 3D texture */ get emptyTexture3D() { return this._emptyTexture3D || (this._emptyTexture3D = this.createRawTexture3D(new Uint8Array(4), 1, 1, 1, 5, !1, !1, 1)), this._emptyTexture3D; } /** * Gets the default empty 2D array texture */ get emptyTexture2DArray() { return this._emptyTexture2DArray || (this._emptyTexture2DArray = this.createRawTexture2DArray(new Uint8Array(4), 1, 1, 1, 5, !1, !1, 1)), this._emptyTexture2DArray; } /** * Gets the default empty cube texture */ get emptyCubeTexture() { if (!this._emptyCubeTexture) { const e = new Uint8Array(4), t = [e, e, e, e, e, e]; this._emptyCubeTexture = this.createRawCubeTexture(t, 1, 5, 0, !1, !1, 1); } return this._emptyCubeTexture; } /** * Gets a boolean indicating if the engine runs in WebGPU or not. */ get isWebGPU() { return this._isWebGPU; } /** * Gets the shader platform name used by the effects. */ get shaderPlatformName() { return this._shaderPlatformName; } /** * Enables or disables the snapshot rendering mode * Note that the WebGL engine does not support snapshot rendering so setting the value won't have any effect for this engine */ get snapshotRendering() { return !1; } set snapshotRendering(e) { } /** * Gets or sets the snapshot rendering mode */ get snapshotRenderingMode() { return this._snapshotRenderingMode; } set snapshotRenderingMode(e) { this._snapshotRenderingMode = e; } /** * Creates a new snapshot at the next frame using the current snapshotRenderingMode */ snapshotRenderingReset() { this.snapshotRendering = !1; } static _CreateCanvas(e, t) { if (typeof document > "u") return new OffscreenCanvas(e, t); const r = document.createElement("canvas"); return r.width = e, r.height = t, r; } /** * Create a canvas. This method is overridden by other engines * @param width width * @param height height * @returns ICanvas interface */ createCanvas(e, t) { return hr._CreateCanvas(e, t); } /** * Create an image to use with canvas * @returns IImage interface */ createCanvasImage() { return document.createElement("img"); } /** * Creates a new engine * @param canvasOrContext defines the canvas or WebGL context to use for rendering. If you provide a WebGL context, Babylon.js will not hook events on the canvas (like pointers, keyboards, etc...) so no event observables will be available. This is mostly used when Babylon.js is used as a plugin on a system which already used the WebGL context * @param antialias defines enable antialiasing (default: false) * @param options defines further options to be sent to the getContext() function * @param adaptToDeviceRatio defines whether to adapt to the device's viewport characteristics (default: false) */ constructor(e, t, r, n) { var i, s, a, f, o, d, v, u, l, P, p; this._name = "WebGL", this._isDisposed = !1, this.forcePOTTextures = !1, this.isFullscreen = !1, this.cullBackFaces = null, this.renderEvenInBackground = !0, this.preventCacheWipeBetweenFrames = !1, this.validateShaderPrograms = !1, this._useReverseDepthBuffer = !1, this.isNDCHalfZRange = !1, this.hasOriginBottomLeft = !0, this.disableUniformBuffers = !1, this.onDisposeObservable = new Oe(), this._frameId = 0, this._uniformBuffers = new Array(), this._storageBuffers = new Array(), this._webGLVersion = 1, this._windowIsBackground = !1, this._highPrecisionShadersAllowed = !0, this._badOS = !1, this._badDesktopOS = !1, this._renderingQueueLaunched = !1, this._activeRenderLoops = new Array(), this.onContextLostObservable = new Oe(), this.onContextRestoredObservable = new Oe(), this._contextWasLost = !1, this._doNotHandleContextLost = !1, this.disableVertexArrayObjects = !1, this._colorWrite = !0, this._colorWriteChanged = !0, this._depthCullingState = new YN(), this._stencilStateComposer = new KN(), this._stencilState = new L2(), this._alphaState = new k$(), this._alphaMode = 1, this._alphaEquation = 0, this._internalTexturesCache = new Array(), this._renderTargetWrapperCache = new Array(), this._activeChannel = 0, this._currentTextureChannel = -1, this._boundTexturesCache = {}, this._compiledEffects = {}, this._vertexAttribArraysEnabled = [], this._currentRenderTarget = null, this._uintIndicesCurrentlySet = !1, this._currentBoundBuffer = new Array(), this._currentFramebuffer = null, this._dummyFramebuffer = null, this._currentBufferPointers = new Array(), this._currentInstanceLocations = new Array(), this._currentInstanceBuffers = new Array(), this._vaoRecordInProgress = !1, this._mustWipeVertexAttributes = !1, this._nextFreeTextureSlots = new Array(), this._maxSimultaneousTextures = 0, this._maxMSAASamplesOverride = null, this._activeRequests = new Array(), this.adaptToDeviceRatio = !1, this._lastDevicePixelRatio = 1, this._transformTextureUrl = null, this.hostInformation = { isMobile: !1 }, this.premultipliedAlpha = !0, this.onBeforeTextureInitObservable = new Oe(), this._isWebGPU = !1, this._snapshotRenderingMode = 0, this._viewportCached = { x: 0, y: 0, z: 0, w: 0 }, this._unpackFlipYCached = null, this.enableUnpackFlipYCached = !0, this._boundUniforms = {}, this.startTime = Yi.Now; let c = null; r = r || {}, this._creationOptions = r, this.adaptToDeviceRatio = n ?? !1, this._stencilStateComposer.stencilGlobal = this._stencilState, k9.SetMatrixPrecision(!!r.useHighPrecisionMatrix), r.antialias = t ?? r.antialias, r.deterministicLockstep = (i = r.deterministicLockstep) !== null && i !== void 0 ? i : !1, r.lockstepMaxSteps = (s = r.lockstepMaxSteps) !== null && s !== void 0 ? s : 4, r.timeStep = (a = r.timeStep) !== null && a !== void 0 ? a : 1 / 60, r.audioEngine = (f = r.audioEngine) !== null && f !== void 0 ? f : !0, r.stencil = (o = r.stencil) !== null && o !== void 0 ? o : !0, this._audioContext = (v = (d = r.audioEngineOptions) === null || d === void 0 ? void 0 : d.audioContext) !== null && v !== void 0 ? v : null, this._audioDestination = (l = (u = r.audioEngineOptions) === null || u === void 0 ? void 0 : u.audioDestination) !== null && l !== void 0 ? l : null, this.premultipliedAlpha = (P = r.premultipliedAlpha) !== null && P !== void 0 ? P : !0, this.useExactSrgbConversions = (p = r.useExactSrgbConversions) !== null && p !== void 0 ? p : !1, this._doNotHandleContextLost = !!r.doNotHandleContextLost, this._isStencilEnable = !!r.stencil, n = n || r.adaptToDeviceRatio || !1; const H = u9() && window.devicePixelRatio || 1, T = r.limitDeviceRatio || H; if (this._hardwareScalingLevel = n ? 1 / Math.min(T, H) : 1, this._lastDevicePixelRatio = H, !e) return; if (e.getContext) { if (c = e, this._renderingCanvas = c, r.preserveDrawingBuffer === void 0 && (r.preserveDrawingBuffer = !1), r.xrCompatible === void 0 && (r.xrCompatible = !0), navigator && navigator.userAgent) { this._setupMobileChecks(); const b = navigator.userAgent; for (const j of hr.ExceptionList) { const w = j.key, m = j.targets; if (new RegExp(w).test(b)) { if (j.capture && j.captureConstraint) { const N = j.capture, k = j.captureConstraint, y = new RegExp(N).exec(b); if (y && y.length > 0 && parseInt(y[y.length - 1]) >= k) continue; } for (const N of m) switch (N) { case "uniformBuffer": this.disableUniformBuffers = !0; break; case "vao": this.disableVertexArrayObjects = !0; break; case "antialias": r.antialias = !1; break; case "maxMSAASamples": this._maxMSAASamplesOverride = 1; break; } } } } if (this._doNotHandleContextLost || (this._onContextLost = (b) => { b.preventDefault(), this._contextWasLost = !0, Se.Warn("WebGL context lost."), this.onContextLostObservable.notifyObservers(this); }, this._onContextRestored = () => { this._restoreEngineAfterContextLost(() => this._initGLContext()); }, c.addEventListener("webglcontextlost", this._onContextLost, !1), c.addEventListener("webglcontextrestored", this._onContextRestored, !1), r.powerPreference = r.powerPreference || "high-performance"), this._badDesktopOS = /^((?!chrome|android).)*safari/i.test(navigator.userAgent), this._badDesktopOS && (r.xrCompatible = !1), !r.disableWebGL2Support) try { this._gl = c.getContext("webgl2", r) || c.getContext("experimental-webgl2", r), this._gl && (this._webGLVersion = 2, this._shaderPlatformName = "WEBGL2", this._gl.deleteQuery || (this._webGLVersion = 1, this._shaderPlatformName = "WEBGL1")); } catch { } if (!this._gl) { if (!c) throw new Error("The provided canvas is null or undefined."); try { this._gl = c.getContext("webgl", r) || c.getContext("experimental-webgl", r); } catch { throw new Error("WebGL not supported"); } } if (!this._gl) throw new Error("WebGL not supported"); } else { this._gl = e, this._renderingCanvas = this._gl.canvas, this._gl.renderbufferStorageMultisample ? (this._webGLVersion = 2, this._shaderPlatformName = "WEBGL2") : this._shaderPlatformName = "WEBGL1"; const b = this._gl.getContextAttributes(); b && (r.stencil = b.stencil); } this._gl.pixelStorei(this._gl.UNPACK_COLORSPACE_CONVERSION_WEBGL, this._gl.NONE), r.useHighPrecisionFloats !== void 0 && (this._highPrecisionShadersAllowed = r.useHighPrecisionFloats), this.resize(), this._initGLContext(), this._initFeatures(); for (let b = 0; b < this._caps.maxVertexAttribs; b++) this._currentBufferPointers[b] = new K9e(); this._shaderProcessor = this.webGLVersion > 1 ? new LN() : new M9e(), this._badOS = /iPad/i.test(navigator.userAgent) || /iPhone/i.test(navigator.userAgent); const q = `Babylon.js v${hr.Version}`; console.log(q + ` - ${this.description}`), this._renderingCanvas && this._renderingCanvas.setAttribute && this._renderingCanvas.setAttribute("data-engine", q); } _setupMobileChecks() { navigator && navigator.userAgent && (this._checkForMobile = () => { const e = navigator.userAgent; this.hostInformation.isMobile = e.indexOf("Mobile") !== -1 || // Needed for iOS 13+ detection on iPad (inspired by solution from https://stackoverflow.com/questions/9038625/detect-if-device-is-ios) e.indexOf("Mac") !== -1 && $w() && "ontouchend" in document; }, this._checkForMobile(), u9() && window.addEventListener("resize", this._checkForMobile)); } _restoreEngineAfterContextLost(e) { setTimeout(async () => { var t; this._dummyFramebuffer = null; const r = this._depthCullingState.depthTest, n = this._depthCullingState.depthFunc, i = this._depthCullingState.depthMask, s = this._stencilState.stencilTest; await e(), this.wipeCaches(!0), this._rebuildEffects(), (t = this._rebuildComputeEffects) === null || t === void 0 || t.call(this), this._rebuildBuffers(), this._rebuildInternalTextures(), this._rebuildRenderTargetWrappers(), this.wipeCaches(!0), this._depthCullingState.depthTest = r, this._depthCullingState.depthFunc = n, this._depthCullingState.depthMask = i, this._stencilState.stencilTest = s, Se.Warn(this.name + " context successfully restored."), this.onContextRestoredObservable.notifyObservers(this), this._contextWasLost = !1; }, 0); } /** * Shared initialization across engines types. * @param canvas The canvas associated with this instance of the engine. */ _sharedInit(e) { this._renderingCanvas = e; } /** * @internal */ _getShaderProcessingContext(e) { return null; } _rebuildInternalTextures() { const e = this._internalTexturesCache.slice(); for (const t of e) t._rebuild(); } _rebuildRenderTargetWrappers() { const e = this._renderTargetWrapperCache.slice(); for (const t of e) t._rebuild(); } _rebuildEffects() { for (const e in this._compiledEffects) { const t = this._compiledEffects[e]; t._pipelineContext = null, t._wasPreviouslyReady = !1, t._prepareEffect(); } An.ResetCache(); } /** * Gets a boolean indicating if all created effects are ready * @returns true if all effects are ready */ areAllEffectsReady() { for (const e in this._compiledEffects) if (!this._compiledEffects[e].isReady()) return !1; return !0; } _rebuildBuffers() { for (const e of this._uniformBuffers) e._rebuild(); for (const e of this._storageBuffers) e._rebuild(); } _initGLContext() { var e; this._caps = { maxTexturesImageUnits: this._gl.getParameter(this._gl.MAX_TEXTURE_IMAGE_UNITS), maxCombinedTexturesImageUnits: this._gl.getParameter(this._gl.MAX_COMBINED_TEXTURE_IMAGE_UNITS), maxVertexTextureImageUnits: this._gl.getParameter(this._gl.MAX_VERTEX_TEXTURE_IMAGE_UNITS), maxTextureSize: this._gl.getParameter(this._gl.MAX_TEXTURE_SIZE), maxSamples: this._webGLVersion > 1 ? this._gl.getParameter(this._gl.MAX_SAMPLES) : 1, maxCubemapTextureSize: this._gl.getParameter(this._gl.MAX_CUBE_MAP_TEXTURE_SIZE), maxRenderTextureSize: this._gl.getParameter(this._gl.MAX_RENDERBUFFER_SIZE), maxVertexAttribs: this._gl.getParameter(this._gl.MAX_VERTEX_ATTRIBS), maxVaryingVectors: this._gl.getParameter(this._gl.MAX_VARYING_VECTORS), maxFragmentUniformVectors: this._gl.getParameter(this._gl.MAX_FRAGMENT_UNIFORM_VECTORS), maxVertexUniformVectors: this._gl.getParameter(this._gl.MAX_VERTEX_UNIFORM_VECTORS), parallelShaderCompile: this._gl.getExtension("KHR_parallel_shader_compile") || void 0, standardDerivatives: this._webGLVersion > 1 || this._gl.getExtension("OES_standard_derivatives") !== null, maxAnisotropy: 1, astc: this._gl.getExtension("WEBGL_compressed_texture_astc") || this._gl.getExtension("WEBKIT_WEBGL_compressed_texture_astc"), bptc: this._gl.getExtension("EXT_texture_compression_bptc") || this._gl.getExtension("WEBKIT_EXT_texture_compression_bptc"), s3tc: this._gl.getExtension("WEBGL_compressed_texture_s3tc") || this._gl.getExtension("WEBKIT_WEBGL_compressed_texture_s3tc"), // eslint-disable-next-line @typescript-eslint/naming-convention s3tc_srgb: this._gl.getExtension("WEBGL_compressed_texture_s3tc_srgb") || this._gl.getExtension("WEBKIT_WEBGL_compressed_texture_s3tc_srgb"), pvrtc: this._gl.getExtension("WEBGL_compressed_texture_pvrtc") || this._gl.getExtension("WEBKIT_WEBGL_compressed_texture_pvrtc"), etc1: this._gl.getExtension("WEBGL_compressed_texture_etc1") || this._gl.getExtension("WEBKIT_WEBGL_compressed_texture_etc1"), etc2: this._gl.getExtension("WEBGL_compressed_texture_etc") || this._gl.getExtension("WEBKIT_WEBGL_compressed_texture_etc") || this._gl.getExtension("WEBGL_compressed_texture_es3_0"), textureAnisotropicFilterExtension: this._gl.getExtension("EXT_texture_filter_anisotropic") || this._gl.getExtension("WEBKIT_EXT_texture_filter_anisotropic") || this._gl.getExtension("MOZ_EXT_texture_filter_anisotropic"), uintIndices: this._webGLVersion > 1 || this._gl.getExtension("OES_element_index_uint") !== null, fragmentDepthSupported: this._webGLVersion > 1 || this._gl.getExtension("EXT_frag_depth") !== null, highPrecisionShaderSupported: !1, timerQuery: this._gl.getExtension("EXT_disjoint_timer_query_webgl2") || this._gl.getExtension("EXT_disjoint_timer_query"), supportOcclusionQuery: this._webGLVersion > 1, canUseTimestampForTimerQuery: !1, drawBuffersExtension: !1, maxMSAASamples: 1, colorBufferFloat: !!(this._webGLVersion > 1 && this._gl.getExtension("EXT_color_buffer_float")), supportFloatTexturesResolve: !1, colorBufferHalfFloat: !!(this._webGLVersion > 1 && this._gl.getExtension("EXT_color_buffer_half_float")), textureFloat: !!(this._webGLVersion > 1 || this._gl.getExtension("OES_texture_float")), textureHalfFloat: !!(this._webGLVersion > 1 || this._gl.getExtension("OES_texture_half_float")), textureHalfFloatRender: !1, textureFloatLinearFiltering: !1, textureFloatRender: !1, textureHalfFloatLinearFiltering: !1, vertexArrayObject: !1, instancedArrays: !1, textureLOD: !!(this._webGLVersion > 1 || this._gl.getExtension("EXT_shader_texture_lod")), texelFetch: this._webGLVersion !== 1, blendMinMax: !1, multiview: this._gl.getExtension("OVR_multiview2"), oculusMultiview: this._gl.getExtension("OCULUS_multiview"), depthTextureExtension: !1, canUseGLInstanceID: this._webGLVersion > 1, canUseGLVertexID: this._webGLVersion > 1, supportComputeShaders: !1, supportSRGBBuffers: !1, supportTransformFeedbacks: this._webGLVersion > 1, textureMaxLevel: this._webGLVersion > 1, texture2DArrayMaxLayerCount: this._webGLVersion > 1 ? this._gl.getParameter(this._gl.MAX_ARRAY_TEXTURE_LAYERS) : 128, disableMorphTargetTexture: !1 }, this._caps.supportFloatTexturesResolve = this._caps.colorBufferFloat, this._glVersion = this._gl.getParameter(this._gl.VERSION); const t = this._gl.getExtension("WEBGL_debug_renderer_info"); if (t != null && (this._glRenderer = this._gl.getParameter(t.UNMASKED_RENDERER_WEBGL), this._glVendor = this._gl.getParameter(t.UNMASKED_VENDOR_WEBGL)), this._glVendor || (this._glVendor = this._gl.getParameter(this._gl.VENDOR) || "Unknown vendor"), this._glRenderer || (this._glRenderer = this._gl.getParameter(this._gl.RENDERER) || "Unknown renderer"), this._gl.HALF_FLOAT_OES !== 36193 && (this._gl.HALF_FLOAT_OES = 36193), this._gl.RGBA16F !== 34842 && (this._gl.RGBA16F = 34842), this._gl.RGBA32F !== 34836 && (this._gl.RGBA32F = 34836), this._gl.DEPTH24_STENCIL8 !== 35056 && (this._gl.DEPTH24_STENCIL8 = 35056), this._caps.timerQuery && (this._webGLVersion === 1 && (this._gl.getQuery = this._caps.timerQuery.getQueryEXT.bind(this._caps.timerQuery)), this._caps.canUseTimestampForTimerQuery = ((e = this._gl.getQuery(this._caps.timerQuery.TIMESTAMP_EXT, this._caps.timerQuery.QUERY_COUNTER_BITS_EXT)) !== null && e !== void 0 ? e : 0) > 0), this._caps.maxAnisotropy = this._caps.textureAnisotropicFilterExtension ? this._gl.getParameter(this._caps.textureAnisotropicFilterExtension.MAX_TEXTURE_MAX_ANISOTROPY_EXT) : 0, this._caps.textureFloatLinearFiltering = !!(this._caps.textureFloat && this._gl.getExtension("OES_texture_float_linear")), this._caps.textureFloatRender = !!(this._caps.textureFloat && this._canRenderToFloatFramebuffer()), this._caps.textureHalfFloatLinearFiltering = !!(this._webGLVersion > 1 || this._caps.textureHalfFloat && this._gl.getExtension("OES_texture_half_float_linear")), this._caps.astc && (this._gl.COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR = this._caps.astc.COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR), this._caps.bptc && (this._gl.COMPRESSED_SRGB_ALPHA_BPTC_UNORM_EXT = this._caps.bptc.COMPRESSED_SRGB_ALPHA_BPTC_UNORM_EXT), this._caps.s3tc_srgb && (this._gl.COMPRESSED_SRGB_S3TC_DXT1_EXT = this._caps.s3tc_srgb.COMPRESSED_SRGB_S3TC_DXT1_EXT, this._gl.COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT = this._caps.s3tc_srgb.COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT, this._gl.COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT = this._caps.s3tc_srgb.COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT), this._caps.etc2 && (this._gl.COMPRESSED_SRGB8_ETC2 = this._caps.etc2.COMPRESSED_SRGB8_ETC2, this._gl.COMPRESSED_SRGB8_ALPHA8_ETC2_EAC = this._caps.etc2.COMPRESSED_SRGB8_ALPHA8_ETC2_EAC), this._webGLVersion > 1 && this._gl.HALF_FLOAT_OES !== 5131 && (this._gl.HALF_FLOAT_OES = 5131), this._caps.textureHalfFloatRender = this._caps.textureHalfFloat && this._canRenderToHalfFloatFramebuffer(), this._webGLVersion > 1) this._caps.drawBuffersExtension = !0, this._caps.maxMSAASamples = this._maxMSAASamplesOverride !== null ? this._maxMSAASamplesOverride : this._gl.getParameter(this._gl.MAX_SAMPLES); else { const r = this._gl.getExtension("WEBGL_draw_buffers"); if (r !== null) { this._caps.drawBuffersExtension = !0, this._gl.drawBuffers = r.drawBuffersWEBGL.bind(r), this._gl.DRAW_FRAMEBUFFER = this._gl.FRAMEBUFFER; for (let n = 0; n < 16; n++) this._gl["COLOR_ATTACHMENT" + n + "_WEBGL"] = r["COLOR_ATTACHMENT" + n + "_WEBGL"]; } } if (this._webGLVersion > 1) this._caps.depthTextureExtension = !0; else { const r = this._gl.getExtension("WEBGL_depth_texture"); r != null && (this._caps.depthTextureExtension = !0, this._gl.UNSIGNED_INT_24_8 = r.UNSIGNED_INT_24_8_WEBGL); } if (this.disableVertexArrayObjects) this._caps.vertexArrayObject = !1; else if (this._webGLVersion > 1) this._caps.vertexArrayObject = !0; else { const r = this._gl.getExtension("OES_vertex_array_object"); r != null && (this._caps.vertexArrayObject = !0, this._gl.createVertexArray = r.createVertexArrayOES.bind(r), this._gl.bindVertexArray = r.bindVertexArrayOES.bind(r), this._gl.deleteVertexArray = r.deleteVertexArrayOES.bind(r)); } if (this._webGLVersion > 1) this._caps.instancedArrays = !0; else { const r = this._gl.getExtension("ANGLE_instanced_arrays"); r != null ? (this._caps.instancedArrays = !0, this._gl.drawArraysInstanced = r.drawArraysInstancedANGLE.bind(r), this._gl.drawElementsInstanced = r.drawElementsInstancedANGLE.bind(r), this._gl.vertexAttribDivisor = r.vertexAttribDivisorANGLE.bind(r)) : this._caps.instancedArrays = !1; } if (this._gl.getShaderPrecisionFormat) { const r = this._gl.getShaderPrecisionFormat(this._gl.VERTEX_SHADER, this._gl.HIGH_FLOAT), n = this._gl.getShaderPrecisionFormat(this._gl.FRAGMENT_SHADER, this._gl.HIGH_FLOAT); r && n && (this._caps.highPrecisionShaderSupported = r.precision !== 0 && n.precision !== 0); } if (this._webGLVersion > 1) this._caps.blendMinMax = !0; else { const r = this._gl.getExtension("EXT_blend_minmax"); r != null && (this._caps.blendMinMax = !0, this._gl.MAX = r.MAX_EXT, this._gl.MIN = r.MIN_EXT); } if (!this._caps.supportSRGBBuffers) { if (this._webGLVersion > 1) this._caps.supportSRGBBuffers = !0, this._glSRGBExtensionValues = { SRGB: WebGL2RenderingContext.SRGB, SRGB8: WebGL2RenderingContext.SRGB8, SRGB8_ALPHA8: WebGL2RenderingContext.SRGB8_ALPHA8 }; else { const r = this._gl.getExtension("EXT_sRGB"); r != null && (this._caps.supportSRGBBuffers = !0, this._glSRGBExtensionValues = { SRGB: r.SRGB_EXT, SRGB8: r.SRGB_ALPHA_EXT, SRGB8_ALPHA8: r.SRGB_ALPHA_EXT }); } this._caps.supportSRGBBuffers = this._caps.supportSRGBBuffers && !!(this._creationOptions && this._creationOptions.forceSRGBBufferSupportState); } this._depthCullingState.depthTest = !0, this._depthCullingState.depthFunc = this._gl.LEQUAL, this._depthCullingState.depthMask = !0, this._maxSimultaneousTextures = this._caps.maxCombinedTexturesImageUnits; for (let r = 0; r < this._maxSimultaneousTextures; r++) this._nextFreeTextureSlots.push(r); this._glRenderer === "Mali-G72" && (this._caps.disableMorphTargetTexture = !0); } _initFeatures() { this._features = { forceBitmapOverHTMLImageElement: !1, supportRenderAndCopyToLodForFloatTextures: this._webGLVersion !== 1, supportDepthStencilTexture: this._webGLVersion !== 1, supportShadowSamplers: this._webGLVersion !== 1, uniformBufferHardCheckMatrix: !1, allowTexturePrefiltering: this._webGLVersion !== 1, trackUbosInFrame: !1, checkUbosContentBeforeUpload: !1, supportCSM: this._webGLVersion !== 1, basisNeedsPOT: this._webGLVersion === 1, support3DTextures: this._webGLVersion !== 1, needTypeSuffixInShaderConstants: this._webGLVersion !== 1, supportMSAA: this._webGLVersion !== 1, supportSSAO2: this._webGLVersion !== 1, supportExtendedTextureFormats: this._webGLVersion !== 1, supportSwitchCaseInShader: this._webGLVersion !== 1, supportSyncTextureRead: !0, needsInvertingBitmap: !0, useUBOBindingCache: !0, needShaderCodeInlining: !1, needToAlwaysBindUniformBuffers: !1, supportRenderPasses: !1, supportSpriteInstancing: !0, forceVertexBufferStrideMultiple4Bytes: !1, _collectUbosUpdatedInFrame: !1 }; } /** * Gets version of the current webGL context * Keep it for back compat - use version instead */ get webGLVersion() { return this._webGLVersion; } /** * Gets a string identifying the name of the class * @returns "Engine" string */ getClassName() { return "ThinEngine"; } /** * Returns true if the stencil buffer has been enabled through the creation option of the context. */ get isStencilEnable() { return this._isStencilEnable; } /** @internal */ _prepareWorkingCanvas() { if (this._workingCanvas) return; this._workingCanvas = this.createCanvas(1, 1); const e = this._workingCanvas.getContext("2d"); e && (this._workingContext = e); } /** * Reset the texture cache to empty state */ resetTextureCache() { for (const e in this._boundTexturesCache) Object.prototype.hasOwnProperty.call(this._boundTexturesCache, e) && (this._boundTexturesCache[e] = null); this._currentTextureChannel = -1; } /** * Gets an object containing information about the current engine context * @returns an object containing the vendor, the renderer and the version of the current engine context */ getInfo() { return this.getGlInfo(); } /** * Gets an object containing information about the current webGL context * @returns an object containing the vendor, the renderer and the version of the current webGL context */ getGlInfo() { return { vendor: this._glVendor, renderer: this._glRenderer, version: this._glVersion }; } /** * Defines the hardware scaling level. * By default the hardware scaling level is computed from the window device ratio. * if level = 1 then the engine will render at the exact resolution of the canvas. If level = 0.5 then the engine will render at twice the size of the canvas. * @param level defines the level to use */ setHardwareScalingLevel(e) { this._hardwareScalingLevel = e, this.resize(); } /** * Gets the current hardware scaling level. * By default the hardware scaling level is computed from the window device ratio. * if level = 1 then the engine will render at the exact resolution of the canvas. If level = 0.5 then the engine will render at twice the size of the canvas. * @returns a number indicating the current hardware scaling level */ getHardwareScalingLevel() { return this._hardwareScalingLevel; } /** * Gets the list of loaded textures * @returns an array containing all loaded textures */ getLoadedTexturesCache() { return this._internalTexturesCache; } /** * Gets the object containing all engine capabilities * @returns the EngineCapabilities object */ getCaps() { return this._caps; } /** * stop executing a render loop function and remove it from the execution array * @param renderFunction defines the function to be removed. If not provided all functions will be removed. */ stopRenderLoop(e) { if (!e) { this._activeRenderLoops.length = 0, this._cancelFrame(); return; } const t = this._activeRenderLoops.indexOf(e); t >= 0 && (this._activeRenderLoops.splice(t, 1), this._activeRenderLoops.length == 0 && this._cancelFrame()); } _cancelFrame() { if (this._renderingQueueLaunched && this._frameHandler) { if (this._renderingQueueLaunched = !1, u9()) { const { cancelAnimationFrame: e } = this.getHostWindow() || window; if (typeof e == "function") return e(this._frameHandler); } else if (typeof cancelAnimationFrame == "function") return cancelAnimationFrame(this._frameHandler); return clearTimeout(this._frameHandler); } } /** @internal */ _renderLoop() { if (!this._contextWasLost) { let e = !0; if ((this._isDisposed || !this.renderEvenInBackground && this._windowIsBackground) && (e = !1), e) { this.beginFrame(); for (let t = 0; t < this._activeRenderLoops.length; t++) { const r = this._activeRenderLoops[t]; r(); } this.endFrame(); } } this._activeRenderLoops.length > 0 ? this._frameHandler = this._queueNewFrame(this._boundRenderFunction, this.getHostWindow()) : this._renderingQueueLaunched = !1; } /** * Gets the HTML canvas attached with the current webGL context * @returns a HTML canvas */ getRenderingCanvas() { return this._renderingCanvas; } /** * Gets the audio context specified in engine initialization options * @returns an Audio Context */ getAudioContext() { return this._audioContext; } /** * Gets the audio destination specified in engine initialization options * @returns an audio destination node */ getAudioDestination() { return this._audioDestination; } /** * Gets host window * @returns the host window object */ getHostWindow() { return u9() ? this._renderingCanvas && this._renderingCanvas.ownerDocument && this._renderingCanvas.ownerDocument.defaultView ? this._renderingCanvas.ownerDocument.defaultView : window : null; } /** * Gets the current render width * @param useScreen defines if screen size must be used (or the current render target if any) * @returns a number defining the current render width */ getRenderWidth(e = !1) { return !e && this._currentRenderTarget ? this._currentRenderTarget.width : this._framebufferDimensionsObject ? this._framebufferDimensionsObject.framebufferWidth : this._gl.drawingBufferWidth; } /** * Gets the current render height * @param useScreen defines if screen size must be used (or the current render target if any) * @returns a number defining the current render height */ getRenderHeight(e = !1) { return !e && this._currentRenderTarget ? this._currentRenderTarget.height : this._framebufferDimensionsObject ? this._framebufferDimensionsObject.framebufferHeight : this._gl.drawingBufferHeight; } /** * Can be used to override the current requestAnimationFrame requester. * @internal */ _queueNewFrame(e, t) { return hr.QueueNewFrame(e, t); } /** * Register and execute a render loop. The engine can have more than one render function * @param renderFunction defines the function to continuously execute */ runRenderLoop(e) { this._activeRenderLoops.indexOf(e) === -1 && (this._activeRenderLoops.push(e), this._renderingQueueLaunched || (this._renderingQueueLaunched = !0, this._boundRenderFunction = () => this._renderLoop(), this._frameHandler = this._queueNewFrame(this._boundRenderFunction, this.getHostWindow()))); } /** * Clear the current render buffer or the current render target (if any is set up) * @param color defines the color to use * @param backBuffer defines if the back buffer must be cleared * @param depth defines if the depth buffer must be cleared * @param stencil defines if the stencil buffer must be cleared */ clear(e, t, r, n = !1) { var i, s; const a = this.stencilStateComposer.useStencilGlobalOnly; this.stencilStateComposer.useStencilGlobalOnly = !0, this.applyStates(), this.stencilStateComposer.useStencilGlobalOnly = a; let f = 0; if (t && e) { let o = !0; if (this._currentRenderTarget) { const d = (i = this._currentRenderTarget.texture) === null || i === void 0 ? void 0 : i.format; if (d === 8 || d === 9 || d === 10 || d === 11) { const v = (s = this._currentRenderTarget.texture) === null || s === void 0 ? void 0 : s.type; v === 7 || v === 5 ? (hr._TempClearColorUint32[0] = e.r * 255, hr._TempClearColorUint32[1] = e.g * 255, hr._TempClearColorUint32[2] = e.b * 255, hr._TempClearColorUint32[3] = e.a * 255, this._gl.clearBufferuiv(this._gl.COLOR, 0, hr._TempClearColorUint32), o = !1) : (hr._TempClearColorInt32[0] = e.r * 255, hr._TempClearColorInt32[1] = e.g * 255, hr._TempClearColorInt32[2] = e.b * 255, hr._TempClearColorInt32[3] = e.a * 255, this._gl.clearBufferiv(this._gl.COLOR, 0, hr._TempClearColorInt32), o = !1); } } o && (this._gl.clearColor(e.r, e.g, e.b, e.a !== void 0 ? e.a : 1), f |= this._gl.COLOR_BUFFER_BIT); } r && (this.useReverseDepthBuffer ? (this._depthCullingState.depthFunc = this._gl.GEQUAL, this._gl.clearDepth(0)) : this._gl.clearDepth(1), f |= this._gl.DEPTH_BUFFER_BIT), n && (this._gl.clearStencil(0), f |= this._gl.STENCIL_BUFFER_BIT), this._gl.clear(f); } /** * @internal */ _viewport(e, t, r, n) { (e !== this._viewportCached.x || t !== this._viewportCached.y || r !== this._viewportCached.z || n !== this._viewportCached.w) && (this._viewportCached.x = e, this._viewportCached.y = t, this._viewportCached.z = r, this._viewportCached.w = n, this._gl.viewport(e, t, r, n)); } /** * Set the WebGL's viewport * @param viewport defines the viewport element to be used * @param requiredWidth defines the width required for rendering. If not provided the rendering canvas' width is used * @param requiredHeight defines the height required for rendering. If not provided the rendering canvas' height is used */ setViewport(e, t, r) { const n = t || this.getRenderWidth(), i = r || this.getRenderHeight(), s = e.x || 0, a = e.y || 0; this._cachedViewport = e, this._viewport(s * n, a * i, n * e.width, i * e.height); } /** * Begin a new frame */ beginFrame() { } /** * Enf the current frame */ endFrame() { this._badOS && this.flushFramebuffer(), this._frameId++; } /** * Resize the view according to the canvas' size * @param forceSetSize true to force setting the sizes of the underlying canvas */ resize(e = !1) { let t, r; if (this.adaptToDeviceRatio) { const n = u9() && window.devicePixelRatio || 1, i = this._lastDevicePixelRatio / n; this._lastDevicePixelRatio = n, this._hardwareScalingLevel *= i; } if (u9() && $w()) if (this._renderingCanvas) { const n = this._renderingCanvas.getBoundingClientRect ? this._renderingCanvas.getBoundingClientRect() : { // fallback to last solution in case the function doesn't exist width: this._renderingCanvas.width * this._hardwareScalingLevel, height: this._renderingCanvas.height * this._hardwareScalingLevel }; t = this._renderingCanvas.clientWidth || n.width || this._renderingCanvas.width || 100, r = this._renderingCanvas.clientHeight || n.height || this._renderingCanvas.height || 100; } else t = window.innerWidth, r = window.innerHeight; else t = this._renderingCanvas ? this._renderingCanvas.width : 100, r = this._renderingCanvas ? this._renderingCanvas.height : 100; this.setSize(t / this._hardwareScalingLevel, r / this._hardwareScalingLevel, e); } /** * Force a specific size of the canvas * @param width defines the new canvas' width * @param height defines the new canvas' height * @param forceSetSize true to force setting the sizes of the underlying canvas * @returns true if the size was changed */ setSize(e, t, r = !1) { return !this._renderingCanvas || (e = e | 0, t = t | 0, !r && this._renderingCanvas.width === e && this._renderingCanvas.height === t) ? !1 : (this._renderingCanvas.width = e, this._renderingCanvas.height = t, !0); } /** * Binds the frame buffer to the specified texture. * @param rtWrapper The render target wrapper to render to * @param faceIndex The face of the texture to render to in case of cube texture and if the render target wrapper is not a multi render target * @param requiredWidth The width of the target to render to * @param requiredHeight The height of the target to render to * @param forceFullscreenViewport Forces the viewport to be the entire texture/screen if true * @param lodLevel Defines the lod level to bind to the frame buffer * @param layer Defines the 2d array index to bind to the frame buffer if the render target wrapper is not a multi render target */ bindFramebuffer(e, t = 0, r, n, i, s = 0, a = 0) { var f, o, d, v, u, l; const P = e; this._currentRenderTarget && this.unBindFramebuffer(this._currentRenderTarget), this._currentRenderTarget = e, this._bindUnboundFramebuffer(P._MSAAFramebuffer ? P._MSAAFramebuffer : P._framebuffer); const p = this._gl; e.isMulti || (e.is2DArray ? p.framebufferTextureLayer(p.FRAMEBUFFER, p.COLOR_ATTACHMENT0, (f = e.texture._hardwareTexture) === null || f === void 0 ? void 0 : f.underlyingResource, s, a) : e.isCube ? p.framebufferTexture2D(p.FRAMEBUFFER, p.COLOR_ATTACHMENT0, p.TEXTURE_CUBE_MAP_POSITIVE_X + t, (o = e.texture._hardwareTexture) === null || o === void 0 ? void 0 : o.underlyingResource, s) : P._currentLOD !== s && (p.framebufferTexture2D(p.FRAMEBUFFER, p.COLOR_ATTACHMENT0, p.TEXTURE_2D, (d = e.texture._hardwareTexture) === null || d === void 0 ? void 0 : d.underlyingResource, s), P._currentLOD = s)); const c = e._depthStencilTexture; if (c) { const H = e._depthStencilTextureWithStencil ? p.DEPTH_STENCIL_ATTACHMENT : p.DEPTH_ATTACHMENT; e.is2DArray ? p.framebufferTextureLayer(p.FRAMEBUFFER, H, (v = c._hardwareTexture) === null || v === void 0 ? void 0 : v.underlyingResource, s, a) : e.isCube ? p.framebufferTexture2D(p.FRAMEBUFFER, H, p.TEXTURE_CUBE_MAP_POSITIVE_X + t, (u = c._hardwareTexture) === null || u === void 0 ? void 0 : u.underlyingResource, s) : p.framebufferTexture2D(p.FRAMEBUFFER, H, p.TEXTURE_2D, (l = c._hardwareTexture) === null || l === void 0 ? void 0 : l.underlyingResource, s); } this._cachedViewport && !i ? this.setViewport(this._cachedViewport, r, n) : (r || (r = e.width, s && (r = r / Math.pow(2, s))), n || (n = e.height, s && (n = n / Math.pow(2, s))), this._viewport(0, 0, r, n)), this.wipeCaches(); } /** * Set various states to the webGL context * @param culling defines culling state: true to enable culling, false to disable it * @param zOffset defines the value to apply to zOffset (0 by default) * @param force defines if states must be applied even if cache is up to date * @param reverseSide defines if culling must be reversed (CCW if false, CW if true) * @param cullBackFaces true to cull back faces, false to cull front faces (if culling is enabled) * @param stencil stencil states to set * @param zOffsetUnits defines the value to apply to zOffsetUnits (0 by default) */ setState(e, t = 0, r, n = !1, i, s, a = 0) { var f, o; (this._depthCullingState.cull !== e || r) && (this._depthCullingState.cull = e); const d = !((o = (f = this.cullBackFaces) !== null && f !== void 0 ? f : i) !== null && o !== void 0) || o ? this._gl.BACK : this._gl.FRONT; (this._depthCullingState.cullFace !== d || r) && (this._depthCullingState.cullFace = d), this.setZOffset(t), this.setZOffsetUnits(a); const v = n ? this._gl.CW : this._gl.CCW; (this._depthCullingState.frontFace !== v || r) && (this._depthCullingState.frontFace = v), this._stencilStateComposer.stencilMaterial = s; } /** * Gets a boolean indicating if depth testing is enabled * @returns the current state */ getDepthBuffer() { return this._depthCullingState.depthTest; } /** * Enable or disable depth buffering * @param enable defines the state to set */ setDepthBuffer(e) { this._depthCullingState.depthTest = e; } /** * Set the z offset Factor to apply to current rendering * @param value defines the offset to apply */ setZOffset(e) { this._depthCullingState.zOffset = this.useReverseDepthBuffer ? -e : e; } /** * Gets the current value of the zOffset Factor * @returns the current zOffset Factor state */ getZOffset() { const e = this._depthCullingState.zOffset; return this.useReverseDepthBuffer ? -e : e; } /** * Set the z offset Units to apply to current rendering * @param value defines the offset to apply */ setZOffsetUnits(e) { this._depthCullingState.zOffsetUnits = this.useReverseDepthBuffer ? -e : e; } /** * Gets the current value of the zOffset Units * @returns the current zOffset Units state */ getZOffsetUnits() { const e = this._depthCullingState.zOffsetUnits; return this.useReverseDepthBuffer ? -e : e; } /** * @internal */ _bindUnboundFramebuffer(e) { this._currentFramebuffer !== e && (this._gl.bindFramebuffer(this._gl.FRAMEBUFFER, e), this._currentFramebuffer = e); } /** @internal */ _currentFrameBufferIsDefaultFrameBuffer() { return this._currentFramebuffer === null; } /** * Generates the mipmaps for a texture * @param texture texture to generate the mipmaps for */ generateMipmaps(e) { this._bindTextureDirectly(this._gl.TEXTURE_2D, e, !0), this._gl.generateMipmap(this._gl.TEXTURE_2D), this._bindTextureDirectly(this._gl.TEXTURE_2D, null); } /** * Unbind the current render target texture from the webGL context * @param texture defines the render target wrapper to unbind * @param disableGenerateMipMaps defines a boolean indicating that mipmaps must not be generated * @param onBeforeUnbind defines a function which will be called before the effective unbind */ unBindFramebuffer(e, t = !1, r) { var n; const i = e; this._currentRenderTarget = null; const s = this._gl; if (i._MSAAFramebuffer) { if (e.isMulti) { this.unBindMultiColorAttachmentFramebuffer(e, t, r); return; } s.bindFramebuffer(s.READ_FRAMEBUFFER, i._MSAAFramebuffer), s.bindFramebuffer(s.DRAW_FRAMEBUFFER, i._framebuffer), s.blitFramebuffer(0, 0, e.width, e.height, 0, 0, e.width, e.height, s.COLOR_BUFFER_BIT, s.NEAREST); } !((n = e.texture) === null || n === void 0) && n.generateMipMaps && !t && !e.isCube && this.generateMipmaps(e.texture), r && (i._MSAAFramebuffer && this._bindUnboundFramebuffer(i._framebuffer), r()), this._bindUnboundFramebuffer(null); } /** * Force a webGL flush (ie. a flush of all waiting webGL commands) */ flushFramebuffer() { this._gl.flush(); } /** * Unbind the current render target and bind the default framebuffer */ restoreDefaultFramebuffer() { this._currentRenderTarget ? this.unBindFramebuffer(this._currentRenderTarget) : this._bindUnboundFramebuffer(null), this._cachedViewport && this.setViewport(this._cachedViewport), this.wipeCaches(); } // VBOs /** @internal */ _resetVertexBufferBinding() { this.bindArrayBuffer(null), this._cachedVertexBuffers = null; } /** * Creates a vertex buffer * @param data the data for the vertex buffer * @param _updatable whether the buffer should be created as updatable * @param _label defines the label of the buffer (for debug purpose) * @returns the new WebGL static buffer */ createVertexBuffer(e, t, r) { return this._createVertexBuffer(e, this._gl.STATIC_DRAW); } _createVertexBuffer(e, t) { const r = this._gl.createBuffer(); if (!r) throw new Error("Unable to create vertex buffer"); const n = new jS(r); return this.bindArrayBuffer(n), e instanceof Array ? this._gl.bufferData(this._gl.ARRAY_BUFFER, new Float32Array(e), t) : this._gl.bufferData(this._gl.ARRAY_BUFFER, e, t), this._resetVertexBufferBinding(), n.references = 1, n; } /** * Creates a dynamic vertex buffer * @param data the data for the dynamic vertex buffer * @param _label defines the label of the buffer (for debug purpose) * @returns the new WebGL dynamic buffer */ createDynamicVertexBuffer(e, t) { return this._createVertexBuffer(e, this._gl.DYNAMIC_DRAW); } _resetIndexBufferBinding() { this.bindIndexBuffer(null), this._cachedIndexBuffer = null; } /** * Creates a new index buffer * @param indices defines the content of the index buffer * @param updatable defines if the index buffer must be updatable * @param _label defines the label of the buffer (for debug purpose) * @returns a new webGL buffer */ createIndexBuffer(e, t, r) { const n = this._gl.createBuffer(), i = new jS(n); if (!n) throw new Error("Unable to create index buffer"); this.bindIndexBuffer(i); const s = this._normalizeIndexData(e); return this._gl.bufferData(this._gl.ELEMENT_ARRAY_BUFFER, s, t ? this._gl.DYNAMIC_DRAW : this._gl.STATIC_DRAW), this._resetIndexBufferBinding(), i.references = 1, i.is32Bits = s.BYTES_PER_ELEMENT === 4, i; } _normalizeIndexData(e) { if (e.BYTES_PER_ELEMENT === 2) return e; if (this._caps.uintIndices) { if (e instanceof Uint32Array) return e; for (let r = 0; r < e.length; r++) if (e[r] >= 65535) return new Uint32Array(e); return new Uint16Array(e); } return new Uint16Array(e); } /** * Bind a webGL buffer to the webGL context * @param buffer defines the buffer to bind */ bindArrayBuffer(e) { this._vaoRecordInProgress || this._unbindVertexArrayObject(), this._bindBuffer(e, this._gl.ARRAY_BUFFER); } /** * Bind a specific block at a given index in a specific shader program * @param pipelineContext defines the pipeline context to use * @param blockName defines the block name * @param index defines the index where to bind the block */ bindUniformBlock(e, t, r) { const n = e.program, i = this._gl.getUniformBlockIndex(n, t); this._gl.uniformBlockBinding(n, i, r); } // eslint-disable-next-line @typescript-eslint/naming-convention bindIndexBuffer(e) { this._vaoRecordInProgress || this._unbindVertexArrayObject(), this._bindBuffer(e, this._gl.ELEMENT_ARRAY_BUFFER); } _bindBuffer(e, t) { (this._vaoRecordInProgress || this._currentBoundBuffer[t] !== e) && (this._gl.bindBuffer(t, e ? e.underlyingResource : null), this._currentBoundBuffer[t] = e); } /** * update the bound buffer with the given data * @param data defines the data to update */ updateArrayBuffer(e) { this._gl.bufferSubData(this._gl.ARRAY_BUFFER, 0, e); } _vertexAttribPointer(e, t, r, n, i, s, a) { const f = this._currentBufferPointers[t]; if (!f) return; let o = !1; f.active ? (f.buffer !== e && (f.buffer = e, o = !0), f.size !== r && (f.size = r, o = !0), f.type !== n && (f.type = n, o = !0), f.normalized !== i && (f.normalized = i, o = !0), f.stride !== s && (f.stride = s, o = !0), f.offset !== a && (f.offset = a, o = !0)) : (o = !0, f.active = !0, f.index = t, f.size = r, f.type = n, f.normalized = i, f.stride = s, f.offset = a, f.buffer = e), (o || this._vaoRecordInProgress) && (this.bindArrayBuffer(e), n === this._gl.UNSIGNED_INT || n === this._gl.INT ? this._gl.vertexAttribIPointer(t, r, n, s, a) : this._gl.vertexAttribPointer(t, r, n, i, s, a)); } /** * @internal */ _bindIndexBufferWithCache(e) { e != null && this._cachedIndexBuffer !== e && (this._cachedIndexBuffer = e, this.bindIndexBuffer(e), this._uintIndicesCurrentlySet = e.is32Bits); } _bindVertexBuffersAttributes(e, t, r) { const n = t.getAttributesNames(); this._vaoRecordInProgress || this._unbindVertexArrayObject(), this.unbindAllAttributes(); for (let i = 0; i < n.length; i++) { const s = t.getAttributeLocation(i); if (s >= 0) { const a = n[i]; let f = null; if (r && (f = r[a]), f || (f = e[a]), !f) continue; this._gl.enableVertexAttribArray(s), this._vaoRecordInProgress || (this._vertexAttribArraysEnabled[s] = !0); const o = f.getBuffer(); o && (this._vertexAttribPointer(o, s, f.getSize(), f.type, f.normalized, f.byteStride, f.byteOffset), f.getIsInstanced() && (this._gl.vertexAttribDivisor(s, f.getInstanceDivisor()), this._vaoRecordInProgress || (this._currentInstanceLocations.push(s), this._currentInstanceBuffers.push(o)))); } } } /** * Records a vertex array object * @see https://doc.babylonjs.com/setup/support/webGL2#vertex-array-objects * @param vertexBuffers defines the list of vertex buffers to store * @param indexBuffer defines the index buffer to store * @param effect defines the effect to store * @param overrideVertexBuffers defines optional list of avertex buffers that overrides the entries in vertexBuffers * @returns the new vertex array object */ recordVertexArrayObject(e, t, r, n) { const i = this._gl.createVertexArray(); if (!i) throw new Error("Unable to create VAO"); return this._vaoRecordInProgress = !0, this._gl.bindVertexArray(i), this._mustWipeVertexAttributes = !0, this._bindVertexBuffersAttributes(e, r, n), this.bindIndexBuffer(t), this._vaoRecordInProgress = !1, this._gl.bindVertexArray(null), i; } /** * Bind a specific vertex array object * @see https://doc.babylonjs.com/setup/support/webGL2#vertex-array-objects * @param vertexArrayObject defines the vertex array object to bind * @param indexBuffer defines the index buffer to bind */ bindVertexArrayObject(e, t) { this._cachedVertexArrayObject !== e && (this._cachedVertexArrayObject = e, this._gl.bindVertexArray(e), this._cachedVertexBuffers = null, this._cachedIndexBuffer = null, this._uintIndicesCurrentlySet = t != null && t.is32Bits, this._mustWipeVertexAttributes = !0); } /** * Bind webGl buffers directly to the webGL context * @param vertexBuffer defines the vertex buffer to bind * @param indexBuffer defines the index buffer to bind * @param vertexDeclaration defines the vertex declaration to use with the vertex buffer * @param vertexStrideSize defines the vertex stride of the vertex buffer * @param effect defines the effect associated with the vertex buffer */ bindBuffersDirectly(e, t, r, n, i) { if (this._cachedVertexBuffers !== e || this._cachedEffectForVertexBuffers !== i) { this._cachedVertexBuffers = e, this._cachedEffectForVertexBuffers = i; const s = i.getAttributesCount(); this._unbindVertexArrayObject(), this.unbindAllAttributes(); let a = 0; for (let f = 0; f < s; f++) if (f < r.length) { const o = i.getAttributeLocation(f); o >= 0 && (this._gl.enableVertexAttribArray(o), this._vertexAttribArraysEnabled[o] = !0, this._vertexAttribPointer(e, o, r[f], this._gl.FLOAT, !1, n, a)), a += r[f] * 4; } } this._bindIndexBufferWithCache(t); } _unbindVertexArrayObject() { this._cachedVertexArrayObject && (this._cachedVertexArrayObject = null, this._gl.bindVertexArray(null)); } /** * Bind a list of vertex buffers to the webGL context * @param vertexBuffers defines the list of vertex buffers to bind * @param indexBuffer defines the index buffer to bind * @param effect defines the effect associated with the vertex buffers * @param overrideVertexBuffers defines optional list of avertex buffers that overrides the entries in vertexBuffers */ bindBuffers(e, t, r, n) { (this._cachedVertexBuffers !== e || this._cachedEffectForVertexBuffers !== r) && (this._cachedVertexBuffers = e, this._cachedEffectForVertexBuffers = r, this._bindVertexBuffersAttributes(e, r, n)), this._bindIndexBufferWithCache(t); } /** * Unbind all instance attributes */ unbindInstanceAttributes() { let e; for (let t = 0, r = this._currentInstanceLocations.length; t < r; t++) { const n = this._currentInstanceBuffers[t]; e != n && n.references && (e = n, this.bindArrayBuffer(n)); const i = this._currentInstanceLocations[t]; this._gl.vertexAttribDivisor(i, 0); } this._currentInstanceBuffers.length = 0, this._currentInstanceLocations.length = 0; } /** * Release and free the memory of a vertex array object * @param vao defines the vertex array object to delete */ releaseVertexArrayObject(e) { this._gl.deleteVertexArray(e); } /** * @internal */ _releaseBuffer(e) { return e.references--, e.references === 0 ? (this._deleteBuffer(e), !0) : !1; } _deleteBuffer(e) { this._gl.deleteBuffer(e.underlyingResource); } /** * Update the content of a webGL buffer used with instantiation and bind it to the webGL context * @param instancesBuffer defines the webGL buffer to update and bind * @param data defines the data to store in the buffer * @param offsetLocations defines the offsets or attributes information used to determine where data must be stored in the buffer */ updateAndBindInstancesBuffer(e, t, r) { if (this.bindArrayBuffer(e), t && this._gl.bufferSubData(this._gl.ARRAY_BUFFER, 0, t), r[0].index !== void 0) this.bindInstancesBuffer(e, r, !0); else for (let n = 0; n < 4; n++) { const i = r[n]; this._vertexAttribArraysEnabled[i] || (this._gl.enableVertexAttribArray(i), this._vertexAttribArraysEnabled[i] = !0), this._vertexAttribPointer(e, i, 4, this._gl.FLOAT, !1, 64, n * 16), this._gl.vertexAttribDivisor(i, 1), this._currentInstanceLocations.push(i), this._currentInstanceBuffers.push(e); } } /** * Bind the content of a webGL buffer used with instantiation * @param instancesBuffer defines the webGL buffer to bind * @param attributesInfo defines the offsets or attributes information used to determine where data must be stored in the buffer * @param computeStride defines Whether to compute the strides from the info or use the default 0 */ bindInstancesBuffer(e, t, r = !0) { this.bindArrayBuffer(e); let n = 0; if (r) for (let i = 0; i < t.length; i++) { const s = t[i]; n += s.attributeSize * 4; } for (let i = 0; i < t.length; i++) { const s = t[i]; s.index === void 0 && (s.index = this._currentEffect.getAttributeLocationByName(s.attributeName)), !(s.index < 0) && (this._vertexAttribArraysEnabled[s.index] || (this._gl.enableVertexAttribArray(s.index), this._vertexAttribArraysEnabled[s.index] = !0), this._vertexAttribPointer(e, s.index, s.attributeSize, s.attributeType || this._gl.FLOAT, s.normalized || !1, n, s.offset), this._gl.vertexAttribDivisor(s.index, s.divisor === void 0 ? 1 : s.divisor), this._currentInstanceLocations.push(s.index), this._currentInstanceBuffers.push(e)); } } /** * Disable the instance attribute corresponding to the name in parameter * @param name defines the name of the attribute to disable */ disableInstanceAttributeByName(e) { if (!this._currentEffect) return; const t = this._currentEffect.getAttributeLocationByName(e); this.disableInstanceAttribute(t); } /** * Disable the instance attribute corresponding to the location in parameter * @param attributeLocation defines the attribute location of the attribute to disable */ disableInstanceAttribute(e) { let t = !1, r; for (; (r = this._currentInstanceLocations.indexOf(e)) !== -1; ) this._currentInstanceLocations.splice(r, 1), this._currentInstanceBuffers.splice(r, 1), t = !0, r = this._currentInstanceLocations.indexOf(e); t && (this._gl.vertexAttribDivisor(e, 0), this.disableAttributeByIndex(e)); } /** * Disable the attribute corresponding to the location in parameter * @param attributeLocation defines the attribute location of the attribute to disable */ disableAttributeByIndex(e) { this._gl.disableVertexAttribArray(e), this._vertexAttribArraysEnabled[e] = !1, this._currentBufferPointers[e].active = !1; } /** * Send a draw order * @param useTriangles defines if triangles must be used to draw (else wireframe will be used) * @param indexStart defines the starting index * @param indexCount defines the number of index to draw * @param instancesCount defines the number of instances to draw (if instantiation is enabled) */ draw(e, t, r, n) { this.drawElementsType(e ? 0 : 1, t, r, n); } /** * Draw a list of points * @param verticesStart defines the index of first vertex to draw * @param verticesCount defines the count of vertices to draw * @param instancesCount defines the number of instances to draw (if instantiation is enabled) */ drawPointClouds(e, t, r) { this.drawArraysType(2, e, t, r); } /** * Draw a list of unindexed primitives * @param useTriangles defines if triangles must be used to draw (else wireframe will be used) * @param verticesStart defines the index of first vertex to draw * @param verticesCount defines the count of vertices to draw * @param instancesCount defines the number of instances to draw (if instantiation is enabled) */ drawUnIndexed(e, t, r, n) { this.drawArraysType(e ? 0 : 1, t, r, n); } /** * Draw a list of indexed primitives * @param fillMode defines the primitive to use * @param indexStart defines the starting index * @param indexCount defines the number of index to draw * @param instancesCount defines the number of instances to draw (if instantiation is enabled) */ drawElementsType(e, t, r, n) { this.applyStates(), this._reportDrawCall(); const i = this._drawMode(e), s = this._uintIndicesCurrentlySet ? this._gl.UNSIGNED_INT : this._gl.UNSIGNED_SHORT, a = this._uintIndicesCurrentlySet ? 4 : 2; n ? this._gl.drawElementsInstanced(i, r, s, t * a, n) : this._gl.drawElements(i, r, s, t * a); } /** * Draw a list of unindexed primitives * @param fillMode defines the primitive to use * @param verticesStart defines the index of first vertex to draw * @param verticesCount defines the count of vertices to draw * @param instancesCount defines the number of instances to draw (if instantiation is enabled) */ drawArraysType(e, t, r, n) { this.applyStates(), this._reportDrawCall(); const i = this._drawMode(e); n ? this._gl.drawArraysInstanced(i, t, r, n) : this._gl.drawArrays(i, t, r); } _drawMode(e) { switch (e) { case 0: return this._gl.TRIANGLES; case 2: return this._gl.POINTS; case 1: return this._gl.LINES; case 3: return this._gl.POINTS; case 4: return this._gl.LINES; case 5: return this._gl.LINE_LOOP; case 6: return this._gl.LINE_STRIP; case 7: return this._gl.TRIANGLE_STRIP; case 8: return this._gl.TRIANGLE_FAN; default: return this._gl.TRIANGLES; } } /** @internal */ _reportDrawCall() { } // Shaders /** * @internal */ _releaseEffect(e) { this._compiledEffects[e._key] && delete this._compiledEffects[e._key]; const t = e.getPipelineContext(); t && this._deletePipelineContext(t); } /** * @internal */ _deletePipelineContext(e) { const t = e; t && t.program && (t.program.__SPECTOR_rebuildProgram = null, this._gl.deleteProgram(t.program)); } /** @internal */ _getGlobalDefines(e) { if (e) { this.isNDCHalfZRange ? e.IS_NDC_HALF_ZRANGE = "" : delete e.IS_NDC_HALF_ZRANGE, this.useReverseDepthBuffer ? e.USE_REVERSE_DEPTHBUFFER = "" : delete e.USE_REVERSE_DEPTHBUFFER, this.useExactSrgbConversions ? e.USE_EXACT_SRGB_CONVERSIONS = "" : delete e.USE_EXACT_SRGB_CONVERSIONS; return; } else { let t = ""; return this.isNDCHalfZRange && (t += "#define IS_NDC_HALF_ZRANGE"), this.useReverseDepthBuffer && (t && (t += ` `), t += "#define USE_REVERSE_DEPTHBUFFER"), this.useExactSrgbConversions && (t && (t += ` `), t += "#define USE_EXACT_SRGB_CONVERSIONS"), t; } } /** * Create a new effect (used to store vertex/fragment shaders) * @param baseName defines the base name of the effect (The name of file without .fragment.fx or .vertex.fx) * @param attributesNamesOrOptions defines either a list of attribute names or an IEffectCreationOptions object * @param uniformsNamesOrEngine defines either a list of uniform names or the engine to use * @param samplers defines an array of string used to represent textures * @param defines defines the string containing the defines to use to compile the shaders * @param fallbacks defines the list of potential fallbacks to use if shader compilation fails * @param onCompiled defines a function to call when the effect creation is successful * @param onError defines a function to call when the effect creation has failed * @param indexParameters defines an object containing the index values to use to compile shaders (like the maximum number of simultaneous lights) * @param shaderLanguage the language the shader is written in (default: GLSL) * @returns the new Effect */ createEffect(e, t, r, n, i, s, a, f, o, d = za.GLSL) { var v; const u = e.vertexElement || e.vertex || e.vertexToken || e.vertexSource || e, l = e.fragmentElement || e.fragment || e.fragmentToken || e.fragmentSource || e, P = this._getGlobalDefines(); let p = (v = i ?? t.defines) !== null && v !== void 0 ? v : ""; P && (p += P); const c = u + "+" + l + "@" + p; if (this._compiledEffects[c]) { const T = this._compiledEffects[c]; return a && T.isReady() && a(T), T; } const H = new An(e, t, r, n, this, i, s, a, f, o, c, d); return this._compiledEffects[c] = H, H; } // eslint-disable-next-line @typescript-eslint/naming-convention static _ConcatenateShader(e, t, r = "") { return r + (t ? t + ` ` : "") + e; } _compileShader(e, t, r, n) { return this._compileRawShader(hr._ConcatenateShader(e, r, n), t); } _compileRawShader(e, t) { const r = this._gl, n = r.createShader(t === "vertex" ? r.VERTEX_SHADER : r.FRAGMENT_SHADER); if (!n) { let i = r.NO_ERROR, s = r.NO_ERROR; for (; (s = r.getError()) !== r.NO_ERROR; ) i = s; throw new Error(`Something went wrong while creating a gl ${t} shader object. gl error=${i}, gl isContextLost=${r.isContextLost()}, _contextWasLost=${this._contextWasLost}`); } return r.shaderSource(n, e), r.compileShader(n), n; } /** * @internal */ _getShaderSource(e) { return this._gl.getShaderSource(e); } /** * Directly creates a webGL program * @param pipelineContext defines the pipeline context to attach to * @param vertexCode defines the vertex shader code to use * @param fragmentCode defines the fragment shader code to use * @param context defines the webGL context to use (if not set, the current one will be used) * @param transformFeedbackVaryings defines the list of transform feedback varyings to use * @returns the new webGL program */ createRawShaderProgram(e, t, r, n, i = null) { n = n || this._gl; const s = this._compileRawShader(t, "vertex"), a = this._compileRawShader(r, "fragment"); return this._createShaderProgram(e, s, a, n, i); } /** * Creates a webGL program * @param pipelineContext defines the pipeline context to attach to * @param vertexCode defines the vertex shader code to use * @param fragmentCode defines the fragment shader code to use * @param defines defines the string containing the defines to use to compile the shaders * @param context defines the webGL context to use (if not set, the current one will be used) * @param transformFeedbackVaryings defines the list of transform feedback varyings to use * @returns the new webGL program */ createShaderProgram(e, t, r, n, i, s = null) { i = i || this._gl; const a = this._webGLVersion > 1 ? `#version 300 es #define WEBGL2 ` : "", f = this._compileShader(t, "vertex", n, a), o = this._compileShader(r, "fragment", n, a); return this._createShaderProgram(e, f, o, i, s); } /** * Inline functions in shader code that are marked to be inlined * @param code code to inline * @returns inlined code */ inlineShaderCode(e) { return e; } /** * Creates a new pipeline context * @param shaderProcessingContext defines the shader processing context used during the processing if available * @returns the new pipeline */ createPipelineContext(e) { const t = new E$(); return t.engine = this, this._caps.parallelShaderCompile && (t.isParallelCompiled = !0), t; } /** * Creates a new material context * @returns the new context */ createMaterialContext() { } /** * Creates a new draw context * @returns the new context */ createDrawContext() { } _createShaderProgram(e, t, r, n, i = null) { const s = n.createProgram(); if (e.program = s, !s) throw new Error("Unable to create program"); return n.attachShader(s, t), n.attachShader(s, r), n.linkProgram(s), e.context = n, e.vertexShader = t, e.fragmentShader = r, e.isParallelCompiled || this._finalizePipelineContext(e), s; } _finalizePipelineContext(e) { const t = e.context, r = e.vertexShader, n = e.fragmentShader, i = e.program; if (!t.getProgramParameter(i, t.LINK_STATUS)) { if (!this._gl.getShaderParameter(r, this._gl.COMPILE_STATUS)) { const f = this._gl.getShaderInfoLog(r); if (f) throw e.vertexCompilationError = f, new Error("VERTEX SHADER " + f); } if (!this._gl.getShaderParameter(n, this._gl.COMPILE_STATUS)) { const f = this._gl.getShaderInfoLog(n); if (f) throw e.fragmentCompilationError = f, new Error("FRAGMENT SHADER " + f); } const a = t.getProgramInfoLog(i); if (a) throw e.programLinkError = a, new Error(a); } if (this.validateShaderPrograms && (t.validateProgram(i), !t.getProgramParameter(i, t.VALIDATE_STATUS))) { const f = t.getProgramInfoLog(i); if (f) throw e.programValidationError = f, new Error(f); } t.deleteShader(r), t.deleteShader(n), e.vertexShader = void 0, e.fragmentShader = void 0, e.onCompiled && (e.onCompiled(), e.onCompiled = void 0); } /** * @internal */ _preparePipelineContext(e, t, r, n, i, s, a, f, o, d) { const v = e; n ? v.program = this.createRawShaderProgram(v, t, r, void 0, o) : v.program = this.createShaderProgram(v, t, r, f, void 0, o), v.program.__SPECTOR_rebuildProgram = a; } /** * @internal */ _isRenderingStateCompiled(e) { const t = e; return this._isDisposed || t._isDisposed ? !1 : this._gl.getProgramParameter(t.program, this._caps.parallelShaderCompile.COMPLETION_STATUS_KHR) ? (this._finalizePipelineContext(t), !0) : !1; } /** * @internal */ _executeWhenRenderingStateIsCompiled(e, t) { const r = e; if (!r.isParallelCompiled) { t(); return; } const n = r.onCompiled; n ? r.onCompiled = () => { n(), t(); } : r.onCompiled = t; } /** * Gets the list of webGL uniform locations associated with a specific program based on a list of uniform names * @param pipelineContext defines the pipeline context to use * @param uniformsNames defines the list of uniform names * @returns an array of webGL uniform locations */ getUniforms(e, t) { const r = new Array(), n = e; for (let i = 0; i < t.length; i++) r.push(this._gl.getUniformLocation(n.program, t[i])); return r; } /** * Gets the list of active attributes for a given webGL program * @param pipelineContext defines the pipeline context to use * @param attributesNames defines the list of attribute names to get * @returns an array of indices indicating the offset of each attribute */ getAttributes(e, t) { const r = [], n = e; for (let i = 0; i < t.length; i++) try { r.push(this._gl.getAttribLocation(n.program, t[i])); } catch { r.push(-1); } return r; } /** * Activates an effect, making it the current one (ie. the one used for rendering) * @param effect defines the effect to activate */ enableEffect(e) { e = e !== null && zo.IsWrapper(e) ? e.effect : e, !(!e || e === this._currentEffect) && (this._stencilStateComposer.stencilMaterial = void 0, e = e, this.bindSamplers(e), this._currentEffect = e, e.onBind && e.onBind(e), e._onBindObservable && e._onBindObservable.notifyObservers(e)); } /** * Set the value of an uniform to a number (int) * @param uniform defines the webGL uniform location where to store the value * @param value defines the int number to store * @returns true if the value was set */ setInt(e, t) { return e ? (this._gl.uniform1i(e, t), !0) : !1; } /** * Set the value of an uniform to a int2 * @param uniform defines the webGL uniform location where to store the value * @param x defines the 1st component of the value * @param y defines the 2nd component of the value * @returns true if the value was set */ setInt2(e, t, r) { return e ? (this._gl.uniform2i(e, t, r), !0) : !1; } /** * Set the value of an uniform to a int3 * @param uniform defines the webGL uniform location where to store the value * @param x defines the 1st component of the value * @param y defines the 2nd component of the value * @param z defines the 3rd component of the value * @returns true if the value was set */ setInt3(e, t, r, n) { return e ? (this._gl.uniform3i(e, t, r, n), !0) : !1; } /** * Set the value of an uniform to a int4 * @param uniform defines the webGL uniform location where to store the value * @param x defines the 1st component of the value * @param y defines the 2nd component of the value * @param z defines the 3rd component of the value * @param w defines the 4th component of the value * @returns true if the value was set */ setInt4(e, t, r, n, i) { return e ? (this._gl.uniform4i(e, t, r, n, i), !0) : !1; } /** * Set the value of an uniform to an array of int32 * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of int32 to store * @returns true if the value was set */ setIntArray(e, t) { return e ? (this._gl.uniform1iv(e, t), !0) : !1; } /** * Set the value of an uniform to an array of int32 (stored as vec2) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of int32 to store * @returns true if the value was set */ setIntArray2(e, t) { return !e || t.length % 2 !== 0 ? !1 : (this._gl.uniform2iv(e, t), !0); } /** * Set the value of an uniform to an array of int32 (stored as vec3) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of int32 to store * @returns true if the value was set */ setIntArray3(e, t) { return !e || t.length % 3 !== 0 ? !1 : (this._gl.uniform3iv(e, t), !0); } /** * Set the value of an uniform to an array of int32 (stored as vec4) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of int32 to store * @returns true if the value was set */ setIntArray4(e, t) { return !e || t.length % 4 !== 0 ? !1 : (this._gl.uniform4iv(e, t), !0); } /** * Set the value of an uniform to a number (unsigned int) * @param uniform defines the webGL uniform location where to store the value * @param value defines the unsigned int number to store * @returns true if the value was set */ setUInt(e, t) { return e ? (this._gl.uniform1ui(e, t), !0) : !1; } /** * Set the value of an uniform to a unsigned int2 * @param uniform defines the webGL uniform location where to store the value * @param x defines the 1st component of the value * @param y defines the 2nd component of the value * @returns true if the value was set */ setUInt2(e, t, r) { return e ? (this._gl.uniform2ui(e, t, r), !0) : !1; } /** * Set the value of an uniform to a unsigned int3 * @param uniform defines the webGL uniform location where to store the value * @param x defines the 1st component of the value * @param y defines the 2nd component of the value * @param z defines the 3rd component of the value * @returns true if the value was set */ setUInt3(e, t, r, n) { return e ? (this._gl.uniform3ui(e, t, r, n), !0) : !1; } /** * Set the value of an uniform to a unsigned int4 * @param uniform defines the webGL uniform location where to store the value * @param x defines the 1st component of the value * @param y defines the 2nd component of the value * @param z defines the 3rd component of the value * @param w defines the 4th component of the value * @returns true if the value was set */ setUInt4(e, t, r, n, i) { return e ? (this._gl.uniform4ui(e, t, r, n, i), !0) : !1; } /** * Set the value of an uniform to an array of unsigned int32 * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of unsigned int32 to store * @returns true if the value was set */ setUIntArray(e, t) { return e ? (this._gl.uniform1uiv(e, t), !0) : !1; } /** * Set the value of an uniform to an array of unsigned int32 (stored as vec2) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of unsigned int32 to store * @returns true if the value was set */ setUIntArray2(e, t) { return !e || t.length % 2 !== 0 ? !1 : (this._gl.uniform2uiv(e, t), !0); } /** * Set the value of an uniform to an array of unsigned int32 (stored as vec3) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of unsigned int32 to store * @returns true if the value was set */ setUIntArray3(e, t) { return !e || t.length % 3 !== 0 ? !1 : (this._gl.uniform3uiv(e, t), !0); } /** * Set the value of an uniform to an array of unsigned int32 (stored as vec4) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of unsigned int32 to store * @returns true if the value was set */ setUIntArray4(e, t) { return !e || t.length % 4 !== 0 ? !1 : (this._gl.uniform4uiv(e, t), !0); } /** * Set the value of an uniform to an array of number * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of number to store * @returns true if the value was set */ setArray(e, t) { return !e || t.length < 1 ? !1 : (this._gl.uniform1fv(e, t), !0); } /** * Set the value of an uniform to an array of number (stored as vec2) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of number to store * @returns true if the value was set */ setArray2(e, t) { return !e || t.length % 2 !== 0 ? !1 : (this._gl.uniform2fv(e, t), !0); } /** * Set the value of an uniform to an array of number (stored as vec3) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of number to store * @returns true if the value was set */ setArray3(e, t) { return !e || t.length % 3 !== 0 ? !1 : (this._gl.uniform3fv(e, t), !0); } /** * Set the value of an uniform to an array of number (stored as vec4) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of number to store * @returns true if the value was set */ setArray4(e, t) { return !e || t.length % 4 !== 0 ? !1 : (this._gl.uniform4fv(e, t), !0); } /** * Set the value of an uniform to an array of float32 (stored as matrices) * @param uniform defines the webGL uniform location where to store the value * @param matrices defines the array of float32 to store * @returns true if the value was set */ setMatrices(e, t) { return e ? (this._gl.uniformMatrix4fv(e, !1, t), !0) : !1; } /** * Set the value of an uniform to a matrix (3x3) * @param uniform defines the webGL uniform location where to store the value * @param matrix defines the Float32Array representing the 3x3 matrix to store * @returns true if the value was set */ setMatrix3x3(e, t) { return e ? (this._gl.uniformMatrix3fv(e, !1, t), !0) : !1; } /** * Set the value of an uniform to a matrix (2x2) * @param uniform defines the webGL uniform location where to store the value * @param matrix defines the Float32Array representing the 2x2 matrix to store * @returns true if the value was set */ setMatrix2x2(e, t) { return e ? (this._gl.uniformMatrix2fv(e, !1, t), !0) : !1; } /** * Set the value of an uniform to a number (float) * @param uniform defines the webGL uniform location where to store the value * @param value defines the float number to store * @returns true if the value was transferred */ setFloat(e, t) { return e ? (this._gl.uniform1f(e, t), !0) : !1; } /** * Set the value of an uniform to a vec2 * @param uniform defines the webGL uniform location where to store the value * @param x defines the 1st component of the value * @param y defines the 2nd component of the value * @returns true if the value was set */ setFloat2(e, t, r) { return e ? (this._gl.uniform2f(e, t, r), !0) : !1; } /** * Set the value of an uniform to a vec3 * @param uniform defines the webGL uniform location where to store the value * @param x defines the 1st component of the value * @param y defines the 2nd component of the value * @param z defines the 3rd component of the value * @returns true if the value was set */ setFloat3(e, t, r, n) { return e ? (this._gl.uniform3f(e, t, r, n), !0) : !1; } /** * Set the value of an uniform to a vec4 * @param uniform defines the webGL uniform location where to store the value * @param x defines the 1st component of the value * @param y defines the 2nd component of the value * @param z defines the 3rd component of the value * @param w defines the 4th component of the value * @returns true if the value was set */ setFloat4(e, t, r, n, i) { return e ? (this._gl.uniform4f(e, t, r, n, i), !0) : !1; } // States /** * Apply all cached states (depth, culling, stencil and alpha) */ applyStates() { if (this._depthCullingState.apply(this._gl), this._stencilStateComposer.apply(this._gl), this._alphaState.apply(this._gl), this._colorWriteChanged) { this._colorWriteChanged = !1; const e = this._colorWrite; this._gl.colorMask(e, e, e, e); } } /** * Enable or disable color writing * @param enable defines the state to set */ setColorWrite(e) { e !== this._colorWrite && (this._colorWriteChanged = !0, this._colorWrite = e); } /** * Gets a boolean indicating if color writing is enabled * @returns the current color writing state */ getColorWrite() { return this._colorWrite; } /** * Gets the depth culling state manager */ get depthCullingState() { return this._depthCullingState; } /** * Gets the alpha state manager */ get alphaState() { return this._alphaState; } /** * Gets the stencil state manager */ get stencilState() { return this._stencilState; } /** * Gets the stencil state composer */ get stencilStateComposer() { return this._stencilStateComposer; } // Textures /** * Clears the list of texture accessible through engine. * This can help preventing texture load conflict due to name collision. */ clearInternalTexturesCache() { this._internalTexturesCache.length = 0; } /** * Force the entire cache to be cleared * You should not have to use this function unless your engine needs to share the webGL context with another engine * @param bruteForce defines a boolean to force clearing ALL caches (including stencil, detoh and alpha states) */ wipeCaches(e) { this.preventCacheWipeBetweenFrames && !e || (this._currentEffect = null, this._viewportCached.x = 0, this._viewportCached.y = 0, this._viewportCached.z = 0, this._viewportCached.w = 0, this._unbindVertexArrayObject(), e && (this._currentProgram = null, this.resetTextureCache(), this._stencilStateComposer.reset(), this._depthCullingState.reset(), this._depthCullingState.depthFunc = this._gl.LEQUAL, this._alphaState.reset(), this._alphaMode = 1, this._alphaEquation = 0, this._colorWrite = !0, this._colorWriteChanged = !0, this._unpackFlipYCached = null, this._gl.pixelStorei(this._gl.UNPACK_COLORSPACE_CONVERSION_WEBGL, this._gl.NONE), this._gl.pixelStorei(this._gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, 0), this._mustWipeVertexAttributes = !0, this.unbindAllAttributes()), this._resetVertexBufferBinding(), this._cachedIndexBuffer = null, this._cachedEffectForVertexBuffers = null, this.bindIndexBuffer(null)); } /** * @internal */ _getSamplingParameters(e, t) { const r = this._gl; let n = r.NEAREST, i = r.NEAREST; switch (e) { case 11: n = r.LINEAR, t ? i = r.LINEAR_MIPMAP_NEAREST : i = r.LINEAR; break; case 3: n = r.LINEAR, t ? i = r.LINEAR_MIPMAP_LINEAR : i = r.LINEAR; break; case 8: n = r.NEAREST, t ? i = r.NEAREST_MIPMAP_LINEAR : i = r.NEAREST; break; case 4: n = r.NEAREST, t ? i = r.NEAREST_MIPMAP_NEAREST : i = r.NEAREST; break; case 5: n = r.NEAREST, t ? i = r.LINEAR_MIPMAP_NEAREST : i = r.LINEAR; break; case 6: n = r.NEAREST, t ? i = r.LINEAR_MIPMAP_LINEAR : i = r.LINEAR; break; case 7: n = r.NEAREST, i = r.LINEAR; break; case 1: n = r.NEAREST, i = r.NEAREST; break; case 9: n = r.LINEAR, t ? i = r.NEAREST_MIPMAP_NEAREST : i = r.NEAREST; break; case 10: n = r.LINEAR, t ? i = r.NEAREST_MIPMAP_LINEAR : i = r.NEAREST; break; case 2: n = r.LINEAR, i = r.LINEAR; break; case 12: n = r.LINEAR, i = r.NEAREST; break; } return { min: i, mag: n }; } /** @internal */ _createTexture() { const e = this._gl.createTexture(); if (!e) throw new Error("Unable to create texture"); return e; } /** @internal */ _createHardwareTexture() { return new jm(this._createTexture(), this._gl); } /** * Creates an internal texture without binding it to a framebuffer * @internal * @param size defines the size of the texture * @param options defines the options used to create the texture * @param delayGPUTextureCreation true to delay the texture creation the first time it is really needed. false to create it right away * @param source source type of the texture * @returns a new internal texture */ _createInternalTexture(e, t, r = !0, n = ri.Unknown) { var i; let s = !1, a = 0, f = 3, o = 5, d = !1, v = 1, u; t !== void 0 && typeof t == "object" ? (s = !!t.generateMipMaps, a = t.type === void 0 ? 0 : t.type, f = t.samplingMode === void 0 ? 3 : t.samplingMode, o = t.format === void 0 ? 5 : t.format, d = t.useSRGBBuffer === void 0 ? !1 : t.useSRGBBuffer, v = (i = t.samples) !== null && i !== void 0 ? i : 1, u = t.label) : s = !!t, d && (d = this._caps.supportSRGBBuffers && (this.webGLVersion > 1 || this.isWebGPU)), (a === 1 && !this._caps.textureFloatLinearFiltering || a === 2 && !this._caps.textureHalfFloatLinearFiltering) && (f = 1), a === 1 && !this._caps.textureFloat && (a = 0, Se.Warn("Float textures are not supported. Type forced to TEXTURETYPE_UNSIGNED_BYTE")); const l = this._gl, P = new As(this, n), p = e.width || e, c = e.height || e, H = e.layers || 0, T = this._getSamplingParameters(f, s), q = H !== 0 ? l.TEXTURE_2D_ARRAY : l.TEXTURE_2D, b = this._getRGBABufferInternalSizedFormat(a, o, d), j = this._getInternalFormat(o), w = this._getWebGLTextureType(a); return this._bindTextureDirectly(q, P), H !== 0 ? (P.is2DArray = !0, l.texImage3D(q, 0, b, p, c, H, 0, j, w, null)) : l.texImage2D(q, 0, b, p, c, 0, j, w, null), l.texParameteri(q, l.TEXTURE_MAG_FILTER, T.mag), l.texParameteri(q, l.TEXTURE_MIN_FILTER, T.min), l.texParameteri(q, l.TEXTURE_WRAP_S, l.CLAMP_TO_EDGE), l.texParameteri(q, l.TEXTURE_WRAP_T, l.CLAMP_TO_EDGE), s && this._gl.generateMipmap(q), this._bindTextureDirectly(q, null), P._useSRGBBuffer = d, P.baseWidth = p, P.baseHeight = c, P.width = p, P.height = c, P.depth = H, P.isReady = !0, P.samples = v, P.generateMipMaps = s, P.samplingMode = f, P.type = a, P.format = o, P.label = u, this._internalTexturesCache.push(P), P; } /** * @internal */ _getUseSRGBBuffer(e, t) { return e && this._caps.supportSRGBBuffers && (this.webGLVersion > 1 || this.isWebGPU || t); } _createTextureBase(e, t, r, n, i = 3, s = null, a = null, f, o, d = null, v = null, u = null, l = null, P, p, c) { e = e || ""; const H = e.substr(0, 5) === "data:", T = e.substr(0, 5) === "blob:", q = H && e.indexOf(";base64,") !== -1, b = v || new As(this, ri.Url); b !== v && (b.label = e.substring(0, 60)); const j = e; this._transformTextureUrl && !q && !v && !d && (e = this._transformTextureUrl(e)), j !== e && (b._originalUrl = j); const w = e.lastIndexOf("."); let m = l || (w > -1 ? e.substring(w).toLowerCase() : ""), I = null; m.indexOf("?") > -1 && (m = m.split("?")[0]); for (const y of hr._TextureLoaders) if (y.canLoad(m, P)) { I = y; break; } n && n.addPendingData(b), b.url = e, b.generateMipMaps = !t, b.samplingMode = i, b.invertY = r, b._useSRGBBuffer = this._getUseSRGBBuffer(!!c, t), this._doNotHandleContextLost || (b._buffer = d); let k = null; s && !v && (k = b.onLoadedObservable.add(s)), v || this._internalTexturesCache.push(b); const R = (y, O) => { n && n.removePendingData(b), e === j ? (k && b.onLoadedObservable.remove(k), gr.UseFallbackTexture && this._createTextureBase(gr.FallbackTexture, t, b.invertY, n, i, null, a, f, o, d, b), y = (y || "Unknown error") + (gr.UseFallbackTexture ? " - Fallback texture was used" : ""), b.onErrorObservable.notifyObservers({ message: y, exception: O }), a && a(y, O)) : (Se.Warn(`Failed to load ${e}, falling back to ${j}`), this._createTextureBase(j, t, b.invertY, n, i, s, a, f, o, d, b, u, l, P, p, c)); }; if (I) { const y = (O) => { I.loadData(O, b, (Y, ee, Z, te, fe, _) => { _ ? R("TextureLoader failed to load data") : f(b, m, n, { width: Y, height: ee }, b.invertY, !Z, te, () => (fe(), !1), i); }, p); }; d ? d instanceof ArrayBuffer ? y(new Uint8Array(d)) : ArrayBuffer.isView(d) ? y(d) : a && a("Unable to load: only ArrayBuffer or ArrayBufferView is supported", null) : this._loadFile(e, (O) => y(new Uint8Array(O)), void 0, n ? n.offlineProvider : void 0, !0, (O, Y) => { R("Unable to load " + (O && O.responseURL, Y)); }); } else { const y = (O) => { T && !this._doNotHandleContextLost && (b._buffer = O), f(b, m, n, O, b.invertY, t, !1, o, i); }; !H || q ? d && (typeof d.decoding == "string" || d.close) ? y(d) : hr._FileToolsLoadImage(e, y, R, n ? n.offlineProvider : null, P, b.invertY && this._features.needsInvertingBitmap ? { imageOrientation: "flipY" } : void 0) : typeof d == "string" || d instanceof ArrayBuffer || ArrayBuffer.isView(d) || d instanceof Blob ? hr._FileToolsLoadImage(d, y, R, n ? n.offlineProvider : null, P, b.invertY && this._features.needsInvertingBitmap ? { imageOrientation: "flipY" } : void 0) : d && y(d); } return b; } /** * Usually called from Texture.ts. * Passed information to create a WebGLTexture * @param url defines a value which contains one of the following: * * A conventional http URL, e.g. 'http://...' or 'file://...' * * A base64 string of in-line texture data, e.g. '...' * * An indicator that data being passed using the buffer parameter, e.g. 'data:mytexture.jpg' * @param noMipmap defines a boolean indicating that no mipmaps shall be generated. Ignored for compressed textures. They must be in the file * @param invertY when true, image is flipped when loaded. You probably want true. Certain compressed textures may invert this if their default is inverted (eg. ktx) * @param scene needed for loading to the correct scene * @param samplingMode mode with should be used sample / access the texture (Default: Texture.TRILINEAR_SAMPLINGMODE) * @param onLoad optional callback to be called upon successful completion * @param onError optional callback to be called upon failure * @param buffer a source of a file previously fetched as either a base64 string, an ArrayBuffer (compressed or image format), HTMLImageElement (image format), or a Blob * @param fallback an internal argument in case the function must be called again, due to etc1 not having alpha capabilities * @param format internal format. Default: RGB when extension is '.jpg' else RGBA. Ignored for compressed textures * @param forcedExtension defines the extension to use to pick the right loader * @param mimeType defines an optional mime type * @param loaderOptions options to be passed to the loader * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg) * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU). * @returns a InternalTexture for assignment back into BABYLON.Texture */ createTexture(e, t, r, n, i = 3, s = null, a = null, f = null, o = null, d = null, v = null, u, l, P, p) { return this._createTextureBase(e, t, r, n, i, s, a, this._prepareWebGLTexture.bind(this), (c, H, T, q, b, j) => { const w = this._gl, m = T.width === c && T.height === H, I = this._getTexImageParametersForCreateTexture(d, q, b._useSRGBBuffer); if (m) return w.texImage2D(w.TEXTURE_2D, 0, I.internalFormat, I.format, I.type, T), !1; const N = this._caps.maxTextureSize; if (T.width > N || T.height > N || !this._supportsHardwareTextureRescaling) return this._prepareWorkingCanvas(), !this._workingCanvas || !this._workingContext || (this._workingCanvas.width = c, this._workingCanvas.height = H, this._workingContext.drawImage(T, 0, 0, T.width, T.height, 0, 0, c, H), w.texImage2D(w.TEXTURE_2D, 0, I.internalFormat, I.format, I.type, this._workingCanvas), b.width = c, b.height = H), !1; { const k = new As(this, ri.Temp); this._bindTextureDirectly(w.TEXTURE_2D, k, !0), w.texImage2D(w.TEXTURE_2D, 0, I.internalFormat, I.format, I.type, T), this._rescaleTexture(k, b, n, I.format, () => { this._releaseTexture(k), this._bindTextureDirectly(w.TEXTURE_2D, b, !0), j(); }); } return !0; }, f, o, d, v, u, l, p); } /** * Calls to the GL texImage2D and texImage3D functions require three arguments describing the pixel format of the texture. * createTexture derives these from the babylonFormat and useSRGBBuffer arguments and also the file extension of the URL it's working with. * This function encapsulates that derivation for easy unit testing. * @param babylonFormat Babylon's format enum, as specified in ITextureCreationOptions. * @param fileExtension The file extension including the dot, e.g. .jpg. * @param useSRGBBuffer Use SRGB not linear. * @returns The options to pass to texImage2D or texImage3D calls. * @internal */ _getTexImageParametersForCreateTexture(e, t, r) { e == null && (e = t === ".jpg" && !r ? 4 : 5); let n, i; return this.webGLVersion === 1 ? (n = this._getInternalFormat(e, r), i = n) : (n = this._getInternalFormat(e, !1), i = this._getRGBABufferInternalSizedFormat(0, e, r)), { internalFormat: i, format: n, type: this._gl.UNSIGNED_BYTE }; } /** * Loads an image as an HTMLImageElement. * @param input url string, ArrayBuffer, or Blob to load * @param onLoad callback called when the image successfully loads * @param onError callback called when the image fails to load * @param offlineProvider offline provider for caching * @param mimeType optional mime type * @param imageBitmapOptions optional the options to use when creating an ImageBitmap * @returns the HTMLImageElement of the loaded image * @internal */ static _FileToolsLoadImage(e, t, r, n, i, s) { throw qn("FileTools"); } /** * @internal */ _rescaleTexture(e, t, r, n, i) { } /** * Creates a raw texture * @param data defines the data to store in the texture * @param width defines the width of the texture * @param height defines the height of the texture * @param format defines the format of the data * @param generateMipMaps defines if the engine should generate the mip levels * @param invertY defines if data must be stored with Y axis inverted * @param samplingMode defines the required sampling mode (Texture.NEAREST_SAMPLINGMODE by default) * @param compression defines the compression used (null by default) * @param type defines the type fo the data (Engine.TEXTURETYPE_UNSIGNED_INT by default) * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg) * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU). * @returns the raw texture inside an InternalTexture */ createRawTexture(e, t, r, n, i, s, a, f = null, o = 0, d = 0, v = !1) { throw qn("Engine.RawTexture"); } /** * Creates a new raw cube texture * @param data defines the array of data to use to create each face * @param size defines the size of the textures * @param format defines the format of the data * @param type defines the type of the data (like Engine.TEXTURETYPE_UNSIGNED_INT) * @param generateMipMaps defines if the engine should generate the mip levels * @param invertY defines if data must be stored with Y axis inverted * @param samplingMode defines the required sampling mode (like Texture.NEAREST_SAMPLINGMODE) * @param compression defines the compression used (null by default) * @returns the cube texture as an InternalTexture */ createRawCubeTexture(e, t, r, n, i, s, a, f = null) { throw qn("Engine.RawTexture"); } /** * Creates a new raw 3D texture * @param data defines the data used to create the texture * @param width defines the width of the texture * @param height defines the height of the texture * @param depth defines the depth of the texture * @param format defines the format of the texture * @param generateMipMaps defines if the engine must generate mip levels * @param invertY defines if data must be stored with Y axis inverted * @param samplingMode defines the required sampling mode (like Texture.NEAREST_SAMPLINGMODE) * @param compression defines the compressed used (can be null) * @param textureType defines the compressed used (can be null) * @returns a new raw 3D texture (stored in an InternalTexture) */ createRawTexture3D(e, t, r, n, i, s, a, f, o = null, d = 0) { throw qn("Engine.RawTexture"); } /** * Creates a new raw 2D array texture * @param data defines the data used to create the texture * @param width defines the width of the texture * @param height defines the height of the texture * @param depth defines the number of layers of the texture * @param format defines the format of the texture * @param generateMipMaps defines if the engine must generate mip levels * @param invertY defines if data must be stored with Y axis inverted * @param samplingMode defines the required sampling mode (like Texture.NEAREST_SAMPLINGMODE) * @param compression defines the compressed used (can be null) * @param textureType defines the compressed used (can be null) * @returns a new raw 2D array texture (stored in an InternalTexture) */ createRawTexture2DArray(e, t, r, n, i, s, a, f, o = null, d = 0) { throw qn("Engine.RawTexture"); } /** * @internal */ _unpackFlipY(e) { this._unpackFlipYCached !== e && (this._gl.pixelStorei(this._gl.UNPACK_FLIP_Y_WEBGL, e ? 1 : 0), this.enableUnpackFlipYCached && (this._unpackFlipYCached = e)); } /** @internal */ _getUnpackAlignement() { return this._gl.getParameter(this._gl.UNPACK_ALIGNMENT); } _getTextureTarget(e) { return e.isCube ? this._gl.TEXTURE_CUBE_MAP : e.is3D ? this._gl.TEXTURE_3D : e.is2DArray || e.isMultiview ? this._gl.TEXTURE_2D_ARRAY : this._gl.TEXTURE_2D; } /** * Update the sampling mode of a given texture * @param samplingMode defines the required sampling mode * @param texture defines the texture to update * @param generateMipMaps defines whether to generate mipmaps for the texture */ updateTextureSamplingMode(e, t, r = !1) { const n = this._getTextureTarget(t), i = this._getSamplingParameters(e, t.useMipMaps || r); this._setTextureParameterInteger(n, this._gl.TEXTURE_MAG_FILTER, i.mag, t), this._setTextureParameterInteger(n, this._gl.TEXTURE_MIN_FILTER, i.min), r && (t.generateMipMaps = !0, this._gl.generateMipmap(n)), this._bindTextureDirectly(n, null), t.samplingMode = e; } /** * Update the dimensions of a texture * @param texture texture to update * @param width new width of the texture * @param height new height of the texture * @param depth new depth of the texture */ updateTextureDimensions(e, t, r, n = 1) { } /** * Update the sampling mode of a given texture * @param texture defines the texture to update * @param wrapU defines the texture wrap mode of the u coordinates * @param wrapV defines the texture wrap mode of the v coordinates * @param wrapR defines the texture wrap mode of the r coordinates */ updateTextureWrappingMode(e, t, r = null, n = null) { const i = this._getTextureTarget(e); t !== null && (this._setTextureParameterInteger(i, this._gl.TEXTURE_WRAP_S, this._getTextureWrapMode(t), e), e._cachedWrapU = t), r !== null && (this._setTextureParameterInteger(i, this._gl.TEXTURE_WRAP_T, this._getTextureWrapMode(r), e), e._cachedWrapV = r), (e.is2DArray || e.is3D) && n !== null && (this._setTextureParameterInteger(i, this._gl.TEXTURE_WRAP_R, this._getTextureWrapMode(n), e), e._cachedWrapR = n), this._bindTextureDirectly(i, null); } /** * @internal */ _setupDepthStencilTexture(e, t, r, n, i, s = 1) { const a = t.width || t, f = t.height || t, o = t.layers || 0; e.baseWidth = a, e.baseHeight = f, e.width = a, e.height = f, e.is2DArray = o > 0, e.depth = o, e.isReady = !0, e.samples = s, e.generateMipMaps = !1, e.samplingMode = n ? 2 : 1, e.type = 0, e._comparisonFunction = i; const d = this._gl, v = this._getTextureTarget(e), u = this._getSamplingParameters(e.samplingMode, !1); d.texParameteri(v, d.TEXTURE_MAG_FILTER, u.mag), d.texParameteri(v, d.TEXTURE_MIN_FILTER, u.min), d.texParameteri(v, d.TEXTURE_WRAP_S, d.CLAMP_TO_EDGE), d.texParameteri(v, d.TEXTURE_WRAP_T, d.CLAMP_TO_EDGE), this.webGLVersion > 1 && (i === 0 ? (d.texParameteri(v, d.TEXTURE_COMPARE_FUNC, 515), d.texParameteri(v, d.TEXTURE_COMPARE_MODE, d.NONE)) : (d.texParameteri(v, d.TEXTURE_COMPARE_FUNC, i), d.texParameteri(v, d.TEXTURE_COMPARE_MODE, d.COMPARE_REF_TO_TEXTURE))); } /** * @internal */ _uploadCompressedDataToTextureDirectly(e, t, r, n, i, s = 0, a = 0) { const f = this._gl; let o = f.TEXTURE_2D; if (e.isCube && (o = f.TEXTURE_CUBE_MAP_POSITIVE_X + s), e._useSRGBBuffer) switch (t) { case 37492: case 36196: this._caps.etc2 ? t = f.COMPRESSED_SRGB8_ETC2 : e._useSRGBBuffer = !1; break; case 37496: this._caps.etc2 ? t = f.COMPRESSED_SRGB8_ALPHA8_ETC2_EAC : e._useSRGBBuffer = !1; break; case 36492: t = f.COMPRESSED_SRGB_ALPHA_BPTC_UNORM_EXT; break; case 37808: t = f.COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR; break; case 33776: this._caps.s3tc_srgb ? t = f.COMPRESSED_SRGB_S3TC_DXT1_EXT : e._useSRGBBuffer = !1; break; case 33777: this._caps.s3tc_srgb ? t = f.COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT : e._useSRGBBuffer = !1; break; case 33779: this._caps.s3tc_srgb ? t = f.COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT : e._useSRGBBuffer = !1; break; default: e._useSRGBBuffer = !1; break; } this._gl.compressedTexImage2D(o, a, t, r, n, 0, i); } /** * @internal */ _uploadDataToTextureDirectly(e, t, r = 0, n = 0, i, s = !1) { const a = this._gl, f = this._getWebGLTextureType(e.type), o = this._getInternalFormat(e.format), d = i === void 0 ? this._getRGBABufferInternalSizedFormat(e.type, e.format, e._useSRGBBuffer) : this._getInternalFormat(i, e._useSRGBBuffer); this._unpackFlipY(e.invertY); let v = a.TEXTURE_2D; e.isCube && (v = a.TEXTURE_CUBE_MAP_POSITIVE_X + r); const u = Math.round(Math.log(e.width) * Math.LOG2E), l = Math.round(Math.log(e.height) * Math.LOG2E), P = s ? e.width : Math.pow(2, Math.max(u - n, 0)), p = s ? e.height : Math.pow(2, Math.max(l - n, 0)); a.texImage2D(v, n, d, P, p, 0, o, f, t); } /** * Update a portion of an internal texture * @param texture defines the texture to update * @param imageData defines the data to store into the texture * @param xOffset defines the x coordinates of the update rectangle * @param yOffset defines the y coordinates of the update rectangle * @param width defines the width of the update rectangle * @param height defines the height of the update rectangle * @param faceIndex defines the face index if texture is a cube (0 by default) * @param lod defines the lod level to update (0 by default) * @param generateMipMaps defines whether to generate mipmaps or not */ updateTextureData(e, t, r, n, i, s, a = 0, f = 0, o = !1) { const d = this._gl, v = this._getWebGLTextureType(e.type), u = this._getInternalFormat(e.format); this._unpackFlipY(e.invertY); let l = d.TEXTURE_2D, P = d.TEXTURE_2D; e.isCube && (P = d.TEXTURE_CUBE_MAP_POSITIVE_X + a, l = d.TEXTURE_CUBE_MAP), this._bindTextureDirectly(l, e, !0), d.texSubImage2D(P, f, r, n, i, s, u, v, t), o && this._gl.generateMipmap(P), this._bindTextureDirectly(l, null); } /** * @internal */ _uploadArrayBufferViewToTexture(e, t, r = 0, n = 0) { const i = this._gl, s = e.isCube ? i.TEXTURE_CUBE_MAP : i.TEXTURE_2D; this._bindTextureDirectly(s, e, !0), this._uploadDataToTextureDirectly(e, t, r, n), this._bindTextureDirectly(s, null, !0); } _prepareWebGLTextureContinuation(e, t, r, n, i) { const s = this._gl; if (!s) return; const a = this._getSamplingParameters(i, !r); s.texParameteri(s.TEXTURE_2D, s.TEXTURE_MAG_FILTER, a.mag), s.texParameteri(s.TEXTURE_2D, s.TEXTURE_MIN_FILTER, a.min), !r && !n && s.generateMipmap(s.TEXTURE_2D), this._bindTextureDirectly(s.TEXTURE_2D, null), t && t.removePendingData(e), e.onLoadedObservable.notifyObservers(e), e.onLoadedObservable.clear(); } _prepareWebGLTexture(e, t, r, n, i, s, a, f, o = 3) { const d = this.getCaps().maxTextureSize, v = Math.min(d, this.needPOTTextures ? hr.GetExponentOfTwo(n.width, d) : n.width), u = Math.min(d, this.needPOTTextures ? hr.GetExponentOfTwo(n.height, d) : n.height), l = this._gl; if (l) { if (!e._hardwareTexture) { r && r.removePendingData(e); return; } this._bindTextureDirectly(l.TEXTURE_2D, e, !0), this._unpackFlipY(i === void 0 ? !0 : !!i), e.baseWidth = n.width, e.baseHeight = n.height, e.width = v, e.height = u, e.isReady = !0, e.type = e.type !== -1 ? e.type : 0, e.format = e.format !== -1 ? e.format : t === ".jpg" && !e._useSRGBBuffer ? 4 : 5, !f(v, u, n, t, e, () => { this._prepareWebGLTextureContinuation(e, r, s, a, o); }) && this._prepareWebGLTextureContinuation(e, r, s, a, o); } } /** * @internal */ _setupFramebufferDepthAttachments(e, t, r, n, i = 1) { const s = this._gl; if (e && t) return this._createRenderBuffer(r, n, i, s.DEPTH_STENCIL, s.DEPTH24_STENCIL8, s.DEPTH_STENCIL_ATTACHMENT); if (t) { let a = s.DEPTH_COMPONENT16; return this._webGLVersion > 1 && (a = s.DEPTH_COMPONENT32F), this._createRenderBuffer(r, n, i, a, a, s.DEPTH_ATTACHMENT); } return e ? this._createRenderBuffer(r, n, i, s.STENCIL_INDEX8, s.STENCIL_INDEX8, s.STENCIL_ATTACHMENT) : null; } /** * @internal */ _createRenderBuffer(e, t, r, n, i, s, a = !0) { const o = this._gl.createRenderbuffer(); return this._updateRenderBuffer(o, e, t, r, n, i, s, a); } _updateRenderBuffer(e, t, r, n, i, s, a, f = !0) { const o = this._gl; return o.bindRenderbuffer(o.RENDERBUFFER, e), n > 1 && o.renderbufferStorageMultisample ? o.renderbufferStorageMultisample(o.RENDERBUFFER, n, s, t, r) : o.renderbufferStorage(o.RENDERBUFFER, i, t, r), o.framebufferRenderbuffer(o.FRAMEBUFFER, a, o.RENDERBUFFER, e), f && o.bindRenderbuffer(o.RENDERBUFFER, null), e; } /** * @internal */ _releaseTexture(e) { var t; this._deleteTexture((t = e._hardwareTexture) === null || t === void 0 ? void 0 : t.underlyingResource), this.unbindAllTextures(); const r = this._internalTexturesCache.indexOf(e); r !== -1 && this._internalTexturesCache.splice(r, 1), e._lodTextureHigh && e._lodTextureHigh.dispose(), e._lodTextureMid && e._lodTextureMid.dispose(), e._lodTextureLow && e._lodTextureLow.dispose(), e._irradianceTexture && e._irradianceTexture.dispose(); } /** * @internal */ _releaseRenderTargetWrapper(e) { const t = this._renderTargetWrapperCache.indexOf(e); t !== -1 && this._renderTargetWrapperCache.splice(t, 1); } _deleteTexture(e) { e && this._gl.deleteTexture(e); } _setProgram(e) { this._currentProgram !== e && (this._gl.useProgram(e), this._currentProgram = e); } /** * Binds an effect to the webGL context * @param effect defines the effect to bind */ bindSamplers(e) { const t = e.getPipelineContext(); this._setProgram(t.program); const r = e.getSamplers(); for (let n = 0; n < r.length; n++) { const i = e.getUniform(r[n]); i && (this._boundUniforms[n] = i); } this._currentEffect = null; } _activateCurrentTexture() { this._currentTextureChannel !== this._activeChannel && (this._gl.activeTexture(this._gl.TEXTURE0 + this._activeChannel), this._currentTextureChannel = this._activeChannel); } /** * @internal */ _bindTextureDirectly(e, t, r = !1, n = !1) { var i, s; let a = !1; const f = t && t._associatedChannel > -1; if (r && f && (this._activeChannel = t._associatedChannel), this._boundTexturesCache[this._activeChannel] !== t || n) { if (this._activateCurrentTexture(), t && t.isMultiview) throw console.error(e, t), "_bindTextureDirectly called with a multiview texture!"; this._gl.bindTexture(e, (s = (i = t == null ? void 0 : t._hardwareTexture) === null || i === void 0 ? void 0 : i.underlyingResource) !== null && s !== void 0 ? s : null), this._boundTexturesCache[this._activeChannel] = t, t && (t._associatedChannel = this._activeChannel); } else r && (a = !0, this._activateCurrentTexture()); return f && !r && this._bindSamplerUniformToChannel(t._associatedChannel, this._activeChannel), a; } /** * @internal */ _bindTexture(e, t, r) { if (e === void 0) return; t && (t._associatedChannel = e), this._activeChannel = e; const n = t ? this._getTextureTarget(t) : this._gl.TEXTURE_2D; this._bindTextureDirectly(n, t); } /** * Unbind all textures from the webGL context */ unbindAllTextures() { for (let e = 0; e < this._maxSimultaneousTextures; e++) this._activeChannel = e, this._bindTextureDirectly(this._gl.TEXTURE_2D, null), this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, null), this.webGLVersion > 1 && (this._bindTextureDirectly(this._gl.TEXTURE_3D, null), this._bindTextureDirectly(this._gl.TEXTURE_2D_ARRAY, null)); } /** * Sets a texture to the according uniform. * @param channel The texture channel * @param uniform The uniform to set * @param texture The texture to apply * @param name The name of the uniform in the effect */ setTexture(e, t, r, n) { e !== void 0 && (t && (this._boundUniforms[e] = t), this._setTexture(e, r)); } _bindSamplerUniformToChannel(e, t) { const r = this._boundUniforms[e]; !r || r._currentState === t || (this._gl.uniform1i(r, t), r._currentState = t); } _getTextureWrapMode(e) { switch (e) { case 1: return this._gl.REPEAT; case 0: return this._gl.CLAMP_TO_EDGE; case 2: return this._gl.MIRRORED_REPEAT; } return this._gl.REPEAT; } _setTexture(e, t, r = !1, n = !1, i = "") { if (!t) return this._boundTexturesCache[e] != null && (this._activeChannel = e, this._bindTextureDirectly(this._gl.TEXTURE_2D, null), this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, null), this.webGLVersion > 1 && (this._bindTextureDirectly(this._gl.TEXTURE_3D, null), this._bindTextureDirectly(this._gl.TEXTURE_2D_ARRAY, null))), !1; if (t.video) { this._activeChannel = e; const o = t.getInternalTexture(); o && (o._associatedChannel = e), t.update(); } else if (t.delayLoadState === 4) return t.delayLoad(), !1; let s; n ? s = t.depthStencilTexture : t.isReady() ? s = t.getInternalTexture() : t.isCube ? s = this.emptyCubeTexture : t.is3D ? s = this.emptyTexture3D : t.is2DArray ? s = this.emptyTexture2DArray : s = this.emptyTexture, !r && s && (s._associatedChannel = e); let a = !0; this._boundTexturesCache[e] === s && (r || this._bindSamplerUniformToChannel(s._associatedChannel, e), a = !1), this._activeChannel = e; const f = this._getTextureTarget(s); if (a && this._bindTextureDirectly(f, s, r), s && !s.isMultiview) { if (s.isCube && s._cachedCoordinatesMode !== t.coordinatesMode) { s._cachedCoordinatesMode = t.coordinatesMode; const o = t.coordinatesMode !== 3 && t.coordinatesMode !== 5 ? 1 : 0; t.wrapU = o, t.wrapV = o; } s._cachedWrapU !== t.wrapU && (s._cachedWrapU = t.wrapU, this._setTextureParameterInteger(f, this._gl.TEXTURE_WRAP_S, this._getTextureWrapMode(t.wrapU), s)), s._cachedWrapV !== t.wrapV && (s._cachedWrapV = t.wrapV, this._setTextureParameterInteger(f, this._gl.TEXTURE_WRAP_T, this._getTextureWrapMode(t.wrapV), s)), s.is3D && s._cachedWrapR !== t.wrapR && (s._cachedWrapR = t.wrapR, this._setTextureParameterInteger(f, this._gl.TEXTURE_WRAP_R, this._getTextureWrapMode(t.wrapR), s)), this._setAnisotropicLevel(f, s, t.anisotropicFilteringLevel); } return !0; } /** * Sets an array of texture to the webGL context * @param channel defines the channel where the texture array must be set * @param uniform defines the associated uniform location * @param textures defines the array of textures to bind * @param name name of the channel */ setTextureArray(e, t, r, n) { if (!(e === void 0 || !t)) { (!this._textureUnits || this._textureUnits.length !== r.length) && (this._textureUnits = new Int32Array(r.length)); for (let i = 0; i < r.length; i++) { const s = r[i].getInternalTexture(); s ? (this._textureUnits[i] = e + i, s._associatedChannel = e + i) : this._textureUnits[i] = -1; } this._gl.uniform1iv(t, this._textureUnits); for (let i = 0; i < r.length; i++) this._setTexture(this._textureUnits[i], r[i], !0); } } /** * @internal */ _setAnisotropicLevel(e, t, r) { const n = this._caps.textureAnisotropicFilterExtension; t.samplingMode !== 11 && t.samplingMode !== 3 && t.samplingMode !== 2 && (r = 1), n && t._cachedAnisotropicFilteringLevel !== r && (this._setTextureParameterFloat(e, n.TEXTURE_MAX_ANISOTROPY_EXT, Math.min(r, this._caps.maxAnisotropy), t), t._cachedAnisotropicFilteringLevel = r); } _setTextureParameterFloat(e, t, r, n) { this._bindTextureDirectly(e, n, !0, !0), this._gl.texParameterf(e, t, r); } _setTextureParameterInteger(e, t, r, n) { n && this._bindTextureDirectly(e, n, !0, !0), this._gl.texParameteri(e, t, r); } /** * Unbind all vertex attributes from the webGL context */ unbindAllAttributes() { if (this._mustWipeVertexAttributes) { this._mustWipeVertexAttributes = !1; for (let e = 0; e < this._caps.maxVertexAttribs; e++) this.disableAttributeByIndex(e); return; } for (let e = 0, t = this._vertexAttribArraysEnabled.length; e < t; e++) e >= this._caps.maxVertexAttribs || !this._vertexAttribArraysEnabled[e] || this.disableAttributeByIndex(e); } /** * Force the engine to release all cached effects. This means that next effect compilation will have to be done completely even if a similar effect was already compiled */ releaseEffects() { for (const e in this._compiledEffects) { const t = this._compiledEffects[e].getPipelineContext(); this._deletePipelineContext(t); } this._compiledEffects = {}; } /** * Dispose and release all associated resources */ dispose() { var e, t; this._isDisposed = !0, this.stopRenderLoop(), this.onBeforeTextureInitObservable && this.onBeforeTextureInitObservable.clear(), this._emptyTexture && (this._releaseTexture(this._emptyTexture), this._emptyTexture = null), this._emptyCubeTexture && (this._releaseTexture(this._emptyCubeTexture), this._emptyCubeTexture = null), this._dummyFramebuffer && this._gl.deleteFramebuffer(this._dummyFramebuffer), this.releaseEffects(), (e = this.releaseComputeEffects) === null || e === void 0 || e.call(this), this.unbindAllAttributes(), this._boundUniforms = {}, u9() && this._renderingCanvas && (this._doNotHandleContextLost || (this._renderingCanvas.removeEventListener("webglcontextlost", this._onContextLost), this._renderingCanvas.removeEventListener("webglcontextrestored", this._onContextRestored)), window.removeEventListener("resize", this._checkForMobile)), this._workingCanvas = null, this._workingContext = null, this._currentBufferPointers.length = 0, this._renderingCanvas = null, this._currentProgram = null, this._boundRenderFunction = null, An.ResetCache(); for (const r of this._activeRequests) r.abort(); this.onDisposeObservable.notifyObservers(this), this.onDisposeObservable.clear(), this._creationOptions.loseContextOnDispose && ((t = this._gl.getExtension("WEBGL_lose_context")) === null || t === void 0 || t.loseContext()); } /** * Attach a new callback raised when context lost event is fired * @param callback defines the callback to call */ attachContextLostEvent(e) { this._renderingCanvas && this._renderingCanvas.addEventListener("webglcontextlost", e, !1); } /** * Attach a new callback raised when context restored event is fired * @param callback defines the callback to call */ attachContextRestoredEvent(e) { this._renderingCanvas && this._renderingCanvas.addEventListener("webglcontextrestored", e, !1); } /** * Get the current error code of the webGL context * @returns the error code * @see https://developer.mozilla.org/en-US/docs/Web/API/WebGLRenderingContext/getError */ getError() { return this._gl.getError(); } _canRenderToFloatFramebuffer() { return this._webGLVersion > 1 ? this._caps.colorBufferFloat : this._canRenderToFramebuffer(1); } _canRenderToHalfFloatFramebuffer() { return this._webGLVersion > 1 ? this._caps.colorBufferFloat : this._canRenderToFramebuffer(2); } // Thank you : http://stackoverflow.com/questions/28827511/webgl-ios-render-to-floating-point-texture _canRenderToFramebuffer(e) { const t = this._gl; for (; t.getError() !== t.NO_ERROR; ) ; let r = !0; const n = t.createTexture(); t.bindTexture(t.TEXTURE_2D, n), t.texImage2D(t.TEXTURE_2D, 0, this._getRGBABufferInternalSizedFormat(e), 1, 1, 0, t.RGBA, this._getWebGLTextureType(e), null), t.texParameteri(t.TEXTURE_2D, t.TEXTURE_MIN_FILTER, t.NEAREST), t.texParameteri(t.TEXTURE_2D, t.TEXTURE_MAG_FILTER, t.NEAREST); const i = t.createFramebuffer(); t.bindFramebuffer(t.FRAMEBUFFER, i), t.framebufferTexture2D(t.FRAMEBUFFER, t.COLOR_ATTACHMENT0, t.TEXTURE_2D, n, 0); const s = t.checkFramebufferStatus(t.FRAMEBUFFER); if (r = r && s === t.FRAMEBUFFER_COMPLETE, r = r && t.getError() === t.NO_ERROR, r && (t.clear(t.COLOR_BUFFER_BIT), r = r && t.getError() === t.NO_ERROR), r) { t.bindFramebuffer(t.FRAMEBUFFER, null); const a = t.RGBA, f = t.UNSIGNED_BYTE, o = new Uint8Array(4); t.readPixels(0, 0, 1, 1, a, f, o), r = r && t.getError() === t.NO_ERROR; } for (t.deleteTexture(n), t.deleteFramebuffer(i), t.bindFramebuffer(t.FRAMEBUFFER, null); !r && t.getError() !== t.NO_ERROR; ) ; return r; } /** * @internal */ _getWebGLTextureType(e) { if (this._webGLVersion === 1) { switch (e) { case 1: return this._gl.FLOAT; case 2: return this._gl.HALF_FLOAT_OES; case 0: return this._gl.UNSIGNED_BYTE; case 8: return this._gl.UNSIGNED_SHORT_4_4_4_4; case 9: return this._gl.UNSIGNED_SHORT_5_5_5_1; case 10: return this._gl.UNSIGNED_SHORT_5_6_5; } return this._gl.UNSIGNED_BYTE; } switch (e) { case 3: return this._gl.BYTE; case 0: return this._gl.UNSIGNED_BYTE; case 4: return this._gl.SHORT; case 5: return this._gl.UNSIGNED_SHORT; case 6: return this._gl.INT; case 7: return this._gl.UNSIGNED_INT; case 1: return this._gl.FLOAT; case 2: return this._gl.HALF_FLOAT; case 8: return this._gl.UNSIGNED_SHORT_4_4_4_4; case 9: return this._gl.UNSIGNED_SHORT_5_5_5_1; case 10: return this._gl.UNSIGNED_SHORT_5_6_5; case 11: return this._gl.UNSIGNED_INT_2_10_10_10_REV; case 12: return this._gl.UNSIGNED_INT_24_8; case 13: return this._gl.UNSIGNED_INT_10F_11F_11F_REV; case 14: return this._gl.UNSIGNED_INT_5_9_9_9_REV; case 15: return this._gl.FLOAT_32_UNSIGNED_INT_24_8_REV; } return this._gl.UNSIGNED_BYTE; } /** * @internal */ _getInternalFormat(e, t = !1) { let r = t ? this._glSRGBExtensionValues.SRGB8_ALPHA8 : this._gl.RGBA; switch (e) { case 0: r = this._gl.ALPHA; break; case 1: r = this._gl.LUMINANCE; break; case 2: r = this._gl.LUMINANCE_ALPHA; break; case 6: r = this._gl.RED; break; case 7: r = this._gl.RG; break; case 4: r = t ? this._glSRGBExtensionValues.SRGB : this._gl.RGB; break; case 5: r = t ? this._glSRGBExtensionValues.SRGB8_ALPHA8 : this._gl.RGBA; break; } if (this._webGLVersion > 1) switch (e) { case 8: r = this._gl.RED_INTEGER; break; case 9: r = this._gl.RG_INTEGER; break; case 10: r = this._gl.RGB_INTEGER; break; case 11: r = this._gl.RGBA_INTEGER; break; } return r; } /** * @internal */ _getRGBABufferInternalSizedFormat(e, t, r = !1) { if (this._webGLVersion === 1) { if (t !== void 0) switch (t) { case 0: return this._gl.ALPHA; case 1: return this._gl.LUMINANCE; case 2: return this._gl.LUMINANCE_ALPHA; case 4: return r ? this._glSRGBExtensionValues.SRGB : this._gl.RGB; } return this._gl.RGBA; } switch (e) { case 3: switch (t) { case 6: return this._gl.R8_SNORM; case 7: return this._gl.RG8_SNORM; case 4: return this._gl.RGB8_SNORM; case 8: return this._gl.R8I; case 9: return this._gl.RG8I; case 10: return this._gl.RGB8I; case 11: return this._gl.RGBA8I; default: return this._gl.RGBA8_SNORM; } case 0: switch (t) { case 6: return this._gl.R8; case 7: return this._gl.RG8; case 4: return r ? this._glSRGBExtensionValues.SRGB8 : this._gl.RGB8; case 5: return r ? this._glSRGBExtensionValues.SRGB8_ALPHA8 : this._gl.RGBA8; case 8: return this._gl.R8UI; case 9: return this._gl.RG8UI; case 10: return this._gl.RGB8UI; case 11: return this._gl.RGBA8UI; case 0: return this._gl.ALPHA; case 1: return this._gl.LUMINANCE; case 2: return this._gl.LUMINANCE_ALPHA; default: return this._gl.RGBA8; } case 4: switch (t) { case 8: return this._gl.R16I; case 9: return this._gl.RG16I; case 10: return this._gl.RGB16I; case 11: return this._gl.RGBA16I; default: return this._gl.RGBA16I; } case 5: switch (t) { case 8: return this._gl.R16UI; case 9: return this._gl.RG16UI; case 10: return this._gl.RGB16UI; case 11: return this._gl.RGBA16UI; default: return this._gl.RGBA16UI; } case 6: switch (t) { case 8: return this._gl.R32I; case 9: return this._gl.RG32I; case 10: return this._gl.RGB32I; case 11: return this._gl.RGBA32I; default: return this._gl.RGBA32I; } case 7: switch (t) { case 8: return this._gl.R32UI; case 9: return this._gl.RG32UI; case 10: return this._gl.RGB32UI; case 11: return this._gl.RGBA32UI; default: return this._gl.RGBA32UI; } case 1: switch (t) { case 6: return this._gl.R32F; case 7: return this._gl.RG32F; case 4: return this._gl.RGB32F; case 5: return this._gl.RGBA32F; default: return this._gl.RGBA32F; } case 2: switch (t) { case 6: return this._gl.R16F; case 7: return this._gl.RG16F; case 4: return this._gl.RGB16F; case 5: return this._gl.RGBA16F; default: return this._gl.RGBA16F; } case 10: return this._gl.RGB565; case 13: return this._gl.R11F_G11F_B10F; case 14: return this._gl.RGB9_E5; case 8: return this._gl.RGBA4; case 9: return this._gl.RGB5_A1; case 11: switch (t) { case 5: return this._gl.RGB10_A2; case 11: return this._gl.RGB10_A2UI; default: return this._gl.RGB10_A2; } } return r ? this._glSRGBExtensionValues.SRGB8_ALPHA8 : this._gl.RGBA8; } /** * @internal */ _loadFile(e, t, r, n, i, s) { const a = hr._FileToolsLoadFile(e, t, r, n, i, s); return this._activeRequests.push(a), a.onCompleteObservable.add((f) => { this._activeRequests.splice(this._activeRequests.indexOf(f), 1); }), a; } /** * Loads a file from a url * @param url url to load * @param onSuccess callback called when the file successfully loads * @param onProgress callback called while file is loading (if the server supports this mode) * @param offlineProvider defines the offline provider for caching * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer * @param onError callback called when the file fails to load * @returns a file request object * @internal */ static _FileToolsLoadFile(e, t, r, n, i, s) { throw qn("FileTools"); } /** * Reads pixels from the current frame buffer. Please note that this function can be slow * @param x defines the x coordinate of the rectangle where pixels must be read * @param y defines the y coordinate of the rectangle where pixels must be read * @param width defines the width of the rectangle where pixels must be read * @param height defines the height of the rectangle where pixels must be read * @param hasAlpha defines whether the output should have alpha or not (defaults to true) * @param flushRenderer true to flush the renderer from the pending commands before reading the pixels * @returns a ArrayBufferView promise (Uint8Array) containing RGBA colors */ readPixels(e, t, r, n, i = !0, s = !0) { const a = i ? 4 : 3, f = i ? this._gl.RGBA : this._gl.RGB, o = new Uint8Array(n * r * a); return s && this.flushFramebuffer(), this._gl.readPixels(e, t, r, n, f, this._gl.UNSIGNED_BYTE, o), Promise.resolve(o); } /** * Gets a Promise indicating if the engine can be instantiated (ie. if a webGL context can be found) */ static get IsSupportedAsync() { return Promise.resolve(this.isSupported()); } /** * Gets a boolean indicating if the engine can be instantiated (ie. if a webGL context can be found) */ static get IsSupported() { return this.isSupported(); } /** * Gets a boolean indicating if the engine can be instantiated (ie. if a webGL context can be found) * @returns true if the engine can be created * @ignorenaming */ // eslint-disable-next-line @typescript-eslint/naming-convention static isSupported() { if (this._HasMajorPerformanceCaveat !== null) return !this._HasMajorPerformanceCaveat; if (this._IsSupported === null) try { const e = this._CreateCanvas(1, 1), t = e.getContext("webgl") || e.getContext("experimental-webgl"); this._IsSupported = t != null && !!window.WebGLRenderingContext; } catch { this._IsSupported = !1; } return this._IsSupported; } /** * Gets a boolean indicating if the engine can be instantiated on a performant device (ie. if a webGL context can be found and it does not use a slow implementation) */ static get HasMajorPerformanceCaveat() { if (this._HasMajorPerformanceCaveat === null) try { const e = this._CreateCanvas(1, 1), t = e.getContext("webgl", { failIfMajorPerformanceCaveat: !0 }) || e.getContext("experimental-webgl", { failIfMajorPerformanceCaveat: !0 }); this._HasMajorPerformanceCaveat = !t; } catch { this._HasMajorPerformanceCaveat = !1; } return this._HasMajorPerformanceCaveat; } /** * Find the next highest power of two. * @param x Number to start search from. * @returns Next highest power of two. */ static CeilingPOT(e) { return e--, e |= e >> 1, e |= e >> 2, e |= e >> 4, e |= e >> 8, e |= e >> 16, e++, e; } /** * Find the next lowest power of two. * @param x Number to start search from. * @returns Next lowest power of two. */ static FloorPOT(e) { return e = e | e >> 1, e = e | e >> 2, e = e | e >> 4, e = e | e >> 8, e = e | e >> 16, e - (e >> 1); } /** * Find the nearest power of two. * @param x Number to start search from. * @returns Next nearest power of two. */ static NearestPOT(e) { const t = hr.CeilingPOT(e), r = hr.FloorPOT(e); return t - e > e - r ? r : t; } /** * Get the closest exponent of two * @param value defines the value to approximate * @param max defines the maximum value to return * @param mode defines how to define the closest value * @returns closest exponent of two of the given value */ static GetExponentOfTwo(e, t, r = 2) { let n; switch (r) { case 1: n = hr.FloorPOT(e); break; case 2: n = hr.NearestPOT(e); break; case 3: default: n = hr.CeilingPOT(e); break; } return Math.min(n, t); } /** * Queue a new function into the requested animation frame pool (ie. this function will be executed by the browser (or the javascript engine) for the next frame) * @param func - the function to be called * @param requester - the object that will request the next frame. Falls back to window. * @returns frame number */ static QueueNewFrame(e, t) { if (u9()) { const { requestAnimationFrame: r } = t || window; if (typeof r == "function") return r(e); } else if (typeof requestAnimationFrame == "function") return requestAnimationFrame(e); return setTimeout(e, 16); } /** * Gets host document * @returns the host document object */ getHostDocument() { return this._renderingCanvas && this._renderingCanvas.ownerDocument ? this._renderingCanvas.ownerDocument : $w() ? document : null; } } hr._TempClearColorUint32 = new Uint32Array(4); hr._TempClearColorInt32 = new Int32Array(4); hr.ExceptionList = [ { key: "Chrome/63.0", capture: "63\\.0\\.3239\\.(\\d+)", captureConstraint: 108, targets: ["uniformBuffer"] }, { key: "Firefox/58", capture: null, captureConstraint: null, targets: ["uniformBuffer"] }, { key: "Firefox/59", capture: null, captureConstraint: null, targets: ["uniformBuffer"] }, { key: "Chrome/72.+?Mobile", capture: null, captureConstraint: null, targets: ["vao"] }, { key: "Chrome/73.+?Mobile", capture: null, captureConstraint: null, targets: ["vao"] }, { key: "Chrome/74.+?Mobile", capture: null, captureConstraint: null, targets: ["vao"] }, { key: "Mac OS.+Chrome/71", capture: null, captureConstraint: null, targets: ["vao"] }, { key: "Mac OS.+Chrome/72", capture: null, captureConstraint: null, targets: ["vao"] }, { key: "Mac OS.+Chrome", capture: null, captureConstraint: null, targets: ["uniformBuffer"] }, // desktop osx safari 15.4 { key: ".*AppleWebKit.*(15.4).*Safari", capture: null, captureConstraint: null, targets: ["antialias", "maxMSAASamples"] }, // mobile browsers using safari 15.4 on ios { key: ".*(15.4).*AppleWebKit.*Safari", capture: null, captureConstraint: null, targets: ["antialias", "maxMSAASamples"] } ]; hr._TextureLoaders = []; hr.CollisionsEpsilon = 1e-3; hr._IsSupported = null; hr._HasMajorPerformanceCaveat = null; class wS { /** * Polyfill for setImmediate * @param action defines the action to execute after the current execution block */ static SetImmediate(e) { u9() && window.setImmediate ? window.setImmediate(e) : setTimeout(e, 1); } } const F$ = new RegExp(/^data:([^,]+\/[^,]+)?;base64,/i); class mS extends O0 { /** * Creates a new LoadFileError * @param message defines the message of the error * @param object defines the optional web request */ constructor(e, t) { super(e, Z2.LoadFileError), this.name = "LoadFileError", dm._setPrototypeOf(this, mS.prototype), t instanceof ho ? this.request = t : this.file = t; } } class OI extends O0 { /** * Creates a new LoadFileError * @param message defines the message of the error * @param request defines the optional web request */ constructor(e, t) { super(e, Z2.RequestFileError), this.request = t, this.name = "RequestFileError", dm._setPrototypeOf(this, OI.prototype); } } class xO extends O0 { /** * Creates a new ReadFileError * @param message defines the message of the error * @param file defines the optional file */ constructor(e, t) { super(e, Z2.ReadFileError), this.file = t, this.name = "ReadFileError", dm._setPrototypeOf(this, xO.prototype); } } const A9 = { /** * Gets or sets the retry strategy to apply when an error happens while loading an asset. * When defining this function, return the wait time before trying again or return -1 to * stop retrying and error out. */ DefaultRetryStrategy: V$.ExponentialBackoff(), /** * Gets or sets the base URL to use to load assets */ BaseUrl: "", /** * Default behaviour for cors in the application. * It can be a string if the expected behavior is identical in the entire app. * Or a callback to be able to set it per url or on a group of them (in case of Video source for instance) */ CorsBehavior: "anonymous", /** * Gets or sets a function used to pre-process url before using them to load assets * @param url */ PreprocessUrl: (A) => A, /** * Gets or sets the base URL to use to load scripts * Used for both JS and WASM */ ScriptBaseUrl: "", /** * Gets or sets a function used to pre-process script url before using them to load. * Used for both JS and WASM * @param url defines the url to process */ ScriptPreprocessUrl: (A) => A }, N$ = (A) => (A = A.replace(/#/gm, "%23"), A), DO = (A, e) => { if (!(A && A.indexOf("data:") === 0) && A9.CorsBehavior) if (typeof A9.CorsBehavior == "string" || A9.CorsBehavior instanceof String) e.crossOrigin = A9.CorsBehavior; else { const t = A9.CorsBehavior(A); t && (e.crossOrigin = t); } }, nU = (A, e, t, r, n = "", i) => { var s; let a, f = !1; A instanceof ArrayBuffer || ArrayBuffer.isView(A) ? typeof Blob < "u" && typeof URL < "u" ? (a = URL.createObjectURL(new Blob([A], { type: n })), f = !0) : a = `data:${n};base64,` + XR(A) : A instanceof Blob ? (a = URL.createObjectURL(A), f = !0) : (a = N$(A), a = A9.PreprocessUrl(A)); const o = gr.LastCreatedEngine, d = (w) => { if (t) { const m = a || A.toString(); t(`Error while trying to load image: ${m.indexOf("http") === 0 || m.length <= 128 ? m : m.slice(0, 128) + "..."}`, w); } }; if (typeof Image > "u" || (s = o == null ? void 0 : o._features.forceBitmapOverHTMLImageElement) !== null && s !== void 0 && s) return gq(a, (w) => { o.createImageBitmap(new Blob([w], { type: n }), Object.assign({ premultiplyAlpha: "none" }, i)).then((m) => { e(m), f && URL.revokeObjectURL(a); }).catch((m) => { t && t("Error while trying to load image: " + A, m); }); }, void 0, r || void 0, !0, (w, m) => { d(m); }), null; const v = new Image(); DO(a, v); const u = [], l = () => { u.forEach((w) => { w.target.addEventListener(w.name, w.handler); }); }, P = () => { u.forEach((w) => { w.target.removeEventListener(w.name, w.handler); }), u.length = 0; }, p = () => { P(), e(v), f && v.src && URL.revokeObjectURL(v.src); }, c = (w) => { P(), d(w), f && v.src && URL.revokeObjectURL(v.src); }, H = (w) => { if (w.blockedURI !== v.src) return; P(); const m = new Error(`CSP violation of policy ${w.effectiveDirective} ${w.blockedURI}. Current policy is ${w.originalPolicy}`); gr.UseFallbackTexture = !1, d(m), f && v.src && URL.revokeObjectURL(v.src), v.src = ""; }; u.push({ target: v, name: "load", handler: p }), u.push({ target: v, name: "error", handler: c }), u.push({ target: document, name: "securitypolicyviolation", handler: H }), l(); const T = a.substring(0, 5) === "blob:", q = a.substring(0, 5) === "data:", b = () => { T || q || !ho.IsCustomRequestAvailable ? v.src = a : gq(a, (w, m, I) => { const N = !n && I ? I : n, k = new Blob([w], { type: N }), R = URL.createObjectURL(k); f = !0, v.src = R; }, void 0, r || void 0, !0, (w, m) => { d(m); }); }, j = () => { r && r.loadImage(a, v); }; if (!T && !q && r && r.enableTexturesOffline) r.open(j, b); else { if (a.indexOf("file:") !== -1) { const w = decodeURIComponent(a.substring(5).toLowerCase()); if (em.FilesToLoad[w] && typeof URL < "u") { try { let m; try { m = URL.createObjectURL(em.FilesToLoad[w]); } catch { m = URL.createObjectURL(em.FilesToLoad[w]); } v.src = m, f = !0; } catch { v.src = ""; } return v; } } b(); } return v; }, BS = (A, e, t, r, n) => { const i = new FileReader(), s = { onCompleteObservable: new Oe(), abort: () => i.abort() }; return i.onloadend = () => s.onCompleteObservable.notifyObservers(s), n && (i.onerror = () => { n(new xO(`Unable to read ${A.name}`, A)); }), i.onload = (a) => { e(a.target.result); }, t && (i.onprogress = t), r ? i.readAsArrayBuffer(A) : i.readAsText(A), s; }, gq = (A, e, t, r, n, i, s) => { if (A.name) return BS(A, e, t, n, i ? (d) => { i(void 0, d); } : void 0); const a = A; if (a.indexOf("file:") !== -1) { let d = decodeURIComponent(a.substring(5).toLowerCase()); d.indexOf("./") === 0 && (d = d.substring(2)); const v = em.FilesToLoad[d]; if (v) return BS(v, e, t, n, i ? (u) => i(void 0, new mS(u.message, u.file)) : void 0); } const { match: f, type: o } = Q$(a); if (f) { const d = { onCompleteObservable: new Oe(), abort: () => () => { } }; try { const v = n ? iU(a) : zN(a); e(v, void 0, o); } catch (v) { i ? i(void 0, v) : Se.Error(v.message || "Failed to parse the Data URL"); } return wS.SetImmediate(() => { d.onCompleteObservable.notifyObservers(d); }), d; } return jO(a, (d, v) => { e(d, v == null ? void 0 : v.responseURL, v == null ? void 0 : v.getResponseHeader("content-type")); }, t, r, n, i ? (d) => { i(d.request, new mS(d.message, d.request)); } : void 0, s); }, jO = (A, e, t, r, n, i, s) => { A = N$(A), A = A9.PreprocessUrl(A); const a = A9.BaseUrl + A; let f = !1; const o = { onCompleteObservable: new Oe(), abort: () => f = !0 }, d = () => { let v = new ho(), u = null, l; const P = () => { v && (t && v.removeEventListener("progress", t), l && v.removeEventListener("readystatechange", l), v.removeEventListener("loadend", p)); }; let p = () => { P(), o.onCompleteObservable.notifyObservers(o), o.onCompleteObservable.clear(), t = void 0, l = null, p = null, i = void 0, s = void 0, e = void 0; }; o.abort = () => { f = !0, p && p(), v && v.readyState !== (XMLHttpRequest.DONE || 4) && v.abort(), u !== null && (clearTimeout(u), u = null), v = null; }; const c = (T) => { const q = T.message || "Unknown error"; i && v ? i(new OI(q, v)) : Se.Error(q); }, H = (T) => { if (v) { if (v.open("GET", a), s) try { s(v); } catch (q) { c(q); return; } n && (v.responseType = "arraybuffer"), t && v.addEventListener("progress", t), p && v.addEventListener("loadend", p), l = () => { if (!(f || !v) && v.readyState === (XMLHttpRequest.DONE || 4)) { if (l && v.removeEventListener("readystatechange", l), v.status >= 200 && v.status < 300 || v.status === 0 && (!u9() || JN())) { try { e && e(n ? v.response : v.responseText, v); } catch (j) { c(j); } return; } const q = A9.DefaultRetryStrategy; if (q) { const j = q(a, v, T); if (j !== -1) { P(), v = new ho(), u = setTimeout(() => H(T + 1), j); return; } } const b = new OI("Error status: " + v.status + " " + v.statusText + " - Unable to load " + a, v); i && i(b); } }, v.addEventListener("readystatechange", l), v.send(); } }; H(0); }; if (r && r.enableSceneOffline) { const v = (l) => { l && l.status > 400 ? i && i(l) : d(); }, u = () => { r && r.loadFile(A9.BaseUrl + A, (l) => { !f && e && e(l), o.onCompleteObservable.notifyObservers(o); }, t ? (l) => { !f && t && t(l); } : void 0, v, n); }; r.open(u, v); } else d(); return o; }, JN = () => typeof location < "u" && location.protocol === "file:", qR = (A) => F$.test(A), Q$ = (A) => { const e = F$.exec(A); return e === null || e.length === 0 ? { match: !1, type: "" } : { match: !0, type: e[0].replace("data:", "").replace("base64,", "") }; }; function iU(A) { return TR(A.split(",")[1]); } const zN = (A) => bO(A.split(",")[1]), J9e = () => { hr._FileToolsLoadImage = nU, hr._FileToolsLoadFile = gq, hp._FileToolsLoadFile = gq; }; J9e(); let kW; const Y$ = (A, e, t, r, n, i, s, a, f, o) => { kW = { DecodeBase64UrlToBinary: A, DecodeBase64UrlToString: e, DefaultRetryStrategy: t.DefaultRetryStrategy, BaseUrl: t.BaseUrl, CorsBehavior: t.CorsBehavior, PreprocessUrl: t.PreprocessUrl, IsBase64DataUrl: r, IsFileURL: n, LoadFile: i, LoadImage: s, ReadFile: a, RequestFile: f, SetCorsBehavior: o }, Object.defineProperty(kW, "DefaultRetryStrategy", { get: function() { return t.DefaultRetryStrategy; }, set: function(d) { t.DefaultRetryStrategy = d; } }), Object.defineProperty(kW, "BaseUrl", { get: function() { return t.BaseUrl; }, set: function(d) { t.BaseUrl = d; } }), Object.defineProperty(kW, "PreprocessUrl", { get: function() { return t.PreprocessUrl; }, set: function(d) { t.PreprocessUrl = d; } }), Object.defineProperty(kW, "CorsBehavior", { get: function() { return t.CorsBehavior; }, set: function(d) { t.CorsBehavior = d; } }); }; Y$(iU, zN, A9, qR, JN, gq, nU, BS, jO, DO); class jI { /** * Tries to instantiate a new object from a given class name * @param className defines the class name to instantiate * @returns the new object or null if the system was not able to do the instantiation */ static Instantiate(e) { if (this.RegisteredExternalClasses && this.RegisteredExternalClasses[e]) return this.RegisteredExternalClasses[e]; const t = Jo(e); if (t) return t; Se.Warn(e + " not found, you may have missed an import."); const r = e.split("."); let n = window || this; for (let i = 0, s = r.length; i < s; i++) n = n[r[i]]; return typeof n != "function" ? null : n; } } jI.RegisteredExternalClasses = {}; function v4() { return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g, (A) => { const e = Math.random() * 16 | 0; return (A === "x" ? e : e & 3 | 8).toString(16); }); } const z9e = { /** * Implementation from http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript/2117523#answer-2117523 * Be aware Math.random() could cause collisions, but: * "All but 6 of the 128 bits of the ID are randomly generated, which means that for any two ids, there's a 1 in 2^^122 (or 5.3x10^^36) chance they'll collide" * @returns a pseudo random id */ // eslint-disable-next-line @typescript-eslint/naming-convention RandomId: v4 }; class ye { /** * Gets or sets the base URL to use to load assets */ static get BaseUrl() { return A9.BaseUrl; } static set BaseUrl(e) { A9.BaseUrl = e; } /** * This function checks whether a URL is absolute or not. * It will also detect data and blob URLs * @param url the url to check * @returns is the url absolute or relative */ static IsAbsoluteUrl(e) { return e.indexOf("//") === 0 ? !0 : e.indexOf("://") === -1 || e.indexOf(".") === -1 || e.indexOf("/") === -1 || e.indexOf(":") > e.indexOf("/") ? !1 : e.indexOf("://") < e.indexOf(".") || e.indexOf("data:") === 0 || e.indexOf("blob:") === 0; } /** * Sets the base URL to use to load scripts */ static set ScriptBaseUrl(e) { A9.ScriptBaseUrl = e; } static get ScriptBaseUrl() { return A9.ScriptBaseUrl; } /** * Sets a preprocessing function to run on a source URL before importing it * Note that this function will execute AFTER the base URL is appended to the URL */ static set ScriptPreprocessUrl(e) { A9.ScriptPreprocessUrl = e; } static get ScriptPreprocessUrl() { return A9.ScriptPreprocessUrl; } /** * Gets or sets the retry strategy to apply when an error happens while loading an asset */ static get DefaultRetryStrategy() { return A9.DefaultRetryStrategy; } static set DefaultRetryStrategy(e) { A9.DefaultRetryStrategy = e; } /** * Default behavior for cors in the application. * It can be a string if the expected behavior is identical in the entire app. * Or a callback to be able to set it per url or on a group of them (in case of Video source for instance) */ static get CorsBehavior() { return A9.CorsBehavior; } static set CorsBehavior(e) { A9.CorsBehavior = e; } /** * Gets or sets a global variable indicating if fallback texture must be used when a texture cannot be loaded * @ignorenaming */ static get UseFallbackTexture() { return gr.UseFallbackTexture; } static set UseFallbackTexture(e) { gr.UseFallbackTexture = e; } /** * Use this object to register external classes like custom textures or material * to allow the loaders to instantiate them */ static get RegisteredExternalClasses() { return jI.RegisteredExternalClasses; } static set RegisteredExternalClasses(e) { jI.RegisteredExternalClasses = e; } /** * Texture content used if a texture cannot loaded * @ignorenaming */ // eslint-disable-next-line @typescript-eslint/naming-convention static get fallbackTexture() { return gr.FallbackTexture; } // eslint-disable-next-line @typescript-eslint/naming-convention static set fallbackTexture(e) { gr.FallbackTexture = e; } /** * Read the content of a byte array at a specified coordinates (taking in account wrapping) * @param u defines the coordinate on X axis * @param v defines the coordinate on Y axis * @param width defines the width of the source data * @param height defines the height of the source data * @param pixels defines the source byte array * @param color defines the output color */ static FetchToRef(e, t, r, n, i, s) { const a = Math.abs(e) * r % r | 0, f = Math.abs(t) * n % n | 0, o = (a + f * r) * 4; s.r = i[o] / 255, s.g = i[o + 1] / 255, s.b = i[o + 2] / 255, s.a = i[o + 3] / 255; } /** * Interpolates between a and b via alpha * @param a The lower value (returned when alpha = 0) * @param b The upper value (returned when alpha = 1) * @param alpha The interpolation-factor * @returns The mixed value */ static Mix(e, t, r) { return e * (1 - r) + t * r; } /** * Tries to instantiate a new object from a given class name * @param className defines the class name to instantiate * @returns the new object or null if the system was not able to do the instantiation */ static Instantiate(e) { return jI.Instantiate(e); } /** * Polyfill for setImmediate * @param action defines the action to execute after the current execution block */ static SetImmediate(e) { wS.SetImmediate(e); } /** * Function indicating if a number is an exponent of 2 * @param value defines the value to test * @returns true if the value is an exponent of 2 */ static IsExponentOfTwo(e) { let t = 1; do t *= 2; while (t < e); return t === e; } /** * Returns the nearest 32-bit single precision float representation of a Number * @param value A Number. If the parameter is of a different type, it will get converted * to a number or to NaN if it cannot be converted * @returns number */ static FloatRound(e) { return Math.fround(e); } /** * Extracts the filename from a path * @param path defines the path to use * @returns the filename */ static GetFilename(e) { const t = e.lastIndexOf("/"); return t < 0 ? e : e.substring(t + 1); } /** * Extracts the "folder" part of a path (everything before the filename). * @param uri The URI to extract the info from * @param returnUnchangedIfNoSlash Do not touch the URI if no slashes are present * @returns The "folder" part of the path */ static GetFolderPath(e, t = !1) { const r = e.lastIndexOf("/"); return r < 0 ? t ? e : "" : e.substring(0, r + 1); } /** * Convert an angle in radians to degrees * @param angle defines the angle to convert * @returns the angle in degrees */ static ToDegrees(e) { return e * 180 / Math.PI; } /** * Convert an angle in degrees to radians * @param angle defines the angle to convert * @returns the angle in radians */ static ToRadians(e) { return e * Math.PI / 180; } /** * Smooth angle changes (kind of low-pass filter), in particular for device orientation "shaking" * Use trigonometric functions to avoid discontinuity (0/360, -180/180) * @param previousAngle defines last angle value, in degrees * @param newAngle defines new angle value, in degrees * @param smoothFactor defines smoothing sensitivity; min 0: no smoothing, max 1: new data ignored * @returns the angle in degrees */ static SmoothAngleChange(e, t, r = 0.9) { const n = this.ToRadians(e), i = this.ToRadians(t); return this.ToDegrees(Math.atan2((1 - r) * Math.sin(i) + r * Math.sin(n), (1 - r) * Math.cos(i) + r * Math.cos(n))); } /** * Returns an array if obj is not an array * @param obj defines the object to evaluate as an array * @param allowsNullUndefined defines a boolean indicating if obj is allowed to be null or undefined * @returns either obj directly if obj is an array or a new array containing obj */ static MakeArray(e, t) { return t !== !0 && (e === void 0 || e == null) ? null : Array.isArray(e) ? e : [e]; } /** * Gets the pointer prefix to use * @param engine defines the engine we are finding the prefix for * @returns "pointer" if touch is enabled. Else returns "mouse" */ static GetPointerPrefix(e) { let t = "pointer"; return u9() && !window.PointerEvent && (t = "mouse"), e._badDesktopOS && !e._badOS && // And not ipad pros who claim to be macs... !(document && "ontouchend" in document) && (t = "mouse"), t; } /** * Sets the cors behavior on a dom element. This will add the required Tools.CorsBehavior to the element. * @param url define the url we are trying * @param element define the dom element where to configure the cors policy * @param element.crossOrigin */ static SetCorsBehavior(e, t) { DO(e, t); } /** * Sets the referrerPolicy behavior on a dom element. * @param referrerPolicy define the referrer policy to use * @param element define the dom element where to configure the referrer policy * @param element.referrerPolicy */ static SetReferrerPolicyBehavior(e, t) { t.referrerPolicy = e; } // External files /** * Removes unwanted characters from an url * @param url defines the url to clean * @returns the cleaned url */ static CleanUrl(e) { return e = e.replace(/#/gm, "%23"), e; } /** * Gets or sets a function used to pre-process url before using them to load assets */ static get PreprocessUrl() { return A9.PreprocessUrl; } static set PreprocessUrl(e) { A9.PreprocessUrl = e; } /** * Loads an image as an HTMLImageElement. * @param input url string, ArrayBuffer, or Blob to load * @param onLoad callback called when the image successfully loads * @param onError callback called when the image fails to load * @param offlineProvider offline provider for caching * @param mimeType optional mime type * @param imageBitmapOptions optional the options to use when creating an ImageBitmap * @returns the HTMLImageElement of the loaded image */ static LoadImage(e, t, r, n, i, s) { return nU(e, t, r, n, i, s); } /** * Loads a file from a url * @param url url string, ArrayBuffer, or Blob to load * @param onSuccess callback called when the file successfully loads * @param onProgress callback called while file is loading (if the server supports this mode) * @param offlineProvider defines the offline provider for caching * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer * @param onError callback called when the file fails to load * @returns a file request object */ static LoadFile(e, t, r, n, i, s) { return gq(e, t, r, n, i, s); } /** * Loads a file from a url * @param url the file url to load * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer * @returns a promise containing an ArrayBuffer corresponding to the loaded file */ static LoadFileAsync(e, t = !0) { return new Promise((r, n) => { gq(e, (i) => { r(i); }, void 0, void 0, t, (i, s) => { n(s); }); }); } /** * Get a script URL including preprocessing * @param scriptUrl the script Url to process * @returns a modified URL to use */ static GetBabylonScriptURL(e, t) { if (!e) return ""; if (ye.ScriptBaseUrl && e.startsWith(ye._DefaultCdnUrl)) { const r = ye.ScriptBaseUrl[ye.ScriptBaseUrl.length - 1] === "/" ? ye.ScriptBaseUrl.substring(0, ye.ScriptBaseUrl.length - 1) : ye.ScriptBaseUrl; e = e.replace(ye._DefaultCdnUrl, r); } return e = ye.ScriptPreprocessUrl(e), t && (e = ye.GetAbsoluteUrl(e)), e; } /** * This function is used internally by babylon components to load a script (identified by an url). When the url returns, the * content of this file is added into a new script element, attached to the DOM (body element) * @param scriptUrl defines the url of the script to load * @param onSuccess defines the callback called when the script is loaded * @param onError defines the callback to call if an error occurs * @param scriptId defines the id of the script element */ static LoadBabylonScript(e, t, r, n) { e = ye.GetBabylonScriptURL(e), ye.LoadScript(e, t, r); } /** * Load an asynchronous script (identified by an url). When the url returns, the * content of this file is added into a new script element, attached to the DOM (body element) * @param scriptUrl defines the url of the script to laod * @returns a promise request object */ static LoadBabylonScriptAsync(e) { return e = ye.GetBabylonScriptURL(e), ye.LoadScriptAsync(e); } /** * This function is used internally by babylon components to load a script (identified by an url). When the url returns, the * content of this file is added into a new script element, attached to the DOM (body element) * @param scriptUrl defines the url of the script to load * @param onSuccess defines the callback called when the script is loaded * @param onError defines the callback to call if an error occurs * @param scriptId defines the id of the script element */ static LoadScript(e, t, r, n) { if (typeof importScripts == "function") { try { importScripts(e), t(); } catch (a) { r == null || r(`Unable to load script '${e}' in worker`, a); } return; } else if (!u9()) { r == null || r(`Cannot load script '${e}' outside of a window or a worker`); return; } const i = document.getElementsByTagName("head")[0], s = document.createElement("script"); s.setAttribute("type", "text/javascript"), s.setAttribute("src", e), n && (s.id = n), s.onload = () => { t && t(); }, s.onerror = (a) => { r && r(`Unable to load script '${e}'`, a); }, i.appendChild(s); } /** * Load an asynchronous script (identified by an url). When the url returns, the * content of this file is added into a new script element, attached to the DOM (body element) * @param scriptUrl defines the url of the script to laod * @returns a promise request object */ static LoadScriptAsync(e) { return new Promise((t, r) => { this.LoadScript(e, () => { t(); }, (n, i) => { r(i || new Error(n)); }); }); } /** * Loads a file from a blob * @param fileToLoad defines the blob to use * @param callback defines the callback to call when data is loaded * @param progressCallback defines the callback to call during loading process * @returns a file request object */ static ReadFileAsDataURL(e, t, r) { const n = new FileReader(), i = { onCompleteObservable: new Oe(), abort: () => n.abort() }; return n.onloadend = () => { i.onCompleteObservable.notifyObservers(i); }, n.onload = (s) => { t(s.target.result); }, n.onprogress = r, n.readAsDataURL(e), i; } /** * Reads a file from a File object * @param file defines the file to load * @param onSuccess defines the callback to call when data is loaded * @param onProgress defines the callback to call during loading process * @param useArrayBuffer defines a boolean indicating that data must be returned as an ArrayBuffer * @param onError defines the callback to call when an error occurs * @returns a file request object */ static ReadFile(e, t, r, n, i) { return BS(e, t, r, n, i); } /** * Creates a data url from a given string content * @param content defines the content to convert * @returns the new data url link */ static FileAsURL(e) { const t = new Blob([e]); return window.URL.createObjectURL(t); } /** * Format the given number to a specific decimal format * @param value defines the number to format * @param decimals defines the number of decimals to use * @returns the formatted string */ static Format(e, t = 2) { return e.toFixed(t); } /** * Tries to copy an object by duplicating every property * @param source defines the source object * @param destination defines the target object * @param doNotCopyList defines a list of properties to avoid * @param mustCopyList defines a list of properties to copy (even if they start with _) */ static DeepCopy(e, t, r, n) { sA.DeepCopy(e, t, r, n); } /** * Gets a boolean indicating if the given object has no own property * @param obj defines the object to test * @returns true if object has no own property */ static IsEmpty(e) { for (const t in e) if (Object.prototype.hasOwnProperty.call(e, t)) return !1; return !0; } /** * Function used to register events at window level * @param windowElement defines the Window object to use * @param events defines the events to register */ static RegisterTopRootEvents(e, t) { for (let r = 0; r < t.length; r++) { const n = t[r]; e.addEventListener(n.name, n.handler, !1); try { window.parent && window.parent.addEventListener(n.name, n.handler, !1); } catch { } } } /** * Function used to unregister events from window level * @param windowElement defines the Window object to use * @param events defines the events to unregister */ static UnregisterTopRootEvents(e, t) { for (let r = 0; r < t.length; r++) { const n = t[r]; e.removeEventListener(n.name, n.handler); try { e.parent && e.parent.removeEventListener(n.name, n.handler); } catch { } } } /** * Dumps the current bound framebuffer * @param width defines the rendering width * @param height defines the rendering height * @param engine defines the hosting engine * @param successCallback defines the callback triggered once the data are available * @param mimeType defines the mime type of the result * @param fileName defines the filename to download. If present, the result will automatically be downloaded * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. * @returns a void promise */ static async DumpFramebuffer(e, t, r, n, i = "image/png", s, a) { throw qn("DumpTools"); } /** * Dumps an array buffer * @param width defines the rendering width * @param height defines the rendering height * @param data the data array * @param successCallback defines the callback triggered once the data are available * @param mimeType defines the mime type of the result * @param fileName defines the filename to download. If present, the result will automatically be downloaded * @param invertY true to invert the picture in the Y dimension * @param toArrayBuffer true to convert the data to an ArrayBuffer (encoded as `mimeType`) instead of a base64 string * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. */ static DumpData(e, t, r, n, i = "image/png", s, a = !1, f = !1, o) { throw qn("DumpTools"); } /** * Dumps an array buffer * @param width defines the rendering width * @param height defines the rendering height * @param data the data array * @param mimeType defines the mime type of the result * @param fileName defines the filename to download. If present, the result will automatically be downloaded * @param invertY true to invert the picture in the Y dimension * @param toArrayBuffer true to convert the data to an ArrayBuffer (encoded as `mimeType`) instead of a base64 string * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. * @returns a promise that resolve to the final data */ static DumpDataAsync(e, t, r, n = "image/png", i, s = !1, a = !1, f) { throw qn("DumpTools"); } static _IsOffScreenCanvas(e) { return e.convertToBlob !== void 0; } /** * Converts the canvas data to blob. * This acts as a polyfill for browsers not supporting the to blob function. * @param canvas Defines the canvas to extract the data from (can be an offscreen canvas) * @param successCallback Defines the callback triggered once the data are available * @param mimeType Defines the mime type of the result * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. */ static ToBlob(e, t, r = "image/png", n) { !ye._IsOffScreenCanvas(e) && !e.toBlob && (e.toBlob = function(i, s, a) { setTimeout(() => { const f = atob(this.toDataURL(s, a).split(",")[1]), o = f.length, d = new Uint8Array(o); for (let v = 0; v < o; v++) d[v] = f.charCodeAt(v); i(new Blob([d])); }); }), ye._IsOffScreenCanvas(e) ? e.convertToBlob({ type: r, quality: n }).then((i) => t(i)) : e.toBlob(function(i) { t(i); }, r, n); } /** * Download a Blob object * @param blob the Blob object * @param fileName the file name to download * @returns */ static DownloadBlob(e, t) { if ("download" in document.createElement("a")) { if (!t) { const r = /* @__PURE__ */ new Date(); t = "screenshot_" + ((r.getFullYear() + "-" + (r.getMonth() + 1)).slice(2) + "-" + r.getDate() + "_" + r.getHours() + "-" + ("0" + r.getMinutes()).slice(-2)) + ".png"; } ye.Download(e, t); } else if (e && typeof URL < "u") { const r = URL.createObjectURL(e), n = window.open(""); if (!n) return; const i = n.document.createElement("img"); i.onload = function() { URL.revokeObjectURL(r); }, i.src = r, n.document.body.appendChild(i); } } /** * Encodes the canvas data to base 64, or automatically downloads the result if `fileName` is defined. * @param canvas The canvas to get the data from, which can be an offscreen canvas. * @param successCallback The callback which is triggered once the data is available. If `fileName` is defined, the callback will be invoked after the download occurs, and the `data` argument will be an empty string. * @param mimeType The mime type of the result. * @param fileName The name of the file to download. If defined, the result will automatically be downloaded. If not defined, and `successCallback` is also not defined, the result will automatically be downloaded with an auto-generated file name. * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. */ static EncodeScreenshotCanvasData(e, t, r = "image/png", n, i) { if (typeof n == "string" || !t) this.ToBlob(e, function(s) { s && ye.DownloadBlob(s, n), t && t(""); }, r, i); else if (t) { if (ye._IsOffScreenCanvas(e)) { e.convertToBlob({ type: r, quality: i }).then((a) => { const f = new FileReader(); f.readAsDataURL(a), f.onloadend = () => { const o = f.result; t(o); }; }); return; } const s = e.toDataURL(r, i); t(s); } } /** * Downloads a blob in the browser * @param blob defines the blob to download * @param fileName defines the name of the downloaded file */ static Download(e, t) { if (typeof URL > "u") return; const r = window.URL.createObjectURL(e), n = document.createElement("a"); document.body.appendChild(n), n.style.display = "none", n.href = r, n.download = t, n.addEventListener("click", () => { n.parentElement && n.parentElement.removeChild(n); }), n.click(), window.URL.revokeObjectURL(r); } /** * Will return the right value of the noPreventDefault variable * Needed to keep backwards compatibility to the old API. * * @param args arguments passed to the attachControl function * @returns the correct value for noPreventDefault */ static BackCompatCameraNoPreventDefault(e) { return typeof e[0] == "boolean" ? e[0] : typeof e[1] == "boolean" ? e[1] : !1; } /** * Captures a screenshot of the current rendering * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG * @param engine defines the rendering engine * @param camera defines the source camera * @param size This parameter can be set to a single number or to an object with the * following (optional) properties: precision, width, height. If a single number is passed, * it will be used for both width and height. If an object is passed, the screenshot size * will be derived from the parameters. The precision property is a multiplier allowing * rendering at a higher or lower resolution * @param successCallback defines the callback receives a single parameter which contains the * screenshot as a string of base64-encoded characters. This string can be assigned to the * src parameter of an to display it * @param mimeType defines the MIME type of the screenshot image (default: image/png). * Check your browser for supported MIME types * @param forceDownload force the system to download the image even if a successCallback is provided * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. */ // eslint-disable-next-line @typescript-eslint/no-unused-vars static CreateScreenshot(e, t, r, n, i = "image/png", s = !1, a) { throw qn("ScreenshotTools"); } /** * Captures a screenshot of the current rendering * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG * @param engine defines the rendering engine * @param camera defines the source camera * @param size This parameter can be set to a single number or to an object with the * following (optional) properties: precision, width, height. If a single number is passed, * it will be used for both width and height. If an object is passed, the screenshot size * will be derived from the parameters. The precision property is a multiplier allowing * rendering at a higher or lower resolution * @param mimeType defines the MIME type of the screenshot image (default: image/png). * Check your browser for supported MIME types * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. * @returns screenshot as a string of base64-encoded characters. This string can be assigned * to the src parameter of an to display it */ // eslint-disable-next-line @typescript-eslint/no-unused-vars static CreateScreenshotAsync(e, t, r, n = "image/png", i) { throw qn("ScreenshotTools"); } /** * Generates an image screenshot from the specified camera. * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG * @param engine The engine to use for rendering * @param camera The camera to use for rendering * @param size This parameter can be set to a single number or to an object with the * following (optional) properties: precision, width, height. If a single number is passed, * it will be used for both width and height. If an object is passed, the screenshot size * will be derived from the parameters. The precision property is a multiplier allowing * rendering at a higher or lower resolution * @param successCallback The callback receives a single parameter which contains the * screenshot as a string of base64-encoded characters. This string can be assigned to the * src parameter of an to display it * @param mimeType The MIME type of the screenshot image (default: image/png). * Check your browser for supported MIME types * @param samples Texture samples (default: 1) * @param antialiasing Whether antialiasing should be turned on or not (default: false) * @param fileName A name for for the downloaded file. * @param renderSprites Whether the sprites should be rendered or not (default: false) * @param enableStencilBuffer Whether the stencil buffer should be enabled or not (default: false) * @param useLayerMask if the camera's layer mask should be used to filter what should be rendered (default: true) * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. */ // eslint-disable-next-line @typescript-eslint/no-unused-vars static CreateScreenshotUsingRenderTarget(e, t, r, n, i = "image/png", s = 1, a = !1, f, o = !1, d = !1, v = !0, u) { throw qn("ScreenshotTools"); } /** * Generates an image screenshot from the specified camera. * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG * @param engine The engine to use for rendering * @param camera The camera to use for rendering * @param size This parameter can be set to a single number or to an object with the * following (optional) properties: precision, width, height. If a single number is passed, * it will be used for both width and height. If an object is passed, the screenshot size * will be derived from the parameters. The precision property is a multiplier allowing * rendering at a higher or lower resolution * @param mimeType The MIME type of the screenshot image (default: image/png). * Check your browser for supported MIME types * @param samples Texture samples (default: 1) * @param antialiasing Whether antialiasing should be turned on or not (default: false) * @param fileName A name for for the downloaded file. * @returns screenshot as a string of base64-encoded characters. This string can be assigned * @param renderSprites Whether the sprites should be rendered or not (default: false) * @param enableStencilBuffer Whether the stencil buffer should be enabled or not (default: false) * @param useLayerMask if the camera's layer mask should be used to filter what should be rendered (default: true) * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. * to the src parameter of an to display it */ // eslint-disable-next-line @typescript-eslint/no-unused-vars static CreateScreenshotUsingRenderTargetAsync(e, t, r, n = "image/png", i = 1, s = !1, a, f = !1, o = !1, d = !0, v) { throw qn("ScreenshotTools"); } /** * Implementation from http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript/2117523#answer-2117523 * Be aware Math.random() could cause collisions, but: * "All but 6 of the 128 bits of the ID are randomly generated, which means that for any two ids, there's a 1 in 2^^122 (or 5.3x10^^36) chance they'll collide" * @returns a pseudo random id */ static RandomId() { return v4(); } /** * Test if the given uri is a base64 string * @deprecated Please use FileTools.IsBase64DataUrl instead. * @param uri The uri to test * @returns True if the uri is a base64 string or false otherwise */ static IsBase64(e) { return qR(e); } /** * Decode the given base64 uri. * @deprecated Please use FileTools.DecodeBase64UrlToBinary instead. * @param uri The uri to decode * @returns The decoded base64 data. */ static DecodeBase64(e) { return iU(e); } /** * Gets a value indicating the number of loading errors * @ignorenaming */ // eslint-disable-next-line @typescript-eslint/naming-convention static get errorsCount() { return Se.errorsCount; } /** * Log a message to the console * @param message defines the message to log */ static Log(e) { Se.Log(e); } /** * Write a warning message to the console * @param message defines the message to log */ static Warn(e) { Se.Warn(e); } /** * Write an error message to the console * @param message defines the message to log */ static Error(e) { Se.Error(e); } /** * Gets current log cache (list of logs) */ static get LogCache() { return Se.LogCache; } /** * Clears the log cache */ static ClearLogCache() { Se.ClearLogCache(); } /** * Sets the current log level (MessageLogLevel / WarningLogLevel / ErrorLogLevel) */ static set LogLevels(e) { Se.LogLevels = e; } /** * Sets the current performance log level */ static set PerformanceLogLevel(e) { if ((e & ye.PerformanceUserMarkLogLevel) === ye.PerformanceUserMarkLogLevel) { ye.StartPerformanceCounter = ye._StartUserMark, ye.EndPerformanceCounter = ye._EndUserMark; return; } if ((e & ye.PerformanceConsoleLogLevel) === ye.PerformanceConsoleLogLevel) { ye.StartPerformanceCounter = ye._StartPerformanceConsole, ye.EndPerformanceCounter = ye._EndPerformanceConsole; return; } ye.StartPerformanceCounter = ye._StartPerformanceCounterDisabled, ye.EndPerformanceCounter = ye._EndPerformanceCounterDisabled; } // eslint-disable-next-line @typescript-eslint/no-unused-vars static _StartPerformanceCounterDisabled(e, t) { } // eslint-disable-next-line @typescript-eslint/no-unused-vars static _EndPerformanceCounterDisabled(e, t) { } static _StartUserMark(e, t = !0) { if (!ye._Performance) { if (!u9()) return; ye._Performance = window.performance; } !t || !ye._Performance.mark || ye._Performance.mark(e + "-Begin"); } static _EndUserMark(e, t = !0) { !t || !ye._Performance.mark || (ye._Performance.mark(e + "-End"), ye._Performance.measure(e, e + "-Begin", e + "-End")); } static _StartPerformanceConsole(e, t = !0) { t && (ye._StartUserMark(e, t), console.time && console.time(e)); } static _EndPerformanceConsole(e, t = !0) { t && (ye._EndUserMark(e, t), console.timeEnd(e)); } /** * Gets either window.performance.now() if supported or Date.now() else */ static get Now() { return Yi.Now; } /** * This method will return the name of the class used to create the instance of the given object. * It will works only on Javascript basic data types (number, string, ...) and instance of class declared with the @className decorator. * @param object the object to get the class name from * @param isType defines if the object is actually a type * @returns the name of the class, will be "object" for a custom data type not using the @className decorator */ static GetClassName(e, t = !1) { let r = null; return !t && e.getClassName ? r = e.getClassName() : (e instanceof Object && (r = (t ? e : Object.getPrototypeOf(e)).constructor.__bjsclassName__), r || (r = typeof e)), r; } /** * Gets the first element of an array satisfying a given predicate * @param array defines the array to browse * @param predicate defines the predicate to use * @returns null if not found or the element */ static First(e, t) { for (const r of e) if (t(r)) return r; return null; } /** * This method will return the name of the full name of the class, including its owning module (if any). * It will works only on Javascript basic data types (number, string, ...) and instance of class declared with the @className decorator or implementing a method getClassName():string (in which case the module won't be specified). * @param object the object to get the class name from * @param isType defines if the object is actually a type * @returns a string that can have two forms: "moduleName.className" if module was specified when the class' Name was registered or "className" if there was not module specified. * @ignorenaming */ // eslint-disable-next-line @typescript-eslint/naming-convention static getFullClassName(e, t = !1) { let r = null, n = null; if (!t && e.getClassName) r = e.getClassName(); else { if (e instanceof Object) { const i = t ? e : Object.getPrototypeOf(e); r = i.constructor.__bjsclassName__, n = i.constructor.__bjsmoduleName__; } r || (r = typeof e); } return r ? (n != null ? n + "." : "") + r : null; } /** * Returns a promise that resolves after the given amount of time. * @param delay Number of milliseconds to delay * @returns Promise that resolves after the given amount of time */ static DelayAsync(e) { return new Promise((t) => { setTimeout(() => { t(); }, e); }); } /** * Utility function to detect if the current user agent is Safari * @returns whether or not the current user agent is safari */ static IsSafari() { return Mw() ? /^((?!chrome|android).)*safari/i.test(navigator.userAgent) : !1; } } ye.UseCustomRequestHeaders = !1; ye.CustomRequestHeaders = ho.CustomRequestHeaders; ye.GetDOMTextContent = gR; ye._DefaultCdnUrl = "https://cdn.babylonjs.com"; ye.GetAbsoluteUrl = typeof document == "object" ? (A) => { const e = document.createElement("a"); return e.href = A, e.href; } : typeof URL == "function" && typeof location == "object" ? (A) => new URL(A, location.origin).href : () => { throw new Error("Unable to get absolute URL. Override BABYLON.Tools.GetAbsoluteUrl to a custom implementation for the current context."); }; ye.NoneLogLevel = Se.NoneLogLevel; ye.MessageLogLevel = Se.MessageLogLevel; ye.WarningLogLevel = Se.WarningLogLevel; ye.ErrorLogLevel = Se.ErrorLogLevel; ye.AllLogLevel = Se.AllLogLevel; ye.IsWindowObjectExist = u9; ye.PerformanceNoneLogLevel = 0; ye.PerformanceUserMarkLogLevel = 1; ye.PerformanceConsoleLogLevel = 2; ye.StartPerformanceCounter = ye._StartPerformanceCounterDisabled; ye.EndPerformanceCounter = ye._EndPerformanceCounterDisabled; function G9e(A, e) { return (t) => { t.__bjsclassName__ = A, t.__bjsmoduleName__ = e ?? null; }; } class up { /** * Constructor. * @param iterations the number of iterations. * @param func the function to run each iteration * @param successCallback the callback that will be called upon successful execution * @param offset starting offset. */ constructor(e, t, r, n = 0) { this.iterations = e, this.index = n - 1, this._done = !1, this._fn = t, this._successCallback = r; } /** * Execute the next iteration. Must be called after the last iteration was finished. */ executeNext() { this._done || (this.index + 1 < this.iterations ? (++this.index, this._fn(this)) : this.breakLoop()); } /** * Break the loop and run the success callback. */ breakLoop() { this._done = !0, this._successCallback(); } /** * Create and run an async loop. * @param iterations the number of iterations. * @param fn the function to run each iteration * @param successCallback the callback that will be called upon successful execution * @param offset starting offset. * @returns the created async loop object */ static Run(e, t, r, n = 0) { const i = new up(e, t, r, n); return i.executeNext(), i; } /** * A for-loop that will run a given number of iterations synchronous and the rest async. * @param iterations total number of iterations * @param syncedIterations number of synchronous iterations in each async iteration. * @param fn the function to call each iteration. * @param callback a success call back that will be called when iterating stops. * @param breakFunction a break condition (optional) * @param timeout timeout settings for the setTimeout function. default - 0. * @returns the created async loop object */ static SyncAsyncForLoop(e, t, r, n, i, s = 0) { return up.Run(Math.ceil(e / t), (a) => { i && i() ? a.breakLoop() : setTimeout(() => { for (let f = 0; f < t; ++f) { const o = a.index * t + f; if (o >= e) break; if (r(o), i && i()) { a.breakLoop(); break; } } a.executeNext(); }, s); }, n); } } gr.FallbackTexture = ""; class qf { /** * Instantiates a Smart Array. * @param capacity defines the default capacity of the array. */ constructor(e) { this.length = 0, this.data = new Array(e), this._id = qf._GlobalId++; } /** * Pushes a value at the end of the active data. * @param value defines the object to push in the array. */ push(e) { this.data[this.length++] = e, this.length > this.data.length && (this.data.length *= 2); } /** * Iterates over the active data and apply the lambda to them. * @param func defines the action to apply on each value. */ forEach(e) { for (let t = 0; t < this.length; t++) e(this.data[t]); } /** * Sorts the full sets of data. * @param compareFn defines the comparison function to apply. */ sort(e) { this.data.sort(e); } /** * Resets the active data to an empty array. */ reset() { this.length = 0; } /** * Releases all the data from the array as well as the array. */ dispose() { this.reset(), this.data && (this.data.length = 0); } /** * Concats the active data with a given array. * @param array defines the data to concatenate with. */ concat(e) { if (e.length !== 0) { this.length + e.length > this.data.length && (this.data.length = (this.length + e.length) * 2); for (let t = 0; t < e.length; t++) this.data[this.length++] = (e.data || e)[t]; } } /** * Returns the position of a value in the active data. * @param value defines the value to find the index for * @returns the index if found in the active data otherwise -1 */ indexOf(e) { const t = this.data.indexOf(e); return t >= this.length ? -1 : t; } /** * Returns whether an element is part of the active data. * @param value defines the value to look for * @returns true if found in the active data otherwise false */ contains(e) { return this.indexOf(e) !== -1; } } qf._GlobalId = 0; class K8 extends qf { constructor() { super(...arguments), this._duplicateId = 0; } /** * Pushes a value at the end of the active data. * THIS DOES NOT PREVENT DUPPLICATE DATA * @param value defines the object to push in the array. */ push(e) { super.push(e), e.__smartArrayFlags || (e.__smartArrayFlags = {}), e.__smartArrayFlags[this._id] = this._duplicateId; } /** * Pushes a value at the end of the active data. * If the data is already present, it won t be added again * @param value defines the object to push in the array. * @returns true if added false if it was already present */ pushNoDuplicate(e) { return e.__smartArrayFlags && e.__smartArrayFlags[this._id] === this._duplicateId ? !1 : (this.push(e), !0); } /** * Resets the active data to an empty array. */ reset() { super.reset(), this._duplicateId++; } /** * Concats the active data with a given array. * This ensures no duplicate will be present in the result. * @param array defines the data to concatenate with. */ concatWithNoDuplicate(e) { if (e.length !== 0) { this.length + e.length > this.data.length && (this.data.length = (this.length + e.length) * 2); for (let t = 0; t < e.length; t++) { const r = (e.data || e)[t]; this.pushNoDuplicate(r); } } } } class YC { constructor() { this._count = 0, this._data = {}; } /** * This will clear this dictionary and copy the content from the 'source' one. * If the T value is a custom object, it won't be copied/cloned, the same object will be used * @param source the dictionary to take the content from and copy to this dictionary */ copyFrom(e) { this.clear(), e.forEach((t, r) => this.add(t, r)); } /** * Get a value based from its key * @param key the given key to get the matching value from * @returns the value if found, otherwise undefined is returned */ get(e) { const t = this._data[e]; if (t !== void 0) return t; } /** * Get a value from its key or add it if it doesn't exist. * This method will ensure you that a given key/data will be present in the dictionary. * @param key the given key to get the matching value from * @param factory the factory that will create the value if the key is not present in the dictionary. * The factory will only be invoked if there's no data for the given key. * @returns the value corresponding to the key. */ getOrAddWithFactory(e, t) { let r = this.get(e); return r !== void 0 || (r = t(e), r && this.add(e, r)), r; } /** * Get a value from its key if present in the dictionary otherwise add it * @param key the key to get the value from * @param val if there's no such key/value pair in the dictionary add it with this value * @returns the value corresponding to the key */ getOrAdd(e, t) { const r = this.get(e); return r !== void 0 ? r : (this.add(e, t), t); } /** * Check if there's a given key in the dictionary * @param key the key to check for * @returns true if the key is present, false otherwise */ contains(e) { return this._data[e] !== void 0; } /** * Add a new key and its corresponding value * @param key the key to add * @param value the value corresponding to the key * @returns true if the operation completed successfully, false if we couldn't insert the key/value because there was already this key in the dictionary */ add(e, t) { return this._data[e] !== void 0 ? !1 : (this._data[e] = t, ++this._count, !0); } /** * Update a specific value associated to a key * @param key defines the key to use * @param value defines the value to store * @returns true if the value was updated (or false if the key was not found) */ set(e, t) { return this._data[e] === void 0 ? !1 : (this._data[e] = t, !0); } /** * Get the element of the given key and remove it from the dictionary * @param key defines the key to search * @returns the value associated with the key or null if not found */ getAndRemove(e) { const t = this.get(e); return t !== void 0 ? (delete this._data[e], --this._count, t) : null; } /** * Remove a key/value from the dictionary. * @param key the key to remove * @returns true if the item was successfully deleted, false if no item with such key exist in the dictionary */ remove(e) { return this.contains(e) ? (delete this._data[e], --this._count, !0) : !1; } /** * Clear the whole content of the dictionary */ clear() { this._data = {}, this._count = 0; } /** * Gets the current count */ get count() { return this._count; } /** * Execute a callback on each key/val of the dictionary. * Note that you can remove any element in this dictionary in the callback implementation * @param callback the callback to execute on a given key/value pair */ forEach(e) { for (const t in this._data) { const r = this._data[t]; e(t, r); } } /** * Execute a callback on every occurrence of the dictionary until it returns a valid TRes object. * If the callback returns null or undefined the method will iterate to the next key/value pair * Note that you can remove any element in this dictionary in the callback implementation * @param callback the callback to execute, if it return a valid T instanced object the enumeration will stop and the object will be returned * @returns the first item */ first(e) { for (const t in this._data) { const r = this._data[t], n = e(t, r); if (n) return n; } return null; } } class na { /** * Creates a new instance * @param externalProperties list of external properties to inject into the object */ constructor(e) { if (this._keys = [], this._isDirty = !0, this._areLightsDirty = !0, this._areLightsDisposed = !1, this._areAttributesDirty = !0, this._areTexturesDirty = !0, this._areFresnelDirty = !0, this._areMiscDirty = !0, this._arePrePassDirty = !0, this._areImageProcessingDirty = !0, this._normals = !1, this._uvs = !1, this._needNormals = !1, this._needUVs = !1, this._externalProperties = e, e) for (const t in e) Object.prototype.hasOwnProperty.call(e, t) && this._setDefaultValue(t); } /** * Specifies if the material needs to be re-calculated */ get isDirty() { return this._isDirty; } /** * Marks the material to indicate that it has been re-calculated */ markAsProcessed() { this._isDirty = !1, this._areAttributesDirty = !1, this._areTexturesDirty = !1, this._areFresnelDirty = !1, this._areLightsDirty = !1, this._areLightsDisposed = !1, this._areMiscDirty = !1, this._arePrePassDirty = !1, this._areImageProcessingDirty = !1; } /** * Marks the material to indicate that it needs to be re-calculated */ markAsUnprocessed() { this._isDirty = !0; } /** * Marks the material to indicate all of its defines need to be re-calculated */ markAllAsDirty() { this._areTexturesDirty = !0, this._areAttributesDirty = !0, this._areLightsDirty = !0, this._areFresnelDirty = !0, this._areMiscDirty = !0, this._arePrePassDirty = !1, this._areImageProcessingDirty = !0, this._isDirty = !0; } /** * Marks the material to indicate that image processing needs to be re-calculated */ markAsImageProcessingDirty() { this._areImageProcessingDirty = !0, this._isDirty = !0; } /** * Marks the material to indicate the lights need to be re-calculated * @param disposed Defines whether the light is dirty due to dispose or not */ markAsLightDirty(e = !1) { this._areLightsDirty = !0, this._areLightsDisposed = this._areLightsDisposed || e, this._isDirty = !0; } /** * Marks the attribute state as changed */ markAsAttributesDirty() { this._areAttributesDirty = !0, this._isDirty = !0; } /** * Marks the texture state as changed */ markAsTexturesDirty() { this._areTexturesDirty = !0, this._isDirty = !0; } /** * Marks the fresnel state as changed */ markAsFresnelDirty() { this._areFresnelDirty = !0, this._isDirty = !0; } /** * Marks the misc state as changed */ markAsMiscDirty() { this._areMiscDirty = !0, this._isDirty = !0; } /** * Marks the prepass state as changed */ markAsPrePassDirty() { this._arePrePassDirty = !0, this._isDirty = !0; } /** * Rebuilds the material defines */ rebuild() { this._keys.length = 0; for (const e of Object.keys(this)) e[0] !== "_" && this._keys.push(e); if (this._externalProperties) for (const e in this._externalProperties) this._keys.indexOf(e) === -1 && this._keys.push(e); } /** * Specifies if two material defines are equal * @param other - A material define instance to compare to * @returns - Boolean indicating if the material defines are equal (true) or not (false) */ isEqual(e) { if (this._keys.length !== e._keys.length) return !1; for (let t = 0; t < this._keys.length; t++) { const r = this._keys[t]; if (this[r] !== e[r]) return !1; } return !0; } /** * Clones this instance's defines to another instance * @param other - material defines to clone values to */ cloneTo(e) { this._keys.length !== e._keys.length && (e._keys = this._keys.slice(0)); for (let t = 0; t < this._keys.length; t++) { const r = this._keys[t]; e[r] = this[r]; } } /** * Resets the material define values */ reset() { this._keys.forEach((e) => this._setDefaultValue(e)); } _setDefaultValue(e) { var t, r, n, i, s; const a = (n = (r = (t = this._externalProperties) === null || t === void 0 ? void 0 : t[e]) === null || r === void 0 ? void 0 : r.type) !== null && n !== void 0 ? n : typeof this[e], f = (s = (i = this._externalProperties) === null || i === void 0 ? void 0 : i[e]) === null || s === void 0 ? void 0 : s.default; switch (a) { case "number": this[e] = f ?? 0; break; case "string": this[e] = f ?? ""; break; default: this[e] = f ?? !1; break; } } /** * Converts the material define values to a string * @returns - String of material define information */ toString() { let e = ""; for (let t = 0; t < this._keys.length; t++) { const r = this._keys[t], n = this[r]; switch (typeof n) { case "number": case "string": e += "#define " + r + " " + n + ` `; break; default: n && (e += "#define " + r + ` `); break; } } return e; } } class so { constructor() { this._dirty = !0, this._tempColor = new xt(0, 0, 0, 0), this._globalCurve = new xt(0, 0, 0, 0), this._highlightsCurve = new xt(0, 0, 0, 0), this._midtonesCurve = new xt(0, 0, 0, 0), this._shadowsCurve = new xt(0, 0, 0, 0), this._positiveCurve = new xt(0, 0, 0, 0), this._negativeCurve = new xt(0, 0, 0, 0), this._globalHue = 30, this._globalDensity = 0, this._globalSaturation = 0, this._globalExposure = 0, this._highlightsHue = 30, this._highlightsDensity = 0, this._highlightsSaturation = 0, this._highlightsExposure = 0, this._midtonesHue = 30, this._midtonesDensity = 0, this._midtonesSaturation = 0, this._midtonesExposure = 0, this._shadowsHue = 30, this._shadowsDensity = 0, this._shadowsSaturation = 0, this._shadowsExposure = 0; } /** * Gets the global Hue value. * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange). */ get globalHue() { return this._globalHue; } /** * Sets the global Hue value. * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange). */ set globalHue(e) { this._globalHue = e, this._dirty = !0; } /** * Gets the global Density value. * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect. * Values less than zero provide a filter of opposite hue. */ get globalDensity() { return this._globalDensity; } /** * Sets the global Density value. * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect. * Values less than zero provide a filter of opposite hue. */ set globalDensity(e) { this._globalDensity = e, this._dirty = !0; } /** * Gets the global Saturation value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation. */ get globalSaturation() { return this._globalSaturation; } /** * Sets the global Saturation value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation. */ set globalSaturation(e) { this._globalSaturation = e, this._dirty = !0; } /** * Gets the global Exposure value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure. */ get globalExposure() { return this._globalExposure; } /** * Sets the global Exposure value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure. */ set globalExposure(e) { this._globalExposure = e, this._dirty = !0; } /** * Gets the highlights Hue value. * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange). */ get highlightsHue() { return this._highlightsHue; } /** * Sets the highlights Hue value. * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange). */ set highlightsHue(e) { this._highlightsHue = e, this._dirty = !0; } /** * Gets the highlights Density value. * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect. * Values less than zero provide a filter of opposite hue. */ get highlightsDensity() { return this._highlightsDensity; } /** * Sets the highlights Density value. * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect. * Values less than zero provide a filter of opposite hue. */ set highlightsDensity(e) { this._highlightsDensity = e, this._dirty = !0; } /** * Gets the highlights Saturation value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation. */ get highlightsSaturation() { return this._highlightsSaturation; } /** * Sets the highlights Saturation value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation. */ set highlightsSaturation(e) { this._highlightsSaturation = e, this._dirty = !0; } /** * Gets the highlights Exposure value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure. */ get highlightsExposure() { return this._highlightsExposure; } /** * Sets the highlights Exposure value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure. */ set highlightsExposure(e) { this._highlightsExposure = e, this._dirty = !0; } /** * Gets the midtones Hue value. * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange). */ get midtonesHue() { return this._midtonesHue; } /** * Sets the midtones Hue value. * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange). */ set midtonesHue(e) { this._midtonesHue = e, this._dirty = !0; } /** * Gets the midtones Density value. * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect. * Values less than zero provide a filter of opposite hue. */ get midtonesDensity() { return this._midtonesDensity; } /** * Sets the midtones Density value. * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect. * Values less than zero provide a filter of opposite hue. */ set midtonesDensity(e) { this._midtonesDensity = e, this._dirty = !0; } /** * Gets the midtones Saturation value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation. */ get midtonesSaturation() { return this._midtonesSaturation; } /** * Sets the midtones Saturation value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation. */ set midtonesSaturation(e) { this._midtonesSaturation = e, this._dirty = !0; } /** * Gets the midtones Exposure value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure. */ get midtonesExposure() { return this._midtonesExposure; } /** * Sets the midtones Exposure value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure. */ set midtonesExposure(e) { this._midtonesExposure = e, this._dirty = !0; } /** * Gets the shadows Hue value. * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange). */ get shadowsHue() { return this._shadowsHue; } /** * Sets the shadows Hue value. * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange). */ set shadowsHue(e) { this._shadowsHue = e, this._dirty = !0; } /** * Gets the shadows Density value. * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect. * Values less than zero provide a filter of opposite hue. */ get shadowsDensity() { return this._shadowsDensity; } /** * Sets the shadows Density value. * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect. * Values less than zero provide a filter of opposite hue. */ set shadowsDensity(e) { this._shadowsDensity = e, this._dirty = !0; } /** * Gets the shadows Saturation value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation. */ get shadowsSaturation() { return this._shadowsSaturation; } /** * Sets the shadows Saturation value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation. */ set shadowsSaturation(e) { this._shadowsSaturation = e, this._dirty = !0; } /** * Gets the shadows Exposure value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure. */ get shadowsExposure() { return this._shadowsExposure; } /** * Sets the shadows Exposure value. * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure. */ set shadowsExposure(e) { this._shadowsExposure = e, this._dirty = !0; } /** * Returns the class name * @returns The class name */ getClassName() { return "ColorCurves"; } /** * Binds the color curves to the shader. * @param colorCurves The color curve to bind * @param effect The effect to bind to * @param positiveUniform The positive uniform shader parameter * @param neutralUniform The neutral uniform shader parameter * @param negativeUniform The negative uniform shader parameter */ static Bind(e, t, r = "vCameraColorCurvePositive", n = "vCameraColorCurveNeutral", i = "vCameraColorCurveNegative") { e._dirty && (e._dirty = !1, e._getColorGradingDataToRef(e._globalHue, e._globalDensity, e._globalSaturation, e._globalExposure, e._globalCurve), e._getColorGradingDataToRef(e._highlightsHue, e._highlightsDensity, e._highlightsSaturation, e._highlightsExposure, e._tempColor), e._tempColor.multiplyToRef(e._globalCurve, e._highlightsCurve), e._getColorGradingDataToRef(e._midtonesHue, e._midtonesDensity, e._midtonesSaturation, e._midtonesExposure, e._tempColor), e._tempColor.multiplyToRef(e._globalCurve, e._midtonesCurve), e._getColorGradingDataToRef(e._shadowsHue, e._shadowsDensity, e._shadowsSaturation, e._shadowsExposure, e._tempColor), e._tempColor.multiplyToRef(e._globalCurve, e._shadowsCurve), e._highlightsCurve.subtractToRef(e._midtonesCurve, e._positiveCurve), e._midtonesCurve.subtractToRef(e._shadowsCurve, e._negativeCurve)), t && (t.setFloat4(r, e._positiveCurve.r, e._positiveCurve.g, e._positiveCurve.b, e._positiveCurve.a), t.setFloat4(n, e._midtonesCurve.r, e._midtonesCurve.g, e._midtonesCurve.b, e._midtonesCurve.a), t.setFloat4(i, e._negativeCurve.r, e._negativeCurve.g, e._negativeCurve.b, e._negativeCurve.a)); } /** * Prepare the list of uniforms associated with the ColorCurves effects. * @param uniformsList The list of uniforms used in the effect */ static PrepareUniforms(e) { e.push("vCameraColorCurveNeutral", "vCameraColorCurvePositive", "vCameraColorCurveNegative"); } /** * Returns color grading data based on a hue, density, saturation and exposure value. * @param hue * @param density * @param saturation The saturation. * @param exposure The exposure. * @param result The result data container. */ _getColorGradingDataToRef(e, t, r, n, i) { e != null && (e = so._Clamp(e, 0, 360), t = so._Clamp(t, -100, 100), r = so._Clamp(r, -100, 100), n = so._Clamp(n, -100, 100), t = so._ApplyColorGradingSliderNonlinear(t), t *= 0.5, n = so._ApplyColorGradingSliderNonlinear(n), t < 0 && (t *= -1, e = (e + 180) % 360), so._FromHSBToRef(e, t, 50 + 0.25 * n, i), i.scaleToRef(2, i), i.a = 1 + 0.01 * r); } /** * Takes an input slider value and returns an adjusted value that provides extra control near the centre. * @param value The input slider value in range [-100,100]. * @returns Adjusted value. */ static _ApplyColorGradingSliderNonlinear(e) { e /= 100; let t = Math.abs(e); return t = Math.pow(t, 2), e < 0 && (t *= -1), t *= 100, t; } /** * Returns an RGBA Color4 based on Hue, Saturation and Brightness (also referred to as value, HSV). * @param hue The hue (H) input. * @param saturation The saturation (S) input. * @param brightness The brightness (B) input. * @param result * @result An RGBA color represented as Vector4. */ static _FromHSBToRef(e, t, r, n) { let i = so._Clamp(e, 0, 360); const s = so._Clamp(t / 100, 0, 1), a = so._Clamp(r / 100, 0, 1); if (s === 0) n.r = a, n.g = a, n.b = a; else { i /= 60; const f = Math.floor(i), o = i - f, d = a * (1 - s), v = a * (1 - s * o), u = a * (1 - s * (1 - o)); switch (f) { case 0: n.r = a, n.g = u, n.b = d; break; case 1: n.r = v, n.g = a, n.b = d; break; case 2: n.r = d, n.g = a, n.b = u; break; case 3: n.r = d, n.g = v, n.b = a; break; case 4: n.r = u, n.g = d, n.b = a; break; default: n.r = a, n.g = d, n.b = v; break; } } n.a = 1; } /** * Returns a value clamped between min and max * @param value The value to clamp * @param min The minimum of value * @param max The maximum of value * @returns The clamped value. */ static _Clamp(e, t, r) { return Math.min(Math.max(e, t), r); } /** * Clones the current color curve instance. * @returns The cloned curves */ clone() { return jt.Clone(() => new so(), this); } /** * Serializes the current color curve instance to a json representation. * @returns a JSON representation */ serialize() { return jt.Serialize(this); } /** * Parses the color curve from a json representation. * @param source the JSON source to parse * @returns The parsed curves */ static Parse(e) { return jt.Parse(() => new so(), e, null, null); } } C([ M() ], so.prototype, "_globalHue", void 0); C([ M() ], so.prototype, "_globalDensity", void 0); C([ M() ], so.prototype, "_globalSaturation", void 0); C([ M() ], so.prototype, "_globalExposure", void 0); C([ M() ], so.prototype, "_highlightsHue", void 0); C([ M() ], so.prototype, "_highlightsDensity", void 0); C([ M() ], so.prototype, "_highlightsSaturation", void 0); C([ M() ], so.prototype, "_highlightsExposure", void 0); C([ M() ], so.prototype, "_midtonesHue", void 0); C([ M() ], so.prototype, "_midtonesDensity", void 0); C([ M() ], so.prototype, "_midtonesSaturation", void 0); C([ M() ], so.prototype, "_midtonesExposure", void 0); jt._ColorCurvesParser = so.Parse; class M$ extends na { constructor() { super(), this.IMAGEPROCESSING = !1, this.VIGNETTE = !1, this.VIGNETTEBLENDMODEMULTIPLY = !1, this.VIGNETTEBLENDMODEOPAQUE = !1, this.TONEMAPPING = !1, this.TONEMAPPING_ACES = !1, this.CONTRAST = !1, this.COLORCURVES = !1, this.COLORGRADING = !1, this.COLORGRADING3D = !1, this.SAMPLER3DGREENDEPTH = !1, this.SAMPLER3DBGRMAP = !1, this.DITHER = !1, this.IMAGEPROCESSINGPOSTPROCESS = !1, this.EXPOSURE = !1, this.SKIPFINALCOLORCLAMP = !1, this.rebuild(); } } class Ui { constructor() { this.colorCurves = new so(), this._colorCurvesEnabled = !1, this._colorGradingEnabled = !1, this._colorGradingWithGreenDepth = !0, this._colorGradingBGR = !0, this._exposure = 1, this._toneMappingEnabled = !1, this._toneMappingType = Ui.TONEMAPPING_STANDARD, this._contrast = 1, this.vignetteStretch = 0, this.vignetteCenterX = 0, this.vignetteCenterY = 0, this.vignetteWeight = 1.5, this.vignetteColor = new xt(0, 0, 0, 0), this.vignetteCameraFov = 0.5, this._vignetteBlendMode = Ui.VIGNETTEMODE_MULTIPLY, this._vignetteEnabled = !1, this._ditheringEnabled = !1, this._ditheringIntensity = 1 / 255, this._skipFinalColorClamp = !1, this._applyByPostProcess = !1, this._isEnabled = !0, this.onUpdateParameters = new Oe(); } /** * Gets whether the color curves effect is enabled. */ get colorCurvesEnabled() { return this._colorCurvesEnabled; } /** * Sets whether the color curves effect is enabled. */ set colorCurvesEnabled(e) { this._colorCurvesEnabled !== e && (this._colorCurvesEnabled = e, this._updateParameters()); } /** * Color grading LUT texture used in the effect if colorGradingEnabled is set to true */ get colorGradingTexture() { return this._colorGradingTexture; } /** * Color grading LUT texture used in the effect if colorGradingEnabled is set to true */ set colorGradingTexture(e) { this._colorGradingTexture !== e && (this._colorGradingTexture = e, this._updateParameters()); } /** * Gets whether the color grading effect is enabled. */ get colorGradingEnabled() { return this._colorGradingEnabled; } /** * Sets whether the color grading effect is enabled. */ set colorGradingEnabled(e) { this._colorGradingEnabled !== e && (this._colorGradingEnabled = e, this._updateParameters()); } /** * Gets whether the color grading effect is using a green depth for the 3d Texture. */ get colorGradingWithGreenDepth() { return this._colorGradingWithGreenDepth; } /** * Sets whether the color grading effect is using a green depth for the 3d Texture. */ set colorGradingWithGreenDepth(e) { this._colorGradingWithGreenDepth !== e && (this._colorGradingWithGreenDepth = e, this._updateParameters()); } /** * Gets whether the color grading texture contains BGR values. */ get colorGradingBGR() { return this._colorGradingBGR; } /** * Sets whether the color grading texture contains BGR values. */ set colorGradingBGR(e) { this._colorGradingBGR !== e && (this._colorGradingBGR = e, this._updateParameters()); } /** * Gets the Exposure used in the effect. */ get exposure() { return this._exposure; } /** * Sets the Exposure used in the effect. */ set exposure(e) { this._exposure !== e && (this._exposure = e, this._updateParameters()); } /** * Gets whether the tone mapping effect is enabled. */ get toneMappingEnabled() { return this._toneMappingEnabled; } /** * Sets whether the tone mapping effect is enabled. */ set toneMappingEnabled(e) { this._toneMappingEnabled !== e && (this._toneMappingEnabled = e, this._updateParameters()); } /** * Gets the type of tone mapping effect. */ get toneMappingType() { return this._toneMappingType; } /** * Sets the type of tone mapping effect used in BabylonJS. */ set toneMappingType(e) { this._toneMappingType !== e && (this._toneMappingType = e, this._updateParameters()); } /** * Gets the contrast used in the effect. */ get contrast() { return this._contrast; } /** * Sets the contrast used in the effect. */ set contrast(e) { this._contrast !== e && (this._contrast = e, this._updateParameters()); } /** * Back Compat: Vignette center Y Offset. * @deprecated use vignetteCenterY instead */ get vignetteCentreY() { return this.vignetteCenterY; } set vignetteCentreY(e) { this.vignetteCenterY = e; } /** * Back Compat: Vignette center X Offset. * @deprecated use vignetteCenterX instead */ get vignetteCentreX() { return this.vignetteCenterX; } set vignetteCentreX(e) { this.vignetteCenterX = e; } /** * Gets the vignette blend mode allowing different kind of effect. */ get vignetteBlendMode() { return this._vignetteBlendMode; } /** * Sets the vignette blend mode allowing different kind of effect. */ set vignetteBlendMode(e) { this._vignetteBlendMode !== e && (this._vignetteBlendMode = e, this._updateParameters()); } /** * Gets whether the vignette effect is enabled. */ get vignetteEnabled() { return this._vignetteEnabled; } /** * Sets whether the vignette effect is enabled. */ set vignetteEnabled(e) { this._vignetteEnabled !== e && (this._vignetteEnabled = e, this._updateParameters()); } /** * Gets whether the dithering effect is enabled. * The dithering effect can be used to reduce banding. */ get ditheringEnabled() { return this._ditheringEnabled; } /** * Sets whether the dithering effect is enabled. * The dithering effect can be used to reduce banding. */ set ditheringEnabled(e) { this._ditheringEnabled !== e && (this._ditheringEnabled = e, this._updateParameters()); } /** * Gets the dithering intensity. 0 is no dithering. Default is 1.0 / 255.0. */ get ditheringIntensity() { return this._ditheringIntensity; } /** * Sets the dithering intensity. 0 is no dithering. Default is 1.0 / 255.0. */ set ditheringIntensity(e) { this._ditheringIntensity !== e && (this._ditheringIntensity = e, this._updateParameters()); } /** * If apply by post process is set to true, setting this to true will skip the final color clamp step in the fragment shader * Applies to PBR materials. */ get skipFinalColorClamp() { return this._skipFinalColorClamp; } /** * If apply by post process is set to true, setting this to true will skip the final color clamp step in the fragment shader * Applies to PBR materials. */ set skipFinalColorClamp(e) { this._skipFinalColorClamp !== e && (this._skipFinalColorClamp = e, this._updateParameters()); } /** * Gets whether the image processing is applied through a post process or not. */ get applyByPostProcess() { return this._applyByPostProcess; } /** * Sets whether the image processing is applied through a post process or not. */ set applyByPostProcess(e) { this._applyByPostProcess !== e && (this._applyByPostProcess = e, this._updateParameters()); } /** * Gets whether the image processing is enabled or not. */ get isEnabled() { return this._isEnabled; } /** * Sets whether the image processing is enabled or not. */ set isEnabled(e) { this._isEnabled !== e && (this._isEnabled = e, this._updateParameters()); } /** * Method called each time the image processing information changes requires to recompile the effect. */ _updateParameters() { this.onUpdateParameters.notifyObservers(this); } /** * Gets the current class name. * @returns "ImageProcessingConfiguration" */ getClassName() { return "ImageProcessingConfiguration"; } /** * Prepare the list of uniforms associated with the Image Processing effects. * @param uniforms The list of uniforms used in the effect * @param defines the list of defines currently in use */ static PrepareUniforms(e, t) { t.EXPOSURE && e.push("exposureLinear"), t.CONTRAST && e.push("contrast"), t.COLORGRADING && e.push("colorTransformSettings"), (t.VIGNETTE || t.DITHER) && e.push("vInverseScreenSize"), t.VIGNETTE && (e.push("vignetteSettings1"), e.push("vignetteSettings2")), t.COLORCURVES && so.PrepareUniforms(e), t.DITHER && e.push("ditherIntensity"); } /** * Prepare the list of samplers associated with the Image Processing effects. * @param samplersList The list of uniforms used in the effect * @param defines the list of defines currently in use */ static PrepareSamplers(e, t) { t.COLORGRADING && e.push("txColorTransform"); } /** * Prepare the list of defines associated to the shader. * @param defines the list of defines to complete * @param forPostProcess Define if we are currently in post process mode or not */ prepareDefines(e, t = !1) { if (t !== this.applyByPostProcess || !this._isEnabled) { e.VIGNETTE = !1, e.TONEMAPPING = !1, e.TONEMAPPING_ACES = !1, e.CONTRAST = !1, e.EXPOSURE = !1, e.COLORCURVES = !1, e.COLORGRADING = !1, e.COLORGRADING3D = !1, e.DITHER = !1, e.IMAGEPROCESSING = !1, e.SKIPFINALCOLORCLAMP = this.skipFinalColorClamp, e.IMAGEPROCESSINGPOSTPROCESS = this.applyByPostProcess && this._isEnabled; return; } switch (e.VIGNETTE = this.vignetteEnabled, e.VIGNETTEBLENDMODEMULTIPLY = this.vignetteBlendMode === Ui._VIGNETTEMODE_MULTIPLY, e.VIGNETTEBLENDMODEOPAQUE = !e.VIGNETTEBLENDMODEMULTIPLY, e.TONEMAPPING = this.toneMappingEnabled, this._toneMappingType) { case Ui.TONEMAPPING_ACES: e.TONEMAPPING_ACES = !0; break; default: e.TONEMAPPING_ACES = !1; break; } e.CONTRAST = this.contrast !== 1, e.EXPOSURE = this.exposure !== 1, e.COLORCURVES = this.colorCurvesEnabled && !!this.colorCurves, e.COLORGRADING = this.colorGradingEnabled && !!this.colorGradingTexture, e.COLORGRADING ? e.COLORGRADING3D = this.colorGradingTexture.is3D : e.COLORGRADING3D = !1, e.SAMPLER3DGREENDEPTH = this.colorGradingWithGreenDepth, e.SAMPLER3DBGRMAP = this.colorGradingBGR, e.DITHER = this._ditheringEnabled, e.IMAGEPROCESSINGPOSTPROCESS = this.applyByPostProcess, e.SKIPFINALCOLORCLAMP = this.skipFinalColorClamp, e.IMAGEPROCESSING = e.VIGNETTE || e.TONEMAPPING || e.CONTRAST || e.EXPOSURE || e.COLORCURVES || e.COLORGRADING || e.DITHER; } /** * Returns true if all the image processing information are ready. * @returns True if ready, otherwise, false */ isReady() { return !this.colorGradingEnabled || !this.colorGradingTexture || this.colorGradingTexture.isReady(); } /** * Binds the image processing to the shader. * @param effect The effect to bind to * @param overrideAspectRatio Override the aspect ratio of the effect */ bind(e, t) { if (this._colorCurvesEnabled && this.colorCurves && so.Bind(this.colorCurves, e), this._vignetteEnabled || this._ditheringEnabled) { const r = 1 / e.getEngine().getRenderWidth(), n = 1 / e.getEngine().getRenderHeight(); if (e.setFloat2("vInverseScreenSize", r, n), this._ditheringEnabled && e.setFloat("ditherIntensity", 0.5 * this._ditheringIntensity), this._vignetteEnabled) { const i = t ?? n / r; let s = Math.tan(this.vignetteCameraFov * 0.5), a = s * i; const f = Math.sqrt(a * s); a = ye.Mix(a, f, this.vignetteStretch), s = ye.Mix(s, f, this.vignetteStretch), e.setFloat4("vignetteSettings1", a, s, -a * this.vignetteCenterX, -s * this.vignetteCenterY); const o = -2 * this.vignetteWeight; e.setFloat4("vignetteSettings2", this.vignetteColor.r, this.vignetteColor.g, this.vignetteColor.b, o); } } if (e.setFloat("exposureLinear", this.exposure), e.setFloat("contrast", this.contrast), this.colorGradingTexture) { e.setTexture("txColorTransform", this.colorGradingTexture); const r = this.colorGradingTexture.getSize().height; e.setFloat4( "colorTransformSettings", (r - 1) / r, // textureScale 0.5 / r, // textureOffset r, // textureSize this.colorGradingTexture.level // weight ); } } /** * Clones the current image processing instance. * @returns The cloned image processing */ clone() { return jt.Clone(() => new Ui(), this); } /** * Serializes the current image processing instance to a json representation. * @returns a JSON representation */ serialize() { return jt.Serialize(this); } /** * Parses the image processing from a json representation. * @param source the JSON source to parse * @returns The parsed image processing */ static Parse(e) { const t = jt.Parse(() => new Ui(), e, null, null); return e.vignetteCentreX !== void 0 && (t.vignetteCenterX = e.vignetteCentreX), e.vignetteCentreY !== void 0 && (t.vignetteCenterY = e.vignetteCentreY), t; } /** * Used to apply the vignette as a mix with the pixel color. */ static get VIGNETTEMODE_MULTIPLY() { return this._VIGNETTEMODE_MULTIPLY; } /** * Used to apply the vignette as a replacement of the pixel color. */ static get VIGNETTEMODE_OPAQUE() { return this._VIGNETTEMODE_OPAQUE; } } Ui.TONEMAPPING_STANDARD = 0; Ui.TONEMAPPING_ACES = 1; Ui._VIGNETTEMODE_MULTIPLY = 0; Ui._VIGNETTEMODE_OPAQUE = 1; C([ W$() ], Ui.prototype, "colorCurves", void 0); C([ M() ], Ui.prototype, "_colorCurvesEnabled", void 0); C([ en("colorGradingTexture") ], Ui.prototype, "_colorGradingTexture", void 0); C([ M() ], Ui.prototype, "_colorGradingEnabled", void 0); C([ M() ], Ui.prototype, "_colorGradingWithGreenDepth", void 0); C([ M() ], Ui.prototype, "_colorGradingBGR", void 0); C([ M() ], Ui.prototype, "_exposure", void 0); C([ M() ], Ui.prototype, "_toneMappingEnabled", void 0); C([ M() ], Ui.prototype, "_toneMappingType", void 0); C([ M() ], Ui.prototype, "_contrast", void 0); C([ M() ], Ui.prototype, "vignetteStretch", void 0); C([ M() ], Ui.prototype, "vignetteCenterX", void 0); C([ M() ], Ui.prototype, "vignetteCenterY", void 0); C([ M() ], Ui.prototype, "vignetteWeight", void 0); C([ rU() ], Ui.prototype, "vignetteColor", void 0); C([ M() ], Ui.prototype, "vignetteCameraFov", void 0); C([ M() ], Ui.prototype, "_vignetteBlendMode", void 0); C([ M() ], Ui.prototype, "_vignetteEnabled", void 0); C([ M() ], Ui.prototype, "_ditheringEnabled", void 0); C([ M() ], Ui.prototype, "_ditheringIntensity", void 0); C([ M() ], Ui.prototype, "_skipFinalColorClamp", void 0); C([ M() ], Ui.prototype, "_applyByPostProcess", void 0); C([ M() ], Ui.prototype, "_isEnabled", void 0); jt._ImageProcessingConfigurationParser = Ui.Parse; hr.prototype.createUniformBuffer = function(A, e) { const t = this._gl.createBuffer(); if (!t) throw new Error("Unable to create uniform buffer"); const r = new jS(t); return this.bindUniformBuffer(r), A instanceof Float32Array ? this._gl.bufferData(this._gl.UNIFORM_BUFFER, A, this._gl.STATIC_DRAW) : this._gl.bufferData(this._gl.UNIFORM_BUFFER, new Float32Array(A), this._gl.STATIC_DRAW), this.bindUniformBuffer(null), r.references = 1, r; }; hr.prototype.createDynamicUniformBuffer = function(A, e) { const t = this._gl.createBuffer(); if (!t) throw new Error("Unable to create dynamic uniform buffer"); const r = new jS(t); return this.bindUniformBuffer(r), A instanceof Float32Array ? this._gl.bufferData(this._gl.UNIFORM_BUFFER, A, this._gl.DYNAMIC_DRAW) : this._gl.bufferData(this._gl.UNIFORM_BUFFER, new Float32Array(A), this._gl.DYNAMIC_DRAW), this.bindUniformBuffer(null), r.references = 1, r; }; hr.prototype.updateUniformBuffer = function(A, e, t, r) { this.bindUniformBuffer(A), t === void 0 && (t = 0), r === void 0 ? e instanceof Float32Array ? this._gl.bufferSubData(this._gl.UNIFORM_BUFFER, t, e) : this._gl.bufferSubData(this._gl.UNIFORM_BUFFER, t, new Float32Array(e)) : e instanceof Float32Array ? this._gl.bufferSubData(this._gl.UNIFORM_BUFFER, 0, e.subarray(t, t + r)) : this._gl.bufferSubData(this._gl.UNIFORM_BUFFER, 0, new Float32Array(e).subarray(t, t + r)), this.bindUniformBuffer(null); }; hr.prototype.bindUniformBuffer = function(A) { this._gl.bindBuffer(this._gl.UNIFORM_BUFFER, A ? A.underlyingResource : null); }; hr.prototype.bindUniformBufferBase = function(A, e, t) { this._gl.bindBufferBase(this._gl.UNIFORM_BUFFER, e, A ? A.underlyingResource : null); }; hr.prototype.bindUniformBlock = function(A, e, t) { const r = A.program, n = this._gl.getUniformBlockIndex(r, e); n !== 4294967295 && this._gl.uniformBlockBinding(r, n, t); }; class yr { /** * Instantiates a new Uniform buffer objects. * * Handles blocks of uniform on the GPU. * * If WebGL 2 is not available, this class falls back on traditional setUniformXXX calls. * * For more information, please refer to : * @see https://www.khronos.org/opengl/wiki/Uniform_Buffer_Object * @param engine Define the engine the buffer is associated with * @param data Define the data contained in the buffer * @param dynamic Define if the buffer is updatable * @param name to assign to the buffer (debugging purpose) * @param forceNoUniformBuffer define that this object must not rely on UBO objects */ constructor(e, t, r, n, i = !1) { this._valueCache = {}, this._engine = e, this._noUBO = !e.supportsUniformBuffers || i, this._dynamic = r, this._name = n ?? "no-name", this._data = t || [], this._uniformLocations = {}, this._uniformSizes = {}, this._uniformArraySizes = {}, this._uniformLocationPointer = 0, this._needSync = !1, this._engine._features.trackUbosInFrame && (this._buffers = [], this._bufferIndex = -1, this._createBufferOnWrite = !1, this._currentFrameId = 0), this._noUBO ? (this.updateMatrix3x3 = this._updateMatrix3x3ForEffect, this.updateMatrix2x2 = this._updateMatrix2x2ForEffect, this.updateFloat = this._updateFloatForEffect, this.updateFloat2 = this._updateFloat2ForEffect, this.updateFloat3 = this._updateFloat3ForEffect, this.updateFloat4 = this._updateFloat4ForEffect, this.updateFloatArray = this._updateFloatArrayForEffect, this.updateArray = this._updateArrayForEffect, this.updateIntArray = this._updateIntArrayForEffect, this.updateUIntArray = this._updateUIntArrayForEffect, this.updateMatrix = this._updateMatrixForEffect, this.updateMatrices = this._updateMatricesForEffect, this.updateVector3 = this._updateVector3ForEffect, this.updateVector4 = this._updateVector4ForEffect, this.updateColor3 = this._updateColor3ForEffect, this.updateColor4 = this._updateColor4ForEffect, this.updateDirectColor4 = this._updateDirectColor4ForEffect, this.updateInt = this._updateIntForEffect, this.updateInt2 = this._updateInt2ForEffect, this.updateInt3 = this._updateInt3ForEffect, this.updateInt4 = this._updateInt4ForEffect, this.updateUInt = this._updateUIntForEffect, this.updateUInt2 = this._updateUInt2ForEffect, this.updateUInt3 = this._updateUInt3ForEffect, this.updateUInt4 = this._updateUInt4ForEffect) : (this._engine._uniformBuffers.push(this), this.updateMatrix3x3 = this._updateMatrix3x3ForUniform, this.updateMatrix2x2 = this._updateMatrix2x2ForUniform, this.updateFloat = this._updateFloatForUniform, this.updateFloat2 = this._updateFloat2ForUniform, this.updateFloat3 = this._updateFloat3ForUniform, this.updateFloat4 = this._updateFloat4ForUniform, this.updateFloatArray = this._updateFloatArrayForUniform, this.updateArray = this._updateArrayForUniform, this.updateIntArray = this._updateIntArrayForUniform, this.updateUIntArray = this._updateUIntArrayForUniform, this.updateMatrix = this._updateMatrixForUniform, this.updateMatrices = this._updateMatricesForUniform, this.updateVector3 = this._updateVector3ForUniform, this.updateVector4 = this._updateVector4ForUniform, this.updateColor3 = this._updateColor3ForUniform, this.updateColor4 = this._updateColor4ForUniform, this.updateDirectColor4 = this._updateDirectColor4ForUniform, this.updateInt = this._updateIntForUniform, this.updateInt2 = this._updateInt2ForUniform, this.updateInt3 = this._updateInt3ForUniform, this.updateInt4 = this._updateInt4ForUniform, this.updateUInt = this._updateUIntForUniform, this.updateUInt2 = this._updateUInt2ForUniform, this.updateUInt3 = this._updateUInt3ForUniform, this.updateUInt4 = this._updateUInt4ForUniform); } /** * Indicates if the buffer is using the WebGL2 UBO implementation, * or just falling back on setUniformXXX calls. */ get useUbo() { return !this._noUBO; } /** * Indicates if the WebGL underlying uniform buffer is in sync * with the javascript cache data. */ get isSync() { return !this._needSync; } /** * Indicates if the WebGL underlying uniform buffer is dynamic. * Also, a dynamic UniformBuffer will disable cache verification and always * update the underlying WebGL uniform buffer to the GPU. * @returns if Dynamic, otherwise false */ isDynamic() { return this._dynamic !== void 0; } /** * The data cache on JS side. * @returns the underlying data as a float array */ getData() { return this._bufferData; } /** * The underlying WebGL Uniform buffer. * @returns the webgl buffer */ getBuffer() { return this._buffer; } /** * std140 layout specifies how to align data within an UBO structure. * See https://khronos.org/registry/OpenGL/specs/gl/glspec45.core.pdf#page=159 * for specs. * @param size */ _fillAlignment(e) { let t; if (e <= 2 ? t = e : t = 4, this._uniformLocationPointer % t !== 0) { const r = this._uniformLocationPointer; this._uniformLocationPointer += t - this._uniformLocationPointer % t; const n = this._uniformLocationPointer - r; for (let i = 0; i < n; i++) this._data.push(0); } } /** * Adds an uniform in the buffer. * Warning : the subsequents calls of this function must be in the same order as declared in the shader * for the layout to be correct ! The addUniform function only handles types like float, vec2, vec3, vec4, mat4, * meaning size=1,2,3,4 or 16. It does not handle struct types. * @param name Name of the uniform, as used in the uniform block in the shader. * @param size Data size, or data directly. * @param arraySize The number of elements in the array, 0 if not an array. */ addUniform(e, t, r = 0) { if (this._noUBO || this._uniformLocations[e] !== void 0) return; let n; if (r > 0) { if (t instanceof Array) throw "addUniform should not be use with Array in UBO: " + e; if (this._fillAlignment(4), this._uniformArraySizes[e] = { strideSize: t, arraySize: r }, t == 16) t = t * r; else { const s = (4 - t) * r; t = t * r + s; } n = []; for (let i = 0; i < t; i++) n.push(0); } else { if (t instanceof Array) n = t, t = n.length; else { t = t, n = []; for (let i = 0; i < t; i++) n.push(0); } this._fillAlignment(t); } this._uniformSizes[e] = t, this._uniformLocations[e] = this._uniformLocationPointer, this._uniformLocationPointer += t; for (let i = 0; i < t; i++) this._data.push(n[i]); this._needSync = !0; } /** * Adds a Matrix 4x4 to the uniform buffer. * @param name Name of the uniform, as used in the uniform block in the shader. * @param mat A 4x4 matrix. */ addMatrix(e, t) { this.addUniform(e, Array.prototype.slice.call(t.toArray())); } /** * Adds a vec2 to the uniform buffer. * @param name Name of the uniform, as used in the uniform block in the shader. * @param x Define the x component value of the vec2 * @param y Define the y component value of the vec2 */ addFloat2(e, t, r) { const n = [t, r]; this.addUniform(e, n); } /** * Adds a vec3 to the uniform buffer. * @param name Name of the uniform, as used in the uniform block in the shader. * @param x Define the x component value of the vec3 * @param y Define the y component value of the vec3 * @param z Define the z component value of the vec3 */ addFloat3(e, t, r, n) { const i = [t, r, n]; this.addUniform(e, i); } /** * Adds a vec3 to the uniform buffer. * @param name Name of the uniform, as used in the uniform block in the shader. * @param color Define the vec3 from a Color */ addColor3(e, t) { const r = [t.r, t.g, t.b]; this.addUniform(e, r); } /** * Adds a vec4 to the uniform buffer. * @param name Name of the uniform, as used in the uniform block in the shader. * @param color Define the rgb components from a Color * @param alpha Define the a component of the vec4 */ addColor4(e, t, r) { const n = [t.r, t.g, t.b, r]; this.addUniform(e, n); } /** * Adds a vec3 to the uniform buffer. * @param name Name of the uniform, as used in the uniform block in the shader. * @param vector Define the vec3 components from a Vector */ addVector3(e, t) { const r = [t.x, t.y, t.z]; this.addUniform(e, r); } /** * Adds a Matrix 3x3 to the uniform buffer. * @param name Name of the uniform, as used in the uniform block in the shader. */ addMatrix3x3(e) { this.addUniform(e, 12); } /** * Adds a Matrix 2x2 to the uniform buffer. * @param name Name of the uniform, as used in the uniform block in the shader. */ addMatrix2x2(e) { this.addUniform(e, 8); } /** * Effectively creates the WebGL Uniform Buffer, once layout is completed with `addUniform`. */ create() { this._noUBO || this._buffer || (this._fillAlignment(4), this._bufferData = new Float32Array(this._data), this._rebuild(), this._needSync = !0); } _getNames() { const e = []; for (const t in this._uniformLocations) e.push(t); return e.join(","); } /** @internal */ _rebuild() { this._noUBO || !this._bufferData || (this._dynamic ? this._buffer = this._engine.createDynamicUniformBuffer(this._bufferData, this._name + "_UniformList:" + this._getNames()) : this._buffer = this._engine.createUniformBuffer(this._bufferData, this._name + "_UniformList:" + this._getNames()), this._engine._features.trackUbosInFrame && (this._buffers.push([this._buffer, this._engine._features.checkUbosContentBeforeUpload ? this._bufferData.slice() : void 0]), this._bufferIndex = this._buffers.length - 1, this._createBufferOnWrite = !1)); } /** @internal */ get _numBuffers() { return this._buffers.length; } /** @internal */ get _indexBuffer() { return this._bufferIndex; } /** Gets the name of this buffer */ get name() { return this._name; } /** Gets the current effect */ get currentEffect() { return this._currentEffect; } _buffersEqual(e, t) { for (let r = 0; r < e.length; ++r) if (e[r] !== t[r]) return !1; return !0; } _copyBuffer(e, t) { for (let r = 0; r < e.length; ++r) t[r] = e[r]; } /** * Updates the WebGL Uniform Buffer on the GPU. * If the `dynamic` flag is set to true, no cache comparison is done. * Otherwise, the buffer will be updated only if the cache differs. */ update() { if (!this._noUBO) { if (this.bindUniformBuffer(), !this._buffer) { this.create(); return; } if (!this._dynamic && !this._needSync) { this._createBufferOnWrite = this._engine._features.trackUbosInFrame; return; } if (this._buffers && this._buffers.length > 1 && this._buffers[this._bufferIndex][1]) if (this._buffersEqual(this._bufferData, this._buffers[this._bufferIndex][1])) { this._needSync = !1, this._createBufferOnWrite = this._engine._features.trackUbosInFrame; return; } else this._copyBuffer(this._bufferData, this._buffers[this._bufferIndex][1]); this._engine.updateUniformBuffer(this._buffer, this._bufferData), this._engine._features._collectUbosUpdatedInFrame && (yr._UpdatedUbosInFrame[this._name] || (yr._UpdatedUbosInFrame[this._name] = 0), yr._UpdatedUbosInFrame[this._name]++), this._needSync = !1, this._createBufferOnWrite = this._engine._features.trackUbosInFrame; } } _createNewBuffer() { this._bufferIndex + 1 < this._buffers.length ? (this._bufferIndex++, this._buffer = this._buffers[this._bufferIndex][0], this._createBufferOnWrite = !1, this._needSync = !0) : this._rebuild(); } _checkNewFrame() { this._engine._features.trackUbosInFrame && this._currentFrameId !== this._engine.frameId && (this._currentFrameId = this._engine.frameId, this._createBufferOnWrite = !1, this._buffers && this._buffers.length > 0 ? (this._needSync = this._bufferIndex !== 0, this._bufferIndex = 0, this._buffer = this._buffers[this._bufferIndex][0]) : this._bufferIndex = -1); } /** * Updates the value of an uniform. The `update` method must be called afterwards to make it effective in the GPU. * @param uniformName Define the name of the uniform, as used in the uniform block in the shader. * @param data Define the flattened data * @param size Define the size of the data. */ updateUniform(e, t, r) { this._checkNewFrame(); let n = this._uniformLocations[e]; if (n === void 0) { if (this._buffer) { Se.Error("Cannot add an uniform after UBO has been created."); return; } this.addUniform(e, r), n = this._uniformLocations[e]; } if (this._buffer || this.create(), this._dynamic) for (let i = 0; i < r; i++) this._bufferData[n + i] = t[i]; else { let i = !1; for (let s = 0; s < r; s++) (r === 16 && !this._engine._features.uniformBufferHardCheckMatrix || this._bufferData[n + s] !== Math.fround(t[s])) && (i = !0, this._createBufferOnWrite && this._createNewBuffer(), this._bufferData[n + s] = t[s]); this._needSync = this._needSync || i; } } /** * Updates the value of an uniform. The `update` method must be called afterwards to make it effective in the GPU. * @param uniformName Define the name of the uniform, as used in the uniform block in the shader. * @param data Define the flattened data * @param size Define the size of the data. */ updateUniformArray(e, t, r) { this._checkNewFrame(); const n = this._uniformLocations[e]; if (n === void 0) { Se.Error("Cannot add an uniform Array dynamically. Please, add it using addUniform and make sure that uniform buffers are supported by the current engine."); return; } this._buffer || this.create(); const i = this._uniformArraySizes[e]; if (this._dynamic) for (let s = 0; s < r; s++) this._bufferData[n + s] = t[s]; else { let s = !1, a = 0, f = 0; for (let o = 0; o < r; o++) if (this._bufferData[n + f * 4 + a] !== ye.FloatRound(t[o]) && (s = !0, this._createBufferOnWrite && this._createNewBuffer(), this._bufferData[n + f * 4 + a] = t[o]), a++, a === i.strideSize) { for (; a < 4; a++) this._bufferData[n + f * 4 + a] = 0; a = 0, f++; } this._needSync = this._needSync || s; } } _cacheMatrix(e, t) { this._checkNewFrame(); const r = this._valueCache[e], n = t.updateFlag; return r !== void 0 && r === n ? !1 : (this._valueCache[e] = n, !0); } // Update methods _updateMatrix3x3ForUniform(e, t) { for (let r = 0; r < 3; r++) yr._TempBuffer[r * 4] = t[r * 3], yr._TempBuffer[r * 4 + 1] = t[r * 3 + 1], yr._TempBuffer[r * 4 + 2] = t[r * 3 + 2], yr._TempBuffer[r * 4 + 3] = 0; this.updateUniform(e, yr._TempBuffer, 12); } _updateMatrix3x3ForEffect(e, t) { this._currentEffect.setMatrix3x3(e, t); } _updateMatrix2x2ForEffect(e, t) { this._currentEffect.setMatrix2x2(e, t); } _updateMatrix2x2ForUniform(e, t) { for (let r = 0; r < 2; r++) yr._TempBuffer[r * 4] = t[r * 2], yr._TempBuffer[r * 4 + 1] = t[r * 2 + 1], yr._TempBuffer[r * 4 + 2] = 0, yr._TempBuffer[r * 4 + 3] = 0; this.updateUniform(e, yr._TempBuffer, 8); } _updateFloatForEffect(e, t) { this._currentEffect.setFloat(e, t); } _updateFloatForUniform(e, t) { yr._TempBuffer[0] = t, this.updateUniform(e, yr._TempBuffer, 1); } _updateFloat2ForEffect(e, t, r, n = "") { this._currentEffect.setFloat2(e + n, t, r); } _updateFloat2ForUniform(e, t, r) { yr._TempBuffer[0] = t, yr._TempBuffer[1] = r, this.updateUniform(e, yr._TempBuffer, 2); } _updateFloat3ForEffect(e, t, r, n, i = "") { this._currentEffect.setFloat3(e + i, t, r, n); } _updateFloat3ForUniform(e, t, r, n) { yr._TempBuffer[0] = t, yr._TempBuffer[1] = r, yr._TempBuffer[2] = n, this.updateUniform(e, yr._TempBuffer, 3); } _updateFloat4ForEffect(e, t, r, n, i, s = "") { this._currentEffect.setFloat4(e + s, t, r, n, i); } _updateFloat4ForUniform(e, t, r, n, i) { yr._TempBuffer[0] = t, yr._TempBuffer[1] = r, yr._TempBuffer[2] = n, yr._TempBuffer[3] = i, this.updateUniform(e, yr._TempBuffer, 4); } _updateFloatArrayForEffect(e, t) { this._currentEffect.setFloatArray(e, t); } _updateFloatArrayForUniform(e, t) { this.updateUniformArray(e, t, t.length); } _updateArrayForEffect(e, t) { this._currentEffect.setArray(e, t); } _updateArrayForUniform(e, t) { this.updateUniformArray(e, t, t.length); } _updateIntArrayForEffect(e, t) { this._currentEffect.setIntArray(e, t); } _updateIntArrayForUniform(e, t) { yr._TempBufferInt32View.set(t), this.updateUniformArray(e, yr._TempBuffer, t.length); } _updateUIntArrayForEffect(e, t) { this._currentEffect.setUIntArray(e, t); } _updateUIntArrayForUniform(e, t) { yr._TempBufferUInt32View.set(t), this.updateUniformArray(e, yr._TempBuffer, t.length); } _updateMatrixForEffect(e, t) { this._currentEffect.setMatrix(e, t); } _updateMatrixForUniform(e, t) { this._cacheMatrix(e, t) && this.updateUniform(e, t.toArray(), 16); } _updateMatricesForEffect(e, t) { this._currentEffect.setMatrices(e, t); } _updateMatricesForUniform(e, t) { this.updateUniform(e, t, t.length); } _updateVector3ForEffect(e, t) { this._currentEffect.setVector3(e, t); } _updateVector3ForUniform(e, t) { yr._TempBuffer[0] = t.x, yr._TempBuffer[1] = t.y, yr._TempBuffer[2] = t.z, this.updateUniform(e, yr._TempBuffer, 3); } _updateVector4ForEffect(e, t) { this._currentEffect.setVector4(e, t); } _updateVector4ForUniform(e, t) { yr._TempBuffer[0] = t.x, yr._TempBuffer[1] = t.y, yr._TempBuffer[2] = t.z, yr._TempBuffer[3] = t.w, this.updateUniform(e, yr._TempBuffer, 4); } _updateColor3ForEffect(e, t, r = "") { this._currentEffect.setColor3(e + r, t); } _updateColor3ForUniform(e, t) { yr._TempBuffer[0] = t.r, yr._TempBuffer[1] = t.g, yr._TempBuffer[2] = t.b, this.updateUniform(e, yr._TempBuffer, 3); } _updateColor4ForEffect(e, t, r, n = "") { this._currentEffect.setColor4(e + n, t, r); } _updateDirectColor4ForEffect(e, t, r = "") { this._currentEffect.setDirectColor4(e + r, t); } _updateColor4ForUniform(e, t, r) { yr._TempBuffer[0] = t.r, yr._TempBuffer[1] = t.g, yr._TempBuffer[2] = t.b, yr._TempBuffer[3] = r, this.updateUniform(e, yr._TempBuffer, 4); } _updateDirectColor4ForUniform(e, t) { yr._TempBuffer[0] = t.r, yr._TempBuffer[1] = t.g, yr._TempBuffer[2] = t.b, yr._TempBuffer[3] = t.a, this.updateUniform(e, yr._TempBuffer, 4); } _updateIntForEffect(e, t, r = "") { this._currentEffect.setInt(e + r, t); } _updateIntForUniform(e, t) { yr._TempBufferInt32View[0] = t, this.updateUniform(e, yr._TempBuffer, 1); } _updateInt2ForEffect(e, t, r, n = "") { this._currentEffect.setInt2(e + n, t, r); } _updateInt2ForUniform(e, t, r) { yr._TempBufferInt32View[0] = t, yr._TempBufferInt32View[1] = r, this.updateUniform(e, yr._TempBuffer, 2); } _updateInt3ForEffect(e, t, r, n, i = "") { this._currentEffect.setInt3(e + i, t, r, n); } _updateInt3ForUniform(e, t, r, n) { yr._TempBufferInt32View[0] = t, yr._TempBufferInt32View[1] = r, yr._TempBufferInt32View[2] = n, this.updateUniform(e, yr._TempBuffer, 3); } _updateInt4ForEffect(e, t, r, n, i, s = "") { this._currentEffect.setInt4(e + s, t, r, n, i); } _updateInt4ForUniform(e, t, r, n, i) { yr._TempBufferInt32View[0] = t, yr._TempBufferInt32View[1] = r, yr._TempBufferInt32View[2] = n, yr._TempBufferInt32View[3] = i, this.updateUniform(e, yr._TempBuffer, 4); } _updateUIntForEffect(e, t, r = "") { this._currentEffect.setUInt(e + r, t); } _updateUIntForUniform(e, t) { yr._TempBufferUInt32View[0] = t, this.updateUniform(e, yr._TempBuffer, 1); } _updateUInt2ForEffect(e, t, r, n = "") { this._currentEffect.setUInt2(e + n, t, r); } _updateUInt2ForUniform(e, t, r) { yr._TempBufferUInt32View[0] = t, yr._TempBufferUInt32View[1] = r, this.updateUniform(e, yr._TempBuffer, 2); } _updateUInt3ForEffect(e, t, r, n, i = "") { this._currentEffect.setUInt3(e + i, t, r, n); } _updateUInt3ForUniform(e, t, r, n) { yr._TempBufferUInt32View[0] = t, yr._TempBufferUInt32View[1] = r, yr._TempBufferUInt32View[2] = n, this.updateUniform(e, yr._TempBuffer, 3); } _updateUInt4ForEffect(e, t, r, n, i, s = "") { this._currentEffect.setUInt4(e + s, t, r, n, i); } _updateUInt4ForUniform(e, t, r, n, i) { yr._TempBufferUInt32View[0] = t, yr._TempBufferUInt32View[1] = r, yr._TempBufferUInt32View[2] = n, yr._TempBufferUInt32View[3] = i, this.updateUniform(e, yr._TempBuffer, 4); } /** * Sets a sampler uniform on the effect. * @param name Define the name of the sampler. * @param texture Define the texture to set in the sampler */ setTexture(e, t) { this._currentEffect.setTexture(e, t); } /** * Sets a sampler uniform on the effect. * @param name Define the name of the sampler. * @param texture Define the (internal) texture to set in the sampler */ bindTexture(e, t) { this._currentEffect._bindTexture(e, t); } /** * Directly updates the value of the uniform in the cache AND on the GPU. * @param uniformName Define the name of the uniform, as used in the uniform block in the shader. * @param data Define the flattened data */ updateUniformDirectly(e, t) { this.updateUniform(e, t, t.length), this.update(); } /** * Associates an effect to this uniform buffer * @param effect Define the effect to associate the buffer to * @param name Name of the uniform block in the shader. */ bindToEffect(e, t) { this._currentEffect = e, this._currentEffectName = t; } /** * Binds the current (GPU) buffer to the effect */ bindUniformBuffer() { !this._noUBO && this._buffer && this._currentEffect && this._currentEffect.bindUniformBuffer(this._buffer, this._currentEffectName); } /** * Dissociates the current effect from this uniform buffer */ unbindEffect() { this._currentEffect = void 0, this._currentEffectName = void 0; } /** * Sets the current state of the class (_bufferIndex, _buffer) to point to the data buffer passed in parameter if this buffer is one of the buffers handled by the class (meaning if it can be found in the _buffers array) * This method is meant to be able to update a buffer at any time: just call setDataBuffer to set the class in the right state, call some updateXXX methods and then call udpate() => that will update the GPU buffer on the graphic card * @param dataBuffer buffer to look for * @returns true if the buffer has been found and the class internal state points to it, else false */ setDataBuffer(e) { if (!this._buffers) return this._buffer === e; for (let t = 0; t < this._buffers.length; ++t) if (this._buffers[t][0] === e) return this._bufferIndex = t, this._buffer = e, this._createBufferOnWrite = !1, this._currentEffect = void 0, !0; return !1; } /** * Disposes the uniform buffer. */ dispose() { if (this._noUBO) return; const e = this._engine._uniformBuffers, t = e.indexOf(this); if (t !== -1 && (e[t] = e[e.length - 1], e.pop()), this._engine._features.trackUbosInFrame && this._buffers) for (let r = 0; r < this._buffers.length; ++r) { const n = this._buffers[r][0]; this._engine._releaseBuffer(n); } else this._buffer && this._engine._releaseBuffer(this._buffer) && (this._buffer = null); } } yr._UpdatedUbosInFrame = {}; yr._MAX_UNIFORM_SIZE = 256; yr._TempBuffer = new Float32Array(yr._MAX_UNIFORM_SIZE); yr._TempBufferInt32View = new Int32Array(yr._TempBuffer.buffer); yr._TempBufferUInt32View = new Uint32Array(yr._TempBuffer.buffer); let P9 = class { /** * Gets a boolean indicating if the Buffer is disposed */ get isDisposed() { return this._isDisposed; } /** * Constructor * @param engine the engine * @param data the data to use for this buffer * @param updatable whether the data is updatable * @param stride the stride (optional) * @param postponeInternalCreation whether to postpone creating the internal WebGL buffer (optional) * @param instanced whether the buffer is instanced (optional) * @param useBytes set to true if the stride in in bytes (optional) * @param divisor sets an optional divisor for instances (1 by default) * @param label defines the label of the buffer (for debug purpose) */ constructor(e, t, r, n = 0, i = !1, s = !1, a = !1, f, o) { this._isAlreadyOwned = !1, this._isDisposed = !1, e && e.getScene ? this._engine = e.getScene().getEngine() : this._engine = e, this._updatable = r, this._instanced = s, this._divisor = f || 1, this._label = o, t instanceof tg ? (this._data = null, this._buffer = t) : (this._data = t, this._buffer = null), this.byteStride = a ? n : n * Float32Array.BYTES_PER_ELEMENT, i || this.create(); } /** * Create a new VertexBuffer based on the current buffer * @param kind defines the vertex buffer kind (position, normal, etc.) * @param offset defines offset in the buffer (0 by default) * @param size defines the size in floats of attributes (position is 3 for instance) * @param stride defines the stride size in floats in the buffer (the offset to apply to reach next value when data is interleaved) * @param instanced defines if the vertex buffer contains indexed data * @param useBytes defines if the offset and stride are in bytes * * @param divisor sets an optional divisor for instances (1 by default) * @returns the new vertex buffer */ createVertexBuffer(e, t, r, n, i, s = !1, a) { const f = s ? t : t * Float32Array.BYTES_PER_ELEMENT, o = n ? s ? n : n * Float32Array.BYTES_PER_ELEMENT : this.byteStride; return new J(this._engine, this, e, this._updatable, !0, o, i === void 0 ? this._instanced : i, f, r, void 0, void 0, !0, this._divisor || a); } // Properties /** * Gets a boolean indicating if the Buffer is updatable? * @returns true if the buffer is updatable */ isUpdatable() { return this._updatable; } /** * Gets current buffer's data * @returns a DataArray or null */ getData() { return this._data; } /** * Gets underlying native buffer * @returns underlying native buffer */ getBuffer() { return this._buffer; } /** * Gets the stride in float32 units (i.e. byte stride / 4). * May not be an integer if the byte stride is not divisible by 4. * @returns the stride in float32 units * @deprecated Please use byteStride instead. */ getStrideSize() { return this.byteStride / Float32Array.BYTES_PER_ELEMENT; } // Methods /** * Store data into the buffer. Creates the buffer if not used already. * If the buffer was already used, it will be updated only if it is updatable, otherwise it will do nothing. * @param data defines the data to store */ create(e = null) { !e && this._buffer || (e = e || this._data, e && (this._buffer ? this._updatable && (this._engine.updateDynamicVertexBuffer(this._buffer, e), this._data = e) : this._updatable ? (this._buffer = this._engine.createDynamicVertexBuffer(e, this._label), this._data = e) : this._buffer = this._engine.createVertexBuffer(e, void 0, this._label))); } /** @internal */ _rebuild() { this._buffer = null, this.create(this._data); } /** * Update current buffer data * @param data defines the data to store */ update(e) { this.create(e); } /** * Updates the data directly. * @param data the new data * @param offset the new offset * @param vertexCount the vertex count (optional) * @param useBytes set to true if the offset is in bytes */ updateDirectly(e, t, r, n = !1) { this._buffer && this._updatable && (this._engine.updateDynamicVertexBuffer(this._buffer, e, n ? t : t * Float32Array.BYTES_PER_ELEMENT, r ? r * this.byteStride : void 0), t === 0 && r === void 0 ? this._data = e : this._data = null); } /** @internal */ _increaseReferences() { if (this._buffer) { if (!this._isAlreadyOwned) { this._isAlreadyOwned = !0; return; } this._buffer.references++; } } /** * Release all resources */ dispose() { this._buffer && this._engine._releaseBuffer(this._buffer) && (this._isDisposed = !0, this._data = null, this._buffer = null); } }; class J { /** * Gets a boolean indicating if the Buffer is disposed */ get isDisposed() { return this._isDisposed; } /** * Gets or sets the instance divisor when in instanced mode */ get instanceDivisor() { return this._instanceDivisor; } set instanceDivisor(e) { const t = e != 0; this._instanceDivisor = e, t !== this._instanced && (this._instanced = t, this._computeHashCode()); } /** * Gets the number of vertices in the buffer */ get totalVertices() { const e = this.getData(); return e ? Array.isArray(e) ? e.length / (this.byteStride / 4) - this.byteOffset / 4 : (e.byteLength - this.byteOffset) / this.byteStride : 0; } /** @internal */ constructor(e, t, r, n, i, s, a, f, o, d, v = !1, u = !1, l = 1, P = !1) { var p, c, H, T, q; this._isDisposed = !1; let b = !1; if (this.engine = e, typeof n == "object" && n !== null ? (b = (p = n.updatable) !== null && p !== void 0 ? p : !1, i = n.postponeInternalCreation, s = n.stride, a = n.instanced, f = n.offset, o = n.size, d = n.type, v = (c = n.normalized) !== null && c !== void 0 ? c : !1, u = (H = n.useBytes) !== null && H !== void 0 ? H : !1, l = (T = n.divisor) !== null && T !== void 0 ? T : 1, P = (q = n.takeBufferOwnership) !== null && q !== void 0 ? q : !1, this._label = n.label) : b = !!n, t instanceof P9 ? (this._buffer = t, this._ownsBuffer = P) : (this._buffer = new P9(e, t, b, s, i, a, u, l, this._label), this._ownsBuffer = !0), this.uniqueId = J._Counter++, this._kind = r, d === void 0) { const w = this.getData(); this.type = w ? J.GetDataType(w) : J.FLOAT; } else this.type = d; const j = J.GetTypeByteLength(this.type); u ? (this._size = o || (s ? s / j : J.DeduceStride(r)), this.byteStride = s || this._buffer.byteStride || this._size * j, this.byteOffset = f || 0) : (this._size = o || s || J.DeduceStride(r), this.byteStride = s ? s * j : this._buffer.byteStride || this._size * j, this.byteOffset = (f || 0) * j), this.normalized = v, this._instanced = a !== void 0 ? a : !1, this._instanceDivisor = a ? l : 0, this._alignBuffer(), this._computeHashCode(); } _computeHashCode() { this.hashCode = (this.type - 5120 << 0) + ((this.normalized ? 1 : 0) << 3) + (this._size << 4) + ((this._instanced ? 1 : 0) << 6) + /* keep 5 bits free */ (this.byteStride << 12); } /** @internal */ _rebuild() { var e; (e = this._buffer) === null || e === void 0 || e._rebuild(); } /** * Returns the kind of the VertexBuffer (string) * @returns a string */ getKind() { return this._kind; } // Properties /** * Gets a boolean indicating if the VertexBuffer is updatable? * @returns true if the buffer is updatable */ isUpdatable() { return this._buffer.isUpdatable(); } /** * Gets current buffer's data * @returns a DataArray or null */ getData() { return this._buffer.getData(); } /** * Gets current buffer's data as a float array. Float data is constructed if the vertex buffer data cannot be returned directly. * @param totalVertices number of vertices in the buffer to take into account * @param forceCopy defines a boolean indicating that the returned array must be cloned upon returning it * @returns a float array containing vertex data */ getFloatData(e, t) { const r = this.getData(); return r ? (e = e ?? this.totalVertices, J.GetFloatData(r, this._size, this.type, this.byteOffset, this.byteStride, this.normalized, e, t)) : null; } /** * Gets underlying native buffer * @returns underlying native buffer */ getBuffer() { return this._buffer.getBuffer(); } /** * Gets the stride in float32 units (i.e. byte stride / 4). * May not be an integer if the byte stride is not divisible by 4. * @returns the stride in float32 units * @deprecated Please use byteStride instead. */ getStrideSize() { return this.byteStride / J.GetTypeByteLength(this.type); } /** * Returns the offset as a multiple of the type byte length. * @returns the offset in bytes * @deprecated Please use byteOffset instead. */ getOffset() { return this.byteOffset / J.GetTypeByteLength(this.type); } /** * Returns the number of components or the byte size per vertex attribute * @param sizeInBytes If true, returns the size in bytes or else the size in number of components of the vertex attribute (default: false) * @returns the number of components */ getSize(e = !1) { return e ? this._size * J.GetTypeByteLength(this.type) : this._size; } /** * Gets a boolean indicating is the internal buffer of the VertexBuffer is instanced * @returns true if this buffer is instanced */ getIsInstanced() { return this._instanced; } /** * Returns the instancing divisor, zero for non-instanced (integer). * @returns a number */ getInstanceDivisor() { return this._instanceDivisor; } // Methods /** * Store data into the buffer. If the buffer was already used it will be either recreated or updated depending on isUpdatable property * @param data defines the data to store */ create(e) { this._buffer.create(e), this._alignBuffer(); } /** * Updates the underlying buffer according to the passed numeric array or Float32Array. * This function will create a new buffer if the current one is not updatable * @param data defines the data to store */ update(e) { this._buffer.update(e), this._alignBuffer(); } /** * Updates directly the underlying WebGLBuffer according to the passed numeric array or Float32Array. * Returns the directly updated WebGLBuffer. * @param data the new data * @param offset the new offset * @param useBytes set to true if the offset is in bytes */ updateDirectly(e, t, r = !1) { this._buffer.updateDirectly(e, t, void 0, r), this._alignBuffer(); } /** * Disposes the VertexBuffer and the underlying WebGLBuffer. */ dispose() { this._ownsBuffer && this._buffer.dispose(), this._isDisposed = !0; } /** * Enumerates each value of this vertex buffer as numbers. * @param count the number of values to enumerate * @param callback the callback function called for each value */ forEach(e, t) { J.ForEach(this._buffer.getData(), this.byteOffset, this.byteStride, this._size, this.type, e, this.normalized, t); } /** @internal */ _alignBuffer() { } /** * Deduces the stride given a kind. * @param kind The kind string to deduce * @returns The deduced stride */ static DeduceStride(e) { switch (e) { case J.UVKind: case J.UV2Kind: case J.UV3Kind: case J.UV4Kind: case J.UV5Kind: case J.UV6Kind: return 2; case J.NormalKind: case J.PositionKind: return 3; case J.ColorKind: case J.ColorInstanceKind: case J.MatricesIndicesKind: case J.MatricesIndicesExtraKind: case J.MatricesWeightsKind: case J.MatricesWeightsExtraKind: case J.TangentKind: return 4; default: throw new Error("Invalid kind '" + e + "'"); } } /** * Gets the vertex buffer type of the given data array. * @param data the data array * @returns the vertex buffer type */ static GetDataType(e) { return e instanceof Int8Array ? J.BYTE : e instanceof Uint8Array ? J.UNSIGNED_BYTE : e instanceof Int16Array ? J.SHORT : e instanceof Uint16Array ? J.UNSIGNED_SHORT : e instanceof Int32Array ? J.INT : e instanceof Uint32Array ? J.UNSIGNED_INT : J.FLOAT; } /** * Gets the byte length of the given type. * @param type the type * @returns the number of bytes */ static GetTypeByteLength(e) { switch (e) { case J.BYTE: case J.UNSIGNED_BYTE: return 1; case J.SHORT: case J.UNSIGNED_SHORT: return 2; case J.INT: case J.UNSIGNED_INT: case J.FLOAT: return 4; default: throw new Error(`Invalid type '${e}'`); } } /** * Enumerates each value of the given parameters as numbers. * @param data the data to enumerate * @param byteOffset the byte offset of the data * @param byteStride the byte stride of the data * @param componentCount the number of components per element * @param componentType the type of the component * @param count the number of values to enumerate * @param normalized whether the data is normalized * @param callback the callback function called for each value */ static ForEach(e, t, r, n, i, s, a, f) { if (e instanceof Array) { let o = t / 4; const d = r / 4; for (let v = 0; v < s; v += n) { for (let u = 0; u < n; u++) f(e[o + u], v + u); o += d; } } else { const o = e instanceof ArrayBuffer ? new DataView(e) : new DataView(e.buffer, e.byteOffset, e.byteLength), d = J.GetTypeByteLength(i); for (let v = 0; v < s; v += n) { let u = t; for (let l = 0; l < n; l++) { const P = J._GetFloatValue(o, i, u, a); f(P, v + l), u += d; } t += r; } } } static _GetFloatValue(e, t, r, n) { switch (t) { case J.BYTE: { let i = e.getInt8(r); return n && (i = Math.max(i / 127, -1)), i; } case J.UNSIGNED_BYTE: { let i = e.getUint8(r); return n && (i = i / 255), i; } case J.SHORT: { let i = e.getInt16(r, !0); return n && (i = Math.max(i / 32767, -1)), i; } case J.UNSIGNED_SHORT: { let i = e.getUint16(r, !0); return n && (i = i / 65535), i; } case J.INT: return e.getInt32(r, !0); case J.UNSIGNED_INT: return e.getUint32(r, !0); case J.FLOAT: return e.getFloat32(r, !0); default: throw new Error(`Invalid component type ${t}`); } } /** * Gets the given data array as a float array. Float data is constructed if the data array cannot be returned directly. * @param data the input data array * @param size the number of components * @param type the component type * @param byteOffset the byte offset of the data * @param byteStride the byte stride of the data * @param normalized whether the data is normalized * @param totalVertices number of vertices in the buffer to take into account * @param forceCopy defines a boolean indicating that the returned array must be cloned upon returning it * @returns a float array containing vertex data */ static GetFloatData(e, t, r, n, i, s, a, f) { const o = t * J.GetTypeByteLength(r), d = a * t; if (r !== J.FLOAT || i !== o) { const v = new Float32Array(d); return J.ForEach(e, n, i, t, r, d, s, (u, l) => v[l] = u), v; } if (!(e instanceof Array || e instanceof Float32Array) || n !== 0 || e.length !== d) if (e instanceof Array) { const v = n / 4; return e.slice(v, v + d); } else { if (e instanceof ArrayBuffer) return new Float32Array(e, n, d); { let v = e.byteOffset + n; if (f) { const l = new Float32Array(d), P = new Float32Array(e.buffer, v, d); return l.set(P), l; } const u = v % 4; return u && (v = Math.max(0, v - u)), new Float32Array(e.buffer, v, d); } } return f ? e.slice() : e; } } J._Counter = 0; J.BYTE = 5120; J.UNSIGNED_BYTE = 5121; J.SHORT = 5122; J.UNSIGNED_SHORT = 5123; J.INT = 5124; J.UNSIGNED_INT = 5125; J.FLOAT = 5126; J.PositionKind = "position"; J.NormalKind = "normal"; J.TangentKind = "tangent"; J.UVKind = "uv"; J.UV2Kind = "uv2"; J.UV3Kind = "uv3"; J.UV4Kind = "uv4"; J.UV5Kind = "uv5"; J.UV6Kind = "uv6"; J.ColorKind = "color"; J.ColorInstanceKind = "instanceColor"; J.MatricesIndicesKind = "matricesIndices"; J.MatricesWeightsKind = "matricesWeights"; J.MatricesIndicesExtraKind = "matricesIndicesExtra"; J.MatricesWeightsExtraKind = "matricesWeightsExtra"; class F9 { constructor() { this.hit = !1, this.distance = 0, this.pickedPoint = null, this.pickedMesh = null, this.bu = 0, this.bv = 0, this.faceId = -1, this.subMeshFaceId = -1, this.subMeshId = 0, this.pickedSprite = null, this.thinInstanceIndex = -1, this.ray = null, this.originMesh = null, this.aimTransform = null, this.gripTransform = null; } /** * Gets the normal corresponding to the face the pick collided with * @param useWorldCoordinates If the resulting normal should be relative to the world (default: false) * @param useVerticesNormals If the vertices normals should be used to calculate the normal instead of the normal map (default: true) * @returns The normal corresponding to the face the pick collided with * @remarks Note that the returned normal will always point towards the picking ray. */ getNormal(e = !1, t = !0) { if (!this.pickedMesh || t && !this.pickedMesh.isVerticesDataPresent(J.NormalKind)) return null; let r = this.pickedMesh.getIndices(); (r == null ? void 0 : r.length) === 0 && (r = null); let n; const i = ue.Vector3[0], s = ue.Vector3[1], a = ue.Vector3[2]; if (t) { const o = this.pickedMesh.getVerticesData(J.NormalKind); let d = r ? S.FromArrayToRef(o, r[this.faceId * 3] * 3, i) : i.copyFromFloats(o[this.faceId * 3 * 3], o[this.faceId * 3 * 3 + 1], o[this.faceId * 3 * 3 + 2]), v = r ? S.FromArrayToRef(o, r[this.faceId * 3 + 1] * 3, s) : s.copyFromFloats(o[(this.faceId * 3 + 1) * 3], o[(this.faceId * 3 + 1) * 3 + 1], o[(this.faceId * 3 + 1) * 3 + 2]), u = r ? S.FromArrayToRef(o, r[this.faceId * 3 + 2] * 3, a) : a.copyFromFloats(o[(this.faceId * 3 + 2) * 3], o[(this.faceId * 3 + 2) * 3 + 1], o[(this.faceId * 3 + 2) * 3 + 2]); d = d.scale(this.bu), v = v.scale(this.bv), u = u.scale(1 - this.bu - this.bv), n = new S(d.x + v.x + u.x, d.y + v.y + u.y, d.z + v.z + u.z); } else { const o = this.pickedMesh.getVerticesData(J.PositionKind), d = r ? S.FromArrayToRef(o, r[this.faceId * 3] * 3, i) : i.copyFromFloats(o[this.faceId * 3 * 3], o[this.faceId * 3 * 3 + 1], o[this.faceId * 3 * 3 + 2]), v = r ? S.FromArrayToRef(o, r[this.faceId * 3 + 1] * 3, s) : s.copyFromFloats(o[(this.faceId * 3 + 1) * 3], o[(this.faceId * 3 + 1) * 3 + 1], o[(this.faceId * 3 + 1) * 3 + 2]), u = r ? S.FromArrayToRef(o, r[this.faceId * 3 + 2] * 3, a) : a.copyFromFloats(o[(this.faceId * 3 + 2) * 3], o[(this.faceId * 3 + 2) * 3 + 1], o[(this.faceId * 3 + 2) * 3 + 2]), l = d.subtract(v), P = u.subtract(v); n = S.Cross(l, P); } const f = (o, d) => { let v = o.getWorldMatrix(); o.nonUniformScaling && (ue.Matrix[0].copyFrom(v), v = ue.Matrix[0], v.setTranslationFromFloats(0, 0, 0), v.invert(), v.transposeToRef(ue.Matrix[1]), v = ue.Matrix[1]), S.TransformNormalToRef(d, v, d); }; if (e && f(this.pickedMesh, n), this.ray) { const o = ue.Vector3[0].copyFrom(n); e || f(this.pickedMesh, o), S.Dot(o, this.ray.direction) > 0 && n.negateInPlace(); } return n.normalize(), n; } /** * Gets the texture coordinates of where the pick occurred * @param uvSet The UV set to use to calculate the texture coordinates (default: VertexBuffer.UVKind) * @returns The vector containing the coordinates of the texture */ getTextureCoordinates(e = J.UVKind) { if (!this.pickedMesh || !this.pickedMesh.isVerticesDataPresent(e)) return null; const t = this.pickedMesh.getIndices(); if (!t) return null; const r = this.pickedMesh.getVerticesData(e); if (!r) return null; let n = at.FromArray(r, t[this.faceId * 3] * 2), i = at.FromArray(r, t[this.faceId * 3 + 1] * 2), s = at.FromArray(r, t[this.faceId * 3 + 2] * 2); return n = n.scale(this.bu), i = i.scale(this.bv), s = s.scale(1 - this.bu - this.bv), new at(n.x + i.x + s.x, n.y + i.y + s.y); } } class yI { /** * Creates a new instance PostProcess * @param scene The scene that the post process is associated with. */ constructor(e) { this._vertexBuffers = {}, this._scene = e; } _prepareBuffers() { if (this._vertexBuffers[J.PositionKind]) return; const e = []; e.push(1, 1), e.push(-1, 1), e.push(-1, -1), e.push(1, -1), this._vertexBuffers[J.PositionKind] = new J(this._scene.getEngine(), e, J.PositionKind, !1, !1, 2), this._buildIndexBuffer(); } _buildIndexBuffer() { const e = []; e.push(0), e.push(1), e.push(2), e.push(0), e.push(2), e.push(3), this._indexBuffer = this._scene.getEngine().createIndexBuffer(e); } /** * Rebuilds the vertex buffers of the manager. * @internal */ _rebuild() { const e = this._vertexBuffers[J.PositionKind]; e && (e._rebuild(), this._buildIndexBuffer()); } // Methods /** * Prepares a frame to be run through a post process. * @param sourceTexture The input texture to the post processes. (default: null) * @param postProcesses An array of post processes to be run. (default: null) * @returns True if the post processes were able to be run. * @internal */ _prepareFrame(e = null, t = null) { const r = this._scene.activeCamera; return !r || (t = t || r._postProcesses.filter((n) => n != null), !t || t.length === 0 || !this._scene.postProcessesEnabled) ? !1 : (t[0].activate(r, e, t != null), !0); } /** * Manually render a set of post processes to a texture. * Please note, the frame buffer won't be unbound after the call in case you have more render to do. * @param postProcesses An array of post processes to be run. * @param targetTexture The render target wrapper to render to. * @param forceFullscreenViewport force gl.viewport to be full screen eg. 0,0,textureWidth,textureHeight * @param faceIndex defines the face to render to if a cubemap is defined as the target * @param lodLevel defines which lod of the texture to render to * @param doNotBindFrambuffer If set to true, assumes that the framebuffer has been bound previously */ directRender(e, t = null, r = !1, n = 0, i = 0, s = !1) { var a; const f = this._scene.getEngine(); for (let o = 0; o < e.length; o++) { o < e.length - 1 ? e[o + 1].activate(this._scene.activeCamera, t == null ? void 0 : t.texture) : (t ? f.bindFramebuffer(t, n, void 0, void 0, r, i) : s || f.restoreDefaultFramebuffer(), (a = f._debugInsertMarker) === null || a === void 0 || a.call(f, `post process ${e[o].name} output`)); const d = e[o], v = d.apply(); v && (d.onBeforeRenderObservable.notifyObservers(v), this._prepareBuffers(), f.bindBuffers(this._vertexBuffers, this._indexBuffer, v), f.drawElementsType(0, 0, 6), d.onAfterRenderObservable.notifyObservers(v)); } f.setDepthBuffer(!0), f.setDepthWrite(!0); } /** * Finalize the result of the output of the postprocesses. * @param doNotPresent If true the result will not be displayed to the screen. * @param targetTexture The render target wrapper to render to. * @param faceIndex The index of the face to bind the target texture to. * @param postProcesses The array of post processes to render. * @param forceFullscreenViewport force gl.viewport to be full screen eg. 0,0,textureWidth,textureHeight (default: false) * @internal */ _finalizeFrame(e, t, r, n, i = !1) { var s; const a = this._scene.activeCamera; if (!a || (n = n || a._postProcesses.filter((o) => o != null), n.length === 0 || !this._scene.postProcessesEnabled)) return; const f = this._scene.getEngine(); for (let o = 0, d = n.length; o < d; o++) { const v = n[o]; if (o < d - 1 ? v._outputTexture = n[o + 1].activate(a, t == null ? void 0 : t.texture) : (t ? (f.bindFramebuffer(t, r, void 0, void 0, i), v._outputTexture = t) : (f.restoreDefaultFramebuffer(), v._outputTexture = null), (s = f._debugInsertMarker) === null || s === void 0 || s.call(f, `post process ${n[o].name} output`)), e) break; const u = v.apply(); u && (v.onBeforeRenderObservable.notifyObservers(u), this._prepareBuffers(), f.bindBuffers(this._vertexBuffers, this._indexBuffer, u), f.drawElementsType(0, 0, 6), v.onAfterRenderObservable.notifyObservers(u)); } f.setDepthBuffer(!0), f.setDepthWrite(!0), f.setAlphaMode(0); } /** * Disposes of the post process manager. */ dispose() { const e = this._vertexBuffers[J.PositionKind]; e && (e.dispose(), this._vertexBuffers[J.PositionKind] = null), this._indexBuffer && (this._scene.getEngine()._releaseBuffer(this._indexBuffer), this._indexBuffer = null); } } class x0 { /** * Set the opaque sort comparison function. * If null the sub meshes will be render in the order they were created */ set opaqueSortCompareFn(e) { e ? this._opaqueSortCompareFn = e : this._opaqueSortCompareFn = x0.PainterSortCompare, this._renderOpaque = this._renderOpaqueSorted; } /** * Set the alpha test sort comparison function. * If null the sub meshes will be render in the order they were created */ set alphaTestSortCompareFn(e) { e ? this._alphaTestSortCompareFn = e : this._alphaTestSortCompareFn = x0.PainterSortCompare, this._renderAlphaTest = this._renderAlphaTestSorted; } /** * Set the transparent sort comparison function. * If null the sub meshes will be render in the order they were created */ set transparentSortCompareFn(e) { e ? this._transparentSortCompareFn = e : this._transparentSortCompareFn = x0.defaultTransparentSortCompare, this._renderTransparent = this._renderTransparentSorted; } /** * Creates a new rendering group. * @param index The rendering group index * @param scene * @param opaqueSortCompareFn The opaque sort comparison function. If null no order is applied * @param alphaTestSortCompareFn The alpha test sort comparison function. If null no order is applied * @param transparentSortCompareFn The transparent sort comparison function. If null back to front + alpha index sort is applied */ constructor(e, t, r = null, n = null, i = null) { this.index = e, this._opaqueSubMeshes = new qf(256), this._transparentSubMeshes = new qf(256), this._alphaTestSubMeshes = new qf(256), this._depthOnlySubMeshes = new qf(256), this._particleSystems = new qf(256), this._spriteManagers = new qf(256), this._empty = !0, this._edgesRenderers = new K8(16), this._scene = t, this.opaqueSortCompareFn = r, this.alphaTestSortCompareFn = n, this.transparentSortCompareFn = i; } /** * Render all the sub meshes contained in the group. * @param customRenderFunction Used to override the default render behaviour of the group. * @param renderSprites * @param renderParticles * @param activeMeshes * @returns true if rendered some submeshes. */ render(e, t, r, n) { if (e) { e(this._opaqueSubMeshes, this._alphaTestSubMeshes, this._transparentSubMeshes, this._depthOnlySubMeshes); return; } const i = this._scene.getEngine(); this._depthOnlySubMeshes.length !== 0 && (i.setColorWrite(!1), this._renderAlphaTest(this._depthOnlySubMeshes), i.setColorWrite(!0)), this._opaqueSubMeshes.length !== 0 && this._renderOpaque(this._opaqueSubMeshes), this._alphaTestSubMeshes.length !== 0 && this._renderAlphaTest(this._alphaTestSubMeshes); const s = i.getStencilBuffer(); if (i.setStencilBuffer(!1), t && this._renderSprites(), r && this._renderParticles(n), this.onBeforeTransparentRendering && this.onBeforeTransparentRendering(), this._transparentSubMeshes.length !== 0 || this._scene.useOrderIndependentTransparency) { if (i.setStencilBuffer(s), this._scene.useOrderIndependentTransparency) { const a = this._scene.depthPeelingRenderer.render(this._transparentSubMeshes); a.length && this._renderTransparent(a); } else this._renderTransparent(this._transparentSubMeshes); i.setAlphaMode(0); } if (i.setStencilBuffer(!1), this._edgesRenderers.length) { for (let a = 0; a < this._edgesRenderers.length; a++) this._edgesRenderers.data[a].render(); i.setAlphaMode(0); } i.setStencilBuffer(s); } /** * Renders the opaque submeshes in the order from the opaqueSortCompareFn. * @param subMeshes The submeshes to render */ _renderOpaqueSorted(e) { return x0._RenderSorted(e, this._opaqueSortCompareFn, this._scene.activeCamera, !1); } /** * Renders the opaque submeshes in the order from the alphatestSortCompareFn. * @param subMeshes The submeshes to render */ _renderAlphaTestSorted(e) { return x0._RenderSorted(e, this._alphaTestSortCompareFn, this._scene.activeCamera, !1); } /** * Renders the opaque submeshes in the order from the transparentSortCompareFn. * @param subMeshes The submeshes to render */ _renderTransparentSorted(e) { return x0._RenderSorted(e, this._transparentSortCompareFn, this._scene.activeCamera, !0); } /** * Renders the submeshes in a specified order. * @param subMeshes The submeshes to sort before render * @param sortCompareFn The comparison function use to sort * @param camera The camera position use to preprocess the submeshes to help sorting * @param transparent Specifies to activate blending if true */ static _RenderSorted(e, t, r, n) { let i = 0, s; const a = r ? r.globalPosition : x0._ZeroVector; if (n) for (; i < e.length; i++) s = e.data[i], s._alphaIndex = s.getMesh().alphaIndex, s._distanceToCamera = S.Distance(s.getBoundingInfo().boundingSphere.centerWorld, a); const f = e.length === e.data.length ? e.data : e.data.slice(0, e.length); t && f.sort(t); const o = f[0].getMesh().getScene(); for (i = 0; i < f.length; i++) if (s = f[i], !(o._activeMeshesFrozenButKeepClipping && !s.isInFrustum(o._frustumPlanes))) { if (n) { const d = s.getMaterial(); if (d && d.needDepthPrePass) { const v = d.getScene().getEngine(); v.setColorWrite(!1), v.setAlphaMode(0), s.render(!1), v.setColorWrite(!0); } } s.render(n); } } /** * Build in function which can be applied to ensure meshes of a special queue (opaque, alpha test, transparent) * are rendered back to front if in the same alpha index. * * @param a The first submesh * @param b The second submesh * @returns The result of the comparison */ // eslint-disable-next-line @typescript-eslint/naming-convention static defaultTransparentSortCompare(e, t) { return e._alphaIndex > t._alphaIndex ? 1 : e._alphaIndex < t._alphaIndex ? -1 : x0.backToFrontSortCompare(e, t); } /** * Build in function which can be applied to ensure meshes of a special queue (opaque, alpha test, transparent) * are rendered back to front. * * @param a The first submesh * @param b The second submesh * @returns The result of the comparison */ // eslint-disable-next-line @typescript-eslint/naming-convention static backToFrontSortCompare(e, t) { return e._distanceToCamera < t._distanceToCamera ? 1 : e._distanceToCamera > t._distanceToCamera ? -1 : 0; } /** * Build in function which can be applied to ensure meshes of a special queue (opaque, alpha test, transparent) * are rendered front to back (prevent overdraw). * * @param a The first submesh * @param b The second submesh * @returns The result of the comparison */ // eslint-disable-next-line @typescript-eslint/naming-convention static frontToBackSortCompare(e, t) { return e._distanceToCamera < t._distanceToCamera ? -1 : e._distanceToCamera > t._distanceToCamera ? 1 : 0; } /** * Build in function which can be applied to ensure meshes of a special queue (opaque, alpha test, transparent) * are grouped by material then geometry. * * @param a The first submesh * @param b The second submesh * @returns The result of the comparison */ static PainterSortCompare(e, t) { const r = e.getMesh(), n = t.getMesh(); return r.material && n.material ? r.material.uniqueId - n.material.uniqueId : r.uniqueId - n.uniqueId; } /** * Resets the different lists of submeshes to prepare a new frame. */ prepare() { this._opaqueSubMeshes.reset(), this._transparentSubMeshes.reset(), this._alphaTestSubMeshes.reset(), this._depthOnlySubMeshes.reset(), this._particleSystems.reset(), this.prepareSprites(), this._edgesRenderers.reset(), this._empty = !0; } /** * Resets the different lists of sprites to prepare a new frame. */ prepareSprites() { this._spriteManagers.reset(); } dispose() { this._opaqueSubMeshes.dispose(), this._transparentSubMeshes.dispose(), this._alphaTestSubMeshes.dispose(), this._depthOnlySubMeshes.dispose(), this._particleSystems.dispose(), this._spriteManagers.dispose(), this._edgesRenderers.dispose(); } /** * Inserts the submesh in its correct queue depending on its material. * @param subMesh The submesh to dispatch * @param [mesh] Optional reference to the submeshes's mesh. Provide if you have an exiting reference to improve performance. * @param [material] Optional reference to the submeshes's material. Provide if you have an exiting reference to improve performance. */ dispatch(e, t, r) { t === void 0 && (t = e.getMesh()), r === void 0 && (r = e.getMaterial()), r != null && (r.needAlphaBlendingForMesh(t) ? this._transparentSubMeshes.push(e) : r.needAlphaTesting() ? (r.needDepthPrePass && this._depthOnlySubMeshes.push(e), this._alphaTestSubMeshes.push(e)) : (r.needDepthPrePass && this._depthOnlySubMeshes.push(e), this._opaqueSubMeshes.push(e)), t._renderingGroup = this, t._edgesRenderer && t._edgesRenderer.isEnabled && this._edgesRenderers.pushNoDuplicate(t._edgesRenderer), this._empty = !1); } dispatchSprites(e) { this._spriteManagers.push(e), this._empty = !1; } dispatchParticles(e) { this._particleSystems.push(e), this._empty = !1; } _renderParticles(e) { if (this._particleSystems.length === 0) return; const t = this._scene.activeCamera; this._scene.onBeforeParticlesRenderingObservable.notifyObservers(this._scene); for (let r = 0; r < this._particleSystems.length; r++) { const n = this._particleSystems.data[r]; if ((t && t.layerMask & n.layerMask) === 0) continue; const i = n.emitter; (!i.position || !e || e.indexOf(i) !== -1) && this._scene._activeParticles.addCount(n.render(), !1); } this._scene.onAfterParticlesRenderingObservable.notifyObservers(this._scene); } _renderSprites() { if (!this._scene.spritesEnabled || this._spriteManagers.length === 0) return; const e = this._scene.activeCamera; this._scene.onBeforeSpritesRenderingObservable.notifyObservers(this._scene); for (let t = 0; t < this._spriteManagers.length; t++) { const r = this._spriteManagers.data[t]; (e && e.layerMask & r.layerMask) !== 0 && r.render(); } this._scene.onAfterSpritesRenderingObservable.notifyObservers(this._scene); } } x0._ZeroVector = S.Zero(); class L$ { } class $6 { /** * Gets or sets a boolean indicating that the manager will not reset between frames. * This means that if a mesh becomes invisible or transparent it will not be visible until this boolean is set to false again. * By default, the rendering manager will dispatch all active meshes per frame (moving them to the transparent, opaque or alpha testing lists). * By turning this property on, you will accelerate the rendering by keeping all these lists unchanged between frames. */ get maintainStateBetweenFrames() { return this._maintainStateBetweenFrames; } set maintainStateBetweenFrames(e) { e !== this._maintainStateBetweenFrames && (this._maintainStateBetweenFrames = e, this._maintainStateBetweenFrames || this.restoreDispachedFlags()); } /** * Restore wasDispatched flags on the lists of elements to render. */ restoreDispachedFlags() { for (const e of this._scene.meshes) if (e.subMeshes) for (const t of e.subMeshes) t._wasDispatched = !1; if (this._scene.spriteManagers) for (const e of this._scene.spriteManagers) e._wasDispatched = !1; for (const e of this._scene.particleSystems) e._wasDispatched = !1; } /** * Instantiates a new rendering group for a particular scene * @param scene Defines the scene the groups belongs to */ constructor(e) { this._useSceneAutoClearSetup = !1, this._renderingGroups = new Array(), this._autoClearDepthStencil = {}, this._customOpaqueSortCompareFn = {}, this._customAlphaTestSortCompareFn = {}, this._customTransparentSortCompareFn = {}, this._renderingGroupInfo = new L$(), this._maintainStateBetweenFrames = !1, this._scene = e; for (let t = $6.MIN_RENDERINGGROUPS; t < $6.MAX_RENDERINGGROUPS; t++) this._autoClearDepthStencil[t] = { autoClear: !0, depth: !0, stencil: !0 }; } /** * Gets the rendering group with the specified id. */ getRenderingGroup(e) { const t = e || 0; return this._prepareRenderingGroup(t), this._renderingGroups[t]; } _clearDepthStencilBuffer(e = !0, t = !0) { this._depthStencilBufferAlreadyCleaned || (this._scene.getEngine().clear(null, !1, e, t), this._depthStencilBufferAlreadyCleaned = !0); } /** * Renders the entire managed groups. This is used by the scene or the different render targets. * @internal */ render(e, t, r, n) { const i = this._renderingGroupInfo; if (i.scene = this._scene, i.camera = this._scene.activeCamera, this._scene.spriteManagers && n) for (let s = 0; s < this._scene.spriteManagers.length; s++) { const a = this._scene.spriteManagers[s]; this.dispatchSprites(a); } for (let s = $6.MIN_RENDERINGGROUPS; s < $6.MAX_RENDERINGGROUPS; s++) { this._depthStencilBufferAlreadyCleaned = s === $6.MIN_RENDERINGGROUPS; const a = this._renderingGroups[s]; if (!a || a._empty) continue; const f = Math.pow(2, s); if (i.renderingGroupId = s, this._scene.onBeforeRenderingGroupObservable.notifyObservers(i, f), $6.AUTOCLEAR) { const o = this._useSceneAutoClearSetup ? this._scene.getAutoClearDepthStencilSetup(s) : this._autoClearDepthStencil[s]; o && o.autoClear && this._clearDepthStencilBuffer(o.depth, o.stencil); } for (const o of this._scene._beforeRenderingGroupDrawStage) o.action(s); a.render(e, n, r, t); for (const o of this._scene._afterRenderingGroupDrawStage) o.action(s); this._scene.onAfterRenderingGroupObservable.notifyObservers(i, f); } } /** * Resets the different information of the group to prepare a new frame * @internal */ reset() { if (!this.maintainStateBetweenFrames) for (let e = $6.MIN_RENDERINGGROUPS; e < $6.MAX_RENDERINGGROUPS; e++) { const t = this._renderingGroups[e]; t && t.prepare(); } } /** * Resets the sprites information of the group to prepare a new frame * @internal */ resetSprites() { if (!this.maintainStateBetweenFrames) for (let e = $6.MIN_RENDERINGGROUPS; e < $6.MAX_RENDERINGGROUPS; e++) { const t = this._renderingGroups[e]; t && t.prepareSprites(); } } /** * Dispose and release the group and its associated resources. * @internal */ dispose() { this.freeRenderingGroups(), this._renderingGroups.length = 0, this._renderingGroupInfo = null; } /** * Clear the info related to rendering groups preventing retention points during dispose. */ freeRenderingGroups() { for (let e = $6.MIN_RENDERINGGROUPS; e < $6.MAX_RENDERINGGROUPS; e++) { const t = this._renderingGroups[e]; t && t.dispose(); } } _prepareRenderingGroup(e) { this._renderingGroups[e] === void 0 && (this._renderingGroups[e] = new x0(e, this._scene, this._customOpaqueSortCompareFn[e], this._customAlphaTestSortCompareFn[e], this._customTransparentSortCompareFn[e])); } /** * Add a sprite manager to the rendering manager in order to render it this frame. * @param spriteManager Define the sprite manager to render */ dispatchSprites(e) { this.maintainStateBetweenFrames && e._wasDispatched || (e._wasDispatched = !0, this.getRenderingGroup(e.renderingGroupId).dispatchSprites(e)); } /** * Add a particle system to the rendering manager in order to render it this frame. * @param particleSystem Define the particle system to render */ dispatchParticles(e) { this.maintainStateBetweenFrames && e._wasDispatched || (e._wasDispatched = !0, this.getRenderingGroup(e.renderingGroupId).dispatchParticles(e)); } /** * Add a submesh to the manager in order to render it this frame * @param subMesh The submesh to dispatch * @param mesh Optional reference to the submeshes's mesh. Provide if you have an exiting reference to improve performance. * @param material Optional reference to the submeshes's material. Provide if you have an exiting reference to improve performance. */ dispatch(e, t, r) { t === void 0 && (t = e.getMesh()), !(this.maintainStateBetweenFrames && e._wasDispatched) && (e._wasDispatched = !0, this.getRenderingGroup(t.renderingGroupId).dispatch(e, t, r)); } /** * Overrides the default sort function applied in the rendering group to prepare the meshes. * This allowed control for front to back rendering or reversely depending of the special needs. * * @param renderingGroupId The rendering group id corresponding to its index * @param opaqueSortCompareFn The opaque queue comparison function use to sort. * @param alphaTestSortCompareFn The alpha test queue comparison function use to sort. * @param transparentSortCompareFn The transparent queue comparison function use to sort. */ setRenderingOrder(e, t = null, r = null, n = null) { if (this._customOpaqueSortCompareFn[e] = t, this._customAlphaTestSortCompareFn[e] = r, this._customTransparentSortCompareFn[e] = n, this._renderingGroups[e]) { const i = this._renderingGroups[e]; i.opaqueSortCompareFn = this._customOpaqueSortCompareFn[e], i.alphaTestSortCompareFn = this._customAlphaTestSortCompareFn[e], i.transparentSortCompareFn = this._customTransparentSortCompareFn[e]; } } /** * Specifies whether or not the stencil and depth buffer are cleared between two rendering groups. * * @param renderingGroupId The rendering group id corresponding to its index * @param autoClearDepthStencil Automatically clears depth and stencil between groups if true. * @param depth Automatically clears depth between groups if true and autoClear is true. * @param stencil Automatically clears stencil between groups if true and autoClear is true. */ setRenderingAutoClearDepthStencil(e, t, r = !0, n = !0) { this._autoClearDepthStencil[e] = { autoClear: t, depth: r, stencil: n }; } /** * Gets the current auto clear configuration for one rendering group of the rendering * manager. * @param index the rendering group index to get the information for * @returns The auto clear setup for the requested rendering group */ getAutoClearDepthStencilSetup(e) { return this._autoClearDepthStencil[e]; } } $6.MAX_RENDERINGGROUPS = 4; $6.MIN_RENDERINGGROUPS = 0; $6.AUTOCLEAR = !0; class Ot { } Ot.NAME_EFFECTLAYER = "EffectLayer"; Ot.NAME_LAYER = "Layer"; Ot.NAME_LENSFLARESYSTEM = "LensFlareSystem"; Ot.NAME_BOUNDINGBOXRENDERER = "BoundingBoxRenderer"; Ot.NAME_PARTICLESYSTEM = "ParticleSystem"; Ot.NAME_GAMEPAD = "Gamepad"; Ot.NAME_SIMPLIFICATIONQUEUE = "SimplificationQueue"; Ot.NAME_GEOMETRYBUFFERRENDERER = "GeometryBufferRenderer"; Ot.NAME_PREPASSRENDERER = "PrePassRenderer"; Ot.NAME_DEPTHRENDERER = "DepthRenderer"; Ot.NAME_DEPTHPEELINGRENDERER = "DepthPeelingRenderer"; Ot.NAME_POSTPROCESSRENDERPIPELINEMANAGER = "PostProcessRenderPipelineManager"; Ot.NAME_SPRITE = "Sprite"; Ot.NAME_SUBSURFACE = "SubSurface"; Ot.NAME_OUTLINERENDERER = "Outline"; Ot.NAME_PROCEDURALTEXTURE = "ProceduralTexture"; Ot.NAME_SHADOWGENERATOR = "ShadowGenerator"; Ot.NAME_OCTREE = "Octree"; Ot.NAME_PHYSICSENGINE = "PhysicsEngine"; Ot.NAME_AUDIO = "Audio"; Ot.NAME_FLUIDRENDERER = "FluidRenderer"; Ot.STEP_ISREADYFORMESH_EFFECTLAYER = 0; Ot.STEP_BEFOREEVALUATEACTIVEMESH_BOUNDINGBOXRENDERER = 0; Ot.STEP_EVALUATESUBMESH_BOUNDINGBOXRENDERER = 0; Ot.STEP_PREACTIVEMESH_BOUNDINGBOXRENDERER = 0; Ot.STEP_CAMERADRAWRENDERTARGET_EFFECTLAYER = 1; Ot.STEP_BEFORECAMERADRAW_PREPASS = 0; Ot.STEP_BEFORECAMERADRAW_EFFECTLAYER = 1; Ot.STEP_BEFORECAMERADRAW_LAYER = 2; Ot.STEP_BEFORERENDERTARGETDRAW_PREPASS = 0; Ot.STEP_BEFORERENDERTARGETDRAW_LAYER = 1; Ot.STEP_BEFORERENDERINGMESH_PREPASS = 0; Ot.STEP_BEFORERENDERINGMESH_OUTLINE = 1; Ot.STEP_AFTERRENDERINGMESH_PREPASS = 0; Ot.STEP_AFTERRENDERINGMESH_OUTLINE = 1; Ot.STEP_AFTERRENDERINGGROUPDRAW_EFFECTLAYER_DRAW = 0; Ot.STEP_AFTERRENDERINGGROUPDRAW_BOUNDINGBOXRENDERER = 1; Ot.STEP_BEFORECAMERAUPDATE_SIMPLIFICATIONQUEUE = 0; Ot.STEP_BEFORECAMERAUPDATE_GAMEPAD = 1; Ot.STEP_BEFORECLEAR_PROCEDURALTEXTURE = 0; Ot.STEP_BEFORECLEAR_PREPASS = 1; Ot.STEP_BEFORERENDERTARGETCLEAR_PREPASS = 0; Ot.STEP_AFTERRENDERTARGETDRAW_PREPASS = 0; Ot.STEP_AFTERRENDERTARGETDRAW_LAYER = 1; Ot.STEP_AFTERCAMERADRAW_PREPASS = 0; Ot.STEP_AFTERCAMERADRAW_EFFECTLAYER = 1; Ot.STEP_AFTERCAMERADRAW_LENSFLARESYSTEM = 2; Ot.STEP_AFTERCAMERADRAW_EFFECTLAYER_DRAW = 3; Ot.STEP_AFTERCAMERADRAW_LAYER = 4; Ot.STEP_AFTERCAMERADRAW_FLUIDRENDERER = 5; Ot.STEP_AFTERCAMERAPOSTPROCESS_LAYER = 0; Ot.STEP_AFTERRENDERTARGETPOSTPROCESS_LAYER = 0; Ot.STEP_AFTERRENDER_AUDIO = 0; Ot.STEP_GATHERRENDERTARGETS_DEPTHRENDERER = 0; Ot.STEP_GATHERRENDERTARGETS_GEOMETRYBUFFERRENDERER = 1; Ot.STEP_GATHERRENDERTARGETS_SHADOWGENERATOR = 2; Ot.STEP_GATHERRENDERTARGETS_POSTPROCESSRENDERPIPELINEMANAGER = 3; Ot.STEP_GATHERACTIVECAMERARENDERTARGETS_DEPTHRENDERER = 0; Ot.STEP_GATHERACTIVECAMERARENDERTARGETS_FLUIDRENDERER = 1; Ot.STEP_POINTERMOVE_SPRITE = 0; Ot.STEP_POINTERDOWN_SPRITE = 0; Ot.STEP_POINTERUP_SPRITE = 0; class Y1 extends Array { /** * Hide ctor from the rest of the world. * @param items The items to add. */ constructor(e) { super(...e); } /** * Creates a new Stage. * @returns A new instance of a Stage */ static Create() { return Object.create(Y1.prototype); } /** * Registers a step in an ordered way in the targeted stage. * @param index Defines the position to register the step in * @param component Defines the component attached to the step * @param action Defines the action to launch during the step */ registerStep(e, t, r) { let n = 0, i = Number.MAX_VALUE; for (; n < this.length && (i = this[n].index, !(e < i)); n++) ; this.splice(n, 0, { index: e, component: t, action: r.bind(t) }); } /** * Clears all the steps from the stage. */ clear() { this.length = 0; } } class ir { } ir.POINTERDOWN = 1; ir.POINTERUP = 2; ir.POINTERMOVE = 4; ir.POINTERWHEEL = 8; ir.POINTERPICK = 16; ir.POINTERTAP = 32; ir.POINTERDOUBLETAP = 64; class GN { /** * Instantiates the base class of pointers info. * @param type Defines the type of event (PointerEventTypes) * @param event Defines the related dom event */ constructor(e, t) { this.type = e, this.event = t; } } class K$ extends GN { /** * Instantiates a PointerInfoPre to store pointer related info to the onPrePointerObservable event. * @param type Defines the type of event (PointerEventTypes) * @param event Defines the related dom event * @param localX Defines the local x coordinates of the pointer when the event occured * @param localY Defines the local y coordinates of the pointer when the event occured */ constructor(e, t, r, n) { super(e, t), this.ray = null, this.originalPickingInfo = null, this.skipOnPointerObservable = !1, this.localPosition = new at(r, n); } } class vp extends GN { /** * Defines the picking info associated with this PointerInfo object (if applicable) */ get pickInfo() { return this._pickInfo || this._generatePickInfo(), this._pickInfo; } /** * Instantiates a PointerInfo to store pointer related info to the onPointerObservable event. * @param type Defines the type of event (PointerEventTypes) * @param event Defines the related dom event * @param pickInfo Defines the picking info associated to the info (if any) * @param inputManager Defines the InputManager to use if there is no pickInfo */ constructor(e, t, r, n = null) { super(e, t), this._pickInfo = r, this._inputManager = n; } /** * Generates the picking info if needed */ /** @internal */ _generatePickInfo() { this._inputManager && (this._pickInfo = this._inputManager._pickMove(this.event), this._inputManager._setRayOnPointerInfo(this._pickInfo, this.event), this._inputManager = null); } } class t4 { } t4.KEYDOWN = 1; t4.KEYUP = 2; class MC { /** * Instantiates a new keyboard info. * This class is used to store keyboard related info for the onKeyboardObservable event. * @param type Defines the type of event (KeyboardEventTypes) * @param event Defines the related dom event */ constructor(e, t) { this.type = e, this.event = t; } } class Y5 extends MC { /** * Defines whether the engine should skip the next onKeyboardObservable associated to this pre. * @deprecated use skipOnKeyboardObservable property instead */ get skipOnPointerObservable() { return this.skipOnKeyboardObservable; } set skipOnPointerObservable(e) { this.skipOnKeyboardObservable = e; } /** * Instantiates a new keyboard pre info. * This class is used to store keyboard related info for the onPreKeyboardObservable event. * @param type Defines the type of event (KeyboardEventTypes) * @param event Defines the related dom event */ constructor(e, t) { super(e, t), this.type = e, this.event = t, this.skipOnKeyboardObservable = !1; } } var vn; (function(A) { A[A.Generic = 0] = "Generic", A[A.Keyboard = 1] = "Keyboard", A[A.Mouse = 2] = "Mouse", A[A.Touch = 3] = "Touch", A[A.DualShock = 4] = "DualShock", A[A.Xbox = 5] = "Xbox", A[A.Switch = 6] = "Switch", A[A.DualSense = 7] = "DualSense"; })(vn || (vn = {})); var Yn; (function(A) { A[A.Horizontal = 0] = "Horizontal", A[A.Vertical = 1] = "Vertical", A[A.LeftClick = 2] = "LeftClick", A[A.MiddleClick = 3] = "MiddleClick", A[A.RightClick = 4] = "RightClick", A[A.BrowserBack = 5] = "BrowserBack", A[A.BrowserForward = 6] = "BrowserForward", A[A.MouseWheelX = 7] = "MouseWheelX", A[A.MouseWheelY = 8] = "MouseWheelY", A[A.MouseWheelZ = 9] = "MouseWheelZ", A[A.Move = 12] = "Move"; })(Yn || (Yn = {})); var kI; (function(A) { A[A.Horizontal = 0] = "Horizontal", A[A.Vertical = 1] = "Vertical", A[A.LeftClick = 2] = "LeftClick", A[A.MiddleClick = 3] = "MiddleClick", A[A.RightClick = 4] = "RightClick", A[A.BrowserBack = 5] = "BrowserBack", A[A.BrowserForward = 6] = "BrowserForward", A[A.MouseWheelX = 7] = "MouseWheelX", A[A.MouseWheelY = 8] = "MouseWheelY", A[A.MouseWheelZ = 9] = "MouseWheelZ", A[A.DeltaHorizontal = 10] = "DeltaHorizontal", A[A.DeltaVertical = 11] = "DeltaVertical"; })(kI || (kI = {})); var M5; (function(A) { A[A.Cross = 0] = "Cross", A[A.Circle = 1] = "Circle", A[A.Square = 2] = "Square", A[A.Triangle = 3] = "Triangle", A[A.L1 = 4] = "L1", A[A.R1 = 5] = "R1", A[A.L2 = 6] = "L2", A[A.R2 = 7] = "R2", A[A.Share = 8] = "Share", A[A.Options = 9] = "Options", A[A.L3 = 10] = "L3", A[A.R3 = 11] = "R3", A[A.DPadUp = 12] = "DPadUp", A[A.DPadDown = 13] = "DPadDown", A[A.DPadLeft = 14] = "DPadLeft", A[A.DPadRight = 15] = "DPadRight", A[A.Home = 16] = "Home", A[A.TouchPad = 17] = "TouchPad", A[A.LStickXAxis = 18] = "LStickXAxis", A[A.LStickYAxis = 19] = "LStickYAxis", A[A.RStickXAxis = 20] = "RStickXAxis", A[A.RStickYAxis = 21] = "RStickYAxis"; })(M5 || (M5 = {})); var L5; (function(A) { A[A.Cross = 0] = "Cross", A[A.Circle = 1] = "Circle", A[A.Square = 2] = "Square", A[A.Triangle = 3] = "Triangle", A[A.L1 = 4] = "L1", A[A.R1 = 5] = "R1", A[A.L2 = 6] = "L2", A[A.R2 = 7] = "R2", A[A.Create = 8] = "Create", A[A.Options = 9] = "Options", A[A.L3 = 10] = "L3", A[A.R3 = 11] = "R3", A[A.DPadUp = 12] = "DPadUp", A[A.DPadDown = 13] = "DPadDown", A[A.DPadLeft = 14] = "DPadLeft", A[A.DPadRight = 15] = "DPadRight", A[A.Home = 16] = "Home", A[A.TouchPad = 17] = "TouchPad", A[A.LStickXAxis = 18] = "LStickXAxis", A[A.LStickYAxis = 19] = "LStickYAxis", A[A.RStickXAxis = 20] = "RStickXAxis", A[A.RStickYAxis = 21] = "RStickYAxis"; })(L5 || (L5 = {})); var K5; (function(A) { A[A.A = 0] = "A", A[A.B = 1] = "B", A[A.X = 2] = "X", A[A.Y = 3] = "Y", A[A.LB = 4] = "LB", A[A.RB = 5] = "RB", A[A.LT = 6] = "LT", A[A.RT = 7] = "RT", A[A.Back = 8] = "Back", A[A.Start = 9] = "Start", A[A.LS = 10] = "LS", A[A.RS = 11] = "RS", A[A.DPadUp = 12] = "DPadUp", A[A.DPadDown = 13] = "DPadDown", A[A.DPadLeft = 14] = "DPadLeft", A[A.DPadRight = 15] = "DPadRight", A[A.Home = 16] = "Home", A[A.LStickXAxis = 17] = "LStickXAxis", A[A.LStickYAxis = 18] = "LStickYAxis", A[A.RStickXAxis = 19] = "RStickXAxis", A[A.RStickYAxis = 20] = "RStickYAxis"; })(K5 || (K5 = {})); var J5; (function(A) { A[A.B = 0] = "B", A[A.A = 1] = "A", A[A.Y = 2] = "Y", A[A.X = 3] = "X", A[A.L = 4] = "L", A[A.R = 5] = "R", A[A.ZL = 6] = "ZL", A[A.ZR = 7] = "ZR", A[A.Minus = 8] = "Minus", A[A.Plus = 9] = "Plus", A[A.LS = 10] = "LS", A[A.RS = 11] = "RS", A[A.DPadUp = 12] = "DPadUp", A[A.DPadDown = 13] = "DPadDown", A[A.DPadLeft = 14] = "DPadLeft", A[A.DPadRight = 15] = "DPadRight", A[A.Home = 16] = "Home", A[A.Capture = 17] = "Capture", A[A.LStickXAxis = 18] = "LStickXAxis", A[A.LStickYAxis = 19] = "LStickYAxis", A[A.RStickXAxis = 20] = "RStickXAxis", A[A.RStickYAxis = 21] = "RStickYAxis"; })(J5 || (J5 = {})); var z5; (function(A) { A[A.PointerMove = 0] = "PointerMove", A[A.PointerDown = 1] = "PointerDown", A[A.PointerUp = 2] = "PointerUp"; })(z5 || (z5 = {})); class wm { } wm.DOM_DELTA_PIXEL = 0; wm.DOM_DELTA_LINE = 1; wm.DOM_DELTA_PAGE = 2; class Fw { /** * Create device input events based on provided type and slot * * @param deviceType Type of device * @param deviceSlot "Slot" or index that device is referenced in * @param inputIndex Id of input to be checked * @param currentState Current value for given input * @param deviceInputSystem Reference to DeviceInputSystem * @param elementToAttachTo HTMLElement to reference as target for inputs * @returns IUIEvent object */ static CreateDeviceEvent(e, t, r, n, i, s, a) { switch (e) { case vn.Keyboard: return this._CreateKeyboardEvent(r, n, i, s); case vn.Mouse: if (r === Yn.MouseWheelX || r === Yn.MouseWheelY || r === Yn.MouseWheelZ) return this._CreateWheelEvent(e, t, r, n, i, s); case vn.Touch: return this._CreatePointerEvent(e, t, r, n, i, s, a); default: throw `Unable to generate event for device ${vn[e]}`; } } /** * Creates pointer event * * @param deviceType Type of device * @param deviceSlot "Slot" or index that device is referenced in * @param inputIndex Id of input to be checked * @param currentState Current value for given input * @param deviceInputSystem Reference to DeviceInputSystem * @param elementToAttachTo HTMLElement to reference as target for inputs * @returns IUIEvent object (Pointer) */ static _CreatePointerEvent(e, t, r, n, i, s, a) { const f = this._CreateMouseEvent(e, t, r, n, i, s); e === vn.Mouse ? (f.deviceType = vn.Mouse, f.pointerId = 1, f.pointerType = "mouse") : (f.deviceType = vn.Touch, f.pointerId = a ?? t, f.pointerType = "touch"); let o = 0; return o += i.pollInput(e, t, Yn.LeftClick), o += i.pollInput(e, t, Yn.RightClick) * 2, o += i.pollInput(e, t, Yn.MiddleClick) * 4, f.buttons = o, r === Yn.Move ? f.type = "pointermove" : r >= Yn.LeftClick && r <= Yn.RightClick && (f.type = n === 1 ? "pointerdown" : "pointerup", f.button = r - 2), f; } /** * Create Mouse Wheel Event * @param deviceType Type of device * @param deviceSlot "Slot" or index that device is referenced in * @param inputIndex Id of input to be checked * @param currentState Current value for given input * @param deviceInputSystem Reference to DeviceInputSystem * @param elementToAttachTo HTMLElement to reference as target for inputs * @returns IUIEvent object (Wheel) */ static _CreateWheelEvent(e, t, r, n, i, s) { const a = this._CreateMouseEvent(e, t, r, n, i, s); switch (a.pointerId = 1, a.type = "wheel", a.deltaMode = wm.DOM_DELTA_PIXEL, a.deltaX = 0, a.deltaY = 0, a.deltaZ = 0, r) { case Yn.MouseWheelX: a.deltaX = n; break; case Yn.MouseWheelY: a.deltaY = n; break; case Yn.MouseWheelZ: a.deltaZ = n; break; } return a; } /** * Create Mouse Event * @param deviceType Type of device * @param deviceSlot "Slot" or index that device is referenced in * @param inputIndex Id of input to be checked * @param currentState Current value for given input * @param deviceInputSystem Reference to DeviceInputSystem * @param elementToAttachTo HTMLElement to reference as target for inputs * @returns IUIEvent object (Mouse) */ static _CreateMouseEvent(e, t, r, n, i, s) { const a = this._CreateEvent(s), f = i.pollInput(e, t, Yn.Horizontal), o = i.pollInput(e, t, Yn.Vertical); return s ? (a.movementX = 0, a.movementY = 0, a.offsetX = a.movementX - s.getBoundingClientRect().x, a.offsetY = a.movementY - s.getBoundingClientRect().y) : (a.movementX = i.pollInput(e, t, kI.DeltaHorizontal), a.movementY = i.pollInput(e, t, kI.DeltaVertical), a.offsetX = 0, a.offsetY = 0), this._CheckNonCharacterKeys(a, i), a.clientX = f, a.clientY = o, a.x = f, a.y = o, a.deviceType = e, a.deviceSlot = t, a.inputIndex = r, a; } /** * Create Keyboard Event * @param inputIndex Id of input to be checked * @param currentState Current value for given input * @param deviceInputSystem Reference to DeviceInputSystem * @param elementToAttachTo HTMLElement to reference as target for inputs * @returns IEvent object (Keyboard) */ static _CreateKeyboardEvent(e, t, r, n) { const i = this._CreateEvent(n); return this._CheckNonCharacterKeys(i, r), i.deviceType = vn.Keyboard, i.deviceSlot = 0, i.inputIndex = e, i.type = t === 1 ? "keydown" : "keyup", i.key = String.fromCharCode(e), i.keyCode = e, i; } /** * Add parameters for non-character keys (Ctrl, Alt, Meta, Shift) * @param evt Event object to add parameters to * @param deviceInputSystem DeviceInputSystem to pull values from */ static _CheckNonCharacterKeys(e, t) { const r = t.isDeviceAvailable(vn.Keyboard), n = r && t.pollInput(vn.Keyboard, 0, 18) === 1, i = r && t.pollInput(vn.Keyboard, 0, 17) === 1, s = r && (t.pollInput(vn.Keyboard, 0, 91) === 1 || t.pollInput(vn.Keyboard, 0, 92) === 1 || t.pollInput(vn.Keyboard, 0, 93) === 1), a = r && t.pollInput(vn.Keyboard, 0, 16) === 1; e.altKey = n, e.ctrlKey = i, e.metaKey = s, e.shiftKey = a; } /** * Create base event object * @param elementToAttachTo Value to use as event target * @returns */ static _CreateEvent(e) { const t = {}; return t.preventDefault = () => { }, t.target = e, t; } } class Z9e { constructor(e, t, r) { this._nativeInput = _native.DeviceInputSystem ? new _native.DeviceInputSystem(e, t, (n, i, s, a) => { const f = Fw.CreateDeviceEvent(n, i, s, a, this); r(n, i, f); }) : this._createDummyNativeInput(); } // Public functions /** * Checks for current device input value, given an id and input index. Throws exception if requested device not initialized. * @param deviceType Enum specifying device type * @param deviceSlot "Slot" or index that device is referenced in * @param inputIndex Id of input to be checked * @returns Current value of input */ pollInput(e, t, r) { return this._nativeInput.pollInput(e, t, r); } /** * Check for a specific device in the DeviceInputSystem * @param deviceType Type of device to check for * @returns bool with status of device's existence */ isDeviceAvailable(e) { return e === vn.Mouse || e === vn.Touch; } /** * Dispose of all the observables */ dispose() { this._nativeInput.dispose(); } /** * For versions of BabylonNative that don't have the NativeInput plugin initialized, create a dummy version * @returns Object with dummy functions */ _createDummyNativeInput() { return { pollInput: () => 0, isDeviceAvailable: () => !1, dispose: () => { } }; } } const AG = 255, dG = Object.keys(Yn).length / 2; class _9e { /** * Constructor for the WebDeviceInputSystem * @param engine Engine to reference * @param onDeviceConnected Callback to execute when device is connected * @param onDeviceDisconnected Callback to execute when device is disconnected * @param onInputChanged Callback to execute when input changes on device */ constructor(e, t, r, n) { this._inputs = [], this._keyboardActive = !1, this._pointerActive = !1, this._usingSafari = ye.IsSafari(), this._usingMacOS = Mw() && /(Mac|iPhone|iPod|iPad)/i.test(navigator.platform), this._keyboardDownEvent = (i) => { }, this._keyboardUpEvent = (i) => { }, this._keyboardBlurEvent = (i) => { }, this._pointerMoveEvent = (i) => { }, this._pointerDownEvent = (i) => { }, this._pointerUpEvent = (i) => { }, this._pointerCancelEvent = (i) => { }, this._pointerWheelEvent = (i) => { }, this._pointerBlurEvent = (i) => { }, this._pointerMacOSChromeOutEvent = (i) => { }, this._eventsAttached = !1, this._mouseId = -1, this._isUsingFirefox = Mw() && navigator.userAgent && navigator.userAgent.indexOf("Firefox") !== -1, this._isUsingChromium = Mw() && navigator.userAgent && navigator.userAgent.indexOf("Chrome") !== -1, this._maxTouchPoints = 0, this._pointerInputClearObserver = null, this._gamepadConnectedEvent = (i) => { }, this._gamepadDisconnectedEvent = (i) => { }, this._eventPrefix = ye.GetPointerPrefix(e), this._engine = e, this._onDeviceConnected = t, this._onDeviceDisconnected = r, this._onInputChanged = n, this._mouseId = this._isUsingFirefox ? 0 : 1, this._enableEvents(), this._usingMacOS && (this._metaKeys = []), this._engine._onEngineViewChanged || (this._engine._onEngineViewChanged = () => { this._enableEvents(); }); } // Public functions /** * Checks for current device input value, given an id and input index. Throws exception if requested device not initialized. * @param deviceType Enum specifying device type * @param deviceSlot "Slot" or index that device is referenced in * @param inputIndex Id of input to be checked * @returns Current value of input */ pollInput(e, t, r) { const n = this._inputs[e][t]; if (!n) throw `Unable to find device ${vn[e]}`; e >= vn.DualShock && e <= vn.DualSense && this._updateDevice(e, t, r); const i = n[r]; if (i === void 0) throw `Unable to find input ${r} for device ${vn[e]} in slot ${t}`; return r === Yn.Move && ye.Warn("Unable to provide information for PointerInput.Move. Try using PointerInput.Horizontal or PointerInput.Vertical for move data."), i; } /** * Check for a specific device in the DeviceInputSystem * @param deviceType Type of device to check for * @returns bool with status of device's existence */ isDeviceAvailable(e) { return this._inputs[e] !== void 0; } /** * Dispose of all the eventlisteners */ dispose() { this._onDeviceConnected = () => { }, this._onDeviceDisconnected = () => { }, this._onInputChanged = () => { }, delete this._engine._onEngineViewChanged, this._elementToAttachTo && this._disableEvents(); } /** * Enable listening for user input events */ _enableEvents() { const e = this === null || this === void 0 ? void 0 : this._engine.getInputElement(); if (e && (!this._eventsAttached || this._elementToAttachTo !== e)) { if (this._disableEvents(), this._inputs) { for (const t of this._inputs) if (t) for (const r in t) { const n = +r, i = t[n]; if (i) for (let s = 0; s < i.length; s++) i[s] = 0; } } this._elementToAttachTo = e, this._elementToAttachTo.tabIndex = this._elementToAttachTo.tabIndex !== -1 ? this._elementToAttachTo.tabIndex : this._engine.canvasTabIndex, this._handleKeyActions(), this._handlePointerActions(), this._handleGamepadActions(), this._eventsAttached = !0, this._checkForConnectedDevices(); } } /** * Disable listening for user input events */ _disableEvents() { this._elementToAttachTo && (this._elementToAttachTo.removeEventListener("blur", this._keyboardBlurEvent), this._elementToAttachTo.removeEventListener("blur", this._pointerBlurEvent), this._elementToAttachTo.removeEventListener("keydown", this._keyboardDownEvent), this._elementToAttachTo.removeEventListener("keyup", this._keyboardUpEvent), this._elementToAttachTo.removeEventListener(this._eventPrefix + "move", this._pointerMoveEvent), this._elementToAttachTo.removeEventListener(this._eventPrefix + "down", this._pointerDownEvent), this._elementToAttachTo.removeEventListener(this._eventPrefix + "up", this._pointerUpEvent), this._elementToAttachTo.removeEventListener(this._eventPrefix + "cancel", this._pointerCancelEvent), this._elementToAttachTo.removeEventListener(this._wheelEventName, this._pointerWheelEvent), this._usingMacOS && this._isUsingChromium && this._elementToAttachTo.removeEventListener("lostpointercapture", this._pointerMacOSChromeOutEvent), window.removeEventListener("gamepadconnected", this._gamepadConnectedEvent), window.removeEventListener("gamepaddisconnected", this._gamepadDisconnectedEvent)), this._pointerInputClearObserver && this._engine.onEndFrameObservable.remove(this._pointerInputClearObserver), this._eventsAttached = !1; } /** * Checks for existing connections to devices and register them, if necessary * Currently handles gamepads and mouse */ _checkForConnectedDevices() { if (navigator.getGamepads) { const e = navigator.getGamepads(); for (const t of e) t && this._addGamePad(t); } typeof matchMedia == "function" && matchMedia("(pointer:fine)").matches && this._addPointerDevice(vn.Mouse, 0, 0, 0); } // Private functions /** * Add a gamepad to the DeviceInputSystem * @param gamepad A single DOM Gamepad object */ _addGamePad(e) { const t = this._getGamepadDeviceType(e.id), r = e.index; this._gamepads = this._gamepads || new Array(e.index + 1), this._registerDevice(t, r, e.buttons.length + e.axes.length), this._gamepads[r] = t; } /** * Add pointer device to DeviceInputSystem * @param deviceType Type of Pointer to add * @param deviceSlot Pointer ID (0 for mouse, pointerId for Touch) * @param currentX Current X at point of adding * @param currentY Current Y at point of adding */ _addPointerDevice(e, t, r, n) { this._pointerActive || (this._pointerActive = !0), this._registerDevice(e, t, dG); const i = this._inputs[e][t]; i[0] = r, i[1] = n; } /** * Add device and inputs to device array * @param deviceType Enum specifying device type * @param deviceSlot "Slot" or index that device is referenced in * @param numberOfInputs Number of input entries to create for given device */ _registerDevice(e, t, r) { if (t === void 0) throw `Unable to register device ${vn[e]} to undefined slot.`; if (this._inputs[e] || (this._inputs[e] = {}), !this._inputs[e][t]) { const n = new Array(r); n.fill(0), this._inputs[e][t] = n, this._onDeviceConnected(e, t); } } /** * Given a specific device name, remove that device from the device map * @param deviceType Enum specifying device type * @param deviceSlot "Slot" or index that device is referenced in */ _unregisterDevice(e, t) { this._inputs[e][t] && (delete this._inputs[e][t], this._onDeviceDisconnected(e, t)); } /** * Handle all actions that come from keyboard interaction */ _handleKeyActions() { this._keyboardDownEvent = (e) => { this._keyboardActive || (this._keyboardActive = !0, this._registerDevice(vn.Keyboard, 0, AG)); const t = this._inputs[vn.Keyboard][0]; if (t) { t[e.keyCode] = 1; const r = e; r.inputIndex = e.keyCode, this._usingMacOS && e.metaKey && e.key !== "Meta" && (this._metaKeys.includes(e.keyCode) || this._metaKeys.push(e.keyCode)), this._onInputChanged(vn.Keyboard, 0, r); } }, this._keyboardUpEvent = (e) => { this._keyboardActive || (this._keyboardActive = !0, this._registerDevice(vn.Keyboard, 0, AG)); const t = this._inputs[vn.Keyboard][0]; if (t) { t[e.keyCode] = 0; const r = e; if (r.inputIndex = e.keyCode, this._usingMacOS && e.key === "Meta" && this._metaKeys.length > 0) { for (const n of this._metaKeys) { const i = Fw.CreateDeviceEvent(vn.Keyboard, 0, n, 0, this, this._elementToAttachTo); t[n] = 0, this._onInputChanged(vn.Keyboard, 0, i); } this._metaKeys.splice(0, this._metaKeys.length); } this._onInputChanged(vn.Keyboard, 0, r); } }, this._keyboardBlurEvent = () => { if (this._keyboardActive) { const e = this._inputs[vn.Keyboard][0]; for (let t = 0; t < e.length; t++) if (e[t] !== 0) { e[t] = 0; const r = Fw.CreateDeviceEvent(vn.Keyboard, 0, t, 0, this, this._elementToAttachTo); this._onInputChanged(vn.Keyboard, 0, r); } this._usingMacOS && this._metaKeys.splice(0, this._metaKeys.length); } }, this._elementToAttachTo.addEventListener("keydown", this._keyboardDownEvent), this._elementToAttachTo.addEventListener("keyup", this._keyboardUpEvent), this._elementToAttachTo.addEventListener("blur", this._keyboardBlurEvent); } /** * Handle all actions that come from pointer interaction */ _handlePointerActions() { this._maxTouchPoints = Mw() && navigator.maxTouchPoints || 2, this._activeTouchIds || (this._activeTouchIds = new Array(this._maxTouchPoints)); for (let r = 0; r < this._maxTouchPoints; r++) this._activeTouchIds[r] = -1; this._pointerMoveEvent = (r) => { const n = this._getPointerType(r); let i = n === vn.Mouse ? 0 : this._activeTouchIds.indexOf(r.pointerId); if (n === vn.Touch && i === -1) { const a = this._activeTouchIds.indexOf(-1); if (a >= 0) i = a, this._activeTouchIds[a] = r.pointerId, this._onDeviceConnected(n, i); else { ye.Warn(`Max number of touches exceeded. Ignoring touches in excess of ${this._maxTouchPoints}`); return; } } this._inputs[n] || (this._inputs[n] = {}), this._inputs[n][i] || this._addPointerDevice(n, i, r.clientX, r.clientY); const s = this._inputs[n][i]; if (s) { const a = r; a.inputIndex = Yn.Move, s[Yn.Horizontal] = r.clientX, s[Yn.Vertical] = r.clientY, n === vn.Touch && s[Yn.LeftClick] === 0 && (s[Yn.LeftClick] = 1), r.pointerId === void 0 && (r.pointerId = this._mouseId), this._onInputChanged(n, i, a), !this._usingSafari && r.button !== -1 && (a.inputIndex = r.button + 2, s[r.button + 2] = s[r.button + 2] ? 0 : 1, this._onInputChanged(n, i, a)); } }, this._pointerDownEvent = (r) => { const n = this._getPointerType(r); let i = n === vn.Mouse ? 0 : r.pointerId; if (n === vn.Touch) { const a = this._activeTouchIds.indexOf(-1); if (a >= 0) i = a, this._activeTouchIds[a] = r.pointerId; else { ye.Warn(`Max number of touches exceeded. Ignoring touches in excess of ${this._maxTouchPoints}`); return; } } this._inputs[n] || (this._inputs[n] = {}), this._inputs[n][i] ? n === vn.Touch && this._onDeviceConnected(n, i) : this._addPointerDevice(n, i, r.clientX, r.clientY); const s = this._inputs[n][i]; if (s) { const a = s[Yn.Horizontal], f = s[Yn.Vertical]; if (n === vn.Mouse) { if (r.pointerId === void 0 && (r.pointerId = this._mouseId), !document.pointerLockElement) try { this._elementToAttachTo.setPointerCapture(this._mouseId); } catch { } } else if (r.pointerId && !document.pointerLockElement) try { this._elementToAttachTo.setPointerCapture(r.pointerId); } catch { } s[Yn.Horizontal] = r.clientX, s[Yn.Vertical] = r.clientY, s[r.button + 2] = 1; const o = r; o.inputIndex = r.button + 2, this._onInputChanged(n, i, o), (a !== r.clientX || f !== r.clientY) && (o.inputIndex = Yn.Move, this._onInputChanged(n, i, o)); } }, this._pointerUpEvent = (r) => { var n, i, s, a, f; const o = this._getPointerType(r), d = o === vn.Mouse ? 0 : this._activeTouchIds.indexOf(r.pointerId); if (o === vn.Touch) { if (d === -1) return; this._activeTouchIds[d] = -1; } const v = (n = this._inputs[o]) === null || n === void 0 ? void 0 : n[d]; if (v && v[r.button + 2] !== 0) { const u = v[Yn.Horizontal], l = v[Yn.Vertical]; v[Yn.Horizontal] = r.clientX, v[Yn.Vertical] = r.clientY, v[r.button + 2] = 0; const P = r; r.pointerId === void 0 && (r.pointerId = this._mouseId), (u !== r.clientX || l !== r.clientY) && (P.inputIndex = Yn.Move, this._onInputChanged(o, d, P)), P.inputIndex = r.button + 2, o === vn.Mouse && this._mouseId >= 0 && (!((s = (i = this._elementToAttachTo).hasPointerCapture) === null || s === void 0) && s.call(i, this._mouseId)) ? this._elementToAttachTo.releasePointerCapture(this._mouseId) : r.pointerId && (!((f = (a = this._elementToAttachTo).hasPointerCapture) === null || f === void 0) && f.call(a, r.pointerId)) && this._elementToAttachTo.releasePointerCapture(r.pointerId), this._onInputChanged(o, d, P), o === vn.Touch && this._onDeviceDisconnected(o, d); } }, this._pointerCancelEvent = (r) => { var n, i, s, a; if (r.pointerType === "mouse") { const f = this._inputs[vn.Mouse][0]; this._mouseId >= 0 && (!((i = (n = this._elementToAttachTo).hasPointerCapture) === null || i === void 0) && i.call(n, this._mouseId)) && this._elementToAttachTo.releasePointerCapture(this._mouseId); for (let o = Yn.LeftClick; o <= Yn.BrowserForward; o++) if (f[o] === 1) { f[o] = 0; const d = Fw.CreateDeviceEvent(vn.Mouse, 0, o, 0, this, this._elementToAttachTo); this._onInputChanged(vn.Mouse, 0, d); } } else { const f = this._activeTouchIds.indexOf(r.pointerId); if (f === -1) return; !((a = (s = this._elementToAttachTo).hasPointerCapture) === null || a === void 0) && a.call(s, r.pointerId) && this._elementToAttachTo.releasePointerCapture(r.pointerId), this._inputs[vn.Touch][f][Yn.LeftClick] = 0; const o = Fw.CreateDeviceEvent(vn.Touch, f, Yn.LeftClick, 0, this, this._elementToAttachTo, r.pointerId); this._onInputChanged(vn.Touch, f, o), this._activeTouchIds[f] = -1, this._onDeviceDisconnected(vn.Touch, f); } }, this._wheelEventName = "onwheel" in document.createElement("div") ? "wheel" : document.onmousewheel !== void 0 ? "mousewheel" : "DOMMouseScroll"; let e = !1; const t = function() { }; try { const r = Object.defineProperty({}, "passive", { get: function() { e = !0; } }); this._elementToAttachTo.addEventListener("test", t, r), this._elementToAttachTo.removeEventListener("test", t, r); } catch { } this._pointerBlurEvent = () => { var r, n, i, s, a; if (this.isDeviceAvailable(vn.Mouse)) { const f = this._inputs[vn.Mouse][0]; this._mouseId >= 0 && (!((n = (r = this._elementToAttachTo).hasPointerCapture) === null || n === void 0) && n.call(r, this._mouseId)) && this._elementToAttachTo.releasePointerCapture(this._mouseId); for (let o = Yn.LeftClick; o <= Yn.BrowserForward; o++) if (f[o] === 1) { f[o] = 0; const d = Fw.CreateDeviceEvent(vn.Mouse, 0, o, 0, this, this._elementToAttachTo); this._onInputChanged(vn.Mouse, 0, d); } } if (this.isDeviceAvailable(vn.Touch)) { const f = this._inputs[vn.Touch]; for (let o = 0; o < this._activeTouchIds.length; o++) { const d = this._activeTouchIds[o]; if (!((s = (i = this._elementToAttachTo).hasPointerCapture) === null || s === void 0) && s.call(i, d) && this._elementToAttachTo.releasePointerCapture(d), d !== -1 && ((a = f[o]) === null || a === void 0 ? void 0 : a[Yn.LeftClick]) === 1) { f[o][Yn.LeftClick] = 0; const v = Fw.CreateDeviceEvent(vn.Touch, o, Yn.LeftClick, 0, this, this._elementToAttachTo, d); this._onInputChanged(vn.Touch, o, v), this._activeTouchIds[o] = -1, this._onDeviceDisconnected(vn.Touch, o); } } } }, this._pointerWheelEvent = (r) => { const n = vn.Mouse, i = 0; this._inputs[n] || (this._inputs[n] = []), this._inputs[n][i] || (this._pointerActive = !0, this._registerDevice(n, i, dG)); const s = this._inputs[n][i]; if (s) { s[Yn.MouseWheelX] = r.deltaX || 0, s[Yn.MouseWheelY] = r.deltaY || r.wheelDelta || 0, s[Yn.MouseWheelZ] = r.deltaZ || 0; const a = r; r.pointerId === void 0 && (r.pointerId = this._mouseId), s[Yn.MouseWheelX] !== 0 && (a.inputIndex = Yn.MouseWheelX, this._onInputChanged(n, i, a)), s[Yn.MouseWheelY] !== 0 && (a.inputIndex = Yn.MouseWheelY, this._onInputChanged(n, i, a)), s[Yn.MouseWheelZ] !== 0 && (a.inputIndex = Yn.MouseWheelZ, this._onInputChanged(n, i, a)); } }, this._usingMacOS && this._isUsingChromium && (this._pointerMacOSChromeOutEvent = (r) => { r.buttons > 1 && this._pointerCancelEvent(r); }, this._elementToAttachTo.addEventListener("lostpointercapture", this._pointerMacOSChromeOutEvent)), this._elementToAttachTo.addEventListener(this._eventPrefix + "move", this._pointerMoveEvent), this._elementToAttachTo.addEventListener(this._eventPrefix + "down", this._pointerDownEvent), this._elementToAttachTo.addEventListener(this._eventPrefix + "up", this._pointerUpEvent), this._elementToAttachTo.addEventListener(this._eventPrefix + "cancel", this._pointerCancelEvent), this._elementToAttachTo.addEventListener("blur", this._pointerBlurEvent), this._elementToAttachTo.addEventListener(this._wheelEventName, this._pointerWheelEvent, e ? { passive: !1 } : !1), this._pointerInputClearObserver = this._engine.onEndFrameObservable.add(() => { if (this.isDeviceAvailable(vn.Mouse)) { const r = this._inputs[vn.Mouse][0]; r[Yn.MouseWheelX] = 0, r[Yn.MouseWheelY] = 0, r[Yn.MouseWheelZ] = 0; } }); } /** * Handle all actions that come from gamepad interaction */ _handleGamepadActions() { this._gamepadConnectedEvent = (e) => { this._addGamePad(e.gamepad); }, this._gamepadDisconnectedEvent = (e) => { if (this._gamepads) { const t = this._getGamepadDeviceType(e.gamepad.id), r = e.gamepad.index; this._unregisterDevice(t, r), delete this._gamepads[r]; } }, window.addEventListener("gamepadconnected", this._gamepadConnectedEvent), window.addEventListener("gamepaddisconnected", this._gamepadDisconnectedEvent); } /** * Update all non-event based devices with each frame * @param deviceType Enum specifying device type * @param deviceSlot "Slot" or index that device is referenced in * @param inputIndex Id of input to be checked */ _updateDevice(e, t, r) { const n = navigator.getGamepads()[t]; if (n && e === this._gamepads[t]) { const i = this._inputs[e][t]; r >= n.buttons.length ? i[r] = n.axes[r - n.buttons.length].valueOf() : i[r] = n.buttons[r].value; } } /** * Gets DeviceType from the device name * @param deviceName Name of Device from DeviceInputSystem * @returns DeviceType enum value */ _getGamepadDeviceType(e) { return e.indexOf("054c") !== -1 ? e.indexOf("0ce6") !== -1 ? vn.DualSense : vn.DualShock : e.indexOf("Xbox One") !== -1 || e.search("Xbox 360") !== -1 || e.search("xinput") !== -1 ? vn.Xbox : e.indexOf("057e") !== -1 ? vn.Switch : vn.Generic; } /** * Get DeviceType from a given pointer/mouse/touch event. * @param evt PointerEvent to evaluate * @returns DeviceType interpreted from event */ _getPointerType(e) { let t = vn.Mouse; return (e.pointerType === "touch" || e.pointerType === "pen" || e.touches) && (t = vn.Touch), t; } } class G5 { /** * Default Constructor * @param deviceInputSystem - Reference to DeviceInputSystem * @param deviceType - Type of device * @param deviceSlot - "Slot" or index that device is referenced in */ constructor(e, t, r = 0) { this.deviceType = t, this.deviceSlot = r, this.onInputChangedObservable = new Oe(), this._deviceInputSystem = e; } /** * Get input for specific input * @param inputIndex - index of specific input on device * @returns Input value from DeviceInputSystem */ getInput(e) { return this._deviceInputSystem.pollInput(this.deviceType, this.deviceSlot, e); } } class $9e { constructor(e) { this._registeredManagers = new Array(), this._refCount = 0, this.registerManager = (s) => { for (let a = 0; a < this._devices.length; a++) { const f = this._devices[a]; for (const o in f) { const d = +o; s._addDevice(new G5(this._deviceInputSystem, a, d)); } } this._registeredManagers.push(s); }, this.unregisterManager = (s) => { const a = this._registeredManagers.indexOf(s); a > -1 && this._registeredManagers.splice(a, 1); }; const t = Object.keys(vn).length / 2; this._devices = new Array(t); const r = (s, a) => { this._devices[s] || (this._devices[s] = new Array()), this._devices[s][a] || (this._devices[s][a] = a); for (const f of this._registeredManagers) { const o = new G5(this._deviceInputSystem, s, a); f._addDevice(o); } }, n = (s, a) => { var f; !((f = this._devices[s]) === null || f === void 0) && f[a] && delete this._devices[s][a]; for (const o of this._registeredManagers) o._removeDevice(s, a); }, i = (s, a, f) => { if (f) for (const o of this._registeredManagers) o._onInputChanged(s, a, f); }; typeof _native < "u" ? this._deviceInputSystem = new Z9e(r, n, i) : this._deviceInputSystem = new _9e(e, r, n, i); } dispose() { this._deviceInputSystem.dispose(); } } class J$ { // Public Functions /** * Gets a DeviceSource, given a type and slot * @param deviceType - Type of Device * @param deviceSlot - Slot or ID of device * @returns DeviceSource */ getDeviceSource(e, t) { if (t === void 0) { if (this._firstDevice[e] === void 0) return null; t = this._firstDevice[e]; } return !this._devices[e] || this._devices[e][t] === void 0 ? null : this._devices[e][t]; } /** * Gets an array of DeviceSource objects for a given device type * @param deviceType - Type of Device * @returns All available DeviceSources of a given type */ getDeviceSources(e) { return this._devices[e] ? this._devices[e].filter((t) => !!t) : []; } /** * Default constructor * @param engine - Used to get canvas (if applicable) */ constructor(e) { const t = Object.keys(vn).length / 2; this._devices = new Array(t), this._firstDevice = new Array(t), this._engine = e, this._engine._deviceSourceManager || (this._engine._deviceSourceManager = new $9e(e)), this._engine._deviceSourceManager._refCount++, this.onDeviceConnectedObservable = new Oe((r) => { for (const n of this._devices) if (n) for (const i of n) i && this.onDeviceConnectedObservable.notifyObserver(r, i); }), this.onDeviceDisconnectedObservable = new Oe(), this._engine._deviceSourceManager.registerManager(this), this._onDisposeObserver = e.onDisposeObservable.add(() => { this.dispose(); }); } /** * Dispose of DeviceSourceManager */ dispose() { this.onDeviceConnectedObservable.clear(), this.onDeviceDisconnectedObservable.clear(), this._engine._deviceSourceManager && (this._engine._deviceSourceManager.unregisterManager(this), --this._engine._deviceSourceManager._refCount < 1 && (this._engine._deviceSourceManager.dispose(), delete this._engine._deviceSourceManager)), this._engine.onDisposeObservable.remove(this._onDisposeObserver); } // Hidden Functions /** * @param deviceSource - Source to add * @internal */ _addDevice(e) { this._devices[e.deviceType] || (this._devices[e.deviceType] = new Array()), this._devices[e.deviceType][e.deviceSlot] || (this._devices[e.deviceType][e.deviceSlot] = e, this._updateFirstDevices(e.deviceType)), this.onDeviceConnectedObservable.notifyObservers(e); } /** * @param deviceType - DeviceType * @param deviceSlot - DeviceSlot * @internal */ _removeDevice(e, t) { var r, n; const i = (r = this._devices[e]) === null || r === void 0 ? void 0 : r[t]; this.onDeviceDisconnectedObservable.notifyObservers(i), !((n = this._devices[e]) === null || n === void 0) && n[t] && delete this._devices[e][t], this._updateFirstDevices(e); } /** * @param deviceType - DeviceType * @param deviceSlot - DeviceSlot * @param eventData - Event * @internal */ _onInputChanged(e, t, r) { var n, i; (i = (n = this._devices[e]) === null || n === void 0 ? void 0 : n[t]) === null || i === void 0 || i.onInputChangedObservable.notifyObservers(r); } // Private Functions _updateFirstDevices(e) { switch (e) { case vn.Keyboard: case vn.Mouse: this._firstDevice[e] = 0; break; case vn.Touch: case vn.DualSense: case vn.DualShock: case vn.Xbox: case vn.Switch: case vn.Generic: { delete this._firstDevice[e]; const t = this._devices[e]; if (t) { for (let r = 0; r < t.length; r++) if (t[r]) { this._firstDevice[e] = r; break; } } break; } } } } class vG { constructor() { this._singleClick = !1, this._doubleClick = !1, this._hasSwiped = !1, this._ignore = !1; } get singleClick() { return this._singleClick; } get doubleClick() { return this._doubleClick; } get hasSwiped() { return this._hasSwiped; } get ignore() { return this._ignore; } set singleClick(e) { this._singleClick = e; } set doubleClick(e) { this._doubleClick = e; } set hasSwiped(e) { this._hasSwiped = e; } set ignore(e) { this._ignore = e; } } class Xf { /** * Creates a new InputManager * @param scene - defines the hosting scene */ constructor(e) { this._alreadyAttached = !1, this._meshPickProceed = !1, this._currentPickResult = null, this._previousPickResult = null, this._totalPointersPressed = 0, this._doubleClickOccured = !1, this._isSwiping = !1, this._swipeButtonPressed = -1, this._skipPointerTap = !1, this._isMultiTouchGesture = !1, this._pointerX = 0, this._pointerY = 0, this._startingPointerPosition = new at(0, 0), this._previousStartingPointerPosition = new at(0, 0), this._startingPointerTime = 0, this._previousStartingPointerTime = 0, this._pointerCaptures = {}, this._meshUnderPointerId = {}, this._movePointerInfo = null, this._cameraObserverCount = 0, this._delayedClicks = [null, null, null, null, null], this._deviceSourceManager = null, this._scene = e || gr.LastCreatedScene, this._scene; } /** * Gets the mesh that is currently under the pointer * @returns Mesh that the pointer is pointer is hovering over */ get meshUnderPointer() { return this._movePointerInfo && (this._movePointerInfo._generatePickInfo(), this._movePointerInfo = null), this._pointerOverMesh; } /** * When using more than one pointer (for example in XR) you can get the mesh under the specific pointer * @param pointerId - the pointer id to use * @returns The mesh under this pointer id or null if not found */ getMeshUnderPointerByPointerId(e) { return this._meshUnderPointerId[e] || null; } /** * Gets the pointer coordinates in 2D without any translation (ie. straight out of the pointer event) * @returns Vector with X/Y values directly from pointer event */ get unTranslatedPointer() { return new at(this._unTranslatedPointerX, this._unTranslatedPointerY); } /** * Gets or sets the current on-screen X position of the pointer * @returns Translated X with respect to screen */ get pointerX() { return this._pointerX; } set pointerX(e) { this._pointerX = e; } /** * Gets or sets the current on-screen Y position of the pointer * @returns Translated Y with respect to screen */ get pointerY() { return this._pointerY; } set pointerY(e) { this._pointerY = e; } _updatePointerPosition(e) { const t = this._scene.getEngine().getInputElementClientRect(); t && (this._pointerX = e.clientX - t.left, this._pointerY = e.clientY - t.top, this._unTranslatedPointerX = this._pointerX, this._unTranslatedPointerY = this._pointerY); } _processPointerMove(e, t) { const r = this._scene, n = r.getEngine(), i = n.getInputElement(); i && (i.tabIndex = n.canvasTabIndex, r.doNotHandleCursors || (i.style.cursor = r.defaultCursor)), this._setCursorAndPointerOverMesh(e, t, r); for (const f of r._pointerMoveStage) { e = e || this._pickMove(t); const o = !!(e != null && e.pickedMesh); e = f.action(this._unTranslatedPointerX, this._unTranslatedPointerY, e, o, i); } const s = t.inputIndex >= Yn.MouseWheelX && t.inputIndex <= Yn.MouseWheelZ ? ir.POINTERWHEEL : ir.POINTERMOVE; r.onPointerMove && (e = e || this._pickMove(t), r.onPointerMove(t, e, s)); let a; e ? (a = new vp(s, t, e), this._setRayOnPointerInfo(e, t)) : (a = new vp(s, t, null, this), this._movePointerInfo = a), r.onPointerObservable.hasObservers() && r.onPointerObservable.notifyObservers(a, s); } // Pointers handling /** @internal */ _setRayOnPointerInfo(e, t) { const r = this._scene; e && r._pickingAvailable && (e.ray || (e.ray = r.createPickingRay(t.offsetX, t.offsetY, he.Identity(), r.activeCamera))); } /** @internal */ _addCameraPointerObserver(e, t) { return this._cameraObserverCount++, this._scene.onPointerObservable.add(e, t); } /** @internal */ _removeCameraPointerObserver(e) { return this._cameraObserverCount--, this._scene.onPointerObservable.remove(e); } _checkForPicking() { return !!(this._scene.onPointerObservable.observers.length > this._cameraObserverCount || this._scene.onPointerPick); } _checkPrePointerObservable(e, t, r) { const n = this._scene, i = new K$(r, t, this._unTranslatedPointerX, this._unTranslatedPointerY); return e && (i.originalPickingInfo = e, i.ray = e.ray, e.originMesh && (i.nearInteractionPickingInfo = e)), n.onPrePointerObservable.notifyObservers(i, r), !!i.skipOnPointerObservable; } /** @internal */ _pickMove(e) { const t = this._scene, r = t.pick(this._unTranslatedPointerX, this._unTranslatedPointerY, t.pointerMovePredicate, t.pointerMoveFastCheck, t.cameraToUseForPointers, t.pointerMoveTrianglePredicate); return this._setCursorAndPointerOverMesh(r, e, t), r; } _setCursorAndPointerOverMesh(e, t, r) { const i = r.getEngine().getInputElement(); if (e != null && e.pickedMesh) { if (this.setPointerOverMesh(e.pickedMesh, t.pointerId, e, t), !r.doNotHandleCursors && i && this._pointerOverMesh) { const s = this._pointerOverMesh._getActionManagerForTrigger(); s && s.hasPointerTriggers && (i.style.cursor = s.hoverCursor || r.hoverCursor); } } else this.setPointerOverMesh(null, t.pointerId, e, t); } /** * Use this method to simulate a pointer move on a mesh * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay * @param pickResult - pickingInfo of the object wished to simulate pointer event on * @param pointerEventInit - pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch) */ simulatePointerMove(e, t) { const r = new PointerEvent("pointermove", t); r.inputIndex = Yn.Move, !this._checkPrePointerObservable(e, r, ir.POINTERMOVE) && this._processPointerMove(e, r); } /** * Use this method to simulate a pointer down on a mesh * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay * @param pickResult - pickingInfo of the object wished to simulate pointer event on * @param pointerEventInit - pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch) */ simulatePointerDown(e, t) { const r = new PointerEvent("pointerdown", t); r.inputIndex = r.button + 2, !this._checkPrePointerObservable(e, r, ir.POINTERDOWN) && this._processPointerDown(e, r); } _processPointerDown(e, t) { const r = this._scene; if (e != null && e.pickedMesh) { this._pickedDownMesh = e.pickedMesh; const s = e.pickedMesh._getActionManagerForTrigger(); if (s) { if (s.hasPickTriggers) switch (s.processTrigger(5, mo.CreateNew(e.pickedMesh, t, e)), t.button) { case 0: s.processTrigger(2, mo.CreateNew(e.pickedMesh, t, e)); break; case 1: s.processTrigger(4, mo.CreateNew(e.pickedMesh, t, e)); break; case 2: s.processTrigger(3, mo.CreateNew(e.pickedMesh, t, e)); break; } s.hasSpecificTrigger(8) && window.setTimeout(() => { const a = r.pick(this._unTranslatedPointerX, this._unTranslatedPointerY, (f) => f.isPickable && f.isVisible && f.isReady() && f.actionManager && f.actionManager.hasSpecificTrigger(8) && f === this._pickedDownMesh, !1, r.cameraToUseForPointers); a != null && a.pickedMesh && s && this._totalPointersPressed !== 0 && Date.now() - this._startingPointerTime > Xf.LongPressDelay && !this._isPointerSwiping() && (this._startingPointerTime = 0, s.processTrigger(8, mo.CreateNew(a.pickedMesh, t))); }, Xf.LongPressDelay); } } else for (const s of r._pointerDownStage) e = s.action(this._unTranslatedPointerX, this._unTranslatedPointerY, e, t, !1); let n; const i = ir.POINTERDOWN; e ? (r.onPointerDown && r.onPointerDown(t, e, i), n = new vp(i, t, e), this._setRayOnPointerInfo(e, t)) : n = new vp(i, t, null, this), r.onPointerObservable.hasObservers() && r.onPointerObservable.notifyObservers(n, i); } /** * @internal * @internals Boolean if delta for pointer exceeds drag movement threshold */ _isPointerSwiping() { return this._isSwiping; } /** * Use this method to simulate a pointer up on a mesh * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay * @param pickResult - pickingInfo of the object wished to simulate pointer event on * @param pointerEventInit - pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch) * @param doubleTap - indicates that the pointer up event should be considered as part of a double click (false by default) */ simulatePointerUp(e, t, r) { const n = new PointerEvent("pointerup", t); n.inputIndex = Yn.Move; const i = new vG(); r ? i.doubleClick = !0 : i.singleClick = !0, !this._checkPrePointerObservable(e, n, ir.POINTERUP) && this._processPointerUp(e, n, i); } _processPointerUp(e, t, r) { const n = this._scene; if (e != null && e.pickedMesh) { if (this._pickedUpMesh = e.pickedMesh, this._pickedDownMesh === this._pickedUpMesh && (n.onPointerPick && n.onPointerPick(t, e), r.singleClick && !r.ignore && n.onPointerObservable.observers.length > this._cameraObserverCount)) { const s = ir.POINTERPICK, a = new vp(s, t, e); this._setRayOnPointerInfo(e, t), n.onPointerObservable.notifyObservers(a, s); } const i = e.pickedMesh._getActionManagerForTrigger(); if (i && !r.ignore) { i.processTrigger(7, mo.CreateNew(e.pickedMesh, t, e)), !r.hasSwiped && r.singleClick && i.processTrigger(1, mo.CreateNew(e.pickedMesh, t, e)); const s = e.pickedMesh._getActionManagerForTrigger(6); r.doubleClick && s && s.processTrigger(6, mo.CreateNew(e.pickedMesh, t, e)); } } else if (!r.ignore) for (const i of n._pointerUpStage) e = i.action(this._unTranslatedPointerX, this._unTranslatedPointerY, e, t, r.doubleClick); if (this._pickedDownMesh && this._pickedDownMesh !== this._pickedUpMesh) { const i = this._pickedDownMesh._getActionManagerForTrigger(16); i && i.processTrigger(16, mo.CreateNew(this._pickedDownMesh, t)); } if (!r.ignore) { const i = new vp(ir.POINTERUP, t, e); if (this._setRayOnPointerInfo(e, t), n.onPointerObservable.notifyObservers(i, ir.POINTERUP), n.onPointerUp && n.onPointerUp(t, e, ir.POINTERUP), !r.hasSwiped && !this._skipPointerTap && !this._isMultiTouchGesture) { let s = 0; if (r.singleClick ? s = ir.POINTERTAP : r.doubleClick && (s = ir.POINTERDOUBLETAP), s) { const a = new vp(s, t, e); n.onPointerObservable.hasObservers() && n.onPointerObservable.hasSpecificMask(s) && n.onPointerObservable.notifyObservers(a, s); } } } } /** * Gets a boolean indicating if the current pointer event is captured (meaning that the scene has already handled the pointer down) * @param pointerId - defines the pointer id to use in a multi-touch scenario (0 by default) * @returns true if the pointer was captured */ isPointerCaptured(e = 0) { return this._pointerCaptures[e]; } /** * Attach events to the canvas (To handle actionManagers triggers and raise onPointerMove, onPointerDown and onPointerUp * @param attachUp - defines if you want to attach events to pointerup * @param attachDown - defines if you want to attach events to pointerdown * @param attachMove - defines if you want to attach events to pointermove * @param elementToAttachTo - defines the target DOM element to attach to (will use the canvas by default) */ attachControl(e = !0, t = !0, r = !0, n = null) { const i = this._scene, s = i.getEngine(); n || (n = s.getInputElement()), this._alreadyAttached && this.detachControl(), n && (this._alreadyAttachedTo = n), this._deviceSourceManager = new J$(s), this._initActionManager = (a) => { if (!this._meshPickProceed) { const f = i.skipPointerUpPicking || i._registeredActions === 0 && !this._checkForPicking() && !i.onPointerUp ? null : i.pick(this._unTranslatedPointerX, this._unTranslatedPointerY, i.pointerUpPredicate, i.pointerUpFastCheck, i.cameraToUseForPointers); this._currentPickResult = f, f && (a = f.hit && f.pickedMesh ? f.pickedMesh._getActionManagerForTrigger() : null), this._meshPickProceed = !0; } return a; }, this._delayedSimpleClick = (a, f, o) => { if ((Date.now() - this._previousStartingPointerTime > Xf.DoubleClickDelay && !this._doubleClickOccured || a !== this._previousButtonPressed) && (this._doubleClickOccured = !1, f.singleClick = !0, f.ignore = !1, this._delayedClicks[a])) { const d = this._delayedClicks[a].evt, v = ir.POINTERTAP, u = new vp(v, d, this._currentPickResult); i.onPointerObservable.hasObservers() && i.onPointerObservable.hasSpecificMask(v) && i.onPointerObservable.notifyObservers(u, v), this._delayedClicks[a] = null; } }, this._initClickEvent = (a, f, o, d) => { var v, u; const l = new vG(); this._currentPickResult = null; let P = null, p = a.hasSpecificMask(ir.POINTERPICK) || f.hasSpecificMask(ir.POINTERPICK) || a.hasSpecificMask(ir.POINTERTAP) || f.hasSpecificMask(ir.POINTERTAP) || a.hasSpecificMask(ir.POINTERDOUBLETAP) || f.hasSpecificMask(ir.POINTERDOUBLETAP); !p && Ml && (P = this._initActionManager(P, l), P && (p = P.hasPickTriggers)); let c = !1; if (p) { const H = o.button; if (l.hasSwiped = this._isPointerSwiping(), !l.hasSwiped) { let T = !Xf.ExclusiveDoubleClickMode; if (T || (T = !a.hasSpecificMask(ir.POINTERDOUBLETAP) && !f.hasSpecificMask(ir.POINTERDOUBLETAP), T && !Ml.HasSpecificTrigger(6) && (P = this._initActionManager(P, l), P && (T = !P.hasSpecificTrigger(6)))), T) (Date.now() - this._previousStartingPointerTime > Xf.DoubleClickDelay || H !== this._previousButtonPressed) && (l.singleClick = !0, d(l, this._currentPickResult), c = !0); else { const b = { evt: o, clickInfo: l, timeoutId: window.setTimeout(this._delayedSimpleClick.bind(this, H, l, d), Xf.DoubleClickDelay) }; this._delayedClicks[H] = b; } let q = a.hasSpecificMask(ir.POINTERDOUBLETAP) || f.hasSpecificMask(ir.POINTERDOUBLETAP); !q && Ml.HasSpecificTrigger(6) && (P = this._initActionManager(P, l), P && (q = P.hasSpecificTrigger(6))), q && (H === this._previousButtonPressed && Date.now() - this._previousStartingPointerTime < Xf.DoubleClickDelay && !this._doubleClickOccured ? (!l.hasSwiped && !this._isPointerSwiping() ? (this._previousStartingPointerTime = 0, this._doubleClickOccured = !0, l.doubleClick = !0, l.ignore = !1, Xf.ExclusiveDoubleClickMode && this._delayedClicks[H] && (clearTimeout((v = this._delayedClicks[H]) === null || v === void 0 ? void 0 : v.timeoutId), this._delayedClicks[H] = null), d(l, this._currentPickResult)) : (this._doubleClickOccured = !1, this._previousStartingPointerTime = this._startingPointerTime, this._previousStartingPointerPosition.x = this._startingPointerPosition.x, this._previousStartingPointerPosition.y = this._startingPointerPosition.y, this._previousButtonPressed = H, Xf.ExclusiveDoubleClickMode ? (this._delayedClicks[H] && (clearTimeout((u = this._delayedClicks[H]) === null || u === void 0 ? void 0 : u.timeoutId), this._delayedClicks[H] = null), d(l, this._previousPickResult)) : d(l, this._currentPickResult)), c = !0) : (this._doubleClickOccured = !1, this._previousStartingPointerTime = this._startingPointerTime, this._previousStartingPointerPosition.x = this._startingPointerPosition.x, this._previousStartingPointerPosition.y = this._startingPointerPosition.y, this._previousButtonPressed = H)); } } c || d(l, this._currentPickResult); }, this._onPointerMove = (a) => { if (this._updatePointerPosition(a), !this._isSwiping && this._swipeButtonPressed !== -1 && (this._isSwiping = Math.abs(this._startingPointerPosition.x - this._pointerX) > Xf.DragMovementThreshold || Math.abs(this._startingPointerPosition.y - this._pointerY) > Xf.DragMovementThreshold), s.isPointerLock && s._verifyPointerLock(), this._checkPrePointerObservable(null, a, a.inputIndex >= Yn.MouseWheelX && a.inputIndex <= Yn.MouseWheelZ ? ir.POINTERWHEEL : ir.POINTERMOVE) || !i.cameraToUseForPointers && !i.activeCamera) return; if (i.skipPointerMovePicking) { this._processPointerMove(new F9(), a); return; } i.pointerMovePredicate || (i.pointerMovePredicate = (o) => o.isPickable && o.isVisible && o.isReady() && o.isEnabled() && (o.enablePointerMoveEvents || i.constantlyUpdateMeshUnderPointer || o._getActionManagerForTrigger() !== null) && (!i.cameraToUseForPointers || (i.cameraToUseForPointers.layerMask & o.layerMask) !== 0)); const f = i._registeredActions > 0 || i.constantlyUpdateMeshUnderPointer ? this._pickMove(a) : null; this._processPointerMove(f, a); }, this._onPointerDown = (a) => { var f; if (this._totalPointersPressed++, this._pickedDownMesh = null, this._meshPickProceed = !1, Xf.ExclusiveDoubleClickMode) { for (let d = 0; d < this._delayedClicks.length; d++) if (this._delayedClicks[d]) if (a.button === d) clearTimeout((f = this._delayedClicks[d]) === null || f === void 0 ? void 0 : f.timeoutId); else { const v = this._delayedClicks[d].clickInfo; this._doubleClickOccured = !1, v.singleClick = !0, v.ignore = !1; const u = this._delayedClicks[d].evt, l = ir.POINTERTAP, P = new vp(l, u, this._currentPickResult); i.onPointerObservable.hasObservers() && i.onPointerObservable.hasSpecificMask(l) && i.onPointerObservable.notifyObservers(P, l), this._delayedClicks[d] = null; } } if (this._updatePointerPosition(a), this._swipeButtonPressed === -1 && (this._swipeButtonPressed = a.button), i.preventDefaultOnPointerDown && n && (a.preventDefault(), n.focus()), this._startingPointerPosition.x = this._pointerX, this._startingPointerPosition.y = this._pointerY, this._startingPointerTime = Date.now(), this._checkPrePointerObservable(null, a, ir.POINTERDOWN) || !i.cameraToUseForPointers && !i.activeCamera) return; this._pointerCaptures[a.pointerId] = !0, i.pointerDownPredicate || (i.pointerDownPredicate = (d) => d.isPickable && d.isVisible && d.isReady() && d.isEnabled() && (!i.cameraToUseForPointers || (i.cameraToUseForPointers.layerMask & d.layerMask) !== 0)), this._pickedDownMesh = null; let o; i.skipPointerDownPicking || i._registeredActions === 0 && !this._checkForPicking() && !i.onPointerDown ? o = new F9() : o = i.pick(this._unTranslatedPointerX, this._unTranslatedPointerY, i.pointerDownPredicate, i.pointerDownFastCheck, i.cameraToUseForPointers), this._processPointerDown(o, a); }, this._onPointerUp = (a) => { this._totalPointersPressed !== 0 && (this._totalPointersPressed--, this._pickedUpMesh = null, this._meshPickProceed = !1, this._updatePointerPosition(a), i.preventDefaultOnPointerUp && n && (a.preventDefault(), n.focus()), this._initClickEvent(i.onPrePointerObservable, i.onPointerObservable, a, (f, o) => { if (i.onPrePointerObservable.hasObservers() && (this._skipPointerTap = !1, !f.ignore)) { if (this._checkPrePointerObservable(null, a, ir.POINTERUP)) { this._swipeButtonPressed === a.button && (this._isSwiping = !1, this._swipeButtonPressed = -1), a.buttons === 0 && (this._pointerCaptures[a.pointerId] = !1); return; } f.hasSwiped || (f.singleClick && i.onPrePointerObservable.hasSpecificMask(ir.POINTERTAP) && this._checkPrePointerObservable(null, a, ir.POINTERTAP) && (this._skipPointerTap = !0), f.doubleClick && i.onPrePointerObservable.hasSpecificMask(ir.POINTERDOUBLETAP) && this._checkPrePointerObservable(null, a, ir.POINTERDOUBLETAP) && (this._skipPointerTap = !0)); } if (!this._pointerCaptures[a.pointerId]) { this._swipeButtonPressed === a.button && (this._isSwiping = !1, this._swipeButtonPressed = -1); return; } a.buttons === 0 && (this._pointerCaptures[a.pointerId] = !1), !(!i.cameraToUseForPointers && !i.activeCamera) && (i.pointerUpPredicate || (i.pointerUpPredicate = (d) => d.isPickable && d.isVisible && d.isReady() && d.isEnabled() && (!i.cameraToUseForPointers || (i.cameraToUseForPointers.layerMask & d.layerMask) !== 0)), !this._meshPickProceed && (Ml && Ml.HasTriggers || this._checkForPicking() || i.onPointerUp) && this._initActionManager(null, f), o || (o = this._currentPickResult), this._processPointerUp(o, a, f), this._previousPickResult = this._currentPickResult, this._swipeButtonPressed === a.button && (this._isSwiping = !1, this._swipeButtonPressed = -1)); })); }, this._onKeyDown = (a) => { const f = t4.KEYDOWN; if (i.onPreKeyboardObservable.hasObservers()) { const o = new Y5(f, a); if (i.onPreKeyboardObservable.notifyObservers(o, f), o.skipOnKeyboardObservable) return; } if (i.onKeyboardObservable.hasObservers()) { const o = new MC(f, a); i.onKeyboardObservable.notifyObservers(o, f); } i.actionManager && i.actionManager.processTrigger(14, mo.CreateNewFromScene(i, a)); }, this._onKeyUp = (a) => { const f = t4.KEYUP; if (i.onPreKeyboardObservable.hasObservers()) { const o = new Y5(f, a); if (i.onPreKeyboardObservable.notifyObservers(o, f), o.skipOnKeyboardObservable) return; } if (i.onKeyboardObservable.hasObservers()) { const o = new MC(f, a); i.onKeyboardObservable.notifyObservers(o, f); } i.actionManager && i.actionManager.processTrigger(15, mo.CreateNewFromScene(i, a)); }, this._deviceSourceManager.onDeviceConnectedObservable.add((a) => { a.deviceType === vn.Mouse ? a.onInputChangedObservable.add((f) => { f.inputIndex === Yn.LeftClick || f.inputIndex === Yn.MiddleClick || f.inputIndex === Yn.RightClick || f.inputIndex === Yn.BrowserBack || f.inputIndex === Yn.BrowserForward ? t && a.getInput(f.inputIndex) === 1 ? this._onPointerDown(f) : e && a.getInput(f.inputIndex) === 0 && this._onPointerUp(f) : r && (f.inputIndex === Yn.Move ? this._onPointerMove(f) : (f.inputIndex === Yn.MouseWheelX || f.inputIndex === Yn.MouseWheelY || f.inputIndex === Yn.MouseWheelZ) && this._onPointerMove(f)); }) : a.deviceType === vn.Touch ? a.onInputChangedObservable.add((f) => { f.inputIndex === Yn.LeftClick && (t && a.getInput(f.inputIndex) === 1 ? (this._onPointerDown(f), this._totalPointersPressed > 1 && (this._isMultiTouchGesture = !0)) : e && a.getInput(f.inputIndex) === 0 && (this._onPointerUp(f), this._totalPointersPressed === 0 && (this._isMultiTouchGesture = !1))), r && f.inputIndex === Yn.Move && this._onPointerMove(f); }) : a.deviceType === vn.Keyboard && a.onInputChangedObservable.add((f) => { f.type === "keydown" ? this._onKeyDown(f) : f.type === "keyup" && this._onKeyUp(f); }); }), this._alreadyAttached = !0; } /** * Detaches all event handlers */ detachControl() { this._alreadyAttached && (this._deviceSourceManager.dispose(), this._deviceSourceManager = null, this._alreadyAttachedTo && !this._scene.doNotHandleCursors && (this._alreadyAttachedTo.style.cursor = this._scene.defaultCursor), this._alreadyAttached = !1, this._alreadyAttachedTo = null); } /** * Force the value of meshUnderPointer * @param mesh - defines the mesh to use * @param pointerId - optional pointer id when using more than one pointer. Defaults to 0 * @param pickResult - optional pickingInfo data used to find mesh * @param evt - optional pointer event */ setPointerOverMesh(e, t = 0, r, n) { if (this._meshUnderPointerId[t] === e && (!e || !e._internalAbstractMeshDataInfo._pointerOverDisableMeshTesting)) return; const i = this._meshUnderPointerId[t]; let s; i && (s = i._getActionManagerForTrigger(10), s && s.processTrigger(10, mo.CreateNew(i, n, { pointerId: t }))), e ? (this._meshUnderPointerId[t] = e, this._pointerOverMesh = e, s = e._getActionManagerForTrigger(9), s && s.processTrigger(9, mo.CreateNew(e, n, { pointerId: t, pickResult: r }))) : (delete this._meshUnderPointerId[t], this._pointerOverMesh = null); } /** * Gets the mesh under the pointer * @returns a Mesh or null if no mesh is under the pointer */ getPointerOverMesh() { return this.meshUnderPointer; } /** * @param mesh - Mesh to invalidate * @internal */ _invalidateMesh(e) { this._pointerOverMesh === e && (this._pointerOverMesh = null), this._pickedDownMesh === e && (this._pickedDownMesh = null), this._pickedUpMesh === e && (this._pickedUpMesh = null); for (const t in this._meshUnderPointerId) this._meshUnderPointerId[t] === e && delete this._meshUnderPointerId[t]; } } Xf.DragMovementThreshold = 10; Xf.LongPressDelay = 500; Xf.DoubleClickDelay = 300; Xf.ExclusiveDoubleClickMode = !1; class v9 { /** * Returns the smallest value ever */ get min() { return this._min; } /** * Returns the biggest value ever */ get max() { return this._max; } /** * Returns the average value since the performance counter is running */ get average() { return this._average; } /** * Returns the average value of the last second the counter was monitored */ get lastSecAverage() { return this._lastSecAverage; } /** * Returns the current value */ get current() { return this._current; } /** * Gets the accumulated total */ get total() { return this._totalAccumulated; } /** * Gets the total value count */ get count() { return this._totalValueCount; } /** * Creates a new counter */ constructor() { this._startMonitoringTime = 0, this._min = 0, this._max = 0, this._average = 0, this._lastSecAverage = 0, this._current = 0, this._totalValueCount = 0, this._totalAccumulated = 0, this._lastSecAccumulated = 0, this._lastSecTime = 0, this._lastSecValueCount = 0; } /** * Call this method to start monitoring a new frame. * This scenario is typically used when you accumulate monitoring time many times for a single frame, you call this method at the start of the frame, then beginMonitoring to start recording and endMonitoring(false) to accumulated the recorded time to the PerfCounter or addCount() to accumulate a monitored count. */ fetchNewFrame() { this._totalValueCount++, this._current = 0, this._lastSecValueCount++; } /** * Call this method to monitor a count of something (e.g. mesh drawn in viewport count) * @param newCount the count value to add to the monitored count * @param fetchResult true when it's the last time in the frame you add to the counter and you wish to update the statistics properties (min/max/average), false if you only want to update statistics. */ addCount(e, t) { v9.Enabled && (this._current += e, t && this._fetchResult()); } /** * Start monitoring this performance counter */ beginMonitoring() { v9.Enabled && (this._startMonitoringTime = Yi.Now); } /** * Compute the time lapsed since the previous beginMonitoring() call. * @param newFrame true by default to fetch the result and monitor a new frame, if false the time monitored will be added to the current frame counter */ endMonitoring(e = !0) { if (!v9.Enabled) return; e && this.fetchNewFrame(); const t = Yi.Now; this._current = t - this._startMonitoringTime, e && this._fetchResult(); } /** * Call this method to end the monitoring of a frame. * This scenario is typically used when you accumulate monitoring time many times for a single frame, you call this method at the end of the frame, after beginMonitoring to start recording and endMonitoring(false) to accumulated the recorded time to the PerfCounter or addCount() to accumulate a monitored count. */ endFrame() { this._fetchResult(); } _fetchResult() { this._totalAccumulated += this._current, this._lastSecAccumulated += this._current, this._min = Math.min(this._min, this._current), this._max = Math.max(this._max, this._current), this._average = this._totalAccumulated / this._totalValueCount; const e = Yi.Now; e - this._lastSecTime > 1e3 && (this._lastSecAverage = this._lastSecAccumulated / this._lastSecValueCount, this._lastSecTime = e, this._lastSecAccumulated = 0, this._lastSecValueCount = 0); } } v9.Enabled = !0; class BA { /** * Creates a Plane object according to the given floats a, b, c, d and the plane equation : ax + by + cz + d = 0 * @param a a component of the plane * @param b b component of the plane * @param c c component of the plane * @param d d component of the plane */ constructor(e, t, r, n) { this.normal = new S(e, t, r), this.d = n; } /** * @returns the plane coordinates as a new array of 4 elements [a, b, c, d]. */ asArray() { return [this.normal.x, this.normal.y, this.normal.z, this.d]; } // Methods /** * @returns a new plane copied from the current Plane. */ clone() { return new BA(this.normal.x, this.normal.y, this.normal.z, this.d); } /** * @returns the string "Plane". */ getClassName() { return "Plane"; } /** * @returns the Plane hash code. */ getHashCode() { let e = this.normal.getHashCode(); return e = e * 397 ^ (this.d | 0), e; } /** * Normalize the current Plane in place. * @returns the updated Plane. */ normalize() { const e = Math.sqrt(this.normal.x * this.normal.x + this.normal.y * this.normal.y + this.normal.z * this.normal.z); let t = 0; return e !== 0 && (t = 1 / e), this.normal.x *= t, this.normal.y *= t, this.normal.z *= t, this.d *= t, this; } /** * Applies a transformation the plane and returns the result * @param transformation the transformation matrix to be applied to the plane * @returns a new Plane as the result of the transformation of the current Plane by the given matrix. */ transform(e) { const t = BA._TmpMatrix; e.invertToRef(t); const r = t.m, n = this.normal.x, i = this.normal.y, s = this.normal.z, a = this.d, f = n * r[0] + i * r[1] + s * r[2] + a * r[3], o = n * r[4] + i * r[5] + s * r[6] + a * r[7], d = n * r[8] + i * r[9] + s * r[10] + a * r[11], v = n * r[12] + i * r[13] + s * r[14] + a * r[15]; return new BA(f, o, d, v); } /** * Compute the dot product between the point and the plane normal * @param point point to calculate the dot product with * @returns the dot product (float) of the point coordinates and the plane normal. */ dotCoordinate(e) { return this.normal.x * e.x + this.normal.y * e.y + this.normal.z * e.z + this.d; } /** * Updates the current Plane from the plane defined by the three given points. * @param point1 one of the points used to construct the plane * @param point2 one of the points used to construct the plane * @param point3 one of the points used to construct the plane * @returns the updated Plane. */ copyFromPoints(e, t, r) { const n = t.x - e.x, i = t.y - e.y, s = t.z - e.z, a = r.x - e.x, f = r.y - e.y, o = r.z - e.z, d = i * o - s * f, v = s * a - n * o, u = n * f - i * a, l = Math.sqrt(d * d + v * v + u * u); let P; return l !== 0 ? P = 1 / l : P = 0, this.normal.x = d * P, this.normal.y = v * P, this.normal.z = u * P, this.d = -(this.normal.x * e.x + this.normal.y * e.y + this.normal.z * e.z), this; } /** * Checks if the plane is facing a given direction (meaning if the plane's normal is pointing in the opposite direction of the given vector). * Note that for this function to work as expected you should make sure that: * - direction and the plane normal are normalized * - epsilon is a number just bigger than -1, something like -0.99 for eg * @param direction the direction to check if the plane is facing * @param epsilon value the dot product is compared against (returns true if dot <= epsilon) * @returns True if the plane is facing the given direction */ isFrontFacingTo(e, t) { return S.Dot(this.normal, e) <= t; } /** * Calculates the distance to a point * @param point point to calculate distance to * @returns the signed distance (float) from the given point to the Plane. */ signedDistanceTo(e) { return S.Dot(e, this.normal) + this.d; } // Statics /** * Creates a plane from an array * @param array the array to create a plane from * @returns a new Plane from the given array. */ static FromArray(e) { return new BA(e[0], e[1], e[2], e[3]); } /** * Creates a plane from three points * @param point1 point used to create the plane * @param point2 point used to create the plane * @param point3 point used to create the plane * @returns a new Plane defined by the three given points. */ static FromPoints(e, t, r) { const n = new BA(0, 0, 0, 0); return n.copyFromPoints(e, t, r), n; } /** * Creates a plane from an origin point and a normal * @param origin origin of the plane to be constructed * @param normal normal of the plane to be constructed * @returns a new Plane the normal vector to this plane at the given origin point. */ static FromPositionAndNormal(e, t) { const r = new BA(0, 0, 0, 0); return this.FromPositionAndNormalToRef(e, t, r); } /** * Updates the given Plane "result" from an origin point and a normal. * @param origin origin of the plane to be constructed * @param normal the normalized normals of the plane to be constructed * @param result defines the Plane where to store the result * @returns result input */ static FromPositionAndNormalToRef(e, t, r) { return r.normal.copyFrom(t), r.normal.normalize(), r.d = -e.dot(r.normal), r; } /** * Calculates the distance from a plane and a point * @param origin origin of the plane to be constructed * @param normal normal of the plane to be constructed * @param point point to calculate distance to * @returns the signed distance between the plane defined by the normal vector at the "origin"" point and the given other point. */ static SignedDistanceToPlaneFromPositionAndNormal(e, t, r) { const n = -(t.x * e.x + t.y * e.y + t.z * e.z); return S.Dot(r, t) + n; } } BA._TmpMatrix = he.Identity(); class Xc { /** * Gets the planes representing the frustum * @param transform matrix to be applied to the returned planes * @returns a new array of 6 Frustum planes computed by the given transformation matrix. */ static GetPlanes(e) { const t = []; for (let r = 0; r < 6; r++) t.push(new BA(0, 0, 0, 0)); return Xc.GetPlanesToRef(e, t), t; } /** * Gets the near frustum plane transformed by the transform matrix * @param transform transformation matrix to be applied to the resulting frustum plane * @param frustumPlane the resulting frustum plane */ static GetNearPlaneToRef(e, t) { const r = e.m; t.normal.x = r[3] + r[2], t.normal.y = r[7] + r[6], t.normal.z = r[11] + r[10], t.d = r[15] + r[14], t.normalize(); } /** * Gets the far frustum plane transformed by the transform matrix * @param transform transformation matrix to be applied to the resulting frustum plane * @param frustumPlane the resulting frustum plane */ static GetFarPlaneToRef(e, t) { const r = e.m; t.normal.x = r[3] - r[2], t.normal.y = r[7] - r[6], t.normal.z = r[11] - r[10], t.d = r[15] - r[14], t.normalize(); } /** * Gets the left frustum plane transformed by the transform matrix * @param transform transformation matrix to be applied to the resulting frustum plane * @param frustumPlane the resulting frustum plane */ static GetLeftPlaneToRef(e, t) { const r = e.m; t.normal.x = r[3] + r[0], t.normal.y = r[7] + r[4], t.normal.z = r[11] + r[8], t.d = r[15] + r[12], t.normalize(); } /** * Gets the right frustum plane transformed by the transform matrix * @param transform transformation matrix to be applied to the resulting frustum plane * @param frustumPlane the resulting frustum plane */ static GetRightPlaneToRef(e, t) { const r = e.m; t.normal.x = r[3] - r[0], t.normal.y = r[7] - r[4], t.normal.z = r[11] - r[8], t.d = r[15] - r[12], t.normalize(); } /** * Gets the top frustum plane transformed by the transform matrix * @param transform transformation matrix to be applied to the resulting frustum plane * @param frustumPlane the resulting frustum plane */ static GetTopPlaneToRef(e, t) { const r = e.m; t.normal.x = r[3] - r[1], t.normal.y = r[7] - r[5], t.normal.z = r[11] - r[9], t.d = r[15] - r[13], t.normalize(); } /** * Gets the bottom frustum plane transformed by the transform matrix * @param transform transformation matrix to be applied to the resulting frustum plane * @param frustumPlane the resulting frustum plane */ static GetBottomPlaneToRef(e, t) { const r = e.m; t.normal.x = r[3] + r[1], t.normal.y = r[7] + r[5], t.normal.z = r[11] + r[9], t.d = r[15] + r[13], t.normalize(); } /** * Sets the given array "frustumPlanes" with the 6 Frustum planes computed by the given transformation matrix. * @param transform transformation matrix to be applied to the resulting frustum planes * @param frustumPlanes the resulting frustum planes */ static GetPlanesToRef(e, t) { Xc.GetNearPlaneToRef(e, t[0]), Xc.GetFarPlaneToRef(e, t[1]), Xc.GetLeftPlaneToRef(e, t[2]), Xc.GetRightPlaneToRef(e, t[3]), Xc.GetTopPlaneToRef(e, t[4]), Xc.GetBottomPlaneToRef(e, t[5]); } /** * Tests if a point is located between the frustum planes. * @param point defines the point to test * @param frustumPlanes defines the frustum planes to test * @returns true if the point is located between the frustum planes */ static IsPointInFrustum(e, t) { for (let r = 0; r < 6; r++) if (t[r].dotCoordinate(e) < 0) return !1; return !0; } } class bR { /** * Gets an unique (relatively to the current scene) Id */ static get UniqueId() { const e = this._UniqueIdCounter; return this._UniqueIdCounter++, e; } } bR._UniqueIdCounter = 1; class ra { /** * Sort function to order lights for rendering. * @param a First Light object to compare to second. * @param b Second Light object to compare first. * @returns -1 to reduce's a's index relative to be, 0 for no change, 1 to increase a's index relative to b. */ static CompareLightsPriority(e, t) { return e.shadowEnabled !== t.shadowEnabled ? (t.shadowEnabled ? 1 : 0) - (e.shadowEnabled ? 1 : 0) : t.renderPriority - e.renderPriority; } } ra.FALLOFF_DEFAULT = 0; ra.FALLOFF_PHYSICAL = 1; ra.FALLOFF_GLTF = 2; ra.FALLOFF_STANDARD = 3; ra.LIGHTMAP_DEFAULT = 0; ra.LIGHTMAP_SPECULAR = 1; ra.LIGHTMAP_SHADOWSONLY = 2; ra.INTENSITYMODE_AUTOMATIC = 0; ra.INTENSITYMODE_LUMINOUSPOWER = 1; ra.INTENSITYMODE_LUMINOUSINTENSITY = 2; ra.INTENSITYMODE_ILLUMINANCE = 3; ra.INTENSITYMODE_LUMINANCE = 4; ra.LIGHTTYPEID_POINTLIGHT = 0; ra.LIGHTTYPEID_DIRECTIONALLIGHT = 1; ra.LIGHTTYPEID_SPOTLIGHT = 2; ra.LIGHTTYPEID_HEMISPHERICLIGHT = 3; class e6e { constructor() { this.pointerDownFastCheck = !1, this.pointerUpFastCheck = !1, this.pointerMoveFastCheck = !1, this.skipPointerMovePicking = !1, this.skipPointerDownPicking = !1, this.skipPointerUpPicking = !1; } } var $H; (function(A) { A[A.BackwardCompatible = 0] = "BackwardCompatible", A[A.Intermediate = 1] = "Intermediate", A[A.Aggressive = 2] = "Aggressive"; })($H || ($H = {})); class sr extends J1 { /** * Factory used to create the default material. * @param scene The scene to create the material for * @returns The default material */ static DefaultMaterialFactory(e) { throw qn("StandardMaterial"); } /** * Factory used to create the a collision coordinator. * @returns The collision coordinator */ static CollisionCoordinatorFactory() { throw qn("DefaultCollisionCoordinator"); } /** * Texture used in all pbr material as the reflection texture. * As in the majority of the scene they are the same (exception for multi room and so on), * this is easier to reference from here than from all the materials. */ get environmentTexture() { return this._environmentTexture; } /** * Texture used in all pbr material as the reflection texture. * As in the majority of the scene they are the same (exception for multi room and so on), * this is easier to set here than in all the materials. */ set environmentTexture(e) { this._environmentTexture !== e && (this._environmentTexture = e, this.markAllMaterialsAsDirty(1)); } /** * Default image processing configuration used either in the rendering * Forward main pass or through the imageProcessingPostProcess if present. * As in the majority of the scene they are the same (exception for multi camera), * this is easier to reference from here than from all the materials and post process. * * No setter as we it is a shared configuration, you can set the values instead. */ get imageProcessingConfiguration() { return this._imageProcessingConfiguration; } /** * Gets or sets a value indicating how to treat performance relatively to ease of use and backward compatibility */ get performancePriority() { return this._performancePriority; } set performancePriority(e) { if (e !== this._performancePriority) { switch (this._performancePriority = e, e) { case $H.BackwardCompatible: this.skipFrustumClipping = !1, this._renderingManager.maintainStateBetweenFrames = !1, this.skipPointerMovePicking = !1, this.autoClear = !0; break; case $H.Intermediate: this.skipFrustumClipping = !1, this._renderingManager.maintainStateBetweenFrames = !1, this.skipPointerMovePicking = !0, this.autoClear = !1; break; case $H.Aggressive: this.skipFrustumClipping = !0, this._renderingManager.maintainStateBetweenFrames = !0, this.skipPointerMovePicking = !0, this.autoClear = !1; break; } this.onScenePerformancePriorityChangedObservable.notifyObservers(e); } } /** * Gets or sets a boolean indicating if all rendering must be done in wireframe */ set forceWireframe(e) { this._forceWireframe !== e && (this._forceWireframe = e, this.markAllMaterialsAsDirty(16)); } get forceWireframe() { return this._forceWireframe; } /** * Gets or sets a boolean indicating if we should skip the frustum clipping part of the active meshes selection */ set skipFrustumClipping(e) { this._skipFrustumClipping !== e && (this._skipFrustumClipping = e); } get skipFrustumClipping() { return this._skipFrustumClipping; } /** * Gets or sets a boolean indicating if all rendering must be done in point cloud */ set forcePointsCloud(e) { this._forcePointsCloud !== e && (this._forcePointsCloud = e, this.markAllMaterialsAsDirty(16)); } get forcePointsCloud() { return this._forcePointsCloud; } /** * Gets or sets the animation properties override */ get animationPropertiesOverride() { return this._animationPropertiesOverride; } set animationPropertiesOverride(e) { this._animationPropertiesOverride = e; } /** Sets a function to be executed when this scene is disposed. */ set onDispose(e) { this._onDisposeObserver && this.onDisposeObservable.remove(this._onDisposeObserver), this._onDisposeObserver = this.onDisposeObservable.add(e); } /** Sets a function to be executed before rendering this scene */ set beforeRender(e) { this._onBeforeRenderObserver && this.onBeforeRenderObservable.remove(this._onBeforeRenderObserver), e && (this._onBeforeRenderObserver = this.onBeforeRenderObservable.add(e)); } /** Sets a function to be executed after rendering this scene */ set afterRender(e) { this._onAfterRenderObserver && this.onAfterRenderObservable.remove(this._onAfterRenderObserver), e && (this._onAfterRenderObserver = this.onAfterRenderObservable.add(e)); } /** Sets a function to be executed before rendering a camera*/ set beforeCameraRender(e) { this._onBeforeCameraRenderObserver && this.onBeforeCameraRenderObservable.remove(this._onBeforeCameraRenderObserver), this._onBeforeCameraRenderObserver = this.onBeforeCameraRenderObservable.add(e); } /** Sets a function to be executed after rendering a camera*/ set afterCameraRender(e) { this._onAfterCameraRenderObserver && this.onAfterCameraRenderObservable.remove(this._onAfterCameraRenderObserver), this._onAfterCameraRenderObserver = this.onAfterCameraRenderObservable.add(e); } /** * Gets or sets a predicate used to select candidate meshes for a pointer down event */ get pointerDownPredicate() { return this._pointerPickingConfiguration.pointerDownPredicate; } set pointerDownPredicate(e) { this._pointerPickingConfiguration.pointerDownPredicate = e; } /** * Gets or sets a predicate used to select candidate meshes for a pointer up event */ get pointerUpPredicate() { return this._pointerPickingConfiguration.pointerUpPredicate; } set pointerUpPredicate(e) { this._pointerPickingConfiguration.pointerUpPredicate = e; } /** * Gets or sets a predicate used to select candidate meshes for a pointer move event */ get pointerMovePredicate() { return this._pointerPickingConfiguration.pointerMovePredicate; } set pointerMovePredicate(e) { this._pointerPickingConfiguration.pointerMovePredicate = e; } /** * Gets or sets a predicate used to select candidate meshes for a pointer down event */ get pointerDownFastCheck() { return this._pointerPickingConfiguration.pointerDownFastCheck; } set pointerDownFastCheck(e) { this._pointerPickingConfiguration.pointerDownFastCheck = e; } /** * Gets or sets a predicate used to select candidate meshes for a pointer up event */ get pointerUpFastCheck() { return this._pointerPickingConfiguration.pointerUpFastCheck; } set pointerUpFastCheck(e) { this._pointerPickingConfiguration.pointerUpFastCheck = e; } /** * Gets or sets a predicate used to select candidate meshes for a pointer move event */ get pointerMoveFastCheck() { return this._pointerPickingConfiguration.pointerMoveFastCheck; } set pointerMoveFastCheck(e) { this._pointerPickingConfiguration.pointerMoveFastCheck = e; } /** * Gets or sets a boolean indicating if the user want to entirely skip the picking phase when a pointer move event occurs. */ get skipPointerMovePicking() { return this._pointerPickingConfiguration.skipPointerMovePicking; } set skipPointerMovePicking(e) { this._pointerPickingConfiguration.skipPointerMovePicking = e; } /** * Gets or sets a boolean indicating if the user want to entirely skip the picking phase when a pointer down event occurs. */ get skipPointerDownPicking() { return this._pointerPickingConfiguration.skipPointerDownPicking; } set skipPointerDownPicking(e) { this._pointerPickingConfiguration.skipPointerDownPicking = e; } /** * Gets or sets a boolean indicating if the user want to entirely skip the picking phase when a pointer up event occurs. Off by default. */ get skipPointerUpPicking() { return this._pointerPickingConfiguration.skipPointerUpPicking; } set skipPointerUpPicking(e) { this._pointerPickingConfiguration.skipPointerUpPicking = e; } /** * Gets the pointer coordinates without any translation (ie. straight out of the pointer event) */ get unTranslatedPointer() { return this._inputManager.unTranslatedPointer; } /** * Gets or sets the distance in pixel that you have to move to prevent some events. Default is 10 pixels */ static get DragMovementThreshold() { return Xf.DragMovementThreshold; } static set DragMovementThreshold(e) { Xf.DragMovementThreshold = e; } /** * Time in milliseconds to wait to raise long press events if button is still pressed. Default is 500 ms */ static get LongPressDelay() { return Xf.LongPressDelay; } static set LongPressDelay(e) { Xf.LongPressDelay = e; } /** * Time in milliseconds to wait to raise long press events if button is still pressed. Default is 300 ms */ static get DoubleClickDelay() { return Xf.DoubleClickDelay; } static set DoubleClickDelay(e) { Xf.DoubleClickDelay = e; } /** If you need to check double click without raising a single click at first click, enable this flag */ static get ExclusiveDoubleClickMode() { return Xf.ExclusiveDoubleClickMode; } static set ExclusiveDoubleClickMode(e) { Xf.ExclusiveDoubleClickMode = e; } /** * Bind the current view position to an effect. * @param effect The effect to be bound * @param variableName name of the shader variable that will hold the eye position * @param isVector3 true to indicates that variableName is a Vector3 and not a Vector4 * @returns the computed eye position */ bindEyePosition(e, t = "vEyePosition", r = !1) { const n = this._forcedViewPosition ? this._forcedViewPosition : this._mirroredCameraPosition ? this._mirroredCameraPosition : this.activeCamera.globalPosition, i = this.useRightHandedSystem === (this._mirroredCameraPosition != null); return ue.Vector4[0].set(n.x, n.y, n.z, i ? -1 : 1), e && (r ? e.setFloat3(t, ue.Vector4[0].x, ue.Vector4[0].y, ue.Vector4[0].z) : e.setVector4(t, ue.Vector4[0])), ue.Vector4[0]; } /** * Update the scene ubo before it can be used in rendering processing * @returns the scene UniformBuffer */ finalizeSceneUbo() { const e = this.getSceneUniformBuffer(), t = this.bindEyePosition(null); return e.updateFloat4("vEyePosition", t.x, t.y, t.z, t.w), e.update(), e; } /** * Gets or sets a boolean indicating if the scene must use right-handed coordinates system */ set useRightHandedSystem(e) { this._useRightHandedSystem !== e && (this._useRightHandedSystem = e, this.markAllMaterialsAsDirty(16)); } get useRightHandedSystem() { return this._useRightHandedSystem; } /** * Sets the step Id used by deterministic lock step * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep * @param newStepId defines the step Id */ setStepId(e) { this._currentStepId = e; } /** * Gets the step Id used by deterministic lock step * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep * @returns the step Id */ getStepId() { return this._currentStepId; } /** * Gets the internal step used by deterministic lock step * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep * @returns the internal step */ getInternalStep() { return this._currentInternalStep; } /** * Gets or sets a boolean indicating if fog is enabled on this scene * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/environment_introduction#fog * (Default is true) */ set fogEnabled(e) { this._fogEnabled !== e && (this._fogEnabled = e, this.markAllMaterialsAsDirty(16)); } get fogEnabled() { return this._fogEnabled; } /** * Gets or sets the fog mode to use * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/environment_introduction#fog * | mode | value | * | --- | --- | * | FOGMODE_NONE | 0 | * | FOGMODE_EXP | 1 | * | FOGMODE_EXP2 | 2 | * | FOGMODE_LINEAR | 3 | */ set fogMode(e) { this._fogMode !== e && (this._fogMode = e, this.markAllMaterialsAsDirty(16)); } get fogMode() { return this._fogMode; } /** * Flag indicating that the frame buffer binding is handled by another component */ get prePass() { return !!this.prePassRenderer && this.prePassRenderer.defaultRT.enabled; } /** * Gets or sets a boolean indicating if shadows are enabled on this scene */ set shadowsEnabled(e) { this._shadowsEnabled !== e && (this._shadowsEnabled = e, this.markAllMaterialsAsDirty(2)); } get shadowsEnabled() { return this._shadowsEnabled; } /** * Gets or sets a boolean indicating if lights are enabled on this scene */ set lightsEnabled(e) { this._lightsEnabled !== e && (this._lightsEnabled = e, this.markAllMaterialsAsDirty(2)); } get lightsEnabled() { return this._lightsEnabled; } /** All of the active cameras added to this scene. */ get activeCameras() { return this._activeCameras; } set activeCameras(e) { this._unObserveActiveCameras && (this._unObserveActiveCameras(), this._unObserveActiveCameras = null), e && (this._unObserveActiveCameras = p$(e, () => { this.onActiveCamerasChanged.notifyObservers(this); })), this._activeCameras = e; } /** Gets or sets the current active camera */ get activeCamera() { return this._activeCamera; } set activeCamera(e) { e !== this._activeCamera && (this._activeCamera = e, this.onActiveCameraChanged.notifyObservers(this)); } /** The default material used on meshes when no material is affected */ get defaultMaterial() { return this._defaultMaterial || (this._defaultMaterial = sr.DefaultMaterialFactory(this)), this._defaultMaterial; } /** The default material used on meshes when no material is affected */ set defaultMaterial(e) { this._defaultMaterial = e; } /** * Gets or sets a boolean indicating if textures are enabled on this scene */ set texturesEnabled(e) { this._texturesEnabled !== e && (this._texturesEnabled = e, this.markAllMaterialsAsDirty(1)); } get texturesEnabled() { return this._texturesEnabled; } /** * Gets or sets a boolean indicating if skeletons are enabled on this scene */ set skeletonsEnabled(e) { this._skeletonsEnabled !== e && (this._skeletonsEnabled = e, this.markAllMaterialsAsDirty(8)); } get skeletonsEnabled() { return this._skeletonsEnabled; } /** @internal */ get collisionCoordinator() { return this._collisionCoordinator || (this._collisionCoordinator = sr.CollisionCoordinatorFactory(), this._collisionCoordinator.init(this)), this._collisionCoordinator; } /** * Gets the scene's rendering manager */ get renderingManager() { return this._renderingManager; } /** * Gets the list of frustum planes (built from the active camera) */ get frustumPlanes() { return this._frustumPlanes; } /** * Registers the transient components if needed. */ _registerTransientComponents() { if (this._transientComponents.length > 0) { for (const e of this._transientComponents) e.register(); this._transientComponents.length = 0; } } /** * @internal * Add a component to the scene. * Note that the ccomponent could be registered on th next frame if this is called after * the register component stage. * @param component Defines the component to add to the scene */ _addComponent(e) { this._components.push(e), this._transientComponents.push(e); const t = e; t.addFromContainer && t.serialize && this._serializableComponents.push(t); } /** * @internal * Gets a component from the scene. * @param name defines the name of the component to retrieve * @returns the component or null if not present */ _getComponent(e) { for (const t of this._components) if (t.name === e) return t; return null; } /** * Creates a new Scene * @param engine defines the engine to use to render this scene * @param options defines the scene options */ constructor(e, t) { super(), this._inputManager = new Xf(this), this.cameraToUseForPointers = null, this._isScene = !0, this._blockEntityCollection = !1, this.autoClear = !0, this.autoClearDepthAndStencil = !0, this.clearColor = new xt(0.2, 0.2, 0.3, 1), this.ambientColor = new Ne(0, 0, 0), this.environmentIntensity = 1, this._performancePriority = $H.BackwardCompatible, this.onScenePerformancePriorityChangedObservable = new Oe(), this._forceWireframe = !1, this._skipFrustumClipping = !1, this._forcePointsCloud = !1, this.animationsEnabled = !0, this._animationPropertiesOverride = null, this.useConstantAnimationDeltaTime = !1, this.constantlyUpdateMeshUnderPointer = !1, this.hoverCursor = "pointer", this.defaultCursor = "", this.doNotHandleCursors = !1, this.preventDefaultOnPointerDown = !0, this.preventDefaultOnPointerUp = !0, this.metadata = null, this.reservedDataStore = null, this.disableOfflineSupportExceptionRules = [], this.onDisposeObservable = new Oe(), this._onDisposeObserver = null, this.onBeforeRenderObservable = new Oe(), this._onBeforeRenderObserver = null, this.onAfterRenderObservable = new Oe(), this.onAfterRenderCameraObservable = new Oe(), this._onAfterRenderObserver = null, this.onBeforeAnimationsObservable = new Oe(), this.onAfterAnimationsObservable = new Oe(), this.onBeforeDrawPhaseObservable = new Oe(), this.onAfterDrawPhaseObservable = new Oe(), this.onReadyObservable = new Oe(), this.onBeforeCameraRenderObservable = new Oe(), this._onBeforeCameraRenderObserver = null, this.onAfterCameraRenderObservable = new Oe(), this._onAfterCameraRenderObserver = null, this.onBeforeActiveMeshesEvaluationObservable = new Oe(), this.onAfterActiveMeshesEvaluationObservable = new Oe(), this.onBeforeParticlesRenderingObservable = new Oe(), this.onAfterParticlesRenderingObservable = new Oe(), this.onDataLoadedObservable = new Oe(), this.onNewCameraAddedObservable = new Oe(), this.onCameraRemovedObservable = new Oe(), this.onNewLightAddedObservable = new Oe(), this.onLightRemovedObservable = new Oe(), this.onNewGeometryAddedObservable = new Oe(), this.onGeometryRemovedObservable = new Oe(), this.onNewTransformNodeAddedObservable = new Oe(), this.onTransformNodeRemovedObservable = new Oe(), this.onNewMeshAddedObservable = new Oe(), this.onMeshRemovedObservable = new Oe(), this.onNewSkeletonAddedObservable = new Oe(), this.onSkeletonRemovedObservable = new Oe(), this.onNewMaterialAddedObservable = new Oe(), this.onNewMultiMaterialAddedObservable = new Oe(), this.onMaterialRemovedObservable = new Oe(), this.onMultiMaterialRemovedObservable = new Oe(), this.onNewTextureAddedObservable = new Oe(), this.onTextureRemovedObservable = new Oe(), this.onBeforeRenderTargetsRenderObservable = new Oe(), this.onAfterRenderTargetsRenderObservable = new Oe(), this.onBeforeStepObservable = new Oe(), this.onAfterStepObservable = new Oe(), this.onActiveCameraChanged = new Oe(), this.onActiveCamerasChanged = new Oe(), this.onBeforeRenderingGroupObservable = new Oe(), this.onAfterRenderingGroupObservable = new Oe(), this.onMeshImportedObservable = new Oe(), this.onAnimationFileImportedObservable = new Oe(), this._registeredForLateAnimationBindings = new K8(256), this._pointerPickingConfiguration = new e6e(), this.onPrePointerObservable = new Oe(), this.onPointerObservable = new Oe(), this.onPreKeyboardObservable = new Oe(), this.onKeyboardObservable = new Oe(), this._useRightHandedSystem = !1, this._timeAccumulator = 0, this._currentStepId = 0, this._currentInternalStep = 0, this._fogEnabled = !0, this._fogMode = sr.FOGMODE_NONE, this.fogColor = new Ne(0.2, 0.2, 0.3), this.fogDensity = 0.1, this.fogStart = 0, this.fogEnd = 1e3, this.needsPreviousWorldMatrices = !1, this._shadowsEnabled = !0, this._lightsEnabled = !0, this._unObserveActiveCameras = null, this._texturesEnabled = !0, this.physicsEnabled = !0, this.particlesEnabled = !0, this.spritesEnabled = !0, this._skeletonsEnabled = !0, this.lensFlaresEnabled = !0, this.collisionsEnabled = !0, this.gravity = new S(0, -9.807, 0), this.postProcessesEnabled = !0, this.renderTargetsEnabled = !0, this.dumpNextRenderTargets = !1, this.customRenderTargets = [], this.importedMeshesFiles = [], this.probesEnabled = !0, this._meshesForIntersections = new K8(256), this.proceduralTexturesEnabled = !0, this._totalVertices = new v9(), this._activeIndices = new v9(), this._activeParticles = new v9(), this._activeBones = new v9(), this._animationTime = 0, this.animationTimeScale = 1, this._renderId = 0, this._frameId = 0, this._executeWhenReadyTimeoutId = null, this._intermediateRendering = !1, this._defaultFrameBufferCleared = !1, this._viewUpdateFlag = -1, this._projectionUpdateFlag = -1, this._toBeDisposed = new Array(256), this._activeRequests = new Array(), this._pendingData = new Array(), this._isDisposed = !1, this.dispatchAllSubMeshesOfActiveMeshes = !1, this._activeMeshes = new qf(256), this._processedMaterials = new qf(256), this._renderTargets = new K8(256), this._materialsRenderTargets = new K8(256), this._activeParticleSystems = new qf(256), this._activeSkeletons = new K8(32), this._softwareSkinnedMeshes = new K8(32), this._activeAnimatables = new Array(), this._transformMatrix = he.Zero(), this.requireLightSorting = !1, this._components = [], this._serializableComponents = [], this._transientComponents = [], this._beforeCameraUpdateStage = Y1.Create(), this._beforeClearStage = Y1.Create(), this._beforeRenderTargetClearStage = Y1.Create(), this._gatherRenderTargetsStage = Y1.Create(), this._gatherActiveCameraRenderTargetsStage = Y1.Create(), this._isReadyForMeshStage = Y1.Create(), this._beforeEvaluateActiveMeshStage = Y1.Create(), this._evaluateSubMeshStage = Y1.Create(), this._preActiveMeshStage = Y1.Create(), this._cameraDrawRenderTargetStage = Y1.Create(), this._beforeCameraDrawStage = Y1.Create(), this._beforeRenderTargetDrawStage = Y1.Create(), this._beforeRenderingGroupDrawStage = Y1.Create(), this._beforeRenderingMeshStage = Y1.Create(), this._afterRenderingMeshStage = Y1.Create(), this._afterRenderingGroupDrawStage = Y1.Create(), this._afterCameraDrawStage = Y1.Create(), this._afterCameraPostProcessStage = Y1.Create(), this._afterRenderTargetDrawStage = Y1.Create(), this._afterRenderTargetPostProcessStage = Y1.Create(), this._afterRenderStage = Y1.Create(), this._pointerMoveStage = Y1.Create(), this._pointerDownStage = Y1.Create(), this._pointerUpStage = Y1.Create(), this._geometriesByUniqueId = null, this._defaultMeshCandidates = { data: [], length: 0 }, this._defaultSubMeshCandidates = { data: [], length: 0 }, this._preventFreeActiveMeshesAndRenderingGroups = !1, this._activeMeshesFrozen = !1, this._activeMeshesFrozenButKeepClipping = !1, this._skipEvaluateActiveMeshesCompletely = !1, this._allowPostProcessClearColor = !0, this.getDeterministicFrameTime = () => this._engine.getTimeStep(), this._registeredActions = 0, this._blockMaterialDirtyMechanism = !1, this._perfCollector = null, this.activeCameras = []; const r = Object.assign({ useGeometryUniqueIdsMap: !0, useMaterialMeshMap: !0, useClonedMeshMap: !0, virtual: !1 }, t); e = this._engine = e || gr.LastCreatedEngine, r.virtual ? e._virtualScenes.push(this) : (gr._LastCreatedScene = this, e.scenes.push(this)), this._uid = null, this._renderingManager = new $6(this), yI && (this.postProcessManager = new yI(this)), u9() && this.attachControl(), this._createUbo(), Ui && (this._imageProcessingConfiguration = new Ui()), this.setDefaultCandidateProviders(), r.useGeometryUniqueIdsMap && (this._geometriesByUniqueId = {}), this.useMaterialMeshMap = r.useMaterialMeshMap, this.useClonedMeshMap = r.useClonedMeshMap, (!t || !t.virtual) && e.onNewSceneAddedObservable.notifyObservers(this); } /** * Gets a string identifying the name of the class * @returns "Scene" string */ getClassName() { return "Scene"; } /** * @internal */ _getDefaultMeshCandidates() { return this._defaultMeshCandidates.data = this.meshes, this._defaultMeshCandidates.length = this.meshes.length, this._defaultMeshCandidates; } /** * @internal */ _getDefaultSubMeshCandidates(e) { return this._defaultSubMeshCandidates.data = e.subMeshes, this._defaultSubMeshCandidates.length = e.subMeshes.length, this._defaultSubMeshCandidates; } /** * Sets the default candidate providers for the scene. * This sets the getActiveMeshCandidates, getActiveSubMeshCandidates, getIntersectingSubMeshCandidates * and getCollidingSubMeshCandidates to their default function */ setDefaultCandidateProviders() { this.getActiveMeshCandidates = () => this._getDefaultMeshCandidates(), this.getActiveSubMeshCandidates = (e) => this._getDefaultSubMeshCandidates(e), this.getIntersectingSubMeshCandidates = (e, t) => this._getDefaultSubMeshCandidates(e), this.getCollidingSubMeshCandidates = (e, t) => this._getDefaultSubMeshCandidates(e); } /** * Gets the mesh that is currently under the pointer */ get meshUnderPointer() { return this._inputManager.meshUnderPointer; } /** * Gets or sets the current on-screen X position of the pointer */ get pointerX() { return this._inputManager.pointerX; } set pointerX(e) { this._inputManager.pointerX = e; } /** * Gets or sets the current on-screen Y position of the pointer */ get pointerY() { return this._inputManager.pointerY; } set pointerY(e) { this._inputManager.pointerY = e; } /** * Gets the cached material (ie. the latest rendered one) * @returns the cached material */ getCachedMaterial() { return this._cachedMaterial; } /** * Gets the cached effect (ie. the latest rendered one) * @returns the cached effect */ getCachedEffect() { return this._cachedEffect; } /** * Gets the cached visibility state (ie. the latest rendered one) * @returns the cached visibility state */ getCachedVisibility() { return this._cachedVisibility; } /** * Gets a boolean indicating if the current material / effect / visibility must be bind again * @param material defines the current material * @param effect defines the current effect * @param visibility defines the current visibility state * @returns true if one parameter is not cached */ isCachedMaterialInvalid(e, t, r = 1) { return this._cachedEffect !== t || this._cachedMaterial !== e || this._cachedVisibility !== r; } /** * Gets the engine associated with the scene * @returns an Engine */ getEngine() { return this._engine; } /** * Gets the total number of vertices rendered per frame * @returns the total number of vertices rendered per frame */ getTotalVertices() { return this._totalVertices.current; } /** * Gets the performance counter for total vertices * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#instrumentation */ get totalVerticesPerfCounter() { return this._totalVertices; } /** * Gets the total number of active indices rendered per frame (You can deduce the number of rendered triangles by dividing this number by 3) * @returns the total number of active indices rendered per frame */ getActiveIndices() { return this._activeIndices.current; } /** * Gets the performance counter for active indices * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#instrumentation */ get totalActiveIndicesPerfCounter() { return this._activeIndices; } /** * Gets the total number of active particles rendered per frame * @returns the total number of active particles rendered per frame */ getActiveParticles() { return this._activeParticles.current; } /** * Gets the performance counter for active particles * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#instrumentation */ get activeParticlesPerfCounter() { return this._activeParticles; } /** * Gets the total number of active bones rendered per frame * @returns the total number of active bones rendered per frame */ getActiveBones() { return this._activeBones.current; } /** * Gets the performance counter for active bones * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#instrumentation */ get activeBonesPerfCounter() { return this._activeBones; } /** * Gets the array of active meshes * @returns an array of AbstractMesh */ getActiveMeshes() { return this._activeMeshes; } /** * Gets the animation ratio (which is 1.0 is the scene renders at 60fps and 2 if the scene renders at 30fps, etc.) * @returns a number */ getAnimationRatio() { return this._animationRatio !== void 0 ? this._animationRatio : 1; } /** * Gets an unique Id for the current render phase * @returns a number */ getRenderId() { return this._renderId; } /** * Gets an unique Id for the current frame * @returns a number */ getFrameId() { return this._frameId; } /** Call this function if you want to manually increment the render Id*/ incrementRenderId() { this._renderId++; } _createUbo() { this.setSceneUniformBuffer(this.createSceneUniformBuffer()); } /** * Use this method to simulate a pointer move on a mesh * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay * @param pickResult pickingInfo of the object wished to simulate pointer event on * @param pointerEventInit pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch) * @returns the current scene */ simulatePointerMove(e, t) { return this._inputManager.simulatePointerMove(e, t), this; } /** * Use this method to simulate a pointer down on a mesh * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay * @param pickResult pickingInfo of the object wished to simulate pointer event on * @param pointerEventInit pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch) * @returns the current scene */ simulatePointerDown(e, t) { return this._inputManager.simulatePointerDown(e, t), this; } /** * Use this method to simulate a pointer up on a mesh * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay * @param pickResult pickingInfo of the object wished to simulate pointer event on * @param pointerEventInit pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch) * @param doubleTap indicates that the pointer up event should be considered as part of a double click (false by default) * @returns the current scene */ simulatePointerUp(e, t, r) { return this._inputManager.simulatePointerUp(e, t, r), this; } /** * Gets a boolean indicating if the current pointer event is captured (meaning that the scene has already handled the pointer down) * @param pointerId defines the pointer id to use in a multi-touch scenario (0 by default) * @returns true if the pointer was captured */ isPointerCaptured(e = 0) { return this._inputManager.isPointerCaptured(e); } /** * Attach events to the canvas (To handle actionManagers triggers and raise onPointerMove, onPointerDown and onPointerUp * @param attachUp defines if you want to attach events to pointerup * @param attachDown defines if you want to attach events to pointerdown * @param attachMove defines if you want to attach events to pointermove */ attachControl(e = !0, t = !0, r = !0) { this._inputManager.attachControl(e, t, r); } /** Detaches all event handlers*/ detachControl() { this._inputManager.detachControl(); } /** * This function will check if the scene can be rendered (textures are loaded, shaders are compiled) * Delay loaded resources are not taking in account * @param checkRenderTargets true to also check that the meshes rendered as part of a render target are ready (default: true) * @returns true if all required resources are ready */ isReady(e = !0) { var t, r, n; if (this._isDisposed) return !1; let i; const s = this.getEngine(), a = s.currentRenderPassId; s.currentRenderPassId = (r = (t = this.activeCamera) === null || t === void 0 ? void 0 : t.renderPassId) !== null && r !== void 0 ? r : a; let f = !0; for (this._pendingData.length > 0 && (f = !1), (n = this.prePassRenderer) === null || n === void 0 || n.update(), this.useOrderIndependentTransparency && this.depthPeelingRenderer && f && (f = this.depthPeelingRenderer.isReady()), e && (this._processedMaterials.reset(), this._materialsRenderTargets.reset()), i = 0; i < this.meshes.length; i++) { const o = this.meshes[i]; if (!o.subMeshes || o.subMeshes.length === 0) continue; if (!o.isReady(!0)) { f = !1; continue; } const d = o.hasThinInstances || o.getClassName() === "InstancedMesh" || o.getClassName() === "InstancedLinesMesh" || s.getCaps().instancedArrays && o.instances.length > 0; for (const u of this._isReadyForMeshStage) u.action(o, d) || (f = !1); if (!e) continue; const v = o.material || this.defaultMaterial; if (v) if (v._storeEffectOnSubMeshes) for (const u of o.subMeshes) { const l = u.getMaterial(); l && l.hasRenderTargetTextures && l.getRenderTargetTextures != null && this._processedMaterials.indexOf(l) === -1 && (this._processedMaterials.push(l), this._materialsRenderTargets.concatWithNoDuplicate(l.getRenderTargetTextures())); } else v.hasRenderTargetTextures && v.getRenderTargetTextures != null && this._processedMaterials.indexOf(v) === -1 && (this._processedMaterials.push(v), this._materialsRenderTargets.concatWithNoDuplicate(v.getRenderTargetTextures())); } if (e) for (i = 0; i < this._materialsRenderTargets.length; ++i) this._materialsRenderTargets.data[i].isReadyForRendering() || (f = !1); for (i = 0; i < this.geometries.length; i++) this.geometries[i].delayLoadState === 2 && (f = !1); if (this.activeCameras && this.activeCameras.length > 0) for (const o of this.activeCameras) o.isReady(!0) || (f = !1); else this.activeCamera && (this.activeCamera.isReady(!0) || (f = !1)); for (const o of this.particleSystems) o.isReady() || (f = !1); if (this.layers) for (const o of this.layers) o.isReady() || (f = !1); return s.areAllEffectsReady() || (f = !1), s.currentRenderPassId = a, f; } /** Resets all cached information relative to material (including effect and visibility) */ resetCachedMaterial() { this._cachedMaterial = null, this._cachedEffect = null, this._cachedVisibility = null; } /** * Registers a function to be called before every frame render * @param func defines the function to register */ registerBeforeRender(e) { this.onBeforeRenderObservable.add(e); } /** * Unregisters a function called before every frame render * @param func defines the function to unregister */ unregisterBeforeRender(e) { this.onBeforeRenderObservable.removeCallback(e); } /** * Registers a function to be called after every frame render * @param func defines the function to register */ registerAfterRender(e) { this.onAfterRenderObservable.add(e); } /** * Unregisters a function called after every frame render * @param func defines the function to unregister */ unregisterAfterRender(e) { this.onAfterRenderObservable.removeCallback(e); } _executeOnceBeforeRender(e) { const t = () => { e(), setTimeout(() => { this.unregisterBeforeRender(t); }); }; this.registerBeforeRender(t); } /** * The provided function will run before render once and will be disposed afterwards. * A timeout delay can be provided so that the function will be executed in N ms. * The timeout is using the browser's native setTimeout so time percision cannot be guaranteed. * @param func The function to be executed. * @param timeout optional delay in ms */ executeOnceBeforeRender(e, t) { t !== void 0 ? setTimeout(() => { this._executeOnceBeforeRender(e); }, t) : this._executeOnceBeforeRender(e); } /** * This function can help adding any object to the list of data awaited to be ready in order to check for a complete scene loading. * @param data defines the object to wait for */ addPendingData(e) { this._pendingData.push(e); } /** * Remove a pending data from the loading list which has previously been added with addPendingData. * @param data defines the object to remove from the pending list */ removePendingData(e) { const t = this.isLoading, r = this._pendingData.indexOf(e); r !== -1 && this._pendingData.splice(r, 1), t && !this.isLoading && this.onDataLoadedObservable.notifyObservers(this); } /** * Returns the number of items waiting to be loaded * @returns the number of items waiting to be loaded */ getWaitingItemsCount() { return this._pendingData.length; } /** * Returns a boolean indicating if the scene is still loading data */ get isLoading() { return this._pendingData.length > 0; } /** * Registers a function to be executed when the scene is ready * @param func - the function to be executed * @param checkRenderTargets true to also check that the meshes rendered as part of a render target are ready (default: false) */ executeWhenReady(e, t = !1) { this.onReadyObservable.addOnce(e), this._executeWhenReadyTimeoutId === null && this._checkIsReady(t); } /** * Returns a promise that resolves when the scene is ready * @param checkRenderTargets true to also check that the meshes rendered as part of a render target are ready (default: false) * @returns A promise that resolves when the scene is ready */ whenReadyAsync(e = !1) { return new Promise((t) => { this.executeWhenReady(() => { t(); }, e); }); } /** * @internal */ _checkIsReady(e = !1) { if (this._registerTransientComponents(), this.isReady(e)) { this.onReadyObservable.notifyObservers(this), this.onReadyObservable.clear(), this._executeWhenReadyTimeoutId = null; return; } if (this._isDisposed) { this.onReadyObservable.clear(), this._executeWhenReadyTimeoutId = null; return; } this._executeWhenReadyTimeoutId = setTimeout(() => { this.incrementRenderId(), this._checkIsReady(e); }, 100); } /** * Gets all animatable attached to the scene */ get animatables() { return this._activeAnimatables; } /** * Resets the last animation time frame. * Useful to override when animations start running when loading a scene for the first time. */ resetLastAnimationTimeFrame() { this._animationTimeLast = Yi.Now; } // Matrix /** * Gets the current view matrix * @returns a Matrix */ getViewMatrix() { return this._viewMatrix; } /** * Gets the current projection matrix * @returns a Matrix */ getProjectionMatrix() { return this._projectionMatrix; } /** * Gets the current transform matrix * @returns a Matrix made of View * Projection */ getTransformMatrix() { return this._transformMatrix; } /** * Sets the current transform matrix * @param viewL defines the View matrix to use * @param projectionL defines the Projection matrix to use * @param viewR defines the right View matrix to use (if provided) * @param projectionR defines the right Projection matrix to use (if provided) */ setTransformMatrix(e, t, r, n) { !r && !n && this._multiviewSceneUbo && (this._multiviewSceneUbo.dispose(), this._multiviewSceneUbo = null), !(this._viewUpdateFlag === e.updateFlag && this._projectionUpdateFlag === t.updateFlag) && (this._viewUpdateFlag = e.updateFlag, this._projectionUpdateFlag = t.updateFlag, this._viewMatrix = e, this._projectionMatrix = t, this._viewMatrix.multiplyToRef(this._projectionMatrix, this._transformMatrix), this._frustumPlanes ? Xc.GetPlanesToRef(this._transformMatrix, this._frustumPlanes) : this._frustumPlanes = Xc.GetPlanes(this._transformMatrix), this._multiviewSceneUbo && this._multiviewSceneUbo.useUbo ? this._updateMultiviewUbo(r, n) : this._sceneUbo.useUbo && (this._sceneUbo.updateMatrix("viewProjection", this._transformMatrix), this._sceneUbo.updateMatrix("view", this._viewMatrix), this._sceneUbo.updateMatrix("projection", this._projectionMatrix))); } /** * Gets the uniform buffer used to store scene data * @returns a UniformBuffer */ getSceneUniformBuffer() { return this._multiviewSceneUbo ? this._multiviewSceneUbo : this._sceneUbo; } /** * Creates a scene UBO * @param name name of the uniform buffer (optional, for debugging purpose only) * @returns a new ubo */ createSceneUniformBuffer(e) { const t = new yr(this._engine, void 0, !1, e ?? "scene"); return t.addUniform("viewProjection", 16), t.addUniform("view", 16), t.addUniform("projection", 16), t.addUniform("vEyePosition", 4), t; } /** * Sets the scene ubo * @param ubo the ubo to set for the scene */ setSceneUniformBuffer(e) { this._sceneUbo = e, this._viewUpdateFlag = -1, this._projectionUpdateFlag = -1; } /** * Gets an unique (relatively to the current scene) Id * @returns an unique number for the scene */ getUniqueId() { return bR.UniqueId; } /** * Add a mesh to the list of scene's meshes * @param newMesh defines the mesh to add * @param recursive if all child meshes should also be added to the scene */ addMesh(e, t = !1) { this._blockEntityCollection || (this.meshes.push(e), e._resyncLightSources(), e.parent || e._addToSceneRootNodes(), this.onNewMeshAddedObservable.notifyObservers(e), t && e.getChildMeshes().forEach((r) => { this.addMesh(r); })); } /** * Remove a mesh for the list of scene's meshes * @param toRemove defines the mesh to remove * @param recursive if all child meshes should also be removed from the scene * @returns the index where the mesh was in the mesh list */ removeMesh(e, t = !1) { const r = this.meshes.indexOf(e); return r !== -1 && (this.meshes[r] = this.meshes[this.meshes.length - 1], this.meshes.pop(), e.parent || e._removeFromSceneRootNodes()), this._inputManager._invalidateMesh(e), this.onMeshRemovedObservable.notifyObservers(e), t && e.getChildMeshes().forEach((n) => { this.removeMesh(n); }), r; } /** * Add a transform node to the list of scene's transform nodes * @param newTransformNode defines the transform node to add */ addTransformNode(e) { this._blockEntityCollection || e.getScene() === this && e._indexInSceneTransformNodesArray !== -1 || (e._indexInSceneTransformNodesArray = this.transformNodes.length, this.transformNodes.push(e), e.parent || e._addToSceneRootNodes(), this.onNewTransformNodeAddedObservable.notifyObservers(e)); } /** * Remove a transform node for the list of scene's transform nodes * @param toRemove defines the transform node to remove * @returns the index where the transform node was in the transform node list */ removeTransformNode(e) { const t = e._indexInSceneTransformNodesArray; if (t !== -1) { if (t !== this.transformNodes.length - 1) { const r = this.transformNodes[this.transformNodes.length - 1]; this.transformNodes[t] = r, r._indexInSceneTransformNodesArray = t; } e._indexInSceneTransformNodesArray = -1, this.transformNodes.pop(), e.parent || e._removeFromSceneRootNodes(); } return this.onTransformNodeRemovedObservable.notifyObservers(e), t; } /** * Remove a skeleton for the list of scene's skeletons * @param toRemove defines the skeleton to remove * @returns the index where the skeleton was in the skeleton list */ removeSkeleton(e) { const t = this.skeletons.indexOf(e); return t !== -1 && (this.skeletons.splice(t, 1), this.onSkeletonRemovedObservable.notifyObservers(e), this._executeActiveContainerCleanup(this._activeSkeletons)), t; } /** * Remove a morph target for the list of scene's morph targets * @param toRemove defines the morph target to remove * @returns the index where the morph target was in the morph target list */ removeMorphTargetManager(e) { const t = this.morphTargetManagers.indexOf(e); return t !== -1 && this.morphTargetManagers.splice(t, 1), t; } /** * Remove a light for the list of scene's lights * @param toRemove defines the light to remove * @returns the index where the light was in the light list */ removeLight(e) { const t = this.lights.indexOf(e); if (t !== -1) { for (const r of this.meshes) r._removeLightSource(e, !1); this.lights.splice(t, 1), this.sortLightsByPriority(), e.parent || e._removeFromSceneRootNodes(); } return this.onLightRemovedObservable.notifyObservers(e), t; } /** * Remove a camera for the list of scene's cameras * @param toRemove defines the camera to remove * @returns the index where the camera was in the camera list */ removeCamera(e) { const t = this.cameras.indexOf(e); if (t !== -1 && (this.cameras.splice(t, 1), e.parent || e._removeFromSceneRootNodes()), this.activeCameras) { const r = this.activeCameras.indexOf(e); r !== -1 && this.activeCameras.splice(r, 1); } return this.activeCamera === e && (this.cameras.length > 0 ? this.activeCamera = this.cameras[0] : this.activeCamera = null), this.onCameraRemovedObservable.notifyObservers(e), t; } /** * Remove a particle system for the list of scene's particle systems * @param toRemove defines the particle system to remove * @returns the index where the particle system was in the particle system list */ removeParticleSystem(e) { const t = this.particleSystems.indexOf(e); return t !== -1 && (this.particleSystems.splice(t, 1), this._executeActiveContainerCleanup(this._activeParticleSystems)), t; } /** * Remove a animation for the list of scene's animations * @param toRemove defines the animation to remove * @returns the index where the animation was in the animation list */ removeAnimation(e) { const t = this.animations.indexOf(e); return t !== -1 && this.animations.splice(t, 1), t; } /** * Will stop the animation of the given target * @param target - the target * @param animationName - the name of the animation to stop (all animations will be stopped if both this and targetMask are empty) * @param targetMask - a function that determines if the animation should be stopped based on its target (all animations will be stopped if both this and animationName are empty) */ stopAnimation(e, t, r) { } /** * Removes the given animation group from this scene. * @param toRemove The animation group to remove * @returns The index of the removed animation group */ removeAnimationGroup(e) { const t = this.animationGroups.indexOf(e); return t !== -1 && this.animationGroups.splice(t, 1), t; } /** * Removes the given multi-material from this scene. * @param toRemove The multi-material to remove * @returns The index of the removed multi-material */ removeMultiMaterial(e) { const t = this.multiMaterials.indexOf(e); return t !== -1 && this.multiMaterials.splice(t, 1), this.onMultiMaterialRemovedObservable.notifyObservers(e), t; } /** * Removes the given material from this scene. * @param toRemove The material to remove * @returns The index of the removed material */ removeMaterial(e) { const t = e._indexInSceneMaterialArray; if (t !== -1 && t < this.materials.length) { if (t !== this.materials.length - 1) { const r = this.materials[this.materials.length - 1]; this.materials[t] = r, r._indexInSceneMaterialArray = t; } e._indexInSceneMaterialArray = -1, this.materials.pop(); } return this.onMaterialRemovedObservable.notifyObservers(e), t; } /** * Removes the given action manager from this scene. * @deprecated * @param toRemove The action manager to remove * @returns The index of the removed action manager */ removeActionManager(e) { const t = this.actionManagers.indexOf(e); return t !== -1 && this.actionManagers.splice(t, 1), t; } /** * Removes the given texture from this scene. * @param toRemove The texture to remove * @returns The index of the removed texture */ removeTexture(e) { const t = this.textures.indexOf(e); return t !== -1 && this.textures.splice(t, 1), this.onTextureRemovedObservable.notifyObservers(e), t; } /** * Adds the given light to this scene * @param newLight The light to add */ addLight(e) { if (!this._blockEntityCollection) { this.lights.push(e), this.sortLightsByPriority(), e.parent || e._addToSceneRootNodes(); for (const t of this.meshes) t.lightSources.indexOf(e) === -1 && (t.lightSources.push(e), t._resyncLightSources()); this.onNewLightAddedObservable.notifyObservers(e); } } /** * Sorts the list list based on light priorities */ sortLightsByPriority() { this.requireLightSorting && this.lights.sort(ra.CompareLightsPriority); } /** * Adds the given camera to this scene * @param newCamera The camera to add */ addCamera(e) { this._blockEntityCollection || (this.cameras.push(e), this.onNewCameraAddedObservable.notifyObservers(e), e.parent || e._addToSceneRootNodes()); } /** * Adds the given skeleton to this scene * @param newSkeleton The skeleton to add */ addSkeleton(e) { this._blockEntityCollection || (this.skeletons.push(e), this.onNewSkeletonAddedObservable.notifyObservers(e)); } /** * Adds the given particle system to this scene * @param newParticleSystem The particle system to add */ addParticleSystem(e) { this._blockEntityCollection || this.particleSystems.push(e); } /** * Adds the given animation to this scene * @param newAnimation The animation to add */ addAnimation(e) { this._blockEntityCollection || this.animations.push(e); } /** * Adds the given animation group to this scene. * @param newAnimationGroup The animation group to add */ addAnimationGroup(e) { this._blockEntityCollection || this.animationGroups.push(e); } /** * Adds the given multi-material to this scene * @param newMultiMaterial The multi-material to add */ addMultiMaterial(e) { this._blockEntityCollection || (this.multiMaterials.push(e), this.onNewMultiMaterialAddedObservable.notifyObservers(e)); } /** * Adds the given material to this scene * @param newMaterial The material to add */ addMaterial(e) { this._blockEntityCollection || e.getScene() === this && e._indexInSceneMaterialArray !== -1 || (e._indexInSceneMaterialArray = this.materials.length, this.materials.push(e), this.onNewMaterialAddedObservable.notifyObservers(e)); } /** * Adds the given morph target to this scene * @param newMorphTargetManager The morph target to add */ addMorphTargetManager(e) { this._blockEntityCollection || this.morphTargetManagers.push(e); } /** * Adds the given geometry to this scene * @param newGeometry The geometry to add */ addGeometry(e) { this._blockEntityCollection || (this._geometriesByUniqueId && (this._geometriesByUniqueId[e.uniqueId] = this.geometries.length), this.geometries.push(e)); } /** * Adds the given action manager to this scene * @deprecated * @param newActionManager The action manager to add */ addActionManager(e) { this.actionManagers.push(e); } /** * Adds the given texture to this scene. * @param newTexture The texture to add */ addTexture(e) { this._blockEntityCollection || (this.textures.push(e), this.onNewTextureAddedObservable.notifyObservers(e)); } /** * Switch active camera * @param newCamera defines the new active camera * @param attachControl defines if attachControl must be called for the new active camera (default: true) */ switchActiveCamera(e, t = !0) { this._engine.getInputElement() && (this.activeCamera && this.activeCamera.detachControl(), this.activeCamera = e, t && e.attachControl()); } /** * sets the active camera of the scene using its Id * @param id defines the camera's Id * @returns the new active camera or null if none found. */ setActiveCameraById(e) { const t = this.getCameraById(e); return t ? (this.activeCamera = t, t) : null; } /** * sets the active camera of the scene using its name * @param name defines the camera's name * @returns the new active camera or null if none found. */ setActiveCameraByName(e) { const t = this.getCameraByName(e); return t ? (this.activeCamera = t, t) : null; } /** * get an animation group using its name * @param name defines the material's name * @returns the animation group or null if none found. */ getAnimationGroupByName(e) { for (let t = 0; t < this.animationGroups.length; t++) if (this.animationGroups[t].name === e) return this.animationGroups[t]; return null; } _getMaterial(e, t) { for (let r = 0; r < this.materials.length; r++) { const n = this.materials[r]; if (t(n)) return n; } if (e) for (let r = 0; r < this.multiMaterials.length; r++) { const n = this.multiMaterials[r]; if (t(n)) return n; } return null; } /** * Get a material using its unique id * @param uniqueId defines the material's unique id * @param allowMultiMaterials determines whether multimaterials should be considered * @returns the material or null if none found. */ getMaterialByUniqueID(e, t = !1) { return this._getMaterial(t, (r) => r.uniqueId === e); } /** * get a material using its id * @param id defines the material's Id * @param allowMultiMaterials determines whether multimaterials should be considered * @returns the material or null if none found. */ getMaterialById(e, t = !1) { return this._getMaterial(t, (r) => r.id === e); } /** * Gets a material using its name * @param name defines the material's name * @param allowMultiMaterials determines whether multimaterials should be considered * @returns the material or null if none found. */ getMaterialByName(e, t = !1) { return this._getMaterial(t, (r) => r.name === e); } /** * Gets a last added material using a given id * @param id defines the material's id * @param allowMultiMaterials determines whether multimaterials should be considered * @returns the last material with the given id or null if none found. */ getLastMaterialById(e, t = !1) { for (let r = this.materials.length - 1; r >= 0; r--) if (this.materials[r].id === e) return this.materials[r]; if (t) { for (let r = this.multiMaterials.length - 1; r >= 0; r--) if (this.multiMaterials[r].id === e) return this.multiMaterials[r]; } return null; } /** * Get a texture using its unique id * @param uniqueId defines the texture's unique id * @returns the texture or null if none found. */ getTextureByUniqueId(e) { for (let t = 0; t < this.textures.length; t++) if (this.textures[t].uniqueId === e) return this.textures[t]; return null; } /** * Gets a texture using its name * @param name defines the texture's name * @returns the texture or null if none found. */ getTextureByName(e) { for (let t = 0; t < this.textures.length; t++) if (this.textures[t].name === e) return this.textures[t]; return null; } /** * Gets a camera using its Id * @param id defines the Id to look for * @returns the camera or null if not found */ getCameraById(e) { for (let t = 0; t < this.cameras.length; t++) if (this.cameras[t].id === e) return this.cameras[t]; return null; } /** * Gets a camera using its unique Id * @param uniqueId defines the unique Id to look for * @returns the camera or null if not found */ getCameraByUniqueId(e) { for (let t = 0; t < this.cameras.length; t++) if (this.cameras[t].uniqueId === e) return this.cameras[t]; return null; } /** * Gets a camera using its name * @param name defines the camera's name * @returns the camera or null if none found. */ getCameraByName(e) { for (let t = 0; t < this.cameras.length; t++) if (this.cameras[t].name === e) return this.cameras[t]; return null; } /** * Gets a bone using its Id * @param id defines the bone's Id * @returns the bone or null if not found */ getBoneById(e) { for (let t = 0; t < this.skeletons.length; t++) { const r = this.skeletons[t]; for (let n = 0; n < r.bones.length; n++) if (r.bones[n].id === e) return r.bones[n]; } return null; } /** * Gets a bone using its id * @param name defines the bone's name * @returns the bone or null if not found */ getBoneByName(e) { for (let t = 0; t < this.skeletons.length; t++) { const r = this.skeletons[t]; for (let n = 0; n < r.bones.length; n++) if (r.bones[n].name === e) return r.bones[n]; } return null; } /** * Gets a light node using its name * @param name defines the light's name * @returns the light or null if none found. */ getLightByName(e) { for (let t = 0; t < this.lights.length; t++) if (this.lights[t].name === e) return this.lights[t]; return null; } /** * Gets a light node using its Id * @param id defines the light's Id * @returns the light or null if none found. */ getLightById(e) { for (let t = 0; t < this.lights.length; t++) if (this.lights[t].id === e) return this.lights[t]; return null; } /** * Gets a light node using its scene-generated unique Id * @param uniqueId defines the light's unique Id * @returns the light or null if none found. */ getLightByUniqueId(e) { for (let t = 0; t < this.lights.length; t++) if (this.lights[t].uniqueId === e) return this.lights[t]; return null; } /** * Gets a particle system by Id * @param id defines the particle system Id * @returns the corresponding system or null if none found */ getParticleSystemById(e) { for (let t = 0; t < this.particleSystems.length; t++) if (this.particleSystems[t].id === e) return this.particleSystems[t]; return null; } /** * Gets a geometry using its Id * @param id defines the geometry's Id * @returns the geometry or null if none found. */ getGeometryById(e) { for (let t = 0; t < this.geometries.length; t++) if (this.geometries[t].id === e) return this.geometries[t]; return null; } _getGeometryByUniqueId(e) { if (this._geometriesByUniqueId) { const t = this._geometriesByUniqueId[e]; if (t !== void 0) return this.geometries[t]; } else for (let t = 0; t < this.geometries.length; t++) if (this.geometries[t].uniqueId === e) return this.geometries[t]; return null; } /** * Add a new geometry to this scene * @param geometry defines the geometry to be added to the scene. * @param force defines if the geometry must be pushed even if a geometry with this id already exists * @returns a boolean defining if the geometry was added or not */ pushGeometry(e, t) { return !t && this._getGeometryByUniqueId(e.uniqueId) ? !1 : (this.addGeometry(e), this.onNewGeometryAddedObservable.notifyObservers(e), !0); } /** * Removes an existing geometry * @param geometry defines the geometry to be removed from the scene * @returns a boolean defining if the geometry was removed or not */ removeGeometry(e) { let t; if (this._geometriesByUniqueId) { if (t = this._geometriesByUniqueId[e.uniqueId], t === void 0) return !1; } else if (t = this.geometries.indexOf(e), t < 0) return !1; if (t !== this.geometries.length - 1) { const r = this.geometries[this.geometries.length - 1]; r && (this.geometries[t] = r, this._geometriesByUniqueId && (this._geometriesByUniqueId[r.uniqueId] = t)); } return this._geometriesByUniqueId && (this._geometriesByUniqueId[e.uniqueId] = void 0), this.geometries.pop(), this.onGeometryRemovedObservable.notifyObservers(e), !0; } /** * Gets the list of geometries attached to the scene * @returns an array of Geometry */ getGeometries() { return this.geometries; } /** * Gets the first added mesh found of a given Id * @param id defines the Id to search for * @returns the mesh found or null if not found at all */ getMeshById(e) { for (let t = 0; t < this.meshes.length; t++) if (this.meshes[t].id === e) return this.meshes[t]; return null; } /** * Gets a list of meshes using their Id * @param id defines the Id to search for * @returns a list of meshes */ getMeshesById(e) { return this.meshes.filter(function(t) { return t.id === e; }); } /** * Gets the first added transform node found of a given Id * @param id defines the Id to search for * @returns the found transform node or null if not found at all. */ getTransformNodeById(e) { for (let t = 0; t < this.transformNodes.length; t++) if (this.transformNodes[t].id === e) return this.transformNodes[t]; return null; } /** * Gets a transform node with its auto-generated unique Id * @param uniqueId defines the unique Id to search for * @returns the found transform node or null if not found at all. */ getTransformNodeByUniqueId(e) { for (let t = 0; t < this.transformNodes.length; t++) if (this.transformNodes[t].uniqueId === e) return this.transformNodes[t]; return null; } /** * Gets a list of transform nodes using their Id * @param id defines the Id to search for * @returns a list of transform nodes */ getTransformNodesById(e) { return this.transformNodes.filter(function(t) { return t.id === e; }); } /** * Gets a mesh with its auto-generated unique Id * @param uniqueId defines the unique Id to search for * @returns the found mesh or null if not found at all. */ getMeshByUniqueId(e) { for (let t = 0; t < this.meshes.length; t++) if (this.meshes[t].uniqueId === e) return this.meshes[t]; return null; } /** * Gets a the last added mesh using a given Id * @param id defines the Id to search for * @returns the found mesh or null if not found at all. */ getLastMeshById(e) { for (let t = this.meshes.length - 1; t >= 0; t--) if (this.meshes[t].id === e) return this.meshes[t]; return null; } /** * Gets a the last transform node using a given Id * @param id defines the Id to search for * @returns the found mesh or null if not found at all. */ getLastTransformNodeById(e) { for (let t = this.transformNodes.length - 1; t >= 0; t--) if (this.transformNodes[t].id === e) return this.transformNodes[t]; return null; } /** * Gets a the last added node (Mesh, Camera, Light) using a given Id * @param id defines the Id to search for * @returns the found node or null if not found at all */ getLastEntryById(e) { let t; for (t = this.meshes.length - 1; t >= 0; t--) if (this.meshes[t].id === e) return this.meshes[t]; for (t = this.transformNodes.length - 1; t >= 0; t--) if (this.transformNodes[t].id === e) return this.transformNodes[t]; for (t = this.cameras.length - 1; t >= 0; t--) if (this.cameras[t].id === e) return this.cameras[t]; for (t = this.lights.length - 1; t >= 0; t--) if (this.lights[t].id === e) return this.lights[t]; return null; } /** * Gets a node (Mesh, Camera, Light) using a given Id * @param id defines the Id to search for * @returns the found node or null if not found at all */ getNodeById(e) { const t = this.getMeshById(e); if (t) return t; const r = this.getTransformNodeById(e); if (r) return r; const n = this.getLightById(e); if (n) return n; const i = this.getCameraById(e); if (i) return i; const s = this.getBoneById(e); return s || null; } /** * Gets a node (Mesh, Camera, Light) using a given name * @param name defines the name to search for * @returns the found node or null if not found at all. */ getNodeByName(e) { const t = this.getMeshByName(e); if (t) return t; const r = this.getTransformNodeByName(e); if (r) return r; const n = this.getLightByName(e); if (n) return n; const i = this.getCameraByName(e); if (i) return i; const s = this.getBoneByName(e); return s || null; } /** * Gets a mesh using a given name * @param name defines the name to search for * @returns the found mesh or null if not found at all. */ getMeshByName(e) { for (let t = 0; t < this.meshes.length; t++) if (this.meshes[t].name === e) return this.meshes[t]; return null; } /** * Gets a transform node using a given name * @param name defines the name to search for * @returns the found transform node or null if not found at all. */ getTransformNodeByName(e) { for (let t = 0; t < this.transformNodes.length; t++) if (this.transformNodes[t].name === e) return this.transformNodes[t]; return null; } /** * Gets a skeleton using a given Id (if many are found, this function will pick the last one) * @param id defines the Id to search for * @returns the found skeleton or null if not found at all. */ getLastSkeletonById(e) { for (let t = this.skeletons.length - 1; t >= 0; t--) if (this.skeletons[t].id === e) return this.skeletons[t]; return null; } /** * Gets a skeleton using a given auto generated unique id * @param uniqueId defines the unique id to search for * @returns the found skeleton or null if not found at all. */ getSkeletonByUniqueId(e) { for (let t = 0; t < this.skeletons.length; t++) if (this.skeletons[t].uniqueId === e) return this.skeletons[t]; return null; } /** * Gets a skeleton using a given id (if many are found, this function will pick the first one) * @param id defines the id to search for * @returns the found skeleton or null if not found at all. */ getSkeletonById(e) { for (let t = 0; t < this.skeletons.length; t++) if (this.skeletons[t].id === e) return this.skeletons[t]; return null; } /** * Gets a skeleton using a given name * @param name defines the name to search for * @returns the found skeleton or null if not found at all. */ getSkeletonByName(e) { for (let t = 0; t < this.skeletons.length; t++) if (this.skeletons[t].name === e) return this.skeletons[t]; return null; } /** * Gets a morph target manager using a given id (if many are found, this function will pick the last one) * @param id defines the id to search for * @returns the found morph target manager or null if not found at all. */ getMorphTargetManagerById(e) { for (let t = 0; t < this.morphTargetManagers.length; t++) if (this.morphTargetManagers[t].uniqueId === e) return this.morphTargetManagers[t]; return null; } /** * Gets a morph target using a given id (if many are found, this function will pick the first one) * @param id defines the id to search for * @returns the found morph target or null if not found at all. */ getMorphTargetById(e) { for (let t = 0; t < this.morphTargetManagers.length; ++t) { const r = this.morphTargetManagers[t]; for (let n = 0; n < r.numTargets; ++n) { const i = r.getTarget(n); if (i.id === e) return i; } } return null; } /** * Gets a morph target using a given name (if many are found, this function will pick the first one) * @param name defines the name to search for * @returns the found morph target or null if not found at all. */ getMorphTargetByName(e) { for (let t = 0; t < this.morphTargetManagers.length; ++t) { const r = this.morphTargetManagers[t]; for (let n = 0; n < r.numTargets; ++n) { const i = r.getTarget(n); if (i.name === e) return i; } } return null; } /** * Gets a post process using a given name (if many are found, this function will pick the first one) * @param name defines the name to search for * @returns the found post process or null if not found at all. */ getPostProcessByName(e) { for (let t = 0; t < this.postProcesses.length; ++t) { const r = this.postProcesses[t]; if (r.name === e) return r; } return null; } /** * Gets a boolean indicating if the given mesh is active * @param mesh defines the mesh to look for * @returns true if the mesh is in the active list */ isActiveMesh(e) { return this._activeMeshes.indexOf(e) !== -1; } /** * Return a unique id as a string which can serve as an identifier for the scene */ get uid() { return this._uid || (this._uid = ye.RandomId()), this._uid; } /** * Add an externally attached data from its key. * This method call will fail and return false, if such key already exists. * If you don't care and just want to get the data no matter what, use the more convenient getOrAddExternalDataWithFactory() method. * @param key the unique key that identifies the data * @param data the data object to associate to the key for this Engine instance * @returns true if no such key were already present and the data was added successfully, false otherwise */ addExternalData(e, t) { return this._externalData || (this._externalData = new YC()), this._externalData.add(e, t); } /** * Get an externally attached data from its key * @param key the unique key that identifies the data * @returns the associated data, if present (can be null), or undefined if not present */ getExternalData(e) { return this._externalData ? this._externalData.get(e) : null; } /** * Get an externally attached data from its key, create it using a factory if it's not already present * @param key the unique key that identifies the data * @param factory the factory that will be called to create the instance if and only if it doesn't exists * @returns the associated data, can be null if the factory returned null. */ getOrAddExternalDataWithFactory(e, t) { return this._externalData || (this._externalData = new YC()), this._externalData.getOrAddWithFactory(e, t); } /** * Remove an externally attached data from the Engine instance * @param key the unique key that identifies the data * @returns true if the data was successfully removed, false if it doesn't exist */ removeExternalData(e) { return this._externalData.remove(e); } _evaluateSubMesh(e, t, r, n) { if (n || e.isInFrustum(this._frustumPlanes)) { for (const s of this._evaluateSubMeshStage) s.action(t, e); const i = e.getMaterial(); i != null && (i.hasRenderTargetTextures && i.getRenderTargetTextures != null && this._processedMaterials.indexOf(i) === -1 && (this._processedMaterials.push(i), this._materialsRenderTargets.concatWithNoDuplicate(i.getRenderTargetTextures())), this._renderingManager.dispatch(e, t, i)); } } /** * Clear the processed materials smart array preventing retention point in material dispose. */ freeProcessedMaterials() { this._processedMaterials.dispose(); } /** Gets or sets a boolean blocking all the calls to freeActiveMeshes and freeRenderingGroups * It can be used in order to prevent going through methods freeRenderingGroups and freeActiveMeshes several times to improve performance * when disposing several meshes in a row or a hierarchy of meshes. * When used, it is the responsibility of the user to blockfreeActiveMeshesAndRenderingGroups back to false. */ get blockfreeActiveMeshesAndRenderingGroups() { return this._preventFreeActiveMeshesAndRenderingGroups; } set blockfreeActiveMeshesAndRenderingGroups(e) { this._preventFreeActiveMeshesAndRenderingGroups !== e && (e && (this.freeActiveMeshes(), this.freeRenderingGroups()), this._preventFreeActiveMeshesAndRenderingGroups = e); } /** * Clear the active meshes smart array preventing retention point in mesh dispose. */ freeActiveMeshes() { if (!this.blockfreeActiveMeshesAndRenderingGroups && (this._activeMeshes.dispose(), this.activeCamera && this.activeCamera._activeMeshes && this.activeCamera._activeMeshes.dispose(), this.activeCameras)) for (let e = 0; e < this.activeCameras.length; e++) { const t = this.activeCameras[e]; t && t._activeMeshes && t._activeMeshes.dispose(); } } /** * Clear the info related to rendering groups preventing retention points during dispose. */ freeRenderingGroups() { if (!this.blockfreeActiveMeshesAndRenderingGroups && (this._renderingManager && this._renderingManager.freeRenderingGroups(), this.textures)) for (let e = 0; e < this.textures.length; e++) { const t = this.textures[e]; t && t.renderList && t.freeRenderingGroups(); } } /** @internal */ _isInIntermediateRendering() { return this._intermediateRendering; } /** * Use this function to stop evaluating active meshes. The current list will be keep alive between frames * @param skipEvaluateActiveMeshes defines an optional boolean indicating that the evaluate active meshes step must be completely skipped * @param onSuccess optional success callback * @param onError optional error callback * @param freezeMeshes defines if meshes should be frozen (true by default) * @param keepFrustumCulling defines if you want to keep running the frustum clipping (false by default) * @returns the current scene */ freezeActiveMeshes(e = !1, t, r, n = !0, i = !1) { return this.executeWhenReady(() => { if (!this.activeCamera) { r && r("No active camera found"); return; } if (this._frustumPlanes || this.updateTransformMatrix(), this._evaluateActiveMeshes(), this._activeMeshesFrozen = !0, this._activeMeshesFrozenButKeepClipping = i, this._skipEvaluateActiveMeshesCompletely = e, n) for (let s = 0; s < this._activeMeshes.length; s++) this._activeMeshes.data[s]._freeze(); t && t(); }), this; } /** * Use this function to restart evaluating active meshes on every frame * @returns the current scene */ unfreezeActiveMeshes() { for (let e = 0; e < this.meshes.length; e++) { const t = this.meshes[e]; t._internalAbstractMeshDataInfo && (t._internalAbstractMeshDataInfo._isActive = !1); } for (let e = 0; e < this._activeMeshes.length; e++) this._activeMeshes.data[e]._unFreeze(); return this._activeMeshesFrozen = !1, this; } _executeActiveContainerCleanup(e) { !(this._engine.snapshotRendering && this._engine.snapshotRenderingMode === 1) && this._activeMeshesFrozen && this._activeMeshes.length || this.onBeforeRenderObservable.addOnce(() => e.dispose()); } _evaluateActiveMeshes() { var e; if (this._engine.snapshotRendering && this._engine.snapshotRenderingMode === 1) { this._activeMeshes.length > 0 && ((e = this.activeCamera) === null || e === void 0 || e._activeMeshes.reset(), this._activeMeshes.reset(), this._renderingManager.reset(), this._processedMaterials.reset(), this._activeParticleSystems.reset(), this._activeSkeletons.reset(), this._softwareSkinnedMeshes.reset()); return; } if (this._activeMeshesFrozen && this._activeMeshes.length) { if (!this._skipEvaluateActiveMeshesCompletely) { const n = this._activeMeshes.length; for (let i = 0; i < n; i++) this._activeMeshes.data[i].computeWorldMatrix(); } if (this._activeParticleSystems) { const n = this._activeParticleSystems.length; for (let i = 0; i < n; i++) this._activeParticleSystems.data[i].animate(); } this._renderingManager.resetSprites(); return; } if (!this.activeCamera) return; this.onBeforeActiveMeshesEvaluationObservable.notifyObservers(this), this.activeCamera._activeMeshes.reset(), this._activeMeshes.reset(), this._renderingManager.reset(), this._processedMaterials.reset(), this._activeParticleSystems.reset(), this._activeSkeletons.reset(), this._softwareSkinnedMeshes.reset(), this._materialsRenderTargets.reset(); for (const n of this._beforeEvaluateActiveMeshStage) n.action(); const t = this.getActiveMeshCandidates(), r = t.length; for (let n = 0; n < r; n++) { const i = t.data[n]; if (i._internalAbstractMeshDataInfo._currentLODIsUpToDate = !1, i.isBlocked || (this._totalVertices.addCount(i.getTotalVertices(), !1), !i.isReady() || !i.isEnabled() || i.scaling.hasAZeroComponent)) continue; i.computeWorldMatrix(), i.actionManager && i.actionManager.hasSpecificTriggers2(12, 13) && this._meshesForIntersections.pushNoDuplicate(i); let s = this.customLODSelector ? this.customLODSelector(i, this.activeCamera) : i.getLOD(this.activeCamera); if (i._internalAbstractMeshDataInfo._currentLOD = s, i._internalAbstractMeshDataInfo._currentLODIsUpToDate = !0, s != null && (s !== i && s.billboardMode !== 0 && s.computeWorldMatrix(), i._preActivate(), i.isVisible && i.visibility > 0 && i.layerMask & this.activeCamera.layerMask && (this._skipFrustumClipping || i.alwaysSelectAsActiveMesh || i.isInFrustum(this._frustumPlanes)))) { this._activeMeshes.push(i), this.activeCamera._activeMeshes.push(i), s !== i && s._activate(this._renderId, !1); for (const a of this._preActiveMeshStage) a.action(i); i._activate(this._renderId, !1) && (i.isAnInstance ? i._internalAbstractMeshDataInfo._actAsRegularMesh && (s = i) : s._internalAbstractMeshDataInfo._onlyForInstances = !1, s._internalAbstractMeshDataInfo._isActive = !0, this._activeMesh(i, s)), i._postActivate(); } } if (this.onAfterActiveMeshesEvaluationObservable.notifyObservers(this), this.particlesEnabled) { this.onBeforeParticlesRenderingObservable.notifyObservers(this); for (let n = 0; n < this.particleSystems.length; n++) { const i = this.particleSystems[n]; if (!i.isStarted() || !i.emitter) continue; const s = i.emitter; (!s.position || s.isEnabled()) && (this._activeParticleSystems.push(i), i.animate(), this._renderingManager.dispatchParticles(i)); } this.onAfterParticlesRenderingObservable.notifyObservers(this); } } _activeMesh(e, t) { this._skeletonsEnabled && t.skeleton !== null && t.skeleton !== void 0 && (this._activeSkeletons.pushNoDuplicate(t.skeleton) && (t.skeleton.prepare(), this._activeBones.addCount(t.skeleton.bones.length, !1)), t.computeBonesUsingShaders || this._softwareSkinnedMeshes.pushNoDuplicate(t)); let r = e.hasInstances || e.isAnInstance || this.dispatchAllSubMeshesOfActiveMeshes || this._skipFrustumClipping || t.alwaysSelectAsActiveMesh; if (t && t.subMeshes && t.subMeshes.length > 0) { const n = this.getActiveSubMeshCandidates(t), i = n.length; r = r || i === 1; for (let s = 0; s < i; s++) { const a = n.data[s]; this._evaluateSubMesh(a, t, e, r); } } } /** * Update the transform matrix to update from the current active camera * @param force defines a boolean used to force the update even if cache is up to date */ updateTransformMatrix(e) { const t = this.activeCamera; if (t) if (t._renderingMultiview) { const r = t._rigCameras[0], n = t._rigCameras[1]; this.setTransformMatrix(r.getViewMatrix(), r.getProjectionMatrix(e), n.getViewMatrix(), n.getProjectionMatrix(e)); } else this.setTransformMatrix(t.getViewMatrix(), t.getProjectionMatrix(e)); } _bindFrameBuffer(e, t = !0) { e && e._multiviewTexture ? e._multiviewTexture._bindFrameBuffer() : e && e.outputRenderTarget ? e.outputRenderTarget._bindFrameBuffer() : this._engine._currentFrameBufferIsDefaultFrameBuffer() || this._engine.restoreDefaultFramebuffer(), t && this._clearFrameBuffer(e); } _clearFrameBuffer(e) { if (!(e && e._multiviewTexture)) if (e && e.outputRenderTarget && !e._renderingMultiview) { const t = e.outputRenderTarget; t.onClearObservable.hasObservers() ? t.onClearObservable.notifyObservers(this._engine) : t.skipInitialClear || (this.autoClear && this._engine.clear(t.clearColor || this.clearColor, !t._cleared, !0, !0), t._cleared = !0); } else this._defaultFrameBufferCleared ? this._engine.clear(null, !1, !0, !0) : (this._defaultFrameBufferCleared = !0, this._clear()); } /** * @internal */ _renderForCamera(e, t, r = !0) { var n, i, s; if (e && e._skipRendering) return; const a = this._engine; if (this._activeCamera = e, !this.activeCamera) throw new Error("Active camera not set"); if (a.setViewport(this.activeCamera.viewport), this.resetCachedMaterial(), this._renderId++, !this.prePass && r) { let o = !0; e._renderingMultiview && e.outputRenderTarget && (o = e.outputRenderTarget.skipInitialClear, this.autoClear && (this._defaultFrameBufferCleared = !1, e.outputRenderTarget.skipInitialClear = !1)), this._bindFrameBuffer(this._activeCamera), e._renderingMultiview && e.outputRenderTarget && (e.outputRenderTarget.skipInitialClear = o); } this.updateTransformMatrix(), this.onBeforeCameraRenderObservable.notifyObservers(this.activeCamera), this._evaluateActiveMeshes(); for (let o = 0; o < this._softwareSkinnedMeshes.length; o++) { const d = this._softwareSkinnedMeshes.data[o]; d.applySkeleton(d.skeleton); } this.onBeforeRenderTargetsRenderObservable.notifyObservers(this), this._renderTargets.concatWithNoDuplicate(this._materialsRenderTargets), e.customRenderTargets && e.customRenderTargets.length > 0 && this._renderTargets.concatWithNoDuplicate(e.customRenderTargets), t && t.customRenderTargets && t.customRenderTargets.length > 0 && this._renderTargets.concatWithNoDuplicate(t.customRenderTargets), this.environmentTexture && this.environmentTexture.isRenderTarget && this._renderTargets.pushNoDuplicate(this.environmentTexture); for (const o of this._gatherActiveCameraRenderTargetsStage) o.action(this._renderTargets); let f = !1; if (this.renderTargetsEnabled) { if (this._intermediateRendering = !0, this._renderTargets.length > 0) { ye.StartPerformanceCounter("Render targets", this._renderTargets.length > 0); for (let o = 0; o < this._renderTargets.length; o++) { const d = this._renderTargets.data[o]; if (d._shouldRender()) { this._renderId++; const v = d.activeCamera && d.activeCamera !== this.activeCamera; d.render(v, this.dumpNextRenderTargets), f = !0; } } ye.EndPerformanceCounter("Render targets", this._renderTargets.length > 0), this._renderId++; } for (const o of this._cameraDrawRenderTargetStage) f = o.action(this.activeCamera) || f; this._intermediateRendering = !1; } this._engine.currentRenderPassId = (s = (i = (n = e.outputRenderTarget) === null || n === void 0 ? void 0 : n.renderPassId) !== null && i !== void 0 ? i : e.renderPassId) !== null && s !== void 0 ? s : 0, f && !this.prePass && this._bindFrameBuffer(this._activeCamera, !1), this.onAfterRenderTargetsRenderObservable.notifyObservers(this), this.postProcessManager && !e._multiviewTexture && !this.prePass && this.postProcessManager._prepareFrame(); for (const o of this._beforeCameraDrawStage) o.action(this.activeCamera); this.onBeforeDrawPhaseObservable.notifyObservers(this), a.snapshotRendering && a.snapshotRenderingMode === 1 && this.finalizeSceneUbo(), this._renderingManager.render(null, null, !0, !0), this.onAfterDrawPhaseObservable.notifyObservers(this); for (const o of this._afterCameraDrawStage) o.action(this.activeCamera); if (this.postProcessManager && !e._multiviewTexture) { const o = e.outputRenderTarget ? e.outputRenderTarget.renderTarget : void 0; this.postProcessManager._finalizeFrame(e.isIntermediate, o); } for (const o of this._afterCameraPostProcessStage) o.action(this.activeCamera); this._renderTargets.reset(), this.onAfterCameraRenderObservable.notifyObservers(this.activeCamera); } _processSubCameras(e, t = !0) { if (e.cameraRigMode === 0 || e._renderingMultiview) { e._renderingMultiview && !this._multiviewSceneUbo && this._createMultiviewUbo(), this._renderForCamera(e, void 0, t), this.onAfterRenderCameraObservable.notifyObservers(e); return; } if (e._useMultiviewToSingleView) this._renderMultiviewToSingleView(e); else { this.onBeforeCameraRenderObservable.notifyObservers(e); for (let r = 0; r < e._rigCameras.length; r++) this._renderForCamera(e._rigCameras[r], e); } this._activeCamera = e, this.updateTransformMatrix(), this.onAfterRenderCameraObservable.notifyObservers(e); } _checkIntersections() { for (let e = 0; e < this._meshesForIntersections.length; e++) { const t = this._meshesForIntersections.data[e]; if (t.actionManager) for (let r = 0; t.actionManager && r < t.actionManager.actions.length; r++) { const n = t.actionManager.actions[r]; if (n.trigger === 12 || n.trigger === 13) { const i = n.getTriggerParameter(), s = i.mesh ? i.mesh : i, a = s.intersectsMesh(t, i.usePreciseIntersection), f = t._intersectionsInProgress.indexOf(s); a && f === -1 ? n.trigger === 12 ? (n._executeCurrent(mo.CreateNew(t, void 0, s)), t._intersectionsInProgress.push(s)) : n.trigger === 13 && t._intersectionsInProgress.push(s) : !a && f > -1 && (n.trigger === 13 && n._executeCurrent(mo.CreateNew(t, void 0, s)), (!t.actionManager.hasSpecificTrigger(13, (o) => { const d = o.mesh ? o.mesh : o; return s === d; }) || n.trigger === 13) && t._intersectionsInProgress.splice(f, 1)); } } } } /** * @internal */ _advancePhysicsEngineStep(e) { } /** @internal */ _animate() { } /** Execute all animations (for a frame) */ animate() { if (this._engine.isDeterministicLockStep()) { let e = Math.max(sr.MinDeltaTime, Math.min(this._engine.getDeltaTime(), sr.MaxDeltaTime)) + this._timeAccumulator; const t = this._engine.getTimeStep(), r = 1e3 / t / 1e3; let n = 0; const i = this._engine.getLockstepMaxSteps(); let s = Math.floor(e / t); for (s = Math.min(s, i); e > 0 && n < s; ) this.onBeforeStepObservable.notifyObservers(this), this._animationRatio = t * r, this._animate(), this.onAfterAnimationsObservable.notifyObservers(this), this.physicsEnabled && this._advancePhysicsEngineStep(t), this.onAfterStepObservable.notifyObservers(this), this._currentStepId++, n++, e -= t; this._timeAccumulator = e < 0 ? 0 : e; } else { const e = this.useConstantAnimationDeltaTime ? 16 : Math.max(sr.MinDeltaTime, Math.min(this._engine.getDeltaTime(), sr.MaxDeltaTime)); this._animationRatio = e * (60 / 1e3), this._animate(), this.onAfterAnimationsObservable.notifyObservers(this), this.physicsEnabled && this._advancePhysicsEngineStep(e); } } _clear() { (this.autoClearDepthAndStencil || this.autoClear) && this._engine.clear(this.clearColor, this.autoClear || this.forceWireframe || this.forcePointsCloud, this.autoClearDepthAndStencil, this.autoClearDepthAndStencil); } _checkCameraRenderTarget(e) { var t; if (e != null && e.outputRenderTarget && !(e != null && e.isRigCamera) && (e.outputRenderTarget._cleared = !1), !((t = e == null ? void 0 : e.rigCameras) === null || t === void 0) && t.length) for (let r = 0; r < e.rigCameras.length; ++r) { const n = e.rigCameras[r].outputRenderTarget; n && (n._cleared = !1); } } /** * Resets the draw wrappers cache of all meshes * @param passId If provided, releases only the draw wrapper corresponding to this render pass id */ resetDrawCache(e) { if (this.meshes) for (const t of this.meshes) t.resetDrawCache(e); } /** * Render the scene * @param updateCameras defines a boolean indicating if cameras must update according to their inputs (true by default) * @param ignoreAnimations defines a boolean indicating if animations should not be executed (false by default) */ render(e = !0, t = !1) { var r, n, i; if (this.isDisposed) return; this.onReadyObservable.hasObservers() && this._executeWhenReadyTimeoutId === null && this._checkIsReady(), this._frameId++, this._defaultFrameBufferCleared = !1, this._checkCameraRenderTarget(this.activeCamera), !((r = this.activeCameras) === null || r === void 0) && r.length && this.activeCameras.forEach(this._checkCameraRenderTarget), this._registerTransientComponents(), this._activeParticles.fetchNewFrame(), this._totalVertices.fetchNewFrame(), this._activeIndices.fetchNewFrame(), this._activeBones.fetchNewFrame(), this._meshesForIntersections.reset(), this.resetCachedMaterial(), this.onBeforeAnimationsObservable.notifyObservers(this), this.actionManager && this.actionManager.processTrigger(11), t || this.animate(); for (const f of this._beforeCameraUpdateStage) f.action(); if (e) { if (this.activeCameras && this.activeCameras.length > 0) for (let f = 0; f < this.activeCameras.length; f++) { const o = this.activeCameras[f]; if (o.update(), o.cameraRigMode !== 0) for (let d = 0; d < o._rigCameras.length; d++) o._rigCameras[d].update(); } else if (this.activeCamera && (this.activeCamera.update(), this.activeCamera.cameraRigMode !== 0)) for (let f = 0; f < this.activeCamera._rigCameras.length; f++) this.activeCamera._rigCameras[f].update(); } this.onBeforeRenderObservable.notifyObservers(this); const s = this.getEngine(); this.onBeforeRenderTargetsRenderObservable.notifyObservers(this); const a = !((n = this.activeCameras) === null || n === void 0) && n.length ? this.activeCameras[0] : this.activeCamera; if (this.renderTargetsEnabled) { ye.StartPerformanceCounter("Custom render targets", this.customRenderTargets.length > 0), this._intermediateRendering = !0; for (let f = 0; f < this.customRenderTargets.length; f++) { const o = this.customRenderTargets[f]; if (o._shouldRender()) { if (this._renderId++, this.activeCamera = o.activeCamera || this.activeCamera, !this.activeCamera) throw new Error("Active camera not set"); s.setViewport(this.activeCamera.viewport), this.updateTransformMatrix(), o.render(a !== this.activeCamera, this.dumpNextRenderTargets); } } ye.EndPerformanceCounter("Custom render targets", this.customRenderTargets.length > 0), this._intermediateRendering = !1, this._renderId++; } this._engine.currentRenderPassId = (i = a == null ? void 0 : a.renderPassId) !== null && i !== void 0 ? i : 0, this.activeCamera = a, this._activeCamera && this._activeCamera.cameraRigMode !== 22 && !this.prePass && this._bindFrameBuffer(this._activeCamera, !1), this.onAfterRenderTargetsRenderObservable.notifyObservers(this); for (const f of this._beforeClearStage) f.action(); this._clearFrameBuffer(this.activeCamera); for (const f of this._gatherRenderTargetsStage) f.action(this._renderTargets); if (this.activeCameras && this.activeCameras.length > 0) for (let f = 0; f < this.activeCameras.length; f++) this._processSubCameras(this.activeCameras[f], f > 0); else { if (!this.activeCamera) throw new Error("No camera defined"); this._processSubCameras(this.activeCamera, !!this.activeCamera.outputRenderTarget); } this._checkIntersections(); for (const f of this._afterRenderStage) f.action(); if (this.afterRender && this.afterRender(), this.onAfterRenderObservable.notifyObservers(this), this._toBeDisposed.length) { for (let f = 0; f < this._toBeDisposed.length; f++) { const o = this._toBeDisposed[f]; o && o.dispose(); } this._toBeDisposed.length = 0; } this.dumpNextRenderTargets && (this.dumpNextRenderTargets = !1), this._activeBones.addCount(0, !0), this._activeIndices.addCount(0, !0), this._activeParticles.addCount(0, !0), this._engine.restoreDefaultFramebuffer(); } /** * Freeze all materials * A frozen material will not be updatable but should be faster to render * Note: multimaterials will not be frozen, but their submaterials will */ freezeMaterials() { for (let e = 0; e < this.materials.length; e++) this.materials[e].freeze(); } /** * Unfreeze all materials * A frozen material will not be updatable but should be faster to render */ unfreezeMaterials() { for (let e = 0; e < this.materials.length; e++) this.materials[e].unfreeze(); } /** * Releases all held resources */ dispose() { if (this.isDisposed) return; this.beforeRender = null, this.afterRender = null, this.metadata = null, this.skeletons.length = 0, this.morphTargetManagers.length = 0, this._transientComponents.length = 0, this._isReadyForMeshStage.clear(), this._beforeEvaluateActiveMeshStage.clear(), this._evaluateSubMeshStage.clear(), this._preActiveMeshStage.clear(), this._cameraDrawRenderTargetStage.clear(), this._beforeCameraDrawStage.clear(), this._beforeRenderTargetDrawStage.clear(), this._beforeRenderingGroupDrawStage.clear(), this._beforeRenderingMeshStage.clear(), this._afterRenderingMeshStage.clear(), this._afterRenderingGroupDrawStage.clear(), this._afterCameraDrawStage.clear(), this._afterRenderTargetDrawStage.clear(), this._afterRenderStage.clear(), this._beforeCameraUpdateStage.clear(), this._beforeClearStage.clear(), this._gatherRenderTargetsStage.clear(), this._gatherActiveCameraRenderTargetsStage.clear(), this._pointerMoveStage.clear(), this._pointerDownStage.clear(), this._pointerUpStage.clear(), this.importedMeshesFiles = [], this.stopAllAnimations && (this._activeAnimatables.forEach((i) => { i.onAnimationEndObservable.clear(), i.onAnimationEnd = null; }), this.stopAllAnimations()), this.resetCachedMaterial(), this.activeCamera && (this.activeCamera._activeMeshes.dispose(), this.activeCamera = null), this.activeCameras = null, this._activeMeshes.dispose(), this._renderingManager.dispose(), this._processedMaterials.dispose(), this._activeParticleSystems.dispose(), this._activeSkeletons.dispose(), this._softwareSkinnedMeshes.dispose(), this._renderTargets.dispose(), this._materialsRenderTargets.dispose(), this._registeredForLateAnimationBindings.dispose(), this._meshesForIntersections.dispose(), this._toBeDisposed.length = 0; const e = this._activeRequests.slice(); for (const i of e) i.abort(); this._activeRequests.length = 0; try { this.onDisposeObservable.notifyObservers(this); } catch (i) { console.error("An error occurred while calling onDisposeObservable!", i); } if (this.detachControl(), this._engine.getInputElement()) for (let i = 0; i < this.cameras.length; i++) this.cameras[i].detachControl(); this._disposeList(this.animationGroups), this._disposeList(this.lights), this._disposeList(this.meshes, (i) => i.dispose(!0)), this._disposeList(this.transformNodes, (i) => i.dispose(!0)); const r = this.cameras; this._disposeList(r), this._defaultMaterial && this._defaultMaterial.dispose(), this._disposeList(this.multiMaterials), this._disposeList(this.materials), this._disposeList(this.particleSystems), this._disposeList(this.postProcesses), this._disposeList(this.textures), this._disposeList(this.morphTargetManagers), this._sceneUbo.dispose(), this._multiviewSceneUbo && this._multiviewSceneUbo.dispose(), this.postProcessManager.dispose(), this._disposeList(this._components); let n = this._engine.scenes.indexOf(this); n > -1 && this._engine.scenes.splice(n, 1), gr._LastCreatedScene === this && (this._engine.scenes.length > 0 ? gr._LastCreatedScene = this._engine.scenes[this._engine.scenes.length - 1] : gr._LastCreatedScene = null), n = this._engine._virtualScenes.indexOf(this), n > -1 && this._engine._virtualScenes.splice(n, 1), this._engine.wipeCaches(!0), this.onDisposeObservable.clear(), this.onBeforeRenderObservable.clear(), this.onAfterRenderObservable.clear(), this.onBeforeRenderTargetsRenderObservable.clear(), this.onAfterRenderTargetsRenderObservable.clear(), this.onAfterStepObservable.clear(), this.onBeforeStepObservable.clear(), this.onBeforeActiveMeshesEvaluationObservable.clear(), this.onAfterActiveMeshesEvaluationObservable.clear(), this.onBeforeParticlesRenderingObservable.clear(), this.onAfterParticlesRenderingObservable.clear(), this.onBeforeDrawPhaseObservable.clear(), this.onAfterDrawPhaseObservable.clear(), this.onBeforeAnimationsObservable.clear(), this.onAfterAnimationsObservable.clear(), this.onDataLoadedObservable.clear(), this.onBeforeRenderingGroupObservable.clear(), this.onAfterRenderingGroupObservable.clear(), this.onMeshImportedObservable.clear(), this.onBeforeCameraRenderObservable.clear(), this.onAfterCameraRenderObservable.clear(), this.onAfterRenderCameraObservable.clear(), this.onReadyObservable.clear(), this.onNewCameraAddedObservable.clear(), this.onCameraRemovedObservable.clear(), this.onNewLightAddedObservable.clear(), this.onLightRemovedObservable.clear(), this.onNewGeometryAddedObservable.clear(), this.onGeometryRemovedObservable.clear(), this.onNewTransformNodeAddedObservable.clear(), this.onTransformNodeRemovedObservable.clear(), this.onNewMeshAddedObservable.clear(), this.onMeshRemovedObservable.clear(), this.onNewSkeletonAddedObservable.clear(), this.onSkeletonRemovedObservable.clear(), this.onNewMaterialAddedObservable.clear(), this.onNewMultiMaterialAddedObservable.clear(), this.onMaterialRemovedObservable.clear(), this.onMultiMaterialRemovedObservable.clear(), this.onNewTextureAddedObservable.clear(), this.onTextureRemovedObservable.clear(), this.onPrePointerObservable.clear(), this.onPointerObservable.clear(), this.onPreKeyboardObservable.clear(), this.onKeyboardObservable.clear(), this.onActiveCameraChanged.clear(), this.onScenePerformancePriorityChangedObservable.clear(), this._isDisposed = !0; } _disposeList(e, t) { const r = e.slice(0); t = t ?? ((n) => n.dispose()); for (const n of r) t(n); e.length = 0; } /** * Gets if the scene is already disposed */ get isDisposed() { return this._isDisposed; } /** * Call this function to reduce memory footprint of the scene. * Vertex buffers will not store CPU data anymore (this will prevent picking, collisions or physics to work correctly) */ clearCachedVertexData() { for (let e = 0; e < this.meshes.length; e++) { const r = this.meshes[e].geometry; r && r.clearCachedData(); } } /** * This function will remove the local cached buffer data from texture. * It will save memory but will prevent the texture from being rebuilt */ cleanCachedTextureBuffer() { for (const e of this.textures) e._buffer && (e._buffer = null); } /** * Get the world extend vectors with an optional filter * * @param filterPredicate the predicate - which meshes should be included when calculating the world size * @returns {{ min: Vector3; max: Vector3 }} min and max vectors */ getWorldExtends(e) { const t = new S(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE), r = new S(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE); return e = e || (() => !0), this.meshes.filter(e).forEach((n) => { if (n.computeWorldMatrix(!0), !n.subMeshes || n.subMeshes.length === 0 || n.infiniteDistance) return; const i = n.getBoundingInfo(), s = i.boundingBox.minimumWorld, a = i.boundingBox.maximumWorld; S.CheckExtends(s, t, r), S.CheckExtends(a, t, r); }), { min: t, max: r }; } // Picking /** * Creates a ray that can be used to pick in the scene * @param x defines the x coordinate of the origin (on-screen) * @param y defines the y coordinate of the origin (on-screen) * @param world defines the world matrix to use if you want to pick in object space (instead of world space) * @param camera defines the camera to use for the picking * @param cameraViewSpace defines if picking will be done in view space (false by default) * @returns a Ray */ createPickingRay(e, t, r, n, i = !1) { throw qn("Ray"); } /** * Creates a ray that can be used to pick in the scene * @param x defines the x coordinate of the origin (on-screen) * @param y defines the y coordinate of the origin (on-screen) * @param world defines the world matrix to use if you want to pick in object space (instead of world space) * @param result defines the ray where to store the picking ray * @param camera defines the camera to use for the picking * @param cameraViewSpace defines if picking will be done in view space (false by default) * @param enableDistantPicking defines if picking should handle large values for mesh position/scaling (false by default) * @returns the current scene */ createPickingRayToRef(e, t, r, n, i, s = !1, a = !1) { throw qn("Ray"); } /** * Creates a ray that can be used to pick in the scene * @param x defines the x coordinate of the origin (on-screen) * @param y defines the y coordinate of the origin (on-screen) * @param camera defines the camera to use for the picking * @returns a Ray */ createPickingRayInCameraSpace(e, t, r) { throw qn("Ray"); } /** * Creates a ray that can be used to pick in the scene * @param x defines the x coordinate of the origin (on-screen) * @param y defines the y coordinate of the origin (on-screen) * @param result defines the ray where to store the picking ray * @param camera defines the camera to use for the picking * @returns the current scene */ createPickingRayInCameraSpaceToRef(e, t, r, n) { throw qn("Ray"); } /** @internal */ get _pickingAvailable() { return !1; } /** Launch a ray to try to pick a mesh in the scene * @param x position on screen * @param y position on screen * @param predicate Predicate function used to determine eligible meshes. Can be set to null. In this case, a mesh must be enabled, visible and with isPickable set to true * @param fastCheck defines if the first intersection will be used (and not the closest) * @param camera to use for computing the picking ray. Can be set to null. In this case, the scene.activeCamera will be used * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected * @returns a PickingInfo */ pick(e, t, r, n, i, s) { const a = qn("Ray", !0); return a && Se.Warn(a), new F9(); } /** Launch a ray to try to pick a mesh in the scene using only bounding information of the main mesh (not using submeshes) * @param x position on screen * @param y position on screen * @param predicate Predicate function used to determine eligible meshes. Can be set to null. In this case, a mesh must be enabled, visible and with isPickable set to true * @param fastCheck defines if the first intersection will be used (and not the closest) * @param camera to use for computing the picking ray. Can be set to null. In this case, the scene.activeCamera will be used * @returns a PickingInfo (Please note that some info will not be set like distance, bv, bu and everything that cannot be capture by only using bounding infos) */ pickWithBoundingInfo(e, t, r, n, i) { const s = qn("Ray", !0); return s && Se.Warn(s), new F9(); } /** * Use the given ray to pick a mesh in the scene. A mesh triangle can be picked both from its front and back sides, * irrespective of orientation. * @param ray The ray to use to pick meshes * @param predicate Predicate function used to determine eligible meshes. Can be set to null. In this case, a mesh must have isPickable set to true * @param fastCheck defines if the first intersection will be used (and not the closest) * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected * @returns a PickingInfo */ pickWithRay(e, t, r, n) { throw qn("Ray"); } /** * Launch a ray to try to pick a mesh in the scene. A mesh triangle can be picked both from its front and back sides, * irrespective of orientation. * @param x X position on screen * @param y Y position on screen * @param predicate Predicate function used to determine eligible meshes. Can be set to null. In this case, a mesh must be enabled, visible and with isPickable set to true * @param camera camera to use for computing the picking ray. Can be set to null. In this case, the scene.activeCamera will be used * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected * @returns an array of PickingInfo */ multiPick(e, t, r, n, i) { throw qn("Ray"); } /** * Launch a ray to try to pick a mesh in the scene * @param ray Ray to use * @param predicate Predicate function used to determine eligible meshes. Can be set to null. In this case, a mesh must be enabled, visible and with isPickable set to true * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected * @returns an array of PickingInfo */ multiPickWithRay(e, t, r) { throw qn("Ray"); } /** * Force the value of meshUnderPointer * @param mesh defines the mesh to use * @param pointerId optional pointer id when using more than one pointer * @param pickResult optional pickingInfo data used to find mesh */ setPointerOverMesh(e, t, r) { this._inputManager.setPointerOverMesh(e, t, r); } /** * Gets the mesh under the pointer * @returns a Mesh or null if no mesh is under the pointer */ getPointerOverMesh() { return this._inputManager.getPointerOverMesh(); } // Misc. /** @internal */ _rebuildGeometries() { for (const e of this.geometries) e._rebuild(); for (const e of this.meshes) e._rebuild(); this.postProcessManager && this.postProcessManager._rebuild(); for (const e of this._components) e.rebuild(); for (const e of this.particleSystems) e.rebuild(); if (this.spriteManagers) for (const e of this.spriteManagers) e.rebuild(); } /** @internal */ _rebuildTextures() { for (const e of this.textures) e._rebuild(); this.markAllMaterialsAsDirty(1); } /** * Get from a list of objects by tags * @param list the list of objects to use * @param tagsQuery the query to use * @param filter a predicate to filter for tags * @returns */ _getByTags(e, t, r) { if (t === void 0) return e; const n = []; for (const i in e) { const s = e[i]; Zi && Zi.MatchesQuery(s, t) && (!r || r(s)) && n.push(s); } return n; } /** * Get a list of meshes by tags * @param tagsQuery defines the tags query to use * @param filter defines a predicate used to filter results * @returns an array of Mesh */ getMeshesByTags(e, t) { return this._getByTags(this.meshes, e, t); } /** * Get a list of cameras by tags * @param tagsQuery defines the tags query to use * @param filter defines a predicate used to filter results * @returns an array of Camera */ getCamerasByTags(e, t) { return this._getByTags(this.cameras, e, t); } /** * Get a list of lights by tags * @param tagsQuery defines the tags query to use * @param filter defines a predicate used to filter results * @returns an array of Light */ getLightsByTags(e, t) { return this._getByTags(this.lights, e, t); } /** * Get a list of materials by tags * @param tagsQuery defines the tags query to use * @param filter defines a predicate used to filter results * @returns an array of Material */ getMaterialByTags(e, t) { return this._getByTags(this.materials, e, t).concat(this._getByTags(this.multiMaterials, e, t)); } /** * Get a list of transform nodes by tags * @param tagsQuery defines the tags query to use * @param filter defines a predicate used to filter results * @returns an array of TransformNode */ getTransformNodesByTags(e, t) { return this._getByTags(this.transformNodes, e, t); } /** * Overrides the default sort function applied in the rendering group to prepare the meshes. * This allowed control for front to back rendering or reversly depending of the special needs. * * @param renderingGroupId The rendering group id corresponding to its index * @param opaqueSortCompareFn The opaque queue comparison function use to sort. * @param alphaTestSortCompareFn The alpha test queue comparison function use to sort. * @param transparentSortCompareFn The transparent queue comparison function use to sort. */ setRenderingOrder(e, t = null, r = null, n = null) { this._renderingManager.setRenderingOrder(e, t, r, n); } /** * Specifies whether or not the stencil and depth buffer are cleared between two rendering groups. * * @param renderingGroupId The rendering group id corresponding to its index * @param autoClearDepthStencil Automatically clears depth and stencil between groups if true. * @param depth Automatically clears depth between groups if true and autoClear is true. * @param stencil Automatically clears stencil between groups if true and autoClear is true. */ setRenderingAutoClearDepthStencil(e, t, r = !0, n = !0) { this._renderingManager.setRenderingAutoClearDepthStencil(e, t, r, n); } /** * Gets the current auto clear configuration for one rendering group of the rendering * manager. * @param index the rendering group index to get the information for * @returns The auto clear setup for the requested rendering group */ getAutoClearDepthStencilSetup(e) { return this._renderingManager.getAutoClearDepthStencilSetup(e); } /** @internal */ _forceBlockMaterialDirtyMechanism(e) { this._blockMaterialDirtyMechanism = e; } /** Gets or sets a boolean blocking all the calls to markAllMaterialsAsDirty (ie. the materials won't be updated if they are out of sync) */ get blockMaterialDirtyMechanism() { return this._blockMaterialDirtyMechanism; } set blockMaterialDirtyMechanism(e) { this._blockMaterialDirtyMechanism !== e && (this._blockMaterialDirtyMechanism = e, e || this.markAllMaterialsAsDirty(63)); } /** * Will flag all materials as dirty to trigger new shader compilation * @param flag defines the flag used to specify which material part must be marked as dirty * @param predicate If not null, it will be used to specify if a material has to be marked as dirty */ markAllMaterialsAsDirty(e, t) { if (!this._blockMaterialDirtyMechanism) for (const r of this.materials) t && !t(r) || r.markAsDirty(e); } /** * @internal */ _loadFile(e, t, r, n, i, s, a) { const f = gq(e, t, r, n ? this.offlineProvider : void 0, i, s, a); return this._activeRequests.push(f), f.onCompleteObservable.add((o) => { this._activeRequests.splice(this._activeRequests.indexOf(o), 1); }), f; } /** * @internal */ _loadFileAsync(e, t, r, n, i) { return new Promise((s, a) => { this._loadFile(e, (f) => { s(f); }, t, r, n, (f, o) => { a(o); }, i); }); } /** * @internal */ _requestFile(e, t, r, n, i, s, a) { const f = jO(e, t, r, n ? this.offlineProvider : void 0, i, s, a); return this._activeRequests.push(f), f.onCompleteObservable.add((o) => { this._activeRequests.splice(this._activeRequests.indexOf(o), 1); }), f; } /** * @internal */ _requestFileAsync(e, t, r, n, i) { return new Promise((s, a) => { this._requestFile(e, (f) => { s(f); }, t, r, n, (f) => { a(f); }, i); }); } /** * @internal */ _readFile(e, t, r, n, i) { const s = BS(e, t, r, n, i); return this._activeRequests.push(s), s.onCompleteObservable.add((a) => { this._activeRequests.splice(this._activeRequests.indexOf(a), 1); }), s; } /** * @internal */ _readFileAsync(e, t, r) { return new Promise((n, i) => { this._readFile(e, (s) => { n(s); }, t, r, (s) => { i(s); }); }); } /** * This method gets the performance collector belonging to the scene, which is generally shared with the inspector. * @returns the perf collector belonging to the scene. */ getPerfCollector() { throw qn("performanceViewerSceneExtension"); } // deprecated /** * Sets the active camera of the scene using its Id * @param id defines the camera's Id * @returns the new active camera or null if none found. * @deprecated Please use setActiveCameraById instead */ setActiveCameraByID(e) { return this.setActiveCameraById(e); } /** * Get a material using its id * @param id defines the material's Id * @returns the material or null if none found. * @deprecated Please use getMaterialById instead */ getMaterialByID(e) { return this.getMaterialById(e); } /** * Gets a the last added material using a given id * @param id defines the material's Id * @returns the last material with the given id or null if none found. * @deprecated Please use getLastMaterialById instead */ getLastMaterialByID(e) { return this.getLastMaterialById(e); } /** * Get a texture using its unique id * @param uniqueId defines the texture's unique id * @returns the texture or null if none found. * @deprecated Please use getTextureByUniqueId instead */ getTextureByUniqueID(e) { return this.getTextureByUniqueId(e); } /** * Gets a camera using its Id * @param id defines the Id to look for * @returns the camera or null if not found * @deprecated Please use getCameraById instead */ getCameraByID(e) { return this.getCameraById(e); } /** * Gets a camera using its unique Id * @param uniqueId defines the unique Id to look for * @returns the camera or null if not found * @deprecated Please use getCameraByUniqueId instead */ getCameraByUniqueID(e) { return this.getCameraByUniqueId(e); } /** * Gets a bone using its Id * @param id defines the bone's Id * @returns the bone or null if not found * @deprecated Please use getBoneById instead */ getBoneByID(e) { return this.getBoneById(e); } /** * Gets a light node using its Id * @param id defines the light's Id * @returns the light or null if none found. * @deprecated Please use getLightById instead */ getLightByID(e) { return this.getLightById(e); } /** * Gets a light node using its scene-generated unique Id * @param uniqueId defines the light's unique Id * @returns the light or null if none found. * @deprecated Please use getLightByUniqueId instead */ getLightByUniqueID(e) { return this.getLightByUniqueId(e); } /** * Gets a particle system by Id * @param id defines the particle system Id * @returns the corresponding system or null if none found * @deprecated Please use getParticleSystemById instead */ getParticleSystemByID(e) { return this.getParticleSystemById(e); } /** * Gets a geometry using its Id * @param id defines the geometry's Id * @returns the geometry or null if none found. * @deprecated Please use getGeometryById instead */ getGeometryByID(e) { return this.getGeometryById(e); } /** * Gets the first added mesh found of a given Id * @param id defines the Id to search for * @returns the mesh found or null if not found at all * @deprecated Please use getMeshById instead */ getMeshByID(e) { return this.getMeshById(e); } /** * Gets a mesh with its auto-generated unique Id * @param uniqueId defines the unique Id to search for * @returns the found mesh or null if not found at all. * @deprecated Please use getMeshByUniqueId instead */ getMeshByUniqueID(e) { return this.getMeshByUniqueId(e); } /** * Gets a the last added mesh using a given Id * @param id defines the Id to search for * @returns the found mesh or null if not found at all. * @deprecated Please use getLastMeshById instead */ getLastMeshByID(e) { return this.getLastMeshById(e); } /** * Gets a list of meshes using their Id * @param id defines the Id to search for * @returns a list of meshes * @deprecated Please use getMeshesById instead */ getMeshesByID(e) { return this.getMeshesById(e); } /** * Gets the first added transform node found of a given Id * @param id defines the Id to search for * @returns the found transform node or null if not found at all. * @deprecated Please use getTransformNodeById instead */ getTransformNodeByID(e) { return this.getTransformNodeById(e); } /** * Gets a transform node with its auto-generated unique Id * @param uniqueId defines the unique Id to search for * @returns the found transform node or null if not found at all. * @deprecated Please use getTransformNodeByUniqueId instead */ getTransformNodeByUniqueID(e) { return this.getTransformNodeByUniqueId(e); } /** * Gets a list of transform nodes using their Id * @param id defines the Id to search for * @returns a list of transform nodes * @deprecated Please use getTransformNodesById instead */ getTransformNodesByID(e) { return this.getTransformNodesById(e); } /** * Gets a node (Mesh, Camera, Light) using a given Id * @param id defines the Id to search for * @returns the found node or null if not found at all * @deprecated Please use getNodeById instead */ getNodeByID(e) { return this.getNodeById(e); } /** * Gets a the last added node (Mesh, Camera, Light) using a given Id * @param id defines the Id to search for * @returns the found node or null if not found at all * @deprecated Please use getLastEntryById instead */ getLastEntryByID(e) { return this.getLastEntryById(e); } /** * Gets a skeleton using a given Id (if many are found, this function will pick the last one) * @param id defines the Id to search for * @returns the found skeleton or null if not found at all. * @deprecated Please use getLastSkeletonById instead */ getLastSkeletonByID(e) { return this.getLastSkeletonById(e); } } sr.FOGMODE_NONE = 0; sr.FOGMODE_EXP = 1; sr.FOGMODE_EXP2 = 2; sr.FOGMODE_LINEAR = 3; sr.MinDeltaTime = 1; sr.MaxDeltaTime = 1e3; var ai; (function(A) { A[A.LOCAL = 0] = "LOCAL", A[A.WORLD = 1] = "WORLD", A[A.BONE = 2] = "BONE"; })(ai || (ai = {})); class bf { } bf.X = new S(1, 0, 0); bf.Y = new S(0, 1, 0); bf.Z = new S(0, 0, 1); var Lx; (function(A) { A[A.X = 0] = "X", A[A.Y = 1] = "Y", A[A.Z = 2] = "Z"; })(Lx || (Lx = {})); class da extends Cs { /** @internal */ get _matrix() { return this._compose(), this._localMatrix; } /** @internal */ set _matrix(e) { e.updateFlag === this._localMatrix.updateFlag && !this._needToCompose || (this._needToCompose = !1, this._localMatrix.copyFrom(e), this._markAsDirtyAndDecompose()); } /** * Create a new bone * @param name defines the bone name * @param skeleton defines the parent skeleton * @param parentBone defines the parent (can be null if the bone is the root) * @param localMatrix defines the local matrix (default: identity) * @param restMatrix defines the rest matrix (default: localMatrix) * @param bindMatrix defines the bind matrix (default: localMatrix) * @param index defines index of the bone in the hierarchy (default: null) */ constructor(e, t, r = null, n = null, i = null, s = null, a = null) { var f; super(e, t.getScene()), this.name = e, this.children = [], this.animations = [], this._index = null, this._scalingDeterminant = 1, this._needToDecompose = !0, this._needToCompose = !1, this._linkedTransformNode = null, this._waitingTransformNodeId = null, this._skeleton = t, this._localMatrix = (f = n == null ? void 0 : n.clone()) !== null && f !== void 0 ? f : he.Identity(), this._restMatrix = i ?? this._localMatrix.clone(), this._bindMatrix = s ?? this._localMatrix.clone(), this._index = a, this._absoluteMatrix = new he(), this._absoluteBindMatrix = new he(), this._absoluteInverseBindMatrix = new he(), this._finalMatrix = new he(), t.bones.push(this), this.setParent(r, !1), this._updateAbsoluteBindMatrices(); } /** * Gets the current object class name. * @returns the class name */ getClassName() { return "Bone"; } // Members /** * Gets the parent skeleton * @returns a skeleton */ getSkeleton() { return this._skeleton; } get parent() { return this._parentNode; } /** * Gets parent bone * @returns a bone or null if the bone is the root of the bone hierarchy */ getParent() { return this.parent; } /** * Returns an array containing the children of the bone * @returns an array containing the children of the bone (can be empty if the bone has no children) */ getChildren() { return this.children; } /** * Gets the node index in matrix array generated for rendering * @returns the node index */ getIndex() { return this._index === null ? this.getSkeleton().bones.indexOf(this) : this._index; } set parent(e) { this.setParent(e); } /** * Sets the parent bone * @param parent defines the parent (can be null if the bone is the root) * @param updateAbsoluteBindMatrices defines if the absolute bind and absolute inverse bind matrices must be updated */ setParent(e, t = !0) { if (this.parent !== e) { if (this.parent) { const r = this.parent.children.indexOf(this); r !== -1 && this.parent.children.splice(r, 1); } this._parentNode = e, this.parent && this.parent.children.push(this), t && this._updateAbsoluteBindMatrices(), this.markAsDirty(); } } /** * Gets the local matrix * @returns the local matrix */ getLocalMatrix() { return this._compose(), this._localMatrix; } /** * Gets the bind matrix * @returns the bind matrix */ getBindMatrix() { return this._bindMatrix; } /** * Gets the bind matrix. * @returns the bind matrix * @deprecated Please use getBindMatrix instead */ getBaseMatrix() { return this.getBindMatrix(); } /** * Gets the rest matrix * @returns the rest matrix */ getRestMatrix() { return this._restMatrix; } /** * Gets the rest matrix * @returns the rest matrix * @deprecated Please use getRestMatrix instead */ getRestPose() { return this.getRestMatrix(); } /** * Sets the rest matrix * @param matrix the local-space rest matrix to set for this bone */ setRestMatrix(e) { this._restMatrix.copyFrom(e); } /** * Sets the rest matrix * @param matrix the local-space rest to set for this bone * @deprecated Please use setRestMatrix instead */ setRestPose(e) { this.setRestMatrix(e); } /** * Gets the bind matrix * @returns the bind matrix * @deprecated Please use getBindMatrix instead */ getBindPose() { return this.getBindMatrix(); } /** * Sets the bind matrix * This will trigger a recomputation of the absolute bind and absolute inverse bind matrices for this bone and its children * Note that the local matrix will also be set with the matrix passed in parameter! * @param matrix the local-space bind matrix to set for this bone */ setBindMatrix(e) { this.updateMatrix(e); } /** * Sets the bind matrix * @param matrix the local-space bind to set for this bone * @deprecated Please use setBindMatrix instead */ setBindPose(e) { this.setBindMatrix(e); } /** * Gets the matrix used to store the final world transformation of the bone (ie. the matrix sent to shaders) */ getFinalMatrix() { return this._finalMatrix; } /** * Gets the matrix used to store the final world transformation of the bone (ie. the matrix sent to shaders) * @deprecated Please use getFinalMatrix instead */ getWorldMatrix() { return this.getFinalMatrix(); } /** * Sets the local matrix to the rest matrix */ returnToRest() { var e; if (this._linkedTransformNode) { const t = ue.Vector3[0], r = ue.Quaternion[0], n = ue.Vector3[1]; this.getRestMatrix().decompose(t, r, n), this._linkedTransformNode.position.copyFrom(n), this._linkedTransformNode.rotationQuaternion = (e = this._linkedTransformNode.rotationQuaternion) !== null && e !== void 0 ? e : Ze.Identity(), this._linkedTransformNode.rotationQuaternion.copyFrom(r), this._linkedTransformNode.scaling.copyFrom(t); } else this._matrix = this._restMatrix; } /** * Gets the inverse of the bind matrix, in world space (relative to the skeleton root) * @returns the inverse bind matrix, in world space */ getAbsoluteInverseBindMatrix() { return this._absoluteInverseBindMatrix; } /** * Gets the inverse of the bind matrix, in world space (relative to the skeleton root) * @returns the inverse bind matrix, in world space * @deprecated Please use getAbsoluteInverseBindMatrix instead */ getInvertedAbsoluteTransform() { return this.getAbsoluteInverseBindMatrix(); } /** * Gets the bone matrix, in world space (relative to the skeleton root) * @returns the bone matrix, in world space */ getAbsoluteMatrix() { return this._absoluteMatrix; } /** * Gets the bone matrix, in world space (relative to the skeleton root) * @returns the bone matrix, in world space * @deprecated Please use getAbsoluteMatrix instead */ getAbsoluteTransform() { return this._absoluteMatrix; } /** * Links with the given transform node. * The local matrix of this bone is overwritten by the transform of the node every frame. * @param transformNode defines the transform node to link to */ linkTransformNode(e) { this._linkedTransformNode && this._skeleton._numBonesWithLinkedTransformNode--, this._linkedTransformNode = e, this._linkedTransformNode && this._skeleton._numBonesWithLinkedTransformNode++; } // Properties (matches TransformNode properties) /** * Gets the node used to drive the bone's transformation * @returns a transform node or null */ getTransformNode() { return this._linkedTransformNode; } /** Gets or sets current position (in local space) */ get position() { return this._decompose(), this._localPosition; } set position(e) { this._decompose(), this._localPosition.copyFrom(e), this._markAsDirtyAndCompose(); } /** Gets or sets current rotation (in local space) */ get rotation() { return this.getRotation(); } set rotation(e) { this.setRotation(e); } /** Gets or sets current rotation quaternion (in local space) */ get rotationQuaternion() { return this._decompose(), this._localRotation; } set rotationQuaternion(e) { this.setRotationQuaternion(e); } /** Gets or sets current scaling (in local space) */ get scaling() { return this.getScale(); } set scaling(e) { this.setScale(e); } /** * Gets the animation properties override */ get animationPropertiesOverride() { return this._skeleton.animationPropertiesOverride; } // Methods _decompose() { this._needToDecompose && (this._needToDecompose = !1, this._localScaling || (this._localScaling = S.Zero(), this._localRotation = Ze.Zero(), this._localPosition = S.Zero()), this._localMatrix.decompose(this._localScaling, this._localRotation, this._localPosition)); } _compose() { if (this._needToCompose) { if (!this._localScaling) { this._needToCompose = !1; return; } this._needToCompose = !1, he.ComposeToRef(this._localScaling, this._localRotation, this._localPosition, this._localMatrix); } } /** * Update the bind (and optionally the local) matrix * @param bindMatrix defines the new matrix to set to the bind/local matrix, in local space * @param updateAbsoluteBindMatrices defines if the absolute bind and absolute inverse bind matrices must be recomputed (default: true) * @param updateLocalMatrix defines if the local matrix should also be updated with the matrix passed in parameter (default: true) */ updateMatrix(e, t = !0, r = !0) { this._bindMatrix.copyFrom(e), t && this._updateAbsoluteBindMatrices(), r ? this._matrix = e : this.markAsDirty(); } /** * @internal */ _updateAbsoluteBindMatrices(e, t = !0) { if (e || (e = this._bindMatrix), this.parent ? e.multiplyToRef(this.parent._absoluteBindMatrix, this._absoluteBindMatrix) : this._absoluteBindMatrix.copyFrom(e), this._absoluteBindMatrix.invertToRef(this._absoluteInverseBindMatrix), t) for (let r = 0; r < this.children.length; r++) this.children[r]._updateAbsoluteBindMatrices(); this._scalingDeterminant = this._absoluteBindMatrix.determinant() < 0 ? -1 : 1; } /** * Flag the bone as dirty (Forcing it to update everything) * @returns this bone */ markAsDirty() { return this._currentRenderId++, this._childUpdateId++, this._skeleton._markAsDirty(), this; } /** @internal */ _markAsDirtyAndCompose() { this.markAsDirty(), this._needToCompose = !0; } _markAsDirtyAndDecompose() { this.markAsDirty(), this._needToDecompose = !0; } _updatePosition(e, t = ai.LOCAL, r, n = !0) { const i = this.getLocalMatrix(); if (t == ai.LOCAL) n ? (i.addAtIndex(12, e.x), i.addAtIndex(13, e.y), i.addAtIndex(14, e.z)) : i.setTranslationFromFloats(e.x, e.y, e.z); else { let s = null; r && (s = r.getWorldMatrix()), this._skeleton.computeAbsoluteMatrices(); const a = da._TmpMats[0], f = da._TmpVecs[0]; this.parent ? r && s ? (a.copyFrom(this.parent.getAbsoluteMatrix()), a.multiplyToRef(s, a)) : a.copyFrom(this.parent.getAbsoluteMatrix()) : he.IdentityToRef(a), n && a.setTranslationFromFloats(0, 0, 0), a.invert(), S.TransformCoordinatesToRef(e, a, f), n ? (i.addAtIndex(12, f.x), i.addAtIndex(13, f.y), i.addAtIndex(14, f.z)) : i.setTranslationFromFloats(f.x, f.y, f.z); } this._markAsDirtyAndDecompose(); } /** * Translate the bone in local or world space * @param vec The amount to translate the bone * @param space The space that the translation is in (default: Space.LOCAL) * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD */ translate(e, t = ai.LOCAL, r) { this._updatePosition(e, t, r, !0); } /** * Set the position of the bone in local or world space * @param position The position to set the bone * @param space The space that the position is in (default: Space.LOCAL) * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD */ setPosition(e, t = ai.LOCAL, r) { this._updatePosition(e, t, r, !1); } /** * Set the absolute position of the bone (world space) * @param position The position to set the bone * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD */ setAbsolutePosition(e, t) { this.setPosition(e, ai.WORLD, t); } /** * Scale the bone on the x, y and z axes (in local space) * @param x The amount to scale the bone on the x axis * @param y The amount to scale the bone on the y axis * @param z The amount to scale the bone on the z axis * @param scaleChildren sets this to true if children of the bone should be scaled as well (false by default) */ scale(e, t, r, n = !1) { const i = this.getLocalMatrix(), s = da._TmpMats[0]; he.ScalingToRef(e, t, r, s), s.multiplyToRef(i, i), s.invert(); for (const a of this.children) { const f = a.getLocalMatrix(); f.multiplyToRef(s, f), f.multiplyAtIndex(12, e), f.multiplyAtIndex(13, t), f.multiplyAtIndex(14, r), a._markAsDirtyAndDecompose(); } if (this._markAsDirtyAndDecompose(), n) for (const a of this.children) a.scale(e, t, r, n); } /** * Set the bone scaling in local space * @param scale defines the scaling vector */ setScale(e) { this._decompose(), this._localScaling.copyFrom(e), this._markAsDirtyAndCompose(); } /** * Gets the current scaling in local space * @returns the current scaling vector */ getScale() { return this._decompose(), this._localScaling; } /** * Gets the current scaling in local space and stores it in a target vector * @param result defines the target vector */ getScaleToRef(e) { this._decompose(), e.copyFrom(this._localScaling); } /** * Set the yaw, pitch, and roll of the bone in local or world space * @param yaw The rotation of the bone on the y axis * @param pitch The rotation of the bone on the x axis * @param roll The rotation of the bone on the z axis * @param space The space that the axes of rotation are in * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD */ setYawPitchRoll(e, t, r, n = ai.LOCAL, i) { if (n === ai.LOCAL) { const f = da._TmpQuat; Ze.RotationYawPitchRollToRef(e, t, r, f), this.setRotationQuaternion(f, n, i); return; } const s = da._TmpMats[0]; if (!this._getAbsoluteInverseMatrixUnscaledToRef(s, i)) return; const a = da._TmpMats[1]; he.RotationYawPitchRollToRef(e, t, r, a), s.multiplyToRef(a, a), this._rotateWithMatrix(a, n, i); } /** * Add a rotation to the bone on an axis in local or world space * @param axis The axis to rotate the bone on * @param amount The amount to rotate the bone * @param space The space that the axis is in * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD */ rotate(e, t, r = ai.LOCAL, n) { const i = da._TmpMats[0]; i.setTranslationFromFloats(0, 0, 0), he.RotationAxisToRef(e, t, i), this._rotateWithMatrix(i, r, n); } /** * Set the rotation of the bone to a particular axis angle in local or world space * @param axis The axis to rotate the bone on * @param angle The angle that the bone should be rotated to * @param space The space that the axis is in * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD */ setAxisAngle(e, t, r = ai.LOCAL, n) { if (r === ai.LOCAL) { const a = da._TmpQuat; Ze.RotationAxisToRef(e, t, a), this.setRotationQuaternion(a, r, n); return; } const i = da._TmpMats[0]; if (!this._getAbsoluteInverseMatrixUnscaledToRef(i, n)) return; const s = da._TmpMats[1]; he.RotationAxisToRef(e, t, s), i.multiplyToRef(s, s), this._rotateWithMatrix(s, r, n); } /** * Set the euler rotation of the bone in local or world space * @param rotation The euler rotation that the bone should be set to * @param space The space that the rotation is in * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD */ setRotation(e, t = ai.LOCAL, r) { this.setYawPitchRoll(e.y, e.x, e.z, t, r); } /** * Set the quaternion rotation of the bone in local or world space * @param quat The quaternion rotation that the bone should be set to * @param space The space that the rotation is in * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD */ setRotationQuaternion(e, t = ai.LOCAL, r) { if (t === ai.LOCAL) { this._decompose(), this._localRotation.copyFrom(e), this._markAsDirtyAndCompose(); return; } const n = da._TmpMats[0]; if (!this._getAbsoluteInverseMatrixUnscaledToRef(n, r)) return; const i = da._TmpMats[1]; he.FromQuaternionToRef(e, i), n.multiplyToRef(i, i), this._rotateWithMatrix(i, t, r); } /** * Set the rotation matrix of the bone in local or world space * @param rotMat The rotation matrix that the bone should be set to * @param space The space that the rotation is in * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD */ setRotationMatrix(e, t = ai.LOCAL, r) { if (t === ai.LOCAL) { const s = da._TmpQuat; Ze.FromRotationMatrixToRef(e, s), this.setRotationQuaternion(s, t, r); return; } const n = da._TmpMats[0]; if (!this._getAbsoluteInverseMatrixUnscaledToRef(n, r)) return; const i = da._TmpMats[1]; i.copyFrom(e), n.multiplyToRef(e, i), this._rotateWithMatrix(i, t, r); } _rotateWithMatrix(e, t = ai.LOCAL, r) { const n = this.getLocalMatrix(), i = n.m[12], s = n.m[13], a = n.m[14], f = this.getParent(), o = da._TmpMats[3], d = da._TmpMats[4]; f && t == ai.WORLD ? (r ? (o.copyFrom(r.getWorldMatrix()), f.getAbsoluteMatrix().multiplyToRef(o, o)) : o.copyFrom(f.getAbsoluteMatrix()), d.copyFrom(o), d.invert(), n.multiplyToRef(o, n), n.multiplyToRef(e, n), n.multiplyToRef(d, n)) : t == ai.WORLD && r ? (o.copyFrom(r.getWorldMatrix()), d.copyFrom(o), d.invert(), n.multiplyToRef(o, n), n.multiplyToRef(e, n), n.multiplyToRef(d, n)) : n.multiplyToRef(e, n), n.setTranslationFromFloats(i, s, a), this.computeAbsoluteMatrices(), this._markAsDirtyAndDecompose(); } _getAbsoluteInverseMatrixUnscaledToRef(e, t) { const r = da._TmpMats[2]; return e.copyFrom(this.getAbsoluteMatrix()), t ? (e.multiplyToRef(t.getWorldMatrix(), e), he.ScalingToRef(t.scaling.x, t.scaling.y, t.scaling.z, r)) : he.IdentityToRef(r), e.invert(), isNaN(e.m[0]) ? !1 : (r.multiplyAtIndex(0, this._scalingDeterminant), e.multiplyToRef(r, e), !0); } /** * Get the position of the bone in local or world space * @param space The space that the returned position is in * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @returns The position of the bone */ getPosition(e = ai.LOCAL, t = null) { const r = S.Zero(); return this.getPositionToRef(e, t, r), r; } /** * Copy the position of the bone to a vector3 in local or world space * @param space The space that the returned position is in * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @param result The vector3 to copy the position to */ getPositionToRef(e = ai.LOCAL, t, r) { if (e == ai.LOCAL) { const n = this.getLocalMatrix(); r.x = n.m[12], r.y = n.m[13], r.z = n.m[14]; } else { let n = null; t && (n = t.getWorldMatrix()), this._skeleton.computeAbsoluteMatrices(); let i = da._TmpMats[0]; t && n ? (i.copyFrom(this.getAbsoluteMatrix()), i.multiplyToRef(n, i)) : i = this.getAbsoluteMatrix(), r.x = i.m[12], r.y = i.m[13], r.z = i.m[14]; } } /** * Get the absolute position of the bone (world space) * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @returns The absolute position of the bone */ getAbsolutePosition(e = null) { const t = S.Zero(); return this.getPositionToRef(ai.WORLD, e, t), t; } /** * Copy the absolute position of the bone (world space) to the result param * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @param result The vector3 to copy the absolute position to */ getAbsolutePositionToRef(e, t) { this.getPositionToRef(ai.WORLD, e, t); } /** * Compute the absolute matrices of this bone and its children */ computeAbsoluteMatrices() { if (this._compose(), this.parent) this._localMatrix.multiplyToRef(this.parent._absoluteMatrix, this._absoluteMatrix); else { this._absoluteMatrix.copyFrom(this._localMatrix); const r = this._skeleton.getPoseMatrix(); r && this._absoluteMatrix.multiplyToRef(r, this._absoluteMatrix); } const e = this.children, t = e.length; for (let r = 0; r < t; r++) e[r].computeAbsoluteMatrices(); } /** * Compute the absolute matrices of this bone and its children * @deprecated Please use computeAbsoluteMatrices instead */ computeAbsoluteTransforms() { this.computeAbsoluteMatrices(); } /** * Get the world direction from an axis that is in the local space of the bone * @param localAxis The local direction that is used to compute the world direction * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @returns The world direction */ getDirection(e, t = null) { const r = S.Zero(); return this.getDirectionToRef(e, t, r), r; } /** * Copy the world direction to a vector3 from an axis that is in the local space of the bone * @param localAxis The local direction that is used to compute the world direction * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @param result The vector3 that the world direction will be copied to */ getDirectionToRef(e, t = null, r) { let n = null; t && (n = t.getWorldMatrix()), this._skeleton.computeAbsoluteMatrices(); const i = da._TmpMats[0]; i.copyFrom(this.getAbsoluteMatrix()), t && n && i.multiplyToRef(n, i), S.TransformNormalToRef(e, i, r), r.normalize(); } /** * Get the euler rotation of the bone in local or world space * @param space The space that the rotation should be in * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @returns The euler rotation */ getRotation(e = ai.LOCAL, t = null) { const r = S.Zero(); return this.getRotationToRef(e, t, r), r; } /** * Copy the euler rotation of the bone to a vector3. The rotation can be in either local or world space * @param space The space that the rotation should be in * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @param result The vector3 that the rotation should be copied to */ getRotationToRef(e = ai.LOCAL, t = null, r) { const n = da._TmpQuat; this.getRotationQuaternionToRef(e, t, n), n.toEulerAnglesToRef(r); } /** * Get the quaternion rotation of the bone in either local or world space * @param space The space that the rotation should be in * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @returns The quaternion rotation */ getRotationQuaternion(e = ai.LOCAL, t = null) { const r = Ze.Identity(); return this.getRotationQuaternionToRef(e, t, r), r; } /** * Copy the quaternion rotation of the bone to a quaternion. The rotation can be in either local or world space * @param space The space that the rotation should be in * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @param result The quaternion that the rotation should be copied to */ getRotationQuaternionToRef(e = ai.LOCAL, t = null, r) { if (e == ai.LOCAL) this._decompose(), r.copyFrom(this._localRotation); else { const n = da._TmpMats[0], i = this.getAbsoluteMatrix(); t ? i.multiplyToRef(t.getWorldMatrix(), n) : n.copyFrom(i), n.multiplyAtIndex(0, this._scalingDeterminant), n.multiplyAtIndex(1, this._scalingDeterminant), n.multiplyAtIndex(2, this._scalingDeterminant), n.decompose(void 0, r, void 0); } } /** * Get the rotation matrix of the bone in local or world space * @param space The space that the rotation should be in * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @returns The rotation matrix */ getRotationMatrix(e = ai.LOCAL, t) { const r = he.Identity(); return this.getRotationMatrixToRef(e, t, r), r; } /** * Copy the rotation matrix of the bone to a matrix. The rotation can be in either local or world space * @param space The space that the rotation should be in * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @param result The quaternion that the rotation should be copied to */ getRotationMatrixToRef(e = ai.LOCAL, t, r) { if (e == ai.LOCAL) this.getLocalMatrix().getRotationMatrixToRef(r); else { const n = da._TmpMats[0], i = this.getAbsoluteMatrix(); t ? i.multiplyToRef(t.getWorldMatrix(), n) : n.copyFrom(i), n.multiplyAtIndex(0, this._scalingDeterminant), n.multiplyAtIndex(1, this._scalingDeterminant), n.multiplyAtIndex(2, this._scalingDeterminant), n.getRotationMatrixToRef(r); } } /** * Get the world position of a point that is in the local space of the bone * @param position The local position * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @returns The world position */ getAbsolutePositionFromLocal(e, t = null) { const r = S.Zero(); return this.getAbsolutePositionFromLocalToRef(e, t, r), r; } /** * Get the world position of a point that is in the local space of the bone and copy it to the result param * @param position The local position * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @param result The vector3 that the world position should be copied to */ getAbsolutePositionFromLocalToRef(e, t = null, r) { let n = null; t && (n = t.getWorldMatrix()), this._skeleton.computeAbsoluteMatrices(); const i = da._TmpMats[0]; i.copyFrom(this.getAbsoluteMatrix()), t && n && i.multiplyToRef(n, i), S.TransformCoordinatesToRef(e, i, r); } /** * Get the local position of a point that is in world space * @param position The world position * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @returns The local position */ getLocalPositionFromAbsolute(e, t = null) { const r = S.Zero(); return this.getLocalPositionFromAbsoluteToRef(e, t, r), r; } /** * Get the local position of a point that is in world space and copy it to the result param * @param position The world position * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD * @param result The vector3 that the local position should be copied to */ getLocalPositionFromAbsoluteToRef(e, t = null, r) { let n = null; t && (n = t.getWorldMatrix()), this._skeleton.computeAbsoluteMatrices(); const i = da._TmpMats[0]; i.copyFrom(this.getAbsoluteMatrix()), t && n && i.multiplyToRef(n, i), i.invert(), S.TransformCoordinatesToRef(e, i, r); } /** * Set the current local matrix as the restMatrix for this bone. */ setCurrentPoseAsRest() { this.setRestMatrix(this.getLocalMatrix()); } } da._TmpVecs = Nf.BuildArray(2, S.Zero); da._TmpQuat = Ze.Identity(); da._TmpMats = Nf.BuildArray(5, he.Identity); class ZN { /** * Gets the root Animatable used to synchronize and normalize animations */ get syncRoot() { return this._syncRoot; } /** * Gets the current frame of the first RuntimeAnimation * Used to synchronize Animatables */ get masterFrame() { return this._runtimeAnimations.length === 0 ? 0 : this._runtimeAnimations[0].currentFrame; } /** * Gets or sets the animatable weight (-1.0 by default meaning not weighted) */ get weight() { return this._weight; } set weight(e) { if (e === -1) { this._weight = -1; return; } this._weight = Math.min(Math.max(e, 0), 1); } /** * Gets or sets the speed ratio to apply to the animatable (1.0 by default) */ get speedRatio() { return this._speedRatio; } set speedRatio(e) { for (let t = 0; t < this._runtimeAnimations.length; t++) this._runtimeAnimations[t]._prepareForSpeedRatioChange(e); this._speedRatio = e, this._goToFrame !== null && this.goToFrame(this._goToFrame); } /** * Gets the elapsed time since the animatable started in milliseconds */ get elapsedTime() { return this._localDelayOffset === null ? 0 : this._scene._animationTime - this._localDelayOffset; } /** * Creates a new Animatable * @param scene defines the hosting scene * @param target defines the target object * @param fromFrame defines the starting frame number (default is 0) * @param toFrame defines the ending frame number (default is 100) * @param loopAnimation defines if the animation must loop (default is false) * @param speedRatio defines the factor to apply to animation speed (default is 1) * @param onAnimationEnd defines a callback to call when animation ends if it is not looping * @param animations defines a group of animation to add to the new Animatable * @param onAnimationLoop defines a callback to call when animation loops * @param isAdditive defines whether the animation should be evaluated additively * @param playOrder defines the order in which this animatable should be processed in the list of active animatables (default: 0) */ constructor(e, t, r = 0, n = 100, i = !1, s = 1, a, f, o, d = !1, v = 0) { this.target = t, this.fromFrame = r, this.toFrame = n, this.loopAnimation = i, this.onAnimationEnd = a, this.onAnimationLoop = o, this.isAdditive = d, this.playOrder = v, this._localDelayOffset = null, this._pausedDelay = null, this._manualJumpDelay = null, this._runtimeAnimations = new Array(), this._paused = !1, this._speedRatio = 1, this._weight = -1, this._syncRoot = null, this._frameToSyncFromJump = null, this._goToFrame = null, this.disposeOnEnd = !0, this.animationStarted = !1, this.onAnimationEndObservable = new Oe(), this.onAnimationLoopObservable = new Oe(), this._scene = e, f && this.appendAnimations(t, f), this._speedRatio = s, e._activeAnimatables.push(this); } // Methods /** * Synchronize and normalize current Animatable with a source Animatable * This is useful when using animation weights and when animations are not of the same length * @param root defines the root Animatable to synchronize with (null to stop synchronizing) * @returns the current Animatable */ syncWith(e) { if (this._syncRoot = e, e) { const t = this._scene._activeAnimatables.indexOf(this); t > -1 && (this._scene._activeAnimatables.splice(t, 1), this._scene._activeAnimatables.push(this)); } return this; } /** * Gets the list of runtime animations * @returns an array of RuntimeAnimation */ getAnimations() { return this._runtimeAnimations; } /** * Adds more animations to the current animatable * @param target defines the target of the animations * @param animations defines the new animations to add */ appendAnimations(e, t) { for (let r = 0; r < t.length; r++) { const n = t[r], i = new R$(e, n, this._scene, this); i._onLoop = () => { this.onAnimationLoopObservable.notifyObservers(this), this.onAnimationLoop && this.onAnimationLoop(); }, this._runtimeAnimations.push(i); } } /** * Gets the source animation for a specific property * @param property defines the property to look for * @returns null or the source animation for the given property */ getAnimationByTargetProperty(e) { const t = this._runtimeAnimations; for (let r = 0; r < t.length; r++) if (t[r].animation.targetProperty === e) return t[r].animation; return null; } /** * Gets the runtime animation for a specific property * @param property defines the property to look for * @returns null or the runtime animation for the given property */ getRuntimeAnimationByTargetProperty(e) { const t = this._runtimeAnimations; for (let r = 0; r < t.length; r++) if (t[r].animation.targetProperty === e) return t[r]; return null; } /** * Resets the animatable to its original state */ reset() { const e = this._runtimeAnimations; for (let t = 0; t < e.length; t++) e[t].reset(!0); this._localDelayOffset = null, this._pausedDelay = null; } /** * Allows the animatable to blend with current running animations * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#animation-blending * @param blendingSpeed defines the blending speed to use */ enableBlending(e) { const t = this._runtimeAnimations; for (let r = 0; r < t.length; r++) t[r].animation.enableBlending = !0, t[r].animation.blendingSpeed = e; } /** * Disable animation blending * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#animation-blending */ disableBlending() { const e = this._runtimeAnimations; for (let t = 0; t < e.length; t++) e[t].animation.enableBlending = !1; } /** * Jump directly to a given frame * @param frame defines the frame to jump to */ goToFrame(e) { var t; const r = this._runtimeAnimations; if (r[0]) { const n = r[0].animation.framePerSecond; this._frameToSyncFromJump = (t = this._frameToSyncFromJump) !== null && t !== void 0 ? t : r[0].currentFrame; const i = this.speedRatio === 0 ? 0 : (e - this._frameToSyncFromJump) / n * 1e3 / this.speedRatio; this._manualJumpDelay = -i; } for (let n = 0; n < r.length; n++) r[n].goToFrame(e); this._goToFrame = e; } /** * Returns true if the animations for this animatable are paused */ get paused() { return this._paused; } /** * Pause the animation */ pause() { this._paused || (this._paused = !0); } /** * Restart the animation */ restart() { this._paused = !1; } _raiseOnAnimationEnd() { this.onAnimationEnd && this.onAnimationEnd(), this.onAnimationEndObservable.notifyObservers(this); } /** * Stop and delete the current animation * @param animationName defines a string used to only stop some of the runtime animations instead of all * @param targetMask a function that determines if the animation should be stopped based on its target (all animations will be stopped if both this and animationName are empty) * @param useGlobalSplice if true, the animatables will be removed by the caller of this function (false by default) */ stop(e, t, r = !1) { if (e || t) { const n = this._scene._activeAnimatables.indexOf(this); if (n > -1) { const i = this._runtimeAnimations; for (let s = i.length - 1; s >= 0; s--) { const a = i[s]; e && a.animation.name != e || t && !t(a.target) || (a.dispose(), i.splice(s, 1)); } i.length == 0 && (r || this._scene._activeAnimatables.splice(n, 1), this._raiseOnAnimationEnd()); } } else { const n = this._scene._activeAnimatables.indexOf(this); if (n > -1) { r || this._scene._activeAnimatables.splice(n, 1); const i = this._runtimeAnimations; for (let s = 0; s < i.length; s++) i[s].dispose(); this._runtimeAnimations.length = 0, this._raiseOnAnimationEnd(); } } } /** * Wait asynchronously for the animation to end * @returns a promise which will be fulfilled when the animation ends */ waitAsync() { return new Promise((e) => { this.onAnimationEndObservable.add(() => { e(this); }, void 0, void 0, this, !0); }); } /** * @internal */ _animate(e) { if (this._paused) return this.animationStarted = !1, this._pausedDelay === null && (this._pausedDelay = e), !0; if (this._localDelayOffset === null ? (this._localDelayOffset = e, this._pausedDelay = null) : this._pausedDelay !== null && (this._localDelayOffset += e - this._pausedDelay, this._pausedDelay = null), this._manualJumpDelay !== null && (this._localDelayOffset += this._manualJumpDelay, this._manualJumpDelay = null, this._frameToSyncFromJump = null), this._goToFrame = null, this._weight === 0) return !0; let t = !1; const r = this._runtimeAnimations; let n; for (n = 0; n < r.length; n++) { const s = r[n].animate(e - this._localDelayOffset, this.fromFrame, this.toFrame, this.loopAnimation, this._speedRatio, this._weight); t = t || s; } if (this.animationStarted = t, !t) { if (this.disposeOnEnd) for (n = this._scene._activeAnimatables.indexOf(this), this._scene._activeAnimatables.splice(n, 1), n = 0; n < r.length; n++) r[n].dispose(); this._raiseOnAnimationEnd(), this.disposeOnEnd && (this.onAnimationEnd = null, this.onAnimationLoop = null, this.onAnimationLoopObservable.clear(), this.onAnimationEndObservable.clear()); } return t; } } sr.prototype._animate = function() { if (!this.animationsEnabled) return; const A = Yi.Now; if (!this._animationTimeLast) { if (this._pendingData.length > 0) return; this._animationTimeLast = A; } this.deltaTime = this.useConstantAnimationDeltaTime ? 16 : (A - this._animationTimeLast) * this.animationTimeScale, this._animationTimeLast = A; const e = this._activeAnimatables; if (e.length === 0) return; this._animationTime += this.deltaTime; const t = this._animationTime; for (let r = 0; r < e.length; r++) { const n = e[r]; !n._animate(t) && n.disposeOnEnd && r--; } this._processLateAnimationBindings(); }; sr.prototype.sortActiveAnimatables = function() { this._activeAnimatables.sort((A, e) => A.playOrder - e.playOrder); }; sr.prototype.beginWeightedAnimation = function(A, e, t, r = 1, n, i = 1, s, a, f, o, d = !1) { const v = this.beginAnimation(A, e, t, n, i, s, a, !1, f, o, d); return v.weight = r, v; }; sr.prototype.beginAnimation = function(A, e, t, r, n = 1, i, s, a = !0, f, o, d = !1) { e > t && n > 0 && (n *= -1), a && this.stopAnimation(A, void 0, f), s || (s = new ZN(this, A, e, t, r, n, i, void 0, o, d)); const v = f ? f(A) : !0; if (A.animations && v && s.appendAnimations(A, A.animations), A.getAnimatables) { const u = A.getAnimatables(); for (let l = 0; l < u.length; l++) this.beginAnimation(u[l], e, t, r, n, i, s, a, f, o); } return s.reset(), s; }; sr.prototype.beginHierarchyAnimation = function(A, e, t, r, n, i = 1, s, a, f = !0, o, d, v = !1) { const u = A.getDescendants(e), l = []; l.push(this.beginAnimation(A, t, r, n, i, s, a, f, o, void 0, v)); for (const P of u) l.push(this.beginAnimation(P, t, r, n, i, s, a, f, o, void 0, v)); return l; }; sr.prototype.beginDirectAnimation = function(A, e, t, r, n, i, s, a, f = !1) { if (i === void 0 && (i = 1), t > r && i > 0) i *= -1; else if (r > t && i < 0) { const d = r; r = t, t = d; } return new ZN(this, A, t, r, n, i, s, e, a, f); }; sr.prototype.beginDirectHierarchyAnimation = function(A, e, t, r, n, i, s, a, f, o = !1) { const d = A.getDescendants(e), v = []; v.push(this.beginDirectAnimation(A, t, r, n, i, s, a, f, o)); for (const u of d) v.push(this.beginDirectAnimation(u, t, r, n, i, s, a, f, o)); return v; }; sr.prototype.getAnimatableByTarget = function(A) { for (let e = 0; e < this._activeAnimatables.length; e++) if (this._activeAnimatables[e].target === A) return this._activeAnimatables[e]; return null; }; sr.prototype.getAllAnimatablesByTarget = function(A) { const e = []; for (let t = 0; t < this._activeAnimatables.length; t++) this._activeAnimatables[t].target === A && e.push(this._activeAnimatables[t]); return e; }; sr.prototype.stopAnimation = function(A, e, t) { const r = this.getAllAnimatablesByTarget(A); for (const n of r) n.stop(e, t); }; sr.prototype.stopAllAnimations = function() { if (this._activeAnimatables) { for (let A = 0; A < this._activeAnimatables.length; A++) this._activeAnimatables[A].stop(void 0, void 0, !0); this._activeAnimatables.length = 0; } for (const A of this.animationGroups) A.stop(); }; sr.prototype._registerTargetForLateAnimationBinding = function(A, e) { const t = A.target; this._registeredForLateAnimationBindings.pushNoDuplicate(t), t._lateAnimationHolders || (t._lateAnimationHolders = {}), t._lateAnimationHolders[A.targetPath] || (t._lateAnimationHolders[A.targetPath] = { totalWeight: 0, totalAdditiveWeight: 0, animations: [], additiveAnimations: [], originalValue: e }), A.isAdditive ? (t._lateAnimationHolders[A.targetPath].additiveAnimations.push(A), t._lateAnimationHolders[A.targetPath].totalAdditiveWeight += A.weight) : (t._lateAnimationHolders[A.targetPath].animations.push(A), t._lateAnimationHolders[A.targetPath].totalWeight += A.weight); }; sr.prototype._processLateAnimationBindingsForMatrices = function(A) { if (A.totalWeight === 0 && A.totalAdditiveWeight === 0) return A.originalValue; let e = 1; const t = ue.Vector3[0], r = ue.Vector3[1], n = ue.Quaternion[0]; let i = 0; const s = A.animations[0], a = A.originalValue; let f = 1, o = !1; if (A.totalWeight < 1) f = 1 - A.totalWeight, a.decompose(r, n, t); else { if (i = 1, e = A.totalWeight, f = s.weight / e, f == 1) if (A.totalAdditiveWeight) o = !0; else return s.currentValue; s.currentValue.decompose(r, n, t); } if (!o) { r.scaleInPlace(f), t.scaleInPlace(f), n.scaleInPlace(f); for (let v = i; v < A.animations.length; v++) { const u = A.animations[v]; if (u.weight === 0) continue; f = u.weight / e; const l = ue.Vector3[2], P = ue.Vector3[3], p = ue.Quaternion[1]; u.currentValue.decompose(P, p, l), P.scaleAndAddToRef(f, r), p.scaleAndAddToRef(Ze.Dot(n, p) > 0 ? f : -f, n), l.scaleAndAddToRef(f, t); } n.normalize(); } for (let v = 0; v < A.additiveAnimations.length; v++) { const u = A.additiveAnimations[v]; if (u.weight === 0) continue; const l = ue.Vector3[2], P = ue.Vector3[3], p = ue.Quaternion[1]; u.currentValue.decompose(P, p, l), P.multiplyToRef(r, P), S.LerpToRef(r, P, u.weight, r), n.multiplyToRef(p, p), Ze.SlerpToRef(n, p, u.weight, n), l.scaleAndAddToRef(u.weight, t); } const d = s ? s._animationState.workValue : ue.Matrix[0].clone(); return he.ComposeToRef(r, n, t, d), d; }; sr.prototype._processLateAnimationBindingsForQuaternions = function(A, e) { if (A.totalWeight === 0 && A.totalAdditiveWeight === 0) return e; const t = A.animations[0], r = A.originalValue; let n = e; if (A.totalWeight === 0 && A.totalAdditiveWeight > 0) n.copyFrom(r); else if (A.animations.length === 1) { if (Ze.SlerpToRef(r, t.currentValue, Math.min(1, A.totalWeight), n), A.totalAdditiveWeight === 0) return n; } else if (A.animations.length > 1) { let i = 1, s, a; if (A.totalWeight < 1) { const o = 1 - A.totalWeight; s = [], a = [], s.push(r), a.push(o); } else { if (A.animations.length === 2 && (Ze.SlerpToRef(A.animations[0].currentValue, A.animations[1].currentValue, A.animations[1].weight / A.totalWeight, e), A.totalAdditiveWeight === 0)) return e; s = [], a = [], i = A.totalWeight; } for (let o = 0; o < A.animations.length; o++) { const d = A.animations[o]; s.push(d.currentValue), a.push(d.weight / i); } let f = 0; for (let o = 0; o < s.length; ) { if (!o) { Ze.SlerpToRef(s[o], s[o + 1], a[o + 1] / (a[o] + a[o + 1]), e), n = e, f = a[o] + a[o + 1], o += 2; continue; } f += a[o], Ze.SlerpToRef(n, s[o], a[o] / f, n), o++; } } for (let i = 0; i < A.additiveAnimations.length; i++) { const s = A.additiveAnimations[i]; s.weight !== 0 && (n.multiplyToRef(s.currentValue, ue.Quaternion[0]), Ze.SlerpToRef(n, ue.Quaternion[0], s.weight, n)); } return n; }; sr.prototype._processLateAnimationBindings = function() { if (this._registeredForLateAnimationBindings.length) { for (let A = 0; A < this._registeredForLateAnimationBindings.length; A++) { const e = this._registeredForLateAnimationBindings.data[A]; for (const t in e._lateAnimationHolders) { const r = e._lateAnimationHolders[t], n = r.animations[0], i = r.originalValue; if (i == null) continue; const s = st.AllowMatrixDecomposeForInterpolation && i.m; let a = e[t]; if (s) a = this._processLateAnimationBindingsForMatrices(r); else if (i.w !== void 0) a = this._processLateAnimationBindingsForQuaternions(r, a || Ze.Identity()); else { let o = 0, d = 1; const v = n && n._animationState.loopMode === st.ANIMATIONLOOPMODE_RELATIVE_FROM_CURRENT; if (r.totalWeight < 1) v ? a = i.clone ? i.clone() : i : n && i.scale ? a = i.scale(1 - r.totalWeight) : n ? a = i * (1 - r.totalWeight) : i.clone ? a = i.clone() : a = i; else if (n) { d = r.totalWeight; const u = n.weight / d; u !== 1 ? n.currentValue.scale ? a = n.currentValue.scale(u) : a = n.currentValue * u : a = n.currentValue, v && (a.addToRef ? a.addToRef(i, a) : a += i), o = 1; } for (let u = o; u < r.animations.length; u++) { const l = r.animations[u], P = l.weight / d; if (P) l.currentValue.scaleAndAddToRef ? l.currentValue.scaleAndAddToRef(P, a) : a += l.currentValue * P; else continue; } for (let u = 0; u < r.additiveAnimations.length; u++) { const l = r.additiveAnimations[u], P = l.weight; if (P) l.currentValue.scaleAndAddToRef ? l.currentValue.scaleAndAddToRef(P, a) : a += l.currentValue * P; else continue; } } e[t] = a; } e._lateAnimationHolders = {}; } this._registeredForLateAnimationBindings.reset(); } }; da.prototype.copyAnimationRange = function(A, e, t, r = !1, n = null) { this.animations.length === 0 && (this.animations.push(new st(this.name, "_matrix", A.animations[0].framePerSecond, st.ANIMATIONTYPE_MATRIX, 0)), this.animations[0].setKeys([])); const i = A.animations[0].getRange(e); if (!i) return !1; const s = i.from, a = i.to, f = A.animations[0].getKeys(), o = A.length, d = A.getParent(), v = this.getParent(), u = r && d && o && this.length && o !== this.length, l = u && v && d ? v.length / d.length : 1, P = r && !v && n && (n.x !== 1 || n.y !== 1 || n.z !== 1), p = this.animations[0].getKeys(); let c, H, T; for (let q = 0, b = f.length; q < b; q++) c = f[q], c.frame >= s && c.frame <= a && (r ? (T = c.value.clone(), u ? (H = T.getTranslation(), T.setTranslation(H.scaleInPlace(l))) : P && n ? (H = T.getTranslation(), T.setTranslation(H.multiplyInPlace(n))) : T = c.value) : T = c.value, p.push({ frame: c.frame + t, value: T })); return this.animations[0].createRange(e, s + t, a + t), !0; }; class t6e { constructor() { this.enableBlending = !1, this.blendingSpeed = 0.01, this.loopMode = st.ANIMATIONLOOPMODE_CYCLE; } } var tm; (function(A) { A[A.CW = 0] = "CW", A[A.CCW = 1] = "CCW"; })(tm || (tm = {})); class z$ { /** * Returns the cubic Bezier interpolated value (float) at "t" (float) from the given x1, y1, x2, y2 floats * @param t defines the time * @param x1 defines the left coordinate on X axis * @param y1 defines the left coordinate on Y axis * @param x2 defines the right coordinate on X axis * @param y2 defines the right coordinate on Y axis * @returns the interpolated value */ static Interpolate(e, t, r, n, i) { const s = 1 - 3 * n + 3 * t, a = 3 * n - 6 * t, f = 3 * t; let o = e; for (let d = 0; d < 5; d++) { const v = o * o, u = v * o, l = s * u + a * v + f * o, P = 1 / (3 * s * v + 2 * a * o + f); o -= (l - e) * P, o = Math.min(1, Math.max(0, o)); } return 3 * Math.pow(1 - o, 2) * o * r + 3 * (1 - o) * Math.pow(o, 2) * i + Math.pow(o, 3); } } class JH { /** * Creates an Angle object of "radians" radians (float). * @param radians the angle in radians */ constructor(e) { this._radians = e, this._radians < 0 && (this._radians += 2 * Math.PI); } /** * Get value in degrees * @returns the Angle value in degrees (float) */ degrees() { return this._radians * 180 / Math.PI; } /** * Get value in radians * @returns the Angle value in radians (float) */ radians() { return this._radians; } /** * Gets a new Angle object with a value of the angle (in radians) between the line connecting the two points and the x-axis * @param a defines first point as the origin * @param b defines point * @returns a new Angle */ static BetweenTwoPoints(e, t) { const r = t.subtract(e), n = Math.atan2(r.y, r.x); return new JH(n); } /** * Gets the angle between the two vectors * @param a defines first vector * @param b defines vector * @returns Returns an new Angle between 0 and PI */ static BetweenTwoVectors(e, t) { let r = e.lengthSquared() * t.lengthSquared(); if (r === 0) return new JH(Math.PI / 2); r = Math.sqrt(r); let n = e.dot(t) / r; n = Xt.Clamp(n, -1, 1); const i = Math.acos(n); return new JH(i); } /** * Gets a new Angle object from the given float in radians * @param radians defines the angle value in radians * @returns a new Angle */ static FromRadians(e) { return new JH(e); } /** * Gets a new Angle object from the given float in degrees * @param degrees defines the angle value in degrees * @returns a new Angle */ static FromDegrees(e) { return new JH(e * Math.PI / 180); } } class G$ { /** * Creates an Arc object from the three given points : start, middle and end. * @param startPoint Defines the start point of the arc * @param midPoint Defines the middle point of the arc * @param endPoint Defines the end point of the arc */ constructor(e, t, r) { this.startPoint = e, this.midPoint = t, this.endPoint = r; const n = Math.pow(t.x, 2) + Math.pow(t.y, 2), i = (Math.pow(e.x, 2) + Math.pow(e.y, 2) - n) / 2, s = (n - Math.pow(r.x, 2) - Math.pow(r.y, 2)) / 2, a = (e.x - t.x) * (t.y - r.y) - (t.x - r.x) * (e.y - t.y); this.centerPoint = new at((i * (t.y - r.y) - s * (e.y - t.y)) / a, ((e.x - t.x) * s - (t.x - r.x) * i) / a), this.radius = this.centerPoint.subtract(this.startPoint).length(), this.startAngle = JH.BetweenTwoPoints(this.centerPoint, this.startPoint); const f = this.startAngle.degrees(); let o = JH.BetweenTwoPoints(this.centerPoint, this.midPoint).degrees(), d = JH.BetweenTwoPoints(this.centerPoint, this.endPoint).degrees(); o - f > 180 && (o -= 360), o - f < -180 && (o += 360), d - o > 180 && (d -= 360), d - o < -180 && (d += 360), this.orientation = o - f < 0 ? tm.CW : tm.CCW, this.angle = JH.FromDegrees(this.orientation === tm.CW ? f - d : d - f); } } class sU { /** * Creates a Path2 object from the starting 2D coordinates x and y. * @param x the starting points x value * @param y the starting points y value */ constructor(e, t) { this._points = new Array(), this._length = 0, this.closed = !1, this._points.push(new at(e, t)); } /** * Adds a new segment until the given coordinates (x, y) to the current Path2. * @param x the added points x value * @param y the added points y value * @returns the updated Path2. */ addLineTo(e, t) { if (this.closed) return this; const r = new at(e, t), n = this._points[this._points.length - 1]; return this._points.push(r), this._length += r.subtract(n).length(), this; } /** * Adds _numberOfSegments_ segments according to the arc definition (middle point coordinates, end point coordinates, the arc start point being the current Path2 last point) to the current Path2. * @param midX middle point x value * @param midY middle point y value * @param endX end point x value * @param endY end point y value * @param numberOfSegments (default: 36) * @returns the updated Path2. */ addArcTo(e, t, r, n, i = 36) { if (this.closed) return this; const s = this._points[this._points.length - 1], a = new at(e, t), f = new at(r, n), o = new G$(s, a, f); let d = o.angle.radians() / i; o.orientation === tm.CW && (d *= -1); let v = o.startAngle.radians() + d; for (let u = 0; u < i; u++) { const l = Math.cos(v) * o.radius + o.centerPoint.x, P = Math.sin(v) * o.radius + o.centerPoint.y; this.addLineTo(l, P), v += d; } return this; } /** * Adds _numberOfSegments_ segments according to the quadratic curve definition to the current Path2. * @param controlX control point x value * @param controlY control point y value * @param endX end point x value * @param endY end point y value * @param numberOfSegments (default: 36) * @returns the updated Path2. */ addQuadraticCurveTo(e, t, r, n, i = 36) { if (this.closed) return this; const s = (f, o, d, v) => (1 - f) * (1 - f) * o + 2 * f * (1 - f) * d + f * f * v, a = this._points[this._points.length - 1]; for (let f = 0; f <= i; f++) { const o = f / i, d = s(o, a.x, e, r), v = s(o, a.y, t, n); this.addLineTo(d, v); } return this; } /** * Adds _numberOfSegments_ segments according to the bezier curve definition to the current Path2. * @param originTangentX tangent vector at the origin point x value * @param originTangentY tangent vector at the origin point y value * @param destinationTangentX tangent vector at the destination point x value * @param destinationTangentY tangent vector at the destination point y value * @param endX end point x value * @param endY end point y value * @param numberOfSegments (default: 36) * @returns the updated Path2. */ addBezierCurveTo(e, t, r, n, i, s, a = 36) { if (this.closed) return this; const f = (d, v, u, l, P) => (1 - d) * (1 - d) * (1 - d) * v + 3 * d * (1 - d) * (1 - d) * u + 3 * d * d * (1 - d) * l + d * d * d * P, o = this._points[this._points.length - 1]; for (let d = 0; d <= a; d++) { const v = d / a, u = f(v, o.x, e, r, i), l = f(v, o.y, t, n, s); this.addLineTo(u, l); } return this; } /** * Defines if a given point is inside the polygon defines by the path * @param point defines the point to test * @returns true if the point is inside */ isPointInside(e) { let t = !1; const r = this._points.length; for (let n = r - 1, i = 0; i < r; n = i++) { let s = this._points[n], a = this._points[i], f = a.x - s.x, o = a.y - s.y; if (Math.abs(o) > Number.EPSILON) { if (o < 0 && (s = this._points[i], f = -f, a = this._points[n], o = -o), e.y < s.y || e.y > a.y) continue; if (e.y === s.y && e.x === s.x) return !0; { const d = o * (e.x - s.x) - f * (e.y - s.y); if (d === 0) return !0; if (d < 0) continue; t = !t; } } else { if (e.y !== s.y) continue; if (a.x <= e.x && e.x <= s.x || s.x <= e.x && e.x <= a.x) return !0; } } return t; } /** * Closes the Path2. * @returns the Path2. */ close() { return this.closed = !0, this; } /** * Gets the sum of the distance between each sequential point in the path * @returns the Path2 total length (float). */ length() { let e = this._length; if (this.closed) { const t = this._points[this._points.length - 1], r = this._points[0]; e += r.subtract(t).length(); } return e; } /** * Gets the area of the polygon defined by the path * @returns area value */ area() { const e = this._points.length; let t = 0; for (let r = e - 1, n = 0; n < e; r = n++) t += this._points[r].x * this._points[n].y - this._points[n].x * this._points[r].y; return t * 0.5; } /** * Gets the points which construct the path * @returns the Path2 internal array of points. */ getPoints() { return this._points; } /** * Retrieves the point at the distance aways from the starting point * @param normalizedLengthPosition the length along the path to retrieve the point from * @returns a new Vector2 located at a percentage of the Path2 total length on this path. */ getPointAtLengthPosition(e) { if (e < 0 || e > 1) return at.Zero(); const t = e * this.length(); let r = 0; for (let n = 0; n < this._points.length; n++) { const i = (n + 1) % this._points.length, s = this._points[n], f = this._points[i].subtract(s), o = f.length() + r; if (t >= r && t <= o) { const d = f.normalize(), v = t - r; return new at(s.x + d.x * v, s.y + d.y * v); } r = o; } return at.Zero(); } /** * Creates a new path starting from an x and y position * @param x starting x value * @param y starting y value * @returns a new Path2 starting at the coordinates (x, y). */ static StartingAt(e, t) { return new sU(e, t); } } class vm { /** * new Path3D(path, normal, raw) * Creates a Path3D. A Path3D is a logical math object, so not a mesh. * please read the description in the tutorial : https://doc.babylonjs.com/features/featuresDeepDive/mesh/path3D * @param path an array of Vector3, the curve axis of the Path3D * @param firstNormal (options) Vector3, the first wanted normal to the curve. Ex (0, 1, 0) for a vertical normal. * @param raw (optional, default false) : boolean, if true the returned Path3D isn't normalized. Useful to depict path acceleration or speed. * @param alignTangentsWithPath (optional, default false) : boolean, if true the tangents will be aligned with the path. */ constructor(e, t = null, r, n = !1) { this.path = e, this._curve = new Array(), this._distances = new Array(), this._tangents = new Array(), this._normals = new Array(), this._binormals = new Array(), this._pointAtData = { id: 0, point: S.Zero(), previousPointArrayIndex: 0, position: 0, subPosition: 0, interpolateReady: !1, interpolationMatrix: he.Identity() }; for (let i = 0; i < e.length; i++) this._curve[i] = e[i].clone(); this._raw = r || !1, this._alignTangentsWithPath = n, this._compute(t, n); } /** * Returns the Path3D array of successive Vector3 designing its curve. * @returns the Path3D array of successive Vector3 designing its curve. */ getCurve() { return this._curve; } /** * Returns the Path3D array of successive Vector3 designing its curve. * @returns the Path3D array of successive Vector3 designing its curve. */ getPoints() { return this._curve; } /** * @returns the computed length (float) of the path. */ length() { return this._distances[this._distances.length - 1]; } /** * Returns an array populated with tangent vectors on each Path3D curve point. * @returns an array populated with tangent vectors on each Path3D curve point. */ getTangents() { return this._tangents; } /** * Returns an array populated with normal vectors on each Path3D curve point. * @returns an array populated with normal vectors on each Path3D curve point. */ getNormals() { return this._normals; } /** * Returns an array populated with binormal vectors on each Path3D curve point. * @returns an array populated with binormal vectors on each Path3D curve point. */ getBinormals() { return this._binormals; } /** * Returns an array populated with distances (float) of the i-th point from the first curve point. * @returns an array populated with distances (float) of the i-th point from the first curve point. */ getDistances() { return this._distances; } /** * Returns an interpolated point along this path * @param position the position of the point along this path, from 0.0 to 1.0 * @returns a new Vector3 as the point */ getPointAt(e) { return this._updatePointAtData(e).point; } /** * Returns the tangent vector of an interpolated Path3D curve point at the specified position along this path. * @param position the position of the point along this path, from 0.0 to 1.0 * @param interpolated (optional, default false) : boolean, if true returns an interpolated tangent instead of the tangent of the previous path point. * @returns a tangent vector corresponding to the interpolated Path3D curve point, if not interpolated, the tangent is taken from the precomputed tangents array. */ getTangentAt(e, t = !1) { return this._updatePointAtData(e, t), t ? S.TransformCoordinates(S.Forward(), this._pointAtData.interpolationMatrix) : this._tangents[this._pointAtData.previousPointArrayIndex]; } /** * Returns the tangent vector of an interpolated Path3D curve point at the specified position along this path. * @param position the position of the point along this path, from 0.0 to 1.0 * @param interpolated (optional, default false) : boolean, if true returns an interpolated normal instead of the normal of the previous path point. * @returns a normal vector corresponding to the interpolated Path3D curve point, if not interpolated, the normal is taken from the precomputed normals array. */ getNormalAt(e, t = !1) { return this._updatePointAtData(e, t), t ? S.TransformCoordinates(S.Right(), this._pointAtData.interpolationMatrix) : this._normals[this._pointAtData.previousPointArrayIndex]; } /** * Returns the binormal vector of an interpolated Path3D curve point at the specified position along this path. * @param position the position of the point along this path, from 0.0 to 1.0 * @param interpolated (optional, default false) : boolean, if true returns an interpolated binormal instead of the binormal of the previous path point. * @returns a binormal vector corresponding to the interpolated Path3D curve point, if not interpolated, the binormal is taken from the precomputed binormals array. */ getBinormalAt(e, t = !1) { return this._updatePointAtData(e, t), t ? S.TransformCoordinates(S.UpReadOnly, this._pointAtData.interpolationMatrix) : this._binormals[this._pointAtData.previousPointArrayIndex]; } /** * Returns the distance (float) of an interpolated Path3D curve point at the specified position along this path. * @param position the position of the point along this path, from 0.0 to 1.0 * @returns the distance of the interpolated Path3D curve point at the specified position along this path. */ getDistanceAt(e) { return this.length() * e; } /** * Returns the array index of the previous point of an interpolated point along this path * @param position the position of the point to interpolate along this path, from 0.0 to 1.0 * @returns the array index */ getPreviousPointIndexAt(e) { return this._updatePointAtData(e), this._pointAtData.previousPointArrayIndex; } /** * Returns the position of an interpolated point relative to the two path points it lies between, from 0.0 (point A) to 1.0 (point B) * @param position the position of the point to interpolate along this path, from 0.0 to 1.0 * @returns the sub position */ getSubPositionAt(e) { return this._updatePointAtData(e), this._pointAtData.subPosition; } /** * Returns the position of the closest virtual point on this path to an arbitrary Vector3, from 0.0 to 1.0 * @param target the vector of which to get the closest position to * @returns the position of the closest virtual point on this path to the target vector */ getClosestPositionTo(e) { let t = Number.MAX_VALUE, r = 0; for (let n = 0; n < this._curve.length - 1; n++) { const i = this._curve[n + 0], s = this._curve[n + 1].subtract(i).normalize(), a = this._distances[n + 1] - this._distances[n + 0], f = Math.min(Math.max(S.Dot(s, e.subtract(i).normalize()), 0) * S.Distance(i, e) / a, 1), o = S.Distance(i.add(s.scale(f * a)), e); o < t && (t = o, r = (this._distances[n + 0] + a * f) / this.length()); } return r; } /** * Returns a sub path (slice) of this path * @param start the position of the fist path point, from 0.0 to 1.0, or a negative value, which will get wrapped around from the end of the path to 0.0 to 1.0 values * @param end the position of the last path point, from 0.0 to 1.0, or a negative value, which will get wrapped around from the end of the path to 0.0 to 1.0 values * @returns a sub path (slice) of this path */ slice(e = 0, t = 1) { if (e < 0 && (e = 1 - e * -1 % 1), t < 0 && (t = 1 - t * -1 % 1), e > t) { const o = e; e = t, t = o; } const r = this.getCurve(), n = this.getPointAt(e); let i = this.getPreviousPointIndexAt(e); const s = this.getPointAt(t), a = this.getPreviousPointIndexAt(t) + 1, f = []; return e !== 0 && (i++, f.push(n)), f.push(...r.slice(i, a)), (t !== 1 || e === 1) && f.push(s), new vm(f, this.getNormalAt(e), this._raw, this._alignTangentsWithPath); } /** * Forces the Path3D tangent, normal, binormal and distance recomputation. * @param path path which all values are copied into the curves points * @param firstNormal which should be projected onto the curve * @param alignTangentsWithPath (optional, default false) : boolean, if true the tangents will be aligned with the path * @returns the same object updated. */ update(e, t = null, r = !1) { for (let n = 0; n < e.length; n++) this._curve[n].x = e[n].x, this._curve[n].y = e[n].y, this._curve[n].z = e[n].z; return this._compute(t, r), this; } // private function compute() : computes tangents, normals and binormals _compute(e, t = !1) { const r = this._curve.length; if (r < 2) return; this._tangents[0] = this._getFirstNonNullVector(0), this._raw || this._tangents[0].normalize(), this._tangents[r - 1] = this._curve[r - 1].subtract(this._curve[r - 2]), this._raw || this._tangents[r - 1].normalize(); const n = this._tangents[0], i = this._normalVector(n, e); this._normals[0] = i, this._raw || this._normals[0].normalize(), this._binormals[0] = S.Cross(n, this._normals[0]), this._raw || this._binormals[0].normalize(), this._distances[0] = 0; let s, a, f, o, d; for (let v = 1; v < r; v++) s = this._getLastNonNullVector(v), v < r - 1 && (a = this._getFirstNonNullVector(v), this._tangents[v] = t ? a : s.add(a), this._tangents[v].normalize()), this._distances[v] = this._distances[v - 1] + this._curve[v].subtract(this._curve[v - 1]).length(), f = this._tangents[v], d = this._binormals[v - 1], this._normals[v] = S.Cross(d, f), this._raw || (this._normals[v].length() === 0 ? (o = this._normals[v - 1], this._normals[v] = o.clone()) : this._normals[v].normalize()), this._binormals[v] = S.Cross(f, this._normals[v]), this._raw || this._binormals[v].normalize(); this._pointAtData.id = NaN; } // private function getFirstNonNullVector(index) // returns the first non null vector from index : curve[index + N].subtract(curve[index]) _getFirstNonNullVector(e) { let t = 1, r = this._curve[e + t].subtract(this._curve[e]); for (; r.length() === 0 && e + t + 1 < this._curve.length; ) t++, r = this._curve[e + t].subtract(this._curve[e]); return r; } // private function getLastNonNullVector(index) // returns the last non null vector from index : curve[index].subtract(curve[index - N]) _getLastNonNullVector(e) { let t = 1, r = this._curve[e].subtract(this._curve[e - t]); for (; r.length() === 0 && e > t + 1; ) t++, r = this._curve[e].subtract(this._curve[e - t]); return r; } // private function normalVector(v0, vt, va) : // returns an arbitrary point in the plane defined by the point v0 and the vector vt orthogonal to this plane // if va is passed, it returns the va projection on the plane orthogonal to vt at the point v0 _normalVector(e, t) { let r, n = e.length(); if (n === 0 && (n = 1), t == null) { let i; Xt.WithinEpsilon(Math.abs(e.y) / n, 1, Dn) ? Xt.WithinEpsilon(Math.abs(e.x) / n, 1, Dn) ? Xt.WithinEpsilon(Math.abs(e.z) / n, 1, Dn) ? i = S.Zero() : i = new S(0, 0, 1) : i = new S(1, 0, 0) : i = new S(0, -1, 0), r = S.Cross(e, i); } else r = S.Cross(e, t), S.CrossToRef(r, e, r); return r.normalize(), r; } /** * Updates the point at data for an interpolated point along this curve * @param position the position of the point along this curve, from 0.0 to 1.0 * @param interpolateTNB * @interpolateTNB whether to compute the interpolated tangent, normal and binormal * @returns the (updated) point at data */ _updatePointAtData(e, t = !1) { if (this._pointAtData.id === e) return this._pointAtData.interpolateReady || this._updateInterpolationMatrix(), this._pointAtData; this._pointAtData.id = e; const r = this.getPoints(); if (e <= 0) return this._setPointAtData(0, 0, r[0], 0, t); if (e >= 1) return this._setPointAtData(1, 1, r[r.length - 1], r.length - 1, t); let n = r[0], i, s = 0; const a = e * this.length(); for (let f = 1; f < r.length; f++) { i = r[f]; const o = S.Distance(n, i); if (s += o, s === a) return this._setPointAtData(e, 1, i, f, t); if (s > a) { const v = (s - a) / o, u = n.subtract(i), l = i.add(u.scaleInPlace(v)); return this._setPointAtData(e, 1 - v, l, f - 1, t); } n = i; } return this._pointAtData; } /** * Updates the point at data from the specified parameters * @param position where along the path the interpolated point is, from 0.0 to 1.0 * @param subPosition * @param point the interpolated point * @param parentIndex the index of an existing curve point that is on, or else positionally the first behind, the interpolated point * @param interpolateTNB */ _setPointAtData(e, t, r, n, i) { return this._pointAtData.point = r, this._pointAtData.position = e, this._pointAtData.subPosition = t, this._pointAtData.previousPointArrayIndex = n, this._pointAtData.interpolateReady = i, i && this._updateInterpolationMatrix(), this._pointAtData; } /** * Updates the point at interpolation matrix for the tangents, normals and binormals */ _updateInterpolationMatrix() { this._pointAtData.interpolationMatrix = he.Identity(); const e = this._pointAtData.previousPointArrayIndex; if (e !== this._tangents.length - 1) { const t = e + 1, r = this._tangents[e].clone(), n = this._normals[e].clone(), i = this._binormals[e].clone(), s = this._tangents[t].clone(), a = this._normals[t].clone(), f = this._binormals[t].clone(), o = Ze.RotationQuaternionFromAxis(n, i, r), d = Ze.RotationQuaternionFromAxis(a, f, s); Ze.Slerp(o, d, this._pointAtData.subPosition).toRotationMatrix(this._pointAtData.interpolationMatrix); } } } class j0 { /** * Returns a Curve3 object along a Quadratic Bezier curve : https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves#quadratic-bezier-curve * @param v0 (Vector3) the origin point of the Quadratic Bezier * @param v1 (Vector3) the control point * @param v2 (Vector3) the end point of the Quadratic Bezier * @param nbPoints (integer) the wanted number of points in the curve * @returns the created Curve3 */ static CreateQuadraticBezier(e, t, r, n) { n = n > 2 ? n : 3; const i = [], s = (a, f, o, d) => (1 - a) * (1 - a) * f + 2 * a * (1 - a) * o + a * a * d; for (let a = 0; a <= n; a++) i.push(new S(s(a / n, e.x, t.x, r.x), s(a / n, e.y, t.y, r.y), s(a / n, e.z, t.z, r.z))); return new j0(i); } /** * Returns a Curve3 object along a Cubic Bezier curve : https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves#cubic-bezier-curve * @param v0 (Vector3) the origin point of the Cubic Bezier * @param v1 (Vector3) the first control point * @param v2 (Vector3) the second control point * @param v3 (Vector3) the end point of the Cubic Bezier * @param nbPoints (integer) the wanted number of points in the curve * @returns the created Curve3 */ static CreateCubicBezier(e, t, r, n, i) { i = i > 3 ? i : 4; const s = [], a = (f, o, d, v, u) => (1 - f) * (1 - f) * (1 - f) * o + 3 * f * (1 - f) * (1 - f) * d + 3 * f * f * (1 - f) * v + f * f * f * u; for (let f = 0; f <= i; f++) s.push(new S(a(f / i, e.x, t.x, r.x, n.x), a(f / i, e.y, t.y, r.y, n.y), a(f / i, e.z, t.z, r.z, n.z))); return new j0(s); } /** * Returns a Curve3 object along a Hermite Spline curve : https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves#hermite-spline * @param p1 (Vector3) the origin point of the Hermite Spline * @param t1 (Vector3) the tangent vector at the origin point * @param p2 (Vector3) the end point of the Hermite Spline * @param t2 (Vector3) the tangent vector at the end point * @param nSeg (integer) the number of curve segments or nSeg + 1 points in the array * @returns the created Curve3 */ static CreateHermiteSpline(e, t, r, n, i) { const s = [], a = 1 / i; for (let f = 0; f <= i; f++) s.push(S.Hermite(e, t, r, n, f * a)); return new j0(s); } /** * Returns a Curve3 object along a CatmullRom Spline curve : * @param points (array of Vector3) the points the spline must pass through. At least, four points required * @param nbPoints (integer) the wanted number of points between each curve control points * @param closed (boolean) optional with default false, when true forms a closed loop from the points * @returns the created Curve3 */ static CreateCatmullRomSpline(e, t, r) { const n = [], i = 1 / t; let s = 0; if (r) { const a = e.length; for (let f = 0; f < a; f++) { s = 0; for (let o = 0; o < t; o++) n.push(S.CatmullRom(e[f % a], e[(f + 1) % a], e[(f + 2) % a], e[(f + 3) % a], s)), s += i; } n.push(n[0]); } else { const a = []; a.push(e[0].clone()), Array.prototype.push.apply(a, e), a.push(e[e.length - 1].clone()); let f = 0; for (; f < a.length - 3; f++) { s = 0; for (let o = 0; o < t; o++) n.push(S.CatmullRom(a[f], a[f + 1], a[f + 2], a[f + 3], s)), s += i; } f--, n.push(S.CatmullRom(a[f], a[f + 1], a[f + 2], a[f + 3], s)); } return new j0(n); } /** * Returns a Curve3 object along an arc through three vector3 points: * The three points should not be colinear. When they are the Curve3 is empty. * @param first (Vector3) the first point the arc must pass through. * @param second (Vector3) the second point the arc must pass through. * @param third (Vector3) the third point the arc must pass through. * @param steps (number) the larger the number of steps the more detailed the arc. * @param closed (boolean) optional with default false, when true forms the chord from the first and third point * @param fullCircle Circle (boolean) optional with default false, when true forms the complete circle through the three points * @returns the created Curve3 */ static ArcThru3Points(e, t, r, n = 32, i = !1, s = !1) { const a = [], f = t.subtract(e), o = r.subtract(t), d = e.subtract(r), v = S.Cross(f, o), u = v.length(); if (u < Math.pow(10, -8)) return new j0(a); const l = f.lengthSquared(), P = o.lengthSquared(), p = d.lengthSquared(), c = v.lengthSquared(), H = f.length(), T = o.length(), q = d.length(), b = 0.5 * H * T * q / u, j = S.Dot(f, d), w = S.Dot(f, o), m = S.Dot(o, d), I = -0.5 * P * j / c, N = -0.5 * p * w / c, k = -0.5 * l * m / c, R = e.scale(I).add(t.scale(N)).add(r.scale(k)), O = e.subtract(R).normalize(), Y = S.Cross(v, O).normalize(); if (s) { const ee = 2 * Math.PI / n; for (let Z = 0; Z <= 2 * Math.PI; Z += ee) a.push(R.add(O.scale(b * Math.cos(Z)).add(Y.scale(b * Math.sin(Z))))); a.push(e); } else { const ee = 1 / n; let Z = 0, te = S.Zero(); do te = R.add(O.scale(b * Math.cos(Z)).add(Y.scale(b * Math.sin(Z)))), a.push(te), Z += ee; while (!te.equalsWithEpsilon(r, b * ee * 1.1)); a.push(r), i && a.push(e); } return new j0(a); } /** * A Curve3 object is a logical object, so not a mesh, to handle curves in the 3D geometric space. * A Curve3 is designed from a series of successive Vector3. * Tuto : https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves#curve3-object * @param points points which make up the curve */ constructor(e) { this._length = 0, this._points = e, this._length = this._computeLength(e); } /** * @returns the Curve3 stored array of successive Vector3 */ getPoints() { return this._points; } /** * @returns the computed length (float) of the curve. */ length() { return this._length; } /** * Returns a new instance of Curve3 object : var curve = curveA.continue(curveB); * This new Curve3 is built by translating and sticking the curveB at the end of the curveA. * curveA and curveB keep unchanged. * @param curve the curve to continue from this curve * @returns the newly constructed curve */ continue(e) { const t = this._points[this._points.length - 1], r = this._points.slice(), n = e.getPoints(); for (let s = 1; s < n.length; s++) r.push(n[s].subtract(n[0]).add(t)); return new j0(r); } _computeLength(e) { let t = 0; for (let r = 1; r < e.length; r++) t += e[r].subtract(e[r - 1]).length(); return t; } } class u1 { constructor() { this._easingMode = u1.EASINGMODE_EASEIN; } /** * Sets the easing mode of the current function. * @param easingMode Defines the willing mode (EASINGMODE_EASEIN, EASINGMODE_EASEOUT or EASINGMODE_EASEINOUT) */ setEasingMode(e) { const t = Math.min(Math.max(e, 0), 2); this._easingMode = t; } /** * Gets the current easing mode. * @returns the easing mode */ getEasingMode() { return this._easingMode; } /** * @internal */ // eslint-disable-next-line @typescript-eslint/no-unused-vars easeInCore(e) { throw new Error("You must implement this method"); } /** * Given an input gradient between 0 and 1, this returns the corresponding value * of the easing function. * @param gradient Defines the value between 0 and 1 we want the easing value for * @returns the corresponding value on the curve defined by the easing function */ ease(e) { switch (this._easingMode) { case u1.EASINGMODE_EASEIN: return this.easeInCore(e); case u1.EASINGMODE_EASEOUT: return 1 - this.easeInCore(1 - e); } return e >= 0.5 ? (1 - this.easeInCore((1 - e) * 2)) * 0.5 + 0.5 : this.easeInCore(e * 2) * 0.5; } } u1.EASINGMODE_EASEIN = 0; u1.EASINGMODE_EASEOUT = 1; u1.EASINGMODE_EASEINOUT = 2; class Z$ extends u1 { /** * @internal */ easeInCore(e) { return e = Math.max(0, Math.min(1, e)), 1 - Math.sqrt(1 - e * e); } } class _$ extends u1 { /** * Instantiates a back ease easing * @see https://easings.net/#easeInBack * @param amplitude Defines the amplitude of the function */ constructor(e = 1) { super(), this.amplitude = e; } /** * @internal */ easeInCore(e) { const t = Math.max(0, this.amplitude); return Math.pow(e, 3) - e * t * Math.sin(3.141592653589793 * e); } } class r6e extends u1 { /** * Instantiates a bounce easing * @see https://easings.net/#easeInBounce * @param bounces Defines the number of bounces * @param bounciness Defines the amplitude of the bounce */ constructor(e = 3, t = 2) { super(), this.bounces = e, this.bounciness = t; } /** * @internal */ easeInCore(e) { const t = Math.max(0, this.bounces); let r = this.bounciness; r <= 1 && (r = 1.001); const n = Math.pow(r, t), i = 1 - r, s = (1 - n) / i + n * 0.5, a = e * s, f = Math.log(-a * (1 - r) + 1) / Math.log(r), o = Math.floor(f), d = o + 1, v = (1 - Math.pow(r, o)) / (i * s), u = (1 - Math.pow(r, d)) / (i * s), l = (v + u) * 0.5, P = e - l, p = l - v; return -Math.pow(1 / r, t - o) / (p * p) * (P - p) * (P + p); } } class n6e extends u1 { /** * @internal */ easeInCore(e) { return e * e * e; } } class i6e extends u1 { /** * Instantiates an elastic easing function * @see https://easings.net/#easeInElastic * @param oscillations Defines the number of oscillations * @param springiness Defines the amplitude of the oscillations */ constructor(e = 3, t = 3) { super(), this.oscillations = e, this.springiness = t; } /** * @internal */ easeInCore(e) { let t; const r = Math.max(0, this.oscillations), n = Math.max(0, this.springiness); return n == 0 ? t = e : t = (Math.exp(n * e) - 1) / (Math.exp(n) - 1), t * Math.sin((6.283185307179586 * r + 1.5707963267948966) * e); } } class $$ extends u1 { /** * Instantiates an exponential easing function * @see https://easings.net/#easeInExpo * @param exponent Defines the exponent of the function */ constructor(e = 2) { super(), this.exponent = e; } /** * @internal */ easeInCore(e) { return this.exponent <= 0 ? e : (Math.exp(this.exponent * e) - 1) / (Math.exp(this.exponent) - 1); } } class s6e extends u1 { /** * Instantiates an power base easing function * @see https://easings.net/#easeInQuad * @param power Defines the power of the function */ constructor(e = 2) { super(), this.power = e; } /** * @internal */ easeInCore(e) { const t = Math.max(0, this.power); return Math.pow(e, t); } } class _N extends u1 { /** * @internal */ easeInCore(e) { return e * e; } } class a6e extends u1 { /** * @internal */ easeInCore(e) { return e * e * e * e; } } class o6e extends u1 { /** * @internal */ easeInCore(e) { return e * e * e * e * e; } } class $N extends u1 { /** * @internal */ easeInCore(e) { return 1 - Math.sin(1.5707963267948966 * (1 - e)); } } class f6e extends u1 { /** * Instantiates a bezier function * @see http://cubic-bezier.com/#.17,.67,.83,.67 * @param x1 Defines the x component of the start tangent in the bezier curve * @param y1 Defines the y component of the start tangent in the bezier curve * @param x2 Defines the x component of the end tangent in the bezier curve * @param y2 Defines the y component of the end tangent in the bezier curve */ constructor(e = 0, t = 0, r = 1, n = 1) { super(), this.x1 = e, this.y1 = t, this.x2 = r, this.y2 = n; } /** * @internal */ easeInCore(e) { return z$.Interpolate(e, this.x1, this.y1, this.x2, this.y2); } } class wO { /** * Initializes the animation event * @param frame The frame for which the event is triggered * @param action The event to perform when triggered * @param onlyOnce Specifies if the event should be triggered only once */ constructor(e, t, r) { this.frame = e, this.action = t, this.onlyOnce = r, this.isDone = !1; } /** @internal */ _clone() { return new wO(this.frame, this.action, this.onlyOnce); } } class eee { /** * Returns the string "TargetedAnimation" * @returns "TargetedAnimation" */ getClassName() { return "TargetedAnimation"; } /** * Serialize the object * @returns the JSON object representing the current entity */ serialize() { const e = {}; return e.animation = this.animation.serialize(), e.targetId = this.target.id, e; } } class w0 { /** * Makes sure that the animations are either played or stopped according to the animation group mask. * Note however that the call won't have any effect if the animation group has not been started yet. * You should call this function if you modify the mask after the animation group has been started. */ syncWithMask() { if (!this.mask) { this._numActiveAnimatables = this._targetedAnimations.length; return; } this._numActiveAnimatables = 0; for (let e = 0; e < this._animatables.length; ++e) { const t = this._animatables[e]; this.mask.retainsTarget(t.target.name) ? (this._numActiveAnimatables++, t.paused && t.restart()) : t.paused || t.pause(); } } /** * Removes all animations for the targets not retained by the animation group mask. * Use this function if you know you won't need those animations anymore and if you want to free memory. */ removeUnmaskedAnimations() { if (this.mask) { for (let e = 0; e < this._animatables.length; ++e) { const t = this._animatables[e]; this.mask.retainsTarget(t.target.name) || (t.stop(), this._animatables.splice(e, 1), --e); } for (let e = 0; e < this._targetedAnimations.length; e++) { const t = this._targetedAnimations[e]; this.mask.retainsTarget(t.target.name) || (this._targetedAnimations.splice(e, 1), --e); } } } /** * Gets the first frame */ get from() { return this._from; } /** * Gets the last frame */ get to() { return this._to; } /** * Define if the animations are started */ get isStarted() { return this._isStarted; } /** * Gets a value indicating that the current group is playing */ get isPlaying() { return this._isStarted && !this._isPaused; } /** * Gets or sets the speed ratio to use for all animations */ get speedRatio() { return this._speedRatio; } /** * Gets or sets the speed ratio to use for all animations */ set speedRatio(e) { if (this._speedRatio !== e) { this._speedRatio = e; for (let t = 0; t < this._animatables.length; t++) { const r = this._animatables[t]; r.speedRatio = this._speedRatio; } } } /** * Gets or sets if all animations should loop or not */ get loopAnimation() { return this._loopAnimation; } set loopAnimation(e) { if (this._loopAnimation !== e) { this._loopAnimation = e; for (let t = 0; t < this._animatables.length; t++) { const r = this._animatables[t]; r.loopAnimation = this._loopAnimation; } } } /** * Gets or sets if all animations should be evaluated additively */ get isAdditive() { return this._isAdditive; } set isAdditive(e) { if (this._isAdditive !== e) { this._isAdditive = e; for (let t = 0; t < this._animatables.length; t++) { const r = this._animatables[t]; r.isAdditive = this._isAdditive; } } } /** * Gets or sets the weight to apply to all animations of the group */ get weight() { return this._weight; } set weight(e) { this._weight !== e && (this._weight = e, this.setWeightForAllAnimatables(this._weight)); } /** * Gets the targeted animations for this animation group */ get targetedAnimations() { return this._targetedAnimations; } /** * returning the list of animatables controlled by this animation group. */ get animatables() { return this._animatables; } /** * Gets the list of target animations */ get children() { return this._targetedAnimations; } /** * Gets or sets the order of play of the animation group (default: 0) */ get playOrder() { return this._playOrder; } set playOrder(e) { if (this._playOrder !== e && (this._playOrder = e, this._animatables.length > 0)) { for (let t = 0; t < this._animatables.length; t++) this._animatables[t].playOrder = this._playOrder; this._scene.sortActiveAnimatables(); } } /** * Allows the animations of the animation group to blend with current running animations * Note that a null value means that each animation will use their own existing blending configuration (Animation.enableBlending) */ get enableBlending() { return this._enableBlending; } set enableBlending(e) { if (this._enableBlending !== e && (this._enableBlending = e, e !== null)) for (let t = 0; t < this._targetedAnimations.length; ++t) this._targetedAnimations[t].animation.enableBlending = e; } /** * Gets or sets the animation blending speed * Note that a null value means that each animation will use their own existing blending configuration (Animation.blendingSpeed) */ get blendingSpeed() { return this._blendingSpeed; } set blendingSpeed(e) { if (this._blendingSpeed !== e && (this._blendingSpeed = e, e !== null)) for (let t = 0; t < this._targetedAnimations.length; ++t) this._targetedAnimations[t].animation.blendingSpeed = e; } /** * Gets the length (in seconds) of the animation group * This function assumes that all animations are played at the same framePerSecond speed! * Note: you can only call this method after you've added at least one targeted animation! * @param from Starting frame range (default is AnimationGroup.from) * @param to Ending frame range (default is AnimationGroup.to) * @returns The length in seconds */ getLength(e, t) { e = e ?? this._from, t = t ?? this._to; const r = this.targetedAnimations[0].animation.framePerSecond * this._speedRatio; return (t - e) / r; } /** * Merge the array of animation groups into a new animation group * @param animationGroups List of animation groups to merge * @param disposeSource If true, animation groups will be disposed after being merged (default: true) * @param normalize If true, animation groups will be normalized before being merged, so that all animations have the same "from" and "to" frame (default: false) * @param weight Weight for the new animation group. If not provided, it will inherit the weight from the first animation group of the array * @returns The new animation group or null if no animation groups were passed */ static MergeAnimationGroups(e, t = !0, r = !1, n) { if (e.length === 0) return null; n = n ?? e[0].weight; let i = Number.MAX_VALUE, s = -Number.MAX_VALUE; if (r) for (const f of e) f.from < i && (i = f.from), f.to > s && (s = f.to); const a = new w0(e[0].name + "_merged", e[0]._scene, n); for (const f of e) { r && f.normalize(i, s); for (const o of f.targetedAnimations) a.addTargetedAnimation(o.animation, o.target); t && f.dispose(); } return a; } /** * Instantiates a new Animation Group. * This helps managing several animations at once. * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/groupAnimations * @param name Defines the name of the group * @param scene Defines the scene the group belongs to * @param weight Defines the weight to use for animations in the group (-1.0 by default, meaning "no weight") * @param playOrder Defines the order of play of the animation group (default is 0) */ constructor(e, t = null, r = -1, n = 0) { this.name = e, this._targetedAnimations = new Array(), this._animatables = new Array(), this._from = Number.MAX_VALUE, this._to = -Number.MAX_VALUE, this._speedRatio = 1, this._loopAnimation = !1, this._isAdditive = !1, this._weight = -1, this._playOrder = 0, this._enableBlending = null, this._blendingSpeed = null, this._numActiveAnimatables = 0, this._parentContainer = null, this.onAnimationEndObservable = new Oe(), this.onAnimationLoopObservable = new Oe(), this.onAnimationGroupLoopObservable = new Oe(), this.onAnimationGroupEndObservable = new Oe(), this.onAnimationGroupPauseObservable = new Oe(), this.onAnimationGroupPlayObservable = new Oe(), this.metadata = null, this._animationLoopFlags = [], this._scene = t || gr.LastCreatedScene, this._weight = r, this._playOrder = n, this.uniqueId = this._scene.getUniqueId(), this._scene.addAnimationGroup(this); } /** * Add an animation (with its target) in the group * @param animation defines the animation we want to add * @param target defines the target of the animation * @returns the TargetedAnimation object */ addTargetedAnimation(e, t) { const r = new eee(); r.animation = e, r.target = t; const n = e.getKeys(); return this._from > n[0].frame && (this._from = n[0].frame), this._to < n[n.length - 1].frame && (this._to = n[n.length - 1].frame), this._enableBlending !== null && (e.enableBlending = this._enableBlending), this._blendingSpeed !== null && (e.blendingSpeed = this._blendingSpeed), this._targetedAnimations.push(r), r; } /** * Remove an animation from the group * @param animation defines the animation we want to remove */ removeTargetedAnimation(e) { for (let t = this._targetedAnimations.length - 1; t > -1; t--) this._targetedAnimations[t].animation === e && this._targetedAnimations.splice(t, 1); } /** * This function will normalize every animation in the group to make sure they all go from beginFrame to endFrame * It can add constant keys at begin or end * @param beginFrame defines the new begin frame for all animations or the smallest begin frame of all animations if null (defaults to null) * @param endFrame defines the new end frame for all animations or the largest end frame of all animations if null (defaults to null) * @returns the animation group */ normalize(e = null, t = null) { e == null && (e = this._from), t == null && (t = this._to); for (let r = 0; r < this._targetedAnimations.length; r++) { const i = this._targetedAnimations[r].animation.getKeys(), s = i[0], a = i[i.length - 1]; if (s.frame > e) { const f = { frame: e, value: s.value, inTangent: s.inTangent, outTangent: s.outTangent, interpolation: s.interpolation }; i.splice(0, 0, f); } if (a.frame < t) { const f = { frame: t, value: a.value, inTangent: a.inTangent, outTangent: a.outTangent, interpolation: a.interpolation }; i.push(f); } } return this._from = e, this._to = t, this; } _processLoop(e, t, r) { e.onAnimationLoop = () => { this.onAnimationLoopObservable.notifyObservers(t), !this._animationLoopFlags[r] && (this._animationLoopFlags[r] = !0, this._animationLoopCount++, this._animationLoopCount === this._numActiveAnimatables && (this.onAnimationGroupLoopObservable.notifyObservers(this), this._animationLoopCount = 0, this._animationLoopFlags.length = 0)); }; } /** * Start all animations on given targets * @param loop defines if animations must loop * @param speedRatio defines the ratio to apply to animation speed (1 by default) * @param from defines the from key (optional) * @param to defines the to key (optional) * @param isAdditive defines the additive state for the resulting animatables (optional) * @returns the current animation group */ start(e = !1, t = 1, r, n, i) { if (this._isStarted || this._targetedAnimations.length === 0) return this; this._loopAnimation = e, this._animationLoopCount = 0, this._animationLoopFlags.length = 0; for (let s = 0; s < this._targetedAnimations.length; s++) { const a = this._targetedAnimations[s], f = this._scene.beginDirectAnimation(a.target, [a.animation], r !== void 0 ? r : this._from, n !== void 0 ? n : this._to, e, t, void 0, void 0, i !== void 0 ? i : this._isAdditive); f.weight = this._weight, f.playOrder = this._playOrder, f.onAnimationEnd = () => { this.onAnimationEndObservable.notifyObservers(a), this._checkAnimationGroupEnded(f); }, this._processLoop(f, a, s), this._animatables.push(f); } return this.syncWithMask(), this._scene.sortActiveAnimatables(), this._speedRatio = t, this._isStarted = !0, this._isPaused = !1, this.onAnimationGroupPlayObservable.notifyObservers(this), this; } /** * Pause all animations * @returns the animation group */ pause() { if (!this._isStarted) return this; this._isPaused = !0; for (let e = 0; e < this._animatables.length; e++) this._animatables[e].pause(); return this.onAnimationGroupPauseObservable.notifyObservers(this), this; } /** * Play all animations to initial state * This function will start() the animations if they were not started or will restart() them if they were paused * @param loop defines if animations must loop * @returns the animation group */ play(e) { return this.isStarted && this._animatables.length === this._targetedAnimations.length ? (e !== void 0 && (this.loopAnimation = e), this.restart()) : (this.stop(), this.start(e, this._speedRatio)), this._isPaused = !1, this; } /** * Reset all animations to initial state * @returns the animation group */ reset() { if (!this._isStarted) return this.play(), this.goToFrame(0), this.stop(), this; for (let e = 0; e < this._animatables.length; e++) this._animatables[e].reset(); return this; } /** * Restart animations from key 0 * @returns the animation group */ restart() { if (!this._isStarted) return this; for (let e = 0; e < this._animatables.length; e++) this._animatables[e].restart(); return this.onAnimationGroupPlayObservable.notifyObservers(this), this; } /** * Stop all animations * @returns the animation group */ stop() { if (!this._isStarted) return this; const e = this._animatables.slice(); for (let r = 0; r < e.length; r++) e[r].stop(void 0, void 0, !0); let t = 0; for (let r = 0; r < this._scene._activeAnimatables.length; r++) { const n = this._scene._activeAnimatables[r]; n._runtimeAnimations.length > 0 && (this._scene._activeAnimatables[t++] = n); } return this._scene._activeAnimatables.length = t, this._isStarted = !1, this; } /** * Set animation weight for all animatables * * @since 6.12.4 * You can pass the weight to the AnimationGroup constructor, or use the weight property to set it after the group has been created, * making it easier to define the overall animation weight than calling setWeightForAllAnimatables() after the animation group has been started * @param weight defines the weight to use * @returns the animationGroup * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#animation-weights */ setWeightForAllAnimatables(e) { for (let t = 0; t < this._animatables.length; t++) { const r = this._animatables[t]; r.weight = e; } return this; } /** * Synchronize and normalize all animatables with a source animatable * @param root defines the root animatable to synchronize with (null to stop synchronizing) * @returns the animationGroup * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#animation-weights */ syncAllAnimationsWith(e) { for (let t = 0; t < this._animatables.length; t++) this._animatables[t].syncWith(e); return this; } /** * Goes to a specific frame in this animation group * @param frame the frame number to go to * @returns the animationGroup */ goToFrame(e) { if (!this._isStarted) return this; for (let t = 0; t < this._animatables.length; t++) this._animatables[t].goToFrame(e); return this; } /** * Dispose all associated resources */ dispose() { this._targetedAnimations.length = 0, this._animatables.length = 0; const e = this._scene.animationGroups.indexOf(this); if (e > -1 && this._scene.animationGroups.splice(e, 1), this._parentContainer) { const t = this._parentContainer.animationGroups.indexOf(this); t > -1 && this._parentContainer.animationGroups.splice(t, 1), this._parentContainer = null; } this.onAnimationEndObservable.clear(), this.onAnimationGroupEndObservable.clear(), this.onAnimationGroupPauseObservable.clear(), this.onAnimationGroupPlayObservable.clear(), this.onAnimationLoopObservable.clear(), this.onAnimationGroupLoopObservable.clear(); } _checkAnimationGroupEnded(e) { const t = this._animatables.indexOf(e); t > -1 && this._animatables.splice(t, 1), this._animatables.length === 0 && (this._isStarted = !1, this.onAnimationGroupEndObservable.notifyObservers(this)); } /** * Clone the current animation group and returns a copy * @param newName defines the name of the new group * @param targetConverter defines an optional function used to convert current animation targets to new ones * @param cloneAnimations defines if the animations should be cloned or referenced * @returns the new animation group */ clone(e, t, r = !1) { const n = new w0(e || this.name, this._scene, this._weight, this._playOrder); n._from = this.from, n._to = this.to, n._speedRatio = this.speedRatio, n._loopAnimation = this.loopAnimation, n._isAdditive = this.isAdditive, n._enableBlending = this.enableBlending, n._blendingSpeed = this.blendingSpeed, n.metadata = this.metadata, n.mask = this.mask; for (const i of this._targetedAnimations) n.addTargetedAnimation(r ? i.animation.clone() : i.animation, t ? t(i.target) : i.target); return n; } /** * Serializes the animationGroup to an object * @returns Serialized object */ serialize() { const e = {}; e.name = this.name, e.from = this.from, e.to = this.to, e.speedRatio = this.speedRatio, e.loopAnimation = this.loopAnimation, e.isAdditive = this.isAdditive, e.weight = this.weight, e.playOrder = this.playOrder, e.enableBlending = this.enableBlending, e.blendingSpeed = this.blendingSpeed, e.targetedAnimations = []; for (let t = 0; t < this.targetedAnimations.length; t++) { const r = this.targetedAnimations[t]; e.targetedAnimations[t] = r.serialize(); } return Zi && Zi.HasTags(this) && (e.tags = Zi.GetTags(this)), this.metadata && (e.metadata = this.metadata), e; } // Statics /** * Returns a new AnimationGroup object parsed from the source provided. * @param parsedAnimationGroup defines the source * @param scene defines the scene that will receive the animationGroup * @returns a new AnimationGroup */ static Parse(e, t) { const r = new w0(e.name, t, e.weight, e.playOrder); for (let n = 0; n < e.targetedAnimations.length; n++) { const i = e.targetedAnimations[n], s = st.Parse(i.animation), a = i.targetId; if (i.animation.property === "influence") { const f = t.getMorphTargetById(a); f && r.addTargetedAnimation(s, f); } else { const f = t.getNodeById(a); f != null && r.addTargetedAnimation(s, f); } } return Zi && Zi.AddTagsTo(r, e.tags), e.from !== null && e.to !== null && r.normalize(e.from, e.to), e.speedRatio !== void 0 && (r._speedRatio = e.speedRatio), e.loopAnimation !== void 0 && (r._loopAnimation = e.loopAnimation), e.isAdditive !== void 0 && (r._isAdditive = e.isAdditive), e.weight !== void 0 && (r._weight = e.weight), e.playOrder !== void 0 && (r._playOrder = e.playOrder), e.enableBlending !== void 0 && (r._enableBlending = e.enableBlending), e.blendingSpeed !== void 0 && (r._blendingSpeed = e.blendingSpeed), e.metadata !== void 0 && (r.metadata = e.metadata), r; } /** @internal */ static MakeAnimationAdditive(e, t, r, n = !1, i) { let s; typeof t == "object" ? s = t : s = { referenceFrame: t, range: r, cloneOriginalAnimationGroup: n, clonedAnimationName: i }; let a = e; s.cloneOriginalAnimationGroup && (a = e.clone(s.clonedAnimationGroupName || a.name)); const f = a.targetedAnimations; for (let o = 0; o < f.length; o++) { const d = f[o]; d.animation = st.MakeAnimationAdditive(d.animation, s); } if (a.isAdditive = !0, s.clipKeys) { let o = Number.MAX_VALUE, d = -Number.MAX_VALUE; const v = a.targetedAnimations; for (let u = 0; u < v.length; u++) { const p = v[u].animation.getKeys(); o > p[0].frame && (o = p[0].frame), d < p[p.length - 1].frame && (d = p[p.length - 1].frame); } a._from = o, a._to = d; } return a; } /** * Creates a new animation, keeping only the keys that are inside a given key range * @param sourceAnimationGroup defines the animation group on which to operate * @param fromKey defines the lower bound of the range * @param toKey defines the upper bound of the range * @param name defines the name of the new animation group. If not provided, use the same name as animationGroup * @param dontCloneAnimations defines whether or not the animations should be cloned before clipping the keys. Default is false, so animations will be cloned * @returns a new animation group stripped from all the keys outside the given range */ static ClipKeys(e, t, r, n, i) { const s = e.clone(n || e.name); return w0.ClipKeysInPlace(s, t, r, i); } /** * Updates an existing animation, keeping only the keys that are inside a given key range * @param animationGroup defines the animation group on which to operate * @param fromKey defines the lower bound of the range * @param toKey defines the upper bound of the range * @param dontCloneAnimations defines whether or not the animations should be cloned before clipping the keys. Default is false, so animations will be cloned * @returns the animationGroup stripped from all the keys outside the given range */ static ClipKeysInPlace(e, t, r, n) { return w0.ClipInPlace(e, t, r, n, !1); } /** * Creates a new animation, keeping only the frames that are inside a given frame range * @param sourceAnimationGroup defines the animation group on which to operate * @param fromFrame defines the lower bound of the range * @param toFrame defines the upper bound of the range * @param name defines the name of the new animation group. If not provided, use the same name as animationGroup * @param dontCloneAnimations defines whether or not the animations should be cloned before clipping the frames. Default is false, so animations will be cloned * @returns a new animation group stripped from all the frames outside the given range */ static ClipFrames(e, t, r, n, i) { const s = e.clone(n || e.name); return w0.ClipFramesInPlace(s, t, r, i); } /** * Updates an existing animation, keeping only the frames that are inside a given frame range * @param animationGroup defines the animation group on which to operate * @param fromFrame defines the lower bound of the range * @param toFrame defines the upper bound of the range * @param dontCloneAnimations defines whether or not the animations should be cloned before clipping the frames. Default is false, so animations will be cloned * @returns the animationGroup stripped from all the frames outside the given range */ static ClipFramesInPlace(e, t, r, n) { return w0.ClipInPlace(e, t, r, n, !0); } /** * Updates an existing animation, keeping only the keys that are inside a given key or frame range * @param animationGroup defines the animation group on which to operate * @param start defines the lower bound of the range * @param end defines the upper bound of the range * @param dontCloneAnimations defines whether or not the animations should be cloned before clipping the keys. Default is false, so animations will be cloned * @param useFrame defines if the range is defined by frame numbers or key indices (default is false which means use key indices) * @returns the animationGroup stripped from all the keys outside the given range */ static ClipInPlace(e, t, r, n, i = !1) { let s = Number.MAX_VALUE, a = -Number.MAX_VALUE; const f = e.targetedAnimations; for (let o = 0; o < f.length; o++) { const d = f[o], v = n ? d.animation : d.animation.clone(); i && (v.createKeyForFrame(t), v.createKeyForFrame(r)); const u = v.getKeys(), l = []; let P = Number.MAX_VALUE; for (let p = 0; p < u.length; p++) { const c = u[p]; if (!i && p >= t && p <= r || i && c.frame >= t && c.frame <= r) { const H = { frame: c.frame, value: c.value.clone ? c.value.clone() : c.value, inTangent: c.inTangent, outTangent: c.outTangent, interpolation: c.interpolation, lockedTangent: c.lockedTangent }; P === Number.MAX_VALUE && (P = H.frame), H.frame -= P, l.push(H); } } if (l.length === 0) { f.splice(o, 1), o--; continue; } s > l[0].frame && (s = l[0].frame), a < l[l.length - 1].frame && (a = l[l.length - 1].frame), v.setKeys(l, !0), d.animation = v; } return e._from = s, e._to = a, e; } /** * Returns the string "AnimationGroup" * @returns "AnimationGroup" */ getClassName() { return "AnimationGroup"; } /** * Creates a detailed string about the object * @param fullDetails defines if the output string will support multiple levels of logging within scene loading * @returns a string representing the object */ toString(e) { let t = "Name: " + this.name; return t += ", type: " + this.getClassName(), e && (t += ", from: " + this._from, t += ", to: " + this._to, t += ", isStarted: " + this._isStarted, t += ", speedRatio: " + this._speedRatio, t += ", targetedAnimations length: " + this._targetedAnimations.length, t += ", animatables length: " + this._animatables), t; } } class A6e { /** * Initializes the path cursor * @param _path The path to track */ constructor(e) { this._path = e, this._onchange = new Array(), this.value = 0, this.animations = []; } /** * Gets the cursor point on the path * @returns A point on the path cursor at the cursor location */ getPoint() { const e = this._path.getPointAtLengthPosition(this.value); return new S(e.x, 0, e.y); } /** * Moves the cursor ahead by the step amount * @param step The amount to move the cursor forward * @returns This path cursor */ moveAhead(e = 2e-3) { return this.move(e), this; } /** * Moves the cursor behind by the step amount * @param step The amount to move the cursor back * @returns This path cursor */ moveBack(e = 2e-3) { return this.move(-e), this; } /** * Moves the cursor by the step amount * If the step amount is greater than one, an exception is thrown * @param step The amount to move the cursor * @returns This path cursor */ move(e) { if (Math.abs(e) > 1) throw "step size should be less than 1."; return this.value += e, this._ensureLimits(), this._raiseOnChange(), this; } /** * Ensures that the value is limited between zero and one * @returns This path cursor */ _ensureLimits() { for (; this.value > 1; ) this.value -= 1; for (; this.value < 0; ) this.value += 1; return this; } /** * Runs onchange callbacks on change (used by the animation engine) * @returns This path cursor */ _raiseOnChange() { return this._onchange.forEach((e) => e(this)), this; } /** * Executes a function on change * @param f A path cursor onchange callback * @returns This path cursor */ onchange(e) { return this._onchange.push(e), this; } } var EI; (function(A) { A[A.Include = 0] = "Include", A[A.Exclude = 1] = "Exclude"; })(EI || (EI = {})); class d6e { /** * Creates a new mask * @param names The list of target names to add to the mask (optional) * @param mode Defines the mode for the mask (default: AnimationGroupMaskMode.Include) */ constructor(e, t = EI.Include) { this.mode = t, this._targetNames = /* @__PURE__ */ new Set(), e && this.addTargetName(e); } /** * Adds one or several target names to the mask * @param name The name(s) to add to the mask */ addTargetName(e) { if (Array.isArray(e)) { for (const t of e) this._targetNames.add(t); return; } this._targetNames.add(e); } /** * Removes one or several target names from the mask * @param name The name(s) to remove from the mask */ removeTargetName(e) { if (Array.isArray(e)) { for (const t of e) this._targetNames.delete(t); return; } this._targetNames.delete(e); } /** * Checks if the mask includes a target name. * This method is intended to know if a given target name is included in the mask, not if the name is actually retained by the mask (see retainsTarget() instead). * @param name The name to check with the mask * @returns True if the mask includes the name, false otherwise */ hasTarget(e) { return this._targetNames.has(e); } /** * Checks if the mask retains a target name. * Note that in the "Exclude" mode, this will return false if the mask includes the name, and true otherwise! * This method is intended to know if a given target name is retained by the mask, not if the name is in the list of target names. * @param name The name to check with the mask * @returns True if the mask retains the name, false otherwise */ retainsTarget(e) { return this._targetNames.has(e) === (this.mode === EI.Include); } } function FI(A, e, t) { try { const r = A.next(); r.done ? e(r) : r.value ? r.value.then(() => { r.value = void 0, e(r); }, t) : e(r); } catch (r) { t(r); } } function tee(A = 25) { let e; return (t, r, n) => { const i = performance.now(); e === void 0 || i - e > A ? (e = i, setTimeout(() => { FI(t, r, n); }, 0)) : FI(t, r, n); }; } function eQ(A, e, t, r, n) { const i = () => { let s; const a = (f) => { f.done ? t(f.value) : s === void 0 ? s = !0 : i(); }; do s = void 0, !n || !n.aborted ? e(A, a, r) : r(new Error("Aborted")), s === void 0 && (s = !1); while (s); }; i(); } function mO(A, e) { let t; return eQ(A, FI, (r) => t = r, (r) => { throw r; }, e), t; } function BO(A, e, t) { return new Promise((r, n) => { eQ(A, e, r, n, t); }); } function ree(A, e) { return (...t) => mO(A(...t), e); } function v6e(A, e, t) { return (...r) => BO(A(...r), e, t); } class WA { /** * Creates a Viewport object located at (x, y) and sized (width, height) * @param x defines viewport left coordinate * @param y defines viewport top coordinate * @param width defines the viewport width * @param height defines the viewport height */ constructor(e, t, r, n) { this.x = e, this.y = t, this.width = r, this.height = n; } /** * Creates a new viewport using absolute sizing (from 0-> width, 0-> height instead of 0->1) * @param renderWidth defines the rendering width * @param renderHeight defines the rendering height * @returns a new Viewport */ toGlobal(e, t) { return new WA(this.x * e, this.y * t, this.width * e, this.height * t); } /** * Stores absolute viewport value into a target viewport (from 0-> width, 0-> height instead of 0->1) * @param renderWidth defines the rendering width * @param renderHeight defines the rendering height * @param ref defines the target viewport * @returns the current viewport */ toGlobalToRef(e, t, r) { return r.x = this.x * e, r.y = this.y * t, r.width = this.width * e, r.height = this.height * t, this; } /** * Returns a new Viewport copied from the current one * @returns a new Viewport */ clone() { return new WA(this.x, this.y, this.width, this.height); } } class Tr extends Cs { /** * Define the current local position of the camera in the scene */ get position() { return this._position; } set position(e) { this._position = e; } /** * The vector the camera should consider as up. * (default is Vector3(0, 1, 0) aka Vector3.Up()) */ set upVector(e) { this._upVector = e; } get upVector() { return this._upVector; } /** * The screen area in scene units squared */ get screenArea() { var e, t, r, n; let i = 0, s = 0; if (this.mode === Tr.PERSPECTIVE_CAMERA) this.fovMode === Tr.FOVMODE_VERTICAL_FIXED ? (s = this.minZ * 2 * Math.tan(this.fov / 2), i = this.getEngine().getAspectRatio(this) * s) : (i = this.minZ * 2 * Math.tan(this.fov / 2), s = i / this.getEngine().getAspectRatio(this)); else { const a = this.getEngine().getRenderWidth() / 2, f = this.getEngine().getRenderHeight() / 2; i = ((e = this.orthoRight) !== null && e !== void 0 ? e : a) - ((t = this.orthoLeft) !== null && t !== void 0 ? t : -a), s = ((r = this.orthoTop) !== null && r !== void 0 ? r : f) - ((n = this.orthoBottom) !== null && n !== void 0 ? n : -f); } return i * s; } set orthoLeft(e) { this._orthoLeft = e; for (const t of this._rigCameras) t.orthoLeft = e; } get orthoLeft() { return this._orthoLeft; } set orthoRight(e) { this._orthoRight = e; for (const t of this._rigCameras) t.orthoRight = e; } get orthoRight() { return this._orthoRight; } set orthoBottom(e) { this._orthoBottom = e; for (const t of this._rigCameras) t.orthoBottom = e; } get orthoBottom() { return this._orthoBottom; } set orthoTop(e) { this._orthoTop = e; for (const t of this._rigCameras) t.orthoTop = e; } get orthoTop() { return this._orthoTop; } set mode(e) { this._mode = e; for (const t of this._rigCameras) t.mode = e; } get mode() { return this._mode; } /** * Instantiates a new camera object. * This should not be used directly but through the inherited cameras: ArcRotate, Free... * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras * @param name Defines the name of the camera in the scene * @param position Defines the position of the camera * @param scene Defines the scene the camera belongs too * @param setActiveOnSceneIfNoneActive Defines if the camera should be set as active after creation if no other camera have been defined in the scene */ constructor(e, t, r, n = !0) { super(e, r), this._position = S.Zero(), this._upVector = S.Up(), this.oblique = null, this._orthoLeft = null, this._orthoRight = null, this._orthoBottom = null, this._orthoTop = null, this.fov = 0.8, this.projectionPlaneTilt = 0, this.minZ = 1, this.maxZ = 1e4, this.inertia = 0.9, this._mode = Tr.PERSPECTIVE_CAMERA, this.isIntermediate = !1, this.viewport = new WA(0, 0, 1, 1), this.layerMask = 268435455, this.fovMode = Tr.FOVMODE_VERTICAL_FIXED, this.cameraRigMode = Tr.RIG_MODE_NONE, this.customRenderTargets = [], this.outputRenderTarget = null, this.onViewMatrixChangedObservable = new Oe(), this.onProjectionMatrixChangedObservable = new Oe(), this.onAfterCheckInputsObservable = new Oe(), this.onRestoreStateObservable = new Oe(), this.isRigCamera = !1, this._rigCameras = new Array(), this._skipRendering = !1, this._projectionMatrix = new he(), this._postProcesses = new Array(), this._activeMeshes = new qf(256), this._globalPosition = S.Zero(), this._computedViewMatrix = he.Identity(), this._doNotComputeProjectionMatrix = !1, this._transformMatrix = he.Zero(), this._refreshFrustumPlanes = !0, this._absoluteRotation = Ze.Identity(), this._isCamera = !0, this._isLeftCamera = !1, this._isRightCamera = !1, this.getScene().addCamera(this), n && !this.getScene().activeCamera && (this.getScene().activeCamera = this), this.position = t, this.renderPassId = this.getScene().getEngine().createRenderPassId(`Camera ${e}`); } /** * Store current camera state (fov, position, etc..) * @returns the camera */ storeState() { return this._stateStored = !0, this._storedFov = this.fov, this; } /** * Restores the camera state values if it has been stored. You must call storeState() first */ _restoreStateValues() { return this._stateStored ? (this.fov = this._storedFov, !0) : !1; } /** * Restored camera state. You must call storeState() first. * @returns true if restored and false otherwise */ restoreState() { return this._restoreStateValues() ? (this.onRestoreStateObservable.notifyObservers(this), !0) : !1; } /** * Gets the class name of the camera. * @returns the class name */ getClassName() { return "Camera"; } /** * Gets a string representation of the camera useful for debug purpose. * @param fullDetails Defines that a more verbose level of logging is required * @returns the string representation */ toString(e) { let t = "Name: " + this.name; if (t += ", type: " + this.getClassName(), this.animations) for (let r = 0; r < this.animations.length; r++) t += ", animation[0]: " + this.animations[r].toString(e); return t; } /** * Automatically tilts the projection plane, using `projectionPlaneTilt`, to correct the perspective effect on vertical lines. */ applyVerticalCorrection() { const e = this.absoluteRotation.toEulerAngles(); this.projectionPlaneTilt = this._scene.useRightHandedSystem ? -e.x : e.x; } /** * Gets the current world space position of the camera. */ get globalPosition() { return this._globalPosition; } /** * Gets the list of active meshes this frame (meshes no culled or excluded by lod s in the frame) * @returns the active meshe list */ getActiveMeshes() { return this._activeMeshes; } /** * Check whether a mesh is part of the current active mesh list of the camera * @param mesh Defines the mesh to check * @returns true if active, false otherwise */ isActiveMesh(e) { return this._activeMeshes.indexOf(e) !== -1; } /** * Is this camera ready to be used/rendered * @param completeCheck defines if a complete check (including post processes) has to be done (false by default) * @returns true if the camera is ready */ isReady(e = !1) { if (e) { for (const t of this._postProcesses) if (t && !t.isReady()) return !1; } return super.isReady(e); } /** @internal */ _initCache() { super._initCache(), this._cache.position = new S(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE), this._cache.upVector = new S(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE), this._cache.mode = void 0, this._cache.minZ = void 0, this._cache.maxZ = void 0, this._cache.fov = void 0, this._cache.fovMode = void 0, this._cache.aspectRatio = void 0, this._cache.orthoLeft = void 0, this._cache.orthoRight = void 0, this._cache.orthoBottom = void 0, this._cache.orthoTop = void 0, this._cache.obliqueAngle = void 0, this._cache.obliqueLength = void 0, this._cache.obliqueOffset = void 0, this._cache.renderWidth = void 0, this._cache.renderHeight = void 0; } /** * @internal */ _updateCache(e) { e || super._updateCache(), this._cache.position.copyFrom(this.position), this._cache.upVector.copyFrom(this.upVector); } /** @internal */ _isSynchronized() { return this._isSynchronizedViewMatrix() && this._isSynchronizedProjectionMatrix(); } /** @internal */ _isSynchronizedViewMatrix() { return super._isSynchronized() ? this._cache.position.equals(this.position) && this._cache.upVector.equals(this.upVector) && this.isSynchronizedWithParent() : !1; } /** @internal */ _isSynchronizedProjectionMatrix() { let e = this._cache.mode === this.mode && this._cache.minZ === this.minZ && this._cache.maxZ === this.maxZ; if (!e) return !1; const t = this.getEngine(); return this.mode === Tr.PERSPECTIVE_CAMERA ? e = this._cache.fov === this.fov && this._cache.fovMode === this.fovMode && this._cache.aspectRatio === t.getAspectRatio(this) && this._cache.projectionPlaneTilt === this.projectionPlaneTilt : (e = this._cache.orthoLeft === this.orthoLeft && this._cache.orthoRight === this.orthoRight && this._cache.orthoBottom === this.orthoBottom && this._cache.orthoTop === this.orthoTop && this._cache.renderWidth === t.getRenderWidth() && this._cache.renderHeight === t.getRenderHeight(), this.oblique && (e = e && this._cache.obliqueAngle === this.oblique.angle && this._cache.obliqueLength === this.oblique.length && this._cache.obliqueOffset === this.oblique.offset)), e; } /** * Attach the input controls to a specific dom element to get the input from. * This function is here because typescript removes the typing of the last function. * @param _ignored defines an ignored parameter kept for backward compatibility. * @param _noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachControl(e, t) { } /** * Detach the current controls from the specified dom element. * This function is here because typescript removes the typing of the last function. * @param _ignored defines an ignored parameter kept for backward compatibility. */ detachControl(e) { } /** * Update the camera state according to the different inputs gathered during the frame. */ update() { this._checkInputs(), this.cameraRigMode !== Tr.RIG_MODE_NONE && this._updateRigCameras(), this.getViewMatrix(), this.getProjectionMatrix(); } /** @internal */ _checkInputs() { this.onAfterCheckInputsObservable.notifyObservers(this); } /** @internal */ get rigCameras() { return this._rigCameras; } /** * Gets the post process used by the rig cameras */ get rigPostProcess() { return this._rigPostProcess; } /** * Internal, gets the first post process. * @returns the first post process to be run on this camera. */ _getFirstPostProcess() { for (let e = 0; e < this._postProcesses.length; e++) if (this._postProcesses[e] !== null) return this._postProcesses[e]; return null; } _cascadePostProcessesToRigCams() { const e = this._getFirstPostProcess(); e && e.markTextureDirty(); for (let t = 0, r = this._rigCameras.length; t < r; t++) { const n = this._rigCameras[t], i = n._rigPostProcess; i ? (i.getEffectName() === "pass" && (n.isIntermediate = this._postProcesses.length === 0), n._postProcesses = this._postProcesses.slice(0).concat(i), i.markTextureDirty()) : n._postProcesses = this._postProcesses.slice(0); } } /** * Attach a post process to the camera. * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/usePostProcesses#attach-postprocess * @param postProcess The post process to attach to the camera * @param insertAt The position of the post process in case several of them are in use in the scene * @returns the position the post process has been inserted at */ attachPostProcess(e, t = null) { return !e.isReusable() && this._postProcesses.indexOf(e) > -1 ? (Se.Error("You're trying to reuse a post process not defined as reusable."), 0) : (t == null || t < 0 ? this._postProcesses.push(e) : this._postProcesses[t] === null ? this._postProcesses[t] = e : this._postProcesses.splice(t, 0, e), this._cascadePostProcessesToRigCams(), this._scene.prePassRenderer && this._scene.prePassRenderer.markAsDirty(), this._postProcesses.indexOf(e)); } /** * Detach a post process to the camera. * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/usePostProcesses#attach-postprocess * @param postProcess The post process to detach from the camera */ detachPostProcess(e) { const t = this._postProcesses.indexOf(e); t !== -1 && (this._postProcesses[t] = null), this._scene.prePassRenderer && this._scene.prePassRenderer.markAsDirty(), this._cascadePostProcessesToRigCams(); } /** * Gets the current world matrix of the camera */ getWorldMatrix() { return this._isSynchronizedViewMatrix() ? this._worldMatrix : (this.getViewMatrix(), this._worldMatrix); } /** @internal */ _getViewMatrix() { return he.Identity(); } /** * Gets the current view matrix of the camera. * @param force forces the camera to recompute the matrix without looking at the cached state * @returns the view matrix */ getViewMatrix(e) { return !e && this._isSynchronizedViewMatrix() ? this._computedViewMatrix : (this.updateCache(), this._computedViewMatrix = this._getViewMatrix(), this._currentRenderId = this.getScene().getRenderId(), this._childUpdateId++, this._refreshFrustumPlanes = !0, this._cameraRigParams && this._cameraRigParams.vrPreViewMatrix && this._computedViewMatrix.multiplyToRef(this._cameraRigParams.vrPreViewMatrix, this._computedViewMatrix), this.parent && this.parent.onViewMatrixChangedObservable && this.parent.onViewMatrixChangedObservable.notifyObservers(this.parent), this.onViewMatrixChangedObservable.notifyObservers(this), this._computedViewMatrix.invertToRef(this._worldMatrix), this._computedViewMatrix); } /** * Freeze the projection matrix. * It will prevent the cache check of the camera projection compute and can speed up perf * if no parameter of the camera are meant to change * @param projection Defines manually a projection if necessary */ freezeProjectionMatrix(e) { this._doNotComputeProjectionMatrix = !0, e !== void 0 && (this._projectionMatrix = e); } /** * Unfreeze the projection matrix if it has previously been freezed by freezeProjectionMatrix. */ unfreezeProjectionMatrix() { this._doNotComputeProjectionMatrix = !1; } /** * Gets the current projection matrix of the camera. * @param force forces the camera to recompute the matrix without looking at the cached state * @returns the projection matrix */ getProjectionMatrix(e) { var t, r, n, i, s, a, f, o, d, v, u, l, P, p, c, H, T, q, b; if (this._doNotComputeProjectionMatrix || !e && this._isSynchronizedProjectionMatrix()) return this._projectionMatrix; this._cache.mode = this.mode, this._cache.minZ = this.minZ, this._cache.maxZ = this.maxZ, this._refreshFrustumPlanes = !0; const j = this.getEngine(), w = this.getScene(), m = j.useReverseDepthBuffer; if (this.mode === Tr.PERSPECTIVE_CAMERA) { this._cache.fov = this.fov, this._cache.fovMode = this.fovMode, this._cache.aspectRatio = j.getAspectRatio(this), this._cache.projectionPlaneTilt = this.projectionPlaneTilt, this.minZ <= 0 && (this.minZ = 0.1); let I; w.useRightHandedSystem ? I = he.PerspectiveFovRHToRef : I = he.PerspectiveFovLHToRef, I(this.fov, j.getAspectRatio(this), m ? this.maxZ : this.minZ, m ? this.minZ : this.maxZ, this._projectionMatrix, this.fovMode === Tr.FOVMODE_VERTICAL_FIXED, j.isNDCHalfZRange, this.projectionPlaneTilt, m); } else { const I = j.getRenderWidth() / 2, N = j.getRenderHeight() / 2; w.useRightHandedSystem ? this.oblique ? he.ObliqueOffCenterRHToRef((t = this.orthoLeft) !== null && t !== void 0 ? t : -I, (r = this.orthoRight) !== null && r !== void 0 ? r : I, (n = this.orthoBottom) !== null && n !== void 0 ? n : -N, (i = this.orthoTop) !== null && i !== void 0 ? i : N, m ? this.maxZ : this.minZ, m ? this.minZ : this.maxZ, this.oblique.length, this.oblique.angle, this._computeObliqueDistance(this.oblique.offset), this._projectionMatrix, j.isNDCHalfZRange) : he.OrthoOffCenterRHToRef((s = this.orthoLeft) !== null && s !== void 0 ? s : -I, (a = this.orthoRight) !== null && a !== void 0 ? a : I, (f = this.orthoBottom) !== null && f !== void 0 ? f : -N, (o = this.orthoTop) !== null && o !== void 0 ? o : N, m ? this.maxZ : this.minZ, m ? this.minZ : this.maxZ, this._projectionMatrix, j.isNDCHalfZRange) : this.oblique ? he.ObliqueOffCenterLHToRef((d = this.orthoLeft) !== null && d !== void 0 ? d : -I, (v = this.orthoRight) !== null && v !== void 0 ? v : I, (u = this.orthoBottom) !== null && u !== void 0 ? u : -N, (l = this.orthoTop) !== null && l !== void 0 ? l : N, m ? this.maxZ : this.minZ, m ? this.minZ : this.maxZ, this.oblique.length, this.oblique.angle, this._computeObliqueDistance(this.oblique.offset), this._projectionMatrix, j.isNDCHalfZRange) : he.OrthoOffCenterLHToRef((P = this.orthoLeft) !== null && P !== void 0 ? P : -I, (p = this.orthoRight) !== null && p !== void 0 ? p : I, (c = this.orthoBottom) !== null && c !== void 0 ? c : -N, (H = this.orthoTop) !== null && H !== void 0 ? H : N, m ? this.maxZ : this.minZ, m ? this.minZ : this.maxZ, this._projectionMatrix, j.isNDCHalfZRange), this._cache.orthoLeft = this.orthoLeft, this._cache.orthoRight = this.orthoRight, this._cache.orthoBottom = this.orthoBottom, this._cache.orthoTop = this.orthoTop, this._cache.obliqueAngle = (T = this.oblique) === null || T === void 0 ? void 0 : T.angle, this._cache.obliqueLength = (q = this.oblique) === null || q === void 0 ? void 0 : q.length, this._cache.obliqueOffset = (b = this.oblique) === null || b === void 0 ? void 0 : b.offset, this._cache.renderWidth = j.getRenderWidth(), this._cache.renderHeight = j.getRenderHeight(); } return this.onProjectionMatrixChangedObservable.notifyObservers(this), this._projectionMatrix; } /** * Gets the transformation matrix (ie. the multiplication of view by projection matrices) * @returns a Matrix */ getTransformationMatrix() { return this._computedViewMatrix.multiplyToRef(this._projectionMatrix, this._transformMatrix), this._transformMatrix; } _computeObliqueDistance(e) { const t = this, r = this; return (t.radius || (r.target ? S.Distance(this.position, r.target) : this.position.length())) + e; } _updateFrustumPlanes() { this._refreshFrustumPlanes && (this.getTransformationMatrix(), this._frustumPlanes ? Xc.GetPlanesToRef(this._transformMatrix, this._frustumPlanes) : this._frustumPlanes = Xc.GetPlanes(this._transformMatrix), this._refreshFrustumPlanes = !1); } /** * Checks if a cullable object (mesh...) is in the camera frustum * This checks the bounding box center. See isCompletelyInFrustum for a full bounding check * @param target The object to check * @param checkRigCameras If the rig cameras should be checked (eg. with VR camera both eyes should be checked) (Default: false) * @returns true if the object is in frustum otherwise false */ isInFrustum(e, t = !1) { if (this._updateFrustumPlanes(), t && this.rigCameras.length > 0) { let r = !1; return this.rigCameras.forEach((n) => { n._updateFrustumPlanes(), r = r || e.isInFrustum(n._frustumPlanes); }), r; } else return e.isInFrustum(this._frustumPlanes); } /** * Checks if a cullable object (mesh...) is in the camera frustum * Unlike isInFrustum this checks the full bounding box * @param target The object to check * @returns true if the object is in frustum otherwise false */ isCompletelyInFrustum(e) { return this._updateFrustumPlanes(), e.isCompletelyInFrustum(this._frustumPlanes); } /** * Gets a ray in the forward direction from the camera. * @param length Defines the length of the ray to create * @param transform Defines the transform to apply to the ray, by default the world matrix is used to create a workd space ray * @param origin Defines the start point of the ray which defaults to the camera position * @returns the forward ray */ // eslint-disable-next-line @typescript-eslint/no-unused-vars getForwardRay(e = 100, t, r) { throw qn("Ray"); } /** * Gets a ray in the forward direction from the camera. * @param refRay the ray to (re)use when setting the values * @param length Defines the length of the ray to create * @param transform Defines the transform to apply to the ray, by default the world matrx is used to create a workd space ray * @param origin Defines the start point of the ray which defaults to the camera position * @returns the forward ray */ // eslint-disable-next-line @typescript-eslint/no-unused-vars getForwardRayToRef(e, t = 100, r, n) { throw qn("Ray"); } /** * Releases resources associated with this node. * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default) * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default) */ dispose(e, t = !1) { for (this.onViewMatrixChangedObservable.clear(), this.onProjectionMatrixChangedObservable.clear(), this.onAfterCheckInputsObservable.clear(), this.onRestoreStateObservable.clear(), this.inputs && this.inputs.clear(), this.getScene().stopAnimation(this), this.getScene().removeCamera(this); this._rigCameras.length > 0; ) { const n = this._rigCameras.pop(); n && n.dispose(); } if (this._parentContainer) { const n = this._parentContainer.cameras.indexOf(this); n > -1 && this._parentContainer.cameras.splice(n, 1), this._parentContainer = null; } if (this._rigPostProcess) this._rigPostProcess.dispose(this), this._rigPostProcess = null, this._postProcesses.length = 0; else if (this.cameraRigMode !== Tr.RIG_MODE_NONE) this._rigPostProcess = null, this._postProcesses.length = 0; else { let n = this._postProcesses.length; for (; --n >= 0; ) { const i = this._postProcesses[n]; i && i.dispose(this); } } let r = this.customRenderTargets.length; for (; --r >= 0; ) this.customRenderTargets[r].dispose(); this.customRenderTargets.length = 0, this._activeMeshes.dispose(), this.getScene().getEngine().releaseRenderPassId(this.renderPassId), super.dispose(e, t); } /** * Gets the left camera of a rig setup in case of Rigged Camera */ get isLeftCamera() { return this._isLeftCamera; } /** * Gets the right camera of a rig setup in case of Rigged Camera */ get isRightCamera() { return this._isRightCamera; } /** * Gets the left camera of a rig setup in case of Rigged Camera */ get leftCamera() { return this._rigCameras.length < 1 ? null : this._rigCameras[0]; } /** * Gets the right camera of a rig setup in case of Rigged Camera */ get rightCamera() { return this._rigCameras.length < 2 ? null : this._rigCameras[1]; } /** * Gets the left camera target of a rig setup in case of Rigged Camera * @returns the target position */ getLeftTarget() { return this._rigCameras.length < 1 ? null : this._rigCameras[0].getTarget(); } /** * Gets the right camera target of a rig setup in case of Rigged Camera * @returns the target position */ getRightTarget() { return this._rigCameras.length < 2 ? null : this._rigCameras[1].getTarget(); } /** * @internal */ setCameraRigMode(e, t) { if (this.cameraRigMode !== e) { for (; this._rigCameras.length > 0; ) { const r = this._rigCameras.pop(); r && r.dispose(); } if (this.cameraRigMode = e, this._cameraRigParams = {}, this._cameraRigParams.interaxialDistance = t.interaxialDistance || 0.0637, this._cameraRigParams.stereoHalfAngle = ye.ToRadians(this._cameraRigParams.interaxialDistance / 0.0637), this.cameraRigMode !== Tr.RIG_MODE_NONE) { const r = this.createRigCamera(this.name + "_L", 0); r && (r._isLeftCamera = !0); const n = this.createRigCamera(this.name + "_R", 1); n && (n._isRightCamera = !0), r && n && (this._rigCameras.push(r), this._rigCameras.push(n)); } this._setRigMode(t), this._cascadePostProcessesToRigCams(), this.update(); } } // eslint-disable-next-line @typescript-eslint/no-unused-vars _setRigMode(e) { } /** @internal */ _getVRProjectionMatrix() { return he.PerspectiveFovLHToRef(this._cameraRigParams.vrMetrics.aspectRatioFov, this._cameraRigParams.vrMetrics.aspectRatio, this.minZ, this.maxZ, this._cameraRigParams.vrWorkMatrix, !0, this.getEngine().isNDCHalfZRange), this._cameraRigParams.vrWorkMatrix.multiplyToRef(this._cameraRigParams.vrHMatrix, this._projectionMatrix), this._projectionMatrix; } /** * @internal */ setCameraRigParameter(e, t) { this._cameraRigParams || (this._cameraRigParams = {}), this._cameraRigParams[e] = t, e === "interaxialDistance" && (this._cameraRigParams.stereoHalfAngle = ye.ToRadians(t / 0.0637)); } /** * needs to be overridden by children so sub has required properties to be copied * @internal */ // eslint-disable-next-line @typescript-eslint/no-unused-vars createRigCamera(e, t) { return null; } /** * May need to be overridden by children * @internal */ _updateRigCameras() { for (let e = 0; e < this._rigCameras.length; e++) this._rigCameras[e].minZ = this.minZ, this._rigCameras[e].maxZ = this.maxZ, this._rigCameras[e].fov = this.fov, this._rigCameras[e].upVector.copyFrom(this.upVector); this.cameraRigMode === Tr.RIG_MODE_STEREOSCOPIC_ANAGLYPH && (this._rigCameras[0].viewport = this._rigCameras[1].viewport = this.viewport); } /** @internal */ _setupInputs() { } /** * Serialiaze the camera setup to a json representation * @returns the JSON representation */ serialize() { const e = jt.Serialize(this); return e.uniqueId = this.uniqueId, e.type = this.getClassName(), this.parent && this.parent._serializeAsParent(e), this.inputs && this.inputs.serialize(e), jt.AppendSerializedAnimations(this, e), e.ranges = this.serializeAnimationRanges(), e.isEnabled = this.isEnabled(), e; } /** * Clones the current camera. * @param name The cloned camera name * @param newParent The cloned camera's new parent (none by default) * @returns the cloned camera */ clone(e, t = null) { const r = jt.Clone(Tr.GetConstructorFromName(this.getClassName(), e, this.getScene(), this.interaxialDistance, this.isStereoscopicSideBySide), this); return r.name = e, r.parent = t, this.onClonedObservable.notifyObservers(r), r; } /** * Gets the direction of the camera relative to a given local axis. * @param localAxis Defines the reference axis to provide a relative direction. * @returns the direction */ getDirection(e) { const t = S.Zero(); return this.getDirectionToRef(e, t), t; } /** * Returns the current camera absolute rotation */ get absoluteRotation() { return this.getWorldMatrix().decompose(void 0, this._absoluteRotation), this._absoluteRotation; } /** * Gets the direction of the camera relative to a given local axis into a passed vector. * @param localAxis Defines the reference axis to provide a relative direction. * @param result Defines the vector to store the result in */ getDirectionToRef(e, t) { S.TransformNormalToRef(e, this.getWorldMatrix(), t); } /** * Gets a camera constructor for a given camera type * @param type The type of the camera to construct (should be equal to one of the camera class name) * @param name The name of the camera the result will be able to instantiate * @param scene The scene the result will construct the camera in * @param interaxial_distance In case of stereoscopic setup, the distance between both eyes * @param isStereoscopicSideBySide In case of stereoscopic setup, should the sereo be side b side * @returns a factory method to construct the camera */ // eslint-disable-next-line @typescript-eslint/naming-convention static GetConstructorFromName(e, t, r, n = 0, i = !0) { const s = Cs.Construct(e, t, r, { // eslint-disable-next-line @typescript-eslint/naming-convention interaxial_distance: n, isStereoscopicSideBySide: i }); return s || (() => Tr._CreateDefaultParsedCamera(t, r)); } /** * Compute the world matrix of the camera. * @returns the camera world matrix */ computeWorldMatrix() { return this.getWorldMatrix(); } /** * Parse a JSON and creates the camera from the parsed information * @param parsedCamera The JSON to parse * @param scene The scene to instantiate the camera in * @returns the newly constructed camera */ static Parse(e, t) { const r = e.type, n = Tr.GetConstructorFromName(r, e.name, t, e.interaxial_distance, e.isStereoscopicSideBySide), i = jt.Parse(n, e, t); if (e.parentId !== void 0 && (i._waitingParentId = e.parentId), e.parentInstanceIndex !== void 0 && (i._waitingParentInstanceIndex = e.parentInstanceIndex), i.inputs && (i.inputs.parse(e), i._setupInputs()), e.upVector && (i.upVector = S.FromArray(e.upVector)), i.setPosition && (i.position.copyFromFloats(0, 0, 0), i.setPosition(S.FromArray(e.position))), e.target && i.setTarget && i.setTarget(S.FromArray(e.target)), e.cameraRigMode) { const s = e.interaxial_distance ? { interaxialDistance: e.interaxial_distance } : {}; i.setCameraRigMode(e.cameraRigMode, s); } if (e.animations) { for (let s = 0; s < e.animations.length; s++) { const a = e.animations[s], f = Jo("BABYLON.Animation"); f && i.animations.push(f.Parse(a)); } Cs.ParseAnimationRanges(i, e, t); } return e.autoAnimate && t.beginAnimation(i, e.autoAnimateFrom, e.autoAnimateTo, e.autoAnimateLoop, e.autoAnimateSpeed || 1), e.isEnabled !== void 0 && i.setEnabled(e.isEnabled), i; } /** @internal */ _calculateHandednessMultiplier() { let e = this.getScene().useRightHandedSystem ? -1 : 1; return this.parent && this.parent._getWorldMatrixDeterminant() < 0 && (e *= -1), e; } } Tr._CreateDefaultParsedCamera = (A, e) => { throw qn("UniversalCamera"); }; Tr.PERSPECTIVE_CAMERA = 0; Tr.ORTHOGRAPHIC_CAMERA = 1; Tr.FOVMODE_VERTICAL_FIXED = 0; Tr.FOVMODE_HORIZONTAL_FIXED = 1; Tr.RIG_MODE_NONE = 0; Tr.RIG_MODE_STEREOSCOPIC_ANAGLYPH = 10; Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL = 11; Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED = 12; Tr.RIG_MODE_STEREOSCOPIC_OVERUNDER = 13; Tr.RIG_MODE_STEREOSCOPIC_INTERLACED = 14; Tr.RIG_MODE_VR = 20; Tr.RIG_MODE_CUSTOM = 22; Tr.ForceAttachControlToAlwaysPreventDefault = !1; C([ fo("position") ], Tr.prototype, "_position", void 0); C([ fo("upVector") ], Tr.prototype, "_upVector", void 0); C([ M() ], Tr.prototype, "orthoLeft", null); C([ M() ], Tr.prototype, "orthoRight", null); C([ M() ], Tr.prototype, "orthoBottom", null); C([ M() ], Tr.prototype, "orthoTop", null); C([ M() ], Tr.prototype, "fov", void 0); C([ M() ], Tr.prototype, "projectionPlaneTilt", void 0); C([ M() ], Tr.prototype, "minZ", void 0); C([ M() ], Tr.prototype, "maxZ", void 0); C([ M() ], Tr.prototype, "inertia", void 0); C([ M() ], Tr.prototype, "mode", null); C([ M() ], Tr.prototype, "layerMask", void 0); C([ M() ], Tr.prototype, "fovMode", void 0); C([ M() ], Tr.prototype, "cameraRigMode", void 0); C([ M() ], Tr.prototype, "interaxialDistance", void 0); C([ M() ], Tr.prototype, "isStereoscopicSideBySide", void 0); class LC { constructor(e, t, r) { this.bu = e, this.bv = t, this.distance = r, this.faceId = 0, this.subMeshId = 0; } } class cp { /** * Creates a new bounding box * @param min defines the minimum vector (in local space) * @param max defines the maximum vector (in local space) * @param worldMatrix defines the new world matrix */ constructor(e, t, r) { this.vectors = Nf.BuildArray(8, S.Zero), this.center = S.Zero(), this.centerWorld = S.Zero(), this.extendSize = S.Zero(), this.extendSizeWorld = S.Zero(), this.directions = Nf.BuildArray(3, S.Zero), this.vectorsWorld = Nf.BuildArray(8, S.Zero), this.minimumWorld = S.Zero(), this.maximumWorld = S.Zero(), this.minimum = S.Zero(), this.maximum = S.Zero(), this._drawWrapperFront = null, this._drawWrapperBack = null, this.reConstruct(e, t, r); } // Methods /** * Recreates the entire bounding box from scratch as if we call the constructor in place * @param min defines the new minimum vector (in local space) * @param max defines the new maximum vector (in local space) * @param worldMatrix defines the new world matrix */ reConstruct(e, t, r) { const n = e.x, i = e.y, s = e.z, a = t.x, f = t.y, o = t.z, d = this.vectors; this.minimum.copyFromFloats(n, i, s), this.maximum.copyFromFloats(a, f, o), d[0].copyFromFloats(n, i, s), d[1].copyFromFloats(a, f, o), d[2].copyFromFloats(a, i, s), d[3].copyFromFloats(n, f, s), d[4].copyFromFloats(n, i, o), d[5].copyFromFloats(a, f, s), d[6].copyFromFloats(n, f, o), d[7].copyFromFloats(a, i, o), t.addToRef(e, this.center).scaleInPlace(0.5), t.subtractToRef(e, this.extendSize).scaleInPlace(0.5), this._worldMatrix = r || he.IdentityReadOnly, this._update(this._worldMatrix); } /** * Scale the current bounding box by applying a scale factor * @param factor defines the scale factor to apply * @returns the current bounding box */ scale(e) { const t = cp._TmpVector3, r = this.maximum.subtractToRef(this.minimum, t[0]), n = r.length(); r.normalizeFromLength(n); const i = n * e, s = r.scaleInPlace(i * 0.5), a = this.center.subtractToRef(s, t[1]), f = this.center.addToRef(s, t[2]); return this.reConstruct(a, f, this._worldMatrix), this; } /** * Gets the world matrix of the bounding box * @returns a matrix */ getWorldMatrix() { return this._worldMatrix; } /** * @internal */ _update(e) { const t = this.minimumWorld, r = this.maximumWorld, n = this.directions, i = this.vectorsWorld, s = this.vectors; if (e.isIdentity()) { t.copyFrom(this.minimum), r.copyFrom(this.maximum); for (let a = 0; a < 8; ++a) i[a].copyFrom(s[a]); this.extendSizeWorld.copyFrom(this.extendSize), this.centerWorld.copyFrom(this.center); } else { t.setAll(Number.MAX_VALUE), r.setAll(-Number.MAX_VALUE); for (let a = 0; a < 8; ++a) { const f = i[a]; S.TransformCoordinatesToRef(s[a], e, f), t.minimizeInPlace(f), r.maximizeInPlace(f); } r.subtractToRef(t, this.extendSizeWorld).scaleInPlace(0.5), r.addToRef(t, this.centerWorld).scaleInPlace(0.5); } S.FromArrayToRef(e.m, 0, n[0]), S.FromArrayToRef(e.m, 4, n[1]), S.FromArrayToRef(e.m, 8, n[2]), this._worldMatrix = e; } /** * Tests if the bounding box is intersecting the frustum planes * @param frustumPlanes defines the frustum planes to test * @returns true if there is an intersection */ isInFrustum(e) { return cp.IsInFrustum(this.vectorsWorld, e); } /** * Tests if the bounding box is entirely inside the frustum planes * @param frustumPlanes defines the frustum planes to test * @returns true if there is an inclusion */ isCompletelyInFrustum(e) { return cp.IsCompletelyInFrustum(this.vectorsWorld, e); } /** * Tests if a point is inside the bounding box * @param point defines the point to test * @returns true if the point is inside the bounding box */ intersectsPoint(e) { const t = this.minimumWorld, r = this.maximumWorld, n = t.x, i = t.y, s = t.z, a = r.x, f = r.y, o = r.z, d = e.x, v = e.y, u = e.z, l = -Dn; return !(a - d < l || l > d - n || f - v < l || l > v - i || o - u < l || l > u - s); } /** * Tests if the bounding box intersects with a bounding sphere * @param sphere defines the sphere to test * @returns true if there is an intersection */ intersectsSphere(e) { return cp.IntersectsSphere(this.minimumWorld, this.maximumWorld, e.centerWorld, e.radiusWorld); } /** * Tests if the bounding box intersects with a box defined by a min and max vectors * @param min defines the min vector to use * @param max defines the max vector to use * @returns true if there is an intersection */ intersectsMinMax(e, t) { const r = this.minimumWorld, n = this.maximumWorld, i = r.x, s = r.y, a = r.z, f = n.x, o = n.y, d = n.z, v = e.x, u = e.y, l = e.z, P = t.x, p = t.y, c = t.z; return !(f < v || i > P || o < u || s > p || d < l || a > c); } /** * Disposes the resources of the class */ dispose() { var e, t; (e = this._drawWrapperFront) === null || e === void 0 || e.dispose(), (t = this._drawWrapperBack) === null || t === void 0 || t.dispose(); } // Statics /** * Tests if two bounding boxes are intersections * @param box0 defines the first box to test * @param box1 defines the second box to test * @returns true if there is an intersection */ static Intersects(e, t) { return e.intersectsMinMax(t.minimumWorld, t.maximumWorld); } /** * Tests if a bounding box defines by a min/max vectors intersects a sphere * @param minPoint defines the minimum vector of the bounding box * @param maxPoint defines the maximum vector of the bounding box * @param sphereCenter defines the sphere center * @param sphereRadius defines the sphere radius * @returns true if there is an intersection */ static IntersectsSphere(e, t, r, n) { const i = cp._TmpVector3[0]; return S.ClampToRef(r, e, t, i), S.DistanceSquared(r, i) <= n * n; } /** * Tests if a bounding box defined with 8 vectors is entirely inside frustum planes * @param boundingVectors defines an array of 8 vectors representing a bounding box * @param frustumPlanes defines the frustum planes to test * @returns true if there is an inclusion */ static IsCompletelyInFrustum(e, t) { for (let r = 0; r < 6; ++r) { const n = t[r]; for (let i = 0; i < 8; ++i) if (n.dotCoordinate(e[i]) < 0) return !1; } return !0; } /** * Tests if a bounding box defined with 8 vectors intersects frustum planes * @param boundingVectors defines an array of 8 vectors representing a bounding box * @param frustumPlanes defines the frustum planes to test * @returns true if there is an intersection */ static IsInFrustum(e, t) { for (let r = 0; r < 6; ++r) { let n = !0; const i = t[r]; for (let s = 0; s < 8; ++s) if (i.dotCoordinate(e[s]) >= 0) { n = !1; break; } if (n) return !1; } return !0; } } cp._TmpVector3 = Nf.BuildArray(3, S.Zero); class rg { /** * Creates a new bounding sphere * @param min defines the minimum vector (in local space) * @param max defines the maximum vector (in local space) * @param worldMatrix defines the new world matrix */ constructor(e, t, r) { this.center = S.Zero(), this.centerWorld = S.Zero(), this.minimum = S.Zero(), this.maximum = S.Zero(), this.reConstruct(e, t, r); } /** * Recreates the entire bounding sphere from scratch as if we call the constructor in place * @param min defines the new minimum vector (in local space) * @param max defines the new maximum vector (in local space) * @param worldMatrix defines the new world matrix */ reConstruct(e, t, r) { this.minimum.copyFrom(e), this.maximum.copyFrom(t); const n = S.Distance(e, t); t.addToRef(e, this.center).scaleInPlace(0.5), this.radius = n * 0.5, this._update(r || he.IdentityReadOnly); } /** * Scale the current bounding sphere by applying a scale factor * @param factor defines the scale factor to apply * @returns the current bounding box */ scale(e) { const t = this.radius * e, r = rg._TmpVector3, n = r[0].setAll(t), i = this.center.subtractToRef(n, r[1]), s = this.center.addToRef(n, r[2]); return this.reConstruct(i, s, this._worldMatrix), this; } /** * Gets the world matrix of the bounding box * @returns a matrix */ getWorldMatrix() { return this._worldMatrix; } // Methods /** * @internal */ _update(e) { if (e.isIdentity()) this.centerWorld.copyFrom(this.center), this.radiusWorld = this.radius; else { S.TransformCoordinatesToRef(this.center, e, this.centerWorld); const t = rg._TmpVector3[0]; S.TransformNormalFromFloatsToRef(1, 1, 1, e, t), this.radiusWorld = Math.max(Math.abs(t.x), Math.abs(t.y), Math.abs(t.z)) * this.radius; } } /** * Tests if the bounding sphere is intersecting the frustum planes * @param frustumPlanes defines the frustum planes to test * @returns true if there is an intersection */ isInFrustum(e) { const t = this.centerWorld, r = this.radiusWorld; for (let n = 0; n < 6; n++) if (e[n].dotCoordinate(t) <= -r) return !1; return !0; } /** * Tests if the bounding sphere center is in between the frustum planes. * Used for optimistic fast inclusion. * @param frustumPlanes defines the frustum planes to test * @returns true if the sphere center is in between the frustum planes */ isCenterInFrustum(e) { const t = this.centerWorld; for (let r = 0; r < 6; r++) if (e[r].dotCoordinate(t) < 0) return !1; return !0; } /** * Tests if a point is inside the bounding sphere * @param point defines the point to test * @returns true if the point is inside the bounding sphere */ intersectsPoint(e) { const t = S.DistanceSquared(this.centerWorld, e); return !(this.radiusWorld * this.radiusWorld < t); } // Statics /** * Checks if two sphere intersect * @param sphere0 sphere 0 * @param sphere1 sphere 1 * @returns true if the spheres intersect */ static Intersects(e, t) { const r = S.DistanceSquared(e.centerWorld, t.centerWorld), n = e.radiusWorld + t.radiusWorld; return !(n * n < r); } /** * Creates a sphere from a center and a radius * @param center The center * @param radius radius * @param matrix Optional worldMatrix * @returns The sphere */ static CreateFromCenterAndRadius(e, t, r) { this._TmpVector3[0].copyFrom(e), this._TmpVector3[1].copyFromFloats(0, 0, t), this._TmpVector3[2].copyFrom(e), this._TmpVector3[0].addInPlace(this._TmpVector3[1]), this._TmpVector3[2].subtractInPlace(this._TmpVector3[1]); const n = new rg(this._TmpVector3[0], this._TmpVector3[2]); return r ? n._worldMatrix = r : n._worldMatrix = he.Identity(), n; } } rg._TmpVector3 = Nf.BuildArray(3, S.Zero); const pE = { min: 0, max: 0 }, hE = { min: 0, max: 0 }, uG = (A, e, t) => { const r = S.Dot(e.centerWorld, A), n = Math.abs(S.Dot(e.directions[0], A)) * e.extendSize.x, i = Math.abs(S.Dot(e.directions[1], A)) * e.extendSize.y, s = Math.abs(S.Dot(e.directions[2], A)) * e.extendSize.z, a = n + i + s; t.min = r - a, t.max = r + a; }, Ol = (A, e, t) => (uG(A, e, pE), uG(A, t, hE), !(pE.min > hE.max || hE.min > pE.max)); class Md { /** * Constructs bounding info * @param minimum min vector of the bounding box/sphere * @param maximum max vector of the bounding box/sphere * @param worldMatrix defines the new world matrix */ constructor(e, t, r) { this._isLocked = !1, this.boundingBox = new cp(e, t, r), this.boundingSphere = new rg(e, t, r); } /** * Recreates the entire bounding info from scratch as if we call the constructor in place * @param min defines the new minimum vector (in local space) * @param max defines the new maximum vector (in local space) * @param worldMatrix defines the new world matrix */ reConstruct(e, t, r) { this.boundingBox.reConstruct(e, t, r), this.boundingSphere.reConstruct(e, t, r); } /** * min vector of the bounding box/sphere */ get minimum() { return this.boundingBox.minimum; } /** * max vector of the bounding box/sphere */ get maximum() { return this.boundingBox.maximum; } /** * If the info is locked and won't be updated to avoid perf overhead */ get isLocked() { return this._isLocked; } set isLocked(e) { this._isLocked = e; } // Methods /** * Updates the bounding sphere and box * @param world world matrix to be used to update */ update(e) { this._isLocked || (this.boundingBox._update(e), this.boundingSphere._update(e)); } /** * Recreate the bounding info to be centered around a specific point given a specific extend. * @param center New center of the bounding info * @param extend New extend of the bounding info * @returns the current bounding info */ centerOn(e, t) { const r = Md._TmpVector3[0].copyFrom(e).subtractInPlace(t), n = Md._TmpVector3[1].copyFrom(e).addInPlace(t); return this.boundingBox.reConstruct(r, n, this.boundingBox.getWorldMatrix()), this.boundingSphere.reConstruct(r, n, this.boundingBox.getWorldMatrix()), this; } /** * Grows the bounding info to include the given point. * @param point The point that will be included in the current bounding info (in local space) * @returns the current bounding info */ encapsulate(e) { const t = S.Minimize(this.minimum, e), r = S.Maximize(this.maximum, e); return this.reConstruct(t, r, this.boundingBox.getWorldMatrix()), this; } /** * Grows the bounding info to encapsulate the given bounding info. * @param toEncapsulate The bounding info that will be encapsulated in the current bounding info * @returns the current bounding info */ encapsulateBoundingInfo(e) { const t = ue.Matrix[0]; this.boundingBox.getWorldMatrix().invertToRef(t); const r = ue.Vector3[0]; return S.TransformCoordinatesToRef(e.boundingBox.minimumWorld, t, r), this.encapsulate(r), S.TransformCoordinatesToRef(e.boundingBox.maximumWorld, t, r), this.encapsulate(r), this; } /** * Scale the current bounding info by applying a scale factor * @param factor defines the scale factor to apply * @returns the current bounding info */ scale(e) { return this.boundingBox.scale(e), this.boundingSphere.scale(e), this; } /** * Returns `true` if the bounding info is within the frustum defined by the passed array of planes. * @param frustumPlanes defines the frustum to test * @param strategy defines the strategy to use for the culling (default is BABYLON.AbstractMesh.CULLINGSTRATEGY_STANDARD) * The different strategies available are: * * BABYLON.AbstractMesh.CULLINGSTRATEGY_STANDARD most accurate but slower @see https://doc.babylonjs.com/typedoc/classes/BABYLON.AbstractMesh#CULLINGSTRATEGY_STANDARD * * BABYLON.AbstractMesh.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY faster but less accurate @see https://doc.babylonjs.com/typedoc/classes/BABYLON.AbstractMesh#CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY * * BABYLON.AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION can be faster if always visible @see https://doc.babylonjs.com/typedoc/classes/BABYLON.AbstractMesh#CULLINGSTRATEGY_OPTIMISTIC_INCLUSION * * BABYLON.AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY can be faster if always visible @see https://doc.babylonjs.com/typedoc/classes/BABYLON.AbstractMesh#CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY * @returns true if the bounding info is in the frustum planes */ isInFrustum(e, t = 0) { return (t === 2 || t === 3) && this.boundingSphere.isCenterInFrustum(e) ? !0 : this.boundingSphere.isInFrustum(e) ? t === 1 || t === 3 ? !0 : this.boundingBox.isInFrustum(e) : !1; } /** * Gets the world distance between the min and max points of the bounding box */ get diagonalLength() { const e = this.boundingBox; return e.maximumWorld.subtractToRef(e.minimumWorld, Md._TmpVector3[0]).length(); } /** * Checks if a cullable object (mesh...) is in the camera frustum * Unlike isInFrustum this checks the full bounding box * @param frustumPlanes Camera near/planes * @returns true if the object is in frustum otherwise false */ isCompletelyInFrustum(e) { return this.boundingBox.isCompletelyInFrustum(e); } /** * @internal */ _checkCollision(e) { return e._canDoCollision(this.boundingSphere.centerWorld, this.boundingSphere.radiusWorld, this.boundingBox.minimumWorld, this.boundingBox.maximumWorld); } /** * Checks if a point is inside the bounding box and bounding sphere or the mesh * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/interactions/mesh_intersect * @param point the point to check intersection with * @returns if the point intersects */ intersectsPoint(e) { return !(!this.boundingSphere.centerWorld || !this.boundingSphere.intersectsPoint(e) || !this.boundingBox.intersectsPoint(e)); } /** * Checks if another bounding info intersects the bounding box and bounding sphere or the mesh * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/interactions/mesh_intersect * @param boundingInfo the bounding info to check intersection with * @param precise if the intersection should be done using OBB * @returns if the bounding info intersects */ intersects(e, t) { if (!rg.Intersects(this.boundingSphere, e.boundingSphere) || !cp.Intersects(this.boundingBox, e.boundingBox)) return !1; if (!t) return !0; const r = this.boundingBox, n = e.boundingBox; return !(!Ol(r.directions[0], r, n) || !Ol(r.directions[1], r, n) || !Ol(r.directions[2], r, n) || !Ol(n.directions[0], r, n) || !Ol(n.directions[1], r, n) || !Ol(n.directions[2], r, n) || !Ol(S.Cross(r.directions[0], n.directions[0]), r, n) || !Ol(S.Cross(r.directions[0], n.directions[1]), r, n) || !Ol(S.Cross(r.directions[0], n.directions[2]), r, n) || !Ol(S.Cross(r.directions[1], n.directions[0]), r, n) || !Ol(S.Cross(r.directions[1], n.directions[1]), r, n) || !Ol(S.Cross(r.directions[1], n.directions[2]), r, n) || !Ol(S.Cross(r.directions[2], n.directions[0]), r, n) || !Ol(S.Cross(r.directions[2], n.directions[1]), r, n) || !Ol(S.Cross(r.directions[2], n.directions[2]), r, n)); } } Md._TmpVector3 = Nf.BuildArray(2, S.Zero); class WO { static extractMinAndMaxIndexed(e, t, r, n, i, s) { for (let a = r; a < r + n; a++) { const f = t[a] * 3, o = e[f], d = e[f + 1], v = e[f + 2]; i.minimizeInPlaceFromFloats(o, d, v), s.maximizeInPlaceFromFloats(o, d, v); } } static extractMinAndMax(e, t, r, n, i, s) { for (let a = t, f = t * n; a < t + r; a++, f += n) { const o = e[f], d = e[f + 1], v = e[f + 2]; i.minimizeInPlaceFromFloats(o, d, v), s.maximizeInPlaceFromFloats(o, d, v); } } } C([ Hq.filter((...[A, e]) => !Array.isArray(A) && !Array.isArray(e)) // eslint-disable-next-line @typescript-eslint/naming-convention ], WO, "extractMinAndMaxIndexed", null); C([ Hq.filter((...[A]) => !Array.isArray(A)) // eslint-disable-next-line @typescript-eslint/naming-convention ], WO, "extractMinAndMax", null); function nee(A, e, t, r, n = null) { const i = new S(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE), s = new S(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE); return WO.extractMinAndMaxIndexed(A, e, t, r, i, s), n && (i.x -= i.x * n.x + n.y, i.y -= i.y * n.x + n.y, i.z -= i.z * n.x + n.y, s.x += s.x * n.x + n.y, s.y += s.y * n.x + n.y, s.z += s.z * n.x + n.y), { minimum: i, maximum: s }; } function WS(A, e, t, r = null, n) { const i = new S(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE), s = new S(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE); return n || (n = 3), WO.extractMinAndMax(A, e, t, n, i, s), r && (i.x -= i.x * r.x + r.y, i.y -= i.y * r.x + r.y, i.z -= i.z * r.x + r.y, s.x += s.x * r.x + r.y, s.y += s.y * r.x + r.y, s.z += s.z * r.x + r.y), { minimum: i, maximum: s }; } class rA { /** * Gets material defines used by the effect associated to the sub mesh */ get materialDefines() { var e; return this._mainDrawWrapperOverride ? this._mainDrawWrapperOverride.defines : (e = this._getDrawWrapper()) === null || e === void 0 ? void 0 : e.defines; } /** * Sets material defines used by the effect associated to the sub mesh */ set materialDefines(e) { var t; const r = (t = this._mainDrawWrapperOverride) !== null && t !== void 0 ? t : this._getDrawWrapper(void 0, !0); r.defines = e; } /** * @internal */ _getDrawWrapper(e, t = !1) { e = e ?? this._engine.currentRenderPassId; let r = this._drawWrappers[e]; return !r && t && (this._drawWrappers[e] = r = new zo(this._mesh.getScene().getEngine())), r; } /** * @internal */ _removeDrawWrapper(e, t = !0) { var r; t && ((r = this._drawWrappers[e]) === null || r === void 0 || r.dispose()), this._drawWrappers[e] = void 0; } /** * Gets associated (main) effect (possibly the effect override if defined) */ get effect() { var e, t; return this._mainDrawWrapperOverride ? this._mainDrawWrapperOverride.effect : (t = (e = this._getDrawWrapper()) === null || e === void 0 ? void 0 : e.effect) !== null && t !== void 0 ? t : null; } /** @internal */ get _drawWrapper() { var e; return (e = this._mainDrawWrapperOverride) !== null && e !== void 0 ? e : this._getDrawWrapper(void 0, !0); } /** @internal */ get _drawWrapperOverride() { return this._mainDrawWrapperOverride; } /** * @internal */ _setMainDrawWrapperOverride(e) { this._mainDrawWrapperOverride = e; } /** * Sets associated effect (effect used to render this submesh) * @param effect defines the effect to associate with * @param defines defines the set of defines used to compile this effect * @param materialContext material context associated to the effect * @param resetContext true to reset the draw context */ setEffect(e, t = null, r, n = !0) { const i = this._drawWrapper; i.setEffect(e, t, n), r !== void 0 && (i.materialContext = r), e || (i.defines = null, i.materialContext = void 0); } /** * Resets the draw wrappers cache * @param passId If provided, releases only the draw wrapper corresponding to this render pass id */ resetDrawCache(e) { if (this._drawWrappers) if (e !== void 0) { this._removeDrawWrapper(e); return; } else for (const t of this._drawWrappers) t == null || t.dispose(); this._drawWrappers = []; } /** * Add a new submesh to a mesh * @param materialIndex defines the material index to use * @param verticesStart defines vertex index start * @param verticesCount defines vertices count * @param indexStart defines index start * @param indexCount defines indices count * @param mesh defines the parent mesh * @param renderingMesh defines an optional rendering mesh * @param createBoundingBox defines if bounding box should be created for this submesh * @returns the new submesh */ static AddToMesh(e, t, r, n, i, s, a, f = !0) { return new rA(e, t, r, n, i, s, a, f); } /** * Creates a new submesh * @param materialIndex defines the material index to use * @param verticesStart defines vertex index start * @param verticesCount defines vertices count * @param indexStart defines index start * @param indexCount defines indices count * @param mesh defines the parent mesh * @param renderingMesh defines an optional rendering mesh * @param createBoundingBox defines if bounding box should be created for this submesh * @param addToMesh defines a boolean indicating that the submesh must be added to the mesh.subMeshes array (true by default) */ constructor(e, t, r, n, i, s, a, f = !0, o = !0) { this.materialIndex = e, this.verticesStart = t, this.verticesCount = r, this.indexStart = n, this.indexCount = i, this._mainDrawWrapperOverride = null, this._linesIndexCount = 0, this._linesIndexBuffer = null, this._lastColliderWorldVertices = null, this._lastColliderTransformMatrix = null, this._wasDispatched = !1, this._renderId = 0, this._alphaIndex = 0, this._distanceToCamera = 0, this._currentMaterial = null, this._mesh = s, this._renderingMesh = a || s, o && s.subMeshes.push(this), this._engine = this._mesh.getScene().getEngine(), this.resetDrawCache(), this._trianglePlanes = [], this._id = s.subMeshes.length - 1, f && (this.refreshBoundingInfo(), s.computeWorldMatrix(!0)); } /** * Returns true if this submesh covers the entire parent mesh * @ignorenaming */ // eslint-disable-next-line @typescript-eslint/naming-convention get IsGlobal() { return this.verticesStart === 0 && this.verticesCount === this._mesh.getTotalVertices() && this.indexStart === 0 && this.indexCount === this._mesh.getTotalIndices(); } /** * Returns the submesh BoundingInfo object * @returns current bounding info (or mesh's one if the submesh is global) */ getBoundingInfo() { return this.IsGlobal || this._mesh.hasThinInstances ? this._mesh.getBoundingInfo() : this._boundingInfo; } /** * Sets the submesh BoundingInfo * @param boundingInfo defines the new bounding info to use * @returns the SubMesh */ setBoundingInfo(e) { return this._boundingInfo = e, this; } /** * Returns the mesh of the current submesh * @returns the parent mesh */ getMesh() { return this._mesh; } /** * Returns the rendering mesh of the submesh * @returns the rendering mesh (could be different from parent mesh) */ getRenderingMesh() { return this._renderingMesh; } /** * Returns the replacement mesh of the submesh * @returns the replacement mesh (could be different from parent mesh) */ getReplacementMesh() { return this._mesh._internalAbstractMeshDataInfo._actAsRegularMesh ? this._mesh : null; } /** * Returns the effective mesh of the submesh * @returns the effective mesh (could be different from parent mesh) */ getEffectiveMesh() { const e = this._mesh._internalAbstractMeshDataInfo._actAsRegularMesh ? this._mesh : null; return e || this._renderingMesh; } /** * Returns the submesh material * @param getDefaultMaterial Defines whether or not to get the default material if nothing has been defined. * @returns null or the current material */ getMaterial(e = !0) { var t; const r = (t = this._renderingMesh.getMaterialForRenderPass(this._engine.currentRenderPassId)) !== null && t !== void 0 ? t : this._renderingMesh.material; if (r) { if (this._isMultiMaterial(r)) { const n = r.getSubMaterial(this.materialIndex); return this._currentMaterial !== n && (this._currentMaterial = n, this.resetDrawCache()), n; } } else return e ? this._mesh.getScene().defaultMaterial : null; return r; } _isMultiMaterial(e) { return e.getSubMaterial !== void 0; } // Methods /** * Sets a new updated BoundingInfo object to the submesh * @param data defines an optional position array to use to determine the bounding info * @returns the SubMesh */ refreshBoundingInfo(e = null) { if (this._lastColliderWorldVertices = null, this.IsGlobal || !this._renderingMesh || !this._renderingMesh.geometry) return this; if (e || (e = this._renderingMesh.getVerticesData(J.PositionKind)), !e) return this._boundingInfo = this._mesh.getBoundingInfo(), this; const t = this._renderingMesh.getIndices(); let r; if (this.indexStart === 0 && this.indexCount === t.length) { const n = this._renderingMesh.getBoundingInfo(); r = { minimum: n.minimum.clone(), maximum: n.maximum.clone() }; } else r = nee(e, t, this.indexStart, this.indexCount, this._renderingMesh.geometry.boundingBias); return this._boundingInfo ? this._boundingInfo.reConstruct(r.minimum, r.maximum) : this._boundingInfo = new Md(r.minimum, r.maximum), this; } /** * @internal */ _checkCollision(e) { return this.getBoundingInfo()._checkCollision(e); } /** * Updates the submesh BoundingInfo * @param world defines the world matrix to use to update the bounding info * @returns the submesh */ updateBoundingInfo(e) { let t = this.getBoundingInfo(); return t || (this.refreshBoundingInfo(), t = this.getBoundingInfo()), t && t.update(e), this; } /** * True is the submesh bounding box intersects the frustum defined by the passed array of planes. * @param frustumPlanes defines the frustum planes * @returns true if the submesh is intersecting with the frustum */ isInFrustum(e) { const t = this.getBoundingInfo(); return t ? t.isInFrustum(e, this._mesh.cullingStrategy) : !1; } /** * True is the submesh bounding box is completely inside the frustum defined by the passed array of planes * @param frustumPlanes defines the frustum planes * @returns true if the submesh is inside the frustum */ isCompletelyInFrustum(e) { const t = this.getBoundingInfo(); return t ? t.isCompletelyInFrustum(e) : !1; } /** * Renders the submesh * @param enableAlphaMode defines if alpha needs to be used * @returns the submesh */ render(e) { return this._renderingMesh.render(this, e, this._mesh._internalAbstractMeshDataInfo._actAsRegularMesh ? this._mesh : void 0), this; } /** * @internal */ _getLinesIndexBuffer(e, t) { if (!this._linesIndexBuffer) { const r = []; for (let n = this.indexStart; n < this.indexStart + this.indexCount; n += 3) r.push(e[n], e[n + 1], e[n + 1], e[n + 2], e[n + 2], e[n]); this._linesIndexBuffer = t.createIndexBuffer(r), this._linesIndexCount = r.length; } return this._linesIndexBuffer; } /** * Checks if the submesh intersects with a ray * @param ray defines the ray to test * @returns true is the passed ray intersects the submesh bounding box */ canIntersects(e) { const t = this.getBoundingInfo(); return t ? e.intersectsBox(t.boundingBox) : !1; } /** * Intersects current submesh with a ray * @param ray defines the ray to test * @param positions defines mesh's positions array * @param indices defines mesh's indices array * @param fastCheck defines if the first intersection will be used (and not the closest) * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected * @returns intersection info or null if no intersection */ intersects(e, t, r, n, i) { const s = this.getMaterial(); if (!s) return null; let a = 3, f = !1; switch (s.fillMode) { case 3: case 5: case 6: case 8: return null; case 7: a = 1, f = !0; break; } return s.fillMode === 4 ? r.length ? this._intersectLines(e, t, r, this._mesh.intersectionThreshold, n) : this._intersectUnIndexedLines(e, t, r, this._mesh.intersectionThreshold, n) : !r.length && this._mesh._unIndexed ? this._intersectUnIndexedTriangles(e, t, r, n, i) : this._intersectTriangles(e, t, r, a, f, n, i); } /** * @internal */ _intersectLines(e, t, r, n, i) { let s = null; for (let a = this.indexStart; a < this.indexStart + this.indexCount; a += 2) { const f = t[r[a]], o = t[r[a + 1]], d = e.intersectionSegment(f, o, n); if (!(d < 0) && (i || !s || d < s.distance) && (s = new LC(null, null, d), s.faceId = a / 2, i)) break; } return s; } /** * @internal */ _intersectUnIndexedLines(e, t, r, n, i) { let s = null; for (let a = this.verticesStart; a < this.verticesStart + this.verticesCount; a += 2) { const f = t[a], o = t[a + 1], d = e.intersectionSegment(f, o, n); if (!(d < 0) && (i || !s || d < s.distance) && (s = new LC(null, null, d), s.faceId = a / 2, i)) break; } return s; } /** * @internal */ _intersectTriangles(e, t, r, n, i, s, a) { let f = null, o = -1; for (let d = this.indexStart; d < this.indexStart + this.indexCount - (3 - n); d += n) { o++; const v = r[d], u = r[d + 1], l = r[d + 2]; if (i && l === 4294967295) { d += 2; continue; } const P = t[v], p = t[u], c = t[l]; if (!P || !p || !c || a && !a(P, p, c, e, v, u, l)) continue; const H = e.intersectsTriangle(P, p, c); if (H) { if (H.distance < 0) continue; if ((s || !f || H.distance < f.distance) && (f = H, f.faceId = o, s)) break; } } return f; } /** * @internal */ _intersectUnIndexedTriangles(e, t, r, n, i) { let s = null; for (let a = this.verticesStart; a < this.verticesStart + this.verticesCount; a += 3) { const f = t[a], o = t[a + 1], d = t[a + 2]; if (i && !i(f, o, d, e, -1, -1, -1)) continue; const v = e.intersectsTriangle(f, o, d); if (v) { if (v.distance < 0) continue; if ((n || !s || v.distance < s.distance) && (s = v, s.faceId = a / 3, n)) break; } } return s; } /** @internal */ _rebuild() { this._linesIndexBuffer && (this._linesIndexBuffer = null); } // Clone /** * Creates a new submesh from the passed mesh * @param newMesh defines the new hosting mesh * @param newRenderingMesh defines an optional rendering mesh * @returns the new submesh */ clone(e, t) { const r = new rA(this.materialIndex, this.verticesStart, this.verticesCount, this.indexStart, this.indexCount, e, t, !1); if (!this.IsGlobal) { const n = this.getBoundingInfo(); if (!n) return r; r._boundingInfo = new Md(n.minimum, n.maximum); } return r; } // Dispose /** * Release associated resources */ dispose() { this._linesIndexBuffer && (this._mesh.getScene().getEngine()._releaseBuffer(this._linesIndexBuffer), this._linesIndexBuffer = null); const e = this._mesh.subMeshes.indexOf(this); this._mesh.subMeshes.splice(e, 1), this.resetDrawCache(); } /** * Gets the class name * @returns the string "SubMesh". */ getClassName() { return "SubMesh"; } // Statics /** * Creates a new submesh from indices data * @param materialIndex the index of the main mesh material * @param startIndex the index where to start the copy in the mesh indices array * @param indexCount the number of indices to copy then from the startIndex * @param mesh the main mesh to create the submesh from * @param renderingMesh the optional rendering mesh * @param createBoundingBox defines if bounding box should be created for this submesh * @returns a new submesh */ static CreateFromIndices(e, t, r, n, i, s = !0) { let a = Number.MAX_VALUE, f = -Number.MAX_VALUE; const d = (i || n).getIndices(); for (let v = t; v < t + r; v++) { const u = d[v]; u < a && (a = u), u > f && (f = u); } return new rA(e, a, f - a + 1, t, r, n, i, s); } } class wI { } class Ut { /** * Creates a new VertexData */ constructor() { this.uniqueId = 0, this.metadata = {}, this._applyTo = ree(this._applyToCoroutine.bind(this)), this.uniqueId = Ut._UniqueIDGenerator, Ut._UniqueIDGenerator++; } /** * Uses the passed data array to set the set the values for the specified kind of data * @param data a linear array of floating numbers * @param kind the type of data that is being set, eg positions, colors etc */ set(e, t) { switch (e.length || Se.Warn(`Setting vertex data kind '${t}' with an empty array`), t) { case J.PositionKind: this.positions = e; break; case J.NormalKind: this.normals = e; break; case J.TangentKind: this.tangents = e; break; case J.UVKind: this.uvs = e; break; case J.UV2Kind: this.uvs2 = e; break; case J.UV3Kind: this.uvs3 = e; break; case J.UV4Kind: this.uvs4 = e; break; case J.UV5Kind: this.uvs5 = e; break; case J.UV6Kind: this.uvs6 = e; break; case J.ColorKind: this.colors = e; break; case J.MatricesIndicesKind: this.matricesIndices = e; break; case J.MatricesWeightsKind: this.matricesWeights = e; break; case J.MatricesIndicesExtraKind: this.matricesIndicesExtra = e; break; case J.MatricesWeightsExtraKind: this.matricesWeightsExtra = e; break; } } /** * Associates the vertexData to the passed Mesh. * Sets it as updatable or not (default `false`) * @param mesh the mesh the vertexData is applied to * @param updatable when used and having the value true allows new data to update the vertexData * @returns the VertexData */ applyToMesh(e, t) { return this._applyTo(e, t, !1), this; } /** * Associates the vertexData to the passed Geometry. * Sets it as updatable or not (default `false`) * @param geometry the geometry the vertexData is applied to * @param updatable when used and having the value true allows new data to update the vertexData * @returns VertexData */ applyToGeometry(e, t) { return this._applyTo(e, t, !1), this; } /** * Updates the associated mesh * @param mesh the mesh to be updated * @returns VertexData */ updateMesh(e) { return this._update(e), this; } /** * Updates the associated geometry * @param geometry the geometry to be updated * @returns VertexData. */ updateGeometry(e) { return this._update(e), this; } /** * @internal */ *_applyToCoroutine(e, t = !1, r) { if (this.positions && (e.setVerticesData(J.PositionKind, this.positions, t), r && (yield)), this.normals && (e.setVerticesData(J.NormalKind, this.normals, t), r && (yield)), this.tangents && (e.setVerticesData(J.TangentKind, this.tangents, t), r && (yield)), this.uvs && (e.setVerticesData(J.UVKind, this.uvs, t), r && (yield)), this.uvs2 && (e.setVerticesData(J.UV2Kind, this.uvs2, t), r && (yield)), this.uvs3 && (e.setVerticesData(J.UV3Kind, this.uvs3, t), r && (yield)), this.uvs4 && (e.setVerticesData(J.UV4Kind, this.uvs4, t), r && (yield)), this.uvs5 && (e.setVerticesData(J.UV5Kind, this.uvs5, t), r && (yield)), this.uvs6 && (e.setVerticesData(J.UV6Kind, this.uvs6, t), r && (yield)), this.colors && (e.setVerticesData(J.ColorKind, this.colors, t), r && (yield)), this.matricesIndices && (e.setVerticesData(J.MatricesIndicesKind, this.matricesIndices, t), r && (yield)), this.matricesWeights && (e.setVerticesData(J.MatricesWeightsKind, this.matricesWeights, t), r && (yield)), this.matricesIndicesExtra && (e.setVerticesData(J.MatricesIndicesExtraKind, this.matricesIndicesExtra, t), r && (yield)), this.matricesWeightsExtra && (e.setVerticesData(J.MatricesWeightsExtraKind, this.matricesWeightsExtra, t), r && (yield)), this.indices ? (e.setIndices(this.indices, null, t), r && (yield)) : e.setIndices([], null), e.subMeshes && this.materialInfos && this.materialInfos.length > 1) { const n = e; n.subMeshes = []; for (const i of this.materialInfos) new rA(i.materialIndex, i.verticesStart, i.verticesCount, i.indexStart, i.indexCount, n); } return this; } _update(e, t, r) { return this.positions && e.updateVerticesData(J.PositionKind, this.positions, t, r), this.normals && e.updateVerticesData(J.NormalKind, this.normals, t, r), this.tangents && e.updateVerticesData(J.TangentKind, this.tangents, t, r), this.uvs && e.updateVerticesData(J.UVKind, this.uvs, t, r), this.uvs2 && e.updateVerticesData(J.UV2Kind, this.uvs2, t, r), this.uvs3 && e.updateVerticesData(J.UV3Kind, this.uvs3, t, r), this.uvs4 && e.updateVerticesData(J.UV4Kind, this.uvs4, t, r), this.uvs5 && e.updateVerticesData(J.UV5Kind, this.uvs5, t, r), this.uvs6 && e.updateVerticesData(J.UV6Kind, this.uvs6, t, r), this.colors && e.updateVerticesData(J.ColorKind, this.colors, t, r), this.matricesIndices && e.updateVerticesData(J.MatricesIndicesKind, this.matricesIndices, t, r), this.matricesWeights && e.updateVerticesData(J.MatricesWeightsKind, this.matricesWeights, t, r), this.matricesIndicesExtra && e.updateVerticesData(J.MatricesIndicesExtraKind, this.matricesIndicesExtra, t, r), this.matricesWeightsExtra && e.updateVerticesData(J.MatricesWeightsExtraKind, this.matricesWeightsExtra, t, r), this.indices && e.setIndices(this.indices, null), this; } static _TransformVector3Coordinates(e, t, r = 0, n = e.length) { const i = ue.Vector3[0], s = ue.Vector3[1]; for (let a = r; a < r + n; a += 3) S.FromArrayToRef(e, a, i), S.TransformCoordinatesToRef(i, t, s), e[a] = s.x, e[a + 1] = s.y, e[a + 2] = s.z; } static _TransformVector3Normals(e, t, r = 0, n = e.length) { const i = ue.Vector3[0], s = ue.Vector3[1]; for (let a = r; a < r + n; a += 3) S.FromArrayToRef(e, a, i), S.TransformNormalToRef(i, t, s), e[a] = s.x, e[a + 1] = s.y, e[a + 2] = s.z; } static _TransformVector4Normals(e, t, r = 0, n = e.length) { const i = ue.Vector4[0], s = ue.Vector4[1]; for (let a = r; a < r + n; a += 4) Ir.FromArrayToRef(e, a, i), Ir.TransformNormalToRef(i, t, s), e[a] = s.x, e[a + 1] = s.y, e[a + 2] = s.z, e[a + 3] = s.w; } static _FlipFaces(e, t = 0, r = e.length) { for (let n = t; n < t + r; n += 3) { const i = e[n + 1]; e[n + 1] = e[n + 2], e[n + 2] = i; } } /** * Transforms each position and each normal of the vertexData according to the passed Matrix * @param matrix the transforming matrix * @returns the VertexData */ transform(e) { const t = e.determinant() < 0; return this.positions && Ut._TransformVector3Coordinates(this.positions, e), this.normals && Ut._TransformVector3Normals(this.normals, e), this.tangents && Ut._TransformVector4Normals(this.tangents, e), t && this.indices && Ut._FlipFaces(this.indices), this; } /** * Generates an array of vertex data where each vertex data only has one material info * @returns An array of VertexData */ splitBasedOnMaterialID() { if (!this.materialInfos || this.materialInfos.length < 2) return [this]; const e = []; for (const t of this.materialInfos) { const r = new Ut(); if (this.positions && (r.positions = this.positions.slice(t.verticesStart * 3, (t.verticesCount + t.verticesStart) * 3)), this.normals && (r.normals = this.normals.slice(t.verticesStart * 3, (t.verticesCount + t.verticesStart) * 3)), this.tangents && (r.tangents = this.tangents.slice(t.verticesStart * 4, (t.verticesCount + t.verticesStart) * 4)), this.colors && (r.colors = this.colors.slice(t.verticesStart * 4, (t.verticesCount + t.verticesStart) * 4)), this.uvs && (r.uvs = this.uvs.slice(t.verticesStart * 2, (t.verticesCount + t.verticesStart) * 2)), this.uvs2 && (r.uvs2 = this.uvs2.slice(t.verticesStart * 2, (t.verticesCount + t.verticesStart) * 2)), this.uvs3 && (r.uvs3 = this.uvs3.slice(t.verticesStart * 2, (t.verticesCount + t.verticesStart) * 2)), this.uvs4 && (r.uvs4 = this.uvs4.slice(t.verticesStart * 2, (t.verticesCount + t.verticesStart) * 2)), this.uvs5 && (r.uvs5 = this.uvs5.slice(t.verticesStart * 2, (t.verticesCount + t.verticesStart) * 2)), this.uvs6 && (r.uvs6 = this.uvs6.slice(t.verticesStart * 2, (t.verticesCount + t.verticesStart) * 2)), this.matricesIndices && (r.matricesIndices = this.matricesIndices.slice(t.verticesStart * 4, (t.verticesCount + t.verticesStart) * 4)), this.matricesIndicesExtra && (r.matricesIndicesExtra = this.matricesIndicesExtra.slice(t.verticesStart * 4, (t.verticesCount + t.verticesStart) * 4)), this.matricesWeights && (r.matricesWeights = this.matricesWeights.slice(t.verticesStart * 4, (t.verticesCount + t.verticesStart) * 4)), this.matricesWeightsExtra && (r.matricesWeightsExtra = this.matricesWeightsExtra.slice(t.verticesStart * 4, (t.verticesCount + t.verticesStart) * 4)), this.indices) { r.indices = []; for (let i = t.indexStart; i < t.indexStart + t.indexCount; i++) r.indices.push(this.indices[i] - t.verticesStart); } const n = new wI(); n.indexStart = 0, n.indexCount = r.indices ? r.indices.length : 0, n.materialIndex = t.materialIndex, n.verticesStart = 0, n.verticesCount = (r.positions ? r.positions.length : 0) / 3, r.materialInfos = [n], e.push(r); } return e; } /** * Merges the passed VertexData into the current one * @param others the VertexData to be merged into the current one * @param use32BitsIndices defines a boolean indicating if indices must be store in a 32 bits array * @param forceCloneIndices defines a boolean indicating if indices are forced to be cloned * @param mergeMaterialIds defines a boolean indicating if we need to merge the material infos * @param enableCompletion defines a boolean indicating if the vertex data should be completed to be compatible * @returns the modified VertexData */ merge(e, t = !1, r = !1, n = !1, i = !1) { const s = Array.isArray(e) ? e.map((a) => ({ vertexData: a })) : [{ vertexData: e }]; return mO(this._mergeCoroutine(void 0, s, t, !1, r, n, i)); } /** * @internal */ *_mergeCoroutine(e, t, r = !1, n, i, s = !1, a = !1) { var f, o, d, v; this._validate(); let u = t.map((H) => H.vertexData), l = this; if (a) for (const H of u) H && (H._validate(), !this.normals && H.normals && (this.normals = new Float32Array(this.positions.length)), !this.tangents && H.tangents && (this.tangents = new Float32Array(this.positions.length / 3 * 4)), !this.uvs && H.uvs && (this.uvs = new Float32Array(this.positions.length / 3 * 2)), !this.uvs2 && H.uvs2 && (this.uvs2 = new Float32Array(this.positions.length / 3 * 2)), !this.uvs3 && H.uvs3 && (this.uvs3 = new Float32Array(this.positions.length / 3 * 2)), !this.uvs4 && H.uvs4 && (this.uvs4 = new Float32Array(this.positions.length / 3 * 2)), !this.uvs5 && H.uvs5 && (this.uvs5 = new Float32Array(this.positions.length / 3 * 2)), !this.uvs6 && H.uvs6 && (this.uvs6 = new Float32Array(this.positions.length / 3 * 2)), !this.colors && H.colors && (this.colors = new Float32Array(this.positions.length / 3 * 4), this.colors.fill(1)), !this.matricesIndices && H.matricesIndices && (this.matricesIndices = new Float32Array(this.positions.length / 3 * 4)), !this.matricesWeights && H.matricesWeights && (this.matricesWeights = new Float32Array(this.positions.length / 3 * 4)), !this.matricesIndicesExtra && H.matricesIndicesExtra && (this.matricesIndicesExtra = new Float32Array(this.positions.length / 3 * 4)), !this.matricesWeightsExtra && H.matricesWeightsExtra && (this.matricesWeightsExtra = new Float32Array(this.positions.length / 3 * 4))); for (const H of u) if (H) { if (a) this.normals && !H.normals && (H.normals = new Float32Array(H.positions.length)), this.tangents && !H.tangents && (H.tangents = new Float32Array(H.positions.length / 3 * 4)), this.uvs && !H.uvs && (H.uvs = new Float32Array(H.positions.length / 3 * 2)), this.uvs2 && !H.uvs2 && (H.uvs2 = new Float32Array(H.positions.length / 3 * 2)), this.uvs3 && !H.uvs3 && (H.uvs3 = new Float32Array(H.positions.length / 3 * 2)), this.uvs4 && !H.uvs4 && (H.uvs4 = new Float32Array(H.positions.length / 3 * 2)), this.uvs5 && !H.uvs5 && (H.uvs5 = new Float32Array(H.positions.length / 3 * 2)), this.uvs6 && !H.uvs6 && (H.uvs6 = new Float32Array(H.positions.length / 3 * 2)), this.colors && !H.colors && (H.colors = new Float32Array(H.positions.length / 3 * 4), H.colors.fill(1)), this.matricesIndices && !H.matricesIndices && (H.matricesIndices = new Float32Array(H.positions.length / 3 * 4)), this.matricesWeights && !H.matricesWeights && (H.matricesWeights = new Float32Array(H.positions.length / 3 * 4)), this.matricesIndicesExtra && !H.matricesIndicesExtra && (H.matricesIndicesExtra = new Float32Array(H.positions.length / 3 * 4)), this.matricesWeightsExtra && !H.matricesWeightsExtra && (H.matricesWeightsExtra = new Float32Array(H.positions.length / 3 * 4)); else if (H._validate(), !this.normals != !H.normals || !this.tangents != !H.tangents || !this.uvs != !H.uvs || !this.uvs2 != !H.uvs2 || !this.uvs3 != !H.uvs3 || !this.uvs4 != !H.uvs4 || !this.uvs5 != !H.uvs5 || !this.uvs6 != !H.uvs6 || !this.colors != !H.colors || !this.matricesIndices != !H.matricesIndices || !this.matricesWeights != !H.matricesWeights || !this.matricesIndicesExtra != !H.matricesIndicesExtra || !this.matricesWeightsExtra != !H.matricesWeightsExtra) throw new Error("Cannot merge vertex data that do not have the same set of attributes"); } if (s) { let H = 0, T = 0, q = 0; const b = []; let j = null; const w = []; for (const I of this.splitBasedOnMaterialID()) w.push({ vertexData: I, transform: e }); for (const I of t) if (I.vertexData) for (const N of I.vertexData.splitBasedOnMaterialID()) w.push({ vertexData: N, transform: I.transform }); w.sort((I, N) => { const k = I.vertexData.materialInfos ? I.vertexData.materialInfos[0].materialIndex : 0, R = N.vertexData.materialInfos ? N.vertexData.materialInfos[0].materialIndex : 0; return k > R ? 1 : k === R ? 0 : -1; }); for (const I of w) { const N = I.vertexData; if (N.materialInfos ? H = N.materialInfos[0].materialIndex : H = 0, j && j.materialIndex === H) j.indexCount += N.indices.length, j.verticesCount += N.positions.length / 3; else { const k = new wI(); k.materialIndex = H, k.indexStart = T, k.indexCount = N.indices.length, k.verticesStart = q, k.verticesCount = N.positions.length / 3, b.push(k), j = k; } T += N.indices.length, q += N.positions.length / 3; } const m = w.splice(0, 1)[0]; l = m.vertexData, e = m.transform, u = w.map((I) => I.vertexData), t = w, this.materialInfos = b; } const P = u.reduce((H, T) => { var q, b; return H + ((b = (q = T.indices) === null || q === void 0 ? void 0 : q.length) !== null && b !== void 0 ? b : 0); }, (o = (f = l.indices) === null || f === void 0 ? void 0 : f.length) !== null && o !== void 0 ? o : 0); let c = i || u.some((H) => H.indices === l.indices) ? (d = l.indices) === null || d === void 0 ? void 0 : d.slice() : l.indices; if (P > 0) { let H = (v = c == null ? void 0 : c.length) !== null && v !== void 0 ? v : 0; if (c || (c = new Array(P)), c.length !== P) { if (Array.isArray(c)) c.length = P; else { const q = r || c instanceof Uint32Array ? new Uint32Array(P) : new Uint16Array(P); q.set(c), c = q; } e && e.determinant() < 0 && Ut._FlipFaces(c, 0, H); } let T = l.positions ? l.positions.length / 3 : 0; for (const { vertexData: q, transform: b } of t) if (q.indices) { for (let j = 0; j < q.indices.length; j++) c[H + j] = q.indices[j] + T; b && b.determinant() < 0 && Ut._FlipFaces(c, H, q.indices.length), T += q.positions.length / 3, H += q.indices.length, n && (yield); } } return this.indices = c, this.positions = Ut._MergeElement(J.PositionKind, l.positions, e, t.map((H) => [H.vertexData.positions, H.transform])), n && (yield), l.normals && (this.normals = Ut._MergeElement(J.NormalKind, l.normals, e, t.map((H) => [H.vertexData.normals, H.transform])), n && (yield)), l.tangents && (this.tangents = Ut._MergeElement(J.TangentKind, l.tangents, e, t.map((H) => [H.vertexData.tangents, H.transform])), n && (yield)), l.uvs && (this.uvs = Ut._MergeElement(J.UVKind, l.uvs, e, t.map((H) => [H.vertexData.uvs, H.transform])), n && (yield)), l.uvs2 && (this.uvs2 = Ut._MergeElement(J.UV2Kind, l.uvs2, e, t.map((H) => [H.vertexData.uvs2, H.transform])), n && (yield)), l.uvs3 && (this.uvs3 = Ut._MergeElement(J.UV3Kind, l.uvs3, e, t.map((H) => [H.vertexData.uvs3, H.transform])), n && (yield)), l.uvs4 && (this.uvs4 = Ut._MergeElement(J.UV4Kind, l.uvs4, e, t.map((H) => [H.vertexData.uvs4, H.transform])), n && (yield)), l.uvs5 && (this.uvs5 = Ut._MergeElement(J.UV5Kind, l.uvs5, e, t.map((H) => [H.vertexData.uvs5, H.transform])), n && (yield)), l.uvs6 && (this.uvs6 = Ut._MergeElement(J.UV6Kind, l.uvs6, e, t.map((H) => [H.vertexData.uvs6, H.transform])), n && (yield)), l.colors && (this.colors = Ut._MergeElement(J.ColorKind, l.colors, e, t.map((H) => [H.vertexData.colors, H.transform])), n && (yield)), l.matricesIndices && (this.matricesIndices = Ut._MergeElement(J.MatricesIndicesKind, l.matricesIndices, e, t.map((H) => [H.vertexData.matricesIndices, H.transform])), n && (yield)), l.matricesWeights && (this.matricesWeights = Ut._MergeElement(J.MatricesWeightsKind, l.matricesWeights, e, t.map((H) => [H.vertexData.matricesWeights, H.transform])), n && (yield)), l.matricesIndicesExtra && (this.matricesIndicesExtra = Ut._MergeElement(J.MatricesIndicesExtraKind, l.matricesIndicesExtra, e, t.map((H) => [H.vertexData.matricesIndicesExtra, H.transform])), n && (yield)), l.matricesWeightsExtra && (this.matricesWeightsExtra = Ut._MergeElement(J.MatricesWeightsExtraKind, l.matricesWeightsExtra, e, t.map((H) => [H.vertexData.matricesWeightsExtra, H.transform]))), this; } static _MergeElement(e, t, r, n) { const i = n.filter((f) => f[0] !== null && f[0] !== void 0); if (!t && i.length == 0) return t; if (!t) return this._MergeElement(e, i[0][0], i[0][1], i.slice(1)); const s = i.reduce((f, o) => f + o[0].length, t.length), a = e === J.PositionKind ? Ut._TransformVector3Coordinates : e === J.NormalKind ? Ut._TransformVector3Normals : e === J.TangentKind ? Ut._TransformVector4Normals : () => { }; if (t instanceof Float32Array) { const f = new Float32Array(s); f.set(t), r && a(f, r, 0, t.length); let o = t.length; for (const [d, v] of i) f.set(d, o), v && a(f, v, o, d.length), o += d.length; return f; } else { const f = new Array(s); for (let d = 0; d < t.length; d++) f[d] = t[d]; r && a(f, r, 0, t.length); let o = t.length; for (const [d, v] of i) { for (let u = 0; u < d.length; u++) f[o + u] = d[u]; v && a(f, v, o, d.length), o += d.length; } return f; } } _validate() { if (!this.positions) throw new O0("Positions are required", Z2.MeshInvalidPositionsError); const e = (n, i) => { const s = J.DeduceStride(n); if (i.length % s !== 0) throw new Error("The " + n + "s array count must be a multiple of " + s); return i.length / s; }, t = e(J.PositionKind, this.positions), r = (n, i) => { const s = e(n, i); if (s !== t) throw new Error("The " + n + "s element count (" + s + ") does not match the positions count (" + t + ")"); }; this.normals && r(J.NormalKind, this.normals), this.tangents && r(J.TangentKind, this.tangents), this.uvs && r(J.UVKind, this.uvs), this.uvs2 && r(J.UV2Kind, this.uvs2), this.uvs3 && r(J.UV3Kind, this.uvs3), this.uvs4 && r(J.UV4Kind, this.uvs4), this.uvs5 && r(J.UV5Kind, this.uvs5), this.uvs6 && r(J.UV6Kind, this.uvs6), this.colors && r(J.ColorKind, this.colors), this.matricesIndices && r(J.MatricesIndicesKind, this.matricesIndices), this.matricesWeights && r(J.MatricesWeightsKind, this.matricesWeights), this.matricesIndicesExtra && r(J.MatricesIndicesExtraKind, this.matricesIndicesExtra), this.matricesWeightsExtra && r(J.MatricesWeightsExtraKind, this.matricesWeightsExtra); } /** * Clone the current vertex data * @returns a copy of the current data */ clone() { const e = this.serialize(); return Ut.Parse(e); } /** * Serializes the VertexData * @returns a serialized object */ serialize() { const e = {}; if (this.positions && (e.positions = Array.from(this.positions)), this.normals && (e.normals = Array.from(this.normals)), this.tangents && (e.tangents = Array.from(this.tangents)), this.uvs && (e.uvs = Array.from(this.uvs)), this.uvs2 && (e.uvs2 = Array.from(this.uvs2)), this.uvs3 && (e.uvs3 = Array.from(this.uvs3)), this.uvs4 && (e.uvs4 = Array.from(this.uvs4)), this.uvs5 && (e.uvs5 = Array.from(this.uvs5)), this.uvs6 && (e.uvs6 = Array.from(this.uvs6)), this.colors && (e.colors = Array.from(this.colors)), this.matricesIndices && (e.matricesIndices = Array.from(this.matricesIndices), e.matricesIndices._isExpanded = !0), this.matricesWeights && (e.matricesWeights = Array.from(this.matricesWeights)), this.matricesIndicesExtra && (e.matricesIndicesExtra = Array.from(this.matricesIndicesExtra), e.matricesIndicesExtra._isExpanded = !0), this.matricesWeightsExtra && (e.matricesWeightsExtra = Array.from(this.matricesWeightsExtra)), e.indices = Array.from(this.indices), this.materialInfos) { e.materialInfos = []; for (const t of this.materialInfos) { const r = { indexStart: t.indexStart, indexCount: t.indexCount, materialIndex: t.materialIndex, verticesStart: t.verticesStart, verticesCount: t.verticesCount }; e.materialInfos.push(r); } } return e; } // Statics /** * Extracts the vertexData from a mesh * @param mesh the mesh from which to extract the VertexData * @param copyWhenShared defines if the VertexData must be cloned when shared between multiple meshes, optional, default false * @param forceCopy indicating that the VertexData must be cloned, optional, default false * @returns the object VertexData associated to the passed mesh */ static ExtractFromMesh(e, t, r) { return Ut._ExtractFrom(e, t, r); } /** * Extracts the vertexData from the geometry * @param geometry the geometry from which to extract the VertexData * @param copyWhenShared defines if the VertexData must be cloned when the geometry is shared between multiple meshes, optional, default false * @param forceCopy indicating that the VertexData must be cloned, optional, default false * @returns the object VertexData associated to the passed mesh */ static ExtractFromGeometry(e, t, r) { return Ut._ExtractFrom(e, t, r); } static _ExtractFrom(e, t, r) { const n = new Ut(); return e.isVerticesDataPresent(J.PositionKind) && (n.positions = e.getVerticesData(J.PositionKind, t, r)), e.isVerticesDataPresent(J.NormalKind) && (n.normals = e.getVerticesData(J.NormalKind, t, r)), e.isVerticesDataPresent(J.TangentKind) && (n.tangents = e.getVerticesData(J.TangentKind, t, r)), e.isVerticesDataPresent(J.UVKind) && (n.uvs = e.getVerticesData(J.UVKind, t, r)), e.isVerticesDataPresent(J.UV2Kind) && (n.uvs2 = e.getVerticesData(J.UV2Kind, t, r)), e.isVerticesDataPresent(J.UV3Kind) && (n.uvs3 = e.getVerticesData(J.UV3Kind, t, r)), e.isVerticesDataPresent(J.UV4Kind) && (n.uvs4 = e.getVerticesData(J.UV4Kind, t, r)), e.isVerticesDataPresent(J.UV5Kind) && (n.uvs5 = e.getVerticesData(J.UV5Kind, t, r)), e.isVerticesDataPresent(J.UV6Kind) && (n.uvs6 = e.getVerticesData(J.UV6Kind, t, r)), e.isVerticesDataPresent(J.ColorKind) && (n.colors = e.getVerticesData(J.ColorKind, t, r)), e.isVerticesDataPresent(J.MatricesIndicesKind) && (n.matricesIndices = e.getVerticesData(J.MatricesIndicesKind, t, r)), e.isVerticesDataPresent(J.MatricesWeightsKind) && (n.matricesWeights = e.getVerticesData(J.MatricesWeightsKind, t, r)), e.isVerticesDataPresent(J.MatricesIndicesExtraKind) && (n.matricesIndicesExtra = e.getVerticesData(J.MatricesIndicesExtraKind, t, r)), e.isVerticesDataPresent(J.MatricesWeightsExtraKind) && (n.matricesWeightsExtra = e.getVerticesData(J.MatricesWeightsExtraKind, t, r)), n.indices = e.getIndices(t, r), n; } /** * Creates the VertexData for a Ribbon * @param options an object used to set the following optional parameters for the ribbon, required but can be empty * * pathArray array of paths, each of which an array of successive Vector3 * * closeArray creates a seam between the first and the last paths of the pathArray, optional, default false * * closePath creates a seam between the first and the last points of each path of the path array, optional, default false * * offset a positive integer, only used when pathArray contains a single path (offset = 10 means the point 1 is joined to the point 11), default rounded half size of the pathArray length * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1) * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1) * * invertUV swaps in the U and V coordinates when applying a texture, optional, default false * * uvs a linear array, of length 2 * number of vertices, of custom UV values, optional * * colors a linear array, of length 4 * number of vertices, of custom color values, optional * @param options.pathArray * @param options.closeArray * @param options.closePath * @param options.offset * @param options.sideOrientation * @param options.frontUVs * @param options.backUVs * @param options.invertUV * @param options.uvs * @param options.colors * @returns the VertexData of the ribbon * @deprecated use CreateRibbonVertexData instead */ static CreateRibbon(e) { throw qn("ribbonBuilder"); } /** * Creates the VertexData for a box * @param options an object used to set the following optional parameters for the box, required but can be empty * * size sets the width, height and depth of the box to the value of size, optional default 1 * * width sets the width (x direction) of the box, overwrites the width set by size, optional, default size * * height sets the height (y direction) of the box, overwrites the height set by size, optional, default size * * depth sets the depth (z direction) of the box, overwrites the depth set by size, optional, default size * * faceUV an array of 6 Vector4 elements used to set different images to each box side * * faceColors an array of 6 Color3 elements used to set different colors to each box side * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1) * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1) * @param options.size * @param options.width * @param options.height * @param options.depth * @param options.faceUV * @param options.faceColors * @param options.sideOrientation * @param options.frontUVs * @param options.backUVs * @returns the VertexData of the box * @deprecated Please use CreateBoxVertexData from the BoxBuilder file instead */ static CreateBox(e) { throw qn("boxBuilder"); } /** * Creates the VertexData for a tiled box * @param options an object used to set the following optional parameters for the box, required but can be empty * * faceTiles sets the pattern, tile size and number of tiles for a face * * faceUV an array of 6 Vector4 elements used to set different images to each box side * * faceColors an array of 6 Color3 elements used to set different colors to each box side * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * @param options.pattern * @param options.width * @param options.height * @param options.depth * @param options.tileSize * @param options.tileWidth * @param options.tileHeight * @param options.alignHorizontal * @param options.alignVertical * @param options.faceUV * @param options.faceColors * @param options.sideOrientation * @returns the VertexData of the box * @deprecated Please use CreateTiledBoxVertexData instead */ static CreateTiledBox(e) { throw qn("tiledBoxBuilder"); } /** * Creates the VertexData for a tiled plane * @param options an object used to set the following optional parameters for the box, required but can be empty * * pattern a limited pattern arrangement depending on the number * * tileSize sets the width, height and depth of the tile to the value of size, optional default 1 * * tileWidth sets the width (x direction) of the tile, overwrites the width set by size, optional, default size * * tileHeight sets the height (y direction) of the tile, overwrites the height set by size, optional, default size * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1) * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1) * @param options.pattern * @param options.tileSize * @param options.tileWidth * @param options.tileHeight * @param options.size * @param options.width * @param options.height * @param options.alignHorizontal * @param options.alignVertical * @param options.sideOrientation * @param options.frontUVs * @param options.backUVs * @returns the VertexData of the tiled plane * @deprecated use CreateTiledPlaneVertexData instead */ static CreateTiledPlane(e) { throw qn("tiledPlaneBuilder"); } /** * Creates the VertexData for an ellipsoid, defaults to a sphere * @param options an object used to set the following optional parameters for the box, required but can be empty * * segments sets the number of horizontal strips optional, default 32 * * diameter sets the axes dimensions, diameterX, diameterY and diameterZ to the value of diameter, optional default 1 * * diameterX sets the diameterX (x direction) of the ellipsoid, overwrites the diameterX set by diameter, optional, default diameter * * diameterY sets the diameterY (y direction) of the ellipsoid, overwrites the diameterY set by diameter, optional, default diameter * * diameterZ sets the diameterZ (z direction) of the ellipsoid, overwrites the diameterZ set by diameter, optional, default diameter * * arc a number from 0 to 1, to create an unclosed ellipsoid based on the fraction of the circumference (latitude) given by the arc value, optional, default 1 * * slice a number from 0 to 1, to create an unclosed ellipsoid based on the fraction of the height (latitude) given by the arc value, optional, default 1 * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1) * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1) * @param options.segments * @param options.diameter * @param options.diameterX * @param options.diameterY * @param options.diameterZ * @param options.arc * @param options.slice * @param options.sideOrientation * @param options.frontUVs * @param options.backUVs * @returns the VertexData of the ellipsoid * @deprecated use CreateSphereVertexData instead */ static CreateSphere(e) { throw qn("sphereBuilder"); } /** * Creates the VertexData for a cylinder, cone or prism * @param options an object used to set the following optional parameters for the box, required but can be empty * * height sets the height (y direction) of the cylinder, optional, default 2 * * diameterTop sets the diameter of the top of the cone, overwrites diameter, optional, default diameter * * diameterBottom sets the diameter of the bottom of the cone, overwrites diameter, optional, default diameter * * diameter sets the diameter of the top and bottom of the cone, optional default 1 * * tessellation the number of prism sides, 3 for a triangular prism, optional, default 24 * * subdivisions` the number of rings along the cylinder height, optional, default 1 * * arc a number from 0 to 1, to create an unclosed cylinder based on the fraction of the circumference given by the arc value, optional, default 1 * * faceColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively * * faceUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively * * hasRings when true makes each subdivision independently treated as a face for faceUV and faceColors, optional, default false * * enclose when true closes an open cylinder by adding extra flat faces between the height axis and vertical edges, think cut cake * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1) * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1) * @param options.height * @param options.diameterTop * @param options.diameterBottom * @param options.diameter * @param options.tessellation * @param options.subdivisions * @param options.arc * @param options.faceColors * @param options.faceUV * @param options.hasRings * @param options.enclose * @param options.sideOrientation * @param options.frontUVs * @param options.backUVs * @returns the VertexData of the cylinder, cone or prism * @deprecated please use CreateCylinderVertexData instead */ static CreateCylinder(e) { throw qn("cylinderBuilder"); } /** * Creates the VertexData for a torus * @param options an object used to set the following optional parameters for the box, required but can be empty * * diameter the diameter of the torus, optional default 1 * * thickness the diameter of the tube forming the torus, optional default 0.5 * * tessellation the number of prism sides, 3 for a triangular prism, optional, default 24 * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1) * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1) * @param options.diameter * @param options.thickness * @param options.tessellation * @param options.sideOrientation * @param options.frontUVs * @param options.backUVs * @returns the VertexData of the torus * @deprecated use CreateTorusVertexData instead */ static CreateTorus(e) { throw qn("torusBuilder"); } /** * Creates the VertexData of the LineSystem * @param options an object used to set the following optional parameters for the LineSystem, required but can be empty * - lines an array of lines, each line being an array of successive Vector3 * - colors an array of line colors, each of the line colors being an array of successive Color4, one per line point * @param options.lines * @param options.colors * @returns the VertexData of the LineSystem * @deprecated use CreateLineSystemVertexData instead */ static CreateLineSystem(e) { throw qn("linesBuilder"); } /** * Create the VertexData for a DashedLines * @param options an object used to set the following optional parameters for the DashedLines, required but can be empty * - points an array successive Vector3 * - dashSize the size of the dashes relative to the dash number, optional, default 3 * - gapSize the size of the gap between two successive dashes relative to the dash number, optional, default 1 * - dashNb the intended total number of dashes, optional, default 200 * @param options.points * @param options.dashSize * @param options.gapSize * @param options.dashNb * @returns the VertexData for the DashedLines * @deprecated use CreateDashedLinesVertexData instead */ static CreateDashedLines(e) { throw qn("linesBuilder"); } /** * Creates the VertexData for a Ground * @param options an object used to set the following optional parameters for the Ground, required but can be empty * - width the width (x direction) of the ground, optional, default 1 * - height the height (z direction) of the ground, optional, default 1 * - subdivisions the number of subdivisions per side, optional, default 1 * @param options.width * @param options.height * @param options.subdivisions * @param options.subdivisionsX * @param options.subdivisionsY * @returns the VertexData of the Ground * @deprecated Please use CreateGroundVertexData instead */ static CreateGround(e) { throw qn("groundBuilder"); } /** * Creates the VertexData for a TiledGround by subdividing the ground into tiles * @param options an object used to set the following optional parameters for the Ground, required but can be empty * * xmin the ground minimum X coordinate, optional, default -1 * * zmin the ground minimum Z coordinate, optional, default -1 * * xmax the ground maximum X coordinate, optional, default 1 * * zmax the ground maximum Z coordinate, optional, default 1 * * subdivisions a javascript object {w: positive integer, h: positive integer}, `w` and `h` are the numbers of subdivisions on the ground width and height creating 'tiles', default {w: 6, h: 6} * * precision a javascript object {w: positive integer, h: positive integer}, `w` and `h` are the numbers of subdivisions on the tile width and height, default {w: 2, h: 2} * @param options.xmin * @param options.zmin * @param options.xmax * @param options.zmax * @param options.subdivisions * @param options.subdivisions.w * @param options.subdivisions.h * @param options.precision * @param options.precision.w * @param options.precision.h * @returns the VertexData of the TiledGround * @deprecated use CreateTiledGroundVertexData instead */ static CreateTiledGround(e) { throw qn("groundBuilder"); } /** * Creates the VertexData of the Ground designed from a heightmap * @param options an object used to set the following parameters for the Ground, required and provided by CreateGroundFromHeightMap * * width the width (x direction) of the ground * * height the height (z direction) of the ground * * subdivisions the number of subdivisions per side * * minHeight the minimum altitude on the ground, optional, default 0 * * maxHeight the maximum altitude on the ground, optional default 1 * * colorFilter the filter to apply to the image pixel colors to compute the height, optional Color3, default (0.3, 0.59, 0.11) * * buffer the array holding the image color data * * bufferWidth the width of image * * bufferHeight the height of image * * alphaFilter Remove any data where the alpha channel is below this value, defaults 0 (all data visible) * @param options.width * @param options.height * @param options.subdivisions * @param options.minHeight * @param options.maxHeight * @param options.colorFilter * @param options.buffer * @param options.bufferWidth * @param options.bufferHeight * @param options.alphaFilter * @returns the VertexData of the Ground designed from a heightmap * @deprecated use CreateGroundFromHeightMapVertexData instead */ static CreateGroundFromHeightMap(e) { throw qn("groundBuilder"); } /** * Creates the VertexData for a Plane * @param options an object used to set the following optional parameters for the plane, required but can be empty * * size sets the width and height of the plane to the value of size, optional default 1 * * width sets the width (x direction) of the plane, overwrites the width set by size, optional, default size * * height sets the height (y direction) of the plane, overwrites the height set by size, optional, default size * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1) * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1) * @param options.size * @param options.width * @param options.height * @param options.sideOrientation * @param options.frontUVs * @param options.backUVs * @returns the VertexData of the box * @deprecated use CreatePlaneVertexData instead */ static CreatePlane(e) { throw qn("planeBuilder"); } /** * Creates the VertexData of the Disc or regular Polygon * @param options an object used to set the following optional parameters for the disc, required but can be empty * * radius the radius of the disc, optional default 0.5 * * tessellation the number of polygon sides, optional, default 64 * * arc a number from 0 to 1, to create an unclosed polygon based on the fraction of the circumference given by the arc value, optional, default 1 * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1) * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1) * @param options.radius * @param options.tessellation * @param options.arc * @param options.sideOrientation * @param options.frontUVs * @param options.backUVs * @returns the VertexData of the box * @deprecated use CreateDiscVertexData instead */ static CreateDisc(e) { throw qn("discBuilder"); } /** * Creates the VertexData for an irregular Polygon in the XoZ plane using a mesh built by polygonTriangulation.build() * All parameters are provided by CreatePolygon as needed * @param polygon a mesh built from polygonTriangulation.build() * @param sideOrientation takes the values Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * @param fUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively * @param fColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively * @param frontUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1) * @param backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1) * @param wrap a boolean, default false, when true and fUVs used texture is wrapped around all sides, when false texture is applied side * @returns the VertexData of the Polygon * @deprecated use CreatePolygonVertexData instead */ static CreatePolygon(e, t, r, n, i, s, a) { throw qn("polygonBuilder"); } /** * Creates the VertexData of the IcoSphere * @param options an object used to set the following optional parameters for the IcoSphere, required but can be empty * * radius the radius of the IcoSphere, optional default 1 * * radiusX allows stretching in the x direction, optional, default radius * * radiusY allows stretching in the y direction, optional, default radius * * radiusZ allows stretching in the z direction, optional, default radius * * flat when true creates a flat shaded mesh, optional, default true * * subdivisions increasing the subdivisions increases the number of faces, optional, default 4 * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1) * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1) * @param options.radius * @param options.radiusX * @param options.radiusY * @param options.radiusZ * @param options.flat * @param options.subdivisions * @param options.sideOrientation * @param options.frontUVs * @param options.backUVs * @returns the VertexData of the IcoSphere * @deprecated use CreateIcoSphereVertexData instead */ static CreateIcoSphere(e) { throw qn("icoSphereBuilder"); } // inspired from // http://stemkoski.github.io/Three.js/Polyhedra.html /** * Creates the VertexData for a Polyhedron * @param options an object used to set the following optional parameters for the polyhedron, required but can be empty * * type provided types are: * * 0 : Tetrahedron, 1 : Octahedron, 2 : Dodecahedron, 3 : Icosahedron, 4 : Rhombicuboctahedron, 5 : Triangular Prism, 6 : Pentagonal Prism, 7 : Hexagonal Prism, 8 : Square Pyramid (J1) * * 9 : Pentagonal Pyramid (J2), 10 : Triangular Dipyramid (J12), 11 : Pentagonal Dipyramid (J13), 12 : Elongated Square Dipyramid (J15), 13 : Elongated Pentagonal Dipyramid (J16), 14 : Elongated Pentagonal Cupola (J20) * * size the size of the IcoSphere, optional default 1 * * sizeX allows stretching in the x direction, optional, default size * * sizeY allows stretching in the y direction, optional, default size * * sizeZ allows stretching in the z direction, optional, default size * * custom a number that overwrites the type to create from an extended set of polyhedron from https://www.babylonjs-playground.com/#21QRSK#15 with minimised editor * * faceUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively * * faceColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively * * flat when true creates a flat shaded mesh, optional, default true * * subdivisions increasing the subdivisions increases the number of faces, optional, default 4 * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1) * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1) * @param options.type * @param options.size * @param options.sizeX * @param options.sizeY * @param options.sizeZ * @param options.custom * @param options.faceUV * @param options.faceColors * @param options.flat * @param options.sideOrientation * @param options.frontUVs * @param options.backUVs * @returns the VertexData of the Polyhedron * @deprecated use CreatePolyhedronVertexData instead */ static CreatePolyhedron(e) { throw qn("polyhedronBuilder"); } /** * Creates the VertexData for a Capsule, inspired from https://github.com/maximeq/three-js-capsule-geometry/blob/master/src/CapsuleBufferGeometry.js * @param options an object used to set the following optional parameters for the capsule, required but can be empty * @returns the VertexData of the Capsule * @deprecated Please use CreateCapsuleVertexData from the capsuleBuilder file instead */ static CreateCapsule(e = { orientation: S.Up(), subdivisions: 2, tessellation: 16, height: 1, radius: 0.25, capSubdivisions: 6 }) { throw qn("capsuleBuilder"); } // based on http://code.google.com/p/away3d/source/browse/trunk/fp10/Away3D/src/away3d/primitives/TorusKnot.as?spec=svn2473&r=2473 /** * Creates the VertexData for a TorusKnot * @param options an object used to set the following optional parameters for the TorusKnot, required but can be empty * * radius the radius of the torus knot, optional, default 2 * * tube the thickness of the tube, optional, default 0.5 * * radialSegments the number of sides on each tube segments, optional, default 32 * * tubularSegments the number of tubes to decompose the knot into, optional, default 32 * * p the number of windings around the z axis, optional, default 2 * * q the number of windings around the x axis, optional, default 3 * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1) * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1) * @param options.radius * @param options.tube * @param options.radialSegments * @param options.tubularSegments * @param options.p * @param options.q * @param options.sideOrientation * @param options.frontUVs * @param options.backUVs * @returns the VertexData of the Torus Knot * @deprecated use CreateTorusKnotVertexData instead */ static CreateTorusKnot(e) { throw qn("torusKnotBuilder"); } // Tools /** * Compute normals for given positions and indices * @param positions an array of vertex positions, [...., x, y, z, ......] * @param indices an array of indices in groups of three for each triangular facet, [...., i, j, k, ......] * @param normals an array of vertex normals, [...., x, y, z, ......] * @param options an object used to set the following optional parameters for the TorusKnot, optional * * facetNormals : optional array of facet normals (vector3) * * facetPositions : optional array of facet positions (vector3) * * facetPartitioning : optional partitioning array. facetPositions is required for facetPartitioning computation * * ratio : optional partitioning ratio / bounding box, required for facetPartitioning computation * * bInfo : optional bounding info, required for facetPartitioning computation * * bbSize : optional bounding box size data, required for facetPartitioning computation * * subDiv : optional partitioning data about subdivisions on each axis (int), required for facetPartitioning computation * * useRightHandedSystem: optional boolean to for right handed system computation * * depthSort : optional boolean to enable the facet depth sort computation * * distanceTo : optional Vector3 to compute the facet depth from this location * * depthSortedFacets : optional array of depthSortedFacets to store the facet distances from the reference location * @param options.facetNormals * @param options.facetPositions * @param options.facetPartitioning * @param options.ratio * @param options.bInfo * @param options.bbSize * @param options.subDiv * @param options.useRightHandedSystem * @param options.depthSort * @param options.distanceTo * @param options.depthSortedFacets */ static ComputeNormals(e, t, r, n) { let i = 0, s = 0, a = 0, f = 0, o = 0, d = 0, v = 0, u = 0, l = 0, P = 0, p = 0, c = 0, H = 0, T = 0, q = 0, b = 0, j = 0, w = 0, m = 0, I = 0, N = !1, k = !1, R = !1, y = !1, O = 1, Y = 0, ee = null; n && (N = !!n.facetNormals, k = !!n.facetPositions, R = !!n.facetPartitioning, O = n.useRightHandedSystem === !0 ? -1 : 1, Y = n.ratio || 0, y = !!n.depthSort, ee = n.distanceTo, y && ee === void 0 && (ee = S.Zero())); let Z = 0, te = 0, fe = 0, _ = 0; for (R && n && n.bbSize && (Z = n.subDiv.X * Y / n.bbSize.x, te = n.subDiv.Y * Y / n.bbSize.y, fe = n.subDiv.Z * Y / n.bbSize.z, _ = n.subDiv.max * n.subDiv.max, n.facetPartitioning.length = 0), i = 0; i < e.length; i++) r[i] = 0; const G = t.length / 3 | 0; for (i = 0; i < G; i++) { if (c = t[i * 3] * 3, H = c + 1, T = c + 2, q = t[i * 3 + 1] * 3, b = q + 1, j = q + 2, w = t[i * 3 + 2] * 3, m = w + 1, I = w + 2, s = e[c] - e[q], a = e[H] - e[b], f = e[T] - e[j], o = e[w] - e[q], d = e[m] - e[b], v = e[I] - e[j], u = O * (a * v - f * d), l = O * (f * o - s * v), P = O * (s * d - a * o), p = Math.sqrt(u * u + l * l + P * P), p = p === 0 ? 1 : p, u /= p, l /= p, P /= p, N && n && (n.facetNormals[i].x = u, n.facetNormals[i].y = l, n.facetNormals[i].z = P), k && n && (n.facetPositions[i].x = (e[c] + e[q] + e[w]) / 3, n.facetPositions[i].y = (e[H] + e[b] + e[m]) / 3, n.facetPositions[i].z = (e[T] + e[j] + e[I]) / 3), R && n) { const L = Math.floor((n.facetPositions[i].x - n.bInfo.minimum.x * Y) * Z), $ = Math.floor((n.facetPositions[i].y - n.bInfo.minimum.y * Y) * te), ae = Math.floor((n.facetPositions[i].z - n.bInfo.minimum.z * Y) * fe), Pe = Math.floor((e[c] - n.bInfo.minimum.x * Y) * Z), ge = Math.floor((e[H] - n.bInfo.minimum.y * Y) * te), me = Math.floor((e[T] - n.bInfo.minimum.z * Y) * fe), Xe = Math.floor((e[q] - n.bInfo.minimum.x * Y) * Z), De = Math.floor((e[b] - n.bInfo.minimum.y * Y) * te), ne = Math.floor((e[j] - n.bInfo.minimum.z * Y) * fe), re = Math.floor((e[w] - n.bInfo.minimum.x * Y) * Z), ve = Math.floor((e[m] - n.bInfo.minimum.y * Y) * te), qe = Math.floor((e[I] - n.bInfo.minimum.z * Y) * fe), ke = Pe + n.subDiv.max * ge + _ * me, be = Xe + n.subDiv.max * De + _ * ne, Fe = re + n.subDiv.max * ve + _ * qe, Ke = L + n.subDiv.max * $ + _ * ae; n.facetPartitioning[Ke] = n.facetPartitioning[Ke] ? n.facetPartitioning[Ke] : new Array(), n.facetPartitioning[ke] = n.facetPartitioning[ke] ? n.facetPartitioning[ke] : new Array(), n.facetPartitioning[be] = n.facetPartitioning[be] ? n.facetPartitioning[be] : new Array(), n.facetPartitioning[Fe] = n.facetPartitioning[Fe] ? n.facetPartitioning[Fe] : new Array(), n.facetPartitioning[ke].push(i), be != ke && n.facetPartitioning[be].push(i), Fe == be || Fe == ke || n.facetPartitioning[Fe].push(i), Ke == ke || Ke == be || Ke == Fe || n.facetPartitioning[Ke].push(i); } if (y && n && n.facetPositions) { const L = n.depthSortedFacets[i]; L.ind = i * 3, L.sqDistance = S.DistanceSquared(n.facetPositions[i], ee); } r[c] += u, r[H] += l, r[T] += P, r[q] += u, r[b] += l, r[j] += P, r[w] += u, r[m] += l, r[I] += P; } for (i = 0; i < r.length / 3; i++) u = r[i * 3], l = r[i * 3 + 1], P = r[i * 3 + 2], p = Math.sqrt(u * u + l * l + P * P), p = p === 0 ? 1 : p, u /= p, l /= p, P /= p, r[i * 3] = u, r[i * 3 + 1] = l, r[i * 3 + 2] = P; } /** * @internal */ static _ComputeSides(e, t, r, n, i, s, a) { const f = r.length, o = n.length; let d, v; switch (e = e || Ut.DEFAULTSIDE, e) { case Ut.FRONTSIDE: break; case Ut.BACKSIDE: for (d = 0; d < f; d += 3) { const u = r[d]; r[d] = r[d + 2], r[d + 2] = u; } for (v = 0; v < o; v++) n[v] = -n[v]; break; case Ut.DOUBLESIDE: { const u = t.length, l = u / 3; for (let c = 0; c < u; c++) t[u + c] = t[c]; for (d = 0; d < f; d += 3) r[d + f] = r[d + 2] + l, r[d + 1 + f] = r[d + 1] + l, r[d + 2 + f] = r[d] + l; for (v = 0; v < o; v++) n[o + v] = -n[v]; const P = i.length; let p = 0; for (p = 0; p < P; p++) i[p + P] = i[p]; for (s = s || new Ir(0, 0, 1, 1), a = a || new Ir(0, 0, 1, 1), p = 0, d = 0; d < P / 2; d++) i[p] = s.x + (s.z - s.x) * i[p], i[p + 1] = s.y + (s.w - s.y) * i[p + 1], i[p + P] = a.x + (a.z - a.x) * i[p + P], i[p + P + 1] = a.y + (a.w - a.y) * i[p + P + 1], p += 2; break; } } } /** * Creates a VertexData from serialized data * @param parsedVertexData the parsed data from an imported file * @returns a VertexData */ static Parse(e) { const t = new Ut(), r = e.positions; r && t.set(r, J.PositionKind); const n = e.normals; n && t.set(n, J.NormalKind); const i = e.tangents; i && t.set(i, J.TangentKind); const s = e.uvs; s && t.set(s, J.UVKind); const a = e.uvs2; a && t.set(a, J.UV2Kind); const f = e.uvs3; f && t.set(f, J.UV3Kind); const o = e.uvs4; o && t.set(o, J.UV4Kind); const d = e.uvs5; d && t.set(d, J.UV5Kind); const v = e.uvs6; v && t.set(v, J.UV6Kind); const u = e.colors; u && t.set(xt.CheckColors4(u, r.length / 3), J.ColorKind); const l = e.matricesIndices; l && t.set(l, J.MatricesIndicesKind); const P = e.matricesWeights; P && t.set(P, J.MatricesWeightsKind); const p = e.indices; p && (t.indices = p); const c = e.materialInfos; if (c) { t.materialInfos = []; for (const H of c) { const T = new wI(); T.indexCount = H.indexCount, T.indexStart = H.indexStart, T.verticesCount = H.verticesCount, T.verticesStart = H.verticesStart, T.materialIndex = H.materialIndex, t.materialInfos.push(T); } } return t; } /** * Applies VertexData created from the imported parameters to the geometry * @param parsedVertexData the parsed data from an imported file * @param geometry the geometry to apply the VertexData to */ static ImportVertexData(e, t) { const r = Ut.Parse(e); t.setAllVerticesData(r, e.updatable); } } Ut.FRONTSIDE = 0; Ut.BACKSIDE = 1; Ut.DOUBLESIDE = 2; Ut.DEFAULTSIDE = 0; Ut._UniqueIDGenerator = 0; C([ Hq.filter((...[A]) => !Array.isArray(A)) ], Ut, "_TransformVector3Coordinates", null); C([ Hq.filter((...[A]) => !Array.isArray(A)) ], Ut, "_TransformVector3Normals", null); C([ Hq.filter((...[A]) => !Array.isArray(A)) ], Ut, "_TransformVector4Normals", null); C([ Hq.filter((...[A]) => !Array.isArray(A)) ], Ut, "_FlipFaces", null); class l9 { /** * Gets or sets a boolean indicating if entire scene must be loaded even if scene contains incremental data */ static get ForceFullSceneLoadingForIncremental() { return l9._ForceFullSceneLoadingForIncremental; } static set ForceFullSceneLoadingForIncremental(e) { l9._ForceFullSceneLoadingForIncremental = e; } /** * Gets or sets a boolean indicating if loading screen must be displayed while loading a scene */ static get ShowLoadingScreen() { return l9._ShowLoadingScreen; } static set ShowLoadingScreen(e) { l9._ShowLoadingScreen = e; } /** * Defines the current logging level (while loading the scene) * @ignorenaming */ // eslint-disable-next-line @typescript-eslint/naming-convention static get loggingLevel() { return l9._LoggingLevel; } // eslint-disable-next-line @typescript-eslint/naming-convention static set loggingLevel(e) { l9._LoggingLevel = e; } /** * Gets or set a boolean indicating if matrix weights must be cleaned upon loading */ static get CleanBoneMatrixWeights() { return l9._CleanBoneMatrixWeights; } static set CleanBoneMatrixWeights(e) { l9._CleanBoneMatrixWeights = e; } } l9._ForceFullSceneLoadingForIncremental = !1; l9._ShowLoadingScreen = !0; l9._CleanBoneMatrixWeights = !1; l9._LoggingLevel = 0; class us { } us.UseOpenGLOrientationForUV = !1; class Tf { /** * Gets or sets the Bias Vector to apply on the bounding elements (box/sphere), the max extend is computed as v += v * bias.x + bias.y, the min is computed as v -= v * bias.x + bias.y */ get boundingBias() { return this._boundingBias; } /** * Gets or sets the Bias Vector to apply on the bounding elements (box/sphere), the max extend is computed as v += v * bias.x + bias.y, the min is computed as v -= v * bias.x + bias.y */ set boundingBias(e) { this._boundingBias ? this._boundingBias.copyFrom(e) : this._boundingBias = e.clone(), this._updateBoundingInfo(!0, null); } /** * Static function used to attach a new empty geometry to a mesh * @param mesh defines the mesh to attach the geometry to * @returns the new Geometry */ static CreateGeometryForMesh(e) { const t = new Tf(Tf.RandomId(), e.getScene()); return t.applyToMesh(e), t; } /** Get the list of meshes using this geometry */ get meshes() { return this._meshes; } /** * Creates a new geometry * @param id defines the unique ID * @param scene defines the hosting scene * @param vertexData defines the VertexData used to get geometry data * @param updatable defines if geometry must be updatable (false by default) * @param mesh defines the mesh that will be associated with the geometry */ constructor(e, t, r, n = !1, i = null) { this.delayLoadState = 0, this._totalVertices = 0, this._isDisposed = !1, this._indexBufferIsUpdatable = !1, this._positionsCache = [], this._parentContainer = null, this.useBoundingInfoFromGeometry = !1, this._scene = t || gr.LastCreatedScene, this._scene && (this.id = e, this.uniqueId = this._scene.getUniqueId(), this._engine = this._scene.getEngine(), this._meshes = [], this._vertexBuffers = {}, this._indices = [], this._updatable = n, r ? this.setAllVerticesData(r, n) : this._totalVertices = 0, this._engine.getCaps().vertexArrayObject && (this._vertexArrayObjects = {}), i && (this.applyToMesh(i), i.computeWorldMatrix(!0))); } /** * Gets the current extend of the geometry */ get extend() { return this._extend; } /** * Gets the hosting scene * @returns the hosting Scene */ getScene() { return this._scene; } /** * Gets the hosting engine * @returns the hosting Engine */ getEngine() { return this._engine; } /** * Defines if the geometry is ready to use * @returns true if the geometry is ready to be used */ isReady() { return this.delayLoadState === 1 || this.delayLoadState === 0; } /** * Gets a value indicating that the geometry should not be serialized */ get doNotSerialize() { for (let e = 0; e < this._meshes.length; e++) if (!this._meshes[e].doNotSerialize) return !1; return !0; } /** @internal */ _rebuild() { this._vertexArrayObjects && (this._vertexArrayObjects = {}), this._meshes.length !== 0 && this._indices && (this._indexBuffer = this._engine.createIndexBuffer(this._indices, this._updatable)); for (const e in this._vertexBuffers) this._vertexBuffers[e]._rebuild(); } /** * Affects all geometry data in one call * @param vertexData defines the geometry data * @param updatable defines if the geometry must be flagged as updatable (false as default) */ setAllVerticesData(e, t) { e.applyToGeometry(this, t), this._notifyUpdate(); } /** * Set specific vertex data * @param kind defines the data kind (Position, normal, etc...) * @param data defines the vertex data to use * @param updatable defines if the vertex must be flagged as updatable (false as default) * @param stride defines the stride to use (0 by default). This value is deduced from the kind value if not specified */ setVerticesData(e, t, r = !1, n) { r && Array.isArray(t) && (t = new Float32Array(t)); const i = new J(this._engine, t, e, { updatable: r, postponeInternalCreation: this._meshes.length === 0, stride: n, label: "Geometry_" + this.id + "_" + e }); this.setVerticesBuffer(i); } /** * Removes a specific vertex data * @param kind defines the data kind (Position, normal, etc...) */ removeVerticesData(e) { this._vertexBuffers[e] && (this._vertexBuffers[e].dispose(), delete this._vertexBuffers[e]), this._vertexArrayObjects && this._disposeVertexArrayObjects(); } /** * Affect a vertex buffer to the geometry. the vertexBuffer.getKind() function is used to determine where to store the data * @param buffer defines the vertex buffer to use * @param totalVertices defines the total number of vertices for position kind (could be null) * @param disposeExistingBuffer disposes the existing buffer, if any (default: true) */ setVerticesBuffer(e, t = null, r = !0) { const n = e.getKind(); this._vertexBuffers[n] && r && this._vertexBuffers[n].dispose(), e._buffer && e._buffer._increaseReferences(), this._vertexBuffers[n] = e; const i = this._meshes, s = i.length; if (n === J.PositionKind) { this._totalVertices = t ?? e.totalVertices, this._updateExtend(e.getFloatData()), this._resetPointsArrayCache(); const a = this._extend && this._extend.minimum || new S(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE), f = this._extend && this._extend.maximum || new S(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE); for (let o = 0; o < s; o++) { const d = i[o]; d.buildBoundingInfo(a, f), d._createGlobalSubMesh(d.isUnIndexed), d.computeWorldMatrix(!0), d.synchronizeInstances(); } } this._notifyUpdate(n); } /** * Update a specific vertex buffer * This function will directly update the underlying DataBuffer according to the passed numeric array or Float32Array * It will do nothing if the buffer is not updatable * @param kind defines the data kind (Position, normal, etc...) * @param data defines the data to use * @param offset defines the offset in the target buffer where to store the data * @param useBytes set to true if the offset is in bytes */ updateVerticesDataDirectly(e, t, r, n = !1) { const i = this.getVertexBuffer(e); i && (i.updateDirectly(t, r, n), this._notifyUpdate(e)); } /** * Update a specific vertex buffer * This function will create a new buffer if the current one is not updatable * @param kind defines the data kind (Position, normal, etc...) * @param data defines the data to use * @param updateExtends defines if the geometry extends must be recomputed (false by default) */ updateVerticesData(e, t, r = !1) { const n = this.getVertexBuffer(e); n && (n.update(t), e === J.PositionKind && this._updateBoundingInfo(r, t), this._notifyUpdate(e)); } _updateBoundingInfo(e, t) { if (e && this._updateExtend(t), this._resetPointsArrayCache(), e) { const r = this._meshes; for (const n of r) { n.hasBoundingInfo ? n.getBoundingInfo().reConstruct(this._extend.minimum, this._extend.maximum) : n.buildBoundingInfo(this._extend.minimum, this._extend.maximum); const i = n.subMeshes; for (const s of i) s.refreshBoundingInfo(); } } } /** * @internal */ _bind(e, t, r, n) { if (!e) return; t === void 0 && (t = this._indexBuffer); const i = this.getVertexBuffers(); if (!i) return; if (t != this._indexBuffer || !this._vertexArrayObjects && !n) { this._engine.bindBuffers(i, t, e, r); return; } const s = n || this._vertexArrayObjects; s[e.key] || (s[e.key] = this._engine.recordVertexArrayObject(i, t, e, r)), this._engine.bindVertexArrayObject(s[e.key], t); } /** * Gets total number of vertices * @returns the total number of vertices */ getTotalVertices() { return this.isReady() ? this._totalVertices : 0; } /** * Gets a specific vertex data attached to this geometry. Float data is constructed if the vertex buffer data cannot be returned directly. * @param kind defines the data kind (Position, normal, etc...) * @param copyWhenShared defines if the returned array must be cloned upon returning it if the current geometry is shared between multiple meshes * @param forceCopy defines a boolean indicating that the returned array must be cloned upon returning it * @returns a float array containing vertex data */ getVerticesData(e, t, r) { const n = this.getVertexBuffer(e); return n ? n.getFloatData(this._totalVertices, r || t && this._meshes.length !== 1) : null; } /** * Returns a boolean defining if the vertex data for the requested `kind` is updatable * @param kind defines the data kind (Position, normal, etc...) * @returns true if the vertex buffer with the specified kind is updatable */ isVertexBufferUpdatable(e) { const t = this._vertexBuffers[e]; return t ? t.isUpdatable() : !1; } /** * Gets a specific vertex buffer * @param kind defines the data kind (Position, normal, etc...) * @returns a VertexBuffer */ getVertexBuffer(e) { return this.isReady() ? this._vertexBuffers[e] : null; } /** * Returns all vertex buffers * @returns an object holding all vertex buffers indexed by kind */ getVertexBuffers() { return this.isReady() ? this._vertexBuffers : null; } /** * Gets a boolean indicating if specific vertex buffer is present * @param kind defines the data kind (Position, normal, etc...) * @returns true if data is present */ isVerticesDataPresent(e) { return this._vertexBuffers ? this._vertexBuffers[e] !== void 0 : this._delayInfo ? this._delayInfo.indexOf(e) !== -1 : !1; } /** * Gets a list of all attached data kinds (Position, normal, etc...) * @returns a list of string containing all kinds */ getVerticesDataKinds() { const e = []; let t; if (!this._vertexBuffers && this._delayInfo) for (t in this._delayInfo) e.push(t); else for (t in this._vertexBuffers) e.push(t); return e; } /** * Update index buffer * @param indices defines the indices to store in the index buffer * @param offset defines the offset in the target buffer where to store the data * @param gpuMemoryOnly defines a boolean indicating that only the GPU memory must be updated leaving the CPU version of the indices unchanged (false by default) */ updateIndices(e, t, r = !1) { if (this._indexBuffer) if (!this._indexBufferIsUpdatable) this.setIndices(e, null, !0); else { const n = e.length !== this._indices.length; if (r || (this._indices = e.slice()), this._engine.updateDynamicIndexBuffer(this._indexBuffer, e, t), n) for (const i of this._meshes) i._createGlobalSubMesh(!0); } } /** * Sets the index buffer for this geometry. * @param indexBuffer Defines the index buffer to use for this geometry * @param totalVertices Defines the total number of vertices used by the buffer * @param totalIndices Defines the total number of indices in the index buffer */ setIndexBuffer(e, t, r) { this._indices = [], this._indexBufferIsUpdatable = !1, this._indexBuffer = e, this._totalVertices = t, this._totalIndices = r, e.is32Bits || (e.is32Bits = this._totalIndices > 65535); for (const n of this._meshes) n._createGlobalSubMesh(!0), n.synchronizeInstances(); this._notifyUpdate(); } /** * Creates a new index buffer * @param indices defines the indices to store in the index buffer * @param totalVertices defines the total number of vertices (could be null) * @param updatable defines if the index buffer must be flagged as updatable (false by default) */ setIndices(e, t = null, r = !1) { this._indexBuffer && this._engine._releaseBuffer(this._indexBuffer), this._indices = e, this._indexBufferIsUpdatable = r, this._meshes.length !== 0 && this._indices && (this._indexBuffer = this._engine.createIndexBuffer(this._indices, r)), t != null && (this._totalVertices = t); for (const n of this._meshes) n._createGlobalSubMesh(!0), n.synchronizeInstances(); this._notifyUpdate(); } /** * Return the total number of indices * @returns the total number of indices */ getTotalIndices() { return this.isReady() ? this._totalIndices !== void 0 ? this._totalIndices : this._indices.length : 0; } /** * Gets the index buffer array * @param copyWhenShared defines if the returned array must be cloned upon returning it if the current geometry is shared between multiple meshes * @param forceCopy defines a boolean indicating that the returned array must be cloned upon returning it * @returns the index buffer array */ getIndices(e, t) { if (!this.isReady()) return null; const r = this._indices; return !t && (!e || this._meshes.length === 1) ? r : r.slice(); } /** * Gets the index buffer * @returns the index buffer */ getIndexBuffer() { return this.isReady() ? this._indexBuffer : null; } /** * @internal */ _releaseVertexArrayObject(e = null) { !e || !this._vertexArrayObjects || this._vertexArrayObjects[e.key] && (this._engine.releaseVertexArrayObject(this._vertexArrayObjects[e.key]), delete this._vertexArrayObjects[e.key]); } /** * Release the associated resources for a specific mesh * @param mesh defines the source mesh * @param shouldDispose defines if the geometry must be disposed if there is no more mesh pointing to it */ releaseForMesh(e, t) { const r = this._meshes, n = r.indexOf(e); n !== -1 && (r.splice(n, 1), this._vertexArrayObjects && e._invalidateInstanceVertexArrayObject(), e._geometry = null, r.length === 0 && t && this.dispose()); } /** * Apply current geometry to a given mesh * @param mesh defines the mesh to apply geometry to */ applyToMesh(e) { if (e._geometry === this) return; const t = e._geometry; t && t.releaseForMesh(e), this._vertexArrayObjects && e._invalidateInstanceVertexArrayObject(); const r = this._meshes; e._geometry = this, e._internalAbstractMeshDataInfo._positions = null, this._scene.pushGeometry(this), r.push(e), this.isReady() ? this._applyToMesh(e) : this._boundingInfo && e.setBoundingInfo(this._boundingInfo); } _updateExtend(e = null) { if (this.useBoundingInfoFromGeometry && this._boundingInfo) this._extend = { minimum: this._boundingInfo.minimum.clone(), maximum: this._boundingInfo.maximum.clone() }; else { if (!e && (e = this.getVerticesData(J.PositionKind), !e)) return; this._extend = WS(e, 0, this._totalVertices, this.boundingBias, 3); } } _applyToMesh(e) { const t = this._meshes.length; for (const r in this._vertexBuffers) t === 1 && this._vertexBuffers[r].create(), r === J.PositionKind && (this._extend || this._updateExtend(), e.buildBoundingInfo(this._extend.minimum, this._extend.maximum), e._createGlobalSubMesh(e.isUnIndexed), e._updateBoundingInfo()); t === 1 && this._indices && this._indices.length > 0 && (this._indexBuffer = this._engine.createIndexBuffer(this._indices, this._updatable)), e._syncGeometryWithMorphTargetManager(), e.synchronizeInstances(); } _notifyUpdate(e) { this.onGeometryUpdated && this.onGeometryUpdated(this, e), this._vertexArrayObjects && this._disposeVertexArrayObjects(); for (const t of this._meshes) t._markSubMeshesAsAttributesDirty(); } /** * Load the geometry if it was flagged as delay loaded * @param scene defines the hosting scene * @param onLoaded defines a callback called when the geometry is loaded */ load(e, t) { if (this.delayLoadState !== 2) { if (this.isReady()) { t && t(); return; } this.delayLoadState = 2, this._queueLoad(e, t); } } _queueLoad(e, t) { this.delayLoadingFile && (e.addPendingData(this), e._loadFile(this.delayLoadingFile, (r) => { if (!this._delayLoadingFunction) return; this._delayLoadingFunction(JSON.parse(r), this), this.delayLoadState = 1, this._delayInfo = [], e.removePendingData(this); const n = this._meshes, i = n.length; for (let s = 0; s < i; s++) this._applyToMesh(n[s]); t && t(); }, void 0, !0)); } /** * Invert the geometry to move from a right handed system to a left handed one. */ toLeftHanded() { const e = this.getIndices(!1); if (e != null && e.length > 0) { for (let n = 0; n < e.length; n += 3) { const i = e[n + 0]; e[n + 0] = e[n + 2], e[n + 2] = i; } this.setIndices(e); } const t = this.getVerticesData(J.PositionKind, !1); if (t != null && t.length > 0) { for (let n = 0; n < t.length; n += 3) t[n + 2] = -t[n + 2]; this.setVerticesData(J.PositionKind, t, !1); } const r = this.getVerticesData(J.NormalKind, !1); if (r != null && r.length > 0) { for (let n = 0; n < r.length; n += 3) r[n + 2] = -r[n + 2]; this.setVerticesData(J.NormalKind, r, !1); } } // Cache /** @internal */ _resetPointsArrayCache() { this._positions = null; } /** @internal */ _generatePointsArray() { if (this._positions) return !0; const e = this.getVerticesData(J.PositionKind); if (!e || e.length === 0) return !1; for (let t = this._positionsCache.length * 3, r = this._positionsCache.length; t < e.length; t += 3, ++r) this._positionsCache[r] = S.FromArray(e, t); for (let t = 0, r = 0; t < e.length; t += 3, ++r) this._positionsCache[r].set(e[0 + t], e[1 + t], e[2 + t]); return this._positionsCache.length = e.length / 3, this._positions = this._positionsCache, !0; } /** * Gets a value indicating if the geometry is disposed * @returns true if the geometry was disposed */ isDisposed() { return this._isDisposed; } _disposeVertexArrayObjects() { if (this._vertexArrayObjects) { for (const r in this._vertexArrayObjects) this._engine.releaseVertexArrayObject(this._vertexArrayObjects[r]); this._vertexArrayObjects = {}; const e = this._meshes, t = e.length; for (let r = 0; r < t; r++) e[r]._invalidateInstanceVertexArrayObject(); } } /** * Free all associated resources */ dispose() { const e = this._meshes, t = e.length; let r; for (r = 0; r < t; r++) this.releaseForMesh(e[r]); this._meshes.length = 0, this._disposeVertexArrayObjects(); for (const n in this._vertexBuffers) this._vertexBuffers[n].dispose(); if (this._vertexBuffers = {}, this._totalVertices = 0, this._indexBuffer && this._engine._releaseBuffer(this._indexBuffer), this._indexBuffer = null, this._indices = [], this.delayLoadState = 0, this.delayLoadingFile = null, this._delayLoadingFunction = null, this._delayInfo = [], this._boundingInfo = null, this._scene.removeGeometry(this), this._parentContainer) { const n = this._parentContainer.geometries.indexOf(this); n > -1 && this._parentContainer.geometries.splice(n, 1), this._parentContainer = null; } this._isDisposed = !0; } /** * Clone the current geometry into a new geometry * @param id defines the unique ID of the new geometry * @returns a new geometry object */ copy(e) { const t = new Ut(); t.indices = []; const r = this.getIndices(); if (r) for (let f = 0; f < r.length; f++) t.indices.push(r[f]); let n = !1, i = !1, s; for (s in this._vertexBuffers) { const f = this.getVerticesData(s); if (f && (f instanceof Float32Array ? t.set(new Float32Array(f), s) : t.set(f.slice(0), s), !i)) { const o = this.getVertexBuffer(s); o && (n = o.isUpdatable(), i = !n); } } const a = new Tf(e, this._scene, t, n); a.delayLoadState = this.delayLoadState, a.delayLoadingFile = this.delayLoadingFile, a._delayLoadingFunction = this._delayLoadingFunction; for (s in this._delayInfo) a._delayInfo = a._delayInfo || [], a._delayInfo.push(s); return a._boundingInfo = new Md(this._extend.minimum, this._extend.maximum), a; } /** * Serialize the current geometry info (and not the vertices data) into a JSON object * @returns a JSON representation of the current geometry data (without the vertices data) */ serialize() { const e = {}; return e.id = this.id, e.uniqueId = this.uniqueId, e.updatable = this._updatable, Zi && Zi.HasTags(this) && (e.tags = Zi.GetTags(this)), e; } _toNumberArray(e) { return Array.isArray(e) ? e : Array.prototype.slice.call(e); } /** * Release any memory retained by the cached data on the Geometry. * * Call this function to reduce memory footprint of the mesh. * Vertex buffers will not store CPU data anymore (this will prevent picking, collisions or physics to work correctly) */ clearCachedData() { this._indices = [], this._resetPointsArrayCache(); for (const e in this._vertexBuffers) Object.prototype.hasOwnProperty.call(this._vertexBuffers, e) && (this._vertexBuffers[e]._buffer._data = null); } /** * Serialize all vertices data into a JSON object * @returns a JSON representation of the current geometry data */ serializeVerticeData() { const e = this.serialize(); return this.isVerticesDataPresent(J.PositionKind) && (e.positions = this._toNumberArray(this.getVerticesData(J.PositionKind)), this.isVertexBufferUpdatable(J.PositionKind) && (e.positions._updatable = !0)), this.isVerticesDataPresent(J.NormalKind) && (e.normals = this._toNumberArray(this.getVerticesData(J.NormalKind)), this.isVertexBufferUpdatable(J.NormalKind) && (e.normals._updatable = !0)), this.isVerticesDataPresent(J.TangentKind) && (e.tangents = this._toNumberArray(this.getVerticesData(J.TangentKind)), this.isVertexBufferUpdatable(J.TangentKind) && (e.tangents._updatable = !0)), this.isVerticesDataPresent(J.UVKind) && (e.uvs = this._toNumberArray(this.getVerticesData(J.UVKind)), this.isVertexBufferUpdatable(J.UVKind) && (e.uvs._updatable = !0)), this.isVerticesDataPresent(J.UV2Kind) && (e.uvs2 = this._toNumberArray(this.getVerticesData(J.UV2Kind)), this.isVertexBufferUpdatable(J.UV2Kind) && (e.uvs2._updatable = !0)), this.isVerticesDataPresent(J.UV3Kind) && (e.uvs3 = this._toNumberArray(this.getVerticesData(J.UV3Kind)), this.isVertexBufferUpdatable(J.UV3Kind) && (e.uvs3._updatable = !0)), this.isVerticesDataPresent(J.UV4Kind) && (e.uvs4 = this._toNumberArray(this.getVerticesData(J.UV4Kind)), this.isVertexBufferUpdatable(J.UV4Kind) && (e.uvs4._updatable = !0)), this.isVerticesDataPresent(J.UV5Kind) && (e.uvs5 = this._toNumberArray(this.getVerticesData(J.UV5Kind)), this.isVertexBufferUpdatable(J.UV5Kind) && (e.uvs5._updatable = !0)), this.isVerticesDataPresent(J.UV6Kind) && (e.uvs6 = this._toNumberArray(this.getVerticesData(J.UV6Kind)), this.isVertexBufferUpdatable(J.UV6Kind) && (e.uvs6._updatable = !0)), this.isVerticesDataPresent(J.ColorKind) && (e.colors = this._toNumberArray(this.getVerticesData(J.ColorKind)), this.isVertexBufferUpdatable(J.ColorKind) && (e.colors._updatable = !0)), this.isVerticesDataPresent(J.MatricesIndicesKind) && (e.matricesIndices = this._toNumberArray(this.getVerticesData(J.MatricesIndicesKind)), e.matricesIndices._isExpanded = !0, this.isVertexBufferUpdatable(J.MatricesIndicesKind) && (e.matricesIndices._updatable = !0)), this.isVerticesDataPresent(J.MatricesWeightsKind) && (e.matricesWeights = this._toNumberArray(this.getVerticesData(J.MatricesWeightsKind)), this.isVertexBufferUpdatable(J.MatricesWeightsKind) && (e.matricesWeights._updatable = !0)), e.indices = this._toNumberArray(this.getIndices()), e; } // Statics /** * Extracts a clone of a mesh geometry * @param mesh defines the source mesh * @param id defines the unique ID of the new geometry object * @returns the new geometry object */ static ExtractFromMesh(e, t) { const r = e._geometry; return r ? r.copy(t) : null; } /** * You should now use Tools.RandomId(), this method is still here for legacy reasons. * Implementation from http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript/2117523#answer-2117523 * Be aware Math.random() could cause collisions, but: * "All but 6 of the 128 bits of the ID are randomly generated, which means that for any two ids, there's a 1 in 2^^122 (or 5.3x10^^36) chance they'll collide" * @returns a string containing a new GUID */ static RandomId() { return ye.RandomId(); } static _GetGeometryByLoadedUniqueId(e, t) { for (let r = 0; r < t.geometries.length; r++) if (t.geometries[r]._loadedUniqueId === e) return t.geometries[r]; return null; } /** * @internal */ static _ImportGeometry(e, t) { const r = t.getScene(), n = e.geometryUniqueId, i = e.geometryId; if (n || i) { const s = n ? this._GetGeometryByLoadedUniqueId(n, r) : r.getGeometryById(i); s && s.applyToMesh(t); } else if (e instanceof ArrayBuffer) { const s = t._binaryInfo; if (s.positionsAttrDesc && s.positionsAttrDesc.count > 0) { const a = new Float32Array(e, s.positionsAttrDesc.offset, s.positionsAttrDesc.count); t.setVerticesData(J.PositionKind, a, !1); } if (s.normalsAttrDesc && s.normalsAttrDesc.count > 0) { const a = new Float32Array(e, s.normalsAttrDesc.offset, s.normalsAttrDesc.count); t.setVerticesData(J.NormalKind, a, !1); } if (s.tangetsAttrDesc && s.tangetsAttrDesc.count > 0) { const a = new Float32Array(e, s.tangetsAttrDesc.offset, s.tangetsAttrDesc.count); t.setVerticesData(J.TangentKind, a, !1); } if (s.uvsAttrDesc && s.uvsAttrDesc.count > 0) { const a = new Float32Array(e, s.uvsAttrDesc.offset, s.uvsAttrDesc.count); if (us.UseOpenGLOrientationForUV) for (let f = 1; f < a.length; f += 2) a[f] = 1 - a[f]; t.setVerticesData(J.UVKind, a, !1); } if (s.uvs2AttrDesc && s.uvs2AttrDesc.count > 0) { const a = new Float32Array(e, s.uvs2AttrDesc.offset, s.uvs2AttrDesc.count); if (us.UseOpenGLOrientationForUV) for (let f = 1; f < a.length; f += 2) a[f] = 1 - a[f]; t.setVerticesData(J.UV2Kind, a, !1); } if (s.uvs3AttrDesc && s.uvs3AttrDesc.count > 0) { const a = new Float32Array(e, s.uvs3AttrDesc.offset, s.uvs3AttrDesc.count); if (us.UseOpenGLOrientationForUV) for (let f = 1; f < a.length; f += 2) a[f] = 1 - a[f]; t.setVerticesData(J.UV3Kind, a, !1); } if (s.uvs4AttrDesc && s.uvs4AttrDesc.count > 0) { const a = new Float32Array(e, s.uvs4AttrDesc.offset, s.uvs4AttrDesc.count); if (us.UseOpenGLOrientationForUV) for (let f = 1; f < a.length; f += 2) a[f] = 1 - a[f]; t.setVerticesData(J.UV4Kind, a, !1); } if (s.uvs5AttrDesc && s.uvs5AttrDesc.count > 0) { const a = new Float32Array(e, s.uvs5AttrDesc.offset, s.uvs5AttrDesc.count); if (us.UseOpenGLOrientationForUV) for (let f = 1; f < a.length; f += 2) a[f] = 1 - a[f]; t.setVerticesData(J.UV5Kind, a, !1); } if (s.uvs6AttrDesc && s.uvs6AttrDesc.count > 0) { const a = new Float32Array(e, s.uvs6AttrDesc.offset, s.uvs6AttrDesc.count); if (us.UseOpenGLOrientationForUV) for (let f = 1; f < a.length; f += 2) a[f] = 1 - a[f]; t.setVerticesData(J.UV6Kind, a, !1); } if (s.colorsAttrDesc && s.colorsAttrDesc.count > 0) { const a = new Float32Array(e, s.colorsAttrDesc.offset, s.colorsAttrDesc.count); t.setVerticesData(J.ColorKind, a, !1, s.colorsAttrDesc.stride); } if (s.matricesIndicesAttrDesc && s.matricesIndicesAttrDesc.count > 0) { const a = new Int32Array(e, s.matricesIndicesAttrDesc.offset, s.matricesIndicesAttrDesc.count), f = []; for (let o = 0; o < a.length; o++) { const d = a[o]; f.push(d & 255), f.push((d & 65280) >> 8), f.push((d & 16711680) >> 16), f.push(d >> 24 & 255); } t.setVerticesData(J.MatricesIndicesKind, f, !1); } if (s.matricesIndicesExtraAttrDesc && s.matricesIndicesExtraAttrDesc.count > 0) { const a = new Int32Array(e, s.matricesIndicesExtraAttrDesc.offset, s.matricesIndicesExtraAttrDesc.count), f = []; for (let o = 0; o < a.length; o++) { const d = a[o]; f.push(d & 255), f.push((d & 65280) >> 8), f.push((d & 16711680) >> 16), f.push(d >> 24 & 255); } t.setVerticesData(J.MatricesIndicesExtraKind, f, !1); } if (s.matricesWeightsAttrDesc && s.matricesWeightsAttrDesc.count > 0) { const a = new Float32Array(e, s.matricesWeightsAttrDesc.offset, s.matricesWeightsAttrDesc.count); t.setVerticesData(J.MatricesWeightsKind, a, !1); } if (s.indicesAttrDesc && s.indicesAttrDesc.count > 0) { const a = new Int32Array(e, s.indicesAttrDesc.offset, s.indicesAttrDesc.count); t.setIndices(a, null); } if (s.subMeshesAttrDesc && s.subMeshesAttrDesc.count > 0) { const a = new Int32Array(e, s.subMeshesAttrDesc.offset, s.subMeshesAttrDesc.count * 5); t.subMeshes = []; for (let f = 0; f < s.subMeshesAttrDesc.count; f++) { const o = a[f * 5 + 0], d = a[f * 5 + 1], v = a[f * 5 + 2], u = a[f * 5 + 3], l = a[f * 5 + 4]; rA.AddToMesh(o, d, v, u, l, t); } } } else if (e.positions && e.normals && e.indices) { if (t.setVerticesData(J.PositionKind, e.positions, e.positions._updatable), t.setVerticesData(J.NormalKind, e.normals, e.normals._updatable), e.tangents && t.setVerticesData(J.TangentKind, e.tangents, e.tangents._updatable), e.uvs && t.setVerticesData(J.UVKind, e.uvs, e.uvs._updatable), e.uvs2 && t.setVerticesData(J.UV2Kind, e.uvs2, e.uvs2._updatable), e.uvs3 && t.setVerticesData(J.UV3Kind, e.uvs3, e.uvs3._updatable), e.uvs4 && t.setVerticesData(J.UV4Kind, e.uvs4, e.uvs4._updatable), e.uvs5 && t.setVerticesData(J.UV5Kind, e.uvs5, e.uvs5._updatable), e.uvs6 && t.setVerticesData(J.UV6Kind, e.uvs6, e.uvs6._updatable), e.colors && t.setVerticesData(J.ColorKind, xt.CheckColors4(e.colors, e.positions.length / 3), e.colors._updatable), e.matricesIndices) if (e.matricesIndices._isExpanded) delete e.matricesIndices._isExpanded, t.setVerticesData(J.MatricesIndicesKind, e.matricesIndices, e.matricesIndices._updatable); else { const s = []; for (let a = 0; a < e.matricesIndices.length; a++) { const f = e.matricesIndices[a]; s.push(f & 255), s.push((f & 65280) >> 8), s.push((f & 16711680) >> 16), s.push(f >> 24 & 255); } t.setVerticesData(J.MatricesIndicesKind, s, e.matricesIndices._updatable); } if (e.matricesIndicesExtra) if (e.matricesIndicesExtra._isExpanded) delete e.matricesIndices._isExpanded, t.setVerticesData(J.MatricesIndicesExtraKind, e.matricesIndicesExtra, e.matricesIndicesExtra._updatable); else { const s = []; for (let a = 0; a < e.matricesIndicesExtra.length; a++) { const f = e.matricesIndicesExtra[a]; s.push(f & 255), s.push((f & 65280) >> 8), s.push((f & 16711680) >> 16), s.push(f >> 24 & 255); } t.setVerticesData(J.MatricesIndicesExtraKind, s, e.matricesIndicesExtra._updatable); } e.matricesWeights && (Tf._CleanMatricesWeights(e, t), t.setVerticesData(J.MatricesWeightsKind, e.matricesWeights, e.matricesWeights._updatable)), e.matricesWeightsExtra && t.setVerticesData(J.MatricesWeightsExtraKind, e.matricesWeightsExtra, e.matricesWeights._updatable), t.setIndices(e.indices, null); } if (e.subMeshes) { t.subMeshes = []; for (let s = 0; s < e.subMeshes.length; s++) { const a = e.subMeshes[s]; rA.AddToMesh(a.materialIndex, a.verticesStart, a.verticesCount, a.indexStart, a.indexCount, t); } } t._shouldGenerateFlatShading && (t.convertToFlatShadedMesh(), t._shouldGenerateFlatShading = !1), t.computeWorldMatrix(!0), r.onMeshImportedObservable.notifyObservers(t); } static _CleanMatricesWeights(e, t) { if (!l9.CleanBoneMatrixWeights) return; let n = 0; if (e.skeletonId > -1) { const v = t.getScene().getLastSkeletonById(e.skeletonId); if (!v) return; n = v.bones.length; } else return; const i = t.getVerticesData(J.MatricesIndicesKind), s = t.getVerticesData(J.MatricesIndicesExtraKind), a = e.matricesWeights, f = e.matricesWeightsExtra, o = e.numBoneInfluencer, d = a.length; for (let v = 0; v < d; v += 4) { let u = 0, l = -1; for (let P = 0; P < 4; P++) { const p = a[v + P]; u += p, p < 1e-3 && l < 0 && (l = P); } if (f) for (let P = 0; P < 4; P++) { const p = f[v + P]; u += p, p < 1e-3 && l < 0 && (l = P + 4); } if ((l < 0 || l > o - 1) && (l = o - 1), u > 1e-3) { const P = 1 / u; for (let p = 0; p < 4; p++) a[v + p] *= P; if (f) for (let p = 0; p < 4; p++) f[v + p] *= P; } else l >= 4 ? (f[v + l - 4] = 1 - u, s[v + l - 4] = n) : (a[v + l] = 1 - u, i[v + l] = n); } t.setVerticesData(J.MatricesIndicesKind, i), e.matricesWeightsExtra && t.setVerticesData(J.MatricesIndicesExtraKind, s); } /** * Create a new geometry from persisted data (Using .babylon file format) * @param parsedVertexData defines the persisted data * @param scene defines the hosting scene * @param rootUrl defines the root url to use to load assets (like delayed data) * @returns the new geometry object */ static Parse(e, t, r) { const n = new Tf(e.id, t, void 0, e.updatable); return n._loadedUniqueId = e.uniqueId, Zi && Zi.AddTagsTo(n, e.tags), e.delayLoadingFile ? (n.delayLoadState = 4, n.delayLoadingFile = r + e.delayLoadingFile, n._boundingInfo = new Md(S.FromArray(e.boundingBoxMinimum), S.FromArray(e.boundingBoxMaximum)), n._delayInfo = [], e.hasUVs && n._delayInfo.push(J.UVKind), e.hasUVs2 && n._delayInfo.push(J.UV2Kind), e.hasUVs3 && n._delayInfo.push(J.UV3Kind), e.hasUVs4 && n._delayInfo.push(J.UV4Kind), e.hasUVs5 && n._delayInfo.push(J.UV5Kind), e.hasUVs6 && n._delayInfo.push(J.UV6Kind), e.hasColors && n._delayInfo.push(J.ColorKind), e.hasMatricesIndices && n._delayInfo.push(J.MatricesIndicesKind), e.hasMatricesWeights && n._delayInfo.push(J.MatricesWeightsKind), n._delayLoadingFunction = Ut.ImportVertexData) : Ut.ImportVertexData(e, n), t.pushGeometry(n, !0), n; } } class iee { /** * constructor * @param frameSampleSize The number of samples required to saturate the sliding window */ constructor(e = 30) { this._enabled = !0, this._rollingFrameTime = new see(e); } /** * Samples current frame * @param timeMs A timestamp in milliseconds of the current frame to compare with other frames */ sampleFrame(e = Yi.Now) { if (this._enabled) { if (this._lastFrameTimeMs != null) { const t = e - this._lastFrameTimeMs; this._rollingFrameTime.add(t); } this._lastFrameTimeMs = e; } } /** * Returns the average frame time in milliseconds over the sliding window (or the subset of frames sampled so far) */ get averageFrameTime() { return this._rollingFrameTime.average; } /** * Returns the variance frame time in milliseconds over the sliding window (or the subset of frames sampled so far) */ get averageFrameTimeVariance() { return this._rollingFrameTime.variance; } /** * Returns the frame time of the most recent frame */ get instantaneousFrameTime() { return this._rollingFrameTime.history(0); } /** * Returns the average framerate in frames per second over the sliding window (or the subset of frames sampled so far) */ get averageFPS() { return 1e3 / this._rollingFrameTime.average; } /** * Returns the average framerate in frames per second using the most recent frame time */ get instantaneousFPS() { const e = this._rollingFrameTime.history(0); return e === 0 ? 0 : 1e3 / e; } /** * Returns true if enough samples have been taken to completely fill the sliding window */ get isSaturated() { return this._rollingFrameTime.isSaturated(); } /** * Enables contributions to the sliding window sample set */ enable() { this._enabled = !0; } /** * Disables contributions to the sliding window sample set * Samples will not be interpolated over the disabled period */ disable() { this._enabled = !1, this._lastFrameTimeMs = null; } /** * Returns true if sampling is enabled */ get isEnabled() { return this._enabled; } /** * Resets performance monitor */ reset() { this._lastFrameTimeMs = null, this._rollingFrameTime.reset(); } } class see { /** * constructor * @param length The number of samples required to saturate the sliding window */ constructor(e) { this._samples = new Array(e), this.reset(); } /** * Adds a sample to the sample set * @param v The sample value */ add(e) { let t; if (this.isSaturated()) { const r = this._samples[this._pos]; t = r - this.average, this.average -= t / (this._sampleCount - 1), this._m2 -= t * (r - this.average); } else this._sampleCount++; t = e - this.average, this.average += t / this._sampleCount, this._m2 += t * (e - this.average), this.variance = this._m2 / (this._sampleCount - 1), this._samples[this._pos] = e, this._pos++, this._pos %= this._samples.length; } /** * Returns previously added values or null if outside of history or outside the sliding window domain * @param i Index in history. For example, pass 0 for the most recent value and 1 for the value before that * @returns Value previously recorded with add() or null if outside of range */ history(e) { if (e >= this._sampleCount || e >= this._samples.length) return 0; const t = this._wrapPosition(this._pos - 1); return this._samples[this._wrapPosition(t - e)]; } /** * Returns true if enough samples have been taken to completely fill the sliding window * @returns true if sample-set saturated */ isSaturated() { return this._sampleCount >= this._samples.length; } /** * Resets the rolling average (equivalent to 0 samples taken so far) */ reset() { this.average = 0, this.variance = 0, this._sampleCount = 0, this._pos = 0, this._m2 = 0; } /** * Wraps a value around the sample range boundaries * @param i Position in sample range, for example if the sample length is 5, and i is -3, then 2 will be returned. * @returns Wrapped position in sample range */ _wrapPosition(e) { const t = this._samples.length; return (e % t + t) % t; } } hr.prototype.setAlphaConstants = function(A, e, t, r) { this._alphaState.setAlphaBlendConstants(A, e, t, r); }; hr.prototype.setAlphaMode = function(A, e = !1) { if (this._alphaMode === A) { if (!e) { const t = A === 0; this.depthCullingState.depthMask !== t && (this.depthCullingState.depthMask = t); } return; } switch (A) { case 0: this._alphaState.alphaBlend = !1; break; case 7: this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA, this._gl.ONE, this._gl.ONE), this._alphaState.alphaBlend = !0; break; case 8: this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA, this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA), this._alphaState.alphaBlend = !0; break; case 2: this._alphaState.setAlphaBlendFunctionParameters(this._gl.SRC_ALPHA, this._gl.ONE_MINUS_SRC_ALPHA, this._gl.ONE, this._gl.ONE), this._alphaState.alphaBlend = !0; break; case 6: this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE, this._gl.ZERO, this._gl.ONE), this._alphaState.alphaBlend = !0; break; case 1: this._alphaState.setAlphaBlendFunctionParameters(this._gl.SRC_ALPHA, this._gl.ONE, this._gl.ZERO, this._gl.ONE), this._alphaState.alphaBlend = !0; break; case 3: this._alphaState.setAlphaBlendFunctionParameters(this._gl.ZERO, this._gl.ONE_MINUS_SRC_COLOR, this._gl.ONE, this._gl.ONE), this._alphaState.alphaBlend = !0; break; case 4: this._alphaState.setAlphaBlendFunctionParameters(this._gl.DST_COLOR, this._gl.ZERO, this._gl.ONE, this._gl.ONE), this._alphaState.alphaBlend = !0; break; case 5: this._alphaState.setAlphaBlendFunctionParameters(this._gl.SRC_ALPHA, this._gl.ONE_MINUS_SRC_COLOR, this._gl.ONE, this._gl.ONE), this._alphaState.alphaBlend = !0; break; case 9: this._alphaState.setAlphaBlendFunctionParameters(this._gl.CONSTANT_COLOR, this._gl.ONE_MINUS_CONSTANT_COLOR, this._gl.CONSTANT_ALPHA, this._gl.ONE_MINUS_CONSTANT_ALPHA), this._alphaState.alphaBlend = !0; break; case 10: this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE_MINUS_SRC_COLOR, this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA), this._alphaState.alphaBlend = !0; break; case 11: this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE, this._gl.ONE, this._gl.ONE), this._alphaState.alphaBlend = !0; break; case 12: this._alphaState.setAlphaBlendFunctionParameters(this._gl.DST_ALPHA, this._gl.ONE, this._gl.ZERO, this._gl.ZERO), this._alphaState.alphaBlend = !0; break; case 13: this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE_MINUS_DST_COLOR, this._gl.ONE_MINUS_SRC_COLOR, this._gl.ONE_MINUS_DST_ALPHA, this._gl.ONE_MINUS_SRC_ALPHA), this._alphaState.alphaBlend = !0; break; case 14: this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA, this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA), this._alphaState.alphaBlend = !0; break; case 15: this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE, this._gl.ONE, this._gl.ZERO), this._alphaState.alphaBlend = !0; break; case 16: this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE_MINUS_DST_COLOR, this._gl.ONE_MINUS_SRC_COLOR, this._gl.ZERO, this._gl.ONE), this._alphaState.alphaBlend = !0; break; case 17: this._alphaState.setAlphaBlendFunctionParameters(this._gl.SRC_ALPHA, this._gl.ONE_MINUS_SRC_ALPHA, this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA), this._alphaState.alphaBlend = !0; break; } e || (this.depthCullingState.depthMask = A === 0), this._alphaMode = A; }; hr.prototype.getAlphaMode = function() { return this._alphaMode; }; hr.prototype.setAlphaEquation = function(A) { if (this._alphaEquation !== A) { switch (A) { case 0: this._alphaState.setAlphaEquationParameters(32774, 32774); break; case 1: this._alphaState.setAlphaEquationParameters(32778, 32778); break; case 2: this._alphaState.setAlphaEquationParameters(32779, 32779); break; case 3: this._alphaState.setAlphaEquationParameters(32776, 32776); break; case 4: this._alphaState.setAlphaEquationParameters(32775, 32775); break; case 5: this._alphaState.setAlphaEquationParameters(32775, 32774); break; } this._alphaEquation = A; } }; hr.prototype.getAlphaEquation = function() { return this._alphaEquation; }; function KC(A, e, t = !1, r) { switch (A) { case 3: { const i = e instanceof ArrayBuffer ? new Int8Array(e) : new Int8Array(e); return r && i.set(new Int8Array(r)), i; } case 0: { const i = e instanceof ArrayBuffer ? new Uint8Array(e) : new Uint8Array(e); return r && i.set(new Uint8Array(r)), i; } case 4: { const i = e instanceof ArrayBuffer ? new Int16Array(e) : new Int16Array(t ? e / 2 : e); return r && i.set(new Int16Array(r)), i; } case 5: case 8: case 9: case 10: case 2: { const i = e instanceof ArrayBuffer ? new Uint16Array(e) : new Uint16Array(t ? e / 2 : e); return r && i.set(new Uint16Array(r)), i; } case 6: { const i = e instanceof ArrayBuffer ? new Int32Array(e) : new Int32Array(t ? e / 4 : e); return r && i.set(new Int32Array(r)), i; } case 7: case 11: case 12: case 13: case 14: case 15: { const i = e instanceof ArrayBuffer ? new Uint32Array(e) : new Uint32Array(t ? e / 4 : e); return r && i.set(new Uint32Array(r)), i; } case 1: { const i = e instanceof ArrayBuffer ? new Float32Array(e) : new Float32Array(t ? e / 4 : e); return r && i.set(new Float32Array(r)), i; } } const n = e instanceof ArrayBuffer ? new Uint8Array(e) : new Uint8Array(e); return r && n.set(new Uint8Array(r)), n; } hr.prototype._readTexturePixelsSync = function(A, e, t, r = -1, n = 0, i = null, s = !0, a = !1, f = 0, o = 0) { var d, v; const u = this._gl; if (!u) throw new Error("Engine does not have gl rendering context."); if (!this._dummyFramebuffer) { const P = u.createFramebuffer(); if (!P) throw new Error("Unable to create dummy framebuffer"); this._dummyFramebuffer = P; } u.bindFramebuffer(u.FRAMEBUFFER, this._dummyFramebuffer), r > -1 ? u.framebufferTexture2D(u.FRAMEBUFFER, u.COLOR_ATTACHMENT0, u.TEXTURE_CUBE_MAP_POSITIVE_X + r, (d = A._hardwareTexture) === null || d === void 0 ? void 0 : d.underlyingResource, n) : u.framebufferTexture2D(u.FRAMEBUFFER, u.COLOR_ATTACHMENT0, u.TEXTURE_2D, (v = A._hardwareTexture) === null || v === void 0 ? void 0 : v.underlyingResource, n); let l = A.type !== void 0 ? this._getWebGLTextureType(A.type) : u.UNSIGNED_BYTE; if (a) i || (i = KC(A.type, 4 * e * t)); else switch (l) { case u.UNSIGNED_BYTE: i || (i = new Uint8Array(4 * e * t)), l = u.UNSIGNED_BYTE; break; default: i || (i = new Float32Array(4 * e * t)), l = u.FLOAT; break; } return s && this.flushFramebuffer(), u.readPixels(f, o, e, t, u.RGBA, l, i), u.bindFramebuffer(u.FRAMEBUFFER, this._currentFramebuffer), i; }; hr.prototype._readTexturePixels = function(A, e, t, r = -1, n = 0, i = null, s = !0, a = !1, f = 0, o = 0) { return Promise.resolve(this._readTexturePixelsSync(A, e, t, r, n, i, s, a, f, o)); }; hr.prototype.updateDynamicIndexBuffer = function(A, e, t = 0) { this._currentBoundBuffer[this._gl.ELEMENT_ARRAY_BUFFER] = null, this.bindIndexBuffer(A); let r; A.is32Bits ? r = e instanceof Uint32Array ? e : new Uint32Array(e) : r = e instanceof Uint16Array ? e : new Uint16Array(e), this._gl.bufferData(this._gl.ELEMENT_ARRAY_BUFFER, r, this._gl.DYNAMIC_DRAW), this._resetIndexBufferBinding(); }; hr.prototype.updateDynamicVertexBuffer = function(A, e, t, r) { this.bindArrayBuffer(A), t === void 0 && (t = 0); const n = e.byteLength || e.length; r === void 0 || r >= n && t === 0 ? e instanceof Array ? this._gl.bufferSubData(this._gl.ARRAY_BUFFER, t, new Float32Array(e)) : this._gl.bufferSubData(this._gl.ARRAY_BUFFER, t, e) : e instanceof Array ? this._gl.bufferSubData(this._gl.ARRAY_BUFFER, 0, new Float32Array(e).subarray(t, t + r)) : (e instanceof ArrayBuffer ? e = new Uint8Array(e, t, r) : e = new Uint8Array(e.buffer, e.byteOffset + t, r), this._gl.bufferSubData(this._gl.ARRAY_BUFFER, 0, e)), this._resetVertexBufferBinding(); }; class Ge extends hr { /** * Returns the current npm package of the sdk */ // Not mixed with Version for tooling purpose. static get NpmPackage() { return hr.NpmPackage; } /** * Returns the current version of the framework */ static get Version() { return hr.Version; } /** Gets the list of created engines */ static get Instances() { return gr.Instances; } /** * Gets the latest created engine */ static get LastCreatedEngine() { return gr.LastCreatedEngine; } /** * Gets the latest created scene */ static get LastCreatedScene() { return gr.LastCreatedScene; } /** @internal */ /** * Engine abstraction for loading and creating an image bitmap from a given source string. * @param imageSource source to load the image from. * @param options An object that sets options for the image's extraction. * @returns ImageBitmap. */ _createImageBitmapFromSource(e, t) { return new Promise((n, i) => { const s = new Image(); s.onload = () => { s.decode().then(() => { this.createImageBitmap(s, t).then((a) => { n(a); }); }); }, s.onerror = () => { i(`Error loading image ${s.src}`); }, s.src = e; }); } /** * Engine abstraction for createImageBitmap * @param image source for image * @param options An object that sets options for the image's extraction. * @returns ImageBitmap */ createImageBitmap(e, t) { return createImageBitmap(e, t); } /** * Resize an image and returns the image data as an uint8array * @param image image to resize * @param bufferWidth destination buffer width * @param bufferHeight destination buffer height * @returns an uint8array containing RGBA values of bufferWidth * bufferHeight size */ resizeImageBitmap(e, t, r) { const i = this.createCanvas(t, r).getContext("2d"); if (!i) throw new Error("Unable to get 2d context for resizeImageBitmap"); return i.drawImage(e, 0, 0), i.getImageData(0, 0, t, r).data; } /** * Will flag all materials in all scenes in all engines as dirty to trigger new shader compilation * @param flag defines which part of the materials must be marked as dirty * @param predicate defines a predicate used to filter which materials should be affected */ static MarkAllMaterialsAsDirty(e, t) { for (let r = 0; r < Ge.Instances.length; r++) { const n = Ge.Instances[r]; for (let i = 0; i < n.scenes.length; i++) n.scenes[i].markAllMaterialsAsDirty(e, t); } } /** * Method called to create the default loading screen. * This can be overridden in your own app. * @param canvas The rendering canvas element * @returns The loading screen */ // eslint-disable-next-line @typescript-eslint/no-unused-vars static DefaultLoadingScreenFactory(e) { throw qn("LoadingScreen"); } get _supportsHardwareTextureRescaling() { return !!Ge._RescalePostProcessFactory; } /** * Gets the performance monitor attached to this engine * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#engineinstrumentation */ get performanceMonitor() { return this._performanceMonitor; } /** * (WebGPU only) True (default) to be in compatibility mode, meaning rendering all existing scenes without artifacts (same rendering than WebGL). * Setting the property to false will improve performances but may not work in some scenes if some precautions are not taken. * See https://doc.babylonjs.com/setup/support/webGPU/webGPUOptimization/webGPUNonCompatibilityMode for more details */ get compatibilityMode() { return this._compatibilityMode; } set compatibilityMode(e) { this._compatibilityMode = !0; } // Events /** * Gets the HTML element used to attach event listeners * @returns a HTML element */ getInputElement() { return this._renderingCanvas; } /** * Creates a new engine * @param canvasOrContext defines the canvas or WebGL context to use for rendering. If you provide a WebGL context, Babylon.js will not hook events on the canvas (like pointers, keyboards, etc...) so no event observables will be available. This is mostly used when Babylon.js is used as a plugin on a system which already used the WebGL context * @param antialias defines enable antialiasing (default: false) * @param options defines further options to be sent to the getContext() function * @param adaptToDeviceRatio defines whether to adapt to the device's viewport characteristics (default: false) */ constructor(e, t, r, n = !1) { if (super(e, t, r, n), this.enableOfflineSupport = !1, this.disableManifestCheck = !1, this.disableContextMenu = !0, this.scenes = [], this._virtualScenes = new Array(), this.onNewSceneAddedObservable = new Oe(), this.postProcesses = [], this.isPointerLock = !1, this.onResizeObservable = new Oe(), this.onCanvasBlurObservable = new Oe(), this.onCanvasFocusObservable = new Oe(), this.onCanvasPointerOutObservable = new Oe(), this.onBeginFrameObservable = new Oe(), this.customAnimationFrameRequester = null, this.onEndFrameObservable = new Oe(), this.onBeforeShaderCompilationObservable = new Oe(), this.onAfterShaderCompilationObservable = new Oe(), this._deterministicLockstep = !1, this._lockstepMaxSteps = 4, this._timeStep = 1 / 60, this._fps = 60, this._deltaTime = 0, this._drawCalls = new v9(), this.canvasTabIndex = 1, this.disablePerformanceMonitorInBackground = !1, this._performanceMonitor = new iee(), this._compatibilityMode = !0, this.currentRenderPassId = 0, this._renderPassNames = ["main"], Ge.Instances.push(this), !!e && (this._features.supportRenderPasses = !0, r = this._creationOptions, e.getContext)) { const i = e; this._sharedInit(i); } } _initGLContext() { super._initGLContext(), this._rescalePostProcess = null; } /** * Shared initialization across engines types. * @param canvas The canvas associated with this instance of the engine. */ _sharedInit(e) { super._sharedInit(e), this._onCanvasFocus = () => { this.onCanvasFocusObservable.notifyObservers(this); }, this._onCanvasBlur = () => { this.onCanvasBlurObservable.notifyObservers(this); }, this._onCanvasContextMenu = (r) => { this.disableContextMenu && r.preventDefault(); }, e.addEventListener("focus", this._onCanvasFocus), e.addEventListener("blur", this._onCanvasBlur), e.addEventListener("contextmenu", this._onCanvasContextMenu), this._onBlur = () => { this.disablePerformanceMonitorInBackground && this._performanceMonitor.disable(), this._windowIsBackground = !0; }, this._onFocus = () => { this.disablePerformanceMonitorInBackground && this._performanceMonitor.enable(), this._windowIsBackground = !1; }, this._onCanvasPointerOut = (r) => { document.elementFromPoint(r.clientX, r.clientY) !== e && this.onCanvasPointerOutObservable.notifyObservers(r); }; const t = this.getHostWindow(); t && typeof t.addEventListener == "function" && (t.addEventListener("blur", this._onBlur), t.addEventListener("focus", this._onFocus)), e.addEventListener("pointerout", this._onCanvasPointerOut), this._creationOptions.doNotHandleTouchAction || this._disableTouchAction(), !Ge.audioEngine && this._creationOptions.audioEngine && Ge.AudioEngineFactory && (Ge.audioEngine = Ge.AudioEngineFactory(this.getRenderingCanvas(), this.getAudioContext(), this.getAudioDestination())), $w() && (this._onFullscreenChange = () => { this.isFullscreen = !!document.fullscreenElement, this.isFullscreen && this._pointerLockRequested && e && Ge._RequestPointerlock(e); }, document.addEventListener("fullscreenchange", this._onFullscreenChange, !1), document.addEventListener("webkitfullscreenchange", this._onFullscreenChange, !1), this._onPointerLockChange = () => { this.isPointerLock = document.pointerLockElement === e; }, document.addEventListener("pointerlockchange", this._onPointerLockChange, !1), document.addEventListener("webkitpointerlockchange", this._onPointerLockChange, !1)), this.enableOfflineSupport = Ge.OfflineProviderFactory !== void 0, this._deterministicLockstep = !!this._creationOptions.deterministicLockstep, this._lockstepMaxSteps = this._creationOptions.lockstepMaxSteps || 0, this._timeStep = this._creationOptions.timeStep || 1 / 60; } /** @internal */ _verifyPointerLock() { var e; (e = this._onPointerLockChange) === null || e === void 0 || e.call(this); } /** * Gets current aspect ratio * @param viewportOwner defines the camera to use to get the aspect ratio * @param useScreen defines if screen size must be used (or the current render target if any) * @returns a number defining the aspect ratio */ getAspectRatio(e, t = !1) { const r = e.viewport; return this.getRenderWidth(t) * r.width / (this.getRenderHeight(t) * r.height); } /** * Gets current screen aspect ratio * @returns a number defining the aspect ratio */ getScreenAspectRatio() { return this.getRenderWidth(!0) / this.getRenderHeight(!0); } /** * Gets the client rect of the HTML canvas attached with the current webGL context * @returns a client rectangle */ getRenderingCanvasClientRect() { return this._renderingCanvas ? this._renderingCanvas.getBoundingClientRect() : null; } /** * Gets the client rect of the HTML element used for events * @returns a client rectangle */ getInputElementClientRect() { return this._renderingCanvas ? this.getInputElement().getBoundingClientRect() : null; } /** * Gets a boolean indicating that the engine is running in deterministic lock step mode * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep * @returns true if engine is in deterministic lock step mode */ isDeterministicLockStep() { return this._deterministicLockstep; } /** * Gets the max steps when engine is running in deterministic lock step * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep * @returns the max steps */ getLockstepMaxSteps() { return this._lockstepMaxSteps; } /** * Returns the time in ms between steps when using deterministic lock step. * @returns time step in (ms) */ getTimeStep() { return this._timeStep * 1e3; } /** * Force the mipmap generation for the given render target texture * @param texture defines the render target texture to use * @param unbind defines whether or not to unbind the texture after generation. Defaults to true. */ generateMipMapsForCubemap(e, t = !0) { if (e.generateMipMaps) { const r = this._gl; this._bindTextureDirectly(r.TEXTURE_CUBE_MAP, e, !0), r.generateMipmap(r.TEXTURE_CUBE_MAP), t && this._bindTextureDirectly(r.TEXTURE_CUBE_MAP, null); } } /** States */ /** * Gets a boolean indicating if depth writing is enabled * @returns the current depth writing state */ getDepthWrite() { return this._depthCullingState.depthMask; } /** * Enable or disable depth writing * @param enable defines the state to set */ setDepthWrite(e) { this._depthCullingState.depthMask = e; } /** * Gets a boolean indicating if stencil buffer is enabled * @returns the current stencil buffer state */ getStencilBuffer() { return this._stencilState.stencilTest; } /** * Enable or disable the stencil buffer * @param enable defines if the stencil buffer must be enabled or disabled */ setStencilBuffer(e) { this._stencilState.stencilTest = e; } /** * Gets the current stencil mask * @returns a number defining the new stencil mask to use */ getStencilMask() { return this._stencilState.stencilMask; } /** * Sets the current stencil mask * @param mask defines the new stencil mask to use */ setStencilMask(e) { this._stencilState.stencilMask = e; } /** * Gets the current stencil function * @returns a number defining the stencil function to use */ getStencilFunction() { return this._stencilState.stencilFunc; } /** * Gets the current stencil reference value * @returns a number defining the stencil reference value to use */ getStencilFunctionReference() { return this._stencilState.stencilFuncRef; } /** * Gets the current stencil mask * @returns a number defining the stencil mask to use */ getStencilFunctionMask() { return this._stencilState.stencilFuncMask; } /** * Sets the current stencil function * @param stencilFunc defines the new stencil function to use */ setStencilFunction(e) { this._stencilState.stencilFunc = e; } /** * Sets the current stencil reference * @param reference defines the new stencil reference to use */ setStencilFunctionReference(e) { this._stencilState.stencilFuncRef = e; } /** * Sets the current stencil mask * @param mask defines the new stencil mask to use */ setStencilFunctionMask(e) { this._stencilState.stencilFuncMask = e; } /** * Gets the current stencil operation when stencil fails * @returns a number defining stencil operation to use when stencil fails */ getStencilOperationFail() { return this._stencilState.stencilOpStencilFail; } /** * Gets the current stencil operation when depth fails * @returns a number defining stencil operation to use when depth fails */ getStencilOperationDepthFail() { return this._stencilState.stencilOpDepthFail; } /** * Gets the current stencil operation when stencil passes * @returns a number defining stencil operation to use when stencil passes */ getStencilOperationPass() { return this._stencilState.stencilOpStencilDepthPass; } /** * Sets the stencil operation to use when stencil fails * @param operation defines the stencil operation to use when stencil fails */ setStencilOperationFail(e) { this._stencilState.stencilOpStencilFail = e; } /** * Sets the stencil operation to use when depth fails * @param operation defines the stencil operation to use when depth fails */ setStencilOperationDepthFail(e) { this._stencilState.stencilOpDepthFail = e; } /** * Sets the stencil operation to use when stencil passes * @param operation defines the stencil operation to use when stencil passes */ setStencilOperationPass(e) { this._stencilState.stencilOpStencilDepthPass = e; } /** * Sets a boolean indicating if the dithering state is enabled or disabled * @param value defines the dithering state */ setDitheringState(e) { e ? this._gl.enable(this._gl.DITHER) : this._gl.disable(this._gl.DITHER); } /** * Sets a boolean indicating if the rasterizer state is enabled or disabled * @param value defines the rasterizer state */ setRasterizerState(e) { e ? this._gl.disable(this._gl.RASTERIZER_DISCARD) : this._gl.enable(this._gl.RASTERIZER_DISCARD); } /** * Gets the current depth function * @returns a number defining the depth function */ getDepthFunction() { return this._depthCullingState.depthFunc; } /** * Sets the current depth function * @param depthFunc defines the function to use */ setDepthFunction(e) { this._depthCullingState.depthFunc = e; } /** * Sets the current depth function to GREATER */ setDepthFunctionToGreater() { this.setDepthFunction(516); } /** * Sets the current depth function to GEQUAL */ setDepthFunctionToGreaterOrEqual() { this.setDepthFunction(518); } /** * Sets the current depth function to LESS */ setDepthFunctionToLess() { this.setDepthFunction(513); } /** * Sets the current depth function to LEQUAL */ setDepthFunctionToLessOrEqual() { this.setDepthFunction(515); } /** * Caches the state of the stencil buffer */ cacheStencilState() { this._cachedStencilBuffer = this.getStencilBuffer(), this._cachedStencilFunction = this.getStencilFunction(), this._cachedStencilMask = this.getStencilMask(), this._cachedStencilOperationPass = this.getStencilOperationPass(), this._cachedStencilOperationFail = this.getStencilOperationFail(), this._cachedStencilOperationDepthFail = this.getStencilOperationDepthFail(), this._cachedStencilReference = this.getStencilFunctionReference(); } /** * Restores the state of the stencil buffer */ restoreStencilState() { this.setStencilFunction(this._cachedStencilFunction), this.setStencilMask(this._cachedStencilMask), this.setStencilBuffer(this._cachedStencilBuffer), this.setStencilOperationPass(this._cachedStencilOperationPass), this.setStencilOperationFail(this._cachedStencilOperationFail), this.setStencilOperationDepthFail(this._cachedStencilOperationDepthFail), this.setStencilFunctionReference(this._cachedStencilReference); } /** * Directly set the WebGL Viewport * @param x defines the x coordinate of the viewport (in screen space) * @param y defines the y coordinate of the viewport (in screen space) * @param width defines the width of the viewport (in screen space) * @param height defines the height of the viewport (in screen space) * @returns the current viewport Object (if any) that is being replaced by this call. You can restore this viewport later on to go back to the original state */ setDirectViewport(e, t, r, n) { const i = this._cachedViewport; return this._cachedViewport = null, this._viewport(e, t, r, n), i; } /** * Executes a scissor clear (ie. a clear on a specific portion of the screen) * @param x defines the x-coordinate of the bottom left corner of the clear rectangle * @param y defines the y-coordinate of the corner of the clear rectangle * @param width defines the width of the clear rectangle * @param height defines the height of the clear rectangle * @param clearColor defines the clear color */ scissorClear(e, t, r, n, i) { this.enableScissor(e, t, r, n), this.clear(i, !0, !0, !0), this.disableScissor(); } /** * Enable scissor test on a specific rectangle (ie. render will only be executed on a specific portion of the screen) * @param x defines the x-coordinate of the bottom left corner of the clear rectangle * @param y defines the y-coordinate of the corner of the clear rectangle * @param width defines the width of the clear rectangle * @param height defines the height of the clear rectangle */ enableScissor(e, t, r, n) { const i = this._gl; i.enable(i.SCISSOR_TEST), i.scissor(e, t, r, n); } /** * Disable previously set scissor test rectangle */ disableScissor() { const e = this._gl; e.disable(e.SCISSOR_TEST); } /** * @internal */ _reportDrawCall(e = 1) { this._drawCalls.addCount(e, !1); } /** * @internal */ _loadFileAsync(e, t, r) { return new Promise((n, i) => { this._loadFile(e, (s) => { n(s); }, void 0, t, r, (s, a) => { i(a); }); }); } /** * Gets the source code of the vertex shader associated with a specific webGL program * @param program defines the program to use * @returns a string containing the source code of the vertex shader associated with the program */ getVertexShaderSource(e) { const t = this._gl.getAttachedShaders(e); return t ? this._gl.getShaderSource(t[0]) : null; } /** * Gets the source code of the fragment shader associated with a specific webGL program * @param program defines the program to use * @returns a string containing the source code of the fragment shader associated with the program */ getFragmentShaderSource(e) { const t = this._gl.getAttachedShaders(e); return t ? this._gl.getShaderSource(t[1]) : null; } /** * Sets a depth stencil texture from a render target to the according uniform. * @param channel The texture channel * @param uniform The uniform to set * @param texture The render target texture containing the depth stencil texture to apply * @param name The texture name */ setDepthStencilTexture(e, t, r, n) { e !== void 0 && (t && (this._boundUniforms[e] = t), !r || !r.depthStencilTexture ? this._setTexture(e, null, void 0, void 0, n) : this._setTexture(e, r, !1, !0, n)); } /** * Sets a texture to the webGL context from a postprocess * @param channel defines the channel to use * @param postProcess defines the source postprocess * @param name name of the channel */ setTextureFromPostProcess(e, t, r) { var n; let i = null; t && (t._forcedOutputTexture ? i = t._forcedOutputTexture : t._textures.data[t._currentRenderTextureInd] && (i = t._textures.data[t._currentRenderTextureInd])), this._bindTexture(e, (n = i == null ? void 0 : i.texture) !== null && n !== void 0 ? n : null, r); } /** * Binds the output of the passed in post process to the texture channel specified * @param channel The channel the texture should be bound to * @param postProcess The post process which's output should be bound * @param name name of the channel */ setTextureFromPostProcessOutput(e, t, r) { var n, i; this._bindTexture(e, (i = (n = t == null ? void 0 : t._outputTexture) === null || n === void 0 ? void 0 : n.texture) !== null && i !== void 0 ? i : null, r); } _rebuildBuffers() { for (const e of this.scenes) e.resetCachedMaterial(), e._rebuildGeometries(), e._rebuildTextures(); for (const e of this._virtualScenes) e.resetCachedMaterial(), e._rebuildGeometries(), e._rebuildTextures(); super._rebuildBuffers(); } /** @internal */ _renderFrame() { for (let e = 0; e < this._activeRenderLoops.length; e++) { const t = this._activeRenderLoops[e]; t(); } } _cancelFrame() { if (this._renderingQueueLaunched && this.customAnimationFrameRequester) { this._renderingQueueLaunched = !1; const { cancelAnimationFrame: e } = this.customAnimationFrameRequester; e && e(this.customAnimationFrameRequester.requestID); } else super._cancelFrame(); } _renderLoop() { if (!this._contextWasLost) { let e = !0; (this.isDisposed || !this.renderEvenInBackground && this._windowIsBackground) && (e = !1), e && (this.beginFrame(), this._renderViews() || this._renderFrame(), this.endFrame()); } this._activeRenderLoops.length > 0 ? this.customAnimationFrameRequester ? (this.customAnimationFrameRequester.requestID = this._queueNewFrame(this.customAnimationFrameRequester.renderFunction || this._boundRenderFunction, this.customAnimationFrameRequester), this._frameHandler = this.customAnimationFrameRequester.requestID) : this._frameHandler = this._queueNewFrame(this._boundRenderFunction, this.getHostWindow()) : this._renderingQueueLaunched = !1; } /** @internal */ _renderViews() { return !1; } /** * Toggle full screen mode * @param requestPointerLock defines if a pointer lock should be requested from the user */ switchFullscreen(e) { this.isFullscreen ? this.exitFullscreen() : this.enterFullscreen(e); } /** * Enters full screen mode * @param requestPointerLock defines if a pointer lock should be requested from the user */ enterFullscreen(e) { this.isFullscreen || (this._pointerLockRequested = e, this._renderingCanvas && Ge._RequestFullscreen(this._renderingCanvas)); } /** * Exits full screen mode */ exitFullscreen() { this.isFullscreen && Ge._ExitFullscreen(); } /** * Enters Pointerlock mode */ enterPointerlock() { this._renderingCanvas && Ge._RequestPointerlock(this._renderingCanvas); } /** * Exits Pointerlock mode */ exitPointerlock() { Ge._ExitPointerlock(); } /** * Begin a new frame */ beginFrame() { this._measureFps(), this.onBeginFrameObservable.notifyObservers(this), super.beginFrame(); } /** * End the current frame */ endFrame() { super.endFrame(), this.onEndFrameObservable.notifyObservers(this); } /** * Force a specific size of the canvas * @param width defines the new canvas' width * @param height defines the new canvas' height * @param forceSetSize true to force setting the sizes of the underlying canvas * @returns true if the size was changed */ setSize(e, t, r = !1) { if (!this._renderingCanvas || !super.setSize(e, t, r)) return !1; if (this.scenes) { for (let n = 0; n < this.scenes.length; n++) { const i = this.scenes[n]; for (let s = 0; s < i.cameras.length; s++) { const a = i.cameras[s]; a._currentRenderId = 0; } } this.onResizeObservable.hasObservers() && this.onResizeObservable.notifyObservers(this); } return !0; } _deletePipelineContext(e) { const t = e; t && t.program && t.transformFeedback && (this.deleteTransformFeedback(t.transformFeedback), t.transformFeedback = null), super._deletePipelineContext(e); } createShaderProgram(e, t, r, n, i, s = null) { i = i || this._gl, this.onBeforeShaderCompilationObservable.notifyObservers(this); const a = super.createShaderProgram(e, t, r, n, i, s); return this.onAfterShaderCompilationObservable.notifyObservers(this), a; } _createShaderProgram(e, t, r, n, i = null) { const s = n.createProgram(); if (e.program = s, !s) throw new Error("Unable to create program"); if (n.attachShader(s, t), n.attachShader(s, r), this.webGLVersion > 1 && i) { const a = this.createTransformFeedback(); this.bindTransformFeedback(a), this.setTranformFeedbackVaryings(s, i), e.transformFeedback = a; } return n.linkProgram(s), this.webGLVersion > 1 && i && this.bindTransformFeedback(null), e.context = n, e.vertexShader = t, e.fragmentShader = r, e.isParallelCompiled || this._finalizePipelineContext(e), s; } /** * @internal */ _releaseTexture(e) { super._releaseTexture(e); } /** * @internal */ _releaseRenderTargetWrapper(e) { super._releaseRenderTargetWrapper(e), this.scenes.forEach((t) => { t.postProcesses.forEach((r) => { r._outputTexture === e && (r._outputTexture = null); }), t.cameras.forEach((r) => { r._postProcesses.forEach((n) => { n && n._outputTexture === e && (n._outputTexture = null); }); }); }); } /** * Gets the names of the render passes that are currently created * @returns list of the render pass names */ getRenderPassNames() { return this._renderPassNames; } /** * Gets the name of the current render pass * @returns name of the current render pass */ getCurrentRenderPassName() { return this._renderPassNames[this.currentRenderPassId]; } /** * Creates a render pass id * @param name Name of the render pass (for debug purpose only) * @returns the id of the new render pass */ createRenderPassId(e) { const t = ++Ge._RenderPassIdCounter; return this._renderPassNames[t] = e ?? "NONAME", t; } /** * Releases a render pass id * @param id id of the render pass to release */ releaseRenderPassId(e) { this._renderPassNames[e] = void 0; for (let t = 0; t < this.scenes.length; ++t) { const r = this.scenes[t]; for (let n = 0; n < r.meshes.length; ++n) { const i = r.meshes[n]; if (i.subMeshes) for (let s = 0; s < i.subMeshes.length; ++s) i.subMeshes[s]._removeDrawWrapper(e); } } } /** * @internal * Rescales a texture * @param source input texture * @param destination destination texture * @param scene scene to use to render the resize * @param internalFormat format to use when resizing * @param onComplete callback to be called when resize has completed */ _rescaleTexture(e, t, r, n, i) { this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_MAG_FILTER, this._gl.LINEAR), this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_MIN_FILTER, this._gl.LINEAR), this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_WRAP_S, this._gl.CLAMP_TO_EDGE), this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_WRAP_T, this._gl.CLAMP_TO_EDGE); const s = this.createRenderTargetTexture({ width: t.width, height: t.height }, { generateMipMaps: !1, type: 0, samplingMode: 2, generateDepthBuffer: !1, generateStencilBuffer: !1 }); !this._rescalePostProcess && Ge._RescalePostProcessFactory && (this._rescalePostProcess = Ge._RescalePostProcessFactory(this)), this._rescalePostProcess && (this._rescalePostProcess.externalTextureSamplerBinding = !0, this._rescalePostProcess.getEffect().executeWhenCompiled(() => { this._rescalePostProcess.onApply = function(f) { f._bindTexture("textureSampler", e); }; let a = r; a || (a = this.scenes[this.scenes.length - 1]), a.postProcessManager.directRender([this._rescalePostProcess], s, !0), this._bindTextureDirectly(this._gl.TEXTURE_2D, t, !0), this._gl.copyTexImage2D(this._gl.TEXTURE_2D, 0, n, 0, 0, t.width, t.height, 0), this.unBindFramebuffer(s), s.dispose(), i && i(); })); } // FPS /** * Gets the current framerate * @returns a number representing the framerate */ getFps() { return this._fps; } /** * Gets the time spent between current and previous frame * @returns a number representing the delta time in ms */ getDeltaTime() { return this._deltaTime; } _measureFps() { this._performanceMonitor.sampleFrame(), this._fps = this._performanceMonitor.averageFPS, this._deltaTime = this._performanceMonitor.instantaneousFrameTime || 0; } /** * Wraps an external web gl texture in a Babylon texture. * @param texture defines the external texture * @param hasMipMaps defines whether the external texture has mip maps (default: false) * @param samplingMode defines the sampling mode for the external texture (default: 3) * @param width defines the width for the external texture (default: 0) * @param height defines the height for the external texture (default: 0) * @returns the babylon internal texture */ wrapWebGLTexture(e, t = !1, r = 3, n = 0, i = 0) { const s = new jm(e, this._gl), a = new As(this, ri.Unknown, !0); return a._hardwareTexture = s, a.baseWidth = n, a.baseHeight = i, a.width = n, a.height = i, a.isReady = !0, a.useMipMaps = t, this.updateTextureSamplingMode(r, a), a; } /** * @internal */ _uploadImageToTexture(e, t, r = 0, n = 0) { const i = this._gl, s = this._getWebGLTextureType(e.type), a = this._getInternalFormat(e.format), f = this._getRGBABufferInternalSizedFormat(e.type, a), o = e.isCube ? i.TEXTURE_CUBE_MAP : i.TEXTURE_2D; this._bindTextureDirectly(o, e, !0), this._unpackFlipY(e.invertY); let d = i.TEXTURE_2D; e.isCube && (d = i.TEXTURE_CUBE_MAP_POSITIVE_X + r), i.texImage2D(d, n, f, a, s, t), this._bindTextureDirectly(o, null, !0); } /** * Updates a depth texture Comparison Mode and Function. * If the comparison Function is equal to 0, the mode will be set to none. * Otherwise, this only works in webgl 2 and requires a shadow sampler in the shader. * @param texture The texture to set the comparison function for * @param comparisonFunction The comparison function to set, 0 if no comparison required */ updateTextureComparisonFunction(e, t) { if (this.webGLVersion === 1) { Se.Error("WebGL 1 does not support texture comparison."); return; } const r = this._gl; e.isCube ? (this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, e, !0), t === 0 ? (r.texParameteri(r.TEXTURE_CUBE_MAP, r.TEXTURE_COMPARE_FUNC, 515), r.texParameteri(r.TEXTURE_CUBE_MAP, r.TEXTURE_COMPARE_MODE, r.NONE)) : (r.texParameteri(r.TEXTURE_CUBE_MAP, r.TEXTURE_COMPARE_FUNC, t), r.texParameteri(r.TEXTURE_CUBE_MAP, r.TEXTURE_COMPARE_MODE, r.COMPARE_REF_TO_TEXTURE)), this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, null)) : (this._bindTextureDirectly(this._gl.TEXTURE_2D, e, !0), t === 0 ? (r.texParameteri(r.TEXTURE_2D, r.TEXTURE_COMPARE_FUNC, 515), r.texParameteri(r.TEXTURE_2D, r.TEXTURE_COMPARE_MODE, r.NONE)) : (r.texParameteri(r.TEXTURE_2D, r.TEXTURE_COMPARE_FUNC, t), r.texParameteri(r.TEXTURE_2D, r.TEXTURE_COMPARE_MODE, r.COMPARE_REF_TO_TEXTURE)), this._bindTextureDirectly(this._gl.TEXTURE_2D, null)), e._comparisonFunction = t; } /** * Creates a webGL buffer to use with instantiation * @param capacity defines the size of the buffer * @returns the webGL buffer */ createInstancesBuffer(e) { const t = this._gl.createBuffer(); if (!t) throw new Error("Unable to create instance buffer"); const r = new jS(t); return r.capacity = e, this.bindArrayBuffer(r), this._gl.bufferData(this._gl.ARRAY_BUFFER, e, this._gl.DYNAMIC_DRAW), r.references = 1, r; } /** * Delete a webGL buffer used with instantiation * @param buffer defines the webGL buffer to delete */ deleteInstancesBuffer(e) { this._gl.deleteBuffer(e); } _clientWaitAsync(e, t = 0, r = 10) { const n = this._gl; return new Promise((i, s) => { const a = () => { const f = n.clientWaitSync(e, t, 0); if (f == n.WAIT_FAILED) { s(); return; } if (f == n.TIMEOUT_EXPIRED) { setTimeout(a, r); return; } i(); }; a(); }); } /** * @internal */ _readPixelsAsync(e, t, r, n, i, s, a) { if (this._webGLVersion < 2) throw new Error("_readPixelsAsync only work on WebGL2+"); const f = this._gl, o = f.createBuffer(); f.bindBuffer(f.PIXEL_PACK_BUFFER, o), f.bufferData(f.PIXEL_PACK_BUFFER, a.byteLength, f.STREAM_READ), f.readPixels(e, t, r, n, i, s, 0), f.bindBuffer(f.PIXEL_PACK_BUFFER, null); const d = f.fenceSync(f.SYNC_GPU_COMMANDS_COMPLETE, 0); return d ? (f.flush(), this._clientWaitAsync(d, 0, 10).then(() => (f.deleteSync(d), f.bindBuffer(f.PIXEL_PACK_BUFFER, o), f.getBufferSubData(f.PIXEL_PACK_BUFFER, 0, a), f.bindBuffer(f.PIXEL_PACK_BUFFER, null), f.deleteBuffer(o), a))) : null; } dispose() { for (this.hideLoadingUI(), this.onNewSceneAddedObservable.clear(); this.postProcesses.length; ) this.postProcesses[0].dispose(); for (this._rescalePostProcess && this._rescalePostProcess.dispose(); this.scenes.length; ) this.scenes[0].dispose(); for (; this._virtualScenes.length; ) this._virtualScenes[0].dispose(); gr.Instances.length === 1 && Ge.audioEngine && (Ge.audioEngine.dispose(), Ge.audioEngine = null); const e = this.getHostWindow(); e && typeof e.removeEventListener == "function" && (e.removeEventListener("blur", this._onBlur), e.removeEventListener("focus", this._onFocus)), this._renderingCanvas && (this._renderingCanvas.removeEventListener("focus", this._onCanvasFocus), this._renderingCanvas.removeEventListener("blur", this._onCanvasBlur), this._renderingCanvas.removeEventListener("pointerout", this._onCanvasPointerOut), this._renderingCanvas.removeEventListener("contextmenu", this._onCanvasContextMenu)), $w() && (document.removeEventListener("fullscreenchange", this._onFullscreenChange), document.removeEventListener("mozfullscreenchange", this._onFullscreenChange), document.removeEventListener("webkitfullscreenchange", this._onFullscreenChange), document.removeEventListener("msfullscreenchange", this._onFullscreenChange), document.removeEventListener("pointerlockchange", this._onPointerLockChange), document.removeEventListener("mspointerlockchange", this._onPointerLockChange), document.removeEventListener("mozpointerlockchange", this._onPointerLockChange), document.removeEventListener("webkitpointerlockchange", this._onPointerLockChange)), super.dispose(); const t = gr.Instances.indexOf(this); t >= 0 && gr.Instances.splice(t, 1), Ge.Instances.length || gr.OnEnginesDisposedObservable.notifyObservers(this), this.onResizeObservable.clear(), this.onCanvasBlurObservable.clear(), this.onCanvasFocusObservable.clear(), this.onCanvasPointerOutObservable.clear(), this.onBeginFrameObservable.clear(), this.onEndFrameObservable.clear(); } _disableTouchAction() { !this._renderingCanvas || !this._renderingCanvas.setAttribute || (this._renderingCanvas.setAttribute("touch-action", "none"), this._renderingCanvas.style.touchAction = "none", this._renderingCanvas.style.webkitTapHighlightColor = "transparent"); } // Loading screen /** * Display the loading screen * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen */ displayLoadingUI() { if (!u9()) return; const e = this.loadingScreen; e && e.displayLoadingUI(); } /** * Hide the loading screen * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen */ hideLoadingUI() { if (!u9()) return; const e = this._loadingScreen; e && e.hideLoadingUI(); } /** * Gets the current loading screen object * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen */ get loadingScreen() { return !this._loadingScreen && this._renderingCanvas && (this._loadingScreen = Ge.DefaultLoadingScreenFactory(this._renderingCanvas)), this._loadingScreen; } /** * Sets the current loading screen object * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen */ set loadingScreen(e) { this._loadingScreen = e; } /** * Sets the current loading screen text * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen */ set loadingUIText(e) { this.loadingScreen.loadingUIText = e; } /** * Sets the current loading screen background color * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen */ set loadingUIBackgroundColor(e) { this.loadingScreen.loadingUIBackgroundColor = e; } /** * creates and returns a new video element * @param constraints video constraints * @returns video element */ // eslint-disable-next-line @typescript-eslint/no-unused-vars createVideoElement(e) { return document.createElement("video"); } /** Pointerlock and fullscreen */ /** * Ask the browser to promote the current element to pointerlock mode * @param element defines the DOM element to promote */ static _RequestPointerlock(e) { if (e.requestPointerLock) { const t = e.requestPointerLock(); t instanceof Promise ? t.then(() => { e.focus(); }).catch(() => { }) : e.focus(); } } /** * Asks the browser to exit pointerlock mode */ static _ExitPointerlock() { document.exitPointerLock && document.exitPointerLock(); } /** * Ask the browser to promote the current element to fullscreen rendering mode * @param element defines the DOM element to promote */ static _RequestFullscreen(e) { const t = e.requestFullscreen || e.webkitRequestFullscreen; t && t.call(e); } /** * Asks the browser to exit fullscreen mode */ static _ExitFullscreen() { const e = document; document.exitFullscreen ? document.exitFullscreen() : e.webkitCancelFullScreen && e.webkitCancelFullScreen(); } /** * Get Font size information * @param font font name * @returns an object containing ascent, height and descent */ getFontOffset(e) { const t = document.createElement("span"); t.innerHTML = "Hg", t.setAttribute("style", `font: ${e} !important`); const r = document.createElement("div"); r.style.display = "inline-block", r.style.width = "1px", r.style.height = "0px", r.style.verticalAlign = "bottom"; const n = document.createElement("div"); n.style.whiteSpace = "nowrap", n.appendChild(t), n.appendChild(r), document.body.appendChild(n); let i = 0, s = 0; try { s = r.getBoundingClientRect().top - t.getBoundingClientRect().top, r.style.verticalAlign = "baseline", i = r.getBoundingClientRect().top - t.getBoundingClientRect().top; } finally { document.body.removeChild(n); } return { ascent: i, height: s, descent: s - i }; } } Ge.ALPHA_DISABLE = 0; Ge.ALPHA_ADD = 1; Ge.ALPHA_COMBINE = 2; Ge.ALPHA_SUBTRACT = 3; Ge.ALPHA_MULTIPLY = 4; Ge.ALPHA_MAXIMIZED = 5; Ge.ALPHA_ONEONE = 6; Ge.ALPHA_PREMULTIPLIED = 7; Ge.ALPHA_PREMULTIPLIED_PORTERDUFF = 8; Ge.ALPHA_INTERPOLATE = 9; Ge.ALPHA_SCREENMODE = 10; Ge.DELAYLOADSTATE_NONE = 0; Ge.DELAYLOADSTATE_LOADED = 1; Ge.DELAYLOADSTATE_LOADING = 2; Ge.DELAYLOADSTATE_NOTLOADED = 4; Ge.NEVER = 512; Ge.ALWAYS = 519; Ge.LESS = 513; Ge.EQUAL = 514; Ge.LEQUAL = 515; Ge.GREATER = 516; Ge.GEQUAL = 518; Ge.NOTEQUAL = 517; Ge.KEEP = 7680; Ge.REPLACE = 7681; Ge.INCR = 7682; Ge.DECR = 7683; Ge.INVERT = 5386; Ge.INCR_WRAP = 34055; Ge.DECR_WRAP = 34056; Ge.TEXTURE_CLAMP_ADDRESSMODE = 0; Ge.TEXTURE_WRAP_ADDRESSMODE = 1; Ge.TEXTURE_MIRROR_ADDRESSMODE = 2; Ge.TEXTUREFORMAT_ALPHA = 0; Ge.TEXTUREFORMAT_LUMINANCE = 1; Ge.TEXTUREFORMAT_LUMINANCE_ALPHA = 2; Ge.TEXTUREFORMAT_RGB = 4; Ge.TEXTUREFORMAT_RGBA = 5; Ge.TEXTUREFORMAT_RED = 6; Ge.TEXTUREFORMAT_R = 6; Ge.TEXTUREFORMAT_RG = 7; Ge.TEXTUREFORMAT_RED_INTEGER = 8; Ge.TEXTUREFORMAT_R_INTEGER = 8; Ge.TEXTUREFORMAT_RG_INTEGER = 9; Ge.TEXTUREFORMAT_RGB_INTEGER = 10; Ge.TEXTUREFORMAT_RGBA_INTEGER = 11; Ge.TEXTURETYPE_UNSIGNED_BYTE = 0; Ge.TEXTURETYPE_UNSIGNED_INT = 0; Ge.TEXTURETYPE_FLOAT = 1; Ge.TEXTURETYPE_HALF_FLOAT = 2; Ge.TEXTURETYPE_BYTE = 3; Ge.TEXTURETYPE_SHORT = 4; Ge.TEXTURETYPE_UNSIGNED_SHORT = 5; Ge.TEXTURETYPE_INT = 6; Ge.TEXTURETYPE_UNSIGNED_INTEGER = 7; Ge.TEXTURETYPE_UNSIGNED_SHORT_4_4_4_4 = 8; Ge.TEXTURETYPE_UNSIGNED_SHORT_5_5_5_1 = 9; Ge.TEXTURETYPE_UNSIGNED_SHORT_5_6_5 = 10; Ge.TEXTURETYPE_UNSIGNED_INT_2_10_10_10_REV = 11; Ge.TEXTURETYPE_UNSIGNED_INT_24_8 = 12; Ge.TEXTURETYPE_UNSIGNED_INT_10F_11F_11F_REV = 13; Ge.TEXTURETYPE_UNSIGNED_INT_5_9_9_9_REV = 14; Ge.TEXTURETYPE_FLOAT_32_UNSIGNED_INT_24_8_REV = 15; Ge.TEXTURE_NEAREST_SAMPLINGMODE = 1; Ge.TEXTURE_BILINEAR_SAMPLINGMODE = 2; Ge.TEXTURE_TRILINEAR_SAMPLINGMODE = 3; Ge.TEXTURE_NEAREST_NEAREST_MIPLINEAR = 8; Ge.TEXTURE_LINEAR_LINEAR_MIPNEAREST = 11; Ge.TEXTURE_LINEAR_LINEAR_MIPLINEAR = 3; Ge.TEXTURE_NEAREST_NEAREST_MIPNEAREST = 4; Ge.TEXTURE_NEAREST_LINEAR_MIPNEAREST = 5; Ge.TEXTURE_NEAREST_LINEAR_MIPLINEAR = 6; Ge.TEXTURE_NEAREST_LINEAR = 7; Ge.TEXTURE_NEAREST_NEAREST = 1; Ge.TEXTURE_LINEAR_NEAREST_MIPNEAREST = 9; Ge.TEXTURE_LINEAR_NEAREST_MIPLINEAR = 10; Ge.TEXTURE_LINEAR_LINEAR = 2; Ge.TEXTURE_LINEAR_NEAREST = 12; Ge.TEXTURE_EXPLICIT_MODE = 0; Ge.TEXTURE_SPHERICAL_MODE = 1; Ge.TEXTURE_PLANAR_MODE = 2; Ge.TEXTURE_CUBIC_MODE = 3; Ge.TEXTURE_PROJECTION_MODE = 4; Ge.TEXTURE_SKYBOX_MODE = 5; Ge.TEXTURE_INVCUBIC_MODE = 6; Ge.TEXTURE_EQUIRECTANGULAR_MODE = 7; Ge.TEXTURE_FIXED_EQUIRECTANGULAR_MODE = 8; Ge.TEXTURE_FIXED_EQUIRECTANGULAR_MIRRORED_MODE = 9; Ge.SCALEMODE_FLOOR = 1; Ge.SCALEMODE_NEAREST = 2; Ge.SCALEMODE_CEILING = 3; Ge._RescalePostProcessFactory = null; Ge._RenderPassIdCounter = 0; const u6e = he.Compose(S.One(), Ze.FromEulerAngles(0, Math.PI, 0), S.Zero()); class Hr extends Cs { /** * Gets or sets the billboard mode. Default is 0. * * | Value | Type | Description | * | --- | --- | --- | * | 0 | BILLBOARDMODE_NONE | | * | 1 | BILLBOARDMODE_X | | * | 2 | BILLBOARDMODE_Y | | * | 4 | BILLBOARDMODE_Z | | * | 7 | BILLBOARDMODE_ALL | | * */ get billboardMode() { return this._billboardMode; } set billboardMode(e) { this._billboardMode !== e && (this._billboardMode = e, this._cache.useBillboardPosition = (this._billboardMode & Hr.BILLBOARDMODE_USE_POSITION) !== 0, this._computeUseBillboardPath()); } /** * Gets or sets a boolean indicating that parent rotation should be preserved when using billboards. * This could be useful for glTF objects where parent rotation helps converting from right handed to left handed */ get preserveParentRotationForBillboard() { return this._preserveParentRotationForBillboard; } set preserveParentRotationForBillboard(e) { e !== this._preserveParentRotationForBillboard && (this._preserveParentRotationForBillboard = e, this._computeUseBillboardPath()); } _computeUseBillboardPath() { this._cache.useBillboardPath = this._billboardMode !== Hr.BILLBOARDMODE_NONE && !this.preserveParentRotationForBillboard; } /** * Gets or sets the distance of the object to max, often used by skybox */ get infiniteDistance() { return this._infiniteDistance; } set infiniteDistance(e) { this._infiniteDistance !== e && (this._infiniteDistance = e); } constructor(e, t = null, r = !0) { super(e, t), this._forward = new S(0, 0, 1), this._up = new S(0, 1, 0), this._right = new S(1, 0, 0), this._position = S.Zero(), this._rotation = S.Zero(), this._rotationQuaternion = null, this._scaling = S.One(), this._transformToBoneReferal = null, this._isAbsoluteSynced = !1, this._billboardMode = Hr.BILLBOARDMODE_NONE, this._preserveParentRotationForBillboard = !1, this.scalingDeterminant = 1, this._infiniteDistance = !1, this.ignoreNonUniformScaling = !1, this.reIntegrateRotationIntoRotationQuaternion = !1, this._poseMatrix = null, this._localMatrix = he.Zero(), this._usePivotMatrix = !1, this._absolutePosition = S.Zero(), this._absoluteScaling = S.Zero(), this._absoluteRotationQuaternion = Ze.Identity(), this._pivotMatrix = he.Identity(), this._postMultiplyPivotMatrix = !1, this._isWorldMatrixFrozen = !1, this._indexInSceneTransformNodesArray = -1, this.onAfterWorldMatrixUpdateObservable = new Oe(), this._nonUniformScaling = !1, r && this.getScene().addTransformNode(this); } /** * Gets a string identifying the name of the class * @returns "TransformNode" string */ getClassName() { return "TransformNode"; } /** * Gets or set the node position (default is (0.0, 0.0, 0.0)) */ get position() { return this._position; } set position(e) { this._position = e, this._isDirty = !0; } /** * return true if a pivot has been set * @returns true if a pivot matrix is used */ isUsingPivotMatrix() { return this._usePivotMatrix; } /** * return true if pivot matrix must be cancelled in the world matrix. When this parameter is set to true (default), the inverse of the pivot matrix is also applied at the end to cancel the transformation effect. */ isUsingPostMultiplyPivotMatrix() { return this._postMultiplyPivotMatrix; } /** * Gets or sets the rotation property : a Vector3 defining the rotation value in radians around each local axis X, Y, Z (default is (0.0, 0.0, 0.0)). * If rotation quaternion is set, this Vector3 will be ignored and copy from the quaternion */ get rotation() { return this._rotation; } set rotation(e) { this._rotation = e, this._rotationQuaternion = null, this._isDirty = !0; } /** * Gets or sets the scaling property : a Vector3 defining the node scaling along each local axis X, Y, Z (default is (1.0, 1.0, 1.0)). */ get scaling() { return this._scaling; } set scaling(e) { this._scaling = e, this._isDirty = !0; } /** * Gets or sets the rotation Quaternion property : this a Quaternion object defining the node rotation by using a unit quaternion (undefined by default, but can be null). * If set, only the rotationQuaternion is then used to compute the node rotation (ie. node.rotation will be ignored) */ get rotationQuaternion() { return this._rotationQuaternion; } set rotationQuaternion(e) { this._rotationQuaternion = e, e && this._rotation.setAll(0), this._isDirty = !0; } /** * The forward direction of that transform in world space. */ get forward() { return S.TransformNormalFromFloatsToRef(0, 0, this.getScene().useRightHandedSystem ? -1 : 1, this.getWorldMatrix(), this._forward), this._forward.normalize(); } /** * The up direction of that transform in world space. */ get up() { return S.TransformNormalFromFloatsToRef(0, 1, 0, this.getWorldMatrix(), this._up), this._up.normalize(); } /** * The right direction of that transform in world space. */ get right() { return S.TransformNormalFromFloatsToRef(this.getScene().useRightHandedSystem ? -1 : 1, 0, 0, this.getWorldMatrix(), this._right), this._right.normalize(); } /** * Copies the parameter passed Matrix into the mesh Pose matrix. * @param matrix the matrix to copy the pose from * @returns this TransformNode. */ updatePoseMatrix(e) { return this._poseMatrix ? (this._poseMatrix.copyFrom(e), this) : (this._poseMatrix = e.clone(), this); } /** * Returns the mesh Pose matrix. * @returns the pose matrix */ getPoseMatrix() { return this._poseMatrix || (this._poseMatrix = he.Identity()), this._poseMatrix; } /** @internal */ _isSynchronized() { const e = this._cache; return !(this._billboardMode !== e.billboardMode || this._billboardMode !== Hr.BILLBOARDMODE_NONE || e.pivotMatrixUpdated || this._infiniteDistance || this._position._isDirty || this._scaling._isDirty || this._rotationQuaternion && this._rotationQuaternion._isDirty || this._rotation._isDirty); } /** @internal */ _initCache() { super._initCache(); const e = this._cache; e.localMatrixUpdated = !1, e.billboardMode = -1, e.infiniteDistance = !1, e.useBillboardPosition = !1, e.useBillboardPath = !1; } /** * Returns the current mesh absolute position. * Returns a Vector3. */ get absolutePosition() { return this.getAbsolutePosition(); } /** * Returns the current mesh absolute scaling. * Returns a Vector3. */ get absoluteScaling() { return this._syncAbsoluteScalingAndRotation(), this._absoluteScaling; } /** * Returns the current mesh absolute rotation. * Returns a Quaternion. */ get absoluteRotationQuaternion() { return this._syncAbsoluteScalingAndRotation(), this._absoluteRotationQuaternion; } /** * Sets a new matrix to apply before all other transformation * @param matrix defines the transform matrix * @returns the current TransformNode */ setPreTransformMatrix(e) { return this.setPivotMatrix(e, !1); } /** * Sets a new pivot matrix to the current node * @param matrix defines the new pivot matrix to use * @param postMultiplyPivotMatrix defines if the pivot matrix must be cancelled in the world matrix. When this parameter is set to true (default), the inverse of the pivot matrix is also applied at the end to cancel the transformation effect * @returns the current TransformNode */ setPivotMatrix(e, t = !0) { return this._pivotMatrix.copyFrom(e), this._usePivotMatrix = !this._pivotMatrix.isIdentity(), this._cache.pivotMatrixUpdated = !0, this._postMultiplyPivotMatrix = t, this._postMultiplyPivotMatrix && (this._pivotMatrixInverse ? this._pivotMatrix.invertToRef(this._pivotMatrixInverse) : this._pivotMatrixInverse = he.Invert(this._pivotMatrix)), this; } /** * Returns the mesh pivot matrix. * Default : Identity. * @returns the matrix */ getPivotMatrix() { return this._pivotMatrix; } /** * Instantiate (when possible) or clone that node with its hierarchy * @param newParent defines the new parent to use for the instance (or clone) * @param options defines options to configure how copy is done * @param options.doNotInstantiate defines if the model must be instantiated or just cloned * @param onNewNodeCreated defines an option callback to call when a clone or an instance is created * @returns an instance (or a clone) of the current node with its hierarchy */ instantiateHierarchy(e = null, t, r) { const n = this.clone("Clone of " + (this.name || this.id), e || this.parent, !0); n && r && r(this, n); for (const i of this.getChildTransformNodes(!0)) i.instantiateHierarchy(n, t, r); return n; } /** * Prevents the World matrix to be computed any longer * @param newWorldMatrix defines an optional matrix to use as world matrix * @param decompose defines whether to decompose the given newWorldMatrix or directly assign * @returns the TransformNode. */ freezeWorldMatrix(e = null, t = !1) { return e ? t ? (this._rotation.setAll(0), this._rotationQuaternion = this._rotationQuaternion || Ze.Identity(), e.decompose(this._scaling, this._rotationQuaternion, this._position), this.computeWorldMatrix(!0)) : (this._worldMatrix = e, this._absolutePosition.copyFromFloats(this._worldMatrix.m[12], this._worldMatrix.m[13], this._worldMatrix.m[14]), this._afterComputeWorldMatrix()) : (this._isWorldMatrixFrozen = !1, this.computeWorldMatrix(!0)), this._isDirty = !1, this._isWorldMatrixFrozen = !0, this; } /** * Allows back the World matrix computation. * @returns the TransformNode. */ unfreezeWorldMatrix() { return this._isWorldMatrixFrozen = !1, this.computeWorldMatrix(!0), this; } /** * True if the World matrix has been frozen. */ get isWorldMatrixFrozen() { return this._isWorldMatrixFrozen; } /** * Returns the mesh absolute position in the World. * @returns a Vector3. */ getAbsolutePosition() { return this.computeWorldMatrix(), this._absolutePosition; } /** * Sets the mesh absolute position in the World from a Vector3 or an Array(3). * @param absolutePosition the absolute position to set * @returns the TransformNode. */ setAbsolutePosition(e) { if (!e) return this; let t, r, n; if (e.x === void 0) { if (arguments.length < 3) return this; t = arguments[0], r = arguments[1], n = arguments[2]; } else t = e.x, r = e.y, n = e.z; if (this.parent) { const i = ue.Matrix[0]; this.parent.getWorldMatrix().invertToRef(i), S.TransformCoordinatesFromFloatsToRef(t, r, n, i, this.position); } else this.position.x = t, this.position.y = r, this.position.z = n; return this._absolutePosition.copyFrom(e), this; } /** * Sets the mesh position in its local space. * @param vector3 the position to set in localspace * @returns the TransformNode. */ setPositionWithLocalVector(e) { return this.computeWorldMatrix(), this.position = S.TransformNormal(e, this._localMatrix), this; } /** * Returns the mesh position in the local space from the current World matrix values. * @returns a new Vector3. */ getPositionExpressedInLocalSpace() { this.computeWorldMatrix(); const e = ue.Matrix[0]; return this._localMatrix.invertToRef(e), S.TransformNormal(this.position, e); } /** * Translates the mesh along the passed Vector3 in its local space. * @param vector3 the distance to translate in localspace * @returns the TransformNode. */ locallyTranslate(e) { return this.computeWorldMatrix(!0), this.position = S.TransformCoordinates(e, this._localMatrix), this; } /** * Orients a mesh towards a target point. Mesh must be drawn facing user. * @param targetPoint the position (must be in same space as current mesh) to look at * @param yawCor optional yaw (y-axis) correction in radians * @param pitchCor optional pitch (x-axis) correction in radians * @param rollCor optional roll (z-axis) correction in radians * @param space the chosen space of the target * @returns the TransformNode. */ lookAt(e, t = 0, r = 0, n = 0, i = ai.LOCAL) { const s = Hr._LookAtVectorCache, a = i === ai.LOCAL ? this.position : this.getAbsolutePosition(); if (e.subtractToRef(a, s), this.setDirection(s, t, r, n), i === ai.WORLD && this.parent) if (this.rotationQuaternion) { const f = ue.Matrix[0]; this.rotationQuaternion.toRotationMatrix(f); const o = ue.Matrix[1]; this.parent.getWorldMatrix().getRotationMatrixToRef(o), o.invert(), f.multiplyToRef(o, f), this.rotationQuaternion.fromRotationMatrix(f); } else { const f = ue.Quaternion[0]; Ze.FromEulerVectorToRef(this.rotation, f); const o = ue.Matrix[0]; f.toRotationMatrix(o); const d = ue.Matrix[1]; this.parent.getWorldMatrix().getRotationMatrixToRef(d), d.invert(), o.multiplyToRef(d, o), f.fromRotationMatrix(o), f.toEulerAnglesToRef(this.rotation); } return this; } /** * Returns a new Vector3 that is the localAxis, expressed in the mesh local space, rotated like the mesh. * This Vector3 is expressed in the World space. * @param localAxis axis to rotate * @returns a new Vector3 that is the localAxis, expressed in the mesh local space, rotated like the mesh. */ getDirection(e) { const t = S.Zero(); return this.getDirectionToRef(e, t), t; } /** * Sets the Vector3 "result" as the rotated Vector3 "localAxis" in the same rotation than the mesh. * localAxis is expressed in the mesh local space. * result is computed in the World space from the mesh World matrix. * @param localAxis axis to rotate * @param result the resulting transformnode * @returns this TransformNode. */ getDirectionToRef(e, t) { return S.TransformNormalToRef(e, this.getWorldMatrix(), t), this; } /** * Sets this transform node rotation to the given local axis. * @param localAxis the axis in local space * @param yawCor optional yaw (y-axis) correction in radians * @param pitchCor optional pitch (x-axis) correction in radians * @param rollCor optional roll (z-axis) correction in radians * @returns this TransformNode */ setDirection(e, t = 0, r = 0, n = 0) { const i = -Math.atan2(e.z, e.x) + Math.PI / 2, s = Math.sqrt(e.x * e.x + e.z * e.z), a = -Math.atan2(e.y, s); return this.rotationQuaternion ? Ze.RotationYawPitchRollToRef(i + t, a + r, n, this.rotationQuaternion) : (this.rotation.x = a + r, this.rotation.y = i + t, this.rotation.z = n), this; } /** * Sets a new pivot point to the current node * @param point defines the new pivot point to use * @param space defines if the point is in world or local space (local by default) * @returns the current TransformNode */ setPivotPoint(e, t = ai.LOCAL) { this.getScene().getRenderId() == 0 && this.computeWorldMatrix(!0); const r = this.getWorldMatrix(); if (t == ai.WORLD) { const n = ue.Matrix[0]; r.invertToRef(n), e = S.TransformCoordinates(e, n); } return this.setPivotMatrix(he.Translation(-e.x, -e.y, -e.z), !0); } /** * Returns a new Vector3 set with the mesh pivot point coordinates in the local space. * @returns the pivot point */ getPivotPoint() { const e = S.Zero(); return this.getPivotPointToRef(e), e; } /** * Sets the passed Vector3 "result" with the coordinates of the mesh pivot point in the local space. * @param result the vector3 to store the result * @returns this TransformNode. */ getPivotPointToRef(e) { return e.x = -this._pivotMatrix.m[12], e.y = -this._pivotMatrix.m[13], e.z = -this._pivotMatrix.m[14], this; } /** * Returns a new Vector3 set with the mesh pivot point World coordinates. * @returns a new Vector3 set with the mesh pivot point World coordinates. */ getAbsolutePivotPoint() { const e = S.Zero(); return this.getAbsolutePivotPointToRef(e), e; } /** * Sets the Vector3 "result" coordinates with the mesh pivot point World coordinates. * @param result vector3 to store the result * @returns this TransformNode. */ getAbsolutePivotPointToRef(e) { return this.getPivotPointToRef(e), S.TransformCoordinatesToRef(e, this.getWorldMatrix(), e), this; } /** * Flag the transform node as dirty (Forcing it to update everything) * @param property if set to "rotation" the objects rotationQuaternion will be set to null * @returns this node */ markAsDirty(e) { if (this._isDirty) return this; if (this._children) for (const t of this._children) t.markAsDirty(e); return super.markAsDirty(e); } /** * Defines the passed node as the parent of the current node. * The node will remain exactly where it is and its position / rotation will be updated accordingly. * Note that if the mesh has a pivot matrix / point defined it will be applied after the parent was updated. * In that case the node will not remain in the same space as it is, as the pivot will be applied. * To avoid this, you can set updatePivot to true and the pivot will be updated to identity * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/parent_pivot/parent * @param node the node ot set as the parent * @param preserveScalingSign if true, keep scaling sign of child. Otherwise, scaling sign might change. * @param updatePivot if true, update the pivot matrix to keep the node in the same space as before * @returns this TransformNode. */ setParent(e, t = !1, r = !1) { if (!e && !this.parent) return this; const n = ue.Quaternion[0], i = ue.Vector3[0], s = ue.Vector3[1], a = ue.Matrix[1]; he.IdentityToRef(a); const f = ue.Matrix[0]; this.computeWorldMatrix(!0); let o = this.rotationQuaternion; return o || (o = Hr._TmpRotation, Ze.RotationYawPitchRollToRef(this._rotation.y, this._rotation.x, this._rotation.z, o)), he.ComposeToRef(this.scaling, o, this.position, f), this.parent && f.multiplyToRef(this.parent.computeWorldMatrix(!0), f), e && (e.computeWorldMatrix(!0).invertToRef(a), f.multiplyToRef(a, f)), f.decompose(s, n, i, t ? this : void 0), this.rotationQuaternion ? this.rotationQuaternion.copyFrom(n) : n.toEulerAnglesToRef(this.rotation), this.scaling.copyFrom(s), this.position.copyFrom(i), this.parent = e, r && this.setPivotMatrix(he.Identity()), this; } /** * True if the scaling property of this object is non uniform eg. (1,2,1) */ get nonUniformScaling() { return this._nonUniformScaling; } /** * @internal */ _updateNonUniformScalingState(e) { return this._nonUniformScaling === e ? !1 : (this._nonUniformScaling = e, !0); } /** * Attach the current TransformNode to another TransformNode associated with a bone * @param bone Bone affecting the TransformNode * @param affectedTransformNode TransformNode associated with the bone * @returns this object */ attachToBone(e, t) { return this._currentParentWhenAttachingToBone = this.parent, this._transformToBoneReferal = t, this.parent = e, e.getSkeleton().prepare(!0), e.getFinalMatrix().determinant() < 0 && (this.scalingDeterminant *= -1), this; } /** * Detach the transform node if its associated with a bone * @param resetToPreviousParent Indicates if the parent that was in effect when attachToBone was called should be set back or if we should set parent to null instead (defaults to the latter) * @returns this object */ detachFromBone(e = !1) { return this.parent ? (this.parent.getWorldMatrix().determinant() < 0 && (this.scalingDeterminant *= -1), this._transformToBoneReferal = null, e ? this.parent = this._currentParentWhenAttachingToBone : this.parent = null, this) : (e && (this.parent = this._currentParentWhenAttachingToBone), this); } /** * Rotates the mesh around the axis vector for the passed angle (amount) expressed in radians, in the given space. * space (default LOCAL) can be either Space.LOCAL, either Space.WORLD. * Note that the property `rotationQuaternion` is then automatically updated and the property `rotation` is set to (0,0,0) and no longer used. * The passed axis is also normalized. * @param axis the axis to rotate around * @param amount the amount to rotate in radians * @param space Space to rotate in (Default: local) * @returns the TransformNode. */ rotate(e, t, r) { e.normalize(), this.rotationQuaternion || (this.rotationQuaternion = this.rotation.toQuaternion(), this.rotation.setAll(0)); let n; if (!r || r === ai.LOCAL) n = Ze.RotationAxisToRef(e, t, Hr._RotationAxisCache), this.rotationQuaternion.multiplyToRef(n, this.rotationQuaternion); else { if (this.parent) { const i = this.parent.getWorldMatrix(), s = ue.Matrix[0]; i.invertToRef(s), e = S.TransformNormal(e, s), i.determinant() < 0 && (t *= -1); } n = Ze.RotationAxisToRef(e, t, Hr._RotationAxisCache), n.multiplyToRef(this.rotationQuaternion, this.rotationQuaternion); } return this; } /** * Rotates the mesh around the axis vector for the passed angle (amount) expressed in radians, in world space. * Note that the property `rotationQuaternion` is then automatically updated and the property `rotation` is set to (0,0,0) and no longer used. * The passed axis is also normalized. . * Method is based on http://www.euclideanspace.com/maths/geometry/affine/aroundPoint/index.htm * @param point the point to rotate around * @param axis the axis to rotate around * @param amount the amount to rotate in radians * @returns the TransformNode */ rotateAround(e, t, r) { t.normalize(), this.rotationQuaternion || (this.rotationQuaternion = Ze.RotationYawPitchRoll(this.rotation.y, this.rotation.x, this.rotation.z), this.rotation.setAll(0)); const n = ue.Vector3[0], i = ue.Vector3[1], s = ue.Vector3[2], a = ue.Quaternion[0], f = ue.Matrix[0], o = ue.Matrix[1], d = ue.Matrix[2], v = ue.Matrix[3]; return e.subtractToRef(this.position, n), he.TranslationToRef(n.x, n.y, n.z, f), he.TranslationToRef(-n.x, -n.y, -n.z, o), he.RotationAxisToRef(t, r, d), o.multiplyToRef(d, v), v.multiplyToRef(f, v), v.decompose(i, a, s), this.position.addInPlace(s), a.multiplyToRef(this.rotationQuaternion, this.rotationQuaternion), this; } /** * Translates the mesh along the axis vector for the passed distance in the given space. * space (default LOCAL) can be either Space.LOCAL, either Space.WORLD. * @param axis the axis to translate in * @param distance the distance to translate * @param space Space to rotate in (Default: local) * @returns the TransformNode. */ translate(e, t, r) { const n = e.scale(t); if (!r || r === ai.LOCAL) { const i = this.getPositionExpressedInLocalSpace().add(n); this.setPositionWithLocalVector(i); } else this.setAbsolutePosition(this.getAbsolutePosition().add(n)); return this; } /** * Adds a rotation step to the mesh current rotation. * x, y, z are Euler angles expressed in radians. * This methods updates the current mesh rotation, either mesh.rotation, either mesh.rotationQuaternion if it's set. * This means this rotation is made in the mesh local space only. * It's useful to set a custom rotation order different from the BJS standard one YXZ. * Example : this rotates the mesh first around its local X axis, then around its local Z axis, finally around its local Y axis. * ```javascript * mesh.addRotation(x1, 0, 0).addRotation(0, 0, z2).addRotation(0, 0, y3); * ``` * Note that `addRotation()` accumulates the passed rotation values to the current ones and computes the .rotation or .rotationQuaternion updated values. * Under the hood, only quaternions are used. So it's a little faster is you use .rotationQuaternion because it doesn't need to translate them back to Euler angles. * @param x Rotation to add * @param y Rotation to add * @param z Rotation to add * @returns the TransformNode. */ addRotation(e, t, r) { let n; this.rotationQuaternion ? n = this.rotationQuaternion : (n = ue.Quaternion[1], Ze.RotationYawPitchRollToRef(this.rotation.y, this.rotation.x, this.rotation.z, n)); const i = ue.Quaternion[0]; return Ze.RotationYawPitchRollToRef(t, e, r, i), n.multiplyInPlace(i), this.rotationQuaternion || n.toEulerAnglesToRef(this.rotation), this; } /** * @internal */ _getEffectiveParent() { return this.parent; } /** * Returns whether the transform node world matrix computation needs the camera information to be computed. * This is the case when the node is a billboard or has an infinite distance for instance. * @returns true if the world matrix computation needs the camera information to be computed */ isWorldMatrixCameraDependent() { return this._infiniteDistance && !this.parent || this._billboardMode !== Hr.BILLBOARDMODE_NONE && !this.preserveParentRotationForBillboard; } /** * Computes the world matrix of the node * @param force defines if the cache version should be invalidated forcing the world matrix to be created from scratch * @param camera defines the camera used if different from the scene active camera (This is used with modes like Billboard or infinite distance) * @returns the world matrix */ computeWorldMatrix(e = !1, t = null) { if (this._isWorldMatrixFrozen && !this._isDirty) return this._worldMatrix; const r = this.getScene().getRenderId(); if (!this._isDirty && !e && (this._currentRenderId === r || this.isSynchronized())) return this._currentRenderId = r, this._worldMatrix; t = t || this.getScene().activeCamera, this._updateCache(); const n = this._cache; n.pivotMatrixUpdated = !1, n.billboardMode = this.billboardMode, n.infiniteDistance = this.infiniteDistance, n.parent = this._parentNode, this._currentRenderId = r, this._childUpdateId += 1, this._isDirty = !1, this._position._isDirty = !1, this._rotation._isDirty = !1, this._scaling._isDirty = !1; const i = this._getEffectiveParent(), s = Hr._TmpScaling; let a = this._position; if (this._infiniteDistance && !this.parent && t) { const o = t.getWorldMatrix(), d = new S(o.m[12], o.m[13], o.m[14]); a = Hr._TmpTranslation, a.copyFromFloats(this._position.x + d.x, this._position.y + d.y, this._position.z + d.z); } s.copyFromFloats(this._scaling.x * this.scalingDeterminant, this._scaling.y * this.scalingDeterminant, this._scaling.z * this.scalingDeterminant); let f; if (this._rotationQuaternion ? (this._rotationQuaternion._isDirty = !1, f = this._rotationQuaternion, this.reIntegrateRotationIntoRotationQuaternion && this.rotation.lengthSquared() && (this._rotationQuaternion.multiplyInPlace(Ze.RotationYawPitchRoll(this._rotation.y, this._rotation.x, this._rotation.z)), this._rotation.copyFromFloats(0, 0, 0))) : (f = Hr._TmpRotation, Ze.RotationYawPitchRollToRef(this._rotation.y, this._rotation.x, this._rotation.z, f)), this._usePivotMatrix) { const o = ue.Matrix[1]; he.ScalingToRef(s.x, s.y, s.z, o); const d = ue.Matrix[0]; f.toRotationMatrix(d), this._pivotMatrix.multiplyToRef(o, ue.Matrix[4]), ue.Matrix[4].multiplyToRef(d, this._localMatrix), this._postMultiplyPivotMatrix && this._localMatrix.multiplyToRef(this._pivotMatrixInverse, this._localMatrix), this._localMatrix.addTranslationFromFloats(a.x, a.y, a.z); } else he.ComposeToRef(s, f, a, this._localMatrix); if (i && i.getWorldMatrix) { if (e && i.computeWorldMatrix(e), n.useBillboardPath) { if (this._transformToBoneReferal) { const u = this.parent; u.getSkeleton().prepare(), u.getFinalMatrix().multiplyToRef(this._transformToBoneReferal.getWorldMatrix(), ue.Matrix[7]); } else ue.Matrix[7].copyFrom(i.getWorldMatrix()); const o = ue.Vector3[5], d = ue.Vector3[6], v = ue.Quaternion[0]; ue.Matrix[7].decompose(d, v, o), he.ScalingToRef(d.x, d.y, d.z, ue.Matrix[7]), ue.Matrix[7].setTranslation(o), Hr.BillboardUseParentOrientation && (this._position.applyRotationQuaternionToRef(v, o), this._localMatrix.setTranslation(o)), this._localMatrix.multiplyToRef(ue.Matrix[7], this._worldMatrix); } else if (this._transformToBoneReferal) { const o = this.parent; o.getSkeleton().prepare(), this._localMatrix.multiplyToRef(o.getFinalMatrix(), ue.Matrix[6]), ue.Matrix[6].multiplyToRef(this._transformToBoneReferal.getWorldMatrix(), this._worldMatrix); } else this._localMatrix.multiplyToRef(i.getWorldMatrix(), this._worldMatrix); this._markSyncedWithParent(); } else this._worldMatrix.copyFrom(this._localMatrix); if (n.useBillboardPath && t && this.billboardMode && !n.useBillboardPosition) { const o = ue.Vector3[0]; if (this._worldMatrix.getTranslationToRef(o), ue.Matrix[1].copyFrom(t.getViewMatrix()), this._scene.useRightHandedSystem && ue.Matrix[1].multiplyToRef(u6e, ue.Matrix[1]), ue.Matrix[1].setTranslationFromFloats(0, 0, 0), ue.Matrix[1].invertToRef(ue.Matrix[0]), (this.billboardMode & Hr.BILLBOARDMODE_ALL) !== Hr.BILLBOARDMODE_ALL) { ue.Matrix[0].decompose(void 0, ue.Quaternion[0], void 0); const d = ue.Vector3[1]; ue.Quaternion[0].toEulerAnglesToRef(d), (this.billboardMode & Hr.BILLBOARDMODE_X) !== Hr.BILLBOARDMODE_X && (d.x = 0), (this.billboardMode & Hr.BILLBOARDMODE_Y) !== Hr.BILLBOARDMODE_Y && (d.y = 0), (this.billboardMode & Hr.BILLBOARDMODE_Z) !== Hr.BILLBOARDMODE_Z && (d.z = 0), he.RotationYawPitchRollToRef(d.y, d.x, d.z, ue.Matrix[0]); } this._worldMatrix.setTranslationFromFloats(0, 0, 0), this._worldMatrix.multiplyToRef(ue.Matrix[0], this._worldMatrix), this._worldMatrix.setTranslation(ue.Vector3[0]); } else if (n.useBillboardPath && t && n.useBillboardPosition) { const o = ue.Vector3[0]; this._worldMatrix.getTranslationToRef(o); const d = t.globalPosition; this._worldMatrix.invertToRef(ue.Matrix[1]); const v = ue.Vector3[1]; S.TransformCoordinatesToRef(d, ue.Matrix[1], v), v.normalize(); const u = -Math.atan2(v.z, v.x) + Math.PI / 2, l = Math.sqrt(v.x * v.x + v.z * v.z), P = -Math.atan2(v.y, l); if (Ze.RotationYawPitchRollToRef(u, P, 0, ue.Quaternion[0]), (this.billboardMode & Hr.BILLBOARDMODE_ALL) !== Hr.BILLBOARDMODE_ALL) { const p = ue.Vector3[1]; ue.Quaternion[0].toEulerAnglesToRef(p), (this.billboardMode & Hr.BILLBOARDMODE_X) !== Hr.BILLBOARDMODE_X && (p.x = 0), (this.billboardMode & Hr.BILLBOARDMODE_Y) !== Hr.BILLBOARDMODE_Y && (p.y = 0), (this.billboardMode & Hr.BILLBOARDMODE_Z) !== Hr.BILLBOARDMODE_Z && (p.z = 0), he.RotationYawPitchRollToRef(p.y, p.x, p.z, ue.Matrix[0]); } else he.FromQuaternionToRef(ue.Quaternion[0], ue.Matrix[0]); this._worldMatrix.setTranslationFromFloats(0, 0, 0), this._worldMatrix.multiplyToRef(ue.Matrix[0], this._worldMatrix), this._worldMatrix.setTranslation(ue.Vector3[0]); } return this.ignoreNonUniformScaling ? this._updateNonUniformScalingState(!1) : this._scaling.isNonUniformWithinEpsilon(1e-6) ? this._updateNonUniformScalingState(!0) : i && i._nonUniformScaling ? this._updateNonUniformScalingState(i._nonUniformScaling) : this._updateNonUniformScalingState(!1), this._afterComputeWorldMatrix(), this._absolutePosition.copyFromFloats(this._worldMatrix.m[12], this._worldMatrix.m[13], this._worldMatrix.m[14]), this._isAbsoluteSynced = !1, this.onAfterWorldMatrixUpdateObservable.notifyObservers(this), this._poseMatrix || (this._poseMatrix = he.Invert(this._worldMatrix)), this._worldMatrixDeterminantIsDirty = !0, this._worldMatrix; } /** * Resets this nodeTransform's local matrix to Matrix.Identity(). * @param independentOfChildren indicates if all child nodeTransform's world-space transform should be preserved. */ resetLocalMatrix(e = !0) { if (this.computeWorldMatrix(), e) { const t = this.getChildren(); for (let r = 0; r < t.length; ++r) { const n = t[r]; if (n) { n.computeWorldMatrix(); const i = ue.Matrix[0]; n._localMatrix.multiplyToRef(this._localMatrix, i); const s = ue.Quaternion[0]; i.decompose(n.scaling, s, n.position), n.rotationQuaternion ? n.rotationQuaternion.copyFrom(s) : s.toEulerAnglesToRef(n.rotation); } } } this.scaling.copyFromFloats(1, 1, 1), this.position.copyFromFloats(0, 0, 0), this.rotation.copyFromFloats(0, 0, 0), this.rotationQuaternion && (this.rotationQuaternion = Ze.Identity()), this._worldMatrix = he.Identity(); } _afterComputeWorldMatrix() { } /** * If you'd like to be called back after the mesh position, rotation or scaling has been updated. * @param func callback function to add * * @returns the TransformNode. */ registerAfterWorldMatrixUpdate(e) { return this.onAfterWorldMatrixUpdateObservable.add(e), this; } /** * Removes a registered callback function. * @param func callback function to remove * @returns the TransformNode. */ unregisterAfterWorldMatrixUpdate(e) { return this.onAfterWorldMatrixUpdateObservable.removeCallback(e), this; } /** * Gets the position of the current mesh in camera space * @param camera defines the camera to use * @returns a position */ getPositionInCameraSpace(e = null) { return e || (e = this.getScene().activeCamera), S.TransformCoordinates(this.getAbsolutePosition(), e.getViewMatrix()); } /** * Returns the distance from the mesh to the active camera * @param camera defines the camera to use * @returns the distance */ getDistanceToCamera(e = null) { return e || (e = this.getScene().activeCamera), this.getAbsolutePosition().subtract(e.globalPosition).length(); } /** * Clone the current transform node * @param name Name of the new clone * @param newParent New parent for the clone * @param doNotCloneChildren Do not clone children hierarchy * @returns the new transform node */ clone(e, t, r) { const n = jt.Clone(() => new Hr(e, this.getScene()), this); if (n.name = e, n.id = e, t && (n.parent = t), !r) { const i = this.getDescendants(!0); for (let s = 0; s < i.length; s++) { const a = i[s]; a.clone && a.clone(e + "." + a.name, n); } } return n; } /** * Serializes the objects information. * @param currentSerializationObject defines the object to serialize in * @returns the serialized object */ serialize(e) { const t = jt.Serialize(this, e); return t.type = this.getClassName(), t.uniqueId = this.uniqueId, this.parent && this.parent._serializeAsParent(t), t.localMatrix = this.getPivotMatrix().asArray(), t.isEnabled = this.isEnabled(), t; } // Statics /** * Returns a new TransformNode object parsed from the source provided. * @param parsedTransformNode is the source. * @param scene the scene the object belongs to * @param rootUrl is a string, it's the root URL to prefix the `delayLoadingFile` property with * @returns a new TransformNode object parsed from the source provided. */ static Parse(e, t, r) { const n = jt.Parse(() => new Hr(e.name, t), e, t, r); return e.localMatrix ? n.setPreTransformMatrix(he.FromArray(e.localMatrix)) : e.pivotMatrix && n.setPivotMatrix(he.FromArray(e.pivotMatrix)), n.setEnabled(e.isEnabled), n._waitingParsedUniqueId = e.uniqueId, e.parentId !== void 0 && (n._waitingParentId = e.parentId), e.parentInstanceIndex !== void 0 && (n._waitingParentInstanceIndex = e.parentInstanceIndex), n; } /** * Get all child-transformNodes of this node * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored * @returns an array of TransformNode */ getChildTransformNodes(e, t) { const r = []; return this._getDescendants(r, e, (n) => (!t || t(n)) && n instanceof Hr), r; } /** * Releases resources associated with this transform node. * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default) * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default) */ dispose(e, t = !1) { if (this.getScene().stopAnimation(this), this.getScene().removeTransformNode(this), this._parentContainer) { const r = this._parentContainer.transformNodes.indexOf(this); r > -1 && this._parentContainer.transformNodes.splice(r, 1), this._parentContainer = null; } if (this.onAfterWorldMatrixUpdateObservable.clear(), e) { const r = this.getChildTransformNodes(!0); for (const n of r) n.parent = null, n.computeWorldMatrix(!0); } super.dispose(e, t); } /** * Uniformly scales the mesh to fit inside of a unit cube (1 X 1 X 1 units) * @param includeDescendants Use the hierarchy's bounding box instead of the mesh's bounding box. Default is false * @param ignoreRotation ignore rotation when computing the scale (ie. object will be axis aligned). Default is false * @param predicate predicate that is passed in to getHierarchyBoundingVectors when selecting which object should be included when scaling * @returns the current mesh */ normalizeToUnitCube(e = !0, t = !1, r) { let n = null, i = null; t && (this.rotationQuaternion ? (i = this.rotationQuaternion.clone(), this.rotationQuaternion.copyFromFloats(0, 0, 0, 1)) : this.rotation && (n = this.rotation.clone(), this.rotation.copyFromFloats(0, 0, 0))); const s = this.getHierarchyBoundingVectors(e, r), a = s.max.subtract(s.min), f = Math.max(a.x, a.y, a.z); if (f === 0) return this; const o = 1 / f; return this.scaling.scaleInPlace(o), t && (this.rotationQuaternion && i ? this.rotationQuaternion.copyFrom(i) : this.rotation && n && this.rotation.copyFrom(n)), this; } _syncAbsoluteScalingAndRotation() { this._isAbsoluteSynced || (this._worldMatrix.decompose(this._absoluteScaling, this._absoluteRotationQuaternion), this._isAbsoluteSynced = !0); } } Hr.BILLBOARDMODE_NONE = 0; Hr.BILLBOARDMODE_X = 1; Hr.BILLBOARDMODE_Y = 2; Hr.BILLBOARDMODE_Z = 4; Hr.BILLBOARDMODE_ALL = 7; Hr.BILLBOARDMODE_USE_POSITION = 128; Hr.BillboardUseParentOrientation = !1; Hr._TmpRotation = Ze.Zero(); Hr._TmpScaling = S.Zero(); Hr._TmpTranslation = S.Zero(); Hr._LookAtVectorCache = new S(0, 0, 0); Hr._RotationAxisCache = new Ze(); C([ fo("position") ], Hr.prototype, "_position", void 0); C([ fo("rotation") ], Hr.prototype, "_rotation", void 0); C([ S$("rotationQuaternion") ], Hr.prototype, "_rotationQuaternion", void 0); C([ fo("scaling") ], Hr.prototype, "_scaling", void 0); C([ M("billboardMode") ], Hr.prototype, "_billboardMode", void 0); C([ M() ], Hr.prototype, "scalingDeterminant", void 0); C([ M("infiniteDistance") ], Hr.prototype, "_infiniteDistance", void 0); C([ M() ], Hr.prototype, "ignoreNonUniformScaling", void 0); C([ M() ], Hr.prototype, "reIntegrateRotationIntoRotationQuaternion", void 0); class aee { constructor() { this._checkCollisions = !1, this._collisionMask = -1, this._collisionGroup = -1, this._surroundingMeshes = null, this._collider = null, this._oldPositionForCollisions = new S(0, 0, 0), this._diffPositionForCollisions = new S(0, 0, 0), this._collisionResponse = !0; } } class l6e { constructor() { this.facetNb = 0, this.partitioningSubdivisions = 10, this.partitioningBBoxRatio = 1.01, this.facetDataEnabled = !1, this.facetParameters = {}, this.bbSize = S.Zero(), this.subDiv = { // actual number of subdivisions per axis for ComputeNormals() max: 1, // eslint-disable-next-line @typescript-eslint/naming-convention X: 1, // eslint-disable-next-line @typescript-eslint/naming-convention Y: 1, // eslint-disable-next-line @typescript-eslint/naming-convention Z: 1 }, this.facetDepthSort = !1, this.facetDepthSortEnabled = !1; } } class P6e { constructor() { this._hasVertexAlpha = !1, this._useVertexColors = !0, this._numBoneInfluencers = 4, this._applyFog = !0, this._receiveShadows = !1, this._facetData = new l6e(), this._visibility = 1, this._skeleton = null, this._layerMask = 268435455, this._computeBonesUsingShaders = !0, this._isActive = !1, this._onlyForInstances = !1, this._isActiveIntermediate = !1, this._onlyForInstancesIntermediate = !1, this._actAsRegularMesh = !1, this._currentLOD = null, this._currentLODIsUpToDate = !1, this._collisionRetryCount = 3, this._morphTargetManager = null, this._renderingGroupId = 0, this._bakedVertexAnimationManager = null, this._material = null, this._positions = null, this._pointerOverDisableMeshTesting = !1, this._meshCollisionData = new aee(), this._enableDistantPicking = !1, this._rawBoundingInfo = null; } } class jn extends Hr { /** * No billboard */ static get BILLBOARDMODE_NONE() { return Hr.BILLBOARDMODE_NONE; } /** Billboard on X axis */ static get BILLBOARDMODE_X() { return Hr.BILLBOARDMODE_X; } /** Billboard on Y axis */ static get BILLBOARDMODE_Y() { return Hr.BILLBOARDMODE_Y; } /** Billboard on Z axis */ static get BILLBOARDMODE_Z() { return Hr.BILLBOARDMODE_Z; } /** Billboard on all axes */ static get BILLBOARDMODE_ALL() { return Hr.BILLBOARDMODE_ALL; } /** Billboard on using position instead of orientation */ static get BILLBOARDMODE_USE_POSITION() { return Hr.BILLBOARDMODE_USE_POSITION; } /** * Gets the number of facets in the mesh * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#what-is-a-mesh-facet */ get facetNb() { return this._internalAbstractMeshDataInfo._facetData.facetNb; } /** * Gets or set the number (integer) of subdivisions per axis in the partitioning space * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#tweaking-the-partitioning */ get partitioningSubdivisions() { return this._internalAbstractMeshDataInfo._facetData.partitioningSubdivisions; } set partitioningSubdivisions(e) { this._internalAbstractMeshDataInfo._facetData.partitioningSubdivisions = e; } /** * The ratio (float) to apply to the bounding box size to set to the partitioning space. * Ex : 1.01 (default) the partitioning space is 1% bigger than the bounding box * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#tweaking-the-partitioning */ get partitioningBBoxRatio() { return this._internalAbstractMeshDataInfo._facetData.partitioningBBoxRatio; } set partitioningBBoxRatio(e) { this._internalAbstractMeshDataInfo._facetData.partitioningBBoxRatio = e; } /** * Gets or sets a boolean indicating that the facets must be depth sorted on next call to `updateFacetData()`. * Works only for updatable meshes. * Doesn't work with multi-materials * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#facet-depth-sort */ get mustDepthSortFacets() { return this._internalAbstractMeshDataInfo._facetData.facetDepthSort; } set mustDepthSortFacets(e) { this._internalAbstractMeshDataInfo._facetData.facetDepthSort = e; } /** * The location (Vector3) where the facet depth sort must be computed from. * By default, the active camera position. * Used only when facet depth sort is enabled * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#facet-depth-sort */ get facetDepthSortFrom() { return this._internalAbstractMeshDataInfo._facetData.facetDepthSortFrom; } set facetDepthSortFrom(e) { this._internalAbstractMeshDataInfo._facetData.facetDepthSortFrom = e; } /** number of collision detection tries. Change this value if not all collisions are detected and handled properly */ get collisionRetryCount() { return this._internalAbstractMeshDataInfo._collisionRetryCount; } set collisionRetryCount(e) { this._internalAbstractMeshDataInfo._collisionRetryCount = e; } /** * gets a boolean indicating if facetData is enabled * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#what-is-a-mesh-facet */ get isFacetDataEnabled() { return this._internalAbstractMeshDataInfo._facetData.facetDataEnabled; } /** * Gets or sets the morph target manager * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/morphTargets */ get morphTargetManager() { return this._internalAbstractMeshDataInfo._morphTargetManager; } set morphTargetManager(e) { this._internalAbstractMeshDataInfo._morphTargetManager !== e && (this._internalAbstractMeshDataInfo._morphTargetManager = e, this._syncGeometryWithMorphTargetManager()); } /** * Gets or sets the baked vertex animation manager * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/baked_texture_animations */ get bakedVertexAnimationManager() { return this._internalAbstractMeshDataInfo._bakedVertexAnimationManager; } set bakedVertexAnimationManager(e) { this._internalAbstractMeshDataInfo._bakedVertexAnimationManager !== e && (this._internalAbstractMeshDataInfo._bakedVertexAnimationManager = e, this._markSubMeshesAsAttributesDirty()); } /** @internal */ _syncGeometryWithMorphTargetManager() { } /** * @internal */ _updateNonUniformScalingState(e) { return super._updateNonUniformScalingState(e) ? (this._markSubMeshesAsMiscDirty(), !0) : !1; } /** @internal */ get rawBoundingInfo() { return this._internalAbstractMeshDataInfo._rawBoundingInfo; } set rawBoundingInfo(e) { this._internalAbstractMeshDataInfo._rawBoundingInfo = e; } /** Set a function to call when this mesh collides with another one */ set onCollide(e) { this._internalAbstractMeshDataInfo._meshCollisionData._onCollideObserver && this.onCollideObservable.remove(this._internalAbstractMeshDataInfo._meshCollisionData._onCollideObserver), this._internalAbstractMeshDataInfo._meshCollisionData._onCollideObserver = this.onCollideObservable.add(e); } /** Set a function to call when the collision's position changes */ set onCollisionPositionChange(e) { this._internalAbstractMeshDataInfo._meshCollisionData._onCollisionPositionChangeObserver && this.onCollisionPositionChangeObservable.remove(this._internalAbstractMeshDataInfo._meshCollisionData._onCollisionPositionChangeObserver), this._internalAbstractMeshDataInfo._meshCollisionData._onCollisionPositionChangeObserver = this.onCollisionPositionChangeObservable.add(e); } /** * Gets or sets mesh visibility between 0 and 1 (default is 1) */ get visibility() { return this._internalAbstractMeshDataInfo._visibility; } /** * Gets or sets mesh visibility between 0 and 1 (default is 1) */ set visibility(e) { if (this._internalAbstractMeshDataInfo._visibility === e) return; const t = this._internalAbstractMeshDataInfo._visibility; this._internalAbstractMeshDataInfo._visibility = e, (t === 1 && e !== 1 || t !== 1 && e === 1) && this._markSubMeshesAsDirty((r) => { r.markAsMiscDirty(), r.markAsPrePassDirty(); }); } /** * Gets or sets the property which disables the test that is checking that the mesh under the pointer is the same than the previous time we tested for it (default: false). * Set this property to true if you want thin instances picking to be reported accurately when moving over the mesh. * Note that setting this property to true will incur some performance penalties when dealing with pointer events for this mesh so use it sparingly. */ get pointerOverDisableMeshTesting() { return this._internalAbstractMeshDataInfo._pointerOverDisableMeshTesting; } set pointerOverDisableMeshTesting(e) { this._internalAbstractMeshDataInfo._pointerOverDisableMeshTesting = e; } /** * Specifies the rendering group id for this mesh (0 by default) * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/advanced/transparent_rendering#rendering-groups */ get renderingGroupId() { return this._internalAbstractMeshDataInfo._renderingGroupId; } set renderingGroupId(e) { this._internalAbstractMeshDataInfo._renderingGroupId = e; } /** Gets or sets current material */ get material() { return this._internalAbstractMeshDataInfo._material; } set material(e) { this._internalAbstractMeshDataInfo._material !== e && (this._internalAbstractMeshDataInfo._material && this._internalAbstractMeshDataInfo._material.meshMap && (this._internalAbstractMeshDataInfo._material.meshMap[this.uniqueId] = void 0), this._internalAbstractMeshDataInfo._material = e, e && e.meshMap && (e.meshMap[this.uniqueId] = this), this.onMaterialChangedObservable.hasObservers() && this.onMaterialChangedObservable.notifyObservers(this), this.subMeshes && (this.resetDrawCache(), this._unBindEffect())); } /** * Gets the material used to render the mesh in a specific render pass * @param renderPassId render pass id * @returns material used for the render pass. If no specific material is used for this render pass, undefined is returned (meaning mesh.material is used for this pass) */ getMaterialForRenderPass(e) { var t; return (t = this._internalAbstractMeshDataInfo._materialForRenderPass) === null || t === void 0 ? void 0 : t[e]; } /** * Sets the material to be used to render the mesh in a specific render pass * @param renderPassId render pass id * @param material material to use for this render pass. If undefined is passed, no specific material will be used for this render pass but the regular material will be used instead (mesh.material) */ setMaterialForRenderPass(e, t) { this.resetDrawCache(e), this._internalAbstractMeshDataInfo._materialForRenderPass || (this._internalAbstractMeshDataInfo._materialForRenderPass = []), this._internalAbstractMeshDataInfo._materialForRenderPass[e] = t; } /** * Gets or sets a boolean indicating that this mesh can receive realtime shadows * @see https://doc.babylonjs.com/features/featuresDeepDive/lights/shadows */ get receiveShadows() { return this._internalAbstractMeshDataInfo._receiveShadows; } set receiveShadows(e) { this._internalAbstractMeshDataInfo._receiveShadows !== e && (this._internalAbstractMeshDataInfo._receiveShadows = e, this._markSubMeshesAsLightDirty()); } /** Gets or sets a boolean indicating that this mesh contains vertex color data with alpha values */ get hasVertexAlpha() { return this._internalAbstractMeshDataInfo._hasVertexAlpha; } set hasVertexAlpha(e) { this._internalAbstractMeshDataInfo._hasVertexAlpha !== e && (this._internalAbstractMeshDataInfo._hasVertexAlpha = e, this._markSubMeshesAsAttributesDirty(), this._markSubMeshesAsMiscDirty()); } /** Gets or sets a boolean indicating that this mesh needs to use vertex color data to render (if this kind of vertex data is available in the geometry) */ get useVertexColors() { return this._internalAbstractMeshDataInfo._useVertexColors; } set useVertexColors(e) { this._internalAbstractMeshDataInfo._useVertexColors !== e && (this._internalAbstractMeshDataInfo._useVertexColors = e, this._markSubMeshesAsAttributesDirty()); } /** * Gets or sets a boolean indicating that bone animations must be computed by the GPU (true by default) */ get computeBonesUsingShaders() { return this._internalAbstractMeshDataInfo._computeBonesUsingShaders; } set computeBonesUsingShaders(e) { this._internalAbstractMeshDataInfo._computeBonesUsingShaders !== e && (this._internalAbstractMeshDataInfo._computeBonesUsingShaders = e, this._markSubMeshesAsAttributesDirty()); } /** Gets or sets the number of allowed bone influences per vertex (4 by default) */ get numBoneInfluencers() { return this._internalAbstractMeshDataInfo._numBoneInfluencers; } set numBoneInfluencers(e) { this._internalAbstractMeshDataInfo._numBoneInfluencers !== e && (this._internalAbstractMeshDataInfo._numBoneInfluencers = e, this._markSubMeshesAsAttributesDirty()); } /** Gets or sets a boolean indicating that this mesh will allow fog to be rendered on it (true by default) */ get applyFog() { return this._internalAbstractMeshDataInfo._applyFog; } set applyFog(e) { this._internalAbstractMeshDataInfo._applyFog !== e && (this._internalAbstractMeshDataInfo._applyFog = e, this._markSubMeshesAsMiscDirty()); } /** When enabled, decompose picking matrices for better precision with large values for mesh position and scling */ get enableDistantPicking() { return this._internalAbstractMeshDataInfo._enableDistantPicking; } set enableDistantPicking(e) { this._internalAbstractMeshDataInfo._enableDistantPicking = e; } /** * Gets or sets the current layer mask (default is 0x0FFFFFFF) * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/layerMasksAndMultiCam */ get layerMask() { return this._internalAbstractMeshDataInfo._layerMask; } set layerMask(e) { e !== this._internalAbstractMeshDataInfo._layerMask && (this._internalAbstractMeshDataInfo._layerMask = e, this._resyncLightSources()); } /** * Gets or sets a collision mask used to mask collisions (default is -1). * A collision between A and B will happen if A.collisionGroup & b.collisionMask !== 0 */ get collisionMask() { return this._internalAbstractMeshDataInfo._meshCollisionData._collisionMask; } set collisionMask(e) { this._internalAbstractMeshDataInfo._meshCollisionData._collisionMask = isNaN(e) ? -1 : e; } /** * Gets or sets a collision response flag (default is true). * when collisionResponse is false, events are still triggered but colliding entity has no response * This helps creating trigger volume when user wants collision feedback events but not position/velocity * to respond to the collision. */ get collisionResponse() { return this._internalAbstractMeshDataInfo._meshCollisionData._collisionResponse; } set collisionResponse(e) { this._internalAbstractMeshDataInfo._meshCollisionData._collisionResponse = e; } /** * Gets or sets the current collision group mask (-1 by default). * A collision between A and B will happen if A.collisionGroup & b.collisionMask !== 0 */ get collisionGroup() { return this._internalAbstractMeshDataInfo._meshCollisionData._collisionGroup; } set collisionGroup(e) { this._internalAbstractMeshDataInfo._meshCollisionData._collisionGroup = isNaN(e) ? -1 : e; } /** * Gets or sets current surrounding meshes (null by default). * * By default collision detection is tested against every mesh in the scene. * It is possible to set surroundingMeshes to a defined list of meshes and then only these specified * meshes will be tested for the collision. * * Note: if set to an empty array no collision will happen when this mesh is moved. */ get surroundingMeshes() { return this._internalAbstractMeshDataInfo._meshCollisionData._surroundingMeshes; } set surroundingMeshes(e) { this._internalAbstractMeshDataInfo._meshCollisionData._surroundingMeshes = e; } /** Gets the list of lights affecting that mesh */ get lightSources() { return this._lightSources; } /** @internal */ get _positions() { return null; } /** * Gets or sets a skeleton to apply skinning transformations * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/bonesSkeletons */ set skeleton(e) { const t = this._internalAbstractMeshDataInfo._skeleton; t && t.needInitialSkinMatrix && t._unregisterMeshWithPoseMatrix(this), e && e.needInitialSkinMatrix && e._registerMeshWithPoseMatrix(this), this._internalAbstractMeshDataInfo._skeleton = e, this._internalAbstractMeshDataInfo._skeleton || (this._bonesTransformMatrices = null), this._markSubMeshesAsAttributesDirty(); } get skeleton() { return this._internalAbstractMeshDataInfo._skeleton; } // Constructor /** * Creates a new AbstractMesh * @param name defines the name of the mesh * @param scene defines the hosting scene */ constructor(e, t = null) { switch (super(e, t, !1), this._internalAbstractMeshDataInfo = new P6e(), this._waitingMaterialId = null, this.cullingStrategy = jn.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY, this.onCollideObservable = new Oe(), this.onCollisionPositionChangeObservable = new Oe(), this.onMaterialChangedObservable = new Oe(), this.definedFacingForward = !0, this._occlusionQuery = null, this._renderingGroup = null, this.alphaIndex = Number.MAX_VALUE, this.isVisible = !0, this.isPickable = !0, this.isNearPickable = !1, this.isNearGrabbable = !1, this.showSubMeshesBoundingBox = !1, this.isBlocker = !1, this.enablePointerMoveEvents = !1, this.outlineColor = Ne.Red(), this.outlineWidth = 0.02, this.overlayColor = Ne.Red(), this.overlayAlpha = 0.5, this.useOctreeForRenderingSelection = !0, this.useOctreeForPicking = !0, this.useOctreeForCollisions = !0, this.alwaysSelectAsActiveMesh = !1, this.doNotSyncBoundingInfo = !1, this.actionManager = null, this.ellipsoid = new S(0.5, 1, 0.5), this.ellipsoidOffset = new S(0, 0, 0), this.edgesWidth = 1, this.edgesColor = new xt(1, 0, 0, 1), this._edgesRenderer = null, this._masterMesh = null, this._boundingInfo = null, this._boundingInfoIsDirty = !0, this._renderId = 0, this._intersectionsInProgress = new Array(), this._unIndexed = !1, this._lightSources = new Array(), this._waitingData = { lods: null, actions: null, freezeWorldMatrix: null }, this._bonesTransformMatrices = null, this._transformMatrixTexture = null, this.onRebuildObservable = new Oe(), this._onCollisionPositionChange = (r, n, i = null) => { n.subtractToRef(this._internalAbstractMeshDataInfo._meshCollisionData._oldPositionForCollisions, this._internalAbstractMeshDataInfo._meshCollisionData._diffPositionForCollisions), this._internalAbstractMeshDataInfo._meshCollisionData._diffPositionForCollisions.length() > Ge.CollisionsEpsilon && this.position.addInPlace(this._internalAbstractMeshDataInfo._meshCollisionData._diffPositionForCollisions), i && this.onCollideObservable.notifyObservers(i), this.onCollisionPositionChangeObservable.notifyObservers(this.position); }, t = this.getScene(), t.addMesh(this), this._resyncLightSources(), this._uniformBuffer = new yr(this.getScene().getEngine(), void 0, void 0, e, !this.getScene().getEngine().isWebGPU), this._buildUniformLayout(), t.performancePriority) { case $H.Aggressive: this.doNotSyncBoundingInfo = !0; case $H.Intermediate: this.alwaysSelectAsActiveMesh = !0, this.isPickable = !1; break; } } _buildUniformLayout() { this._uniformBuffer.addUniform("world", 16), this._uniformBuffer.addUniform("visibility", 1), this._uniformBuffer.create(); } /** * Transfer the mesh values to its UBO. * @param world The world matrix associated with the mesh */ transferToEffect(e) { const t = this._uniformBuffer; t.updateMatrix("world", e), t.updateFloat("visibility", this._internalAbstractMeshDataInfo._visibility), t.update(); } /** * Gets the mesh uniform buffer. * @returns the uniform buffer of the mesh. */ getMeshUniformBuffer() { return this._uniformBuffer; } /** * Returns the string "AbstractMesh" * @returns "AbstractMesh" */ getClassName() { return "AbstractMesh"; } /** * Gets a string representation of the current mesh * @param fullDetails defines a boolean indicating if full details must be included * @returns a string representation of the current mesh */ toString(e) { let t = "Name: " + this.name + ", isInstance: " + (this.getClassName() !== "InstancedMesh" ? "YES" : "NO"); t += ", # of submeshes: " + (this.subMeshes ? this.subMeshes.length : 0); const r = this._internalAbstractMeshDataInfo._skeleton; return r && (t += ", skeleton: " + r.name), e && (t += ", billboard mode: " + ["NONE", "X", "Y", null, "Z", null, null, "ALL"][this.billboardMode], t += ", freeze wrld mat: " + (this._isWorldMatrixFrozen || this._waitingData.freezeWorldMatrix ? "YES" : "NO")), t; } /** * @internal */ _getEffectiveParent() { return this._masterMesh && this.billboardMode !== Hr.BILLBOARDMODE_NONE ? this._masterMesh : super._getEffectiveParent(); } /** * @internal */ _getActionManagerForTrigger(e, t = !0) { if (this.actionManager && (t || this.actionManager.isRecursive)) if (e) { if (this.actionManager.hasSpecificTrigger(e)) return this.actionManager; } else return this.actionManager; return this.parent ? this.parent._getActionManagerForTrigger(e, !1) : null; } /** * @internal */ // eslint-disable-next-line @typescript-eslint/no-unused-vars _rebuild(e = !1) { if (this.onRebuildObservable.notifyObservers(this), this._occlusionQuery !== null && (this._occlusionQuery = null), !!this.subMeshes) for (const t of this.subMeshes) t._rebuild(); } /** @internal */ _resyncLightSources() { this._lightSources.length = 0; for (const e of this.getScene().lights) e.isEnabled() && e.canAffectMesh(this) && this._lightSources.push(e); this._markSubMeshesAsLightDirty(); } /** * @internal */ _resyncLightSource(e) { const t = e.isEnabled() && e.canAffectMesh(this), r = this._lightSources.indexOf(e); let n = !1; if (r === -1) { if (!t) return; this._lightSources.push(e); } else { if (t) return; n = !0, this._lightSources.splice(r, 1); } this._markSubMeshesAsLightDirty(n); } /** @internal */ _unBindEffect() { for (const e of this.subMeshes) e.setEffect(null); } /** * @internal */ _removeLightSource(e, t) { const r = this._lightSources.indexOf(e); r !== -1 && (this._lightSources.splice(r, 1), this._markSubMeshesAsLightDirty(t)); } _markSubMeshesAsDirty(e) { if (this.subMeshes) for (const t of this.subMeshes) for (let r = 0; r < t._drawWrappers.length; ++r) { const n = t._drawWrappers[r]; !n || !n.defines || !n.defines.markAllAsDirty || e(n.defines); } } /** * @internal */ _markSubMeshesAsLightDirty(e = !1) { this._markSubMeshesAsDirty((t) => t.markAsLightDirty(e)); } /** @internal */ _markSubMeshesAsAttributesDirty() { this._markSubMeshesAsDirty((e) => e.markAsAttributesDirty()); } /** @internal */ _markSubMeshesAsMiscDirty() { this._markSubMeshesAsDirty((e) => e.markAsMiscDirty()); } /** * Flag the AbstractMesh as dirty (Forcing it to update everything) * @param property if set to "rotation" the objects rotationQuaternion will be set to null * @returns this AbstractMesh */ // eslint-disable-next-line @typescript-eslint/no-unused-vars markAsDirty(e) { return this._currentRenderId = Number.MAX_VALUE, this._isDirty = !0, this; } /** * Resets the draw wrappers cache for all submeshes of this abstract mesh * @param passId If provided, releases only the draw wrapper corresponding to this render pass id */ resetDrawCache(e) { if (this.subMeshes) for (const t of this.subMeshes) t.resetDrawCache(e); } // Methods /** * Returns true if the mesh is blocked. Implemented by child classes */ get isBlocked() { return !1; } /** * Returns the mesh itself by default. Implemented by child classes * @param camera defines the camera to use to pick the right LOD level * @returns the currentAbstractMesh */ // eslint-disable-next-line @typescript-eslint/no-unused-vars getLOD(e) { return this; } /** * Returns 0 by default. Implemented by child classes * @returns an integer */ getTotalVertices() { return 0; } /** * Returns a positive integer : the total number of indices in this mesh geometry. * @returns the number of indices or zero if the mesh has no geometry. */ getTotalIndices() { return 0; } /** * Returns null by default. Implemented by child classes * @returns null */ getIndices() { return null; } /** * Returns the array of the requested vertex data kind. Implemented by child classes * @param kind defines the vertex data kind to use * @returns null */ // eslint-disable-next-line @typescript-eslint/no-unused-vars getVerticesData(e) { return null; } /** * Sets the vertex data of the mesh geometry for the requested `kind`. * If the mesh has no geometry, a new Geometry object is set to the mesh and then passed this vertex data. * Note that a new underlying VertexBuffer object is created each call. * If the `kind` is the `PositionKind`, the mesh BoundingInfo is renewed, so the bounding box and sphere, and the mesh World Matrix is recomputed. * @param kind defines vertex data kind: * * VertexBuffer.PositionKind * * VertexBuffer.UVKind * * VertexBuffer.UV2Kind * * VertexBuffer.UV3Kind * * VertexBuffer.UV4Kind * * VertexBuffer.UV5Kind * * VertexBuffer.UV6Kind * * VertexBuffer.ColorKind * * VertexBuffer.MatricesIndicesKind * * VertexBuffer.MatricesIndicesExtraKind * * VertexBuffer.MatricesWeightsKind * * VertexBuffer.MatricesWeightsExtraKind * @param data defines the data source * @param updatable defines if the data must be flagged as updatable (or static) * @param stride defines the vertex stride (size of an entire vertex). Can be null and in this case will be deduced from vertex data kind * @returns the current mesh */ // eslint-disable-next-line @typescript-eslint/no-unused-vars setVerticesData(e, t, r, n) { return this; } /** * Updates the existing vertex data of the mesh geometry for the requested `kind`. * If the mesh has no geometry, it is simply returned as it is. * @param kind defines vertex data kind: * * VertexBuffer.PositionKind * * VertexBuffer.UVKind * * VertexBuffer.UV2Kind * * VertexBuffer.UV3Kind * * VertexBuffer.UV4Kind * * VertexBuffer.UV5Kind * * VertexBuffer.UV6Kind * * VertexBuffer.ColorKind * * VertexBuffer.MatricesIndicesKind * * VertexBuffer.MatricesIndicesExtraKind * * VertexBuffer.MatricesWeightsKind * * VertexBuffer.MatricesWeightsExtraKind * @param data defines the data source * @param updateExtends If `kind` is `PositionKind` and if `updateExtends` is true, the mesh BoundingInfo is renewed, so the bounding box and sphere, and the mesh World Matrix is recomputed * @param makeItUnique If true, a new global geometry is created from this data and is set to the mesh * @returns the current mesh */ // eslint-disable-next-line @typescript-eslint/no-unused-vars updateVerticesData(e, t, r, n) { return this; } /** * Sets the mesh indices, * If the mesh has no geometry, a new Geometry object is created and set to the mesh. * @param indices Expects an array populated with integers or a typed array (Int32Array, Uint32Array, Uint16Array) * @param totalVertices Defines the total number of vertices * @returns the current mesh */ // eslint-disable-next-line @typescript-eslint/no-unused-vars setIndices(e, t) { return this; } /** * Gets a boolean indicating if specific vertex data is present * @param kind defines the vertex data kind to use * @returns true is data kind is present */ // eslint-disable-next-line @typescript-eslint/no-unused-vars isVerticesDataPresent(e) { return !1; } /** * Returns the mesh BoundingInfo object or creates a new one and returns if it was undefined. * Note that it returns a shallow bounding of the mesh (i.e. it does not include children). * However, if the mesh contains thin instances, it will be expanded to include them. If you want the "raw" bounding data instead, then use `getRawBoundingInfo()`. * To get the full bounding of all children, call `getHierarchyBoundingVectors` instead. * @returns a BoundingInfo */ getBoundingInfo() { return this._masterMesh ? this._masterMesh.getBoundingInfo() : (this._boundingInfoIsDirty && (this._boundingInfoIsDirty = !1, this._updateBoundingInfo()), this._boundingInfo); } /** * Returns the bounding info unnafected by instance data. * @returns the bounding info of the mesh unaffected by instance data. */ getRawBoundingInfo() { var e; return (e = this.rawBoundingInfo) !== null && e !== void 0 ? e : this.getBoundingInfo(); } /** * Overwrite the current bounding info * @param boundingInfo defines the new bounding info * @returns the current mesh */ setBoundingInfo(e) { return this._boundingInfo = e, this; } /** * Returns true if there is already a bounding info */ get hasBoundingInfo() { return this._boundingInfo !== null; } /** * Creates a new bounding info for the mesh * @param minimum min vector of the bounding box/sphere * @param maximum max vector of the bounding box/sphere * @param worldMatrix defines the new world matrix * @returns the new bounding info */ buildBoundingInfo(e, t, r) { return this._boundingInfo = new Md(e, t, r), this._boundingInfo; } /** * Uniformly scales the mesh to fit inside of a unit cube (1 X 1 X 1 units) * @param includeDescendants Use the hierarchy's bounding box instead of the mesh's bounding box. Default is false * @param ignoreRotation ignore rotation when computing the scale (ie. object will be axis aligned). Default is false * @param predicate predicate that is passed in to getHierarchyBoundingVectors when selecting which object should be included when scaling * @returns the current mesh */ normalizeToUnitCube(e = !0, t = !1, r) { return super.normalizeToUnitCube(e, t, r); } /** Gets a boolean indicating if this mesh has skinning data and an attached skeleton */ get useBones() { return this.skeleton && this.getScene().skeletonsEnabled && this.isVerticesDataPresent(J.MatricesIndicesKind) && this.isVerticesDataPresent(J.MatricesWeightsKind); } /** @internal */ _preActivate() { } /** * @internal */ // eslint-disable-next-line @typescript-eslint/no-unused-vars _preActivateForIntermediateRendering(e) { } /** * @internal */ // eslint-disable-next-line @typescript-eslint/no-unused-vars _activate(e, t) { return this._renderId = e, !0; } /** @internal */ _postActivate() { } /** @internal */ _freeze() { } /** @internal */ _unFreeze() { } /** * Gets the current world matrix * @returns a Matrix */ getWorldMatrix() { return this._masterMesh && this.billboardMode === Hr.BILLBOARDMODE_NONE ? this._masterMesh.getWorldMatrix() : super.getWorldMatrix(); } /** @internal */ _getWorldMatrixDeterminant() { return this._masterMesh ? this._masterMesh._getWorldMatrixDeterminant() : super._getWorldMatrixDeterminant(); } /** * Gets a boolean indicating if this mesh is an instance or a regular mesh */ get isAnInstance() { return !1; } /** * Gets a boolean indicating if this mesh has instances */ get hasInstances() { return !1; } /** * Gets a boolean indicating if this mesh has thin instances */ get hasThinInstances() { return !1; } // ================================== Point of View Movement ================================= /** * Perform relative position change from the point of view of behind the front of the mesh. * This is performed taking into account the meshes current rotation, so you do not have to care. * Supports definition of mesh facing forward or backward {@link definedFacingForwardSearch | See definedFacingForwardSearch }. * @param amountRight defines the distance on the right axis * @param amountUp defines the distance on the up axis * @param amountForward defines the distance on the forward axis * @returns the current mesh */ movePOV(e, t, r) { return this.position.addInPlace(this.calcMovePOV(e, t, r)), this; } /** * Calculate relative position change from the point of view of behind the front of the mesh. * This is performed taking into account the meshes current rotation, so you do not have to care. * Supports definition of mesh facing forward or backward {@link definedFacingForwardSearch | See definedFacingForwardSearch }. * @param amountRight defines the distance on the right axis * @param amountUp defines the distance on the up axis * @param amountForward defines the distance on the forward axis * @returns the new displacement vector */ calcMovePOV(e, t, r) { const n = new he(); (this.rotationQuaternion ? this.rotationQuaternion : Ze.RotationYawPitchRoll(this.rotation.y, this.rotation.x, this.rotation.z)).toRotationMatrix(n); const s = S.Zero(), a = this.definedFacingForward ? -1 : 1; return S.TransformCoordinatesFromFloatsToRef(e * a, t, r * a, n, s), s; } // ================================== Point of View Rotation ================================= /** * Perform relative rotation change from the point of view of behind the front of the mesh. * Supports definition of mesh facing forward or backward {@link definedFacingForwardSearch | See definedFacingForwardSearch }. * @param flipBack defines the flip * @param twirlClockwise defines the twirl * @param tiltRight defines the tilt * @returns the current mesh */ rotatePOV(e, t, r) { return this.rotation.addInPlace(this.calcRotatePOV(e, t, r)), this; } /** * Calculate relative rotation change from the point of view of behind the front of the mesh. * Supports definition of mesh facing forward or backward {@link definedFacingForwardSearch | See definedFacingForwardSearch }. * @param flipBack defines the flip * @param twirlClockwise defines the twirl * @param tiltRight defines the tilt * @returns the new rotation vector */ calcRotatePOV(e, t, r) { const n = this.definedFacingForward ? 1 : -1; return new S(e * n, t, r * n); } /** * This method recomputes and sets a new BoundingInfo to the mesh unless it is locked. * This means the mesh underlying bounding box and sphere are recomputed. * @param applySkeleton defines whether to apply the skeleton before computing the bounding info * @param applyMorph defines whether to apply the morph target before computing the bounding info * @returns the current mesh */ refreshBoundingInfo(e = !1, t = !1) { return this._boundingInfo && this._boundingInfo.isLocked ? this : (this._refreshBoundingInfo(this._getPositionData(e, t), null), this); } /** * @internal */ _refreshBoundingInfo(e, t) { if (e) { const r = WS(e, 0, this.getTotalVertices(), t); this._boundingInfo ? this._boundingInfo.reConstruct(r.minimum, r.maximum) : this._boundingInfo = new Md(r.minimum, r.maximum); } if (this.subMeshes) for (let r = 0; r < this.subMeshes.length; r++) this.subMeshes[r].refreshBoundingInfo(e); this._updateBoundingInfo(); } /** * Internal function to get buffer data and possibly apply morphs and normals * @param applySkeleton * @param applyMorph * @param data * @param kind the kind of data you want. Can be Normal or Position */ _getData(e = !1, t = !1, r, n = J.PositionKind) { if (r = r ?? this.getVerticesData(n).slice(), r && t && this.morphTargetManager) { let i = 0, s = 0; for (let a = 0; a < r.length; a++) { for (let f = 0; f < this.morphTargetManager.numTargets; f++) { const o = this.morphTargetManager.getTarget(f), d = o.influence; if (d > 0) { const v = o.getPositions(); v && (r[a] += (v[a] - r[a]) * d); } } if (i++, n === J.PositionKind && this._positions && i === 3) { i = 0; const f = s * 3; this._positions[s++].copyFromFloats(r[f], r[f + 1], r[f + 2]); } } } if (r && e && this.skeleton) { const i = this.getVerticesData(J.MatricesIndicesKind), s = this.getVerticesData(J.MatricesWeightsKind); if (s && i) { const a = this.numBoneInfluencers > 4, f = a ? this.getVerticesData(J.MatricesIndicesExtraKind) : null, o = a ? this.getVerticesData(J.MatricesWeightsExtraKind) : null, d = this.skeleton.getTransformMatrices(this), v = ue.Vector3[0], u = ue.Matrix[0], l = ue.Matrix[1]; let P = 0; for (let p = 0; p < r.length; p += 3, P += 4) { u.reset(); let c, H; for (c = 0; c < 4; c++) H = s[P + c], H > 0 && (he.FromFloat32ArrayToRefScaled(d, Math.floor(i[P + c] * 16), H, l), u.addToSelf(l)); if (a) for (c = 0; c < 4; c++) H = o[P + c], H > 0 && (he.FromFloat32ArrayToRefScaled(d, Math.floor(f[P + c] * 16), H, l), u.addToSelf(l)); n === J.NormalKind ? S.TransformNormalFromFloatsToRef(r[p], r[p + 1], r[p + 2], u, v) : S.TransformCoordinatesFromFloatsToRef(r[p], r[p + 1], r[p + 2], u, v), v.toArray(r, p), n === J.PositionKind && this._positions && this._positions[p / 3].copyFrom(v); } } } return r; } /** * Get the normals vertex data and optionally apply skeleton and morphing. * @param applySkeleton defines whether to apply the skeleton * @param applyMorph defines whether to apply the morph target * @returns the normals data */ getNormalsData(e = !1, t = !1) { return this._getData(e, t, null, J.NormalKind); } /** * Get the position vertex data and optionally apply skeleton and morphing. * @param applySkeleton defines whether to apply the skeleton * @param applyMorph defines whether to apply the morph target * @param data defines the position data to apply the skeleton and morph to * @returns the position data */ getPositionData(e = !1, t = !1, r) { return this._getData(e, t, r, J.PositionKind); } /** * @internal */ _getPositionData(e, t) { var r; let n = this.getVerticesData(J.PositionKind); if (this._internalAbstractMeshDataInfo._positions && (this._internalAbstractMeshDataInfo._positions = null), n && (e && this.skeleton || t && this.morphTargetManager)) { if (n = n.slice(), this._generatePointsArray(), this._positions) { const i = this._positions; this._internalAbstractMeshDataInfo._positions = new Array(i.length); for (let s = 0; s < i.length; s++) this._internalAbstractMeshDataInfo._positions[s] = ((r = i[s]) === null || r === void 0 ? void 0 : r.clone()) || new S(); } return this.getPositionData(e, t, n); } return n; } /** @internal */ _updateBoundingInfo() { return this._boundingInfo ? this._boundingInfo.update(this.worldMatrixFromCache) : this._boundingInfo = new Md(S.Zero(), S.Zero(), this.worldMatrixFromCache), this._updateSubMeshesBoundingInfo(this.worldMatrixFromCache), this; } /** * @internal */ _updateSubMeshesBoundingInfo(e) { if (!this.subMeshes) return this; const t = this.subMeshes.length; for (let r = 0; r < t; r++) { const n = this.subMeshes[r]; (t > 1 || !n.IsGlobal) && n.updateBoundingInfo(e); } return this; } /** @internal */ _afterComputeWorldMatrix() { this.doNotSyncBoundingInfo || (this._boundingInfoIsDirty = !0); } /** * Returns `true` if the mesh is within the frustum defined by the passed array of planes. * A mesh is in the frustum if its bounding box intersects the frustum * @param frustumPlanes defines the frustum to test * @returns true if the mesh is in the frustum planes */ isInFrustum(e) { return this.getBoundingInfo().isInFrustum(e, this.cullingStrategy); } /** * Returns `true` if the mesh is completely in the frustum defined be the passed array of planes. * A mesh is completely in the frustum if its bounding box it completely inside the frustum. * @param frustumPlanes defines the frustum to test * @returns true if the mesh is completely in the frustum planes */ isCompletelyInFrustum(e) { return this.getBoundingInfo().isCompletelyInFrustum(e); } /** * True if the mesh intersects another mesh or a SolidParticle object * @param mesh defines a target mesh or SolidParticle to test * @param precise Unless the parameter `precise` is set to `true` the intersection is computed according to Axis Aligned Bounding Boxes (AABB), else according to OBB (Oriented BBoxes) * @param includeDescendants Can be set to true to test if the mesh defined in parameters intersects with the current mesh or any child meshes * @returns true if there is an intersection */ intersectsMesh(e, t = !1, r) { const n = this.getBoundingInfo(), i = e.getBoundingInfo(); if (n.intersects(i, t)) return !0; if (r) { for (const s of this.getChildMeshes()) if (s.intersectsMesh(e, t, !0)) return !0; } return !1; } /** * Returns true if the passed point (Vector3) is inside the mesh bounding box * @param point defines the point to test * @returns true if there is an intersection */ intersectsPoint(e) { return this.getBoundingInfo().intersectsPoint(e); } // Collisions /** * Gets or sets a boolean indicating that this mesh can be used in the collision engine * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions */ get checkCollisions() { return this._internalAbstractMeshDataInfo._meshCollisionData._checkCollisions; } set checkCollisions(e) { this._internalAbstractMeshDataInfo._meshCollisionData._checkCollisions = e; } /** * Gets Collider object used to compute collisions (not physics) * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions */ get collider() { return this._internalAbstractMeshDataInfo._meshCollisionData._collider; } /** * Move the mesh using collision engine * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions * @param displacement defines the requested displacement vector * @returns the current mesh */ moveWithCollisions(e) { this.getAbsolutePosition().addToRef(this.ellipsoidOffset, this._internalAbstractMeshDataInfo._meshCollisionData._oldPositionForCollisions); const r = this.getScene().collisionCoordinator; return this._internalAbstractMeshDataInfo._meshCollisionData._collider || (this._internalAbstractMeshDataInfo._meshCollisionData._collider = r.createCollider()), this._internalAbstractMeshDataInfo._meshCollisionData._collider._radius = this.ellipsoid, r.getNewPosition(this._internalAbstractMeshDataInfo._meshCollisionData._oldPositionForCollisions, e, this._internalAbstractMeshDataInfo._meshCollisionData._collider, this.collisionRetryCount, this, this._onCollisionPositionChange, this.uniqueId), this; } // Collisions /** * @internal */ _collideForSubMesh(e, t, r) { var n; if (this._generatePointsArray(), !this._positions) return this; if (!e._lastColliderWorldVertices || !e._lastColliderTransformMatrix.equals(t)) { e._lastColliderTransformMatrix = t.clone(), e._lastColliderWorldVertices = [], e._trianglePlanes = []; const i = e.verticesStart, s = e.verticesStart + e.verticesCount; for (let a = i; a < s; a++) e._lastColliderWorldVertices.push(S.TransformCoordinates(this._positions[a], t)); } return r._collide(e._trianglePlanes, e._lastColliderWorldVertices, this.getIndices(), e.indexStart, e.indexStart + e.indexCount, e.verticesStart, !!e.getMaterial(), this, this._shouldConvertRHS(), ((n = e.getMaterial()) === null || n === void 0 ? void 0 : n.fillMode) === 7), this; } /** * @internal */ _processCollisionsForSubMeshes(e, t) { const r = this._scene.getCollidingSubMeshCandidates(this, e), n = r.length; for (let i = 0; i < n; i++) { const s = r.data[i]; n > 1 && !s._checkCollision(e) || this._collideForSubMesh(s, t, e); } return this; } /** @internal */ _shouldConvertRHS() { return !1; } /** * @internal */ _checkCollision(e) { if (!this.getBoundingInfo()._checkCollision(e)) return this; const t = ue.Matrix[0], r = ue.Matrix[1]; return he.ScalingToRef(1 / e._radius.x, 1 / e._radius.y, 1 / e._radius.z, t), this.worldMatrixFromCache.multiplyToRef(t, r), this._processCollisionsForSubMeshes(e, r), this; } // Picking /** @internal */ _generatePointsArray() { return !1; } /** * Checks if the passed Ray intersects with the mesh. A mesh triangle can be picked both from its front and back sides, * irrespective of orientation. * @param ray defines the ray to use. It should be in the mesh's LOCAL coordinate space. * @param fastCheck defines if fast mode (but less precise) must be used (false by default) * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected * @param onlyBoundingInfo defines a boolean indicating if picking should only happen using bounding info (false by default) * @param worldToUse defines the world matrix to use to get the world coordinate of the intersection point * @param skipBoundingInfo a boolean indicating if we should skip the bounding info check * @returns the picking info * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/interactions/mesh_intersect */ intersects(e, t, r, n = !1, i, s = !1) { const a = new F9(), f = this.getClassName(), o = f === "InstancedLinesMesh" || f === "LinesMesh" || f === "GreasedLineMesh" ? this.intersectionThreshold : 0, d = this.getBoundingInfo(); if (!this.subMeshes || !s && (!e.intersectsSphere(d.boundingSphere, o) || !e.intersectsBox(d.boundingBox, o))) return a; if (n) return a.hit = !s, a.pickedMesh = s ? null : this, a.distance = s ? 0 : S.Distance(e.origin, d.boundingSphere.center), a.subMeshId = 0, a; if (!this._generatePointsArray()) return a; let v = null; const u = this._scene.getIntersectingSubMeshCandidates(this, e), l = u.length; let P = !1; for (let p = 0; p < l; p++) { const H = u.data[p].getMaterial(); if (H && (H.fillMode == 7 || H.fillMode == 0 || H.fillMode == 1 || H.fillMode == 2 || H.fillMode == 4)) { P = !0; break; } } if (!P) return a.hit = !0, a.pickedMesh = this, a.distance = S.Distance(e.origin, d.boundingSphere.center), a.subMeshId = -1, a; for (let p = 0; p < l; p++) { const c = u.data[p]; if (l > 1 && !s && !c.canIntersects(e)) continue; const H = c.intersects(e, this._positions, this.getIndices(), t, r); if (H && (t || !v || H.distance < v.distance) && (v = H, v.subMeshId = p, t)) break; } if (v) { const p = i ?? this.getWorldMatrix(), c = ue.Vector3[0], H = ue.Vector3[1]; S.TransformCoordinatesToRef(e.origin, p, c), e.direction.scaleToRef(v.distance, H); const q = S.TransformNormal(H, p).addInPlace(c); return a.hit = !0, a.distance = S.Distance(c, q), a.pickedPoint = q, a.pickedMesh = this, a.bu = v.bu || 0, a.bv = v.bv || 0, a.subMeshFaceId = v.faceId, a.faceId = v.faceId + u.data[v.subMeshId].indexStart / (this.getClassName().indexOf("LinesMesh") !== -1 ? 2 : 3), a.subMeshId = v.subMeshId, a; } return a; } /** * Clones the current mesh * @param name defines the mesh name * @param newParent defines the new mesh parent * @param doNotCloneChildren defines a boolean indicating that children must not be cloned (false by default) * @returns the new mesh */ // eslint-disable-next-line @typescript-eslint/no-unused-vars clone(e, t, r) { return null; } /** * Disposes all the submeshes of the current meshnp * @returns the current mesh */ releaseSubMeshes() { if (this.subMeshes) for (; this.subMeshes.length; ) this.subMeshes[0].dispose(); else this.subMeshes = []; return this; } /** * Releases resources associated with this abstract mesh. * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default) * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default) */ dispose(e, t = !1) { let r; const n = this.getScene(); for (this._scene.useMaterialMeshMap && this._internalAbstractMeshDataInfo._material && this._internalAbstractMeshDataInfo._material.meshMap && (this._internalAbstractMeshDataInfo._material.meshMap[this.uniqueId] = void 0), n.freeActiveMeshes(), n.freeRenderingGroups(), n.renderingManager.maintainStateBetweenFrames && n.renderingManager.restoreDispachedFlags(), this.actionManager !== void 0 && this.actionManager !== null && (this._scene.meshes.some((a) => a !== this && a.actionManager === this.actionManager) || this.actionManager.dispose(), this.actionManager = null), this._internalAbstractMeshDataInfo._skeleton = null, this._transformMatrixTexture && (this._transformMatrixTexture.dispose(), this._transformMatrixTexture = null), r = 0; r < this._intersectionsInProgress.length; r++) { const a = this._intersectionsInProgress[r], f = a._intersectionsInProgress.indexOf(this); a._intersectionsInProgress.splice(f, 1); } this._intersectionsInProgress.length = 0, n.lights.forEach((a) => { let f = a.includedOnlyMeshes.indexOf(this); f !== -1 && a.includedOnlyMeshes.splice(f, 1), f = a.excludedMeshes.indexOf(this), f !== -1 && a.excludedMeshes.splice(f, 1); const o = a.getShadowGenerators(); if (o) { const d = o.values(); for (let v = d.next(); v.done !== !0; v = d.next()) { const l = v.value.getShadowMap(); l && l.renderList && (f = l.renderList.indexOf(this), f !== -1 && l.renderList.splice(f, 1)); } } }), (this.getClassName() !== "InstancedMesh" || this.getClassName() !== "InstancedLinesMesh") && this.releaseSubMeshes(); const s = n.getEngine(); if (this._occlusionQuery !== null && (this.isOcclusionQueryInProgress = !1, s.deleteQuery(this._occlusionQuery), this._occlusionQuery = null), s.wipeCaches(), n.removeMesh(this), this._parentContainer) { const a = this._parentContainer.meshes.indexOf(this); a > -1 && this._parentContainer.meshes.splice(a, 1), this._parentContainer = null; } if (t && this.material && (this.material.getClassName() === "MultiMaterial" ? this.material.dispose(!1, !0, !0) : this.material.dispose(!1, !0)), !e) for (r = 0; r < n.particleSystems.length; r++) n.particleSystems[r].emitter === this && (n.particleSystems[r].dispose(), r--); this._internalAbstractMeshDataInfo._facetData.facetDataEnabled && this.disableFacetData(), this._uniformBuffer.dispose(), this.onAfterWorldMatrixUpdateObservable.clear(), this.onCollideObservable.clear(), this.onCollisionPositionChangeObservable.clear(), this.onRebuildObservable.clear(), super.dispose(e, t); } /** * Adds the passed mesh as a child to the current mesh * @param mesh defines the child mesh * @param preserveScalingSign if true, keep scaling sign of child. Otherwise, scaling sign might change. * @returns the current mesh */ addChild(e, t = !1) { return e.setParent(this, t), this; } /** * Removes the passed mesh from the current mesh children list * @param mesh defines the child mesh * @param preserveScalingSign if true, keep scaling sign of child. Otherwise, scaling sign might change. * @returns the current mesh */ removeChild(e, t = !1) { return e.setParent(null, t), this; } // Facet data /** @internal */ _initFacetData() { const e = this._internalAbstractMeshDataInfo._facetData; e.facetNormals || (e.facetNormals = []), e.facetPositions || (e.facetPositions = []), e.facetPartitioning || (e.facetPartitioning = new Array()), e.facetNb = this.getIndices().length / 3 | 0, e.partitioningSubdivisions = e.partitioningSubdivisions ? e.partitioningSubdivisions : 10, e.partitioningBBoxRatio = e.partitioningBBoxRatio ? e.partitioningBBoxRatio : 1.01; for (let t = 0; t < e.facetNb; t++) e.facetNormals[t] = S.Zero(), e.facetPositions[t] = S.Zero(); return e.facetDataEnabled = !0, this; } /** * Updates the mesh facetData arrays and the internal partitioning when the mesh is morphed or updated. * This method can be called within the render loop. * You don't need to call this method by yourself in the render loop when you update/morph a mesh with the methods CreateXXX() as they automatically manage this computation * @returns the current mesh * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData */ updateFacetData() { const e = this._internalAbstractMeshDataInfo._facetData; e.facetDataEnabled || this._initFacetData(); const t = this.getVerticesData(J.PositionKind), r = this.getIndices(), n = this.getVerticesData(J.NormalKind), i = this.getBoundingInfo(); if (e.facetDepthSort && !e.facetDepthSortEnabled) { if (e.facetDepthSortEnabled = !0, r instanceof Uint16Array) e.depthSortedIndices = new Uint16Array(r); else if (r instanceof Uint32Array) e.depthSortedIndices = new Uint32Array(r); else { let a = !1; for (let f = 0; f < r.length; f++) if (r[f] > 65535) { a = !0; break; } a ? e.depthSortedIndices = new Uint32Array(r) : e.depthSortedIndices = new Uint16Array(r); } if (e.facetDepthSortFunction = function(a, f) { return f.sqDistance - a.sqDistance; }, !e.facetDepthSortFrom) { const a = this.getScene().activeCamera; e.facetDepthSortFrom = a ? a.position : S.Zero(); } e.depthSortedFacets = []; for (let a = 0; a < e.facetNb; a++) { const f = { ind: a * 3, sqDistance: 0 }; e.depthSortedFacets.push(f); } e.invertedMatrix = he.Identity(), e.facetDepthSortOrigin = S.Zero(); } e.bbSize.x = i.maximum.x - i.minimum.x > Dn ? i.maximum.x - i.minimum.x : Dn, e.bbSize.y = i.maximum.y - i.minimum.y > Dn ? i.maximum.y - i.minimum.y : Dn, e.bbSize.z = i.maximum.z - i.minimum.z > Dn ? i.maximum.z - i.minimum.z : Dn; let s = e.bbSize.x > e.bbSize.y ? e.bbSize.x : e.bbSize.y; if (s = s > e.bbSize.z ? s : e.bbSize.z, e.subDiv.max = e.partitioningSubdivisions, e.subDiv.X = Math.floor(e.subDiv.max * e.bbSize.x / s), e.subDiv.Y = Math.floor(e.subDiv.max * e.bbSize.y / s), e.subDiv.Z = Math.floor(e.subDiv.max * e.bbSize.z / s), e.subDiv.X = e.subDiv.X < 1 ? 1 : e.subDiv.X, e.subDiv.Y = e.subDiv.Y < 1 ? 1 : e.subDiv.Y, e.subDiv.Z = e.subDiv.Z < 1 ? 1 : e.subDiv.Z, e.facetParameters.facetNormals = this.getFacetLocalNormals(), e.facetParameters.facetPositions = this.getFacetLocalPositions(), e.facetParameters.facetPartitioning = this.getFacetLocalPartitioning(), e.facetParameters.bInfo = i, e.facetParameters.bbSize = e.bbSize, e.facetParameters.subDiv = e.subDiv, e.facetParameters.ratio = this.partitioningBBoxRatio, e.facetParameters.depthSort = e.facetDepthSort, e.facetDepthSort && e.facetDepthSortEnabled && (this.computeWorldMatrix(!0), this._worldMatrix.invertToRef(e.invertedMatrix), S.TransformCoordinatesToRef(e.facetDepthSortFrom, e.invertedMatrix, e.facetDepthSortOrigin), e.facetParameters.distanceTo = e.facetDepthSortOrigin), e.facetParameters.depthSortedFacets = e.depthSortedFacets, n && Ut.ComputeNormals(t, r, n, e.facetParameters), e.facetDepthSort && e.facetDepthSortEnabled) { e.depthSortedFacets.sort(e.facetDepthSortFunction); const a = e.depthSortedIndices.length / 3 | 0; for (let f = 0; f < a; f++) { const o = e.depthSortedFacets[f].ind; e.depthSortedIndices[f * 3] = r[o], e.depthSortedIndices[f * 3 + 1] = r[o + 1], e.depthSortedIndices[f * 3 + 2] = r[o + 2]; } this.updateIndices(e.depthSortedIndices, void 0, !0); } return this; } /** * Returns the facetLocalNormals array. * The normals are expressed in the mesh local spac * @returns an array of Vector3 * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData */ getFacetLocalNormals() { const e = this._internalAbstractMeshDataInfo._facetData; return e.facetNormals || this.updateFacetData(), e.facetNormals; } /** * Returns the facetLocalPositions array. * The facet positions are expressed in the mesh local space * @returns an array of Vector3 * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData */ getFacetLocalPositions() { const e = this._internalAbstractMeshDataInfo._facetData; return e.facetPositions || this.updateFacetData(), e.facetPositions; } /** * Returns the facetLocalPartitioning array * @returns an array of array of numbers * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData */ getFacetLocalPartitioning() { const e = this._internalAbstractMeshDataInfo._facetData; return e.facetPartitioning || this.updateFacetData(), e.facetPartitioning; } /** * Returns the i-th facet position in the world system. * This method allocates a new Vector3 per call * @param i defines the facet index * @returns a new Vector3 * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData */ getFacetPosition(e) { const t = S.Zero(); return this.getFacetPositionToRef(e, t), t; } /** * Sets the reference Vector3 with the i-th facet position in the world system * @param i defines the facet index * @param ref defines the target vector * @returns the current mesh * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData */ getFacetPositionToRef(e, t) { const r = this.getFacetLocalPositions()[e], n = this.getWorldMatrix(); return S.TransformCoordinatesToRef(r, n, t), this; } /** * Returns the i-th facet normal in the world system. * This method allocates a new Vector3 per call * @param i defines the facet index * @returns a new Vector3 * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData */ getFacetNormal(e) { const t = S.Zero(); return this.getFacetNormalToRef(e, t), t; } /** * Sets the reference Vector3 with the i-th facet normal in the world system * @param i defines the facet index * @param ref defines the target vector * @returns the current mesh * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData */ getFacetNormalToRef(e, t) { const r = this.getFacetLocalNormals()[e]; return S.TransformNormalToRef(r, this.getWorldMatrix(), t), this; } /** * Returns the facets (in an array) in the same partitioning block than the one the passed coordinates are located (expressed in the mesh local system) * @param x defines x coordinate * @param y defines y coordinate * @param z defines z coordinate * @returns the array of facet indexes * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData */ getFacetsAtLocalCoordinates(e, t, r) { const n = this.getBoundingInfo(), i = this._internalAbstractMeshDataInfo._facetData, s = Math.floor((e - n.minimum.x * i.partitioningBBoxRatio) * i.subDiv.X * i.partitioningBBoxRatio / i.bbSize.x), a = Math.floor((t - n.minimum.y * i.partitioningBBoxRatio) * i.subDiv.Y * i.partitioningBBoxRatio / i.bbSize.y), f = Math.floor((r - n.minimum.z * i.partitioningBBoxRatio) * i.subDiv.Z * i.partitioningBBoxRatio / i.bbSize.z); return s < 0 || s > i.subDiv.max || a < 0 || a > i.subDiv.max || f < 0 || f > i.subDiv.max ? null : i.facetPartitioning[s + i.subDiv.max * a + i.subDiv.max * i.subDiv.max * f]; } /** * Returns the closest mesh facet index at (x,y,z) World coordinates, null if not found * @param x defines x coordinate * @param y defines y coordinate * @param z defines z coordinate * @param projected sets as the (x,y,z) world projection on the facet * @param checkFace if true (default false), only the facet "facing" to (x,y,z) or only the ones "turning their backs", according to the parameter "facing" are returned * @param facing if facing and checkFace are true, only the facet "facing" to (x, y, z) are returned : positive dot (x, y, z) * facet position. If facing si false and checkFace is true, only the facet "turning their backs" to (x, y, z) are returned : negative dot (x, y, z) * facet position * @returns the face index if found (or null instead) * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData */ getClosestFacetAtCoordinates(e, t, r, n, i = !1, s = !0) { const a = this.getWorldMatrix(), f = ue.Matrix[5]; a.invertToRef(f); const o = ue.Vector3[8]; S.TransformCoordinatesFromFloatsToRef(e, t, r, f, o); const d = this.getClosestFacetAtLocalCoordinates(o.x, o.y, o.z, n, i, s); return n && S.TransformCoordinatesFromFloatsToRef(n.x, n.y, n.z, a, n), d; } /** * Returns the closest mesh facet index at (x,y,z) local coordinates, null if not found * @param x defines x coordinate * @param y defines y coordinate * @param z defines z coordinate * @param projected sets as the (x,y,z) local projection on the facet * @param checkFace if true (default false), only the facet "facing" to (x,y,z) or only the ones "turning their backs", according to the parameter "facing" are returned * @param facing if facing and checkFace are true, only the facet "facing" to (x, y, z) are returned : positive dot (x, y, z) * facet position. If facing si false and checkFace is true, only the facet "turning their backs" to (x, y, z) are returned : negative dot (x, y, z) * facet position * @returns the face index if found (or null instead) * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData */ getClosestFacetAtLocalCoordinates(e, t, r, n, i = !1, s = !0) { let a = null, f = 0, o = 0, d = 0, v = 0, u = 0, l = 0, P = 0, p = 0; const c = this.getFacetLocalPositions(), H = this.getFacetLocalNormals(), T = this.getFacetsAtLocalCoordinates(e, t, r); if (!T) return null; let q = Number.MAX_VALUE, b = q, j, w, m; for (let I = 0; I < T.length; I++) j = T[I], w = H[j], m = c[j], v = (e - m.x) * w.x + (t - m.y) * w.y + (r - m.z) * w.z, (!i || i && s && v >= 0 || i && !s && v <= 0) && (v = w.x * m.x + w.y * m.y + w.z * m.z, u = -(w.x * e + w.y * t + w.z * r - v) / (w.x * w.x + w.y * w.y + w.z * w.z), l = e + w.x * u, P = t + w.y * u, p = r + w.z * u, f = l - e, o = P - t, d = p - r, b = f * f + o * o + d * d, b < q && (q = b, a = j, n && (n.x = l, n.y = P, n.z = p))); return a; } /** * Returns the object "parameter" set with all the expected parameters for facetData computation by ComputeNormals() * @returns the parameters * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData */ getFacetDataParameters() { return this._internalAbstractMeshDataInfo._facetData.facetParameters; } /** * Disables the feature FacetData and frees the related memory * @returns the current mesh * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData */ disableFacetData() { const e = this._internalAbstractMeshDataInfo._facetData; return e.facetDataEnabled && (e.facetDataEnabled = !1, e.facetPositions = [], e.facetNormals = [], e.facetPartitioning = new Array(), e.facetParameters = null, e.depthSortedIndices = new Uint32Array(0)), this; } /** * Updates the AbstractMesh indices array * @param indices defines the data source * @param offset defines the offset in the index buffer where to store the new data (can be null) * @param gpuMemoryOnly defines a boolean indicating that only the GPU memory must be updated leaving the CPU version of the indices unchanged (false by default) * @returns the current mesh */ // eslint-disable-next-line @typescript-eslint/no-unused-vars updateIndices(e, t, r = !1) { return this; } /** * Creates new normals data for the mesh * @param updatable defines if the normal vertex buffer must be flagged as updatable * @returns the current mesh */ createNormals(e) { const t = this.getVerticesData(J.PositionKind), r = this.getIndices(); let n; return this.isVerticesDataPresent(J.NormalKind) ? n = this.getVerticesData(J.NormalKind) : n = [], Ut.ComputeNormals(t, r, n, { useRightHandedSystem: this.getScene().useRightHandedSystem }), this.setVerticesData(J.NormalKind, n, e), this; } /** * Align the mesh with a normal * @param normal defines the normal to use * @param upDirection can be used to redefined the up vector to use (will use the (0, 1, 0) by default) * @returns the current mesh */ alignWithNormal(e, t) { t || (t = bf.Y); const r = ue.Vector3[0], n = ue.Vector3[1]; return S.CrossToRef(t, e, n), S.CrossToRef(e, n, r), this.rotationQuaternion ? Ze.RotationQuaternionFromAxisToRef(r, e, n, this.rotationQuaternion) : S.RotationFromAxisToRef(r, e, n, this.rotation), this; } /** @internal */ _checkOcclusionQuery() { return !1; } /** * Disables the mesh edge rendering mode * @returns the currentAbstractMesh */ disableEdgesRendering() { throw qn("EdgesRenderer"); } /** * Enables the edge rendering mode on the mesh. * This mode makes the mesh edges visible * @param epsilon defines the maximal distance between two angles to detect a face * @param checkVerticesInsteadOfIndices indicates that we should check vertex list directly instead of faces * @param options options to the edge renderer * @returns the currentAbstractMesh * @see https://www.babylonjs-playground.com/#19O9TU#0 */ // eslint-disable-next-line @typescript-eslint/no-unused-vars enableEdgesRendering(e, t, r) { throw qn("EdgesRenderer"); } /** * This function returns all of the particle systems in the scene that use the mesh as an emitter. * @returns an array of particle systems in the scene that use the mesh as an emitter */ getConnectedParticleSystems() { return this._scene.particleSystems.filter((e) => e.emitter === this); } } jn.OCCLUSION_TYPE_NONE = 0; jn.OCCLUSION_TYPE_OPTIMISTIC = 1; jn.OCCLUSION_TYPE_STRICT = 2; jn.OCCLUSION_ALGORITHM_TYPE_ACCURATE = 0; jn.OCCLUSION_ALGORITHM_TYPE_CONSERVATIVE = 1; jn.CULLINGSTRATEGY_STANDARD = 0; jn.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY = 1; jn.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION = 2; jn.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY = 3; Ue("BABYLON.AbstractMesh", jn); function Mf(A) { A.indexOf("vClipPlane") === -1 && A.push("vClipPlane"), A.indexOf("vClipPlane2") === -1 && A.push("vClipPlane2"), A.indexOf("vClipPlane3") === -1 && A.push("vClipPlane3"), A.indexOf("vClipPlane4") === -1 && A.push("vClipPlane4"), A.indexOf("vClipPlane5") === -1 && A.push("vClipPlane5"), A.indexOf("vClipPlane6") === -1 && A.push("vClipPlane6"); } function xq(A, e, t) { var r, n, i, s, a, f; const o = !!((r = A.clipPlane) !== null && r !== void 0 ? r : e.clipPlane), d = !!((n = A.clipPlane2) !== null && n !== void 0 ? n : e.clipPlane2), v = !!((i = A.clipPlane3) !== null && i !== void 0 ? i : e.clipPlane3), u = !!((s = A.clipPlane4) !== null && s !== void 0 ? s : e.clipPlane4), l = !!((a = A.clipPlane5) !== null && a !== void 0 ? a : e.clipPlane5), P = !!((f = A.clipPlane6) !== null && f !== void 0 ? f : e.clipPlane6); o && t.push("#define CLIPPLANE"), d && t.push("#define CLIPPLANE2"), v && t.push("#define CLIPPLANE3"), u && t.push("#define CLIPPLANE4"), l && t.push("#define CLIPPLANE5"), P && t.push("#define CLIPPLANE6"); } function oee(A, e, t) { var r, n, i, s, a, f; let o = !1; const d = !!((r = A.clipPlane) !== null && r !== void 0 ? r : e.clipPlane), v = !!((n = A.clipPlane2) !== null && n !== void 0 ? n : e.clipPlane2), u = !!((i = A.clipPlane3) !== null && i !== void 0 ? i : e.clipPlane3), l = !!((s = A.clipPlane4) !== null && s !== void 0 ? s : e.clipPlane4), P = !!((a = A.clipPlane5) !== null && a !== void 0 ? a : e.clipPlane5), p = !!((f = A.clipPlane6) !== null && f !== void 0 ? f : e.clipPlane6); return t.CLIPPLANE !== d && (t.CLIPPLANE = d, o = !0), t.CLIPPLANE2 !== v && (t.CLIPPLANE2 = v, o = !0), t.CLIPPLANE3 !== u && (t.CLIPPLANE3 = u, o = !0), t.CLIPPLANE4 !== l && (t.CLIPPLANE4 = l, o = !0), t.CLIPPLANE5 !== P && (t.CLIPPLANE5 = P, o = !0), t.CLIPPLANE6 !== p && (t.CLIPPLANE6 = p, o = !0), o; } function Df(A, e, t) { var r, n, i, s, a, f; let o = (r = e.clipPlane) !== null && r !== void 0 ? r : t.clipPlane; wW(A, "vClipPlane", o), o = (n = e.clipPlane2) !== null && n !== void 0 ? n : t.clipPlane2, wW(A, "vClipPlane2", o), o = (i = e.clipPlane3) !== null && i !== void 0 ? i : t.clipPlane3, wW(A, "vClipPlane3", o), o = (s = e.clipPlane4) !== null && s !== void 0 ? s : t.clipPlane4, wW(A, "vClipPlane4", o), o = (a = e.clipPlane5) !== null && a !== void 0 ? a : t.clipPlane5, wW(A, "vClipPlane5", o), o = (f = e.clipPlane6) !== null && f !== void 0 ? f : t.clipPlane6, wW(A, "vClipPlane6", o); } function wW(A, e, t) { t && A.setFloat4(e, t.normal.x, t.normal.y, t.normal.z, t.d); } class Ye { /** * Binds the scene's uniform buffer to the effect. * @param effect defines the effect to bind to the scene uniform buffer * @param sceneUbo defines the uniform buffer storing scene data */ static BindSceneUniformBuffer(e, t) { t.bindToEffect(e, "Scene"); } /** * Helps preparing the defines values about the UVs in used in the effect. * UVs are shared as much as we can across channels in the shaders. * @param texture The texture we are preparing the UVs for * @param defines The defines to update * @param key The channel key "diffuse", "specular"... used in the shader */ static PrepareDefinesForMergedUV(e, t, r) { t._needUVs = !0, t[r] = !0, e.optimizeUVAllocation && e.getTextureMatrix().isIdentityAs3x2() ? (t[r + "DIRECTUV"] = e.coordinatesIndex + 1, t["MAINUV" + (e.coordinatesIndex + 1)] = !0) : t[r + "DIRECTUV"] = 0; } /** * Binds a texture matrix value to its corresponding uniform * @param texture The texture to bind the matrix for * @param uniformBuffer The uniform buffer receiving the data * @param key The channel key "diffuse", "specular"... used in the shader */ static BindTextureMatrix(e, t, r) { const n = e.getTextureMatrix(); t.updateMatrix(r + "Matrix", n); } /** * Gets the current status of the fog (should it be enabled?) * @param mesh defines the mesh to evaluate for fog support * @param scene defines the hosting scene * @returns true if fog must be enabled */ static GetFogState(e, t) { return t.fogEnabled && e.applyFog && t.fogMode !== sr.FOGMODE_NONE; } /** * Helper used to prepare the list of defines associated with misc. values for shader compilation * @param mesh defines the current mesh * @param scene defines the current scene * @param useLogarithmicDepth defines if logarithmic depth has to be turned on * @param pointsCloud defines if point cloud rendering has to be turned on * @param fogEnabled defines if fog has to be turned on * @param alphaTest defines if alpha testing has to be turned on * @param defines defines the current list of defines * @param applyDecalAfterDetail Defines if the decal is applied after or before the detail */ static PrepareDefinesForMisc(e, t, r, n, i, s, a, f = !1) { a._areMiscDirty && (a.LOGARITHMICDEPTH = r, a.POINTSIZE = n, a.FOG = i && this.GetFogState(e, t), a.NONUNIFORMSCALING = e.nonUniformScaling, a.ALPHATEST = s, a.DECAL_AFTER_DETAIL = f); } /** * Helper used to prepare the defines relative to the active camera * @param scene defines the current scene * @param defines specifies the list of active defines * @returns true if the defines have been updated, else false */ static PrepareDefinesForCamera(e, t) { let r = !1; if (e.activeCamera) { const n = t.CAMERA_ORTHOGRAPHIC ? 1 : 0, i = t.CAMERA_PERSPECTIVE ? 1 : 0, s = e.activeCamera.mode === Tr.ORTHOGRAPHIC_CAMERA ? 1 : 0, a = e.activeCamera.mode === Tr.PERSPECTIVE_CAMERA ? 1 : 0; (n ^ s || i ^ a) && (t.CAMERA_ORTHOGRAPHIC = s === 1, t.CAMERA_PERSPECTIVE = a === 1, r = !0); } return r; } /** * Helper used to prepare the list of defines associated with frame values for shader compilation * @param scene defines the current scene * @param engine defines the current engine * @param material defines the material we are compiling the shader for * @param defines specifies the list of active defines * @param useInstances defines if instances have to be turned on * @param useClipPlane defines if clip plane have to be turned on * @param useThinInstances defines if thin instances have to be turned on */ static PrepareDefinesForFrameBoundValues(e, t, r, n, i, s = null, a = !1) { let f = Ye.PrepareDefinesForCamera(e, n); s !== !1 && (f = oee(r, e, n)), n.DEPTHPREPASS !== !t.getColorWrite() && (n.DEPTHPREPASS = !n.DEPTHPREPASS, f = !0), n.INSTANCES !== i && (n.INSTANCES = i, f = !0), n.THIN_INSTANCES !== a && (n.THIN_INSTANCES = a, f = !0), f && n.markAsUnprocessed(); } /** * Prepares the defines for bones * @param mesh The mesh containing the geometry data we will draw * @param defines The defines to update */ static PrepareDefinesForBones(e, t) { if (e.useBones && e.computeBonesUsingShaders && e.skeleton) { t.NUM_BONE_INFLUENCERS = e.numBoneInfluencers; const r = t.BONETEXTURE !== void 0; if (e.skeleton.isUsingTextureForMatrices && r) t.BONETEXTURE = !0; else { t.BonesPerMesh = e.skeleton.bones.length + 1, t.BONETEXTURE = r ? !1 : void 0; const n = e.getScene().prePassRenderer; if (n && n.enabled) { const i = n.excludedSkinnedMesh.indexOf(e) === -1; t.BONES_VELOCITY_ENABLED = i; } } } else t.NUM_BONE_INFLUENCERS = 0, t.BonesPerMesh = 0, t.BONETEXTURE !== void 0 && (t.BONETEXTURE = !1); } /** * Prepares the defines for morph targets * @param mesh The mesh containing the geometry data we will draw * @param defines The defines to update */ static PrepareDefinesForMorphTargets(e, t) { const r = e.morphTargetManager; r ? (t.MORPHTARGETS_UV = r.supportsUVs && t.UV1, t.MORPHTARGETS_TANGENT = r.supportsTangents && t.TANGENT, t.MORPHTARGETS_NORMAL = r.supportsNormals && t.NORMAL, t.MORPHTARGETS = r.numInfluencers > 0, t.NUM_MORPH_INFLUENCERS = r.numInfluencers, t.MORPHTARGETS_TEXTURE = r.isUsingTextureForTargets) : (t.MORPHTARGETS_UV = !1, t.MORPHTARGETS_TANGENT = !1, t.MORPHTARGETS_NORMAL = !1, t.MORPHTARGETS = !1, t.NUM_MORPH_INFLUENCERS = 0); } /** * Prepares the defines for baked vertex animation * @param mesh The mesh containing the geometry data we will draw * @param defines The defines to update */ static PrepareDefinesForBakedVertexAnimation(e, t) { const r = e.bakedVertexAnimationManager; t.BAKED_VERTEX_ANIMATION_TEXTURE = !!(r && r.isEnabled); } /** * Prepares the defines used in the shader depending on the attributes data available in the mesh * @param mesh The mesh containing the geometry data we will draw * @param defines The defines to update * @param useVertexColor Precise whether vertex colors should be used or not (override mesh info) * @param useBones Precise whether bones should be used or not (override mesh info) * @param useMorphTargets Precise whether morph targets should be used or not (override mesh info) * @param useVertexAlpha Precise whether vertex alpha should be used or not (override mesh info) * @param useBakedVertexAnimation Precise whether baked vertex animation should be used or not (override mesh info) * @returns false if defines are considered not dirty and have not been checked */ static PrepareDefinesForAttributes(e, t, r, n, i = !1, s = !0, a = !0) { if (!t._areAttributesDirty && t._needNormals === t._normals && t._needUVs === t._uvs) return !1; t._normals = t._needNormals, t._uvs = t._needUVs, t.NORMAL = t._needNormals && e.isVerticesDataPresent(J.NormalKind), t._needNormals && e.isVerticesDataPresent(J.TangentKind) && (t.TANGENT = !0); for (let f = 1; f <= 6; ++f) t["UV" + f] = t._needUVs ? e.isVerticesDataPresent(`uv${f === 1 ? "" : f}`) : !1; if (r) { const f = e.useVertexColors && e.isVerticesDataPresent(J.ColorKind); t.VERTEXCOLOR = f, t.VERTEXALPHA = e.hasVertexAlpha && f && s; } return e.isVerticesDataPresent(J.ColorInstanceKind) && (e.hasInstances || e.hasThinInstances) && (t.INSTANCESCOLOR = !0), n && this.PrepareDefinesForBones(e, t), i && this.PrepareDefinesForMorphTargets(e, t), a && this.PrepareDefinesForBakedVertexAnimation(e, t), !0; } /** * Prepares the defines related to multiview * @param scene The scene we are intending to draw * @param defines The defines to update */ static PrepareDefinesForMultiview(e, t) { if (e.activeCamera) { const r = t.MULTIVIEW; t.MULTIVIEW = e.activeCamera.outputRenderTarget !== null && e.activeCamera.outputRenderTarget.getViewCount() > 1, t.MULTIVIEW != r && t.markAsUnprocessed(); } } /** * Prepares the defines related to order independant transparency * @param scene The scene we are intending to draw * @param defines The defines to update * @param needAlphaBlending Determines if the material needs alpha blending */ static PrepareDefinesForOIT(e, t, r) { const n = t.ORDER_INDEPENDENT_TRANSPARENCY, i = t.ORDER_INDEPENDENT_TRANSPARENCY_16BITS; t.ORDER_INDEPENDENT_TRANSPARENCY = e.useOrderIndependentTransparency && r, t.ORDER_INDEPENDENT_TRANSPARENCY_16BITS = !e.getEngine().getCaps().textureFloatLinearFiltering, (n !== t.ORDER_INDEPENDENT_TRANSPARENCY || i !== t.ORDER_INDEPENDENT_TRANSPARENCY_16BITS) && t.markAsUnprocessed(); } /** * Prepares the defines related to the prepass * @param scene The scene we are intending to draw * @param defines The defines to update * @param canRenderToMRT Indicates if this material renders to several textures in the prepass */ static PrepareDefinesForPrePass(e, t, r) { const n = t.PREPASS; if (!t._arePrePassDirty) return; const i = [ { type: 1, define: "PREPASS_POSITION", index: "PREPASS_POSITION_INDEX" }, { type: 2, define: "PREPASS_VELOCITY", index: "PREPASS_VELOCITY_INDEX" }, { type: 3, define: "PREPASS_REFLECTIVITY", index: "PREPASS_REFLECTIVITY_INDEX" }, { type: 0, define: "PREPASS_IRRADIANCE", index: "PREPASS_IRRADIANCE_INDEX" }, { type: 7, define: "PREPASS_ALBEDO_SQRT", index: "PREPASS_ALBEDO_SQRT_INDEX" }, { type: 5, define: "PREPASS_DEPTH", index: "PREPASS_DEPTH_INDEX" }, { type: 6, define: "PREPASS_NORMAL", index: "PREPASS_NORMAL_INDEX" } ]; if (e.prePassRenderer && e.prePassRenderer.enabled && r) { t.PREPASS = !0, t.SCENE_MRT_COUNT = e.prePassRenderer.mrtCount, t.PREPASS_NORMAL_WORLDSPACE = e.prePassRenderer.generateNormalsInWorldSpace; for (let s = 0; s < i.length; s++) { const a = e.prePassRenderer.getIndex(i[s].type); a !== -1 ? (t[i[s].define] = !0, t[i[s].index] = a) : t[i[s].define] = !1; } } else { t.PREPASS = !1; for (let s = 0; s < i.length; s++) t[i[s].define] = !1; } t.PREPASS != n && (t.markAsUnprocessed(), t.markAsImageProcessingDirty()); } /** * Prepares the defines related to the light information passed in parameter * @param scene The scene we are intending to draw * @param mesh The mesh the effect is compiling for * @param light The light the effect is compiling for * @param lightIndex The index of the light * @param defines The defines to update * @param specularSupported Specifies whether specular is supported or not (override lights data) * @param state Defines the current state regarding what is needed (normals, etc...) * @param state.needNormals * @param state.needRebuild * @param state.shadowEnabled * @param state.specularEnabled * @param state.lightmapMode */ static PrepareDefinesForLight(e, t, r, n, i, s, a) { var f; switch (a.needNormals = !0, i["LIGHT" + n] === void 0 && (a.needRebuild = !0), i["LIGHT" + n] = !0, i["SPOTLIGHT" + n] = !1, i["HEMILIGHT" + n] = !1, i["POINTLIGHT" + n] = !1, i["DIRLIGHT" + n] = !1, r.prepareLightSpecificDefines(i, n), i["LIGHT_FALLOFF_PHYSICAL" + n] = !1, i["LIGHT_FALLOFF_GLTF" + n] = !1, i["LIGHT_FALLOFF_STANDARD" + n] = !1, r.falloffType) { case ra.FALLOFF_GLTF: i["LIGHT_FALLOFF_GLTF" + n] = !0; break; case ra.FALLOFF_PHYSICAL: i["LIGHT_FALLOFF_PHYSICAL" + n] = !0; break; case ra.FALLOFF_STANDARD: i["LIGHT_FALLOFF_STANDARD" + n] = !0; break; } if (s && !r.specular.equalsFloats(0, 0, 0) && (a.specularEnabled = !0), i["SHADOW" + n] = !1, i["SHADOWCSM" + n] = !1, i["SHADOWCSMDEBUG" + n] = !1, i["SHADOWCSMNUM_CASCADES" + n] = !1, i["SHADOWCSMUSESHADOWMAXZ" + n] = !1, i["SHADOWCSMNOBLEND" + n] = !1, i["SHADOWCSM_RIGHTHANDED" + n] = !1, i["SHADOWPCF" + n] = !1, i["SHADOWPCSS" + n] = !1, i["SHADOWPOISSON" + n] = !1, i["SHADOWESM" + n] = !1, i["SHADOWCLOSEESM" + n] = !1, i["SHADOWCUBE" + n] = !1, i["SHADOWLOWQUALITY" + n] = !1, i["SHADOWMEDIUMQUALITY" + n] = !1, t && t.receiveShadows && e.shadowsEnabled && r.shadowEnabled) { const o = (f = r.getShadowGenerator(e.activeCamera)) !== null && f !== void 0 ? f : r.getShadowGenerator(); if (o) { const d = o.getShadowMap(); d && d.renderList && d.renderList.length > 0 && (a.shadowEnabled = !0, o.prepareDefines(i, n)); } } r.lightmapMode != ra.LIGHTMAP_DEFAULT ? (a.lightmapMode = !0, i["LIGHTMAPEXCLUDED" + n] = !0, i["LIGHTMAPNOSPECULAR" + n] = r.lightmapMode == ra.LIGHTMAP_SHADOWSONLY) : (i["LIGHTMAPEXCLUDED" + n] = !1, i["LIGHTMAPNOSPECULAR" + n] = !1); } /** * Prepares the defines related to the light information passed in parameter * @param scene The scene we are intending to draw * @param mesh The mesh the effect is compiling for * @param defines The defines to update * @param specularSupported Specifies whether specular is supported or not (override lights data) * @param maxSimultaneousLights Specifies how manuy lights can be added to the effect at max * @param disableLighting Specifies whether the lighting is disabled (override scene and light) * @returns true if normals will be required for the rest of the effect */ static PrepareDefinesForLights(e, t, r, n, i = 4, s = !1) { if (!r._areLightsDirty) return r._needNormals; let a = 0; const f = { needNormals: r._needNormals, needRebuild: !1, lightmapMode: !1, shadowEnabled: !1, specularEnabled: !1 }; if (e.lightsEnabled && !s) { for (const d of t.lightSources) if (this.PrepareDefinesForLight(e, t, d, a, r, n, f), a++, a === i) break; } r.SPECULARTERM = f.specularEnabled, r.SHADOWS = f.shadowEnabled; for (let d = a; d < i; d++) r["LIGHT" + d] !== void 0 && (r["LIGHT" + d] = !1, r["HEMILIGHT" + d] = !1, r["POINTLIGHT" + d] = !1, r["DIRLIGHT" + d] = !1, r["SPOTLIGHT" + d] = !1, r["SHADOW" + d] = !1, r["SHADOWCSM" + d] = !1, r["SHADOWCSMDEBUG" + d] = !1, r["SHADOWCSMNUM_CASCADES" + d] = !1, r["SHADOWCSMUSESHADOWMAXZ" + d] = !1, r["SHADOWCSMNOBLEND" + d] = !1, r["SHADOWCSM_RIGHTHANDED" + d] = !1, r["SHADOWPCF" + d] = !1, r["SHADOWPCSS" + d] = !1, r["SHADOWPOISSON" + d] = !1, r["SHADOWESM" + d] = !1, r["SHADOWCLOSEESM" + d] = !1, r["SHADOWCUBE" + d] = !1, r["SHADOWLOWQUALITY" + d] = !1, r["SHADOWMEDIUMQUALITY" + d] = !1); const o = e.getEngine().getCaps(); return r.SHADOWFLOAT === void 0 && (f.needRebuild = !0), r.SHADOWFLOAT = f.shadowEnabled && (o.textureFloatRender && o.textureFloatLinearFiltering || o.textureHalfFloatRender && o.textureHalfFloatLinearFiltering), r.LIGHTMAPEXCLUDED = f.lightmapMode, f.needRebuild && r.rebuild(), f.needNormals; } /** * Prepares the uniforms and samplers list to be used in the effect (for a specific light) * @param lightIndex defines the light index * @param uniformsList The uniform list * @param samplersList The sampler list * @param projectedLightTexture defines if projected texture must be used * @param uniformBuffersList defines an optional list of uniform buffers * @param updateOnlyBuffersList True to only update the uniformBuffersList array */ static PrepareUniformsAndSamplersForLight(e, t, r, n, i = null, s = !1) { i && i.push("Light" + e), !s && (t.push("vLightData" + e, "vLightDiffuse" + e, "vLightSpecular" + e, "vLightDirection" + e, "vLightFalloff" + e, "vLightGround" + e, "lightMatrix" + e, "shadowsInfo" + e, "depthValues" + e), r.push("shadowSampler" + e), r.push("depthSampler" + e), t.push("viewFrustumZ" + e, "cascadeBlendFactor" + e, "lightSizeUVCorrection" + e, "depthCorrection" + e, "penumbraDarkness" + e, "frustumLengths" + e), n && (r.push("projectionLightSampler" + e), t.push("textureProjectionMatrix" + e))); } /** * Prepares the uniforms and samplers list to be used in the effect * @param uniformsListOrOptions The uniform names to prepare or an EffectCreationOptions containing the list and extra information * @param samplersList The sampler list * @param defines The defines helping in the list generation * @param maxSimultaneousLights The maximum number of simultaneous light allowed in the effect */ static PrepareUniformsAndSamplersList(e, t, r, n = 4) { let i, s = null; if (e.uniformsNames) { const a = e; i = a.uniformsNames, s = a.uniformBuffersNames, t = a.samplers, r = a.defines, n = a.maxSimultaneousLights || 0; } else i = e, t || (t = []); for (let a = 0; a < n && r["LIGHT" + a]; a++) this.PrepareUniformsAndSamplersForLight(a, i, t, r["PROJECTEDLIGHTTEXTURE" + a], s); r.NUM_MORPH_INFLUENCERS && i.push("morphTargetInfluences"), r.BAKED_VERTEX_ANIMATION_TEXTURE && (i.push("bakedVertexAnimationSettings"), i.push("bakedVertexAnimationTextureSizeInverted"), i.push("bakedVertexAnimationTime"), t.push("bakedVertexAnimationTexture")); } /** * This helps decreasing rank by rank the shadow quality (0 being the highest rank and quality) * @param defines The defines to update while falling back * @param fallbacks The authorized effect fallbacks * @param maxSimultaneousLights The maximum number of lights allowed * @param rank the current rank of the Effect * @returns The newly affected rank */ static HandleFallbacksForShadows(e, t, r = 4, n = 0) { let i = 0; for (let s = 0; s < r && e["LIGHT" + s]; s++) s > 0 && (i = n + s, t.addFallback(i, "LIGHT" + s)), e.SHADOWS || (e["SHADOW" + s] && t.addFallback(n, "SHADOW" + s), e["SHADOWPCF" + s] && t.addFallback(n, "SHADOWPCF" + s), e["SHADOWPCSS" + s] && t.addFallback(n, "SHADOWPCSS" + s), e["SHADOWPOISSON" + s] && t.addFallback(n, "SHADOWPOISSON" + s), e["SHADOWESM" + s] && t.addFallback(n, "SHADOWESM" + s), e["SHADOWCLOSEESM" + s] && t.addFallback(n, "SHADOWCLOSEESM" + s)); return i++; } /** * Prepares the list of attributes required for morph targets according to the effect defines. * @param attribs The current list of supported attribs * @param mesh The mesh to prepare the morph targets attributes for * @param influencers The number of influencers */ static PrepareAttributesForMorphTargetsInfluencers(e, t, r) { this._TmpMorphInfluencers.NUM_MORPH_INFLUENCERS = r, this.PrepareAttributesForMorphTargets(e, t, this._TmpMorphInfluencers); } /** * Prepares the list of attributes required for morph targets according to the effect defines. * @param attribs The current list of supported attribs * @param mesh The mesh to prepare the morph targets attributes for * @param defines The current Defines of the effect */ static PrepareAttributesForMorphTargets(e, t, r) { const n = r.NUM_MORPH_INFLUENCERS; if (n > 0 && gr.LastCreatedEngine) { const i = gr.LastCreatedEngine.getCaps().maxVertexAttribs, s = t.morphTargetManager; if (s != null && s.isUsingTextureForTargets) return; const a = s && s.supportsNormals && r.NORMAL, f = s && s.supportsTangents && r.TANGENT, o = s && s.supportsUVs && r.UV1; for (let d = 0; d < n; d++) e.push(J.PositionKind + d), a && e.push(J.NormalKind + d), f && e.push(J.TangentKind + d), o && e.push(J.UVKind + "_" + d), e.length > i && Se.Error("Cannot add more vertex attributes for mesh " + t.name); } } /** * Prepares the list of attributes required for baked vertex animations according to the effect defines. * @param attribs The current list of supported attribs * @param mesh The mesh to prepare the morph targets attributes for * @param defines The current Defines of the effect */ static PrepareAttributesForBakedVertexAnimation(e, t, r) { r.BAKED_VERTEX_ANIMATION_TEXTURE && r.INSTANCES && e.push("bakedVertexAnimationSettingsInstanced"); } /** * Prepares the list of attributes required for bones according to the effect defines. * @param attribs The current list of supported attribs * @param mesh The mesh to prepare the bones attributes for * @param defines The current Defines of the effect * @param fallbacks The current effect fallback strategy */ static PrepareAttributesForBones(e, t, r, n) { r.NUM_BONE_INFLUENCERS > 0 && (n.addCPUSkinningFallback(0, t), e.push(J.MatricesIndicesKind), e.push(J.MatricesWeightsKind), r.NUM_BONE_INFLUENCERS > 4 && (e.push(J.MatricesIndicesExtraKind), e.push(J.MatricesWeightsExtraKind))); } /** * Check and prepare the list of attributes required for instances according to the effect defines. * @param attribs The current list of supported attribs * @param defines The current MaterialDefines of the effect */ static PrepareAttributesForInstances(e, t) { (t.INSTANCES || t.THIN_INSTANCES) && this.PushAttributesForInstances(e, !!t.PREPASS_VELOCITY), t.INSTANCESCOLOR && e.push(J.ColorInstanceKind); } /** * Add the list of attributes required for instances to the attribs array. * @param attribs The current list of supported attribs * @param needsPreviousMatrices If the shader needs previous matrices */ static PushAttributesForInstances(e, t = !1) { e.push("world0"), e.push("world1"), e.push("world2"), e.push("world3"), t && (e.push("previousWorld0"), e.push("previousWorld1"), e.push("previousWorld2"), e.push("previousWorld3")); } /** * Binds the light information to the effect. * @param light The light containing the generator * @param effect The effect we are binding the data to * @param lightIndex The light index in the effect used to render */ static BindLightProperties(e, t, r) { e.transferToEffect(t, r + ""); } /** * Binds the lights information from the scene to the effect for the given mesh. * @param light Light to bind * @param lightIndex Light index * @param scene The scene where the light belongs to * @param effect The effect we are binding the data to * @param useSpecular Defines if specular is supported * @param receiveShadows Defines if the effect (mesh) we bind the light for receives shadows */ static BindLight(e, t, r, n, i, s = !0) { e._bindLight(t, r, n, i, s); } /** * Binds the lights information from the scene to the effect for the given mesh. * @param scene The scene the lights belongs to * @param mesh The mesh we are binding the information to render * @param effect The effect we are binding the data to * @param defines The generated defines for the effect * @param maxSimultaneousLights The maximum number of light that can be bound to the effect */ static BindLights(e, t, r, n, i = 4) { const s = Math.min(t.lightSources.length, i); for (let a = 0; a < s; a++) { const f = t.lightSources[a]; this.BindLight(f, a, e, r, typeof n == "boolean" ? n : n.SPECULARTERM, t.receiveShadows); } } /** * Binds the fog information from the scene to the effect for the given mesh. * @param scene The scene the lights belongs to * @param mesh The mesh we are binding the information to render * @param effect The effect we are binding the data to * @param linearSpace Defines if the fog effect is applied in linear space */ static BindFogParameters(e, t, r, n = !1) { e.fogEnabled && t.applyFog && e.fogMode !== sr.FOGMODE_NONE && (r.setFloat4("vFogInfos", e.fogMode, e.fogStart, e.fogEnd, e.fogDensity), n ? (e.fogColor.toLinearSpaceToRef(this._TempFogColor, e.getEngine().useExactSrgbConversions), r.setColor3("vFogColor", this._TempFogColor)) : r.setColor3("vFogColor", e.fogColor)); } /** * Binds the bones information from the mesh to the effect. * @param mesh The mesh we are binding the information to render * @param effect The effect we are binding the data to * @param prePassConfiguration Configuration for the prepass, in case prepass is activated */ static BindBonesParameters(e, t, r) { if (!(!t || !e) && (e.computeBonesUsingShaders && t._bonesComputationForcedToCPU && (e.computeBonesUsingShaders = !1), e.useBones && e.computeBonesUsingShaders && e.skeleton)) { const n = e.skeleton; if (n.isUsingTextureForMatrices && t.getUniformIndex("boneTextureWidth") > -1) { const i = n.getTransformMatrixTexture(e); t.setTexture("boneSampler", i), t.setFloat("boneTextureWidth", 4 * (n.bones.length + 1)); } else { const i = n.getTransformMatrices(e); i && (t.setMatrices("mBones", i), r && e.getScene().prePassRenderer && e.getScene().prePassRenderer.getIndex(2) && (r.previousBones[e.uniqueId] || (r.previousBones[e.uniqueId] = i.slice()), t.setMatrices("mPreviousBones", r.previousBones[e.uniqueId]), Ye._CopyBonesTransformationMatrices(i, r.previousBones[e.uniqueId]))); } } } // Copies the bones transformation matrices into the target array and returns the target's reference static _CopyBonesTransformationMatrices(e, t) { return t.set(e), t; } /** * Binds the morph targets information from the mesh to the effect. * @param abstractMesh The mesh we are binding the information to render * @param effect The effect we are binding the data to */ static BindMorphTargetParameters(e, t) { const r = e.morphTargetManager; !e || !r || t.setFloatArray("morphTargetInfluences", r.influences); } /** * Binds the logarithmic depth information from the scene to the effect for the given defines. * @param defines The generated defines used in the effect * @param effect The effect we are binding the data to * @param scene The scene we are willing to render with logarithmic scale for */ static BindLogDepth(e, t, r) { if (!e || e.LOGARITHMICDEPTH || e.indexOf && e.indexOf("LOGARITHMICDEPTH") >= 0) { const n = r.activeCamera; n.mode === Tr.ORTHOGRAPHIC_CAMERA && Se.Error("Logarithmic depth is not compatible with orthographic cameras!", 20), t.setFloat("logarithmicDepthConstant", 2 / (Math.log(n.maxZ + 1) / Math.LN2)); } } } Ye._TmpMorphInfluencers = { NUM_MORPH_INFLUENCERS: 0 }; Ye._TempFogColor = Ne.Black(); class Dq { /** * Creates a material stencil state instance */ constructor() { this.reset(); } /** * Resets all the stencil states to default values */ reset() { this.enabled = !1, this.mask = 255, this.func = 519, this.funcRef = 1, this.funcMask = 255, this.opStencilFail = 7680, this.opDepthFail = 7680, this.opStencilDepthPass = 7681; } /** * Gets or sets the stencil function */ get func() { return this._func; } set func(e) { this._func = e; } /** * Gets or sets the stencil function reference */ get funcRef() { return this._funcRef; } set funcRef(e) { this._funcRef = e; } /** * Gets or sets the stencil function mask */ get funcMask() { return this._funcMask; } set funcMask(e) { this._funcMask = e; } /** * Gets or sets the operation when the stencil test fails */ get opStencilFail() { return this._opStencilFail; } set opStencilFail(e) { this._opStencilFail = e; } /** * Gets or sets the operation when the depth test fails */ get opDepthFail() { return this._opDepthFail; } set opDepthFail(e) { this._opDepthFail = e; } /** * Gets or sets the operation when the stencil+depth test succeeds */ get opStencilDepthPass() { return this._opStencilDepthPass; } set opStencilDepthPass(e) { this._opStencilDepthPass = e; } /** * Gets or sets the stencil mask */ get mask() { return this._mask; } set mask(e) { this._mask = e; } /** * Enables or disables the stencil test */ get enabled() { return this._enabled; } set enabled(e) { this._enabled = e; } /** * Get the current class name, useful for serialization or dynamic coding. * @returns "MaterialStencilState" */ getClassName() { return "MaterialStencilState"; } /** * Makes a duplicate of the current configuration into another one. * @param stencilState defines stencil state where to copy the info */ copyTo(e) { jt.Clone(() => e, this); } /** * Serializes this stencil configuration. * @returns - An object with the serialized config. */ serialize() { return jt.Serialize(this); } /** * Parses a stencil state configuration from a serialized object. * @param source - Serialized object. * @param scene Defines the scene we are parsing for * @param rootUrl Defines the rootUrl to load from */ parse(e, t, r) { jt.Parse(() => this, e, t, r); } } C([ M() ], Dq.prototype, "func", null); C([ M() ], Dq.prototype, "funcRef", null); C([ M() ], Dq.prototype, "funcMask", null); C([ M() ], Dq.prototype, "opStencilFail", null); C([ M() ], Dq.prototype, "opDepthFail", null); C([ M() ], Dq.prototype, "opStencilDepthPass", null); C([ M() ], Dq.prototype, "mask", null); C([ M() ], Dq.prototype, "enabled", null); var x6; (function(A) { A[A.Created = 1] = "Created", A[A.Disposed = 2] = "Disposed", A[A.GetDefineNames = 4] = "GetDefineNames", A[A.PrepareUniformBuffer = 8] = "PrepareUniformBuffer", A[A.IsReadyForSubMesh = 16] = "IsReadyForSubMesh", A[A.PrepareDefines = 32] = "PrepareDefines", A[A.BindForSubMesh = 64] = "BindForSubMesh", A[A.PrepareEffect = 128] = "PrepareEffect", A[A.GetAnimatables = 256] = "GetAnimatables", A[A.GetActiveTextures = 512] = "GetActiveTextures", A[A.HasTexture = 1024] = "HasTexture", A[A.FillRenderTargetTextures = 2048] = "FillRenderTargetTextures", A[A.HasRenderTargetTextures = 4096] = "HasRenderTargetTextures", A[A.HardBindForSubMesh = 8192] = "HardBindForSubMesh"; })(x6 || (x6 = {})); class gt { /** * If the material can be rendered to several textures with MRT extension */ get canRenderToMRT() { return !1; } /** * Sets the alpha value of the material */ set alpha(e) { if (this._alpha === e) return; const t = this._alpha; this._alpha = e, (t === 1 || e === 1) && this.markAsDirty(gt.MiscDirtyFlag + gt.PrePassDirtyFlag); } /** * Gets the alpha value of the material */ get alpha() { return this._alpha; } /** * Sets the culling state (true to enable culling, false to disable) */ set backFaceCulling(e) { this._backFaceCulling !== e && (this._backFaceCulling = e, this.markAsDirty(gt.TextureDirtyFlag)); } /** * Gets the culling state */ get backFaceCulling() { return this._backFaceCulling; } /** * Sets the type of faces that should be culled (true for back faces, false for front faces) */ set cullBackFaces(e) { this._cullBackFaces !== e && (this._cullBackFaces = e, this.markAsDirty(gt.TextureDirtyFlag)); } /** * Gets the type of faces that should be culled */ get cullBackFaces() { return this._cullBackFaces; } /** * Block the dirty-mechanism for this specific material * When set to false after being true the material will be marked as dirty. */ get blockDirtyMechanism() { return this._blockDirtyMechanism; } set blockDirtyMechanism(e) { this._blockDirtyMechanism !== e && (this._blockDirtyMechanism = e, e || this.markDirty()); } /** * This allows you to modify the material without marking it as dirty after every change. * This function should be used if you need to make more than one dirty-enabling change to the material - adding a texture, setting a new fill mode and so on. * The callback will pass the material as an argument, so you can make your changes to it. * @param callback the callback to be executed that will update the material */ atomicMaterialsUpdate(e) { this.blockDirtyMechanism = !0; try { e(this); } finally { this.blockDirtyMechanism = !1; } } /** * Gets a boolean indicating that current material needs to register RTT */ get hasRenderTargetTextures() { return this._eventInfo.hasRenderTargetTextures = !1, this._callbackPluginEventHasRenderTargetTextures(this._eventInfo), this._eventInfo.hasRenderTargetTextures; } /** * Called during a dispose event */ set onDispose(e) { this._onDisposeObserver && this.onDisposeObservable.remove(this._onDisposeObserver), this._onDisposeObserver = this.onDisposeObservable.add(e); } /** * An event triggered when the material is bound */ get onBindObservable() { return this._onBindObservable || (this._onBindObservable = new Oe()), this._onBindObservable; } /** * Called during a bind event */ set onBind(e) { this._onBindObserver && this.onBindObservable.remove(this._onBindObserver), this._onBindObserver = this.onBindObservable.add(e); } /** * An event triggered when the material is unbound */ get onUnBindObservable() { return this._onUnBindObservable || (this._onUnBindObservable = new Oe()), this._onUnBindObservable; } /** * An event triggered when the effect is (re)created */ get onEffectCreatedObservable() { return this._onEffectCreatedObservable || (this._onEffectCreatedObservable = new Oe()), this._onEffectCreatedObservable; } /** * Sets the value of the alpha mode. * * | Value | Type | Description | * | --- | --- | --- | * | 0 | ALPHA_DISABLE | | * | 1 | ALPHA_ADD | | * | 2 | ALPHA_COMBINE | | * | 3 | ALPHA_SUBTRACT | | * | 4 | ALPHA_MULTIPLY | | * | 5 | ALPHA_MAXIMIZED | | * | 6 | ALPHA_ONEONE | | * | 7 | ALPHA_PREMULTIPLIED | | * | 8 | ALPHA_PREMULTIPLIED_PORTERDUFF | | * | 9 | ALPHA_INTERPOLATE | | * | 10 | ALPHA_SCREENMODE | | * */ set alphaMode(e) { this._alphaMode !== e && (this._alphaMode = e, this.markAsDirty(gt.TextureDirtyFlag)); } /** * Gets the value of the alpha mode */ get alphaMode() { return this._alphaMode; } /** * Sets the need depth pre-pass value */ set needDepthPrePass(e) { this._needDepthPrePass !== e && (this._needDepthPrePass = e, this._needDepthPrePass && (this.checkReadyOnEveryCall = !0)); } /** * Gets the depth pre-pass value */ get needDepthPrePass() { return this._needDepthPrePass; } /** * Can this material render to prepass */ get isPrePassCapable() { return !1; } /** * Sets the state for enabling fog */ set fogEnabled(e) { this._fogEnabled !== e && (this._fogEnabled = e, this.markAsDirty(gt.MiscDirtyFlag)); } /** * Gets the value of the fog enabled state */ get fogEnabled() { return this._fogEnabled; } get wireframe() { switch (this._fillMode) { case gt.WireFrameFillMode: case gt.LineListDrawMode: case gt.LineLoopDrawMode: case gt.LineStripDrawMode: return !0; } return this._scene.forceWireframe; } /** * Sets the state of wireframe mode */ set wireframe(e) { this.fillMode = e ? gt.WireFrameFillMode : gt.TriangleFillMode; } /** * Gets the value specifying if point clouds are enabled */ get pointsCloud() { switch (this._fillMode) { case gt.PointFillMode: case gt.PointListDrawMode: return !0; } return this._scene.forcePointsCloud; } /** * Sets the state of point cloud mode */ set pointsCloud(e) { this.fillMode = e ? gt.PointFillMode : gt.TriangleFillMode; } /** * Gets the material fill mode */ get fillMode() { return this._fillMode; } /** * Sets the material fill mode */ set fillMode(e) { this._fillMode !== e && (this._fillMode = e, this.markAsDirty(gt.MiscDirtyFlag)); } /** * In case the depth buffer does not allow enough depth precision for your scene (might be the case in large scenes) * You can try switching to logarithmic depth. * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/advanced/logarithmicDepthBuffer */ get useLogarithmicDepth() { return this._useLogarithmicDepth; } set useLogarithmicDepth(e) { const t = this.getScene().getEngine().getCaps().fragmentDepthSupported; e && !t && Se.Warn("Logarithmic depth has been requested for a material on a device that doesn't support it."), this._useLogarithmicDepth = e && t, this._markAllSubMeshesAsMiscDirty(); } /** @internal */ _getDrawWrapper() { return this._drawWrapper; } /** * @internal */ _setDrawWrapper(e) { this._drawWrapper = e; } /** * Creates a material instance * @param name defines the name of the material * @param scene defines the scene to reference * @param doNotAdd specifies if the material should be added to the scene */ constructor(e, t, r) { this.shadowDepthWrapper = null, this.allowShaderHotSwapping = !0, this.metadata = null, this.reservedDataStore = null, this.checkReadyOnEveryCall = !1, this.checkReadyOnlyOnce = !1, this.state = "", this._alpha = 1, this._backFaceCulling = !0, this._cullBackFaces = !0, this._blockDirtyMechanism = !1, this.onCompiled = null, this.onError = null, this.getRenderTargetTextures = null, this.doNotSerialize = !1, this._storeEffectOnSubMeshes = !1, this.animations = null, this.onDisposeObservable = new Oe(), this._onDisposeObserver = null, this._onUnBindObservable = null, this._onBindObserver = null, this._alphaMode = 2, this._needDepthPrePass = !1, this.disableDepthWrite = !1, this.disableColorWrite = !1, this.forceDepthWrite = !1, this.depthFunction = 0, this.separateCullingPass = !1, this._fogEnabled = !0, this.pointSize = 1, this.zOffset = 0, this.zOffsetUnits = 0, this.stencil = new Dq(), this._useUBO = !1, this._fillMode = gt.TriangleFillMode, this._cachedDepthWriteState = !1, this._cachedColorWriteState = !1, this._cachedDepthFunctionState = 0, this._indexInSceneMaterialArray = -1, this.meshMap = null, this._parentContainer = null, this._uniformBufferLayoutBuilt = !1, this._eventInfo = {}, this._callbackPluginEventGeneric = () => { }, this._callbackPluginEventIsReadyForSubMesh = () => { }, this._callbackPluginEventPrepareDefines = () => { }, this._callbackPluginEventPrepareDefinesBeforeAttributes = () => { }, this._callbackPluginEventHardBindForSubMesh = () => { }, this._callbackPluginEventBindForSubMesh = () => { }, this._callbackPluginEventHasRenderTargetTextures = () => { }, this._callbackPluginEventFillRenderTargetTextures = () => { }, this._forceAlphaTest = !1, this._transparencyMode = null, this.name = e; const n = t || gr.LastCreatedScene; n && (this._scene = n, this._dirtyCallbacks = {}, this._dirtyCallbacks[1] = this._markAllSubMeshesAsTexturesDirty.bind(this), this._dirtyCallbacks[2] = this._markAllSubMeshesAsLightsDirty.bind(this), this._dirtyCallbacks[4] = this._markAllSubMeshesAsFresnelDirty.bind(this), this._dirtyCallbacks[8] = this._markAllSubMeshesAsAttributesDirty.bind(this), this._dirtyCallbacks[16] = this._markAllSubMeshesAsMiscDirty.bind(this), this._dirtyCallbacks[32] = this._markAllSubMeshesAsPrePassDirty.bind(this), this._dirtyCallbacks[63] = this._markAllSubMeshesAsAllDirty.bind(this), this.id = e || ye.RandomId(), this.uniqueId = this._scene.getUniqueId(), this._materialContext = this._scene.getEngine().createMaterialContext(), this._drawWrapper = new zo(this._scene.getEngine(), !1), this._drawWrapper.materialContext = this._materialContext, this._scene.useRightHandedSystem ? this.sideOrientation = gt.ClockWiseSideOrientation : this.sideOrientation = gt.CounterClockWiseSideOrientation, this._uniformBuffer = new yr(this._scene.getEngine(), void 0, void 0, e), this._useUBO = this.getScene().getEngine().supportsUniformBuffers, r || this._scene.addMaterial(this), this._scene.useMaterialMeshMap && (this.meshMap = {}), gt.OnEventObservable.notifyObservers(this, x6.Created)); } /** * Returns a string representation of the current material * @param fullDetails defines a boolean indicating which levels of logging is desired * @returns a string with material information */ // eslint-disable-next-line @typescript-eslint/no-unused-vars toString(e) { return "Name: " + this.name; } /** * Gets the class name of the material * @returns a string with the class name of the material */ getClassName() { return "Material"; } /** @internal */ get _isMaterial() { return !0; } /** * Specifies if updates for the material been locked */ get isFrozen() { return this.checkReadyOnlyOnce; } /** * Locks updates for the material */ freeze() { this.markDirty(), this.checkReadyOnlyOnce = !0; } /** * Unlocks updates for the material */ unfreeze() { this.markDirty(), this.checkReadyOnlyOnce = !1; } /** * Specifies if the material is ready to be used * @param mesh defines the mesh to check * @param useInstances specifies if instances should be used * @returns a boolean indicating if the material is ready to be used */ // eslint-disable-next-line @typescript-eslint/no-unused-vars isReady(e, t) { return !0; } /** * Specifies that the submesh is ready to be used * @param mesh defines the mesh to check * @param subMesh defines which submesh to check * @param useInstances specifies that instances should be used * @returns a boolean indicating that the submesh is ready or not */ // eslint-disable-next-line @typescript-eslint/no-unused-vars isReadyForSubMesh(e, t, r) { const n = t.materialDefines; return n ? (this._eventInfo.isReadyForSubMesh = !0, this._eventInfo.defines = n, this._callbackPluginEventIsReadyForSubMesh(this._eventInfo), this._eventInfo.isReadyForSubMesh) : !1; } /** * Returns the material effect * @returns the effect associated with the material */ getEffect() { return this._drawWrapper.effect; } /** * Returns the current scene * @returns a Scene */ getScene() { return this._scene; } /** * Gets the current transparency mode. */ get transparencyMode() { return this._transparencyMode; } /** * Sets the transparency mode of the material. * * | Value | Type | Description | * | ----- | ----------------------------------- | ----------- | * | 0 | OPAQUE | | * | 1 | ALPHATEST | | * | 2 | ALPHABLEND | | * | 3 | ALPHATESTANDBLEND | | * */ set transparencyMode(e) { this._transparencyMode !== e && (this._transparencyMode = e, this._forceAlphaTest = e === gt.MATERIAL_ALPHATESTANDBLEND, this._markAllSubMeshesAsTexturesAndMiscDirty()); } /** * Returns true if alpha blending should be disabled. */ get _disableAlphaBlending() { return this._transparencyMode === gt.MATERIAL_OPAQUE || this._transparencyMode === gt.MATERIAL_ALPHATEST; } /** * Specifies whether or not this material should be rendered in alpha blend mode. * @returns a boolean specifying if alpha blending is needed */ needAlphaBlending() { return this._disableAlphaBlending ? !1 : this.alpha < 1; } /** * Specifies if the mesh will require alpha blending * @param mesh defines the mesh to check * @returns a boolean specifying if alpha blending is needed for the mesh */ needAlphaBlendingForMesh(e) { return e.visibility < 1 ? !0 : this._disableAlphaBlending ? !1 : e.hasVertexAlpha || this.needAlphaBlending(); } /** * Specifies whether or not this material should be rendered in alpha test mode. * @returns a boolean specifying if an alpha test is needed. */ needAlphaTesting() { return !!this._forceAlphaTest; } /** * Specifies if material alpha testing should be turned on for the mesh * @param mesh defines the mesh to check */ _shouldTurnAlphaTestOn(e) { return !this.needAlphaBlendingForMesh(e) && this.needAlphaTesting(); } /** * Gets the texture used for the alpha test * @returns the texture to use for alpha testing */ getAlphaTestTexture() { return null; } /** * Marks the material to indicate that it needs to be re-calculated * @param forceMaterialDirty - Forces the material to be marked as dirty for all components (same as this.markAsDirty(Material.AllDirtyFlag)). You should use this flag if the material is frozen and you want to force a recompilation. */ markDirty(e = !1) { const t = this.getScene().meshes; for (const r of t) if (r.subMeshes) for (const n of r.subMeshes) n.getMaterial() === this && n.effect && (n.effect._wasPreviouslyReady = !1, n.effect._wasPreviouslyUsingInstances = null, n.effect._forceRebindOnNextCall = e); e && this.markAsDirty(gt.AllDirtyFlag); } /** * @internal */ _preBind(e, t = null) { const r = this._scene.getEngine(), i = (t ?? this.sideOrientation) === gt.ClockWiseSideOrientation; return r.enableEffect(e || this._getDrawWrapper()), r.setState(this.backFaceCulling, this.zOffset, !1, i, this._scene._mirroredCameraPosition ? !this.cullBackFaces : this.cullBackFaces, this.stencil, this.zOffsetUnits), i; } /** * Binds the material to the mesh * @param world defines the world transformation matrix * @param mesh defines the mesh to bind the material to */ // eslint-disable-next-line @typescript-eslint/no-unused-vars bind(e, t) { } /** * Initializes the uniform buffer layout for the shader. */ buildUniformLayout() { const e = this._uniformBuffer; this._eventInfo.ubo = e, this._callbackPluginEventGeneric(x6.PrepareUniformBuffer, this._eventInfo), e.create(), this._uniformBufferLayoutBuilt = !0; } /** * Binds the submesh to the material * @param world defines the world transformation matrix * @param mesh defines the mesh containing the submesh * @param subMesh defines the submesh to bind the material to */ bindForSubMesh(e, t, r) { const n = r.effect; n && (this._eventInfo.subMesh = r, this._callbackPluginEventBindForSubMesh(this._eventInfo), n._forceRebindOnNextCall = !1); } /** * Binds the world matrix to the material * @param world defines the world transformation matrix */ // eslint-disable-next-line @typescript-eslint/no-unused-vars bindOnlyWorldMatrix(e) { } /** * Binds the view matrix to the effect * @param effect defines the effect to bind the view matrix to */ bindView(e) { this._useUBO ? this._needToBindSceneUbo = !0 : e.setMatrix("view", this.getScene().getViewMatrix()); } /** * Binds the view projection and projection matrices to the effect * @param effect defines the effect to bind the view projection and projection matrices to */ bindViewProjection(e) { this._useUBO ? this._needToBindSceneUbo = !0 : (e.setMatrix("viewProjection", this.getScene().getTransformMatrix()), e.setMatrix("projection", this.getScene().getProjectionMatrix())); } /** * Binds the view matrix to the effect * @param effect defines the effect to bind the view matrix to * @param variableName name of the shader variable that will hold the eye position */ bindEyePosition(e, t) { this._useUBO ? this._needToBindSceneUbo = !0 : this._scene.bindEyePosition(e, t); } /** * Processes to execute after binding the material to a mesh * @param mesh defines the rendered mesh * @param effect */ _afterBind(e, t = null) { if (this._scene._cachedMaterial = this, this._needToBindSceneUbo && t && (this._needToBindSceneUbo = !1, Ye.BindSceneUniformBuffer(t, this.getScene().getSceneUniformBuffer()), this._scene.finalizeSceneUbo()), e ? this._scene._cachedVisibility = e.visibility : this._scene._cachedVisibility = 1, this._onBindObservable && e && this._onBindObservable.notifyObservers(e), this.disableDepthWrite) { const r = this._scene.getEngine(); this._cachedDepthWriteState = r.getDepthWrite(), r.setDepthWrite(!1); } if (this.disableColorWrite) { const r = this._scene.getEngine(); this._cachedColorWriteState = r.getColorWrite(), r.setColorWrite(!1); } if (this.depthFunction !== 0) { const r = this._scene.getEngine(); this._cachedDepthFunctionState = r.getDepthFunction() || 0, r.setDepthFunction(this.depthFunction); } } /** * Unbinds the material from the mesh */ unbind() { this._onUnBindObservable && this._onUnBindObservable.notifyObservers(this), this.depthFunction !== 0 && this._scene.getEngine().setDepthFunction(this._cachedDepthFunctionState), this.disableDepthWrite && this._scene.getEngine().setDepthWrite(this._cachedDepthWriteState), this.disableColorWrite && this._scene.getEngine().setColorWrite(this._cachedColorWriteState); } /** * Returns the animatable textures. * @returns - Array of animatable textures. */ getAnimatables() { return this._eventInfo.animatables = [], this._callbackPluginEventGeneric(x6.GetAnimatables, this._eventInfo), this._eventInfo.animatables; } /** * Gets the active textures from the material * @returns an array of textures */ getActiveTextures() { return this._eventInfo.activeTextures = [], this._callbackPluginEventGeneric(x6.GetActiveTextures, this._eventInfo), this._eventInfo.activeTextures; } /** * Specifies if the material uses a texture * @param texture defines the texture to check against the material * @returns a boolean specifying if the material uses the texture */ hasTexture(e) { return this._eventInfo.hasTexture = !1, this._eventInfo.texture = e, this._callbackPluginEventGeneric(x6.HasTexture, this._eventInfo), this._eventInfo.hasTexture; } /** * Makes a duplicate of the material, and gives it a new name * @param name defines the new name for the duplicated material * @returns the cloned material */ // eslint-disable-next-line @typescript-eslint/no-unused-vars clone(e) { return null; } _clonePlugins(e, t) { const r = {}; if (this._serializePlugins(r), gt._parsePlugins(r, e, this._scene, t), this.pluginManager) for (const n of this.pluginManager._plugins) { const i = e.pluginManager.getPlugin(n.name); n.copyTo(i); } } /** * Gets the meshes bound to the material * @returns an array of meshes bound to the material */ getBindedMeshes() { if (this.meshMap) { const e = []; for (const t in this.meshMap) { const r = this.meshMap[t]; r && e.push(r); } return e; } else return this._scene.meshes.filter((t) => t.material === this); } /** * Force shader compilation * @param mesh defines the mesh associated with this material * @param onCompiled defines a function to execute once the material is compiled * @param options defines the options to configure the compilation * @param onError defines a function to execute if the material fails compiling */ forceCompilation(e, t, r, n) { const i = Object.assign({ clipPlane: !1, useInstances: !1 }, r), s = this.getScene(), a = this.allowShaderHotSwapping; this.allowShaderHotSwapping = !1; const f = () => { if (!this._scene || !this._scene.getEngine()) return; const o = s.clipPlane; if (i.clipPlane && (s.clipPlane = new BA(0, 0, 0, 1)), this._storeEffectOnSubMeshes) { let d = !0, v = null; if (e.subMeshes) { const u = new rA(0, 0, 0, 0, 0, e, void 0, !1, !1); u.materialDefines && (u.materialDefines._renderId = -1), this.isReadyForSubMesh(e, u, i.useInstances) || (u.effect && u.effect.getCompilationError() && u.effect.allFallbacksProcessed() ? v = u.effect.getCompilationError() : (d = !1, setTimeout(f, 16))); } d && (this.allowShaderHotSwapping = a, v && n && n(v), t && t(this)); } else this.isReady() ? (this.allowShaderHotSwapping = a, t && t(this)) : setTimeout(f, 16); i.clipPlane && (s.clipPlane = o); }; f(); } /** * Force shader compilation * @param mesh defines the mesh that will use this material * @param options defines additional options for compiling the shaders * @returns a promise that resolves when the compilation completes */ forceCompilationAsync(e, t) { return new Promise((r, n) => { this.forceCompilation(e, () => { r(); }, t, (i) => { n(i); }); }); } /** * Marks a define in the material to indicate that it needs to be re-computed * @param flag defines a flag used to determine which parts of the material have to be marked as dirty */ markAsDirty(e) { this.getScene().blockMaterialDirtyMechanism || this._blockDirtyMechanism || (gt._DirtyCallbackArray.length = 0, e & gt.TextureDirtyFlag && gt._DirtyCallbackArray.push(gt._TextureDirtyCallBack), e & gt.LightDirtyFlag && gt._DirtyCallbackArray.push(gt._LightsDirtyCallBack), e & gt.FresnelDirtyFlag && gt._DirtyCallbackArray.push(gt._FresnelDirtyCallBack), e & gt.AttributesDirtyFlag && gt._DirtyCallbackArray.push(gt._AttributeDirtyCallBack), e & gt.MiscDirtyFlag && gt._DirtyCallbackArray.push(gt._MiscDirtyCallBack), e & gt.PrePassDirtyFlag && gt._DirtyCallbackArray.push(gt._PrePassDirtyCallBack), gt._DirtyCallbackArray.length && this._markAllSubMeshesAsDirty(gt._RunDirtyCallBacks), this.getScene().resetCachedMaterial()); } /** * Resets the draw wrappers cache for all submeshes that are using this material */ resetDrawCache() { const e = this.getScene().meshes; for (const t of e) if (t.subMeshes) for (const r of t.subMeshes) r.getMaterial() === this && r.resetDrawCache(); } /** * Marks all submeshes of a material to indicate that their material defines need to be re-calculated * @param func defines a function which checks material defines against the submeshes */ _markAllSubMeshesAsDirty(e) { if (this.getScene().blockMaterialDirtyMechanism || this._blockDirtyMechanism) return; const t = this.getScene().meshes; for (const r of t) if (r.subMeshes) { for (const n of r.subMeshes) if (n.getMaterial(!1) === this) for (const i of n._drawWrappers) !i || !i.defines || !i.defines.markAllAsDirty || this._materialContext === i.materialContext && e(i.defines); } } /** * Indicates that the scene should check if the rendering now needs a prepass */ _markScenePrePassDirty() { if (this.getScene().blockMaterialDirtyMechanism || this._blockDirtyMechanism) return; const e = this.getScene().enablePrePassRenderer(); e && e.markAsDirty(); } /** * Indicates that we need to re-calculated for all submeshes */ _markAllSubMeshesAsAllDirty() { this._markAllSubMeshesAsDirty(gt._AllDirtyCallBack); } /** * Indicates that image processing needs to be re-calculated for all submeshes */ _markAllSubMeshesAsImageProcessingDirty() { this._markAllSubMeshesAsDirty(gt._ImageProcessingDirtyCallBack); } /** * Indicates that textures need to be re-calculated for all submeshes */ _markAllSubMeshesAsTexturesDirty() { this._markAllSubMeshesAsDirty(gt._TextureDirtyCallBack); } /** * Indicates that fresnel needs to be re-calculated for all submeshes */ _markAllSubMeshesAsFresnelDirty() { this._markAllSubMeshesAsDirty(gt._FresnelDirtyCallBack); } /** * Indicates that fresnel and misc need to be re-calculated for all submeshes */ _markAllSubMeshesAsFresnelAndMiscDirty() { this._markAllSubMeshesAsDirty(gt._FresnelAndMiscDirtyCallBack); } /** * Indicates that lights need to be re-calculated for all submeshes */ _markAllSubMeshesAsLightsDirty() { this._markAllSubMeshesAsDirty(gt._LightsDirtyCallBack); } /** * Indicates that attributes need to be re-calculated for all submeshes */ _markAllSubMeshesAsAttributesDirty() { this._markAllSubMeshesAsDirty(gt._AttributeDirtyCallBack); } /** * Indicates that misc needs to be re-calculated for all submeshes */ _markAllSubMeshesAsMiscDirty() { this._markAllSubMeshesAsDirty(gt._MiscDirtyCallBack); } /** * Indicates that prepass needs to be re-calculated for all submeshes */ _markAllSubMeshesAsPrePassDirty() { this._markAllSubMeshesAsDirty(gt._MiscDirtyCallBack); } /** * Indicates that textures and misc need to be re-calculated for all submeshes */ _markAllSubMeshesAsTexturesAndMiscDirty() { this._markAllSubMeshesAsDirty(gt._TextureAndMiscDirtyCallBack); } _checkScenePerformancePriority() { if (this._scene.performancePriority !== $H.BackwardCompatible) { this.checkReadyOnlyOnce = !0; const e = this._scene.onScenePerformancePriorityChangedObservable.addOnce(() => { this.checkReadyOnlyOnce = !1; }); this.onDisposeObservable.add(() => { this._scene.onScenePerformancePriorityChangedObservable.remove(e); }); } } /** * Sets the required values to the prepass renderer. * @param prePassRenderer defines the prepass renderer to setup. * @returns true if the pre pass is needed. */ // eslint-disable-next-line @typescript-eslint/no-unused-vars setPrePassRenderer(e) { return !1; } /** * Disposes the material * @param forceDisposeEffect specifies if effects should be forcefully disposed * @param forceDisposeTextures specifies if textures should be forcefully disposed * @param notBoundToMesh specifies if the material that is being disposed is known to be not bound to any mesh */ dispose(e, t, r) { const n = this.getScene(); if (n.stopAnimation(this), n.freeProcessedMaterials(), n.removeMaterial(this), this._eventInfo.forceDisposeTextures = t, this._callbackPluginEventGeneric(x6.Disposed, this._eventInfo), this._parentContainer) { const i = this._parentContainer.materials.indexOf(this); i > -1 && this._parentContainer.materials.splice(i, 1), this._parentContainer = null; } if (r !== !0) if (this.meshMap) for (const i in this.meshMap) { const s = this.meshMap[i]; s && (s.material = null, this.releaseVertexArrayObject(s, e)); } else { const i = n.meshes; for (const s of i) s.material === this && !s.sourceMesh && (s.material = null, this.releaseVertexArrayObject(s, e)); } this._uniformBuffer.dispose(), e && this._drawWrapper.effect && (this._storeEffectOnSubMeshes || this._drawWrapper.effect.dispose(), this._drawWrapper.effect = null), this.metadata = null, this.onDisposeObservable.notifyObservers(this), this.onDisposeObservable.clear(), this._onBindObservable && this._onBindObservable.clear(), this._onUnBindObservable && this._onUnBindObservable.clear(), this._onEffectCreatedObservable && this._onEffectCreatedObservable.clear(), this._eventInfo && (this._eventInfo = {}); } /** * @internal */ // eslint-disable-next-line @typescript-eslint/naming-convention releaseVertexArrayObject(e, t) { const r = e.geometry; if (r) if (this._storeEffectOnSubMeshes) { if (e.subMeshes) for (const n of e.subMeshes) r._releaseVertexArrayObject(n.effect), t && n.effect && n.effect.dispose(); } else r._releaseVertexArrayObject(this._drawWrapper.effect); } /** * Serializes this material * @returns the serialized material object */ serialize() { const e = jt.Serialize(this); return e.stencil = this.stencil.serialize(), e.uniqueId = this.uniqueId, this._serializePlugins(e), e; } _serializePlugins(e) { if (e.plugins = {}, this.pluginManager) for (const t of this.pluginManager._plugins) e.plugins[t.getClassName()] = t.serialize(); } /** * Creates a material from parsed material data * @param parsedMaterial defines parsed material data * @param scene defines the hosting scene * @param rootUrl defines the root URL to use to load textures * @returns a new material */ static Parse(e, t, r) { if (!e.customType) e.customType = "BABYLON.StandardMaterial"; else if (e.customType === "BABYLON.PBRMaterial" && e.overloadedAlbedo && (e.customType = "BABYLON.LegacyPBRMaterial", !BABYLON.LegacyPBRMaterial)) return Se.Error("Your scene is trying to load a legacy version of the PBRMaterial, please, include it from the materials library."), null; const i = ye.Instantiate(e.customType).Parse(e, t, r); return i._loadedUniqueId = e.uniqueId, i; } static _parsePlugins(e, t, r, n) { var i; if (e.plugins) for (const s in e.plugins) { const a = e.plugins[s]; let f = (i = t.pluginManager) === null || i === void 0 ? void 0 : i.getPlugin(a.name); if (!f) { const o = ye.Instantiate("BABYLON." + s); o && (f = new o(t)); } f == null || f.parse(a, r, n); } } } gt.TriangleFillMode = 0; gt.WireFrameFillMode = 1; gt.PointFillMode = 2; gt.PointListDrawMode = 3; gt.LineListDrawMode = 4; gt.LineLoopDrawMode = 5; gt.LineStripDrawMode = 6; gt.TriangleStripDrawMode = 7; gt.TriangleFanDrawMode = 8; gt.ClockWiseSideOrientation = 0; gt.CounterClockWiseSideOrientation = 1; gt.TextureDirtyFlag = 1; gt.LightDirtyFlag = 2; gt.FresnelDirtyFlag = 4; gt.AttributesDirtyFlag = 8; gt.MiscDirtyFlag = 16; gt.PrePassDirtyFlag = 32; gt.AllDirtyFlag = 63; gt.MATERIAL_OPAQUE = 0; gt.MATERIAL_ALPHATEST = 1; gt.MATERIAL_ALPHABLEND = 2; gt.MATERIAL_ALPHATESTANDBLEND = 3; gt.MATERIAL_NORMALBLENDMETHOD_WHITEOUT = 0; gt.MATERIAL_NORMALBLENDMETHOD_RNM = 1; gt.OnEventObservable = new Oe(); gt._AllDirtyCallBack = (A) => A.markAllAsDirty(); gt._ImageProcessingDirtyCallBack = (A) => A.markAsImageProcessingDirty(); gt._TextureDirtyCallBack = (A) => A.markAsTexturesDirty(); gt._FresnelDirtyCallBack = (A) => A.markAsFresnelDirty(); gt._MiscDirtyCallBack = (A) => A.markAsMiscDirty(); gt._PrePassDirtyCallBack = (A) => A.markAsPrePassDirty(); gt._LightsDirtyCallBack = (A) => A.markAsLightDirty(); gt._AttributeDirtyCallBack = (A) => A.markAsAttributesDirty(); gt._FresnelAndMiscDirtyCallBack = (A) => { gt._FresnelDirtyCallBack(A), gt._MiscDirtyCallBack(A); }; gt._TextureAndMiscDirtyCallBack = (A) => { gt._TextureDirtyCallBack(A), gt._MiscDirtyCallBack(A); }; gt._DirtyCallbackArray = []; gt._RunDirtyCallBacks = (A) => { for (const e of gt._DirtyCallbackArray) e(A); }; C([ M() ], gt.prototype, "id", void 0); C([ M() ], gt.prototype, "uniqueId", void 0); C([ M() ], gt.prototype, "name", void 0); C([ M() ], gt.prototype, "metadata", void 0); C([ M() ], gt.prototype, "checkReadyOnEveryCall", void 0); C([ M() ], gt.prototype, "checkReadyOnlyOnce", void 0); C([ M() ], gt.prototype, "state", void 0); C([ M("alpha") ], gt.prototype, "_alpha", void 0); C([ M("backFaceCulling") ], gt.prototype, "_backFaceCulling", void 0); C([ M("cullBackFaces") ], gt.prototype, "_cullBackFaces", void 0); C([ M() ], gt.prototype, "sideOrientation", void 0); C([ M("alphaMode") ], gt.prototype, "_alphaMode", void 0); C([ M() ], gt.prototype, "_needDepthPrePass", void 0); C([ M() ], gt.prototype, "disableDepthWrite", void 0); C([ M() ], gt.prototype, "disableColorWrite", void 0); C([ M() ], gt.prototype, "forceDepthWrite", void 0); C([ M() ], gt.prototype, "depthFunction", void 0); C([ M() ], gt.prototype, "separateCullingPass", void 0); C([ M("fogEnabled") ], gt.prototype, "_fogEnabled", void 0); C([ M() ], gt.prototype, "pointSize", void 0); C([ M() ], gt.prototype, "zOffset", void 0); C([ M() ], gt.prototype, "zOffsetUnits", void 0); C([ M() ], gt.prototype, "pointsCloud", null); C([ M() ], gt.prototype, "fillMode", null); C([ M() ], gt.prototype, "useLogarithmicDepth", null); C([ M() ], gt.prototype, "transparencyMode", null); class Dc extends gt { /** * Gets or Sets the list of Materials used within the multi material. * They need to be ordered according to the submeshes order in the associated mesh */ get subMaterials() { return this._subMaterials; } set subMaterials(e) { this._subMaterials = e, this._hookArray(e); } /** * Function used to align with Node.getChildren() * @returns the list of Materials used within the multi material */ getChildren() { return this.subMaterials; } /** * Instantiates a new Multi Material * A multi-material is used to apply different materials to different parts of the same object without the need of * separate meshes. This can be use to improve performances. * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/multiMaterials * @param name Define the name in the scene * @param scene Define the scene the material belongs to */ constructor(e, t) { super(e, t, !0), this._waitingSubMaterialsUniqueIds = [], this.getScene().addMultiMaterial(this), this.subMaterials = [], this._storeEffectOnSubMeshes = !0; } _hookArray(e) { const t = e.push; e.push = (...n) => { const i = t.apply(e, n); return this._markAllSubMeshesAsTexturesDirty(), i; }; const r = e.splice; e.splice = (n, i) => { const s = r.apply(e, [n, i]); return this._markAllSubMeshesAsTexturesDirty(), s; }; } /** * Get one of the submaterial by its index in the submaterials array * @param index The index to look the sub material at * @returns The Material if the index has been defined */ getSubMaterial(e) { return e < 0 || e >= this.subMaterials.length ? this.getScene().defaultMaterial : this.subMaterials[e]; } /** * Get the list of active textures for the whole sub materials list. * @returns All the textures that will be used during the rendering */ getActiveTextures() { return super.getActiveTextures().concat(...this.subMaterials.map((e) => e ? e.getActiveTextures() : [])); } /** * Specifies if any sub-materials of this multi-material use a given texture. * @param texture Defines the texture to check against this multi-material's sub-materials. * @returns A boolean specifying if any sub-material of this multi-material uses the texture. */ hasTexture(e) { var t; if (super.hasTexture(e)) return !0; for (let r = 0; r < this.subMaterials.length; r++) if (!((t = this.subMaterials[r]) === null || t === void 0) && t.hasTexture(e)) return !0; return !1; } /** * Gets the current class name of the material e.g. "MultiMaterial" * Mainly use in serialization. * @returns the class name */ getClassName() { return "MultiMaterial"; } /** * Checks if the material is ready to render the requested sub mesh * @param mesh Define the mesh the submesh belongs to * @param subMesh Define the sub mesh to look readiness for * @param useInstances Define whether or not the material is used with instances * @returns true if ready, otherwise false */ isReadyForSubMesh(e, t, r) { for (let n = 0; n < this.subMaterials.length; n++) { const i = this.subMaterials[n]; if (i) { if (i._storeEffectOnSubMeshes) { if (!i.isReadyForSubMesh(e, t, r)) return !1; continue; } if (!i.isReady(e)) return !1; } } return !0; } /** * Clones the current material and its related sub materials * @param name Define the name of the newly cloned material * @param cloneChildren Define if submaterial will be cloned or shared with the parent instance * @returns the cloned material */ clone(e, t) { const r = new Dc(e, this.getScene()); for (let n = 0; n < this.subMaterials.length; n++) { let i = null; const s = this.subMaterials[n]; t && s ? i = s.clone(e + "-" + s.name) : i = this.subMaterials[n], r.subMaterials.push(i); } return r; } /** * Serializes the materials into a JSON representation. * @returns the JSON representation */ serialize() { const e = {}; e.name = this.name, e.id = this.id, e.uniqueId = this.uniqueId, Zi && (e.tags = Zi.GetTags(this)), e.materialsUniqueIds = [], e.materials = []; for (let t = 0; t < this.subMaterials.length; t++) { const r = this.subMaterials[t]; r ? (e.materialsUniqueIds.push(r.uniqueId), e.materials.push(r.id)) : (e.materialsUniqueIds.push(null), e.materials.push(null)); } return e; } /** * Dispose the material and release its associated resources * @param forceDisposeEffect Define if we want to force disposing the associated effect (if false the shader is not released and could be reuse later on) * @param forceDisposeTextures Define if we want to force disposing the associated textures (if false, they will not be disposed and can still be use elsewhere in the app) * @param forceDisposeChildren Define if we want to force disposing the associated submaterials (if false, they will not be disposed and can still be use elsewhere in the app) */ dispose(e, t, r) { const n = this.getScene(); if (!n) return; if (r) for (let s = 0; s < this.subMaterials.length; s++) { const a = this.subMaterials[s]; a && a.dispose(e, t); } const i = n.multiMaterials.indexOf(this); i >= 0 && n.multiMaterials.splice(i, 1), super.dispose(e, t); } /** * Creates a MultiMaterial from parsed MultiMaterial data. * @param parsedMultiMaterial defines parsed MultiMaterial data. * @param scene defines the hosting scene * @returns a new MultiMaterial */ static ParseMultiMaterial(e, t) { const r = new Dc(e.name, t); return r.id = e.id, r._loadedUniqueId = e.uniqueId, Zi && Zi.AddTagsTo(r, e.tags), e.materialsUniqueIds ? r._waitingSubMaterialsUniqueIds = e.materialsUniqueIds : e.materials.forEach((n) => r.subMaterials.push(t.getLastMaterialById(n))), r; } } Ue("BABYLON.MultiMaterial", Dc); class fee { /** * Creates a new LOD level * @param distanceOrScreenCoverage defines either the distance or the screen coverage where this level should start being displayed * @param mesh defines the mesh to use to render this level */ constructor(e, t) { this.distanceOrScreenCoverage = e, this.mesh = t; } } class tQ { } class c6e { constructor() { this.visibleInstances = {}, this.batchCache = new Z5(), this.batchCacheReplacementModeInFrozenMode = new Z5(), this.instancesBufferSize = 32 * 16 * 4; } } class Z5 { constructor() { this.mustReturn = !1, this.visibleInstances = new Array(), this.renderSelf = [], this.hardwareInstancedRendering = []; } } class p6e { constructor() { this.instancesCount = 0, this.matrixBuffer = null, this.previousMatrixBuffer = null, this.matrixBufferSize = 32 * 16, this.matrixData = null, this.boundingVectors = [], this.worldMatrices = null; } } class h6e { constructor() { this._areNormalsFrozen = !1, this._source = null, this.meshMap = null, this._preActivateId = -1, this._LODLevels = new Array(), this._useLODScreenCoverage = !1, this._effectiveMaterial = null, this._forcedInstanceCount = 0, this._overrideRenderingFillMode = null; } } class Ee extends jn { /** * Gets the default side orientation. * @param orientation the orientation to value to attempt to get * @returns the default orientation * @internal */ static _GetDefaultSideOrientation(e) { return e || Ee.FRONTSIDE; } /** * Determines if the LOD levels are intended to be calculated using screen coverage (surface area ratio) instead of distance. */ get useLODScreenCoverage() { return this._internalMeshDataInfo._useLODScreenCoverage; } set useLODScreenCoverage(e) { this._internalMeshDataInfo._useLODScreenCoverage = e, this._sortLODLevels(); } get computeBonesUsingShaders() { return this._internalAbstractMeshDataInfo._computeBonesUsingShaders; } set computeBonesUsingShaders(e) { this._internalAbstractMeshDataInfo._computeBonesUsingShaders !== e && (e && this._internalMeshDataInfo._sourcePositions && (this.setVerticesData(J.PositionKind, this._internalMeshDataInfo._sourcePositions, !0), this._internalMeshDataInfo._sourceNormals && this.setVerticesData(J.NormalKind, this._internalMeshDataInfo._sourceNormals, !0), this._internalMeshDataInfo._sourcePositions = null, this._internalMeshDataInfo._sourceNormals = null), this._internalAbstractMeshDataInfo._computeBonesUsingShaders = e, this._markSubMeshesAsAttributesDirty()); } /** * An event triggered before rendering the mesh */ get onBeforeRenderObservable() { return this._internalMeshDataInfo._onBeforeRenderObservable || (this._internalMeshDataInfo._onBeforeRenderObservable = new Oe()), this._internalMeshDataInfo._onBeforeRenderObservable; } /** * An event triggered before binding the mesh */ get onBeforeBindObservable() { return this._internalMeshDataInfo._onBeforeBindObservable || (this._internalMeshDataInfo._onBeforeBindObservable = new Oe()), this._internalMeshDataInfo._onBeforeBindObservable; } /** * An event triggered after rendering the mesh */ get onAfterRenderObservable() { return this._internalMeshDataInfo._onAfterRenderObservable || (this._internalMeshDataInfo._onAfterRenderObservable = new Oe()), this._internalMeshDataInfo._onAfterRenderObservable; } /** * An event triggeredbetween rendering pass when using separateCullingPass = true */ get onBetweenPassObservable() { return this._internalMeshDataInfo._onBetweenPassObservable || (this._internalMeshDataInfo._onBetweenPassObservable = new Oe()), this._internalMeshDataInfo._onBetweenPassObservable; } /** * An event triggered before drawing the mesh */ get onBeforeDrawObservable() { return this._internalMeshDataInfo._onBeforeDrawObservable || (this._internalMeshDataInfo._onBeforeDrawObservable = new Oe()), this._internalMeshDataInfo._onBeforeDrawObservable; } /** * Sets a callback to call before drawing the mesh. It is recommended to use onBeforeDrawObservable instead */ set onBeforeDraw(e) { this._onBeforeDrawObserver && this.onBeforeDrawObservable.remove(this._onBeforeDrawObserver), this._onBeforeDrawObserver = this.onBeforeDrawObservable.add(e); } get hasInstances() { return this.instances.length > 0; } get hasThinInstances() { return (this.forcedInstanceCount || this._thinInstanceDataStorage.instancesCount || 0) > 0; } /** * Gets or sets the forced number of instances to display. * If 0 (default value), the number of instances is not forced and depends on the draw type * (regular / instance / thin instances mesh) */ get forcedInstanceCount() { return this._internalMeshDataInfo._forcedInstanceCount; } set forcedInstanceCount(e) { this._internalMeshDataInfo._forcedInstanceCount = e; } /** * Use this property to override the Material's fillMode value */ get overrideRenderingFillMode() { return this._internalMeshDataInfo._overrideRenderingFillMode; } set overrideRenderingFillMode(e) { this._internalMeshDataInfo._overrideRenderingFillMode = e; } /** * Gets the source mesh (the one used to clone this one from) */ get source() { return this._internalMeshDataInfo._source; } /** * Gets the list of clones of this mesh * The scene must have been constructed with useClonedMeshMap=true for this to work! * Note that useClonedMeshMap=true is the default setting */ get cloneMeshMap() { return this._internalMeshDataInfo.meshMap; } /** * Gets or sets a boolean indicating that this mesh does not use index buffer */ get isUnIndexed() { return this._unIndexed; } set isUnIndexed(e) { this._unIndexed !== e && (this._unIndexed = e, this._markSubMeshesAsAttributesDirty()); } /** Gets the array buffer used to store the instanced buffer used for instances' world matrices */ get worldMatrixInstancedBuffer() { return this._instanceDataStorage.instancesData; } /** Gets the array buffer used to store the instanced buffer used for instances' previous world matrices */ get previousWorldMatrixInstancedBuffer() { return this._instanceDataStorage.instancesPreviousData; } /** Gets or sets a boolean indicating that the update of the instance buffer of the world matrices is manual */ get manualUpdateOfWorldMatrixInstancedBuffer() { return this._instanceDataStorage.manualUpdate; } set manualUpdateOfWorldMatrixInstancedBuffer(e) { this._instanceDataStorage.manualUpdate = e; } /** Gets or sets a boolean indicating that the update of the instance buffer of the world matrices is manual */ get manualUpdateOfPreviousWorldMatrixInstancedBuffer() { return this._instanceDataStorage.previousManualUpdate; } set manualUpdateOfPreviousWorldMatrixInstancedBuffer(e) { this._instanceDataStorage.previousManualUpdate = e; } /** Gets or sets a boolean indicating that the update of the instance buffer of the world matrices must be performed in all cases (and notably even in frozen mode) */ get forceWorldMatrixInstancedBufferUpdate() { return this._instanceDataStorage.forceMatrixUpdates; } set forceWorldMatrixInstancedBufferUpdate(e) { this._instanceDataStorage.forceMatrixUpdates = e; } /** * @constructor * @param name The value used by scene.getMeshByName() to do a lookup. * @param scene The scene to add this mesh to. * @param parent The parent of this mesh, if it has one * @param source An optional Mesh from which geometry is shared, cloned. * @param doNotCloneChildren When cloning, skip cloning child meshes of source, default False. * When false, achieved by calling a clone(), also passing False. * This will make creation of children, recursive. * @param clonePhysicsImpostor When cloning, include cloning mesh physics impostor, default True. */ constructor(e, t = null, r = null, n = null, i, s = !0) { if (super(e, t), this._internalMeshDataInfo = new h6e(), this.delayLoadState = 0, this.instances = [], this._creationDataStorage = null, this._geometry = null, this._instanceDataStorage = new c6e(), this._thinInstanceDataStorage = new p6e(), this._shouldGenerateFlatShading = !1, this._originalBuilderSideOrientation = Ee.DEFAULTSIDE, this.overrideMaterialSideOrientation = null, this.ignoreCameraMaxZ = !1, t = this.getScene(), this._onBeforeDraw = (a, f, o) => { a && o && (this._uniformBuffer ? this.transferToEffect(f) : o.bindOnlyWorldMatrix(f)); }, n) { if (n._geometry && n._geometry.applyToMesh(this), sA.DeepCopy(n, this, [ "name", "material", "skeleton", "instances", "parent", "uniqueId", "source", "metadata", "morphTargetManager", "hasInstances", "worldMatrixInstancedBuffer", "previousWorldMatrixInstancedBuffer", "hasLODLevels", "geometry", "isBlocked", "areNormalsFrozen", "facetNb", "isFacetDataEnabled", "lightSources", "useBones", "isAnInstance", "collider", "edgesRenderer", "forward", "up", "right", "absolutePosition", "absoluteScaling", "absoluteRotationQuaternion", "isWorldMatrixFrozen", "nonUniformScaling", "behaviors", "worldMatrixFromCache", "hasThinInstances", "cloneMeshMap", "hasBoundingInfo", "physicsBody", "physicsImpostor" ], ["_poseMatrix"]), this._internalMeshDataInfo._source = n, t.useClonedMeshMap && (n._internalMeshDataInfo.meshMap || (n._internalMeshDataInfo.meshMap = {}), n._internalMeshDataInfo.meshMap[this.uniqueId] = this), this._originalBuilderSideOrientation = n._originalBuilderSideOrientation, this._creationDataStorage = n._creationDataStorage, n._ranges) { const a = n._ranges; for (const f in a) Object.prototype.hasOwnProperty.call(a, f) && a[f] && this.createAnimationRange(f, a[f].from, a[f].to); } if (n.metadata && n.metadata.clone ? this.metadata = n.metadata.clone() : this.metadata = n.metadata, this._internalMetadata = n._internalMetadata, Zi && Zi.HasTags(n) && Zi.AddTagsTo(this, Zi.GetTags(n, !0)), this.setEnabled(n.isEnabled(!1)), this.parent = n.parent, this.setPivotMatrix(n.getPivotMatrix()), this.id = e + "." + n.id, this.material = n.material, !i) { const a = n.getDescendants(!0); for (let f = 0; f < a.length; f++) { const o = a[f]; o.clone && o.clone(e + "." + o.name, this); } } if (n.morphTargetManager && (this.morphTargetManager = n.morphTargetManager), t.getPhysicsEngine) { const a = t.getPhysicsEngine(); if (s && a) if (a.getPluginVersion() === 1) { const f = a.getImpostorForPhysicsObject(n); f && (this.physicsImpostor = f.clone(this)); } else a.getPluginVersion() === 2 && n.physicsBody && n.physicsBody.clone(this); } for (let a = 0; a < t.particleSystems.length; a++) { const f = t.particleSystems[a]; f.emitter === n && f.clone(f.name, this); } this.skeleton = n.skeleton, this.refreshBoundingInfo(!0, !0), this.computeWorldMatrix(!0); } r !== null && (this.parent = r), this._instanceDataStorage.hardwareInstancedRendering = this.getEngine().getCaps().instancedArrays, this._internalMeshDataInfo._onMeshReadyObserverAdded = (a) => { a.unregisterOnNextCall = !0, this.isReady(!0) ? this.onMeshReadyObservable.notifyObservers(this) : this._internalMeshDataInfo._checkReadinessObserver || (this._internalMeshDataInfo._checkReadinessObserver = this._scene.onBeforeRenderObservable.add(() => { this.isReady(!0) && (this._scene.onBeforeRenderObservable.remove(this._internalMeshDataInfo._checkReadinessObserver), this._internalMeshDataInfo._checkReadinessObserver = null, this.onMeshReadyObservable.notifyObservers(this)); })); }, this.onMeshReadyObservable = new Oe(this._internalMeshDataInfo._onMeshReadyObserverAdded), n && n.onClonedObservable.notifyObservers(this); } instantiateHierarchy(e = null, t, r) { const n = this.getTotalVertices() === 0 || t && t.doNotInstantiate && (t.doNotInstantiate === !0 || t.doNotInstantiate(this)) ? this.clone("Clone of " + (this.name || this.id), e || this.parent, !0) : this.createInstance("instance of " + (this.name || this.id)); n.parent = e || this.parent, n.position = this.position.clone(), n.scaling = this.scaling.clone(), this.rotationQuaternion ? n.rotationQuaternion = this.rotationQuaternion.clone() : n.rotation = this.rotation.clone(), r && r(this, n); for (const i of this.getChildTransformNodes(!0)) i.getClassName() === "InstancedMesh" && n.getClassName() === "Mesh" && i.sourceMesh === this ? i.instantiateHierarchy(n, { doNotInstantiate: t && t.doNotInstantiate || !1, newSourcedMesh: n }, r) : i.instantiateHierarchy(n, t, r); return n; } /** * Gets the class name * @returns the string "Mesh". */ getClassName() { return "Mesh"; } /** @internal */ get _isMesh() { return !0; } /** * Returns a description of this mesh * @param fullDetails define if full details about this mesh must be used * @returns a descriptive string representing this mesh */ toString(e) { let t = super.toString(e); if (t += ", n vertices: " + this.getTotalVertices(), t += ", parent: " + (this._waitingParentId ? this._waitingParentId : this.parent ? this.parent.name : "NONE"), this.animations) for (let r = 0; r < this.animations.length; r++) t += ", animation[0]: " + this.animations[r].toString(e); if (e) if (this._geometry) { const r = this.getIndices(), n = this.getVerticesData(J.PositionKind); n && r && (t += ", flat shading: " + (n.length / 3 === r.length ? "YES" : "NO")); } else t += ", flat shading: UNKNOWN"; return t; } /** @internal */ _unBindEffect() { super._unBindEffect(); for (const e of this.instances) e._unBindEffect(); } /** * Gets a boolean indicating if this mesh has LOD */ get hasLODLevels() { return this._internalMeshDataInfo._LODLevels.length > 0; } /** * Gets the list of MeshLODLevel associated with the current mesh * @returns an array of MeshLODLevel */ getLODLevels() { return this._internalMeshDataInfo._LODLevels; } _sortLODLevels() { const e = this._internalMeshDataInfo._useLODScreenCoverage ? -1 : 1; this._internalMeshDataInfo._LODLevels.sort((t, r) => t.distanceOrScreenCoverage < r.distanceOrScreenCoverage ? e : t.distanceOrScreenCoverage > r.distanceOrScreenCoverage ? -e : 0); } /** * Add a mesh as LOD level triggered at the given distance. * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/LOD * @param distanceOrScreenCoverage Either distance from the center of the object to show this level or the screen coverage if `useScreenCoverage` is set to `true`. * If screen coverage, value is a fraction of the screen's total surface, between 0 and 1. * Example Playground for distance https://playground.babylonjs.com/#QE7KM#197 * Example Playground for screen coverage https://playground.babylonjs.com/#QE7KM#196 * @param mesh The mesh to be added as LOD level (can be null) * @returns This mesh (for chaining) */ addLODLevel(e, t) { if (t && t._masterMesh) return Se.Warn("You cannot use a mesh as LOD level twice"), this; const r = new fee(e, t); return this._internalMeshDataInfo._LODLevels.push(r), t && (t._masterMesh = this), this._sortLODLevels(), this; } /** * Returns the LOD level mesh at the passed distance or null if not found. * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/LOD * @param distance The distance from the center of the object to show this level * @returns a Mesh or `null` */ getLODLevelAtDistance(e) { const t = this._internalMeshDataInfo; for (let r = 0; r < t._LODLevels.length; r++) { const n = t._LODLevels[r]; if (n.distanceOrScreenCoverage === e) return n.mesh; } return null; } /** * Remove a mesh from the LOD array * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/LOD * @param mesh defines the mesh to be removed * @returns This mesh (for chaining) */ removeLODLevel(e) { const t = this._internalMeshDataInfo; for (let r = 0; r < t._LODLevels.length; r++) t._LODLevels[r].mesh === e && (t._LODLevels.splice(r, 1), e && (e._masterMesh = null)); return this._sortLODLevels(), this; } /** * Returns the registered LOD mesh distant from the parameter `camera` position if any, else returns the current mesh. * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/LOD * @param camera defines the camera to use to compute distance * @param boundingSphere defines a custom bounding sphere to use instead of the one from this mesh * @returns This mesh (for chaining) */ getLOD(e, t) { const r = this._internalMeshDataInfo; if (!r._LODLevels || r._LODLevels.length === 0) return this; const n = t || this.getBoundingInfo().boundingSphere, i = e.mode === Tr.ORTHOGRAPHIC_CAMERA ? e.minZ : n.centerWorld.subtract(e.globalPosition).length(); let s = i, a = 1; if (r._useLODScreenCoverage) { const f = e.screenArea; let o = n.radiusWorld * e.minZ / i; o = o * o * Math.PI, s = o / f, a = -1; } if (a * r._LODLevels[r._LODLevels.length - 1].distanceOrScreenCoverage > a * s) return this.onLODLevelSelection && this.onLODLevelSelection(s, this, this), this; for (let f = 0; f < r._LODLevels.length; f++) { const o = r._LODLevels[f]; if (a * o.distanceOrScreenCoverage < a * s) { if (o.mesh) { if (o.mesh.delayLoadState === 4) return o.mesh._checkDelayState(), this; if (o.mesh.delayLoadState === 2) return this; o.mesh._preActivate(), o.mesh._updateSubMeshesBoundingInfo(this.worldMatrixFromCache); } return this.onLODLevelSelection && this.onLODLevelSelection(s, this, o.mesh), o.mesh; } } return this.onLODLevelSelection && this.onLODLevelSelection(s, this, this), this; } /** * Gets the mesh internal Geometry object */ get geometry() { return this._geometry; } /** * Returns the total number of vertices within the mesh geometry or zero if the mesh has no geometry. * @returns the total number of vertices */ getTotalVertices() { return this._geometry === null || this._geometry === void 0 ? 0 : this._geometry.getTotalVertices(); } /** * Returns the content of an associated vertex buffer * @param kind defines which buffer to read from (positions, indices, normals, etc). Possible `kind` values : * - VertexBuffer.PositionKind * - VertexBuffer.UVKind * - VertexBuffer.UV2Kind * - VertexBuffer.UV3Kind * - VertexBuffer.UV4Kind * - VertexBuffer.UV5Kind * - VertexBuffer.UV6Kind * - VertexBuffer.ColorKind * - VertexBuffer.MatricesIndicesKind * - VertexBuffer.MatricesIndicesExtraKind * - VertexBuffer.MatricesWeightsKind * - VertexBuffer.MatricesWeightsExtraKind * @param copyWhenShared defines a boolean indicating that if the mesh geometry is shared among some other meshes, the returned array is a copy of the internal one * @param forceCopy defines a boolean forcing the copy of the buffer no matter what the value of copyWhenShared is * @param bypassInstanceData defines a boolean indicating that the function should not take into account the instance data (applies only if the mesh has instances). Default: false * @returns a FloatArray or null if the mesh has no geometry or no vertex buffer for this kind. */ getVerticesData(e, t, r, n) { var i, s; if (!this._geometry) return null; let a = n || (s = (i = this._userInstancedBuffersStorage) === null || i === void 0 ? void 0 : i.vertexBuffers[e]) === null || s === void 0 ? void 0 : s.getFloatData( this.instances.length + 1, // +1 because the master mesh is not included in the instances array r || t && this._geometry.meshes.length !== 1 ); return a || (a = this._geometry.getVerticesData(e, t, r)), a; } /** * Returns the mesh VertexBuffer object from the requested `kind` * @param kind defines which buffer to read from (positions, indices, normals, etc). Possible `kind` values : * - VertexBuffer.PositionKind * - VertexBuffer.NormalKind * - VertexBuffer.UVKind * - VertexBuffer.UV2Kind * - VertexBuffer.UV3Kind * - VertexBuffer.UV4Kind * - VertexBuffer.UV5Kind * - VertexBuffer.UV6Kind * - VertexBuffer.ColorKind * - VertexBuffer.MatricesIndicesKind * - VertexBuffer.MatricesIndicesExtraKind * - VertexBuffer.MatricesWeightsKind * - VertexBuffer.MatricesWeightsExtraKind * @param bypassInstanceData defines a boolean indicating that the function should not take into account the instance data (applies only if the mesh has instances). Default: false * @returns a FloatArray or null if the mesh has no vertex buffer for this kind. */ getVertexBuffer(e, t) { var r, n; return this._geometry ? (n = t || (r = this._userInstancedBuffersStorage) === null || r === void 0 ? void 0 : r.vertexBuffers[e]) !== null && n !== void 0 ? n : this._geometry.getVertexBuffer(e) : null; } /** * Tests if a specific vertex buffer is associated with this mesh * @param kind defines which buffer to check (positions, indices, normals, etc). Possible `kind` values : * - VertexBuffer.PositionKind * - VertexBuffer.NormalKind * - VertexBuffer.UVKind * - VertexBuffer.UV2Kind * - VertexBuffer.UV3Kind * - VertexBuffer.UV4Kind * - VertexBuffer.UV5Kind * - VertexBuffer.UV6Kind * - VertexBuffer.ColorKind * - VertexBuffer.MatricesIndicesKind * - VertexBuffer.MatricesIndicesExtraKind * - VertexBuffer.MatricesWeightsKind * - VertexBuffer.MatricesWeightsExtraKind * @param bypassInstanceData defines a boolean indicating that the function should not take into account the instance data (applies only if the mesh has instances). Default: false * @returns a boolean */ isVerticesDataPresent(e, t) { var r; return this._geometry ? !t && ((r = this._userInstancedBuffersStorage) === null || r === void 0 ? void 0 : r.vertexBuffers[e]) !== void 0 || this._geometry.isVerticesDataPresent(e) : this._delayInfo ? this._delayInfo.indexOf(e) !== -1 : !1; } /** * Returns a boolean defining if the vertex data for the requested `kind` is updatable. * @param kind defines which buffer to check (positions, indices, normals, etc). Possible `kind` values : * - VertexBuffer.PositionKind * - VertexBuffer.UVKind * - VertexBuffer.UV2Kind * - VertexBuffer.UV3Kind * - VertexBuffer.UV4Kind * - VertexBuffer.UV5Kind * - VertexBuffer.UV6Kind * - VertexBuffer.ColorKind * - VertexBuffer.MatricesIndicesKind * - VertexBuffer.MatricesIndicesExtraKind * - VertexBuffer.MatricesWeightsKind * - VertexBuffer.MatricesWeightsExtraKind * @param bypassInstanceData defines a boolean indicating that the function should not take into account the instance data (applies only if the mesh has instances). Default: false * @returns a boolean */ isVertexBufferUpdatable(e, t) { var r; if (!this._geometry) return this._delayInfo ? this._delayInfo.indexOf(e) !== -1 : !1; if (!t) { const n = (r = this._userInstancedBuffersStorage) === null || r === void 0 ? void 0 : r.vertexBuffers[e]; if (n) return n.isUpdatable(); } return this._geometry.isVertexBufferUpdatable(e); } /** * Returns a string which contains the list of existing `kinds` of Vertex Data associated with this mesh. * @param bypassInstanceData defines a boolean indicating that the function should not take into account the instance data (applies only if the mesh has instances). Default: false * @returns an array of strings */ getVerticesDataKinds(e) { if (!this._geometry) { const r = []; return this._delayInfo && this._delayInfo.forEach(function(n) { r.push(n); }), r; } const t = this._geometry.getVerticesDataKinds(); if (!e && this._userInstancedBuffersStorage) for (const r in this._userInstancedBuffersStorage.vertexBuffers) t.indexOf(r) === -1 && t.push(r); return t; } /** * Returns a positive integer : the total number of indices in this mesh geometry. * @returns the numner of indices or zero if the mesh has no geometry. */ getTotalIndices() { return this._geometry ? this._geometry.getTotalIndices() : 0; } /** * Returns an array of integers or a typed array (Int32Array, Uint32Array, Uint16Array) populated with the mesh indices. * @param copyWhenShared If true (default false) and and if the mesh geometry is shared among some other meshes, the returned array is a copy of the internal one. * @param forceCopy defines a boolean indicating that the returned array must be cloned upon returning it * @returns the indices array or an empty array if the mesh has no geometry */ getIndices(e, t) { return this._geometry ? this._geometry.getIndices(e, t) : []; } get isBlocked() { return this._masterMesh !== null && this._masterMesh !== void 0; } /** * Determine if the current mesh is ready to be rendered * @param completeCheck defines if a complete check (including materials and lights) has to be done (false by default) * @param forceInstanceSupport will check if the mesh will be ready when used with instances (false by default) * @returns true if all associated assets are ready (material, textures, shaders) */ isReady(e = !1, t = !1) { var r, n, i, s, a, f, o; if (this.delayLoadState === 2 || !super.isReady(e)) return !1; if (!this.subMeshes || this.subMeshes.length === 0 || !e) return !0; const d = this.getEngine(), v = this.getScene(), u = t || d.getCaps().instancedArrays && (this.instances.length > 0 || this.hasThinInstances); this.computeWorldMatrix(); const l = this.material || v.defaultMaterial; if (l) { if (l._storeEffectOnSubMeshes) for (const p of this.subMeshes) { const c = p.getMaterial(); if (c) { if (c._storeEffectOnSubMeshes) { if (!c.isReadyForSubMesh(this, p, u)) return !1; } else if (!c.isReady(this, u)) return !1; } } else if (!l.isReady(this, u)) return !1; } const P = d.currentRenderPassId; for (const p of this.lightSources) { const c = p.getShadowGenerators(); if (!c) continue; const H = c.values(); for (let T = H.next(); T.done !== !0; T = H.next()) { const q = T.value; if (q && (!(!((r = q.getShadowMap()) === null || r === void 0) && r.renderList) || !((n = q.getShadowMap()) === null || n === void 0) && n.renderList && ((s = (i = q.getShadowMap()) === null || i === void 0 ? void 0 : i.renderList) === null || s === void 0 ? void 0 : s.indexOf(this)) !== -1)) { const j = (a = q.getShadowMap().renderPassIds) !== null && a !== void 0 ? a : [d.currentRenderPassId]; for (let w = 0; w < j.length; ++w) { d.currentRenderPassId = j[w]; for (const m of this.subMeshes) if (!q.isReady(m, u, (o = (f = m.getMaterial()) === null || f === void 0 ? void 0 : f.needAlphaBlendingForMesh(this)) !== null && o !== void 0 ? o : !1)) return d.currentRenderPassId = P, !1; } d.currentRenderPassId = P; } } } for (const p of this._internalMeshDataInfo._LODLevels) if (p.mesh && !p.mesh.isReady(u)) return !1; return !0; } /** * Gets a boolean indicating if the normals aren't to be recomputed on next mesh `positions` array update. This property is pertinent only for updatable parametric shapes. */ get areNormalsFrozen() { return this._internalMeshDataInfo._areNormalsFrozen; } /** * This function affects parametric shapes on vertex position update only : ribbons, tubes, etc. It has no effect at all on other shapes. It prevents the mesh normals from being recomputed on next `positions` array update. * @returns the current mesh */ freezeNormals() { return this._internalMeshDataInfo._areNormalsFrozen = !0, this; } /** * This function affects parametric shapes on vertex position update only : ribbons, tubes, etc. It has no effect at all on other shapes. It reactivates the mesh normals computation if it was previously frozen * @returns the current mesh */ unfreezeNormals() { return this._internalMeshDataInfo._areNormalsFrozen = !1, this; } /** * Sets a value overriding the instance count. Only applicable when custom instanced InterleavedVertexBuffer are used rather than InstancedMeshs */ set overridenInstanceCount(e) { this._instanceDataStorage.overridenInstanceCount = e; } // Methods /** @internal */ _preActivate() { const e = this._internalMeshDataInfo, t = this.getScene().getRenderId(); return e._preActivateId === t ? this : (e._preActivateId = t, this._instanceDataStorage.visibleInstances = null, this); } /** * @internal */ _preActivateForIntermediateRendering(e) { return this._instanceDataStorage.visibleInstances && (this._instanceDataStorage.visibleInstances.intermediateDefaultRenderId = e), this; } /** * @internal */ _registerInstanceForRenderId(e, t) { return this._instanceDataStorage.visibleInstances || (this._instanceDataStorage.visibleInstances = { defaultRenderId: t, selfDefaultRenderId: this._renderId }), this._instanceDataStorage.visibleInstances[t] || (this._instanceDataStorage.previousRenderId !== void 0 && this._instanceDataStorage.isFrozen && (this._instanceDataStorage.visibleInstances[this._instanceDataStorage.previousRenderId] = null), this._instanceDataStorage.previousRenderId = t, this._instanceDataStorage.visibleInstances[t] = new Array()), this._instanceDataStorage.visibleInstances[t].push(e), this; } _afterComputeWorldMatrix() { super._afterComputeWorldMatrix(), this.hasThinInstances && (this.doNotSyncBoundingInfo || this.thinInstanceRefreshBoundingInfo(!1)); } /** @internal */ _postActivate() { this.edgesShareWithInstances && this.edgesRenderer && this.edgesRenderer.isEnabled && this._renderingGroup && (this._renderingGroup._edgesRenderers.pushNoDuplicate(this.edgesRenderer), this.edgesRenderer.customInstances.push(this.getWorldMatrix())); } /** * This method recomputes and sets a new BoundingInfo to the mesh unless it is locked. * This means the mesh underlying bounding box and sphere are recomputed. * @param applySkeleton defines whether to apply the skeleton before computing the bounding info * @param applyMorph defines whether to apply the morph target before computing the bounding info * @returns the current mesh */ refreshBoundingInfo(e = !1, t = !1) { if (this.hasBoundingInfo && this.getBoundingInfo().isLocked) return this; const r = this.geometry ? this.geometry.boundingBias : null; return this._refreshBoundingInfo(this._getPositionData(e, t), r), this; } /** * @internal */ _createGlobalSubMesh(e) { const t = this.getTotalVertices(); if (!t || !this.getIndices()) return null; if (this.subMeshes && this.subMeshes.length > 0) { const r = this.getIndices(); if (!r) return null; const n = r.length; let i = !1; if (e) i = !0; else for (const s of this.subMeshes) { if (s.indexStart + s.indexCount > n) { i = !0; break; } if (s.verticesStart + s.verticesCount > t) { i = !0; break; } } if (!i) return this.subMeshes[0]; } return this.releaseSubMeshes(), new rA(0, 0, t, 0, this.getTotalIndices(), this); } /** * This function will subdivide the mesh into multiple submeshes * @param count defines the expected number of submeshes */ subdivide(e) { if (e < 1) return; const t = this.getTotalIndices(); let r = t / e | 0, n = 0; for (; r % 3 !== 0; ) r++; this.releaseSubMeshes(); for (let i = 0; i < e && !(n >= t); i++) rA.CreateFromIndices(0, n, i === e - 1 ? t - n : r, this, void 0, !1), n += r; this.refreshBoundingInfo(), this.synchronizeInstances(); } /** * Copy a FloatArray into a specific associated vertex buffer * @param kind defines which buffer to write to (positions, indices, normals, etc). Possible `kind` values : * - VertexBuffer.PositionKind * - VertexBuffer.UVKind * - VertexBuffer.UV2Kind * - VertexBuffer.UV3Kind * - VertexBuffer.UV4Kind * - VertexBuffer.UV5Kind * - VertexBuffer.UV6Kind * - VertexBuffer.ColorKind * - VertexBuffer.MatricesIndicesKind * - VertexBuffer.MatricesIndicesExtraKind * - VertexBuffer.MatricesWeightsKind * - VertexBuffer.MatricesWeightsExtraKind * @param data defines the data source * @param updatable defines if the updated vertex buffer must be flagged as updatable * @param stride defines the data stride size (can be null) * @returns the current mesh */ setVerticesData(e, t, r = !1, n) { if (this._geometry) this._geometry.setVerticesData(e, t, r, n); else { const i = new Ut(); i.set(t, e); const s = this.getScene(); new Tf(Tf.RandomId(), s, i, r, this); } return this; } /** * Delete a vertex buffer associated with this mesh * @param kind defines which buffer to delete (positions, indices, normals, etc). Possible `kind` values : * - VertexBuffer.PositionKind * - VertexBuffer.UVKind * - VertexBuffer.UV2Kind * - VertexBuffer.UV3Kind * - VertexBuffer.UV4Kind * - VertexBuffer.UV5Kind * - VertexBuffer.UV6Kind * - VertexBuffer.ColorKind * - VertexBuffer.MatricesIndicesKind * - VertexBuffer.MatricesIndicesExtraKind * - VertexBuffer.MatricesWeightsKind * - VertexBuffer.MatricesWeightsExtraKind */ removeVerticesData(e) { this._geometry && this._geometry.removeVerticesData(e); } /** * Flags an associated vertex buffer as updatable * @param kind defines which buffer to use (positions, indices, normals, etc). Possible `kind` values : * - VertexBuffer.PositionKind * - VertexBuffer.UVKind * - VertexBuffer.UV2Kind * - VertexBuffer.UV3Kind * - VertexBuffer.UV4Kind * - VertexBuffer.UV5Kind * - VertexBuffer.UV6Kind * - VertexBuffer.ColorKind * - VertexBuffer.MatricesIndicesKind * - VertexBuffer.MatricesIndicesExtraKind * - VertexBuffer.MatricesWeightsKind * - VertexBuffer.MatricesWeightsExtraKind * @param updatable defines if the updated vertex buffer must be flagged as updatable */ markVerticesDataAsUpdatable(e, t = !0) { const r = this.getVertexBuffer(e); !r || r.isUpdatable() === t || this.setVerticesData(e, this.getVerticesData(e), t); } /** * Sets the mesh global Vertex Buffer * @param buffer defines the buffer to use * @param disposeExistingBuffer disposes the existing buffer, if any (default: true) * @returns the current mesh */ setVerticesBuffer(e, t = !0) { return this._geometry || (this._geometry = Tf.CreateGeometryForMesh(this)), this._geometry.setVerticesBuffer(e, null, t), this; } /** * Update a specific associated vertex buffer * @param kind defines which buffer to write to (positions, indices, normals, etc). Possible `kind` values : * - VertexBuffer.PositionKind * - VertexBuffer.UVKind * - VertexBuffer.UV2Kind * - VertexBuffer.UV3Kind * - VertexBuffer.UV4Kind * - VertexBuffer.UV5Kind * - VertexBuffer.UV6Kind * - VertexBuffer.ColorKind * - VertexBuffer.MatricesIndicesKind * - VertexBuffer.MatricesIndicesExtraKind * - VertexBuffer.MatricesWeightsKind * - VertexBuffer.MatricesWeightsExtraKind * @param data defines the data source * @param updateExtends defines if extends info of the mesh must be updated (can be null). This is mostly useful for "position" kind * @param makeItUnique defines if the geometry associated with the mesh must be cloned to make the change only for this mesh (and not all meshes associated with the same geometry) * @returns the current mesh */ updateVerticesData(e, t, r, n) { return this._geometry ? (n ? (this.makeGeometryUnique(), this.updateVerticesData(e, t, r, !1)) : this._geometry.updateVerticesData(e, t, r), this) : this; } /** * This method updates the vertex positions of an updatable mesh according to the `positionFunction` returned values. * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#other-shapes-updatemeshpositions * @param positionFunction is a simple JS function what is passed the mesh `positions` array. It doesn't need to return anything * @param computeNormals is a boolean (default true) to enable/disable the mesh normal recomputation after the vertex position update * @returns the current mesh */ updateMeshPositions(e, t = !0) { const r = this.getVerticesData(J.PositionKind); if (!r) return this; if (e(r), this.updateVerticesData(J.PositionKind, r, !1, !1), t) { const n = this.getIndices(), i = this.getVerticesData(J.NormalKind); if (!i) return this; Ut.ComputeNormals(r, n, i), this.updateVerticesData(J.NormalKind, i, !1, !1); } return this; } /** * Creates a un-shared specific occurence of the geometry for the mesh. * @returns the current mesh */ makeGeometryUnique() { if (!this._geometry) return this; if (this._geometry.meshes.length === 1) return this; const e = this._geometry, t = this._geometry.copy(Tf.RandomId()); return e.releaseForMesh(this, !0), t.applyToMesh(this), this; } /** * Sets the index buffer of this mesh. * @param indexBuffer Defines the index buffer to use for this mesh * @param totalVertices Defines the total number of vertices used by the buffer * @param totalIndices Defines the total number of indices in the index buffer */ setIndexBuffer(e, t, r) { let n = this._geometry; n || (n = new Tf(Tf.RandomId(), this.getScene(), void 0, void 0, this)), n.setIndexBuffer(e, t, r); } /** * Set the index buffer of this mesh * @param indices defines the source data * @param totalVertices defines the total number of vertices referenced by this index data (can be null) * @param updatable defines if the updated index buffer must be flagged as updatable (default is false) * @returns the current mesh */ setIndices(e, t = null, r = !1) { if (this._geometry) this._geometry.setIndices(e, t, r); else { const n = new Ut(); n.indices = e; const i = this.getScene(); new Tf(Tf.RandomId(), i, n, r, this); } return this; } /** * Update the current index buffer * @param indices defines the source data * @param offset defines the offset in the index buffer where to store the new data (can be null) * @param gpuMemoryOnly defines a boolean indicating that only the GPU memory must be updated leaving the CPU version of the indices unchanged (false by default) * @returns the current mesh */ updateIndices(e, t, r = !1) { return this._geometry ? (this._geometry.updateIndices(e, t, r), this) : this; } /** * Invert the geometry to move from a right handed system to a left handed one. * @returns the current mesh */ toLeftHanded() { return this._geometry ? (this._geometry.toLeftHanded(), this) : this; } /** * @internal */ _bind(e, t, r, n = !0) { if (!this._geometry) return this; const i = this.getScene().getEngine(); this.morphTargetManager && this.morphTargetManager.isUsingTextureForTargets && this.morphTargetManager._bind(t); let s; if (this._unIndexed) s = null; else switch (this._getRenderingFillMode(r)) { case gt.PointFillMode: s = null; break; case gt.WireFrameFillMode: s = e._getLinesIndexBuffer(this.getIndices(), i); break; default: case gt.TriangleFillMode: s = this._geometry.getIndexBuffer(); break; } return !n || !this._userInstancedBuffersStorage || this.hasThinInstances ? this._geometry._bind(t, s) : this._geometry._bind(t, s, this._userInstancedBuffersStorage.vertexBuffers, this._userInstancedBuffersStorage.vertexArrayObjects), this; } /** * @internal */ _draw(e, t, r) { if (!this._geometry || !this._geometry.getVertexBuffers() || !this._unIndexed && !this._geometry.getIndexBuffer()) return this; this._internalMeshDataInfo._onBeforeDrawObservable && this._internalMeshDataInfo._onBeforeDrawObservable.notifyObservers(this); const i = this.getScene().getEngine(); return this._unIndexed || t == gt.PointFillMode ? i.drawArraysType(t, e.verticesStart, e.verticesCount, this.forcedInstanceCount || r) : t == gt.WireFrameFillMode ? i.drawElementsType(t, 0, e._linesIndexCount, this.forcedInstanceCount || r) : i.drawElementsType(t, e.indexStart, e.indexCount, this.forcedInstanceCount || r), this; } /** * Registers for this mesh a javascript function called just before the rendering process * @param func defines the function to call before rendering this mesh * @returns the current mesh */ registerBeforeRender(e) { return this.onBeforeRenderObservable.add(e), this; } /** * Disposes a previously registered javascript function called before the rendering * @param func defines the function to remove * @returns the current mesh */ unregisterBeforeRender(e) { return this.onBeforeRenderObservable.removeCallback(e), this; } /** * Registers for this mesh a javascript function called just after the rendering is complete * @param func defines the function to call after rendering this mesh * @returns the current mesh */ registerAfterRender(e) { return this.onAfterRenderObservable.add(e), this; } /** * Disposes a previously registered javascript function called after the rendering. * @param func defines the function to remove * @returns the current mesh */ unregisterAfterRender(e) { return this.onAfterRenderObservable.removeCallback(e), this; } /** * @internal */ _getInstancesRenderList(e, t = !1) { if (this._instanceDataStorage.isFrozen) { if (t) return this._instanceDataStorage.batchCacheReplacementModeInFrozenMode.hardwareInstancedRendering[e] = !1, this._instanceDataStorage.batchCacheReplacementModeInFrozenMode.renderSelf[e] = !0, this._instanceDataStorage.batchCacheReplacementModeInFrozenMode; if (this._instanceDataStorage.previousBatch) return this._instanceDataStorage.previousBatch; } const r = this.getScene(), n = r._isInIntermediateRendering(), i = n ? this._internalAbstractMeshDataInfo._onlyForInstancesIntermediate : this._internalAbstractMeshDataInfo._onlyForInstances, s = this._instanceDataStorage.batchCache; if (s.mustReturn = !1, s.renderSelf[e] = t || !i && this.isEnabled() && this.isVisible, s.visibleInstances[e] = null, this._instanceDataStorage.visibleInstances && !t) { const a = this._instanceDataStorage.visibleInstances, f = r.getRenderId(), o = n ? a.intermediateDefaultRenderId : a.defaultRenderId; s.visibleInstances[e] = a[f], !s.visibleInstances[e] && o && (s.visibleInstances[e] = a[o]); } return s.hardwareInstancedRendering[e] = !t && this._instanceDataStorage.hardwareInstancedRendering && s.visibleInstances[e] !== null && s.visibleInstances[e] !== void 0, this._instanceDataStorage.previousBatch = s, s; } /** * @internal */ _renderWithInstances(e, t, r, n, i) { var s; const a = r.visibleInstances[e._id], f = a ? a.length : 0, o = this._instanceDataStorage, d = o.instancesBufferSize; let v = o.instancesBuffer, u = o.instancesPreviousBuffer; const P = (f + 1) * 16 * 4; for (; o.instancesBufferSize < P; ) o.instancesBufferSize *= 2; (!o.instancesData || d != o.instancesBufferSize) && (o.instancesData = new Float32Array(o.instancesBufferSize / 4)), (this._scene.needsPreviousWorldMatrices && !o.instancesPreviousData || d != o.instancesBufferSize) && (o.instancesPreviousData = new Float32Array(o.instancesBufferSize / 4)); let p = 0, c = 0; const H = r.renderSelf[e._id], T = !v || d !== o.instancesBufferSize || this._scene.needsPreviousWorldMatrices && !o.instancesPreviousBuffer; if (!this._instanceDataStorage.manualUpdate && (!o.isFrozen || T)) { const q = this.getWorldMatrix(); if (H && (this._scene.needsPreviousWorldMatrices && (o.masterMeshPreviousWorldMatrix ? (o.masterMeshPreviousWorldMatrix.copyToArray(o.instancesPreviousData, p), o.masterMeshPreviousWorldMatrix.copyFrom(q)) : (o.masterMeshPreviousWorldMatrix = q.clone(), o.masterMeshPreviousWorldMatrix.copyToArray(o.instancesPreviousData, p))), q.copyToArray(o.instancesData, p), p += 16, c++), a) { if (Ee.INSTANCEDMESH_SORT_TRANSPARENT && this._scene.activeCamera && (!((s = e.getMaterial()) === null || s === void 0) && s.needAlphaBlendingForMesh(e.getRenderingMesh()))) { const b = this._scene.activeCamera.globalPosition; for (let j = 0; j < a.length; j++) { const w = a[j]; w._distanceToCamera = S.Distance(w.getBoundingInfo().boundingSphere.centerWorld, b); } a.sort((j, w) => j._distanceToCamera > w._distanceToCamera ? -1 : j._distanceToCamera < w._distanceToCamera ? 1 : 0); } for (let b = 0; b < a.length; b++) { const j = a[b], w = j.getWorldMatrix(); w.copyToArray(o.instancesData, p), this._scene.needsPreviousWorldMatrices && (j._previousWorldMatrix ? (j._previousWorldMatrix.copyToArray(o.instancesPreviousData, p), j._previousWorldMatrix.copyFrom(w)) : (j._previousWorldMatrix = w.clone(), j._previousWorldMatrix.copyToArray(o.instancesPreviousData, p))), p += 16, c++; } } } else c = (H ? 1 : 0) + f; return T ? (v && v.dispose(), u && u.dispose(), v = new P9(i, o.instancesData, !0, 16, !1, !0), o.instancesBuffer = v, this._userInstancedBuffersStorage || (this._userInstancedBuffersStorage = { data: {}, vertexBuffers: {}, strides: {}, sizes: {}, vertexArrayObjects: this.getEngine().getCaps().vertexArrayObject ? {} : void 0 }), this._userInstancedBuffersStorage.vertexBuffers.world0 = v.createVertexBuffer("world0", 0, 4), this._userInstancedBuffersStorage.vertexBuffers.world1 = v.createVertexBuffer("world1", 4, 4), this._userInstancedBuffersStorage.vertexBuffers.world2 = v.createVertexBuffer("world2", 8, 4), this._userInstancedBuffersStorage.vertexBuffers.world3 = v.createVertexBuffer("world3", 12, 4), this._scene.needsPreviousWorldMatrices && (u = new P9(i, o.instancesPreviousData, !0, 16, !1, !0), o.instancesPreviousBuffer = u, this._userInstancedBuffersStorage.vertexBuffers.previousWorld0 = u.createVertexBuffer("previousWorld0", 0, 4), this._userInstancedBuffersStorage.vertexBuffers.previousWorld1 = u.createVertexBuffer("previousWorld1", 4, 4), this._userInstancedBuffersStorage.vertexBuffers.previousWorld2 = u.createVertexBuffer("previousWorld2", 8, 4), this._userInstancedBuffersStorage.vertexBuffers.previousWorld3 = u.createVertexBuffer("previousWorld3", 12, 4)), this._invalidateInstanceVertexArrayObject()) : (!this._instanceDataStorage.isFrozen || this._instanceDataStorage.forceMatrixUpdates) && (v.updateDirectly(o.instancesData, 0, c), this._scene.needsPreviousWorldMatrices && (!this._instanceDataStorage.manualUpdate || this._instanceDataStorage.previousManualUpdate) && u.updateDirectly(o.instancesPreviousData, 0, c)), this._processInstancedBuffers(a, H), this.getScene()._activeIndices.addCount(e.indexCount * c, !1), i._currentDrawContext && (i._currentDrawContext.useInstancing = !0), this._bind(e, n, t), this._draw(e, t, c), this._scene.needsPreviousWorldMatrices && !T && this._instanceDataStorage.manualUpdate && (!this._instanceDataStorage.isFrozen || this._instanceDataStorage.forceMatrixUpdates) && !this._instanceDataStorage.previousManualUpdate && u.updateDirectly(o.instancesData, 0, c), i.unbindInstanceAttributes(), this; } /** * @internal */ _renderWithThinInstances(e, t, r, n) { var i, s; const a = (s = (i = this._thinInstanceDataStorage) === null || i === void 0 ? void 0 : i.instancesCount) !== null && s !== void 0 ? s : 0; this.getScene()._activeIndices.addCount(e.indexCount * a, !1), n._currentDrawContext && (n._currentDrawContext.useInstancing = !0), this._bind(e, r, t), this._draw(e, t, a), this._scene.needsPreviousWorldMatrices && !this._thinInstanceDataStorage.previousMatrixData && this._thinInstanceDataStorage.matrixData && (this._thinInstanceDataStorage.previousMatrixBuffer ? this._thinInstanceDataStorage.previousMatrixBuffer.updateDirectly(this._thinInstanceDataStorage.matrixData, 0, a) : this._thinInstanceDataStorage.previousMatrixBuffer = this._thinInstanceCreateMatrixBuffer("previousWorld", this._thinInstanceDataStorage.matrixData, !1)), n.unbindInstanceAttributes(); } /** * @internal */ // eslint-disable-next-line @typescript-eslint/no-unused-vars _processInstancedBuffers(e, t) { } /** * @internal */ _processRendering(e, t, r, n, i, s, a, f) { const o = this.getScene(), d = o.getEngine(); if (n = this._getRenderingFillMode(n), s && t.getRenderingMesh().hasThinInstances) return this._renderWithThinInstances(t, n, r, d), this; if (s) this._renderWithInstances(t, n, i, r, d); else { d._currentDrawContext && (d._currentDrawContext.useInstancing = !1); let v = 0; i.renderSelf[t._id] && (a && a(!1, e.getWorldMatrix(), f), v++, this._draw(t, n, this._instanceDataStorage.overridenInstanceCount)); const u = i.visibleInstances[t._id]; if (u) { const l = u.length; v += l; for (let P = 0; P < l; P++) { const c = u[P].getWorldMatrix(); a && a(!0, c, f), this._draw(t, n); } } o._activeIndices.addCount(t.indexCount * v, !1); } return this; } /** * @internal */ _rebuild(e = !1) { if (this._instanceDataStorage.instancesBuffer && (e && this._instanceDataStorage.instancesBuffer.dispose(), this._instanceDataStorage.instancesBuffer = null), this._userInstancedBuffersStorage) { for (const t in this._userInstancedBuffersStorage.vertexBuffers) { const r = this._userInstancedBuffersStorage.vertexBuffers[t]; r && (e && r.dispose(), this._userInstancedBuffersStorage.vertexBuffers[t] = null); } this._userInstancedBuffersStorage.vertexArrayObjects && (this._userInstancedBuffersStorage.vertexArrayObjects = {}); } this._internalMeshDataInfo._effectiveMaterial = null, super._rebuild(e); } /** @internal */ _freeze() { if (this.subMeshes) { for (let e = 0; e < this.subMeshes.length; e++) this._getInstancesRenderList(e); this._internalMeshDataInfo._effectiveMaterial = null, this._instanceDataStorage.isFrozen = !0; } } /** @internal */ _unFreeze() { this._instanceDataStorage.isFrozen = !1, this._instanceDataStorage.previousBatch = null; } /** * Triggers the draw call for the mesh (or a submesh), for a specific render pass id * @param renderPassId defines the render pass id to use to draw the mesh / submesh. If not provided, use the current renderPassId of the engine. * @param enableAlphaMode defines if alpha mode can be changed (default: false) * @param effectiveMeshReplacement defines an optional mesh used to provide info for the rendering (default: undefined) * @param subMesh defines the subMesh to render. If not provided, draw all mesh submeshes (default: undefined) * @param checkFrustumCulling defines if frustum culling must be checked (default: true). If you know the mesh is in the frustum (or if you don't care!), you can pass false to optimize. * @returns the current mesh */ renderWithRenderPassId(e, t, r, n, i = !0) { const s = this._scene.getEngine(), a = s.currentRenderPassId; if (e !== void 0 && (s.currentRenderPassId = e), n) (!i || i && n.isInFrustum(this._scene._frustumPlanes)) && this.render(n, !!t, r); else for (let f = 0; f < this.subMeshes.length; f++) { const o = this.subMeshes[f]; (!i || i && o.isInFrustum(this._scene._frustumPlanes)) && this.render(o, !!t, r); } return e !== void 0 && (s.currentRenderPassId = a), this; } /** * Triggers the draw call for the mesh. Usually, you don't need to call this method by your own because the mesh rendering is handled by the scene rendering manager * @param subMesh defines the subMesh to render * @param enableAlphaMode defines if alpha mode can be changed * @param effectiveMeshReplacement defines an optional mesh used to provide info for the rendering * @returns the current mesh */ render(e, t, r) { var n, i, s, a, f; const o = this.getScene(); this._internalAbstractMeshDataInfo._isActiveIntermediate ? this._internalAbstractMeshDataInfo._isActiveIntermediate = !1 : this._internalAbstractMeshDataInfo._isActive = !1; const d = (i = (n = o.activeCameras) === null || n === void 0 ? void 0 : n.length) !== null && i !== void 0 ? i : 0; if ((d > 1 && o.activeCamera === o.activeCameras[0] || d <= 1) && this._checkOcclusionQuery() && !this._occlusionDataStorage.forceRenderingWhenOccluded) return this; const u = this._getInstancesRenderList(e._id, !!r); if (u.mustReturn) return this; if (!this._geometry || !this._geometry.getVertexBuffers() || !this._unIndexed && !this._geometry.getIndexBuffer()) return this; const l = o.getEngine(); let P = 0, p = null; this.ignoreCameraMaxZ && o.activeCamera && !o._isInIntermediateRendering() && (P = o.activeCamera.maxZ, p = o.activeCamera, o.activeCamera.maxZ = 0, o.updateTransformMatrix(!0)), this._internalMeshDataInfo._onBeforeRenderObservable && this._internalMeshDataInfo._onBeforeRenderObservable.notifyObservers(this); const c = e.getRenderingMesh(), H = u.hardwareInstancedRendering[e._id] || c.hasThinInstances || !!this._userInstancedBuffersStorage && !e.getMesh()._internalAbstractMeshDataInfo._actAsRegularMesh, T = this._instanceDataStorage, q = e.getMaterial(); if (!q) return p && (p.maxZ = P, o.updateTransformMatrix(!0)), this; if (!T.isFrozen || !this._internalMeshDataInfo._effectiveMaterial || this._internalMeshDataInfo._effectiveMaterial !== q) { if (q._storeEffectOnSubMeshes) { if (!q.isReadyForSubMesh(this, e, H)) return p && (p.maxZ = P, o.updateTransformMatrix(!0)), this; } else if (!q.isReady(this, H)) return p && (p.maxZ = P, o.updateTransformMatrix(!0)), this; this._internalMeshDataInfo._effectiveMaterial = q; } else if (q._storeEffectOnSubMeshes && !(!((s = e.effect) === null || s === void 0) && s._wasPreviouslyReady) || !q._storeEffectOnSubMeshes && !(!((a = q.getEffect()) === null || a === void 0) && a._wasPreviouslyReady)) return p && (p.maxZ = P, o.updateTransformMatrix(!0)), this; t && l.setAlphaMode(this._internalMeshDataInfo._effectiveMaterial.alphaMode); let b; this._internalMeshDataInfo._effectiveMaterial._storeEffectOnSubMeshes ? b = e._drawWrapper : b = this._internalMeshDataInfo._effectiveMaterial._getDrawWrapper(); const j = (f = b == null ? void 0 : b.effect) !== null && f !== void 0 ? f : null; for (const y of o._beforeRenderingMeshStage) y.action(this, e, u, j); if (!b || !j) return p && (p.maxZ = P, o.updateTransformMatrix(!0)), this; const w = r || this; let m; if (!T.isFrozen && (this._internalMeshDataInfo._effectiveMaterial.backFaceCulling || this.overrideMaterialSideOrientation !== null || this._internalMeshDataInfo._effectiveMaterial.twoSidedLighting)) { const y = w._getWorldMatrixDeterminant(); m = this.overrideMaterialSideOrientation, m == null && (m = this._internalMeshDataInfo._effectiveMaterial.sideOrientation), y < 0 && (m = m === gt.ClockWiseSideOrientation ? gt.CounterClockWiseSideOrientation : gt.ClockWiseSideOrientation), T.sideOrientation = m; } else m = T.sideOrientation; const I = this._internalMeshDataInfo._effectiveMaterial._preBind(b, m); this._internalMeshDataInfo._effectiveMaterial.forceDepthWrite && l.setDepthWrite(!0); const N = this._internalMeshDataInfo._effectiveMaterial, k = N.fillMode; this._internalMeshDataInfo._onBeforeBindObservable && this._internalMeshDataInfo._onBeforeBindObservable.notifyObservers(this), H || this._bind(e, j, k, !1); const R = w.getWorldMatrix(); N._storeEffectOnSubMeshes ? N.bindForSubMesh(R, this, e) : N.bind(R, this), !N.backFaceCulling && N.separateCullingPass && (l.setState(!0, N.zOffset, !1, !I, N.cullBackFaces, N.stencil, N.zOffsetUnits), this._processRendering(this, e, j, k, u, H, this._onBeforeDraw, this._internalMeshDataInfo._effectiveMaterial), l.setState(!0, N.zOffset, !1, I, N.cullBackFaces, N.stencil, N.zOffsetUnits), this._internalMeshDataInfo._onBetweenPassObservable && this._internalMeshDataInfo._onBetweenPassObservable.notifyObservers(e)), this._processRendering(this, e, j, k, u, H, this._onBeforeDraw, this._internalMeshDataInfo._effectiveMaterial), this._internalMeshDataInfo._effectiveMaterial.unbind(); for (const y of o._afterRenderingMeshStage) y.action(this, e, u, j); return this._internalMeshDataInfo._onAfterRenderObservable && this._internalMeshDataInfo._onAfterRenderObservable.notifyObservers(this), p && (p.maxZ = P, o.updateTransformMatrix(!0)), o.performancePriority === $H.Aggressive && !T.isFrozen && this._freeze(), this; } /** * Renormalize the mesh and patch it up if there are no weights * Similar to normalization by adding the weights compute the reciprocal and multiply all elements, this wil ensure that everything adds to 1. * However in the case of zero weights then we set just a single influence to 1. * We check in the function for extra's present and if so we use the normalizeSkinWeightsWithExtras rather than the FourWeights version. */ cleanMatrixWeights() { this.isVerticesDataPresent(J.MatricesWeightsKind) && (this.isVerticesDataPresent(J.MatricesWeightsExtraKind) ? this._normalizeSkinWeightsAndExtra() : this._normalizeSkinFourWeights()); } // faster 4 weight version. _normalizeSkinFourWeights() { const e = this.getVerticesData(J.MatricesWeightsKind), t = e.length; for (let r = 0; r < t; r += 4) { const n = e[r] + e[r + 1] + e[r + 2] + e[r + 3]; if (n === 0) e[r] = 1; else { const i = 1 / n; e[r] *= i, e[r + 1] *= i, e[r + 2] *= i, e[r + 3] *= i; } } this.setVerticesData(J.MatricesWeightsKind, e); } // handle special case of extra verts. (in theory gltf can handle 12 influences) _normalizeSkinWeightsAndExtra() { const e = this.getVerticesData(J.MatricesWeightsExtraKind), t = this.getVerticesData(J.MatricesWeightsKind), r = t.length; for (let n = 0; n < r; n += 4) { let i = t[n] + t[n + 1] + t[n + 2] + t[n + 3]; if (i += e[n] + e[n + 1] + e[n + 2] + e[n + 3], i === 0) t[n] = 1; else { const s = 1 / i; t[n] *= s, t[n + 1] *= s, t[n + 2] *= s, t[n + 3] *= s, e[n] *= s, e[n + 1] *= s, e[n + 2] *= s, e[n + 3] *= s; } } this.setVerticesData(J.MatricesWeightsKind, t), this.setVerticesData(J.MatricesWeightsKind, e); } /** * ValidateSkinning is used to determine that a mesh has valid skinning data along with skin metrics, if missing weights, * or not normalized it is returned as invalid mesh the string can be used for console logs, or on screen messages to let * the user know there was an issue with importing the mesh * @returns a validation object with skinned, valid and report string */ validateSkinning() { const e = this.getVerticesData(J.MatricesWeightsExtraKind), t = this.getVerticesData(J.MatricesWeightsKind); if (t === null || this.skeleton == null) return { skinned: !1, valid: !0, report: "not skinned" }; const r = t.length; let n = 0, i = 0, s = 0, a = 0; const f = e === null ? 4 : 8, o = []; for (let c = 0; c <= f; c++) o[c] = 0; const d = 1e-3; for (let c = 0; c < r; c += 4) { let H = t[c], T = H, q = T === 0 ? 0 : 1; for (let b = 1; b < f; b++) { const j = b < 4 ? t[c + b] : e[c + b - 4]; j > H && n++, j !== 0 && q++, T += j, H = j; } if (o[q]++, q > s && (s = q), T === 0) i++; else { const b = 1 / T; let j = 0; for (let w = 0; w < f; w++) w < 4 ? j += Math.abs(t[c + w] - t[c + w] * b) : j += Math.abs(e[c + w - 4] - e[c + w - 4] * b); j > d && a++; } } const v = this.skeleton.bones.length, u = this.getVerticesData(J.MatricesIndicesKind), l = this.getVerticesData(J.MatricesIndicesExtraKind); let P = 0; for (let c = 0; c < r; c += 4) for (let H = 0; H < f; H++) { const T = H < 4 ? u[c + H] : l[c + H - 4]; (T >= v || T < 0) && P++; } const p = "Number of Weights = " + r / 4 + ` Maximum influences = ` + s + ` Missing Weights = ` + i + ` Not Sorted = ` + n + ` Not Normalized = ` + a + ` WeightCounts = [` + o + `] Number of bones = ` + v + ` Bad Bone Indices = ` + P; return { skinned: !0, valid: i === 0 && a === 0 && P === 0, report: p }; } /** @internal */ _checkDelayState() { const e = this.getScene(); return this._geometry ? this._geometry.load(e) : this.delayLoadState === 4 && (this.delayLoadState = 2, this._queueLoad(e)), this; } _queueLoad(e) { e.addPendingData(this); const t = this.delayLoadingFile.indexOf(".babylonbinarymeshdata") !== -1; return ye.LoadFile(this.delayLoadingFile, (r) => { r instanceof ArrayBuffer ? this._delayLoadingFunction(r, this) : this._delayLoadingFunction(JSON.parse(r), this), this.instances.forEach((n) => { n.refreshBoundingInfo(), n._syncSubMeshes(); }), this.delayLoadState = 1, e.removePendingData(this); }, () => { }, e.offlineProvider, t), this; } /** * Returns `true` if the mesh is within the frustum defined by the passed array of planes. * A mesh is in the frustum if its bounding box intersects the frustum * @param frustumPlanes defines the frustum to test * @returns true if the mesh is in the frustum planes */ isInFrustum(e) { return this.delayLoadState === 2 || !super.isInFrustum(e) ? !1 : (this._checkDelayState(), !0); } /** * Sets the mesh material by the material or multiMaterial `id` property * @param id is a string identifying the material or the multiMaterial * @returns the current mesh */ setMaterialById(e) { const t = this.getScene().materials; let r; for (r = t.length - 1; r > -1; r--) if (t[r].id === e) return this.material = t[r], this; const n = this.getScene().multiMaterials; for (r = n.length - 1; r > -1; r--) if (n[r].id === e) return this.material = n[r], this; return this; } /** * Returns as a new array populated with the mesh material and/or skeleton, if any. * @returns an array of IAnimatable */ getAnimatables() { const e = []; return this.material && e.push(this.material), this.skeleton && e.push(this.skeleton), e; } /** * Modifies the mesh geometry according to the passed transformation matrix. * This method returns nothing, but it really modifies the mesh even if it's originally not set as updatable. * The mesh normals are modified using the same transformation. * Note that, under the hood, this method sets a new VertexBuffer each call. * @param transform defines the transform matrix to use * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/bakingTransforms * @returns the current mesh */ bakeTransformIntoVertices(e) { if (!this.isVerticesDataPresent(J.PositionKind)) return this; const t = this.subMeshes.splice(0); this._resetPointsArrayCache(); let r = this.getVerticesData(J.PositionKind); const n = S.Zero(); let i; for (i = 0; i < r.length; i += 3) S.TransformCoordinatesFromFloatsToRef(r[i], r[i + 1], r[i + 2], e, n).toArray(r, i); if (this.setVerticesData(J.PositionKind, r, this.getVertexBuffer(J.PositionKind).isUpdatable()), this.isVerticesDataPresent(J.NormalKind)) { for (r = this.getVerticesData(J.NormalKind), i = 0; i < r.length; i += 3) S.TransformNormalFromFloatsToRef(r[i], r[i + 1], r[i + 2], e, n).normalize().toArray(r, i); this.setVerticesData(J.NormalKind, r, this.getVertexBuffer(J.NormalKind).isUpdatable()); } return e.determinant() < 0 && this.flipFaces(), this.releaseSubMeshes(), this.subMeshes = t, this; } /** * Modifies the mesh geometry according to its own current World Matrix. * The mesh World Matrix is then reset. * This method returns nothing but really modifies the mesh even if it's originally not set as updatable. * Note that, under the hood, this method sets a new VertexBuffer each call. * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/bakingTransforms * @param bakeIndependentlyOfChildren indicates whether to preserve all child nodes' World Matrix during baking * @returns the current mesh */ bakeCurrentTransformIntoVertices(e = !0) { return this.bakeTransformIntoVertices(this.computeWorldMatrix(!0)), this.resetLocalMatrix(e), this; } // Cache /** @internal */ get _positions() { return this._internalAbstractMeshDataInfo._positions ? this._internalAbstractMeshDataInfo._positions : this._geometry ? this._geometry._positions : null; } /** @internal */ _resetPointsArrayCache() { return this._geometry && this._geometry._resetPointsArrayCache(), this; } /** @internal */ _generatePointsArray() { return this._geometry ? this._geometry._generatePointsArray() : !1; } /** * Returns a new Mesh object generated from the current mesh properties. * This method must not get confused with createInstance() * @param name is a string, the name given to the new mesh * @param newParent can be any Node object (default `null`) * @param doNotCloneChildren allows/denies the recursive cloning of the original mesh children if any (default `false`) * @param clonePhysicsImpostor allows/denies the cloning in the same time of the original mesh `body` used by the physics engine, if any (default `true`) * @returns a new mesh */ clone(e = "", t = null, r, n = !0) { return new Ee(e, this.getScene(), t, this, r, n); } /** * Releases resources associated with this mesh. * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default) * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default) */ dispose(e, t = !1) { this.morphTargetManager = null, this._geometry && this._geometry.releaseForMesh(this, !0); const r = this._internalMeshDataInfo; if (r._onBeforeDrawObservable && r._onBeforeDrawObservable.clear(), r._onBeforeBindObservable && r._onBeforeBindObservable.clear(), r._onBeforeRenderObservable && r._onBeforeRenderObservable.clear(), r._onAfterRenderObservable && r._onAfterRenderObservable.clear(), r._onBetweenPassObservable && r._onBetweenPassObservable.clear(), this._scene.useClonedMeshMap) { if (r.meshMap) for (const n in r.meshMap) { const i = r.meshMap[n]; i && (i._internalMeshDataInfo._source = null, r.meshMap[n] = void 0); } r._source && r._source._internalMeshDataInfo.meshMap && (r._source._internalMeshDataInfo.meshMap[this.uniqueId] = void 0); } else { const n = this.getScene().meshes; for (const i of n) { const s = i; s._internalMeshDataInfo && s._internalMeshDataInfo._source && s._internalMeshDataInfo._source === this && (s._internalMeshDataInfo._source = null); } } r._source = null, this._instanceDataStorage.visibleInstances = {}, this._disposeInstanceSpecificData(), this._disposeThinInstanceSpecificData(), this._internalMeshDataInfo._checkReadinessObserver && this._scene.onBeforeRenderObservable.remove(this._internalMeshDataInfo._checkReadinessObserver), super.dispose(e, t); } /** @internal */ _disposeInstanceSpecificData() { } /** @internal */ _disposeThinInstanceSpecificData() { } /** @internal */ _invalidateInstanceVertexArrayObject() { } /** * Modifies the mesh geometry according to a displacement map. * A displacement map is a colored image. Each pixel color value (actually a gradient computed from red, green, blue values) will give the displacement to apply to each mesh vertex. * The mesh must be set as updatable. Its internal geometry is directly modified, no new buffer are allocated. * @param url is a string, the URL from the image file is to be downloaded. * @param minHeight is the lower limit of the displacement. * @param maxHeight is the upper limit of the displacement. * @param onSuccess is an optional Javascript function to be called just after the mesh is modified. It is passed the modified mesh and must return nothing. * @param uvOffset is an optional vector2 used to offset UV. * @param uvScale is an optional vector2 used to scale UV. * @param forceUpdate defines whether or not to force an update of the generated buffers. This is useful to apply on a deserialized model for instance. * @returns the Mesh. */ applyDisplacementMap(e, t, r, n, i, s, a = !1) { const f = this.getScene(), o = (d) => { const v = d.width, u = d.height, P = this.getEngine().createCanvas(v, u).getContext("2d"); P.drawImage(d, 0, 0); const p = P.getImageData(0, 0, v, u).data; this.applyDisplacementMapFromBuffer(p, v, u, t, r, i, s, a), n && n(this); }; return ye.LoadImage(e, o, () => { }, f.offlineProvider), this; } /** * Modifies the mesh geometry according to a displacementMap buffer. * A displacement map is a colored image. Each pixel color value (actually a gradient computed from red, green, blue values) will give the displacement to apply to each mesh vertex. * The mesh must be set as updatable. Its internal geometry is directly modified, no new buffer are allocated. * @param buffer is a `Uint8Array` buffer containing series of `Uint8` lower than 255, the red, green, blue and alpha values of each successive pixel. * @param heightMapWidth is the width of the buffer image. * @param heightMapHeight is the height of the buffer image. * @param minHeight is the lower limit of the displacement. * @param maxHeight is the upper limit of the displacement. * @param uvOffset is an optional vector2 used to offset UV. * @param uvScale is an optional vector2 used to scale UV. * @param forceUpdate defines whether or not to force an update of the generated buffers. This is useful to apply on a deserialized model for instance. * @returns the Mesh. */ applyDisplacementMapFromBuffer(e, t, r, n, i, s, a, f = !1) { if (!this.isVerticesDataPresent(J.PositionKind) || !this.isVerticesDataPresent(J.NormalKind) || !this.isVerticesDataPresent(J.UVKind)) return Se.Warn("Cannot call applyDisplacementMap: Given mesh is not complete. Position, Normal or UV are missing"), this; const o = this.getVerticesData(J.PositionKind, !0, !0), d = this.getVerticesData(J.NormalKind), v = this.getVerticesData(J.UVKind); let u = S.Zero(); const l = S.Zero(), P = at.Zero(); s = s || at.Zero(), a = a || new at(1, 1); for (let p = 0; p < o.length; p += 3) { S.FromArrayToRef(o, p, u), S.FromArrayToRef(d, p, l), at.FromArrayToRef(v, p / 3 * 2, P); const c = Math.abs(P.x * a.x + s.x % 1) * (t - 1) % t | 0, H = Math.abs(P.y * a.y + s.y % 1) * (r - 1) % r | 0, T = (c + H * t) * 4, q = e[T] / 255, b = e[T + 1] / 255, j = e[T + 2] / 255, w = q * 0.3 + b * 0.59 + j * 0.11; l.normalize(), l.scaleInPlace(n + (i - n) * w), u = u.add(l), u.toArray(o, p); } return Ut.ComputeNormals(o, this.getIndices(), d), f ? (this.setVerticesData(J.PositionKind, o), this.setVerticesData(J.NormalKind, d), this.setVerticesData(J.UVKind, v)) : (this.updateVerticesData(J.PositionKind, o), this.updateVerticesData(J.NormalKind, d)), this; } _getFlattenedNormals(e, t) { const r = new Float32Array(e.length * 3); let n = 0; const i = this.overrideMaterialSideOrientation === (this._scene.useRightHandedSystem ? 1 : 0); for (let s = 0; s < e.length; s += 3) { const a = S.FromArray(t, e[s] * 3), f = S.FromArray(t, e[s + 1] * 3), o = S.FromArray(t, e[s + 2] * 3), d = a.subtract(f), v = o.subtract(f), u = S.Normalize(S.Cross(d, v)); i && u.scaleInPlace(-1); for (let l = 0; l < 3; l++) r[n++] = u.x, r[n++] = u.y, r[n++] = u.z; } return r; } _convertToUnIndexedMesh(e = !1) { const t = this.getVerticesDataKinds(), r = this.getIndices(), n = {}, i = (a, f) => { const o = new Float32Array(r.length * f); let d = 0; for (let v = 0; v < r.length; v++) for (let u = 0; u < f; u++) o[d++] = a[r[v] * f + u]; return o; }, s = this.geometry ? this.subMeshes.slice(0) : []; for (const a of t) n[a] = this.getVerticesData(a); for (const a of t) { const f = this.getVertexBuffer(a), o = f.getStrideSize(); if (e && a === J.NormalKind) { const d = this._getFlattenedNormals(r, n[J.PositionKind]); this.setVerticesData(J.NormalKind, d, f.isUpdatable(), o); } else this.setVerticesData(a, i(n[a], o), f.isUpdatable(), o); } if (this.morphTargetManager) { for (let a = 0; a < this.morphTargetManager.numTargets; a++) { const f = this.morphTargetManager.getTarget(a), o = f.getPositions(); f.setPositions(i(o, 3)); const d = f.getNormals(); d && f.setNormals(e ? this._getFlattenedNormals(r, o) : i(d, 3)); const v = f.getTangents(); v && f.setTangents(i(v, 3)); const u = f.getUVs(); u && f.setUVs(i(u, 2)); } this.morphTargetManager.synchronize(); } for (let a = 0; a < r.length; a++) r[a] = a; this.setIndices(r), this._unIndexed = !0, this.releaseSubMeshes(); for (const a of s) rA.AddToMesh(a.materialIndex, a.indexStart, a.indexCount, a.indexStart, a.indexCount, this); return this.synchronizeInstances(), this; } /** * Modify the mesh to get a flat shading rendering. * This means each mesh facet will then have its own normals. Usually new vertices are added in the mesh geometry to get this result. * Warning : the mesh is really modified even if not set originally as updatable and, under the hood, a new VertexBuffer is allocated. * @returns current mesh */ convertToFlatShadedMesh() { return this._convertToUnIndexedMesh(!0); } /** * This method removes all the mesh indices and add new vertices (duplication) in order to unfold facets into buffers. * In other words, more vertices, no more indices and a single bigger VBO. * The mesh is really modified even if not set originally as updatable. Under the hood, a new VertexBuffer is allocated. * @returns current mesh */ convertToUnIndexedMesh() { return this._convertToUnIndexedMesh(); } /** * Inverses facet orientations. * Warning : the mesh is really modified even if not set originally as updatable. A new VertexBuffer is created under the hood each call. * @param flipNormals will also inverts the normals * @returns current mesh */ flipFaces(e = !1) { const t = Ut.ExtractFromMesh(this); let r; if (e && this.isVerticesDataPresent(J.NormalKind) && t.normals) for (r = 0; r < t.normals.length; r++) t.normals[r] *= -1; if (t.indices) { let n; for (r = 0; r < t.indices.length; r += 3) n = t.indices[r + 1], t.indices[r + 1] = t.indices[r + 2], t.indices[r + 2] = n; } return t.applyToMesh(this, this.isVertexBufferUpdatable(J.PositionKind)), this; } /** * Increase the number of facets and hence vertices in a mesh * Vertex normals are interpolated from existing vertex normals * Warning : the mesh is really modified even if not set originally as updatable. A new VertexBuffer is created under the hood each call. * @param numberPerEdge the number of new vertices to add to each edge of a facet, optional default 1 */ increaseVertices(e = 1) { const t = Ut.ExtractFromMesh(this), r = t.indices && !Array.isArray(t.indices) && Array.from ? Array.from(t.indices) : t.indices, n = t.positions && !Array.isArray(t.positions) && Array.from ? Array.from(t.positions) : t.positions, i = t.uvs && !Array.isArray(t.uvs) && Array.from ? Array.from(t.uvs) : t.uvs, s = t.normals && !Array.isArray(t.normals) && Array.from ? Array.from(t.normals) : t.normals; if (!r || !n) Se.Warn("Couldn't increase number of vertices : VertexData must contain at least indices and positions"); else { t.indices = r, t.positions = n, i && (t.uvs = i), s && (t.normals = s); const a = e + 1, f = new Array(); for (let j = 0; j < a + 1; j++) f[j] = new Array(); let o, d; const v = new S(0, 0, 0), u = new S(0, 0, 0), l = new at(0, 0), P = new Array(), p = new Array(), c = new Array(); let H, T = n.length, q; i && (q = i.length); let b; s && (b = s.length); for (let j = 0; j < r.length; j += 3) { p[0] = r[j], p[1] = r[j + 1], p[2] = r[j + 2]; for (let w = 0; w < 3; w++) if (o = p[w], d = p[(w + 1) % 3], c[o] === void 0 && c[d] === void 0 ? (c[o] = new Array(), c[d] = new Array()) : (c[o] === void 0 && (c[o] = new Array()), c[d] === void 0 && (c[d] = new Array())), c[o][d] === void 0 && c[d][o] === void 0) { c[o][d] = [], v.x = (n[3 * d] - n[3 * o]) / a, v.y = (n[3 * d + 1] - n[3 * o + 1]) / a, v.z = (n[3 * d + 2] - n[3 * o + 2]) / a, s && (u.x = (s[3 * d] - s[3 * o]) / a, u.y = (s[3 * d + 1] - s[3 * o + 1]) / a, u.z = (s[3 * d + 2] - s[3 * o + 2]) / a), i && (l.x = (i[2 * d] - i[2 * o]) / a, l.y = (i[2 * d + 1] - i[2 * o + 1]) / a), c[o][d].push(o); for (let m = 1; m < a; m++) c[o][d].push(n.length / 3), n[T++] = n[3 * o] + m * v.x, n[T++] = n[3 * o + 1] + m * v.y, n[T++] = n[3 * o + 2] + m * v.z, s && (s[b++] = s[3 * o] + m * u.x, s[b++] = s[3 * o + 1] + m * u.y, s[b++] = s[3 * o + 2] + m * u.z), i && (i[q++] = i[2 * o] + m * l.x, i[q++] = i[2 * o + 1] + m * l.y); c[o][d].push(d), c[d][o] = new Array(), H = c[o][d].length; for (let m = 0; m < H; m++) c[d][o][m] = c[o][d][H - 1 - m]; } f[0][0] = r[j], f[1][0] = c[r[j]][r[j + 1]][1], f[1][1] = c[r[j]][r[j + 2]][1]; for (let w = 2; w < a; w++) { f[w][0] = c[r[j]][r[j + 1]][w], f[w][w] = c[r[j]][r[j + 2]][w], v.x = (n[3 * f[w][w]] - n[3 * f[w][0]]) / w, v.y = (n[3 * f[w][w] + 1] - n[3 * f[w][0] + 1]) / w, v.z = (n[3 * f[w][w] + 2] - n[3 * f[w][0] + 2]) / w, s && (u.x = (s[3 * f[w][w]] - s[3 * f[w][0]]) / w, u.y = (s[3 * f[w][w] + 1] - s[3 * f[w][0] + 1]) / w, u.z = (s[3 * f[w][w] + 2] - s[3 * f[w][0] + 2]) / w), i && (l.x = (i[2 * f[w][w]] - i[2 * f[w][0]]) / w, l.y = (i[2 * f[w][w] + 1] - i[2 * f[w][0] + 1]) / w); for (let m = 1; m < w; m++) f[w][m] = n.length / 3, n[T++] = n[3 * f[w][0]] + m * v.x, n[T++] = n[3 * f[w][0] + 1] + m * v.y, n[T++] = n[3 * f[w][0] + 2] + m * v.z, s && (s[b++] = s[3 * f[w][0]] + m * u.x, s[b++] = s[3 * f[w][0] + 1] + m * u.y, s[b++] = s[3 * f[w][0] + 2] + m * u.z), i && (i[q++] = i[2 * f[w][0]] + m * l.x, i[q++] = i[2 * f[w][0] + 1] + m * l.y); } f[a] = c[r[j + 1]][r[j + 2]], P.push(f[0][0], f[1][0], f[1][1]); for (let w = 1; w < a; w++) { let m; for (m = 0; m < w; m++) P.push(f[w][m], f[w + 1][m], f[w + 1][m + 1]), P.push(f[w][m], f[w + 1][m + 1], f[w][m + 1]); P.push(f[w][m], f[w + 1][m], f[w + 1][m + 1]); } } t.indices = P, t.applyToMesh(this, this.isVertexBufferUpdatable(J.PositionKind)); } } /** * Force adjacent facets to share vertices and remove any facets that have all vertices in a line * This will undo any application of covertToFlatShadedMesh * Warning : the mesh is really modified even if not set originally as updatable. A new VertexBuffer is created under the hood each call. */ forceSharedVertices() { const e = Ut.ExtractFromMesh(this), t = e.uvs, r = e.indices, n = e.positions, i = e.colors, s = e.matricesIndices, a = e.matricesWeights, f = e.matricesIndicesExtra, o = e.matricesWeightsExtra; if (r === void 0 || n === void 0 || r === null || n === null) Se.Warn("VertexData contains empty entries"); else { const d = new Array(), v = new Array(), u = new Array(), l = new Array(), P = new Array(), p = new Array(), c = new Array(), H = new Array(); let T = new Array(), q = 0; const b = {}; let j, w; for (let I = 0; I < r.length; I += 3) { w = [r[I], r[I + 1], r[I + 2]], T = []; for (let N = 0; N < 3; N++) { T[N] = ""; for (let k = 0; k < 3; k++) Math.abs(n[3 * w[N] + k]) < 1e-8 && (n[3 * w[N] + k] = 0), T[N] += n[3 * w[N] + k] + "|"; } if (!(T[0] == T[1] || T[0] == T[2] || T[1] == T[2])) for (let N = 0; N < 3; N++) { if (j = b[T[N]], j === void 0) { b[T[N]] = q, j = q++; for (let k = 0; k < 3; k++) d.push(n[3 * w[N] + k]); if (i != null) for (let k = 0; k < 4; k++) l.push(i[4 * w[N] + k]); if (t != null) for (let k = 0; k < 2; k++) u.push(t[2 * w[N] + k]); if (s != null) for (let k = 0; k < 4; k++) P.push(s[4 * w[N] + k]); if (a != null) for (let k = 0; k < 4; k++) p.push(a[4 * w[N] + k]); if (f != null) for (let k = 0; k < 4; k++) c.push(f[4 * w[N] + k]); if (o != null) for (let k = 0; k < 4; k++) H.push(o[4 * w[N] + k]); } v.push(j); } } const m = new Array(); Ut.ComputeNormals(d, v, m), e.positions = d, e.indices = v, e.normals = m, t != null && (e.uvs = u), i != null && (e.colors = l), s != null && (e.matricesIndices = P), a != null && (e.matricesWeights = p), f != null && (e.matricesIndicesExtra = c), a != null && (e.matricesWeightsExtra = H), e.applyToMesh(this, this.isVertexBufferUpdatable(J.PositionKind)); } } // Instances /** * @internal */ // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/naming-convention static _instancedMeshFactory(e, t) { throw qn("InstancedMesh"); } /** * @internal */ // eslint-disable-next-line @typescript-eslint/no-unused-vars static _PhysicsImpostorParser(e, t, r) { throw qn("PhysicsImpostor"); } /** * Creates a new InstancedMesh object from the mesh model. * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/copies/instances * @param name defines the name of the new instance * @returns a new InstancedMesh */ createInstance(e) { return Ee._instancedMeshFactory(e, this); } /** * Synchronises all the mesh instance submeshes to the current mesh submeshes, if any. * After this call, all the mesh instances have the same submeshes than the current mesh. * @returns the current mesh */ synchronizeInstances() { for (let e = 0; e < this.instances.length; e++) this.instances[e]._syncSubMeshes(); return this; } /** * Optimization of the mesh's indices, in case a mesh has duplicated vertices. * The function will only reorder the indices and will not remove unused vertices to avoid problems with submeshes. * This should be used together with the simplification to avoid disappearing triangles. * @param successCallback an optional success callback to be called after the optimization finished. * @returns the current mesh */ optimizeIndices(e) { const t = this.getIndices(), r = this.getVerticesData(J.PositionKind); if (!r || !t) return this; const n = []; for (let s = 0; s < r.length; s = s + 3) n.push(S.FromArray(r, s)); const i = []; return up.SyncAsyncForLoop(n.length, 40, (s) => { const a = n.length - 1 - s, f = n[a]; for (let o = 0; o < a; ++o) { const d = n[o]; if (f.equals(d)) { i[a] = o; break; } } }, () => { for (let a = 0; a < t.length; ++a) t[a] = i[t[a]] || t[a]; const s = this.subMeshes.slice(0); this.setIndices(t), this.subMeshes = s, e && e(this); }), this; } /** * Serialize current mesh * @param serializationObject defines the object which will receive the serialization data */ serialize(e = {}) { e.name = this.name, e.id = this.id, e.uniqueId = this.uniqueId, e.type = this.getClassName(), Zi && Zi.HasTags(this) && (e.tags = Zi.GetTags(this)), e.position = this.position.asArray(), this.rotationQuaternion ? e.rotationQuaternion = this.rotationQuaternion.asArray() : this.rotation && (e.rotation = this.rotation.asArray()), e.scaling = this.scaling.asArray(), this._postMultiplyPivotMatrix ? e.pivotMatrix = this.getPivotMatrix().asArray() : e.localMatrix = this.getPivotMatrix().asArray(), e.isEnabled = this.isEnabled(!1), e.isVisible = this.isVisible, e.infiniteDistance = this.infiniteDistance, e.pickable = this.isPickable, e.receiveShadows = this.receiveShadows, e.billboardMode = this.billboardMode, e.visibility = this.visibility, e.checkCollisions = this.checkCollisions, e.isBlocker = this.isBlocker, e.overrideMaterialSideOrientation = this.overrideMaterialSideOrientation, this.parent && this.parent._serializeAsParent(e), e.isUnIndexed = this.isUnIndexed; const t = this._geometry; if (t && this.subMeshes) { e.geometryUniqueId = t.uniqueId, e.geometryId = t.id, e.subMeshes = []; for (let r = 0; r < this.subMeshes.length; r++) { const n = this.subMeshes[r]; e.subMeshes.push({ materialIndex: n.materialIndex, verticesStart: n.verticesStart, verticesCount: n.verticesCount, indexStart: n.indexStart, indexCount: n.indexCount }); } } if (this.material ? this.material.doNotSerialize || (e.materialUniqueId = this.material.uniqueId, e.materialId = this.material.id) : (this.material = null, e.materialUniqueId = this._scene.defaultMaterial.uniqueId, e.materialId = this._scene.defaultMaterial.id), this.morphTargetManager && (e.morphTargetManagerId = this.morphTargetManager.uniqueId), this.skeleton && (e.skeletonId = this.skeleton.id, e.numBoneInfluencers = this.numBoneInfluencers), this.getScene()._getComponent(Ot.NAME_PHYSICSENGINE)) { const r = this.getPhysicsImpostor(); r && (e.physicsMass = r.getParam("mass"), e.physicsFriction = r.getParam("friction"), e.physicsRestitution = r.getParam("mass"), e.physicsImpostor = r.type); } this.metadata && (e.metadata = this.metadata), e.instances = []; for (let r = 0; r < this.instances.length; r++) { const n = this.instances[r]; if (n.doNotSerialize) continue; const i = { name: n.name, id: n.id, isEnabled: n.isEnabled(!1), isVisible: n.isVisible, isPickable: n.isPickable, checkCollisions: n.checkCollisions, position: n.position.asArray(), scaling: n.scaling.asArray() }; if (n.parent && n.parent._serializeAsParent(i), n.rotationQuaternion ? i.rotationQuaternion = n.rotationQuaternion.asArray() : n.rotation && (i.rotation = n.rotation.asArray()), this.getScene()._getComponent(Ot.NAME_PHYSICSENGINE)) { const s = n.getPhysicsImpostor(); s && (i.physicsMass = s.getParam("mass"), i.physicsFriction = s.getParam("friction"), i.physicsRestitution = s.getParam("mass"), i.physicsImpostor = s.type); } n.metadata && (i.metadata = n.metadata), n.actionManager && (i.actions = n.actionManager.serialize(n.name)), e.instances.push(i), jt.AppendSerializedAnimations(n, i), i.ranges = n.serializeAnimationRanges(); } if (this._thinInstanceDataStorage.instancesCount && this._thinInstanceDataStorage.matrixData && (e.thinInstances = { instancesCount: this._thinInstanceDataStorage.instancesCount, matrixData: Array.from(this._thinInstanceDataStorage.matrixData), matrixBufferSize: this._thinInstanceDataStorage.matrixBufferSize, enablePicking: this.thinInstanceEnablePicking }, this._userThinInstanceBuffersStorage)) { const r = { data: {}, sizes: {}, strides: {} }; for (const n in this._userThinInstanceBuffersStorage.data) r.data[n] = Array.from(this._userThinInstanceBuffersStorage.data[n]), r.sizes[n] = this._userThinInstanceBuffersStorage.sizes[n], r.strides[n] = this._userThinInstanceBuffersStorage.strides[n]; e.thinInstances.userThinInstance = r; } return jt.AppendSerializedAnimations(this, e), e.ranges = this.serializeAnimationRanges(), e.layerMask = this.layerMask, e.alphaIndex = this.alphaIndex, e.hasVertexAlpha = this.hasVertexAlpha, e.overlayAlpha = this.overlayAlpha, e.overlayColor = this.overlayColor.asArray(), e.renderOverlay = this.renderOverlay, e.applyFog = this.applyFog, this.actionManager && (e.actions = this.actionManager.serialize(this.name)), e; } /** @internal */ _syncGeometryWithMorphTargetManager() { if (!this.geometry) return; this._markSubMeshesAsAttributesDirty(); const e = this._internalAbstractMeshDataInfo._morphTargetManager; if (e && e.vertexCount) { if (e.vertexCount !== this.getTotalVertices()) { Se.Error("Mesh is incompatible with morph targets. Targets and mesh must all have the same vertices count."), this.morphTargetManager = null; return; } if (e.isUsingTextureForTargets) return; for (let t = 0; t < e.numInfluencers; t++) { const r = e.getActiveTarget(t), n = r.getPositions(); if (!n) { Se.Error("Invalid morph target. Target must have positions."); return; } this.geometry.setVerticesData(J.PositionKind + t, n, !1, 3); const i = r.getNormals(); i && this.geometry.setVerticesData(J.NormalKind + t, i, !1, 3); const s = r.getTangents(); s && this.geometry.setVerticesData(J.TangentKind + t, s, !1, 3); const a = r.getUVs(); a && this.geometry.setVerticesData(J.UVKind + "_" + t, a, !1, 2); } } else { let t = 0; for (; this.geometry.isVerticesDataPresent(J.PositionKind + t); ) this.geometry.removeVerticesData(J.PositionKind + t), this.geometry.isVerticesDataPresent(J.NormalKind + t) && this.geometry.removeVerticesData(J.NormalKind + t), this.geometry.isVerticesDataPresent(J.TangentKind + t) && this.geometry.removeVerticesData(J.TangentKind + t), this.geometry.isVerticesDataPresent(J.UVKind + t) && this.geometry.removeVerticesData(J.UVKind + "_" + t), t++; } } /** * Returns a new Mesh object parsed from the source provided. * @param parsedMesh is the source * @param scene defines the hosting scene * @param rootUrl is the root URL to prefix the `delayLoadingFile` property with * @returns a new Mesh */ static Parse(e, t, r) { let n; if (e.type && e.type === "LinesMesh" ? n = Ee._LinesMeshParser(e, t) : e.type && e.type === "GroundMesh" ? n = Ee._GroundMeshParser(e, t) : e.type && e.type === "GoldbergMesh" ? n = Ee._GoldbergMeshParser(e, t) : e.type && e.type === "GreasedLineMesh" ? n = Ee._GreasedLineMeshParser(e, t) : e.type && e.type === "TrailMesh" ? n = Ee._TrailMeshParser(e, t) : n = new Ee(e.name, t), n.id = e.id, n._waitingParsedUniqueId = e.uniqueId, Zi && Zi.AddTagsTo(n, e.tags), n.position = S.FromArray(e.position), e.metadata !== void 0 && (n.metadata = e.metadata), e.rotationQuaternion ? n.rotationQuaternion = Ze.FromArray(e.rotationQuaternion) : e.rotation && (n.rotation = S.FromArray(e.rotation)), n.scaling = S.FromArray(e.scaling), e.localMatrix ? n.setPreTransformMatrix(he.FromArray(e.localMatrix)) : e.pivotMatrix && n.setPivotMatrix(he.FromArray(e.pivotMatrix)), n.setEnabled(e.isEnabled), n.isVisible = e.isVisible, n.infiniteDistance = e.infiniteDistance, n.showBoundingBox = e.showBoundingBox, n.showSubMeshesBoundingBox = e.showSubMeshesBoundingBox, e.applyFog !== void 0 && (n.applyFog = e.applyFog), e.pickable !== void 0 && (n.isPickable = e.pickable), e.alphaIndex !== void 0 && (n.alphaIndex = e.alphaIndex), n.receiveShadows = e.receiveShadows, e.billboardMode !== void 0 && (n.billboardMode = e.billboardMode), e.visibility !== void 0 && (n.visibility = e.visibility), n.checkCollisions = e.checkCollisions, e.overrideMaterialSideOrientation !== void 0 && (n.overrideMaterialSideOrientation = e.overrideMaterialSideOrientation), e.isBlocker !== void 0 && (n.isBlocker = e.isBlocker), n._shouldGenerateFlatShading = e.useFlatShading, e.freezeWorldMatrix && (n._waitingData.freezeWorldMatrix = e.freezeWorldMatrix), e.parentId !== void 0 && (n._waitingParentId = e.parentId), e.parentInstanceIndex !== void 0 && (n._waitingParentInstanceIndex = e.parentInstanceIndex), e.actions !== void 0 && (n._waitingData.actions = e.actions), e.overlayAlpha !== void 0 && (n.overlayAlpha = e.overlayAlpha), e.overlayColor !== void 0 && (n.overlayColor = Ne.FromArray(e.overlayColor)), e.renderOverlay !== void 0 && (n.renderOverlay = e.renderOverlay), n.isUnIndexed = !!e.isUnIndexed, n.hasVertexAlpha = e.hasVertexAlpha, e.delayLoadingFile ? (n.delayLoadState = 4, n.delayLoadingFile = r + e.delayLoadingFile, n.buildBoundingInfo(S.FromArray(e.boundingBoxMinimum), S.FromArray(e.boundingBoxMaximum)), e._binaryInfo && (n._binaryInfo = e._binaryInfo), n._delayInfo = [], e.hasUVs && n._delayInfo.push(J.UVKind), e.hasUVs2 && n._delayInfo.push(J.UV2Kind), e.hasUVs3 && n._delayInfo.push(J.UV3Kind), e.hasUVs4 && n._delayInfo.push(J.UV4Kind), e.hasUVs5 && n._delayInfo.push(J.UV5Kind), e.hasUVs6 && n._delayInfo.push(J.UV6Kind), e.hasColors && n._delayInfo.push(J.ColorKind), e.hasMatricesIndices && n._delayInfo.push(J.MatricesIndicesKind), e.hasMatricesWeights && n._delayInfo.push(J.MatricesWeightsKind), n._delayLoadingFunction = Tf._ImportGeometry, l9.ForceFullSceneLoadingForIncremental && n._checkDelayState()) : Tf._ImportGeometry(e, n), e.materialUniqueId ? n._waitingMaterialId = e.materialUniqueId : e.materialId && (n._waitingMaterialId = e.materialId), e.morphTargetManagerId > -1 && (n.morphTargetManager = t.getMorphTargetManagerById(e.morphTargetManagerId)), e.skeletonId !== void 0 && e.skeletonId !== null && (n.skeleton = t.getLastSkeletonById(e.skeletonId), e.numBoneInfluencers && (n.numBoneInfluencers = e.numBoneInfluencers)), e.animations) { for (let i = 0; i < e.animations.length; i++) { const s = e.animations[i], a = Jo("BABYLON.Animation"); a && n.animations.push(a.Parse(s)); } Cs.ParseAnimationRanges(n, e, t); } if (e.autoAnimate && t.beginAnimation(n, e.autoAnimateFrom, e.autoAnimateTo, e.autoAnimateLoop, e.autoAnimateSpeed || 1), e.layerMask && !isNaN(e.layerMask) ? n.layerMask = Math.abs(parseInt(e.layerMask)) : n.layerMask = 268435455, e.physicsImpostor && Ee._PhysicsImpostorParser(t, n, e), e.lodMeshIds && (n._waitingData.lods = { ids: e.lodMeshIds, distances: e.lodDistances ? e.lodDistances : null, coverages: e.lodCoverages ? e.lodCoverages : null }), e.instances) for (let i = 0; i < e.instances.length; i++) { const s = e.instances[i], a = n.createInstance(s.name); if (s.id && (a.id = s.id), Zi && (s.tags ? Zi.AddTagsTo(a, s.tags) : Zi.AddTagsTo(a, e.tags)), a.position = S.FromArray(s.position), s.metadata !== void 0 && (a.metadata = s.metadata), s.parentId !== void 0 && (a._waitingParentId = s.parentId), s.parentInstanceIndex !== void 0 && (a._waitingParentInstanceIndex = s.parentInstanceIndex), s.isEnabled !== void 0 && s.isEnabled !== null && a.setEnabled(s.isEnabled), s.isVisible !== void 0 && s.isVisible !== null && (a.isVisible = s.isVisible), s.isPickable !== void 0 && s.isPickable !== null && (a.isPickable = s.isPickable), s.rotationQuaternion ? a.rotationQuaternion = Ze.FromArray(s.rotationQuaternion) : s.rotation && (a.rotation = S.FromArray(s.rotation)), a.scaling = S.FromArray(s.scaling), s.checkCollisions != null && s.checkCollisions != null && (a.checkCollisions = s.checkCollisions), s.pickable != null && s.pickable != null && (a.isPickable = s.pickable), s.showBoundingBox != null && s.showBoundingBox != null && (a.showBoundingBox = s.showBoundingBox), s.showSubMeshesBoundingBox != null && s.showSubMeshesBoundingBox != null && (a.showSubMeshesBoundingBox = s.showSubMeshesBoundingBox), s.alphaIndex != null && s.showSubMeshesBoundingBox != null && (a.alphaIndex = s.alphaIndex), s.physicsImpostor && Ee._PhysicsImpostorParser(t, a, s), s.actions !== void 0 && (a._waitingData.actions = s.actions), s.animations) { for (let f = 0; f < s.animations.length; f++) { const o = s.animations[f], d = Jo("BABYLON.Animation"); d && a.animations.push(d.Parse(o)); } Cs.ParseAnimationRanges(a, s, t), s.autoAnimate && t.beginAnimation(a, s.autoAnimateFrom, s.autoAnimateTo, s.autoAnimateLoop, s.autoAnimateSpeed || 1); } } if (e.thinInstances) { const i = e.thinInstances; if (n.thinInstanceEnablePicking = !!i.enablePicking, i.matrixData ? (n.thinInstanceSetBuffer("matrix", new Float32Array(i.matrixData), 16, !1), n._thinInstanceDataStorage.matrixBufferSize = i.matrixBufferSize, n._thinInstanceDataStorage.instancesCount = i.instancesCount) : n._thinInstanceDataStorage.matrixBufferSize = i.matrixBufferSize, e.thinInstances.userThinInstance) { const s = e.thinInstances.userThinInstance; for (const a in s.data) n.thinInstanceSetBuffer(a, new Float32Array(s.data[a]), s.strides[a], !1), n._userThinInstanceBuffersStorage.sizes[a] = s.sizes[a]; } } return n; } // Skeletons /** * Prepare internal position array for software CPU skinning * @returns original positions used for CPU skinning. Useful for integrating Morphing with skeletons in same mesh */ setPositionsForCPUSkinning() { const e = this._internalMeshDataInfo; if (!e._sourcePositions) { const t = this.getVerticesData(J.PositionKind); if (!t) return e._sourcePositions; e._sourcePositions = new Float32Array(t), this.isVertexBufferUpdatable(J.PositionKind) || this.setVerticesData(J.PositionKind, t, !0); } return e._sourcePositions; } /** * Prepare internal normal array for software CPU skinning * @returns original normals used for CPU skinning. Useful for integrating Morphing with skeletons in same mesh. */ setNormalsForCPUSkinning() { const e = this._internalMeshDataInfo; if (!e._sourceNormals) { const t = this.getVerticesData(J.NormalKind); if (!t) return e._sourceNormals; e._sourceNormals = new Float32Array(t), this.isVertexBufferUpdatable(J.NormalKind) || this.setVerticesData(J.NormalKind, t, !0); } return e._sourceNormals; } /** * Updates the vertex buffer by applying transformation from the bones * @param skeleton defines the skeleton to apply to current mesh * @returns the current mesh */ applySkeleton(e) { if (!this.geometry) return this; if (this.geometry._softwareSkinningFrameId == this.getScene().getFrameId()) return this; if (this.geometry._softwareSkinningFrameId = this.getScene().getFrameId(), !this.isVerticesDataPresent(J.PositionKind)) return this; if (!this.isVerticesDataPresent(J.MatricesIndicesKind)) return this; if (!this.isVerticesDataPresent(J.MatricesWeightsKind)) return this; const t = this.isVerticesDataPresent(J.NormalKind), r = this._internalMeshDataInfo; if (!r._sourcePositions) { const H = this.subMeshes.slice(); this.setPositionsForCPUSkinning(), this.subMeshes = H; } t && !r._sourceNormals && this.setNormalsForCPUSkinning(); let n = this.getVerticesData(J.PositionKind); if (!n) return this; n instanceof Float32Array || (n = new Float32Array(n)); let i = this.getVerticesData(J.NormalKind); if (t) { if (!i) return this; i instanceof Float32Array || (i = new Float32Array(i)); } const s = this.getVerticesData(J.MatricesIndicesKind), a = this.getVerticesData(J.MatricesWeightsKind); if (!a || !s) return this; const f = this.numBoneInfluencers > 4, o = f ? this.getVerticesData(J.MatricesIndicesExtraKind) : null, d = f ? this.getVerticesData(J.MatricesWeightsExtraKind) : null, v = e.getTransformMatrices(this), u = S.Zero(), l = new he(), P = new he(); let p = 0, c; for (let H = 0; H < n.length; H += 3, p += 4) { let T; for (c = 0; c < 4; c++) T = a[p + c], T > 0 && (he.FromFloat32ArrayToRefScaled(v, Math.floor(s[p + c] * 16), T, P), l.addToSelf(P)); if (f) for (c = 0; c < 4; c++) T = d[p + c], T > 0 && (he.FromFloat32ArrayToRefScaled(v, Math.floor(o[p + c] * 16), T, P), l.addToSelf(P)); S.TransformCoordinatesFromFloatsToRef(r._sourcePositions[H], r._sourcePositions[H + 1], r._sourcePositions[H + 2], l, u), u.toArray(n, H), t && (S.TransformNormalFromFloatsToRef(r._sourceNormals[H], r._sourceNormals[H + 1], r._sourceNormals[H + 2], l, u), u.toArray(i, H)), l.reset(); } return this.updateVerticesData(J.PositionKind, n), t && this.updateVerticesData(J.NormalKind, i), this; } // Tools /** * Returns an object containing a min and max Vector3 which are the minimum and maximum vectors of each mesh bounding box from the passed array, in the world coordinates * @param meshes defines the list of meshes to scan * @returns an object `{min:` Vector3`, max:` Vector3`}` */ static MinMax(e) { let t = null, r = null; return e.forEach(function(n) { const s = n.getBoundingInfo().boundingBox; !t || !r ? (t = s.minimumWorld, r = s.maximumWorld) : (t.minimizeInPlace(s.minimumWorld), r.maximizeInPlace(s.maximumWorld)); }), !t || !r ? { min: S.Zero(), max: S.Zero() } : { min: t, max: r }; } /** * Returns the center of the `{min:` Vector3`, max:` Vector3`}` or the center of MinMax vector3 computed from a mesh array * @param meshesOrMinMaxVector could be an array of meshes or a `{min:` Vector3`, max:` Vector3`}` object * @returns a vector3 */ static Center(e) { const t = e instanceof Array ? Ee.MinMax(e) : e; return S.Center(t.min, t.max); } /** * Merge the array of meshes into a single mesh for performance reasons. * @param meshes array of meshes with the vertices to merge. Entries cannot be empty meshes. * @param disposeSource when true (default), dispose of the vertices from the source meshes. * @param allow32BitsIndices when the sum of the vertices > 64k, this must be set to true. * @param meshSubclass (optional) can be set to a Mesh where the merged vertices will be inserted. * @param subdivideWithSubMeshes when true (false default), subdivide mesh into subMeshes. * @param multiMultiMaterials when true (false default), subdivide mesh into subMeshes with multiple materials, ignores subdivideWithSubMeshes. * @returns a new mesh */ static MergeMeshes(e, t = !0, r, n, i, s) { return mO(Ee._MergeMeshesCoroutine(e, t, r, n, i, s, !1)); } /** * Merge the array of meshes into a single mesh for performance reasons. * @param meshes array of meshes with the vertices to merge. Entries cannot be empty meshes. * @param disposeSource when true (default), dispose of the vertices from the source meshes. * @param allow32BitsIndices when the sum of the vertices > 64k, this must be set to true. * @param meshSubclass (optional) can be set to a Mesh where the merged vertices will be inserted. * @param subdivideWithSubMeshes when true (false default), subdivide mesh into subMeshes. * @param multiMultiMaterials when true (false default), subdivide mesh into subMeshes with multiple materials, ignores subdivideWithSubMeshes. * @returns a new mesh */ static MergeMeshesAsync(e, t = !0, r, n, i, s) { return BO(Ee._MergeMeshesCoroutine(e, t, r, n, i, s, !0), tee()); } static *_MergeMeshesCoroutine(e, t = !0, r, n, i, s, a) { if (e = e.filter(Boolean), e.length === 0) return null; let f; if (!r) { let m = 0; for (f = 0; f < e.length; f++) if (m += e[f].getTotalVertices(), m >= 65536) return Se.Warn("Cannot merge meshes because resulting mesh will have more than 65536 vertices. Please use allow32BitsIndices = true to use 32 bits indices"), null; } s && (i = !1); const o = new Array(), d = new Array(), v = new Array(), u = e[0].overrideMaterialSideOrientation; for (f = 0; f < e.length; f++) { const m = e[f]; if (m.isAnInstance) return Se.Warn("Cannot merge instance meshes."), null; if (u !== m.overrideMaterialSideOrientation) return Se.Warn("Cannot merge meshes with different overrideMaterialSideOrientation values."), null; if (i && v.push(m.getTotalIndices()), s) if (m.material) { const I = m.material; if (I instanceof Dc) { for (let N = 0; N < I.subMaterials.length; N++) o.indexOf(I.subMaterials[N]) < 0 && o.push(I.subMaterials[N]); for (let N = 0; N < m.subMeshes.length; N++) d.push(o.indexOf(I.subMaterials[m.subMeshes[N].materialIndex])), v.push(m.subMeshes[N].indexCount); } else { o.indexOf(I) < 0 && o.push(I); for (let N = 0; N < m.subMeshes.length; N++) d.push(o.indexOf(I)), v.push(m.subMeshes[N].indexCount); } } else for (let I = 0; I < m.subMeshes.length; I++) d.push(0), v.push(m.subMeshes[I].indexCount); } const l = e[0], P = (m) => { const I = m.computeWorldMatrix(!0); return { vertexData: Ut.ExtractFromMesh(m, !1, !1), transform: I }; }, { vertexData: p, transform: c } = P(l); a && (yield); const H = new Array(e.length - 1); for (let m = 1; m < e.length; m++) H[m - 1] = P(e[m]), a && (yield); const T = p._mergeCoroutine(c, H, r, a, !t); let q = T.next(); for (; !q.done; ) a && (yield), q = T.next(); const b = q.value; n || (n = new Ee(l.name + "_merged", l.getScene())); const j = b._applyToCoroutine(n, void 0, a); let w = j.next(); for (; !w.done; ) a && (yield), w = j.next(); if (n.checkCollisions = l.checkCollisions, n.overrideMaterialSideOrientation = l.overrideMaterialSideOrientation, t) for (f = 0; f < e.length; f++) e[f].dispose(); if (i || s) { n.releaseSubMeshes(), f = 0; let m = 0; for (; f < v.length; ) rA.CreateFromIndices(0, m, v[f], n, void 0, !1), m += v[f], f++; for (const I of n.subMeshes) I.refreshBoundingInfo(); n.computeWorldMatrix(!0); } if (s) { const m = new Dc(l.name + "_merged", l.getScene()); m.subMaterials = o; for (let I = 0; I < n.subMeshes.length; I++) n.subMeshes[I].materialIndex = d[I]; n.material = m; } else n.material = l.material; return n; } /** * @internal */ addInstance(e) { e._indexInSourceMeshInstanceArray = this.instances.length, this.instances.push(e); } /** * @internal */ removeInstance(e) { const t = e._indexInSourceMeshInstanceArray; if (t != -1) { if (t !== this.instances.length - 1) { const r = this.instances[this.instances.length - 1]; this.instances[t] = r, r._indexInSourceMeshInstanceArray = t; } e._indexInSourceMeshInstanceArray = -1, this.instances.pop(); } } /** @internal */ _shouldConvertRHS() { return this.overrideMaterialSideOrientation === gt.CounterClockWiseSideOrientation; } /** @internal */ _getRenderingFillMode(e) { var t; const r = this.getScene(); return r.forcePointsCloud ? gt.PointFillMode : r.forceWireframe ? gt.WireFrameFillMode : (t = this.overrideRenderingFillMode) !== null && t !== void 0 ? t : e; } // deprecated methods /** * Sets the mesh material by the material or multiMaterial `id` property * @param id is a string identifying the material or the multiMaterial * @returns the current mesh * @deprecated Please use MeshBuilder instead Please use setMaterialById instead */ setMaterialByID(e) { return this.setMaterialById(e); } /** * Creates a ribbon mesh. * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param * @param name defines the name of the mesh to create * @param pathArray is a required array of paths, what are each an array of successive Vector3. The pathArray parameter depicts the ribbon geometry. * @param closeArray creates a seam between the first and the last paths of the path array (default is false) * @param closePath creates a seam between the first and the last points of each path of the path array * @param offset is taken in account only if the `pathArray` is containing a single path * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation) * @param instance defines an instance of an existing Ribbon object to be updated with the passed `pathArray` parameter (https://doc.babylonjs.com/how_to/How_to_dynamically_morph_a_mesh#ribbon) * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateRibbon(e, t, r, n, i, s, a, f, o) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a plane polygonal mesh. By default, this is a disc. * @param name defines the name of the mesh to create * @param radius sets the radius size (float) of the polygon (default 0.5) * @param tessellation sets the number of polygon sides (positive integer, default 64). So a tessellation valued to 3 will build a triangle, to 4 a square, etc * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation) * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateDisc(e, t, r, n, i, s) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a box mesh. * @param name defines the name of the mesh to create * @param size sets the size (float) of each box side (default 1) * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation) * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateBox(e, t, r, n, i) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a sphere mesh. * @param name defines the name of the mesh to create * @param segments sets the sphere number of horizontal stripes (positive integer, default 32) * @param diameter sets the diameter size (float) of the sphere (default 1) * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation) * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateSphere(e, t, r, n, i, s) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a hemisphere mesh. * @param name defines the name of the mesh to create * @param segments sets the sphere number of horizontal stripes (positive integer, default 32) * @param diameter sets the diameter size (float) of the sphere (default 1) * @param scene defines the hosting scene * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateHemisphere(e, t, r, n) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a cylinder or a cone mesh. * @param name defines the name of the mesh to create * @param height sets the height size (float) of the cylinder/cone (float, default 2) * @param diameterTop set the top cap diameter (floats, default 1) * @param diameterBottom set the bottom cap diameter (floats, default 1). This value can't be zero * @param tessellation sets the number of cylinder sides (positive integer, default 24). Set it to 3 to get a prism for instance * @param subdivisions sets the number of rings along the cylinder height (positive integer, default 1) * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation) * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateCylinder(e, t, r, n, i, s, a, f, o) { throw new Error("Import MeshBuilder to populate this function"); } // Torus (Code from SharpDX.org) /** * Creates a torus mesh. * @param name defines the name of the mesh to create * @param diameter sets the diameter size (float) of the torus (default 1) * @param thickness sets the diameter size of the tube of the torus (float, default 0.5) * @param tessellation sets the number of torus sides (positive integer, default 16) * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation) * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateTorus(e, t, r, n, i, s, a) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a torus knot mesh. * @param name defines the name of the mesh to create * @param radius sets the global radius size (float) of the torus knot (default 2) * @param tube sets the diameter size of the tube of the torus (float, default 0.5) * @param radialSegments sets the number of sides on each tube segments (positive integer, default 32) * @param tubularSegments sets the number of tubes to decompose the knot into (positive integer, default 32) * @param p the number of windings on X axis (positive integers, default 2) * @param q the number of windings on Y axis (positive integers, default 3) * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation) * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateTorusKnot(e, t, r, n, i, s, a, f, o, d) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a line mesh.. * @param name defines the name of the mesh to create * @param points is an array successive Vector3 * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @param instance is an instance of an existing LineMesh object to be updated with the passed `points` parameter (https://doc.babylonjs.com/how_to/How_to_dynamically_morph_a_mesh#lines-and-dashedlines). * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateLines(e, t, r, n, i) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a dashed line mesh. * @param name defines the name of the mesh to create * @param points is an array successive Vector3 * @param dashSize is the size of the dashes relatively the dash number (positive float, default 3) * @param gapSize is the size of the gap between two successive dashes relatively the dash number (positive float, default 1) * @param dashNb is the intended total number of dashes (positive integer, default 200) * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @param instance is an instance of an existing LineMesh object to be updated with the passed `points` parameter (https://doc.babylonjs.com/how_to/How_to_dynamically_morph_a_mesh#lines-and-dashedlines) * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateDashedLines(e, t, r, n, i, s, a, f) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a polygon mesh.Please consider using the same method from the MeshBuilder class instead * The polygon's shape will depend on the input parameters and is constructed parallel to a ground mesh. * The parameter `shape` is a required array of successive Vector3 representing the corners of the polygon in th XoZ plane, that is y = 0 for all vectors. * You can set the mesh side orientation with the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created. * Remember you can only change the shape positions, not their number when updating a polygon. * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#non-regular-polygon * @param name defines the name of the mesh to create * @param shape is a required array of successive Vector3 representing the corners of the polygon in th XoZ plane, that is y = 0 for all vectors * @param scene defines the hosting scene * @param holes is a required array of arrays of successive Vector3 used to defines holes in the polygon * @param updatable defines if the mesh must be flagged as updatable * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation) * @param earcutInjection can be used to inject your own earcut reference * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreatePolygon(e, t, r, n, i, s, a) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates an extruded polygon mesh, with depth in the Y direction.. * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#extruded-non-regular-polygon * @param name defines the name of the mesh to create * @param shape is a required array of successive Vector3 representing the corners of the polygon in th XoZ plane, that is y = 0 for all vectors * @param depth defines the height of extrusion * @param scene defines the hosting scene * @param holes is a required array of arrays of successive Vector3 used to defines holes in the polygon * @param updatable defines if the mesh must be flagged as updatable * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation) * @param earcutInjection can be used to inject your own earcut reference * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static ExtrudePolygon(e, t, r, n, i, s, a, f) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates an extruded shape mesh. * The extrusion is a parametric shape. It has no predefined shape. Its final shape will depend on the input parameters. * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#extruded-shapes * @param name defines the name of the mesh to create * @param shape is a required array of successive Vector3. This array depicts the shape to be extruded in its local space : the shape must be designed in the xOy plane and will be extruded along the Z axis * @param path is a required array of successive Vector3. This is the axis curve the shape is extruded along * @param scale is the value to scale the shape * @param rotation is the angle value to rotate the shape each step (each path point), from the former step (so rotation added each step) along the curve * @param cap sets the way the extruded shape is capped. Possible values : Mesh.NO_CAP (default), Mesh.CAP_START, Mesh.CAP_END, Mesh.CAP_ALL * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation) * @param instance is an instance of an existing ExtrudedShape object to be updated with the passed `shape`, `path`, `scale` or `rotation` parameters (https://doc.babylonjs.com/how_to/How_to_dynamically_morph_a_mesh#extruded-shape) * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static ExtrudeShape(e, t, r, n, i, s, a, f, o, d) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates an custom extruded shape mesh. * The custom extrusion is a parametric shape. * It has no predefined shape. Its final shape will depend on the input parameters. * * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#extruded-shapes * @param name defines the name of the mesh to create * @param shape is a required array of successive Vector3. This array depicts the shape to be extruded in its local space : the shape must be designed in the xOy plane and will be extruded along the Z axis * @param path is a required array of successive Vector3. This is the axis curve the shape is extruded along * @param scaleFunction is a custom Javascript function called on each path point * @param rotationFunction is a custom Javascript function called on each path point * @param ribbonCloseArray forces the extrusion underlying ribbon to close all the paths in its `pathArray` * @param ribbonClosePath forces the extrusion underlying ribbon to close its `pathArray` * @param cap sets the way the extruded shape is capped. Possible values : Mesh.NO_CAP (default), Mesh.CAP_START, Mesh.CAP_END, Mesh.CAP_ALL * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation) * @param instance is an instance of an existing ExtrudedShape object to be updated with the passed `shape`, `path`, `scale` or `rotation` parameters (https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#extruded-shape) * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static ExtrudeShapeCustom(e, t, r, n, i, s, a, f, o, d, v, u) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates lathe mesh. * The lathe is a shape with a symmetry axis : a 2D model shape is rotated around this axis to design the lathe. * @param name defines the name of the mesh to create * @param shape is a required array of successive Vector3. This array depicts the shape to be rotated in its local space : the shape must be designed in the xOy plane and will be rotated around the Y axis. It's usually a 2D shape, so the Vector3 z coordinates are often set to zero * @param radius is the radius value of the lathe * @param tessellation is the side number of the lathe. * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation) * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateLathe(e, t, r, n, i, s, a) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a plane mesh. * @param name defines the name of the mesh to create * @param size sets the size (float) of both sides of the plane at once (default 1) * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation) * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreatePlane(e, t, r, n, i) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a ground mesh. * @param name defines the name of the mesh to create * @param width set the width of the ground * @param height set the height of the ground * @param subdivisions sets the number of subdivisions per side * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateGround(e, t, r, n, i, s) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a tiled ground mesh. * @param name defines the name of the mesh to create * @param xmin set the ground minimum X coordinate * @param zmin set the ground minimum Y coordinate * @param xmax set the ground maximum X coordinate * @param zmax set the ground maximum Z coordinate * @param subdivisions is an object `{w: positive integer, h: positive integer}` (default `{w: 6, h: 6}`). `w` and `h` are the numbers of subdivisions on the ground width and height. Each subdivision is called a tile * @param precision is an object `{w: positive integer, h: positive integer}` (default `{w: 2, h: 2}`). `w` and `h` are the numbers of subdivisions on the ground width and height of each tile * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateTiledGround(e, t, r, n, i, s, a, f, o) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a ground mesh from a height map. * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/height_map * @param name defines the name of the mesh to create * @param url sets the URL of the height map image resource * @param width set the ground width size * @param height set the ground height size * @param subdivisions sets the number of subdivision per side * @param minHeight is the minimum altitude on the ground * @param maxHeight is the maximum altitude on the ground * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @param onReady is a callback function that will be called once the mesh is built (the height map download can last some time) * @param alphaFilter will filter any data where the alpha channel is below this value, defaults 0 (all data visible) * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateGroundFromHeightMap(e, t, r, n, i, s, a, f, o, d, v) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a tube mesh. * The tube is a parametric shape. * It has no predefined shape. Its final shape will depend on the input parameters. * * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param * @param name defines the name of the mesh to create * @param path is a required array of successive Vector3. It is the curve used as the axis of the tube * @param radius sets the tube radius size * @param tessellation is the number of sides on the tubular surface * @param radiusFunction is a custom function. If it is not null, it overrides the parameter `radius`. This function is called on each point of the tube path and is passed the index `i` of the i-th point and the distance of this point from the first point of the path * @param cap sets the way the extruded shape is capped. Possible values : Mesh.NO_CAP (default), Mesh.CAP_START, Mesh.CAP_END, Mesh.CAP_ALL * @param scene defines the hosting scene * @param updatable defines if the mesh must be flagged as updatable * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation) * @param instance is an instance of an existing Tube object to be updated with the passed `pathArray` parameter (https://doc.babylonjs.com/how_to/How_to_dynamically_morph_a_mesh#tube) * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateTube(e, t, r, n, i, s, a, f, o, d) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a polyhedron mesh. *. * * The parameter `type` (positive integer, max 14, default 0) sets the polyhedron type to build among the 15 embedded types. Please refer to the type sheet in the tutorial to choose the wanted type * * The parameter `size` (positive float, default 1) sets the polygon size * * You can overwrite the `size` on each dimension bu using the parameters `sizeX`, `sizeY` or `sizeZ` (positive floats, default to `size` value) * * You can build other polyhedron types than the 15 embbeded ones by setting the parameter `custom` (`polyhedronObject`, default null). If you set the parameter `custom`, this overwrittes the parameter `type` * * A `polyhedronObject` is a formatted javascript object. You'll find a full file with pre-set polyhedra here : https://github.com/BabylonJS/Extensions/tree/master/Polyhedron * * You can set the color and the UV of each side of the polyhedron with the parameters `faceColors` (Color4, default `(1, 1, 1, 1)`) and faceUV (Vector4, default `(0, 0, 1, 1)`) * * To understand how to set `faceUV` or `faceColors`, please read this by considering the right number of faces of your polyhedron, instead of only 6 for the box : https://doc.babylonjs.com/features/featuresDeepDive/materials/using/texturePerBoxFace * * The parameter `flat` (boolean, default true). If set to false, it gives the polyhedron a single global face, so less vertices and shared normals. In this case, `faceColors` and `faceUV` are ignored * * You can also set the mesh side orientation with the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created * @param name defines the name of the mesh to create * @param options defines the options used to create the mesh * @param scene defines the hosting scene * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreatePolyhedron(e, t, r) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a sphere based upon an icosahedron with 20 triangular faces which can be subdivided * * The parameter `radius` sets the radius size (float) of the icosphere (default 1) * * You can set some different icosphere dimensions, for instance to build an ellipsoid, by using the parameters `radiusX`, `radiusY` and `radiusZ` (all by default have the same value than `radius`) * * The parameter `subdivisions` sets the number of subdivisions (positive integer, default 4). The more subdivisions, the more faces on the icosphere whatever its size * * The parameter `flat` (boolean, default true) gives each side its own normals. Set it to false to get a smooth continuous light reflection on the surface * * You can also set the mesh side orientation with the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/polyhedra#icosphere * @param name defines the name of the mesh * @param options defines the options used to create the mesh * @param scene defines the hosting scene * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateIcoSphere(e, t, r) { throw new Error("Import MeshBuilder to populate this function"); } /** * Creates a decal mesh. *. * A decal is a mesh usually applied as a model onto the surface of another mesh * @param name defines the name of the mesh * @param sourceMesh defines the mesh receiving the decal * @param position sets the position of the decal in world coordinates * @param normal sets the normal of the mesh where the decal is applied onto in world coordinates * @param size sets the decal scaling * @param angle sets the angle to rotate the decal * @returns a new Mesh * @deprecated Please use MeshBuilder instead */ static CreateDecal(e, t, r, n, i, s) { throw new Error("Import MeshBuilder to populate this function"); } /** Creates a Capsule Mesh * @param name defines the name of the mesh. * @param options the constructors options used to shape the mesh. * @param scene defines the scene the mesh is scoped to. * @returns the capsule mesh * @see https://doc.babylonjs.com/how_to/capsule_shape * @deprecated Please use MeshBuilder instead */ static CreateCapsule(e, t, r) { throw new Error("Import MeshBuilder to populate this function"); } /** * Extends a mesh to a Goldberg mesh * Warning the mesh to convert MUST be an import of a perviously exported Goldberg mesh * @param mesh the mesh to convert * @returns the extended mesh * @deprecated Please use ExtendMeshToGoldberg instead */ static ExtendToGoldberg(e) { throw new Error("Import MeshBuilder to populate this function"); } } Ee.FRONTSIDE = Ut.FRONTSIDE; Ee.BACKSIDE = Ut.BACKSIDE; Ee.DOUBLESIDE = Ut.DOUBLESIDE; Ee.DEFAULTSIDE = Ut.DEFAULTSIDE; Ee.NO_CAP = 0; Ee.CAP_START = 1; Ee.CAP_END = 2; Ee.CAP_ALL = 3; Ee.NO_FLIP = 0; Ee.FLIP_TILE = 1; Ee.ROTATE_TILE = 2; Ee.FLIP_ROW = 3; Ee.ROTATE_ROW = 4; Ee.FLIP_N_ROTATE_TILE = 5; Ee.FLIP_N_ROTATE_ROW = 6; Ee.CENTER = 0; Ee.LEFT = 1; Ee.RIGHT = 2; Ee.TOP = 3; Ee.BOTTOM = 4; Ee.INSTANCEDMESH_SORT_TRANSPARENT = !1; Ee._GroundMeshParser = (A, e) => { throw qn("GroundMesh"); }; Ee._GoldbergMeshParser = (A, e) => { throw qn("GoldbergMesh"); }; Ee._LinesMeshParser = (A, e) => { throw qn("LinesMesh"); }; Ee._GreasedLineMeshParser = (A, e) => { throw qn("GreasedLineMesh"); }; Ee._GreasedLineRibbonMeshParser = (A, e) => { throw qn("GreasedLineRibbonMesh"); }; Ee._TrailMeshParser = (A, e) => { throw qn("TrailMesh"); }; Ue("BABYLON.Mesh", Ee); Ee._instancedMeshFactory = (A, e) => { const t = new bp(A, e); if (e.instancedBuffers) { t.instancedBuffers = {}; for (const r in e.instancedBuffers) t.instancedBuffers[r] = e.instancedBuffers[r]; } return t; }; class bp extends jn { /** * Creates a new InstancedMesh object from the mesh source. * @param name defines the name of the instance * @param source the mesh to create the instance from */ constructor(e, t) { super(e, t.getScene()), this._indexInSourceMeshInstanceArray = -1, this._distanceToCamera = 0, t.addInstance(this), this._sourceMesh = t, this._unIndexed = t._unIndexed, this.position.copyFrom(t.position), this.rotation.copyFrom(t.rotation), this.scaling.copyFrom(t.scaling), t.rotationQuaternion && (this.rotationQuaternion = t.rotationQuaternion.clone()), this.animations = t.animations.slice(); for (const r of t.getAnimationRanges()) r != null && this.createAnimationRange(r.name, r.from, r.to); this.infiniteDistance = t.infiniteDistance, this.setPivotMatrix(t.getPivotMatrix()), this.refreshBoundingInfo(!0, !0), this._syncSubMeshes(); } /** * Returns the string "InstancedMesh". */ getClassName() { return "InstancedMesh"; } /** Gets the list of lights affecting that mesh */ get lightSources() { return this._sourceMesh._lightSources; } _resyncLightSources() { } _resyncLightSource() { } _removeLightSource() { } // Methods /** * If the source mesh receives shadows */ get receiveShadows() { return this._sourceMesh.receiveShadows; } set receiveShadows(e) { var t; ((t = this._sourceMesh) === null || t === void 0 ? void 0 : t.receiveShadows) !== e && ye.Warn("Setting receiveShadows on an instanced mesh has no effect"); } /** * The material of the source mesh */ get material() { return this._sourceMesh.material; } set material(e) { var t; ((t = this._sourceMesh) === null || t === void 0 ? void 0 : t.material) !== e && ye.Warn("Setting material on an instanced mesh has no effect"); } /** * Visibility of the source mesh */ get visibility() { return this._sourceMesh.visibility; } set visibility(e) { var t; ((t = this._sourceMesh) === null || t === void 0 ? void 0 : t.visibility) !== e && ye.Warn("Setting visibility on an instanced mesh has no effect"); } /** * Skeleton of the source mesh */ get skeleton() { return this._sourceMesh.skeleton; } set skeleton(e) { var t; ((t = this._sourceMesh) === null || t === void 0 ? void 0 : t.skeleton) !== e && ye.Warn("Setting skeleton on an instanced mesh has no effect"); } /** * Rendering ground id of the source mesh */ get renderingGroupId() { return this._sourceMesh.renderingGroupId; } set renderingGroupId(e) { !this._sourceMesh || e === this._sourceMesh.renderingGroupId || Se.Warn("Note - setting renderingGroupId of an instanced mesh has no effect on the scene"); } /** * Returns the total number of vertices (integer). */ getTotalVertices() { return this._sourceMesh ? this._sourceMesh.getTotalVertices() : 0; } /** * Returns a positive integer : the total number of indices in this mesh geometry. * @returns the number of indices or zero if the mesh has no geometry. */ getTotalIndices() { return this._sourceMesh.getTotalIndices(); } /** * The source mesh of the instance */ get sourceMesh() { return this._sourceMesh; } /** * Creates a new InstancedMesh object from the mesh model. * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/copies/instances * @param name defines the name of the new instance * @returns a new InstancedMesh */ createInstance(e) { return this._sourceMesh.createInstance(e); } /** * Is this node ready to be used/rendered * @param completeCheck defines if a complete check (including materials and lights) has to be done (false by default) * @returns {boolean} is it ready */ isReady(e = !1) { return this._sourceMesh.isReady(e, !0); } /** * Returns an array of integers or a typed array (Int32Array, Uint32Array, Uint16Array) populated with the mesh indices. * @param kind kind of verticies to retrieve (eg. positions, normals, uvs, etc.) * @param copyWhenShared If true (default false) and and if the mesh geometry is shared among some other meshes, the returned array is a copy of the internal one. * @param forceCopy defines a boolean forcing the copy of the buffer no matter what the value of copyWhenShared is * @returns a float array or a Float32Array of the requested kind of data : positions, normals, uvs, etc. */ getVerticesData(e, t, r) { return this._sourceMesh.getVerticesData(e, t, r); } /** * Sets the vertex data of the mesh geometry for the requested `kind`. * If the mesh has no geometry, a new Geometry object is set to the mesh and then passed this vertex data. * The `data` are either a numeric array either a Float32Array. * The parameter `updatable` is passed as is to the underlying Geometry object constructor (if initially none) or updater. * The parameter `stride` is an optional positive integer, it is usually automatically deducted from the `kind` (3 for positions or normals, 2 for UV, etc). * Note that a new underlying VertexBuffer object is created each call. * If the `kind` is the `PositionKind`, the mesh BoundingInfo is renewed, so the bounding box and sphere, and the mesh World Matrix is recomputed. * * Possible `kind` values : * - VertexBuffer.PositionKind * - VertexBuffer.UVKind * - VertexBuffer.UV2Kind * - VertexBuffer.UV3Kind * - VertexBuffer.UV4Kind * - VertexBuffer.UV5Kind * - VertexBuffer.UV6Kind * - VertexBuffer.ColorKind * - VertexBuffer.MatricesIndicesKind * - VertexBuffer.MatricesIndicesExtraKind * - VertexBuffer.MatricesWeightsKind * - VertexBuffer.MatricesWeightsExtraKind * * Returns the Mesh. * @param kind * @param data * @param updatable * @param stride */ setVerticesData(e, t, r, n) { return this.sourceMesh && this.sourceMesh.setVerticesData(e, t, r, n), this.sourceMesh; } /** * Updates the existing vertex data of the mesh geometry for the requested `kind`. * If the mesh has no geometry, it is simply returned as it is. * The `data` are either a numeric array either a Float32Array. * No new underlying VertexBuffer object is created. * If the `kind` is the `PositionKind` and if `updateExtends` is true, the mesh BoundingInfo is renewed, so the bounding box and sphere, and the mesh World Matrix is recomputed. * If the parameter `makeItUnique` is true, a new global geometry is created from this positions and is set to the mesh. * * Possible `kind` values : * - VertexBuffer.PositionKind * - VertexBuffer.UVKind * - VertexBuffer.UV2Kind * - VertexBuffer.UV3Kind * - VertexBuffer.UV4Kind * - VertexBuffer.UV5Kind * - VertexBuffer.UV6Kind * - VertexBuffer.ColorKind * - VertexBuffer.MatricesIndicesKind * - VertexBuffer.MatricesIndicesExtraKind * - VertexBuffer.MatricesWeightsKind * - VertexBuffer.MatricesWeightsExtraKind * * Returns the Mesh. * @param kind * @param data * @param updateExtends * @param makeItUnique */ updateVerticesData(e, t, r, n) { return this.sourceMesh && this.sourceMesh.updateVerticesData(e, t, r, n), this.sourceMesh; } /** * Sets the mesh indices. * Expects an array populated with integers or a typed array (Int32Array, Uint32Array, Uint16Array). * If the mesh has no geometry, a new Geometry object is created and set to the mesh. * This method creates a new index buffer each call. * Returns the Mesh. * @param indices * @param totalVertices */ setIndices(e, t = null) { return this.sourceMesh && this.sourceMesh.setIndices(e, t), this.sourceMesh; } /** * Boolean : True if the mesh owns the requested kind of data. * @param kind */ isVerticesDataPresent(e) { return this._sourceMesh.isVerticesDataPresent(e); } /** * Returns an array of indices (IndicesArray). */ getIndices() { return this._sourceMesh.getIndices(); } get _positions() { return this._sourceMesh._positions; } /** * This method recomputes and sets a new BoundingInfo to the mesh unless it is locked. * This means the mesh underlying bounding box and sphere are recomputed. * @param applySkeleton defines whether to apply the skeleton before computing the bounding info * @param applyMorph defines whether to apply the morph target before computing the bounding info * @returns the current mesh */ refreshBoundingInfo(e = !1, t = !1) { if (this.hasBoundingInfo && this.getBoundingInfo().isLocked) return this; const r = this._sourceMesh.geometry ? this._sourceMesh.geometry.boundingBias : null; return this._refreshBoundingInfo(this._sourceMesh._getPositionData(e, t), r), this; } /** @internal */ _preActivate() { return this._currentLOD && this._currentLOD._preActivate(), this; } /** * @internal */ _activate(e, t) { if (super._activate(e, t), this._sourceMesh.subMeshes || Se.Warn("Instances should only be created for meshes with geometry."), this._currentLOD) { if (this._currentLOD._getWorldMatrixDeterminant() >= 0 != this._getWorldMatrixDeterminant() >= 0) return this._internalAbstractMeshDataInfo._actAsRegularMesh = !0, !0; if (this._internalAbstractMeshDataInfo._actAsRegularMesh = !1, this._currentLOD._registerInstanceForRenderId(this, e), t) { if (!this._currentLOD._internalAbstractMeshDataInfo._isActiveIntermediate) return this._currentLOD._internalAbstractMeshDataInfo._onlyForInstancesIntermediate = !0, !0; } else if (!this._currentLOD._internalAbstractMeshDataInfo._isActive) return this._currentLOD._internalAbstractMeshDataInfo._onlyForInstances = !0, !0; } return !1; } /** @internal */ _postActivate() { this._sourceMesh.edgesShareWithInstances && this._sourceMesh._edgesRenderer && this._sourceMesh._edgesRenderer.isEnabled && this._sourceMesh._renderingGroup ? (this._sourceMesh._renderingGroup._edgesRenderers.pushNoDuplicate(this._sourceMesh._edgesRenderer), this._sourceMesh._edgesRenderer.customInstances.push(this.getWorldMatrix())) : this._edgesRenderer && this._edgesRenderer.isEnabled && this._sourceMesh._renderingGroup && this._sourceMesh._renderingGroup._edgesRenderers.push(this._edgesRenderer); } getWorldMatrix() { if (this._currentLOD && this._currentLOD.billboardMode !== Hr.BILLBOARDMODE_NONE && this._currentLOD._masterMesh !== this) { this._billboardWorldMatrix || (this._billboardWorldMatrix = new he()); const e = this._currentLOD._masterMesh; return this._currentLOD._masterMesh = this, ue.Vector3[7].copyFrom(this._currentLOD.position), this._currentLOD.position.set(0, 0, 0), this._billboardWorldMatrix.copyFrom(this._currentLOD.computeWorldMatrix(!0)), this._currentLOD.position.copyFrom(ue.Vector3[7]), this._currentLOD._masterMesh = e, this._billboardWorldMatrix; } return super.getWorldMatrix(); } get isAnInstance() { return !0; } /** * Returns the current associated LOD AbstractMesh. * @param camera */ getLOD(e) { if (!e) return this; const t = this.sourceMesh.getLODLevels(); if (!t || t.length === 0) this._currentLOD = this.sourceMesh; else { const r = this.getBoundingInfo(); this._currentLOD = this.sourceMesh.getLOD(e, r.boundingSphere); } return this._currentLOD; } /** * @internal */ _preActivateForIntermediateRendering(e) { return this.sourceMesh._preActivateForIntermediateRendering(e); } /** @internal */ _syncSubMeshes() { if (this.releaseSubMeshes(), this._sourceMesh.subMeshes) for (let e = 0; e < this._sourceMesh.subMeshes.length; e++) this._sourceMesh.subMeshes[e].clone(this, this._sourceMesh); return this; } /** @internal */ _generatePointsArray() { return this._sourceMesh._generatePointsArray(); } /** @internal */ _updateBoundingInfo() { return this.hasBoundingInfo ? this.getBoundingInfo().update(this.worldMatrixFromCache) : this.buildBoundingInfo(this.absolutePosition, this.absolutePosition, this.worldMatrixFromCache), this._updateSubMeshesBoundingInfo(this.worldMatrixFromCache), this; } /** * Creates a new InstancedMesh from the current mesh. * * Returns the clone. * @param name the cloned mesh name * @param newParent the optional Node to parent the clone to. * @param doNotCloneChildren if `true` the model children aren't cloned. * @param newSourceMesh if set this mesh will be used as the source mesh instead of ths instance's one * @returns the clone */ clone(e, t = null, r, n) { const i = (n || this._sourceMesh).createInstance(e); if (sA.DeepCopy(this, i, [ "name", "subMeshes", "uniqueId", "parent", "lightSources", "receiveShadows", "material", "visibility", "skeleton", "sourceMesh", "isAnInstance", "facetNb", "isFacetDataEnabled", "isBlocked", "useBones", "hasInstances", "collider", "edgesRenderer", "forward", "up", "right", "absolutePosition", "absoluteScaling", "absoluteRotationQuaternion", "isWorldMatrixFrozen", "nonUniformScaling", "behaviors", "worldMatrixFromCache", "hasThinInstances", "hasBoundingInfo" ], []), this.refreshBoundingInfo(), t && (i.parent = t), !r) for (let s = 0; s < this.getScene().meshes.length; s++) { const a = this.getScene().meshes[s]; a.parent === this && a.clone(a.name, i); } return i.computeWorldMatrix(!0), this.onClonedObservable.notifyObservers(i), i; } /** * Disposes the InstancedMesh. * Returns nothing. * @param doNotRecurse * @param disposeMaterialAndTextures */ dispose(e, t = !1) { this._sourceMesh.removeInstance(this), super.dispose(e, t); } /** * @internal */ _serializeAsParent(e) { super._serializeAsParent(e), e.parentId = this._sourceMesh.uniqueId, e.parentInstanceIndex = this._indexInSourceMeshInstanceArray; } /** * Instantiate (when possible) or clone that node with its hierarchy * @param newParent defines the new parent to use for the instance (or clone) * @param options defines options to configure how copy is done * @param options.doNotInstantiate defines if the model must be instantiated or just cloned * @param options.newSourcedMesh newSourcedMesh the new source mesh for the instance (or clone) * @param onNewNodeCreated defines an option callback to call when a clone or an instance is created * @returns an instance (or a clone) of the current node with its hierarchy */ instantiateHierarchy(e = null, t, r) { const n = this.clone("Clone of " + (this.name || this.id), e || this.parent, !0, t && t.newSourcedMesh); n && r && r(this, n); for (const i of this.getChildTransformNodes(!0)) i.instantiateHierarchy(n, t, r); return n; } } Ee.prototype.registerInstancedBuffer = function(A, e) { var t, r; if ((r = (t = this._userInstancedBuffersStorage) === null || t === void 0 ? void 0 : t.vertexBuffers[A]) === null || r === void 0 || r.dispose(), !this.instancedBuffers) { this.instancedBuffers = {}; for (const n of this.instances) n.instancedBuffers = {}; } this._userInstancedBuffersStorage || (this._userInstancedBuffersStorage = { data: {}, vertexBuffers: {}, strides: {}, sizes: {}, vertexArrayObjects: this.getEngine().getCaps().vertexArrayObject ? {} : void 0 }), this.instancedBuffers[A] = null, this._userInstancedBuffersStorage.strides[A] = e, this._userInstancedBuffersStorage.sizes[A] = e * 32, this._userInstancedBuffersStorage.data[A] = new Float32Array(this._userInstancedBuffersStorage.sizes[A]), this._userInstancedBuffersStorage.vertexBuffers[A] = new J(this.getEngine(), this._userInstancedBuffersStorage.data[A], A, !0, !1, e, !0); for (const n of this.instances) n.instancedBuffers[A] = null; this._invalidateInstanceVertexArrayObject(), this._markSubMeshesAsAttributesDirty(); }; Ee.prototype._processInstancedBuffers = function(A, e) { const t = A ? A.length : 0; for (const r in this.instancedBuffers) { let n = this._userInstancedBuffersStorage.sizes[r]; const i = this._userInstancedBuffersStorage.strides[r], s = (t + 1) * i; for (; n < s; ) n *= 2; this._userInstancedBuffersStorage.data[r].length != n && (this._userInstancedBuffersStorage.data[r] = new Float32Array(n), this._userInstancedBuffersStorage.sizes[r] = n, this._userInstancedBuffersStorage.vertexBuffers[r] && (this._userInstancedBuffersStorage.vertexBuffers[r].dispose(), this._userInstancedBuffersStorage.vertexBuffers[r] = null)); const a = this._userInstancedBuffersStorage.data[r]; let f = 0; if (e) { const o = this.instancedBuffers[r]; o.toArray ? o.toArray(a, f) : o.copyToArray ? o.copyToArray(a, f) : a[f] = o, f += i; } for (let o = 0; o < t; o++) { const v = A[o].instancedBuffers[r]; v.toArray ? v.toArray(a, f) : v.copyToArray ? v.copyToArray(a, f) : a[f] = v, f += i; } this._userInstancedBuffersStorage.vertexBuffers[r] ? this._userInstancedBuffersStorage.vertexBuffers[r].updateDirectly(a, 0) : (this._userInstancedBuffersStorage.vertexBuffers[r] = new J(this.getEngine(), this._userInstancedBuffersStorage.data[r], r, !0, !1, i, !0), this._invalidateInstanceVertexArrayObject()); } }; Ee.prototype._invalidateInstanceVertexArrayObject = function() { if (!(!this._userInstancedBuffersStorage || this._userInstancedBuffersStorage.vertexArrayObjects === void 0)) { for (const A in this._userInstancedBuffersStorage.vertexArrayObjects) this.getEngine().releaseVertexArrayObject(this._userInstancedBuffersStorage.vertexArrayObjects[A]); this._userInstancedBuffersStorage.vertexArrayObjects = {}; } }; Ee.prototype._disposeInstanceSpecificData = function() { for (this._instanceDataStorage.instancesBuffer && (this._instanceDataStorage.instancesBuffer.dispose(), this._instanceDataStorage.instancesBuffer = null); this.instances.length; ) this.instances[0].dispose(); for (const A in this.instancedBuffers) this._userInstancedBuffersStorage.vertexBuffers[A] && this._userInstancedBuffersStorage.vertexBuffers[A].dispose(); this._invalidateInstanceVertexArrayObject(), this.instancedBuffers = {}; }; class ci extends Cs { /** * Defines how far from the source the light is impacting in scene units. * Note: Unused in PBR material as the distance light falloff is defined following the inverse squared falloff. */ get range() { return this._range; } /** * Defines how far from the source the light is impacting in scene units. * Note: Unused in PBR material as the distance light falloff is defined following the inverse squared falloff. */ set range(e) { this._range = e, this._inverseSquaredRange = 1 / (this.range * this.range); } /** * Gets the photometric scale used to interpret the intensity. * This is only relevant with PBR Materials where the light intensity can be defined in a physical way. */ get intensityMode() { return this._intensityMode; } /** * Sets the photometric scale used to interpret the intensity. * This is only relevant with PBR Materials where the light intensity can be defined in a physical way. */ set intensityMode(e) { this._intensityMode = e, this._computePhotometricScale(); } /** * Gets the light radius used by PBR Materials to simulate soft area lights. */ get radius() { return this._radius; } /** * sets the light radius used by PBR Materials to simulate soft area lights. */ set radius(e) { this._radius = e, this._computePhotometricScale(); } /** * Gets whether or not the shadows are enabled for this light. This can help turning off/on shadow without detaching * the current shadow generator. */ get shadowEnabled() { return this._shadowEnabled; } /** * Sets whether or not the shadows are enabled for this light. This can help turning off/on shadow without detaching * the current shadow generator. */ set shadowEnabled(e) { this._shadowEnabled !== e && (this._shadowEnabled = e, this._markMeshesAsLightDirty()); } /** * Gets the only meshes impacted by this light. */ get includedOnlyMeshes() { return this._includedOnlyMeshes; } /** * Sets the only meshes impacted by this light. */ set includedOnlyMeshes(e) { this._includedOnlyMeshes = e, this._hookArrayForIncludedOnly(e); } /** * Gets the meshes not impacted by this light. */ get excludedMeshes() { return this._excludedMeshes; } /** * Sets the meshes not impacted by this light. */ set excludedMeshes(e) { this._excludedMeshes = e, this._hookArrayForExcluded(e); } /** * Gets the layer id use to find what meshes are not impacted by the light. * Inactive if 0 */ get excludeWithLayerMask() { return this._excludeWithLayerMask; } /** * Sets the layer id use to find what meshes are not impacted by the light. * Inactive if 0 */ set excludeWithLayerMask(e) { this._excludeWithLayerMask = e, this._resyncMeshes(); } /** * Gets the layer id use to find what meshes are impacted by the light. * Inactive if 0 */ get includeOnlyWithLayerMask() { return this._includeOnlyWithLayerMask; } /** * Sets the layer id use to find what meshes are impacted by the light. * Inactive if 0 */ set includeOnlyWithLayerMask(e) { this._includeOnlyWithLayerMask = e, this._resyncMeshes(); } /** * Gets the lightmap mode of this light (should be one of the constants defined by Light.LIGHTMAP_x) */ get lightmapMode() { return this._lightmapMode; } /** * Sets the lightmap mode of this light (should be one of the constants defined by Light.LIGHTMAP_x) */ set lightmapMode(e) { this._lightmapMode !== e && (this._lightmapMode = e, this._markMeshesAsLightDirty()); } /** * Creates a Light object in the scene. * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction * @param name The friendly name of the light * @param scene The scene the light belongs too */ constructor(e, t) { super(e, t), this.diffuse = new Ne(1, 1, 1), this.specular = new Ne(1, 1, 1), this.falloffType = ci.FALLOFF_DEFAULT, this.intensity = 1, this._range = Number.MAX_VALUE, this._inverseSquaredRange = 0, this._photometricScale = 1, this._intensityMode = ci.INTENSITYMODE_AUTOMATIC, this._radius = 1e-5, this.renderPriority = 0, this._shadowEnabled = !0, this._excludeWithLayerMask = 0, this._includeOnlyWithLayerMask = 0, this._lightmapMode = 0, this._shadowGenerators = null, this._excludedMeshesIds = new Array(), this._includedOnlyMeshesIds = new Array(), this._isLight = !0, this.getScene().addLight(this), this._uniformBuffer = new yr(this.getScene().getEngine(), void 0, void 0, e), this._buildUniformLayout(), this.includedOnlyMeshes = [], this.excludedMeshes = [], this._resyncMeshes(); } /** * Sets the passed Effect "effect" with the Light textures. * @param effect The effect to update * @param lightIndex The index of the light in the effect to update * @returns The light */ // eslint-disable-next-line @typescript-eslint/no-unused-vars transferTexturesToEffect(e, t) { return this; } /** * Binds the lights information from the scene to the effect for the given mesh. * @param lightIndex Light index * @param scene The scene where the light belongs to * @param effect The effect we are binding the data to * @param useSpecular Defines if specular is supported * @param receiveShadows Defines if the effect (mesh) we bind the light for receives shadows */ _bindLight(e, t, r, n, i = !0) { var s; const a = e.toString(); let f = !1; if (this._uniformBuffer.bindToEffect(r, "Light" + a), this._renderId !== t.getRenderId() || this._lastUseSpecular !== n || !this._uniformBuffer.useUbo) { this._renderId = t.getRenderId(), this._lastUseSpecular = n; const o = this.getScaledIntensity(); this.transferToEffect(r, a), this.diffuse.scaleToRef(o, Hs.Color3[0]), this._uniformBuffer.updateColor4("vLightDiffuse", Hs.Color3[0], this.range, a), n && (this.specular.scaleToRef(o, Hs.Color3[1]), this._uniformBuffer.updateColor4("vLightSpecular", Hs.Color3[1], this.radius, a)), f = !0; } if (this.transferTexturesToEffect(r, a), t.shadowsEnabled && this.shadowEnabled && i) { const o = (s = this.getShadowGenerator(t.activeCamera)) !== null && s !== void 0 ? s : this.getShadowGenerator(); o && (o.bindShadowLight(a, r), f = !0); } f ? this._uniformBuffer.update() : this._uniformBuffer.bindUniformBuffer(); } /** * Returns the string "Light". * @returns the class name */ getClassName() { return "Light"; } /** * Converts the light information to a readable string for debug purpose. * @param fullDetails Supports for multiple levels of logging within scene loading * @returns the human readable light info */ toString(e) { let t = "Name: " + this.name; if (t += ", type: " + ["Point", "Directional", "Spot", "Hemispheric"][this.getTypeID()], this.animations) for (let r = 0; r < this.animations.length; r++) t += ", animation[0]: " + this.animations[r].toString(e); return t; } /** @internal */ _syncParentEnabledState() { super._syncParentEnabledState(), this.isDisposed() || this._resyncMeshes(); } /** * Set the enabled state of this node. * @param value - the new enabled state */ setEnabled(e) { super.setEnabled(e), this._resyncMeshes(); } /** * Returns the Light associated shadow generator if any. * @param camera Camera for which the shadow generator should be retrieved (default: null). If null, retrieves the default shadow generator * @returns the associated shadow generator. */ getShadowGenerator(e = null) { var t; return this._shadowGenerators === null ? null : (t = this._shadowGenerators.get(e)) !== null && t !== void 0 ? t : null; } /** * Returns all the shadow generators associated to this light * @returns */ getShadowGenerators() { return this._shadowGenerators; } /** * Returns a Vector3, the absolute light position in the World. * @returns the world space position of the light */ getAbsolutePosition() { return S.Zero(); } /** * Specifies if the light will affect the passed mesh. * @param mesh The mesh to test against the light * @returns true the mesh is affected otherwise, false. */ canAffectMesh(e) { return e ? !(this.includedOnlyMeshes && this.includedOnlyMeshes.length > 0 && this.includedOnlyMeshes.indexOf(e) === -1 || this.excludedMeshes && this.excludedMeshes.length > 0 && this.excludedMeshes.indexOf(e) !== -1 || this.includeOnlyWithLayerMask !== 0 && !(this.includeOnlyWithLayerMask & e.layerMask) || this.excludeWithLayerMask !== 0 && this.excludeWithLayerMask & e.layerMask) : !0; } /** * Releases resources associated with this node. * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default) * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default) */ dispose(e, t = !1) { if (this._shadowGenerators) { const r = this._shadowGenerators.values(); for (let n = r.next(); n.done !== !0; n = r.next()) n.value.dispose(); this._shadowGenerators = null; } if (this.getScene().stopAnimation(this), this._parentContainer) { const r = this._parentContainer.lights.indexOf(this); r > -1 && this._parentContainer.lights.splice(r, 1), this._parentContainer = null; } for (const r of this.getScene().meshes) r._removeLightSource(this, !0); this._uniformBuffer.dispose(), this.getScene().removeLight(this), super.dispose(e, t); } /** * Returns the light type ID (integer). * @returns The light Type id as a constant defines in Light.LIGHTTYPEID_x */ getTypeID() { return 0; } /** * Returns the intensity scaled by the Photometric Scale according to the light type and intensity mode. * @returns the scaled intensity in intensity mode unit */ getScaledIntensity() { return this._photometricScale * this.intensity; } /** * Returns a new Light object, named "name", from the current one. * @param name The name of the cloned light * @param newParent The parent of this light, if it has one * @returns the new created light */ clone(e, t = null) { const r = ci.GetConstructorFromName(this.getTypeID(), e, this.getScene()); if (!r) return null; const n = jt.Clone(r, this); return e && (n.name = e), t && (n.parent = t), n.setEnabled(this.isEnabled()), this.onClonedObservable.notifyObservers(n), n; } /** * Serializes the current light into a Serialization object. * @returns the serialized object. */ serialize() { const e = jt.Serialize(this); return e.uniqueId = this.uniqueId, e.type = this.getTypeID(), this.parent && this.parent._serializeAsParent(e), this.excludedMeshes.length > 0 && (e.excludedMeshesIds = [], this.excludedMeshes.forEach((t) => { e.excludedMeshesIds.push(t.id); })), this.includedOnlyMeshes.length > 0 && (e.includedOnlyMeshesIds = [], this.includedOnlyMeshes.forEach((t) => { e.includedOnlyMeshesIds.push(t.id); })), jt.AppendSerializedAnimations(this, e), e.ranges = this.serializeAnimationRanges(), e.isEnabled = this.isEnabled(), e; } /** * Creates a new typed light from the passed type (integer) : point light = 0, directional light = 1, spot light = 2, hemispheric light = 3. * This new light is named "name" and added to the passed scene. * @param type Type according to the types available in Light.LIGHTTYPEID_x * @param name The friendly name of the light * @param scene The scene the new light will belong to * @returns the constructor function */ static GetConstructorFromName(e, t, r) { const n = Cs.Construct("Light_Type_" + e, t, r); return n || null; } /** * Parses the passed "parsedLight" and returns a new instanced Light from this parsing. * @param parsedLight The JSON representation of the light * @param scene The scene to create the parsed light in * @returns the created light after parsing */ static Parse(e, t) { const r = ci.GetConstructorFromName(e.type, e.name, t); if (!r) return null; const n = jt.Parse(r, e, t); if (e.excludedMeshesIds && (n._excludedMeshesIds = e.excludedMeshesIds), e.includedOnlyMeshesIds && (n._includedOnlyMeshesIds = e.includedOnlyMeshesIds), e.parentId !== void 0 && (n._waitingParentId = e.parentId), e.parentInstanceIndex !== void 0 && (n._waitingParentInstanceIndex = e.parentInstanceIndex), e.falloffType !== void 0 && (n.falloffType = e.falloffType), e.lightmapMode !== void 0 && (n.lightmapMode = e.lightmapMode), e.animations) { for (let i = 0; i < e.animations.length; i++) { const s = e.animations[i], a = Jo("BABYLON.Animation"); a && n.animations.push(a.Parse(s)); } Cs.ParseAnimationRanges(n, e, t); } return e.autoAnimate && t.beginAnimation(n, e.autoAnimateFrom, e.autoAnimateTo, e.autoAnimateLoop, e.autoAnimateSpeed || 1), e.isEnabled !== void 0 && n.setEnabled(e.isEnabled), n; } _hookArrayForExcluded(e) { const t = e.push; e.push = (...n) => { const i = t.apply(e, n); for (const s of n) s._resyncLightSource(this); return i; }; const r = e.splice; e.splice = (n, i) => { const s = r.apply(e, [n, i]); for (const a of s) a._resyncLightSource(this); return s; }; for (const n of e) n._resyncLightSource(this); } _hookArrayForIncludedOnly(e) { const t = e.push; e.push = (...n) => { const i = t.apply(e, n); return this._resyncMeshes(), i; }; const r = e.splice; e.splice = (n, i) => { const s = r.apply(e, [n, i]); return this._resyncMeshes(), s; }, this._resyncMeshes(); } _resyncMeshes() { for (const e of this.getScene().meshes) e._resyncLightSource(this); } /** * Forces the meshes to update their light related information in their rendering used effects * @internal Internal Use Only */ _markMeshesAsLightDirty() { for (const e of this.getScene().meshes) e.lightSources.indexOf(this) !== -1 && e._markSubMeshesAsLightDirty(); } /** * Recomputes the cached photometric scale if needed. */ _computePhotometricScale() { this._photometricScale = this._getPhotometricScale(), this.getScene().resetCachedMaterial(); } /** * Returns the Photometric Scale according to the light type and intensity mode. */ _getPhotometricScale() { let e = 0; const t = this.getTypeID(); let r = this.intensityMode; switch (r === ci.INTENSITYMODE_AUTOMATIC && (t === ci.LIGHTTYPEID_DIRECTIONALLIGHT ? r = ci.INTENSITYMODE_ILLUMINANCE : r = ci.INTENSITYMODE_LUMINOUSINTENSITY), t) { case ci.LIGHTTYPEID_POINTLIGHT: case ci.LIGHTTYPEID_SPOTLIGHT: switch (r) { case ci.INTENSITYMODE_LUMINOUSPOWER: e = 1 / (4 * Math.PI); break; case ci.INTENSITYMODE_LUMINOUSINTENSITY: e = 1; break; case ci.INTENSITYMODE_LUMINANCE: e = this.radius * this.radius; break; } break; case ci.LIGHTTYPEID_DIRECTIONALLIGHT: switch (r) { case ci.INTENSITYMODE_ILLUMINANCE: e = 1; break; case ci.INTENSITYMODE_LUMINANCE: { let n = this.radius; n = Math.max(n, 1e-3), e = 2 * Math.PI * (1 - Math.cos(n)); break; } } break; case ci.LIGHTTYPEID_HEMISPHERICLIGHT: e = 1; break; } return e; } /** * Reorder the light in the scene according to their defined priority. * @internal Internal Use Only */ _reorderLightsInScene() { const e = this.getScene(); this._renderPriority != 0 && (e.requireLightSorting = !0), this.getScene().sortLightsByPriority(); } } ci.FALLOFF_DEFAULT = ra.FALLOFF_DEFAULT; ci.FALLOFF_PHYSICAL = ra.FALLOFF_PHYSICAL; ci.FALLOFF_GLTF = ra.FALLOFF_GLTF; ci.FALLOFF_STANDARD = ra.FALLOFF_STANDARD; ci.LIGHTMAP_DEFAULT = ra.LIGHTMAP_DEFAULT; ci.LIGHTMAP_SPECULAR = ra.LIGHTMAP_SPECULAR; ci.LIGHTMAP_SHADOWSONLY = ra.LIGHTMAP_SHADOWSONLY; ci.INTENSITYMODE_AUTOMATIC = ra.INTENSITYMODE_AUTOMATIC; ci.INTENSITYMODE_LUMINOUSPOWER = ra.INTENSITYMODE_LUMINOUSPOWER; ci.INTENSITYMODE_LUMINOUSINTENSITY = ra.INTENSITYMODE_LUMINOUSINTENSITY; ci.INTENSITYMODE_ILLUMINANCE = ra.INTENSITYMODE_ILLUMINANCE; ci.INTENSITYMODE_LUMINANCE = ra.INTENSITYMODE_LUMINANCE; ci.LIGHTTYPEID_POINTLIGHT = ra.LIGHTTYPEID_POINTLIGHT; ci.LIGHTTYPEID_DIRECTIONALLIGHT = ra.LIGHTTYPEID_DIRECTIONALLIGHT; ci.LIGHTTYPEID_SPOTLIGHT = ra.LIGHTTYPEID_SPOTLIGHT; ci.LIGHTTYPEID_HEMISPHERICLIGHT = ra.LIGHTTYPEID_HEMISPHERICLIGHT; C([ Oi() ], ci.prototype, "diffuse", void 0); C([ Oi() ], ci.prototype, "specular", void 0); C([ M() ], ci.prototype, "falloffType", void 0); C([ M() ], ci.prototype, "intensity", void 0); C([ M() ], ci.prototype, "range", null); C([ M() ], ci.prototype, "intensityMode", null); C([ M() ], ci.prototype, "radius", null); C([ M() ], ci.prototype, "_renderPriority", void 0); C([ At("_reorderLightsInScene") ], ci.prototype, "renderPriority", void 0); C([ M("shadowEnabled") ], ci.prototype, "_shadowEnabled", void 0); C([ M("excludeWithLayerMask") ], ci.prototype, "_excludeWithLayerMask", void 0); C([ M("includeOnlyWithLayerMask") ], ci.prototype, "_includeOnlyWithLayerMask", void 0); C([ M("lightmapMode") ], ci.prototype, "_lightmapMode", void 0); class Aee extends J1 { } class dee { constructor() { this.rootNodes = [], this.skeletons = [], this.animationGroups = []; } /** * Disposes the instantiated entries from the scene */ dispose() { this.rootNodes.slice(0).forEach((e) => { e.dispose(); }), this.rootNodes.length = 0, this.skeletons.slice(0).forEach((e) => { e.dispose(); }), this.skeletons.length = 0, this.animationGroups.slice(0).forEach((e) => { e.dispose(); }), this.animationGroups.length = 0; } } class xR extends J1 { /** * Instantiates an AssetContainer. * @param scene The scene the AssetContainer belongs to. */ constructor(e) { super(), this._wasAddedToScene = !1, e = e || gr.LastCreatedScene, e && (this.scene = e, this.sounds = [], this.effectLayers = [], this.layers = [], this.lensFlareSystems = [], this.proceduralTextures = [], this.reflectionProbes = [], e.onDisposeObservable.add(() => { this._wasAddedToScene || this.dispose(); }), this._onContextRestoredObserver = e.getEngine().onContextRestoredObservable.add(() => { for (const t of this.geometries) t._rebuild(); for (const t of this.meshes) t._rebuild(); for (const t of this.particleSystems) t.rebuild(); for (const t of this.textures) t._rebuild(); })); } /** * Given a list of nodes, return a topological sorting of them. * @param nodes */ _topologicalSort(e) { const t = /* @__PURE__ */ new Map(); for (const a of e) t.set(a.uniqueId, a); const r = { dependsOn: /* @__PURE__ */ new Map(), dependedBy: /* @__PURE__ */ new Map() // given a node id, what are the ids of the nodes that depend on it }; for (const a of e) { const f = a.uniqueId; r.dependsOn.set(f, /* @__PURE__ */ new Set()), r.dependedBy.set(f, /* @__PURE__ */ new Set()); } for (const a of e) { const f = a.uniqueId, o = r.dependsOn.get(f); if (a instanceof bp) { const v = a.sourceMesh; t.has(v.uniqueId) && (o.add(v.uniqueId), r.dependedBy.get(v.uniqueId).add(f)); } const d = r.dependedBy.get(f); for (const v of a.getDescendants()) { const u = v.uniqueId; t.has(u) && (d.add(u), r.dependsOn.get(u).add(f)); } } const n = [], i = []; for (const a of e) { const f = a.uniqueId; r.dependsOn.get(f).size === 0 && (i.push(a), t.delete(f)); } const s = i; for (; s.length > 0; ) { const a = s.shift(); n.push(a); const f = r.dependedBy.get(a.uniqueId); for (const o of Array.from(f.values())) { const d = r.dependsOn.get(o); d.delete(a.uniqueId), d.size === 0 && t.get(o) && (s.push(t.get(o)), t.delete(o)); } } return t.size > 0 && (console.error("SceneSerializer._topologicalSort: There were unvisited nodes:"), t.forEach((a) => console.error(a.name))), n; } _addNodeAndDescendantsToList(e, t, r, n) { if (!(!r || n && !n(r) || t.has(r.uniqueId))) { e.push(r), t.add(r.uniqueId); for (const i of r.getDescendants(!0)) this._addNodeAndDescendantsToList(e, t, i, n); } } /** * Check if a specific node is contained in this asset container. * @param node */ _isNodeInContainer(e) { return e instanceof Ee && this.meshes.indexOf(e) !== -1 || e instanceof Hr && this.transformNodes.indexOf(e) !== -1 || e instanceof ci && this.lights.indexOf(e) !== -1 || e instanceof Tr && this.cameras.indexOf(e) !== -1; } /** * For every node in the scene, check if its parent node is also in the scene. */ _isValidHierarchy() { for (const e of this.meshes) if (e.parent && !this._isNodeInContainer(e.parent)) return Se.Warn(`Node ${e.name} has a parent that is not in the container.`), !1; for (const e of this.transformNodes) if (e.parent && !this._isNodeInContainer(e.parent)) return Se.Warn(`Node ${e.name} has a parent that is not in the container.`), !1; for (const e of this.lights) if (e.parent && !this._isNodeInContainer(e.parent)) return Se.Warn(`Node ${e.name} has a parent that is not in the container.`), !1; for (const e of this.cameras) if (e.parent && !this._isNodeInContainer(e.parent)) return Se.Warn(`Node ${e.name} has a parent that is not in the container.`), !1; return !0; } /** * Instantiate or clone all meshes and add the new ones to the scene. * Skeletons and animation groups will all be cloned * @param nameFunction defines an optional function used to get new names for clones * @param cloneMaterials defines an optional boolean that defines if materials must be cloned as well (false by default) * @param options defines an optional list of options to control how to instantiate / clone models * @param options.doNotInstantiate defines if the model must be instantiated or just cloned * @param options.predicate defines a predicate used to filter whih mesh to instantiate/clone * @returns a list of rootNodes, skeletons and animation groups that were duplicated */ instantiateModelsToScene(e, t = !1, r) { this._isValidHierarchy() || ye.Warn("SceneSerializer.InstantiateModelsToScene: The Asset Container hierarchy is not valid."); const n = {}, i = {}, s = new dee(), a = [], f = [], o = Object.assign({ doNotInstantiate: !0 }, r), d = (p, c) => { if (n[p.uniqueId] = c.uniqueId, i[c.uniqueId] = c, e && (c.name = e(p.name)), c instanceof Ee) { const H = c; if (H.morphTargetManager) { const T = p.morphTargetManager; H.morphTargetManager = T.clone(); for (let q = 0; q < T.numTargets; q++) { const b = T.getTarget(q), j = H.morphTargetManager.getTarget(q); n[b.uniqueId] = j.uniqueId, i[j.uniqueId] = j; } } } }, v = [], u = /* @__PURE__ */ new Set(); for (const p of this.transformNodes) p.parent === null && this._addNodeAndDescendantsToList(v, u, p, o.predicate); for (const p of this.meshes) p.parent === null && this._addNodeAndDescendantsToList(v, u, p, o.predicate); const l = this._topologicalSort(v), P = (p, c) => { if (d(p, c), p.parent) { const H = n[p.parent.uniqueId], T = i[H]; T ? c.parent = T : c.parent = p.parent; } if (c.position && p.position && c.position.copyFrom(p.position), c.rotationQuaternion && p.rotationQuaternion && c.rotationQuaternion.copyFrom(p.rotationQuaternion), c.rotation && p.rotation && c.rotation.copyFrom(p.rotation), c.scaling && p.scaling && c.scaling.copyFrom(p.scaling), c.material) { const H = c; if (H.material) if (t) { const T = p.material; if (f.indexOf(T) === -1) { let q = T.clone(e ? e(T.name) : "Clone of " + T.name); if (f.push(T), n[T.uniqueId] = q.uniqueId, i[q.uniqueId] = q, T.getClassName() === "MultiMaterial") { const b = T; for (const j of b.subMaterials) j && (q = j.clone(e ? e(j.name) : "Clone of " + j.name), f.push(j), n[j.uniqueId] = q.uniqueId, i[q.uniqueId] = q); b.subMaterials = b.subMaterials.map((j) => j && i[n[j.uniqueId]]); } } H.getClassName() !== "InstancedMesh" && (H.material = i[n[T.uniqueId]]); } else H.material.getClassName() === "MultiMaterial" ? this.scene.multiMaterials.indexOf(H.material) === -1 && this.scene.addMultiMaterial(H.material) : this.scene.materials.indexOf(H.material) === -1 && this.scene.addMaterial(H.material); } c.parent === null && s.rootNodes.push(c); }; return l.forEach((p) => { if (p.getClassName() === "InstancedMesh") { const c = p, H = c.sourceMesh, T = n[H.uniqueId], b = (typeof T == "number" ? i[T] : H).createInstance(c.name); P(c, b); } else { let c = !0; p.getClassName() === "TransformNode" || p.getClassName() === "Node" || p.skeleton || !p.getTotalVertices || p.getTotalVertices() === 0 ? c = !1 : o.doNotInstantiate && (typeof o.doNotInstantiate == "function" ? c = !o.doNotInstantiate(p) : c = !o.doNotInstantiate); const H = c ? p.createInstance(`instance of ${p.name}`) : p.clone(`Clone of ${p.name}`, null, !0); if (!H) throw new Error(`Could not clone or instantiate node on Asset Container ${p.name}`); P(p, H); } }), this.skeletons.forEach((p) => { if (o.predicate && !o.predicate(p)) return; const c = p.clone(e ? e(p.name) : "Clone of " + p.name); for (const H of this.meshes) if (H.skeleton === p && !H.isAnInstance) { const T = i[n[H.uniqueId]]; if (!T || T.isAnInstance || (T.skeleton = c, a.indexOf(c) !== -1)) continue; a.push(c); for (const q of c.bones) q._linkedTransformNode && (q._linkedTransformNode = i[n[q._linkedTransformNode.uniqueId]]); } s.skeletons.push(c); }), this.animationGroups.forEach((p) => { if (o.predicate && !o.predicate(p)) return; const c = p.clone(e ? e(p.name) : "Clone of " + p.name, (H) => i[n[H.uniqueId]] || H); s.animationGroups.push(c); }), s; } /** * Adds all the assets from the container to the scene. */ addAllToScene() { if (!this._wasAddedToScene) { this._isValidHierarchy() || ye.Warn("SceneSerializer.addAllToScene: The Asset Container hierarchy is not valid."), this._wasAddedToScene = !0, this.addToScene(null), this.environmentTexture && (this.scene.environmentTexture = this.environmentTexture); for (const e of this.scene._serializableComponents) e.addFromContainer(this); this.scene.getEngine().onContextRestoredObservable.remove(this._onContextRestoredObserver), this._onContextRestoredObserver = null; } } /** * Adds assets from the container to the scene. * @param predicate defines a predicate used to select which entity will be added (can be null) */ addToScene(e = null) { const t = []; this.cameras.forEach((r) => { e && !e(r) || (this.scene.addCamera(r), t.push(r)); }), this.lights.forEach((r) => { e && !e(r) || (this.scene.addLight(r), t.push(r)); }), this.meshes.forEach((r) => { e && !e(r) || (this.scene.addMesh(r), t.push(r)); }), this.skeletons.forEach((r) => { e && !e(r) || this.scene.addSkeleton(r); }), this.animations.forEach((r) => { e && !e(r) || this.scene.addAnimation(r); }), this.animationGroups.forEach((r) => { e && !e(r) || this.scene.addAnimationGroup(r); }), this.multiMaterials.forEach((r) => { e && !e(r) || this.scene.addMultiMaterial(r); }), this.materials.forEach((r) => { e && !e(r) || this.scene.addMaterial(r); }), this.morphTargetManagers.forEach((r) => { e && !e(r) || this.scene.addMorphTargetManager(r); }), this.geometries.forEach((r) => { e && !e(r) || this.scene.addGeometry(r); }), this.transformNodes.forEach((r) => { e && !e(r) || (this.scene.addTransformNode(r), t.push(r)); }), this.actionManagers.forEach((r) => { e && !e(r) || this.scene.addActionManager(r); }), this.textures.forEach((r) => { e && !e(r) || this.scene.addTexture(r); }), this.reflectionProbes.forEach((r) => { e && !e(r) || this.scene.addReflectionProbe(r); }); for (const r of t) r.parent && this.scene.getNodes().indexOf(r.parent) === -1 && (r.setParent ? r.setParent(null) : r.parent = null); } /** * Removes all the assets in the container from the scene */ removeAllFromScene() { this._isValidHierarchy() || ye.Warn("SceneSerializer.removeAllFromScene: The Asset Container hierarchy is not valid."), this._wasAddedToScene = !1, this.removeFromScene(null), this.environmentTexture === this.scene.environmentTexture && (this.scene.environmentTexture = null); for (const e of this.scene._serializableComponents) e.removeFromContainer(this); } /** * Removes assets in the container from the scene * @param predicate defines a predicate used to select which entity will be added (can be null) */ removeFromScene(e = null) { this.cameras.forEach((t) => { e && !e(t) || this.scene.removeCamera(t); }), this.lights.forEach((t) => { e && !e(t) || this.scene.removeLight(t); }), this.meshes.forEach((t) => { e && !e(t) || this.scene.removeMesh(t, !0); }), this.skeletons.forEach((t) => { e && !e(t) || this.scene.removeSkeleton(t); }), this.animations.forEach((t) => { e && !e(t) || this.scene.removeAnimation(t); }), this.animationGroups.forEach((t) => { e && !e(t) || this.scene.removeAnimationGroup(t); }), this.multiMaterials.forEach((t) => { e && !e(t) || this.scene.removeMultiMaterial(t); }), this.materials.forEach((t) => { e && !e(t) || this.scene.removeMaterial(t); }), this.morphTargetManagers.forEach((t) => { e && !e(t) || this.scene.removeMorphTargetManager(t); }), this.geometries.forEach((t) => { e && !e(t) || this.scene.removeGeometry(t); }), this.transformNodes.forEach((t) => { e && !e(t) || this.scene.removeTransformNode(t); }), this.actionManagers.forEach((t) => { e && !e(t) || this.scene.removeActionManager(t); }), this.textures.forEach((t) => { e && !e(t) || this.scene.removeTexture(t); }), this.reflectionProbes.forEach((t) => { e && !e(t) || this.scene.removeReflectionProbe(t); }); } /** * Disposes all the assets in the container */ dispose() { this.cameras.slice(0).forEach((e) => { e.dispose(); }), this.cameras.length = 0, this.lights.slice(0).forEach((e) => { e.dispose(); }), this.lights.length = 0, this.meshes.slice(0).forEach((e) => { e.dispose(); }), this.meshes.length = 0, this.skeletons.slice(0).forEach((e) => { e.dispose(); }), this.skeletons.length = 0, this.animationGroups.slice(0).forEach((e) => { e.dispose(); }), this.animationGroups.length = 0, this.multiMaterials.slice(0).forEach((e) => { e.dispose(); }), this.multiMaterials.length = 0, this.materials.slice(0).forEach((e) => { e.dispose(); }), this.materials.length = 0, this.geometries.slice(0).forEach((e) => { e.dispose(); }), this.geometries.length = 0, this.transformNodes.slice(0).forEach((e) => { e.dispose(); }), this.transformNodes.length = 0, this.actionManagers.slice(0).forEach((e) => { e.dispose(); }), this.actionManagers.length = 0, this.textures.slice(0).forEach((e) => { e.dispose(); }), this.textures.length = 0, this.reflectionProbes.slice(0).forEach((e) => { e.dispose(); }), this.reflectionProbes.length = 0, this.morphTargetManagers.slice(0).forEach((e) => { e.dispose(); }), this.morphTargetManagers.length = 0, this.environmentTexture && (this.environmentTexture.dispose(), this.environmentTexture = null); for (const e of this.scene._serializableComponents) e.removeFromContainer(this, !0); this._onContextRestoredObserver && (this.scene.getEngine().onContextRestoredObservable.remove(this._onContextRestoredObserver), this._onContextRestoredObserver = null); } _moveAssets(e, t, r) { if (!(!e || !t)) for (const n of e) { let i = !0; if (r) { for (const s of r) if (n === s) { i = !1; break; } } i && (t.push(n), n._parentContainer = this); } } /** * Removes all the assets contained in the scene and adds them to the container. * @param keepAssets Set of assets to keep in the scene. (default: empty) */ moveAllFromScene(e) { this._wasAddedToScene = !1, e === void 0 && (e = new Aee()); for (const t in this) Object.prototype.hasOwnProperty.call(this, t) && (this[t] = this[t] || (t === "_environmentTexture" ? null : []), this._moveAssets(this.scene[t], this[t], e[t])); this.environmentTexture = this.scene.environmentTexture, this.removeAllFromScene(); } /** * Adds all meshes in the asset container to a root mesh that can be used to position all the contained meshes. The root mesh is then added to the front of the meshes in the assetContainer. * @returns the root mesh */ createRootMesh() { const e = new Ee("assetContainerRootMesh", this.scene); return this.meshes.forEach((t) => { t.parent || e.addChild(t); }), this.meshes.unshift(e), e; } /** * Merge animations (direct and animation groups) from this asset container into a scene * @param scene is the instance of BABYLON.Scene to append to (default: last created scene) * @param animatables set of animatables to retarget to a node from the scene * @param targetConverter defines a function used to convert animation targets from the asset container to the scene (default: search node by name) * @returns an array of the new AnimationGroup added to the scene (empty array if none) */ mergeAnimationsTo(e = gr.LastCreatedScene, t, r = null) { if (!e) return Se.Error("No scene available to merge animations to"), []; const n = r || ((a) => { let f = null; const o = a.animations.length ? a.animations[0].targetProperty : "", d = a.name.split(".").join("").split("_primitive")[0]; switch (o) { case "position": case "rotationQuaternion": f = e.getTransformNodeByName(a.name) || e.getTransformNodeByName(d); break; case "influence": f = e.getMorphTargetByName(a.name) || e.getMorphTargetByName(d); break; default: f = e.getNodeByName(a.name) || e.getNodeByName(d); } return f; }); this.getNodes().forEach((a) => { const f = n(a); if (f !== null) { for (const o of a.animations) { const d = f.animations.filter((v) => v.targetProperty === o.targetProperty); for (const v of d) { const u = f.animations.indexOf(v, 0); u > -1 && f.animations.splice(u, 1); } } f.animations = f.animations.concat(a.animations); } }); const s = []; return this.animationGroups.slice().forEach((a) => { s.push(a.clone(a.name, n)), a.animatables.forEach((f) => { f.stop(); }); }), t.forEach((a) => { const f = n(a.target); f && (e.beginAnimation(f, a.fromFrame, a.toFrame, a.loopAnimation, a.speedRatio, a.onAnimationEnd ? a.onAnimationEnd : void 0, void 0, !0, void 0, a.onAnimationLoop ? a.onAnimationLoop : void 0), e.stopAnimation(a.target)); }), s; } /** * @since 6.15.0 * This method checks for any node that has no parent * and is not in the rootNodes array, and adds the node * there, if so. */ populateRootNodes() { this.rootNodes.length = 0, this.meshes.forEach((e) => { !e.parent && this.rootNodes.indexOf(e) === -1 && this.rootNodes.push(e); }), this.transformNodes.forEach((e) => { !e.parent && this.rootNodes.indexOf(e) === -1 && this.rootNodes.push(e); }), this.lights.forEach((e) => { !e.parent && this.rootNodes.indexOf(e) === -1 && this.rootNodes.push(e); }), this.cameras.forEach((e) => { !e.parent && this.rootNodes.indexOf(e) === -1 && this.rootNodes.push(e); }); } /** * @since 6.26.0 * Given a root asset, this method will traverse its hierarchy and add it, its children and any materials/skeletons/animation groups to the container. * @param root */ addAllAssetsToContainer(e) { if (!e) return; const t = [], r = /* @__PURE__ */ new Set(); for (t.push(e); t.length > 0; ) { const n = t.pop(); if (n instanceof Ee ? (n.geometry && this.geometries.indexOf(n.geometry) === -1 && this.geometries.push(n.geometry), this.meshes.push(n)) : n instanceof Hr ? this.transformNodes.push(n) : n instanceof ci ? this.lights.push(n) : n instanceof Tr && this.cameras.push(n), n instanceof jn) { if (n.material && this.materials.indexOf(n.material) === -1) { this.materials.push(n.material); for (const i of n.material.getActiveTextures()) this.textures.indexOf(i) === -1 && this.textures.push(i); } n.skeleton && this.skeletons.indexOf(n.skeleton) === -1 && this.skeletons.push(n.skeleton), n.morphTargetManager && this.morphTargetManagers.indexOf(n.morphTargetManager) === -1 && this.morphTargetManagers.push(n.morphTargetManager); } for (const i of n.getChildren()) r.has(i) || t.push(i); r.add(n); } this.populateRootNodes(); } } class H6e { /** * Creates a new analyser * @param scene defines hosting scene */ constructor(e) { if (this.SMOOTHING = 0.75, this.FFT_SIZE = 512, this.BARGRAPHAMPLITUDE = 256, this.DEBUGCANVASPOS = { x: 20, y: 20 }, this.DEBUGCANVASSIZE = { width: 320, height: 200 }, e = e || gr.LastCreatedScene, !!e) { if (this._scene = e, !Ge.audioEngine) { ye.Warn("No audio engine initialized, failed to create an audio analyser"); return; } this._audioEngine = Ge.audioEngine, this._audioEngine.canUseWebAudio && this._audioEngine.audioContext && (this._webAudioAnalyser = this._audioEngine.audioContext.createAnalyser(), this._webAudioAnalyser.minDecibels = -140, this._webAudioAnalyser.maxDecibels = 0, this._byteFreqs = new Uint8Array(this._webAudioAnalyser.frequencyBinCount), this._byteTime = new Uint8Array(this._webAudioAnalyser.frequencyBinCount), this._floatFreqs = new Float32Array(this._webAudioAnalyser.frequencyBinCount)); } } /** * Get the number of data values you will have to play with for the visualization * @see https://developer.mozilla.org/en-US/docs/Web/API/AnalyserNode/frequencyBinCount * @returns a number */ getFrequencyBinCount() { return this._audioEngine.canUseWebAudio ? this._webAudioAnalyser.frequencyBinCount : 0; } /** * Gets the current frequency data as a byte array * @see https://developer.mozilla.org/en-US/docs/Web/API/AnalyserNode/getByteFrequencyData * @returns a Uint8Array */ getByteFrequencyData() { return this._audioEngine.canUseWebAudio && (this._webAudioAnalyser.smoothingTimeConstant = this.SMOOTHING, this._webAudioAnalyser.fftSize = this.FFT_SIZE, this._webAudioAnalyser.getByteFrequencyData(this._byteFreqs)), this._byteFreqs; } /** * Gets the current waveform as a byte array * @see https://developer.mozilla.org/en-US/docs/Web/API/AnalyserNode/getByteTimeDomainData * @returns a Uint8Array */ getByteTimeDomainData() { return this._audioEngine.canUseWebAudio && (this._webAudioAnalyser.smoothingTimeConstant = this.SMOOTHING, this._webAudioAnalyser.fftSize = this.FFT_SIZE, this._webAudioAnalyser.getByteTimeDomainData(this._byteTime)), this._byteTime; } /** * Gets the current frequency data as a float array * @see https://developer.mozilla.org/en-US/docs/Web/API/AnalyserNode/getByteFrequencyData * @returns a Float32Array */ getFloatFrequencyData() { return this._audioEngine.canUseWebAudio && (this._webAudioAnalyser.smoothingTimeConstant = this.SMOOTHING, this._webAudioAnalyser.fftSize = this.FFT_SIZE, this._webAudioAnalyser.getFloatFrequencyData(this._floatFreqs)), this._floatFreqs; } /** * Renders the debug canvas */ drawDebugCanvas() { if (this._audioEngine.canUseWebAudio && (this._debugCanvas || (this._debugCanvas = document.createElement("canvas"), this._debugCanvas.width = this.DEBUGCANVASSIZE.width, this._debugCanvas.height = this.DEBUGCANVASSIZE.height, this._debugCanvas.style.position = "absolute", this._debugCanvas.style.top = this.DEBUGCANVASPOS.y + "px", this._debugCanvas.style.left = this.DEBUGCANVASPOS.x + "px", this._debugCanvasContext = this._debugCanvas.getContext("2d"), document.body.appendChild(this._debugCanvas), this._registerFunc = () => { this.drawDebugCanvas(); }, this._scene.registerBeforeRender(this._registerFunc)), this._registerFunc && this._debugCanvasContext)) { const e = this.getByteFrequencyData(); this._debugCanvasContext.fillStyle = "rgb(0, 0, 0)", this._debugCanvasContext.fillRect(0, 0, this.DEBUGCANVASSIZE.width, this.DEBUGCANVASSIZE.height); for (let t = 0; t < this.getFrequencyBinCount(); t++) { const n = e[t] / this.BARGRAPHAMPLITUDE, i = this.DEBUGCANVASSIZE.height * n, s = this.DEBUGCANVASSIZE.height - i - 1, a = this.DEBUGCANVASSIZE.width / this.getFrequencyBinCount(), f = t / this.getFrequencyBinCount() * 360; this._debugCanvasContext.fillStyle = "hsl(" + f + ", 100%, 50%)", this._debugCanvasContext.fillRect(t * a, s, a, i); } } } /** * Stops rendering the debug canvas and removes it */ stopDebugCanvas() { this._debugCanvas && (this._registerFunc && (this._scene.unregisterBeforeRender(this._registerFunc), this._registerFunc = null), document.body.removeChild(this._debugCanvas), this._debugCanvas = null, this._debugCanvasContext = null); } /** * Connects two audio nodes * @param inputAudioNode defines first node to connect * @param outputAudioNode defines second node to connect */ connectAudioNodes(e, t) { this._audioEngine.canUseWebAudio && (e.connect(this._webAudioAnalyser), this._webAudioAnalyser.connect(t)); } /** * Releases all associated resources */ dispose() { this._audioEngine.canUseWebAudio && this._webAudioAnalyser.disconnect(); } } Ge.AudioEngineFactory = (A, e, t) => new vee(A, e, t); class vee { /** * Gets the current AudioContext if available. */ get audioContext() { return this._audioContextInitialized || this._initializeAudioContext(), this._audioContext; } /** * Instantiates a new audio engine. * * There should be only one per page as some browsers restrict the number * of audio contexts you can create. * @param hostElement defines the host element where to display the mute icon if necessary * @param audioContext defines the audio context to be used by the audio engine * @param audioDestination defines the audio destination node to be used by audio engine */ constructor(e = null, t = null, r = null) { if (this._audioContext = null, this._audioContextInitialized = !1, this._muteButton = null, this._audioDestination = null, this.canUseWebAudio = !1, this.WarnedWebAudioUnsupported = !1, this.isMP3supported = !1, this.isOGGsupported = !1, this.unlocked = !1, this.useCustomUnlockedButton = !1, this.onAudioUnlockedObservable = new Oe(), this.onAudioLockedObservable = new Oe(), this._tryToRun = !1, this._onResize = () => { this._moveButtonToTopLeft(); }, !u9()) return; typeof window.AudioContext < "u" && (this.canUseWebAudio = !0); const n = document.createElement("audio"); this._hostElement = e, this._audioContext = t, this._audioDestination = r; try { n && n.canPlayType && (n.canPlayType('audio/mpeg; codecs="mp3"').replace(/^no$/, "") || n.canPlayType("audio/mp3").replace(/^no$/, "")) && (this.isMP3supported = !0); } catch { } try { n && n.canPlayType && n.canPlayType('audio/ogg; codecs="vorbis"').replace(/^no$/, "") && (this.isOGGsupported = !0); } catch { } } /** * Flags the audio engine in Locked state. * This happens due to new browser policies preventing audio to autoplay. */ lock() { this._triggerSuspendedState(); } /** * Unlocks the audio engine once a user action has been done on the dom. * This is helpful to resume play once browser policies have been satisfied. */ unlock() { var e, t; if (((e = this._audioContext) === null || e === void 0 ? void 0 : e.state) === "running") { this._hideMuteButton(); return; } this._tryToRun ? (t = this._audioContext) === null || t === void 0 || t.suspend().then(() => { this._tryToRun = !1, this._triggerRunningState(); }) : this._triggerRunningState(); } _resumeAudioContext() { var e; return !((e = this._audioContext) === null || e === void 0) && e.resume ? this._audioContext.resume() : Promise.resolve(); } _initializeAudioContext() { try { this.canUseWebAudio && (this._audioContext || (this._audioContext = new AudioContext()), this.masterGain = this._audioContext.createGain(), this.masterGain.gain.value = 1, this._audioDestination || (this._audioDestination = this._audioContext.destination), this.masterGain.connect(this._audioDestination), this._audioContextInitialized = !0, this._audioContext.state === "running" && this._triggerRunningState()); } catch (e) { this.canUseWebAudio = !1, Se.Error("Web Audio: " + e.message); } } _triggerRunningState() { this._tryToRun || (this._tryToRun = !0, this._resumeAudioContext().then(() => { this._tryToRun = !1, this._muteButton && this._hideMuteButton(), this.unlocked = !0, this.onAudioUnlockedObservable.notifyObservers(this); }).catch(() => { this._tryToRun = !1, this.unlocked = !1; })); } _triggerSuspendedState() { this.unlocked = !1, this.onAudioLockedObservable.notifyObservers(this), this._displayMuteButton(); } _displayMuteButton() { if (this.useCustomUnlockedButton || this._muteButton) return; this._muteButton = document.createElement("BUTTON"), this._muteButton.className = "babylonUnmuteIcon", this._muteButton.id = "babylonUnmuteIconBtn", this._muteButton.title = "Unmute"; const t = ".babylonUnmuteIcon { position: absolute; left: 20px; top: 20px; height: 40px; width: 60px; background-color: rgba(51,51,51,0.7); background-image: url(" + (window.SVGSVGElement ? "data:image/svg+xml;charset=UTF-8,%3Csvg%20version%3D%221.1%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20width%3D%2239%22%20height%3D%2232%22%20viewBox%3D%220%200%2039%2032%22%3E%3Cpath%20fill%3D%22white%22%20d%3D%22M9.625%2018.938l-0.031%200.016h-4.953q-0.016%200-0.031-0.016v-12.453q0-0.016%200.031-0.016h4.953q0.031%200%200.031%200.016v12.453zM12.125%207.688l8.719-8.703v27.453l-8.719-8.719-0.016-0.047v-9.938zM23.359%207.875l1.406-1.406%204.219%204.203%204.203-4.203%201.422%201.406-4.219%204.219%204.219%204.203-1.484%201.359-4.141-4.156-4.219%204.219-1.406-1.422%204.219-4.203z%22%3E%3C%2Fpath%3E%3C%2Fsvg%3E" : "https://cdn.babylonjs.com/Assets/audio.png") + "); background-size: 80%; background-repeat:no-repeat; background-position: center; background-position-y: 4px; border: none; outline: none; transition: transform 0.125s ease-out; cursor: pointer; z-index: 9999; } .babylonUnmuteIcon:hover { transform: scale(1.05) } .babylonUnmuteIcon:active { background-color: rgba(51,51,51,1) }", r = document.createElement("style"); r.appendChild(document.createTextNode(t)), document.getElementsByTagName("head")[0].appendChild(r), document.body.appendChild(this._muteButton), this._moveButtonToTopLeft(), this._muteButton.addEventListener("touchend", () => { this._triggerRunningState(); }, !0), this._muteButton.addEventListener("click", () => { this.unlock(); }, !0), window.addEventListener("resize", this._onResize); } _moveButtonToTopLeft() { this._hostElement && this._muteButton && (this._muteButton.style.top = this._hostElement.offsetTop + 20 + "px", this._muteButton.style.left = this._hostElement.offsetLeft + 20 + "px"); } _hideMuteButton() { this._muteButton && (document.body.removeChild(this._muteButton), this._muteButton = null); } /** * Destroy and release the resources associated with the audio context. */ dispose() { this.canUseWebAudio && this._audioContextInitialized && (this._connectedAnalyser && this._audioContext && (this._connectedAnalyser.stopDebugCanvas(), this._connectedAnalyser.dispose(), this.masterGain.disconnect(), this.masterGain.connect(this._audioContext.destination), this._connectedAnalyser = null), this.masterGain.gain.value = 1), this.WarnedWebAudioUnsupported = !1, this._hideMuteButton(), window.removeEventListener("resize", this._onResize), this.onAudioUnlockedObservable.clear(), this.onAudioLockedObservable.clear(); } /** * Gets the global volume sets on the master gain. * @returns the global volume if set or -1 otherwise */ getGlobalVolume() { return this.canUseWebAudio && this._audioContextInitialized ? this.masterGain.gain.value : -1; } /** * Sets the global volume of your experience (sets on the master gain). * @param newVolume Defines the new global volume of the application */ setGlobalVolume(e) { this.canUseWebAudio && this._audioContextInitialized && (this.masterGain.gain.value = e); } /** * Connect the audio engine to an audio analyser allowing some amazing * synchronization between the sounds/music and your visualization (VuMeter for instance). * @see https://doc.babylonjs.com/features/featuresDeepDive/audio/playingSoundsMusic#using-the-analyser * @param analyser The analyser to connect to the engine */ connectToAnalyser(e) { this._connectedAnalyser && this._connectedAnalyser.stopDebugCanvas(), this.canUseWebAudio && this._audioContextInitialized && this._audioContext && (this._connectedAnalyser = e, this.masterGain.disconnect(), this._connectedAnalyser.connectAudioNodes(this.masterGain, this._audioContext.destination)); } } class qc { /** * Does the sound loop after it finishes playing once. */ get loop() { return this._loop; } set loop(e) { e !== this._loop && (this._loop = e, this.updateOptions({ loop: e })); } /** * Gets the current time for the sound. */ get currentTime() { var e; if (this._htmlAudioElement) return this._htmlAudioElement.currentTime; if (!((e = Ge.audioEngine) === null || e === void 0) && e.audioContext && (this.isPlaying || this.isPaused)) { const t = this.isPaused ? 0 : Ge.audioEngine.audioContext.currentTime - this._startTime; return this._currentTime + t; } return 0; } /** * Does this sound enables spatial sound. * @see https://doc.babylonjs.com/features/featuresDeepDive/audio/playingSoundsMusic#creating-a-spatial-3d-sound */ get spatialSound() { return this._spatialSound; } /** * Does this sound enables spatial sound. * @see https://doc.babylonjs.com/features/featuresDeepDive/audio/playingSoundsMusic#creating-a-spatial-3d-sound */ set spatialSound(e) { if (e == this._spatialSound) return; const t = this.isPlaying; this.pause(), e ? (this._spatialSound = e, this._updateSpatialParameters()) : this._disableSpatialSound(), t && this.play(); } /** * Create a sound and attach it to a scene * @param name Name of your sound * @param urlOrArrayBuffer Url to the sound to load async or ArrayBuffer, it also works with MediaStreams and AudioBuffers * @param scene defines the scene the sound belongs to * @param readyToPlayCallback Provide a callback function if you'd like to load your code once the sound is ready to be played * @param options Objects to provide with the current available options: autoplay, loop, volume, spatialSound, maxDistance, rolloffFactor, refDistance, distanceModel, panningModel, streaming */ constructor(e, t, r, n = null, i) { var s, a, f, o, d; if (this.autoplay = !1, this._loop = !1, this.useCustomAttenuation = !1, this.isPlaying = !1, this.isPaused = !1, this.refDistance = 1, this.rolloffFactor = 1, this.maxDistance = 100, this.distanceModel = "linear", this.metadata = null, this.onEndedObservable = new Oe(), this._spatialSound = !1, this._panningModel = "equalpower", this._playbackRate = 1, this._streaming = !1, this._startTime = 0, this._currentTime = 0, this._position = S.Zero(), this._localDirection = new S(1, 0, 0), this._volume = 1, this._isReadyToPlay = !1, this._isDirectional = !1, this._coneInnerAngle = 360, this._coneOuterAngle = 360, this._coneOuterGain = 0, this._isOutputConnected = !1, this._urlType = "Unknown", this.name = e, r = r || gr.LastCreatedScene, !!r) if (this._scene = r, qc._SceneComponentInitialization(r), this._readyToPlayCallback = n, this._customAttenuationFunction = (v, u, l, P, p) => u < l ? v * (1 - u / l) : 0, i && (this.autoplay = i.autoplay || !1, this._loop = i.loop || !1, i.volume !== void 0 && (this._volume = i.volume), this._spatialSound = (s = i.spatialSound) !== null && s !== void 0 ? s : !1, this.maxDistance = (a = i.maxDistance) !== null && a !== void 0 ? a : 100, this.useCustomAttenuation = (f = i.useCustomAttenuation) !== null && f !== void 0 ? f : !1, this.rolloffFactor = i.rolloffFactor || 1, this.refDistance = i.refDistance || 1, this.distanceModel = i.distanceModel || "linear", this._playbackRate = i.playbackRate || 1, this._streaming = (o = i.streaming) !== null && o !== void 0 ? o : !1, this._length = i.length, this._offset = i.offset), !((d = Ge.audioEngine) === null || d === void 0) && d.canUseWebAudio && Ge.audioEngine.audioContext) { this._soundGain = Ge.audioEngine.audioContext.createGain(), this._soundGain.gain.value = this._volume, this._inputAudioNode = this._soundGain, this._outputAudioNode = this._soundGain, this._spatialSound && this._createSpatialParameters(), this._scene.mainSoundTrack.addSound(this); let v = !0; if (t) try { typeof t == "string" ? (this._urlType = "String", this._url = t) : t instanceof ArrayBuffer ? this._urlType = "ArrayBuffer" : t instanceof HTMLMediaElement ? this._urlType = "MediaElement" : t instanceof MediaStream ? this._urlType = "MediaStream" : t instanceof AudioBuffer ? this._urlType = "AudioBuffer" : Array.isArray(t) && (this._urlType = "Array"); let u = [], l = !1; switch (this._urlType) { case "MediaElement": this._streaming = !0, this._isReadyToPlay = !0, this._streamingSource = Ge.audioEngine.audioContext.createMediaElementSource(t), this.autoplay && this.play(0, this._offset, this._length), this._readyToPlayCallback && this._readyToPlayCallback(); break; case "MediaStream": this._streaming = !0, this._isReadyToPlay = !0, this._streamingSource = Ge.audioEngine.audioContext.createMediaStreamSource(t), this.autoplay && this.play(0, this._offset, this._length), this._readyToPlayCallback && this._readyToPlayCallback(); break; case "ArrayBuffer": t.byteLength > 0 && (l = !0, this._soundLoaded(t)); break; case "AudioBuffer": this._audioBufferLoaded(t); break; case "String": u.push(t); case "Array": u.length === 0 && (u = t); for (let P = 0; P < u.length; P++) { const p = u[P]; if (l = i && i.skipCodecCheck || p.indexOf(".mp3", p.length - 4) !== -1 && Ge.audioEngine.isMP3supported || p.indexOf(".ogg", p.length - 4) !== -1 && Ge.audioEngine.isOGGsupported || p.indexOf(".wav", p.length - 4) !== -1 || p.indexOf(".m4a", p.length - 4) !== -1 || p.indexOf(".mp4", p.length - 4) !== -1 || p.indexOf("blob:") !== -1, l) { this._streaming ? (this._htmlAudioElement = new Audio(p), this._htmlAudioElement.controls = !1, this._htmlAudioElement.loop = this.loop, ye.SetCorsBehavior(p, this._htmlAudioElement), this._htmlAudioElement.preload = "auto", this._htmlAudioElement.addEventListener("canplaythrough", () => { this._isReadyToPlay = !0, this.autoplay && this.play(0, this._offset, this._length), this._readyToPlayCallback && this._readyToPlayCallback(); }), document.body.appendChild(this._htmlAudioElement), this._htmlAudioElement.load()) : this._scene._loadFile(p, (c) => { this._soundLoaded(c); }, void 0, !0, !0, (c) => { c && Se.Error("XHR " + c.status + " error on: " + p + "."), Se.Error("Sound creation aborted."), this._scene.mainSoundTrack.removeSound(this); }); break; } } break; default: v = !1; break; } v ? l || (this._isReadyToPlay = !0, this._readyToPlayCallback && setTimeout(() => { this._readyToPlayCallback && this._readyToPlayCallback(); }, 1e3)) : Se.Error("Parameter must be a URL to the sound, an Array of URLs (.mp3 & .ogg) or an ArrayBuffer of the sound."); } catch { Se.Error("Unexpected error. Sound creation aborted."), this._scene.mainSoundTrack.removeSound(this); } } else this._scene.mainSoundTrack.addSound(this), Ge.audioEngine && !Ge.audioEngine.WarnedWebAudioUnsupported && (Se.Error("Web Audio is not supported by your browser."), Ge.audioEngine.WarnedWebAudioUnsupported = !0), this._readyToPlayCallback && setTimeout(() => { this._readyToPlayCallback && this._readyToPlayCallback(); }, 1e3); } /** * Release the sound and its associated resources */ dispose() { var e; !((e = Ge.audioEngine) === null || e === void 0) && e.canUseWebAudio && (this.isPlaying && this.stop(), this._isReadyToPlay = !1, this.soundTrackId === -1 ? this._scene.mainSoundTrack.removeSound(this) : this._scene.soundTracks && this._scene.soundTracks[this.soundTrackId].removeSound(this), this._soundGain && (this._soundGain.disconnect(), this._soundGain = null), this._soundPanner && (this._soundPanner.disconnect(), this._soundPanner = null), this._soundSource && (this._soundSource.disconnect(), this._soundSource = null), this._audioBuffer = null, this._htmlAudioElement && (this._htmlAudioElement.pause(), this._htmlAudioElement.src = "", document.body.removeChild(this._htmlAudioElement)), this._streamingSource && this._streamingSource.disconnect(), this._connectedTransformNode && this._registerFunc && (this._connectedTransformNode.unregisterAfterWorldMatrixUpdate(this._registerFunc), this._connectedTransformNode = null), this._clearTimeoutsAndObservers()); } /** * Gets if the sounds is ready to be played or not. * @returns true if ready, otherwise false */ isReady() { return this._isReadyToPlay; } /** * Get the current class name. * @returns current class name */ getClassName() { return "Sound"; } _audioBufferLoaded(e) { var t; !((t = Ge.audioEngine) === null || t === void 0) && t.audioContext && (this._audioBuffer = e, this._isReadyToPlay = !0, this.autoplay && this.play(0, this._offset, this._length), this._readyToPlayCallback && this._readyToPlayCallback()); } _soundLoaded(e) { var t; !((t = Ge.audioEngine) === null || t === void 0) && t.audioContext && Ge.audioEngine.audioContext.decodeAudioData(e, (r) => { this._audioBufferLoaded(r); }, (r) => { Se.Error("Error while decoding audio data for: " + this.name + " / Error: " + r); }); } /** * Sets the data of the sound from an audiobuffer * @param audioBuffer The audioBuffer containing the data */ setAudioBuffer(e) { var t; !((t = Ge.audioEngine) === null || t === void 0) && t.canUseWebAudio && (this._audioBuffer = e, this._isReadyToPlay = !0); } /** * Updates the current sounds options such as maxdistance, loop... * @param options A JSON object containing values named as the object properties */ updateOptions(e) { var t, r, n, i, s, a, f, o, d, v, u; e && (this.loop = (t = e.loop) !== null && t !== void 0 ? t : this.loop, this.maxDistance = (r = e.maxDistance) !== null && r !== void 0 ? r : this.maxDistance, this.useCustomAttenuation = (n = e.useCustomAttenuation) !== null && n !== void 0 ? n : this.useCustomAttenuation, this.rolloffFactor = (i = e.rolloffFactor) !== null && i !== void 0 ? i : this.rolloffFactor, this.refDistance = (s = e.refDistance) !== null && s !== void 0 ? s : this.refDistance, this.distanceModel = (a = e.distanceModel) !== null && a !== void 0 ? a : this.distanceModel, this._playbackRate = (f = e.playbackRate) !== null && f !== void 0 ? f : this._playbackRate, this._length = (o = e.length) !== null && o !== void 0 ? o : void 0, this.spatialSound = (d = e.spatialSound) !== null && d !== void 0 ? d : this._spatialSound, this._setOffset((v = e.offset) !== null && v !== void 0 ? v : void 0), this.setVolume((u = e.volume) !== null && u !== void 0 ? u : this._volume), this._updateSpatialParameters(), this.isPlaying && (this._streaming && this._htmlAudioElement ? (this._htmlAudioElement.playbackRate = this._playbackRate, this._htmlAudioElement.loop !== this.loop && (this._htmlAudioElement.loop = this.loop)) : this._soundSource && (this._soundSource.playbackRate.value = this._playbackRate, this._soundSource.loop !== this.loop && (this._soundSource.loop = this.loop), this._offset !== void 0 && this._soundSource.loopStart !== this._offset && (this._soundSource.loopStart = this._offset), this._length !== void 0 && this._length !== this._soundSource.loopEnd && (this._soundSource.loopEnd = (this._offset | 0) + this._length)))); } _createSpatialParameters() { var e, t; !((e = Ge.audioEngine) === null || e === void 0) && e.canUseWebAudio && Ge.audioEngine.audioContext && (this._scene.headphone && (this._panningModel = "HRTF"), this._soundPanner = (t = this._soundPanner) !== null && t !== void 0 ? t : Ge.audioEngine.audioContext.createPanner(), this._soundPanner && this._outputAudioNode && (this._updateSpatialParameters(), this._soundPanner.connect(this._outputAudioNode), this._inputAudioNode = this._soundPanner)); } _disableSpatialSound() { var e; this._spatialSound && (this._inputAudioNode = this._soundGain, (e = this._soundPanner) === null || e === void 0 || e.disconnect(), this._soundPanner = null, this._spatialSound = !1); } _updateSpatialParameters() { this._spatialSound && (this._soundPanner ? this.useCustomAttenuation ? (this._soundPanner.distanceModel = "linear", this._soundPanner.maxDistance = Number.MAX_VALUE, this._soundPanner.refDistance = 1, this._soundPanner.rolloffFactor = 1, this._soundPanner.panningModel = this._panningModel) : (this._soundPanner.distanceModel = this.distanceModel, this._soundPanner.maxDistance = this.maxDistance, this._soundPanner.refDistance = this.refDistance, this._soundPanner.rolloffFactor = this.rolloffFactor, this._soundPanner.panningModel = this._panningModel) : this._createSpatialParameters()); } /** * Switch the panning model to HRTF: * Renders a stereo output of higher quality than equalpower — it uses a convolution with measured impulse responses from human subjects. * @see https://doc.babylonjs.com/features/featuresDeepDive/audio/playingSoundsMusic#creating-a-spatial-3d-sound */ switchPanningModelToHRTF() { this._panningModel = "HRTF", this._switchPanningModel(); } /** * Switch the panning model to Equal Power: * Represents the equal-power panning algorithm, generally regarded as simple and efficient. equalpower is the default value. * @see https://doc.babylonjs.com/features/featuresDeepDive/audio/playingSoundsMusic#creating-a-spatial-3d-sound */ switchPanningModelToEqualPower() { this._panningModel = "equalpower", this._switchPanningModel(); } _switchPanningModel() { var e; !((e = Ge.audioEngine) === null || e === void 0) && e.canUseWebAudio && this._spatialSound && this._soundPanner && (this._soundPanner.panningModel = this._panningModel); } /** * Connect this sound to a sound track audio node like gain... * @param soundTrackAudioNode the sound track audio node to connect to */ connectToSoundTrackAudioNode(e) { var t; !((t = Ge.audioEngine) === null || t === void 0) && t.canUseWebAudio && this._outputAudioNode && (this._isOutputConnected && this._outputAudioNode.disconnect(), this._outputAudioNode.connect(e), this._isOutputConnected = !0); } /** * Transform this sound into a directional source * @param coneInnerAngle Size of the inner cone in degree * @param coneOuterAngle Size of the outer cone in degree * @param coneOuterGain Volume of the sound outside the outer cone (between 0.0 and 1.0) */ setDirectionalCone(e, t, r) { if (t < e) { Se.Error("setDirectionalCone(): outer angle of the cone must be superior or equal to the inner angle."); return; } this._coneInnerAngle = e, this._coneOuterAngle = t, this._coneOuterGain = r, this._isDirectional = !0, this.isPlaying && this.loop && (this.stop(), this.play(0, this._offset, this._length)); } /** * Gets or sets the inner angle for the directional cone. */ get directionalConeInnerAngle() { return this._coneInnerAngle; } /** * Gets or sets the inner angle for the directional cone. */ set directionalConeInnerAngle(e) { var t; if (e != this._coneInnerAngle) { if (this._coneOuterAngle < e) { Se.Error("directionalConeInnerAngle: outer angle of the cone must be superior or equal to the inner angle."); return; } this._coneInnerAngle = e, !((t = Ge.audioEngine) === null || t === void 0) && t.canUseWebAudio && this._spatialSound && this._soundPanner && (this._soundPanner.coneInnerAngle = this._coneInnerAngle); } } /** * Gets or sets the outer angle for the directional cone. */ get directionalConeOuterAngle() { return this._coneOuterAngle; } /** * Gets or sets the outer angle for the directional cone. */ set directionalConeOuterAngle(e) { var t; if (e != this._coneOuterAngle) { if (e < this._coneInnerAngle) { Se.Error("directionalConeOuterAngle: outer angle of the cone must be superior or equal to the inner angle."); return; } this._coneOuterAngle = e, !((t = Ge.audioEngine) === null || t === void 0) && t.canUseWebAudio && this._spatialSound && this._soundPanner && (this._soundPanner.coneOuterAngle = this._coneOuterAngle); } } /** * Sets the position of the emitter if spatial sound is enabled * @param newPosition Defines the new position */ setPosition(e) { var t; e.equals(this._position) || (this._position.copyFrom(e), !((t = Ge.audioEngine) === null || t === void 0) && t.canUseWebAudio && this._spatialSound && this._soundPanner && !isNaN(this._position.x) && !isNaN(this._position.y) && !isNaN(this._position.z) && (this._soundPanner.positionX.value = this._position.x, this._soundPanner.positionY.value = this._position.y, this._soundPanner.positionZ.value = this._position.z)); } /** * Sets the local direction of the emitter if spatial sound is enabled * @param newLocalDirection Defines the new local direction */ setLocalDirectionToMesh(e) { var t; this._localDirection = e, !((t = Ge.audioEngine) === null || t === void 0) && t.canUseWebAudio && this._connectedTransformNode && this.isPlaying && this._updateDirection(); } _updateDirection() { if (!this._connectedTransformNode || !this._soundPanner) return; const e = this._connectedTransformNode.getWorldMatrix(), t = S.TransformNormal(this._localDirection, e); t.normalize(), this._soundPanner.orientationX.value = t.x, this._soundPanner.orientationY.value = t.y, this._soundPanner.orientationZ.value = t.z; } /** @internal */ updateDistanceFromListener() { var e; if (!((e = Ge.audioEngine) === null || e === void 0) && e.canUseWebAudio && this._connectedTransformNode && this.useCustomAttenuation && this._soundGain && this._scene.activeCamera) { const t = this._scene.audioListenerPositionProvider ? this._connectedTransformNode.position.subtract(this._scene.audioListenerPositionProvider()).length() : this._connectedTransformNode.getDistanceToCamera(this._scene.activeCamera); this._soundGain.gain.value = this._customAttenuationFunction(this._volume, t, this.maxDistance, this.refDistance, this.rolloffFactor); } } /** * Sets a new custom attenuation function for the sound. * @param callback Defines the function used for the attenuation * @see https://doc.babylonjs.com/features/featuresDeepDive/audio/playingSoundsMusic#creating-your-own-custom-attenuation-function */ setAttenuationFunction(e) { this._customAttenuationFunction = e; } /** * Play the sound * @param time (optional) Start the sound after X seconds. Start immediately (0) by default. * @param offset (optional) Start the sound at a specific time in seconds * @param length (optional) Sound duration (in seconds) */ play(e, t, r) { var n, i, s, a; if (this._isReadyToPlay && this._scene.audioEnabled && (!((n = Ge.audioEngine) === null || n === void 0) && n.audioContext)) try { this._clearTimeoutsAndObservers(); let f = e ? ((i = Ge.audioEngine) === null || i === void 0 ? void 0 : i.audioContext.currentTime) + e : (s = Ge.audioEngine) === null || s === void 0 ? void 0 : s.audioContext.currentTime; if ((!this._soundSource || !this._streamingSource) && this._spatialSound && this._soundPanner && (!isNaN(this._position.x) && !isNaN(this._position.y) && !isNaN(this._position.z) && (this._soundPanner.positionX.value = this._position.x, this._soundPanner.positionY.value = this._position.y, this._soundPanner.positionZ.value = this._position.z), this._isDirectional && (this._soundPanner.coneInnerAngle = this._coneInnerAngle, this._soundPanner.coneOuterAngle = this._coneOuterAngle, this._soundPanner.coneOuterGain = this._coneOuterGain, this._connectedTransformNode ? this._updateDirection() : this._soundPanner.setOrientation(this._localDirection.x, this._localDirection.y, this._localDirection.z))), this._streaming) { if (this._streamingSource || (this._streamingSource = Ge.audioEngine.audioContext.createMediaElementSource(this._htmlAudioElement), this._htmlAudioElement.onended = () => { this._onended(); }, this._htmlAudioElement.playbackRate = this._playbackRate), this._streamingSource.disconnect(), this._inputAudioNode && this._streamingSource.connect(this._inputAudioNode), this._htmlAudioElement) { const o = () => { var d, v; if (!((d = Ge.audioEngine) === null || d === void 0) && d.unlocked) { const u = this._htmlAudioElement.play(); u !== void 0 && u.catch(() => { var l, P; (l = Ge.audioEngine) === null || l === void 0 || l.lock(), (this.loop || this.autoplay) && (this._audioUnlockedObserver = (P = Ge.audioEngine) === null || P === void 0 ? void 0 : P.onAudioUnlockedObservable.addOnce(() => { o(); })); }); } else (this.loop || this.autoplay) && (this._audioUnlockedObserver = (v = Ge.audioEngine) === null || v === void 0 ? void 0 : v.onAudioUnlockedObservable.addOnce(() => { o(); })); }; o(); } } else { const o = () => { var d, v, u, l; if (!((d = Ge.audioEngine) === null || d === void 0) && d.audioContext) { if (r = r || this._length, t !== void 0 && this._setOffset(t), this._soundSource) { const P = this._soundSource; P.onended = () => { P.disconnect(); }; } if (this._soundSource = (v = Ge.audioEngine) === null || v === void 0 ? void 0 : v.audioContext.createBufferSource(), this._soundSource && this._inputAudioNode) { this._soundSource.buffer = this._audioBuffer, this._soundSource.connect(this._inputAudioNode), this._soundSource.loop = this.loop, t !== void 0 && (this._soundSource.loopStart = t), r !== void 0 && (this._soundSource.loopEnd = (t | 0) + r), this._soundSource.playbackRate.value = this._playbackRate, this._soundSource.onended = () => { this._onended(); }, f = e ? ((u = Ge.audioEngine) === null || u === void 0 ? void 0 : u.audioContext.currentTime) + e : Ge.audioEngine.audioContext.currentTime; const P = ((this.isPaused ? this.currentTime : 0) + ((l = this._offset) !== null && l !== void 0 ? l : 0)) % this._soundSource.buffer.duration; this._soundSource.start(f, P, this.loop ? void 0 : r); } } }; ((a = Ge.audioEngine) === null || a === void 0 ? void 0 : a.audioContext.state) === "suspended" ? this._tryToPlayTimeout = setTimeout(() => { var d; ((d = Ge.audioEngine) === null || d === void 0 ? void 0 : d.audioContext.state) === "suspended" ? (Ge.audioEngine.lock(), (this.loop || this.autoplay) && (this._audioUnlockedObserver = Ge.audioEngine.onAudioUnlockedObservable.addOnce(() => { o(); }))) : o(); }, 500) : o(); } this._startTime = f, this.isPlaying = !0, this.isPaused = !1; } catch (f) { Se.Error("Error while trying to play audio: " + this.name + ", " + f.message); } } _onended() { this.isPlaying = !1, this._startTime = 0, this._currentTime = 0, this.onended && this.onended(), this.onEndedObservable.notifyObservers(this); } /** * Stop the sound * @param time (optional) Stop the sound after X seconds. Stop immediately (0) by default. */ stop(e) { var t; if (this.isPlaying) if (this._clearTimeoutsAndObservers(), this._streaming) this._htmlAudioElement ? (this._htmlAudioElement.pause(), this._htmlAudioElement.currentTime > 0 && (this._htmlAudioElement.currentTime = 0)) : this._streamingSource.disconnect(), this.isPlaying = !1; else if (!((t = Ge.audioEngine) === null || t === void 0) && t.audioContext && this._soundSource) { const r = e ? Ge.audioEngine.audioContext.currentTime + e : void 0; this._soundSource.onended = () => { this.isPlaying = !1, this.isPaused = !1, this._startTime = 0, this._currentTime = 0, this._soundSource && (this._soundSource.onended = () => { }), this._onended(); }, this._soundSource.stop(r); } else this.isPlaying = !1; else this.isPaused && (this.isPaused = !1, this._startTime = 0, this._currentTime = 0); } /** * Put the sound in pause */ pause() { var e; this.isPlaying && (this._clearTimeoutsAndObservers(), this._streaming ? (this._htmlAudioElement ? this._htmlAudioElement.pause() : this._streamingSource.disconnect(), this.isPlaying = !1, this.isPaused = !0) : !((e = Ge.audioEngine) === null || e === void 0) && e.audioContext && this._soundSource && (this._soundSource.onended = () => { }, this._soundSource.stop(), this.isPlaying = !1, this.isPaused = !0, this._currentTime += Ge.audioEngine.audioContext.currentTime - this._startTime)); } /** * Sets a dedicated volume for this sounds * @param newVolume Define the new volume of the sound * @param time Define time for gradual change to new volume */ setVolume(e, t) { var r; !((r = Ge.audioEngine) === null || r === void 0) && r.canUseWebAudio && this._soundGain && (t && Ge.audioEngine.audioContext ? (this._soundGain.gain.cancelScheduledValues(Ge.audioEngine.audioContext.currentTime), this._soundGain.gain.setValueAtTime(this._soundGain.gain.value, Ge.audioEngine.audioContext.currentTime), this._soundGain.gain.linearRampToValueAtTime(e, Ge.audioEngine.audioContext.currentTime + t)) : this._soundGain.gain.value = e), this._volume = e; } /** * Set the sound play back rate * @param newPlaybackRate Define the playback rate the sound should be played at */ setPlaybackRate(e) { this._playbackRate = e, this.isPlaying && (this._streaming && this._htmlAudioElement ? this._htmlAudioElement.playbackRate = this._playbackRate : this._soundSource && (this._soundSource.playbackRate.value = this._playbackRate)); } /** * Gets the sound play back rate. * @returns the play back rate of the sound */ getPlaybackRate() { return this._playbackRate; } /** * Gets the volume of the sound. * @returns the volume of the sound */ getVolume() { return this._volume; } /** * Attach the sound to a dedicated mesh * @param transformNode The transform node to connect the sound with * @see https://doc.babylonjs.com/features/featuresDeepDive/audio/playingSoundsMusic#attaching-a-sound-to-a-mesh */ attachToMesh(e) { this._connectedTransformNode && this._registerFunc && (this._connectedTransformNode.unregisterAfterWorldMatrixUpdate(this._registerFunc), this._registerFunc = null), this._connectedTransformNode = e, this._spatialSound || (this._spatialSound = !0, this._createSpatialParameters(), this.isPlaying && this.loop && (this.stop(), this.play(0, this._offset, this._length))), this._onRegisterAfterWorldMatrixUpdate(this._connectedTransformNode), this._registerFunc = (t) => this._onRegisterAfterWorldMatrixUpdate(t), this._connectedTransformNode.registerAfterWorldMatrixUpdate(this._registerFunc); } /** * Detach the sound from the previously attached mesh * @see https://doc.babylonjs.com/features/featuresDeepDive/audio/playingSoundsMusic#attaching-a-sound-to-a-mesh */ detachFromMesh() { this._connectedTransformNode && this._registerFunc && (this._connectedTransformNode.unregisterAfterWorldMatrixUpdate(this._registerFunc), this._registerFunc = null, this._connectedTransformNode = null); } _onRegisterAfterWorldMatrixUpdate(e) { var t; if (!e.getBoundingInfo) this.setPosition(e.absolutePosition); else { const n = e.getBoundingInfo(); this.setPosition(n.boundingSphere.centerWorld); } !((t = Ge.audioEngine) === null || t === void 0) && t.canUseWebAudio && this._isDirectional && this.isPlaying && this._updateDirection(); } /** * Clone the current sound in the scene. * @returns the new sound clone */ clone() { if (this._streaming) return null; { const e = () => { this._isReadyToPlay ? (r._audioBuffer = this.getAudioBuffer(), r._isReadyToPlay = !0, r.autoplay && r.play(0, this._offset, this._length)) : setTimeout(e, 300); }, t = { autoplay: this.autoplay, loop: this.loop, volume: this._volume, spatialSound: this._spatialSound, maxDistance: this.maxDistance, useCustomAttenuation: this.useCustomAttenuation, rolloffFactor: this.rolloffFactor, refDistance: this.refDistance, distanceModel: this.distanceModel }, r = new qc(this.name + "_cloned", new ArrayBuffer(0), this._scene, null, t); return this.useCustomAttenuation && r.setAttenuationFunction(this._customAttenuationFunction), r.setPosition(this._position), r.setPlaybackRate(this._playbackRate), e(), r; } } /** * Gets the current underlying audio buffer containing the data * @returns the audio buffer */ getAudioBuffer() { return this._audioBuffer; } /** * Gets the WebAudio AudioBufferSourceNode, lets you keep track of and stop instances of this Sound. * @returns the source node */ getSoundSource() { return this._soundSource; } /** * Gets the WebAudio GainNode, gives you precise control over the gain of instances of this Sound. * @returns the gain node */ getSoundGain() { return this._soundGain; } /** * Serializes the Sound in a JSON representation * @returns the JSON representation of the sound */ serialize() { const e = { name: this.name, url: this._url, autoplay: this.autoplay, loop: this.loop, volume: this._volume, spatialSound: this._spatialSound, maxDistance: this.maxDistance, rolloffFactor: this.rolloffFactor, refDistance: this.refDistance, distanceModel: this.distanceModel, playbackRate: this._playbackRate, panningModel: this._panningModel, soundTrackId: this.soundTrackId, metadata: this.metadata }; return this._spatialSound && (this._connectedTransformNode && (e.connectedMeshId = this._connectedTransformNode.id), e.position = this._position.asArray(), e.refDistance = this.refDistance, e.distanceModel = this.distanceModel, e.isDirectional = this._isDirectional, e.localDirectionToMesh = this._localDirection.asArray(), e.coneInnerAngle = this._coneInnerAngle, e.coneOuterAngle = this._coneOuterAngle, e.coneOuterGain = this._coneOuterGain), e; } /** * Parse a JSON representation of a sound to instantiate in a given scene * @param parsedSound Define the JSON representation of the sound (usually coming from the serialize method) * @param scene Define the scene the new parsed sound should be created in * @param rootUrl Define the rooturl of the load in case we need to fetch relative dependencies * @param sourceSound Define a sound place holder if do not need to instantiate a new one * @returns the newly parsed sound */ static Parse(e, t, r, n) { const i = e.name; let s; e.url ? s = r + e.url : s = r + i; const a = { autoplay: e.autoplay, loop: e.loop, volume: e.volume, spatialSound: e.spatialSound, maxDistance: e.maxDistance, rolloffFactor: e.rolloffFactor, refDistance: e.refDistance, distanceModel: e.distanceModel, playbackRate: e.playbackRate }; let f; if (!n) f = new qc(i, s, t, () => { t.removePendingData(f); }, a), t.addPendingData(f); else { const o = () => { n._isReadyToPlay ? (f._audioBuffer = n.getAudioBuffer(), f._isReadyToPlay = !0, f.autoplay && f.play(0, f._offset, f._length)) : setTimeout(o, 300); }; f = new qc(i, new ArrayBuffer(0), t, null, a), o(); } if (e.position) { const o = S.FromArray(e.position); f.setPosition(o); } if (e.isDirectional && (f.setDirectionalCone(e.coneInnerAngle || 360, e.coneOuterAngle || 360, e.coneOuterGain || 0), e.localDirectionToMesh)) { const o = S.FromArray(e.localDirectionToMesh); f.setLocalDirectionToMesh(o); } if (e.connectedMeshId) { const o = t.getMeshById(e.connectedMeshId); o && f.attachToMesh(o); } return e.metadata && (f.metadata = e.metadata), f; } _setOffset(e) { this._offset !== e && (this.isPaused && (this.stop(), this.isPaused = !1), this._offset = e); } _clearTimeoutsAndObservers() { var e; this._tryToPlayTimeout && (clearTimeout(this._tryToPlayTimeout), this._tryToPlayTimeout = null), this._audioUnlockedObserver && ((e = Ge.audioEngine) === null || e === void 0 || e.onAudioUnlockedObservable.remove(this._audioUnlockedObserver), this._audioUnlockedObserver = null); } } qc._SceneComponentInitialization = (A) => { throw qn("AudioSceneComponent"); }; class uee { /** * Creates a new sound track. * @see https://doc.babylonjs.com/features/featuresDeepDive/audio/playingSoundsMusic#using-sound-tracks * @param scene Define the scene the sound track belongs to * @param options */ constructor(e, t = {}) { this.id = -1, this._isInitialized = !1, e = e || gr.LastCreatedScene, e && (this._scene = e, this.soundCollection = [], this._options = t, !this._options.mainTrack && this._scene.soundTracks && (this._scene.soundTracks.push(this), this.id = this._scene.soundTracks.length - 1)); } _initializeSoundTrackAudioGraph() { var e; !((e = Ge.audioEngine) === null || e === void 0) && e.canUseWebAudio && Ge.audioEngine.audioContext && (this._outputAudioNode = Ge.audioEngine.audioContext.createGain(), this._outputAudioNode.connect(Ge.audioEngine.masterGain), this._options && this._options.volume && (this._outputAudioNode.gain.value = this._options.volume), this._isInitialized = !0); } /** * Release the sound track and its associated resources */ dispose() { if (Ge.audioEngine && Ge.audioEngine.canUseWebAudio) { for (this._connectedAnalyser && this._connectedAnalyser.stopDebugCanvas(); this.soundCollection.length; ) this.soundCollection[0].dispose(); this._outputAudioNode && this._outputAudioNode.disconnect(), this._outputAudioNode = null; } } /** * Adds a sound to this sound track * @param sound define the sound to add * @ignoreNaming */ addSound(e) { var t; this._isInitialized || this._initializeSoundTrackAudioGraph(), !((t = Ge.audioEngine) === null || t === void 0) && t.canUseWebAudio && this._outputAudioNode && e.connectToSoundTrackAudioNode(this._outputAudioNode), e.soundTrackId !== void 0 && (e.soundTrackId === -1 ? this._scene.mainSoundTrack.removeSound(e) : this._scene.soundTracks && this._scene.soundTracks[e.soundTrackId].removeSound(e)), this.soundCollection.push(e), e.soundTrackId = this.id; } /** * Removes a sound to this sound track * @param sound define the sound to remove * @ignoreNaming */ removeSound(e) { const t = this.soundCollection.indexOf(e); t !== -1 && this.soundCollection.splice(t, 1); } /** * Set a global volume for the full sound track. * @param newVolume Define the new volume of the sound track */ setVolume(e) { var t; !((t = Ge.audioEngine) === null || t === void 0) && t.canUseWebAudio && this._outputAudioNode && (this._outputAudioNode.gain.value = e); } /** * Switch the panning model to HRTF: * Renders a stereo output of higher quality than equalpower — it uses a convolution with measured impulse responses from human subjects. * @see https://doc.babylonjs.com/features/featuresDeepDive/audio/playingSoundsMusic#creating-a-spatial-3d-sound */ switchPanningModelToHRTF() { var e; if (!((e = Ge.audioEngine) === null || e === void 0) && e.canUseWebAudio) for (let t = 0; t < this.soundCollection.length; t++) this.soundCollection[t].switchPanningModelToHRTF(); } /** * Switch the panning model to Equal Power: * Represents the equal-power panning algorithm, generally regarded as simple and efficient. equalpower is the default value. * @see https://doc.babylonjs.com/features/featuresDeepDive/audio/playingSoundsMusic#creating-a-spatial-3d-sound */ switchPanningModelToEqualPower() { var e; if (!((e = Ge.audioEngine) === null || e === void 0) && e.canUseWebAudio) for (let t = 0; t < this.soundCollection.length; t++) this.soundCollection[t].switchPanningModelToEqualPower(); } /** * Connect the sound track to an audio analyser allowing some amazing * synchronization between the sounds/music and your visualization (VuMeter for instance). * @see https://doc.babylonjs.com/features/featuresDeepDive/audio/playingSoundsMusic#using-the-analyser * @param analyser The analyser to connect to the engine */ connectToAnalyser(e) { var t; this._connectedAnalyser && this._connectedAnalyser.stopDebugCanvas(), this._connectedAnalyser = e, !((t = Ge.audioEngine) === null || t === void 0) && t.canUseWebAudio && this._outputAudioNode && (this._outputAudioNode.disconnect(), this._connectedAnalyser.connectAudioNodes(this._outputAudioNode, Ge.audioEngine.masterGain)); } } J1.AddParser(Ot.NAME_AUDIO, (A, e, t, r) => { var n; let i = [], s; if (t.sounds = t.sounds || [], A.sounds !== void 0 && A.sounds !== null) for (let a = 0, f = A.sounds.length; a < f; a++) { const o = A.sounds[a]; !((n = Ge.audioEngine) === null || n === void 0) && n.canUseWebAudio ? (o.url || (o.url = o.name), i[o.url] ? t.sounds.push(qc.Parse(o, e, r, i[o.url])) : (s = qc.Parse(o, e, r), i[o.url] = s, t.sounds.push(s))) : t.sounds.push(new qc(o.name, null, e)); } i = []; }); Object.defineProperty(sr.prototype, "mainSoundTrack", { get: function() { let A = this._getComponent(Ot.NAME_AUDIO); return A || (A = new yu(this), this._addComponent(A)), this._mainSoundTrack || (this._mainSoundTrack = new uee(this, { mainTrack: !0 })), this._mainSoundTrack; }, enumerable: !0, configurable: !0 }); sr.prototype.getSoundByName = function(A) { let e; for (e = 0; e < this.mainSoundTrack.soundCollection.length; e++) if (this.mainSoundTrack.soundCollection[e].name === A) return this.mainSoundTrack.soundCollection[e]; if (this.soundTracks) { for (let t = 0; t < this.soundTracks.length; t++) for (e = 0; e < this.soundTracks[t].soundCollection.length; e++) if (this.soundTracks[t].soundCollection[e].name === A) return this.soundTracks[t].soundCollection[e]; } return null; }; Object.defineProperty(sr.prototype, "audioEnabled", { get: function() { let A = this._getComponent(Ot.NAME_AUDIO); return A || (A = new yu(this), this._addComponent(A)), A.audioEnabled; }, set: function(A) { let e = this._getComponent(Ot.NAME_AUDIO); e || (e = new yu(this), this._addComponent(e)), A ? e.enableAudio() : e.disableAudio(); }, enumerable: !0, configurable: !0 }); Object.defineProperty(sr.prototype, "headphone", { get: function() { let A = this._getComponent(Ot.NAME_AUDIO); return A || (A = new yu(this), this._addComponent(A)), A.headphone; }, set: function(A) { let e = this._getComponent(Ot.NAME_AUDIO); e || (e = new yu(this), this._addComponent(e)), A ? e.switchAudioModeForHeadphones() : e.switchAudioModeForNormalSpeakers(); }, enumerable: !0, configurable: !0 }); Object.defineProperty(sr.prototype, "audioListenerPositionProvider", { get: function() { let A = this._getComponent(Ot.NAME_AUDIO); return A || (A = new yu(this), this._addComponent(A)), A.audioListenerPositionProvider; }, set: function(A) { let e = this._getComponent(Ot.NAME_AUDIO); if (e || (e = new yu(this), this._addComponent(e)), A && typeof A != "function") throw new Error("The value passed to [Scene.audioListenerPositionProvider] must be a function that returns a Vector3"); e.audioListenerPositionProvider = A; }, enumerable: !0, configurable: !0 }); Object.defineProperty(sr.prototype, "audioListenerRotationProvider", { get: function() { let A = this._getComponent(Ot.NAME_AUDIO); return A || (A = new yu(this), this._addComponent(A)), A.audioListenerRotationProvider; }, set: function(A) { let e = this._getComponent(Ot.NAME_AUDIO); if (e || (e = new yu(this), this._addComponent(e)), A && typeof A != "function") throw new Error("The value passed to [Scene.audioListenerRotationProvider] must be a function that returns a Vector3"); e.audioListenerRotationProvider = A; }, enumerable: !0, configurable: !0 }); Object.defineProperty(sr.prototype, "audioPositioningRefreshRate", { get: function() { let A = this._getComponent(Ot.NAME_AUDIO); return A || (A = new yu(this), this._addComponent(A)), A.audioPositioningRefreshRate; }, set: function(A) { let e = this._getComponent(Ot.NAME_AUDIO); e || (e = new yu(this), this._addComponent(e)), e.audioPositioningRefreshRate = A; }, enumerable: !0, configurable: !0 }); class yu { /** * Gets whether audio is enabled or not. * Please use related enable/disable method to switch state. */ get audioEnabled() { return this._audioEnabled; } /** * Gets whether audio is outputting to headphone or not. * Please use the according Switch methods to change output. */ get headphone() { return this._headphone; } /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_AUDIO, this._audioEnabled = !0, this._headphone = !1, this.audioPositioningRefreshRate = 500, this.audioListenerPositionProvider = null, this.audioListenerRotationProvider = null, this._cachedCameraDirection = new S(), this._cachedCameraPosition = new S(), this._lastCheck = 0, this._invertMatrixTemp = new he(), this._cameraDirectionTemp = new S(), e = e || gr.LastCreatedScene, e && (this.scene = e, e.soundTracks = [], e.sounds = []); } /** * Registers the component in a given scene */ register() { this.scene._afterRenderStage.registerStep(Ot.STEP_AFTERRENDER_AUDIO, this, this._afterRender); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { } /** * Serializes the component data to the specified json object * @param serializationObject The object to serialize to */ serialize(e) { if (e.sounds = [], this.scene.soundTracks) for (let t = 0; t < this.scene.soundTracks.length; t++) { const r = this.scene.soundTracks[t]; for (let n = 0; n < r.soundCollection.length; n++) e.sounds.push(r.soundCollection[n].serialize()); } } /** * Adds all the elements from the container to the scene * @param container the container holding the elements */ addFromContainer(e) { e.sounds && e.sounds.forEach((t) => { t.play(), t.autoplay = !0, this.scene.mainSoundTrack.addSound(t); }); } /** * Removes all the elements in the container from the scene * @param container contains the elements to remove * @param dispose if the removed element should be disposed (default: false) */ removeFromContainer(e, t = !1) { e.sounds && e.sounds.forEach((r) => { r.stop(), r.autoplay = !1, this.scene.mainSoundTrack.removeSound(r), t && r.dispose(); }); } /** * Disposes the component and the associated resources. */ dispose() { const e = this.scene; if (e._mainSoundTrack && e.mainSoundTrack.dispose(), e.soundTracks) for (let t = 0; t < e.soundTracks.length; t++) e.soundTracks[t].dispose(); } /** * Disables audio in the associated scene. */ disableAudio() { const e = this.scene; this._audioEnabled = !1, Ge.audioEngine && Ge.audioEngine.audioContext && Ge.audioEngine.audioContext.suspend(); let t; for (t = 0; t < e.mainSoundTrack.soundCollection.length; t++) e.mainSoundTrack.soundCollection[t].pause(); if (e.soundTracks) for (t = 0; t < e.soundTracks.length; t++) for (let r = 0; r < e.soundTracks[t].soundCollection.length; r++) e.soundTracks[t].soundCollection[r].pause(); } /** * Enables audio in the associated scene. */ enableAudio() { const e = this.scene; this._audioEnabled = !0, Ge.audioEngine && Ge.audioEngine.audioContext && Ge.audioEngine.audioContext.resume(); let t; for (t = 0; t < e.mainSoundTrack.soundCollection.length; t++) e.mainSoundTrack.soundCollection[t].isPaused && e.mainSoundTrack.soundCollection[t].play(); if (e.soundTracks) for (t = 0; t < e.soundTracks.length; t++) for (let r = 0; r < e.soundTracks[t].soundCollection.length; r++) e.soundTracks[t].soundCollection[r].isPaused && e.soundTracks[t].soundCollection[r].play(); } /** * Switch audio to headphone output. */ switchAudioModeForHeadphones() { const e = this.scene; if (this._headphone = !0, e.mainSoundTrack.switchPanningModelToHRTF(), e.soundTracks) for (let t = 0; t < e.soundTracks.length; t++) e.soundTracks[t].switchPanningModelToHRTF(); } /** * Switch audio to normal speakers. */ switchAudioModeForNormalSpeakers() { const e = this.scene; if (this._headphone = !1, e.mainSoundTrack.switchPanningModelToEqualPower(), e.soundTracks) for (let t = 0; t < e.soundTracks.length; t++) e.soundTracks[t].switchPanningModelToEqualPower(); } _afterRender() { const e = Yi.Now; if (this._lastCheck && e - this._lastCheck < this.audioPositioningRefreshRate) return; this._lastCheck = e; const t = this.scene; if (!this._audioEnabled || !t._mainSoundTrack || !t.soundTracks || t._mainSoundTrack.soundCollection.length === 0 && t.soundTracks.length === 1) return; const r = Ge.audioEngine; if (r && r.audioContext) { let n = t.activeCamera; if (t.activeCameras && t.activeCameras.length > 0 && (n = t.activeCameras[0]), this.audioListenerPositionProvider) { const s = this.audioListenerPositionProvider(); r.audioContext.listener.setPosition(s.x || 0, s.y || 0, s.z || 0); } else n ? this._cachedCameraPosition.equals(n.globalPosition) || (this._cachedCameraPosition.copyFrom(n.globalPosition), r.audioContext.listener.setPosition(n.globalPosition.x, n.globalPosition.y, n.globalPosition.z)) : r.audioContext.listener.setPosition(0, 0, 0); if (this.audioListenerRotationProvider) { const s = this.audioListenerRotationProvider(); r.audioContext.listener.setOrientation(s.x || 0, s.y || 0, s.z || 0, 0, 1, 0); } else n ? (n.rigCameras && n.rigCameras.length > 0 && (n = n.rigCameras[0]), n.getViewMatrix().invertToRef(this._invertMatrixTemp), S.TransformNormalToRef(yu._CameraDirection, this._invertMatrixTemp, this._cameraDirectionTemp), this._cameraDirectionTemp.normalize(), !isNaN(this._cameraDirectionTemp.x) && !isNaN(this._cameraDirectionTemp.y) && !isNaN(this._cameraDirectionTemp.z) && (this._cachedCameraDirection.equals(this._cameraDirectionTemp) || (this._cachedCameraDirection.copyFrom(this._cameraDirectionTemp), r.audioContext.listener.setOrientation(this._cameraDirectionTemp.x, this._cameraDirectionTemp.y, this._cameraDirectionTemp.z, 0, 1, 0)))) : r.audioContext.listener.setOrientation(0, 0, 0, 0, 1, 0); let i; for (i = 0; i < t.mainSoundTrack.soundCollection.length; i++) { const s = t.mainSoundTrack.soundCollection[i]; s.useCustomAttenuation && s.updateDistanceFromListener(); } if (t.soundTracks) for (i = 0; i < t.soundTracks.length; i++) for (let s = 0; s < t.soundTracks[i].soundCollection.length; s++) { const a = t.soundTracks[i].soundCollection[s]; a.useCustomAttenuation && a.updateDistanceFromListener(); } } } } yu._CameraDirection = new S(0, 0, -1); qc._SceneComponentInitialization = (A) => { let e = A._getComponent(Ot.NAME_AUDIO); e || (e = new yu(A), A._addComponent(e)); }; class lee { /** * Creates a new WeightedSound from the list of sounds given. * @param loop When true a Sound will be selected and played when the current playing Sound completes. * @param sounds Array of Sounds that will be selected from. * @param weights Array of number values for selection weights; length must equal sounds, values will be normalized to 1 */ constructor(e, t, r) { if (this.loop = !1, this._coneInnerAngle = 360, this._coneOuterAngle = 360, this._volume = 1, this.isPlaying = !1, this.isPaused = !1, this._sounds = [], this._weights = [], t.length !== r.length) throw new Error("Sounds length does not equal weights length"); this.loop = e, this._weights = r; let n = 0; for (const s of r) n += s; const i = n > 0 ? 1 / n : 0; for (let s = 0; s < this._weights.length; s++) this._weights[s] *= i; this._sounds = t; for (const s of this._sounds) s.onEndedObservable.add(() => { this._onended(); }); } /** * The size of cone in degrees for a directional sound in which there will be no attenuation. */ get directionalConeInnerAngle() { return this._coneInnerAngle; } /** * The size of cone in degrees for a directional sound in which there will be no attenuation. */ set directionalConeInnerAngle(e) { if (e !== this._coneInnerAngle) { if (this._coneOuterAngle < e) { Se.Error("directionalConeInnerAngle: outer angle of the cone must be superior or equal to the inner angle."); return; } this._coneInnerAngle = e; for (const t of this._sounds) t.directionalConeInnerAngle = e; } } /** * Size of cone in degrees for a directional sound outside of which there will be no sound. * Listener angles between innerAngle and outerAngle will falloff linearly. */ get directionalConeOuterAngle() { return this._coneOuterAngle; } /** * Size of cone in degrees for a directional sound outside of which there will be no sound. * Listener angles between innerAngle and outerAngle will falloff linearly. */ set directionalConeOuterAngle(e) { if (e !== this._coneOuterAngle) { if (e < this._coneInnerAngle) { Se.Error("directionalConeOuterAngle: outer angle of the cone must be superior or equal to the inner angle."); return; } this._coneOuterAngle = e; for (const t of this._sounds) t.directionalConeOuterAngle = e; } } /** * Playback volume. */ get volume() { return this._volume; } /** * Playback volume. */ set volume(e) { if (e !== this._volume) for (const t of this._sounds) t.setVolume(e); } _onended() { this._currentIndex !== void 0 && (this._sounds[this._currentIndex].autoplay = !1), this.loop && this.isPlaying ? this.play() : this.isPlaying = !1; } /** * Suspend playback */ pause() { this.isPaused = !0, this._currentIndex !== void 0 && this._sounds[this._currentIndex].pause(); } /** * Stop playback */ stop() { this.isPlaying = !1, this._currentIndex !== void 0 && this._sounds[this._currentIndex].stop(); } /** * Start playback. * @param startOffset Position the clip head at a specific time in seconds. */ play(e) { if (!this.isPaused) { this.stop(); const r = Math.random(); let n = 0; for (let i = 0; i < this._weights.length; i++) if (n += this._weights[i], r <= n) { this._currentIndex = i; break; } } const t = this._sounds[this._currentIndex]; t.isReady() ? t.play(0, this.isPaused ? void 0 : e) : t.autoplay = !0, this.isPlaying = !0, this.isPaused = !1; } } class mm { /** * Creates a new BakedVertexAnimationManager * @param scene defines the current scene */ constructor(e) { this._texture = null, this._isEnabled = !0, this.isEnabled = !0, this.time = 0, e = e || gr.LastCreatedScene, e && (this._scene = e, this.animationParameters = new Ir(0, 0, 0, 30)); } /** @internal */ _markSubMeshesAsAttributesDirty() { for (const e of this._scene.meshes) e.bakedVertexAnimationManager === this && e._markSubMeshesAsAttributesDirty(); } /** * Binds to the effect. * @param effect The effect to bind to. * @param useInstances True when it's an instance. */ bind(e, t = !1) { if (!this._texture || !this._isEnabled) return; const r = this._texture.getSize(); e.setFloat2("bakedVertexAnimationTextureSizeInverted", 1 / r.width, 1 / r.height), e.setFloat("bakedVertexAnimationTime", this.time), t || e.setVector4("bakedVertexAnimationSettings", this.animationParameters), e.setTexture("bakedVertexAnimationTexture", this._texture); } /** * Clone the current manager * @returns a new BakedVertexAnimationManager */ clone() { const e = new mm(this._scene); return this.copyTo(e), e; } /** * Sets animation parameters. * @param startFrame The first frame of the animation. * @param endFrame The last frame of the animation. * @param offset The offset when starting the animation. * @param speedFramesPerSecond The frame rate. */ setAnimationParameters(e, t, r = 0, n = 30) { this.animationParameters = new Ir(e, t, r, n); } /** * Disposes the resources of the manager. * @param forceDisposeTextures - Forces the disposal of all textures. */ dispose(e) { var t; e && ((t = this._texture) === null || t === void 0 || t.dispose()); } /** * Get the current class name useful for serialization or dynamic coding. * @returns "BakedVertexAnimationManager" */ getClassName() { return "BakedVertexAnimationManager"; } /** * Makes a duplicate of the current instance into another one. * @param vatMap define the instance where to copy the info */ copyTo(e) { jt.Clone(() => e, this); } /** * Serializes this vertex animation instance * @returns - An object with the serialized instance. */ serialize() { return jt.Serialize(this); } /** * Parses a vertex animation setting from a serialized object. * @param source - Serialized object. * @param scene Defines the scene we are parsing for * @param rootUrl Defines the rootUrl to load from */ parse(e, t, r) { jt.Parse(() => this, e, t, r); } } C([ en(), At("_markSubMeshesAsAttributesDirty") ], mm.prototype, "texture", void 0); C([ M(), At("_markSubMeshesAsAttributesDirty") ], mm.prototype, "isEnabled", void 0); C([ M() ], mm.prototype, "animationParameters", void 0); C([ M() ], mm.prototype, "time", void 0); class nq { /** * | Value | Type | Description | * | ----- | ------------------ | ----------- | * | 0 | CLAMP_ADDRESSMODE | | * | 1 | WRAP_ADDRESSMODE | | * | 2 | MIRROR_ADDRESSMODE | | */ get wrapU() { return this._wrapU; } set wrapU(e) { this._wrapU = e; } /** * | Value | Type | Description | * | ----- | ------------------ | ----------- | * | 0 | CLAMP_ADDRESSMODE | | * | 1 | WRAP_ADDRESSMODE | | * | 2 | MIRROR_ADDRESSMODE | | */ get wrapV() { return this._wrapV; } set wrapV(e) { this._wrapV = e; } /** * How a texture is mapped. * Unused in thin texture mode. */ get coordinatesMode() { return 0; } /** * Define if the texture is a cube texture or if false a 2d texture. */ get isCube() { return this._texture ? this._texture.isCube : !1; } set isCube(e) { this._texture && (this._texture.isCube = e); } /** * Define if the texture is a 3d texture (webgl 2) or if false a 2d texture. */ get is3D() { return this._texture ? this._texture.is3D : !1; } set is3D(e) { this._texture && (this._texture.is3D = e); } /** * Define if the texture is a 2d array texture (webgl 2) or if false a 2d texture. */ get is2DArray() { return this._texture ? this._texture.is2DArray : !1; } set is2DArray(e) { this._texture && (this._texture.is2DArray = e); } /** * Get the class name of the texture. * @returns "ThinTexture" */ getClassName() { return "ThinTexture"; } static _IsRenderTargetWrapper(e) { return (e == null ? void 0 : e._shareDepth) !== void 0; } /** * Instantiates a new ThinTexture. * Base class of all the textures in babylon. * This can be used as an internal texture wrapper in ThinEngine to benefit from the cache * @param internalTexture Define the internalTexture to wrap. You can also pass a RenderTargetWrapper, in which case the texture will be the render target's texture */ constructor(e) { this._wrapU = 1, this._wrapV = 1, this.wrapR = 1, this.anisotropicFilteringLevel = 4, this.delayLoadState = 0, this._texture = null, this._engine = null, this._cachedSize = Qd.Zero(), this._cachedBaseSize = Qd.Zero(), this._initialSamplingMode = 2, this._texture = nq._IsRenderTargetWrapper(e) ? e.texture : e, this._texture && (this._engine = this._texture.getEngine()); } /** * Get if the texture is ready to be used (downloaded, converted, mip mapped...). * @returns true if fully ready */ isReady() { return this.delayLoadState === 4 ? (this.delayLoad(), !1) : this._texture ? this._texture.isReady : !1; } /** * Triggers the load sequence in delayed load mode. */ delayLoad() { } /** * Get the underlying lower level texture from Babylon. * @returns the internal texture */ getInternalTexture() { return this._texture; } /** * Get the size of the texture. * @returns the texture size. */ getSize() { if (this._texture) { if (this._texture.width) return this._cachedSize.width = this._texture.width, this._cachedSize.height = this._texture.height, this._cachedSize; if (this._texture._size) return this._cachedSize.width = this._texture._size, this._cachedSize.height = this._texture._size, this._cachedSize; } return this._cachedSize; } /** * Get the base size of the texture. * It can be different from the size if the texture has been resized for POT for instance * @returns the base size */ getBaseSize() { return !this.isReady() || !this._texture ? (this._cachedBaseSize.width = 0, this._cachedBaseSize.height = 0, this._cachedBaseSize) : this._texture._size ? (this._cachedBaseSize.width = this._texture._size, this._cachedBaseSize.height = this._texture._size, this._cachedBaseSize) : (this._cachedBaseSize.width = this._texture.baseWidth, this._cachedBaseSize.height = this._texture.baseHeight, this._cachedBaseSize); } /** * Get the current sampling mode associated with the texture. */ get samplingMode() { return this._texture ? this._texture.samplingMode : this._initialSamplingMode; } /** * Update the sampling mode of the texture. * Default is Trilinear mode. * * | Value | Type | Description | * | ----- | ------------------ | ----------- | * | 1 | NEAREST_SAMPLINGMODE or NEAREST_NEAREST_MIPLINEAR | Nearest is: mag = nearest, min = nearest, mip = linear | * | 2 | BILINEAR_SAMPLINGMODE or LINEAR_LINEAR_MIPNEAREST | Bilinear is: mag = linear, min = linear, mip = nearest | * | 3 | TRILINEAR_SAMPLINGMODE or LINEAR_LINEAR_MIPLINEAR | Trilinear is: mag = linear, min = linear, mip = linear | * | 4 | NEAREST_NEAREST_MIPNEAREST | | * | 5 | NEAREST_LINEAR_MIPNEAREST | | * | 6 | NEAREST_LINEAR_MIPLINEAR | | * | 7 | NEAREST_LINEAR | | * | 8 | NEAREST_NEAREST | | * | 9 | LINEAR_NEAREST_MIPNEAREST | | * | 10 | LINEAR_NEAREST_MIPLINEAR | | * | 11 | LINEAR_LINEAR | | * | 12 | LINEAR_NEAREST | | * * > _mag_: magnification filter (close to the viewer) * > _min_: minification filter (far from the viewer) * > _mip_: filter used between mip map levels *@param samplingMode Define the new sampling mode of the texture */ updateSamplingMode(e) { this._texture && this._engine && this._engine.updateTextureSamplingMode(e, this._texture); } /** * Release and destroy the underlying lower level texture aka internalTexture. */ releaseInternalTexture() { this._texture && (this._texture.dispose(), this._texture = null); } /** * Dispose the texture and release its associated resources. */ dispose() { this._texture && (this.releaseInternalTexture(), this._engine = null); } } class ls extends nq { /** * Define if the texture is having a usable alpha value (can be use for transparency or glossiness for instance). */ set hasAlpha(e) { this._hasAlpha !== e && (this._hasAlpha = e, this._scene && this._scene.markAllMaterialsAsDirty(1, (t) => t.hasTexture(this))); } get hasAlpha() { return this._hasAlpha; } /** * Defines if the alpha value should be determined via the rgb values. * If true the luminance of the pixel might be used to find the corresponding alpha value. */ set getAlphaFromRGB(e) { this._getAlphaFromRGB !== e && (this._getAlphaFromRGB = e, this._scene && this._scene.markAllMaterialsAsDirty(1, (t) => t.hasTexture(this))); } get getAlphaFromRGB() { return this._getAlphaFromRGB; } /** * Define the UV channel to use starting from 0 and defaulting to 0. * This is part of the texture as textures usually maps to one uv set. */ set coordinatesIndex(e) { this._coordinatesIndex !== e && (this._coordinatesIndex = e, this._scene && this._scene.markAllMaterialsAsDirty(1, (t) => t.hasTexture(this))); } get coordinatesIndex() { return this._coordinatesIndex; } /** * How a texture is mapped. * * | Value | Type | Description | * | ----- | ----------------------------------- | ----------- | * | 0 | EXPLICIT_MODE | | * | 1 | SPHERICAL_MODE | | * | 2 | PLANAR_MODE | | * | 3 | CUBIC_MODE | | * | 4 | PROJECTION_MODE | | * | 5 | SKYBOX_MODE | | * | 6 | INVCUBIC_MODE | | * | 7 | EQUIRECTANGULAR_MODE | | * | 8 | FIXED_EQUIRECTANGULAR_MODE | | * | 9 | FIXED_EQUIRECTANGULAR_MIRRORED_MODE | | */ set coordinatesMode(e) { this._coordinatesMode !== e && (this._coordinatesMode = e, this._scene && this._scene.markAllMaterialsAsDirty(1, (t) => t.hasTexture(this))); } get coordinatesMode() { return this._coordinatesMode; } /** * | Value | Type | Description | * | ----- | ------------------ | ----------- | * | 0 | CLAMP_ADDRESSMODE | | * | 1 | WRAP_ADDRESSMODE | | * | 2 | MIRROR_ADDRESSMODE | | */ get wrapU() { return this._wrapU; } set wrapU(e) { this._wrapU = e; } /** * | Value | Type | Description | * | ----- | ------------------ | ----------- | * | 0 | CLAMP_ADDRESSMODE | | * | 1 | WRAP_ADDRESSMODE | | * | 2 | MIRROR_ADDRESSMODE | | */ get wrapV() { return this._wrapV; } set wrapV(e) { this._wrapV = e; } /** * Define if the texture is a cube texture or if false a 2d texture. */ get isCube() { return this._texture ? this._texture.isCube : this._isCube; } set isCube(e) { this._texture ? this._texture.isCube = e : this._isCube = e; } /** * Define if the texture is a 3d texture (webgl 2) or if false a 2d texture. */ get is3D() { return this._texture ? this._texture.is3D : !1; } set is3D(e) { this._texture && (this._texture.is3D = e); } /** * Define if the texture is a 2d array texture (webgl 2) or if false a 2d texture. */ get is2DArray() { return this._texture ? this._texture.is2DArray : !1; } set is2DArray(e) { this._texture && (this._texture.is2DArray = e); } /** * Define if the texture contains data in gamma space (most of the png/jpg aside bump). * HDR texture are usually stored in linear space. * This only impacts the PBR and Background materials */ get gammaSpace() { if (this._texture) this._texture._gammaSpace === null && (this._texture._gammaSpace = this._gammaSpace); else return this._gammaSpace; return this._texture._gammaSpace && !this._texture._useSRGBBuffer; } set gammaSpace(e) { var t; if (this._texture) { if (this._texture._gammaSpace === e) return; this._texture._gammaSpace = e; } else { if (this._gammaSpace === e) return; this._gammaSpace = e; } (t = this.getScene()) === null || t === void 0 || t.markAllMaterialsAsDirty(1, (r) => r.hasTexture(this)); } /** * Gets or sets whether or not the texture contains RGBD data. */ get isRGBD() { return this._texture != null && this._texture._isRGBD; } set isRGBD(e) { var t; e !== this.isRGBD && (this._texture && (this._texture._isRGBD = e), (t = this.getScene()) === null || t === void 0 || t.markAllMaterialsAsDirty(1, (r) => r.hasTexture(this))); } /** * Are mip maps generated for this texture or not. */ get noMipmap() { return !1; } /** * With prefiltered texture, defined the offset used during the prefiltering steps. */ get lodGenerationOffset() { return this._texture ? this._texture._lodGenerationOffset : 0; } set lodGenerationOffset(e) { this._texture && (this._texture._lodGenerationOffset = e); } /** * With prefiltered texture, defined the scale used during the prefiltering steps. */ get lodGenerationScale() { return this._texture ? this._texture._lodGenerationScale : 0; } set lodGenerationScale(e) { this._texture && (this._texture._lodGenerationScale = e); } /** * With prefiltered texture, defined if the specular generation is based on a linear ramp. * By default we are using a log2 of the linear roughness helping to keep a better resolution for * average roughness values. */ get linearSpecularLOD() { return this._texture ? this._texture._linearSpecularLOD : !1; } set linearSpecularLOD(e) { this._texture && (this._texture._linearSpecularLOD = e); } /** * In case a better definition than spherical harmonics is required for the diffuse part of the environment. * You can set the irradiance texture to rely on a texture instead of the spherical approach. * This texture need to have the same characteristics than its parent (Cube vs 2d, coordinates mode, Gamma/Linear, RGBD). */ get irradianceTexture() { return this._texture ? this._texture._irradianceTexture : null; } set irradianceTexture(e) { this._texture && (this._texture._irradianceTexture = e); } /** * Define the unique id of the texture in the scene. */ get uid() { return this._uid || (this._uid = v4()), this._uid; } /** * Return a string representation of the texture. * @returns the texture as a string */ toString() { return this.name; } /** * Get the class name of the texture. * @returns "BaseTexture" */ getClassName() { return "BaseTexture"; } /** * Callback triggered when the texture has been disposed. * Kept for back compatibility, you can use the onDisposeObservable instead. */ set onDispose(e) { this._onDisposeObserver && this.onDisposeObservable.remove(this._onDisposeObserver), this._onDisposeObserver = this.onDisposeObservable.add(e); } /** * Define if the texture is preventing a material to render or not. * If not and the texture is not ready, the engine will use a default black texture instead. */ get isBlocking() { return !0; } /** * Was there any loading error? */ get loadingError() { return this._loadingError; } /** * If a loading error occurred this object will be populated with information about the error. */ get errorObject() { return this._errorObject; } /** * Instantiates a new BaseTexture. * Base class of all the textures in babylon. * It groups all the common properties the materials, post process, lights... might need * in order to make a correct use of the texture. * @param sceneOrEngine Define the scene or engine the texture belongs to * @param internalTexture Define the internal texture associated with the texture */ constructor(e, t = null) { super(null), this.metadata = null, this.reservedDataStore = null, this._hasAlpha = !1, this._getAlphaFromRGB = !1, this.level = 1, this._coordinatesIndex = 0, this.optimizeUVAllocation = !0, this._coordinatesMode = 0, this.wrapR = 1, this.anisotropicFilteringLevel = ls.DEFAULT_ANISOTROPIC_FILTERING_LEVEL, this._isCube = !1, this._gammaSpace = !0, this.invertZ = !1, this.lodLevelInAlpha = !1, this.isRenderTarget = !1, this._prefiltered = !1, this._forceSerialize = !1, this.animations = [], this.onDisposeObservable = new Oe(), this._onDisposeObserver = null, this._scene = null, this._uid = null, this._parentContainer = null, this._loadingError = !1, e ? ls._IsScene(e) ? this._scene = e : this._engine = e : this._scene = gr.LastCreatedScene, this._scene && (this.uniqueId = this._scene.getUniqueId(), this._scene.addTexture(this), this._engine = this._scene.getEngine()), this._texture = t, this._uid = null; } /** * Get the scene the texture belongs to. * @returns the scene or null if undefined */ getScene() { return this._scene; } /** @internal */ _getEngine() { return this._engine; } /** * Checks if the texture has the same transform matrix than another texture * @param texture texture to check against * @returns true if the transforms are the same, else false */ checkTransformsAreIdentical(e) { return e !== null; } /** * Get the texture transform matrix used to offset tile the texture for instance. * @returns the transformation matrix */ getTextureMatrix() { return he.IdentityReadOnly; } /** * Get the texture reflection matrix used to rotate/transform the reflection. * @returns the reflection matrix */ getReflectionTextureMatrix() { return he.IdentityReadOnly; } /** * Gets a suitable rotate/transform matrix when the texture is used for refraction. * There's a separate function from getReflectionTextureMatrix because refraction requires a special configuration of the matrix in right-handed mode. * @returns The refraction matrix */ getRefractionTextureMatrix() { return this.getReflectionTextureMatrix(); } /** * Get if the texture is ready to be consumed (either it is ready or it is not blocking) * @returns true if ready, not blocking or if there was an error loading the texture */ isReadyOrNotBlocking() { return !this.isBlocking || this.isReady() || this.loadingError; } /** * Scales the texture if is `canRescale()` * @param ratio the resize factor we want to use to rescale */ // eslint-disable-next-line @typescript-eslint/no-unused-vars scale(e) { } /** * Get if the texture can rescale. */ get canRescale() { return !1; } /** * @internal */ _getFromCache(e, t, r, n, i, s) { const a = this._getEngine(); if (!a) return null; const f = a._getUseSRGBBuffer(!!i, t), o = a.getLoadedTexturesCache(); for (let d = 0; d < o.length; d++) { const v = o[d]; if ((i === void 0 || f === v._useSRGBBuffer) && (n === void 0 || n === v.invertY) && v.url === e && v.generateMipMaps === !t && (!r || r === v.samplingMode) && (s === void 0 || s === v.isCube)) return v.incrementReferences(), v; } return null; } /** @internal */ _rebuild() { } /** * Clones the texture. * @returns the cloned texture */ clone() { return null; } /** * Get the texture underlying type (INT, FLOAT...) */ get textureType() { return this._texture && this._texture.type !== void 0 ? this._texture.type : 0; } /** * Get the texture underlying format (RGB, RGBA...) */ get textureFormat() { return this._texture && this._texture.format !== void 0 ? this._texture.format : 5; } /** * Indicates that textures need to be re-calculated for all materials */ _markAllSubMeshesAsTexturesDirty() { const e = this.getScene(); e && e.markAllMaterialsAsDirty(1); } /** * Reads the pixels stored in the webgl texture and returns them as an ArrayBuffer. * This will returns an RGBA array buffer containing either in values (0-255) or * float values (0-1) depending of the underlying buffer type. * @param faceIndex defines the face of the texture to read (in case of cube texture) * @param level defines the LOD level of the texture to read (in case of Mip Maps) * @param buffer defines a user defined buffer to fill with data (can be null) * @param flushRenderer true to flush the renderer from the pending commands before reading the pixels * @param noDataConversion false to convert the data to Uint8Array (if texture type is UNSIGNED_BYTE) or to Float32Array (if texture type is anything but UNSIGNED_BYTE). If true, the type of the generated buffer (if buffer==null) will depend on the type of the texture * @param x defines the region x coordinates to start reading from (default to 0) * @param y defines the region y coordinates to start reading from (default to 0) * @param width defines the region width to read from (default to the texture size at level) * @param height defines the region width to read from (default to the texture size at level) * @returns The Array buffer promise containing the pixels data. */ readPixels(e = 0, t = 0, r = null, n = !0, i = !1, s = 0, a = 0, f = Number.MAX_VALUE, o = Number.MAX_VALUE) { if (!this._texture) return null; const d = this._getEngine(); if (!d) return null; const v = this.getSize(); let u = v.width, l = v.height; t !== 0 && (u = u / Math.pow(2, t), l = l / Math.pow(2, t), u = Math.round(u), l = Math.round(l)), f = Math.min(u, f), o = Math.min(l, o); try { return this._texture.isCube ? d._readTexturePixels(this._texture, f, o, e, t, r, n, i, s, a) : d._readTexturePixels(this._texture, f, o, -1, t, r, n, i, s, a); } catch { return null; } } /** * @internal */ _readPixelsSync(e = 0, t = 0, r = null, n = !0, i = !1) { if (!this._texture) return null; const s = this.getSize(); let a = s.width, f = s.height; const o = this._getEngine(); if (!o) return null; t != 0 && (a = a / Math.pow(2, t), f = f / Math.pow(2, t), a = Math.round(a), f = Math.round(f)); try { return this._texture.isCube ? o._readTexturePixelsSync(this._texture, a, f, e, t, r, n, i) : o._readTexturePixelsSync(this._texture, a, f, -1, t, r, n, i); } catch { return null; } } /** @internal */ get _lodTextureHigh() { return this._texture ? this._texture._lodTextureHigh : null; } /** @internal */ get _lodTextureMid() { return this._texture ? this._texture._lodTextureMid : null; } /** @internal */ get _lodTextureLow() { return this._texture ? this._texture._lodTextureLow : null; } /** * Dispose the texture and release its associated resources. */ dispose() { if (this._scene) { this._scene.stopAnimation && this._scene.stopAnimation(this), this._scene.removePendingData(this); const e = this._scene.textures.indexOf(this); if (e >= 0 && this._scene.textures.splice(e, 1), this._scene.onTextureRemovedObservable.notifyObservers(this), this._scene = null, this._parentContainer) { const t = this._parentContainer.textures.indexOf(this); t > -1 && this._parentContainer.textures.splice(t, 1), this._parentContainer = null; } } this.onDisposeObservable.notifyObservers(this), this.onDisposeObservable.clear(), this.metadata = null, super.dispose(); } /** * Serialize the texture into a JSON representation that can be parsed later on. * @param allowEmptyName True to force serialization even if name is empty. Default: false * @returns the JSON representation of the texture */ serialize(e = !1) { if (!this.name && !e) return null; const t = jt.Serialize(this); return jt.AppendSerializedAnimations(this, t), t; } /** * Helper function to be called back once a list of texture contains only ready textures. * @param textures Define the list of textures to wait for * @param callback Define the callback triggered once the entire list will be ready */ static WhenAllReady(e, t) { let r = e.length; if (r === 0) { t(); return; } for (let n = 0; n < e.length; n++) { const i = e[n]; if (i.isReady()) --r === 0 && t(); else { const s = i.onLoadObservable; s ? s.addOnce(() => { --r === 0 && t(); }) : --r === 0 && t(); } } } static _IsScene(e) { return e.getClassName() === "Scene"; } } ls.DEFAULT_ANISOTROPIC_FILTERING_LEVEL = 4; C([ M() ], ls.prototype, "uniqueId", void 0); C([ M() ], ls.prototype, "name", void 0); C([ M() ], ls.prototype, "metadata", void 0); C([ M("hasAlpha") ], ls.prototype, "_hasAlpha", void 0); C([ M("getAlphaFromRGB") ], ls.prototype, "_getAlphaFromRGB", void 0); C([ M() ], ls.prototype, "level", void 0); C([ M("coordinatesIndex") ], ls.prototype, "_coordinatesIndex", void 0); C([ M() ], ls.prototype, "optimizeUVAllocation", void 0); C([ M("coordinatesMode") ], ls.prototype, "_coordinatesMode", void 0); C([ M() ], ls.prototype, "wrapU", null); C([ M() ], ls.prototype, "wrapV", null); C([ M() ], ls.prototype, "wrapR", void 0); C([ M() ], ls.prototype, "anisotropicFilteringLevel", void 0); C([ M() ], ls.prototype, "isCube", null); C([ M() ], ls.prototype, "is3D", null); C([ M() ], ls.prototype, "is2DArray", null); C([ M() ], ls.prototype, "gammaSpace", null); C([ M() ], ls.prototype, "invertZ", void 0); C([ M() ], ls.prototype, "lodLevelInAlpha", void 0); C([ M() ], ls.prototype, "lodGenerationOffset", null); C([ M() ], ls.prototype, "lodGenerationScale", null); C([ M() ], ls.prototype, "linearSpecularLOD", null); C([ en() ], ls.prototype, "irradianceTexture", null); C([ M() ], ls.prototype, "isRenderTarget", void 0); function SO(A, e, t = !1) { const r = e.width, n = e.height; if (A instanceof Float32Array) { let o = A.byteLength / A.BYTES_PER_ELEMENT; const d = new Uint8Array(o); for (; --o >= 0; ) { let v = A[o]; v < 0 ? v = 0 : v > 1 && (v = 1), d[o] = v * 255; } A = d; } const i = document.createElement("canvas"); i.width = r, i.height = n; const s = i.getContext("2d"); if (!s) return null; const a = s.createImageData(r, n); if (a.data.set(A), s.putImageData(a, 0, 0), t) { const o = document.createElement("canvas"); o.width = r, o.height = n; const d = o.getContext("2d"); return d ? (d.translate(0, n), d.scale(1, -1), d.drawImage(i, 0, 0), o.toDataURL("image/png")) : null; } return i.toDataURL("image/png"); } function rQ(A, e = 0, t = 0) { const r = A.getInternalTexture(); if (!r) return null; const n = A._readPixelsSync(e, t); return n ? SO(n, A.getSize(), r.invertY) : null; } async function nQ(A, e = 0, t = 0) { const r = A.getInternalTexture(); if (!r) return null; const n = await A.readPixels(e, t); return n ? SO(n, A.getSize(), r.invertY) : null; } const g6e = { /** * Transform some pixel data to a base64 string * @param pixels defines the pixel data to transform to base64 * @param size defines the width and height of the (texture) data * @param invertY true if the data must be inverted for the Y coordinate during the conversion * @returns The base64 encoded string or null */ GenerateBase64StringFromPixelData: SO, /** * Reads the pixels stored in the webgl texture and returns them as a base64 string * @param texture defines the texture to read pixels from * @param faceIndex defines the face of the texture to read (in case of cube texture) * @param level defines the LOD level of the texture to read (in case of Mip Maps) * @returns The base64 encoded string or null */ GenerateBase64StringFromTexture: rQ, /** * Reads the pixels stored in the webgl texture and returns them as a base64 string * @param texture defines the texture to read pixels from * @param faceIndex defines the face of the texture to read (in case of cube texture) * @param level defines the LOD level of the texture to read (in case of Mip Maps) * @returns The base64 encoded string or null wrapped in a promise */ GenerateBase64StringFromTextureAsync: nQ }; class We extends ls { static _CreateVideoTexture(e, t, r, n = !1, i = !1, s = We.TRILINEAR_SAMPLINGMODE, a = {}, f, o = 5) { throw qn("VideoTexture"); } /** * Are mip maps generated for this texture or not. */ get noMipmap() { return this._noMipmap; } /** Returns the texture mime type if it was defined by a loader (undefined else) */ get mimeType() { return this._mimeType; } /** * Is the texture preventing material to render while loading. * If false, a default texture will be used instead of the loading one during the preparation step. */ set isBlocking(e) { this._isBlocking = e; } get isBlocking() { return this._isBlocking; } /** * Gets a boolean indicating if the texture needs to be inverted on the y axis during loading */ get invertY() { return this._invertY; } /** * Instantiates a new texture. * This represents a texture in babylon. It can be easily loaded from a network, base64 or html input. * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/materials_introduction#texture * @param url defines the url of the picture to load as a texture * @param sceneOrEngine defines the scene or engine the texture will belong to * @param noMipmapOrOptions defines if the texture will require mip maps or not or set of all options to create the texture * @param invertY defines if the texture needs to be inverted on the y axis during loading * @param samplingMode defines the sampling mode we want for the texture while fetching from it (Texture.NEAREST_SAMPLINGMODE...) * @param onLoad defines a callback triggered when the texture has been loaded * @param onError defines a callback triggered when an error occurred during the loading session * @param buffer defines the buffer to load the texture from in case the texture is loaded from a buffer representation * @param deleteBuffer defines if the buffer we are loading the texture from should be deleted after load * @param format defines the format of the texture we are trying to load (Engine.TEXTUREFORMAT_RGBA...) * @param mimeType defines an optional mime type information * @param loaderOptions options to be passed to the loader * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg) * @param forcedExtension defines the extension to use to pick the right loader */ constructor(e, t, r, n, i = We.TRILINEAR_SAMPLINGMODE, s = null, a = null, f = null, o = !1, d, v, u, l, P) { var p, c, H, T, q, b, j, w, m, I; super(t), this.url = null, this.uOffset = 0, this.vOffset = 0, this.uScale = 1, this.vScale = 1, this.uAng = 0, this.vAng = 0, this.wAng = 0, this.uRotationCenter = 0.5, this.vRotationCenter = 0.5, this.wRotationCenter = 0.5, this.homogeneousRotationInUVTransform = !1, this.inspectableCustomProperties = null, this._noMipmap = !1, this._invertY = !1, this._rowGenerationMatrix = null, this._cachedTextureMatrix = null, this._projectionModeMatrix = null, this._t0 = null, this._t1 = null, this._t2 = null, this._cachedUOffset = -1, this._cachedVOffset = -1, this._cachedUScale = 0, this._cachedVScale = 0, this._cachedUAng = -1, this._cachedVAng = -1, this._cachedWAng = -1, this._cachedReflectionProjectionMatrixId = -1, this._cachedURotationCenter = -1, this._cachedVRotationCenter = -1, this._cachedWRotationCenter = -1, this._cachedHomogeneousRotationInUVTransform = !1, this._cachedReflectionTextureMatrix = null, this._cachedReflectionUOffset = -1, this._cachedReflectionVOffset = -1, this._cachedReflectionUScale = 0, this._cachedReflectionVScale = 0, this._cachedReflectionCoordinatesMode = -1, this._buffer = null, this._deleteBuffer = !1, this._format = null, this._delayedOnLoad = null, this._delayedOnError = null, this.onLoadObservable = new Oe(), this._isBlocking = !0, this.name = e || "", this.url = e; let N, k = !1, R = null, y = !0; typeof r == "object" && r !== null ? (N = (p = r.noMipmap) !== null && p !== void 0 ? p : !1, n = (c = r.invertY) !== null && c !== void 0 ? c : !us.UseOpenGLOrientationForUV, i = (H = r.samplingMode) !== null && H !== void 0 ? H : We.TRILINEAR_SAMPLINGMODE, s = (T = r.onLoad) !== null && T !== void 0 ? T : null, a = (q = r.onError) !== null && q !== void 0 ? q : null, f = (b = r.buffer) !== null && b !== void 0 ? b : null, o = (j = r.deleteBuffer) !== null && j !== void 0 ? j : !1, d = r.format, v = r.mimeType, u = r.loaderOptions, l = r.creationFlags, k = (w = r.useSRGBBuffer) !== null && w !== void 0 ? w : !1, R = (m = r.internalTexture) !== null && m !== void 0 ? m : null, y = (I = r.gammaSpace) !== null && I !== void 0 ? I : y) : N = !!r, this._gammaSpace = y, this._noMipmap = N, this._invertY = n === void 0 ? !us.UseOpenGLOrientationForUV : n, this._initialSamplingMode = i, this._buffer = f, this._deleteBuffer = o, this._mimeType = v, this._loaderOptions = u, this._creationFlags = l, this._useSRGBBuffer = k, this._forcedExtension = P, d && (this._format = d); const O = this.getScene(), Y = this._getEngine(); if (!Y) return; Y.onBeforeTextureInitObservable.notifyObservers(this); const ee = () => { this._texture && (this._texture._invertVScale && (this.vScale *= -1, this.vOffset += 1), this._texture._cachedWrapU !== null && (this.wrapU = this._texture._cachedWrapU, this._texture._cachedWrapU = null), this._texture._cachedWrapV !== null && (this.wrapV = this._texture._cachedWrapV, this._texture._cachedWrapV = null), this._texture._cachedWrapR !== null && (this.wrapR = this._texture._cachedWrapR, this._texture._cachedWrapR = null)), this.onLoadObservable.hasObservers() && this.onLoadObservable.notifyObservers(this), s && s(), !this.isBlocking && O && O.resetCachedMaterial(); }, Z = (te, fe) => { this._loadingError = !0, this._errorObject = { message: te, exception: fe }, a && a(te, fe), We.OnTextureLoadErrorObservable.notifyObservers(this); }; if (!this.url && !R) { this._delayedOnLoad = ee, this._delayedOnError = Z; return; } if (this._texture = R ?? this._getFromCache(this.url, N, i, this._invertY, k, this.isCube), this._texture) if (this._texture.isReady) wS.SetImmediate(() => ee()); else { const te = this._texture.onLoadedObservable.add(ee); this._texture.onErrorObservable.add((fe) => { var _; Z(fe.message, fe.exception), (_ = this._texture) === null || _ === void 0 || _.onLoadedObservable.remove(te); }); } else if (!O || !O.useDelayedTextureLoading) { try { this._texture = Y.createTexture(this.url, N, this._invertY, O, i, ee, Z, this._buffer, void 0, this._format, this._forcedExtension, v, u, l, k); } catch (te) { throw Z("error loading", te), te; } o && (this._buffer = null); } else this.delayLoadState = 4, this._delayedOnLoad = ee, this._delayedOnError = Z; } /** * Update the url (and optional buffer) of this texture if url was null during construction. * @param url the url of the texture * @param buffer the buffer of the texture (defaults to null) * @param onLoad callback called when the texture is loaded (defaults to null) * @param forcedExtension defines the extension to use to pick the right loader */ updateURL(e, t = null, r, n) { this.url && (this.releaseInternalTexture(), this.getScene().markAllMaterialsAsDirty(1, (i) => i.hasTexture(this))), (!this.name || this.name.startsWith("data:")) && (this.name = e), this.url = e, this._buffer = t, this._forcedExtension = n, this.delayLoadState = 4, r && (this._delayedOnLoad = r), this.delayLoad(); } /** * Finish the loading sequence of a texture flagged as delayed load. * @internal */ delayLoad() { if (this.delayLoadState !== 4) return; const e = this.getScene(); e && (this.delayLoadState = 1, this._texture = this._getFromCache(this.url, this._noMipmap, this.samplingMode, this._invertY, this._useSRGBBuffer, this.isCube), this._texture ? this._delayedOnLoad && (this._texture.isReady ? wS.SetImmediate(this._delayedOnLoad) : this._texture.onLoadedObservable.add(this._delayedOnLoad)) : (this._texture = e.getEngine().createTexture(this.url, this._noMipmap, this._invertY, e, this.samplingMode, this._delayedOnLoad, this._delayedOnError, this._buffer, null, this._format, this._forcedExtension, this._mimeType, this._loaderOptions, this._creationFlags, this._useSRGBBuffer), this._deleteBuffer && (this._buffer = null)), this._delayedOnLoad = null, this._delayedOnError = null); } _prepareRowForTextureGeneration(e, t, r, n) { e *= this._cachedUScale, t *= this._cachedVScale, e -= this.uRotationCenter * this._cachedUScale, t -= this.vRotationCenter * this._cachedVScale, r -= this.wRotationCenter, S.TransformCoordinatesFromFloatsToRef(e, t, r, this._rowGenerationMatrix, n), n.x += this.uRotationCenter * this._cachedUScale + this._cachedUOffset, n.y += this.vRotationCenter * this._cachedVScale + this._cachedVOffset, n.z += this.wRotationCenter; } /** * Checks if the texture has the same transform matrix than another texture * @param texture texture to check against * @returns true if the transforms are the same, else false */ checkTransformsAreIdentical(e) { return e !== null && this.uOffset === e.uOffset && this.vOffset === e.vOffset && this.uScale === e.uScale && this.vScale === e.vScale && this.uAng === e.uAng && this.vAng === e.vAng && this.wAng === e.wAng; } /** * Get the current texture matrix which includes the requested offsetting, tiling and rotation components. * @param uBase * @returns the transform matrix of the texture. */ getTextureMatrix(e = 1) { if (this.uOffset === this._cachedUOffset && this.vOffset === this._cachedVOffset && this.uScale * e === this._cachedUScale && this.vScale === this._cachedVScale && this.uAng === this._cachedUAng && this.vAng === this._cachedVAng && this.wAng === this._cachedWAng && this.uRotationCenter === this._cachedURotationCenter && this.vRotationCenter === this._cachedVRotationCenter && this.wRotationCenter === this._cachedWRotationCenter && this.homogeneousRotationInUVTransform === this._cachedHomogeneousRotationInUVTransform) return this._cachedTextureMatrix; this._cachedUOffset = this.uOffset, this._cachedVOffset = this.vOffset, this._cachedUScale = this.uScale * e, this._cachedVScale = this.vScale, this._cachedUAng = this.uAng, this._cachedVAng = this.vAng, this._cachedWAng = this.wAng, this._cachedURotationCenter = this.uRotationCenter, this._cachedVRotationCenter = this.vRotationCenter, this._cachedWRotationCenter = this.wRotationCenter, this._cachedHomogeneousRotationInUVTransform = this.homogeneousRotationInUVTransform, (!this._cachedTextureMatrix || !this._rowGenerationMatrix) && (this._cachedTextureMatrix = he.Zero(), this._rowGenerationMatrix = new he(), this._t0 = S.Zero(), this._t1 = S.Zero(), this._t2 = S.Zero()), he.RotationYawPitchRollToRef(this.vAng, this.uAng, this.wAng, this._rowGenerationMatrix), this.homogeneousRotationInUVTransform ? (he.TranslationToRef(-this._cachedURotationCenter, -this._cachedVRotationCenter, -this._cachedWRotationCenter, ue.Matrix[0]), he.TranslationToRef(this._cachedURotationCenter, this._cachedVRotationCenter, this._cachedWRotationCenter, ue.Matrix[1]), he.ScalingToRef(this._cachedUScale, this._cachedVScale, 0, ue.Matrix[2]), he.TranslationToRef(this._cachedUOffset, this._cachedVOffset, 0, ue.Matrix[3]), ue.Matrix[0].multiplyToRef(this._rowGenerationMatrix, this._cachedTextureMatrix), this._cachedTextureMatrix.multiplyToRef(ue.Matrix[1], this._cachedTextureMatrix), this._cachedTextureMatrix.multiplyToRef(ue.Matrix[2], this._cachedTextureMatrix), this._cachedTextureMatrix.multiplyToRef(ue.Matrix[3], this._cachedTextureMatrix), this._cachedTextureMatrix.setRowFromFloats(2, this._cachedTextureMatrix.m[12], this._cachedTextureMatrix.m[13], this._cachedTextureMatrix.m[14], 1)) : (this._prepareRowForTextureGeneration(0, 0, 0, this._t0), this._prepareRowForTextureGeneration(1, 0, 0, this._t1), this._prepareRowForTextureGeneration(0, 1, 0, this._t2), this._t1.subtractInPlace(this._t0), this._t2.subtractInPlace(this._t0), he.FromValuesToRef(this._t1.x, this._t1.y, this._t1.z, 0, this._t2.x, this._t2.y, this._t2.z, 0, this._t0.x, this._t0.y, this._t0.z, 0, 0, 0, 0, 1, this._cachedTextureMatrix)); const t = this.getScene(); return t ? (this.optimizeUVAllocation && t.markAllMaterialsAsDirty(1, (r) => r.hasTexture(this)), this._cachedTextureMatrix) : this._cachedTextureMatrix; } /** * Get the current matrix used to apply reflection. This is useful to rotate an environment texture for instance. * @returns The reflection texture transform */ getReflectionTextureMatrix() { const e = this.getScene(); if (!e) return this._cachedReflectionTextureMatrix; if (this.uOffset === this._cachedReflectionUOffset && this.vOffset === this._cachedReflectionVOffset && this.uScale === this._cachedReflectionUScale && this.vScale === this._cachedReflectionVScale && this.coordinatesMode === this._cachedReflectionCoordinatesMode) if (this.coordinatesMode === We.PROJECTION_MODE) { if (this._cachedReflectionProjectionMatrixId === e.getProjectionMatrix().updateFlag) return this._cachedReflectionTextureMatrix; } else return this._cachedReflectionTextureMatrix; this._cachedReflectionTextureMatrix || (this._cachedReflectionTextureMatrix = he.Zero()), this._projectionModeMatrix || (this._projectionModeMatrix = he.Zero()); const t = this._cachedReflectionCoordinatesMode !== this.coordinatesMode; switch (this._cachedReflectionUOffset = this.uOffset, this._cachedReflectionVOffset = this.vOffset, this._cachedReflectionUScale = this.uScale, this._cachedReflectionVScale = this.vScale, this._cachedReflectionCoordinatesMode = this.coordinatesMode, this.coordinatesMode) { case We.PLANAR_MODE: { he.IdentityToRef(this._cachedReflectionTextureMatrix), this._cachedReflectionTextureMatrix[0] = this.uScale, this._cachedReflectionTextureMatrix[5] = this.vScale, this._cachedReflectionTextureMatrix[12] = this.uOffset, this._cachedReflectionTextureMatrix[13] = this.vOffset; break; } case We.PROJECTION_MODE: { he.FromValuesToRef(0.5, 0, 0, 0, 0, -0.5, 0, 0, 0, 0, 0, 0, 0.5, 0.5, 1, 1, this._projectionModeMatrix); const r = e.getProjectionMatrix(); this._cachedReflectionProjectionMatrixId = r.updateFlag, r.multiplyToRef(this._projectionModeMatrix, this._cachedReflectionTextureMatrix); break; } default: he.IdentityToRef(this._cachedReflectionTextureMatrix); break; } return t && e.markAllMaterialsAsDirty(1, (r) => r.hasTexture(this)), this._cachedReflectionTextureMatrix; } /** * Clones the texture. * @returns the cloned texture */ clone() { const e = { noMipmap: this._noMipmap, invertY: this._invertY, samplingMode: this.samplingMode, onLoad: void 0, onError: void 0, buffer: this._texture ? this._texture._buffer : void 0, deleteBuffer: this._deleteBuffer, format: this.textureFormat, mimeType: this.mimeType, loaderOptions: this._loaderOptions, creationFlags: this._creationFlags, useSRGBBuffer: this._useSRGBBuffer }; return jt.Clone(() => new We(this._texture ? this._texture.url : null, this.getScene(), e), this); } /** * Serialize the texture to a JSON representation we can easily use in the respective Parse function. * @returns The JSON representation of the texture */ serialize() { var e, t; const r = this.name; We.SerializeBuffers || this.name.startsWith("data:") && (this.name = ""), this.name.startsWith("data:") && this.url === this.name && (this.url = ""); const n = super.serialize(We._SerializeInternalTextureUniqueId); return n ? ((We.SerializeBuffers || We.ForceSerializeBuffers) && (typeof this._buffer == "string" && this._buffer.substr(0, 5) === "data:" ? (n.base64String = this._buffer, n.name = n.name.replace("data:", "")) : this.url && this.url.startsWith("data:") && this._buffer instanceof Uint8Array ? n.base64String = "data:image/png;base64," + XR(this._buffer) : (We.ForceSerializeBuffers || this.url && this.url.startsWith("blob:") || this._forceSerialize) && (n.base64String = !this._engine || this._engine._features.supportSyncTextureRead ? rQ(this) : nQ(this))), n.invertY = this._invertY, n.samplingMode = this.samplingMode, n._creationFlags = this._creationFlags, n._useSRGBBuffer = this._useSRGBBuffer, We._SerializeInternalTextureUniqueId && (n.internalTextureUniqueId = (t = (e = this._texture) === null || e === void 0 ? void 0 : e.uniqueId) !== null && t !== void 0 ? t : void 0), n.noMipmap = this._noMipmap, this.name = r, n) : null; } /** * Get the current class name of the texture useful for serialization or dynamic coding. * @returns "Texture" */ getClassName() { return "Texture"; } /** * Dispose the texture and release its associated resources. */ dispose() { super.dispose(), this.onLoadObservable.clear(), this._delayedOnLoad = null, this._delayedOnError = null, this._buffer = null; } /** * Parse the JSON representation of a texture in order to recreate the texture in the given scene. * @param parsedTexture Define the JSON representation of the texture * @param scene Define the scene the parsed texture should be instantiated in * @param rootUrl Define the root url of the parsing sequence in the case of relative dependencies * @returns The parsed texture if successful */ static Parse(e, t, r) { if (e.customType) { const o = jI.Instantiate(e.customType).Parse(e, t, r); return e.samplingMode && o.updateSamplingMode && o._samplingMode && o._samplingMode !== e.samplingMode && o.updateSamplingMode(e.samplingMode), o; } if (e.isCube && !e.isRenderTarget) return We._CubeTextureParser(e, t, r); const n = e.internalTextureUniqueId !== void 0; if (!e.name && !e.isRenderTarget && !n) return null; let i; if (n) { const f = t.getEngine().getLoadedTexturesCache(); for (const o of f) if (o.uniqueId === e.internalTextureUniqueId) { i = o; break; } } const s = (f) => { var o; if (f && f._texture && (f._texture._cachedWrapU = null, f._texture._cachedWrapV = null, f._texture._cachedWrapR = null), e.samplingMode) { const d = e.samplingMode; f && f.samplingMode !== d && f.updateSamplingMode(d); } if (f && e.animations) for (let d = 0; d < e.animations.length; d++) { const v = e.animations[d], u = Jo("BABYLON.Animation"); u && f.animations.push(u.Parse(v)); } n && !i && ((o = f == null ? void 0 : f._texture) === null || o === void 0 || o._setUniqueId(e.internalTextureUniqueId)); }; return jt.Parse(() => { var f, o, d; let v = !0; if (e.noMipmap && (v = !1), e.mirrorPlane) { const u = We._CreateMirror(e.name, e.renderTargetSize, t, v); return u._waitingRenderList = e.renderList, u.mirrorPlane = BA.FromArray(e.mirrorPlane), s(u), u; } else if (e.isRenderTarget) { let u = null; if (e.isCube) { if (t.reflectionProbes) for (let l = 0; l < t.reflectionProbes.length; l++) { const P = t.reflectionProbes[l]; if (P.name === e.name) return P.cubeTexture; } } else u = We._CreateRenderTargetTexture(e.name, e.renderTargetSize, t, v, (f = e._creationFlags) !== null && f !== void 0 ? f : 0), u._waitingRenderList = e.renderList; return s(u), u; } else if (e.isVideo) { const u = We._CreateVideoTexture(r + (e.url || e.name), r + (e.src || e.url), t, v, e.invertY, e.samplingMode, e.settings || {}); return s(u), u; } else { let u; if (e.base64String && !i) u = We.CreateFromBase64String(e.base64String, e.base64String, t, !v, e.invertY, e.samplingMode, () => { s(u); }, (o = e._creationFlags) !== null && o !== void 0 ? o : 0, (d = e._useSRGBBuffer) !== null && d !== void 0 ? d : !1), u.name = e.name; else { let l; e.name && (e.name.indexOf("://") > 0 || e.name.startsWith("data:")) ? l = e.name : l = r + e.name, e.url && (e.url.startsWith("data:") || We.UseSerializedUrlIfAny) && (l = e.url); const P = { noMipmap: !v, invertY: e.invertY, samplingMode: e.samplingMode, onLoad: () => { s(u); }, internalTexture: i }; u = new We(l, t, P); } return u; } }, e, t); } /** * Creates a texture from its base 64 representation. * @param data Define the base64 payload without the data: prefix * @param name Define the name of the texture in the scene useful fo caching purpose for instance * @param scene Define the scene the texture should belong to * @param noMipmapOrOptions defines if the texture will require mip maps or not or set of all options to create the texture * @param invertY define if the texture needs to be inverted on the y axis during loading * @param samplingMode define the sampling mode we want for the texture while fetching from it (Texture.NEAREST_SAMPLINGMODE...) * @param onLoad define a callback triggered when the texture has been loaded * @param onError define a callback triggered when an error occurred during the loading session * @param format define the format of the texture we are trying to load (Engine.TEXTUREFORMAT_RGBA...) * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg) * @returns the created texture */ static CreateFromBase64String(e, t, r, n, i, s = We.TRILINEAR_SAMPLINGMODE, a = null, f = null, o = 5, d) { return new We("data:" + t, r, n, i, s, a, f, e, !1, o, void 0, void 0, d); } /** * Creates a texture from its data: representation. (data: will be added in case only the payload has been passed in) * @param name Define the name of the texture in the scene useful fo caching purpose for instance * @param buffer define the buffer to load the texture from in case the texture is loaded from a buffer representation * @param scene Define the scene the texture should belong to * @param deleteBuffer define if the buffer we are loading the texture from should be deleted after load * @param noMipmapOrOptions defines if the texture will require mip maps or not or set of all options to create the texture * @param invertY define if the texture needs to be inverted on the y axis during loading * @param samplingMode define the sampling mode we want for the texture while fetching from it (Texture.NEAREST_SAMPLINGMODE...) * @param onLoad define a callback triggered when the texture has been loaded * @param onError define a callback triggered when an error occurred during the loading session * @param format define the format of the texture we are trying to load (Engine.TEXTUREFORMAT_RGBA...) * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg) * @returns the created texture */ static LoadFromDataString(e, t, r, n = !1, i, s = !0, a = We.TRILINEAR_SAMPLINGMODE, f = null, o = null, d = 5, v) { return e.substr(0, 5) !== "data:" && (e = "data:" + e), new We(e, r, i, s, a, f, o, t, n, d, void 0, void 0, v); } } We.SerializeBuffers = !0; We.ForceSerializeBuffers = !1; We.OnTextureLoadErrorObservable = new Oe(); We._SerializeInternalTextureUniqueId = !1; We._CubeTextureParser = (A, e, t) => { throw qn("CubeTexture"); }; We._CreateMirror = (A, e, t, r) => { throw qn("MirrorTexture"); }; We._CreateRenderTargetTexture = (A, e, t, r, n) => { throw qn("RenderTargetTexture"); }; We.NEAREST_SAMPLINGMODE = 1; We.NEAREST_NEAREST_MIPLINEAR = 8; We.BILINEAR_SAMPLINGMODE = 2; We.LINEAR_LINEAR_MIPNEAREST = 11; We.TRILINEAR_SAMPLINGMODE = 3; We.LINEAR_LINEAR_MIPLINEAR = 3; We.NEAREST_NEAREST_MIPNEAREST = 4; We.NEAREST_LINEAR_MIPNEAREST = 5; We.NEAREST_LINEAR_MIPLINEAR = 6; We.NEAREST_LINEAR = 7; We.NEAREST_NEAREST = 1; We.LINEAR_NEAREST_MIPNEAREST = 9; We.LINEAR_NEAREST_MIPLINEAR = 10; We.LINEAR_LINEAR = 2; We.LINEAR_NEAREST = 12; We.EXPLICIT_MODE = 0; We.SPHERICAL_MODE = 1; We.PLANAR_MODE = 2; We.CUBIC_MODE = 3; We.PROJECTION_MODE = 4; We.SKYBOX_MODE = 5; We.INVCUBIC_MODE = 6; We.EQUIRECTANGULAR_MODE = 7; We.FIXED_EQUIRECTANGULAR_MODE = 8; We.FIXED_EQUIRECTANGULAR_MIRRORED_MODE = 9; We.CLAMP_ADDRESSMODE = 0; We.WRAP_ADDRESSMODE = 1; We.MIRROR_ADDRESSMODE = 2; We.UseSerializedUrlIfAny = !1; C([ M() ], We.prototype, "url", void 0); C([ M() ], We.prototype, "uOffset", void 0); C([ M() ], We.prototype, "vOffset", void 0); C([ M() ], We.prototype, "uScale", void 0); C([ M() ], We.prototype, "vScale", void 0); C([ M() ], We.prototype, "uAng", void 0); C([ M() ], We.prototype, "vAng", void 0); C([ M() ], We.prototype, "wAng", void 0); C([ M() ], We.prototype, "uRotationCenter", void 0); C([ M() ], We.prototype, "vRotationCenter", void 0); C([ M() ], We.prototype, "wRotationCenter", void 0); C([ M() ], We.prototype, "homogeneousRotationInUVTransform", void 0); C([ M() ], We.prototype, "isBlocking", null); Ue("BABYLON.Texture", We); jt._TextureParser = We.Parse; hr.prototype.updateRawTexture = function(A, e, t, r, n = null, i = 0, s = !1) { if (!A) return; const a = this._getRGBABufferInternalSizedFormat(i, t, s), f = this._getInternalFormat(t), o = this._getWebGLTextureType(i); this._bindTextureDirectly(this._gl.TEXTURE_2D, A, !0), this._unpackFlipY(r === void 0 ? !0 : !!r), this._doNotHandleContextLost || (A._bufferView = e, A.format = t, A.type = i, A.invertY = r, A._compression = n), A.width % 4 !== 0 && this._gl.pixelStorei(this._gl.UNPACK_ALIGNMENT, 1), n && e ? this._gl.compressedTexImage2D(this._gl.TEXTURE_2D, 0, this.getCaps().s3tc[n], A.width, A.height, 0, e) : this._gl.texImage2D(this._gl.TEXTURE_2D, 0, a, A.width, A.height, 0, f, o, e), A.generateMipMaps && this._gl.generateMipmap(this._gl.TEXTURE_2D), this._bindTextureDirectly(this._gl.TEXTURE_2D, null), A.isReady = !0; }; hr.prototype.createRawTexture = function(A, e, t, r, n, i, s, a = null, f = 0, o = 0, d = !1) { const v = new As(this, ri.Raw); v.baseWidth = e, v.baseHeight = t, v.width = e, v.height = t, v.format = r, v.generateMipMaps = n, v.samplingMode = s, v.invertY = i, v._compression = a, v.type = f, v._useSRGBBuffer = this._getUseSRGBBuffer(d, !n), this._doNotHandleContextLost || (v._bufferView = A), this.updateRawTexture(v, A, r, i, a, f, v._useSRGBBuffer), this._bindTextureDirectly(this._gl.TEXTURE_2D, v, !0); const u = this._getSamplingParameters(s, n); return this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_MAG_FILTER, u.mag), this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_MIN_FILTER, u.min), n && this._gl.generateMipmap(this._gl.TEXTURE_2D), this._bindTextureDirectly(this._gl.TEXTURE_2D, null), this._internalTexturesCache.push(v), v; }; hr.prototype.createRawCubeTexture = function(A, e, t, r, n, i, s, a = null) { const f = this._gl, o = new As(this, ri.CubeRaw); o.isCube = !0, o.format = t, o.type = r, this._doNotHandleContextLost || (o._bufferViewArray = A); const d = this._getWebGLTextureType(r); let v = this._getInternalFormat(t); v === f.RGB && (v = f.RGBA), d === f.FLOAT && !this._caps.textureFloatLinearFiltering ? (n = !1, s = 1, Se.Warn("Float texture filtering is not supported. Mipmap generation and sampling mode are forced to false and TEXTURE_NEAREST_SAMPLINGMODE, respectively.")) : d === this._gl.HALF_FLOAT_OES && !this._caps.textureHalfFloatLinearFiltering ? (n = !1, s = 1, Se.Warn("Half float texture filtering is not supported. Mipmap generation and sampling mode are forced to false and TEXTURE_NEAREST_SAMPLINGMODE, respectively.")) : d === f.FLOAT && !this._caps.textureFloatRender ? (n = !1, Se.Warn("Render to float textures is not supported. Mipmap generation forced to false.")) : d === f.HALF_FLOAT && !this._caps.colorBufferFloat && (n = !1, Se.Warn("Render to half float textures is not supported. Mipmap generation forced to false.")); const u = e, l = u; if (o.width = u, o.height = l, o.invertY = i, o._compression = a, !this.needPOTTextures || ye.IsExponentOfTwo(o.width) && ye.IsExponentOfTwo(o.height) || (n = !1), A) this.updateRawCubeTexture(o, A, t, r, i, a); else { const c = this._getRGBABufferInternalSizedFormat(r), H = 0; this._bindTextureDirectly(f.TEXTURE_CUBE_MAP, o, !0); for (let T = 0; T < 6; T++) a ? f.compressedTexImage2D(f.TEXTURE_CUBE_MAP_POSITIVE_X + T, H, this.getCaps().s3tc[a], o.width, o.height, 0, void 0) : f.texImage2D(f.TEXTURE_CUBE_MAP_POSITIVE_X + T, H, c, o.width, o.height, 0, v, d, null); this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, null); } this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, o, !0), A && n && this._gl.generateMipmap(this._gl.TEXTURE_CUBE_MAP); const p = this._getSamplingParameters(s, n); return f.texParameteri(f.TEXTURE_CUBE_MAP, f.TEXTURE_MAG_FILTER, p.mag), f.texParameteri(f.TEXTURE_CUBE_MAP, f.TEXTURE_MIN_FILTER, p.min), f.texParameteri(f.TEXTURE_CUBE_MAP, f.TEXTURE_WRAP_S, f.CLAMP_TO_EDGE), f.texParameteri(f.TEXTURE_CUBE_MAP, f.TEXTURE_WRAP_T, f.CLAMP_TO_EDGE), this._bindTextureDirectly(f.TEXTURE_CUBE_MAP, null), o.generateMipMaps = n, o.samplingMode = s, o.isReady = !0, o; }; hr.prototype.updateRawCubeTexture = function(A, e, t, r, n, i = null, s = 0) { A._bufferViewArray = e, A.format = t, A.type = r, A.invertY = n, A._compression = i; const a = this._gl, f = this._getWebGLTextureType(r); let o = this._getInternalFormat(t); const d = this._getRGBABufferInternalSizedFormat(r); let v = !1; o === a.RGB && (o = a.RGBA, v = !0), this._bindTextureDirectly(a.TEXTURE_CUBE_MAP, A, !0), this._unpackFlipY(n === void 0 ? !0 : !!n), A.width % 4 !== 0 && a.pixelStorei(a.UNPACK_ALIGNMENT, 1); for (let l = 0; l < 6; l++) { let P = e[l]; i ? a.compressedTexImage2D(a.TEXTURE_CUBE_MAP_POSITIVE_X + l, s, this.getCaps().s3tc[i], A.width, A.height, 0, P) : (v && (P = Pee(P, A.width, A.height, r)), a.texImage2D(a.TEXTURE_CUBE_MAP_POSITIVE_X + l, s, d, A.width, A.height, 0, o, f, P)); } (!this.needPOTTextures || ye.IsExponentOfTwo(A.width) && ye.IsExponentOfTwo(A.height)) && A.generateMipMaps && s === 0 && this._gl.generateMipmap(this._gl.TEXTURE_CUBE_MAP), this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, null), A.isReady = !0; }; hr.prototype.createRawCubeTextureFromUrl = function(A, e, t, r, n, i, s, a, f = null, o = null, d = 3, v = !1) { const u = this._gl, l = this.createRawCubeTexture(null, t, r, n, !i, v, d, null); e == null || e.addPendingData(l), l.url = A, l.isReady = !1, this._internalTexturesCache.push(l); const P = (c, H) => { e == null || e.removePendingData(l), o && c && o(c.status + " " + c.statusText, H); }, p = (c) => { const H = l.width, T = s(c); if (T) { if (a) { const q = this._getWebGLTextureType(n); let b = this._getInternalFormat(r); const j = this._getRGBABufferInternalSizedFormat(n); let w = !1; b === u.RGB && (b = u.RGBA, w = !0), this._bindTextureDirectly(u.TEXTURE_CUBE_MAP, l, !0), this._unpackFlipY(!1); const m = a(T); for (let I = 0; I < m.length; I++) { const N = H >> I; for (let k = 0; k < 6; k++) { let R = m[I][k]; w && (R = Pee(R, N, N, n)), u.texImage2D(k, I, j, N, N, 0, b, q, R); } } this._bindTextureDirectly(u.TEXTURE_CUBE_MAP, null); } else this.updateRawCubeTexture(l, T, r, n, v); l.isReady = !0, e == null || e.removePendingData(l), l.onLoadedObservable.notifyObservers(l), l.onLoadedObservable.clear(), f && f(); } }; return this._loadFile(A, (c) => { p(c); }, void 0, e == null ? void 0 : e.offlineProvider, !0, P), l; }; function Pee(A, e, t, r) { let n, i = 1; r === 1 ? n = new Float32Array(e * t * 4) : r === 2 ? (n = new Uint16Array(e * t * 4), i = 15360) : r === 7 ? n = new Uint32Array(e * t * 4) : n = new Uint8Array(e * t * 4); for (let s = 0; s < e; s++) for (let a = 0; a < t; a++) { const f = (a * e + s) * 3, o = (a * e + s) * 4; n[o + 0] = A[f + 0], n[o + 1] = A[f + 1], n[o + 2] = A[f + 2], n[o + 3] = i; } return n; } function cee(A) { return function(e, t, r, n, i, s, a, f, o = null, d = 0) { const v = A ? this._gl.TEXTURE_3D : this._gl.TEXTURE_2D_ARRAY, u = A ? ri.Raw3D : ri.Raw2DArray, l = new As(this, u); l.baseWidth = t, l.baseHeight = r, l.baseDepth = n, l.width = t, l.height = r, l.depth = n, l.format = i, l.type = d, l.generateMipMaps = s, l.samplingMode = f, A ? l.is3D = !0 : l.is2DArray = !0, this._doNotHandleContextLost || (l._bufferView = e), A ? this.updateRawTexture3D(l, e, i, a, o, d) : this.updateRawTexture2DArray(l, e, i, a, o, d), this._bindTextureDirectly(v, l, !0); const P = this._getSamplingParameters(f, s); return this._gl.texParameteri(v, this._gl.TEXTURE_MAG_FILTER, P.mag), this._gl.texParameteri(v, this._gl.TEXTURE_MIN_FILTER, P.min), s && this._gl.generateMipmap(v), this._bindTextureDirectly(v, null), this._internalTexturesCache.push(l), l; }; } hr.prototype.createRawTexture2DArray = cee(!1); hr.prototype.createRawTexture3D = cee(!0); function pee(A) { return function(e, t, r, n, i = null, s = 0) { const a = A ? this._gl.TEXTURE_3D : this._gl.TEXTURE_2D_ARRAY, f = this._getWebGLTextureType(s), o = this._getInternalFormat(r), d = this._getRGBABufferInternalSizedFormat(s, r); this._bindTextureDirectly(a, e, !0), this._unpackFlipY(n === void 0 ? !0 : !!n), this._doNotHandleContextLost || (e._bufferView = t, e.format = r, e.invertY = n, e._compression = i), e.width % 4 !== 0 && this._gl.pixelStorei(this._gl.UNPACK_ALIGNMENT, 1), i && t ? this._gl.compressedTexImage3D(a, 0, this.getCaps().s3tc[i], e.width, e.height, e.depth, 0, t) : this._gl.texImage3D(a, 0, d, e.width, e.height, e.depth, 0, o, f, t), e.generateMipMaps && this._gl.generateMipmap(a), this._bindTextureDirectly(a, null), e.isReady = !0; }; } hr.prototype.updateRawTexture2DArray = pee(!1); hr.prototype.updateRawTexture3D = pee(!0); class Bo extends We { /** * Instantiates a new RawTexture. * Raw texture can help creating a texture directly from an array of data. * This can be super useful if you either get the data from an uncompressed source or * if you wish to create your texture pixel by pixel. * @param data define the array of data to use to create the texture (null to create an empty texture) * @param width define the width of the texture * @param height define the height of the texture * @param format define the format of the data (RGB, RGBA... Engine.TEXTUREFORMAT_xxx) * @param sceneOrEngine defines the scene or engine the texture will belong to * @param generateMipMaps define whether mip maps should be generated or not * @param invertY define if the data should be flipped on Y when uploaded to the GPU * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE) * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx) * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg) * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU). */ constructor(e, t, r, n, i, s = !0, a = !1, f = 3, o = 0, d, v) { super(null, i, !s, a, void 0, void 0, void 0, void 0, void 0, void 0, void 0, void 0, d), this.format = n, this._engine && (!this._engine._caps.textureFloatLinearFiltering && o === 1 && (f = 1), !this._engine._caps.textureHalfFloatLinearFiltering && o === 2 && (f = 1), this._texture = this._engine.createRawTexture(e, t, r, n, s, a, f, null, o, d ?? 0, v ?? !1), this.wrapU = We.CLAMP_ADDRESSMODE, this.wrapV = We.CLAMP_ADDRESSMODE); } /** * Updates the texture underlying data. * @param data Define the new data of the texture */ update(e) { this._getEngine().updateRawTexture(this._texture, e, this._texture.format, this._texture.invertY, null, this._texture.type, this._texture._useSRGBBuffer); } /** * Creates a luminance texture from some data. * @param data Define the texture data * @param width Define the width of the texture * @param height Define the height of the texture * @param sceneOrEngine defines the scene or engine the texture will belong to * @param generateMipMaps Define whether or not to create mip maps for the texture * @param invertY define if the data should be flipped on Y when uploaded to the GPU * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE) * @returns the luminance texture */ static CreateLuminanceTexture(e, t, r, n, i = !0, s = !1, a = 3) { return new Bo(e, t, r, 1, n, i, s, a); } /** * Creates a luminance alpha texture from some data. * @param data Define the texture data * @param width Define the width of the texture * @param height Define the height of the texture * @param sceneOrEngine defines the scene or engine the texture will belong to * @param generateMipMaps Define whether or not to create mip maps for the texture * @param invertY define if the data should be flipped on Y when uploaded to the GPU * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE) * @returns the luminance alpha texture */ static CreateLuminanceAlphaTexture(e, t, r, n, i = !0, s = !1, a = 3) { return new Bo(e, t, r, 2, n, i, s, a); } /** * Creates an alpha texture from some data. * @param data Define the texture data * @param width Define the width of the texture * @param height Define the height of the texture * @param sceneOrEngine defines the scene or engine the texture will belong to * @param generateMipMaps Define whether or not to create mip maps for the texture * @param invertY define if the data should be flipped on Y when uploaded to the GPU * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE) * @returns the alpha texture */ static CreateAlphaTexture(e, t, r, n, i = !0, s = !1, a = 3) { return new Bo(e, t, r, 0, n, i, s, a); } /** * Creates a RGB texture from some data. * @param data Define the texture data * @param width Define the width of the texture * @param height Define the height of the texture * @param sceneOrEngine defines the scene or engine the texture will belong to * @param generateMipMaps Define whether or not to create mip maps for the texture * @param invertY define if the data should be flipped on Y when uploaded to the GPU * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE) * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx) * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg) * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU). * @returns the RGB alpha texture */ static CreateRGBTexture(e, t, r, n, i = !0, s = !1, a = 3, f = 0, o = 0, d = !1) { return new Bo(e, t, r, 4, n, i, s, a, f, o, d); } /** * Creates a RGBA texture from some data. * @param data Define the texture data * @param width Define the width of the texture * @param height Define the height of the texture * @param sceneOrEngine defines the scene or engine the texture will belong to * @param generateMipMaps Define whether or not to create mip maps for the texture * @param invertY define if the data should be flipped on Y when uploaded to the GPU * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE) * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx) * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg) * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU). * @returns the RGBA texture */ static CreateRGBATexture(e, t, r, n, i = !0, s = !1, a = 3, f = 0, o = 0, d = !1) { return new Bo(e, t, r, 5, n, i, s, a, f, o, d); } /** * Creates a RGBA storage texture from some data. * @param data Define the texture data * @param width Define the width of the texture * @param height Define the height of the texture * @param sceneOrEngine defines the scene or engine the texture will belong to * @param generateMipMaps Define whether or not to create mip maps for the texture * @param invertY define if the data should be flipped on Y when uploaded to the GPU * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE) * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx) * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU). * @returns the RGBA texture */ static CreateRGBAStorageTexture(e, t, r, n, i = !0, s = !1, a = 3, f = 0, o = !1) { return new Bo(e, t, r, 5, n, i, s, a, f, 1, o); } /** * Creates a R texture from some data. * @param data Define the texture data * @param width Define the width of the texture * @param height Define the height of the texture * @param sceneOrEngine defines the scene or engine the texture will belong to * @param generateMipMaps Define whether or not to create mip maps for the texture * @param invertY define if the data should be flipped on Y when uploaded to the GPU * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE) * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx) * @returns the R texture */ static CreateRTexture(e, t, r, n, i = !0, s = !1, a = We.TRILINEAR_SAMPLINGMODE, f = 1) { return new Bo(e, t, r, 6, n, i, s, a, f); } /** * Creates a R storage texture from some data. * @param data Define the texture data * @param width Define the width of the texture * @param height Define the height of the texture * @param sceneOrEngine defines the scene or engine the texture will belong to * @param generateMipMaps Define whether or not to create mip maps for the texture * @param invertY define if the data should be flipped on Y when uploaded to the GPU * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE) * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx) * @returns the R texture */ static CreateRStorageTexture(e, t, r, n, i = !0, s = !1, a = We.TRILINEAR_SAMPLINGMODE, f = 1) { return new Bo(e, t, r, 6, n, i, s, a, f, 1); } } class X6e { /** * Create a new VertexAnimationBaker object which can help baking animations into a texture. * @param scene Defines the scene the VAT belongs to * @param mesh Defines the mesh the VAT belongs to */ constructor(e, t) { this._scene = e, this._mesh = t; } /** * Bakes the animation into the texture. This should be called once, when the * scene starts, so the VAT is generated and associated to the mesh. * @param ranges Defines the ranges in the animation that will be baked. * @returns The array of matrix transforms for each vertex (columns) and frame (rows), as a Float32Array. */ async bakeVertexData(e) { if (!this._mesh.skeleton) throw new Error("No skeleton in this mesh."); const t = this._mesh.skeleton.bones.length, r = e.reduce((a, f) => a + f.to - f.from + 1, 0); if (isNaN(r)) throw new Error("Invalid animation ranges."); let n = 0; const i = (t + 1) * 4 * 4 * r, s = new Float32Array(i); this._scene.stopAnimation(this._mesh), this._mesh.skeleton.returnToRest(); for (const a of e) for (let f = a.from; f <= a.to; f++) await this._executeAnimationFrame(s, f, n++); return s; } /** * Runs an animation frame and stores its vertex data * * @param vertexData The array to save data to. * @param frameIndex Current frame in the skeleton animation to render. * @param textureIndex Current index of the texture data. */ async _executeAnimationFrame(e, t, r) { return new Promise((n, i) => { this._scene.beginAnimation(this._mesh.skeleton, t, t, !1, 1, () => { const s = this._mesh.skeleton.getTransformMatrices(this._mesh); e.set(s, r * s.length), n(); }); }); } /** * Builds a vertex animation texture given the vertexData in an array. * @param vertexData The vertex animation data. You can generate it with bakeVertexData(). * @returns The vertex animation texture to be used with BakedVertexAnimationManager. */ textureFromBakedVertexData(e) { if (!this._mesh.skeleton) throw new Error("No skeleton in this mesh."); const t = this._mesh.skeleton.bones.length, r = Bo.CreateRGBATexture(e, (t + 1) * 4, e.length / ((t + 1) * 4 * 4), this._scene, !1, !1, We.NEAREST_NEAREST, 1); return r.name = "VAT" + this._mesh.skeleton.name, r; } /** * Serializes our vertexData to an object, with a nice string for the vertexData. * @param vertexData The vertex array data. * @returns This object serialized to a JS dict. */ serializeBakedVertexDataToObject(e) { if (!this._mesh.skeleton) throw new Error("No skeleton in this mesh."); const t = this._mesh.skeleton.bones.length, r = (t + 1) * 4, n = e.length / ((t + 1) * 4 * 4); return { vertexData: XR(e), width: r, height: n }; } /** * Loads previously baked data. * @param data The object as serialized by serializeBakedVertexDataToObject() * @returns The array of matrix transforms for each vertex (columns) and frame (rows), as a Float32Array. */ loadBakedVertexDataFromObject(e) { return new Float32Array(TR(e.vertexData)); } /** * Serializes our vertexData to a JSON string, with a nice string for the vertexData. * Should be called right after bakeVertexData(). * @param vertexData The vertex array data. * @returns This object serialized to a safe string. */ serializeBakedVertexDataToJSON(e) { return JSON.stringify(this.serializeBakedVertexDataToObject(e)); } /** * Loads previously baked data in string format. * @param json The json string as serialized by serializeBakedVertexDataToJSON(). * @returns The array of matrix transforms for each vertex (columns) and frame (rows), as a Float32Array. */ loadBakedVertexDataFromJSON(e) { return this.loadBakedVertexDataFromObject(JSON.parse(e)); } } class hee { constructor() { this._zoomStopsAnimation = !1, this._idleRotationSpeed = 0.05, this._idleRotationWaitTime = 2e3, this._idleRotationSpinupTime = 2e3, this.targetAlpha = null, this._isPointerDown = !1, this._lastFrameTime = null, this._lastInteractionTime = -1 / 0, this._cameraRotationSpeed = 0, this._lastFrameRadius = 0; } /** * Gets the name of the behavior. */ get name() { return "AutoRotation"; } /** * Sets the flag that indicates if user zooming should stop animation. */ set zoomStopsAnimation(e) { this._zoomStopsAnimation = e; } /** * Gets the flag that indicates if user zooming should stop animation. */ get zoomStopsAnimation() { return this._zoomStopsAnimation; } /** * Sets the default speed at which the camera rotates around the model. */ set idleRotationSpeed(e) { this._idleRotationSpeed = e; } /** * Gets the default speed at which the camera rotates around the model. */ get idleRotationSpeed() { return this._idleRotationSpeed; } /** * Sets the time (in milliseconds) to wait after user interaction before the camera starts rotating. */ set idleRotationWaitTime(e) { this._idleRotationWaitTime = e; } /** * Gets the time (milliseconds) to wait after user interaction before the camera starts rotating. */ get idleRotationWaitTime() { return this._idleRotationWaitTime; } /** * Sets the time (milliseconds) to take to spin up to the full idle rotation speed. */ set idleRotationSpinupTime(e) { this._idleRotationSpinupTime = e; } /** * Gets the time (milliseconds) to take to spin up to the full idle rotation speed. */ get idleRotationSpinupTime() { return this._idleRotationSpinupTime; } /** * Gets a value indicating if the camera is currently rotating because of this behavior */ get rotationInProgress() { return Math.abs(this._cameraRotationSpeed) > 0; } /** * Initializes the behavior. */ init() { } /** * Attaches the behavior to its arc rotate camera. * @param camera Defines the camera to attach the behavior to */ attach(e) { this._attachedCamera = e; const t = this._attachedCamera.getScene(); this._onPrePointerObservableObserver = t.onPrePointerObservable.add((r) => { if (r.type === ir.POINTERDOWN) { this._isPointerDown = !0; return; } r.type === ir.POINTERUP && (this._isPointerDown = !1); }), this._onAfterCheckInputsObserver = e.onAfterCheckInputsObservable.add(() => { if (this._reachTargetAlpha()) return; const r = Yi.Now; let n = 0; this._lastFrameTime != null && (n = r - this._lastFrameTime), this._lastFrameTime = r, this._applyUserInteraction(); const i = r - this._lastInteractionTime - this._idleRotationWaitTime, s = Math.max(Math.min(i / this._idleRotationSpinupTime, 1), 0); this._cameraRotationSpeed = this._idleRotationSpeed * s, this._attachedCamera && (this._attachedCamera.alpha -= this._cameraRotationSpeed * (n / 1e3)); }); } /** * Detaches the behavior from its current arc rotate camera. */ detach() { if (!this._attachedCamera) return; const e = this._attachedCamera.getScene(); this._onPrePointerObservableObserver && e.onPrePointerObservable.remove(this._onPrePointerObservableObserver), this._attachedCamera.onAfterCheckInputsObservable.remove(this._onAfterCheckInputsObserver), this._attachedCamera = null; } /** * Force-reset the last interaction time * @param customTime an optional time that will be used instead of the current last interaction time. For example `Date.now()` */ resetLastInteractionTime(e) { this._lastInteractionTime = e ?? Yi.Now; } /** * Returns true if camera alpha reaches the target alpha * @returns true if camera alpha reaches the target alpha */ _reachTargetAlpha() { return this._attachedCamera && this.targetAlpha ? Math.abs(this._attachedCamera.alpha - this.targetAlpha) < Dn : !1; } /** * Returns true if user is scrolling. * @returns true if user is scrolling. */ _userIsZooming() { return this._attachedCamera ? this._attachedCamera.inertialRadiusOffset !== 0 : !1; } _shouldAnimationStopForInteraction() { if (!this._attachedCamera) return !1; let e = !1; return this._lastFrameRadius === this._attachedCamera.radius && this._attachedCamera.inertialRadiusOffset !== 0 && (e = !0), this._lastFrameRadius = this._attachedCamera.radius, this._zoomStopsAnimation ? e : this._userIsZooming(); } /** * Applies any current user interaction to the camera. Takes into account maximum alpha rotation. */ _applyUserInteraction() { this._userIsMoving() && !this._shouldAnimationStopForInteraction() && (this._lastInteractionTime = Yi.Now); } // Tools _userIsMoving() { return this._attachedCamera ? this._attachedCamera.inertialAlphaOffset !== 0 || this._attachedCamera.inertialBetaOffset !== 0 || this._attachedCamera.inertialRadiusOffset !== 0 || this._attachedCamera.inertialPanningX !== 0 || this._attachedCamera.inertialPanningY !== 0 || this._isPointerDown : !1; } } class fD { constructor() { this.transitionDuration = 450, this.lowerRadiusTransitionRange = 2, this.upperRadiusTransitionRange = -2, this._autoTransitionRange = !1, this._radiusIsAnimating = !1, this._radiusBounceTransition = null, this._animatables = new Array(); } /** * Gets the name of the behavior. */ get name() { return "Bouncing"; } /** * Gets a value indicating if the lowerRadiusTransitionRange and upperRadiusTransitionRange are defined automatically */ get autoTransitionRange() { return this._autoTransitionRange; } /** * Sets a value indicating if the lowerRadiusTransitionRange and upperRadiusTransitionRange are defined automatically * Transition ranges will be set to 5% of the bounding box diagonal in world space */ set autoTransitionRange(e) { if (this._autoTransitionRange === e) return; this._autoTransitionRange = e; const t = this._attachedCamera; t && (e ? this._onMeshTargetChangedObserver = t.onMeshTargetChangedObservable.add((r) => { if (!r) return; r.computeWorldMatrix(!0); const n = r.getBoundingInfo().diagonalLength; this.lowerRadiusTransitionRange = n * 0.05, this.upperRadiusTransitionRange = n * 0.05; }) : this._onMeshTargetChangedObserver && t.onMeshTargetChangedObservable.remove(this._onMeshTargetChangedObserver)); } /** * Initializes the behavior. */ init() { } /** * Attaches the behavior to its arc rotate camera. * @param camera Defines the camera to attach the behavior to */ attach(e) { this._attachedCamera = e, this._onAfterCheckInputsObserver = e.onAfterCheckInputsObservable.add(() => { this._attachedCamera && (this._isRadiusAtLimit(this._attachedCamera.lowerRadiusLimit) && this._applyBoundRadiusAnimation(this.lowerRadiusTransitionRange), this._isRadiusAtLimit(this._attachedCamera.upperRadiusLimit) && this._applyBoundRadiusAnimation(this.upperRadiusTransitionRange)); }); } /** * Detaches the behavior from its current arc rotate camera. */ detach() { this._attachedCamera && (this._onAfterCheckInputsObserver && this._attachedCamera.onAfterCheckInputsObservable.remove(this._onAfterCheckInputsObserver), this._onMeshTargetChangedObserver && this._attachedCamera.onMeshTargetChangedObservable.remove(this._onMeshTargetChangedObserver), this._attachedCamera = null); } /** * Checks if the camera radius is at the specified limit. Takes into account animation locks. * @param radiusLimit The limit to check against. * @returns Bool to indicate if at limit. */ _isRadiusAtLimit(e) { return this._attachedCamera ? this._attachedCamera.radius === e && !this._radiusIsAnimating : !1; } /** * Applies an animation to the radius of the camera, extending by the radiusDelta. * @param radiusDelta The delta by which to animate to. Can be negative. */ _applyBoundRadiusAnimation(e) { if (!this._attachedCamera) return; this._radiusBounceTransition || (fD.EasingFunction.setEasingMode(fD.EasingMode), this._radiusBounceTransition = st.CreateAnimation("radius", st.ANIMATIONTYPE_FLOAT, 60, fD.EasingFunction)), this._cachedWheelPrecision = this._attachedCamera.wheelPrecision, this._attachedCamera.wheelPrecision = 1 / 0, this._attachedCamera.inertialRadiusOffset = 0, this.stopAllAnimations(), this._radiusIsAnimating = !0; const t = st.TransitionTo("radius", this._attachedCamera.radius + e, this._attachedCamera, this._attachedCamera.getScene(), 60, this._radiusBounceTransition, this.transitionDuration, () => this._clearAnimationLocks()); t && this._animatables.push(t); } /** * Removes all animation locks. Allows new animations to be added to any of the camera properties. */ _clearAnimationLocks() { this._radiusIsAnimating = !1, this._attachedCamera && (this._attachedCamera.wheelPrecision = this._cachedWheelPrecision); } /** * Stops and removes all animations that have been applied to the camera */ stopAllAnimations() { for (this._attachedCamera && (this._attachedCamera.animations = []); this._animatables.length; ) this._animatables[0].onAnimationEnd = null, this._animatables[0].stop(), this._animatables.shift(); } } fD.EasingFunction = new _$(0.3); fD.EasingMode = u1.EASINGMODE_EASEOUT; class Vu { constructor() { this.onTargetFramingAnimationEndObservable = new Oe(), this._mode = Vu.FitFrustumSidesMode, this._radiusScale = 1, this._positionScale = 0.5, this._defaultElevation = 0.3, this._elevationReturnTime = 1500, this._elevationReturnWaitTime = 1e3, this._zoomStopsAnimation = !1, this._framingTime = 1500, this.autoCorrectCameraLimitsAndSensibility = !0, this._isPointerDown = !1, this._lastInteractionTime = -1 / 0, this._animatables = new Array(), this._betaIsAnimating = !1; } /** * Gets the name of the behavior. */ get name() { return "Framing"; } /** * Sets the current mode used by the behavior */ set mode(e) { this._mode = e; } /** * Gets current mode used by the behavior. */ get mode() { return this._mode; } /** * Sets the scale applied to the radius (1 by default) */ set radiusScale(e) { this._radiusScale = e; } /** * Gets the scale applied to the radius */ get radiusScale() { return this._radiusScale; } /** * Sets the scale to apply on Y axis to position camera focus. 0.5 by default which means the center of the bounding box. */ set positionScale(e) { this._positionScale = e; } /** * Gets the scale to apply on Y axis to position camera focus. 0.5 by default which means the center of the bounding box. */ get positionScale() { return this._positionScale; } /** * Sets the angle above/below the horizontal plane to return to when the return to default elevation idle * behaviour is triggered, in radians. */ set defaultElevation(e) { this._defaultElevation = e; } /** * Gets the angle above/below the horizontal plane to return to when the return to default elevation idle * behaviour is triggered, in radians. */ get defaultElevation() { return this._defaultElevation; } /** * Sets the time (in milliseconds) taken to return to the default beta position. * Negative value indicates camera should not return to default. */ set elevationReturnTime(e) { this._elevationReturnTime = e; } /** * Gets the time (in milliseconds) taken to return to the default beta position. * Negative value indicates camera should not return to default. */ get elevationReturnTime() { return this._elevationReturnTime; } /** * Sets the delay (in milliseconds) taken before the camera returns to the default beta position. */ set elevationReturnWaitTime(e) { this._elevationReturnWaitTime = e; } /** * Gets the delay (in milliseconds) taken before the camera returns to the default beta position. */ get elevationReturnWaitTime() { return this._elevationReturnWaitTime; } /** * Sets the flag that indicates if user zooming should stop animation. */ set zoomStopsAnimation(e) { this._zoomStopsAnimation = e; } /** * Gets the flag that indicates if user zooming should stop animation. */ get zoomStopsAnimation() { return this._zoomStopsAnimation; } /** * Sets the transition time when framing the mesh, in milliseconds */ set framingTime(e) { this._framingTime = e; } /** * Gets the transition time when framing the mesh, in milliseconds */ get framingTime() { return this._framingTime; } /** * Initializes the behavior. */ init() { } /** * Attaches the behavior to its arc rotate camera. * @param camera Defines the camera to attach the behavior to */ attach(e) { this._attachedCamera = e; const t = this._attachedCamera.getScene(); Vu.EasingFunction.setEasingMode(Vu.EasingMode), this._onPrePointerObservableObserver = t.onPrePointerObservable.add((r) => { if (r.type === ir.POINTERDOWN) { this._isPointerDown = !0; return; } r.type === ir.POINTERUP && (this._isPointerDown = !1); }), this._onMeshTargetChangedObserver = e.onMeshTargetChangedObservable.add((r) => { r && this.zoomOnMesh(r, void 0, () => { this.onTargetFramingAnimationEndObservable.notifyObservers(); }); }), this._onAfterCheckInputsObserver = e.onAfterCheckInputsObservable.add(() => { this._applyUserInteraction(), this._maintainCameraAboveGround(); }); } /** * Detaches the behavior from its current arc rotate camera. */ detach() { if (!this._attachedCamera) return; const e = this._attachedCamera.getScene(); this._onPrePointerObservableObserver && e.onPrePointerObservable.remove(this._onPrePointerObservableObserver), this._onAfterCheckInputsObserver && this._attachedCamera.onAfterCheckInputsObservable.remove(this._onAfterCheckInputsObserver), this._onMeshTargetChangedObserver && this._attachedCamera.onMeshTargetChangedObservable.remove(this._onMeshTargetChangedObserver), this._attachedCamera = null; } /** * Targets the given mesh and updates zoom level accordingly. * @param mesh The mesh to target. * @param focusOnOriginXZ Determines if the camera should focus on 0 in the X and Z axis instead of the mesh * @param onAnimationEnd Callback triggered at the end of the framing animation */ zoomOnMesh(e, t = !1, r = null) { e.computeWorldMatrix(!0); const n = e.getBoundingInfo().boundingBox; this.zoomOnBoundingInfo(n.minimumWorld, n.maximumWorld, t, r); } /** * Targets the given mesh with its children and updates zoom level accordingly. * @param mesh The mesh to target. * @param focusOnOriginXZ Determines if the camera should focus on 0 in the X and Z axis instead of the mesh * @param onAnimationEnd Callback triggered at the end of the framing animation */ zoomOnMeshHierarchy(e, t = !1, r = null) { e.computeWorldMatrix(!0); const n = e.getHierarchyBoundingVectors(!0); this.zoomOnBoundingInfo(n.min, n.max, t, r); } /** * Targets the given meshes with their children and updates zoom level accordingly. * @param meshes The mesh to target. * @param focusOnOriginXZ Determines if the camera should focus on 0 in the X and Z axis instead of the mesh * @param onAnimationEnd Callback triggered at the end of the framing animation */ zoomOnMeshesHierarchy(e, t = !1, r = null) { const n = new S(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE), i = new S(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE); for (let s = 0; s < e.length; s++) { const a = e[s].getHierarchyBoundingVectors(!0); S.CheckExtends(a.min, n, i), S.CheckExtends(a.max, n, i); } this.zoomOnBoundingInfo(n, i, t, r); } /** * Targets the bounding box info defined by its extends and updates zoom level accordingly. * @param minimumWorld Determines the smaller position of the bounding box extend * @param maximumWorld Determines the bigger position of the bounding box extend * @param focusOnOriginXZ Determines if the camera should focus on 0 in the X and Z axis instead of the mesh * @param onAnimationEnd Callback triggered at the end of the framing animation * @returns true if the zoom was done */ zoomOnBoundingInfo(e, t, r = !1, n = null) { let i; if (!this._attachedCamera) return !1; const s = e.y, a = t.y, f = s + (a - s) * this._positionScale, o = t.subtract(e).scale(0.5); if (r) i = new S(0, f, 0); else { const u = e.add(o); i = new S(u.x, f, u.z); } this._vectorTransition || (this._vectorTransition = st.CreateAnimation("target", st.ANIMATIONTYPE_VECTOR3, 60, Vu.EasingFunction)), this._betaIsAnimating = !0; let d = st.TransitionTo("target", i, this._attachedCamera, this._attachedCamera.getScene(), 60, this._vectorTransition, this._framingTime); d && this._animatables.push(d); let v = 0; if (this._mode === Vu.FitFrustumSidesMode) { const u = this._calculateLowerRadiusFromModelBoundingSphere(e, t); this.autoCorrectCameraLimitsAndSensibility && (this._attachedCamera.lowerRadiusLimit = o.length() + this._attachedCamera.minZ), v = u; } else this._mode === Vu.IgnoreBoundsSizeMode && (v = this._calculateLowerRadiusFromModelBoundingSphere(e, t), this.autoCorrectCameraLimitsAndSensibility && this._attachedCamera.lowerRadiusLimit === null && (this._attachedCamera.lowerRadiusLimit = this._attachedCamera.minZ)); if (this.autoCorrectCameraLimitsAndSensibility) { const u = t.subtract(e).length(); this._attachedCamera.panningSensibility = 5e3 / u, this._attachedCamera.wheelPrecision = 100 / v; } return this._radiusTransition || (this._radiusTransition = st.CreateAnimation("radius", st.ANIMATIONTYPE_FLOAT, 60, Vu.EasingFunction)), d = st.TransitionTo("radius", v, this._attachedCamera, this._attachedCamera.getScene(), 60, this._radiusTransition, this._framingTime, () => { this.stopAllAnimations(), n && n(), this._attachedCamera && this._attachedCamera.useInputToRestoreState && this._attachedCamera.storeState(); }), d && this._animatables.push(d), !0; } /** * Calculates the lowest radius for the camera based on the bounding box of the mesh. * @param minimumWorld * @param maximumWorld * @returns The minimum distance from the primary mesh's center point at which the camera must be kept in order * to fully enclose the mesh in the viewing frustum. */ _calculateLowerRadiusFromModelBoundingSphere(e, t) { const r = this._attachedCamera; if (!r) return 0; let n = r._calculateLowerRadiusFromModelBoundingSphere(e, t, this._radiusScale); return r.lowerRadiusLimit && this._mode === Vu.IgnoreBoundsSizeMode && (n = n < r.lowerRadiusLimit ? r.lowerRadiusLimit : n), r.upperRadiusLimit && (n = n > r.upperRadiusLimit ? r.upperRadiusLimit : n), n; } /** * Keeps the camera above the ground plane. If the user pulls the camera below the ground plane, the camera * is automatically returned to its default position (expected to be above ground plane). */ _maintainCameraAboveGround() { if (this._elevationReturnTime < 0) return; const e = Yi.Now - this._lastInteractionTime, t = Math.PI * 0.5 - this._defaultElevation, r = Math.PI * 0.5; if (this._attachedCamera && !this._betaIsAnimating && this._attachedCamera.beta > r && e >= this._elevationReturnWaitTime) { this._betaIsAnimating = !0, this.stopAllAnimations(), this._betaTransition || (this._betaTransition = st.CreateAnimation("beta", st.ANIMATIONTYPE_FLOAT, 60, Vu.EasingFunction)); const n = st.TransitionTo("beta", t, this._attachedCamera, this._attachedCamera.getScene(), 60, this._betaTransition, this._elevationReturnTime, () => { this._clearAnimationLocks(), this.stopAllAnimations(); }); n && this._animatables.push(n); } } /** * Removes all animation locks. Allows new animations to be added to any of the arcCamera properties. */ _clearAnimationLocks() { this._betaIsAnimating = !1; } /** * Applies any current user interaction to the camera. Takes into account maximum alpha rotation. */ _applyUserInteraction() { this.isUserIsMoving && (this._lastInteractionTime = Yi.Now, this.stopAllAnimations(), this._clearAnimationLocks()); } /** * Stops and removes all animations that have been applied to the camera */ stopAllAnimations() { for (this._attachedCamera && (this._attachedCamera.animations = []); this._animatables.length; ) this._animatables[0] && (this._animatables[0].onAnimationEnd = null, this._animatables[0].stop()), this._animatables.shift(); } /** * Gets a value indicating if the user is moving the camera */ get isUserIsMoving() { return this._attachedCamera ? this._attachedCamera.inertialAlphaOffset !== 0 || this._attachedCamera.inertialBetaOffset !== 0 || this._attachedCamera.inertialRadiusOffset !== 0 || this._attachedCamera.inertialPanningX !== 0 || this._attachedCamera.inertialPanningY !== 0 || this._isPointerDown : !1; } } Vu.EasingFunction = new $$(); Vu.EasingMode = u1.EASINGMODE_EASEINOUT; Vu.IgnoreBoundsSizeMode = 0; Vu.FitFrustumSidesMode = 1; class mW { constructor(e, t = new S(), r = 0, n = !1) { this.direction = e, this.rotatedDirection = t, this.diff = r, this.ignore = n; } } class T6e { /** * Creates the AttachToBoxBehavior, used to attach UI to the closest face of the box to a camera * @param _ui The transform node that should be attached to the mesh */ constructor(e) { this._ui = e, this.name = "AttachToBoxBehavior", this.distanceAwayFromFace = 0.15, this.distanceAwayFromBottomOfFace = 0.15, this._faceVectors = [ new mW(S.Up()), new mW(S.Down()), new mW(S.Left()), new mW(S.Right()), new mW(S.Forward()), new mW(S.Forward().scaleInPlace(-1)) ], this._tmpMatrix = new he(), this._tmpVector = new S(), this._zeroVector = S.Zero(), this._lookAtTmpMatrix = new he(); } /** * Initializes the behavior */ init() { } _closestFace(e) { return this._faceVectors.forEach((t) => { this._target.rotationQuaternion || (this._target.rotationQuaternion = Ze.RotationYawPitchRoll(this._target.rotation.y, this._target.rotation.x, this._target.rotation.z)), this._target.rotationQuaternion.toRotationMatrix(this._tmpMatrix), S.TransformCoordinatesToRef(t.direction, this._tmpMatrix, t.rotatedDirection), t.diff = S.GetAngleBetweenVectors(t.rotatedDirection, e, S.Cross(t.rotatedDirection, e)); }), this._faceVectors.reduce((t, r) => t.ignore ? r : r.ignore || t.diff < r.diff ? t : r, this._faceVectors[0]); } _lookAtToRef(e, t = new S(0, 1, 0), r) { he.LookAtLHToRef(this._zeroVector, e, t, this._lookAtTmpMatrix), this._lookAtTmpMatrix.invert(), Ze.FromRotationMatrixToRef(this._lookAtTmpMatrix, r); } /** * Attaches the AttachToBoxBehavior to the passed in mesh * @param target The mesh that the specified node will be attached to */ attach(e) { this._target = e, this._scene = this._target.getScene(), this._onRenderObserver = this._scene.onBeforeRenderObservable.add(() => { if (!this._scene.activeCamera) return; let t = this._scene.activeCamera.position; this._scene.activeCamera.devicePosition && (t = this._scene.activeCamera.devicePosition); const r = this._closestFace(t.subtract(e.position)); this._scene.activeCamera.leftCamera ? this._scene.activeCamera.leftCamera.computeWorldMatrix().getRotationMatrixToRef(this._tmpMatrix) : this._scene.activeCamera.computeWorldMatrix().getRotationMatrixToRef(this._tmpMatrix), S.TransformCoordinatesToRef(S.Up(), this._tmpMatrix, this._tmpVector), this._faceVectors.forEach((i) => { r.direction.x && i.direction.x && (i.ignore = !0), r.direction.y && i.direction.y && (i.ignore = !0), r.direction.z && i.direction.z && (i.ignore = !0); }); const n = this._closestFace(this._tmpVector); this._faceVectors.forEach((i) => { i.ignore = !1; }), this._ui.position.copyFrom(e.position), r.direction.x && (r.rotatedDirection.scaleToRef(e.scaling.x / 2 + this.distanceAwayFromFace, this._tmpVector), this._ui.position.addInPlace(this._tmpVector)), r.direction.y && (r.rotatedDirection.scaleToRef(e.scaling.y / 2 + this.distanceAwayFromFace, this._tmpVector), this._ui.position.addInPlace(this._tmpVector)), r.direction.z && (r.rotatedDirection.scaleToRef(e.scaling.z / 2 + this.distanceAwayFromFace, this._tmpVector), this._ui.position.addInPlace(this._tmpVector)), this._ui.rotationQuaternion || (this._ui.rotationQuaternion = Ze.RotationYawPitchRoll(this._ui.rotation.y, this._ui.rotation.x, this._ui.rotation.z)), r.rotatedDirection.scaleToRef(-1, this._tmpVector), this._lookAtToRef(this._tmpVector, n.rotatedDirection, this._ui.rotationQuaternion), n.direction.x && this._ui.up.scaleToRef(this.distanceAwayFromBottomOfFace - e.scaling.x / 2, this._tmpVector), n.direction.y && this._ui.up.scaleToRef(this.distanceAwayFromBottomOfFace - e.scaling.y / 2, this._tmpVector), n.direction.z && this._ui.up.scaleToRef(this.distanceAwayFromBottomOfFace - e.scaling.z / 2, this._tmpVector), this._ui.position.addInPlace(this._tmpVector); }); } /** * Detaches the behavior from the mesh */ detach() { this._scene.onBeforeRenderObservable.remove(this._onRenderObserver); } } class q6e { /** * Time in milliseconds to delay before fading in (Default: 0) * Will set both fade in and out delay to the same value */ get delay() { return this.fadeInDelay; } set delay(e) { this.fadeInDelay = e, this.fadeOutDelay = e; } /** * Instantiates the FadeInOutBehavior */ constructor() { this.fadeInDelay = 0, this.fadeOutDelay = 0, this.fadeInTime = 300, this.fadeOutTime = 300, this._millisecondsPerFrame = 1e3 / 60, this._hovered = !1, this._hoverValue = 0, this._ownerNode = null, this._delay = 0, this._time = 300, this._update = () => { if (this._ownerNode) { if (this._hoverValue += this._hovered ? this._millisecondsPerFrame : -this._millisecondsPerFrame, this._setAllVisibility(this._ownerNode, (this._hoverValue - this._delay) / this._time), this._ownerNode.visibility > 1) { if (this._setAllVisibility(this._ownerNode, 1), this._hoverValue > this._time) { this._hoverValue = this._time, this._detachObserver(); return; } } else if (this._ownerNode.visibility < 0 && (this._setAllVisibility(this._ownerNode, 0), this._hoverValue < 0)) { this._hoverValue = 0, this._detachObserver(); return; } this._attachObserver(); } }; } /** * The name of the behavior */ get name() { return "FadeInOut"; } /** * Initializes the behavior */ init() { } /** * Attaches the fade behavior on the passed in mesh * @param ownerNode The mesh that will be faded in/out once attached */ attach(e) { this._ownerNode = e, this._setAllVisibility(this._ownerNode, 0); } /** * Detaches the behavior from the mesh */ detach() { this._ownerNode = null; } /** * Triggers the mesh to begin fading in (or out) * @param fadeIn if the object should fade in or out (true to fade in) */ fadeIn(e = !0) { this._delay = e ? this.fadeInDelay : this.fadeOutDelay, this._time = e ? this.fadeInTime : this.fadeOutTime, this._detachObserver(), !(this._ownerNode && (e && this._ownerNode.visibility >= 1 || !e && this._ownerNode.visibility <= 0)) && (this._hovered = e, this._hovered || (this._delay *= -1), this._ownerNode.visibility >= 1 ? this._hoverValue = this._time : this._ownerNode.visibility <= 0 && (this._hoverValue = 0), this._update()); } /** * Triggers the mesh to begin fading out */ fadeOut() { this.fadeIn(!1); } _setAllVisibility(e, t) { e.visibility = t, e.getChildMeshes().forEach((r) => { this._setAllVisibility(r, t); }); } _attachObserver() { var e; this._onBeforeRenderObserver || (this._onBeforeRenderObserver = (e = this._ownerNode) === null || e === void 0 ? void 0 : e.getScene().onBeforeRenderObservable.add(this._update)); } _detachObserver() { var e; this._onBeforeRenderObserver && ((e = this._ownerNode) === null || e === void 0 || e.getScene().onBeforeRenderObservable.remove(this._onBeforeRenderObserver), this._onBeforeRenderObserver = null); } } class Hi { /** * Creates a new ray * @param origin origin point * @param direction direction * @param length length of the ray */ constructor(e, t, r = Number.MAX_VALUE) { this.origin = e, this.direction = t, this.length = r; } // Methods /** * Clone the current ray * @returns a new ray */ clone() { return new Hi(this.origin.clone(), this.direction.clone(), this.length); } /** * Checks if the ray intersects a box * This does not account for the ray length by design to improve perfs. * @param minimum bound of the box * @param maximum bound of the box * @param intersectionTreshold extra extend to be added to the box in all direction * @returns if the box was hit */ intersectsBoxMinMax(e, t, r = 0) { const n = Hi._TmpVector3[0].copyFromFloats(e.x - r, e.y - r, e.z - r), i = Hi._TmpVector3[1].copyFromFloats(t.x + r, t.y + r, t.z + r); let s = 0, a = Number.MAX_VALUE, f, o, d, v; if (Math.abs(this.direction.x) < 1e-7) { if (this.origin.x < n.x || this.origin.x > i.x) return !1; } else if (f = 1 / this.direction.x, o = (n.x - this.origin.x) * f, d = (i.x - this.origin.x) * f, d === -1 / 0 && (d = 1 / 0), o > d && (v = o, o = d, d = v), s = Math.max(o, s), a = Math.min(d, a), s > a) return !1; if (Math.abs(this.direction.y) < 1e-7) { if (this.origin.y < n.y || this.origin.y > i.y) return !1; } else if (f = 1 / this.direction.y, o = (n.y - this.origin.y) * f, d = (i.y - this.origin.y) * f, d === -1 / 0 && (d = 1 / 0), o > d && (v = o, o = d, d = v), s = Math.max(o, s), a = Math.min(d, a), s > a) return !1; if (Math.abs(this.direction.z) < 1e-7) { if (this.origin.z < n.z || this.origin.z > i.z) return !1; } else if (f = 1 / this.direction.z, o = (n.z - this.origin.z) * f, d = (i.z - this.origin.z) * f, d === -1 / 0 && (d = 1 / 0), o > d && (v = o, o = d, d = v), s = Math.max(o, s), a = Math.min(d, a), s > a) return !1; return !0; } /** * Checks if the ray intersects a box * This does not account for the ray lenght by design to improve perfs. * @param box the bounding box to check * @param intersectionTreshold extra extend to be added to the BoundingBox in all direction * @returns if the box was hit */ intersectsBox(e, t = 0) { return this.intersectsBoxMinMax(e.minimum, e.maximum, t); } /** * If the ray hits a sphere * @param sphere the bounding sphere to check * @param intersectionTreshold extra extend to be added to the BoundingSphere in all direction * @returns true if it hits the sphere */ intersectsSphere(e, t = 0) { const r = e.center.x - this.origin.x, n = e.center.y - this.origin.y, i = e.center.z - this.origin.z, s = r * r + n * n + i * i, a = e.radius + t, f = a * a; if (s <= f) return !0; const o = r * this.direction.x + n * this.direction.y + i * this.direction.z; return o < 0 ? !1 : s - o * o <= f; } /** * If the ray hits a triange * @param vertex0 triangle vertex * @param vertex1 triangle vertex * @param vertex2 triangle vertex * @returns intersection information if hit */ intersectsTriangle(e, t, r) { const n = Hi._TmpVector3[0], i = Hi._TmpVector3[1], s = Hi._TmpVector3[2], a = Hi._TmpVector3[3], f = Hi._TmpVector3[4]; t.subtractToRef(e, n), r.subtractToRef(e, i), S.CrossToRef(this.direction, i, s); const o = S.Dot(n, s); if (o === 0) return null; const d = 1 / o; this.origin.subtractToRef(e, a); const v = S.Dot(a, s) * d; if (v < 0 || v > 1) return null; S.CrossToRef(a, n, f); const u = S.Dot(this.direction, f) * d; if (u < 0 || v + u > 1) return null; const l = S.Dot(i, f) * d; return l > this.length ? null : new LC(1 - v - u, v, l); } /** * Checks if ray intersects a plane * @param plane the plane to check * @returns the distance away it was hit */ intersectsPlane(e) { let t; const r = S.Dot(e.normal, this.direction); if (Math.abs(r) < 999999997475243e-21) return null; { const n = S.Dot(e.normal, this.origin); return t = (-e.d - n) / r, t < 0 ? t < -999999997475243e-21 ? null : 0 : t; } } /** * Calculate the intercept of a ray on a given axis * @param axis to check 'x' | 'y' | 'z' * @param offset from axis interception (i.e. an offset of 1y is intercepted above ground) * @returns a vector containing the coordinates where 'axis' is equal to zero (else offset), or null if there is no intercept. */ intersectsAxis(e, t = 0) { switch (e) { case "y": { const r = (this.origin.y - t) / this.direction.y; return r > 0 ? null : new S(this.origin.x + this.direction.x * -r, t, this.origin.z + this.direction.z * -r); } case "x": { const r = (this.origin.x - t) / this.direction.x; return r > 0 ? null : new S(t, this.origin.y + this.direction.y * -r, this.origin.z + this.direction.z * -r); } case "z": { const r = (this.origin.z - t) / this.direction.z; return r > 0 ? null : new S(this.origin.x + this.direction.x * -r, this.origin.y + this.direction.y * -r, t); } default: return null; } } /** * Checks if ray intersects a mesh. The ray is defined in WORLD space. A mesh triangle can be picked both from its front and back sides, * irrespective of orientation. * @param mesh the mesh to check * @param fastCheck defines if the first intersection will be used (and not the closest) * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected * @param onlyBoundingInfo defines a boolean indicating if picking should only happen using bounding info (false by default) * @param worldToUse defines the world matrix to use to get the world coordinate of the intersection point * @param skipBoundingInfo a boolean indicating if we should skip the bounding info check * @returns picking info of the intersection */ intersectsMesh(e, t, r, n = !1, i, s = !1) { const a = ue.Matrix[0]; return e.getWorldMatrix().invertToRef(a), this._tmpRay ? Hi.TransformToRef(this, a, this._tmpRay) : this._tmpRay = Hi.Transform(this, a), e.intersects(this._tmpRay, t, r, n, i, s); } /** * Checks if ray intersects a mesh * @param meshes the meshes to check * @param fastCheck defines if the first intersection will be used (and not the closest) * @param results array to store result in * @returns Array of picking infos */ intersectsMeshes(e, t, r) { r ? r.length = 0 : r = []; for (let n = 0; n < e.length; n++) { const i = this.intersectsMesh(e[n], t); i.hit && r.push(i); } return r.sort(this._comparePickingInfo), r; } _comparePickingInfo(e, t) { return e.distance < t.distance ? -1 : e.distance > t.distance ? 1 : 0; } /** * Intersection test between the ray and a given segment within a given tolerance (threshold) * @param sega the first point of the segment to test the intersection against * @param segb the second point of the segment to test the intersection against * @param threshold the tolerance margin, if the ray doesn't intersect the segment but is close to the given threshold, the intersection is successful * @returns the distance from the ray origin to the intersection point if there's intersection, or -1 if there's no intersection */ intersectionSegment(e, t, r) { const n = this.origin, i = ue.Vector3[0], s = ue.Vector3[1], a = ue.Vector3[2], f = ue.Vector3[3]; t.subtractToRef(e, i), this.direction.scaleToRef(Hi._Rayl, a), n.addToRef(a, s), e.subtractToRef(n, f); const o = S.Dot(i, i), d = S.Dot(i, a), v = S.Dot(a, a), u = S.Dot(i, f), l = S.Dot(a, f), P = o * v - d * d; let p, c = P, H, T = P; P < Hi._Smallnum ? (p = 0, c = 1, H = l, T = v) : (p = d * l - v * u, H = o * l - d * u, p < 0 ? (p = 0, H = l, T = v) : p > c && (p = c, H = l + d, T = v)), H < 0 ? (H = 0, -u < 0 ? p = 0 : -u > o ? p = c : (p = -u, c = o)) : H > T && (H = T, -u + d < 0 ? p = 0 : -u + d > o ? p = c : (p = -u + d, c = o)); const q = Math.abs(p) < Hi._Smallnum ? 0 : p / c, b = Math.abs(H) < Hi._Smallnum ? 0 : H / T, j = ue.Vector3[4]; a.scaleToRef(b, j); const w = ue.Vector3[5]; i.scaleToRef(q, w), w.addInPlace(f); const m = ue.Vector3[6]; return w.subtractToRef(j, m), b > 0 && b <= this.length && m.lengthSquared() < r * r ? w.length() : -1; } /** * Update the ray from viewport position * @param x position * @param y y position * @param viewportWidth viewport width * @param viewportHeight viewport height * @param world world matrix * @param view view matrix * @param projection projection matrix * @param enableDistantPicking defines if picking should handle large values for mesh position/scaling (false by default) * @returns this ray updated */ update(e, t, r, n, i, s, a, f = !1) { if (f) { Hi._RayDistant || (Hi._RayDistant = Hi.Zero()), Hi._RayDistant.unprojectRayToRef(e, t, r, n, he.IdentityReadOnly, s, a); const o = ue.Matrix[0]; i.invertToRef(o), Hi.TransformToRef(Hi._RayDistant, o, this); } else this.unprojectRayToRef(e, t, r, n, i, s, a); return this; } // Statics /** * Creates a ray with origin and direction of 0,0,0 * @returns the new ray */ static Zero() { return new Hi(S.Zero(), S.Zero()); } /** * Creates a new ray from screen space and viewport * @param x position * @param y y position * @param viewportWidth viewport width * @param viewportHeight viewport height * @param world world matrix * @param view view matrix * @param projection projection matrix * @returns new ray */ static CreateNew(e, t, r, n, i, s, a) { return Hi.Zero().update(e, t, r, n, i, s, a); } /** * Function will create a new transformed ray starting from origin and ending at the end point. Ray's length will be set, and ray will be * transformed to the given world matrix. * @param origin The origin point * @param end The end point * @param world a matrix to transform the ray to. Default is the identity matrix. * @returns the new ray */ static CreateNewFromTo(e, t, r = he.IdentityReadOnly) { const n = t.subtract(e), i = Math.sqrt(n.x * n.x + n.y * n.y + n.z * n.z); return n.normalize(), Hi.Transform(new Hi(e, n, i), r); } /** * Transforms a ray by a matrix * @param ray ray to transform * @param matrix matrix to apply * @returns the resulting new ray */ static Transform(e, t) { const r = new Hi(new S(0, 0, 0), new S(0, 0, 0)); return Hi.TransformToRef(e, t, r), r; } /** * Transforms a ray by a matrix * @param ray ray to transform * @param matrix matrix to apply * @param result ray to store result in */ static TransformToRef(e, t, r) { S.TransformCoordinatesToRef(e.origin, t, r.origin), S.TransformNormalToRef(e.direction, t, r.direction), r.length = e.length; const n = r.direction, i = n.length(); if (!(i === 0 || i === 1)) { const s = 1 / i; n.x *= s, n.y *= s, n.z *= s, r.length *= i; } } /** * Unproject a ray from screen space to object space * @param sourceX defines the screen space x coordinate to use * @param sourceY defines the screen space y coordinate to use * @param viewportWidth defines the current width of the viewport * @param viewportHeight defines the current height of the viewport * @param world defines the world matrix to use (can be set to Identity to go to world space) * @param view defines the view matrix to use * @param projection defines the projection matrix to use */ unprojectRayToRef(e, t, r, n, i, s, a) { const f = ue.Matrix[0]; i.multiplyToRef(s, f), f.multiplyToRef(a, f), f.invert(); const o = gr.LastCreatedEngine, d = ue.Vector3[0]; d.x = e / r * 2 - 1, d.y = -(t / n * 2 - 1), d.z = o != null && o.useReverseDepthBuffer ? 1 : o != null && o.isNDCHalfZRange ? 0 : -1; const v = ue.Vector3[1].copyFromFloats(d.x, d.y, 1 - 1e-8), u = ue.Vector3[2], l = ue.Vector3[3]; S._UnprojectFromInvertedMatrixToRef(d, f, u), S._UnprojectFromInvertedMatrixToRef(v, f, l), this.origin.copyFrom(u), l.subtractToRef(u, this.direction), this.direction.normalize(); } } Hi._TmpVector3 = Nf.BuildArray(6, S.Zero); Hi._RayDistant = Hi.Zero(); Hi._Smallnum = 1e-8; Hi._Rayl = 1e9; sr.prototype.createPickingRay = function(A, e, t, r, n = !1) { const i = Hi.Zero(); return this.createPickingRayToRef(A, e, t, i, r, n), i; }; sr.prototype.createPickingRayToRef = function(A, e, t, r, n, i = !1, s = !1) { const a = this.getEngine(); if (!n && !(n = this.activeCamera)) return this; const f = n.viewport, o = a.getRenderHeight(), { x: d, y: v, width: u, height: l } = f.toGlobal(a.getRenderWidth(), o), P = 1 / a.getHardwareScalingLevel(); return A = A * P - d, e = e * P - (o - v - l), r.update(A, e, u, l, t || he.IdentityReadOnly, i ? he.IdentityReadOnly : n.getViewMatrix(), n.getProjectionMatrix(), s), this; }; sr.prototype.createPickingRayInCameraSpace = function(A, e, t) { const r = Hi.Zero(); return this.createPickingRayInCameraSpaceToRef(A, e, r, t), r; }; sr.prototype.createPickingRayInCameraSpaceToRef = function(A, e, t, r) { if (!F9) return this; const n = this.getEngine(); if (!r && !(r = this.activeCamera)) throw new Error("Active camera not set"); const i = r.viewport, s = n.getRenderHeight(), { x: a, y: f, width: o, height: d } = i.toGlobal(n.getRenderWidth(), s), v = he.Identity(), u = 1 / n.getHardwareScalingLevel(); return A = A * u - a, e = e * u - (s - f - d), t.update(A, e, o, d, v, v, r.getProjectionMatrix()), this; }; sr.prototype._internalPickForMesh = function(A, e, t, r, n, i, s, a) { const f = e(r, t.enableDistantPicking), o = t.intersects(f, n, s, i, r, a); return !o || !o.hit || !n && A != null && o.distance >= A.distance ? null : o; }; sr.prototype._internalPick = function(A, e, t, r, n) { let i = null; const s = !!(this.activeCameras && this.activeCameras.length > 1 && this.cameraToUseForPointers !== this.activeCamera), a = this.cameraToUseForPointers || this.activeCamera; for (let f = 0; f < this.meshes.length; f++) { const o = this.meshes[f]; if (e) { if (!e(o)) continue; } else if (!o.isEnabled() || !o.isVisible || !o.isPickable) continue; const d = s && o.isWorldMatrixCameraDependent(), v = o.computeWorldMatrix(d, a); if (o.hasThinInstances && o.thinInstanceEnablePicking) { const u = this._internalPickForMesh(i, A, o, v, !0, !0, n); if (u) { if (r) return u; const l = ue.Matrix[1], P = o.thinInstanceGetWorldMatrices(); for (let p = 0; p < P.length; p++) { P[p].multiplyToRef(v, l); const H = this._internalPickForMesh(i, A, o, l, t, r, n, !0); if (H && (i = H, i.thinInstanceIndex = p, t)) return i; } } } else { const u = this._internalPickForMesh(i, A, o, v, t, r, n); if (u && (i = u, t)) return i; } } return i || new F9(); }; sr.prototype._internalMultiPick = function(A, e, t) { if (!F9) return null; const r = [], n = !!(this.activeCameras && this.activeCameras.length > 1 && this.cameraToUseForPointers !== this.activeCamera), i = this.cameraToUseForPointers || this.activeCamera; for (let s = 0; s < this.meshes.length; s++) { const a = this.meshes[s]; if (e) { if (!e(a)) continue; } else if (!a.isEnabled() || !a.isVisible || !a.isPickable) continue; const f = n && a.isWorldMatrixCameraDependent(), o = a.computeWorldMatrix(f, i); if (a.hasThinInstances && a.thinInstanceEnablePicking) { if (this._internalPickForMesh(null, A, a, o, !0, !0, t)) { const v = ue.Matrix[1], u = a.thinInstanceGetWorldMatrices(); for (let l = 0; l < u.length; l++) { u[l].multiplyToRef(o, v); const p = this._internalPickForMesh(null, A, a, v, !1, !1, t, !0); p && (p.thinInstanceIndex = l, r.push(p)); } } } else { const d = this._internalPickForMesh(null, A, a, o, !1, !1, t); d && r.push(d); } } return r; }; sr.prototype.pickWithBoundingInfo = function(A, e, t, r, n) { if (!F9) return null; const i = this._internalPick((s) => (this._tempPickingRay || (this._tempPickingRay = Hi.Zero()), this.createPickingRayToRef(A, e, s, this._tempPickingRay, n || null), this._tempPickingRay), t, r, !0); return i && (i.ray = this.createPickingRay(A, e, he.Identity(), n || null)), i; }; Object.defineProperty(sr.prototype, "_pickingAvailable", { get: () => !0, enumerable: !1, configurable: !1 }); sr.prototype.pick = function(A, e, t, r, n, i, s = !1) { const a = this._internalPick((f, o) => (this._tempPickingRay || (this._tempPickingRay = Hi.Zero()), this.createPickingRayToRef(A, e, f, this._tempPickingRay, n || null, !1, o), this._tempPickingRay), t, r, !1, i); return a && (a.ray = this.createPickingRay(A, e, he.Identity(), n || null)), a; }; sr.prototype.pickWithRay = function(A, e, t, r) { const n = this._internalPick((i) => (this._pickWithRayInverseMatrix || (this._pickWithRayInverseMatrix = he.Identity()), i.invertToRef(this._pickWithRayInverseMatrix), this._cachedRayForTransform || (this._cachedRayForTransform = Hi.Zero()), Hi.TransformToRef(A, this._pickWithRayInverseMatrix, this._cachedRayForTransform), this._cachedRayForTransform), e, t, !1, r); return n && (n.ray = A), n; }; sr.prototype.multiPick = function(A, e, t, r, n) { return this._internalMultiPick((i) => this.createPickingRay(A, e, i, r || null), t, n); }; sr.prototype.multiPickWithRay = function(A, e, t) { return this._internalMultiPick((r) => (this._pickWithRayInverseMatrix || (this._pickWithRayInverseMatrix = he.Identity()), r.invertToRef(this._pickWithRayInverseMatrix), this._cachedRayForTransform || (this._cachedRayForTransform = Hi.Zero()), Hi.TransformToRef(A, this._pickWithRayInverseMatrix, this._cachedRayForTransform), this._cachedRayForTransform), e, t); }; Tr.prototype.getForwardRay = function(A = 100, e, t) { return this.getForwardRayToRef(new Hi(S.Zero(), S.Zero(), A), A, e, t); }; Tr.prototype.getForwardRayToRef = function(A, e = 100, t, r) { t || (t = this.getWorldMatrix()), A.length = e, r ? A.origin.copyFrom(r) : A.origin.copyFrom(this.position); const n = ue.Vector3[2]; n.set(0, 0, this._scene.useRightHandedSystem ? -1 : 1); const i = ue.Vector3[3]; return S.TransformNormalToRef(n, t, i), S.NormalizeToRef(i, A.direction), A; }; class Vs { /** * @internal */ static _RemoveAndStorePivotPoint(e) { e && Vs._PivotCached === 0 && (e.getPivotPointToRef(Vs._OldPivotPoint), Vs._PivotPostMultiplyPivotMatrix = e._postMultiplyPivotMatrix, Vs._OldPivotPoint.equalsToFloats(0, 0, 0) || (e.setPivotMatrix(he.IdentityReadOnly), Vs._OldPivotPoint.subtractToRef(e.getPivotPoint(), Vs._PivotTranslation), Vs._PivotTmpVector.copyFromFloats(1, 1, 1), Vs._PivotTmpVector.subtractInPlace(e.scaling), Vs._PivotTmpVector.multiplyInPlace(Vs._PivotTranslation), e.position.addInPlace(Vs._PivotTmpVector))), Vs._PivotCached++; } /** * @internal */ static _RestorePivotPoint(e) { e && !Vs._OldPivotPoint.equalsToFloats(0, 0, 0) && Vs._PivotCached === 1 && (e.setPivotPoint(Vs._OldPivotPoint), e._postMultiplyPivotMatrix = Vs._PivotPostMultiplyPivotMatrix, Vs._PivotTmpVector.copyFromFloats(1, 1, 1), Vs._PivotTmpVector.subtractInPlace(e.scaling), Vs._PivotTmpVector.multiplyInPlace(Vs._PivotTranslation), e.position.subtractInPlace(Vs._PivotTmpVector)), this._PivotCached--; } } Vs._PivotCached = 0; Vs._OldPivotPoint = new S(); Vs._PivotTranslation = new S(); Vs._PivotTmpVector = new S(); Vs._PivotPostMultiplyPivotMatrix = !1; function UO(A) { const e = [], t = [], r = [], n = [], i = A.width || A.size || 1, s = A.height || A.size || 1, a = A.sideOrientation === 0 ? 0 : A.sideOrientation || Ut.DEFAULTSIDE, f = i / 2, o = s / 2; t.push(-f, -o, 0), r.push(0, 0, -1), n.push(0, us.UseOpenGLOrientationForUV ? 1 : 0), t.push(f, -o, 0), r.push(0, 0, -1), n.push(1, us.UseOpenGLOrientationForUV ? 1 : 0), t.push(f, o, 0), r.push(0, 0, -1), n.push(1, us.UseOpenGLOrientationForUV ? 0 : 1), t.push(-f, o, 0), r.push(0, 0, -1), n.push(0, us.UseOpenGLOrientationForUV ? 0 : 1), e.push(0), e.push(1), e.push(2), e.push(0), e.push(2), e.push(3), Ut._ComputeSides(a, t, e, r, n, A.frontUVs, A.backUVs); const d = new Ut(); return d.indices = e, d.positions = t, d.normals = r, d.uvs = n, d; } function u4(A, e = {}, t = null) { const r = new Ee(A, t); return e.sideOrientation = Ee._GetDefaultSideOrientation(e.sideOrientation), r._originalBuilderSideOrientation = e.sideOrientation, UO(e).applyToMesh(r, e.updatable), e.sourcePlane && (r.translate(e.sourcePlane.normal, -e.sourcePlane.d), r.setDirection(e.sourcePlane.normal.scale(-1))), r; } const b6e = { // eslint-disable-next-line @typescript-eslint/naming-convention CreatePlane: u4 }; Ut.CreatePlane = UO; Ee.CreatePlane = (A, e, t, r, n) => u4(A, { size: e, width: e, height: e, sideOrientation: n, updatable: r }, t); class O9 { /** * Get or set the currentDraggingPointerId * @deprecated Please use currentDraggingPointerId instead */ get currentDraggingPointerID() { return this.currentDraggingPointerId; } set currentDraggingPointerID(e) { this.currentDraggingPointerId = e; } /** * If the drag behavior will react to drag events (Default: true) */ set enabled(e) { e != this._enabled && this.onEnabledObservable.notifyObservers(e), this._enabled = e; } get enabled() { return this._enabled; } /** * Gets the options used by the behavior */ get options() { return this._options; } /** * Sets the options used by the behavior */ set options(e) { this._options = e; } /** * Creates a pointer drag behavior that can be attached to a mesh * @param options The drag axis or normal of the plane that will be dragged across. If no options are specified the drag plane will always face the ray's origin (eg. camera) * @param options.dragAxis * @param options.dragPlaneNormal */ constructor(e) { this._useAlternatePickedPointAboveMaxDragAngleDragSpeed = -1.1, this._activeDragButton = -1, this.maxDragAngle = 0, this.dragButtons = [0, 1, 2], this._useAlternatePickedPointAboveMaxDragAngle = !1, this.currentDraggingPointerId = -1, this.dragging = !1, this.dragDeltaRatio = 0.2, this.updateDragPlane = !0, this._debugMode = !1, this._moving = !1, this.onDragObservable = new Oe(), this.onDragStartObservable = new Oe(), this.onDragEndObservable = new Oe(), this.onEnabledObservable = new Oe(), this.moveAttached = !0, this._enabled = !0, this.startAndReleaseDragOnPointerEvents = !0, this.detachCameraControls = !0, this.useObjectOrientationForDragging = !0, this.validateDrag = (r) => !0, this._tmpVector = new S(0, 0, 0), this._alternatePickedPoint = new S(0, 0, 0), this._worldDragAxis = new S(0, 0, 0), this._targetPosition = new S(0, 0, 0), this._attachedToElement = !1, this._startDragRay = new Hi(new S(), new S()), this._lastPointerRay = {}, this._dragDelta = new S(), this._pointA = new S(0, 0, 0), this._pointC = new S(0, 0, 0), this._localAxis = new S(0, 0, 0), this._lookAt = new S(0, 0, 0), this._options = e || {}; let t = 0; if (this._options.dragAxis && t++, this._options.dragPlaneNormal && t++, t > 1) throw "Multiple drag modes specified in dragBehavior options. Only one expected"; } /** * The name of the behavior */ get name() { return "PointerDrag"; } /** * Initializes the behavior */ init() { } /** * Attaches the drag behavior the passed in mesh * @param ownerNode The mesh that will be dragged around once attached * @param predicate Predicate to use for pick filtering */ attach(e, t) { this._scene = e.getScene(), e.isNearGrabbable = !0, this.attachedNode = e, O9._PlaneScene || (this._debugMode ? O9._PlaneScene = this._scene : (O9._PlaneScene = new sr(this._scene.getEngine(), { virtual: !0 }), O9._PlaneScene.detachControl(), this._scene.onDisposeObservable.addOnce(() => { O9._PlaneScene.dispose(), O9._PlaneScene = null; }))), this._dragPlane = u4("pointerDragPlane", { size: this._debugMode ? 1 : 1e4, updatable: !1, sideOrientation: Ee.DOUBLESIDE }, O9._PlaneScene), this.lastDragPosition = new S(0, 0, 0); const r = t || ((n) => this.attachedNode == n || n.isDescendantOf(this.attachedNode)); this._pointerObserver = this._scene.onPointerObservable.add((n) => { if (!this.enabled) { this._attachedToElement && this.releaseDrag(); return; } if (n.type == ir.POINTERDOWN) this.startAndReleaseDragOnPointerEvents && !this.dragging && n.pickInfo && n.pickInfo.hit && n.pickInfo.pickedMesh && n.pickInfo.pickedPoint && n.pickInfo.ray && r(n.pickInfo.pickedMesh) && this._activeDragButton === -1 && this.dragButtons.indexOf(n.event.button) !== -1 && (this._activeDragButton = n.event.button, this._activePointerInfo = n, this._startDrag(n.event.pointerId, n.pickInfo.ray, n.pickInfo.pickedPoint)); else if (n.type == ir.POINTERUP) this.startAndReleaseDragOnPointerEvents && this.currentDraggingPointerId == n.event.pointerId && (this._activeDragButton === n.event.button || this._activeDragButton === -1) && this.releaseDrag(); else if (n.type == ir.POINTERMOVE) { const i = n.event.pointerId; if (this.currentDraggingPointerId === O9._AnyMouseId && i !== O9._AnyMouseId) { const s = n.event; (s.pointerType === "mouse" || !this._scene.getEngine().hostInformation.isMobile && s instanceof MouseEvent) && (this._lastPointerRay[this.currentDraggingPointerId] && (this._lastPointerRay[i] = this._lastPointerRay[this.currentDraggingPointerId], delete this._lastPointerRay[this.currentDraggingPointerId]), this.currentDraggingPointerId = i); } this._lastPointerRay[i] || (this._lastPointerRay[i] = new Hi(new S(), new S())), n.pickInfo && n.pickInfo.ray && (this._lastPointerRay[i].origin.copyFrom(n.pickInfo.ray.origin), this._lastPointerRay[i].direction.copyFrom(n.pickInfo.ray.direction), this.currentDraggingPointerId == i && this.dragging && this._moveDrag(n.pickInfo.ray)); } }), this._beforeRenderObserver = this._scene.onBeforeRenderObservable.add(() => { if (this._moving && this.moveAttached) { let n = !1; Vs._RemoveAndStorePivotPoint(this.attachedNode), this._targetPosition.subtractToRef(this.attachedNode.absolutePosition, this._tmpVector), this._tmpVector.scaleInPlace(this.dragDeltaRatio), this.attachedNode.getAbsolutePosition().addToRef(this._tmpVector, this._tmpVector), this.validateDrag(this._tmpVector) && (this.attachedNode.setAbsolutePosition(this._tmpVector), n = !0), Vs._RestorePivotPoint(this.attachedNode), n && this.attachedNode.computeWorldMatrix(); } }); } /** * Force release the drag action by code. */ releaseDrag() { if (this.dragging && (this.dragging = !1, this.onDragEndObservable.notifyObservers({ dragPlanePoint: this.lastDragPosition, pointerId: this.currentDraggingPointerId, pointerInfo: this._activePointerInfo })), this.currentDraggingPointerId = -1, this._activeDragButton = -1, this._activePointerInfo = null, this._moving = !1, this.detachCameraControls && this._attachedToElement && this._scene.activeCamera && !this._scene.activeCamera.leftCamera) { if (this._scene.activeCamera.getClassName() === "ArcRotateCamera") { const e = this._scene.activeCamera; e.attachControl(e.inputs ? e.inputs.noPreventDefault : !0, e._useCtrlForPanning, e._panningMouseButton); } else this._scene.activeCamera.attachControl(this._scene.activeCamera.inputs ? this._scene.activeCamera.inputs.noPreventDefault : !0); this._attachedToElement = !1; } } /** * Simulates the start of a pointer drag event on the behavior * @param pointerId pointerID of the pointer that should be simulated (Default: Any mouse pointer ID) * @param fromRay initial ray of the pointer to be simulated (Default: Ray from camera to attached mesh) * @param startPickedPoint picked point of the pointer to be simulated (Default: attached mesh position) */ startDrag(e = O9._AnyMouseId, t, r) { this._startDrag(e, t, r); let n = this._lastPointerRay[e]; e === O9._AnyMouseId && (n = this._lastPointerRay[Object.keys(this._lastPointerRay)[0]]), n && this._moveDrag(n); } _startDrag(e, t, r) { if (!this._scene.activeCamera || this.dragging || !this.attachedNode) return; Vs._RemoveAndStorePivotPoint(this.attachedNode), t ? (this._startDragRay.direction.copyFrom(t.direction), this._startDragRay.origin.copyFrom(t.origin)) : (this._startDragRay.origin.copyFrom(this._scene.activeCamera.position), this.attachedNode.getWorldMatrix().getTranslationToRef(this._tmpVector), this._tmpVector.subtractToRef(this._scene.activeCamera.position, this._startDragRay.direction)), this._updateDragPlanePosition(this._startDragRay, r || this._tmpVector); const n = this._pickWithRayOnDragPlane(this._startDragRay); n ? (this.dragging = !0, this.currentDraggingPointerId = e, this.lastDragPosition.copyFrom(n), this.onDragStartObservable.notifyObservers({ dragPlanePoint: n, pointerId: this.currentDraggingPointerId, pointerInfo: this._activePointerInfo }), this._targetPosition.copyFrom(this.attachedNode.getAbsolutePosition()), this.detachCameraControls && this._scene.activeCamera && this._scene.activeCamera.inputs && !this._scene.activeCamera.leftCamera && (this._scene.activeCamera.inputs.attachedToElement ? (this._scene.activeCamera.detachControl(), this._attachedToElement = !0) : this._attachedToElement = !1)) : this.releaseDrag(), Vs._RestorePivotPoint(this.attachedNode); } _moveDrag(e) { this._moving = !0; const t = this._pickWithRayOnDragPlane(e); if (t) { Vs._RemoveAndStorePivotPoint(this.attachedNode), this.updateDragPlane && this._updateDragPlanePosition(e, t); let r = 0; this._options.dragAxis ? (this.useObjectOrientationForDragging ? S.TransformCoordinatesToRef(this._options.dragAxis, this.attachedNode.getWorldMatrix().getRotationMatrix(), this._worldDragAxis) : this._worldDragAxis.copyFrom(this._options.dragAxis), t.subtractToRef(this.lastDragPosition, this._tmpVector), r = S.Dot(this._tmpVector, this._worldDragAxis), this._worldDragAxis.scaleToRef(r, this._dragDelta)) : (r = this._dragDelta.length(), t.subtractToRef(this.lastDragPosition, this._dragDelta)), this._targetPosition.addInPlace(this._dragDelta), this.onDragObservable.notifyObservers({ dragDistance: r, delta: this._dragDelta, dragPlanePoint: t, dragPlaneNormal: this._dragPlane.forward, pointerId: this.currentDraggingPointerId, pointerInfo: this._activePointerInfo }), this.lastDragPosition.copyFrom(t), Vs._RestorePivotPoint(this.attachedNode); } } _pickWithRayOnDragPlane(e) { if (!e) return null; let t = Math.acos(S.Dot(this._dragPlane.forward, e.direction)); if (t > Math.PI / 2 && (t = Math.PI - t), this.maxDragAngle > 0 && t > this.maxDragAngle) if (this._useAlternatePickedPointAboveMaxDragAngle) { this._tmpVector.copyFrom(e.direction), this.attachedNode.absolutePosition.subtractToRef(e.origin, this._alternatePickedPoint), this._alternatePickedPoint.normalize(), this._alternatePickedPoint.scaleInPlace(this._useAlternatePickedPointAboveMaxDragAngleDragSpeed * S.Dot(this._alternatePickedPoint, this._tmpVector)), this._tmpVector.addInPlace(this._alternatePickedPoint); const f = S.Dot(this._dragPlane.forward, this._tmpVector); return this._dragPlane.forward.scaleToRef(-f, this._alternatePickedPoint), this._alternatePickedPoint.addInPlace(this._tmpVector), this._alternatePickedPoint.addInPlace(this.attachedNode.absolutePosition), this._alternatePickedPoint; } else return null; const r = this._dragPlane.forward, n = this._dragPlane.position, i = e.direction.dot(r); if (Math.abs(i) < Dn) return null; n.subtractToRef(e.origin, ue.Vector3[0]); const s = ue.Vector3[0].dot(r) / i; return s < 0 ? null : (e.direction.scaleToRef(s, ue.Vector3[0]), e.origin.add(ue.Vector3[0])); } // Position the drag plane based on the attached mesh position, for single axis rotate the plane along the axis to face the camera _updateDragPlanePosition(e, t) { this._pointA.copyFrom(t), this._options.dragAxis ? (this.useObjectOrientationForDragging ? S.TransformCoordinatesToRef(this._options.dragAxis, this.attachedNode.getWorldMatrix().getRotationMatrix(), this._localAxis) : this._localAxis.copyFrom(this._options.dragAxis), e.origin.subtractToRef(this._pointA, this._pointC), this._pointC.normalize(), Math.abs(S.Dot(this._localAxis, this._pointC)) > 0.999 ? Math.abs(S.Dot(S.UpReadOnly, this._pointC)) > 0.999 ? this._lookAt.copyFrom(S.Right()) : this._lookAt.copyFrom(S.UpReadOnly) : (S.CrossToRef(this._localAxis, this._pointC, this._lookAt), S.CrossToRef(this._localAxis, this._lookAt, this._lookAt), this._lookAt.normalize()), this._dragPlane.position.copyFrom(this._pointA), this._pointA.addToRef(this._lookAt, this._lookAt), this._dragPlane.lookAt(this._lookAt)) : this._options.dragPlaneNormal ? (this.useObjectOrientationForDragging ? S.TransformCoordinatesToRef(this._options.dragPlaneNormal, this.attachedNode.getWorldMatrix().getRotationMatrix(), this._localAxis) : this._localAxis.copyFrom(this._options.dragPlaneNormal), this._dragPlane.position.copyFrom(this._pointA), this._pointA.addToRef(this._localAxis, this._lookAt), this._dragPlane.lookAt(this._lookAt)) : (this._dragPlane.position.copyFrom(this._pointA), this._dragPlane.lookAt(e.origin)), this._dragPlane.position.copyFrom(this.attachedNode.getAbsolutePosition()), this._dragPlane.computeWorldMatrix(!0); } /** * Detaches the behavior from the mesh */ detach() { this._lastPointerRay = {}, this.attachedNode && (this.attachedNode.isNearGrabbable = !1), this._pointerObserver && this._scene.onPointerObservable.remove(this._pointerObserver), this._beforeRenderObserver && this._scene.onBeforeRenderObservable.remove(this._beforeRenderObserver), this._dragPlane && this._dragPlane.dispose(), this.releaseDrag(); } } O9._AnyMouseId = -2; class x6e { /** * Instantiate a new behavior that when attached to a mesh will allow the mesh to be scaled */ constructor() { this._startDistance = 0, this._initialScale = new S(0, 0, 0), this._targetScale = new S(0, 0, 0), this._sceneRenderObserver = null, this._dragBehaviorA = new O9({}), this._dragBehaviorA.moveAttached = !1, this._dragBehaviorB = new O9({}), this._dragBehaviorB.moveAttached = !1; } /** * The name of the behavior */ get name() { return "MultiPointerScale"; } /** * Initializes the behavior */ init() { } _getCurrentDistance() { return this._dragBehaviorA.lastDragPosition.subtract(this._dragBehaviorB.lastDragPosition).length(); } /** * Attaches the scale behavior the passed in mesh * @param ownerNode The mesh that will be scaled around once attached */ attach(e) { this._ownerNode = e, this._dragBehaviorA.onDragStartObservable.add(() => { this._dragBehaviorA.dragging && this._dragBehaviorB.dragging && (this._dragBehaviorA.currentDraggingPointerId == this._dragBehaviorB.currentDraggingPointerId ? this._dragBehaviorA.releaseDrag() : (this._initialScale.copyFrom(e.scaling), this._startDistance = this._getCurrentDistance())); }), this._dragBehaviorB.onDragStartObservable.add(() => { this._dragBehaviorA.dragging && this._dragBehaviorB.dragging && (this._dragBehaviorA.currentDraggingPointerId == this._dragBehaviorB.currentDraggingPointerId ? this._dragBehaviorB.releaseDrag() : (this._initialScale.copyFrom(e.scaling), this._startDistance = this._getCurrentDistance())); }), [this._dragBehaviorA, this._dragBehaviorB].forEach((t) => { t.onDragObservable.add(() => { if (this._dragBehaviorA.dragging && this._dragBehaviorB.dragging) { const r = this._getCurrentDistance() / this._startDistance; this._initialScale.scaleToRef(r, this._targetScale); } }); }), e.addBehavior(this._dragBehaviorA), e.addBehavior(this._dragBehaviorB), this._sceneRenderObserver = e.getScene().onBeforeRenderObservable.add(() => { if (this._dragBehaviorA.dragging && this._dragBehaviorB.dragging) { const t = this._targetScale.subtract(e.scaling).scaleInPlace(0.1); t.length() > 0.01 && e.scaling.addInPlace(t); } }); } /** * Detaches the behavior from the mesh */ detach() { this._ownerNode.getScene().onBeforeRenderObservable.remove(this._sceneRenderObserver), [this._dragBehaviorA, this._dragBehaviorB].forEach((e) => { e.onDragStartObservable.clear(), e.onDragObservable.clear(), this._ownerNode.removeBehavior(e); }); } } class C2 { constructor() { this._attachedToElement = !1, this._virtualMeshesInfo = {}, this._tmpVector = new S(), this._tmpQuaternion = new Ze(), this._dragType = { NONE: 0, DRAG: 1, DRAG_WITH_CONTROLLER: 2, NEAR_DRAG: 3 }, this._moving = !1, this._dragging = this._dragType.NONE, this.draggableMeshes = null, this.zDragFactor = 3, this.currentDraggingPointerIds = [], this.detachCameraControls = !0, this.onDragStartObservable = new Oe(), this.onDragObservable = new Oe(), this.onDragEndObservable = new Oe(), this.allowMultiPointer = !0; } /** * The id of the pointer that is currently interacting with the behavior (-1 when no pointer is active) */ get currentDraggingPointerId() { return this.currentDraggingPointerIds[0] !== void 0 ? this.currentDraggingPointerIds[0] : -1; } set currentDraggingPointerId(e) { this.currentDraggingPointerIds[0] = e; } /** * Get or set the currentDraggingPointerId * @deprecated Please use currentDraggingPointerId instead */ get currentDraggingPointerID() { return this.currentDraggingPointerId; } set currentDraggingPointerID(e) { this.currentDraggingPointerId = e; } /** * The name of the behavior */ get name() { return "BaseSixDofDrag"; } /** * Returns true if the attached mesh is currently moving with this behavior */ get isMoving() { return this._moving; } /** * Initializes the behavior */ init() { } /** * In the case of multiple active cameras, the cameraToUseForPointers should be used if set instead of active camera */ get _pointerCamera() { return this._scene.cameraToUseForPointers ? this._scene.cameraToUseForPointers : this._scene.activeCamera; } _createVirtualMeshInfo() { const e = new jn("", C2._virtualScene); e.rotationQuaternion = new Ze(); const t = new jn("", C2._virtualScene); t.rotationQuaternion = new Ze(); const r = new jn("", C2._virtualScene); return r.rotationQuaternion = new Ze(), { dragging: !1, moving: !1, dragMesh: e, originMesh: t, pivotMesh: r, startingPivotPosition: new S(), startingPivotOrientation: new Ze(), startingPosition: new S(), startingOrientation: new Ze(), lastOriginPosition: new S(), lastDragPosition: new S() }; } _resetVirtualMeshesPosition() { for (let e = 0; e < this.currentDraggingPointerIds.length; e++) this._virtualMeshesInfo[this.currentDraggingPointerIds[e]].pivotMesh.position.copyFrom(this._ownerNode.getAbsolutePivotPoint()), this._virtualMeshesInfo[this.currentDraggingPointerIds[e]].pivotMesh.rotationQuaternion.copyFrom(this._ownerNode.rotationQuaternion), this._virtualMeshesInfo[this.currentDraggingPointerIds[e]].startingPivotPosition.copyFrom(this._virtualMeshesInfo[this.currentDraggingPointerIds[e]].pivotMesh.position), this._virtualMeshesInfo[this.currentDraggingPointerIds[e]].startingPivotOrientation.copyFrom(this._virtualMeshesInfo[this.currentDraggingPointerIds[e]].pivotMesh.rotationQuaternion), this._virtualMeshesInfo[this.currentDraggingPointerIds[e]].startingPosition.copyFrom(this._virtualMeshesInfo[this.currentDraggingPointerIds[e]].dragMesh.position), this._virtualMeshesInfo[this.currentDraggingPointerIds[e]].startingOrientation.copyFrom(this._virtualMeshesInfo[this.currentDraggingPointerIds[e]].dragMesh.rotationQuaternion); } _pointerUpdate2D(e, t, r) { this._pointerCamera && this._pointerCamera.cameraRigMode == Tr.RIG_MODE_NONE && !this._pointerCamera._isLeftCamera && !this._pointerCamera._isRightCamera && (e.origin.copyFrom(this._pointerCamera.globalPosition), r = 0); const n = this._virtualMeshesInfo[t], i = ue.Vector3[0]; e.origin.subtractToRef(n.lastOriginPosition, i), n.lastOriginPosition.copyFrom(e.origin); const s = -S.Dot(i, e.direction); n.originMesh.addChild(n.dragMesh), n.originMesh.addChild(n.pivotMesh), this._applyZOffset(n.dragMesh, s, r), this._applyZOffset(n.pivotMesh, s, r), n.originMesh.position.copyFrom(e.origin); const a = ue.Vector3[0]; e.origin.addToRef(e.direction, a), n.originMesh.lookAt(a), n.originMesh.removeChild(n.dragMesh), n.originMesh.removeChild(n.pivotMesh); } _pointerUpdateXR(e, t, r, n) { const i = this._virtualMeshesInfo[r]; if (i.originMesh.position.copyFrom(e.position), this._dragging === this._dragType.NEAR_DRAG && t ? i.originMesh.rotationQuaternion.copyFrom(t.rotationQuaternion) : i.originMesh.rotationQuaternion.copyFrom(e.rotationQuaternion), i.pivotMesh.computeWorldMatrix(!0), i.dragMesh.computeWorldMatrix(!0), n !== 0) { const s = ue.Vector3[0], a = ue.Vector3[1]; s.copyFrom(this._pointerCamera.getForwardRay().direction), i.originMesh.position.subtractToRef(i.lastOriginPosition, a), i.lastOriginPosition.copyFrom(i.originMesh.position); const f = a.length(); a.normalize(); const o = ue.Vector3[2], d = ue.Vector3[3]; i.dragMesh.absolutePosition.subtractToRef(this._pointerCamera.globalPosition, o), i.dragMesh.absolutePosition.subtractToRef(i.originMesh.position, d); const v = d.length(); o.normalize(), d.normalize(); let l = Math.abs(S.Dot(a, d)) * S.Dot(a, s) * n * f * v; const P = 0.01; l < 0 && P - v > l && (l = Math.min(P - v, 0)), d.scaleInPlace(l), d.addToRef(i.pivotMesh.absolutePosition, this._tmpVector), i.pivotMesh.setAbsolutePosition(this._tmpVector), d.addToRef(i.dragMesh.absolutePosition, this._tmpVector), i.dragMesh.setAbsolutePosition(this._tmpVector); } } /** * Attaches the scale behavior the passed in mesh * @param ownerNode The mesh that will be scaled around once attached */ attach(e) { this._ownerNode = e, this._scene = this._ownerNode.getScene(), C2._virtualScene || (C2._virtualScene = new sr(this._scene.getEngine(), { virtual: !0 }), C2._virtualScene.detachControl()); const t = (r) => this._ownerNode === r || r.isDescendantOf(this._ownerNode) && (!this.draggableMeshes || this.draggableMeshes.indexOf(r) !== -1); this._pointerObserver = this._scene.onPointerObservable.add((r) => { const n = r.event.pointerId; this._virtualMeshesInfo[n] || (this._virtualMeshesInfo[n] = this._createVirtualMeshInfo()); const i = this._virtualMeshesInfo[n], s = r.event.pointerType === "xr-near"; if (r.type == ir.POINTERDOWN) { if (!i.dragging && r.pickInfo && r.pickInfo.hit && r.pickInfo.pickedMesh && r.pickInfo.pickedPoint && r.pickInfo.ray && (!s || r.pickInfo.aimTransform) && t(r.pickInfo.pickedMesh)) { if (!this.allowMultiPointer && this.currentDraggingPointerIds.length > 0) return; this._pointerCamera && this._pointerCamera.cameraRigMode === Tr.RIG_MODE_NONE && !this._pointerCamera._isLeftCamera && !this._pointerCamera._isRightCamera && r.pickInfo.ray.origin.copyFrom(this._pointerCamera.globalPosition), this._ownerNode.computeWorldMatrix(!0); const a = this._virtualMeshesInfo[n]; s ? (this._dragging = r.pickInfo.originMesh ? this._dragType.NEAR_DRAG : this._dragType.DRAG_WITH_CONTROLLER, a.originMesh.position.copyFrom(r.pickInfo.aimTransform.position), this._dragging === this._dragType.NEAR_DRAG && r.pickInfo.gripTransform ? a.originMesh.rotationQuaternion.copyFrom(r.pickInfo.gripTransform.rotationQuaternion) : a.originMesh.rotationQuaternion.copyFrom(r.pickInfo.aimTransform.rotationQuaternion)) : (this._dragging = this._dragType.DRAG, a.originMesh.position.copyFrom(r.pickInfo.ray.origin)), a.lastOriginPosition.copyFrom(a.originMesh.position), a.dragMesh.position.copyFrom(r.pickInfo.pickedPoint), a.lastDragPosition.copyFrom(r.pickInfo.pickedPoint), a.pivotMesh.position.copyFrom(this._ownerNode.getAbsolutePivotPoint()), a.pivotMesh.rotationQuaternion.copyFrom(this._ownerNode.absoluteRotationQuaternion), a.startingPosition.copyFrom(a.dragMesh.position), a.startingPivotPosition.copyFrom(a.pivotMesh.position), a.startingOrientation.copyFrom(a.dragMesh.rotationQuaternion), a.startingPivotOrientation.copyFrom(a.pivotMesh.rotationQuaternion), s ? (a.originMesh.addChild(a.dragMesh), a.originMesh.addChild(a.pivotMesh)) : a.originMesh.lookAt(a.dragMesh.position), a.dragging = !0, this.currentDraggingPointerIds.indexOf(n) === -1 && this.currentDraggingPointerIds.push(n), this.detachCameraControls && this._pointerCamera && !this._pointerCamera.leftCamera && (this._pointerCamera.inputs && this._pointerCamera.inputs.attachedToElement ? (this._pointerCamera.detachControl(), this._attachedToElement = !0) : this._attachedToElement = !1), this._targetDragStart(a.pivotMesh.position, a.pivotMesh.rotationQuaternion, n), this.onDragStartObservable.notifyObservers({ position: a.pivotMesh.position }); } } else if (r.type == ir.POINTERUP || r.type == ir.POINTERDOUBLETAP) { const a = this.currentDraggingPointerIds.indexOf(n); i.dragging = !1, a !== -1 && (this.currentDraggingPointerIds.splice(a, 1), this.currentDraggingPointerIds.length === 0 && (this._moving = !1, this._dragging = this._dragType.NONE, this.detachCameraControls && this._attachedToElement && this._pointerCamera && !this._pointerCamera.leftCamera && (this._reattachCameraControls(), this._attachedToElement = !1)), i.originMesh.removeChild(i.dragMesh), i.originMesh.removeChild(i.pivotMesh), this._targetDragEnd(n), this.onDragEndObservable.notifyObservers({})); } else if (r.type == ir.POINTERMOVE && this.currentDraggingPointerIds.indexOf(n) !== -1 && i.dragging && r.pickInfo && (r.pickInfo.ray || r.pickInfo.aimTransform)) { let f = this.zDragFactor; (this.currentDraggingPointerIds.length > 1 || r.pickInfo.originMesh) && (f = 0), this._ownerNode.computeWorldMatrix(!0), s ? this._pointerUpdateXR(r.pickInfo.aimTransform, r.pickInfo.gripTransform, n, f) : this._pointerUpdate2D(r.pickInfo.ray, n, f), this._tmpQuaternion.copyFrom(i.startingPivotOrientation), this._tmpQuaternion.x = -this._tmpQuaternion.x, this._tmpQuaternion.y = -this._tmpQuaternion.y, this._tmpQuaternion.z = -this._tmpQuaternion.z, i.pivotMesh.absoluteRotationQuaternion.multiplyToRef(this._tmpQuaternion, this._tmpQuaternion), i.pivotMesh.absolutePosition.subtractToRef(i.startingPivotPosition, this._tmpVector), this.onDragObservable.notifyObservers({ delta: this._tmpVector, position: i.pivotMesh.position, pickInfo: r.pickInfo }), this._targetDrag(this._tmpVector, this._tmpQuaternion, n), i.lastDragPosition.copyFrom(i.dragMesh.absolutePosition), this._moving = !0; } }); } _applyZOffset(e, t, r) { e.position.z -= e.position.z < 1 ? t * r : t * r * e.position.z, e.position.z < 0 && (e.position.z = 0); } // eslint-disable-next-line @typescript-eslint/no-unused-vars _targetDragStart(e, t, r) { } _targetDrag(e, t, r) { } _targetDragEnd(e) { } _reattachCameraControls() { if (this._pointerCamera) if (this._pointerCamera.getClassName() === "ArcRotateCamera") { const e = this._pointerCamera; e.attachControl(e.inputs ? e.inputs.noPreventDefault : !0, e._useCtrlForPanning, e._panningMouseButton); } else this._pointerCamera.attachControl(this._pointerCamera.inputs ? this._pointerCamera.inputs.noPreventDefault : !0); } /** * Detaches the behavior from the mesh */ detach() { this._scene && (this.detachCameraControls && this._attachedToElement && this._pointerCamera && !this._pointerCamera.leftCamera && (this._reattachCameraControls(), this._attachedToElement = !1), this._scene.onPointerObservable.remove(this._pointerObserver)); for (const e in this._virtualMeshesInfo) this._virtualMeshesInfo[e].originMesh.dispose(), this._virtualMeshesInfo[e].dragMesh.dispose(); this.onDragEndObservable.clear(), this.onDragObservable.clear(), this.onDragStartObservable.clear(); } } class Hee extends C2 { constructor() { super(...arguments), this._sceneRenderObserver = null, this._targetPosition = new S(0, 0, 0), this._targetOrientation = new Ze(), this._targetScaling = new S(1, 1, 1), this._startingPosition = new S(0, 0, 0), this._startingOrientation = new Ze(), this._startingScaling = new S(1, 1, 1), this.onPositionChangedObservable = new Oe(), this.dragDeltaRatio = 0.2, this.rotateDraggedObject = !0, this.rotateAroundYOnly = !1, this.rotateWithMotionController = !0, this.disableMovement = !1, this.faceCameraOnDragStart = !1; } /** * The name of the behavior */ get name() { return "SixDofDrag"; } /** * Attaches the six DoF drag behavior * @param ownerNode The mesh that will be dragged around once attached */ attach(e) { super.attach(e), e.isNearGrabbable = !0, this._virtualTransformNode = new Hr("virtual_sixDof", C2._virtualScene), this._virtualTransformNode.rotationQuaternion = Ze.Identity(), this._sceneRenderObserver = e.getScene().onBeforeRenderObservable.add(() => { if (this.currentDraggingPointerIds.length === 1 && this._moving && !this.disableMovement) { const t = e.parent; e.setParent(null), e.position.addInPlace(this._targetPosition.subtract(e.position).scale(this.dragDeltaRatio)), this.onPositionChangedObservable.notifyObservers({ position: e.absolutePosition }), (!t || t.scaling && !t.scaling.isNonUniformWithinEpsilon(1e-3)) && Ze.SlerpToRef(e.rotationQuaternion, this._targetOrientation, this.dragDeltaRatio, e.rotationQuaternion), e.setParent(t); } }); } _getPositionOffsetAround(e, t, r) { const n = ue.Matrix[0], i = ue.Matrix[1], s = ue.Matrix[2], a = ue.Matrix[3], f = ue.Matrix[4]; return he.TranslationToRef(e.x, e.y, e.z, n), he.TranslationToRef(-e.x, -e.y, -e.z, i), he.FromQuaternionToRef(r, s), he.ScalingToRef(t, t, t, a), i.multiplyToRef(s, f), f.multiplyToRef(a, f), f.multiplyToRef(n, f), f.getTranslation(); } _onePointerPositionUpdated(e, t) { ue.Vector3[0].setAll(0), this._dragging === this._dragType.DRAG ? this.rotateDraggedObject && (this.rotateAroundYOnly ? Ze.RotationYawPitchRollToRef(t.toEulerAngles().y, 0, 0, ue.Quaternion[0]) : ue.Quaternion[0].copyFrom(t), ue.Quaternion[0].multiplyToRef(this._startingOrientation, this._targetOrientation)) : (this._dragging === this._dragType.NEAR_DRAG || this._dragging === this._dragType.DRAG_WITH_CONTROLLER && this.rotateWithMotionController) && t.multiplyToRef(this._startingOrientation, this._targetOrientation), this._targetPosition.copyFrom(this._startingPosition).addInPlace(e); } _twoPointersPositionUpdated() { const e = this._virtualMeshesInfo[this.currentDraggingPointerIds[0]].startingPosition, t = this._virtualMeshesInfo[this.currentDraggingPointerIds[1]].startingPosition, r = ue.Vector3[0]; e.addToRef(t, r), r.scaleInPlace(0.5); const n = ue.Vector3[1]; t.subtractToRef(e, n); const i = this._virtualMeshesInfo[this.currentDraggingPointerIds[0]].dragMesh.absolutePosition, s = this._virtualMeshesInfo[this.currentDraggingPointerIds[1]].dragMesh.absolutePosition, a = ue.Vector3[2]; i.addToRef(s, a), a.scaleInPlace(0.5); const f = ue.Vector3[3]; s.subtractToRef(i, f); const o = f.length() / n.length(), d = a.subtract(r), v = Ze.FromEulerAngles(0, S.GetAngleBetweenVectorsOnPlane(n.normalize(), f.normalize(), S.UpReadOnly), 0), u = this._ownerNode.parent; this._ownerNode.setParent(null); const l = this._getPositionOffsetAround(r.subtract(this._virtualTransformNode.getAbsolutePivotPoint()), o, v); this._virtualTransformNode.rotationQuaternion.multiplyToRef(v, this._ownerNode.rotationQuaternion), this._virtualTransformNode.scaling.scaleToRef(o, this._ownerNode.scaling), this._virtualTransformNode.position.addToRef(d.addInPlace(l), this._ownerNode.position), this.onPositionChangedObservable.notifyObservers({ position: this._ownerNode.position }), this._ownerNode.setParent(u); } _targetDragStart() { const e = this.currentDraggingPointerIds.length, t = this._ownerNode.parent; this._ownerNode.rotationQuaternion || (this._ownerNode.rotationQuaternion = Ze.RotationYawPitchRoll(this._ownerNode.rotation.y, this._ownerNode.rotation.x, this._ownerNode.rotation.z)); const r = this._ownerNode.getAbsolutePivotPoint(); if (this._ownerNode.setParent(null), e === 1) { if (this._targetPosition.copyFrom(this._ownerNode.position), this._targetOrientation.copyFrom(this._ownerNode.rotationQuaternion), this._targetScaling.copyFrom(this._ownerNode.scaling), this.faceCameraOnDragStart && this._scene.activeCamera) { const n = ue.Vector3[0]; this._scene.activeCamera.position.subtractToRef(r, n), n.normalize(); const i = ue.Quaternion[0]; this._scene.useRightHandedSystem ? Ze.FromLookDirectionRHToRef(n, new S(0, 1, 0), i) : Ze.FromLookDirectionLHToRef(n, new S(0, 1, 0), i), i.normalize(), Ze.RotationYawPitchRollToRef(i.toEulerAngles().y, 0, 0, ue.Quaternion[0]), this._targetOrientation.copyFrom(ue.Quaternion[0]); } this._startingPosition.copyFrom(this._targetPosition), this._startingOrientation.copyFrom(this._targetOrientation), this._startingScaling.copyFrom(this._targetScaling); } else e === 2 && (this._virtualTransformNode.setPivotPoint(new S(0, 0, 0), ai.LOCAL), this._virtualTransformNode.position.copyFrom(this._ownerNode.position), this._virtualTransformNode.scaling.copyFrom(this._ownerNode.scaling), this._virtualTransformNode.rotationQuaternion.copyFrom(this._ownerNode.rotationQuaternion), this._virtualTransformNode.setPivotPoint(r, ai.WORLD), this._resetVirtualMeshesPosition()); this._ownerNode.setParent(t); } _targetDrag(e, t) { this.currentDraggingPointerIds.length === 1 ? this._onePointerPositionUpdated(e, t) : this.currentDraggingPointerIds.length === 2 && this._twoPointersPositionUpdated(); } _targetDragEnd() { if (this.currentDraggingPointerIds.length === 1) { this._resetVirtualMeshesPosition(); const e = this.faceCameraOnDragStart; this.faceCameraOnDragStart = !1, this._targetDragStart(), this.faceCameraOnDragStart = e; } } /** * Detaches the behavior from the mesh */ detach() { super.detach(), this._ownerNode && (this._ownerNode.isNearGrabbable = !1, this._ownerNode.getScene().onBeforeRenderObservable.remove(this._sceneRenderObserver)), this._virtualTransformNode && this._virtualTransformNode.dispose(); } } class D6e { constructor() { this._attachPointLocalOffset = new S(), this._workingPosition = new S(), this._workingQuaternion = new Ze(), this._lastTick = -1, this._hit = !1, this.hitNormalOffset = 0.05, this.meshes = [], this.interpolatePose = !0, this.lerpTime = 250, this.keepOrientationVertical = !0, this.enabled = !0, this.maxStickingDistance = 0.8; } /** * Name of the behavior */ get name() { return "SurfaceMagnetism"; } /** * Function called when the behavior needs to be initialized (after attaching it to a target) */ init() { } /** * Attaches the behavior to a transform node * @param target defines the target where the behavior is attached to * @param scene the scene */ attach(e, t) { this._attachedMesh = e, this._scene = t || e.getScene(), this._attachedMesh.rotationQuaternion || (this._attachedMesh.rotationQuaternion = Ze.RotationYawPitchRoll(this._attachedMesh.rotation.y, this._attachedMesh.rotation.x, this._attachedMesh.rotation.z)), this.updateAttachPoint(), this._workingPosition.copyFrom(this._attachedMesh.position), this._workingQuaternion.copyFrom(this._attachedMesh.rotationQuaternion), this._addObservables(); } /** * Detaches the behavior */ detach() { this._attachedMesh = null, this._removeObservables(); } _getTargetPose(e) { if (!this._attachedMesh) return null; if (e && e.hit) { const t = e.getNormal(!0, !0), r = e.pickedPoint; if (!t || !r) return null; t.normalize(); const n = ue.Vector3[0]; return n.copyFrom(t), n.scaleInPlace(this.hitNormalOffset), n.addInPlace(r), this._attachedMesh.parent && (ue.Matrix[0].copyFrom(this._attachedMesh.parent.getWorldMatrix()).invert(), S.TransformNormalToRef(n, ue.Matrix[0], n)), { position: n, quaternion: Ze.RotationYawPitchRoll(-Math.atan2(t.x, -t.z), this.keepOrientationVertical ? 0 : Math.atan2(t.y, Math.sqrt(t.z * t.z + t.x * t.x)), 0) }; } return null; } /** * Updates the attach point with the current geometry extents of the attached mesh */ updateAttachPoint() { this._getAttachPointOffsetToRef(this._attachPointLocalOffset); } /** * Finds the intersection point of the given ray onto the meshes and updates the target. * Transformation will be interpolated according to `interpolatePose` and `lerpTime` properties. * If no mesh of `meshes` are hit, this does nothing. * @param pickInfo The input pickingInfo that will be used to intersect the meshes * @returns a boolean indicating if we found a hit to stick to */ findAndUpdateTarget(e) { if (this._hit = !1, !e.ray) return !1; const t = e.ray.intersectsMeshes(this.meshes)[0]; if (this._attachedMesh && t && t.hit && t.pickedMesh) { const r = this._getTargetPose(t); r && S.Distance(this._attachedMesh.position, r.position) < this.maxStickingDistance && (this._workingPosition.copyFrom(r.position), this._workingQuaternion.copyFrom(r.quaternion), this._hit = !0); } return this._hit; } _getAttachPointOffsetToRef(e) { if (!this._attachedMesh) { e.setAll(0); return; } const t = ue.Quaternion[0]; t.copyFrom(this._attachedMesh.rotationQuaternion), this._attachedMesh.rotationQuaternion.copyFromFloats(0, 0, 0, 1), this._attachedMesh.computeWorldMatrix(); const r = this._attachedMesh.getHierarchyBoundingVectors(), n = ue.Vector3[0]; r.max.addToRef(r.min, n), n.scaleInPlace(0.5), n.z = r.max.z; const i = ue.Matrix[0]; this._attachedMesh.getWorldMatrix().invertToRef(i), S.TransformCoordinatesToRef(n, i, e), this._attachedMesh.rotationQuaternion.copyFrom(t); } _updateTransformToGoal(e) { if (!this._attachedMesh || !this._hit) return; const t = this._attachedMesh.parent; this._attachedMesh.setParent(null); const r = ue.Vector3[0]; if (S.TransformNormalToRef(this._attachPointLocalOffset, this._attachedMesh.getWorldMatrix(), r), !this.interpolatePose) { this._attachedMesh.position.copyFrom(this._workingPosition).subtractInPlace(r), this._attachedMesh.rotationQuaternion.copyFrom(this._workingQuaternion); return; } const n = new S(); S.SmoothToRef(this._attachedMesh.position, this._workingPosition, e, this.lerpTime, n), this._attachedMesh.position.copyFrom(n); const i = new Ze(); i.copyFrom(this._attachedMesh.rotationQuaternion), Ze.SmoothToRef(i, this._workingQuaternion, e, this.lerpTime, this._attachedMesh.rotationQuaternion), this._attachedMesh.setParent(t); } _addObservables() { this._pointerObserver = this._scene.onPointerObservable.add((e) => { this.enabled && e.type == ir.POINTERMOVE && e.pickInfo && this.findAndUpdateTarget(e.pickInfo); }), this._lastTick = Date.now(), this._onBeforeRender = this._scene.onBeforeRenderObservable.add(() => { const e = Date.now(); this._updateTransformToGoal(e - this._lastTick), this._lastTick = e; }); } _removeObservables() { this._scene.onPointerObservable.remove(this._pointerObserver), this._scene.onBeforeRenderObservable.remove(this._onBeforeRender), this._pointerObserver = null, this._onBeforeRender = null; } } class j6e { constructor() { this._tmpQuaternion = new Ze(), this._tmpVectors = [new S(), new S(), new S(), new S(), new S(), new S(), new S()], this._tmpMatrix = new he(), this._tmpInvertView = new he(), this._tmpForward = new S(), this._tmpNodeForward = new S(), this._tmpPosition = new S(), this._workingPosition = new S(), this._workingQuaternion = new Ze(), this._lastTick = -1, this._recenterNextUpdate = !0, this.interpolatePose = !0, this.lerpTime = 500, this.ignoreCameraPitchAndRoll = !1, this.pitchOffset = 15, this.maxViewVerticalDegrees = 30, this.maxViewHorizontalDegrees = 30, this.orientToCameraDeadzoneDegrees = 60, this.ignoreDistanceClamp = !1, this.ignoreAngleClamp = !1, this.verticalMaxDistance = 0, this.defaultDistance = 0.8, this.maximumDistance = 2, this.minimumDistance = 0.3, this.useFixedVerticalOffset = !1, this.fixedVerticalOffset = 0, this._enabled = !0; } /** * The camera that should be followed by this behavior */ get followedCamera() { return this._followedCamera || this._scene.activeCamera; } set followedCamera(e) { this._followedCamera = e; } /** * The name of the behavior */ get name() { return "Follow"; } /** * Initializes the behavior */ init() { } /** * Attaches the follow behavior * @param ownerNode The mesh that will be following once attached * @param followedCamera The camera that should be followed by the node */ attach(e, t) { this._scene = e.getScene(), this.attachedNode = e, t && (this.followedCamera = t), this._addObservables(); } /** * Detaches the behavior from the mesh */ detach() { this.attachedNode = null, this._removeObservables(); } /** * Recenters the attached node in front of the camera on the next update */ recenter() { this._recenterNextUpdate = !0; } _angleBetweenVectorAndPlane(e, t) { return this._tmpVectors[0].copyFrom(e), e = this._tmpVectors[0], this._tmpVectors[1].copyFrom(t), t = this._tmpVectors[1], e.normalize(), t.normalize(), Math.PI / 2 - Math.acos(S.Dot(e, t)); } _length2D(e) { return Math.sqrt(e.x * e.x + e.z * e.z); } _distanceClamp(e, t = !1) { let r = this.minimumDistance, n = this.maximumDistance; const i = this.defaultDistance, s = this._tmpVectors[0]; s.copyFrom(e); let a = s.length(); if (s.normalizeFromLength(a), this.ignoreCameraPitchAndRoll) { r = this._length2D(s) * r, n = this._length2D(s) * n; const o = this._length2D(e); s.scaleInPlace(a / o), a = o; } let f = a; return t ? f = i : f = Xt.Clamp(a, r, n), e.copyFrom(s).scaleInPlace(f), a !== f; } _applyVerticalClamp(e) { this.verticalMaxDistance !== 0 && (e.y = Xt.Clamp(e.y, -this.verticalMaxDistance, this.verticalMaxDistance)); } _toOrientationQuatToRef(e, t) { Ze.RotationYawPitchRollToRef(Math.atan2(e.x, e.z), Math.atan2(e.y, Math.sqrt(e.z * e.z + e.x * e.x)), 0, t); } _applyPitchOffset(e) { const t = this._tmpVectors[0], r = this._tmpVectors[1]; t.copyFromFloats(0, 0, this._scene.useRightHandedSystem ? -1 : 1), r.copyFromFloats(1, 0, 0), S.TransformNormalToRef(t, e, t), t.y = 0, t.normalize(), S.TransformNormalToRef(r, e, r), Ze.RotationAxisToRef(r, this.pitchOffset * Math.PI / 180, this._tmpQuaternion), t.rotateByQuaternionToRef(this._tmpQuaternion, t), this._toOrientationQuatToRef(t, this._tmpQuaternion), this._tmpQuaternion.toRotationMatrix(this._tmpMatrix), e.copyFrom(this._tmpMatrix); } _angularClamp(e, t) { const r = this._tmpVectors[5]; r.copyFromFloats(0, 0, this._scene.useRightHandedSystem ? -1 : 1); const n = this._tmpVectors[6]; n.copyFromFloats(1, 0, 0), S.TransformNormalToRef(r, e, r), S.TransformNormalToRef(n, e, n); const i = S.UpReadOnly; if (t.length() < Dn) return !1; let a = !1; const f = this._tmpQuaternion; if (this.ignoreCameraPitchAndRoll) { const v = S.GetAngleBetweenVectorsOnPlane(t, r, n); Ze.RotationAxisToRef(n, v, f), t.rotateByQuaternionToRef(f, t); } else { const v = -S.GetAngleBetweenVectorsOnPlane(t, r, n), u = this.maxViewVerticalDegrees * Math.PI / 180 * 0.5; v < -u ? (Ze.RotationAxisToRef(n, -v - u, f), t.rotateByQuaternionToRef(f, t), a = !0) : v > u && (Ze.RotationAxisToRef(n, -v + u, f), t.rotateByQuaternionToRef(f, t), a = !0); } const o = this._angleBetweenVectorAndPlane(t, n) * (this._scene.useRightHandedSystem ? -1 : 1), d = this.maxViewHorizontalDegrees * Math.PI / 180 * 0.5; return o < -d ? (Ze.RotationAxisToRef(i, -o - d, f), t.rotateByQuaternionToRef(f, t), a = !0) : o > d && (Ze.RotationAxisToRef(i, -o + d, f), t.rotateByQuaternionToRef(f, t), a = !0), a; } _orientationClamp(e, t) { var r; const n = this._tmpVectors[0]; n.copyFrom(e).scaleInPlace(-1).normalize(); const i = this._tmpVectors[1], s = this._tmpVectors[2]; i.copyFromFloats(0, 1, 0), S.CrossToRef(n, i, s); const a = s.length(); a < Dn || (s.normalizeFromLength(a), S.CrossToRef(s, n, i), !((r = this.attachedNode) === null || r === void 0) && r.getScene().useRightHandedSystem ? Ze.FromLookDirectionRHToRef(n, i, t) : Ze.FromLookDirectionLHToRef(n, i, t)); } _passedOrientationDeadzone(e, t) { const r = this._tmpVectors[5]; return r.copyFrom(e), r.normalize(), Math.abs(S.GetAngleBetweenVectorsOnPlane(t, r, S.UpReadOnly)) * 180 / Math.PI > this.orientToCameraDeadzoneDegrees; } _updateLeashing(e) { if (this.attachedNode && this._enabled) { const t = this.attachedNode.parent; this.attachedNode.setParent(null); const r = this.attachedNode.getWorldMatrix(), n = this._workingPosition, i = this._workingQuaternion, s = this.attachedNode.getPivotPoint(), a = this._tmpInvertView; a.copyFrom(e.getViewMatrix()), a.invert(), S.TransformCoordinatesToRef(s, r, n); const f = this._tmpPosition; f.copyFromFloats(0, 0, 0), S.TransformCoordinatesToRef(f, r, f), f.scaleInPlace(-1).subtractInPlace(s), n.subtractInPlace(e.globalPosition), this.ignoreCameraPitchAndRoll && this._applyPitchOffset(a); let o = !1; const d = this._tmpForward; d.copyFromFloats(0, 0, this._scene.useRightHandedSystem ? -1 : 1), S.TransformNormalToRef(d, a, d); const v = this._tmpNodeForward; if (v.copyFromFloats(0, 0, this._scene.useRightHandedSystem ? -1 : 1), S.TransformNormalToRef(v, r, v), this._recenterNextUpdate) n.copyFrom(d).scaleInPlace(this.defaultDistance); else if (this.ignoreAngleClamp) { const l = n.length(); n.copyFrom(d).scaleInPlace(l); } else o = this._angularClamp(a, n); let u = !1; this.ignoreDistanceClamp || (u = this._distanceClamp(n, o), this._applyVerticalClamp(n)), this.useFixedVerticalOffset && (n.y = f.y - e.globalPosition.y + this.fixedVerticalOffset), (o || u || this._passedOrientationDeadzone(n, v) || this._recenterNextUpdate) && this._orientationClamp(n, i), this._workingPosition.subtractInPlace(s), this._recenterNextUpdate = !1, this.attachedNode.setParent(t); } } _updateTransformToGoal(e) { if (!this.attachedNode || !this.followedCamera || !this._enabled) return; this.attachedNode.rotationQuaternion || (this.attachedNode.rotationQuaternion = Ze.Identity()); const t = this.attachedNode.parent; if (this.attachedNode.setParent(null), !this.interpolatePose) { this.attachedNode.position.copyFrom(this.followedCamera.globalPosition).addInPlace(this._workingPosition), this.attachedNode.rotationQuaternion.copyFrom(this._workingQuaternion); return; } const r = new S(); r.copyFrom(this.attachedNode.position).subtractInPlace(this.followedCamera.globalPosition), S.SmoothToRef(r, this._workingPosition, e, this.lerpTime, r), r.addInPlace(this.followedCamera.globalPosition), this.attachedNode.position.copyFrom(r); const n = new Ze(); n.copyFrom(this.attachedNode.rotationQuaternion), Ze.SmoothToRef(n, this._workingQuaternion, e, this.lerpTime, this.attachedNode.rotationQuaternion), this.attachedNode.setParent(t); } _addObservables() { this._lastTick = Date.now(), this._onBeforeRender = this._scene.onBeforeRenderObservable.add(() => { if (!this.followedCamera) return; const e = Date.now(); this._updateLeashing(this.followedCamera), this._updateTransformToGoal(e - this._lastTick), this._lastTick = e; }); } _removeObservables() { this._onBeforeRender && this._scene.onBeforeRenderObservable.remove(this._onBeforeRender); } } class Gi { } Gi.ANCHOR_SYSTEM = "xr-anchor-system"; Gi.BACKGROUND_REMOVER = "xr-background-remover"; Gi.HIT_TEST = "xr-hit-test"; Gi.MESH_DETECTION = "xr-mesh-detection"; Gi.PHYSICS_CONTROLLERS = "xr-physics-controller"; Gi.PLANE_DETECTION = "xr-plane-detection"; Gi.POINTER_SELECTION = "xr-controller-pointer-selection"; Gi.TELEPORTATION = "xr-controller-teleportation"; Gi.FEATURE_POINTS = "xr-feature-points"; Gi.HAND_TRACKING = "xr-hand-tracking"; Gi.IMAGE_TRACKING = "xr-image-tracking"; Gi.NEAR_INTERACTION = "xr-near-interaction"; Gi.DOM_OVERLAY = "xr-dom-overlay"; Gi.MOVEMENT = "xr-controller-movement"; Gi.LIGHT_ESTIMATION = "xr-light-estimation"; Gi.EYE_TRACKING = "xr-eye-tracking"; Gi.WALKING_LOCOMOTION = "xr-walking-locomotion"; Gi.LAYERS = "xr-layers"; Gi.DEPTH_SENSING = "xr-depth-sensing"; Gi.SPACE_WARP = "xr-space-warp"; Gi.RAW_CAMERA_ACCESS = "xr-raw-camera-access"; class So { /** * constructs a new features manages. * * @param _xrSessionManager an instance of WebXRSessionManager */ constructor(e) { this._xrSessionManager = e, this._features = {}, this._xrSessionManager.onXRSessionInit.add(() => { this.getEnabledFeatures().forEach((t) => { const r = this._features[t]; r.enabled && !r.featureImplementation.attached && !r.featureImplementation.disableAutoAttach && this.attachFeature(t); }); }), this._xrSessionManager.onXRSessionEnded.add(() => { this.getEnabledFeatures().forEach((t) => { const r = this._features[t]; r.enabled && r.featureImplementation.attached && this.detachFeature(t); }); }); } /** * Used to register a module. After calling this function a developer can use this feature in the scene. * Mainly used internally. * * @param featureName the name of the feature to register * @param constructorFunction the function used to construct the module * @param version the (babylon) version of the module * @param stable is that a stable version of this module */ static AddWebXRFeature(e, t, r = 1, n = !1) { this._AvailableFeatures[e] = this._AvailableFeatures[e] || { latest: r }, r > this._AvailableFeatures[e].latest && (this._AvailableFeatures[e].latest = r), n && (this._AvailableFeatures[e].stable = r), this._AvailableFeatures[e][r] = t; } /** * Returns a constructor of a specific feature. * * @param featureName the name of the feature to construct * @param version the version of the feature to load * @param xrSessionManager the xrSessionManager. Used to construct the module * @param options optional options provided to the module. * @returns a function that, when called, will return a new instance of this feature */ static ConstructFeature(e, t = 1, r, n) { const i = this._AvailableFeatures[e][t]; if (!i) throw new Error("feature not found"); return i(r, n); } /** * Can be used to return the list of features currently registered * * @returns an Array of available features */ static GetAvailableFeatures() { return Object.keys(this._AvailableFeatures); } /** * Gets the versions available for a specific feature * @param featureName the name of the feature * @returns an array with the available versions */ static GetAvailableVersions(e) { return Object.keys(this._AvailableFeatures[e]); } /** * Return the latest unstable version of this feature * @param featureName the name of the feature to search * @returns the version number. if not found will return -1 */ static GetLatestVersionOfFeature(e) { return this._AvailableFeatures[e] && this._AvailableFeatures[e].latest || -1; } /** * Return the latest stable version of this feature * @param featureName the name of the feature to search * @returns the version number. if not found will return -1 */ static GetStableVersionOfFeature(e) { return this._AvailableFeatures[e] && this._AvailableFeatures[e].stable || -1; } /** * Attach a feature to the current session. Mainly used when session started to start the feature effect. * Can be used during a session to start a feature * @param featureName the name of feature to attach */ attachFeature(e) { const t = this._features[e]; t && t.enabled && !t.featureImplementation.attached && t.featureImplementation.attach(); } /** * Can be used inside a session or when the session ends to detach a specific feature * @param featureName the name of the feature to detach */ detachFeature(e) { const t = this._features[e]; t && t.featureImplementation.attached && t.featureImplementation.detach(); } /** * Used to disable an already-enabled feature * The feature will be disposed and will be recreated once enabled. * @param featureName the feature to disable * @returns true if disable was successful */ // eslint-disable-next-line @typescript-eslint/naming-convention disableFeature(e) { const t = typeof e == "string" ? e : e.Name, r = this._features[t]; return r && r.enabled ? (r.enabled = !1, this.detachFeature(t), r.featureImplementation.dispose(), delete this._features[t], !0) : !1; } /** * dispose this features manager */ dispose() { this.getEnabledFeatures().forEach((e) => { this.disableFeature(e); }); } /** * Enable a feature using its name and a version. This will enable it in the scene, and will be responsible to attach it when the session starts. * If used twice, the old version will be disposed and a new one will be constructed. This way you can re-enable with different configuration. * * @param featureName the name of the feature to load or the class of the feature * @param version optional version to load. if not provided the latest version will be enabled * @param moduleOptions options provided to the module. Ses the module documentation / constructor * @param attachIfPossible if set to true (default) the feature will be automatically attached, if it is currently possible * @param required is this feature required to the app. If set to true the session init will fail if the feature is not available. * @returns a new constructed feature or throws an error if feature not found or conflicts with another enabled feature. */ enableFeature(e, t = "latest", r = {}, n = !0, i = !0) { const s = typeof e == "string" ? e : e.Name; let a = 0; if (typeof t == "string") { if (!t) throw new Error(`Error in provided version - ${s} (${t})`); if (t === "stable" ? a = So.GetStableVersionOfFeature(s) : t === "latest" ? a = So.GetLatestVersionOfFeature(s) : a = +t, a === -1 || isNaN(a)) throw new Error(`feature not found - ${s} (${t})`); } else a = t; const f = So._ConflictingFeatures[s]; if (f !== void 0 && this.getEnabledFeatures().indexOf(f) !== -1) throw new Error(`Feature ${s} cannot be enabled while ${f} is enabled.`); const o = this._features[s], d = So.ConstructFeature(s, a, this._xrSessionManager, r); if (!d) throw new Error(`feature not found - ${s}`); o && this.disableFeature(s); const v = d(); if (v.dependsOn && !v.dependsOn.every((l) => !!this._features[l])) throw new Error(`Dependant features missing. Make sure the following features are enabled - ${v.dependsOn.join(", ")}`); if (v.isCompatible()) return this._features[s] = { featureImplementation: v, enabled: !0, version: a, required: i }, n ? this._xrSessionManager.session && !this._features[s].featureImplementation.attached && this.attachFeature(s) : this._features[s].featureImplementation.disableAutoAttach = !0, this._features[s].featureImplementation; if (i) throw new Error("required feature not compatible"); return ye.Warn(`Feature ${s} not compatible with the current environment/browser and was not enabled.`), v; } /** * get the implementation of an enabled feature. * @param featureName the name of the feature to load * @returns the feature class, if found */ getEnabledFeature(e) { return this._features[e] && this._features[e].featureImplementation; } /** * Get the list of enabled features * @returns an array of enabled features */ getEnabledFeatures() { return Object.keys(this._features); } /** * This function will extend the session creation configuration object with enabled features. * If, for example, the anchors feature is enabled, it will be automatically added to the optional or required features list, * according to the defined "required" variable, provided during enableFeature call * @param xrSessionInit the xr Session init object to extend * * @returns an extended XRSessionInit object */ async _extendXRSessionInitObject(e) { const t = this.getEnabledFeatures(); for (const r of t) { const n = this._features[r], i = n.featureImplementation.xrNativeFeatureName; if (i && (n.required ? (e.requiredFeatures = e.requiredFeatures || [], e.requiredFeatures.indexOf(i) === -1 && e.requiredFeatures.push(i)) : (e.optionalFeatures = e.optionalFeatures || [], e.optionalFeatures.indexOf(i) === -1 && e.optionalFeatures.push(i))), n.featureImplementation.getXRSessionInitExtension) { const s = await n.featureImplementation.getXRSessionInitExtension(); e = Object.assign(Object.assign({}, e), s); } } return e; } } So._AvailableFeatures = {}; So._ConflictingFeatures = { [Gi.TELEPORTATION]: Gi.MOVEMENT, [Gi.MOVEMENT]: Gi.TELEPORTATION }; class L9 { /** * Construct a new (abstract) WebXR feature * @param _xrSessionManager the xr session manager for this feature */ constructor(e) { this._xrSessionManager = e, this._attached = !1, this._removeOnDetach = [], this.isDisposed = !1, this.disableAutoAttach = !1, this.xrNativeFeatureName = "", this.onFeatureAttachObservable = new Oe(), this.onFeatureDetachObservable = new Oe(); } /** * Is this feature attached */ get attached() { return this._attached; } /** * attach this feature * * @param force should attachment be forced (even when already attached) * @returns true if successful, false is failed or already attached */ attach(e) { if (this.isDisposed) return !1; if (e) this.attached && this.detach(); else if (this.attached) return !1; return this._attached = !0, this._addNewAttachObserver(this._xrSessionManager.onXRFrameObservable, (t) => this._onXRFrame(t)), this.onFeatureAttachObservable.notifyObservers(this), !0; } /** * detach this feature. * * @returns true if successful, false if failed or already detached */ detach() { return this._attached ? (this._attached = !1, this._removeOnDetach.forEach((e) => { e.observable.remove(e.observer); }), this.onFeatureDetachObservable.notifyObservers(this), !0) : (this.disableAutoAttach = !0, !1); } /** * Dispose this feature and all of the resources attached */ dispose() { this.detach(), this.isDisposed = !0, this.onFeatureAttachObservable.clear(), this.onFeatureDetachObservable.clear(); } /** * This function will be executed during before enabling the feature and can be used to not-allow enabling it. * Note that at this point the session has NOT started, so this is purely checking if the browser supports it * * @returns whether or not the feature is compatible in this environment */ isCompatible() { return !0; } /** * This is used to register callbacks that will automatically be removed when detach is called. * @param observable the observable to which the observer will be attached * @param callback the callback to register */ _addNewAttachObserver(e, t) { this._removeOnDetach.push({ observable: e, observer: e.add(t) }); } } class ta { /** * Initializes the physics joint * @param type The type of the physics joint * @param jointData The data for the physics joint */ constructor(e, t) { this.type = e, this.jointData = t, t.nativeParams = t.nativeParams || {}; } /** * Gets the physics joint */ get physicsJoint() { return this._physicsJoint; } /** * Sets the physics joint */ set physicsJoint(e) { this._physicsJoint, this._physicsJoint = e; } /** * Sets the physics plugin */ set physicsPlugin(e) { this._physicsPlugin = e; } /** * Execute a function that is physics-plugin specific. * @param {Function} func the function that will be executed. * It accepts two parameters: the physics world and the physics joint */ executeNativeFunction(e) { e(this._physicsPlugin.world, this._physicsJoint); } } ta.DistanceJoint = 0; ta.HingeJoint = 1; ta.BallAndSocketJoint = 2; ta.WheelJoint = 3; ta.SliderJoint = 4; ta.PrismaticJoint = 5; ta.UniversalJoint = 6; ta.Hinge2Joint = ta.WheelJoint; ta.PointToPointJoint = 8; ta.SpringJoint = 9; ta.LockJoint = 10; class w6e extends ta { /** * * @param jointData The data for the Distance-Joint */ constructor(e) { super(ta.DistanceJoint, e); } /** * Update the predefined distance. * @param maxDistance The maximum preferred distance * @param minDistance The minimum preferred distance */ updateDistance(e, t) { this._physicsPlugin.updateDistanceJoint(this, e, t); } } class iQ extends ta { /** * Initializes the Motor-Enabled Joint * @param type The type of the joint * @param jointData The physical joint data for the joint */ constructor(e, t) { super(e, t); } /** * Set the motor values. * Attention, this function is plugin specific. Engines won't react 100% the same. * @param force the force to apply * @param maxForce max force for this motor. */ setMotor(e, t) { this._physicsPlugin.setMotor(this, e || 0, t); } /** * Set the motor's limits. * Attention, this function is plugin specific. Engines won't react 100% the same. * @param upperLimit The upper limit of the motor * @param lowerLimit The lower limit of the motor */ setLimit(e, t) { this._physicsPlugin.setLimit(this, e, t); } } class m6e extends iQ { /** * Initializes the Hinge-Joint * @param jointData The joint data for the Hinge-Joint */ constructor(e) { super(ta.HingeJoint, e); } /** * Set the motor values. * Attention, this function is plugin specific. Engines won't react 100% the same. * @param {number} force the force to apply * @param {number} maxForce max force for this motor. */ setMotor(e, t) { this._physicsPlugin.setMotor(this, e || 0, t); } /** * Set the motor's limits. * Attention, this function is plugin specific. Engines won't react 100% the same. * @param upperLimit The upper limit of the motor * @param lowerLimit The lower limit of the motor */ setLimit(e, t) { this._physicsPlugin.setLimit(this, e, t); } } class B6e extends iQ { /** * Initializes the Hinge2-Joint * @param jointData The joint data for the Hinge2-Joint */ constructor(e) { super(ta.Hinge2Joint, e); } /** * Set the motor values. * Attention, this function is plugin specific. Engines won't react 100% the same. * @param targetSpeed the speed the motor is to reach * @param maxForce max force for this motor. * @param motorIndex motor's index, 0 or 1. */ setMotor(e, t, r = 0) { this._physicsPlugin.setMotor(this, e || 0, t, r); } /** * Set the motor limits. * Attention, this function is plugin specific. Engines won't react 100% the same. * @param upperLimit the upper limit * @param lowerLimit lower limit * @param motorIndex the motor's index, 0 or 1. */ setLimit(e, t, r = 0) { this._physicsPlugin.setLimit(this, e, t, r); } } Ee._PhysicsImpostorParser = function(A, e, t) { return new tn(e, t.physicsImpostor, { mass: t.physicsMass, friction: t.physicsFriction, restitution: t.physicsRestitution }, A); }; class tn { /** * Specifies if the physics imposter is disposed */ get isDisposed() { return this._isDisposed; } /** * Gets the mass of the physics imposter */ get mass() { return this._physicsEngine ? this._physicsEngine.getPhysicsPlugin().getBodyMass(this) : 0; } set mass(e) { this.setMass(e); } /** * Gets the coefficient of friction */ get friction() { return this._physicsEngine ? this._physicsEngine.getPhysicsPlugin().getBodyFriction(this) : 0; } /** * Sets the coefficient of friction */ set friction(e) { this._physicsEngine && this._physicsEngine.getPhysicsPlugin().setBodyFriction(this, e); } /** * Gets the coefficient of restitution */ get restitution() { return this._physicsEngine ? this._physicsEngine.getPhysicsPlugin().getBodyRestitution(this) : 0; } /** * Sets the coefficient of restitution */ set restitution(e) { this._physicsEngine && this._physicsEngine.getPhysicsPlugin().setBodyRestitution(this, e); } /** * Gets the pressure of a soft body; only supported by the AmmoJSPlugin */ get pressure() { if (!this._physicsEngine) return 0; const e = this._physicsEngine.getPhysicsPlugin(); return e.setBodyPressure ? e.getBodyPressure(this) : 0; } /** * Sets the pressure of a soft body; only supported by the AmmoJSPlugin */ set pressure(e) { if (!this._physicsEngine) return; const t = this._physicsEngine.getPhysicsPlugin(); t.setBodyPressure && t.setBodyPressure(this, e); } /** * Gets the stiffness of a soft body; only supported by the AmmoJSPlugin */ get stiffness() { if (!this._physicsEngine) return 0; const e = this._physicsEngine.getPhysicsPlugin(); return e.getBodyStiffness ? e.getBodyStiffness(this) : 0; } /** * Sets the stiffness of a soft body; only supported by the AmmoJSPlugin */ set stiffness(e) { if (!this._physicsEngine) return; const t = this._physicsEngine.getPhysicsPlugin(); t.setBodyStiffness && t.setBodyStiffness(this, e); } /** * Gets the velocityIterations of a soft body; only supported by the AmmoJSPlugin */ get velocityIterations() { if (!this._physicsEngine) return 0; const e = this._physicsEngine.getPhysicsPlugin(); return e.getBodyVelocityIterations ? e.getBodyVelocityIterations(this) : 0; } /** * Sets the velocityIterations of a soft body; only supported by the AmmoJSPlugin */ set velocityIterations(e) { if (!this._physicsEngine) return; const t = this._physicsEngine.getPhysicsPlugin(); t.setBodyVelocityIterations && t.setBodyVelocityIterations(this, e); } /** * Gets the positionIterations of a soft body; only supported by the AmmoJSPlugin */ get positionIterations() { if (!this._physicsEngine) return 0; const e = this._physicsEngine.getPhysicsPlugin(); return e.getBodyPositionIterations ? e.getBodyPositionIterations(this) : 0; } /** * Sets the positionIterations of a soft body; only supported by the AmmoJSPlugin */ set positionIterations(e) { if (!this._physicsEngine) return; const t = this._physicsEngine.getPhysicsPlugin(); t.setBodyPositionIterations && t.setBodyPositionIterations(this, e); } /** * Initializes the physics imposter * @param object The physics-enabled object used as the physics imposter * @param type The type of the physics imposter. Types are available as static members of this class. * @param _options The options for the physics imposter * @param _scene The Babylon scene */ constructor(e, t, r = { mass: 0 }, n) { if (this.object = e, this.type = t, this._options = r, this._scene = n, this._pluginData = {}, this._bodyUpdateRequired = !1, this._onBeforePhysicsStepCallbacks = new Array(), this._onAfterPhysicsStepCallbacks = new Array(), this._onPhysicsCollideCallbacks = [], this._deltaPosition = S.Zero(), this._isDisposed = !1, this.soft = !1, this.segments = 0, this._tmpQuat = new Ze(), this._tmpQuat2 = new Ze(), this.beforeStep = () => { this._physicsEngine && (this.object.translate(this._deltaPosition, -1), this._deltaRotationConjugated && this.object.rotationQuaternion && this.object.rotationQuaternion.multiplyToRef(this._deltaRotationConjugated, this.object.rotationQuaternion), this.object.computeWorldMatrix(!1), this.object.parent && this.object.rotationQuaternion ? (this.getParentsRotation(), this._tmpQuat.multiplyToRef(this.object.rotationQuaternion, this._tmpQuat)) : this._tmpQuat.copyFrom(this.object.rotationQuaternion || new Ze()), this._options.disableBidirectionalTransformation || this.object.rotationQuaternion && this._physicsEngine.getPhysicsPlugin().setPhysicsBodyTransformation( this, /*bInfo.boundingBox.centerWorld*/ this.object.getAbsolutePosition(), this._tmpQuat ), this._onBeforePhysicsStepCallbacks.forEach((i) => { i(this); })); }, this.afterStep = () => { this._physicsEngine && (this._onAfterPhysicsStepCallbacks.forEach((i) => { i(this); }), this._physicsEngine.getPhysicsPlugin().setTransformationFromPhysicsBody(this), this.object.parent && this.object.rotationQuaternion && (this.getParentsRotation(), this._tmpQuat.conjugateInPlace(), this._tmpQuat.multiplyToRef(this.object.rotationQuaternion, this.object.rotationQuaternion)), this.object.setAbsolutePosition(this.object.position), this._deltaRotation ? (this.object.rotationQuaternion && this.object.rotationQuaternion.multiplyToRef(this._deltaRotation, this.object.rotationQuaternion), this._deltaPosition.applyRotationQuaternionToRef(this._deltaRotation, tn._TmpVecs[0]), this.object.translate(tn._TmpVecs[0], 1)) : this.object.translate(this._deltaPosition, 1), this.object.computeWorldMatrix(!0)); }, this.onCollideEvent = null, this.onCollide = (i) => { if (!this._onPhysicsCollideCallbacks.length && !this.onCollideEvent || !this._physicsEngine) return; const s = this._physicsEngine.getImpostorWithPhysicsBody(i.body); s && (this.onCollideEvent && this.onCollideEvent(this, s), this._onPhysicsCollideCallbacks.filter((a) => a.otherImpostors.indexOf(s) !== -1).forEach((a) => { a.callback(this, s, i.point, i.distance, i.impulse, i.normal); })); }, !this.object) { Se.Error("No object was provided. A physics object is obligatory"); return; } this.object.parent && r.mass !== 0 && Se.Warn("A physics impostor has been created for an object which has a parent. Babylon physics currently works in local space so unexpected issues may occur."), !this._scene && e.getScene && (this._scene = e.getScene()), this._scene && (this.type > 100 && (this.soft = !0), this._physicsEngine = this._scene.getPhysicsEngine(), this._physicsEngine ? (this.object.rotationQuaternion || (this.object.rotation ? this.object.rotationQuaternion = Ze.RotationYawPitchRoll(this.object.rotation.y, this.object.rotation.x, this.object.rotation.z) : this.object.rotationQuaternion = new Ze()), this._options.mass = r.mass === void 0 ? 0 : r.mass, this._options.friction = r.friction === void 0 ? 0.2 : r.friction, this._options.restitution = r.restitution === void 0 ? 0.2 : r.restitution, this.soft && (this._options.mass = this._options.mass > 0 ? this._options.mass : 1, this._options.pressure = r.pressure === void 0 ? 200 : r.pressure, this._options.stiffness = r.stiffness === void 0 ? 1 : r.stiffness, this._options.velocityIterations = r.velocityIterations === void 0 ? 20 : r.velocityIterations, this._options.positionIterations = r.positionIterations === void 0 ? 20 : r.positionIterations, this._options.fixedPoints = r.fixedPoints === void 0 ? 0 : r.fixedPoints, this._options.margin = r.margin === void 0 ? 0 : r.margin, this._options.damping = r.damping === void 0 ? 0 : r.damping, this._options.path = r.path === void 0 ? null : r.path, this._options.shape = r.shape === void 0 ? null : r.shape), this._joints = [], !this.object.parent || this._options.ignoreParent ? this._init() : this.object.parent.physicsImpostor && Se.Warn("You must affect impostors to children before affecting impostor to parent.")) : Se.Error("Physics not enabled. Please use scene.enablePhysics(...) before creating impostors.")); } /** * This function will completely initialize this impostor. * It will create a new body - but only if this mesh has no parent. * If it has, this impostor will not be used other than to define the impostor * of the child mesh. * @internal */ _init() { this._physicsEngine && (this._physicsEngine.removeImpostor(this), this.physicsBody = null, this._parent = this._parent || this._getPhysicsParent(), !this._isDisposed && (!this.parent || this._options.ignoreParent) && this._physicsEngine.addImpostor(this)); } _getPhysicsParent() { return this.object.parent instanceof jn ? this.object.parent.physicsImpostor : null; } /** * Should a new body be generated. * @returns boolean specifying if body initialization is required */ isBodyInitRequired() { return this._bodyUpdateRequired || !this._physicsBody && (!this._parent || !!this._options.ignoreParent); } /** * Sets the updated scaling */ setScalingUpdated() { this.forceUpdate(); } /** * Force a regeneration of this or the parent's impostor's body. * Use with caution - This will remove all previously-instantiated joints. */ forceUpdate() { this._init(), this.parent && !this._options.ignoreParent && this.parent.forceUpdate(); } /*public get mesh(): AbstractMesh { return this._mesh; }*/ /** * Gets the body that holds this impostor. Either its own, or its parent. */ get physicsBody() { return this._parent && !this._options.ignoreParent ? this._parent.physicsBody : this._physicsBody; } /** * Get the parent of the physics imposter * @returns Physics imposter or null */ get parent() { return !this._options.ignoreParent && this._parent ? this._parent : null; } /** * Sets the parent of the physics imposter */ set parent(e) { this._parent = e; } /** * Set the physics body. Used mainly by the physics engine/plugin */ set physicsBody(e) { this._physicsBody && this._physicsEngine && this._physicsEngine.getPhysicsPlugin().removePhysicsBody(this), this._physicsBody = e, this.resetUpdateFlags(); } /** * Resets the update flags */ resetUpdateFlags() { this._bodyUpdateRequired = !1; } /** * Gets the object extents * @returns the object extents */ getObjectExtents() { if (this.object.getBoundingInfo) { const e = this.object.rotationQuaternion, t = this.object.scaling.clone(); this.object.rotationQuaternion = tn.IDENTITY_QUATERNION; const r = this.object.computeWorldMatrix && this.object.computeWorldMatrix(!0); r && r.decompose(t, void 0, void 0); const i = this.object.getBoundingInfo().boundingBox.extendSize.scale(2).multiplyInPlace(t); return i.x = Math.abs(i.x), i.y = Math.abs(i.y), i.z = Math.abs(i.z), this.object.rotationQuaternion = e, this.object.computeWorldMatrix && this.object.computeWorldMatrix(!0), i; } else return tn.DEFAULT_OBJECT_SIZE; } /** * Gets the object center * @returns The object center */ getObjectCenter() { return this.object.getBoundingInfo ? this.object.getBoundingInfo().boundingBox.centerWorld : this.object.position; } /** * Get a specific parameter from the options parameters * @param paramName The object parameter name * @returns The object parameter */ getParam(e) { return this._options[e]; } /** * Sets a specific parameter in the options given to the physics plugin * @param paramName The parameter name * @param value The value of the parameter */ setParam(e, t) { this._options[e] = t, this._bodyUpdateRequired = !0; } /** * Specifically change the body's mass. Won't recreate the physics body object * @param mass The mass of the physics imposter */ setMass(e) { this.getParam("mass") !== e && this.setParam("mass", e), this._physicsEngine && this._physicsEngine.getPhysicsPlugin().setBodyMass(this, e); } /** * Gets the linear velocity * @returns linear velocity or null */ getLinearVelocity() { return this._physicsEngine ? this._physicsEngine.getPhysicsPlugin().getLinearVelocity(this) : S.Zero(); } /** * Sets the linear velocity * @param velocity linear velocity or null */ setLinearVelocity(e) { this._physicsEngine && this._physicsEngine.getPhysicsPlugin().setLinearVelocity(this, e); } /** * Gets the angular velocity * @returns angular velocity or null */ getAngularVelocity() { return this._physicsEngine ? this._physicsEngine.getPhysicsPlugin().getAngularVelocity(this) : S.Zero(); } /** * Sets the angular velocity * @param velocity The velocity or null */ setAngularVelocity(e) { this._physicsEngine && this._physicsEngine.getPhysicsPlugin().setAngularVelocity(this, e); } /** * Execute a function with the physics plugin native code * Provide a function the will have two variables - the world object and the physics body object * @param func The function to execute with the physics plugin native code */ executeNativeFunction(e) { this._physicsEngine && e(this._physicsEngine.getPhysicsPlugin().world, this.physicsBody); } /** * Register a function that will be executed before the physics world is stepping forward * @param func The function to execute before the physics world is stepped forward */ registerBeforePhysicsStep(e) { this._onBeforePhysicsStepCallbacks.push(e); } /** * Unregister a function that will be executed before the physics world is stepping forward * @param func The function to execute before the physics world is stepped forward */ unregisterBeforePhysicsStep(e) { const t = this._onBeforePhysicsStepCallbacks.indexOf(e); t > -1 ? this._onBeforePhysicsStepCallbacks.splice(t, 1) : Se.Warn("Function to remove was not found"); } /** * Register a function that will be executed after the physics step * @param func The function to execute after physics step */ registerAfterPhysicsStep(e) { this._onAfterPhysicsStepCallbacks.push(e); } /** * Unregisters a function that will be executed after the physics step * @param func The function to execute after physics step */ unregisterAfterPhysicsStep(e) { const t = this._onAfterPhysicsStepCallbacks.indexOf(e); t > -1 ? this._onAfterPhysicsStepCallbacks.splice(t, 1) : Se.Warn("Function to remove was not found"); } /** * register a function that will be executed when this impostor collides against a different body * @param collideAgainst Physics imposter, or array of physics imposters to collide against * @param func Callback that is executed on collision */ registerOnPhysicsCollide(e, t) { const r = e instanceof Array ? e : [e]; this._onPhysicsCollideCallbacks.push({ callback: t, otherImpostors: r }); } /** * Unregisters the physics imposter's collision callback * @param collideAgainst The physics object to collide against * @param func Callback to execute on collision */ unregisterOnPhysicsCollide(e, t) { const r = e instanceof Array ? e : [e]; let n = -1; this._onPhysicsCollideCallbacks.some((s, a) => { if (s.callback === t && s.otherImpostors.length === r.length) { const f = s.otherImpostors.every((o) => r.indexOf(o) > -1); return f && (n = a), f; } return !1; }) ? this._onPhysicsCollideCallbacks.splice(n, 1) : Se.Warn("Function to remove was not found"); } /** * Get the parent rotation * @returns The parent rotation */ getParentsRotation() { let e = this.object.parent; for (this._tmpQuat.copyFromFloats(0, 0, 0, 1); e; ) e.rotationQuaternion ? this._tmpQuat2.copyFrom(e.rotationQuaternion) : Ze.RotationYawPitchRollToRef(e.rotation.y, e.rotation.x, e.rotation.z, this._tmpQuat2), this._tmpQuat.multiplyToRef(this._tmpQuat2, this._tmpQuat), e = e.parent; return this._tmpQuat; } /** * Apply a force * @param force The force to apply * @param contactPoint The contact point for the force * @returns The physics imposter */ applyForce(e, t) { return this._physicsEngine && this._physicsEngine.getPhysicsPlugin().applyForce(this, e, t), this; } /** * Apply an impulse * @param force The impulse force * @param contactPoint The contact point for the impulse force * @returns The physics imposter */ applyImpulse(e, t) { return this._physicsEngine && this._physicsEngine.getPhysicsPlugin().applyImpulse(this, e, t), this; } /** * A help function to create a joint * @param otherImpostor A physics imposter used to create a joint * @param jointType The type of joint * @param jointData The data for the joint * @returns The physics imposter */ createJoint(e, t, r) { const n = new ta(t, r); return this.addJoint(e, n), this; } /** * Add a joint to this impostor with a different impostor * @param otherImpostor A physics imposter used to add a joint * @param joint The joint to add * @returns The physics imposter */ addJoint(e, t) { return this._joints.push({ otherImpostor: e, joint: t }), this._physicsEngine && this._physicsEngine.addJoint(this, e, t), this; } /** * Add an anchor to a cloth impostor * @param otherImpostor rigid impostor to anchor to * @param width ratio across width from 0 to 1 * @param height ratio up height from 0 to 1 * @param influence the elasticity between cloth impostor and anchor from 0, very stretchy to 1, little stretch * @param noCollisionBetweenLinkedBodies when true collisions between cloth impostor and anchor are ignored; default false * @returns impostor the soft imposter */ addAnchor(e, t, r, n, i) { if (!this._physicsEngine) return this; const s = this._physicsEngine.getPhysicsPlugin(); return s.appendAnchor ? (this._physicsEngine && s.appendAnchor(this, e, t, r, n, i), this) : this; } /** * Add a hook to a rope impostor * @param otherImpostor rigid impostor to anchor to * @param length ratio across rope from 0 to 1 * @param influence the elasticity between rope impostor and anchor from 0, very stretchy to 1, little stretch * @param noCollisionBetweenLinkedBodies when true collisions between soft impostor and anchor are ignored; default false * @returns impostor the rope imposter */ addHook(e, t, r, n) { if (!this._physicsEngine) return this; const i = this._physicsEngine.getPhysicsPlugin(); return i.appendAnchor ? (this._physicsEngine && i.appendHook(this, e, t, r, n), this) : this; } /** * Will keep this body still, in a sleep mode. * @returns the physics imposter */ sleep() { return this._physicsEngine && this._physicsEngine.getPhysicsPlugin().sleepBody(this), this; } /** * Wake the body up. * @returns The physics imposter */ wakeUp() { return this._physicsEngine && this._physicsEngine.getPhysicsPlugin().wakeUpBody(this), this; } /** * Clones the physics imposter * @param newObject The physics imposter clones to this physics-enabled object * @returns A nullable physics imposter */ clone(e) { return e ? new tn(e, this.type, this._options, this._scene) : null; } /** * Disposes the physics imposter */ dispose() { this._physicsEngine && (this._joints.forEach((e) => { this._physicsEngine && this._physicsEngine.removeJoint(this, e.otherImpostor, e.joint); }), this._physicsEngine.removeImpostor(this), this.parent && this.parent.forceUpdate(), this._isDisposed = !0); } /** * Sets the delta position * @param position The delta position amount */ setDeltaPosition(e) { this._deltaPosition.copyFrom(e); } /** * Sets the delta rotation * @param rotation The delta rotation amount */ setDeltaRotation(e) { this._deltaRotation || (this._deltaRotation = new Ze()), this._deltaRotation.copyFrom(e), this._deltaRotationConjugated = this._deltaRotation.conjugate(); } /** * Gets the box size of the physics imposter and stores the result in the input parameter * @param result Stores the box size * @returns The physics imposter */ getBoxSizeToRef(e) { return this._physicsEngine && this._physicsEngine.getPhysicsPlugin().getBoxSizeToRef(this, e), this; } /** * Gets the radius of the physics imposter * @returns Radius of the physics imposter */ getRadius() { return this._physicsEngine ? this._physicsEngine.getPhysicsPlugin().getRadius(this) : 0; } /** * Sync a bone with this impostor * @param bone The bone to sync to the impostor. * @param boneMesh The mesh that the bone is influencing. * @param jointPivot The pivot of the joint / bone in local space. * @param distToJoint Optional distance from the impostor to the joint. * @param adjustRotation Optional quaternion for adjusting the local rotation of the bone. */ syncBoneWithImpostor(e, t, r, n, i) { const s = tn._TmpVecs[0], a = this.object; if (a.rotationQuaternion) if (i) { const f = tn._TmpQuat; a.rotationQuaternion.multiplyToRef(i, f), e.setRotationQuaternion(f, ai.WORLD, t); } else e.setRotationQuaternion(a.rotationQuaternion, ai.WORLD, t); s.x = 0, s.y = 0, s.z = 0, r && (s.x = r.x, s.y = r.y, s.z = r.z, e.getDirectionToRef(s, t, s), n == null && (n = r.length()), s.x *= n, s.y *= n, s.z *= n), e.getParent() ? (s.addInPlace(a.getAbsolutePosition()), e.setAbsolutePosition(s, t)) : (t.setAbsolutePosition(a.getAbsolutePosition()), t.position.x -= s.x, t.position.y -= s.y, t.position.z -= s.z); } /** * Sync impostor to a bone * @param bone The bone that the impostor will be synced to. * @param boneMesh The mesh that the bone is influencing. * @param jointPivot The pivot of the joint / bone in local space. * @param distToJoint Optional distance from the impostor to the joint. * @param adjustRotation Optional quaternion for adjusting the local rotation of the bone. * @param boneAxis Optional vector3 axis the bone is aligned with */ syncImpostorWithBone(e, t, r, n, i, s) { const a = this.object; if (a.rotationQuaternion) if (i) { const d = tn._TmpQuat; e.getRotationQuaternionToRef(ai.WORLD, t, d), d.multiplyToRef(i, a.rotationQuaternion); } else e.getRotationQuaternionToRef(ai.WORLD, t, a.rotationQuaternion); const f = tn._TmpVecs[0], o = tn._TmpVecs[1]; s || (s = tn._TmpVecs[2], s.x = 0, s.y = 1, s.z = 0), e.getDirectionToRef(s, t, o), e.getAbsolutePositionToRef(t, f), n == null && r && (n = r.length()), n != null && (f.x += o.x * n, f.y += o.y * n, f.z += o.z * n), a.setAbsolutePosition(f); } } tn.DEFAULT_OBJECT_SIZE = new S(1, 1, 1); tn.IDENTITY_QUATERNION = Ze.Identity(); tn._TmpVecs = Nf.BuildArray(3, S.Zero); tn._TmpQuat = Ze.Identity(); tn.NoImpostor = 0; tn.SphereImpostor = 1; tn.BoxImpostor = 2; tn.PlaneImpostor = 3; tn.MeshImpostor = 4; tn.CapsuleImpostor = 6; tn.CylinderImpostor = 7; tn.ParticleImpostor = 8; tn.HeightmapImpostor = 9; tn.ConvexHullImpostor = 10; tn.CustomImpostor = 100; tn.RopeImpostor = 101; tn.ClothImpostor = 102; tn.SoftbodyImpostor = 103; var O2; (function(A) { A[A.Clean = 0] = "Clean", A[A.Stop = 1] = "Stop", A[A.Sync = 2] = "Sync", A[A.NoSync = 3] = "NoSync"; })(O2 || (O2 = {})); class Hn { /** * Gets or sets a boolean indicating if entire scene must be loaded even if scene contains incremental data */ static get ForceFullSceneLoadingForIncremental() { return l9.ForceFullSceneLoadingForIncremental; } static set ForceFullSceneLoadingForIncremental(e) { l9.ForceFullSceneLoadingForIncremental = e; } /** * Gets or sets a boolean indicating if loading screen must be displayed while loading a scene */ static get ShowLoadingScreen() { return l9.ShowLoadingScreen; } static set ShowLoadingScreen(e) { l9.ShowLoadingScreen = e; } /** * Defines the current logging level (while loading the scene) * @ignorenaming */ // eslint-disable-next-line @typescript-eslint/naming-convention static get loggingLevel() { return l9.loggingLevel; } // eslint-disable-next-line @typescript-eslint/naming-convention static set loggingLevel(e) { l9.loggingLevel = e; } /** * Gets or set a boolean indicating if matrix weights must be cleaned upon loading */ static get CleanBoneMatrixWeights() { return l9.CleanBoneMatrixWeights; } static set CleanBoneMatrixWeights(e) { l9.CleanBoneMatrixWeights = e; } /** * Gets the default plugin (used to load Babylon files) * @returns the .babylon plugin */ static GetDefaultPlugin() { return Hn._RegisteredPlugins[".babylon"]; } static _GetPluginForExtension(e) { const t = Hn._RegisteredPlugins[e]; return t || (Se.Warn("Unable to find a plugin to load " + e + " files. Trying to use .babylon default plugin. To load from a specific filetype (eg. gltf) see: https://doc.babylonjs.com/features/featuresDeepDive/importers/loadingFileTypes"), Hn.GetDefaultPlugin()); } static _GetPluginForDirectLoad(e) { for (const t in Hn._RegisteredPlugins) { const r = Hn._RegisteredPlugins[t].plugin; if (r.canDirectLoad && r.canDirectLoad(e)) return Hn._RegisteredPlugins[t]; } return Hn.GetDefaultPlugin(); } static _GetPluginForFilename(e) { const t = e.indexOf("?"); t !== -1 && (e = e.substring(0, t)); const r = e.lastIndexOf("."), n = e.substring(r, e.length).toLowerCase(); return Hn._GetPluginForExtension(n); } static _GetDirectLoad(e) { return e.substr(0, 5) === "data:" ? e.substr(5) : null; } static _FormatErrorMessage(e, t, r) { let i = "Unable to load from " + (e.rawData ? "binary data" : e.url); return t ? i += `: ${t}` : r && (i += `: ${r}`), i; } static _LoadData(e, t, r, n, i, s, a, f) { const o = Hn._GetDirectLoad(e.url); if (e.rawData && !a) throw "When using ArrayBufferView to load data the file extension must be provided."; const d = a ? Hn._GetPluginForExtension(a) : o ? Hn._GetPluginForDirectLoad(e.url) : Hn._GetPluginForFilename(e.url); if (e.rawData && !d.isBinary) throw "Loading from ArrayBufferView can not be used with plugins that don't support binary loading."; let v; if (d.plugin.createPlugin !== void 0 ? v = d.plugin.createPlugin() : v = d.plugin, !v) throw "The loader plugin corresponding to the file type you are trying to load has not been found. If using es6, please import the plugin you wish to use before."; if (Hn.OnPluginActivatedObservable.notifyObservers(v), o && (v.canDirectLoad && v.canDirectLoad(e.url) || !qR(e.url))) { if (v.directLoad) { const b = v.directLoad(t, o); b.then ? b.then((j) => { r(v, j); }).catch((j) => { i("Error in directLoad of _loadData: " + j, j); }) : r(v, b); } else r(v, o); return v; } const u = d.isBinary, l = (b, j) => { if (t.isDisposed) { i("Scene has been disposed"); return; } r(v, b, j); }; let P = null, p = !1; const c = v.onDisposeObservable; c && c.add(() => { p = !0, P && (P.abort(), P = null), s(); }); const H = () => { if (p) return; const b = (j, w) => { i(j == null ? void 0 : j.statusText, w); }; if (!v.loadFile && e.rawData) throw "Plugin does not support loading ArrayBufferView."; P = v.loadFile ? v.loadFile(t, e.rawData || e.file || e.url, e.rootUrl, l, n, u, b, f) : t._loadFile(e.file || e.url, l, n, !0, u, b); }, T = t.getEngine(); let q = T.enableOfflineSupport; if (q) { let b = !1; for (const j of t.disableOfflineSupportExceptionRules) if (j.test(e.url)) { b = !0; break; } q = !b; } return q && Ge.OfflineProviderFactory ? t.offlineProvider = Ge.OfflineProviderFactory(e.url, H, T.disableManifestCheck) : H(), v; } static _GetFileInfo(e, t) { let r, n, i = null, s = null; if (!t) r = e, n = ye.GetFilename(e), e = ye.GetFolderPath(e); else if (t.name) { const a = t; r = `file:${a.name}`, n = a.name, i = a; } else if (ArrayBuffer.isView(t)) r = "", n = "arrayBuffer", s = t; else if (typeof t == "string" && t.startsWith("data:")) r = t, n = ""; else { const a = t; if (a.substr(0, 1) === "/") return ye.Error("Wrong sceneFilename parameter"), null; r = e + a, n = a; } return { url: r, rootUrl: e, name: n, file: i, rawData: s }; } // Public functions /** * Gets a plugin that can load the given extension * @param extension defines the extension to load * @returns a plugin or null if none works */ static GetPluginForExtension(e) { return Hn._GetPluginForExtension(e).plugin; } /** * Gets a boolean indicating that the given extension can be loaded * @param extension defines the extension to load * @returns true if the extension is supported */ static IsPluginForExtensionAvailable(e) { return !!Hn._RegisteredPlugins[e]; } /** * Adds a new plugin to the list of registered plugins * @param plugin defines the plugin to add */ static RegisterPlugin(e) { if (typeof e.extensions == "string") { const t = e.extensions; Hn._RegisteredPlugins[t.toLowerCase()] = { plugin: e, isBinary: !1 }; } else { const t = e.extensions; Object.keys(t).forEach((r) => { Hn._RegisteredPlugins[r.toLowerCase()] = { plugin: e, isBinary: t[r].isBinary }; }); } } /** * Import meshes into a scene * @param meshNames an array of mesh names, a single mesh name, or empty string for all meshes that filter what meshes are imported * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb) * @param sceneFilename a string that defines the name of the scene file or starts with "data:" following by the stringified version of the scene or a File object (default: empty string) * @param scene the instance of BABYLON.Scene to append to * @param onSuccess a callback with a list of imported meshes, particleSystems, skeletons, and animationGroups when import succeeds * @param onProgress a callback with a progress event for each file being loaded * @param onError a callback with the scene, a message, and possibly an exception when import fails * @param pluginExtension the extension used to determine the plugin * @returns The loaded plugin */ static ImportMesh(e, t, r = "", n = gr.LastCreatedScene, i = null, s = null, a = null, f = null, o = "") { if (!n) return Se.Error("No scene available to import mesh to"), null; const d = Hn._GetFileInfo(t, r); if (!d) return null; const v = {}; n.addPendingData(v); const u = () => { n.removePendingData(v); }, l = (c, H) => { const T = Hn._FormatErrorMessage(d, c, H); a ? a(n, T, new O0(T, Z2.SceneLoaderError, H)) : Se.Error(T), u(); }, P = s ? (c) => { try { s(c); } catch (H) { l("Error in onProgress callback: " + H, H); } } : void 0, p = (c, H, T, q, b, j, w) => { if (n.importedMeshesFiles.push(d.url), i) try { i(c, H, T, q, b, j, w); } catch (m) { l("Error in onSuccess callback: " + m, m); } n.removePendingData(v); }; return Hn._LoadData(d, n, (c, H, T) => { if (c.rewriteRootURL && (d.rootUrl = c.rewriteRootURL(d.rootUrl, T)), c.importMesh) { const q = c, b = [], j = [], w = []; if (!q.importMesh(e, n, H, d.rootUrl, b, j, w, l)) return; n.loadingPluginName = c.name, p(b, j, w, [], [], [], []); } else c.importMeshAsync(e, n, H, d.rootUrl, P, d.name).then((b) => { n.loadingPluginName = c.name, p(b.meshes, b.particleSystems, b.skeletons, b.animationGroups, b.transformNodes, b.geometries, b.lights); }).catch((b) => { l(b.message, b); }); }, P, l, u, f, o); } /** * Import meshes into a scene * @param meshNames an array of mesh names, a single mesh name, or empty string for all meshes that filter what meshes are imported * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb) * @param sceneFilename a string that defines the name of the scene file or starts with "data:" following by the stringified version of the scene or a File object (default: empty string) * @param scene the instance of BABYLON.Scene to append to * @param onProgress a callback with a progress event for each file being loaded * @param pluginExtension the extension used to determine the plugin * @returns The loaded list of imported meshes, particle systems, skeletons, and animation groups */ static ImportMeshAsync(e, t, r = "", n = gr.LastCreatedScene, i = null, s = null, a = "") { return new Promise((f, o) => { Hn.ImportMesh(e, t, r, n, (d, v, u, l, P, p, c) => { f({ meshes: d, particleSystems: v, skeletons: u, animationGroups: l, transformNodes: P, geometries: p, lights: c }); }, i, (d, v, u) => { o(u || new Error(v)); }, s, a); }); } /** * Load a scene * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb) * @param sceneFilename a string that defines the name of the scene file or starts with "data:" following by the stringified version of the scene or a File object (default: empty string) * @param engine is the instance of BABYLON.Engine to use to create the scene * @param onSuccess a callback with the scene when import succeeds * @param onProgress a callback with a progress event for each file being loaded * @param onError a callback with the scene, a message, and possibly an exception when import fails * @param pluginExtension the extension used to determine the plugin * @returns The loaded plugin */ static Load(e, t = "", r = gr.LastCreatedEngine, n = null, i = null, s = null, a = null, f = "") { return r ? Hn.Append(e, t, new sr(r), n, i, s, a, f) : (ye.Error("No engine available"), null); } /** * Load a scene * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb) * @param sceneFilename a string that defines the name of the scene file or starts with "data:" following by the stringified version of the scene or a File object (default: empty string) * @param engine is the instance of BABYLON.Engine to use to create the scene * @param onProgress a callback with a progress event for each file being loaded * @param pluginExtension the extension used to determine the plugin * @returns The loaded scene */ static LoadAsync(e, t = "", r = gr.LastCreatedEngine, n = null, i = null, s = "") { return new Promise((a, f) => { Hn.Load(e, t, r, (o) => { a(o); }, n, (o, d, v) => { f(v || new Error(d)); }, i, s); }); } /** * Append a scene * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb) * @param sceneFilename a string that defines the name of the scene file or starts with "data:" following by the stringified version of the scene or a File object (default: empty string) * @param scene is the instance of BABYLON.Scene to append to * @param onSuccess a callback with the scene when import succeeds * @param onProgress a callback with a progress event for each file being loaded * @param onError a callback with the scene, a message, and possibly an exception when import fails * @param pluginExtension the extension used to determine the plugin * @returns The loaded plugin */ static Append(e, t = "", r = gr.LastCreatedScene, n = null, i = null, s = null, a = null, f = "") { if (!r) return Se.Error("No scene available to append to"), null; const o = Hn._GetFileInfo(e, t); if (!o) return null; const d = {}; r.addPendingData(d); const v = () => { r.removePendingData(d); }; Hn.ShowLoadingScreen && !this._ShowingLoadingScreen && (this._ShowingLoadingScreen = !0, r.getEngine().displayLoadingUI(), r.executeWhenReady(() => { r.getEngine().hideLoadingUI(), this._ShowingLoadingScreen = !1; })); const u = (p, c) => { const H = Hn._FormatErrorMessage(o, p, c); s ? s(r, H, new O0(H, Z2.SceneLoaderError, c)) : Se.Error(H), v(); }, l = i ? (p) => { try { i(p); } catch (c) { u("Error in onProgress callback", c); } } : void 0, P = () => { if (n) try { n(r); } catch (p) { u("Error in onSuccess callback", p); } r.removePendingData(d); }; return Hn._LoadData(o, r, (p, c) => { if (p.load) { if (!p.load(r, c, o.rootUrl, u)) return; r.loadingPluginName = p.name, P(); } else p.loadAsync(r, c, o.rootUrl, l, o.name).then(() => { r.loadingPluginName = p.name, P(); }).catch((T) => { u(T.message, T); }); }, l, u, v, a, f); } /** * Append a scene * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb) * @param sceneFilename a string that defines the name of the scene file or starts with "data:" following by the stringified version of the scene or a File object (default: empty string) * @param scene is the instance of BABYLON.Scene to append to * @param onProgress a callback with a progress event for each file being loaded * @param pluginExtension the extension used to determine the plugin * @returns The given scene */ static AppendAsync(e, t = "", r = gr.LastCreatedScene, n = null, i = null, s = "") { return new Promise((a, f) => { Hn.Append(e, t, r, (o) => { a(o); }, n, (o, d, v) => { f(v || new Error(d)); }, i, s); }); } /** * Load a scene into an asset container * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb) * @param sceneFilename a string that defines the name of the scene file or starts with "data:" following by the stringified version of the scene or a File object (default: empty string) * @param scene is the instance of BABYLON.Scene to append to (default: last created scene) * @param onSuccess a callback with the scene when import succeeds * @param onProgress a callback with a progress event for each file being loaded * @param onError a callback with the scene, a message, and possibly an exception when import fails * @param pluginExtension the extension used to determine the plugin * @returns The loaded plugin */ static LoadAssetContainer(e, t = "", r = gr.LastCreatedScene, n = null, i = null, s = null, a = null, f = "") { if (!r) return Se.Error("No scene available to load asset container to"), null; const o = Hn._GetFileInfo(e, t); if (!o) return null; const d = {}; r.addPendingData(d); const v = () => { r.removePendingData(d); }, u = (p, c) => { const H = Hn._FormatErrorMessage(o, p, c); s ? s(r, H, new O0(H, Z2.SceneLoaderError, c)) : Se.Error(H), v(); }, l = i ? (p) => { try { i(p); } catch (c) { u("Error in onProgress callback", c); } } : void 0, P = (p) => { if (n) try { n(p); } catch (c) { u("Error in onSuccess callback", c); } r.removePendingData(d); }; return Hn._LoadData(o, r, (p, c) => { if (p.loadAssetContainer) { const T = p.loadAssetContainer(r, c, o.rootUrl, u); if (!T) return; T.populateRootNodes(), r.loadingPluginName = p.name, P(T); } else p.loadAssetContainerAsync ? p.loadAssetContainerAsync(r, c, o.rootUrl, l, o.name).then((T) => { T.populateRootNodes(), r.loadingPluginName = p.name, P(T); }).catch((T) => { u(T.message, T); }) : u("LoadAssetContainer is not supported by this plugin. Plugin did not provide a loadAssetContainer or loadAssetContainerAsync method."); }, l, u, v, a, f); } /** * Load a scene into an asset container * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb) * @param sceneFilename a string that defines the name of the scene file or starts with "data:" following by the stringified version of the scene (default: empty string) * @param scene is the instance of Scene to append to * @param onProgress a callback with a progress event for each file being loaded * @param pluginExtension the extension used to determine the plugin * @returns The loaded asset container */ static LoadAssetContainerAsync(e, t = "", r = gr.LastCreatedScene, n = null, i = null) { return new Promise((s, a) => { Hn.LoadAssetContainer(e, t, r, (f) => { s(f); }, n, (f, o, d) => { a(d || new Error(o)); }, i); }); } /** * Import animations from a file into a scene * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb) * @param sceneFilename a string that defines the name of the scene file or starts with "data:" following by the stringified version of the scene or a File object (default: empty string) * @param scene is the instance of BABYLON.Scene to append to (default: last created scene) * @param overwriteAnimations when true, animations are cleaned before importing new ones. Animations are appended otherwise * @param animationGroupLoadingMode defines how to handle old animations groups before importing new ones * @param targetConverter defines a function used to convert animation targets from loaded scene to current scene (default: search node by name) * @param onSuccess a callback with the scene when import succeeds * @param onProgress a callback with a progress event for each file being loaded * @param onError a callback with the scene, a message, and possibly an exception when import fails * @param pluginExtension the extension used to determine the plugin */ static ImportAnimations(e, t = "", r = gr.LastCreatedScene, n = !0, i = O2.Clean, s = null, a = null, f = null, o = null, d = null) { if (!r) { Se.Error("No scene available to load animations to"); return; } if (n) { for (const P of r.animatables) P.reset(); r.stopAllAnimations(), r.animationGroups.slice().forEach((P) => { P.dispose(); }), r.getNodes().forEach((P) => { P.animations && (P.animations = []); }); } else switch (i) { case O2.Clean: r.animationGroups.slice().forEach((l) => { l.dispose(); }); break; case O2.Stop: r.animationGroups.forEach((l) => { l.stop(); }); break; case O2.Sync: r.animationGroups.forEach((l) => { l.reset(), l.restart(); }); break; case O2.NoSync: break; default: Se.Error("Unknown animation group loading mode value '" + i + "'"); return; } const v = r.animatables.length, u = (l) => { l.mergeAnimationsTo(r, r.animatables.slice(v), s), l.dispose(), r.onAnimationFileImportedObservable.notifyObservers(r), a && a(r); }; this.LoadAssetContainer(e, t, r, u, f, o, d); } /** * Import animations from a file into a scene * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb) * @param sceneFilename a string that defines the name of the scene file or starts with "data:" following by the stringified version of the scene or a File object (default: empty string) * @param scene is the instance of BABYLON.Scene to append to (default: last created scene) * @param overwriteAnimations when true, animations are cleaned before importing new ones. Animations are appended otherwise * @param animationGroupLoadingMode defines how to handle old animations groups before importing new ones * @param targetConverter defines a function used to convert animation targets from loaded scene to current scene (default: search node by name) * @param onSuccess a callback with the scene when import succeeds * @param onProgress a callback with a progress event for each file being loaded * @param onError a callback with the scene, a message, and possibly an exception when import fails * @param pluginExtension the extension used to determine the plugin * @returns the updated scene with imported animations */ static ImportAnimationsAsync(e, t = "", r = gr.LastCreatedScene, n = !0, i = O2.Clean, s = null, a = null, f = null, o = null, d = null) { return new Promise((v, u) => { Hn.ImportAnimations(e, t, r, n, i, s, (l) => { v(l); }, f, (l, P, p) => { u(p || new Error(P)); }, d); }); } } Hn.NO_LOGGING = 0; Hn.MINIMAL_LOGGING = 1; Hn.SUMMARY_LOGGING = 2; Hn.DETAILED_LOGGING = 3; Hn.OnPluginActivatedObservable = new Oe(); Hn._RegisteredPlugins = {}; Hn._ShowingLoadingScreen = !1; class P1 extends gt { constructor(e, t, r = !0) { super(e, t), this._normalMatrix = new he(), this._storeEffectOnSubMeshes = r; } getEffect() { return this._storeEffectOnSubMeshes ? this._activeEffect : super.getEffect(); } isReady(e, t) { return e ? !this._storeEffectOnSubMeshes || !e.subMeshes || e.subMeshes.length === 0 ? !0 : this.isReadyForSubMesh(e, e.subMeshes[0], t) : !1; } _isReadyForSubMesh(e) { const t = e.materialDefines; return !!(!this.checkReadyOnEveryCall && e.effect && t && t._renderId === this.getScene().getRenderId()); } /** * Binds the given world matrix to the active effect * * @param world the matrix to bind */ bindOnlyWorldMatrix(e) { this._activeEffect.setMatrix("world", e); } /** * Binds the given normal matrix to the active effect * * @param normalMatrix the matrix to bind */ bindOnlyNormalMatrix(e) { this._activeEffect.setMatrix("normalMatrix", e); } bind(e, t) { t && this.bindForSubMesh(e, t, t.subMeshes[0]); } _afterBind(e, t = null) { super._afterBind(e, t), this.getScene()._cachedEffect = t, t && (t._forceRebindOnNextCall = !1); } _mustRebind(e, t, r = 1) { return e.isCachedMaterialInvalid(this, t, r); } dispose(e, t, r) { this._activeEffect = void 0, super.dispose(e, t, r); } } var de; (function(A) { A[A.Float = 1] = "Float", A[A.Int = 2] = "Int", A[A.Vector2 = 4] = "Vector2", A[A.Vector3 = 8] = "Vector3", A[A.Vector4 = 16] = "Vector4", A[A.Color3 = 32] = "Color3", A[A.Color4 = 64] = "Color4", A[A.Matrix = 128] = "Matrix", A[A.Object = 256] = "Object", A[A.AutoDetect = 1024] = "AutoDetect", A[A.BasedOnInput = 2048] = "BasedOnInput", A[A.All = 4095] = "All"; })(de || (de = {})); var Ve; (function(A) { A[A.Vertex = 1] = "Vertex", A[A.Fragment = 2] = "Fragment", A[A.Neutral = 4] = "Neutral", A[A.VertexAndFragment = 3] = "VertexAndFragment"; })(Ve || (Ve = {})); class lG { constructor() { this.supportUniformBuffers = !1, this.attributes = [], this.uniforms = [], this.constants = [], this.samplers = [], this.functions = {}, this.extensions = {}, this.prePassOutput = {}, this.counters = {}, this._attributeDeclaration = "", this._uniformDeclaration = "", this._constantDeclaration = "", this._samplerDeclaration = "", this._varyingTransfer = "", this._injectAtEnd = "", this._repeatableContentAnchorIndex = 0, this._builtCompilationString = "", this.compilationString = ""; } /** * Finalize the compilation strings * @param state defines the current compilation state */ finalize(e) { const t = e.sharedData.emitComments, r = this.target === Ve.Fragment; this.compilationString = ` ${t ? `//Entry point ` : ""}void main(void) { ${this.compilationString}`, this._constantDeclaration && (this.compilationString = ` ${t ? `//Constants ` : ""}${this._constantDeclaration} ${this.compilationString}`); let n = ""; for (const i in this.functions) n += this.functions[i] + ` `; this.compilationString = ` ${n} ${this.compilationString}`, !r && this._varyingTransfer && (this.compilationString = `${this.compilationString} ${this._varyingTransfer}`), this._injectAtEnd && (this.compilationString = `${this.compilationString} ${this._injectAtEnd}`), this.compilationString = `${this.compilationString} }`, this.sharedData.varyingDeclaration && (this.compilationString = ` ${t ? `//Varyings ` : ""}${this.sharedData.varyingDeclaration} ${this.compilationString}`), this._samplerDeclaration && (this.compilationString = ` ${t ? `//Samplers ` : ""}${this._samplerDeclaration} ${this.compilationString}`), this._uniformDeclaration && (this.compilationString = ` ${t ? `//Uniforms ` : ""}${this._uniformDeclaration} ${this.compilationString}`), this._attributeDeclaration && !r && (this.compilationString = ` ${t ? `//Attributes ` : ""}${this._attributeDeclaration} ${this.compilationString}`), this.compilationString = `precision highp float; ` + this.compilationString, this.compilationString = `#if defined(WEBGL2) || defines(WEBGPU) precision highp sampler2DArray; #endif ` + this.compilationString, r && (this.compilationString = `#if defined(PREPASS)\r #extension GL_EXT_draw_buffers : require\r layout(location = 0) out highp vec4 glFragData[SCENE_MRT_COUNT];\r highp vec4 gl_FragColor;\r #endif\r ` + this.compilationString); for (const i in this.extensions) { const s = this.extensions[i]; this.compilationString = ` ${s} ${this.compilationString}`; } this._builtCompilationString = this.compilationString; } /** @internal */ get _repeatableContentAnchor() { return `###___ANCHOR${this._repeatableContentAnchorIndex++}___###`; } /** * @internal */ _getFreeVariableName(e) { return e = e.replace(/[^a-zA-Z_]+/g, ""), this.sharedData.variableNames[e] === void 0 ? (this.sharedData.variableNames[e] = 0, e === "output" || e === "texture" ? e + this.sharedData.variableNames[e] : e) : (this.sharedData.variableNames[e]++, e + this.sharedData.variableNames[e]); } /** * @internal */ _getFreeDefineName(e) { return this.sharedData.defineNames[e] === void 0 ? this.sharedData.defineNames[e] = 0 : this.sharedData.defineNames[e]++, e + this.sharedData.defineNames[e]; } /** * @internal */ _excludeVariableName(e) { this.sharedData.variableNames[e] = 0; } /** * @internal */ _emit2DSampler(e) { this.samplers.indexOf(e) < 0 && (this._samplerDeclaration += `uniform sampler2D ${e}; `, this.samplers.push(e)); } /** * @internal */ _emit2DArraySampler(e) { this.samplers.indexOf(e) < 0 && (this._samplerDeclaration += `uniform sampler2DArray ${e}; `, this.samplers.push(e)); } /** * @internal */ _getGLType(e) { switch (e) { case de.Float: return "float"; case de.Int: return "int"; case de.Vector2: return "vec2"; case de.Color3: case de.Vector3: return "vec3"; case de.Color4: case de.Vector4: return "vec4"; case de.Matrix: return "mat4"; } return ""; } /** * @internal */ _emitExtension(e, t, r = "") { this.extensions[e] || (r && (t = `#if ${r} ${t} #endif`), this.extensions[e] = t); } /** * @internal */ _emitFunction(e, t, r) { this.functions[e] || (this.sharedData.emitComments && (t = r + ` ` + t), this.functions[e] = t); } /** * @internal */ _emitCodeFromInclude(e, t, r) { if (r && r.repeatKey) return `#include<${e}>${r.substitutionVars ? "(" + r.substitutionVars + ")" : ""}[0..${r.repeatKey}] `; let n = An.IncludesShadersStore[e] + ` `; if (this.sharedData.emitComments && (n = t + ` ` + n), !r) return n; if (r.replaceStrings) for (let i = 0; i < r.replaceStrings.length; i++) { const s = r.replaceStrings[i]; n = n.replace(s.search, s.replace); } return n; } /** * @internal */ _emitFunctionFromInclude(e, t, r, n = "") { const i = e + n; if (!this.functions[i]) { if (!r || !r.removeAttributes && !r.removeUniforms && !r.removeVaryings && !r.removeIfDef && !r.replaceStrings) { r && r.repeatKey ? this.functions[i] = `#include<${e}>${r.substitutionVars ? "(" + r.substitutionVars + ")" : ""}[0..${r.repeatKey}] ` : this.functions[i] = `#include<${e}>${r != null && r.substitutionVars ? "(" + (r == null ? void 0 : r.substitutionVars) + ")" : ""} `, this.sharedData.emitComments && (this.functions[i] = t + ` ` + this.functions[i]); return; } if (this.functions[i] = An.IncludesShadersStore[e], this.sharedData.emitComments && (this.functions[i] = t + ` ` + this.functions[i]), r.removeIfDef && (this.functions[i] = this.functions[i].replace(/^\s*?#ifdef.+$/gm, ""), this.functions[i] = this.functions[i].replace(/^\s*?#endif.*$/gm, ""), this.functions[i] = this.functions[i].replace(/^\s*?#else.*$/gm, ""), this.functions[i] = this.functions[i].replace(/^\s*?#elif.*$/gm, "")), r.removeAttributes && (this.functions[i] = this.functions[i].replace(/\s*?attribute .+?;/g, ` `)), r.removeUniforms && (this.functions[i] = this.functions[i].replace(/\s*?uniform .*?;/g, ` `)), r.removeVaryings && (this.functions[i] = this.functions[i].replace(/\s*?(varying|in) .+?;/g, ` `)), r.replaceStrings) for (let s = 0; s < r.replaceStrings.length; s++) { const a = r.replaceStrings[s]; this.functions[i] = this.functions[i].replace(a.search, a.replace); } } } /** * @internal */ _registerTempVariable(e) { return this.sharedData.temps.indexOf(e) !== -1 ? !1 : (this.sharedData.temps.push(e), !0); } /** * @internal */ _emitVaryingFromString(e, t, r = "", n = !1) { return this.sharedData.varyings.indexOf(e) !== -1 ? !1 : (this.sharedData.varyings.push(e), r && (r.startsWith("defined(") ? this.sharedData.varyingDeclaration += `#if ${r} ` : this.sharedData.varyingDeclaration += `${n ? "#ifndef" : "#ifdef"} ${r} `), this.sharedData.varyingDeclaration += `varying ${t} ${e}; `, r && (this.sharedData.varyingDeclaration += `#endif `), !0); } /** * @internal */ _emitUniformFromString(e, t, r = "", n = !1) { this.uniforms.indexOf(e) === -1 && (this.uniforms.push(e), r && (r.startsWith("defined(") ? this._uniformDeclaration += `#if ${r} ` : this._uniformDeclaration += `${n ? "#ifndef" : "#ifdef"} ${r} `), this._uniformDeclaration += `uniform ${t} ${e}; `, r && (this._uniformDeclaration += `#endif `)); } /** * @internal */ _emitFloat(e) { return e.toString() === e.toFixed(0) ? `${e}.0` : e.toString(); } } class W6e { /** Creates a new shared data */ constructor() { this.temps = [], this.varyings = [], this.varyingDeclaration = "", this.inputBlocks = [], this.textureBlocks = [], this.bindableBlocks = [], this.forcedBindableBlocks = [], this.blocksWithFallbacks = [], this.blocksWithDefines = [], this.repeatableContentBlocks = [], this.dynamicUniformBlocks = [], this.blockingBlocks = [], this.animatedInputs = [], this.variableNames = {}, this.defineNames = {}, this.hints = { needWorldViewMatrix: !1, needWorldViewProjectionMatrix: !1, needAlphaBlending: !1, needAlphaTesting: !1 }, this.checks = { emitVertex: !1, emitFragment: !1, notConnectedNonOptionalInputs: new Array() }, this.allowEmptyVertexProgram = !1, this.variableNames.position = 0, this.variableNames.normal = 0, this.variableNames.tangent = 0, this.variableNames.uv = 0, this.variableNames.uv2 = 0, this.variableNames.uv3 = 0, this.variableNames.uv4 = 0, this.variableNames.uv5 = 0, this.variableNames.uv6 = 0, this.variableNames.color = 0, this.variableNames.matricesIndices = 0, this.variableNames.matricesWeights = 0, this.variableNames.matricesIndicesExtra = 0, this.variableNames.matricesWeightsExtra = 0, this.variableNames.diffuseBase = 0, this.variableNames.specularBase = 0, this.variableNames.worldPos = 0, this.variableNames.shadow = 0, this.variableNames.view = 0, this.variableNames.vTBN = 0, this.defineNames.MAINUV0 = 0, this.defineNames.MAINUV1 = 0, this.defineNames.MAINUV2 = 0, this.defineNames.MAINUV3 = 0, this.defineNames.MAINUV4 = 0, this.defineNames.MAINUV5 = 0, this.defineNames.MAINUV6 = 0, this.defineNames.MAINUV7 = 0; } /** * Emits console errors and exceptions if there is a failing check */ emitErrors() { let e = ""; !this.checks.emitVertex && !this.allowEmptyVertexProgram && (e += `NodeMaterial does not have a vertex output. You need to at least add a block that generates a glPosition value. `), this.checks.emitFragment || (e += `NodeMaterial does not have a fragment output. You need to at least add a block that generates a glFragColor value. `); for (const t of this.checks.notConnectedNonOptionalInputs) e += `input ${t.name} from block ${t.ownerBlock.name}[${t.ownerBlock.getClassName()}] is not connected and is not optional. `; if (e) throw `Build of NodeMaterial failed: ` + e; } } var pc; (function(A) { A[A.Compatible = 0] = "Compatible", A[A.TypeIncompatible = 1] = "TypeIncompatible", A[A.TargetIncompatible = 2] = "TargetIncompatible", A[A.HierarchyIssue = 3] = "HierarchyIssue"; })(pc || (pc = {})); var ao; (function(A) { A[A.Input = 0] = "Input", A[A.Output = 1] = "Output"; })(ao || (ao = {})); class um { /** * Checks if two types are equivalent * @param type1 type 1 to check * @param type2 type 2 to check * @returns true if both types are equivalent, else false */ static AreEquivalentTypes(e, t) { switch (e) { case de.Vector3: { if (t === de.Color3) return !0; break; } case de.Vector4: { if (t === de.Color4) return !0; break; } case de.Color3: { if (t === de.Vector3) return !0; break; } case de.Color4: { if (t === de.Vector4) return !0; break; } } return !1; } /** Gets the direction of the point */ get direction() { return this._direction; } /** * Gets or sets the associated variable name in the shader */ get associatedVariableName() { return this._ownerBlock.isInput ? this._ownerBlock.associatedVariableName : (!this._enforceAssociatedVariableName || !this._associatedVariableName) && this._connectedPoint ? this._connectedPoint.associatedVariableName : this._associatedVariableName; } set associatedVariableName(e) { this._associatedVariableName = e; } /** Get the inner type (ie AutoDetect for instance instead of the inferred one) */ get innerType() { return this._linkedConnectionSource && this._linkedConnectionSource.isConnected ? this.type : this._type; } /** * Gets or sets the connection point type (default is float) */ get type() { if (this._type === de.AutoDetect) { if (this._ownerBlock.isInput) return this._ownerBlock.type; if (this._connectedPoint) return this._connectedPoint.type; if (this._linkedConnectionSource && this._linkedConnectionSource.isConnected) return this._linkedConnectionSource.type; } if (this._type === de.BasedOnInput) { if (this._typeConnectionSource) return !this._typeConnectionSource.isConnected && this._defaultConnectionPointType ? this._defaultConnectionPointType : this._typeConnectionSource.type; if (this._defaultConnectionPointType) return this._defaultConnectionPointType; } return this._type; } set type(e) { this._type = e; } /** Gets or sets the target of that connection point */ get target() { return !this._prioritizeVertex || !this._ownerBlock ? this._target : this._target !== Ve.VertexAndFragment ? this._target : this._ownerBlock.target === Ve.Fragment ? Ve.Fragment : Ve.Vertex; } set target(e) { this._target = e; } /** * Gets a boolean indicating that the current point is connected to another NodeMaterialBlock */ get isConnected() { return this.connectedPoint !== null || this.hasEndpoints; } /** * Gets a boolean indicating that the current point is connected to an input block */ get isConnectedToInputBlock() { return this.connectedPoint !== null && this.connectedPoint.ownerBlock.isInput; } /** * Gets a the connected input block (if any) */ get connectInputBlock() { return this.isConnectedToInputBlock ? this.connectedPoint.ownerBlock : null; } /** Get the other side of the connection (if any) */ get connectedPoint() { return this._connectedPoint; } /** Get the block that owns this connection point */ get ownerBlock() { return this._ownerBlock; } /** Get the block connected on the other side of this connection (if any) */ get sourceBlock() { return this._connectedPoint ? this._connectedPoint.ownerBlock : null; } /** Get the block connected on the endpoints of this connection (if any) */ get connectedBlocks() { return this._endpoints.length === 0 ? [] : this._endpoints.map((e) => e.ownerBlock); } /** Gets the list of connected endpoints */ get endpoints() { return this._endpoints; } /** Gets a boolean indicating if that output point is connected to at least one input */ get hasEndpoints() { return this._endpoints && this._endpoints.length > 0; } /** Gets a boolean indicating that this connection has a path to the vertex output*/ get isDirectlyConnectedToVertexOutput() { if (!this.hasEndpoints) return !1; for (const e of this._endpoints) if (e.ownerBlock.target === Ve.Vertex || (e.ownerBlock.target === Ve.Neutral || e.ownerBlock.target === Ve.VertexAndFragment) && e.ownerBlock.outputs.some((t) => t.isDirectlyConnectedToVertexOutput)) return !0; return !1; } /** Gets a boolean indicating that this connection will be used in the vertex shader */ get isConnectedInVertexShader() { if (this.target === Ve.Vertex) return !0; if (!this.hasEndpoints) return !1; for (const e of this._endpoints) if (e.ownerBlock.target === Ve.Vertex || e.target === Ve.Vertex || (e.ownerBlock.target === Ve.Neutral || e.ownerBlock.target === Ve.VertexAndFragment) && e.ownerBlock.outputs.some((t) => t.isConnectedInVertexShader)) return !0; return !1; } /** Gets a boolean indicating that this connection will be used in the fragment shader */ get isConnectedInFragmentShader() { if (this.target === Ve.Fragment) return !0; if (!this.hasEndpoints) return !1; for (const e of this._endpoints) if (e.ownerBlock.target === Ve.Fragment || (e.ownerBlock.target === Ve.Neutral || e.ownerBlock.target === Ve.VertexAndFragment) && e.ownerBlock.isConnectedInFragmentShader()) return !0; return !1; } /** * Creates a block suitable to be used as an input for this input point. * If null is returned, a block based on the point type will be created. * @returns The returned string parameter is the name of the output point of NodeMaterialBlock (first parameter of the returned array) that can be connected to the input */ createCustomInputBlock() { return null; } /** * Creates a new connection point * @param name defines the connection point name * @param ownerBlock defines the block hosting this connection point * @param direction defines the direction of the connection point */ constructor(e, t, r) { this._connectedPoint = null, this._endpoints = new Array(), this._typeConnectionSource = null, this._defaultConnectionPointType = null, this._linkedConnectionSource = null, this._acceptedConnectionPointType = null, this._type = de.Float, this._enforceAssociatedVariableName = !1, this.needDualDirectionValidation = !1, this.acceptedConnectionPointTypes = [], this.excludedConnectionPointTypes = [], this.onConnectionObservable = new Oe(), this.isExposedOnFrame = !1, this.exposedPortPosition = -1, this._prioritizeVertex = !1, this._target = Ve.VertexAndFragment, this._ownerBlock = t, this.name = e, this._direction = r; } /** * Gets the current class name e.g. "NodeMaterialConnectionPoint" * @returns the class name */ getClassName() { return "NodeMaterialConnectionPoint"; } /** * Gets a boolean indicating if the current point can be connected to another point * @param connectionPoint defines the other connection point * @returns a boolean */ canConnectTo(e) { return this.checkCompatibilityState(e) === pc.Compatible; } /** * Gets a number indicating if the current point can be connected to another point * @param connectionPoint defines the other connection point * @returns a number defining the compatibility state */ checkCompatibilityState(e) { const t = this._ownerBlock, r = e.ownerBlock; if (t.target === Ve.Fragment) { if (r.target === Ve.Vertex) return pc.TargetIncompatible; for (const s of r.outputs) if (s.ownerBlock.target != Ve.Neutral && s.isConnectedInVertexShader) return pc.TargetIncompatible; } if (this.type !== e.type && e.innerType !== de.AutoDetect) return um.AreEquivalentTypes(this.type, e.type) || e.acceptedConnectionPointTypes && e.acceptedConnectionPointTypes.indexOf(this.type) !== -1 || e._acceptedConnectionPointType && um.AreEquivalentTypes(e._acceptedConnectionPointType.type, this.type) ? pc.Compatible : pc.TypeIncompatible; if (e.excludedConnectionPointTypes && e.excludedConnectionPointTypes.indexOf(this.type) !== -1) return pc.TypeIncompatible; let n = r, i = t; return this.direction === ao.Input && (n = t, i = r), n.isAnAncestorOf(i) ? pc.HierarchyIssue : pc.Compatible; } /** * Connect this point to another connection point * @param connectionPoint defines the other connection point * @param ignoreConstraints defines if the system will ignore connection type constraints (default is false) * @returns the current connection point */ connectTo(e, t = !1) { if (!t && !this.canConnectTo(e)) throw "Cannot connect these two connectors."; return this._endpoints.push(e), e._connectedPoint = this, this._enforceAssociatedVariableName = !1, this.onConnectionObservable.notifyObservers(e), e.onConnectionObservable.notifyObservers(this), this; } /** * Disconnect this point from one of his endpoint * @param endpoint defines the other connection point * @returns the current connection point */ disconnectFrom(e) { const t = this._endpoints.indexOf(e); return t === -1 ? this : (this._endpoints.splice(t, 1), e._connectedPoint = null, this._enforceAssociatedVariableName = !1, e._enforceAssociatedVariableName = !1, this); } /** * Fill the list of excluded connection point types with all types other than those passed in the parameter * @param mask Types (ORed values of NodeMaterialBlockConnectionPointTypes) that are allowed, and thus will not be pushed to the excluded list */ addExcludedConnectionPointFromAllowedTypes(e) { let t = 1; for (; t < de.All; ) e & t || this.excludedConnectionPointTypes.push(t), t = t << 1; } /** * Serializes this point in a JSON representation * @param isInput defines if the connection point is an input (default is true) * @returns the serialized point object */ serialize(e = !0) { const t = {}; return t.name = this.name, t.displayName = this.displayName, e && this.connectedPoint && (t.inputName = this.name, t.targetBlockId = this.connectedPoint.ownerBlock.uniqueId, t.targetConnectionName = this.connectedPoint.name, t.isExposedOnFrame = !0, t.exposedPortPosition = this.exposedPortPosition), (this.isExposedOnFrame || this.exposedPortPosition >= 0) && (t.isExposedOnFrame = !0, t.exposedPortPosition = this.exposedPortPosition), t; } /** * Release resources */ dispose() { this.onConnectionObservable.clear(); } } class Mr { /** * Gets the name of the block */ get name() { return this._name; } /** * Sets the name of the block. Will check if the name is valid. */ set name(e) { this.validateBlockName(e) && (this._name = e); } /** * Gets a boolean indicating that this block can only be used once per NodeMaterial */ get isUnique() { return this._isUnique; } /** * Gets a boolean indicating that this block is an end block (e.g. it is generating a system value) */ get isFinalMerger() { return this._isFinalMerger; } /** * Gets a boolean indicating that this block is an input (e.g. it sends data to the shader) */ get isInput() { return this._isInput; } /** * Gets a boolean indicating if this block is a teleport out */ get isTeleportOut() { return this._isTeleportOut; } /** * Gets a boolean indicating if this block is a teleport in */ get isTeleportIn() { return this._isTeleportIn; } /** * Gets or sets the build Id */ get buildId() { return this._buildId; } set buildId(e) { this._buildId = e; } /** * Gets or sets the target of the block */ get target() { return this._target; } set target(e) { this._target & e || (this._target = e); } /** * Gets the list of input points */ get inputs() { return this._inputs; } /** Gets the list of output points */ get outputs() { return this._outputs; } /** * Find an input by its name * @param name defines the name of the input to look for * @returns the input or null if not found */ getInputByName(e) { const t = this._inputs.filter((r) => r.name === e); return t.length ? t[0] : null; } /** * Find an output by its name * @param name defines the name of the output to look for * @returns the output or null if not found */ getOutputByName(e) { const t = this._outputs.filter((r) => r.name === e); return t.length ? t[0] : null; } /** * Creates a new NodeMaterialBlock * @param name defines the block name * @param target defines the target of that block (Vertex by default) * @param isFinalMerger defines a boolean indicating that this block is an end block (e.g. it is generating a system value). Default is false */ constructor(e, t = Ve.Vertex, r = !1) { this._isFinalMerger = !1, this._isInput = !1, this._isTeleportOut = !1, this._isTeleportIn = !1, this._name = "", this._isUnique = !1, this.inputsAreExclusive = !1, this._codeVariableName = "", this._inputs = new Array(), this._outputs = new Array(), this.comments = "", this.visibleInInspector = !1, this.visibleOnFrame = !1, this._target = t, this._originalTargetIsNeutral = t === Ve.Neutral, this._isFinalMerger = r, this._isInput = this.getClassName() === "InputBlock", this._isTeleportOut = this.getClassName() === "NodeMaterialTeleportOutBlock", this._isTeleportIn = this.getClassName() === "NodeMaterialTeleportInBlock", this._name = e, this.uniqueId = bR.UniqueId; } /** @internal */ _setInitialTarget(e) { this._target = e, this._originalTargetIsNeutral = e === Ve.Neutral; } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ // eslint-disable-next-line @typescript-eslint/no-unused-vars initialize(e) { } /** * Bind data to effect. Will only be called for blocks with isBindable === true * @param effect defines the effect to bind data to * @param nodeMaterial defines the hosting NodeMaterial * @param mesh defines the mesh that will be rendered * @param subMesh defines the submesh that will be rendered */ // eslint-disable-next-line @typescript-eslint/no-unused-vars bind(e, t, r, n) { } _declareOutput(e, t) { return `${t._getGLType(e.type)} ${e.associatedVariableName}`; } _writeVariable(e) { return e.connectedPoint ? `${e.associatedVariableName}` : "0."; } _writeFloat(e) { let t = e.toString(); return t.indexOf(".") === -1 && (t += ".0"), `${t}`; } /** * Gets the current class name e.g. "NodeMaterialBlock" * @returns the class name */ getClassName() { return "NodeMaterialBlock"; } /** Gets a boolean indicating that this connection will be used in the fragment shader */ isConnectedInFragmentShader() { return this.outputs.some((e) => e.isConnectedInFragmentShader); } /** * Register a new input. Must be called inside a block constructor * @param name defines the connection point name * @param type defines the connection point type * @param isOptional defines a boolean indicating that this input can be omitted * @param target defines the target to use to limit the connection point (will be VertexAndFragment by default) * @param point an already created connection point. If not provided, create a new one * @returns the current block */ registerInput(e, t, r = !1, n, i) { return i = i ?? new um(e, this, ao.Input), i.type = t, i.isOptional = r, n && (i.target = n), this._inputs.push(i), this; } /** * Register a new output. Must be called inside a block constructor * @param name defines the connection point name * @param type defines the connection point type * @param target defines the target to use to limit the connection point (will be VertexAndFragment by default) * @param point an already created connection point. If not provided, create a new one * @returns the current block */ registerOutput(e, t, r, n) { return n = n ?? new um(e, this, ao.Output), n.type = t, r && (n.target = r), this._outputs.push(n), this; } /** * Will return the first available input e.g. the first one which is not an uniform or an attribute * @param forOutput defines an optional connection point to check compatibility with * @returns the first available input or null */ getFirstAvailableInput(e = null) { for (const t of this._inputs) if (!t.connectedPoint && (!e || e.type === t.type || t.type === de.AutoDetect)) return t; return null; } /** * Will return the first available output e.g. the first one which is not yet connected and not a varying * @param forBlock defines an optional block to check compatibility with * @returns the first available input or null */ getFirstAvailableOutput(e = null) { for (const t of this._outputs) if (!e || !e.target || e.target === Ve.Neutral || e.target & t.target) return t; return null; } /** * Gets the sibling of the given output * @param current defines the current output * @returns the next output in the list or null */ getSiblingOutput(e) { const t = this._outputs.indexOf(e); return t === -1 || t >= this._outputs.length ? null : this._outputs[t + 1]; } /** * Checks if the current block is an ancestor of a given block * @param block defines the potential descendant block to check * @returns true if block is a descendant */ isAnAncestorOf(e) { for (const t of this._outputs) if (t.hasEndpoints) { for (const r of t.endpoints) if (r.ownerBlock === e || r.ownerBlock.isAnAncestorOf(e)) return !0; } return !1; } /** * Connect current block with another block * @param other defines the block to connect with * @param options define the various options to help pick the right connections * @param options.input * @param options.output * @param options.outputSwizzle * @returns the current block */ connectTo(e, t) { if (this._outputs.length === 0) return; let r = t && t.output ? this.getOutputByName(t.output) : this.getFirstAvailableOutput(e), n = !0; for (; n; ) { const i = t && t.input ? e.getInputByName(t.input) : e.getFirstAvailableInput(r); if (r && i && r.canConnectTo(i)) r.connectTo(i), n = !1; else if (r) r = this.getSiblingOutput(r); else throw "Unable to find a compatible match"; } return this; } // eslint-disable-next-line @typescript-eslint/no-unused-vars _buildBlock(e) { } /** * Add uniforms, samplers and uniform buffers at compilation time * @param state defines the state to update * @param nodeMaterial defines the node material requesting the update * @param defines defines the material defines to update * @param uniformBuffers defines the list of uniform buffer names */ // eslint-disable-next-line @typescript-eslint/no-unused-vars updateUniformsAndSamples(e, t, r, n) { } /** * Add potential fallbacks if shader compilation fails * @param mesh defines the mesh to be rendered * @param fallbacks defines the current prioritized list of fallbacks */ // eslint-disable-next-line @typescript-eslint/no-unused-vars provideFallbacks(e, t) { } /** * Initialize defines for shader compilation * @param mesh defines the mesh to be rendered * @param nodeMaterial defines the node material requesting the update * @param defines defines the material defines to update * @param useInstances specifies that instances should be used */ // eslint-disable-next-line @typescript-eslint/no-unused-vars initializeDefines(e, t, r, n = !1) { } /** * Update defines for shader compilation * @param mesh defines the mesh to be rendered * @param nodeMaterial defines the node material requesting the update * @param defines defines the material defines to update * @param useInstances specifies that instances should be used * @param subMesh defines which submesh to render */ // eslint-disable-next-line @typescript-eslint/no-unused-vars prepareDefines(e, t, r, n = !1, i) { } /** * Lets the block try to connect some inputs automatically * @param material defines the hosting NodeMaterial * @param additionalFilteringInfo optional additional filtering condition when looking for compatible blocks */ // eslint-disable-next-line @typescript-eslint/no-unused-vars autoConfigure(e, t = () => !0) { } /** * Function called when a block is declared as repeatable content generator * @param vertexShaderState defines the current compilation state for the vertex shader * @param fragmentShaderState defines the current compilation state for the fragment shader * @param mesh defines the mesh to be rendered * @param defines defines the material defines to update */ // eslint-disable-next-line @typescript-eslint/no-unused-vars replaceRepeatableContent(e, t, r, n) { } /** Gets a boolean indicating that the code of this block will be promoted to vertex shader even if connected to fragment output */ get willBeGeneratedIntoVertexShaderFromFragmentShader() { return this.isInput || this.isFinalMerger || this._outputs.some((e) => e.isDirectlyConnectedToVertexOutput) || this.target === Ve.Vertex ? !1 : !!((this.target === Ve.VertexAndFragment || this.target === Ve.Neutral) && this._outputs.some((e) => e.isConnectedInVertexShader)); } /** * Checks if the block is ready * @param mesh defines the mesh to be rendered * @param nodeMaterial defines the node material requesting the update * @param defines defines the material defines to update * @param useInstances specifies that instances should be used * @returns true if the block is ready */ // eslint-disable-next-line @typescript-eslint/no-unused-vars isReady(e, t, r, n = !1) { return !0; } _linkConnectionTypes(e, t, r = !1) { r ? this._inputs[t]._acceptedConnectionPointType = this._inputs[e] : this._inputs[e]._linkedConnectionSource = this._inputs[t], this._inputs[t]._linkedConnectionSource = this._inputs[e]; } _processBuild(e, t, r, n) { e.build(t, n); const i = t._vertexState != null, s = e._buildTarget === Ve.Vertex && e.target !== Ve.VertexAndFragment; if (i && (!(e.target & e._buildTarget) || !(e.target & r.target) || this.target !== Ve.VertexAndFragment && s) && (!e.isInput && t.target !== e._buildTarget || // block was already emitted by vertex shader e.isInput && e.isAttribute && !e._noContextSwitch)) { const a = r.connectedPoint; t._vertexState._emitVaryingFromString("v_" + a.associatedVariableName, t._getGLType(a.type)) && (t._vertexState.compilationString += `${"v_" + a.associatedVariableName} = ${a.associatedVariableName}; `), r.associatedVariableName = "v_" + a.associatedVariableName, r._enforceAssociatedVariableName = !0; } } /** * Validates the new name for the block node. * @param newName the new name to be given to the node. * @returns false if the name is a reserve word, else true. */ validateBlockName(e) { const t = [ "position", "normal", "tangent", "particle_positionw", "uv", "uv2", "uv3", "uv4", "uv5", "uv6", "position2d", "particle_uv", "matricesIndices", "matricesWeights", "world0", "world1", "world2", "world3", "particle_color", "particle_texturemask" ]; for (const r of t) if (e === r) return !1; return !0; } // eslint-disable-next-line @typescript-eslint/no-unused-vars _customBuildStep(e, t) { } /** * Compile the current node and generate the shader code * @param state defines the current compilation state (uniforms, samplers, current string) * @param activeBlocks defines the list of active blocks (i.e. blocks to compile) * @returns true if already built */ build(e, t) { if (this._buildId === e.sharedData.buildId) return !0; if (!this.isInput) for (const r of this._outputs) r.associatedVariableName || (r.associatedVariableName = e._getFreeVariableName(r.name)); for (const r of this._inputs) { if (!r.connectedPoint) { r.isOptional || e.sharedData.checks.notConnectedNonOptionalInputs.push(r); continue; } if (this.target !== Ve.Neutral && (!(r.target & this.target) || !(r.target & e.target))) continue; const n = r.connectedPoint.ownerBlock; n && n !== this && this._processBuild(n, e, r, t); } if (this._customBuildStep(e, t), this._buildId === e.sharedData.buildId) return !0; if (e.sharedData.verbose && console.log(`${e.target === Ve.Vertex ? "Vertex shader" : "Fragment shader"}: Building ${this.name} [${this.getClassName()}]`), this.isFinalMerger) switch (e.target) { case Ve.Vertex: e.sharedData.checks.emitVertex = !0; break; case Ve.Fragment: e.sharedData.checks.emitFragment = !0; break; } !this.isInput && e.sharedData.emitComments && (e.compilationString += ` //${this.name} `), this._buildBlock(e), this._buildId = e.sharedData.buildId, this._buildTarget = e.target; for (const r of this._outputs) if (r.target & e.target) for (const n of r.endpoints) { const i = n.ownerBlock; i && i.target & e.target && t.indexOf(i) !== -1 && this._processBuild(i, e, n, t); } return !1; } _inputRename(e) { return e; } _outputRename(e) { return e; } _dumpPropertiesCode() { const e = this._codeVariableName; return `${e}.visibleInInspector = ${this.visibleInInspector}; ${e}.visibleOnFrame = ${this.visibleOnFrame}; ${e}.target = ${this.target}; `; } /** * @internal */ _dumpCode(e, t) { t.push(this); const r = this.name.replace(/[^A-Za-z_]+/g, ""); if (this._codeVariableName = r || `${this.getClassName()}_${this.uniqueId}`, e.indexOf(this._codeVariableName) !== -1) { let i = 0; do i++, this._codeVariableName = r + i; while (e.indexOf(this._codeVariableName) !== -1); } e.push(this._codeVariableName); let n = ` // ${this.getClassName()} `; this.comments && (n += `// ${this.comments} `), n += `var ${this._codeVariableName} = new BABYLON.${this.getClassName()}("${this.name}"); `, n += this._dumpPropertiesCode(); for (const i of this.inputs) { if (!i.isConnected) continue; const a = i.connectedPoint.ownerBlock; t.indexOf(a) === -1 && (n += a._dumpCode(e, t)); } for (const i of this.outputs) if (i.hasEndpoints) for (const s of i.endpoints) { const a = s.ownerBlock; a && t.indexOf(a) === -1 && (n += a._dumpCode(e, t)); } return n; } /** * @internal */ _dumpCodeForOutputConnections(e) { let t = ""; if (e.indexOf(this) !== -1) return t; e.push(this); for (const r of this.inputs) { if (!r.isConnected) continue; const n = r.connectedPoint, i = n.ownerBlock; t += i._dumpCodeForOutputConnections(e), t += `${i._codeVariableName}.${i._outputRename(n.name)}.connectTo(${this._codeVariableName}.${this._inputRename(r.name)}); `; } return t; } /** * Clone the current block to a new identical block * @param scene defines the hosting scene * @param rootUrl defines the root URL to use to load textures and relative dependencies * @returns a copy of the current block */ clone(e, t = "") { const r = this.serialize(), n = Jo(r.customType); if (n) { const i = new n(); return i._deserialize(r, e, t), i; } return null; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = {}; e.customType = "BABYLON." + this.getClassName(), e.id = this.uniqueId, e.name = this.name, e.comments = this.comments, e.visibleInInspector = this.visibleInInspector, e.visibleOnFrame = this.visibleOnFrame, e.target = this.target, e.inputs = [], e.outputs = []; for (const t of this.inputs) e.inputs.push(t.serialize()); for (const t of this.outputs) e.outputs.push(t.serialize(!1)); return e; } /** * @internal */ // eslint-disable-next-line @typescript-eslint/no-unused-vars _deserialize(e, t, r) { var n; this.name = e.name, this.comments = e.comments, this.visibleInInspector = !!e.visibleInInspector, this.visibleOnFrame = !!e.visibleOnFrame, this._target = (n = e.target) !== null && n !== void 0 ? n : this.target, this._deserializePortDisplayNamesAndExposedOnFrame(e); } _deserializePortDisplayNamesAndExposedOnFrame(e) { const t = e.inputs, r = e.outputs; t && t.forEach((n, i) => { n.displayName && (this.inputs[i].displayName = n.displayName), n.isExposedOnFrame && (this.inputs[i].isExposedOnFrame = n.isExposedOnFrame, this.inputs[i].exposedPortPosition = n.exposedPortPosition); }), r && r.forEach((n, i) => { n.displayName && (this.outputs[i].displayName = n.displayName), n.isExposedOnFrame && (this.outputs[i].isExposedOnFrame = n.isExposedOnFrame, this.outputs[i].exposedPortPosition = n.exposedPortPosition); }); } /** * Release resources */ dispose() { for (const e of this.inputs) e.dispose(); for (const e of this.outputs) e.dispose(); } } class JC extends Mr { /** * Creates a new TransformBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.complementW = 1, this.complementZ = 0, this.target = Ve.Vertex, this.registerInput("vector", de.AutoDetect), this.registerInput("transform", de.Matrix), this.registerOutput("output", de.Vector4), this.registerOutput("xyz", de.Vector3), this._inputs[0].onConnectionObservable.add((t) => { if (t.ownerBlock.isInput) { const r = t.ownerBlock; (r.name === "normal" || r.name === "tangent") && (this.complementW = 0); } }); } /** * Gets the current class name * @returns the class name */ getClassName() { return "TransformBlock"; } /** * Gets the vector input */ get vector() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } /** * Gets the xyz output component */ get xyz() { return this._outputs[1]; } /** * Gets the matrix transform input */ get transform() { return this._inputs[1]; } _buildBlock(e) { super._buildBlock(e); const t = this.vector, r = this.transform; if (t.connectedPoint) { if (this.complementW === 0) { const n = `//${this.name}`; e._emitFunctionFromInclude("helperFunctions", n), e.sharedData.blocksWithDefines.push(this); const i = e._getFreeVariableName(`${r.associatedVariableName}_NUS`); switch (e.compilationString += `mat3 ${i} = mat3(${r.associatedVariableName}); `, e.compilationString += `#ifdef NONUNIFORMSCALING `, e.compilationString += `${i} = transposeMat3(inverseMat3(${i})); `, e.compilationString += `#endif `, t.connectedPoint.type) { case de.Vector2: e.compilationString += this._declareOutput(this.output, e) + ` = vec4(${i} * vec3(${t.associatedVariableName}, ${this._writeFloat(this.complementZ)}), ${this._writeFloat(this.complementW)}); `; break; case de.Vector3: case de.Color3: e.compilationString += this._declareOutput(this.output, e) + ` = vec4(${i} * ${t.associatedVariableName}, ${this._writeFloat(this.complementW)}); `; break; default: e.compilationString += this._declareOutput(this.output, e) + ` = vec4(${i} * ${t.associatedVariableName}.xyz, ${this._writeFloat(this.complementW)}); `; break; } } else { const n = r.associatedVariableName; switch (t.connectedPoint.type) { case de.Vector2: e.compilationString += this._declareOutput(this.output, e) + ` = ${n} * vec4(${t.associatedVariableName}, ${this._writeFloat(this.complementZ)}, ${this._writeFloat(this.complementW)}); `; break; case de.Vector3: case de.Color3: e.compilationString += this._declareOutput(this.output, e) + ` = ${n} * vec4(${t.associatedVariableName}, ${this._writeFloat(this.complementW)}); `; break; default: e.compilationString += this._declareOutput(this.output, e) + ` = ${n} * ${t.associatedVariableName}; `; break; } } this.xyz.hasEndpoints && (e.compilationString += this._declareOutput(this.xyz, e) + ` = ${this.output.associatedVariableName}.xyz; `); } return this; } /** * Update defines for shader compilation * @param mesh defines the mesh to be rendered * @param nodeMaterial defines the node material requesting the update * @param defines defines the material defines to update */ prepareDefines(e, t, r) { e.nonUniformScaling && r.setValue("NONUNIFORMSCALING", !0); } serialize() { const e = super.serialize(); return e.complementZ = this.complementZ, e.complementW = this.complementW, e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.complementZ = e.complementZ !== void 0 ? e.complementZ : 0, this.complementW = e.complementW !== void 0 ? e.complementW : 1; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode() + `${this._codeVariableName}.complementZ = ${this.complementZ}; `; return e += `${this._codeVariableName}.complementW = ${this.complementW}; `, e; } } Ue("BABYLON.TransformBlock", JC); class mI extends Mr { /** * Creates a new VertexOutputBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Vertex, !0), this.registerInput("vector", de.Vector4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "VertexOutputBlock"; } /** * Gets the vector input component */ get vector() { return this._inputs[0]; } _isLogarithmicDepthEnabled(e, t) { if (t) return !0; for (const r of e) if (r.useLogarithmicDepth) return !0; return !1; } _buildBlock(e) { super._buildBlock(e); const t = this.vector; return e.compilationString += `gl_Position = ${t.associatedVariableName}; `, this._isLogarithmicDepthEnabled(e.sharedData.fragmentOutputNodes, e.sharedData.nodeMaterial.useLogarithmicDepth) && (e._emitUniformFromString("logarithmicDepthConstant", "float"), e._emitVaryingFromString("vFragmentDepth", "float"), e.compilationString += `vFragmentDepth = 1.0 + gl_Position.w; `, e.compilationString += `gl_Position.z = log2(max(0.000001, vFragmentDepth)) * logarithmicDepthConstant; `), this; } } Ue("BABYLON.VertexOutputBlock", mI); var Gr; (function(A) { A[A.Boolean = 0] = "Boolean", A[A.Float = 1] = "Float", A[A.Int = 2] = "Int", A[A.Vector2 = 3] = "Vector2", A[A.List = 4] = "List"; })(Gr || (Gr = {})); function rn(A, e = Gr.Boolean, t = "PROPERTIES", r) { return (n, i) => { let s = n._propStore; s || (s = [], n._propStore = s), s.push({ propertyName: i, displayName: A, type: e, groupName: t, options: r ?? {} }); }; } class iq extends Mr { /** * Create a new FragmentOutputBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment, !0), this.convertToGammaSpace = !1, this.convertToLinearSpace = !1, this.useLogarithmicDepth = !1, this.registerInput("rgba", de.Color4, !0), this.registerInput("rgb", de.AutoDetect, !0), this.registerInput("a", de.Float, !0), this.rgb.addExcludedConnectionPointFromAllowedTypes(de.Color3 | de.Vector3 | de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "FragmentOutputBlock"; } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("logarithmicDepthConstant"), e._excludeVariableName("vFragmentDepth"); } /** * Gets the rgba input component */ get rgba() { return this._inputs[0]; } /** * Gets the rgb input component */ get rgb() { return this._inputs[1]; } /** * Gets the a input component */ get a() { return this._inputs[2]; } prepareDefines(e, t, r) { r.setValue(this._linearDefineName, this.convertToLinearSpace, !0), r.setValue(this._gammaDefineName, this.convertToGammaSpace, !0); } bind(e, t, r) { (this.useLogarithmicDepth || t.useLogarithmicDepth) && r && Ye.BindLogDepth(void 0, e, r.getScene()); } _buildBlock(e) { super._buildBlock(e); const t = this.rgba, r = this.rgb, n = this.a; e.sharedData.hints.needAlphaBlending = t.isConnected || n.isConnected, e.sharedData.blocksWithDefines.push(this), (this.useLogarithmicDepth || e.sharedData.nodeMaterial.useLogarithmicDepth) && (e._emitUniformFromString("logarithmicDepthConstant", "float"), e._emitVaryingFromString("vFragmentDepth", "float"), e.sharedData.bindableBlocks.push(this)), this._linearDefineName = e._getFreeDefineName("CONVERTTOLINEAR"), this._gammaDefineName = e._getFreeDefineName("CONVERTTOGAMMA"); const i = `//${this.name}`; if (e._emitFunctionFromInclude("helperFunctions", i), t.connectedPoint) n.isConnected ? e.compilationString += `gl_FragColor = vec4(${t.associatedVariableName}.rgb, ${n.associatedVariableName}); ` : e.compilationString += `gl_FragColor = ${t.associatedVariableName}; `; else if (r.connectedPoint) { let s = "1.0"; n.connectedPoint && (s = n.associatedVariableName), r.connectedPoint.type === de.Float ? e.compilationString += `gl_FragColor = vec4(${r.associatedVariableName}, ${r.associatedVariableName}, ${r.associatedVariableName}, ${s}); ` : e.compilationString += `gl_FragColor = vec4(${r.associatedVariableName}, ${s}); `; } else e.sharedData.checks.notConnectedNonOptionalInputs.push(t); return e.compilationString += `#ifdef ${this._linearDefineName} `, e.compilationString += `gl_FragColor = toLinearSpace(gl_FragColor); `, e.compilationString += `#endif `, e.compilationString += `#ifdef ${this._gammaDefineName} `, e.compilationString += `gl_FragColor = toGammaSpace(gl_FragColor); `, e.compilationString += `#endif `, (this.useLogarithmicDepth || e.sharedData.nodeMaterial.useLogarithmicDepth) && (e.compilationString += `gl_FragDepthEXT = log2(vFragmentDepth) * logarithmicDepthConstant * 0.5; `), e.compilationString += `#if defined(PREPASS)\r `, e.compilationString += `gl_FragData[0] = gl_FragColor;\r `, e.compilationString += `#endif\r `, this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return e += `${this._codeVariableName}.convertToGammaSpace = ${this.convertToGammaSpace}; `, e += `${this._codeVariableName}.convertToLinearSpace = ${this.convertToLinearSpace}; `, e += `${this._codeVariableName}.useLogarithmicDepth = ${this.useLogarithmicDepth}; `, e; } serialize() { const e = super.serialize(); return e.convertToGammaSpace = this.convertToGammaSpace, e.convertToLinearSpace = this.convertToLinearSpace, e.useLogarithmicDepth = this.useLogarithmicDepth, e; } _deserialize(e, t, r) { var n; super._deserialize(e, t, r), this.convertToGammaSpace = e.convertToGammaSpace, this.convertToLinearSpace = e.convertToLinearSpace, this.useLogarithmicDepth = (n = e.useLogarithmicDepth) !== null && n !== void 0 ? n : !1; } } C([ rn("Convert to gamma space", Gr.Boolean, "PROPERTIES", { notifiers: { update: !0 } }) ], iq.prototype, "convertToGammaSpace", void 0); C([ rn("Convert to linear space", Gr.Boolean, "PROPERTIES", { notifiers: { update: !0 } }) ], iq.prototype, "convertToLinearSpace", void 0); C([ rn("Use logarithmic depth", Gr.Boolean, "PROPERTIES") ], iq.prototype, "useLogarithmicDepth", void 0); Ue("BABYLON.FragmentOutputBlock", iq); var jA; (function(A) { A[A.Uniform = 0] = "Uniform", A[A.Attribute = 1] = "Attribute", A[A.Varying = 2] = "Varying", A[A.Undefined = 3] = "Undefined"; })(jA || (jA = {})); var Bi; (function(A) { A[A.World = 1] = "World", A[A.View = 2] = "View", A[A.Projection = 3] = "Projection", A[A.ViewProjection = 4] = "ViewProjection", A[A.WorldView = 5] = "WorldView", A[A.WorldViewProjection = 6] = "WorldViewProjection", A[A.CameraPosition = 7] = "CameraPosition", A[A.FogColor = 8] = "FogColor", A[A.DeltaTime = 9] = "DeltaTime", A[A.CameraParameters = 10] = "CameraParameters", A[A.MaterialAlpha = 11] = "MaterialAlpha"; })(Bi || (Bi = {})); class sQ { /** * Creates a PositionNormalVertex * @param position the position of the vertex (defaut: 0,0,0) * @param normal the normal of the vertex (defaut: 0,1,0) */ constructor(e = S.Zero(), t = S.Up()) { this.position = e, this.normal = t; } /** * Clones the PositionNormalVertex * @returns the cloned PositionNormalVertex */ clone() { return new sQ(this.position.clone(), this.normal.clone()); } } class aQ { /** * Creates a PositionNormalTextureVertex * @param position the position of the vertex (defaut: 0,0,0) * @param normal the normal of the vertex (defaut: 0,1,0) * @param uv the uv of the vertex (default: 0,0) */ constructor(e = S.Zero(), t = S.Up(), r = at.Zero()) { this.position = e, this.normal = t, this.uv = r; } /** * Clones the PositionNormalTextureVertex * @returns the cloned PositionNormalTextureVertex */ clone() { return new aQ(this.position.clone(), this.normal.clone(), this.uv.clone()); } } var tq; (function(A) { A[A.None = 0] = "None", A[A.Time = 1] = "Time", A[A.RealTime = 2] = "RealTime"; })(tq || (tq = {})); const S6e = { position2d: "position", particle_uv: "vUV", particle_color: "vColor", particle_texturemask: "textureMask", particle_positionw: "vPositionW" }, HE = { particle_uv: !0, particle_color: !0, particle_texturemask: !0, particle_positionw: !0 }, PG = { particle_texturemask: !0 }; class gi extends Mr { /** * Gets or sets the connection point type (default is float) */ get type() { if (this._type === de.AutoDetect) { if (this.isUniform && this.value != null) { if (!isNaN(this.value)) return this._type = de.Float, this._type; switch (this.value.getClassName()) { case "Vector2": return this._type = de.Vector2, this._type; case "Vector3": return this._type = de.Vector3, this._type; case "Vector4": return this._type = de.Vector4, this._type; case "Color3": return this._type = de.Color3, this._type; case "Color4": return this._type = de.Color4, this._type; case "Matrix": return this._type = de.Matrix, this._type; } } if (this.isAttribute) switch (this.name) { case "position": case "normal": case "particle_positionw": return this._type = de.Vector3, this._type; case "uv": case "uv2": case "uv3": case "uv4": case "uv5": case "uv6": case "position2d": case "particle_uv": return this._type = de.Vector2, this._type; case "matricesIndices": case "matricesWeights": case "matricesIndicesExtra": case "matricesWeightsExtra": case "world0": case "world1": case "world2": case "world3": case "tangent": return this._type = de.Vector4, this._type; case "color": case "instanceColor": case "particle_color": case "particle_texturemask": return this._type = de.Color4, this._type; } if (this.isSystemValue) switch (this._systemValue) { case Bi.World: case Bi.WorldView: case Bi.WorldViewProjection: case Bi.View: case Bi.ViewProjection: case Bi.Projection: return this._type = de.Matrix, this._type; case Bi.CameraPosition: return this._type = de.Vector3, this._type; case Bi.FogColor: return this._type = de.Color3, this._type; case Bi.DeltaTime: case Bi.MaterialAlpha: return this._type = de.Float, this._type; case Bi.CameraParameters: return this._type = de.Vector4, this._type; } } return this._type; } /** * Creates a new InputBlock * @param name defines the block name * @param target defines the target of that block (Vertex by default) * @param type defines the type of the input (can be set to NodeMaterialBlockConnectionPointTypes.AutoDetect) */ constructor(e, t = Ve.Vertex, r = de.AutoDetect) { super(e, t, !1), this._mode = jA.Undefined, this._animationType = tq.None, this.min = 0, this.max = 0, this.isBoolean = !1, this.matrixMode = 0, this._systemValue = null, this.isConstant = !1, this.groupInInspector = "", this.onValueChangedObservable = new Oe(), this.convertToGammaSpace = !1, this.convertToLinearSpace = !1, this._type = r, this.setDefaultValue(), this.registerOutput("output", r); } /** * Validates if a name is a reserve word. * @param newName the new name to be given to the node. * @returns false if the name is a reserve word, else true. */ validateBlockName(e) { return this.isAttribute ? !0 : super.validateBlockName(e); } /** * Gets the output component */ get output() { return this._outputs[0]; } /** * Set the source of this connection point to a vertex attribute * @param attributeName defines the attribute name (position, uv, normal, etc...). If not specified it will take the connection point name * @returns the current connection point */ setAsAttribute(e) { return this._mode = jA.Attribute, e && (this.name = e), this; } /** * Set the source of this connection point to a system value * @param value define the system value to use (world, view, etc...) or null to switch to manual value * @returns the current connection point */ setAsSystemValue(e) { return this.systemValue = e, this; } /** * Gets or sets the value of that point. * Please note that this value will be ignored if valueCallback is defined */ get value() { return this._storedValue; } set value(e) { this.type === de.Float && (this.isBoolean ? e = e ? 1 : 0 : this.min !== this.max && (e = Math.max(this.min, e), e = Math.min(this.max, e))), this._storedValue = e, this._mode = jA.Uniform, this.onValueChangedObservable.notifyObservers(this); } /** * Gets or sets a callback used to get the value of that point. * Please note that setting this value will force the connection point to ignore the value property */ get valueCallback() { return this._valueCallback; } set valueCallback(e) { this._valueCallback = e, this._mode = jA.Uniform; } /** * Gets or sets the associated variable name in the shader */ get associatedVariableName() { return this._associatedVariableName; } set associatedVariableName(e) { this._associatedVariableName = e; } /** Gets or sets the type of animation applied to the input */ get animationType() { return this._animationType; } set animationType(e) { this._animationType = e; } /** * Gets a boolean indicating that this connection point not defined yet */ get isUndefined() { return this._mode === jA.Undefined; } /** * Gets or sets a boolean indicating that this connection point is coming from an uniform. * In this case the connection point name must be the name of the uniform to use. * Can only be set on inputs */ get isUniform() { return this._mode === jA.Uniform; } set isUniform(e) { this._mode = e ? jA.Uniform : jA.Undefined, this.associatedVariableName = ""; } /** * Gets or sets a boolean indicating that this connection point is coming from an attribute. * In this case the connection point name must be the name of the attribute to use * Can only be set on inputs */ get isAttribute() { return this._mode === jA.Attribute; } set isAttribute(e) { this._mode = e ? jA.Attribute : jA.Undefined, this.associatedVariableName = ""; } /** * Gets or sets a boolean indicating that this connection point is generating a varying variable. * Can only be set on exit points */ get isVarying() { return this._mode === jA.Varying; } set isVarying(e) { this._mode = e ? jA.Varying : jA.Undefined, this.associatedVariableName = ""; } /** * Gets a boolean indicating that the current connection point is a system value */ get isSystemValue() { return this._systemValue != null; } /** * Gets or sets the current well known value or null if not defined as a system value */ get systemValue() { return this._systemValue; } set systemValue(e) { this._mode = jA.Uniform, this.associatedVariableName = "", this._systemValue = e; } /** * Gets the current class name * @returns the class name */ getClassName() { return "InputBlock"; } /** * Animate the input if animationType !== None * @param scene defines the rendering scene */ animate(e) { switch (this._animationType) { case tq.Time: { this.type === de.Float && (this.value += e.getAnimationRatio() * 0.01); break; } case tq.RealTime: { this.type === de.Float && (this.value = (Yi.Now - e.getEngine().startTime) / 1e3); break; } } } _emitDefine(e) { return e[0] === "!" ? `#ifndef ${e.substring(1)} ` : `#ifdef ${e} `; } initialize() { this.associatedVariableName = ""; } /** * Set the input block to its default value (based on its type) */ setDefaultValue() { switch (this.type) { case de.Float: this.value = 0; break; case de.Vector2: this.value = at.Zero(); break; case de.Vector3: this.value = S.Zero(); break; case de.Vector4: this.value = Ir.Zero(); break; case de.Color3: this.value = Ne.White(); break; case de.Color4: this.value = new xt(1, 1, 1, 1); break; case de.Matrix: this.value = he.Identity(); break; } } _emitConstant(e) { switch (this.type) { case de.Float: return `${e._emitFloat(this.value)}`; case de.Vector2: return `vec2(${this.value.x}, ${this.value.y})`; case de.Vector3: return `vec3(${this.value.x}, ${this.value.y}, ${this.value.z})`; case de.Vector4: return `vec4(${this.value.x}, ${this.value.y}, ${this.value.z}, ${this.value.w})`; case de.Color3: return Hs.Color3[0].set(this.value.r, this.value.g, this.value.b), this.convertToGammaSpace && Hs.Color3[0].toGammaSpaceToRef(Hs.Color3[0], e.sharedData.scene.getEngine().useExactSrgbConversions), this.convertToLinearSpace && Hs.Color3[0].toLinearSpaceToRef(Hs.Color3[0], e.sharedData.scene.getEngine().useExactSrgbConversions), `vec3(${Hs.Color3[0].r}, ${Hs.Color3[0].g}, ${Hs.Color3[0].b})`; case de.Color4: return Hs.Color4[0].set(this.value.r, this.value.g, this.value.b, this.value.a), this.convertToGammaSpace && Hs.Color4[0].toGammaSpaceToRef(Hs.Color4[0], e.sharedData.scene.getEngine().useExactSrgbConversions), this.convertToLinearSpace && Hs.Color4[0].toLinearSpaceToRef(Hs.Color4[0], e.sharedData.scene.getEngine().useExactSrgbConversions), `vec4(${Hs.Color4[0].r}, ${Hs.Color4[0].g}, ${Hs.Color4[0].b}, ${Hs.Color4[0].a})`; } return ""; } /** @internal */ get _noContextSwitch() { return HE[this.name]; } _emit(e, t) { var r; if (this.isUniform) { if (this.associatedVariableName || (this.associatedVariableName = e._getFreeVariableName("u_" + this.name)), this.isConstant) { if (e.constants.indexOf(this.associatedVariableName) !== -1) return; e.constants.push(this.associatedVariableName), e._constantDeclaration += this._declareOutput(this.output, e) + ` = ${this._emitConstant(e)}; `; return; } if (e.uniforms.indexOf(this.associatedVariableName) !== -1) return; e.uniforms.push(this.associatedVariableName), t && (e._uniformDeclaration += this._emitDefine(t)), e._uniformDeclaration += `uniform ${e._getGLType(this.type)} ${this.associatedVariableName}; `, t && (e._uniformDeclaration += `#endif `); const n = e.sharedData.hints; if (this._systemValue !== null && this._systemValue !== void 0) switch (this._systemValue) { case Bi.WorldView: n.needWorldViewMatrix = !0; break; case Bi.WorldViewProjection: n.needWorldViewProjectionMatrix = !0; break; } else this._animationType !== tq.None && e.sharedData.animatedInputs.push(this); return; } if (this.isAttribute) { if (this.associatedVariableName = (r = S6e[this.name]) !== null && r !== void 0 ? r : this.name, this.target === Ve.Vertex && e._vertexState) { HE[this.name] ? PG[this.name] ? e._emitUniformFromString(this.associatedVariableName, e._getGLType(this.type), t) : e._emitVaryingFromString(this.associatedVariableName, e._getGLType(this.type), t) : this._emit(e._vertexState, t); return; } if (e.attributes.indexOf(this.associatedVariableName) !== -1) return; e.attributes.push(this.associatedVariableName), HE[this.name] ? PG[this.name] ? e._emitUniformFromString(this.associatedVariableName, e._getGLType(this.type), t) : e._emitVaryingFromString(this.associatedVariableName, e._getGLType(this.type), t) : (t && (e._attributeDeclaration += this._emitDefine(t)), e._attributeDeclaration += `attribute ${e._getGLType(this.type)} ${this.associatedVariableName}; `, t && (e._attributeDeclaration += `#endif `)); } } /** * @internal */ _transmitWorld(e, t, r, n) { if (!this._systemValue) return; const i = this.associatedVariableName; switch (this._systemValue) { case Bi.World: e.setMatrix(i, t); break; case Bi.WorldView: e.setMatrix(i, r); break; case Bi.WorldViewProjection: e.setMatrix(i, n); break; } } /** * @internal */ _transmit(e, t, r) { if (this.isAttribute) return; const n = this.associatedVariableName; if (this._systemValue) { switch (this._systemValue) { case Bi.World: case Bi.WorldView: case Bi.WorldViewProjection: return; case Bi.View: e.setMatrix(n, t.getViewMatrix()); break; case Bi.Projection: e.setMatrix(n, t.getProjectionMatrix()); break; case Bi.ViewProjection: e.setMatrix(n, t.getTransformMatrix()); break; case Bi.CameraPosition: t.bindEyePosition(e, n, !0); break; case Bi.FogColor: e.setColor3(n, t.fogColor); break; case Bi.DeltaTime: e.setFloat(n, t.deltaTime / 1e3); break; case Bi.CameraParameters: t.activeCamera && e.setFloat4(n, t.getEngine().hasOriginBottomLeft ? -1 : 1, t.activeCamera.minZ, t.activeCamera.maxZ, 1 / t.activeCamera.maxZ); break; case Bi.MaterialAlpha: e.setFloat(n, r.alpha); break; } return; } const i = this._valueCallback ? this._valueCallback() : this._storedValue; if (i !== null) switch (this.type) { case de.Float: e.setFloat(n, i); break; case de.Int: e.setInt(n, i); break; case de.Color3: Hs.Color3[0].set(this.value.r, this.value.g, this.value.b), this.convertToGammaSpace && Hs.Color3[0].toGammaSpaceToRef(Hs.Color3[0], t.getEngine().useExactSrgbConversions), this.convertToLinearSpace && Hs.Color3[0].toLinearSpaceToRef(Hs.Color3[0], t.getEngine().useExactSrgbConversions), e.setColor3(n, Hs.Color3[0]); break; case de.Color4: Hs.Color4[0].set(this.value.r, this.value.g, this.value.b, this.value.a), this.convertToGammaSpace && Hs.Color4[0].toGammaSpaceToRef(Hs.Color4[0], t.getEngine().useExactSrgbConversions), this.convertToLinearSpace && Hs.Color4[0].toLinearSpaceToRef(Hs.Color4[0], t.getEngine().useExactSrgbConversions), e.setDirectColor4(n, Hs.Color4[0]); break; case de.Vector2: e.setVector2(n, i); break; case de.Vector3: e.setVector3(n, i); break; case de.Vector4: e.setVector4(n, i); break; case de.Matrix: e.setMatrix(n, i); break; } } _buildBlock(e) { super._buildBlock(e), (this.isUniform || this.isSystemValue) && e.sharedData.inputBlocks.push(this), this._emit(e); } _dumpPropertiesCode() { const e = this._codeVariableName; if (this.isAttribute) return super._dumpPropertiesCode() + `${e}.setAsAttribute("${this.name}"); `; if (this.isSystemValue) return super._dumpPropertiesCode() + `${e}.setAsSystemValue(BABYLON.NodeMaterialSystemValues.${Bi[this._systemValue]}); `; if (this.isUniform) { const t = []; let r = ""; switch (this.type) { case de.Float: r = `${this.value}`; break; case de.Vector2: r = `new BABYLON.Vector2(${this.value.x}, ${this.value.y})`; break; case de.Vector3: r = `new BABYLON.Vector3(${this.value.x}, ${this.value.y}, ${this.value.z})`; break; case de.Vector4: r = `new BABYLON.Vector4(${this.value.x}, ${this.value.y}, ${this.value.z}, ${this.value.w})`; break; case de.Color3: r = `new BABYLON.Color3(${this.value.r}, ${this.value.g}, ${this.value.b})`, this.convertToGammaSpace && (r += ".toGammaSpace()"), this.convertToLinearSpace && (r += ".toLinearSpace()"); break; case de.Color4: r = `new BABYLON.Color4(${this.value.r}, ${this.value.g}, ${this.value.b}, ${this.value.a})`, this.convertToGammaSpace && (r += ".toGammaSpace()"), this.convertToLinearSpace && (r += ".toLinearSpace()"); break; case de.Matrix: r = `BABYLON.Matrix.FromArray([${this.value.m}])`; break; } return t.push(`${e}.value = ${r}`), this.type === de.Float && t.push(`${e}.min = ${this.min}`, `${e}.max = ${this.max}`, `${e}.isBoolean = ${this.isBoolean}`, `${e}.matrixMode = ${this.matrixMode}`, `${e}.animationType = BABYLON.AnimatedInputBlockTypes.${tq[this.animationType]}`), t.push(`${e}.isConstant = ${this.isConstant}`), t.push(""), super._dumpPropertiesCode() + t.join(`; `); } return super._dumpPropertiesCode(); } dispose() { this.onValueChangedObservable.clear(), super.dispose(); } serialize() { const e = super.serialize(); return e.type = this.type, e.mode = this._mode, e.systemValue = this._systemValue, e.animationType = this._animationType, e.min = this.min, e.max = this.max, e.isBoolean = this.isBoolean, e.matrixMode = this.matrixMode, e.isConstant = this.isConstant, e.groupInInspector = this.groupInInspector, e.convertToGammaSpace = this.convertToGammaSpace, e.convertToLinearSpace = this.convertToLinearSpace, this._storedValue != null && this._mode === jA.Uniform && (this._storedValue.asArray ? (e.valueType = "BABYLON." + this._storedValue.getClassName(), e.value = this._storedValue.asArray()) : (e.valueType = "number", e.value = this._storedValue)), e; } _deserialize(e, t, r) { if (this._mode = e.mode, super._deserialize(e, t, r), this._type = e.type, this._systemValue = e.systemValue || e.wellKnownValue, this._animationType = e.animationType, this.min = e.min || 0, this.max = e.max || 0, this.isBoolean = !!e.isBoolean, this.matrixMode = e.matrixMode || 0, this.isConstant = !!e.isConstant, this.groupInInspector = e.groupInInspector || "", this.convertToGammaSpace = !!e.convertToGammaSpace, this.convertToLinearSpace = !!e.convertToLinearSpace, e.name === "tangent" && e.mode === jA.Attribute && e.type === de.Vector3 && (this._type = de.Vector4), !!e.valueType) if (e.valueType === "number") this._storedValue = e.value; else { const n = Jo(e.valueType); n && (this._storedValue = n.FromArray(e.value)); } } } Ue("BABYLON.InputBlock", gi); class oQ extends Mr { /** * Create a new CurrentScreenBlock * @param name defines the block name */ constructor(e) { super(e, Ve.VertexAndFragment), this._samplerName = "textureSampler", this.convertToGammaSpace = !1, this.convertToLinearSpace = !1, this._isUnique = !1, this.registerInput("uv", de.AutoDetect, !1, Ve.VertexAndFragment), this.registerOutput("rgba", de.Color4, Ve.Neutral), this.registerOutput("rgb", de.Color3, Ve.Neutral), this.registerOutput("r", de.Float, Ve.Neutral), this.registerOutput("g", de.Float, Ve.Neutral), this.registerOutput("b", de.Float, Ve.Neutral), this.registerOutput("a", de.Float, Ve.Neutral), this._inputs[0].addExcludedConnectionPointFromAllowedTypes(de.Vector2 | de.Vector3 | de.Vector4), this._inputs[0]._prioritizeVertex = !1; } /** * Gets the current class name * @returns the class name */ getClassName() { return "CurrentScreenBlock"; } /** * Gets the uv input component */ get uv() { return this._inputs[0]; } /** * Gets the rgba output component */ get rgba() { return this._outputs[0]; } /** * Gets the rgb output component */ get rgb() { return this._outputs[1]; } /** * Gets the r output component */ get r() { return this._outputs[2]; } /** * Gets the g output component */ get g() { return this._outputs[3]; } /** * Gets the b output component */ get b() { return this._outputs[4]; } /** * Gets the a output component */ get a() { return this._outputs[5]; } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("textureSampler"); } get target() { return !this.uv.isConnected || this.uv.sourceBlock.isInput ? Ve.VertexAndFragment : Ve.Fragment; } prepareDefines(e, t, r) { r.setValue(this._linearDefineName, this.convertToGammaSpace, !0), r.setValue(this._gammaDefineName, this.convertToLinearSpace, !0); } isReady() { return !(this.texture && !this.texture.isReadyOrNotBlocking()); } _injectVertexCode(e) { const t = this.uv; if (t.connectedPoint.ownerBlock.isInput && (t.connectedPoint.ownerBlock.isAttribute || e._emitUniformFromString(t.associatedVariableName, "vec2")), this._mainUVName = "vMain" + t.associatedVariableName, e._emitVaryingFromString(this._mainUVName, "vec2"), e.compilationString += `${this._mainUVName} = ${t.associatedVariableName}.xy; `, !!this._outputs.some((r) => r.isConnectedInVertexShader)) { this._writeTextureRead(e, !0); for (const r of this._outputs) r.hasEndpoints && this._writeOutput(e, r, r.name, !0); } } _writeTextureRead(e, t = !1) { const r = this.uv; if (t) { if (e.target === Ve.Fragment) return; e.compilationString += `vec4 ${this._tempTextureRead} = texture2D(${this._samplerName}, ${r.associatedVariableName}); `; return; } if (this.uv.ownerBlock.target === Ve.Fragment) { e.compilationString += `vec4 ${this._tempTextureRead} = texture2D(${this._samplerName}, ${r.associatedVariableName}); `; return; } e.compilationString += `vec4 ${this._tempTextureRead} = texture2D(${this._samplerName}, ${this._mainUVName}); `; } _writeOutput(e, t, r, n = !1) { if (n) { if (e.target === Ve.Fragment) return; e.compilationString += `${this._declareOutput(t, e)} = ${this._tempTextureRead}.${r}; `; return; } if (this.uv.ownerBlock.target === Ve.Fragment) { e.compilationString += `${this._declareOutput(t, e)} = ${this._tempTextureRead}.${r}; `; return; } e.compilationString += `${this._declareOutput(t, e)} = ${this._tempTextureRead}.${r}; `, e.compilationString += `#ifdef ${this._linearDefineName} `, e.compilationString += `${t.associatedVariableName} = toGammaSpace(${t.associatedVariableName}); `, e.compilationString += `#endif `, e.compilationString += `#ifdef ${this._gammaDefineName} `, e.compilationString += `${t.associatedVariableName} = toLinearSpace(${t.associatedVariableName}); `, e.compilationString += `#endif `; } _buildBlock(e) { if (super._buildBlock(e), this._tempTextureRead = e._getFreeVariableName("tempTextureRead"), e.sharedData.blockingBlocks.indexOf(this) < 0 && e.sharedData.blockingBlocks.push(this), e.sharedData.textureBlocks.indexOf(this) < 0 && e.sharedData.textureBlocks.push(this), e.sharedData.blocksWithDefines.indexOf(this) < 0 && e.sharedData.blocksWithDefines.push(this), e.target !== Ve.Fragment) { e._emit2DSampler(this._samplerName), this._injectVertexCode(e); return; } if (!this._outputs.some((r) => r.isConnectedInFragmentShader)) return; e._emit2DSampler(this._samplerName), this._linearDefineName = e._getFreeDefineName("ISLINEAR"), this._gammaDefineName = e._getFreeDefineName("ISGAMMA"); const t = `//${this.name}`; e._emitFunctionFromInclude("helperFunctions", t), this._writeTextureRead(e); for (const r of this._outputs) r.hasEndpoints && this._writeOutput(e, r, r.name); return this; } serialize() { const e = super.serialize(); return e.convertToGammaSpace = this.convertToGammaSpace, e.convertToLinearSpace = this.convertToLinearSpace, this.texture && !this.texture.isRenderTarget && (e.texture = this.texture.serialize()), e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.convertToGammaSpace = e.convertToGammaSpace, this.convertToLinearSpace = !!e.convertToLinearSpace, e.texture && (r = e.texture.url.indexOf("data:") === 0 ? "" : r, this.texture = We.Parse(e.texture, t, r)); } } Ue("BABYLON.CurrentScreenBlock", oQ); class fQ extends Mr { /** * Create a new ParticleTextureBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this._samplerName = "diffuseSampler", this.convertToGammaSpace = !1, this.convertToLinearSpace = !1, this._isUnique = !1, this.registerInput("uv", de.AutoDetect, !1, Ve.VertexAndFragment), this.registerOutput("rgba", de.Color4, Ve.Neutral), this.registerOutput("rgb", de.Color3, Ve.Neutral), this.registerOutput("r", de.Float, Ve.Neutral), this.registerOutput("g", de.Float, Ve.Neutral), this.registerOutput("b", de.Float, Ve.Neutral), this.registerOutput("a", de.Float, Ve.Neutral), this._inputs[0].addExcludedConnectionPointFromAllowedTypes(de.Vector2 | de.Vector3 | de.Vector4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ParticleTextureBlock"; } /** * Gets the uv input component */ get uv() { return this._inputs[0]; } /** * Gets the rgba output component */ get rgba() { return this._outputs[0]; } /** * Gets the rgb output component */ get rgb() { return this._outputs[1]; } /** * Gets the r output component */ get r() { return this._outputs[2]; } /** * Gets the g output component */ get g() { return this._outputs[3]; } /** * Gets the b output component */ get b() { return this._outputs[4]; } /** * Gets the a output component */ get a() { return this._outputs[5]; } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("diffuseSampler"); } autoConfigure(e, t = () => !0) { if (!this.uv.isConnected) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "particle_uv" && t(n)); r || (r = new gi("uv"), r.setAsAttribute("particle_uv")), r.output.connectTo(this.uv); } } prepareDefines(e, t, r) { r.setValue(this._linearDefineName, this.convertToGammaSpace, !0), r.setValue(this._gammaDefineName, this.convertToLinearSpace, !0); } isReady() { return !(this.texture && !this.texture.isReadyOrNotBlocking()); } _writeOutput(e, t, r) { e.compilationString += `${this._declareOutput(t, e)} = ${this._tempTextureRead}.${r}; `, e.compilationString += `#ifdef ${this._linearDefineName} `, e.compilationString += `${t.associatedVariableName} = toGammaSpace(${t.associatedVariableName}); `, e.compilationString += `#endif `, e.compilationString += `#ifdef ${this._gammaDefineName} `, e.compilationString += `${t.associatedVariableName} = toLinearSpace(${t.associatedVariableName}); `, e.compilationString += `#endif `; } _buildBlock(e) { if (super._buildBlock(e), e.target === Ve.Vertex) return; this._tempTextureRead = e._getFreeVariableName("tempTextureRead"), e._emit2DSampler(this._samplerName), e.sharedData.blockingBlocks.push(this), e.sharedData.textureBlocks.push(this), e.sharedData.blocksWithDefines.push(this), this._linearDefineName = e._getFreeDefineName("ISLINEAR"), this._gammaDefineName = e._getFreeDefineName("ISGAMMA"); const t = `//${this.name}`; e._emitFunctionFromInclude("helperFunctions", t), e.compilationString += `vec4 ${this._tempTextureRead} = texture2D(${this._samplerName}, ${this.uv.associatedVariableName}); `; for (const r of this._outputs) r.hasEndpoints && this._writeOutput(e, r, r.name); return this; } serialize() { const e = super.serialize(); return e.convertToGammaSpace = this.convertToGammaSpace, e.convertToLinearSpace = this.convertToLinearSpace, this.texture && !this.texture.isRenderTarget && (e.texture = this.texture.serialize()), e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.convertToGammaSpace = e.convertToGammaSpace, this.convertToLinearSpace = !!e.convertToLinearSpace, e.texture && (r = e.texture.url.indexOf("data:") === 0 ? "" : r, this.texture = We.Parse(e.texture, t, r)); } } Ue("BABYLON.ParticleTextureBlock", fQ); class AQ extends Mr { /** * Create a new ParticleRampGradientBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this._isUnique = !0, this.registerInput("color", de.Color4, !1, Ve.Fragment), this.registerOutput("rampColor", de.Color4, Ve.Fragment); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ParticleRampGradientBlock"; } /** * Gets the color input component */ get color() { return this._inputs[0]; } /** * Gets the rampColor output component */ get rampColor() { return this._outputs[0]; } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("remapRanges"), e._excludeVariableName("rampSampler"), e._excludeVariableName("baseColor"), e._excludeVariableName("alpha"), e._excludeVariableName("remappedColorIndex"), e._excludeVariableName("rampColor"), e._excludeVariableName("finalAlpha"); } _buildBlock(e) { if (super._buildBlock(e), e.target !== Ve.Vertex) return e._emit2DSampler("rampSampler"), e._emitVaryingFromString("remapRanges", "vec4", "RAMPGRADIENT"), e.compilationString += ` #ifdef RAMPGRADIENT vec4 baseColor = ${this.color.associatedVariableName}; float alpha = ${this.color.associatedVariableName}.a; float remappedColorIndex = clamp((alpha - remapRanges.x) / remapRanges.y, 0.0, 1.0); vec4 rampColor = texture2D(rampSampler, vec2(1.0 - remappedColorIndex, 0.)); baseColor.rgb *= rampColor.rgb; // Remapped alpha float finalAlpha = baseColor.a; baseColor.a = clamp((alpha * rampColor.a - remapRanges.z) / remapRanges.w, 0.0, 1.0); ${this._declareOutput(this.rampColor, e)} = baseColor; #else ${this._declareOutput(this.rampColor, e)} = ${this.color.associatedVariableName}; #endif `, this; } } Ue("BABYLON.ParticleRampGradientBlock", AQ); class dQ extends Mr { /** * Create a new ParticleBlendMultiplyBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this._isUnique = !0, this.registerInput("color", de.Color4, !1, Ve.Fragment), this.registerInput("alphaTexture", de.Float, !1, Ve.Fragment), this.registerInput("alphaColor", de.Float, !1, Ve.Fragment), this.registerOutput("blendColor", de.Color4, Ve.Fragment); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ParticleBlendMultiplyBlock"; } /** * Gets the color input component */ get color() { return this._inputs[0]; } /** * Gets the alphaTexture input component */ get alphaTexture() { return this._inputs[1]; } /** * Gets the alphaColor input component */ get alphaColor() { return this._inputs[2]; } /** * Gets the blendColor output component */ get blendColor() { return this._outputs[0]; } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("sourceAlpha"); } _buildBlock(e) { if (super._buildBlock(e), e.target !== Ve.Vertex) return e.compilationString += ` #ifdef BLENDMULTIPLYMODE ${this._declareOutput(this.blendColor, e)}; float sourceAlpha = ${this.alphaColor.associatedVariableName} * ${this.alphaTexture.associatedVariableName}; ${this.blendColor.associatedVariableName}.rgb = ${this.color.associatedVariableName}.rgb * sourceAlpha + vec3(1.0) * (1.0 - sourceAlpha); ${this.blendColor.associatedVariableName}.a = ${this.color.associatedVariableName}.a; #else ${this._declareOutput(this.blendColor, e)} = ${this.color.associatedVariableName}; #endif `, this; } } Ue("BABYLON.ParticleBlendMultiplyBlock", dQ); class c1 { constructor() { this._defines = {}, this._currentRank = 32, this._maxRank = -1, this._mesh = null; } /** * Removes the fallback from the bound mesh. */ unBindMesh() { this._mesh = null; } /** * Adds a fallback on the specified property. * @param rank The rank of the fallback (Lower ranks will be fallbacked to first) * @param define The name of the define in the shader */ addFallback(e, t) { this._defines[e] || (e < this._currentRank && (this._currentRank = e), e > this._maxRank && (this._maxRank = e), this._defines[e] = new Array()), this._defines[e].push(t); } /** * Sets the mesh to use CPU skinning when needing to fallback. * @param rank The rank of the fallback (Lower ranks will be fallbacked to first) * @param mesh The mesh to use the fallbacks. */ addCPUSkinningFallback(e, t) { this._mesh = t, e < this._currentRank && (this._currentRank = e), e > this._maxRank && (this._maxRank = e); } /** * Checks to see if more fallbacks are still available. */ get hasMoreFallbacks() { return this._currentRank <= this._maxRank; } /** * Removes the defines that should be removed when falling back. * @param currentDefines defines the current define statements for the shader. * @param effect defines the current effect we try to compile * @returns The resulting defines with defines of the current rank removed. */ reduce(e, t) { if (this._mesh && this._mesh.computeBonesUsingShaders && this._mesh.numBoneInfluencers > 0) { this._mesh.computeBonesUsingShaders = !1, e = e.replace("#define NUM_BONE_INFLUENCERS " + this._mesh.numBoneInfluencers, "#define NUM_BONE_INFLUENCERS 0"), t._bonesComputationForcedToCPU = !0; const r = this._mesh.getScene(); for (let n = 0; n < r.meshes.length; n++) { const i = r.meshes[n]; if (!i.material) { !this._mesh.material && i.computeBonesUsingShaders && i.numBoneInfluencers > 0 && (i.computeBonesUsingShaders = !1); continue; } if (!(!i.computeBonesUsingShaders || i.numBoneInfluencers === 0)) { if (i.material.getEffect() === t) i.computeBonesUsingShaders = !1; else if (i.subMeshes) { for (const s of i.subMeshes) if (s.effect === t) { i.computeBonesUsingShaders = !1; break; } } } } } else { const r = this._defines[this._currentRank]; if (r) for (let n = 0; n < r.length; n++) e = e.replace("#define " + r[n], ""); this._currentRank++; } return e; } } const U6e = "postprocessVertexShader", I6e = `attribute vec2 position;uniform vec2 scale;varying vec2 vUV;const vec2 madd=vec2(0.5,0.5); #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN vUV=(position*madd+madd)*scale;gl_Position=vec4(position,0.0,1.0); #define CUSTOM_VERTEX_MAIN_END }`; Le.ShadersStore[U6e] = I6e; class DR { /** * Gets the depth/stencil texture (if created by a createDepthStencilTexture() call) */ get depthStencilTexture() { return this._depthStencilTexture; } /** * Indicates if the depth/stencil texture has a stencil aspect */ get depthStencilTextureWithStencil() { return this._depthStencilTextureWithStencil; } /** * Defines if the render target wrapper is for a cube texture or if false a 2d texture */ get isCube() { return this._isCube; } /** * Defines if the render target wrapper is for a single or multi target render wrapper */ get isMulti() { return this._isMulti; } /** * Defines if the render target wrapper is for a single or an array of textures */ get is2DArray() { return this.layers > 0; } /** * Gets the size of the render target wrapper (used for cubes, as width=height in this case) */ get size() { return this.width; } /** * Gets the width of the render target wrapper */ get width() { return this._size.width || this._size; } /** * Gets the height of the render target wrapper */ get height() { return this._size.height || this._size; } /** * Gets the number of layers of the render target wrapper (only used if is2DArray is true and wrapper is not a multi render target) */ get layers() { return this._size.layers || 0; } /** * Gets the render texture. If this is a multi render target, gets the first texture */ get texture() { var e, t; return (t = (e = this._textures) === null || e === void 0 ? void 0 : e[0]) !== null && t !== void 0 ? t : null; } /** * Gets the list of render textures. If we are not in a multi render target, the list will be null (use the texture getter instead) */ get textures() { return this._textures; } /** * Gets the face indices that correspond to the list of render textures. If we are not in a multi render target, the list will be null */ get faceIndices() { return this._faceIndices; } /** * Gets the layer indices that correspond to the list of render textures. If we are not in a multi render target, the list will be null */ get layerIndices() { return this._layerIndices; } /** * Gets the sample count of the render target */ get samples() { return this._samples; } /** * Sets the sample count of the render target * @param value sample count * @param initializeBuffers If set to true, the engine will make an initializing call to drawBuffers (only used when isMulti=true). * @param force true to force calling the update sample count engine function even if the current sample count is equal to value * @returns the sample count that has been set */ setSamples(e, t = !0, r = !1) { if (this.samples === e && !r) return e; const n = this._isMulti ? this._engine.updateMultipleRenderTargetTextureSampleCount(this, e, t) : this._engine.updateRenderTargetTextureSampleCount(this, e); return this._samples = e, n; } /** * Initializes the render target wrapper * @param isMulti true if the wrapper is a multi render target * @param isCube true if the wrapper should render to a cube texture * @param size size of the render target (width/height/layers) * @param engine engine used to create the render target * @param label defines the label to use for the wrapper (for debugging purpose only) */ constructor(e, t, r, n, i) { this._textures = null, this._faceIndices = null, this._layerIndices = null, this._samples = 1, this._attachments = null, this._generateStencilBuffer = !1, this._generateDepthBuffer = !1, this._depthStencilTextureWithStencil = !1, this._isMulti = e, this._isCube = t, this._size = r, this._engine = n, this._depthStencilTexture = null, this.label = i; } /** * Sets the render target texture(s) * @param textures texture(s) to set */ setTextures(e) { Array.isArray(e) ? this._textures = e : e ? this._textures = [e] : this._textures = null; } /** * Set a texture in the textures array * @param texture The texture to set * @param index The index in the textures array to set * @param disposePrevious If this function should dispose the previous texture */ setTexture(e, t = 0, r = !0) { this._textures || (this._textures = []), this._textures[t] !== e && (this._textures[t] && r && this._textures[t].dispose(), this._textures[t] = e); } /** * Sets the layer and face indices of every render target texture bound to each color attachment * @param layers The layers of each texture to be set * @param faces The faces of each texture to be set */ setLayerAndFaceIndices(e, t) { this._layerIndices = e, this._faceIndices = t; } /** * Sets the layer and face indices of a texture in the textures array that should be bound to each color attachment * @param index The index of the texture in the textures array to modify * @param layer The layer of the texture to be set * @param face The face of the texture to be set */ setLayerAndFaceIndex(e = 0, t, r) { this._layerIndices || (this._layerIndices = []), this._faceIndices || (this._faceIndices = []), t !== void 0 && t >= 0 && (this._layerIndices[e] = t), r !== void 0 && r >= 0 && (this._faceIndices[e] = r); } /** * Creates the depth/stencil texture * @param comparisonFunction Comparison function to use for the texture * @param bilinearFiltering true if bilinear filtering should be used when sampling the texture * @param generateStencil true if the stencil aspect should also be created * @param samples sample count to use when creating the texture * @param format format of the depth texture * @param label defines the label to use for the texture (for debugging purpose only) * @returns the depth/stencil created texture */ createDepthStencilTexture(e = 0, t = !0, r = !1, n = 1, i = 14, s) { var a; return (a = this._depthStencilTexture) === null || a === void 0 || a.dispose(), this._depthStencilTextureWithStencil = r, this._depthStencilTexture = this._engine.createDepthStencilTexture(this._size, { bilinearFiltering: t, comparisonFunction: e, generateStencil: r, isCube: this._isCube, samples: n, depthTextureFormat: i, label: s }, this), this._depthStencilTexture; } /** * Shares the depth buffer of this render target with another render target. * @internal * @param renderTarget Destination renderTarget */ _shareDepth(e) { this._depthStencilTexture && (e._depthStencilTexture && e._depthStencilTexture.dispose(), e._depthStencilTexture = this._depthStencilTexture, this._depthStencilTexture.incrementReferences()); } /** * @internal */ _swapAndDie(e) { this.texture && this.texture._swapAndDie(e), this._textures = null, this.dispose(!0); } _cloneRenderTargetWrapper() { var e, t, r, n, i, s, a, f; let o = null; if (this._isMulti) { const d = this.textures; if (d && d.length > 0) { let v = !1, u = d.length; const l = d[d.length - 1]._source; (l === ri.Depth || l === ri.DepthStencil) && (v = !0, u--); const P = [], p = [], c = [], H = [], T = [], q = [], b = [], j = {}; for (let I = 0; I < u; ++I) { const N = d[I]; P.push(N.samplingMode), p.push(N.type), c.push(N.format), j[N.uniqueId] !== void 0 ? (H.push(-1), b.push(0)) : (j[N.uniqueId] = I, N.is2DArray ? (H.push(35866), b.push(N.depth)) : N.isCube ? (H.push(34067), b.push(0)) : N.is3D ? (H.push(32879), b.push(N.depth)) : (H.push(3553), b.push(0))), this._faceIndices && T.push((e = this._faceIndices[I]) !== null && e !== void 0 ? e : 0), this._layerIndices && q.push((t = this._layerIndices[I]) !== null && t !== void 0 ? t : 0); } const w = { samplingModes: P, generateMipMaps: d[0].generateMipMaps, generateDepthBuffer: this._generateDepthBuffer, generateStencilBuffer: this._generateStencilBuffer, generateDepthTexture: v, types: p, formats: c, textureCount: u, targetTypes: H, faceIndex: T, layerIndex: q, layerCounts: b }, m = { width: this.width, height: this.height }; o = this._engine.createMultipleRenderTarget(m, w); for (let I = 0; I < u; ++I) { if (H[I] !== -1) continue; const N = j[d[I].uniqueId]; o.setTexture(o.textures[N], I); } } } else { const d = {}; if (d.generateDepthBuffer = this._generateDepthBuffer, d.generateMipMaps = (n = (r = this.texture) === null || r === void 0 ? void 0 : r.generateMipMaps) !== null && n !== void 0 ? n : !1, d.generateStencilBuffer = this._generateStencilBuffer, d.samplingMode = (i = this.texture) === null || i === void 0 ? void 0 : i.samplingMode, d.type = (s = this.texture) === null || s === void 0 ? void 0 : s.type, d.format = (a = this.texture) === null || a === void 0 ? void 0 : a.format, this.isCube) o = this._engine.createRenderTargetCubeTexture(this.width, d); else { const v = { width: this.width, height: this.height, layers: this.is2DArray ? (f = this.texture) === null || f === void 0 ? void 0 : f.depth : void 0 }; o = this._engine.createRenderTargetTexture(v, d); } o.texture.isReady = !0; } return o; } _swapRenderTargetWrapper(e) { if (this._textures && e._textures) for (let t = 0; t < this._textures.length; ++t) this._textures[t]._swapAndDie(e._textures[t], !1), e._textures[t].isReady = !0; this._depthStencilTexture && e._depthStencilTexture && (this._depthStencilTexture._swapAndDie(e._depthStencilTexture), e._depthStencilTexture.isReady = !0), this._textures = null, this._depthStencilTexture = null; } /** @internal */ _rebuild() { const e = this._cloneRenderTargetWrapper(); if (e) { if (this._depthStencilTexture) { const t = this._depthStencilTexture.samplingMode, r = t === 2 || t === 3 || t === 11; e.createDepthStencilTexture(this._depthStencilTexture._comparisonFunction, r, this._depthStencilTextureWithStencil, this._depthStencilTexture.samples); } this.samples > 1 && e.setSamples(this.samples), e._swapRenderTargetWrapper(this), e.dispose(); } } /** * Releases the internal render textures */ releaseTextures() { var e, t; if (this._textures) for (let r = 0; (t = r < ((e = this._textures) === null || e === void 0 ? void 0 : e.length)) !== null && t !== void 0 && t; ++r) this._textures[r].dispose(); this._textures = null; } /** * Disposes the whole render target wrapper * @param disposeOnlyFramebuffers true if only the frame buffers should be released (used for the WebGL engine). If false, all the textures will also be released */ dispose(e = !1) { var t; e || ((t = this._depthStencilTexture) === null || t === void 0 || t.dispose(), this._depthStencilTexture = null, this.releaseTextures()), this._engine._releaseRenderTargetWrapper(this); } } class R6e extends DR { constructor(e, t, r, n, i) { super(e, t, r, n), this._framebuffer = null, this._depthStencilBuffer = null, this._MSAAFramebuffer = null, this._colorTextureArray = null, this._depthStencilTextureArray = null, this._disposeOnlyFramebuffers = !1, this._currentLOD = 0, this._context = i; } _cloneRenderTargetWrapper() { let e = null; return this._colorTextureArray && this._depthStencilTextureArray ? (e = this._engine.createMultiviewRenderTargetTexture(this.width, this.height), e.texture.isReady = !0) : e = super._cloneRenderTargetWrapper(), e; } _swapRenderTargetWrapper(e) { super._swapRenderTargetWrapper(e), e._framebuffer = this._framebuffer, e._depthStencilBuffer = this._depthStencilBuffer, e._MSAAFramebuffer = this._MSAAFramebuffer, e._colorTextureArray = this._colorTextureArray, e._depthStencilTextureArray = this._depthStencilTextureArray, this._framebuffer = this._depthStencilBuffer = this._MSAAFramebuffer = this._colorTextureArray = this._depthStencilTextureArray = null; } /** * Shares the depth buffer of this render target with another render target. * @internal * @param renderTarget Destination renderTarget */ _shareDepth(e) { super._shareDepth(e); const t = this._context, r = this._depthStencilBuffer, n = e._MSAAFramebuffer || e._framebuffer; e._depthStencilBuffer && e._depthStencilBuffer !== r && t.deleteRenderbuffer(e._depthStencilBuffer), e._depthStencilBuffer = r; const i = e._generateStencilBuffer ? t.DEPTH_STENCIL_ATTACHMENT : t.DEPTH_ATTACHMENT; this._engine._bindUnboundFramebuffer(n), t.framebufferRenderbuffer(t.FRAMEBUFFER, i, t.RENDERBUFFER, r), this._engine._bindUnboundFramebuffer(null); } /** * Binds a texture to this render target on a specific attachment * @param texture The texture to bind to the framebuffer * @param attachmentIndex Index of the attachment * @param faceIndexOrLayer The face or layer of the texture to render to in case of cube texture or array texture * @param lodLevel defines the lod level to bind to the frame buffer */ _bindTextureRenderTarget(e, t = 0, r, n = 0) { var i, s, a, f; if (!e._hardwareTexture) return; const o = this._framebuffer, d = this._engine._currentFramebuffer; if (this._engine._bindUnboundFramebuffer(o), this._engine.webGLVersion > 1) { const v = this._context, u = v["COLOR_ATTACHMENT" + t]; e.is2DArray || e.is3D ? (r = (s = r ?? ((i = this.layerIndices) === null || i === void 0 ? void 0 : i[t])) !== null && s !== void 0 ? s : 0, v.framebufferTextureLayer(v.FRAMEBUFFER, u, e._hardwareTexture.underlyingResource, n, r)) : e.isCube ? (r = (f = r ?? ((a = this.faceIndices) === null || a === void 0 ? void 0 : a[t])) !== null && f !== void 0 ? f : 0, v.framebufferTexture2D(v.FRAMEBUFFER, u, v.TEXTURE_CUBE_MAP_POSITIVE_X + r, e._hardwareTexture.underlyingResource, n)) : v.framebufferTexture2D(v.FRAMEBUFFER, u, v.TEXTURE_2D, e._hardwareTexture.underlyingResource, n); } else { const v = this._context, u = v["COLOR_ATTACHMENT" + t + "_WEBGL"], l = r !== void 0 ? v.TEXTURE_CUBE_MAP_POSITIVE_X + r : v.TEXTURE_2D; v.framebufferTexture2D(v.FRAMEBUFFER, u, l, e._hardwareTexture.underlyingResource, n); } this._engine._bindUnboundFramebuffer(d); } /** * Set a texture in the textures array * @param texture the texture to set * @param index the index in the textures array to set * @param disposePrevious If this function should dispose the previous texture */ setTexture(e, t = 0, r = !0) { super.setTexture(e, t, r), this._bindTextureRenderTarget(e, t); } /** * Sets the layer and face indices of every render target texture * @param layers The layer of the texture to be set (make negative to not modify) * @param faces The face of the texture to be set (make negative to not modify) */ setLayerAndFaceIndices(e, t) { var r, n; if (super.setLayerAndFaceIndices(e, t), !this.textures || !this.layerIndices || !this.faceIndices) return; const i = (n = (r = this._attachments) === null || r === void 0 ? void 0 : r.length) !== null && n !== void 0 ? n : this.textures.length; for (let s = 0; s < i; s++) { const a = this.textures[s]; a && (a.is2DArray || a.is3D ? this._bindTextureRenderTarget(a, s, this.layerIndices[s]) : a.isCube ? this._bindTextureRenderTarget(a, s, this.faceIndices[s]) : this._bindTextureRenderTarget(a, s)); } } /** * Set the face and layer indices of a texture in the textures array * @param index The index of the texture in the textures array to modify * @param layer The layer of the texture to be set * @param face The face of the texture to be set */ setLayerAndFaceIndex(e = 0, t, r) { if (super.setLayerAndFaceIndex(e, t, r), !this.textures || !this.layerIndices || !this.faceIndices) return; const n = this.textures[e]; n.is2DArray || n.is3D ? this._bindTextureRenderTarget(this.textures[e], e, this.layerIndices[e]) : n.isCube && this._bindTextureRenderTarget(this.textures[e], e, this.faceIndices[e]); } dispose(e = this._disposeOnlyFramebuffers) { const t = this._context; e || (this._colorTextureArray && (this._context.deleteTexture(this._colorTextureArray), this._colorTextureArray = null), this._depthStencilTextureArray && (this._context.deleteTexture(this._depthStencilTextureArray), this._depthStencilTextureArray = null)), this._framebuffer && (t.deleteFramebuffer(this._framebuffer), this._framebuffer = null), this._depthStencilBuffer && (t.deleteRenderbuffer(this._depthStencilBuffer), this._depthStencilBuffer = null), this._MSAAFramebuffer && (t.deleteFramebuffer(this._MSAAFramebuffer), this._MSAAFramebuffer = null), super.dispose(e); } } hr.prototype._createHardwareRenderTargetWrapper = function(A, e, t) { const r = new R6e(A, e, t, this, this._gl); return this._renderTargetWrapperCache.push(r), r; }; hr.prototype.createRenderTargetTexture = function(A, e) { var t, r; const n = this._createHardwareRenderTargetWrapper(!1, !1, A); let i = !0, s = !1, a = !1, f, o = 1; e !== void 0 && typeof e == "object" && (i = (t = e.generateDepthBuffer) !== null && t !== void 0 ? t : !0, s = !!e.generateStencilBuffer, a = !!e.noColorAttachment, f = e.colorAttachment, o = (r = e.samples) !== null && r !== void 0 ? r : 1); const d = f || (a ? null : this._createInternalTexture(A, e, !0, ri.RenderTarget)), v = A.width || A, u = A.height || A, l = this._currentFramebuffer, P = this._gl, p = P.createFramebuffer(); return this._bindUnboundFramebuffer(p), n._depthStencilBuffer = this._setupFramebufferDepthAttachments(s, i, v, u), d && !d.is2DArray && P.framebufferTexture2D(P.FRAMEBUFFER, P.COLOR_ATTACHMENT0, P.TEXTURE_2D, d._hardwareTexture.underlyingResource, 0), this._bindUnboundFramebuffer(l), n._framebuffer = p, n._generateDepthBuffer = i, n._generateStencilBuffer = s, n.setTextures(d), this.updateRenderTargetTextureSampleCount(n, o), n; }; hr.prototype.createDepthStencilTexture = function(A, e, t) { if (e.isCube) { const r = A.width || A; return this._createDepthStencilCubeTexture(r, e, t); } else return this._createDepthStencilTexture(A, e, t); }; hr.prototype._createDepthStencilTexture = function(A, e, t) { const r = this._gl, n = A.layers || 0, i = n !== 0 ? r.TEXTURE_2D_ARRAY : r.TEXTURE_2D, s = new As(this, ri.DepthStencil); if (!this._caps.depthTextureExtension) return Se.Error("Depth texture is not supported by your browser or hardware."), s; const a = Object.assign({ bilinearFiltering: !1, comparisonFunction: 0, generateStencil: !1 }, e); if (this._bindTextureDirectly(i, s, !0), this._setupDepthStencilTexture(s, A, a.generateStencil, a.comparisonFunction === 0 ? !1 : a.bilinearFiltering, a.comparisonFunction, a.samples), a.depthTextureFormat !== void 0) { if (a.depthTextureFormat !== 15 && a.depthTextureFormat !== 16 && a.depthTextureFormat !== 17 && a.depthTextureFormat !== 13 && a.depthTextureFormat !== 14 && a.depthTextureFormat !== 18) return Se.Error("Depth texture format is not supported."), s; s.format = a.depthTextureFormat; } else s.format = a.generateStencil ? 13 : 16; const f = s.format === 17 || s.format === 13 || s.format === 18; t._depthStencilTexture = s, t._depthStencilTextureWithStencil = f; let o = r.UNSIGNED_INT; s.format === 15 ? o = r.UNSIGNED_SHORT : s.format === 17 || s.format === 13 ? o = r.UNSIGNED_INT_24_8 : s.format === 14 ? o = r.FLOAT : s.format === 18 && (o = r.FLOAT_32_UNSIGNED_INT_24_8_REV); const d = f ? r.DEPTH_STENCIL : r.DEPTH_COMPONENT; let v = d; this.webGLVersion > 1 && (s.format === 15 ? v = r.DEPTH_COMPONENT16 : s.format === 16 ? v = r.DEPTH_COMPONENT24 : s.format === 17 || s.format === 13 ? v = r.DEPTH24_STENCIL8 : s.format === 14 ? v = r.DEPTH_COMPONENT32F : s.format === 18 && (v = r.DEPTH32F_STENCIL8)), s.is2DArray ? r.texImage3D(i, 0, v, s.width, s.height, n, 0, d, o, null) : r.texImage2D(i, 0, v, s.width, s.height, 0, d, o, null), this._bindTextureDirectly(i, null), this._internalTexturesCache.push(s); const u = t; if (u._depthStencilBuffer) { const l = this._currentFramebuffer; this._bindUnboundFramebuffer(u._framebuffer), r.framebufferRenderbuffer(r.FRAMEBUFFER, r.DEPTH_STENCIL_ATTACHMENT, r.RENDERBUFFER, null), r.framebufferRenderbuffer(r.FRAMEBUFFER, r.DEPTH_ATTACHMENT, r.RENDERBUFFER, null), r.framebufferRenderbuffer(r.FRAMEBUFFER, r.STENCIL_ATTACHMENT, r.RENDERBUFFER, null), this._bindUnboundFramebuffer(l), r.deleteRenderbuffer(u._depthStencilBuffer), u._depthStencilBuffer = null; } return s; }; hr.prototype.updateRenderTargetTextureSampleCount = function(A, e) { if (this.webGLVersion < 2 || !A || !A.texture) return 1; if (A.samples === e) return e; const t = this._gl; e = Math.min(e, this.getCaps().maxMSAASamples), A._depthStencilBuffer && (t.deleteRenderbuffer(A._depthStencilBuffer), A._depthStencilBuffer = null), A._MSAAFramebuffer && (t.deleteFramebuffer(A._MSAAFramebuffer), A._MSAAFramebuffer = null); const r = A.texture._hardwareTexture; if (r.releaseMSAARenderBuffers(), e > 1 && typeof t.renderbufferStorageMultisample == "function") { const n = t.createFramebuffer(); if (!n) throw new Error("Unable to create multi sampled framebuffer"); A._MSAAFramebuffer = n, this._bindUnboundFramebuffer(A._MSAAFramebuffer); const i = this._createRenderBuffer(A.texture.width, A.texture.height, e, -1, this._getRGBABufferInternalSizedFormat(A.texture.type, A.texture.format, A.texture._useSRGBBuffer), t.COLOR_ATTACHMENT0, !1); if (!i) throw new Error("Unable to create multi sampled framebuffer"); r.addMSAARenderBuffer(i); } else this._bindUnboundFramebuffer(A._framebuffer); return A.texture.samples = e, A._samples = e, A._depthStencilBuffer = this._setupFramebufferDepthAttachments(A._generateStencilBuffer, A._generateDepthBuffer, A.texture.width, A.texture.height, e), this._bindUnboundFramebuffer(null), e; }; class kr { /** * Registers a shader code processing with a post process name. * @param postProcessName name of the post process. Use null for the fallback shader code processing. This is the shader code processing that will be used in case no specific shader code processing has been associated to a post process name * @param customShaderCodeProcessing shader code processing to associate to the post process name * @returns */ static RegisterShaderCodeProcessing(e, t) { if (!t) { delete kr._CustomShaderCodeProcessing[e ?? ""]; return; } kr._CustomShaderCodeProcessing[e ?? ""] = t; } static _GetShaderCodeProcessing(e) { var t; return (t = kr._CustomShaderCodeProcessing[e]) !== null && t !== void 0 ? t : kr._CustomShaderCodeProcessing[""]; } /** * Number of sample textures (default: 1) */ get samples() { return this._samples; } set samples(e) { this._samples = Math.min(e, this._engine.getCaps().maxMSAASamples), this._textures.forEach((t) => { t.setSamples(this._samples); }); } /** * Returns the fragment url or shader name used in the post process. * @returns the fragment url or name in the shader store. */ getEffectName() { return this._fragmentUrl; } /** * A function that is added to the onActivateObservable */ set onActivate(e) { this._onActivateObserver && this.onActivateObservable.remove(this._onActivateObserver), e && (this._onActivateObserver = this.onActivateObservable.add(e)); } /** * A function that is added to the onSizeChangedObservable */ set onSizeChanged(e) { this._onSizeChangedObserver && this.onSizeChangedObservable.remove(this._onSizeChangedObserver), this._onSizeChangedObserver = this.onSizeChangedObservable.add(e); } /** * A function that is added to the onApplyObservable */ set onApply(e) { this._onApplyObserver && this.onApplyObservable.remove(this._onApplyObserver), this._onApplyObserver = this.onApplyObservable.add(e); } /** * A function that is added to the onBeforeRenderObservable */ set onBeforeRender(e) { this._onBeforeRenderObserver && this.onBeforeRenderObservable.remove(this._onBeforeRenderObserver), this._onBeforeRenderObserver = this.onBeforeRenderObservable.add(e); } /** * A function that is added to the onAfterRenderObservable */ set onAfterRender(e) { this._onAfterRenderObserver && this.onAfterRenderObservable.remove(this._onAfterRenderObserver), this._onAfterRenderObserver = this.onAfterRenderObservable.add(e); } /** * The input texture for this post process and the output texture of the previous post process. When added to a pipeline the previous post process will * render it's output into this texture and this texture will be used as textureSampler in the fragment shader of this post process. */ get inputTexture() { return this._textures.data[this._currentRenderTextureInd]; } set inputTexture(e) { this._forcedOutputTexture = e; } /** * Since inputTexture should always be defined, if we previously manually set `inputTexture`, * the only way to unset it is to use this function to restore its internal state */ restoreDefaultInputTexture() { this._forcedOutputTexture && (this._forcedOutputTexture = null, this.markTextureDirty()); } /** * Gets the camera which post process is applied to. * @returns The camera the post process is applied to. */ getCamera() { return this._camera; } /** * Gets the texel size of the postprocess. * See https://en.wikipedia.org/wiki/Texel_(graphics) */ get texelSize() { return this._shareOutputWithPostProcess ? this._shareOutputWithPostProcess.texelSize : (this._forcedOutputTexture && this._texelSize.copyFromFloats(1 / this._forcedOutputTexture.width, 1 / this._forcedOutputTexture.height), this._texelSize); } /** @internal */ constructor(e, t, r, n, i, s, a = 1, f, o, d = null, v = 0, u = "postprocess", l, P = !1, p = 5, c = za.GLSL) { var H, T, q, b, j, w, m, I, N, k, R, y; this._parentContainer = null, this.width = -1, this.height = -1, this.nodeMaterialSource = null, this._outputTexture = null, this.autoClear = !0, this.forceAutoClearInAlphaMode = !1, this.alphaMode = 0, this.animations = [], this.enablePixelPerfectMode = !1, this.forceFullscreenViewport = !0, this.scaleMode = 1, this.alwaysForcePOT = !1, this._samples = 1, this.adaptScaleToCurrentViewport = !1, this._reusable = !1, this._renderId = 0, this.externalTextureSamplerBinding = !1, this._textures = new qf(2), this._textureCache = [], this._currentRenderTextureInd = 0, this._scaleRatio = new at(1, 1), this._texelSize = at.Zero(), this.onActivateObservable = new Oe(), this.onSizeChangedObservable = new Oe(), this.onApplyObservable = new Oe(), this.onBeforeRenderObservable = new Oe(), this.onAfterRenderObservable = new Oe(), this.name = e; let O = 1, Y = null; if (r && !Array.isArray(r)) { const ee = r; r = (H = ee.uniforms) !== null && H !== void 0 ? H : null, n = (T = ee.samplers) !== null && T !== void 0 ? T : null, O = (q = ee.size) !== null && q !== void 0 ? q : 1, s = (b = ee.camera) !== null && b !== void 0 ? b : null, a = (j = ee.samplingMode) !== null && j !== void 0 ? j : 1, f = ee.engine, o = ee.reusable, d = (w = ee.defines) !== null && w !== void 0 ? w : null, v = (m = ee.textureType) !== null && m !== void 0 ? m : 0, u = (I = ee.vertexUrl) !== null && I !== void 0 ? I : "postprocess", l = ee.indexParameters, P = (N = ee.blockCompilation) !== null && N !== void 0 ? N : !1, p = (k = ee.textureFormat) !== null && k !== void 0 ? k : 5, c = (R = ee.shaderLanguage) !== null && R !== void 0 ? R : za.GLSL, Y = (y = ee.uniformBuffers) !== null && y !== void 0 ? y : null; } else i && (typeof i == "number" ? O = i : O = { width: i.width, height: i.height }); s != null ? (this._camera = s, this._scene = s.getScene(), s.attachPostProcess(this), this._engine = this._scene.getEngine(), this._scene.postProcesses.push(this), this.uniqueId = this._scene.getUniqueId()) : f && (this._engine = f, this._engine.postProcesses.push(this)), this._options = O, this.renderTargetSamplingMode = a || 1, this._reusable = o || !1, this._textureType = v, this._textureFormat = p, this._shaderLanguage = c, this._samplers = n || [], this._samplers.push("textureSampler"), this._fragmentUrl = t, this._vertexUrl = u, this._parameters = r || [], this._parameters.push("scale"), this._uniformBuffers = Y || [], this._indexParameters = l, this._drawWrapper = new zo(this._engine), P || this.updateEffect(d); } /** * Gets a string identifying the name of the class * @returns "PostProcess" string */ getClassName() { return "PostProcess"; } /** * Gets the engine which this post process belongs to. * @returns The engine the post process was enabled with. */ getEngine() { return this._engine; } /** * The effect that is created when initializing the post process. * @returns The created effect corresponding the postprocess. */ getEffect() { return this._drawWrapper.effect; } /** * To avoid multiple redundant textures for multiple post process, the output the output texture for this post process can be shared with another. * @param postProcess The post process to share the output with. * @returns This post process. */ shareOutputWith(e) { return this._disposeTextures(), this._shareOutputWithPostProcess = e, this; } /** * Reverses the effect of calling shareOutputWith and returns the post process back to its original state. * This should be called if the post process that shares output with this post process is disabled/disposed. */ useOwnOutput() { this._textures.length == 0 && (this._textures = new qf(2)), this._shareOutputWithPostProcess = null; } /** * Updates the effect with the current post process compile time values and recompiles the shader. * @param defines Define statements that should be added at the beginning of the shader. (default: null) * @param uniforms Set of uniform variables that will be passed to the shader. (default: null) * @param samplers Set of Texture2D variables that will be passed to the shader. (default: null) * @param indexParameters The index parameters to be used for babylons include syntax "#include[0..varyingCount]". (default: undefined) See usage in babylon.blurPostProcess.ts and kernelBlur.vertex.fx * @param onCompiled Called when the shader has been compiled. * @param onError Called if there is an error when compiling a shader. * @param vertexUrl The url of the vertex shader to be used (default: the one given at construction time) * @param fragmentUrl The url of the fragment shader to be used (default: the one given at construction time) */ updateEffect(e = null, t = null, r = null, n, i, s, a, f) { var o, d; const v = kr._GetShaderCodeProcessing(this.name); if (v != null && v.defineCustomBindings) { const u = (o = t == null ? void 0 : t.slice()) !== null && o !== void 0 ? o : []; u.push(...this._parameters); const l = (d = r == null ? void 0 : r.slice()) !== null && d !== void 0 ? d : []; l.push(...this._samplers), e = v.defineCustomBindings(this.name, e, u, l), t = u, r = l; } this._postProcessDefines = e, this._drawWrapper.effect = this._engine.createEffect({ vertex: a ?? this._vertexUrl, fragment: f ?? this._fragmentUrl }, { attributes: ["position"], uniformsNames: t || this._parameters, uniformBuffersNames: this._uniformBuffers, samplers: r || this._samplers, defines: e !== null ? e : "", fallbacks: null, onCompiled: i ?? null, onError: s ?? null, indexParameters: n || this._indexParameters, processCodeAfterIncludes: v != null && v.processCodeAfterIncludes ? (u, l) => v.processCodeAfterIncludes(this.name, u, l) : null, processFinalCode: v != null && v.processFinalCode ? (u, l) => v.processFinalCode(this.name, u, l) : null, shaderLanguage: this._shaderLanguage }, this._engine); } /** * The post process is reusable if it can be used multiple times within one frame. * @returns If the post process is reusable */ isReusable() { return this._reusable; } /** invalidate frameBuffer to hint the postprocess to create a depth buffer */ markTextureDirty() { this.width = -1; } _createRenderTargetTexture(e, t, r = 0) { for (let i = 0; i < this._textureCache.length; i++) if (this._textureCache[i].texture.width === e.width && this._textureCache[i].texture.height === e.height && this._textureCache[i].postProcessChannel === r && this._textureCache[i].texture._generateDepthBuffer === t.generateDepthBuffer && this._textureCache[i].texture.samples === t.samples) return this._textureCache[i].texture; const n = this._engine.createRenderTargetTexture(e, t); return this._textureCache.push({ texture: n, postProcessChannel: r, lastUsedRenderId: -1 }), n; } _flushTextureCache() { const e = this._renderId; for (let t = this._textureCache.length - 1; t >= 0; t--) if (e - this._textureCache[t].lastUsedRenderId > 100) { let r = !1; for (let n = 0; n < this._textures.length; n++) if (this._textures.data[n] === this._textureCache[t].texture) { r = !0; break; } r || (this._textureCache[t].texture.dispose(), this._textureCache.splice(t, 1)); } } /** * Resizes the post-process texture * @param width Width of the texture * @param height Height of the texture * @param camera The camera this post-process is applied to. Pass null if the post-process is used outside the context of a camera post-process chain (default: null) * @param needMipMaps True if mip maps need to be generated after render (default: false) * @param forceDepthStencil True to force post-process texture creation with stencil depth and buffer (default: false) */ resize(e, t, r = null, n = !1, i = !1) { this._textures.length > 0 && this._textures.reset(), this.width = e, this.height = t; let s = null; if (r) { for (let o = 0; o < r._postProcesses.length; o++) if (r._postProcesses[o] !== null) { s = r._postProcesses[o]; break; } } const a = { width: this.width, height: this.height }, f = { generateMipMaps: n, generateDepthBuffer: i || s === this, generateStencilBuffer: (i || s === this) && this._engine.isStencilEnable, samplingMode: this.renderTargetSamplingMode, type: this._textureType, format: this._textureFormat, samples: this._samples, label: "PostProcessRTT-" + this.name }; this._textures.push(this._createRenderTargetTexture(a, f, 0)), this._reusable && this._textures.push(this._createRenderTargetTexture(a, f, 1)), this._texelSize.copyFromFloats(1 / this.width, 1 / this.height), this.onSizeChangedObservable.notifyObservers(this); } _getTarget() { let e; if (this._shareOutputWithPostProcess) e = this._shareOutputWithPostProcess.inputTexture; else if (this._forcedOutputTexture) e = this._forcedOutputTexture, this.width = this._forcedOutputTexture.width, this.height = this._forcedOutputTexture.height; else { e = this.inputTexture; let t; for (let r = 0; r < this._textureCache.length; r++) if (this._textureCache[r].texture === e) { t = this._textureCache[r]; break; } t && (t.lastUsedRenderId = this._renderId); } return e; } /** * Activates the post process by intializing the textures to be used when executed. Notifies onActivateObservable. * When this post process is used in a pipeline, this is call will bind the input texture of this post process to the output of the previous. * @param camera The camera that will be used in the post process. This camera will be used when calling onActivateObservable. * @param sourceTexture The source texture to be inspected to get the width and height if not specified in the post process constructor. (default: null) * @param forceDepthStencil If true, a depth and stencil buffer will be generated. (default: false) * @returns The render target wrapper that was bound to be written to. */ activate(e, t = null, r) { var n, i; e = e || this._camera; const s = e.getScene(), a = s.getEngine(), f = a.getCaps().maxTextureSize, o = (t ? t.width : this._engine.getRenderWidth(!0)) * this._options | 0, d = (t ? t.height : this._engine.getRenderHeight(!0)) * this._options | 0; let v = this._options.width || o, u = this._options.height || d; const l = this.renderTargetSamplingMode !== 7 && this.renderTargetSamplingMode !== 1 && this.renderTargetSamplingMode !== 2; let P = null; if (!this._shareOutputWithPostProcess && !this._forcedOutputTexture) { if (this.adaptScaleToCurrentViewport) { const p = a.currentViewport; p && (v *= p.width, u *= p.height); } (l || this.alwaysForcePOT) && (this._options.width || (v = a.needPOTTextures ? Ge.GetExponentOfTwo(v, f, this.scaleMode) : v), this._options.height || (u = a.needPOTTextures ? Ge.GetExponentOfTwo(u, f, this.scaleMode) : u)), (this.width !== v || this.height !== u || !(P = this._getTarget())) && this.resize(v, u, e, l, r), this._textures.forEach((p) => { p.samples !== this.samples && this._engine.updateRenderTargetTextureSampleCount(p, this.samples); }), this._flushTextureCache(), this._renderId++; } return P || (P = this._getTarget()), this.enablePixelPerfectMode ? (this._scaleRatio.copyFromFloats(o / v, d / u), this._engine.bindFramebuffer(P, 0, o, d, this.forceFullscreenViewport)) : (this._scaleRatio.copyFromFloats(1, 1), this._engine.bindFramebuffer(P, 0, void 0, void 0, this.forceFullscreenViewport)), (i = (n = this._engine)._debugInsertMarker) === null || i === void 0 || i.call(n, `post process ${this.name} input`), this.onActivateObservable.notifyObservers(e), this.autoClear && (this.alphaMode === 0 || this.forceAutoClearInAlphaMode) && this._engine.clear(this.clearColor ? this.clearColor : s.clearColor, s._allowPostProcessClearColor, !0, !0), this._reusable && (this._currentRenderTextureInd = (this._currentRenderTextureInd + 1) % 2), P; } /** * If the post process is supported. */ get isSupported() { return this._drawWrapper.effect.isSupported; } /** * The aspect ratio of the output texture. */ get aspectRatio() { return this._shareOutputWithPostProcess ? this._shareOutputWithPostProcess.aspectRatio : this._forcedOutputTexture ? this._forcedOutputTexture.width / this._forcedOutputTexture.height : this.width / this.height; } /** * Get a value indicating if the post-process is ready to be used * @returns true if the post-process is ready (shader is compiled) */ isReady() { var e, t; return (t = (e = this._drawWrapper.effect) === null || e === void 0 ? void 0 : e.isReady()) !== null && t !== void 0 ? t : !1; } /** * Binds all textures and uniforms to the shader, this will be run on every pass. * @returns the effect corresponding to this post process. Null if not compiled or not ready. */ apply() { var e, t, r; if (!(!((e = this._drawWrapper.effect) === null || e === void 0) && e.isReady())) return null; this._engine.enableEffect(this._drawWrapper), this._engine.setState(!1), this._engine.setDepthBuffer(!1), this._engine.setDepthWrite(!1), this._engine.setAlphaMode(this.alphaMode), this.alphaConstants && this.getEngine().setAlphaConstants(this.alphaConstants.r, this.alphaConstants.g, this.alphaConstants.b, this.alphaConstants.a); let n; return this._shareOutputWithPostProcess ? n = this._shareOutputWithPostProcess.inputTexture : this._forcedOutputTexture ? n = this._forcedOutputTexture : n = this.inputTexture, this.externalTextureSamplerBinding || this._drawWrapper.effect._bindTexture("textureSampler", n == null ? void 0 : n.texture), this._drawWrapper.effect.setVector2("scale", this._scaleRatio), this.onApplyObservable.notifyObservers(this._drawWrapper.effect), (r = (t = kr._GetShaderCodeProcessing(this.name)) === null || t === void 0 ? void 0 : t.bindCustomBindings) === null || r === void 0 || r.call(t, this.name, this._drawWrapper.effect), this._drawWrapper.effect; } _disposeTextures() { if (this._shareOutputWithPostProcess || this._forcedOutputTexture) { this._disposeTextureCache(); return; } this._disposeTextureCache(), this._textures.dispose(); } _disposeTextureCache() { for (let e = this._textureCache.length - 1; e >= 0; e--) this._textureCache[e].texture.dispose(); this._textureCache.length = 0; } /** * Sets the required values to the prepass renderer. * @param prePassRenderer defines the prepass renderer to setup. * @returns true if the pre pass is needed. */ setPrePassRenderer(e) { return this._prePassEffectConfiguration ? (this._prePassEffectConfiguration = e.addEffectConfiguration(this._prePassEffectConfiguration), this._prePassEffectConfiguration.enabled = !0, !0) : !1; } /** * Disposes the post process. * @param camera The camera to dispose the post process on. */ dispose(e) { e = e || this._camera, this._disposeTextures(); let t; if (this._scene && (t = this._scene.postProcesses.indexOf(this), t !== -1 && this._scene.postProcesses.splice(t, 1)), this._parentContainer) { const r = this._parentContainer.postProcesses.indexOf(this); r > -1 && this._parentContainer.postProcesses.splice(r, 1), this._parentContainer = null; } if (t = this._engine.postProcesses.indexOf(this), t !== -1 && this._engine.postProcesses.splice(t, 1), !!e) { if (e.detachPostProcess(this), t = e._postProcesses.indexOf(this), t === 0 && e._postProcesses.length > 0) { const r = this._camera._getFirstPostProcess(); r && r.markTextureDirty(); } this.onActivateObservable.clear(), this.onAfterRenderObservable.clear(), this.onApplyObservable.clear(), this.onBeforeRenderObservable.clear(), this.onSizeChangedObservable.clear(); } } /** * Serializes the post process to a JSON object * @returns the JSON object */ serialize() { const e = jt.Serialize(this), t = this.getCamera() || this._scene && this._scene.activeCamera; return e.customType = "BABYLON." + this.getClassName(), e.cameraId = t ? t.id : null, e.reusable = this._reusable, e.textureType = this._textureType, e.fragmentUrl = this._fragmentUrl, e.parameters = this._parameters, e.samplers = this._samplers, e.options = this._options, e.defines = this._postProcessDefines, e.textureFormat = this._textureFormat, e.vertexUrl = this._vertexUrl, e.indexParameters = this._indexParameters, e; } /** * Clones this post process * @returns a new post process similar to this one */ clone() { const e = this.serialize(); e._engine = this._engine, e.cameraId = null; const t = kr.Parse(e, this._scene, ""); return t ? (t.onActivateObservable = this.onActivateObservable.clone(), t.onSizeChangedObservable = this.onSizeChangedObservable.clone(), t.onApplyObservable = this.onApplyObservable.clone(), t.onBeforeRenderObservable = this.onBeforeRenderObservable.clone(), t.onAfterRenderObservable = this.onAfterRenderObservable.clone(), t._prePassEffectConfiguration = this._prePassEffectConfiguration, t) : null; } /** * Creates a material from parsed material data * @param parsedPostProcess defines parsed post process data * @param scene defines the hosting scene * @param rootUrl defines the root URL to use to load textures * @returns a new post process */ static Parse(e, t, r) { const n = Jo(e.customType); if (!n || !n._Parse) return null; const i = t ? t.getCameraById(e.cameraId) : null; return n._Parse(e, i, t, r); } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new kr(e.name, e.fragmentUrl, e.parameters, e.samplers, e.options, t, e.renderTargetSamplingMode, e._engine, e.reusable, e.defines, e.textureType, e.vertexUrl, e.indexParameters, !1, e.textureFormat), e, r, n); } } kr._CustomShaderCodeProcessing = {}; C([ M() ], kr.prototype, "uniqueId", void 0); C([ M() ], kr.prototype, "name", void 0); C([ M() ], kr.prototype, "width", void 0); C([ M() ], kr.prototype, "height", void 0); C([ M() ], kr.prototype, "renderTargetSamplingMode", void 0); C([ rU() ], kr.prototype, "clearColor", void 0); C([ M() ], kr.prototype, "autoClear", void 0); C([ M() ], kr.prototype, "forceAutoClearInAlphaMode", void 0); C([ M() ], kr.prototype, "alphaMode", void 0); C([ M() ], kr.prototype, "alphaConstants", void 0); C([ M() ], kr.prototype, "enablePixelPerfectMode", void 0); C([ M() ], kr.prototype, "forceFullscreenViewport", void 0); C([ M() ], kr.prototype, "scaleMode", void 0); C([ M() ], kr.prototype, "alwaysForcePOT", void 0); C([ M("samples") ], kr.prototype, "_samples", void 0); C([ M() ], kr.prototype, "adaptScaleToCurrentViewport", void 0); Ue("BABYLON.PostProcess", kr); class BI extends Mr { /** * Create a new VectorMergerBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.xSwizzle = "x", this.ySwizzle = "y", this.zSwizzle = "z", this.wSwizzle = "w", this.registerInput("xyzw ", de.Vector4, !0), this.registerInput("xyz ", de.Vector3, !0), this.registerInput("xy ", de.Vector2, !0), this.registerInput("zw ", de.Vector2, !0), this.registerInput("x", de.Float, !0), this.registerInput("y", de.Float, !0), this.registerInput("z", de.Float, !0), this.registerInput("w", de.Float, !0), this.registerOutput("xyzw", de.Vector4), this.registerOutput("xyz", de.Vector3), this.registerOutput("xy", de.Vector2), this.registerOutput("zw", de.Vector2); } /** * Gets the current class name * @returns the class name */ getClassName() { return "VectorMergerBlock"; } /** * Gets the xyzw component (input) */ get xyzwIn() { return this._inputs[0]; } /** * Gets the xyz component (input) */ get xyzIn() { return this._inputs[1]; } /** * Gets the xy component (input) */ get xyIn() { return this._inputs[2]; } /** * Gets the zw component (input) */ get zwIn() { return this._inputs[3]; } /** * Gets the x component (input) */ get x() { return this._inputs[4]; } /** * Gets the y component (input) */ get y() { return this._inputs[5]; } /** * Gets the z component (input) */ get z() { return this._inputs[6]; } /** * Gets the w component (input) */ get w() { return this._inputs[7]; } /** * Gets the xyzw component (output) */ get xyzw() { return this._outputs[0]; } /** * Gets the xyz component (output) */ get xyzOut() { return this._outputs[1]; } /** * Gets the xy component (output) */ get xyOut() { return this._outputs[2]; } /** * Gets the zw component (output) */ get zwOut() { return this._outputs[3]; } /** * Gets the xy component (output) * @deprecated Please use xyOut instead. */ get xy() { return this.xyOut; } /** * Gets the xyz component (output) * @deprecated Please use xyzOut instead. */ get xyz() { return this.xyzOut; } _inputRename(e) { return e === "xyzw " ? "xyzwIn" : e === "xyz " ? "xyzIn" : e === "xy " ? "xyIn" : e === "zw " ? "zwIn" : e; } _buildSwizzle(e) { return "." + (this.xSwizzle + this.ySwizzle + this.zSwizzle + this.wSwizzle).substr(0, e); } _buildBlock(e) { super._buildBlock(e); const t = this.x, r = this.y, n = this.z, i = this.w, s = this.xyIn, a = this.zwIn, f = this.xyzIn, o = this.xyzwIn, d = this._outputs[0], v = this._outputs[1], u = this._outputs[2], l = this._outputs[3]; return o.isConnected ? (d.hasEndpoints && (e.compilationString += this._declareOutput(d, e) + ` = ${o.associatedVariableName}${this._buildSwizzle(4)}; `), v.hasEndpoints && (e.compilationString += this._declareOutput(v, e) + ` = ${o.associatedVariableName}${this._buildSwizzle(3)}; `), u.hasEndpoints && (e.compilationString += this._declareOutput(u, e) + ` = ${o.associatedVariableName}${this._buildSwizzle(2)}; `)) : f.isConnected ? (d.hasEndpoints && (e.compilationString += this._declareOutput(d, e) + ` = vec4(${f.associatedVariableName}, ${i.isConnected ? this._writeVariable(i) : "0.0"})${this._buildSwizzle(4)}; `), v.hasEndpoints && (e.compilationString += this._declareOutput(v, e) + ` = ${f.associatedVariableName}${this._buildSwizzle(3)}; `), u.hasEndpoints && (e.compilationString += this._declareOutput(u, e) + ` = ${f.associatedVariableName}${this._buildSwizzle(2)}; `)) : s.isConnected ? (d.hasEndpoints && (a.isConnected ? e.compilationString += this._declareOutput(d, e) + ` = vec4(${s.associatedVariableName}, ${a.associatedVariableName})${this._buildSwizzle(4)}; ` : e.compilationString += this._declareOutput(d, e) + ` = vec4(${s.associatedVariableName}, ${n.isConnected ? this._writeVariable(n) : "0.0"}, ${i.isConnected ? this._writeVariable(i) : "0.0"})${this._buildSwizzle(4)}; `), v.hasEndpoints && (e.compilationString += this._declareOutput(v, e) + ` = vec3(${s.associatedVariableName}, ${n.isConnected ? this._writeVariable(n) : "0.0"})${this._buildSwizzle(3)}; `), u.hasEndpoints && (e.compilationString += this._declareOutput(u, e) + ` = ${s.associatedVariableName}${this._buildSwizzle(2)}; `), l.hasEndpoints && (a.isConnected ? e.compilationString += this._declareOutput(l, e) + ` = ${a.associatedVariableName}${this._buildSwizzle(2)}; ` : e.compilationString += this._declareOutput(l, e) + ` = vec2(${n.isConnected ? this._writeVariable(n) : "0.0"}, ${i.isConnected ? this._writeVariable(i) : "0.0"})${this._buildSwizzle(2)}; `)) : (d.hasEndpoints && (a.isConnected ? e.compilationString += this._declareOutput(d, e) + ` = vec4(${t.isConnected ? this._writeVariable(t) : "0.0"}, ${r.isConnected ? this._writeVariable(r) : "0.0"}, ${a.associatedVariableName})${this._buildSwizzle(4)}; ` : e.compilationString += this._declareOutput(d, e) + ` = vec4(${t.isConnected ? this._writeVariable(t) : "0.0"}, ${r.isConnected ? this._writeVariable(r) : "0.0"}, ${n.isConnected ? this._writeVariable(n) : "0.0"}, ${i.isConnected ? this._writeVariable(i) : "0.0"})${this._buildSwizzle(4)}; `), v.hasEndpoints && (e.compilationString += this._declareOutput(v, e) + ` = vec3(${t.isConnected ? this._writeVariable(t) : "0.0"}, ${r.isConnected ? this._writeVariable(r) : "0.0"}, ${n.isConnected ? this._writeVariable(n) : "0.0"})${this._buildSwizzle(3)}; `), u.hasEndpoints && (e.compilationString += this._declareOutput(u, e) + ` = vec2(${t.isConnected ? this._writeVariable(t) : "0.0"}, ${r.isConnected ? this._writeVariable(r) : "0.0"})${this._buildSwizzle(2)}; `), l.hasEndpoints && (a.isConnected ? e.compilationString += this._declareOutput(l, e) + ` = ${a.associatedVariableName}${this._buildSwizzle(2)}; ` : e.compilationString += this._declareOutput(l, e) + ` = vec2(${n.isConnected ? this._writeVariable(n) : "0.0"}, ${i.isConnected ? this._writeVariable(i) : "0.0"})${this._buildSwizzle(2)}; `)), this; } serialize() { const e = super.serialize(); return e.xSwizzle = this.xSwizzle, e.ySwizzle = this.ySwizzle, e.zSwizzle = this.zSwizzle, e.wSwizzle = this.wSwizzle, e; } _deserialize(e, t, r) { var n, i, s, a; super._deserialize(e, t, r), this.xSwizzle = (n = e.xSwizzle) !== null && n !== void 0 ? n : "x", this.ySwizzle = (i = e.ySwizzle) !== null && i !== void 0 ? i : "y", this.zSwizzle = (s = e.zSwizzle) !== null && s !== void 0 ? s : "z", this.wSwizzle = (a = e.wSwizzle) !== null && a !== void 0 ? a : "w"; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return e += `${this._codeVariableName}.xSwizzle = "${this.xSwizzle}"; `, e += `${this._codeVariableName}.ySwizzle = "${this.ySwizzle}"; `, e += `${this._codeVariableName}.zSwizzle = "${this.zSwizzle}"; `, e += `${this._codeVariableName}.wSwizzle = "${this.wSwizzle}"; `, e; } } Ue("BABYLON.VectorMergerBlock", BI); class jR extends Mr { /** * Creates a new RemapBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.sourceRange = new at(-1, 1), this.targetRange = new at(0, 1), this.registerInput("input", de.AutoDetect), this.registerInput("sourceMin", de.Float, !0), this.registerInput("sourceMax", de.Float, !0), this.registerInput("targetMin", de.Float, !0), this.registerInput("targetMax", de.Float, !0), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0]; } /** * Gets the current class name * @returns the class name */ getClassName() { return "RemapBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the source min input component */ get sourceMin() { return this._inputs[1]; } /** * Gets the source max input component */ get sourceMax() { return this._inputs[2]; } /** * Gets the target min input component */ get targetMin() { return this._inputs[3]; } /** * Gets the target max input component */ get targetMax() { return this._inputs[4]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0], r = this.sourceMin.isConnected ? this.sourceMin.associatedVariableName : this._writeFloat(this.sourceRange.x), n = this.sourceMax.isConnected ? this.sourceMax.associatedVariableName : this._writeFloat(this.sourceRange.y), i = this.targetMin.isConnected ? this.targetMin.associatedVariableName : this._writeFloat(this.targetRange.x), s = this.targetMax.isConnected ? this.targetMax.associatedVariableName : this._writeFloat(this.targetRange.y); return e.compilationString += this._declareOutput(t, e) + ` = ${i} + (${this._inputs[0].associatedVariableName} - ${r}) * (${s} - ${i}) / (${n} - ${r}); `, this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode() + `${this._codeVariableName}.sourceRange = new BABYLON.Vector2(${this.sourceRange.x}, ${this.sourceRange.y}); `; return e += `${this._codeVariableName}.targetRange = new BABYLON.Vector2(${this.targetRange.x}, ${this.targetRange.y}); `, e; } serialize() { const e = super.serialize(); return e.sourceRange = this.sourceRange.asArray(), e.targetRange = this.targetRange.asArray(), e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.sourceRange = at.FromArray(e.sourceRange), this.targetRange = at.FromArray(e.targetRange); } } C([ rn("From", Gr.Vector2) ], jR.prototype, "sourceRange", void 0); C([ rn("To", Gr.Vector2) ], jR.prototype, "targetRange", void 0); Ue("BABYLON.RemapBlock", jR); class zC extends Mr { /** * Creates a new MultiplyBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("left", de.AutoDetect), this.registerInput("right", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._linkConnectionTypes(0, 1), this._inputs[0].acceptedConnectionPointTypes.push(de.Float), this._inputs[1].acceptedConnectionPointTypes.push(de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "MultiplyBlock"; } /** * Gets the left operand input component */ get left() { return this._inputs[0]; } /** * Gets the right operand input component */ get right() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = ${this.left.associatedVariableName} * ${this.right.associatedVariableName}; `, this; } } Ue("BABYLON.MultiplyBlock", zC); var Vv; (function(A) { A[A.Material = 0] = "Material", A[A.PostProcess = 1] = "PostProcess", A[A.Particle = 2] = "Particle", A[A.ProceduralTexture = 3] = "ProceduralTexture"; })(Vv || (Vv = {})); class AD { /** * Creates a new instance BoxParticleEmitter */ constructor() { this.direction1 = new S(0, 1, 0), this.direction2 = new S(0, 1, 0), this.minEmitBox = new S(-0.5, -0.5, -0.5), this.maxEmitBox = new S(0.5, 0.5, 0.5); } /** * Called by the particle System when the direction is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param directionToUpdate is the direction vector to update with the result * @param particle is the particle we are computed the direction for * @param isLocal defines if the direction should be set in local space */ startDirectionFunction(e, t, r, n) { const i = Xt.RandomRange(this.direction1.x, this.direction2.x), s = Xt.RandomRange(this.direction1.y, this.direction2.y), a = Xt.RandomRange(this.direction1.z, this.direction2.z); if (n) { t.x = i, t.y = s, t.z = a; return; } S.TransformNormalFromFloatsToRef(i, s, a, e, t); } /** * Called by the particle System when the position is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param positionToUpdate is the position vector to update with the result * @param particle is the particle we are computed the position for * @param isLocal defines if the position should be set in local space */ startPositionFunction(e, t, r, n) { const i = Xt.RandomRange(this.minEmitBox.x, this.maxEmitBox.x), s = Xt.RandomRange(this.minEmitBox.y, this.maxEmitBox.y), a = Xt.RandomRange(this.minEmitBox.z, this.maxEmitBox.z); if (n) { t.x = i, t.y = s, t.z = a; return; } S.TransformCoordinatesFromFloatsToRef(i, s, a, e, t); } /** * Clones the current emitter and returns a copy of it * @returns the new emitter */ clone() { const e = new AD(); return sA.DeepCopy(this, e), e; } /** * Called by the GPUParticleSystem to setup the update shader * @param uboOrEffect defines the update shader */ applyToShader(e) { e.setVector3("direction1", this.direction1), e.setVector3("direction2", this.direction2), e.setVector3("minEmitBox", this.minEmitBox), e.setVector3("maxEmitBox", this.maxEmitBox); } /** * Creates the structure of the ubo for this particle emitter * @param ubo ubo to create the structure for */ buildUniformLayout(e) { e.addUniform("direction1", 3), e.addUniform("direction2", 3), e.addUniform("minEmitBox", 3), e.addUniform("maxEmitBox", 3); } /** * Returns a string to use to update the GPU particles update shader * @returns a string containing the defines string */ getEffectDefines() { return "#define BOXEMITTER"; } /** * Returns the string "BoxParticleEmitter" * @returns a string containing the class name */ getClassName() { return "BoxParticleEmitter"; } /** * Serializes the particle system to a JSON object. * @returns the JSON object */ serialize() { const e = {}; return e.type = this.getClassName(), e.direction1 = this.direction1.asArray(), e.direction2 = this.direction2.asArray(), e.minEmitBox = this.minEmitBox.asArray(), e.maxEmitBox = this.maxEmitBox.asArray(), e; } /** * Parse properties from a JSON object * @param serializationObject defines the JSON object */ parse(e) { S.FromArrayToRef(e.direction1, 0, this.direction1), S.FromArrayToRef(e.direction2, 0, this.direction2), S.FromArrayToRef(e.minEmitBox, 0, this.minEmitBox), S.FromArrayToRef(e.maxEmitBox, 0, this.maxEmitBox); } } class wR { /** * Gets or sets the radius of the emission cone */ get radius() { return this._radius; } set radius(e) { this._radius = e, this._buildHeight(); } /** * Gets or sets the angle of the emission cone */ get angle() { return this._angle; } set angle(e) { this._angle = e, this._buildHeight(); } _buildHeight() { this._angle !== 0 ? this._height = this._radius / Math.tan(this._angle / 2) : this._height = 1; } /** * Creates a new instance ConeParticleEmitter * @param radius the radius of the emission cone (1 by default) * @param angle the cone base angle (PI by default) * @param directionRandomizer defines how much to randomize the particle direction [0-1] (default is 0) */ constructor(e = 1, t = Math.PI, r = 0) { this.directionRandomizer = r, this.radiusRange = 1, this.heightRange = 1, this.emitFromSpawnPointOnly = !1, this.angle = t, this.radius = e; } /** * Called by the particle System when the direction is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param directionToUpdate is the direction vector to update with the result * @param particle is the particle we are computed the direction for * @param isLocal defines if the direction should be set in local space */ startDirectionFunction(e, t, r, n) { n ? ue.Vector3[0].copyFrom(r._localPosition).normalize() : r.position.subtractToRef(e.getTranslation(), ue.Vector3[0]).normalize(); const i = Xt.RandomRange(0, this.directionRandomizer), s = Xt.RandomRange(0, this.directionRandomizer), a = Xt.RandomRange(0, this.directionRandomizer); t.x = ue.Vector3[0].x + i, t.y = ue.Vector3[0].y + s, t.z = ue.Vector3[0].z + a, t.normalize(); } /** * Called by the particle System when the position is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param positionToUpdate is the position vector to update with the result * @param particle is the particle we are computed the position for * @param isLocal defines if the position should be set in local space */ startPositionFunction(e, t, r, n) { const i = Xt.RandomRange(0, Math.PI * 2); let s; this.emitFromSpawnPointOnly ? s = 1e-4 : (s = Xt.RandomRange(0, this.heightRange), s = 1 - s * s); let a = this._radius - Xt.RandomRange(0, this._radius * this.radiusRange); a = a * s; const f = a * Math.sin(i), o = a * Math.cos(i), d = s * this._height; if (n) { t.x = f, t.y = d, t.z = o; return; } S.TransformCoordinatesFromFloatsToRef(f, d, o, e, t); } /** * Clones the current emitter and returns a copy of it * @returns the new emitter */ clone() { const e = new wR(this._radius, this._angle, this.directionRandomizer); return sA.DeepCopy(this, e), e; } /** * Called by the GPUParticleSystem to setup the update shader * @param uboOrEffect defines the update shader */ applyToShader(e) { e.setFloat2("radius", this._radius, this.radiusRange), e.setFloat("coneAngle", this._angle), e.setFloat2("height", this._height, this.heightRange), e.setFloat("directionRandomizer", this.directionRandomizer); } /** * Creates the structure of the ubo for this particle emitter * @param ubo ubo to create the structure for */ buildUniformLayout(e) { e.addUniform("radius", 2), e.addUniform("coneAngle", 1), e.addUniform("height", 2), e.addUniform("directionRandomizer", 1); } /** * Returns a string to use to update the GPU particles update shader * @returns a string containing the defines string */ getEffectDefines() { let e = "#define CONEEMITTER"; return this.emitFromSpawnPointOnly && (e += ` #define CONEEMITTERSPAWNPOINT`), e; } /** * Returns the string "ConeParticleEmitter" * @returns a string containing the class name */ getClassName() { return "ConeParticleEmitter"; } /** * Serializes the particle system to a JSON object. * @returns the JSON object */ serialize() { const e = {}; return e.type = this.getClassName(), e.radius = this._radius, e.angle = this._angle, e.directionRandomizer = this.directionRandomizer, e.radiusRange = this.radiusRange, e.heightRange = this.heightRange, e.emitFromSpawnPointOnly = this.emitFromSpawnPointOnly, e; } /** * Parse properties from a JSON object * @param serializationObject defines the JSON object */ parse(e) { this.radius = e.radius, this.angle = e.angle, this.directionRandomizer = e.directionRandomizer, this.radiusRange = e.radiusRange !== void 0 ? e.radiusRange : 1, this.heightRange = e.radiusRange !== void 0 ? e.heightRange : 1, this.emitFromSpawnPointOnly = e.emitFromSpawnPointOnly !== void 0 ? e.emitFromSpawnPointOnly : !1; } } class aU { /** * Creates a new instance CylinderParticleEmitter * @param radius the radius of the emission cylinder (1 by default) * @param height the height of the emission cylinder (1 by default) * @param radiusRange the range of the emission cylinder [0-1] 0 Surface only, 1 Entire Radius (1 by default) * @param directionRandomizer defines how much to randomize the particle direction [0-1] */ constructor(e = 1, t = 1, r = 1, n = 0) { this.radius = e, this.height = t, this.radiusRange = r, this.directionRandomizer = n, this._tempVector = S.Zero(); } /** * Called by the particle System when the direction is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param directionToUpdate is the direction vector to update with the result * @param particle is the particle we are computed the direction for * @param isLocal defines if the direction should be set in local space * @param inverseWorldMatrix defines the inverted world matrix to use if isLocal is false */ startDirectionFunction(e, t, r, n, i) { r.position.subtractToRef(e.getTranslation(), this._tempVector), this._tempVector.normalize(), S.TransformNormalToRef(this._tempVector, i, this._tempVector); const s = Xt.RandomRange(-this.directionRandomizer / 2, this.directionRandomizer / 2); let a = Math.atan2(this._tempVector.x, this._tempVector.z); if (a += Xt.RandomRange(-Math.PI / 2, Math.PI / 2) * this.directionRandomizer, this._tempVector.y = s, this._tempVector.x = Math.sin(a), this._tempVector.z = Math.cos(a), this._tempVector.normalize(), n) { t.copyFrom(this._tempVector); return; } S.TransformNormalFromFloatsToRef(this._tempVector.x, this._tempVector.y, this._tempVector.z, e, t); } /** * Called by the particle System when the position is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param positionToUpdate is the position vector to update with the result * @param particle is the particle we are computed the position for * @param isLocal defines if the position should be set in local space */ startPositionFunction(e, t, r, n) { const i = Xt.RandomRange(-this.height / 2, this.height / 2), s = Xt.RandomRange(0, 2 * Math.PI), a = Xt.RandomRange((1 - this.radiusRange) * (1 - this.radiusRange), 1), f = Math.sqrt(a) * this.radius, o = f * Math.cos(s), d = f * Math.sin(s); if (n) { t.copyFromFloats(o, i, d); return; } S.TransformCoordinatesFromFloatsToRef(o, i, d, e, t); } /** * Clones the current emitter and returns a copy of it * @returns the new emitter */ clone() { const e = new aU(this.radius, this.directionRandomizer); return sA.DeepCopy(this, e), e; } /** * Called by the GPUParticleSystem to setup the update shader * @param uboOrEffect defines the update shader */ applyToShader(e) { e.setFloat("radius", this.radius), e.setFloat("height", this.height), e.setFloat("radiusRange", this.radiusRange), e.setFloat("directionRandomizer", this.directionRandomizer); } /** * Creates the structure of the ubo for this particle emitter * @param ubo ubo to create the structure for */ buildUniformLayout(e) { e.addUniform("radius", 1), e.addUniform("height", 1), e.addUniform("radiusRange", 1), e.addUniform("directionRandomizer", 1); } /** * Returns a string to use to update the GPU particles update shader * @returns a string containing the defines string */ getEffectDefines() { return "#define CYLINDEREMITTER"; } /** * Returns the string "CylinderParticleEmitter" * @returns a string containing the class name */ getClassName() { return "CylinderParticleEmitter"; } /** * Serializes the particle system to a JSON object. * @returns the JSON object */ serialize() { const e = {}; return e.type = this.getClassName(), e.radius = this.radius, e.height = this.height, e.radiusRange = this.radiusRange, e.directionRandomizer = this.directionRandomizer, e; } /** * Parse properties from a JSON object * @param serializationObject defines the JSON object */ parse(e) { this.radius = e.radius, this.height = e.height, this.radiusRange = e.radiusRange, this.directionRandomizer = e.directionRandomizer; } } class mR extends aU { /** * Creates a new instance CylinderDirectedParticleEmitter * @param radius the radius of the emission cylinder (1 by default) * @param height the height of the emission cylinder (1 by default) * @param radiusRange the range of the emission cylinder [0-1] 0 Surface only, 1 Entire Radius (1 by default) * @param direction1 the min limit of the emission direction (up vector by default) * @param direction2 the max limit of the emission direction (up vector by default) */ constructor(e = 1, t = 1, r = 1, n = new S(0, 1, 0), i = new S(0, 1, 0)) { super(e, t, r), this.direction1 = n, this.direction2 = i; } /** * Called by the particle System when the direction is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param directionToUpdate is the direction vector to update with the result */ startDirectionFunction(e, t) { const r = Xt.RandomRange(this.direction1.x, this.direction2.x), n = Xt.RandomRange(this.direction1.y, this.direction2.y), i = Xt.RandomRange(this.direction1.z, this.direction2.z); S.TransformNormalFromFloatsToRef(r, n, i, e, t); } /** * Clones the current emitter and returns a copy of it * @returns the new emitter */ clone() { const e = new mR(this.radius, this.height, this.radiusRange, this.direction1, this.direction2); return sA.DeepCopy(this, e), e; } /** * Called by the GPUParticleSystem to setup the update shader * @param uboOrEffect defines the update shader */ applyToShader(e) { e.setFloat("radius", this.radius), e.setFloat("height", this.height), e.setFloat("radiusRange", this.radiusRange), e.setVector3("direction1", this.direction1), e.setVector3("direction2", this.direction2); } /** * Creates the structure of the ubo for this particle emitter * @param ubo ubo to create the structure for */ buildUniformLayout(e) { e.addUniform("radius", 1), e.addUniform("height", 1), e.addUniform("radiusRange", 1), e.addUniform("direction1", 3), e.addUniform("direction2", 3); } /** * Returns a string to use to update the GPU particles update shader * @returns a string containing the defines string */ getEffectDefines() { return `#define CYLINDEREMITTER #define DIRECTEDCYLINDEREMITTER`; } /** * Returns the string "CylinderDirectedParticleEmitter" * @returns a string containing the class name */ getClassName() { return "CylinderDirectedParticleEmitter"; } /** * Serializes the particle system to a JSON object. * @returns the JSON object */ serialize() { const e = super.serialize(); return e.direction1 = this.direction1.asArray(), e.direction2 = this.direction2.asArray(), e; } /** * Parse properties from a JSON object * @param serializationObject defines the JSON object */ parse(e) { super.parse(e), this.direction1.copyFrom(e.direction1), this.direction2.copyFrom(e.direction2); } } class BR { /** * Creates a new instance HemisphericParticleEmitter * @param radius the radius of the emission hemisphere (1 by default) * @param radiusRange the range of the emission hemisphere [0-1] 0 Surface only, 1 Entire Radius (1 by default) * @param directionRandomizer defines how much to randomize the particle direction [0-1] */ constructor(e = 1, t = 1, r = 0) { this.radius = e, this.radiusRange = t, this.directionRandomizer = r; } /** * Called by the particle System when the direction is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param directionToUpdate is the direction vector to update with the result * @param particle is the particle we are computed the direction for * @param isLocal defines if the direction should be set in local space */ startDirectionFunction(e, t, r, n) { const i = r.position.subtract(e.getTranslation()).normalize(), s = Xt.RandomRange(0, this.directionRandomizer), a = Xt.RandomRange(0, this.directionRandomizer), f = Xt.RandomRange(0, this.directionRandomizer); if (i.x += s, i.y += a, i.z += f, i.normalize(), n) { t.copyFrom(i); return; } S.TransformNormalFromFloatsToRef(i.x, i.y, i.z, e, t); } /** * Called by the particle System when the position is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param positionToUpdate is the position vector to update with the result * @param particle is the particle we are computed the position for * @param isLocal defines if the position should be set in local space */ startPositionFunction(e, t, r, n) { const i = this.radius - Xt.RandomRange(0, this.radius * this.radiusRange), s = Xt.RandomRange(0, 1), a = Xt.RandomRange(0, 2 * Math.PI), f = Math.acos(2 * s - 1), o = i * Math.cos(a) * Math.sin(f), d = i * Math.cos(f), v = i * Math.sin(a) * Math.sin(f); if (n) { t.copyFromFloats(o, Math.abs(d), v); return; } S.TransformCoordinatesFromFloatsToRef(o, Math.abs(d), v, e, t); } /** * Clones the current emitter and returns a copy of it * @returns the new emitter */ clone() { const e = new BR(this.radius, this.directionRandomizer); return sA.DeepCopy(this, e), e; } /** * Called by the GPUParticleSystem to setup the update shader * @param uboOrEffect defines the update shader */ applyToShader(e) { e.setFloat("radius", this.radius), e.setFloat("radiusRange", this.radiusRange), e.setFloat("directionRandomizer", this.directionRandomizer); } /** * Creates the structure of the ubo for this particle emitter * @param ubo ubo to create the structure for */ buildUniformLayout(e) { e.addUniform("radius", 1), e.addUniform("radiusRange", 1), e.addUniform("directionRandomizer", 1); } /** * Returns a string to use to update the GPU particles update shader * @returns a string containing the defines string */ getEffectDefines() { return "#define HEMISPHERICEMITTER"; } /** * Returns the string "HemisphericParticleEmitter" * @returns a string containing the class name */ getClassName() { return "HemisphericParticleEmitter"; } /** * Serializes the particle system to a JSON object. * @returns the JSON object */ serialize() { const e = {}; return e.type = this.getClassName(), e.radius = this.radius, e.radiusRange = this.radiusRange, e.directionRandomizer = this.directionRandomizer, e; } /** * Parse properties from a JSON object * @param serializationObject defines the JSON object */ parse(e) { this.radius = e.radius, this.radiusRange = e.radiusRange, this.directionRandomizer = e.directionRandomizer; } } class WR { /** * Creates a new instance PointParticleEmitter */ constructor() { this.direction1 = new S(0, 1, 0), this.direction2 = new S(0, 1, 0); } /** * Called by the particle System when the direction is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param directionToUpdate is the direction vector to update with the result * @param particle is the particle we are computed the direction for * @param isLocal defines if the direction should be set in local space */ startDirectionFunction(e, t, r, n) { const i = Xt.RandomRange(this.direction1.x, this.direction2.x), s = Xt.RandomRange(this.direction1.y, this.direction2.y), a = Xt.RandomRange(this.direction1.z, this.direction2.z); if (n) { t.copyFromFloats(i, s, a); return; } S.TransformNormalFromFloatsToRef(i, s, a, e, t); } /** * Called by the particle System when the position is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param positionToUpdate is the position vector to update with the result * @param particle is the particle we are computed the position for * @param isLocal defines if the position should be set in local space */ startPositionFunction(e, t, r, n) { if (n) { t.copyFromFloats(0, 0, 0); return; } S.TransformCoordinatesFromFloatsToRef(0, 0, 0, e, t); } /** * Clones the current emitter and returns a copy of it * @returns the new emitter */ clone() { const e = new WR(); return sA.DeepCopy(this, e), e; } /** * Called by the GPUParticleSystem to setup the update shader * @param uboOrEffect defines the update shader */ applyToShader(e) { e.setVector3("direction1", this.direction1), e.setVector3("direction2", this.direction2); } /** * Creates the structure of the ubo for this particle emitter * @param ubo ubo to create the structure for */ buildUniformLayout(e) { e.addUniform("direction1", 3), e.addUniform("direction2", 3); } /** * Returns a string to use to update the GPU particles update shader * @returns a string containing the defines string */ getEffectDefines() { return "#define POINTEMITTER"; } /** * Returns the string "PointParticleEmitter" * @returns a string containing the class name */ getClassName() { return "PointParticleEmitter"; } /** * Serializes the particle system to a JSON object. * @returns the JSON object */ serialize() { const e = {}; return e.type = this.getClassName(), e.direction1 = this.direction1.asArray(), e.direction2 = this.direction2.asArray(), e; } /** * Parse properties from a JSON object * @param serializationObject defines the JSON object */ parse(e) { S.FromArrayToRef(e.direction1, 0, this.direction1), S.FromArrayToRef(e.direction2, 0, this.direction2); } } class oU { /** * Creates a new instance SphereParticleEmitter * @param radius the radius of the emission sphere (1 by default) * @param radiusRange the range of the emission sphere [0-1] 0 Surface only, 1 Entire Radius (1 by default) * @param directionRandomizer defines how much to randomize the particle direction [0-1] */ constructor(e = 1, t = 1, r = 0) { this.radius = e, this.radiusRange = t, this.directionRandomizer = r; } /** * Called by the particle System when the direction is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param directionToUpdate is the direction vector to update with the result * @param particle is the particle we are computed the direction for * @param isLocal defines if the direction should be set in local space */ startDirectionFunction(e, t, r, n) { const i = r.position.subtract(e.getTranslation()).normalize(), s = Xt.RandomRange(0, this.directionRandomizer), a = Xt.RandomRange(0, this.directionRandomizer), f = Xt.RandomRange(0, this.directionRandomizer); if (i.x += s, i.y += a, i.z += f, i.normalize(), n) { t.copyFrom(i); return; } S.TransformNormalFromFloatsToRef(i.x, i.y, i.z, e, t); } /** * Called by the particle System when the position is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param positionToUpdate is the position vector to update with the result * @param particle is the particle we are computed the position for * @param isLocal defines if the position should be set in local space */ startPositionFunction(e, t, r, n) { const i = this.radius - Xt.RandomRange(0, this.radius * this.radiusRange), s = Xt.RandomRange(0, 1), a = Xt.RandomRange(0, 2 * Math.PI), f = Math.acos(2 * s - 1), o = i * Math.cos(a) * Math.sin(f), d = i * Math.cos(f), v = i * Math.sin(a) * Math.sin(f); if (n) { t.copyFromFloats(o, d, v); return; } S.TransformCoordinatesFromFloatsToRef(o, d, v, e, t); } /** * Clones the current emitter and returns a copy of it * @returns the new emitter */ clone() { const e = new oU(this.radius, this.directionRandomizer); return sA.DeepCopy(this, e), e; } /** * Called by the GPUParticleSystem to setup the update shader * @param uboOrEffect defines the update shader */ applyToShader(e) { e.setFloat("radius", this.radius), e.setFloat("radiusRange", this.radiusRange), e.setFloat("directionRandomizer", this.directionRandomizer); } /** * Creates the structure of the ubo for this particle emitter * @param ubo ubo to create the structure for */ buildUniformLayout(e) { e.addUniform("radius", 1), e.addUniform("radiusRange", 1), e.addUniform("directionRandomizer", 1); } /** * Returns a string to use to update the GPU particles update shader * @returns a string containing the defines string */ getEffectDefines() { return "#define SPHEREEMITTER"; } /** * Returns the string "SphereParticleEmitter" * @returns a string containing the class name */ getClassName() { return "SphereParticleEmitter"; } /** * Serializes the particle system to a JSON object. * @returns the JSON object */ serialize() { const e = {}; return e.type = this.getClassName(), e.radius = this.radius, e.radiusRange = this.radiusRange, e.directionRandomizer = this.directionRandomizer, e; } /** * Parse properties from a JSON object * @param serializationObject defines the JSON object */ parse(e) { this.radius = e.radius, this.radiusRange = e.radiusRange, this.directionRandomizer = e.directionRandomizer; } } class SR extends oU { /** * Creates a new instance SphereDirectedParticleEmitter * @param radius the radius of the emission sphere (1 by default) * @param direction1 the min limit of the emission direction (up vector by default) * @param direction2 the max limit of the emission direction (up vector by default) */ constructor(e = 1, t = new S(0, 1, 0), r = new S(0, 1, 0)) { super(e), this.direction1 = t, this.direction2 = r; } /** * Called by the particle System when the direction is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param directionToUpdate is the direction vector to update with the result */ startDirectionFunction(e, t) { const r = Xt.RandomRange(this.direction1.x, this.direction2.x), n = Xt.RandomRange(this.direction1.y, this.direction2.y), i = Xt.RandomRange(this.direction1.z, this.direction2.z); S.TransformNormalFromFloatsToRef(r, n, i, e, t); } /** * Clones the current emitter and returns a copy of it * @returns the new emitter */ clone() { const e = new SR(this.radius, this.direction1, this.direction2); return sA.DeepCopy(this, e), e; } /** * Called by the GPUParticleSystem to setup the update shader * @param uboOrEffect defines the update shader */ applyToShader(e) { e.setFloat("radius", this.radius), e.setFloat("radiusRange", this.radiusRange), e.setVector3("direction1", this.direction1), e.setVector3("direction2", this.direction2); } /** * Creates the structure of the ubo for this particle emitter * @param ubo ubo to create the structure for */ buildUniformLayout(e) { e.addUniform("radius", 1), e.addUniform("radiusRange", 1), e.addUniform("direction1", 3), e.addUniform("direction2", 3); } /** * Returns a string to use to update the GPU particles update shader * @returns a string containing the defines string */ getEffectDefines() { return `#define SPHEREEMITTER #define DIRECTEDSPHEREEMITTER`; } /** * Returns the string "SphereDirectedParticleEmitter" * @returns a string containing the class name */ getClassName() { return "SphereDirectedParticleEmitter"; } /** * Serializes the particle system to a JSON object. * @returns the JSON object */ serialize() { const e = super.serialize(); return e.direction1 = this.direction1.asArray(), e.direction2 = this.direction2.asArray(), e; } /** * Parse properties from a JSON object * @param serializationObject defines the JSON object */ parse(e) { super.parse(e), this.direction1.copyFrom(e.direction1), this.direction2.copyFrom(e.direction2); } } class dD { /** * Creates a new instance CustomParticleEmitter */ constructor() { this.particlePositionGenerator = () => { }, this.particleDestinationGenerator = () => { }; } /** * Called by the particle System when the direction is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param directionToUpdate is the direction vector to update with the result * @param particle is the particle we are computed the direction for * @param isLocal defines if the direction should be set in local space */ startDirectionFunction(e, t, r, n) { const i = ue.Vector3[0]; if (this.particleDestinationGenerator) { this.particleDestinationGenerator(-1, r, i); const s = ue.Vector3[1]; i.subtractToRef(r.position, s), s.scaleToRef(1 / r.lifeTime, i); } else i.set(0, 0, 0); if (n) { t.copyFrom(i); return; } S.TransformNormalToRef(i, e, t); } /** * Called by the particle System when the position is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param positionToUpdate is the position vector to update with the result * @param particle is the particle we are computed the position for * @param isLocal defines if the position should be set in local space */ startPositionFunction(e, t, r, n) { const i = ue.Vector3[0]; if (this.particlePositionGenerator ? this.particlePositionGenerator(-1, r, i) : i.set(0, 0, 0), n) { t.copyFrom(i); return; } S.TransformCoordinatesToRef(i, e, t); } /** * Clones the current emitter and returns a copy of it * @returns the new emitter */ clone() { const e = new dD(); return sA.DeepCopy(this, e), e; } /** * Called by the GPUParticleSystem to setup the update shader * @param uboOrEffect defines the update shader */ // eslint-disable-next-line @typescript-eslint/no-unused-vars applyToShader(e) { } /** * Creates the structure of the ubo for this particle emitter * @param ubo ubo to create the structure for */ // eslint-disable-next-line @typescript-eslint/no-unused-vars buildUniformLayout(e) { } /** * Returns a string to use to update the GPU particles update shader * @returns a string containing the defines string */ getEffectDefines() { return "#define CUSTOMEMITTER"; } /** * Returns the string "PointParticleEmitter" * @returns a string containing the class name */ getClassName() { return "CustomParticleEmitter"; } /** * Serializes the particle system to a JSON object. * @returns the JSON object */ serialize() { const e = {}; return e.type = this.getClassName(), e; } /** * Parse properties from a JSON object * @param serializationObject defines the JSON object */ // eslint-disable-next-line @typescript-eslint/no-unused-vars parse(e) { } } class IO { /** Defines the mesh to use as source */ get mesh() { return this._mesh; } set mesh(e) { this._mesh !== e && (this._mesh = e, e ? (this._indices = e.getIndices(), this._positions = e.getVerticesData(J.PositionKind), this._normals = e.getVerticesData(J.NormalKind)) : (this._indices = null, this._positions = null, this._normals = null)); } /** * Creates a new instance MeshParticleEmitter * @param mesh defines the mesh to use as source */ constructor(e = null) { this._indices = null, this._positions = null, this._normals = null, this._storedNormal = S.Zero(), this._mesh = null, this.direction1 = new S(0, 1, 0), this.direction2 = new S(0, 1, 0), this.useMeshNormalsForDirection = !0, this.mesh = e; } /** * Called by the particle System when the direction is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param directionToUpdate is the direction vector to update with the result * @param particle is the particle we are computed the direction for * @param isLocal defines if the direction should be set in local space */ startDirectionFunction(e, t, r, n) { if (this.useMeshNormalsForDirection && this._normals) { S.TransformNormalToRef(this._storedNormal, e, t); return; } const i = Xt.RandomRange(this.direction1.x, this.direction2.x), s = Xt.RandomRange(this.direction1.y, this.direction2.y), a = Xt.RandomRange(this.direction1.z, this.direction2.z); if (n) { t.copyFromFloats(i, s, a); return; } S.TransformNormalFromFloatsToRef(i, s, a, e, t); } /** * Called by the particle System when the position is computed for the created particle. * @param worldMatrix is the world matrix of the particle system * @param positionToUpdate is the position vector to update with the result * @param particle is the particle we are computed the position for * @param isLocal defines if the position should be set in local space */ startPositionFunction(e, t, r, n) { if (!this._indices || !this._positions) return; const i = 3 * Math.random() * (this._indices.length / 3) | 0, s = Math.random(), a = Math.random() * (1 - s), f = 1 - s - a, o = this._indices[i], d = this._indices[i + 1], v = this._indices[i + 2], u = ue.Vector3[0], l = ue.Vector3[1], P = ue.Vector3[2], p = ue.Vector3[3]; S.FromArrayToRef(this._positions, o * 3, u), S.FromArrayToRef(this._positions, d * 3, l), S.FromArrayToRef(this._positions, v * 3, P), p.x = s * u.x + a * l.x + f * P.x, p.y = s * u.y + a * l.y + f * P.y, p.z = s * u.z + a * l.z + f * P.z, n ? t.copyFromFloats(p.x, p.y, p.z) : S.TransformCoordinatesFromFloatsToRef(p.x, p.y, p.z, e, t), this.useMeshNormalsForDirection && this._normals && (S.FromArrayToRef(this._normals, o * 3, u), S.FromArrayToRef(this._normals, d * 3, l), S.FromArrayToRef(this._normals, v * 3, P), this._storedNormal.x = s * u.x + a * l.x + f * P.x, this._storedNormal.y = s * u.y + a * l.y + f * P.y, this._storedNormal.z = s * u.z + a * l.z + f * P.z); } /** * Clones the current emitter and returns a copy of it * @returns the new emitter */ clone() { const e = new IO(this.mesh); return sA.DeepCopy(this, e), e; } /** * Called by the GPUParticleSystem to setup the update shader * @param uboOrEffect defines the update shader */ applyToShader(e) { e.setVector3("direction1", this.direction1), e.setVector3("direction2", this.direction2); } /** * Creates the structure of the ubo for this particle emitter * @param ubo ubo to create the structure for */ buildUniformLayout(e) { e.addUniform("direction1", 3), e.addUniform("direction2", 3); } /** * Returns a string to use to update the GPU particles update shader * @returns a string containing the defines string */ getEffectDefines() { return ""; } /** * Returns the string "BoxParticleEmitter" * @returns a string containing the class name */ getClassName() { return "MeshParticleEmitter"; } /** * Serializes the particle system to a JSON object. * @returns the JSON object */ serialize() { var e; const t = {}; return t.type = this.getClassName(), t.direction1 = this.direction1.asArray(), t.direction2 = this.direction2.asArray(), t.meshId = (e = this.mesh) === null || e === void 0 ? void 0 : e.id, t.useMeshNormalsForDirection = this.useMeshNormalsForDirection, t; } /** * Parse properties from a JSON object * @param serializationObject defines the JSON object * @param scene defines the hosting scene */ parse(e, t) { S.FromArrayToRef(e.direction1, 0, this.direction1), S.FromArrayToRef(e.direction2, 0, this.direction2), e.meshId && t && (this.mesh = t.getLastMeshById(e.meshId)), this.useMeshNormalsForDirection = e.useMeshNormalsForDirection; } } class N0 { /** * Gets or sets a texture used to add random noise to particle positions */ get noiseTexture() { return this._noiseTexture; } set noiseTexture(e) { this._noiseTexture !== e && (this._noiseTexture = e, this._reset()); } /** * Gets or sets whether an animation sprite sheet is enabled or not on the particle system */ get isAnimationSheetEnabled() { return this._isAnimationSheetEnabled; } set isAnimationSheetEnabled(e) { this._isAnimationSheetEnabled != e && (this._isAnimationSheetEnabled = e, this._reset()); } /** * Gets or sets a boolean enabling the use of logarithmic depth buffers, which is good for wide depth buffers. */ get useLogarithmicDepth() { return this._useLogarithmicDepth; } set useLogarithmicDepth(e) { this._useLogarithmicDepth = e && this.getScene().getEngine().getCaps().fragmentDepthSupported; } /** * Get hosting scene * @returns the scene */ getScene() { return this._scene; } _hasTargetStopDurationDependantGradient() { return this._startSizeGradients && this._startSizeGradients.length > 0 || this._emitRateGradients && this._emitRateGradients.length > 0 || this._lifeTimeGradients && this._lifeTimeGradients.length > 0; } /** * Gets the current list of drag gradients. * You must use addDragGradient and removeDragGradient to update this list * @returns the list of drag gradients */ getDragGradients() { return this._dragGradients; } /** * Gets the current list of limit velocity gradients. * You must use addLimitVelocityGradient and removeLimitVelocityGradient to update this list * @returns the list of limit velocity gradients */ getLimitVelocityGradients() { return this._limitVelocityGradients; } /** * Gets the current list of color gradients. * You must use addColorGradient and removeColorGradient to update this list * @returns the list of color gradients */ getColorGradients() { return this._colorGradients; } /** * Gets the current list of size gradients. * You must use addSizeGradient and removeSizeGradient to update this list * @returns the list of size gradients */ getSizeGradients() { return this._sizeGradients; } /** * Gets the current list of color remap gradients. * You must use addColorRemapGradient and removeColorRemapGradient to update this list * @returns the list of color remap gradients */ getColorRemapGradients() { return this._colorRemapGradients; } /** * Gets the current list of alpha remap gradients. * You must use addAlphaRemapGradient and removeAlphaRemapGradient to update this list * @returns the list of alpha remap gradients */ getAlphaRemapGradients() { return this._alphaRemapGradients; } /** * Gets the current list of life time gradients. * You must use addLifeTimeGradient and removeLifeTimeGradient to update this list * @returns the list of life time gradients */ getLifeTimeGradients() { return this._lifeTimeGradients; } /** * Gets the current list of angular speed gradients. * You must use addAngularSpeedGradient and removeAngularSpeedGradient to update this list * @returns the list of angular speed gradients */ getAngularSpeedGradients() { return this._angularSpeedGradients; } /** * Gets the current list of velocity gradients. * You must use addVelocityGradient and removeVelocityGradient to update this list * @returns the list of velocity gradients */ getVelocityGradients() { return this._velocityGradients; } /** * Gets the current list of start size gradients. * You must use addStartSizeGradient and removeStartSizeGradient to update this list * @returns the list of start size gradients */ getStartSizeGradients() { return this._startSizeGradients; } /** * Gets the current list of emit rate gradients. * You must use addEmitRateGradient and removeEmitRateGradient to update this list * @returns the list of emit rate gradients */ getEmitRateGradients() { return this._emitRateGradients; } /** * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors. * This only works when particleEmitterTyps is a BoxParticleEmitter */ get direction1() { return this.particleEmitterType.direction1 ? this.particleEmitterType.direction1 : S.Zero(); } set direction1(e) { this.particleEmitterType.direction1 && (this.particleEmitterType.direction1 = e); } /** * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors. * This only works when particleEmitterTyps is a BoxParticleEmitter */ get direction2() { return this.particleEmitterType.direction2 ? this.particleEmitterType.direction2 : S.Zero(); } set direction2(e) { this.particleEmitterType.direction2 && (this.particleEmitterType.direction2 = e); } /** * Minimum box point around our emitter. Our emitter is the center of particles source, but if you want your particles to emit from more than one point, then you can tell it to do so. * This only works when particleEmitterTyps is a BoxParticleEmitter */ get minEmitBox() { return this.particleEmitterType.minEmitBox ? this.particleEmitterType.minEmitBox : S.Zero(); } set minEmitBox(e) { this.particleEmitterType.minEmitBox && (this.particleEmitterType.minEmitBox = e); } /** * Maximum box point around our emitter. Our emitter is the center of particles source, but if you want your particles to emit from more than one point, then you can tell it to do so. * This only works when particleEmitterTyps is a BoxParticleEmitter */ get maxEmitBox() { return this.particleEmitterType.maxEmitBox ? this.particleEmitterType.maxEmitBox : S.Zero(); } set maxEmitBox(e) { this.particleEmitterType.maxEmitBox && (this.particleEmitterType.maxEmitBox = e); } /** * Gets or sets the billboard mode to use when isBillboardBased = true. * Value can be: ParticleSystem.BILLBOARDMODE_ALL, ParticleSystem.BILLBOARDMODE_Y, ParticleSystem.BILLBOARDMODE_STRETCHED */ get billboardMode() { return this._billboardMode; } set billboardMode(e) { this._billboardMode !== e && (this._billboardMode = e, this._reset()); } /** * Gets or sets a boolean indicating if the particles must be rendered as billboard or aligned with the direction */ get isBillboardBased() { return this._isBillboardBased; } set isBillboardBased(e) { this._isBillboardBased !== e && (this._isBillboardBased = e, this._reset()); } /** * Gets the image processing configuration used either in this material. */ get imageProcessingConfiguration() { return this._imageProcessingConfiguration; } /** * Sets the Default image processing configuration used either in the this material. * * If sets to null, the scene one is in use. */ set imageProcessingConfiguration(e) { this._attachImageProcessingConfiguration(e); } /** * Attaches a new image processing configuration to the Standard Material. * @param configuration */ _attachImageProcessingConfiguration(e) { e !== this._imageProcessingConfiguration && (!e && this._scene ? this._imageProcessingConfiguration = this._scene.imageProcessingConfiguration : this._imageProcessingConfiguration = e); } /** @internal */ _reset() { } /** * @internal */ _removeGradientAndTexture(e, t, r) { if (!t) return this; let n = 0; for (const i of t) { if (i.gradient === e) { t.splice(n, 1); break; } n++; } return r && r.dispose(), this; } /** * Instantiates a particle system. * Particles are often small sprites used to simulate hard-to-reproduce phenomena like fire, smoke, water, or abstract visual effects like magic glitter and faery dust. * @param name The name of the particle system */ constructor(e) { this.animations = [], this.renderingGroupId = 0, this.emitter = S.Zero(), this.emitRate = 10, this.manualEmitCount = -1, this.updateSpeed = 0.01, this.targetStopDuration = 0, this.disposeOnStop = !1, this.minEmitPower = 1, this.maxEmitPower = 1, this.minLifeTime = 1, this.maxLifeTime = 1, this.minSize = 1, this.maxSize = 1, this.minScaleX = 1, this.maxScaleX = 1, this.minScaleY = 1, this.maxScaleY = 1, this.minInitialRotation = 0, this.maxInitialRotation = 0, this.minAngularSpeed = 0, this.maxAngularSpeed = 0, this.layerMask = 268435455, this.customShader = null, this.preventAutoStart = !1, this._wasDispatched = !1, this._rootUrl = "", this.noiseStrength = new S(10, 10, 10), this.onAnimationEnd = null, this.blendMode = N0.BLENDMODE_ONEONE, this.forceDepthWrite = !1, this.preWarmCycles = 0, this.preWarmStepOffset = 1, this.spriteCellChangeSpeed = 1, this.startSpriteCellID = 0, this.endSpriteCellID = 0, this.spriteCellWidth = 0, this.spriteCellHeight = 0, this.spriteCellLoop = !0, this.spriteRandomStartCell = !1, this.translationPivot = new at(0, 0), this.beginAnimationOnStart = !1, this.beginAnimationFrom = 0, this.beginAnimationTo = 60, this.beginAnimationLoop = !1, this.worldOffset = new S(0, 0, 0), this._useLogarithmicDepth = !1, this.gravity = S.Zero(), this._colorGradients = null, this._sizeGradients = null, this._lifeTimeGradients = null, this._angularSpeedGradients = null, this._velocityGradients = null, this._limitVelocityGradients = null, this._dragGradients = null, this._emitRateGradients = null, this._startSizeGradients = null, this._rampGradients = null, this._colorRemapGradients = null, this._alphaRemapGradients = null, this.startDelay = 0, this.limitVelocityDamping = 0.4, this.color1 = new xt(1, 1, 1, 1), this.color2 = new xt(1, 1, 1, 1), this.colorDead = new xt(0, 0, 0, 1), this.textureMask = new xt(1, 1, 1, 1), this._isSubEmitter = !1, this._billboardMode = 7, this._isBillboardBased = !0, this._imageProcessingConfigurationDefines = new M$(), this.id = e, this.name = e; } /** * Creates a Point Emitter for the particle system (emits directly from the emitter position) * @param direction1 Particles are emitted between the direction1 and direction2 from within the box * @param direction2 Particles are emitted between the direction1 and direction2 from within the box * @returns the emitter */ createPointEmitter(e, t) { const r = new WR(); return r.direction1 = e, r.direction2 = t, this.particleEmitterType = r, r; } /** * Creates a Hemisphere Emitter for the particle system (emits along the hemisphere radius) * @param radius The radius of the hemisphere to emit from * @param radiusRange The range of the hemisphere to emit from [0-1] 0 Surface Only, 1 Entire Radius * @returns the emitter */ createHemisphericEmitter(e = 1, t = 1) { const r = new BR(e, t); return this.particleEmitterType = r, r; } /** * Creates a Sphere Emitter for the particle system (emits along the sphere radius) * @param radius The radius of the sphere to emit from * @param radiusRange The range of the sphere to emit from [0-1] 0 Surface Only, 1 Entire Radius * @returns the emitter */ createSphereEmitter(e = 1, t = 1) { const r = new oU(e, t); return this.particleEmitterType = r, r; } /** * Creates a Directed Sphere Emitter for the particle system (emits between direction1 and direction2) * @param radius The radius of the sphere to emit from * @param direction1 Particles are emitted between the direction1 and direction2 from within the sphere * @param direction2 Particles are emitted between the direction1 and direction2 from within the sphere * @returns the emitter */ createDirectedSphereEmitter(e = 1, t = new S(0, 1, 0), r = new S(0, 1, 0)) { const n = new SR(e, t, r); return this.particleEmitterType = n, n; } /** * Creates a Cylinder Emitter for the particle system (emits from the cylinder to the particle position) * @param radius The radius of the emission cylinder * @param height The height of the emission cylinder * @param radiusRange The range of emission [0-1] 0 Surface only, 1 Entire Radius * @param directionRandomizer How much to randomize the particle direction [0-1] * @returns the emitter */ createCylinderEmitter(e = 1, t = 1, r = 1, n = 0) { const i = new aU(e, t, r, n); return this.particleEmitterType = i, i; } /** * Creates a Directed Cylinder Emitter for the particle system (emits between direction1 and direction2) * @param radius The radius of the cylinder to emit from * @param height The height of the emission cylinder * @param radiusRange the range of the emission cylinder [0-1] 0 Surface only, 1 Entire Radius (1 by default) * @param direction1 Particles are emitted between the direction1 and direction2 from within the cylinder * @param direction2 Particles are emitted between the direction1 and direction2 from within the cylinder * @returns the emitter */ createDirectedCylinderEmitter(e = 1, t = 1, r = 1, n = new S(0, 1, 0), i = new S(0, 1, 0)) { const s = new mR(e, t, r, n, i); return this.particleEmitterType = s, s; } /** * Creates a Cone Emitter for the particle system (emits from the cone to the particle position) * @param radius The radius of the cone to emit from * @param angle The base angle of the cone * @returns the emitter */ createConeEmitter(e = 1, t = Math.PI / 4) { const r = new wR(e, t); return this.particleEmitterType = r, r; } /** * Creates a Box Emitter for the particle system. (emits between direction1 and direction2 from withing the box defined by minEmitBox and maxEmitBox) * @param direction1 Particles are emitted between the direction1 and direction2 from within the box * @param direction2 Particles are emitted between the direction1 and direction2 from within the box * @param minEmitBox Particles are emitted from the box between minEmitBox and maxEmitBox * @param maxEmitBox Particles are emitted from the box between minEmitBox and maxEmitBox * @returns the emitter */ createBoxEmitter(e, t, r, n) { const i = new AD(); return this.particleEmitterType = i, this.direction1 = e, this.direction2 = t, this.minEmitBox = r, this.maxEmitBox = n, i; } } N0.BLENDMODE_ONEONE = 0; N0.BLENDMODE_STANDARD = 1; N0.BLENDMODE_ADD = 2; N0.BLENDMODE_MULTIPLY = 3; N0.BLENDMODE_MULTIPLYADD = 4; class vQ extends Mr { /** * Create a new ColorSplitterBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("rgba", de.Color4, !0), this.registerInput("rgb ", de.Color3, !0), this.registerOutput("rgb", de.Color3), this.registerOutput("r", de.Float), this.registerOutput("g", de.Float), this.registerOutput("b", de.Float), this.registerOutput("a", de.Float), this.inputsAreExclusive = !0; } /** * Gets the current class name * @returns the class name */ getClassName() { return "ColorSplitterBlock"; } /** * Gets the rgba component (input) */ get rgba() { return this._inputs[0]; } /** * Gets the rgb component (input) */ get rgbIn() { return this._inputs[1]; } /** * Gets the rgb component (output) */ get rgbOut() { return this._outputs[0]; } /** * Gets the r component (output) */ get r() { return this._outputs[1]; } /** * Gets the g component (output) */ get g() { return this._outputs[2]; } /** * Gets the b component (output) */ get b() { return this._outputs[3]; } /** * Gets the a component (output) */ get a() { return this._outputs[4]; } _inputRename(e) { return e === "rgb " ? "rgbIn" : e; } _outputRename(e) { return e === "rgb" ? "rgbOut" : e; } _buildBlock(e) { super._buildBlock(e); const t = this.rgba.isConnected ? this.rgba : this.rgbIn; if (!t.isConnected) return; const r = this._outputs[0], n = this._outputs[1], i = this._outputs[2], s = this._outputs[3], a = this._outputs[4]; return r.hasEndpoints && (e.compilationString += this._declareOutput(r, e) + ` = ${t.associatedVariableName}.rgb; `), n.hasEndpoints && (e.compilationString += this._declareOutput(n, e) + ` = ${t.associatedVariableName}.r; `), i.hasEndpoints && (e.compilationString += this._declareOutput(i, e) + ` = ${t.associatedVariableName}.g; `), s.hasEndpoints && (e.compilationString += this._declareOutput(s, e) + ` = ${t.associatedVariableName}.b; `), a.hasEndpoints && (e.compilationString += this._declareOutput(a, e) + ` = ${t.associatedVariableName}.a; `), this; } } Ue("BABYLON.ColorSplitterBlock", vQ); hr.prototype.createRenderTargetCubeTexture = function(A, e) { const t = this._createHardwareRenderTargetWrapper(!1, !0, A), r = Object.assign({ generateMipMaps: !0, generateDepthBuffer: !0, generateStencilBuffer: !1, type: 0, samplingMode: 3, format: 5 }, e); r.generateStencilBuffer = r.generateDepthBuffer && r.generateStencilBuffer, (r.type === 1 && !this._caps.textureFloatLinearFiltering || r.type === 2 && !this._caps.textureHalfFloatLinearFiltering) && (r.samplingMode = 1); const n = this._gl, i = new As(this, ri.RenderTarget); this._bindTextureDirectly(n.TEXTURE_CUBE_MAP, i, !0); const s = this._getSamplingParameters(r.samplingMode, r.generateMipMaps); r.type === 1 && !this._caps.textureFloat && (r.type = 0, Se.Warn("Float textures are not supported. Cube render target forced to TEXTURETYPE_UNESIGNED_BYTE type")), n.texParameteri(n.TEXTURE_CUBE_MAP, n.TEXTURE_MAG_FILTER, s.mag), n.texParameteri(n.TEXTURE_CUBE_MAP, n.TEXTURE_MIN_FILTER, s.min), n.texParameteri(n.TEXTURE_CUBE_MAP, n.TEXTURE_WRAP_S, n.CLAMP_TO_EDGE), n.texParameteri(n.TEXTURE_CUBE_MAP, n.TEXTURE_WRAP_T, n.CLAMP_TO_EDGE); for (let f = 0; f < 6; f++) n.texImage2D(n.TEXTURE_CUBE_MAP_POSITIVE_X + f, 0, this._getRGBABufferInternalSizedFormat(r.type, r.format), A, A, 0, this._getInternalFormat(r.format), this._getWebGLTextureType(r.type), null); const a = n.createFramebuffer(); return this._bindUnboundFramebuffer(a), t._depthStencilBuffer = this._setupFramebufferDepthAttachments(r.generateStencilBuffer, r.generateDepthBuffer, A, A), r.generateMipMaps && n.generateMipmap(n.TEXTURE_CUBE_MAP), this._bindTextureDirectly(n.TEXTURE_CUBE_MAP, null), this._bindUnboundFramebuffer(null), t._framebuffer = a, t._generateDepthBuffer = r.generateDepthBuffer, t._generateStencilBuffer = r.generateStencilBuffer, i.width = A, i.height = A, i.isReady = !0, i.isCube = !0, i.samples = 1, i.generateMipMaps = r.generateMipMaps, i.samplingMode = r.samplingMode, i.type = r.type, i.format = r.format, this._internalTexturesCache.push(i), t.setTextures(i), t; }; const gE = { positions: [1, 1, -1, 1, -1, -1, 1, -1], indices: [0, 1, 2, 0, 2, 3] }; class fU { /** * Creates an effect renderer * @param engine the engine to use for rendering * @param options defines the options of the effect renderer */ constructor(e, t = gE) { var r, n; this._fullscreenViewport = new WA(0, 0, 1, 1); const i = (r = t.positions) !== null && r !== void 0 ? r : gE.positions, s = (n = t.indices) !== null && n !== void 0 ? n : gE.indices; this.engine = e, this._vertexBuffers = { [J.PositionKind]: new J(e, i, J.PositionKind, !1, !1, 2) }, this._indexBuffer = e.createIndexBuffer(s), this._onContextRestoredObserver = e.onContextRestoredObservable.add(() => { this._indexBuffer = e.createIndexBuffer(s); for (const a in this._vertexBuffers) this._vertexBuffers[a]._rebuild(); }); } /** * Sets the current viewport in normalized coordinates 0-1 * @param viewport Defines the viewport to set (defaults to 0 0 1 1) */ setViewport(e = this._fullscreenViewport) { this.engine.setViewport(e); } /** * Binds the embedded attributes buffer to the effect. * @param effect Defines the effect to bind the attributes for */ bindBuffers(e) { this.engine.bindBuffers(this._vertexBuffers, this._indexBuffer, e); } /** * Sets the current effect wrapper to use during draw. * The effect needs to be ready before calling this api. * This also sets the default full screen position attribute. * @param effectWrapper Defines the effect to draw with */ applyEffectWrapper(e) { this.engine.setState(!0), this.engine.depthCullingState.depthTest = !1, this.engine.stencilState.stencilTest = !1, this.engine.enableEffect(e._drawWrapper), this.bindBuffers(e.effect), e.onApplyObservable.notifyObservers({}); } /** * Saves engine states */ saveStates() { this._savedStateDepthTest = this.engine.depthCullingState.depthTest, this._savedStateStencilTest = this.engine.stencilState.stencilTest; } /** * Restores engine states */ restoreStates() { this.engine.depthCullingState.depthTest = this._savedStateDepthTest, this.engine.stencilState.stencilTest = this._savedStateStencilTest; } /** * Draws a full screen quad. */ draw() { this.engine.drawElementsType(0, 0, 6); } _isRenderTargetTexture(e) { return e.renderTarget !== void 0; } /** * renders one or more effects to a specified texture * @param effectWrapper the effect to renderer * @param outputTexture texture to draw to, if null it will render to the screen. */ render(e, t = null) { if (!e.effect.isReady()) return; this.saveStates(), this.setViewport(); const r = t === null ? null : this._isRenderTargetTexture(t) ? t.renderTarget : t; r && this.engine.bindFramebuffer(r), this.applyEffectWrapper(e), this.draw(), r && this.engine.unBindFramebuffer(r), this.restoreStates(); } /** * Disposes of the effect renderer */ dispose() { const e = this._vertexBuffers[J.PositionKind]; e && (e.dispose(), delete this._vertexBuffers[J.PositionKind]), this._indexBuffer && this.engine._releaseBuffer(this._indexBuffer), this._onContextRestoredObserver && (this.engine.onContextRestoredObservable.remove(this._onContextRestoredObserver), this._onContextRestoredObserver = null); } } class ng { /** * The underlying effect */ get effect() { return this._drawWrapper.effect; } set effect(e) { this._drawWrapper.effect = e; } /** * Creates an effect to be renderer * @param creationOptions options to create the effect */ constructor(e) { this.onApplyObservable = new Oe(); let t; const r = e.uniformNames || []; e.vertexShader ? t = { fragmentSource: e.fragmentShader, vertexSource: e.vertexShader, spectorName: e.name || "effectWrapper" } : (r.push("scale"), t = { fragmentSource: e.fragmentShader, vertex: "postprocess", spectorName: e.name || "effectWrapper" }, this.onApplyObservable.add(() => { this.effect.setFloat2("scale", 1, 1); })); const n = e.defines ? e.defines.join(` `) : ""; this._drawWrapper = new zo(e.engine), e.useShaderStore ? (t.fragment = t.fragmentSource, t.vertex || (t.vertex = t.vertexSource), delete t.fragmentSource, delete t.vertexSource, this.effect = e.engine.createEffect(t, e.attributeNames || ["position"], r, e.samplerNames, n, void 0, e.onCompiled, void 0, void 0, e.shaderLanguage)) : (this.effect = new An(t, e.attributeNames || ["position"], r, e.samplerNames, e.engine, n, void 0, e.onCompiled, void 0, void 0, void 0, e.shaderLanguage), this._onContextRestoredObserver = e.engine.onContextRestoredObservable.add(() => { this.effect._pipelineContext = null, this.effect._wasPreviouslyReady = !1, this.effect._prepareEffect(); })); } /** * Disposes of the effect wrapper */ dispose() { this._onContextRestoredObserver && (this.effect.getEngine().onContextRestoredObservable.remove(this._onContextRestoredObserver), this._onContextRestoredObserver = null), this.effect.dispose(); } } const gee = "passPixelShader", Xee = `varying vec2 vUV;uniform sampler2D textureSampler; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {gl_FragColor=texture2D(textureSampler,vUV);}`; Le.ShadersStore[gee] = Xee; const cG = { name: gee, shader: Xee }; class eA { static _CreateDumpRenderer() { if (!eA._DumpToolsEngine) { let e, t = null; const r = { preserveDrawingBuffer: !0, depth: !1, stencil: !1, alpha: !0, premultipliedAlpha: !1, antialias: !1, failIfMajorPerformanceCaveat: !1 }; try { e = new OffscreenCanvas(100, 100), t = new hr(e, !1, r); } catch { e = document.createElement("canvas"), t = new hr(e, !1, r); } t.getCaps().parallelShaderCompile = void 0; const n = new fU(t), i = new ng({ engine: t, name: cG.name, fragmentShader: cG.shader, samplerNames: ["textureSampler"] }); eA._DumpToolsEngine = { canvas: e, engine: t, renderer: n, wrapper: i }; } return eA._DumpToolsEngine; } /** * Dumps the current bound framebuffer * @param width defines the rendering width * @param height defines the rendering height * @param engine defines the hosting engine * @param successCallback defines the callback triggered once the data are available * @param mimeType defines the mime type of the result * @param fileName defines the filename to download. If present, the result will automatically be downloaded * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. * @returns a void promise */ static async DumpFramebuffer(e, t, r, n, i = "image/png", s, a) { const f = await r.readPixels(0, 0, e, t), o = new Uint8Array(f.buffer); eA.DumpData(e, t, o, n, i, s, !0, void 0, a); } /** * Dumps an array buffer * @param width defines the rendering width * @param height defines the rendering height * @param data the data array * @param mimeType defines the mime type of the result * @param fileName defines the filename to download. If present, the result will automatically be downloaded * @param invertY true to invert the picture in the Y dimension * @param toArrayBuffer true to convert the data to an ArrayBuffer (encoded as `mimeType`) instead of a base64 string * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. * @returns a promise that resolve to the final data */ static DumpDataAsync(e, t, r, n = "image/png", i, s = !1, a = !1, f) { return new Promise((o) => { eA.DumpData(e, t, r, (d) => o(d), n, i, s, a, f); }); } /** * Dumps an array buffer * @param width defines the rendering width * @param height defines the rendering height * @param data the data array * @param successCallback defines the callback triggered once the data are available * @param mimeType defines the mime type of the result * @param fileName defines the filename to download. If present, the result will automatically be downloaded * @param invertY true to invert the picture in the Y dimension * @param toArrayBuffer true to convert the data to an ArrayBuffer (encoded as `mimeType`) instead of a base64 string * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. */ static DumpData(e, t, r, n, i = "image/png", s, a = !1, f = !1, o) { const d = eA._CreateDumpRenderer(); if (d.engine.setSize(e, t, !0), r instanceof Float32Array) { const u = new Uint8Array(r.length); let l = r.length; for (; l--; ) { const P = r[l]; u[l] = Math.round(Xt.Clamp(P) * 255); } r = u; } const v = d.engine.createRawTexture(r, e, t, 5, !1, !a, 1); d.renderer.setViewport(), d.renderer.applyEffectWrapper(d.wrapper), d.wrapper.effect._bindTexture("textureSampler", v), d.renderer.draw(), f ? ye.ToBlob(d.canvas, (u) => { const l = new FileReader(); l.onload = (P) => { const p = P.target.result; n && n(p); }, l.readAsArrayBuffer(u); }, i, o) : ye.EncodeScreenshotCanvasData(d.canvas, n, i, s, o), v.dispose(); } /** * Dispose the dump tools associated resources */ static Dispose() { eA._DumpToolsEngine && (eA._DumpToolsEngine.wrapper.dispose(), eA._DumpToolsEngine.renderer.dispose(), eA._DumpToolsEngine.engine.dispose()), eA._DumpToolsEngine = null; } } const V6e = () => { ye.DumpData = eA.DumpData, ye.DumpDataAsync = eA.DumpDataAsync, ye.DumpFramebuffer = eA.DumpFramebuffer; }; V6e(); class Ta extends We { /** * Use this list to define the list of mesh you want to render. */ get renderList() { return this._renderList; } set renderList(e) { this._unObserveRenderList && (this._unObserveRenderList(), this._unObserveRenderList = null), e && (this._unObserveRenderList = p$(e, this._renderListHasChanged)), this._renderList = e; } /** * Post-processes for this render target */ get postProcesses() { return this._postProcesses; } get _prePassEnabled() { return !!this._prePassRenderTarget && this._prePassRenderTarget.enabled; } /** * Set a after unbind callback in the texture. * This has been kept for backward compatibility and use of onAfterUnbindObservable is recommended. */ set onAfterUnbind(e) { this._onAfterUnbindObserver && this.onAfterUnbindObservable.remove(this._onAfterUnbindObserver), this._onAfterUnbindObserver = this.onAfterUnbindObservable.add(e); } /** * Set a before render callback in the texture. * This has been kept for backward compatibility and use of onBeforeRenderObservable is recommended. */ set onBeforeRender(e) { this._onBeforeRenderObserver && this.onBeforeRenderObservable.remove(this._onBeforeRenderObserver), this._onBeforeRenderObserver = this.onBeforeRenderObservable.add(e); } /** * Set a after render callback in the texture. * This has been kept for backward compatibility and use of onAfterRenderObservable is recommended. */ set onAfterRender(e) { this._onAfterRenderObserver && this.onAfterRenderObservable.remove(this._onAfterRenderObserver), this._onAfterRenderObserver = this.onAfterRenderObservable.add(e); } /** * Set a clear callback in the texture. * This has been kept for backward compatibility and use of onClearObservable is recommended. */ set onClear(e) { this._onClearObserver && this.onClearObservable.remove(this._onClearObserver), this._onClearObserver = this.onClearObservable.add(e); } /** * Gets the render pass ids used by the render target texture. For a single render target the array length will be 1, for a cube texture it will be 6 and for * a 2D texture array it will return an array of ids the size of the 2D texture array */ get renderPassIds() { return this._renderPassIds; } /** * Gets the current value of the refreshId counter */ get currentRefreshId() { return this._currentRefreshId; } /** * Sets a specific material to be used to render a mesh/a list of meshes in this render target texture * @param mesh mesh or array of meshes * @param material material or array of materials to use for this render pass. If undefined is passed, no specific material will be used but the regular material instead (mesh.material). It's possible to provide an array of materials to use a different material for each rendering in the case of a cube texture (6 rendering) and a 2D texture array (as many rendering as the length of the array) */ setMaterialForRendering(e, t) { let r; Array.isArray(e) ? r = e : r = [e]; for (let n = 0; n < r.length; ++n) for (let i = 0; i < this._renderPassIds.length; ++i) r[n].setMaterialForRenderPass(this._renderPassIds[i], t !== void 0 ? Array.isArray(t) ? t[i] : t : void 0); } /** * Define if the texture has multiple draw buffers or if false a single draw buffer. */ get isMulti() { var e, t; return (t = (e = this._renderTarget) === null || e === void 0 ? void 0 : e.isMulti) !== null && t !== void 0 ? t : !1; } /** * Gets render target creation options that were used. */ get renderTargetOptions() { return this._renderTargetOptions; } /** * Gets the render target wrapper associated with this render target */ get renderTarget() { return this._renderTarget; } _onRatioRescale() { this._sizeRatio && this.resize(this._initialSizeParameter); } /** * Gets or sets the size of the bounding box associated with the texture (when in cube mode) * When defined, the cubemap will switch to local mode * @see https://community.arm.com/graphics/b/blog/posts/reflections-based-on-local-cubemaps-in-unity * @example https://www.babylonjs-playground.com/#RNASML */ set boundingBoxSize(e) { if (this._boundingBoxSize && this._boundingBoxSize.equals(e)) return; this._boundingBoxSize = e; const t = this.getScene(); t && t.markAllMaterialsAsDirty(1); } get boundingBoxSize() { return this._boundingBoxSize; } /** * In case the RTT has been created with a depth texture, get the associated * depth texture. * Otherwise, return null. */ get depthStencilTexture() { var e, t; return (t = (e = this._renderTarget) === null || e === void 0 ? void 0 : e._depthStencilTexture) !== null && t !== void 0 ? t : null; } /** @internal */ constructor(e, t, r, n = !1, i = !0, s = 0, a = !1, f = We.TRILINEAR_SAMPLINGMODE, o = !0, d = !1, v = !1, u = 5, l = !1, P, p, c = !1, H = !1) { var T, q, b, j, w, m, I; let N, k = !0; if (typeof n == "object") { const y = n; n = !!y.generateMipMaps, i = (T = y.doNotChangeAspectRatio) !== null && T !== void 0 ? T : !0, s = (q = y.type) !== null && q !== void 0 ? q : 0, a = !!y.isCube, f = (b = y.samplingMode) !== null && b !== void 0 ? b : We.TRILINEAR_SAMPLINGMODE, o = (j = y.generateDepthBuffer) !== null && j !== void 0 ? j : !0, d = !!y.generateStencilBuffer, v = !!y.isMulti, u = (w = y.format) !== null && w !== void 0 ? w : 5, l = !!y.delayAllocation, P = y.samples, p = y.creationFlags, c = !!y.noColorAttachment, H = !!y.useSRGBBuffer, N = y.colorAttachment, k = (m = y.gammaSpace) !== null && m !== void 0 ? m : k; } if (super(null, r, !n, void 0, f, void 0, void 0, void 0, void 0, u), this._unObserveRenderList = null, this._renderListHasChanged = (y, O) => { var Y; const ee = this._renderList ? this._renderList.length : 0; (O === 0 && ee > 0 || ee === 0) && ((Y = this.getScene()) === null || Y === void 0 || Y.meshes.forEach((Z) => { Z._markSubMeshesAsLightDirty(); })); }, this.renderParticles = !0, this.renderSprites = !1, this.forceLayerMaskCheck = !1, this.ignoreCameraViewport = !1, this.onBeforeBindObservable = new Oe(), this.onAfterUnbindObservable = new Oe(), this.onBeforeRenderObservable = new Oe(), this.onAfterRenderObservable = new Oe(), this.onClearObservable = new Oe(), this.onResizeObservable = new Oe(), this._cleared = !1, this.skipInitialClear = !1, this._currentRefreshId = -1, this._refreshRate = 1, this._samples = 1, this._canRescale = !0, this._renderTarget = null, this.boundingBoxPosition = S.Zero(), r = this.getScene(), !r) return; const R = this.getScene().getEngine(); this._gammaSpace = k, this._coordinatesMode = We.PROJECTION_MODE, this.renderList = [], this.name = e, this.isRenderTarget = !0, this._initialSizeParameter = t, this._renderPassIds = [], this._isCubeData = a, this._processSizeParameter(t), this.renderPassId = this._renderPassIds[0], this._resizeObserver = R.onResizeObservable.add(() => { }), this._generateMipMaps = !!n, this._doNotChangeAspectRatio = i, this._renderingManager = new $6(r), this._renderingManager._useSceneAutoClearSetup = !0, !v && (this._renderTargetOptions = { generateMipMaps: n, type: s, format: (I = this._format) !== null && I !== void 0 ? I : void 0, samplingMode: this.samplingMode, generateDepthBuffer: o, generateStencilBuffer: d, samples: P, creationFlags: p, noColorAttachment: c, useSRGBBuffer: H, colorAttachment: N, label: this.name }, this.samplingMode === We.NEAREST_SAMPLINGMODE && (this.wrapU = We.CLAMP_ADDRESSMODE, this.wrapV = We.CLAMP_ADDRESSMODE), l || (a ? (this._renderTarget = r.getEngine().createRenderTargetCubeTexture(this.getRenderSize(), this._renderTargetOptions), this.coordinatesMode = We.INVCUBIC_MODE, this._textureMatrix = he.Identity()) : this._renderTarget = r.getEngine().createRenderTargetTexture(this._size, this._renderTargetOptions), this._texture = this._renderTarget.texture, P !== void 0 && (this.samples = P))); } /** * Creates a depth stencil texture. * This is only available in WebGL 2 or with the depth texture extension available. * @param comparisonFunction Specifies the comparison function to set on the texture. If 0 or undefined, the texture is not in comparison mode (default: 0) * @param bilinearFiltering Specifies whether or not bilinear filtering is enable on the texture (default: true) * @param generateStencil Specifies whether or not a stencil should be allocated in the texture (default: false) * @param samples sample count of the depth/stencil texture (default: 1) * @param format format of the depth texture (default: 14) */ createDepthStencilTexture(e = 0, t = !0, r = !1, n = 1, i = 14) { var s; (s = this._renderTarget) === null || s === void 0 || s.createDepthStencilTexture(e, t, r, n, i); } _releaseRenderPassId() { if (this._scene) { const e = this._scene.getEngine(); for (let t = 0; t < this._renderPassIds.length; ++t) e.releaseRenderPassId(this._renderPassIds[t]); } this._renderPassIds = []; } _createRenderPassId() { this._releaseRenderPassId(); const e = this._scene.getEngine(), t = this._isCubeData ? 6 : this.getRenderLayers() || 1; for (let r = 0; r < t; ++r) this._renderPassIds[r] = e.createRenderPassId(`RenderTargetTexture - ${this.name}#${r}`); } _processSizeParameter(e, t = !0) { if (e.ratio) { this._sizeRatio = e.ratio; const r = this._getEngine(); this._size = { width: this._bestReflectionRenderTargetDimension(r.getRenderWidth(), this._sizeRatio), height: this._bestReflectionRenderTargetDimension(r.getRenderHeight(), this._sizeRatio) }; } else this._size = e; t && this._createRenderPassId(); } /** * Define the number of samples to use in case of MSAA. * It defaults to one meaning no MSAA has been enabled. */ get samples() { var e, t; return (t = (e = this._renderTarget) === null || e === void 0 ? void 0 : e.samples) !== null && t !== void 0 ? t : this._samples; } set samples(e) { this._renderTarget && (this._samples = this._renderTarget.setSamples(e)); } /** * Resets the refresh counter of the texture and start bak from scratch. * Could be useful to regenerate the texture if it is setup to render only once. */ resetRefreshCounter() { this._currentRefreshId = -1; } /** * Define the refresh rate of the texture or the rendering frequency. * Use 0 to render just once, 1 to render on every frame, 2 to render every two frames and so on... */ get refreshRate() { return this._refreshRate; } set refreshRate(e) { this._refreshRate = e, this.resetRefreshCounter(); } /** * Adds a post process to the render target rendering passes. * @param postProcess define the post process to add */ addPostProcess(e) { if (!this._postProcessManager) { const t = this.getScene(); if (!t) return; this._postProcessManager = new yI(t), this._postProcesses = new Array(); } this._postProcesses.push(e), this._postProcesses[0].autoClear = !1; } /** * Clear all the post processes attached to the render target * @param dispose define if the cleared post processes should also be disposed (false by default) */ clearPostProcesses(e = !1) { if (this._postProcesses) { if (e) for (const t of this._postProcesses) t.dispose(); this._postProcesses = []; } } /** * Remove one of the post process from the list of attached post processes to the texture * @param postProcess define the post process to remove from the list */ removePostProcess(e) { if (!this._postProcesses) return; const t = this._postProcesses.indexOf(e); t !== -1 && (this._postProcesses.splice(t, 1), this._postProcesses.length > 0 && (this._postProcesses[0].autoClear = !1)); } /** @internal */ _shouldRender() { return this._currentRefreshId === -1 ? (this._currentRefreshId = 1, !0) : this.refreshRate === this._currentRefreshId ? (this._currentRefreshId = 1, !0) : (this._currentRefreshId++, !1); } /** * Gets the actual render size of the texture. * @returns the width of the render size */ getRenderSize() { return this.getRenderWidth(); } /** * Gets the actual render width of the texture. * @returns the width of the render size */ getRenderWidth() { return this._size.width ? this._size.width : this._size; } /** * Gets the actual render height of the texture. * @returns the height of the render size */ getRenderHeight() { return this._size.width ? this._size.height : this._size; } /** * Gets the actual number of layers of the texture. * @returns the number of layers */ getRenderLayers() { const e = this._size.layers; return e || 0; } /** * Don't allow this render target texture to rescale. Mainly used to prevent rescaling by the scene optimizer. */ disableRescaling() { this._canRescale = !1; } /** * Get if the texture can be rescaled or not. */ get canRescale() { return this._canRescale; } /** * Resize the texture using a ratio. * @param ratio the ratio to apply to the texture size in order to compute the new target size */ scale(e) { const t = Math.max(1, this.getRenderSize() * e); this.resize(t); } /** * Get the texture reflection matrix used to rotate/transform the reflection. * @returns the reflection matrix */ getReflectionTextureMatrix() { return this.isCube ? this._textureMatrix : super.getReflectionTextureMatrix(); } /** * Resize the texture to a new desired size. * Be careful as it will recreate all the data in the new texture. * @param size Define the new size. It can be: * - a number for squared texture, * - an object containing { width: number, height: number } * - or an object containing a ratio { ratio: number } */ resize(e) { var t; const r = this.isCube; (t = this._renderTarget) === null || t === void 0 || t.dispose(), this._renderTarget = null; const n = this.getScene(); n && (this._processSizeParameter(e, !1), r ? this._renderTarget = n.getEngine().createRenderTargetCubeTexture(this.getRenderSize(), this._renderTargetOptions) : this._renderTarget = n.getEngine().createRenderTargetTexture(this._size, this._renderTargetOptions), this._texture = this._renderTarget.texture, this._renderTargetOptions.samples !== void 0 && (this.samples = this._renderTargetOptions.samples), this.onResizeObservable.hasObservers() && this.onResizeObservable.notifyObservers(this)); } /** * Renders all the objects from the render list into the texture. * @param useCameraPostProcess Define if camera post processes should be used during the rendering * @param dumpForDebug Define if the rendering result should be dumped (copied) for debugging purpose */ render(e = !1, t = !1) { this._render(e, t); } /** * This function will check if the render target texture can be rendered (textures are loaded, shaders are compiled) * @returns true if all required resources are ready */ isReadyForRendering() { return this._render(!1, !1, !0); } _render(e = !1, t = !1, r = !1) { var n; const i = this.getScene(); if (!i) return r; const s = i.getEngine(); if (this.useCameraPostProcesses !== void 0 && (e = this.useCameraPostProcesses), this._waitingRenderList) { this.renderList = []; for (let v = 0; v < this._waitingRenderList.length; v++) { const u = this._waitingRenderList[v], l = i.getMeshById(u); l && this.renderList.push(l); } this._waitingRenderList = void 0; } if (this.renderListPredicate) { this.renderList ? this.renderList.length = 0 : this.renderList = []; const v = this.getScene(); if (!v) return r; const u = v.meshes; for (let l = 0; l < u.length; l++) { const P = u[l]; this.renderListPredicate(P) && this.renderList.push(P); } } const a = s.currentRenderPassId; this.onBeforeBindObservable.notifyObservers(this); const f = (n = this.activeCamera) !== null && n !== void 0 ? n : i.activeCamera, o = i.activeCamera; f && (f !== i.activeCamera && (i.setTransformMatrix(f.getViewMatrix(), f.getProjectionMatrix(!0)), i.activeCamera = f), s.setViewport(f.rigParent ? f.rigParent.viewport : f.viewport, this.getRenderWidth(), this.getRenderHeight())), this._defaultRenderListPrepared = !1; let d = r; if (r) { i.getViewMatrix() || i.updateTransformMatrix(); const v = this.is2DArray ? this.getRenderLayers() : this.isCube ? 6 : 1; for (let u = 0; u < v && d; u++) { let l = null; const P = this.renderList ? this.renderList : i.getActiveMeshes().data, p = this.renderList ? this.renderList.length : i.getActiveMeshes().length; s.currentRenderPassId = this._renderPassIds[u], this.onBeforeRenderObservable.notifyObservers(u), this.getCustomRenderList && (l = this.getCustomRenderList(u, P, p)), l || (l = P), this._doNotChangeAspectRatio || i.updateTransformMatrix(!0); for (let c = 0; c < l.length && d; ++c) { const H = l[c]; if (!(!H.isEnabled() || H.isBlocked || !H.isVisible || !H.subMeshes)) { if (this.customIsReadyFunction) { if (!this.customIsReadyFunction(H, this.refreshRate, r)) { d = !1; continue; } } else if (!H.isReady(!0)) { d = !1; continue; } } } this.onAfterRenderObservable.notifyObservers(u), (this.is2DArray || this.isCube) && (i.incrementRenderId(), i.resetCachedMaterial()); } } else if (this.is2DArray && !this.isMulti) for (let v = 0; v < this.getRenderLayers(); v++) this._renderToTarget(0, e, t, v, f), i.incrementRenderId(), i.resetCachedMaterial(); else if (this.isCube && !this.isMulti) for (let v = 0; v < 6; v++) this._renderToTarget(v, e, t, void 0, f), i.incrementRenderId(), i.resetCachedMaterial(); else this._renderToTarget(0, e, t, void 0, f); return this.onAfterUnbindObservable.notifyObservers(this), s.currentRenderPassId = a, o && (i.activeCamera = o, (i.getEngine().scenes.length > 1 || this.activeCamera && this.activeCamera !== i.activeCamera) && i.setTransformMatrix(i.activeCamera.getViewMatrix(), i.activeCamera.getProjectionMatrix(!0)), s.setViewport(i.activeCamera.viewport)), i.resetCachedMaterial(), d; } _bestReflectionRenderTargetDimension(e, t) { const n = e * t, i = Ge.NearestPOT(n + 128 * 128 / (128 + n)); return Math.min(Ge.FloorPOT(e), i); } _prepareRenderingManager(e, t, r, n) { const i = this.getScene(); if (!i) return; this._renderingManager.reset(); const s = i.getRenderId(); for (let a = 0; a < t; a++) { const f = e[a]; if (f && !f.isBlocked) { if (this.customIsReadyFunction) { if (!this.customIsReadyFunction(f, this.refreshRate, !1)) { this.resetRefreshCounter(); continue; } } else if (!f.isReady(this.refreshRate === 0)) { this.resetRefreshCounter(); continue; } if (!f._internalAbstractMeshDataInfo._currentLODIsUpToDate && i.activeCamera && (f._internalAbstractMeshDataInfo._currentLOD = i.customLODSelector ? i.customLODSelector(f, this.activeCamera || i.activeCamera) : f.getLOD(this.activeCamera || i.activeCamera), f._internalAbstractMeshDataInfo._currentLODIsUpToDate = !0), !f._internalAbstractMeshDataInfo._currentLOD) continue; let o = f._internalAbstractMeshDataInfo._currentLOD; o._preActivateForIntermediateRendering(s); let d; if (n && r ? d = (f.layerMask & r.layerMask) === 0 : d = !1, f.isEnabled() && f.isVisible && f.subMeshes && !d && (o !== f && o._activate(s, !0), f._activate(s, !0) && f.subMeshes.length)) { f.isAnInstance ? f._internalAbstractMeshDataInfo._actAsRegularMesh && (o = f) : o._internalAbstractMeshDataInfo._onlyForInstancesIntermediate = !1, o._internalAbstractMeshDataInfo._isActiveIntermediate = !0; for (let v = 0; v < o.subMeshes.length; v++) { const u = o.subMeshes[v]; this._renderingManager.dispatch(u, o); } } } } for (let a = 0; a < i.particleSystems.length; a++) { const f = i.particleSystems[a], o = f.emitter; !f.isStarted() || !o || o.position && !o.isEnabled() || this._renderingManager.dispatchParticles(f); } } /** * @internal * @param faceIndex face index to bind to if this is a cubetexture * @param layer defines the index of the texture to bind in the array */ _bindFrameBuffer(e = 0, t = 0) { const r = this.getScene(); if (!r) return; const n = r.getEngine(); this._renderTarget && n.bindFramebuffer(this._renderTarget, this.isCube ? e : void 0, void 0, void 0, this.ignoreCameraViewport, 0, t); } _unbindFrameBuffer(e, t) { this._renderTarget && e.unBindFramebuffer(this._renderTarget, this.isCube, () => { this.onAfterRenderObservable.notifyObservers(t); }); } /** * @internal */ _prepareFrame(e, t, r, n) { this._postProcessManager ? this._prePassEnabled || this._postProcessManager._prepareFrame(this._texture, this._postProcesses) : (!n || !e.postProcessManager._prepareFrame(this._texture)) && this._bindFrameBuffer(t, r); } _renderToTarget(e, t, r, n = 0, i = null) { var s, a, f, o, d, v; const u = this.getScene(); if (!u) return; const l = u.getEngine(); if ((s = l._debugPushGroup) === null || s === void 0 || s.call(l, `render to face #${e} layer #${n}`, 1), this._prepareFrame(u, e, n, t), this.is2DArray ? (l.currentRenderPassId = this._renderPassIds[n], this.onBeforeRenderObservable.notifyObservers(n)) : (l.currentRenderPassId = this._renderPassIds[e], this.onBeforeRenderObservable.notifyObservers(e)), l.snapshotRendering && l.snapshotRenderingMode === 1) this.onClearObservable.hasObservers() ? this.onClearObservable.notifyObservers(l) : this.skipInitialClear || l.clear(this.clearColor || u.clearColor, !0, !0, !0); else { let p = null; const c = this.renderList ? this.renderList : u.getActiveMeshes().data, H = this.renderList ? this.renderList.length : u.getActiveMeshes().length; this.getCustomRenderList && (p = this.getCustomRenderList(this.is2DArray ? n : e, c, H)), p ? this._prepareRenderingManager(p, p.length, i, this.forceLayerMaskCheck) : (this._defaultRenderListPrepared || (this._prepareRenderingManager(c, H, i, !this.renderList || this.forceLayerMaskCheck), this._defaultRenderListPrepared = !0), p = c); for (const q of u._beforeRenderTargetClearStage) q.action(this, e, n); this.onClearObservable.hasObservers() ? this.onClearObservable.notifyObservers(l) : this.skipInitialClear || l.clear(this.clearColor || u.clearColor, !0, !0, !0), this._doNotChangeAspectRatio || u.updateTransformMatrix(!0); for (const q of u._beforeRenderTargetDrawStage) q.action(this, e, n); this._renderingManager.render(this.customRenderFunction, p, this.renderParticles, this.renderSprites); for (const q of u._afterRenderTargetDrawStage) q.action(this, e, n); const T = (f = (a = this._texture) === null || a === void 0 ? void 0 : a.generateMipMaps) !== null && f !== void 0 ? f : !1; this._texture && (this._texture.generateMipMaps = !1), this._postProcessManager ? this._postProcessManager._finalizeFrame(!1, (o = this._renderTarget) !== null && o !== void 0 ? o : void 0, e, this._postProcesses, this.ignoreCameraViewport) : t && u.postProcessManager._finalizeFrame(!1, (d = this._renderTarget) !== null && d !== void 0 ? d : void 0, e); for (const q of u._afterRenderTargetPostProcessStage) q.action(this, e, n); this._texture && (this._texture.generateMipMaps = T), this._doNotChangeAspectRatio || u.updateTransformMatrix(!0), r && eA.DumpFramebuffer(this.getRenderWidth(), this.getRenderHeight(), l); } this._unbindFrameBuffer(l, e), this._texture && this.isCube && e === 5 && l.generateMipMapsForCubemap(this._texture), (v = l._debugPopGroup) === null || v === void 0 || v.call(l, 1); } /** * Overrides the default sort function applied in the rendering group to prepare the meshes. * This allowed control for front to back rendering or reversely depending of the special needs. * * @param renderingGroupId The rendering group id corresponding to its index * @param opaqueSortCompareFn The opaque queue comparison function use to sort. * @param alphaTestSortCompareFn The alpha test queue comparison function use to sort. * @param transparentSortCompareFn The transparent queue comparison function use to sort. */ setRenderingOrder(e, t = null, r = null, n = null) { this._renderingManager.setRenderingOrder(e, t, r, n); } /** * Specifies whether or not the stencil and depth buffer are cleared between two rendering groups. * * @param renderingGroupId The rendering group id corresponding to its index * @param autoClearDepthStencil Automatically clears depth and stencil between groups if true. */ setRenderingAutoClearDepthStencil(e, t) { this._renderingManager.setRenderingAutoClearDepthStencil(e, t), this._renderingManager._useSceneAutoClearSetup = !1; } /** * Clones the texture. * @returns the cloned texture */ clone() { const e = this.getSize(), t = new Ta(this.name, e, this.getScene(), this._renderTargetOptions.generateMipMaps, this._doNotChangeAspectRatio, this._renderTargetOptions.type, this.isCube, this._renderTargetOptions.samplingMode, this._renderTargetOptions.generateDepthBuffer, this._renderTargetOptions.generateStencilBuffer, void 0, this._renderTargetOptions.format, void 0, this._renderTargetOptions.samples); return t.hasAlpha = this.hasAlpha, t.level = this.level, t.coordinatesMode = this.coordinatesMode, this.renderList && (t.renderList = this.renderList.slice(0)), t; } /** * Serialize the texture to a JSON representation we can easily use in the respective Parse function. * @returns The JSON representation of the texture */ serialize() { if (!this.name) return null; const e = super.serialize(); if (e.renderTargetSize = this.getRenderSize(), e.renderList = [], this.renderList) for (let t = 0; t < this.renderList.length; t++) e.renderList.push(this.renderList[t].id); return e; } /** * This will remove the attached framebuffer objects. The texture will not be able to be used as render target anymore */ disposeFramebufferObjects() { var e; (e = this._renderTarget) === null || e === void 0 || e.dispose(!0); } /** * Release and destroy the underlying lower level texture aka internalTexture. */ releaseInternalTexture() { var e; (e = this._renderTarget) === null || e === void 0 || e.releaseTextures(), this._texture = null; } /** * Dispose the texture and release its associated resources. */ dispose() { var e; this.onResizeObservable.clear(), this.onClearObservable.clear(), this.onAfterRenderObservable.clear(), this.onAfterUnbindObservable.clear(), this.onBeforeBindObservable.clear(), this.onBeforeRenderObservable.clear(), this._postProcessManager && (this._postProcessManager.dispose(), this._postProcessManager = null), this._prePassRenderTarget && this._prePassRenderTarget.dispose(), this._releaseRenderPassId(), this.clearPostProcesses(!0), this._resizeObserver && (this.getScene().getEngine().onResizeObservable.remove(this._resizeObserver), this._resizeObserver = null), this.renderList = null; const t = this.getScene(); if (!t) return; let r = t.customRenderTargets.indexOf(this); r >= 0 && t.customRenderTargets.splice(r, 1); for (const n of t.cameras) r = n.customRenderTargets.indexOf(this), r >= 0 && n.customRenderTargets.splice(r, 1); (e = this._renderTarget) === null || e === void 0 || e.dispose(), this._renderTarget = null, this._texture = null, super.dispose(); } /** @internal */ _rebuild() { this.refreshRate === Ta.REFRESHRATE_RENDER_ONCE && (this.refreshRate = Ta.REFRESHRATE_RENDER_ONCE), this._postProcessManager && this._postProcessManager._rebuild(); } /** * Clear the info related to rendering groups preventing retention point in material dispose. */ freeRenderingGroups() { this._renderingManager && this._renderingManager.freeRenderingGroups(); } /** * Gets the number of views the corresponding to the texture (eg. a MultiviewRenderTarget will have > 1) * @returns the view count */ getViewCount() { return 1; } } Ta.REFRESHRATE_RENDER_ONCE = 0; Ta.REFRESHRATE_RENDER_ONEVERYFRAME = 1; Ta.REFRESHRATE_RENDER_ONEVERYTWOFRAMES = 2; We._CreateRenderTargetTexture = (A, e, t, r, n) => new Ta(A, e, t, r); class Tee { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_PROCEDURALTEXTURE, this.scene = e, this.scene.proceduralTextures = []; } /** * Registers the component in a given scene */ register() { this.scene._beforeClearStage.registerStep(Ot.STEP_BEFORECLEAR_PROCEDURALTEXTURE, this, this._beforeClear); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { } /** * Disposes the component and the associated resources. */ dispose() { } _beforeClear() { if (this.scene.proceduralTexturesEnabled) { ye.StartPerformanceCounter("Procedural textures", this.scene.proceduralTextures.length > 0); for (let e = 0; e < this.scene.proceduralTextures.length; e++) { const t = this.scene.proceduralTextures[e]; t._shouldRender() && t.render(); } ye.EndPerformanceCounter("Procedural textures", this.scene.proceduralTextures.length > 0); } } } const C6e = "proceduralVertexShader", O6e = `attribute vec2 position;varying vec2 vPosition;varying vec2 vUV;const vec2 madd=vec2(0.5,0.5); #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN vPosition=position;vUV=position*madd+madd;gl_Position=vec4(position,0.0,1.0); #define CUSTOM_VERTEX_MAIN_END }`; Le.ShadersStore[C6e] = O6e; class Y0 extends We { /** * Instantiates a new procedural texture. * Procedural texturing is a way to programmatically create a texture. There are 2 types of procedural textures: code-only, and code that references some classic 2D images, sometimes called 'refMaps' or 'sampler' images. * This is the base class of any Procedural texture and contains most of the shareable code. * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/proceduralTextures * @param name Define the name of the texture * @param size Define the size of the texture to create * @param fragment Define the fragment shader to use to generate the texture or null if it is defined later: * * object: \{ fragmentElement: "fragmentShaderCode" \}, used with shader code in script tags * * object: \{ fragmentSource: "fragment shader code string" \}, the string contains the shader code * * string: the string contains a name "XXX" to lookup in Effect.ShadersStore["XXXFragmentShader"] * @param scene Define the scene the texture belongs to * @param fallbackTexture Define a fallback texture in case there were issues to create the custom texture * @param generateMipMaps Define if the texture should creates mip maps or not * @param isCube Define if the texture is a cube texture or not (this will render each faces of the cube) * @param textureType The FBO internal texture type */ constructor(e, t, r, n, i = null, s = !0, a = !1, f = 0) { super(null, n, !s), this.isEnabled = !0, this.autoClear = !0, this.onGeneratedObservable = new Oe(), this.onBeforeGenerationObservable = new Oe(), this.nodeMaterialSource = null, this._textures = {}, this._currentRefreshId = -1, this._frameId = -1, this._refreshRate = 1, this._vertexBuffers = {}, this._uniforms = new Array(), this._samplers = new Array(), this._floats = {}, this._ints = {}, this._floatsArrays = {}, this._colors3 = {}, this._colors4 = {}, this._vectors2 = {}, this._vectors3 = {}, this._matrices = {}, this._fallbackTextureUsed = !1, this._cachedDefines = null, this._contentUpdateId = -1, this._rtWrapper = null, n = this.getScene() || gr.LastCreatedScene; let o = n._getComponent(Ot.NAME_PROCEDURALTEXTURE); o || (o = new Tee(n), n._addComponent(o)), n.proceduralTextures.push(this), this._fullEngine = n.getEngine(), this.name = e, this.isRenderTarget = !0, this._size = t, this._textureType = f, this._generateMipMaps = s, this._drawWrapper = new zo(this._fullEngine), this.setFragment(r), this._fallbackTexture = i; const d = this._createRtWrapper(a, t, s, f); this._texture = d.texture; const v = []; v.push(1, 1), v.push(-1, 1), v.push(-1, -1), v.push(1, -1), this._vertexBuffers[J.PositionKind] = new J(this._fullEngine, v, J.PositionKind, !1, !1, 2), this._createIndexBuffer(); } _createRtWrapper(e, t, r, n) { return e ? (this._rtWrapper = this._fullEngine.createRenderTargetCubeTexture(t, { generateMipMaps: r, generateDepthBuffer: !1, generateStencilBuffer: !1, type: n }), this.setFloat("face", 0)) : this._rtWrapper = this._fullEngine.createRenderTargetTexture(t, { generateMipMaps: r, generateDepthBuffer: !1, generateStencilBuffer: !1, type: n }), this._rtWrapper; } /** * The effect that is created when initializing the post process. * @returns The created effect corresponding the postprocess. */ getEffect() { return this._drawWrapper.effect; } /** * @internal* */ _setEffect(e) { this._drawWrapper.effect = e; } /** * Gets texture content (Use this function wisely as reading from a texture can be slow) * @returns an ArrayBufferView promise (Uint8Array or Float32Array) */ getContent() { return this._contentData && this._frameId === this._contentUpdateId ? this._contentData : (this._contentData ? this._contentData.then((e) => { this._contentData = this.readPixels(0, 0, e), this._contentUpdateId = this._frameId; }) : (this._contentData = this.readPixels(0, 0), this._contentUpdateId = this._frameId), this._contentData); } _createIndexBuffer() { const e = this._fullEngine, t = []; t.push(0), t.push(1), t.push(2), t.push(0), t.push(2), t.push(3), this._indexBuffer = e.createIndexBuffer(t); } /** @internal */ _rebuild() { const e = this._vertexBuffers[J.PositionKind]; e && e._rebuild(), this._createIndexBuffer(), this.refreshRate === Ta.REFRESHRATE_RENDER_ONCE && (this.refreshRate = Ta.REFRESHRATE_RENDER_ONCE); } /** * Resets the texture in order to recreate its associated resources. * This can be called in case of context loss or if you change the shader code and need to regenerate the texture with the new code */ reset() { var e; (e = this._drawWrapper.effect) === null || e === void 0 || e.dispose(), this._drawWrapper.effect = null, this._cachedDefines = null; } _getDefines() { return ""; } /** * Executes a function when the texture will be ready to be drawn. * @param func The callback to be used. */ executeWhenReady(e) { if (this.isReady()) { e(this); return; } const t = this.getEffect(); t && t.executeWhenCompiled(() => { e(this); }); } /** * Is the texture ready to be used ? (rendered at least once) * @returns true if ready, otherwise, false. */ isReady() { const e = this._fullEngine; if (this.nodeMaterialSource) return this._drawWrapper.effect.isReady(); if (!this._fragment) return !1; if (this._fallbackTextureUsed) return !0; if (!this._texture) return !1; const t = this._getDefines(); if (this._drawWrapper.effect && t === this._cachedDefines && this._drawWrapper.effect.isReady()) return !0; const r = { vertex: "procedural", fragmentElement: this._fragment.fragmentElement, fragmentSource: this._fragment.fragmentSource, fragment: typeof this._fragment == "string" ? this._fragment : void 0 }; return this._cachedDefines !== t && (this._cachedDefines = t, this._drawWrapper.effect = e.createEffect(r, [J.PositionKind], this._uniforms, this._samplers, t, void 0, void 0, () => { var n; (n = this._rtWrapper) === null || n === void 0 || n.dispose(), this._rtWrapper = this._texture = null, this._fallbackTexture && (this._texture = this._fallbackTexture._texture, this._texture && this._texture.incrementReferences()), this._fallbackTextureUsed = !0; })), this._drawWrapper.effect.isReady(); } /** * Resets the refresh counter of the texture and start bak from scratch. * Could be useful to regenerate the texture if it is setup to render only once. */ resetRefreshCounter() { this._currentRefreshId = -1; } /** * Set the fragment shader to use in order to render the texture. * @param fragment This can be set to a path (into the shader store) or to a json object containing a fragmentElement property. */ setFragment(e) { this._fragment = e; } /** * Define the refresh rate of the texture or the rendering frequency. * Use 0 to render just once, 1 to render on every frame, 2 to render every two frames and so on... */ get refreshRate() { return this._refreshRate; } set refreshRate(e) { this._refreshRate = e, this.resetRefreshCounter(); } /** @internal */ _shouldRender() { return !this.isEnabled || !this.isReady() || !this._texture ? (this._texture && (this._texture.isReady = !1), !1) : this._fallbackTextureUsed ? !1 : this._currentRefreshId === -1 ? (this._currentRefreshId = 1, this._frameId++, !0) : this.refreshRate === this._currentRefreshId ? (this._currentRefreshId = 1, this._frameId++, !0) : (this._currentRefreshId++, !1); } /** * Get the size the texture is rendering at. * @returns the size (on cube texture it is always squared) */ getRenderSize() { return this._size; } /** * Resize the texture to new value. * @param size Define the new size the texture should have * @param generateMipMaps Define whether the new texture should create mip maps */ resize(e, t) { if (this._fallbackTextureUsed || !this._rtWrapper || !this._texture) return; const r = this._texture.isCube; this._rtWrapper.dispose(); const n = this._createRtWrapper(r, e, t, this._textureType); this._texture = n.texture, this._size = e, this._generateMipMaps = t; } _checkUniform(e) { this._uniforms.indexOf(e) === -1 && this._uniforms.push(e); } /** * Set a texture in the shader program used to render. * @param name Define the name of the uniform samplers as defined in the shader * @param texture Define the texture to bind to this sampler * @returns the texture itself allowing "fluent" like uniform updates */ setTexture(e, t) { return this._samplers.indexOf(e) === -1 && this._samplers.push(e), this._textures[e] = t, this; } /** * Set a float in the shader. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the texture itself allowing "fluent" like uniform updates */ setFloat(e, t) { return this._checkUniform(e), this._floats[e] = t, this; } /** * Set a int in the shader. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the texture itself allowing "fluent" like uniform updates */ setInt(e, t) { return this._checkUniform(e), this._ints[e] = t, this; } /** * Set an array of floats in the shader. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the texture itself allowing "fluent" like uniform updates */ setFloats(e, t) { return this._checkUniform(e), this._floatsArrays[e] = t, this; } /** * Set a vec3 in the shader from a Color3. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the texture itself allowing "fluent" like uniform updates */ setColor3(e, t) { return this._checkUniform(e), this._colors3[e] = t, this; } /** * Set a vec4 in the shader from a Color4. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the texture itself allowing "fluent" like uniform updates */ setColor4(e, t) { return this._checkUniform(e), this._colors4[e] = t, this; } /** * Set a vec2 in the shader from a Vector2. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the texture itself allowing "fluent" like uniform updates */ setVector2(e, t) { return this._checkUniform(e), this._vectors2[e] = t, this; } /** * Set a vec3 in the shader from a Vector3. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the texture itself allowing "fluent" like uniform updates */ setVector3(e, t) { return this._checkUniform(e), this._vectors3[e] = t, this; } /** * Set a mat4 in the shader from a MAtrix. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the texture itself allowing "fluent" like uniform updates */ setMatrix(e, t) { return this._checkUniform(e), this._matrices[e] = t, this; } /** * Render the texture to its associated render target. * @param useCameraPostProcess Define if camera post process should be applied to the texture */ // eslint-disable-next-line @typescript-eslint/no-unused-vars render(e) { var t, r; const n = this.getScene(); if (!n) return; const i = this._fullEngine; if (i.enableEffect(this._drawWrapper), this.onBeforeGenerationObservable.notifyObservers(this), i.setState(!1), !this.nodeMaterialSource) { for (const a in this._textures) this._drawWrapper.effect.setTexture(a, this._textures[a]); for (const a in this._ints) this._drawWrapper.effect.setInt(a, this._ints[a]); for (const a in this._floats) this._drawWrapper.effect.setFloat(a, this._floats[a]); for (const a in this._floatsArrays) this._drawWrapper.effect.setArray(a, this._floatsArrays[a]); for (const a in this._colors3) this._drawWrapper.effect.setColor3(a, this._colors3[a]); for (const a in this._colors4) { const f = this._colors4[a]; this._drawWrapper.effect.setFloat4(a, f.r, f.g, f.b, f.a); } for (const a in this._vectors2) this._drawWrapper.effect.setVector2(a, this._vectors2[a]); for (const a in this._vectors3) this._drawWrapper.effect.setVector3(a, this._vectors3[a]); for (const a in this._matrices) this._drawWrapper.effect.setMatrix(a, this._matrices[a]); } if (!this._texture || !this._rtWrapper) return; (t = i._debugPushGroup) === null || t === void 0 || t.call(i, `procedural texture generation for ${this.name}`, 1); const s = i.currentViewport; if (this.isCube) for (let a = 0; a < 6; a++) i.bindFramebuffer(this._rtWrapper, a, void 0, void 0, !0), i.bindBuffers(this._vertexBuffers, this._indexBuffer, this._drawWrapper.effect), this._drawWrapper.effect.setFloat("face", a), this.autoClear && i.clear(n.clearColor, !0, !1, !1), i.drawElementsType(gt.TriangleFillMode, 0, 6); else i.bindFramebuffer(this._rtWrapper, 0, void 0, void 0, !0), i.bindBuffers(this._vertexBuffers, this._indexBuffer, this._drawWrapper.effect), this.autoClear && i.clear(n.clearColor, !0, !1, !1), i.drawElementsType(gt.TriangleFillMode, 0, 6); i.unBindFramebuffer(this._rtWrapper, this.isCube), s && i.setViewport(s), this.isCube && i.generateMipMapsForCubemap(this._texture), (r = i._debugPopGroup) === null || r === void 0 || r.call(i, 1), this.onGenerated && this.onGenerated(), this.onGeneratedObservable.notifyObservers(this); } /** * Clone the texture. * @returns the cloned texture */ clone() { const e = this.getSize(), t = new Y0(this.name, e.width, this._fragment, this.getScene(), this._fallbackTexture, this._generateMipMaps); return t.hasAlpha = this.hasAlpha, t.level = this.level, t.coordinatesMode = this.coordinatesMode, t; } /** * Dispose the texture and release its associated resources. */ dispose() { const e = this.getScene(); if (!e) return; const t = e.proceduralTextures.indexOf(this); t >= 0 && e.proceduralTextures.splice(t, 1); const r = this._vertexBuffers[J.PositionKind]; r && (r.dispose(), this._vertexBuffers[J.PositionKind] = null), this._indexBuffer && this._fullEngine._releaseBuffer(this._indexBuffer) && (this._indexBuffer = null), this.onGeneratedObservable.clear(), this.onBeforeGenerationObservable.clear(), super.dispose(); } } C([ M() ], Y0.prototype, "isEnabled", void 0); C([ M() ], Y0.prototype, "autoClear", void 0); C([ M() ], Y0.prototype, "_generateMipMaps", void 0); C([ M() ], Y0.prototype, "_size", void 0); C([ M() ], Y0.prototype, "refreshRate", null); Ue("BABYLON.ProceduralTexture", Y0); var o9; (function(A) { A[A.Cos = 0] = "Cos", A[A.Sin = 1] = "Sin", A[A.Abs = 2] = "Abs", A[A.Exp = 3] = "Exp", A[A.Exp2 = 4] = "Exp2", A[A.Round = 5] = "Round", A[A.Floor = 6] = "Floor", A[A.Ceiling = 7] = "Ceiling", A[A.Sqrt = 8] = "Sqrt", A[A.Log = 9] = "Log", A[A.Tan = 10] = "Tan", A[A.ArcTan = 11] = "ArcTan", A[A.ArcCos = 12] = "ArcCos", A[A.ArcSin = 13] = "ArcSin", A[A.Fract = 14] = "Fract", A[A.Sign = 15] = "Sign", A[A.Radians = 16] = "Radians", A[A.Degrees = 17] = "Degrees"; })(o9 || (o9 = {})); class uQ extends Mr { /** * Creates a new TrigonometryBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.operation = o9.Cos, this.registerInput("input", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0]; } /** * Gets the current class name * @returns the class name */ getClassName() { return "TrigonometryBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; let r = ""; switch (this.operation) { case o9.Cos: { r = "cos"; break; } case o9.Sin: { r = "sin"; break; } case o9.Abs: { r = "abs"; break; } case o9.Exp: { r = "exp"; break; } case o9.Exp2: { r = "exp2"; break; } case o9.Round: { r = "round"; break; } case o9.Floor: { r = "floor"; break; } case o9.Ceiling: { r = "ceil"; break; } case o9.Sqrt: { r = "sqrt"; break; } case o9.Log: { r = "log"; break; } case o9.Tan: { r = "tan"; break; } case o9.ArcTan: { r = "atan"; break; } case o9.ArcCos: { r = "acos"; break; } case o9.ArcSin: { r = "asin"; break; } case o9.Fract: { r = "fract"; break; } case o9.Sign: { r = "sign"; break; } case o9.Radians: { r = "radians"; break; } case o9.Degrees: { r = "degrees"; break; } } return e.compilationString += this._declareOutput(t, e) + ` = ${r}(${this.input.associatedVariableName}); `, this; } serialize() { const e = super.serialize(); return e.operation = this.operation, e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.operation = e.operation; } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.operation = BABYLON.TrigonometryBlockOperations.${o9[this.operation]}; `; } } Ue("BABYLON.TrigonometryBlock", uQ); const XE = { effect: null, subMesh: null }; class HI extends na { constructor() { super(), this.NORMAL = !1, this.TANGENT = !1, this.VERTEXCOLOR_NME = !1, this.UV1 = !1, this.UV2 = !1, this.UV3 = !1, this.UV4 = !1, this.UV5 = !1, this.UV6 = !1, this.PREPASS = !1, this.PREPASS_NORMAL = !1, this.PREPASS_NORMAL_INDEX = -1, this.PREPASS_POSITION = !1, this.PREPASS_POSITION_INDEX = -1, this.PREPASS_DEPTH = !1, this.PREPASS_DEPTH_INDEX = -1, this.SCENE_MRT_COUNT = 0, this.NUM_BONE_INFLUENCERS = 0, this.BonesPerMesh = 0, this.BONETEXTURE = !1, this.MORPHTARGETS = !1, this.MORPHTARGETS_NORMAL = !1, this.MORPHTARGETS_TANGENT = !1, this.MORPHTARGETS_UV = !1, this.NUM_MORPH_INFLUENCERS = 0, this.MORPHTARGETS_TEXTURE = !1, this.IMAGEPROCESSING = !1, this.VIGNETTE = !1, this.VIGNETTEBLENDMODEMULTIPLY = !1, this.VIGNETTEBLENDMODEOPAQUE = !1, this.TONEMAPPING = !1, this.TONEMAPPING_ACES = !1, this.CONTRAST = !1, this.EXPOSURE = !1, this.COLORCURVES = !1, this.COLORGRADING = !1, this.COLORGRADING3D = !1, this.SAMPLER3DGREENDEPTH = !1, this.SAMPLER3DBGRMAP = !1, this.DITHER = !1, this.IMAGEPROCESSINGPOSTPROCESS = !1, this.SKIPFINALCOLORCLAMP = !1, this.BUMPDIRECTUV = 0, this.CAMERA_ORTHOGRAPHIC = !1, this.CAMERA_PERSPECTIVE = !1, this.rebuild(); } setValue(e, t, r = !1) { this[e] === void 0 && this._keys.push(e), r && this[e] !== t && this.markAsUnprocessed(), this[e] = t; } } class ja extends P1 { /** * Checks if a block is a texture block * @param block The block to check * @returns True if the block is a texture block */ static _BlockIsTextureBlock(e) { return e.getClassName() === "TextureBlock" || e.getClassName() === "ReflectionTextureBaseBlock" || e.getClassName() === "RefractionBlock" || e.getClassName() === "CurrentScreenBlock" || e.getClassName() === "ParticleTextureBlock" || e.getClassName() === "ImageSourceBlock" || e.getClassName() === "TriPlanarBlock" || e.getClassName() === "BiPlanarBlock" || e.getClassName() === "PrePassTextureBlock"; } /** Get the inspector from bundle or global */ _getGlobalNodeMaterialEditor() { if (typeof NODEEDITOR < "u") return NODEEDITOR; if (typeof BABYLON < "u" && typeof BABYLON.NodeEditor < "u") return BABYLON; } /** Gets or sets options to control the node material overall behavior */ get options() { return this._options; } set options(e) { this._options = e; } /** * Gets the image processing configuration used either in this material. */ get imageProcessingConfiguration() { return this._imageProcessingConfiguration; } /** * Sets the Default image processing configuration used either in the this material. * * If sets to null, the scene one is in use. */ set imageProcessingConfiguration(e) { this._attachImageProcessingConfiguration(e), this._markAllSubMeshesAsTexturesDirty(); } /** * Gets or sets the mode property */ get mode() { return this._mode; } set mode(e) { this._mode = e; } /** Gets or sets the unique identifier used to identified the effect associated with the material */ get buildId() { return this._buildId; } set buildId(e) { this._buildId = e; } /** * Create a new node based material * @param name defines the material name * @param scene defines the hosting scene * @param options defines creation option */ constructor(e, t, r = {}) { super(e, t || gr.LastCreatedScene), this._buildId = ja._BuildIdGenerator++, this._buildWasSuccessful = !1, this._cachedWorldViewMatrix = new he(), this._cachedWorldViewProjectionMatrix = new he(), this._optimizers = new Array(), this._animationFrame = -1, this.BJSNODEMATERIALEDITOR = this._getGlobalNodeMaterialEditor(), this.editorData = null, this.ignoreAlpha = !1, this.maxSimultaneousLights = 4, this.onBuildObservable = new Oe(), this._vertexOutputNodes = new Array(), this._fragmentOutputNodes = new Array(), this.attachedBlocks = [], this._mode = Vv.Material, this.forceAlphaBlending = !1, this._options = Object.assign({ emitComments: !1 }, r), this._attachImageProcessingConfiguration(null); } /** * Gets the current class name of the material e.g. "NodeMaterial" * @returns the class name */ getClassName() { return "NodeMaterial"; } /** * Attaches a new image processing configuration to the Standard Material. * @param configuration */ _attachImageProcessingConfiguration(e) { e !== this._imageProcessingConfiguration && (this._imageProcessingConfiguration && this._imageProcessingObserver && this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver), e ? this._imageProcessingConfiguration = e : this._imageProcessingConfiguration = this.getScene().imageProcessingConfiguration, this._imageProcessingConfiguration && (this._imageProcessingObserver = this._imageProcessingConfiguration.onUpdateParameters.add(() => { this._markAllSubMeshesAsImageProcessingDirty(); }))); } /** * Get a block by its name * @param name defines the name of the block to retrieve * @returns the required block or null if not found */ getBlockByName(e) { let t = null; for (const r of this.attachedBlocks) if (r.name === e) if (!t) t = r; else return ye.Warn("More than one block was found with the name `" + e + "`"), t; return t; } /** * Get a block by its name * @param predicate defines the predicate used to find the good candidate * @returns the required block or null if not found */ getBlockByPredicate(e) { for (const t of this.attachedBlocks) if (e(t)) return t; return null; } /** * Get an input block by its name * @param predicate defines the predicate used to find the good candidate * @returns the required input block or null if not found */ getInputBlockByPredicate(e) { for (const t of this.attachedBlocks) if (t.isInput && e(t)) return t; return null; } /** * Gets the list of input blocks attached to this material * @returns an array of InputBlocks */ getInputBlocks() { const e = []; for (const t of this.attachedBlocks) t.isInput && e.push(t); return e; } /** * Adds a new optimizer to the list of optimizers * @param optimizer defines the optimizers to add * @returns the current material */ registerOptimizer(e) { if (!(this._optimizers.indexOf(e) > -1)) return this._optimizers.push(e), this; } /** * Remove an optimizer from the list of optimizers * @param optimizer defines the optimizers to remove * @returns the current material */ unregisterOptimizer(e) { const t = this._optimizers.indexOf(e); if (t !== -1) return this._optimizers.splice(t, 1), this; } /** * Add a new block to the list of output nodes * @param node defines the node to add * @returns the current material */ addOutputNode(e) { if (e.target === null) throw "This node is not meant to be an output node. You may want to explicitly set its target value."; return e.target & Ve.Vertex && this._addVertexOutputNode(e), e.target & Ve.Fragment && this._addFragmentOutputNode(e), this; } /** * Remove a block from the list of root nodes * @param node defines the node to remove * @returns the current material */ removeOutputNode(e) { return e.target === null ? this : (e.target & Ve.Vertex && this._removeVertexOutputNode(e), e.target & Ve.Fragment && this._removeFragmentOutputNode(e), this); } _addVertexOutputNode(e) { if (this._vertexOutputNodes.indexOf(e) === -1) return e.target = Ve.Vertex, this._vertexOutputNodes.push(e), this; } _removeVertexOutputNode(e) { const t = this._vertexOutputNodes.indexOf(e); if (t !== -1) return this._vertexOutputNodes.splice(t, 1), this; } _addFragmentOutputNode(e) { if (this._fragmentOutputNodes.indexOf(e) === -1) return e.target = Ve.Fragment, this._fragmentOutputNodes.push(e), this; } _removeFragmentOutputNode(e) { const t = this._fragmentOutputNodes.indexOf(e); if (t !== -1) return this._fragmentOutputNodes.splice(t, 1), this; } /** * Specifies if the material will require alpha blending * @returns a boolean specifying if alpha blending is needed */ needAlphaBlending() { return this.ignoreAlpha ? !1 : this.forceAlphaBlending || this.alpha < 1 || this._sharedData && this._sharedData.hints.needAlphaBlending; } /** * Specifies if this material should be rendered in alpha test mode * @returns a boolean specifying if an alpha test is needed. */ needAlphaTesting() { return this._sharedData && this._sharedData.hints.needAlphaTesting; } _processInitializeOnLink(e, t, r, n = !0) { (e.target === Ve.VertexAndFragment || t.target === Ve.Fragment && e.target === Ve.Vertex && e._preparationId !== this._buildId) && r.push(e), this._initializeBlock(e, t, r, n); } _initializeBlock(e, t, r, n = !0) { if (e.initialize(t), n && e.autoConfigure(this), e._preparationId = this._buildId, this.attachedBlocks.indexOf(e) === -1) { if (e.isUnique) { const i = e.getClassName(); for (const s of this.attachedBlocks) if (s.getClassName() === i) throw `Cannot have multiple blocks of type ${i} in the same NodeMaterial`; } this.attachedBlocks.push(e); } for (const i of e.inputs) { i.associatedVariableName = ""; const s = i.connectedPoint; if (s) { const a = s.ownerBlock; a !== e && this._processInitializeOnLink(a, t, r, n); } } if (e.isTeleportOut) { const i = e; i.entryPoint && this._processInitializeOnLink(i.entryPoint, t, r, n); } for (const i of e.outputs) i.associatedVariableName = ""; } _resetDualBlocks(e, t) { e.target === Ve.VertexAndFragment && (e.buildId = t); for (const r of e.inputs) { const n = r.connectedPoint; if (n) { const i = n.ownerBlock; i !== e && this._resetDualBlocks(i, t); } } if (e.isTeleportOut) { const r = e; r.entryPoint && this._resetDualBlocks(r.entryPoint, t); } } /** * Remove a block from the current node material * @param block defines the block to remove */ removeBlock(e) { const t = this.attachedBlocks.indexOf(e); t > -1 && this.attachedBlocks.splice(t, 1), e.isFinalMerger && this.removeOutputNode(e); } /** * Build the material and generates the inner effect * @param verbose defines if the build should log activity * @param updateBuildId defines if the internal build Id should be updated (default is true) * @param autoConfigure defines if the autoConfigure method should be called when initializing blocks (default is false) */ build(e = !1, t = !0, r = !1) { !this._vertexCompilationState && !r && (r = !0), this._buildWasSuccessful = !1; const n = this.getScene().getEngine(), i = this._mode === Vv.Particle; if (this._vertexOutputNodes.length === 0 && !i) throw "You must define at least one vertexOutputNode"; if (this._fragmentOutputNodes.length === 0) throw "You must define at least one fragmentOutputNode"; this._vertexCompilationState = new lG(), this._vertexCompilationState.supportUniformBuffers = n.supportsUniformBuffers, this._vertexCompilationState.target = Ve.Vertex, this._fragmentCompilationState = new lG(), this._fragmentCompilationState.supportUniformBuffers = n.supportsUniformBuffers, this._fragmentCompilationState.target = Ve.Fragment, this._sharedData = new W6e(), this._sharedData.nodeMaterial = this, this._sharedData.fragmentOutputNodes = this._fragmentOutputNodes, this._vertexCompilationState.sharedData = this._sharedData, this._fragmentCompilationState.sharedData = this._sharedData, this._sharedData.buildId = this._buildId, this._sharedData.emitComments = this._options.emitComments, this._sharedData.verbose = e, this._sharedData.scene = this.getScene(), this._sharedData.allowEmptyVertexProgram = i; const s = [], a = []; for (const d of this._vertexOutputNodes) s.push(d), this._initializeBlock(d, this._vertexCompilationState, a, r); for (const d of this._fragmentOutputNodes) a.push(d), this._initializeBlock(d, this._fragmentCompilationState, s, r); this.optimize(); for (const d of s) d.build(this._vertexCompilationState, s); this._fragmentCompilationState.uniforms = this._vertexCompilationState.uniforms.slice(0), this._fragmentCompilationState._uniformDeclaration = this._vertexCompilationState._uniformDeclaration, this._fragmentCompilationState._constantDeclaration = this._vertexCompilationState._constantDeclaration, this._fragmentCompilationState._vertexState = this._vertexCompilationState; for (const d of a) this._resetDualBlocks(d, this._buildId - 1); for (const d of a) d.build(this._fragmentCompilationState, a); this._vertexCompilationState.finalize(this._vertexCompilationState), this._fragmentCompilationState.finalize(this._fragmentCompilationState), t && (this._buildId = ja._BuildIdGenerator++), this._sharedData.emitErrors(), e && (console.log("Vertex shader:"), console.log(this._vertexCompilationState.compilationString), console.log("Fragment shader:"), console.log(this._fragmentCompilationState.compilationString)), this._buildWasSuccessful = !0, this.onBuildObservable.notifyObservers(this); const f = this.getScene().meshes; for (const d of f) if (d.subMeshes) for (const v of d.subMeshes) { if (v.getMaterial() !== this || !v.materialDefines) continue; const u = v.materialDefines; u.markAllAsDirty(), u.reset(); } this.prePassTextureInputs.length && this.getScene().enablePrePassRenderer(); const o = this.getScene().prePassRenderer; o && o.markAsDirty(); } /** * Runs an otpimization phase to try to improve the shader code */ optimize() { for (const e of this._optimizers) e.optimize(this._vertexOutputNodes, this._fragmentOutputNodes); } _prepareDefinesForAttributes(e, t) { const r = t.NORMAL, n = t.TANGENT, i = t.VERTEXCOLOR_NME; t.NORMAL = e.isVerticesDataPresent(J.NormalKind), t.TANGENT = e.isVerticesDataPresent(J.TangentKind); const s = e.useVertexColors && e.isVerticesDataPresent(J.ColorKind); t.VERTEXCOLOR_NME = s; let a = !1; for (let o = 1; o <= 6; ++o) { const d = t["UV" + o]; t["UV" + o] = e.isVerticesDataPresent(`uv${o === 1 ? "" : o}`), a = a || t["UV" + o] !== d; } const f = this.needAlphaBlendingForMesh(e) && this.getScene().useOrderIndependentTransparency; Ye.PrepareDefinesForPrePass(this.getScene(), t, !f), (r !== t.NORMAL || n !== t.TANGENT || i !== t.VERTEXCOLOR_NME || a) && t.markAsAttributesDirty(); } /** * Can this material render to prepass */ get isPrePassCapable() { return !0; } /** * Outputs written to the prepass */ get prePassTextureOutputs() { const e = this.getBlockByPredicate((r) => r.getClassName() === "PrePassOutputBlock"), t = [4]; return !e || this.prePassTextureInputs.length || (e.viewDepth.isConnected && t.push(5), e.viewNormal.isConnected && t.push(6), e.worldPosition.isConnected && t.push(1)), t; } /** * Gets the list of prepass texture required */ get prePassTextureInputs() { const e = this.getAllTextureBlocks().filter((r) => r.getClassName() === "PrePassTextureBlock"), t = []; for (const r of e) r.position.isConnected && !t.includes(1) && t.push(1), r.depth.isConnected && !t.includes(5) && t.push(5), r.normal.isConnected && !t.includes(6) && t.push(6); return t; } /** * Sets the required values to the prepass renderer. */ setPrePassRenderer(e) { const t = this.prePassTextureInputs.concat(this.prePassTextureOutputs); if (e && t.length > 1) { let r = e.getEffectConfiguration("nodeMaterial"); r || (r = e.addEffectConfiguration({ enabled: !0, needsImageProcessing: !1, name: "nodeMaterial", texturesRequired: [] })); for (const n of t) r.texturesRequired.includes(n) || r.texturesRequired.push(n); r.enabled = !0; } return t.length > 1; } /** * Create a post process from the material * @param camera The camera to apply the render pass to. * @param options The required width/height ratio to downsize to before computing the render pass. (Use 1.0 for full size) * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) * @param textureType Type of textures used when performing the post process. (default: 0) * @param textureFormat Format of textures used when performing the post process. (default: TEXTUREFORMAT_RGBA) * @returns the post process created */ createPostProcess(e, t = 1, r = 1, n, i, s = 0, a = 5) { return this.mode !== Vv.PostProcess ? (console.log("Incompatible material mode"), null) : this._createEffectForPostProcess(null, e, t, r, n, i, s, a); } /** * Create the post process effect from the material * @param postProcess The post process to create the effect for */ createEffectForPostProcess(e) { this._createEffectForPostProcess(e); } _createEffectForPostProcess(e, t, r = 1, n = 1, i, s, a = 0, f = 5) { let o = this.name + this._buildId; const d = new HI(), v = new jn(o + "PostProcess", this.getScene()); let u = this._buildId; return this._processDefines(v, d), An.RegisterShader(o, this._fragmentCompilationState._builtCompilationString, this._vertexCompilationState._builtCompilationString), e ? e.updateEffect(d.toString(), this._fragmentCompilationState.uniforms, this._fragmentCompilationState.samplers, { maxSimultaneousLights: this.maxSimultaneousLights }, void 0, void 0, o, o) : e = new kr(this.name + "PostProcess", o, this._fragmentCompilationState.uniforms, this._fragmentCompilationState.samplers, r, t, n, i, s, d.toString(), a, o, { maxSimultaneousLights: this.maxSimultaneousLights }, !1, f), e.nodeMaterialSource = this, e.onApplyObservable.add((l) => { u !== this._buildId && (delete An.ShadersStore[o + "VertexShader"], delete An.ShadersStore[o + "PixelShader"], o = this.name + this._buildId, d.markAllAsDirty(), u = this._buildId), this._processDefines(v, d) && (An.RegisterShader(o, this._fragmentCompilationState._builtCompilationString, this._vertexCompilationState._builtCompilationString), wS.SetImmediate(() => e.updateEffect(d.toString(), this._fragmentCompilationState.uniforms, this._fragmentCompilationState.samplers, { maxSimultaneousLights: this.maxSimultaneousLights }, void 0, void 0, o, o))), this._checkInternals(l); }), e; } /** * Create a new procedural texture based on this node material * @param size defines the size of the texture * @param scene defines the hosting scene * @returns the new procedural texture attached to this node material */ createProceduralTexture(e, t) { if (this.mode !== Vv.ProceduralTexture) return console.log("Incompatible material mode"), null; let r = this.name + this._buildId; const n = new Y0(r, e, null, t), i = new jn(r + "Procedural", this.getScene()); i.reservedDataStore = { hidden: !0 }; const s = new HI(), a = this._processDefines(i, s); An.RegisterShader(r, this._fragmentCompilationState._builtCompilationString, this._vertexCompilationState._builtCompilationString); let f = this.getScene().getEngine().createEffect({ vertexElement: r, fragmentElement: r }, [J.PositionKind], this._fragmentCompilationState.uniforms, this._fragmentCompilationState.samplers, s.toString(), a == null ? void 0 : a.fallbacks, void 0); n.nodeMaterialSource = this, n._setEffect(f); let o = this._buildId; return n.onBeforeGenerationObservable.add(() => { o !== this._buildId && (delete An.ShadersStore[r + "VertexShader"], delete An.ShadersStore[r + "PixelShader"], r = this.name + this._buildId, s.markAllAsDirty(), o = this._buildId); const d = this._processDefines(i, s); d && (An.RegisterShader(r, this._fragmentCompilationState._builtCompilationString, this._vertexCompilationState._builtCompilationString), wS.SetImmediate(() => { f = this.getScene().getEngine().createEffect({ vertexElement: r, fragmentElement: r }, [J.PositionKind], this._fragmentCompilationState.uniforms, this._fragmentCompilationState.samplers, s.toString(), d == null ? void 0 : d.fallbacks, void 0), n._setEffect(f); })), this._checkInternals(f); }), n; } _createEffectForParticles(e, t, r, n, i, s, a, f = "") { let o = this.name + this._buildId + "_" + t; s || (s = new HI()), a || (a = this.getScene().getMeshByName(this.name + "Particle"), a || (a = new jn(this.name + "Particle", this.getScene()), a.reservedDataStore = { hidden: !0 })); let d = this._buildId; const v = []; let u = f; if (!i) { const l = this._processDefines(a, s); An.RegisterShader(o, this._fragmentCompilationState._builtCompilationString), e.fillDefines(v, t), u = v.join(` `), i = this.getScene().getEngine().createEffectForParticles(o, this._fragmentCompilationState.uniforms, this._fragmentCompilationState.samplers, s.toString() + ` ` + u, l == null ? void 0 : l.fallbacks, r, n, e), e.setCustomEffect(i, t); } i.onBindObservable.add((l) => { d !== this._buildId && (delete An.ShadersStore[o + "PixelShader"], o = this.name + this._buildId + "_" + t, s.markAllAsDirty(), d = this._buildId), v.length = 0, e.fillDefines(v, t); const P = v.join(` `); P !== u && (s.markAllAsDirty(), u = P); const p = this._processDefines(a, s); if (p) { An.RegisterShader(o, this._fragmentCompilationState._builtCompilationString), l = this.getScene().getEngine().createEffectForParticles(o, this._fragmentCompilationState.uniforms, this._fragmentCompilationState.samplers, s.toString() + ` ` + u, p == null ? void 0 : p.fallbacks, r, n, e), e.setCustomEffect(l, t), this._createEffectForParticles(e, t, r, n, l, s, a, f); return; } this._checkInternals(l); }); } _checkInternals(e) { if (this._sharedData.animatedInputs) { const t = this.getScene(), r = t.getFrameId(); if (this._animationFrame !== r) { for (const n of this._sharedData.animatedInputs) n.animate(t); this._animationFrame = r; } } for (const t of this._sharedData.bindableBlocks) t.bind(e, this); for (const t of this._sharedData.inputBlocks) t._transmit(e, this.getScene(), this); } /** * Create the effect to be used as the custom effect for a particle system * @param particleSystem Particle system to create the effect for * @param onCompiled defines a function to call when the effect creation is successful * @param onError defines a function to call when the effect creation has failed */ createEffectForParticles(e, t, r) { if (this.mode !== Vv.Particle) { console.log("Incompatible material mode"); return; } this._createEffectForParticles(e, N0.BLENDMODE_ONEONE, t, r), this._createEffectForParticles(e, N0.BLENDMODE_MULTIPLY, t, r); } /** * Use this material as the shadow depth wrapper of a target material * @param targetMaterial defines the target material */ createAsShadowDepthWrapper(e) { if (this.mode !== Vv.Material) { console.log("Incompatible material mode"); return; } e.shadowDepthWrapper = new BABYLON.ShadowDepthWrapper(this, this.getScene()); } _processDefines(e, t, r = !1, n) { let i = null; const s = this.getScene(); if (Ye.PrepareDefinesForCamera(s, t) && t.markAsMiscDirty(), this._sharedData.blocksWithDefines.forEach((a) => { a.initializeDefines(e, this, t, r); }), this._sharedData.blocksWithDefines.forEach((a) => { a.prepareDefines(e, this, t, r, n); }), t.isDirty) { const a = t._areLightsDisposed; t.markAsProcessed(), this._vertexCompilationState.compilationString = this._vertexCompilationState._builtCompilationString, this._fragmentCompilationState.compilationString = this._fragmentCompilationState._builtCompilationString, this._sharedData.repeatableContentBlocks.forEach((u) => { u.replaceRepeatableContent(this._vertexCompilationState, this._fragmentCompilationState, e, t); }); const f = []; this._sharedData.dynamicUniformBlocks.forEach((u) => { u.updateUniformsAndSamples(this._vertexCompilationState, this, t, f); }); const o = this._vertexCompilationState.uniforms; this._fragmentCompilationState.uniforms.forEach((u) => { o.indexOf(u) === -1 && o.push(u); }); const d = this._vertexCompilationState.samplers; this._fragmentCompilationState.samplers.forEach((u) => { d.indexOf(u) === -1 && d.push(u); }); const v = new c1(); this._sharedData.blocksWithFallbacks.forEach((u) => { u.provideFallbacks(e, v); }), i = { lightDisposed: a, uniformBuffers: f, mergedUniforms: o, mergedSamplers: d, fallbacks: v }; } return i; } /** * Get if the submesh is ready to be used and all its information available. * Child classes can use it to update shaders * @param mesh defines the mesh to check * @param subMesh defines which submesh to check * @param useInstances specifies that instances should be used * @returns a boolean indicating that the submesh is ready or not */ isReadyForSubMesh(e, t, r = !1) { if (!this._buildWasSuccessful) return !1; const n = this.getScene(); if (this._sharedData.animatedInputs) { const f = n.getFrameId(); if (this._animationFrame !== f) { for (const o of this._sharedData.animatedInputs) o.animate(n); this._animationFrame = f; } } if (t.effect && this.isFrozen && t.effect._wasPreviouslyReady && t.effect._wasPreviouslyUsingInstances === r) return !0; t.materialDefines || (t.materialDefines = new HI()); const i = t.materialDefines; if (this._isReadyForSubMesh(t)) return !0; const s = n.getEngine(); if (this._prepareDefinesForAttributes(e, i), this._sharedData.blockingBlocks.some((f) => !f.isReady(e, this, i, r))) return !1; const a = this._processDefines(e, i, r, t); if (a) { const f = t.effect, o = i.toString(); let d = s.createEffect({ vertex: "nodeMaterial" + this._buildId, fragment: "nodeMaterial" + this._buildId, vertexSource: this._vertexCompilationState.compilationString, fragmentSource: this._fragmentCompilationState.compilationString }, { attributes: this._vertexCompilationState.attributes, uniformsNames: a.mergedUniforms, uniformBuffersNames: a.uniformBuffers, samplers: a.mergedSamplers, defines: o, fallbacks: a.fallbacks, onCompiled: this.onCompiled, onError: this.onError, multiTarget: i.PREPASS, indexParameters: { maxSimultaneousLights: this.maxSimultaneousLights, maxSimultaneousMorphTargets: i.NUM_MORPH_INFLUENCERS } }, s); if (d) if (this._onEffectCreatedObservable && (XE.effect = d, XE.subMesh = t, this._onEffectCreatedObservable.notifyObservers(XE)), this.allowShaderHotSwapping && f && !d.isReady()) { if (d = f, i.markAsUnprocessed(), a.lightDisposed) return i._areLightsDisposed = !0, !1; } else n.resetCachedMaterial(), t.setEffect(d, i, this._materialContext); } return !t.effect || !t.effect.isReady() ? !1 : (i._renderId = n.getRenderId(), t.effect._wasPreviouslyReady = !0, t.effect._wasPreviouslyUsingInstances = r, this._checkScenePerformancePriority(), !0); } /** * Get a string representing the shaders built by the current node graph */ get compiledShaders() { return `// Vertex shader ${this._vertexCompilationState.compilationString} // Fragment shader ${this._fragmentCompilationState.compilationString}`; } /** * Binds the world matrix to the material * @param world defines the world transformation matrix */ bindOnlyWorldMatrix(e) { const t = this.getScene(); if (!this._activeEffect) return; const r = this._sharedData.hints; r.needWorldViewMatrix && e.multiplyToRef(t.getViewMatrix(), this._cachedWorldViewMatrix), r.needWorldViewProjectionMatrix && e.multiplyToRef(t.getTransformMatrix(), this._cachedWorldViewProjectionMatrix); for (const n of this._sharedData.inputBlocks) n._transmitWorld(this._activeEffect, e, this._cachedWorldViewMatrix, this._cachedWorldViewProjectionMatrix); } /** * Binds the submesh to this material by preparing the effect and shader to draw * @param world defines the world transformation matrix * @param mesh defines the mesh containing the submesh * @param subMesh defines the submesh to bind the material to */ bindForSubMesh(e, t, r) { const n = this.getScene(), i = r.effect; if (!i) return; this._activeEffect = i, this.bindOnlyWorldMatrix(e); const s = this._mustRebind(n, i, t.visibility), a = this._sharedData; if (s) { for (const f of a.bindableBlocks) f.bind(i, this, t, r); for (const f of a.forcedBindableBlocks) f.bind(i, this, t, r); for (const f of a.inputBlocks) f._transmit(i, n, this); } else if (!this.isFrozen) for (const f of a.forcedBindableBlocks) f.bind(i, this, t, r); this._afterBind(t, this._activeEffect); } /** * Gets the active textures from the material * @returns an array of textures */ getActiveTextures() { const e = super.getActiveTextures(); return this._sharedData && e.push(...this._sharedData.textureBlocks.filter((t) => t.texture).map((t) => t.texture)), e; } /** * Gets the list of texture blocks * Note that this method will only return blocks that are reachable from the final block(s) and only after the material has been built! * @returns an array of texture blocks */ getTextureBlocks() { return this._sharedData ? this._sharedData.textureBlocks : []; } /** * Gets the list of all texture blocks * Note that this method will scan all attachedBlocks and return blocks that are texture blocks * @returns */ getAllTextureBlocks() { const e = []; for (const t of this.attachedBlocks) ja._BlockIsTextureBlock(t) && e.push(t); return e; } /** * Specifies if the material uses a texture * @param texture defines the texture to check against the material * @returns a boolean specifying if the material uses the texture */ hasTexture(e) { if (super.hasTexture(e)) return !0; if (!this._sharedData) return !1; for (const t of this._sharedData.textureBlocks) if (t.texture === e) return !0; return !1; } /** * Disposes the material * @param forceDisposeEffect specifies if effects should be forcefully disposed * @param forceDisposeTextures specifies if textures should be forcefully disposed * @param notBoundToMesh specifies if the material that is being disposed is known to be not bound to any mesh */ dispose(e, t, r) { if (t) for (const n of this.getTextureBlocks().filter((i) => i.texture).map((i) => i.texture)) n.dispose(); for (const n of this.attachedBlocks) n.dispose(); this.attachedBlocks.length = 0, this._sharedData = null, this._vertexCompilationState = null, this._fragmentCompilationState = null, this.onBuildObservable.clear(), this._imageProcessingObserver && (this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver), this._imageProcessingObserver = null), super.dispose(e, t, r); } /** Creates the node editor window. */ _createNodeEditor(e) { const t = Object.assign({ nodeMaterial: this }, e); this.BJSNODEMATERIALEDITOR.NodeEditor.Show(t); } /** * Launch the node material editor * @param config Define the configuration of the editor * @returns a promise fulfilled when the node editor is visible */ edit(e) { return new Promise((t) => { if (this.BJSNODEMATERIALEDITOR = this.BJSNODEMATERIALEDITOR || this._getGlobalNodeMaterialEditor(), typeof this.BJSNODEMATERIALEDITOR > "u") { const r = e && e.editorURL ? e.editorURL : ja.EditorURL; ye.LoadBabylonScript(r, () => { this.BJSNODEMATERIALEDITOR = this.BJSNODEMATERIALEDITOR || this._getGlobalNodeMaterialEditor(), this._createNodeEditor(e == null ? void 0 : e.nodeEditorConfig), t(); }); } else this._createNodeEditor(e == null ? void 0 : e.nodeEditorConfig), t(); }); } /** * Clear the current material */ clear() { this._vertexOutputNodes.length = 0, this._fragmentOutputNodes.length = 0, this.attachedBlocks.length = 0; } /** * Clear the current material and set it to a default state */ setToDefault() { this.clear(), this.editorData = null; const e = new gi("Position"); e.setAsAttribute("position"); const t = new gi("World"); t.setAsSystemValue(Bi.World); const r = new JC("WorldPos"); e.connectTo(r), t.connectTo(r); const n = new gi("ViewProjection"); n.setAsSystemValue(Bi.ViewProjection); const i = new JC("WorldPos * ViewProjectionTransform"); r.connectTo(i), n.connectTo(i); const s = new mI("VertexOutput"); i.connectTo(s); const a = new gi("color"); a.value = new xt(0.8, 0.8, 0.8, 1); const f = new iq("FragmentOutput"); a.connectTo(f), this.addOutputNode(s), this.addOutputNode(f), this._mode = Vv.Material; } /** * Clear the current material and set it to a default state for post process */ setToDefaultPostProcess() { this.clear(), this.editorData = null; const e = new gi("Position"); e.setAsAttribute("position2d"); const t = new gi("Constant1"); t.isConstant = !0, t.value = 1; const r = new BI("Position3D"); e.connectTo(r), t.connectTo(r, { input: "w" }); const n = new mI("VertexOutput"); r.connectTo(n); const i = new gi("Scale"); i.visibleInInspector = !0, i.value = new at(1, 1); const s = new jR("uv0"); e.connectTo(s); const a = new zC("UV scale"); s.connectTo(a), i.connectTo(a); const f = new oQ("CurrentScreen"); a.connectTo(f), f.texture = new We("https://assets.babylonjs.com/nme/currentScreenPostProcess.png", this.getScene()); const o = new iq("FragmentOutput"); f.connectTo(o, { output: "rgba" }), this.addOutputNode(n), this.addOutputNode(o), this._mode = Vv.PostProcess; } /** * Clear the current material and set it to a default state for procedural texture */ setToDefaultProceduralTexture() { this.clear(), this.editorData = null; const e = new gi("Position"); e.setAsAttribute("position2d"); const t = new gi("Constant1"); t.isConstant = !0, t.value = 1; const r = new BI("Position3D"); e.connectTo(r), t.connectTo(r, { input: "w" }); const n = new mI("VertexOutput"); r.connectTo(n); const i = new gi("Time"); i.value = 0, i.min = 0, i.max = 0, i.isBoolean = !1, i.matrixMode = 0, i.animationType = tq.Time, i.isConstant = !1; const s = new gi("Color3"); s.value = new Ne(1, 1, 1), s.isConstant = !1; const a = new iq("FragmentOutput"), f = new BI("VectorMerger"); f.visibleInInspector = !1; const o = new uQ("Cos"); o.operation = o9.Cos, e.connectTo(f), i.output.connectTo(o.input), o.output.connectTo(f.z), f.xyzOut.connectTo(a.rgb), this.addOutputNode(n), this.addOutputNode(a), this._mode = Vv.ProceduralTexture; } /** * Clear the current material and set it to a default state for particle */ setToDefaultParticle() { this.clear(), this.editorData = null; const e = new gi("uv"); e.setAsAttribute("particle_uv"); const t = new fQ("ParticleTexture"); e.connectTo(t); const r = new gi("Color"); r.setAsAttribute("particle_color"); const n = new zC("Texture * Color"); t.connectTo(n), r.connectTo(n); const i = new AQ("ParticleRampGradient"); n.connectTo(i); const s = new vQ("ColorSplitter"); r.connectTo(s); const a = new dQ("ParticleBlendMultiply"); i.connectTo(a), t.connectTo(a, { output: "a" }), s.connectTo(a, { output: "a" }); const f = new iq("FragmentOutput"); a.connectTo(f), this.addOutputNode(f), this._mode = Vv.Particle; } /** * Loads the current Node Material from a url pointing to a file save by the Node Material Editor * @deprecated Please use NodeMaterial.ParseFromFileAsync instead * @param url defines the url to load from * @param rootUrl defines the root URL for nested url in the node material * @returns a promise that will fulfil when the material is fully loaded */ async loadAsync(e, t = "") { return ja.ParseFromFileAsync("", e, this.getScene(), t, !0, this); } _gatherBlocks(e, t) { if (t.indexOf(e) === -1) { t.push(e); for (const r of e.inputs) { const n = r.connectedPoint; if (n) { const i = n.ownerBlock; i !== e && this._gatherBlocks(i, t); } } if (e.isTeleportOut) { const r = e; r.entryPoint && this._gatherBlocks(r.entryPoint, t); } } } /** * Generate a string containing the code declaration required to create an equivalent of this material * @returns a string */ generateCode() { let e = []; const t = [], r = ["const", "var", "let"]; for (const s of this._vertexOutputNodes) this._gatherBlocks(s, t); const n = []; for (const s of this._fragmentOutputNodes) this._gatherBlocks(s, n); let i = `var nodeMaterial = new BABYLON.NodeMaterial("${this.name || "node material"}"); `; i += `nodeMaterial.mode = BABYLON.NodeMaterialModes.${Vv[this.mode]}; `; for (const s of t) s.isInput && e.indexOf(s) === -1 && (i += s._dumpCode(r, e)); for (const s of n) s.isInput && e.indexOf(s) === -1 && (i += s._dumpCode(r, e)); e = [], i += ` // Connections `; for (const s of this._vertexOutputNodes) i += s._dumpCodeForOutputConnections(e); for (const s of this._fragmentOutputNodes) i += s._dumpCodeForOutputConnections(e); i += ` // Output nodes `; for (const s of this._vertexOutputNodes) i += `nodeMaterial.addOutputNode(${s._codeVariableName}); `; for (const s of this._fragmentOutputNodes) i += `nodeMaterial.addOutputNode(${s._codeVariableName}); `; return i += `nodeMaterial.build(); `, i; } /** * Serializes this material in a JSON representation * @param selectedBlocks * @returns the serialized material object */ serialize(e) { const t = e ? {} : jt.Serialize(this); t.editorData = JSON.parse(JSON.stringify(this.editorData)); let r = []; if (e) r = e; else { t.customType = "BABYLON.NodeMaterial", t.outputNodes = []; for (const n of this._vertexOutputNodes) this._gatherBlocks(n, r), t.outputNodes.push(n.uniqueId); for (const n of this._fragmentOutputNodes) this._gatherBlocks(n, r), t.outputNodes.indexOf(n.uniqueId) === -1 && t.outputNodes.push(n.uniqueId); } t.blocks = []; for (const n of r) t.blocks.push(n.serialize()); if (!e) for (const n of this.attachedBlocks) r.indexOf(n) === -1 && t.blocks.push(n.serialize()); return t; } _restoreConnections(e, t, r) { for (const n of e.outputs) for (const i of t.blocks) { const s = r[i.id]; if (s) { for (const a of i.inputs) if (r[a.targetBlockId] === e && a.targetConnectionName === n.name) { const f = s.getInputByName(a.inputName); if (!f || f.isConnected) continue; n.connectTo(f, !0), this._restoreConnections(s, t, r); continue; } } } } /** * Clear the current graph and load a new one from a serialization object * @param source defines the JSON representation of the material * @param rootUrl defines the root URL to use to load textures and relative dependencies * @param merge defines whether or not the source must be merged or replace the current content */ parseSerializedObject(e, t = "", r = !1) { var n; r || this.clear(); const i = {}; for (const s of e.blocks) { const a = Jo(s.customType); if (a) { const f = new a(); f._deserialize(s, this.getScene(), t), i[s.id] = f, this.attachedBlocks.push(f); } } for (const s of this.attachedBlocks) if (s.isTeleportOut) { const a = s, f = a._tempEntryPointUniqueId; f && i[f].attachToEndpoint(a); } for (let s = 0; s < e.blocks.length; s++) { const a = e.blocks[s], f = i[a.id]; f && (f.inputs.length && !r || this._restoreConnections(f, e, i)); } if (e.outputNodes) for (const s of e.outputNodes) this.addOutputNode(i[s]); if (e.locations || e.editorData && e.editorData.locations) { const s = e.locations || e.editorData.locations; for (const f of s) i[f.blockId] && (f.blockId = i[f.blockId].uniqueId); r && this.editorData && this.editorData.locations && s.concat(this.editorData.locations), e.locations ? this.editorData = { locations: s } : (this.editorData = e.editorData, this.editorData.locations = s); const a = []; for (const f in i) a[f] = i[f].uniqueId; this.editorData.map = a; } this.comment = e.comment, e.forceAlphaBlending !== void 0 && (this.forceAlphaBlending = e.forceAlphaBlending), r || (this._mode = (n = e.mode) !== null && n !== void 0 ? n : Vv.Material); } /** * Clear the current graph and load a new one from a serialization object * @param source defines the JSON representation of the material * @param rootUrl defines the root URL to use to load textures and relative dependencies * @param merge defines whether or not the source must be merged or replace the current content * @deprecated Please use the parseSerializedObject method instead */ loadFromSerialization(e, t = "", r = !1) { this.parseSerializedObject(e, t, r); } /** * Makes a duplicate of the current material. * @param name defines the name to use for the new material * @param shareEffect defines if the clone material should share the same effect (default is false) */ clone(e, t = !1) { const r = this.serialize(), n = jt.Clone(() => new ja(e, this.getScene(), this.options), this); return n.id = e, n.name = e, n.parseSerializedObject(r), n._buildId = this._buildId, n.build(!1, !t), n; } /** * Awaits for all the material textures to be ready before resolving the returned promise. */ whenTexturesReadyAsync() { const e = []; return this.getActiveTextures().forEach((t) => { const r = t.getInternalTexture(); r && !r.isReady && e.push(new Promise((n, i) => { r.onLoadedObservable.addOnce(() => { n(); }), r.onErrorObservable.addOnce((s) => { i(s); }); })); }), Promise.all(e); } /** * Creates a node material from parsed material data * @param source defines the JSON representation of the material * @param scene defines the hosting scene * @param rootUrl defines the root URL to use to load textures and relative dependencies * @returns a new node material */ static Parse(e, t, r = "") { const n = jt.Parse(() => new ja(e.name, t), e, t, r); return n.parseSerializedObject(e, r), n.build(), n; } /** * Creates a node material from a snippet saved in a remote file * @param name defines the name of the material to create * @param url defines the url to load from * @param scene defines the hosting scene * @param rootUrl defines the root URL for nested url in the node material * @param skipBuild defines whether to build the node material * @param targetMaterial defines a material to use instead of creating a new one * @returns a promise that will resolve to the new node material */ static async ParseFromFileAsync(e, t, r, n = "", i = !1, s) { const a = s ?? new ja(e, r), f = await r._loadFileAsync(t), o = JSON.parse(f); return a.parseSerializedObject(o, n), i || a.build(), a; } /** * Creates a node material from a snippet saved by the node material editor * @param snippetId defines the snippet to load * @param scene defines the hosting scene * @param rootUrl defines the root URL to use to load textures and relative dependencies * @param nodeMaterial defines a node material to update (instead of creating a new one) * @param skipBuild defines whether to build the node material * @param waitForTextureReadyness defines whether to wait for texture readiness resolving the promise (default: false) * @returns a promise that will resolve to the new node material */ static ParseFromSnippetAsync(e, t = gr.LastCreatedScene, r = "", n, i = !1, s = !1) { return e === "_BLANK" ? Promise.resolve(ja.CreateDefault("blank", t)) : new Promise((a, f) => { const o = new ho(); o.addEventListener("readystatechange", () => { if (o.readyState == 4) if (o.status == 200) { const d = JSON.parse(JSON.parse(o.responseText).jsonPayload), v = JSON.parse(d.nodeMaterial); n || (n = jt.Parse(() => new ja(e, t), v, t, r), n.uniqueId = t.getUniqueId()), n.parseSerializedObject(v), n.snippetId = e; try { i || n.build(); } catch (u) { f(u); } s ? n.whenTexturesReadyAsync().then(() => { a(n); }).catch((u) => { f(u); }) : a(n); } else f("Unable to load the snippet " + e); }), o.open("GET", this.SnippetUrl + "/" + e.replace(/#/g, "/")), o.send(); }); } /** * Creates a new node material set to default basic configuration * @param name defines the name of the material * @param scene defines the hosting scene * @returns a new NodeMaterial */ static CreateDefault(e, t) { const r = new ja(e, t); return r.setToDefault(), r.build(), r; } } ja._BuildIdGenerator = 0; ja.EditorURL = `${ye._DefaultCdnUrl}/v${Ge.Version}/nodeEditor/babylon.nodeEditor.js`; ja.SnippetUrl = "https://snippet.babylonjs.com"; ja.IgnoreTexturesAtLoadTime = !1; C([ M() ], ja.prototype, "ignoreAlpha", void 0); C([ M() ], ja.prototype, "maxSimultaneousLights", void 0); C([ M("mode") ], ja.prototype, "_mode", void 0); C([ M("comment") ], ja.prototype, "comment", void 0); C([ M() ], ja.prototype, "forceAlphaBlending", void 0); Ue("BABYLON.NodeMaterial", ja); function RO(A) { const e = A.sideOrientation || Ut.DEFAULTSIDE, t = A.radius || 1, r = A.flat === void 0 ? !0 : A.flat, n = (A.subdivisions || 4) | 0, i = A.radiusX || t, s = A.radiusY || t, a = A.radiusZ || t, f = (1 + Math.sqrt(5)) / 2, o = [ -1, f, -0, 1, f, 0, -1, -f, 0, 1, -f, 0, 0, -1, -f, 0, 1, -f, 0, -1, f, 0, 1, f, f, 0, 1, f, 0, -1, -f, 0, 1, -f, 0, -1 // v8-11 ], d = [ 0, 11, 5, 0, 5, 1, 0, 1, 7, 0, 7, 10, 12, 22, 23, 1, 5, 20, 5, 11, 4, 23, 22, 13, 22, 18, 6, 7, 1, 8, 14, 21, 4, 14, 4, 2, 16, 13, 6, 15, 6, 19, 3, 8, 9, 4, 21, 5, 13, 17, 23, 6, 13, 22, 19, 6, 18, 9, 8, 1 ], v = [ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, // vertex alias 0, 2, 3, 3, 3, 4, 7, 8, 9, 9, 10, 11 // 23: B + 12 ], u = [ 5, 1, 3, 1, 6, 4, 0, 0, 5, 3, 4, 2, 2, 2, 4, 0, 2, 0, 1, 1, 6, 0, 6, 2, // vertex alias (for same vertex on different faces) 0, 4, 3, 3, 4, 4, 3, 1, 4, 2, 4, 4, 0, 2, 1, 1, 2, 2, 3, 3, 1, 3, 2, 4 // 23: B + 12 ], l = 138 / 1024, P = 239 / 1024, p = 60 / 1024, c = 26 / 1024, H = -40 / 1024, T = 20 / 1024, q = [ 0, 0, 0, 0, 1, 0, 0, 1, 1, 0, 0, 0, 1, 1, 0, 0, 1, 1, 1, 0 // 15 - 19 ], b = [], j = [], w = [], m = []; let I = 0; const N = new Array(3), k = new Array(3); let R; for (R = 0; R < 3; R++) N[R] = S.Zero(), k[R] = at.Zero(); for (let O = 0; O < 20; O++) { for (R = 0; R < 3; R++) { const ee = d[3 * O + R]; N[R].copyFromFloats(o[3 * v[ee]], o[3 * v[ee] + 1], o[3 * v[ee] + 2]), N[R].normalize(), k[R].copyFromFloats(u[2 * ee] * l + p + q[O] * H, u[2 * ee + 1] * P + c + q[O] * T); } const Y = (ee, Z, te, fe) => { const _ = S.Lerp(N[0], N[2], Z / n), G = S.Lerp(N[1], N[2], Z / n), L = n === Z ? N[2] : S.Lerp(_, G, ee / (n - Z)); L.normalize(); let $; if (r) { const me = S.Lerp(N[0], N[2], fe / n), Xe = S.Lerp(N[1], N[2], fe / n); $ = S.Lerp(me, Xe, te / (n - fe)); } else $ = new S(L.x, L.y, L.z); $.x /= i, $.y /= s, $.z /= a, $.normalize(); const ae = at.Lerp(k[0], k[2], Z / n), Pe = at.Lerp(k[1], k[2], Z / n), ge = n === Z ? k[2] : at.Lerp(ae, Pe, ee / (n - Z)); j.push(L.x * i, L.y * s, L.z * a), w.push($.x, $.y, $.z), m.push(ge.x, us.UseOpenGLOrientationForUV ? 1 - ge.y : ge.y), b.push(I), I++; }; for (let ee = 0; ee < n; ee++) for (let Z = 0; Z + ee < n; Z++) Y(Z, ee, Z + 1 / 3, ee + 1 / 3), Y(Z + 1, ee, Z + 1 / 3, ee + 1 / 3), Y(Z, ee + 1, Z + 1 / 3, ee + 1 / 3), Z + ee + 1 < n && (Y(Z + 1, ee, Z + 2 / 3, ee + 2 / 3), Y(Z + 1, ee + 1, Z + 2 / 3, ee + 2 / 3), Y(Z, ee + 1, Z + 2 / 3, ee + 2 / 3)); } Ut._ComputeSides(e, j, b, w, m, A.frontUVs, A.backUVs); const y = new Ut(); return y.indices = b, y.positions = j, y.normals = w, y.uvs = m, y; } function UR(A, e = {}, t = null) { const r = new Ee(A, t); return e.sideOrientation = Ee._GetDefaultSideOrientation(e.sideOrientation), r._originalBuilderSideOrientation = e.sideOrientation, RO(e).applyToMesh(r, e.updatable), r; } const y6e = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateIcoSphere: UR }; Ut.CreateIcoSphere = RO; Ee.CreateIcoSphere = (A, e, t) => UR(A, e, t); var J8; (function(A) { A.WRIST = "wrist", A.THUMB = "thumb", A.INDEX = "index", A.MIDDLE = "middle", A.RING = "ring", A.LITTLE = "little"; })(J8 || (J8 = {})); var gn; (function(A) { A.WRIST = "wrist", A.THUMB_METACARPAL = "thumb-metacarpal", A.THUMB_PHALANX_PROXIMAL = "thumb-phalanx-proximal", A.THUMB_PHALANX_DISTAL = "thumb-phalanx-distal", A.THUMB_TIP = "thumb-tip", A.INDEX_FINGER_METACARPAL = "index-finger-metacarpal", A.INDEX_FINGER_PHALANX_PROXIMAL = "index-finger-phalanx-proximal", A.INDEX_FINGER_PHALANX_INTERMEDIATE = "index-finger-phalanx-intermediate", A.INDEX_FINGER_PHALANX_DISTAL = "index-finger-phalanx-distal", A.INDEX_FINGER_TIP = "index-finger-tip", A.MIDDLE_FINGER_METACARPAL = "middle-finger-metacarpal", A.MIDDLE_FINGER_PHALANX_PROXIMAL = "middle-finger-phalanx-proximal", A.MIDDLE_FINGER_PHALANX_INTERMEDIATE = "middle-finger-phalanx-intermediate", A.MIDDLE_FINGER_PHALANX_DISTAL = "middle-finger-phalanx-distal", A.MIDDLE_FINGER_TIP = "middle-finger-tip", A.RING_FINGER_METACARPAL = "ring-finger-metacarpal", A.RING_FINGER_PHALANX_PROXIMAL = "ring-finger-phalanx-proximal", A.RING_FINGER_PHALANX_INTERMEDIATE = "ring-finger-phalanx-intermediate", A.RING_FINGER_PHALANX_DISTAL = "ring-finger-phalanx-distal", A.RING_FINGER_TIP = "ring-finger-tip", A.PINKY_FINGER_METACARPAL = "pinky-finger-metacarpal", A.PINKY_FINGER_PHALANX_PROXIMAL = "pinky-finger-phalanx-proximal", A.PINKY_FINGER_PHALANX_INTERMEDIATE = "pinky-finger-phalanx-intermediate", A.PINKY_FINGER_PHALANX_DISTAL = "pinky-finger-phalanx-distal", A.PINKY_FINGER_TIP = "pinky-finger-tip"; })(gn || (gn = {})); const NH = [ gn.WRIST, gn.THUMB_METACARPAL, gn.THUMB_PHALANX_PROXIMAL, gn.THUMB_PHALANX_DISTAL, gn.THUMB_TIP, gn.INDEX_FINGER_METACARPAL, gn.INDEX_FINGER_PHALANX_PROXIMAL, gn.INDEX_FINGER_PHALANX_INTERMEDIATE, gn.INDEX_FINGER_PHALANX_DISTAL, gn.INDEX_FINGER_TIP, gn.MIDDLE_FINGER_METACARPAL, gn.MIDDLE_FINGER_PHALANX_PROXIMAL, gn.MIDDLE_FINGER_PHALANX_INTERMEDIATE, gn.MIDDLE_FINGER_PHALANX_DISTAL, gn.MIDDLE_FINGER_TIP, gn.RING_FINGER_METACARPAL, gn.RING_FINGER_PHALANX_PROXIMAL, gn.RING_FINGER_PHALANX_INTERMEDIATE, gn.RING_FINGER_PHALANX_DISTAL, gn.RING_FINGER_TIP, gn.PINKY_FINGER_METACARPAL, gn.PINKY_FINGER_PHALANX_PROXIMAL, gn.PINKY_FINGER_PHALANX_INTERMEDIATE, gn.PINKY_FINGER_PHALANX_DISTAL, gn.PINKY_FINGER_TIP ], k6e = { [J8.WRIST]: [gn.WRIST], [J8.THUMB]: [gn.THUMB_METACARPAL, gn.THUMB_PHALANX_PROXIMAL, gn.THUMB_PHALANX_DISTAL, gn.THUMB_TIP], [J8.INDEX]: [ gn.INDEX_FINGER_METACARPAL, gn.INDEX_FINGER_PHALANX_PROXIMAL, gn.INDEX_FINGER_PHALANX_INTERMEDIATE, gn.INDEX_FINGER_PHALANX_DISTAL, gn.INDEX_FINGER_TIP ], [J8.MIDDLE]: [ gn.MIDDLE_FINGER_METACARPAL, gn.MIDDLE_FINGER_PHALANX_PROXIMAL, gn.MIDDLE_FINGER_PHALANX_INTERMEDIATE, gn.MIDDLE_FINGER_PHALANX_DISTAL, gn.MIDDLE_FINGER_TIP ], [J8.RING]: [ gn.RING_FINGER_METACARPAL, gn.RING_FINGER_PHALANX_PROXIMAL, gn.RING_FINGER_PHALANX_INTERMEDIATE, gn.RING_FINGER_PHALANX_DISTAL, gn.RING_FINGER_TIP ], [J8.LITTLE]: [ gn.PINKY_FINGER_METACARPAL, gn.PINKY_FINGER_PHALANX_PROXIMAL, gn.PINKY_FINGER_PHALANX_INTERMEDIATE, gn.PINKY_FINGER_PHALANX_DISTAL, gn.PINKY_FINGER_TIP ] }; class qee { /** * Get the hand mesh. */ get handMesh() { return this._handMesh; } /** * Get meshes of part of the hand. * @param part The part of hand to get. * @returns An array of meshes that correlate to the hand part requested. */ getHandPartMeshes(e) { return k6e[e].map((t) => this._jointMeshes[NH.indexOf(t)]); } /** * Retrieves a mesh linked to a named joint in the hand. * @param jointName The name of the joint. * @returns An AbstractMesh whose position corresponds with the joint position. */ getJointMesh(e) { return this._jointMeshes[NH.indexOf(e)]; } /** * Construct a new hand object * @param xrController The controller to which the hand correlates. * @param _jointMeshes The meshes to be used to track the hand joints. * @param _handMesh An optional hand mesh. * @param rigMapping An optional rig mapping for the hand mesh. * If not provided (but a hand mesh is provided), * it will be assumed that the hand mesh's bones are named * directly after the WebXR bone names. * @param _leftHandedMeshes Are the hand meshes left-handed-system meshes * @param _jointsInvisible Are the tracked joint meshes visible * @param _jointScaleFactor Scale factor for all joint meshes */ constructor(e, t, r, n, i = !1, s = !1, a = 1) { this.xrController = e, this._jointMeshes = t, this._handMesh = r, this.rigMapping = n, this._leftHandedMeshes = i, this._jointsInvisible = s, this._jointScaleFactor = a, this._jointTransforms = new Array(NH.length), this._jointTransformMatrices = new Float32Array(NH.length * 16), this._tempJointMatrix = new he(), this._jointRadii = new Float32Array(NH.length), this._scene = t[0].getScene(); for (let f = 0; f < this._jointTransforms.length; f++) { const o = this._jointTransforms[f] = new Hr(NH[f], this._scene); o.rotationQuaternion = new Ze(), t[f].rotationQuaternion = new Ze(); } r && this.setHandMesh(r, n), this.xrController.motionController && (this.xrController.motionController.rootMesh ? this.xrController.motionController.rootMesh.setEnabled(!1) : this.xrController.motionController.onModelLoadedObservable.add((f) => { f.rootMesh && f.rootMesh.setEnabled(!1); })), this.xrController.onMotionControllerInitObservable.add((f) => { f.onModelLoadedObservable.add((o) => { o.rootMesh && o.rootMesh.setEnabled(!1); }), f.rootMesh && f.rootMesh.setEnabled(!1); }); } /** * Sets the current hand mesh to render for the WebXRHand. * @param handMesh The rigged hand mesh that will be tracked to the user's hand. * @param rigMapping The mapping from XRHandJoint to bone names to use with the mesh. */ setHandMesh(e, t) { if (this._handMesh = e, e.alwaysSelectAsActiveMesh = !0, e.getChildMeshes().forEach((r) => r.alwaysSelectAsActiveMesh = !0), this._handMesh.skeleton) { const r = this._handMesh.skeleton; NH.forEach((n, i) => { const s = r.getBoneIndexByName(t ? t[n] : n); s !== -1 && r.bones[s].linkTransformNode(this._jointTransforms[i]); }); } } /** * Update this hand from the latest xr frame. * @param xrFrame The latest frame received from WebXR. * @param referenceSpace The current viewer reference space. */ updateFromXRFrame(e, t) { const r = this.xrController.inputSource.hand; if (!r) return; const n = r, i = NH.map((a) => n[a] || r.get(a)); let s = !1; if (e.fillPoses && e.fillJointRadii) s = e.fillPoses(i, t, this._jointTransformMatrices) && e.fillJointRadii(i, this._jointRadii); else if (e.getJointPose) { s = !0; for (let a = 0; a < i.length; a++) { const f = e.getJointPose(i[a], t); if (f) this._jointTransformMatrices.set(f.transform.matrix, a * 16), this._jointRadii[a] = f.radius || 8e-3; else { s = !1; break; } } } s && (NH.forEach((a, f) => { const o = this._jointTransforms[f]; he.FromArrayToRef(this._jointTransformMatrices, f * 16, this._tempJointMatrix), this._tempJointMatrix.decompose(void 0, o.rotationQuaternion, o.position); const d = this._jointRadii[f] * this._jointScaleFactor, v = this._jointMeshes[f]; v.isVisible = !this._handMesh && !this._jointsInvisible, v.position.copyFrom(o.position), v.rotationQuaternion.copyFrom(o.rotationQuaternion), v.scaling.setAll(d), this._scene.useRightHandedSystem || (v.position.z *= -1, v.rotationQuaternion.z *= -1, v.rotationQuaternion.w *= -1, this._leftHandedMeshes && this._handMesh && (o.position.z *= -1, o.rotationQuaternion.z *= -1, o.rotationQuaternion.w *= -1)); }), this._handMesh && (this._handMesh.isVisible = !0)); } /** * Dispose this Hand object */ dispose() { this._handMesh && (this._handMesh.isVisible = !1); } } class va extends L9 { static _GenerateTrackedJointMeshes(e) { const t = {}; return ["left", "right"].map((r) => { var n, i, s, a, f; const o = [], d = ((n = e.jointMeshes) === null || n === void 0 ? void 0 : n.sourceMesh) || UR("jointParent", va._ICOSPHERE_PARAMS); d.isVisible = !!(!((i = e.jointMeshes) === null || i === void 0) && i.keepOriginalVisible); for (let v = 0; v < NH.length; ++v) { let u = d.createInstance(`${r}-handJoint-${v}`); if (!((s = e.jointMeshes) === null || s === void 0) && s.onHandJointMeshGenerated) { const l = e.jointMeshes.onHandJointMeshGenerated(u, v, r); l && l !== u && (u.dispose(), u = l); } if (u.isPickable = !1, !((a = e.jointMeshes) === null || a === void 0) && a.enablePhysics) { const l = ((f = e.jointMeshes) === null || f === void 0 ? void 0 : f.physicsProps) || {}; u.scaling.setAll(0.02); const P = l.impostorType !== void 0 ? l.impostorType : tn.SphereImpostor; u.physicsImpostor = new tn(u, P, Object.assign({ mass: 0 }, l)); } u.rotationQuaternion = new Ze(), u.isVisible = !1, o.push(u); } t[r] = o; }), { left: t.left, right: t.right }; } static _GenerateDefaultHandMeshesAsync(e, t) { return new Promise(async (r) => { var n, i, s, a, f; const o = {}; !((i = (n = va._RightHandGLB) === null || n === void 0 ? void 0 : n.meshes[1]) === null || i === void 0) && i.isDisposed() && (va._RightHandGLB = null), !((a = (s = va._LeftHandGLB) === null || s === void 0 ? void 0 : s.meshes[1]) === null || a === void 0) && a.isDisposed() && (va._LeftHandGLB = null); const d = !!(va._RightHandGLB && va._LeftHandGLB), v = await Promise.all([ va._RightHandGLB || Hn.ImportMeshAsync("", va.DEFAULT_HAND_MODEL_BASE_URL, va.DEFAULT_HAND_MODEL_RIGHT_FILENAME, e), va._LeftHandGLB || Hn.ImportMeshAsync("", va.DEFAULT_HAND_MODEL_BASE_URL, va.DEFAULT_HAND_MODEL_LEFT_FILENAME, e) ]); va._RightHandGLB = v[0], va._LeftHandGLB = v[1]; const u = new ja("handShader", e, { emitComments: !1 }); await u.loadAsync(va.DEFAULT_HAND_MODEL_SHADER_URL), u.needDepthPrePass = !0, u.transparencyMode = gt.MATERIAL_ALPHABLEND, u.alphaMode = 2, u.build(!1); const l = Object.assign({ base: Ne.FromInts(116, 63, 203), fresnel: Ne.FromInts(149, 102, 229), fingerColor: Ne.FromInts(177, 130, 255), tipFresnel: Ne.FromInts(220, 200, 255) }, (f = t == null ? void 0 : t.handMeshes) === null || f === void 0 ? void 0 : f.customColors), P = { base: u.getBlockByName("baseColor"), fresnel: u.getBlockByName("fresnelColor"), fingerColor: u.getBlockByName("fingerColor"), tipFresnel: u.getBlockByName("tipFresnelColor") }; P.base.value = l.base, P.fresnel.value = l.fresnel, P.fingerColor.value = l.fingerColor, P.tipFresnel.value = l.tipFresnel, ["left", "right"].forEach((p) => { const c = p == "left" ? va._LeftHandGLB : va._RightHandGLB; if (!c) throw new Error("Could not load hand model"); const H = c.meshes[1]; H._internalAbstractMeshDataInfo._computeBonesUsingShaders = !0, H.material = u.clone(`${p}HandShaderClone`, !0), H.isVisible = !1, o[p] = H, !d && !e.useRightHandedSystem && c.meshes[1].rotate(bf.Y, Math.PI); }), u.dispose(), r({ left: o.left, right: o.right }); }); } /** * Generates a mapping from XRHandJoint to bone name for the default hand mesh. * @param handedness The handedness being mapped for. */ static _GenerateDefaultHandMeshRigMapping(e) { const t = e == "right" ? "R" : "L"; return { [gn.WRIST]: `wrist_${t}`, [gn.THUMB_METACARPAL]: `thumb_metacarpal_${t}`, [gn.THUMB_PHALANX_PROXIMAL]: `thumb_proxPhalanx_${t}`, [gn.THUMB_PHALANX_DISTAL]: `thumb_distPhalanx_${t}`, [gn.THUMB_TIP]: `thumb_tip_${t}`, [gn.INDEX_FINGER_METACARPAL]: `index_metacarpal_${t}`, [gn.INDEX_FINGER_PHALANX_PROXIMAL]: `index_proxPhalanx_${t}`, [gn.INDEX_FINGER_PHALANX_INTERMEDIATE]: `index_intPhalanx_${t}`, [gn.INDEX_FINGER_PHALANX_DISTAL]: `index_distPhalanx_${t}`, [gn.INDEX_FINGER_TIP]: `index_tip_${t}`, [gn.MIDDLE_FINGER_METACARPAL]: `middle_metacarpal_${t}`, [gn.MIDDLE_FINGER_PHALANX_PROXIMAL]: `middle_proxPhalanx_${t}`, [gn.MIDDLE_FINGER_PHALANX_INTERMEDIATE]: `middle_intPhalanx_${t}`, [gn.MIDDLE_FINGER_PHALANX_DISTAL]: `middle_distPhalanx_${t}`, [gn.MIDDLE_FINGER_TIP]: `middle_tip_${t}`, [gn.RING_FINGER_METACARPAL]: `ring_metacarpal_${t}`, [gn.RING_FINGER_PHALANX_PROXIMAL]: `ring_proxPhalanx_${t}`, [gn.RING_FINGER_PHALANX_INTERMEDIATE]: `ring_intPhalanx_${t}`, [gn.RING_FINGER_PHALANX_DISTAL]: `ring_distPhalanx_${t}`, [gn.RING_FINGER_TIP]: `ring_tip_${t}`, [gn.PINKY_FINGER_METACARPAL]: `little_metacarpal_${t}`, [gn.PINKY_FINGER_PHALANX_PROXIMAL]: `little_proxPhalanx_${t}`, [gn.PINKY_FINGER_PHALANX_INTERMEDIATE]: `little_intPhalanx_${t}`, [gn.PINKY_FINGER_PHALANX_DISTAL]: `little_distPhalanx_${t}`, [gn.PINKY_FINGER_TIP]: `little_tip_${t}` }; } /** * Check if the needed objects are defined. * This does not mean that the feature is enabled, but that the objects needed are well defined. */ isCompatible() { return typeof XRHand < "u"; } /** * Get the hand object according to the controller id * @param controllerId the controller id to which we want to get the hand * @returns null if not found or the WebXRHand object if found */ getHandByControllerId(e) { return this._attachedHands[e]; } /** * Get a hand object according to the requested handedness * @param handedness the handedness to request * @returns null if not found or the WebXRHand object if found */ getHandByHandedness(e) { return e == "none" ? null : this._trackingHands[e]; } /** * Creates a new instance of the XR hand tracking feature. * @param _xrSessionManager An instance of WebXRSessionManager. * @param options Options to use when constructing this feature. */ constructor(e, t) { super(e), this.options = t, this._attachedHands = {}, this._trackingHands = { left: null, right: null }, this._handResources = { jointMeshes: null, handMeshes: null, rigMappings: null }, this.onHandAddedObservable = new Oe(), this.onHandRemovedObservable = new Oe(), this._attachHand = (i) => { var s, a, f; if (!i.inputSource.hand || i.inputSource.handedness == "none" || !this._handResources.jointMeshes) return; const o = i.inputSource.handedness, d = new qee(i, this._handResources.jointMeshes[o], this._handResources.handMeshes && this._handResources.handMeshes[o], this._handResources.rigMappings && this._handResources.rigMappings[o], (s = this.options.handMeshes) === null || s === void 0 ? void 0 : s.meshesUseLeftHandedCoordinates, (a = this.options.jointMeshes) === null || a === void 0 ? void 0 : a.invisible, (f = this.options.jointMeshes) === null || f === void 0 ? void 0 : f.scaleFactor); this._attachedHands[i.uniqueId] = d, this._trackingHands[o] = d, this.onHandAddedObservable.notifyObservers(d); }, this._detachHand = (i) => { this._detachHandById(i.uniqueId); }, this.xrNativeFeatureName = "hand-tracking"; const n = t.jointMeshes; if (n && (typeof n.disableDefaultHandMesh < "u" && (t.handMeshes = t.handMeshes || {}, t.handMeshes.disableDefaultMeshes = n.disableDefaultHandMesh), typeof n.handMeshes < "u" && (t.handMeshes = t.handMeshes || {}, t.handMeshes.customMeshes = n.handMeshes), typeof n.leftHandedSystemMeshes < "u" && (t.handMeshes = t.handMeshes || {}, t.handMeshes.meshesUseLeftHandedCoordinates = n.leftHandedSystemMeshes), typeof n.rigMapping < "u")) { t.handMeshes = t.handMeshes || {}; const i = {}, s = {}; [ [n.rigMapping.left, i], [n.rigMapping.right, s] ].forEach((a) => { const f = a[0], o = a[1]; f.forEach((d, v) => { o[NH[v]] = d; }); }), t.handMeshes.customRigMappings = { left: i, right: s }; } } /** * Attach this feature. * Will usually be called by the features manager. * * @returns true if successful. */ attach() { var e, t, r, n; return super.attach() ? (this._handResources = { jointMeshes: va._GenerateTrackedJointMeshes(this.options), handMeshes: ((e = this.options.handMeshes) === null || e === void 0 ? void 0 : e.customMeshes) || null, rigMappings: ((t = this.options.handMeshes) === null || t === void 0 ? void 0 : t.customRigMappings) || null }, !(!((r = this.options.handMeshes) === null || r === void 0) && r.customMeshes) && !(!((n = this.options.handMeshes) === null || n === void 0) && n.disableDefaultMeshes) && va._GenerateDefaultHandMeshesAsync(gr.LastCreatedScene, this.options).then((i) => { var s, a; this._handResources.handMeshes = i, this._handResources.rigMappings = { left: va._GenerateDefaultHandMeshRigMapping("left"), right: va._GenerateDefaultHandMeshRigMapping("right") }, (s = this._trackingHands.left) === null || s === void 0 || s.setHandMesh(this._handResources.handMeshes.left, this._handResources.rigMappings.left), (a = this._trackingHands.right) === null || a === void 0 || a.setHandMesh(this._handResources.handMeshes.right, this._handResources.rigMappings.right); }), this.options.xrInput.controllers.forEach(this._attachHand), this._addNewAttachObserver(this.options.xrInput.onControllerAddedObservable, this._attachHand), this._addNewAttachObserver(this.options.xrInput.onControllerRemovedObservable, this._detachHand), !0) : !1; } _onXRFrame(e) { var t, r; (t = this._trackingHands.left) === null || t === void 0 || t.updateFromXRFrame(e, this._xrSessionManager.referenceSpace), (r = this._trackingHands.right) === null || r === void 0 || r.updateFromXRFrame(e, this._xrSessionManager.referenceSpace); } _detachHandById(e) { var t; const r = this.getHandByControllerId(e); if (r) { const n = r.xrController.inputSource.handedness == "left" ? "left" : "right"; ((t = this._trackingHands[n]) === null || t === void 0 ? void 0 : t.xrController.uniqueId) === e && (this._trackingHands[n] = null), this.onHandRemovedObservable.notifyObservers(r), r.dispose(), delete this._attachedHands[e]; } } /** * Detach this feature. * Will usually be called by the features manager. * * @returns true if successful. */ detach() { return super.detach() ? (Object.keys(this._attachedHands).forEach((e) => this._detachHandById(e)), !0) : !1; } /** * Dispose this feature and all of the resources attached. */ dispose() { var e; super.dispose(), this.onHandAddedObservable.clear(), this.onHandRemovedObservable.clear(), this._handResources.handMeshes && !(!((e = this.options.handMeshes) === null || e === void 0) && e.customMeshes) && (this._handResources.handMeshes.left.dispose(), this._handResources.handMeshes.right.dispose(), va._RightHandGLB = null, va._LeftHandGLB = null), this._handResources.jointMeshes && (this._handResources.jointMeshes.left.forEach((t) => t.dispose()), this._handResources.jointMeshes.right.forEach((t) => t.dispose())); } } va.Name = Gi.HAND_TRACKING; va.Version = 1; va.DEFAULT_HAND_MODEL_BASE_URL = "https://assets.babylonjs.com/meshes/HandMeshes/"; va.DEFAULT_HAND_MODEL_RIGHT_FILENAME = "r_hand_rhs.glb"; va.DEFAULT_HAND_MODEL_LEFT_FILENAME = "l_hand_rhs.glb"; va.DEFAULT_HAND_MODEL_SHADER_URL = "https://assets.babylonjs.com/meshes/HandMeshes/handsShader.json"; va._ICOSPHERE_PARAMS = { radius: 0.5, flat: !1, subdivisions: 2 }; va._RightHandGLB = null; va._LeftHandGLB = null; So.AddWebXRFeature(va.Name, (A, e) => () => new va(A, e), va.Version, !1); var Kx; (function(A) { A[A.ABOVE_FINGER_TIPS = 0] = "ABOVE_FINGER_TIPS", A[A.RADIAL_SIDE = 1] = "RADIAL_SIDE", A[A.ULNAR_SIDE = 2] = "ULNAR_SIDE", A[A.BELOW_WRIST = 3] = "BELOW_WRIST"; })(Kx || (Kx = {})); var z8; (function(A) { A[A.LOOK_AT_CAMERA = 0] = "LOOK_AT_CAMERA", A[A.HAND_ROTATION = 1] = "HAND_ROTATION"; })(z8 || (z8 = {})); var Jx; (function(A) { A[A.ALWAYS_VISIBLE = 0] = "ALWAYS_VISIBLE", A[A.PALM_UP = 1] = "PALM_UP", A[A.GAZE_FOCUS = 2] = "GAZE_FOCUS", A[A.PALM_AND_GAZE = 3] = "PALM_AND_GAZE"; })(Jx || (Jx = {})); class E6e { /** * Builds a hand constraint behavior */ constructor() { this._sceneRenderObserver = null, this._zoneAxis = {}, this.handConstraintVisibility = Jx.PALM_AND_GAZE, this.palmUpStrictness = 0.95, this.gazeProximityRadius = 0.15, this.targetOffset = 0.1, this.targetZone = Kx.ULNAR_SIDE, this.zoneOrientationMode = z8.HAND_ROTATION, this.nodeOrientationMode = z8.HAND_ROTATION, this.handedness = "none", this.lerpTime = 100, this._zoneAxis[Kx.ABOVE_FINGER_TIPS] = new S(0, 1, 0), this._zoneAxis[Kx.RADIAL_SIDE] = new S(-1, 0, 0), this._zoneAxis[Kx.ULNAR_SIDE] = new S(1, 0, 0), this._zoneAxis[Kx.BELOW_WRIST] = new S(0, -1, 0); } /** gets or sets behavior's name */ get name() { return "HandConstraint"; } /** Enable the behavior */ enable() { this._node.setEnabled(!0); } /** Disable the behavior */ disable() { this._node.setEnabled(!1); } _getHandPose() { if (!this._handTracking) return null; let e; if (this.handedness === "none" ? e = this._handTracking.getHandByHandedness("left") || this._handTracking.getHandByHandedness("right") : e = this._handTracking.getHandByHandedness(this.handedness), e) { const t = e.getJointMesh(gn.PINKY_FINGER_METACARPAL), r = e.getJointMesh(gn.MIDDLE_FINGER_METACARPAL), n = e.getJointMesh(gn.WRIST); if (n && r && t) { const i = { position: r.absolutePosition, quaternion: new Ze(), id: e.xrController.uniqueId }, s = ue.Vector3[0], a = ue.Vector3[1], f = ue.Vector3[2]; return s.copyFrom(r.absolutePosition).subtractInPlace(n.absolutePosition).normalize(), a.copyFrom(t.absolutePosition).subtractInPlace(r.absolutePosition).normalize(), S.CrossToRef(s, a, a), S.CrossToRef(a, s, f), Ze.FromLookDirectionLHToRef(a, s, i.quaternion), i; } } return null; } /** * Initializes the hand constraint behavior */ init() { } /** * Attaches the hand constraint to a `TransformNode` * @param node defines the node to attach the behavior to */ attach(e) { this._node = e, this._scene = e.getScene(), this._node.rotationQuaternion || (this._node.rotationQuaternion = Ze.RotationYawPitchRoll(this._node.rotation.y, this._node.rotation.x, this._node.rotation.z)); let t = Date.now(); this._sceneRenderObserver = this._scene.onBeforeRenderObservable.add(() => { const r = this._getHandPose(); if (this._node.reservedDataStore = this._node.reservedDataStore || {}, this._node.reservedDataStore.nearInteraction = this._node.reservedDataStore.nearInteraction || {}, this._node.reservedDataStore.nearInteraction.excludedControllerId = null, r) { const n = ue.Vector3[0], i = this._scene.activeCamera; n.copyFrom(this._zoneAxis[this.targetZone]); const s = ue.Quaternion[0]; if (i && (this.zoneOrientationMode === z8.LOOK_AT_CAMERA || this.nodeOrientationMode === z8.LOOK_AT_CAMERA)) { const d = ue.Vector3[1]; d.copyFrom(i.position).subtractInPlace(r.position).normalize(), this._scene.useRightHandedSystem ? Ze.FromLookDirectionRHToRef(d, S.UpReadOnly, s) : Ze.FromLookDirectionLHToRef(d, S.UpReadOnly, s); } this.zoneOrientationMode === z8.HAND_ROTATION ? r.quaternion.toRotationMatrix(ue.Matrix[0]) : s.toRotationMatrix(ue.Matrix[0]), S.TransformNormalToRef(n, ue.Matrix[0], n), n.scaleInPlace(this.targetOffset); const a = ue.Vector3[2], f = ue.Quaternion[1]; a.copyFrom(r.position).addInPlace(n), this.nodeOrientationMode === z8.HAND_ROTATION ? f.copyFrom(r.quaternion) : f.copyFrom(s); const o = Date.now() - t; S.SmoothToRef(this._node.position, a, o, this.lerpTime, this._node.position), Ze.SmoothToRef(this._node.rotationQuaternion, f, o, this.lerpTime, this._node.rotationQuaternion), this._node.reservedDataStore.nearInteraction.excludedControllerId = r.id; } this._setVisibility(r), t = Date.now(); }); } _setVisibility(e) { let t = !0, r = !0; const n = this._scene.activeCamera; if (n) { const i = n.getForwardRay(); if (this.handConstraintVisibility === Jx.GAZE_FOCUS || this.handConstraintVisibility === Jx.PALM_AND_GAZE) { r = !1; let s; this._eyeTracking && (s = this._eyeTracking.getEyeGaze()), s = s || i; const a = ue.Vector3[0]; e ? e.position.subtractToRef(s.origin, a) : this._node.getAbsolutePosition().subtractToRef(s.origin, a); const f = S.Dot(a, s.direction), o = f * f; f > 0 && a.lengthSquared() - o < this.gazeProximityRadius * this.gazeProximityRadius && (r = !0); } if ((this.handConstraintVisibility === Jx.PALM_UP || this.handConstraintVisibility === Jx.PALM_AND_GAZE) && (t = !1, e)) { const s = ue.Vector3[0]; S.LeftHandedForwardReadOnly.rotateByQuaternionToRef(e.quaternion, s), S.Dot(s, i.direction) > this.palmUpStrictness * 2 - 1 && (t = !0); } } this._node.setEnabled(t && r); } /** * Detaches the behavior from the `TransformNode` */ detach() { this._scene.onBeforeRenderObservable.remove(this._sceneRenderObserver); } /** * Links the behavior to the XR experience in which to retrieve hand transform information. * @param xr xr experience */ linkToXRExperience(e) { const t = e.featuresManager ? e.featuresManager : e; if (!t) ye.Error("XR features manager must be available or provided directly for the Hand Menu to work"); else { try { this._eyeTracking = t.getEnabledFeature(Gi.EYE_TRACKING); } catch { } try { this._handTracking = t.getEnabledFeature(Gi.HAND_TRACKING); } catch { ye.Error("Hand tracking must be enabled for the Hand Menu to work"); } } } } class hc { /** * Gets or sets maximum allowed angle */ get maxAngle() { return this._maxAngle; } set maxAngle(e) { this._setMaxAngle(e); } /** * Creates a new BoneIKController * @param mesh defines the TransformNode to control * @param bone defines the bone to control. The bone needs to have a parent bone. It also needs to have a length greater than 0 or a children we can use to infer its length. * @param options defines options to set up the controller * @param options.targetMesh * @param options.poleTargetMesh * @param options.poleTargetBone * @param options.poleTargetLocalOffset * @param options.poleAngle * @param options.bendAxis * @param options.maxAngle * @param options.slerpAmount */ constructor(e, t, r) { this.targetPosition = S.Zero(), this.poleTargetPosition = S.Zero(), this.poleTargetLocalOffset = S.Zero(), this.poleAngle = 0, this.slerpAmount = 1, this._bone1Quat = Ze.Identity(), this._bone1Mat = he.Identity(), this._bone2Ang = Math.PI, this._maxAngle = Math.PI, this._rightHandedSystem = !1, this._bendAxis = S.Right(), this._slerping = !1, this._adjustRoll = 0, this._notEnoughInformation = !1, this._bone2 = t; const n = t.getParent(); if (!n) { this._notEnoughInformation = !0, Se.Error("BoneIKController: bone must have a parent for IK to work."); return; } if (this._bone1 = n, this._bone2.children.length === 0 && !this._bone2.length) { this._notEnoughInformation = !0, Se.Error("BoneIKController: bone must not be a leaf or it should have a length for IK to work."); return; } this.mesh = e, t.getSkeleton().computeAbsoluteMatrices(); const i = t.getPosition(); if (t.getAbsoluteMatrix().determinant() > 0 && (this._rightHandedSystem = !0, this._bendAxis.x = 0, this._bendAxis.y = 0, this._bendAxis.z = -1, i.x > i.y && i.x > i.z && (this._adjustRoll = Math.PI * 0.5, this._bendAxis.z = 1)), this._bone1.length && this._bone2.length) { const s = this._bone1.getScale(), a = this._bone2.getScale(); this._bone1Length = this._bone1.length * s.y * this.mesh.scaling.y, this._bone2Length = this._bone2.length * a.y * this.mesh.scaling.y; } else if (this._bone2.children[0]) { e.computeWorldMatrix(!0); const s = this._bone2.children[0].getAbsolutePosition(e), a = this._bone2.getAbsolutePosition(e), f = this._bone1.getAbsolutePosition(e); this._bone2Length = S.Distance(s, a), this._bone1Length = S.Distance(a, f); } else { e.computeWorldMatrix(!0); const s = this._bone2.getScale(); this._bone2Length = this._bone2.length * s.y * this.mesh.scaling.y; const a = this._bone2.getAbsolutePosition(e), f = this._bone1.getAbsolutePosition(e); this._bone1Length = S.Distance(a, f); } this._bone1.getRotationMatrixToRef(ai.WORLD, e, this._bone1Mat), this.maxAngle = Math.PI, r && (r.targetMesh && (this.targetMesh = r.targetMesh, this.targetMesh.computeWorldMatrix(!0)), r.poleTargetMesh ? (this.poleTargetMesh = r.poleTargetMesh, this.poleTargetMesh.computeWorldMatrix(!0)) : r.poleTargetBone ? this.poleTargetBone = r.poleTargetBone : this._bone1.getParent() && (this.poleTargetBone = this._bone1.getParent()), r.poleTargetLocalOffset && this.poleTargetLocalOffset.copyFrom(r.poleTargetLocalOffset), r.poleAngle && (this.poleAngle = r.poleAngle), r.bendAxis && this._bendAxis.copyFrom(r.bendAxis), r.maxAngle && (this.maxAngle = r.maxAngle), r.slerpAmount && (this.slerpAmount = r.slerpAmount)); } _setMaxAngle(e) { e < 0 && (e = 0), (e > Math.PI || e == null) && (e = Math.PI), this._maxAngle = e; const t = this._bone1Length, r = this._bone2Length; this._maxReach = Math.sqrt(t * t + r * r - 2 * t * r * Math.cos(e)); } /** * Force the controller to update the bones */ update() { if (this._notEnoughInformation) return; const e = this.targetPosition, t = this.poleTargetPosition, r = hc._TmpMats[0], n = hc._TmpMats[1]; this.targetMesh && e.copyFrom(this.targetMesh.getAbsolutePosition()), this.poleTargetBone ? this.poleTargetBone.getAbsolutePositionFromLocalToRef(this.poleTargetLocalOffset, this.mesh, t) : this.poleTargetMesh && S.TransformCoordinatesToRef(this.poleTargetLocalOffset, this.poleTargetMesh.getWorldMatrix(), t); const i = hc._TmpVecs[0], s = hc._TmpVecs[1], a = hc._TmpVecs[2], f = hc._TmpVecs[3], o = hc._TmpVecs[4], d = hc._TmpQuat; this._bone1.getAbsolutePositionToRef(this.mesh, i), t.subtractToRef(i, o), o.x == 0 && o.y == 0 && o.z == 0 ? o.y = 1 : o.normalize(), e.subtractToRef(i, f), f.normalize(), S.CrossToRef(f, o, s), s.normalize(), S.CrossToRef(f, s, a), a.normalize(), he.FromXYZAxesToRef(a, f, s, r); const v = this._bone1Length, u = this._bone2Length; let l = S.Distance(i, e); this._maxReach > 0 && (l = Math.min(this._maxReach, l)); let P = (u * u + l * l - v * v) / (2 * u * l), p = (l * l + v * v - u * u) / (2 * l * v); P > 1 && (P = 1), p > 1 && (p = 1), P < -1 && (P = -1), p < -1 && (p = -1); const c = Math.acos(P), H = Math.acos(p); let T = -c - H; if (this._rightHandedSystem) he.RotationYawPitchRollToRef(0, 0, this._adjustRoll, n), n.multiplyToRef(r, r), he.RotationAxisToRef(this._bendAxis, H, n), n.multiplyToRef(r, r); else { const q = hc._TmpVecs[5]; q.copyFrom(this._bendAxis), q.x *= -1, he.RotationAxisToRef(q, -H, n), n.multiplyToRef(r, r); } this.poleAngle && (he.RotationAxisToRef(f, this.poleAngle, n), r.multiplyToRef(n, r)), this._bone1 && (this.slerpAmount < 1 ? (this._slerping || Ze.FromRotationMatrixToRef(this._bone1Mat, this._bone1Quat), Ze.FromRotationMatrixToRef(r, d), Ze.SlerpToRef(this._bone1Quat, d, this.slerpAmount, this._bone1Quat), T = this._bone2Ang * (1 - this.slerpAmount) + T * this.slerpAmount, this._bone1.setRotationQuaternion(this._bone1Quat, ai.WORLD, this.mesh), this._slerping = !0) : (this._bone1.setRotationMatrix(r, ai.WORLD, this.mesh), this._bone1Mat.copyFrom(r), this._slerping = !1), this._updateLinkedTransformRotation(this._bone1)), this._bone2.setAxisAngle(this._bendAxis, T, ai.LOCAL), this._updateLinkedTransformRotation(this._bone2), this._bone2Ang = T; } _updateLinkedTransformRotation(e) { e._linkedTransformNode && (e._linkedTransformNode.rotationQuaternion || (e._linkedTransformNode.rotationQuaternion = new Ze()), e.getRotationQuaternionToRef(ai.LOCAL, null, e._linkedTransformNode.rotationQuaternion)); } } hc._TmpVecs = [S.Zero(), S.Zero(), S.Zero(), S.Zero(), S.Zero(), S.Zero()]; hc._TmpQuat = Ze.Identity(); hc._TmpMats = [he.Identity(), he.Identity()]; class wA { /** * Gets or sets the minimum yaw angle that the bone can look to */ get minYaw() { return this._minYaw; } set minYaw(e) { this._minYaw = e, this._minYawSin = Math.sin(e), this._minYawCos = Math.cos(e), this._maxYaw != null && (this._midYawConstraint = this._getAngleDiff(this._minYaw, this._maxYaw) * 0.5 + this._minYaw, this._yawRange = this._maxYaw - this._minYaw); } /** * Gets or sets the maximum yaw angle that the bone can look to */ get maxYaw() { return this._maxYaw; } set maxYaw(e) { this._maxYaw = e, this._maxYawSin = Math.sin(e), this._maxYawCos = Math.cos(e), this._minYaw != null && (this._midYawConstraint = this._getAngleDiff(this._minYaw, this._maxYaw) * 0.5 + this._minYaw, this._yawRange = this._maxYaw - this._minYaw); } /** * Gets or sets the minimum pitch angle that the bone can look to */ get minPitch() { return this._minPitch; } set minPitch(e) { this._minPitch = e, this._minPitchTan = Math.tan(e); } /** * Gets or sets the maximum pitch angle that the bone can look to */ get maxPitch() { return this._maxPitch; } set maxPitch(e) { this._maxPitch = e, this._maxPitchTan = Math.tan(e); } /** * Create a BoneLookController * @param mesh the TransformNode that the bone belongs to * @param bone the bone that will be looking to the target * @param target the target Vector3 to look at * @param options optional settings: * * maxYaw: the maximum angle the bone will yaw to * * minYaw: the minimum angle the bone will yaw to * * maxPitch: the maximum angle the bone will pitch to * * minPitch: the minimum angle the bone will yaw to * * slerpAmount: set the between 0 and 1 to make the bone slerp to the target. * * upAxis: the up axis of the coordinate system * * upAxisSpace: the space that the up axis is in - Space.BONE, Space.LOCAL (default), or Space.WORLD. * * yawAxis: set yawAxis if the bone does not yaw on the y axis * * pitchAxis: set pitchAxis if the bone does not pitch on the x axis * * adjustYaw: used to make an adjustment to the yaw of the bone * * adjustPitch: used to make an adjustment to the pitch of the bone * * adjustRoll: used to make an adjustment to the roll of the bone * @param options.maxYaw * @param options.minYaw * @param options.maxPitch * @param options.minPitch * @param options.slerpAmount * @param options.upAxis * @param options.upAxisSpace * @param options.yawAxis * @param options.pitchAxis * @param options.adjustYaw * @param options.adjustPitch * @param options.adjustRoll **/ constructor(e, t, r, n) { if (this.upAxis = S.Up(), this.upAxisSpace = ai.LOCAL, this.adjustYaw = 0, this.adjustPitch = 0, this.adjustRoll = 0, this.slerpAmount = 1, this._boneQuat = Ze.Identity(), this._slerping = !1, this._firstFrameSkipped = !1, this._fowardAxis = S.Forward(), this.useAbsoluteValueForYaw = !1, this.mesh = e, this.bone = t, this.target = r, n) { if (n.adjustYaw && (this.adjustYaw = n.adjustYaw), n.adjustPitch && (this.adjustPitch = n.adjustPitch), n.adjustRoll && (this.adjustRoll = n.adjustRoll), n.maxYaw != null ? this.maxYaw = n.maxYaw : this.maxYaw = Math.PI, n.minYaw != null ? this.minYaw = n.minYaw : this.minYaw = -Math.PI, n.maxPitch != null ? this.maxPitch = n.maxPitch : this.maxPitch = Math.PI, n.minPitch != null ? this.minPitch = n.minPitch : this.minPitch = -Math.PI, n.slerpAmount != null && (this.slerpAmount = n.slerpAmount), n.upAxis != null && (this.upAxis = n.upAxis), n.upAxisSpace != null && (this.upAxisSpace = n.upAxisSpace), n.yawAxis != null || n.pitchAxis != null) { let i = bf.Y, s = bf.X; n.yawAxis != null && (i = n.yawAxis.clone(), i.normalize()), n.pitchAxis != null && (s = n.pitchAxis.clone(), s.normalize()); const a = S.Cross(s, i); this._transformYawPitch = he.Identity(), he.FromXYZAxesToRef(s, i, a, this._transformYawPitch), this._transformYawPitchInv = this._transformYawPitch.clone(), this._transformYawPitch.invert(); } n.useAbsoluteValueForYaw !== void 0 && (this.useAbsoluteValueForYaw = n.useAbsoluteValueForYaw); } !t.getParent() && this.upAxisSpace == ai.BONE && (this.upAxisSpace = ai.LOCAL); } /** * Update the bone to look at the target. This should be called before the scene is rendered (use scene.registerBeforeRender()) */ update() { if (this.slerpAmount < 1 && !this._firstFrameSkipped) { this._firstFrameSkipped = !0; return; } const e = this.bone, t = wA._TmpVecs[0]; e.getAbsolutePositionToRef(this.mesh, t); let r = this.target; const n = wA._TmpMats[0], i = wA._TmpMats[1], s = this.mesh, a = e.getParent(), f = wA._TmpVecs[1]; f.copyFrom(this.upAxis), this.upAxisSpace == ai.BONE && a ? (this._transformYawPitch && S.TransformCoordinatesToRef(f, this._transformYawPitchInv, f), a.getDirectionToRef(f, this.mesh, f)) : this.upAxisSpace == ai.LOCAL && (s.getDirectionToRef(f, f), (s.scaling.x != 1 || s.scaling.y != 1 || s.scaling.z != 1) && f.normalize()); let o = !1, d = !1; if ((this._maxYaw != Math.PI || this._minYaw != -Math.PI) && (o = !0), (this._maxPitch != Math.PI || this._minPitch != -Math.PI) && (d = !0), o || d) { const p = wA._TmpMats[2], c = wA._TmpMats[3]; if (this.upAxisSpace == ai.BONE && f.y == 1 && a) a.getRotationMatrixToRef(ai.WORLD, this.mesh, p); else if (this.upAxisSpace == ai.LOCAL && f.y == 1 && !a) p.copyFrom(s.getWorldMatrix()); else { let T = wA._TmpVecs[2]; T.copyFrom(this._fowardAxis), this._transformYawPitch && S.TransformCoordinatesToRef(T, this._transformYawPitchInv, T), a ? a.getDirectionToRef(T, this.mesh, T) : s.getDirectionToRef(T, T); const q = S.Cross(f, T); q.normalize(), T = S.Cross(q, f), he.FromXYZAxesToRef(q, f, T, p); } p.invertToRef(c); let H = null; if (d) { const T = wA._TmpVecs[3]; r.subtractToRef(t, T), S.TransformCoordinatesToRef(T, c, T), H = Math.sqrt(T.x * T.x + T.z * T.z); const q = Math.atan2(T.y, H); let b = q; q > this._maxPitch ? (T.y = this._maxPitchTan * H, b = this._maxPitch) : q < this._minPitch && (T.y = this._minPitchTan * H, b = this._minPitch), q != b && (S.TransformCoordinatesToRef(T, p, T), T.addInPlace(t), r = T); } if (o) { const T = wA._TmpVecs[4]; r.subtractToRef(t, T), S.TransformCoordinatesToRef(T, c, T); const q = Math.atan2(T.x, T.z), b = this.useAbsoluteValueForYaw ? Math.abs(q) : q; let j = q; if ((b > this._maxYaw || b < this._minYaw) && (H == null && (H = Math.sqrt(T.x * T.x + T.z * T.z)), this._yawRange > Math.PI ? this._isAngleBetween(q, this._maxYaw, this._midYawConstraint) ? (T.z = this._maxYawCos * H, T.x = this._maxYawSin * H, j = this._maxYaw) : this._isAngleBetween(q, this._midYawConstraint, this._minYaw) && (T.z = this._minYawCos * H, T.x = this._minYawSin * H, j = this._minYaw) : b > this._maxYaw ? (T.z = this._maxYawCos * H, T.x = this._maxYawSin * H, q < 0 && this.useAbsoluteValueForYaw && (T.x *= -1), j = this._maxYaw) : b < this._minYaw && (T.z = this._minYawCos * H, T.x = this._minYawSin * H, q < 0 && this.useAbsoluteValueForYaw && (T.x *= -1), j = this._minYaw)), this._slerping && this._yawRange > Math.PI) { const w = wA._TmpVecs[8]; w.copyFrom(bf.Z), this._transformYawPitch && S.TransformCoordinatesToRef(w, this._transformYawPitchInv, w); const m = wA._TmpMats[4]; this._boneQuat.toRotationMatrix(m), this.mesh.getWorldMatrix().multiplyToRef(m, m), S.TransformCoordinatesToRef(w, m, w), S.TransformCoordinatesToRef(w, c, w); const I = Math.atan2(w.x, w.z), N = this._getAngleBetween(I, q), k = this._getAngleBetween(I, this._midYawConstraint); if (N > k) { H == null && (H = Math.sqrt(T.x * T.x + T.z * T.z)); const R = this._getAngleBetween(I, this._maxYaw); this._getAngleBetween(I, this._minYaw) < R ? (j = I + Math.PI * 0.75, T.z = Math.cos(j) * H, T.x = Math.sin(j) * H) : (j = I - Math.PI * 0.75, T.z = Math.cos(j) * H, T.x = Math.sin(j) * H); } } q != j && (S.TransformCoordinatesToRef(T, p, T), T.addInPlace(t), r = T); } } const v = wA._TmpVecs[5], u = wA._TmpVecs[6], l = wA._TmpVecs[7], P = wA._TmpQuat; r.subtractToRef(t, v), v.normalize(), S.CrossToRef(f, v, u), u.normalize(), S.CrossToRef(v, u, l), l.normalize(), he.FromXYZAxesToRef(u, l, v, n), !(u.x === 0 && u.y === 0 && u.z === 0) && (l.x === 0 && l.y === 0 && l.z === 0 || v.x === 0 && v.y === 0 && v.z === 0 || ((this.adjustYaw || this.adjustPitch || this.adjustRoll) && (he.RotationYawPitchRollToRef(this.adjustYaw, this.adjustPitch, this.adjustRoll, i), i.multiplyToRef(n, n)), this.slerpAmount < 1 ? (this._slerping || this.bone.getRotationQuaternionToRef(ai.WORLD, this.mesh, this._boneQuat), this._transformYawPitch && this._transformYawPitch.multiplyToRef(n, n), Ze.FromRotationMatrixToRef(n, P), Ze.SlerpToRef(this._boneQuat, P, this.slerpAmount, this._boneQuat), this.bone.setRotationQuaternion(this._boneQuat, ai.WORLD, this.mesh), this._slerping = !0) : (this._transformYawPitch && this._transformYawPitch.multiplyToRef(n, n), this.bone.setRotationMatrix(n, ai.WORLD, this.mesh), this._slerping = !1), this._updateLinkedTransformRotation())); } _getAngleDiff(e, t) { let r = t - e; return r %= Math.PI * 2, r > Math.PI ? r -= Math.PI * 2 : r < -Math.PI && (r += Math.PI * 2), r; } _getAngleBetween(e, t) { e %= 2 * Math.PI, e = e < 0 ? e + 2 * Math.PI : e, t %= 2 * Math.PI, t = t < 0 ? t + 2 * Math.PI : t; let r = 0; return e < t ? r = t - e : r = e - t, r > Math.PI && (r = Math.PI * 2 - r), r; } _isAngleBetween(e, t, r) { if (e %= 2 * Math.PI, e = e < 0 ? e + 2 * Math.PI : e, t %= 2 * Math.PI, t = t < 0 ? t + 2 * Math.PI : t, r %= 2 * Math.PI, r = r < 0 ? r + 2 * Math.PI : r, t < r) { if (e > t && e < r) return !0; } else if (e > r && e < t) return !0; return !1; } _updateLinkedTransformRotation() { const e = this.bone; e._linkedTransformNode && (e._linkedTransformNode.rotationQuaternion || (e._linkedTransformNode.rotationQuaternion = new Ze()), e.getRotationQuaternionToRef(ai.LOCAL, null, e._linkedTransformNode.rotationQuaternion)); } } wA._TmpVecs = Nf.BuildArray(10, S.Zero); wA._TmpQuat = Ze.Identity(); wA._TmpMats = Nf.BuildArray(5, he.Identity); class r4 { /** * Gets or sets a boolean indicating that bone matrices should be stored as a texture instead of using shader uniforms (default is true). * Please note that this option is not available if the hardware does not support it */ get useTextureToStoreBoneMatrices() { return this._useTextureToStoreBoneMatrices; } set useTextureToStoreBoneMatrices(e) { this._useTextureToStoreBoneMatrices = e, this._markAsDirty(); } /** * Gets or sets the animation properties override */ get animationPropertiesOverride() { return this._animationPropertiesOverride ? this._animationPropertiesOverride : this._scene.animationPropertiesOverride; } set animationPropertiesOverride(e) { this._animationPropertiesOverride = e; } /** * Gets a boolean indicating that the skeleton effectively stores matrices into a texture */ get isUsingTextureForMatrices() { return this.useTextureToStoreBoneMatrices && this._canUseTextureForBones; } /** * Gets the unique ID of this skeleton */ get uniqueId() { return this._uniqueId; } /** * Creates a new skeleton * @param name defines the skeleton name * @param id defines the skeleton Id * @param scene defines the hosting scene */ constructor(e, t, r) { this.name = e, this.id = t, this.bones = [], this.needInitialSkinMatrix = !1, this._isDirty = !0, this._meshesWithPoseMatrix = new Array(), this._identity = he.Identity(), this._currentRenderId = -1, this._ranges = {}, this._absoluteTransformIsDirty = !0, this._canUseTextureForBones = !1, this._uniqueId = 0, this._numBonesWithLinkedTransformNode = 0, this._hasWaitingData = null, this._parentContainer = null, this.doNotSerialize = !1, this._useTextureToStoreBoneMatrices = !0, this._animationPropertiesOverride = null, this.onBeforeComputeObservable = new Oe(), this.bones = [], this._scene = r || gr.LastCreatedScene, this._uniqueId = this._scene.getUniqueId(), this._scene.addSkeleton(this), this._isDirty = !0; const n = this._scene.getEngine().getCaps(); this._canUseTextureForBones = n.textureFloat && n.maxVertexTextureImageUnits > 0; } /** * Gets the current object class name. * @returns the class name */ getClassName() { return "Skeleton"; } /** * Returns an array containing the root bones * @returns an array containing the root bones */ getChildren() { return this.bones.filter((e) => !e.getParent()); } // Members /** * Gets the list of transform matrices to send to shaders (one matrix per bone) * @param mesh defines the mesh to use to get the root matrix (if needInitialSkinMatrix === true) * @returns a Float32Array containing matrices data */ getTransformMatrices(e) { return this.needInitialSkinMatrix ? (e._bonesTransformMatrices || this.prepare(), e._bonesTransformMatrices) : ((!this._transformMatrices || this._isDirty) && this.prepare(), this._transformMatrices); } /** * Gets the list of transform matrices to send to shaders inside a texture (one matrix per bone) * @param mesh defines the mesh to use to get the root matrix (if needInitialSkinMatrix === true) * @returns a raw texture containing the data */ getTransformMatrixTexture(e) { return this.needInitialSkinMatrix && e._transformMatrixTexture ? e._transformMatrixTexture : this._transformMatrixTexture; } /** * Gets the current hosting scene * @returns a scene object */ getScene() { return this._scene; } // Methods /** * Gets a string representing the current skeleton data * @param fullDetails defines a boolean indicating if we want a verbose version * @returns a string representing the current skeleton data */ toString(e) { let t = `Name: ${this.name}, nBones: ${this.bones.length}`; if (t += `, nAnimationRanges: ${this._ranges ? Object.keys(this._ranges).length : "none"}`, e) { t += ", Ranges: {"; let r = !0; for (const n in this._ranges) r && (t += ", ", r = !1), t += n; t += "}"; } return t; } /** * Get bone's index searching by name * @param name defines bone's name to search for * @returns the indice of the bone. Returns -1 if not found */ getBoneIndexByName(e) { for (let t = 0, r = this.bones.length; t < r; t++) if (this.bones[t].name === e) return t; return -1; } /** * Create a new animation range * @param name defines the name of the range * @param from defines the start key * @param to defines the end key */ createAnimationRange(e, t, r) { if (!this._ranges[e]) { this._ranges[e] = new Am(e, t, r); for (let n = 0, i = this.bones.length; n < i; n++) this.bones[n].animations[0] && this.bones[n].animations[0].createRange(e, t, r); } } /** * Delete a specific animation range * @param name defines the name of the range * @param deleteFrames defines if frames must be removed as well */ deleteAnimationRange(e, t = !0) { for (let r = 0, n = this.bones.length; r < n; r++) this.bones[r].animations[0] && this.bones[r].animations[0].deleteRange(e, t); this._ranges[e] = null; } /** * Gets a specific animation range * @param name defines the name of the range to look for * @returns the requested animation range or null if not found */ getAnimationRange(e) { return this._ranges[e] || null; } /** * Gets the list of all animation ranges defined on this skeleton * @returns an array */ getAnimationRanges() { const e = []; let t; for (t in this._ranges) e.push(this._ranges[t]); return e; } /** * Copy animation range from a source skeleton. * This is not for a complete retargeting, only between very similar skeleton's with only possible bone length differences * @param source defines the source skeleton * @param name defines the name of the range to copy * @param rescaleAsRequired defines if rescaling must be applied if required * @returns true if operation was successful */ copyAnimationRange(e, t, r = !1) { if (this._ranges[t] || !e.getAnimationRange(t)) return !1; let n = !0; const i = this._getHighestAnimationFrame() + 1, s = {}, a = e.bones; let f, o; for (o = 0, f = a.length; o < f; o++) s[a[o].name] = a[o]; this.bones.length !== a.length && (Se.Warn(`copyAnimationRange: this rig has ${this.bones.length} bones, while source as ${a.length}`), n = !1); const d = r && this.dimensionsAtRest && e.dimensionsAtRest ? this.dimensionsAtRest.divide(e.dimensionsAtRest) : null; for (o = 0, f = this.bones.length; o < f; o++) { const u = this.bones[o].name, l = s[u]; l ? n = n && this.bones[o].copyAnimationRange(l, t, i, r, d) : (Se.Warn("copyAnimationRange: not same rig, missing source bone " + u), n = !1); } const v = e.getAnimationRange(t); return v && (this._ranges[t] = new Am(t, v.from + i, v.to + i)), n; } /** * Forces the skeleton to go to rest pose */ returnToRest() { for (const e of this.bones) e._index !== -1 && e.returnToRest(); } _getHighestAnimationFrame() { let e = 0; for (let t = 0, r = this.bones.length; t < r; t++) if (this.bones[t].animations[0]) { const n = this.bones[t].animations[0].getHighestFrame(); e < n && (e = n); } return e; } /** * Begin a specific animation range * @param name defines the name of the range to start * @param loop defines if looping must be turned on (false by default) * @param speedRatio defines the speed ratio to apply (1 by default) * @param onAnimationEnd defines a callback which will be called when animation will end * @returns a new animatable */ beginAnimation(e, t, r, n) { const i = this.getAnimationRange(e); return i ? this._scene.beginAnimation(this, i.from, i.to, t, r, n) : null; } /** * Convert the keyframes for a range of animation on a skeleton to be relative to a given reference frame. * @param skeleton defines the Skeleton containing the animation range to convert * @param referenceFrame defines the frame that keyframes in the range will be relative to * @param range defines the name of the AnimationRange belonging to the Skeleton to convert * @returns the original skeleton */ static MakeAnimationAdditive(e, t = 0, r) { const n = e.getAnimationRange(r); if (!n) return null; const i = e._scene.getAllAnimatablesByTarget(e); let s = null; for (let f = 0; f < i.length; f++) { const o = i[f]; if (o.fromFrame === (n == null ? void 0 : n.from) && o.toFrame === (n == null ? void 0 : n.to)) { s = o; break; } } const a = e.getAnimatables(); for (let f = 0; f < a.length; f++) { const d = a[f].animations; if (d) for (let v = 0; v < d.length; v++) st.MakeAnimationAdditive(d[v], t, r); } return s && (s.isAdditive = !0), e; } /** @internal */ _markAsDirty() { this._isDirty = !0, this._absoluteTransformIsDirty = !0; } /** * @internal */ _registerMeshWithPoseMatrix(e) { this._meshesWithPoseMatrix.push(e); } /** * @internal */ _unregisterMeshWithPoseMatrix(e) { const t = this._meshesWithPoseMatrix.indexOf(e); t > -1 && this._meshesWithPoseMatrix.splice(t, 1); } _computeTransformMatrices(e, t) { this.onBeforeComputeObservable.notifyObservers(this); for (let r = 0; r < this.bones.length; r++) { const n = this.bones[r]; n._childUpdateId++; const i = n.getParent(); if (i ? n.getLocalMatrix().multiplyToRef(i.getFinalMatrix(), n.getFinalMatrix()) : t ? n.getLocalMatrix().multiplyToRef(t, n.getFinalMatrix()) : n.getFinalMatrix().copyFrom(n.getLocalMatrix()), n._index !== -1) { const s = n._index === null ? r : n._index; n.getAbsoluteInverseBindMatrix().multiplyToArray(n.getFinalMatrix(), e, s * 16); } } this._identity.copyToArray(e, this.bones.length * 16); } /** * Build all resources required to render a skeleton * @param dontCheckFrameId defines a boolean indicating if prepare should be run without checking first the current frame id (default: false) */ prepare(e = !1) { if (!e) { const t = this.getScene().getRenderId(); if (this._currentRenderId === t) return; this._currentRenderId = t; } if (this._numBonesWithLinkedTransformNode > 0) { for (const t of this.bones) if (t._linkedTransformNode) { const r = t._linkedTransformNode; t.position = r.position, r.rotationQuaternion ? t.rotationQuaternion = r.rotationQuaternion : t.rotation = r.rotation, t.scaling = r.scaling; } } if (this.needInitialSkinMatrix) for (const t of this._meshesWithPoseMatrix) { const r = t.getPoseMatrix(); let n = this._isDirty; if ((!t._bonesTransformMatrices || t._bonesTransformMatrices.length !== 16 * (this.bones.length + 1)) && (t._bonesTransformMatrices = new Float32Array(16 * (this.bones.length + 1)), n = !0), !!n) { if (this._synchronizedWithMesh !== t) { this._synchronizedWithMesh = t; for (const i of this.bones) i.getParent() || (i.getBindMatrix().multiplyToRef(r, ue.Matrix[1]), i._updateAbsoluteBindMatrices(ue.Matrix[1])); if (this.isUsingTextureForMatrices) { const i = (this.bones.length + 1) * 4; (!t._transformMatrixTexture || t._transformMatrixTexture.getSize().width !== i) && (t._transformMatrixTexture && t._transformMatrixTexture.dispose(), t._transformMatrixTexture = Bo.CreateRGBATexture(t._bonesTransformMatrices, (this.bones.length + 1) * 4, 1, this._scene, !1, !1, 1, 1)); } } this._computeTransformMatrices(t._bonesTransformMatrices, r), this.isUsingTextureForMatrices && t._transformMatrixTexture && t._transformMatrixTexture.update(t._bonesTransformMatrices); } } else { if (!this._isDirty) return; (!this._transformMatrices || this._transformMatrices.length !== 16 * (this.bones.length + 1)) && (this._transformMatrices = new Float32Array(16 * (this.bones.length + 1)), this.isUsingTextureForMatrices && (this._transformMatrixTexture && this._transformMatrixTexture.dispose(), this._transformMatrixTexture = Bo.CreateRGBATexture(this._transformMatrices, (this.bones.length + 1) * 4, 1, this._scene, !1, !1, 1, 1))), this._computeTransformMatrices(this._transformMatrices, null), this.isUsingTextureForMatrices && this._transformMatrixTexture && this._transformMatrixTexture.update(this._transformMatrices); } this._isDirty = !1; } /** * Gets the list of animatables currently running for this skeleton * @returns an array of animatables */ getAnimatables() { if (!this._animatables || this._animatables.length !== this.bones.length) { this._animatables = []; for (let e = 0; e < this.bones.length; e++) this._animatables.push(this.bones[e]); } return this._animatables; } /** * Clone the current skeleton * @param name defines the name of the new skeleton * @param id defines the id of the new skeleton * @returns the new skeleton */ clone(e, t) { const r = new r4(e, t || e, this._scene); r.needInitialSkinMatrix = this.needInitialSkinMatrix; for (let n = 0; n < this.bones.length; n++) { const i = this.bones[n]; let s = null; const a = i.getParent(); if (a) { const o = this.bones.indexOf(a); s = r.bones[o]; } const f = new da(i.name, r, s, i.getBindMatrix().clone(), i.getRestMatrix().clone()); f._index = i._index, i._linkedTransformNode && f.linkTransformNode(i._linkedTransformNode), sA.DeepCopy(i.animations, f.animations); } if (this._ranges) { r._ranges = {}; for (const n in this._ranges) { const i = this._ranges[n]; i && (r._ranges[n] = i.clone()); } } return this._isDirty = !0, r.prepare(!0), r; } /** * Enable animation blending for this skeleton * @param blendingSpeed defines the blending speed to apply * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#animation-blending */ enableBlending(e = 0.01) { this.bones.forEach((t) => { t.animations.forEach((r) => { r.enableBlending = !0, r.blendingSpeed = e; }); }); } /** * Releases all resources associated with the current skeleton */ dispose() { if (this._meshesWithPoseMatrix.length = 0, this.getScene().stopAnimation(this), this.getScene().removeSkeleton(this), this._parentContainer) { const e = this._parentContainer.skeletons.indexOf(this); e > -1 && this._parentContainer.skeletons.splice(e, 1), this._parentContainer = null; } this._transformMatrixTexture && (this._transformMatrixTexture.dispose(), this._transformMatrixTexture = null); } /** * Serialize the skeleton in a JSON object * @returns a JSON object */ serialize() { var e; const t = {}; t.name = this.name, t.id = this.id, this.dimensionsAtRest && (t.dimensionsAtRest = this.dimensionsAtRest.asArray()), t.bones = [], t.needInitialSkinMatrix = this.needInitialSkinMatrix; for (let r = 0; r < this.bones.length; r++) { const n = this.bones[r], i = n.getParent(), s = { parentBoneIndex: i ? this.bones.indexOf(i) : -1, index: n.getIndex(), name: n.name, id: n.id, matrix: n.getBindMatrix().toArray(), rest: n.getRestMatrix().toArray(), linkedTransformNodeId: (e = n.getTransformNode()) === null || e === void 0 ? void 0 : e.id }; t.bones.push(s), n.length && (s.length = n.length), n.metadata && (s.metadata = n.metadata), n.animations && n.animations.length > 0 && (s.animation = n.animations[0].serialize()), t.ranges = []; for (const a in this._ranges) { const f = this._ranges[a]; if (!f) continue; const o = {}; o.name = a, o.from = f.from, o.to = f.to, t.ranges.push(o); } } return t; } /** * Creates a new skeleton from serialized data * @param parsedSkeleton defines the serialized data * @param scene defines the hosting scene * @returns a new skeleton */ static Parse(e, t) { const r = new r4(e.name, e.id, t); e.dimensionsAtRest && (r.dimensionsAtRest = S.FromArray(e.dimensionsAtRest)), r.needInitialSkinMatrix = e.needInitialSkinMatrix; let n; for (n = 0; n < e.bones.length; n++) { const i = e.bones[n], s = e.bones[n].index; let a = null; i.parentBoneIndex > -1 && (a = r.bones[i.parentBoneIndex]); const f = i.rest ? he.FromArray(i.rest) : null, o = new da(i.name, r, a, he.FromArray(i.matrix), f, null, s); i.id !== void 0 && i.id !== null && (o.id = i.id), i.length && (o.length = i.length), i.metadata && (o.metadata = i.metadata), i.animation && o.animations.push(st.Parse(i.animation)), i.linkedTransformNodeId !== void 0 && i.linkedTransformNodeId !== null && (r._hasWaitingData = !0, o._waitingTransformNodeId = i.linkedTransformNodeId); } if (e.ranges) for (n = 0; n < e.ranges.length; n++) { const i = e.ranges[n]; r.createAnimationRange(i.name, i.from, i.to); } return r; } /** * Compute all node absolute matrices * @param forceUpdate defines if computation must be done even if cache is up to date */ computeAbsoluteMatrices(e = !1) { (this._absoluteTransformIsDirty || e) && (this.bones[0].computeAbsoluteMatrices(), this._absoluteTransformIsDirty = !1); } /** * Compute all node absolute matrices * @param forceUpdate defines if computation must be done even if cache is up to date * @deprecated Please use computeAbsoluteMatrices instead */ computeAbsoluteTransforms(e = !1) { this.computeAbsoluteMatrices(e); } /** * Gets the root pose matrix * @returns a matrix */ getPoseMatrix() { let e = null; return this._meshesWithPoseMatrix.length > 0 && (e = this._meshesWithPoseMatrix[0].getPoseMatrix()), e; } /** * Sorts bones per internal index */ sortBones() { const e = [], t = new Array(this.bones.length); for (let r = 0; r < this.bones.length; r++) this._sortBones(r, e, t); this.bones = e; } _sortBones(e, t, r) { if (r[e]) return; r[e] = !0; const n = this.bones[e]; if (!n) return; n._index === void 0 && (n._index = e); const i = n.getParent(); i && this._sortBones(this.bones.indexOf(i), t, r), t.push(n); } /** * Set the current local matrix as the restPose for all bones in the skeleton. */ setCurrentPoseAsRest() { this.bones.forEach((e) => { e.setCurrentPoseAsRest(); }); } } class bee { /** * Creates a new storage buffer instance * @param engine The engine the buffer will be created inside * @param size The size of the buffer in bytes * @param creationFlags flags to use when creating the buffer (see undefined). The BUFFER_CREATIONFLAG_STORAGE flag will be automatically added. * @param label defines the label of the buffer (for debug purpose) */ constructor(e, t, r = 3, n) { this._engine = e, this._label = n, this._engine._storageBuffers.push(this), this._create(t, r); } _create(e, t) { this._bufferSize = e, this._creationFlags = t, this._buffer = this._engine.createStorageBuffer(e, t, this._label); } /** @internal */ _rebuild() { this._create(this._bufferSize, this._creationFlags); } /** * Gets underlying native buffer * @returns underlying native buffer */ getBuffer() { return this._buffer; } /** * Updates the storage buffer * @param data the data used to update the storage buffer * @param byteOffset the byte offset of the data (optional) * @param byteLength the byte length of the data (optional) */ update(e, t, r) { this._buffer && this._engine.updateStorageBuffer(this._buffer, e, t, r); } /** * Reads data from the storage buffer * @param offset The offset in the storage buffer to start reading from (default: 0) * @param size The number of bytes to read from the storage buffer (default: capacity of the buffer) * @param buffer The buffer to write the data we have read from the storage buffer to (optional) * @returns If not undefined, returns the (promise) buffer (as provided by the 4th parameter) filled with the data, else it returns a (promise) Uint8Array with the data read from the storage buffer */ read(e, t, r) { return this._engine.readFromStorageBuffer(this._buffer, e, t, r); } /** * Disposes the storage buffer */ dispose() { const e = this._engine._storageBuffers, t = e.indexOf(this); t !== -1 && (e[t] = e[e.length - 1], e.pop()), this._engine._releaseBuffer(this._buffer), this._buffer = null; } } const fI = (() => { const A = new Uint8Array(4), e = new Uint32Array(A.buffer); return !!((e[0] = 1) & A[0]); })(); Object.defineProperty(J.prototype, "effectiveByteStride", { get: function() { return this._alignedBuffer && this._alignedBuffer.byteStride || this.byteStride; }, enumerable: !0, configurable: !0 }); Object.defineProperty(J.prototype, "effectiveByteOffset", { get: function() { return this._alignedBuffer ? 0 : this.byteOffset; }, enumerable: !0, configurable: !0 }); Object.defineProperty(J.prototype, "effectiveBuffer", { get: function() { return this._alignedBuffer && this._alignedBuffer.getBuffer() || this._buffer.getBuffer(); }, enumerable: !0, configurable: !0 }); J.prototype._rebuild = function() { var A, e; (A = this._buffer) === null || A === void 0 || A._rebuild(), (e = this._alignedBuffer) === null || e === void 0 || e._rebuild(); }; J.prototype.dispose = function() { var A; this._ownsBuffer && this._buffer.dispose(), (A = this._alignedBuffer) === null || A === void 0 || A.dispose(), this._alignedBuffer = void 0, this._isDisposed = !0; }; J.prototype._alignBuffer = function() { var A, e; const t = this._buffer.getData(); if (!this.engine._features.forceVertexBufferStrideMultiple4Bytes || this.byteStride % 4 === 0 || !t) return; const r = J.GetTypeByteLength(this.type), n = this.byteStride + 3 & -4, i = n / r, s = this.totalVertices, f = s * n / r; let o; if (Array.isArray(t)) { const l = new Float32Array(t); o = new DataView(l.buffer, l.byteOffset, l.byteLength); } else t instanceof ArrayBuffer ? o = new DataView(t, 0, t.byteLength) : o = new DataView(t.buffer, t.byteOffset, t.byteLength); let d; this.type === J.BYTE ? d = new Int8Array(f) : this.type === J.UNSIGNED_BYTE ? d = new Uint8Array(f) : this.type === J.SHORT ? d = new Int16Array(f) : this.type === J.UNSIGNED_SHORT ? d = new Uint16Array(f) : this.type === J.INT ? d = new Int32Array(f) : this.type === J.UNSIGNED_INT ? d = new Uint32Array(f) : d = new Float32Array(f); const v = this.getSize(); let u = this.byteOffset; for (let l = 0; l < s; ++l) { for (let P = 0; P < v; ++P) switch (this.type) { case J.BYTE: d[l * i + P] = o.getInt8(u + P); break; case J.UNSIGNED_BYTE: d[l * i + P] = o.getUint8(u + P); break; case J.SHORT: d[l * i + P] = o.getInt16(u + P * 2, fI); break; case J.UNSIGNED_SHORT: d[l * i + P] = o.getUint16(u + P * 2, fI); break; case J.INT: d[l * i + P] = o.getInt32(u + P * 4, fI); break; case J.UNSIGNED_INT: d[l * i + P] = o.getUint32(u + P * 4, fI); break; case J.FLOAT: d[l * i + P] = o.getFloat32(u + P * 4, fI); break; } u += this.byteStride; } (A = this._alignedBuffer) === null || A === void 0 || A.dispose(), this._alignedBuffer = new P9(this.engine, d, !1, n, !1, this.getIsInstanced(), !0, this.instanceDivisor, ((e = this._label) !== null && e !== void 0 ? e : "VertexBuffer") + "_aligned"); }; class IR { constructor() { this.wheelPrecisionX = 3, this.wheelPrecisionY = 3, this.wheelPrecisionZ = 3, this.onChangedObservable = new Oe(), this._wheelDeltaX = 0, this._wheelDeltaY = 0, this._wheelDeltaZ = 0, this._ffMultiplier = 12, this._normalize = 120; } /** * Attach the input controls to a specific dom element to get the input from. * @param noPreventDefault Defines whether event caught by the controls * should call preventdefault(). * (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachControl(e) { e = ye.BackCompatCameraNoPreventDefault(arguments), this._wheel = (t) => { if (t.type !== ir.POINTERWHEEL) return; const r = t.event, n = r.deltaMode === wm.DOM_DELTA_LINE ? this._ffMultiplier : 1; this._wheelDeltaX += this.wheelPrecisionX * n * r.deltaX / this._normalize, this._wheelDeltaY -= this.wheelPrecisionY * n * r.deltaY / this._normalize, this._wheelDeltaZ += this.wheelPrecisionZ * n * r.deltaZ / this._normalize, r.preventDefault && (e || r.preventDefault()); }, this._observer = this.camera.getScene()._inputManager._addCameraPointerObserver(this._wheel, ir.POINTERWHEEL); } /** * Detach the current controls from the specified dom element. */ detachControl() { this._observer && (this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer), this._observer = null, this._wheel = null), this.onChangedObservable && this.onChangedObservable.clear(); } /** * Called for each rendered frame. */ checkInputs() { this.onChangedObservable.notifyObservers({ wheelDeltaX: this._wheelDeltaX, wheelDeltaY: this._wheelDeltaY, wheelDeltaZ: this._wheelDeltaZ }), this._wheelDeltaX = 0, this._wheelDeltaY = 0, this._wheelDeltaZ = 0; } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "BaseCameraMouseWheelInput"; } /** * Get the friendly name associated with the input class. * @returns the input friendly name */ getSimpleName() { return "mousewheel"; } } C([ M() ], IR.prototype, "wheelPrecisionX", void 0); C([ M() ], IR.prototype, "wheelPrecisionY", void 0); C([ M() ], IR.prototype, "wheelPrecisionZ", void 0); class VO { constructor() { this._currentActiveButton = -1, this.buttons = [0, 1, 2]; } /** * Attach the input controls to a specific dom element to get the input from. * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachControl(e) { e = ye.BackCompatCameraNoPreventDefault(arguments); const t = this.camera.getEngine(), r = t.getInputElement(); let n = 0, i = null; this._pointA = null, this._pointB = null, this._altKey = !1, this._ctrlKey = !1, this._metaKey = !1, this._shiftKey = !1, this._buttonsPressed = 0, this._pointerInput = (a) => { var f, o; const d = a.event, v = d.pointerType === "touch"; if (a.type !== ir.POINTERMOVE && this.buttons.indexOf(d.button) === -1) return; const u = d.target; if (this._altKey = d.altKey, this._ctrlKey = d.ctrlKey, this._metaKey = d.metaKey, this._shiftKey = d.shiftKey, this._buttonsPressed = d.buttons, t.isPointerLock) { const l = d.movementX, P = d.movementY; this.onTouch(null, l, P), this._pointA = null, this._pointB = null; } else { if (a.type !== ir.POINTERDOWN && v && ((f = this._pointA) === null || f === void 0 ? void 0 : f.pointerId) !== d.pointerId && ((o = this._pointB) === null || o === void 0 ? void 0 : o.pointerId) !== d.pointerId) return; if (a.type === ir.POINTERDOWN && (this._currentActiveButton === -1 || v)) { try { u == null || u.setPointerCapture(d.pointerId); } catch { } if (this._pointA === null) this._pointA = { x: d.clientX, y: d.clientY, pointerId: d.pointerId, type: d.pointerType }; else if (this._pointB === null) this._pointB = { x: d.clientX, y: d.clientY, pointerId: d.pointerId, type: d.pointerType }; else return; this._currentActiveButton === -1 && !v && (this._currentActiveButton = d.button), this.onButtonDown(d), e || (d.preventDefault(), r && r.focus()); } else if (a.type === ir.POINTERDOUBLETAP) this.onDoubleTap(d.pointerType); else if (a.type === ir.POINTERUP && (this._currentActiveButton === d.button || v)) { try { u == null || u.releasePointerCapture(d.pointerId); } catch { } v || (this._pointB = null), t._badOS ? this._pointA = this._pointB = null : this._pointB && this._pointA && this._pointA.pointerId == d.pointerId ? (this._pointA = this._pointB, this._pointB = null) : this._pointA && this._pointB && this._pointB.pointerId == d.pointerId ? this._pointB = null : this._pointA = this._pointB = null, (n !== 0 || i) && (this.onMultiTouch( this._pointA, this._pointB, n, 0, // pinchSquaredDistance i, null // multiTouchPanPosition ), n = 0, i = null), this._currentActiveButton = -1, this.onButtonUp(d), e || d.preventDefault(); } else if (a.type === ir.POINTERMOVE) { if (e || d.preventDefault(), this._pointA && this._pointB === null) { const l = d.clientX - this._pointA.x, P = d.clientY - this._pointA.y; this.onTouch(this._pointA, l, P), this._pointA.x = d.clientX, this._pointA.y = d.clientY; } else if (this._pointA && this._pointB) { const l = this._pointA.pointerId === d.pointerId ? this._pointA : this._pointB; l.x = d.clientX, l.y = d.clientY; const P = this._pointA.x - this._pointB.x, p = this._pointA.y - this._pointB.y, c = P * P + p * p, H = { x: (this._pointA.x + this._pointB.x) / 2, y: (this._pointA.y + this._pointB.y) / 2, pointerId: d.pointerId, type: a.type }; this.onMultiTouch(this._pointA, this._pointB, n, c, i, H), i = H, n = c; } } } }, this._observer = this.camera.getScene()._inputManager._addCameraPointerObserver(this._pointerInput, ir.POINTERDOWN | ir.POINTERUP | ir.POINTERMOVE | ir.POINTERDOUBLETAP), this._onLostFocus = () => { this._pointA = this._pointB = null, n = 0, i = null, this.onLostFocus(); }, this._contextMenuBind = (a) => this.onContextMenu(a), r && r.addEventListener("contextmenu", this._contextMenuBind, !1); const s = this.camera.getScene().getEngine().getHostWindow(); s && ye.RegisterTopRootEvents(s, [{ name: "blur", handler: this._onLostFocus }]); } /** * Detach the current controls from the specified dom element. */ detachControl() { if (this._onLostFocus) { const e = this.camera.getScene().getEngine().getHostWindow(); e && ye.UnregisterTopRootEvents(e, [{ name: "blur", handler: this._onLostFocus }]); } if (this._observer) { if (this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer), this._observer = null, this._contextMenuBind) { const e = this.camera.getScene().getEngine().getInputElement(); e && e.removeEventListener("contextmenu", this._contextMenuBind); } this._onLostFocus = null; } this._altKey = !1, this._ctrlKey = !1, this._metaKey = !1, this._shiftKey = !1, this._buttonsPressed = 0, this._currentActiveButton = -1; } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "BaseCameraPointersInput"; } /** * Get the friendly name associated with the input class. * @returns the input friendly name */ getSimpleName() { return "pointers"; } /** * Called on pointer POINTERDOUBLETAP event. * Override this method to provide functionality on POINTERDOUBLETAP event. * @param type */ // eslint-disable-next-line @typescript-eslint/no-unused-vars onDoubleTap(e) { } /** * Called on pointer POINTERMOVE event if only a single touch is active. * Override this method to provide functionality. * @param point * @param offsetX * @param offsetY */ // eslint-disable-next-line @typescript-eslint/no-unused-vars onTouch(e, t, r) { } /** * Called on pointer POINTERMOVE event if multiple touches are active. * Override this method to provide functionality. * @param _pointA * @param _pointB * @param previousPinchSquaredDistance * @param pinchSquaredDistance * @param previousMultiTouchPanPosition * @param multiTouchPanPosition */ // eslint-disable-next-line @typescript-eslint/no-unused-vars onMultiTouch(e, t, r, n, i, s) { } /** * Called on JS contextmenu event. * Override this method to provide functionality. * @param evt */ onContextMenu(e) { e.preventDefault(); } /** * Called each time a new POINTERDOWN event occurs. Ie, for each button * press. * Override this method to provide functionality. * @param _evt Defines the event to track */ onButtonDown(e) { } /** * Called each time a new POINTERUP event occurs. Ie, for each button * release. * Override this method to provide functionality. * @param _evt Defines the event to track */ onButtonUp(e) { } /** * Called when window becomes inactive. * Override this method to provide functionality. */ onLostFocus() { } } C([ M() ], VO.prototype, "buttons", void 0); var VA = {}; class RR { /** * Instantiate a new Camera Input Manager. * @param camera Defines the camera the input manager belongs to */ constructor(e) { this.attachedToElement = !1, this.attached = {}, this.camera = e, this.checkInputs = () => { }; } /** * Add an input method to a camera * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs * @param input Camera input method */ add(e) { const t = e.getSimpleName(); if (this.attached[t]) { Se.Warn("camera input of type " + t + " already exists on camera"); return; } this.attached[t] = e, e.camera = this.camera, e.checkInputs && (this.checkInputs = this._addCheckInputs(e.checkInputs.bind(e))), this.attachedToElement && e.attachControl(this.noPreventDefault); } /** * Remove a specific input method from a camera * example: camera.inputs.remove(camera.inputs.attached.mouse); * @param inputToRemove camera input method */ remove(e) { for (const t in this.attached) { const r = this.attached[t]; if (r === e) { r.detachControl(), r.camera = null, delete this.attached[t], this.rebuildInputCheck(); return; } } } /** * Remove a specific input type from a camera * example: camera.inputs.remove("ArcRotateCameraGamepadInput"); * @param inputType the type of the input to remove */ removeByType(e) { for (const t in this.attached) { const r = this.attached[t]; r.getClassName() === e && (r.detachControl(), r.camera = null, delete this.attached[t], this.rebuildInputCheck()); } } _addCheckInputs(e) { const t = this.checkInputs; return () => { t(), e(); }; } /** * Attach the input controls to the currently attached dom element to listen the events from. * @param input Defines the input to attach */ attachInput(e) { this.attachedToElement && e.attachControl(this.noPreventDefault); } /** * Attach the current manager inputs controls to a specific dom element to listen the events from. * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachElement(e = !1) { if (!this.attachedToElement) { e = Tr.ForceAttachControlToAlwaysPreventDefault ? !1 : e, this.attachedToElement = !0, this.noPreventDefault = e; for (const t in this.attached) this.attached[t].attachControl(e); } } /** * Detach the current manager inputs controls from a specific dom element. * @param disconnect Defines whether the input should be removed from the current list of attached inputs */ detachElement(e = !1) { for (const t in this.attached) this.attached[t].detachControl(), e && (this.attached[t].camera = null); this.attachedToElement = !1; } /** * Rebuild the dynamic inputCheck function from the current list of * defined inputs in the manager. */ rebuildInputCheck() { this.checkInputs = () => { }; for (const e in this.attached) { const t = this.attached[e]; t.checkInputs && (this.checkInputs = this._addCheckInputs(t.checkInputs.bind(t))); } } /** * Remove all attached input methods from a camera */ clear() { this.attachedToElement && this.detachElement(!0), this.attached = {}, this.attachedToElement = !1, this.checkInputs = () => { }; } /** * Serialize the current input manager attached to a camera. * This ensures than once parsed, * the input associated to the camera will be identical to the current ones * @param serializedCamera Defines the camera serialization JSON the input serialization should write to */ serialize(e) { const t = {}; for (const r in this.attached) { const n = this.attached[r], i = jt.Serialize(n); t[n.getClassName()] = i; } e.inputsmgr = t; } /** * Parses an input manager serialized JSON to restore the previous list of inputs * and states associated to a camera. * @param parsedCamera Defines the JSON to parse */ parse(e) { const t = e.inputsmgr; if (t) { this.clear(); for (const r in t) { const n = VA[r]; if (n) { const i = t[r], s = jt.Parse(() => new n(), i, null); this.add(s); } } } else for (const r in this.attached) { const n = VA[this.attached[r].getClassName()]; if (n) { const i = jt.Parse(() => new n(), e, null); this.remove(this.attached[r]), this.add(i); } } } } class F6e { /** * Initializes the gamepad x and y control stick values * @param x The x component of the gamepad control stick value * @param y The y component of the gamepad control stick value */ constructor(e, t) { this.x = e, this.y = t; } } class N9 { /** * Specifies if the gamepad has been connected */ get isConnected() { return this._isConnected; } /** * Initializes the gamepad * @param id The id of the gamepad * @param index The index of the gamepad * @param browserGamepad The browser gamepad * @param leftStickX The x component of the left joystick * @param leftStickY The y component of the left joystick * @param rightStickX The x component of the right joystick * @param rightStickY The y component of the right joystick */ constructor(e, t, r, n = 0, i = 1, s = 2, a = 3) { this.id = e, this.index = t, this.browserGamepad = r, this._leftStick = { x: 0, y: 0 }, this._rightStick = { x: 0, y: 0 }, this._isConnected = !0, this._invertLeftStickY = !1, this.type = N9.GAMEPAD, this._leftStickAxisX = n, this._leftStickAxisY = i, this._rightStickAxisX = s, this._rightStickAxisY = a, this.browserGamepad.axes.length >= 2 && (this._leftStick = { x: this.browserGamepad.axes[this._leftStickAxisX], y: this.browserGamepad.axes[this._leftStickAxisY] }), this.browserGamepad.axes.length >= 4 && (this._rightStick = { x: this.browserGamepad.axes[this._rightStickAxisX], y: this.browserGamepad.axes[this._rightStickAxisY] }); } /** * Callback triggered when the left joystick has changed * @param callback */ onleftstickchanged(e) { this._onleftstickchanged = e; } /** * Callback triggered when the right joystick has changed * @param callback */ onrightstickchanged(e) { this._onrightstickchanged = e; } /** * Gets the left joystick */ get leftStick() { return this._leftStick; } /** * Sets the left joystick values */ set leftStick(e) { this._onleftstickchanged && (this._leftStick.x !== e.x || this._leftStick.y !== e.y) && this._onleftstickchanged(e), this._leftStick = e; } /** * Gets the right joystick */ get rightStick() { return this._rightStick; } /** * Sets the right joystick value */ set rightStick(e) { this._onrightstickchanged && (this._rightStick.x !== e.x || this._rightStick.y !== e.y) && this._onrightstickchanged(e), this._rightStick = e; } /** * Updates the gamepad joystick positions */ update() { this._leftStick && (this.leftStick = { x: this.browserGamepad.axes[this._leftStickAxisX], y: this.browserGamepad.axes[this._leftStickAxisY] }, this._invertLeftStickY && (this.leftStick.y *= -1)), this._rightStick && (this.rightStick = { x: this.browserGamepad.axes[this._rightStickAxisX], y: this.browserGamepad.axes[this._rightStickAxisY] }); } /** * Disposes the gamepad */ dispose() { } } N9.GAMEPAD = 0; N9.GENERIC = 1; N9.XBOX = 2; N9.POSE_ENABLED = 3; N9.DUALSHOCK = 4; class xee extends N9 { /** * Callback triggered when a button has been pressed * @param callback Called when a button has been pressed */ onbuttondown(e) { this._onbuttondown = e; } /** * Callback triggered when a button has been released * @param callback Called when a button has been released */ onbuttonup(e) { this._onbuttonup = e; } /** * Initializes the generic gamepad * @param id The id of the generic gamepad * @param index The index of the generic gamepad * @param browserGamepad The browser gamepad */ constructor(e, t, r) { super(e, t, r), this.onButtonDownObservable = new Oe(), this.onButtonUpObservable = new Oe(), this.type = N9.GENERIC, this._buttons = new Array(r.buttons.length); } _setButtonValue(e, t, r) { return e !== t && (e === 1 && (this._onbuttondown && this._onbuttondown(r), this.onButtonDownObservable.notifyObservers(r)), e === 0 && (this._onbuttonup && this._onbuttonup(r), this.onButtonUpObservable.notifyObservers(r))), e; } /** * Updates the generic gamepad */ update() { super.update(); for (let e = 0; e < this._buttons.length; e++) this._buttons[e] = this._setButtonValue(this.browserGamepad.buttons[e].value, this._buttons[e], e); } /** * Disposes the generic gamepad */ dispose() { super.dispose(), this.onButtonDownObservable.clear(), this.onButtonUpObservable.clear(); } } class VR { constructor() { this.gamepadRotationSensibility = 80, this.gamepadMoveSensibility = 40, this._yAxisScale = 1; } /** * Gets or sets a boolean indicating that Yaxis (for right stick) should be inverted */ get invertYAxis() { return this._yAxisScale !== 1; } set invertYAxis(e) { this._yAxisScale = e ? -1 : 1; } /** * Attach the input controls to a specific dom element to get the input from. */ attachControl() { const e = this.camera.getScene().gamepadManager; this._onGamepadConnectedObserver = e.onGamepadConnectedObservable.add((t) => { t.type !== N9.POSE_ENABLED && (!this.gamepad || t.type === N9.XBOX) && (this.gamepad = t); }), this._onGamepadDisconnectedObserver = e.onGamepadDisconnectedObservable.add((t) => { this.gamepad === t && (this.gamepad = null); }), this.gamepad = e.getGamepadByType(N9.XBOX); } /** * Detach the current controls from the specified dom element. */ detachControl() { this.camera.getScene().gamepadManager.onGamepadConnectedObservable.remove(this._onGamepadConnectedObserver), this.camera.getScene().gamepadManager.onGamepadDisconnectedObservable.remove(this._onGamepadDisconnectedObserver), this.gamepad = null; } /** * Update the current camera state depending on the inputs that have been used this frame. * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop. */ checkInputs() { if (this.gamepad) { const e = this.camera, t = this.gamepad.rightStick; if (t) { if (t.x != 0) { const n = t.x / this.gamepadRotationSensibility; n != 0 && Math.abs(n) > 5e-3 && (e.inertialAlphaOffset += n); } if (t.y != 0) { const n = t.y / this.gamepadRotationSensibility * this._yAxisScale; n != 0 && Math.abs(n) > 5e-3 && (e.inertialBetaOffset += n); } } const r = this.gamepad.leftStick; if (r && r.y != 0) { const n = r.y / this.gamepadMoveSensibility; n != 0 && Math.abs(n) > 5e-3 && (this.camera.inertialRadiusOffset -= n); } } } /** * Gets the class name of the current intput. * @returns the class name */ getClassName() { return "ArcRotateCameraGamepadInput"; } /** * Get the friendly name associated with the input class. * @returns the input friendly name */ getSimpleName() { return "gamepad"; } } C([ M() ], VR.prototype, "gamepadRotationSensibility", void 0); C([ M() ], VR.prototype, "gamepadMoveSensibility", void 0); VA.ArcRotateCameraGamepadInput = VR; class M0 { constructor() { this.keysUp = [38], this.keysDown = [40], this.keysLeft = [37], this.keysRight = [39], this.keysReset = [220], this.panningSensibility = 50, this.zoomingSensibility = 25, this.useAltToZoom = !0, this.angularSpeed = 0.01, this._keys = new Array(); } /** * Attach the input controls to a specific dom element to get the input from. * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachControl(e) { e = ye.BackCompatCameraNoPreventDefault(arguments), !this._onCanvasBlurObserver && (this._scene = this.camera.getScene(), this._engine = this._scene.getEngine(), this._onCanvasBlurObserver = this._engine.onCanvasBlurObservable.add(() => { this._keys.length = 0; }), this._onKeyboardObserver = this._scene.onKeyboardObservable.add((t) => { const r = t.event; if (!r.metaKey) { if (t.type === t4.KEYDOWN) this._ctrlPressed = r.ctrlKey, this._altPressed = r.altKey, (this.keysUp.indexOf(r.keyCode) !== -1 || this.keysDown.indexOf(r.keyCode) !== -1 || this.keysLeft.indexOf(r.keyCode) !== -1 || this.keysRight.indexOf(r.keyCode) !== -1 || this.keysReset.indexOf(r.keyCode) !== -1) && (this._keys.indexOf(r.keyCode) === -1 && this._keys.push(r.keyCode), r.preventDefault && (e || r.preventDefault())); else if (this.keysUp.indexOf(r.keyCode) !== -1 || this.keysDown.indexOf(r.keyCode) !== -1 || this.keysLeft.indexOf(r.keyCode) !== -1 || this.keysRight.indexOf(r.keyCode) !== -1 || this.keysReset.indexOf(r.keyCode) !== -1) { const n = this._keys.indexOf(r.keyCode); n >= 0 && this._keys.splice(n, 1), r.preventDefault && (e || r.preventDefault()); } } })); } /** * Detach the current controls from the specified dom element. */ detachControl() { this._scene && (this._onKeyboardObserver && this._scene.onKeyboardObservable.remove(this._onKeyboardObserver), this._onCanvasBlurObserver && this._engine.onCanvasBlurObservable.remove(this._onCanvasBlurObserver), this._onKeyboardObserver = null, this._onCanvasBlurObserver = null), this._keys.length = 0; } /** * Update the current camera state depending on the inputs that have been used this frame. * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop. */ checkInputs() { if (this._onKeyboardObserver) { const e = this.camera; for (let t = 0; t < this._keys.length; t++) { const r = this._keys[t]; this.keysLeft.indexOf(r) !== -1 ? this._ctrlPressed && this.camera._useCtrlForPanning ? e.inertialPanningX -= 1 / this.panningSensibility : e.inertialAlphaOffset -= this.angularSpeed : this.keysUp.indexOf(r) !== -1 ? this._ctrlPressed && this.camera._useCtrlForPanning ? e.inertialPanningY += 1 / this.panningSensibility : this._altPressed && this.useAltToZoom ? e.inertialRadiusOffset += 1 / this.zoomingSensibility : e.inertialBetaOffset -= this.angularSpeed : this.keysRight.indexOf(r) !== -1 ? this._ctrlPressed && this.camera._useCtrlForPanning ? e.inertialPanningX += 1 / this.panningSensibility : e.inertialAlphaOffset += this.angularSpeed : this.keysDown.indexOf(r) !== -1 ? this._ctrlPressed && this.camera._useCtrlForPanning ? e.inertialPanningY -= 1 / this.panningSensibility : this._altPressed && this.useAltToZoom ? e.inertialRadiusOffset -= 1 / this.zoomingSensibility : e.inertialBetaOffset += this.angularSpeed : this.keysReset.indexOf(r) !== -1 && e.useInputToRestoreState && e.restoreState(); } } } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "ArcRotateCameraKeyboardMoveInput"; } /** * Get the friendly name associated with the input class. * @returns the input friendly name */ getSimpleName() { return "keyboard"; } } C([ M() ], M0.prototype, "keysUp", void 0); C([ M() ], M0.prototype, "keysDown", void 0); C([ M() ], M0.prototype, "keysLeft", void 0); C([ M() ], M0.prototype, "keysRight", void 0); C([ M() ], M0.prototype, "keysReset", void 0); C([ M() ], M0.prototype, "panningSensibility", void 0); C([ M() ], M0.prototype, "zoomingSensibility", void 0); C([ M() ], M0.prototype, "useAltToZoom", void 0); C([ M() ], M0.prototype, "angularSpeed", void 0); VA.ArcRotateCameraKeyboardMoveInput = M0; const N6e = 40; class AU { constructor() { this.wheelPrecision = 3, this.zoomToMouseLocation = !1, this.wheelDeltaPercentage = 0, this.customComputeDeltaFromMouseWheel = null, this._viewOffset = new S(0, 0, 0), this._globalOffset = new S(0, 0, 0), this._inertialPanning = S.Zero(); } _computeDeltaFromMouseWheelLegacyEvent(e, t) { let r = 0; const n = e * 0.01 * this.wheelDeltaPercentage * t; return e > 0 ? r = n / (1 + this.wheelDeltaPercentage) : r = n * (1 + this.wheelDeltaPercentage), r; } /** * Attach the input controls to a specific dom element to get the input from. * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachControl(e) { e = ye.BackCompatCameraNoPreventDefault(arguments), this._wheel = (t) => { if (t.type !== ir.POINTERWHEEL) return; const r = t.event; let n = 0; const i = r.deltaMode === wm.DOM_DELTA_LINE ? N6e : 1, s = -(r.deltaY * i); if (this.customComputeDeltaFromMouseWheel) n = this.customComputeDeltaFromMouseWheel(s, this, r); else if (this.wheelDeltaPercentage) { if (n = this._computeDeltaFromMouseWheelLegacyEvent(s, this.camera.radius), n > 0) { let a = this.camera.radius, f = this.camera.inertialRadiusOffset + n; for (let o = 0; o < 20 && Math.abs(f) > 1e-3; o++) a -= f, f *= this.camera.inertia; a = Xt.Clamp(a, 0, Number.MAX_VALUE), n = this._computeDeltaFromMouseWheelLegacyEvent(s, a); } } else n = s / (this.wheelPrecision * 40); n && (this.zoomToMouseLocation ? (this._hitPlane || this._updateHitPlane(), this._zoomToMouse(n)) : this.camera.inertialRadiusOffset += n), r.preventDefault && (e || r.preventDefault()); }, this._observer = this.camera.getScene()._inputManager._addCameraPointerObserver(this._wheel, ir.POINTERWHEEL), this.zoomToMouseLocation && this._inertialPanning.setAll(0); } /** * Detach the current controls from the specified dom element. */ detachControl() { this._observer && (this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer), this._observer = null, this._wheel = null); } /** * Update the current camera state depending on the inputs that have been used this frame. * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop. */ checkInputs() { if (!this.zoomToMouseLocation) return; const e = this.camera; 0 + e.inertialAlphaOffset + e.inertialBetaOffset + e.inertialRadiusOffset && (this._updateHitPlane(), e.target.addInPlace(this._inertialPanning), this._inertialPanning.scaleInPlace(e.inertia), this._zeroIfClose(this._inertialPanning)); } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "ArcRotateCameraMouseWheelInput"; } /** * Get the friendly name associated with the input class. * @returns the input friendly name */ getSimpleName() { return "mousewheel"; } _updateHitPlane() { const e = this.camera, t = e.target.subtract(e.position); this._hitPlane = BA.FromPositionAndNormal(e.target, t); } // Get position on the hit plane _getPosition() { var e; const t = this.camera, r = t.getScene(), n = r.createPickingRay(r.pointerX, r.pointerY, he.Identity(), t, !1); (t.targetScreenOffset.x !== 0 || t.targetScreenOffset.y !== 0) && (this._viewOffset.set(t.targetScreenOffset.x, t.targetScreenOffset.y, 0), t.getViewMatrix().invertToRef(t._cameraTransformMatrix), this._globalOffset = S.TransformNormal(this._viewOffset, t._cameraTransformMatrix), n.origin.addInPlace(this._globalOffset)); let i = 0; return this._hitPlane && (i = (e = n.intersectsPlane(this._hitPlane)) !== null && e !== void 0 ? e : 0), n.origin.addInPlace(n.direction.scaleInPlace(i)); } _zoomToMouse(e) { var t, r; const n = this.camera, i = 1 - n.inertia; if (n.lowerRadiusLimit) { const d = (t = n.lowerRadiusLimit) !== null && t !== void 0 ? t : 0; n.radius - (n.inertialRadiusOffset + e) / i < d && (e = (n.radius - d) * i - n.inertialRadiusOffset); } if (n.upperRadiusLimit) { const d = (r = n.upperRadiusLimit) !== null && r !== void 0 ? r : 0; n.radius - (n.inertialRadiusOffset + e) / i > d && (e = (n.radius - d) * i - n.inertialRadiusOffset); } const a = e / i / n.radius, f = this._getPosition(), o = ue.Vector3[6]; f.subtractToRef(n.target, o), o.scaleInPlace(a), o.scaleInPlace(i), this._inertialPanning.addInPlace(o), n.inertialRadiusOffset += e; } // Sets x y or z of passed in vector to zero if less than Epsilon. _zeroIfClose(e) { Math.abs(e.x) < Dn && (e.x = 0), Math.abs(e.y) < Dn && (e.y = 0), Math.abs(e.z) < Dn && (e.z = 0); } } C([ M() ], AU.prototype, "wheelPrecision", void 0); C([ M() ], AU.prototype, "zoomToMouseLocation", void 0); C([ M() ], AU.prototype, "wheelDeltaPercentage", void 0); VA.ArcRotateCameraMouseWheelInput = AU; class ku extends VO { constructor() { super(...arguments), this.buttons = [0, 1, 2], this.angularSensibilityX = 1e3, this.angularSensibilityY = 1e3, this.pinchPrecision = 12, this.pinchDeltaPercentage = 0, this.useNaturalPinchZoom = !1, this.pinchZoom = !0, this.panningSensibility = 1e3, this.multiTouchPanning = !0, this.multiTouchPanAndZoom = !0, this.pinchInwards = !0, this._isPanClick = !1, this._twoFingerActivityCount = 0, this._isPinching = !1; } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "ArcRotateCameraPointersInput"; } /** * Move camera from multi touch panning positions. * @param previousMultiTouchPanPosition * @param multiTouchPanPosition */ _computeMultiTouchPanning(e, t) { if (this.panningSensibility !== 0 && e && t) { const r = t.x - e.x, n = t.y - e.y; this.camera.inertialPanningX += -r / this.panningSensibility, this.camera.inertialPanningY += n / this.panningSensibility; } } /** * Move camera from pinch zoom distances. * @param previousPinchSquaredDistance * @param pinchSquaredDistance */ _computePinchZoom(e, t) { const r = this.camera.radius || ku.MinimumRadiusForPinch; this.useNaturalPinchZoom ? this.camera.radius = r * Math.sqrt(e) / Math.sqrt(t) : this.pinchDeltaPercentage ? this.camera.inertialRadiusOffset += (t - e) * 1e-3 * r * this.pinchDeltaPercentage : this.camera.inertialRadiusOffset += (t - e) / (this.pinchPrecision * (this.pinchInwards ? 1 : -1) * (this.angularSensibilityX + this.angularSensibilityY) / 2); } /** * Called on pointer POINTERMOVE event if only a single touch is active. * @param point * @param offsetX * @param offsetY */ onTouch(e, t, r) { this.panningSensibility !== 0 && (this._ctrlKey && this.camera._useCtrlForPanning || this._isPanClick) ? (this.camera.inertialPanningX += -t / this.panningSensibility, this.camera.inertialPanningY += r / this.panningSensibility) : (this.camera.inertialAlphaOffset -= t / this.angularSensibilityX, this.camera.inertialBetaOffset -= r / this.angularSensibilityY); } /** * Called on pointer POINTERDOUBLETAP event. */ onDoubleTap() { this.camera.useInputToRestoreState && this.camera.restoreState(); } /** * Called on pointer POINTERMOVE event if multiple touches are active. * @param pointA * @param pointB * @param previousPinchSquaredDistance * @param pinchSquaredDistance * @param previousMultiTouchPanPosition * @param multiTouchPanPosition */ onMultiTouch(e, t, r, n, i, s) { r === 0 && i === null || n === 0 && s === null || (this.multiTouchPanAndZoom ? (this._computePinchZoom(r, n), this._computeMultiTouchPanning(i, s)) : this.multiTouchPanning && this.pinchZoom ? (this._twoFingerActivityCount++, this._isPinching || this._twoFingerActivityCount < 20 && Math.abs(Math.sqrt(n) - Math.sqrt(r)) > this.camera.pinchToPanMaxDistance ? (this._computePinchZoom(r, n), this._isPinching = !0) : this._computeMultiTouchPanning(i, s)) : this.multiTouchPanning ? this._computeMultiTouchPanning(i, s) : this.pinchZoom && this._computePinchZoom(r, n)); } /** * Called each time a new POINTERDOWN event occurs. Ie, for each button * press. * @param evt Defines the event to track */ onButtonDown(e) { this._isPanClick = e.button === this.camera._panningMouseButton; } /** * Called each time a new POINTERUP event occurs. Ie, for each button * release. * @param _evt Defines the event to track */ onButtonUp(e) { this._twoFingerActivityCount = 0, this._isPinching = !1; } /** * Called when window becomes inactive. */ onLostFocus() { this._isPanClick = !1, this._twoFingerActivityCount = 0, this._isPinching = !1; } } ku.MinimumRadiusForPinch = 1e-3; C([ M() ], ku.prototype, "buttons", void 0); C([ M() ], ku.prototype, "angularSensibilityX", void 0); C([ M() ], ku.prototype, "angularSensibilityY", void 0); C([ M() ], ku.prototype, "pinchPrecision", void 0); C([ M() ], ku.prototype, "pinchDeltaPercentage", void 0); C([ M() ], ku.prototype, "useNaturalPinchZoom", void 0); C([ M() ], ku.prototype, "pinchZoom", void 0); C([ M() ], ku.prototype, "panningSensibility", void 0); C([ M() ], ku.prototype, "multiTouchPanning", void 0); C([ M() ], ku.prototype, "multiTouchPanAndZoom", void 0); VA.ArcRotateCameraPointersInput = ku; class CO extends RR { /** * Instantiates a new ArcRotateCameraInputsManager. * @param camera Defines the camera the inputs belong to */ constructor(e) { super(e); } /** * Add mouse wheel input support to the input manager. * @returns the current input manager */ addMouseWheel() { return this.add(new AU()), this; } /** * Add pointers input support to the input manager. * @returns the current input manager */ addPointers() { return this.add(new ku()), this; } /** * Add keyboard input support to the input manager. * @returns the current input manager */ addKeyboard() { return this.add(new M0()), this; } } CO.prototype.addVRDeviceOrientation = function() { return this.add(new lQ()), this; }; class lQ { /** * Instantiate a new ArcRotateCameraVRDeviceOrientationInput. */ constructor() { this.alphaCorrection = 1, this.gammaCorrection = 1, this._alpha = 0, this._gamma = 0, this._dirty = !1, this._deviceOrientationHandler = (e) => this._onOrientationEvent(e); } /** * Attach the input controls to a specific dom element to get the input from. * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachControl(e) { e = ye.BackCompatCameraNoPreventDefault(arguments), this.camera.attachControl(e); const t = this.camera.getScene().getEngine().getHostWindow(); t && (typeof DeviceOrientationEvent < "u" && typeof DeviceOrientationEvent.requestPermission == "function" ? DeviceOrientationEvent.requestPermission().then((r) => { r === "granted" ? t.addEventListener("deviceorientation", this._deviceOrientationHandler) : ye.Warn("Permission not granted."); }).catch((r) => { ye.Error(r); }) : t.addEventListener("deviceorientation", this._deviceOrientationHandler)); } /** * @internal */ _onOrientationEvent(e) { e.alpha !== null && (this._alpha = (+e.alpha | 0) * this.alphaCorrection), e.gamma !== null && (this._gamma = (+e.gamma | 0) * this.gammaCorrection), this._dirty = !0; } /** * Update the current camera state depending on the inputs that have been used this frame. * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop. */ checkInputs() { this._dirty && (this._dirty = !1, this._gamma < 0 && (this._gamma = 180 + this._gamma), this.camera.alpha = -this._alpha / 180 * Math.PI % Math.PI * 2, this.camera.beta = this._gamma / 180 * Math.PI); } /** * Detach the current controls from the specified dom element. */ detachControl() { window.removeEventListener("deviceorientation", this._deviceOrientationHandler); } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "ArcRotateCameraVRDeviceOrientationInput"; } /** * Get the friendly name associated with the input class. * @returns the input friendly name */ getSimpleName() { return "VRDeviceOrientation"; } } VA.ArcRotateCameraVRDeviceOrientationInput = lQ; class jq { constructor() { this.keysForward = [87], this.keysBackward = [83], this.keysUp = [69], this.keysDown = [81], this.keysRight = [68], this.keysLeft = [65], this._keys = new Array(); } /** * Attach the input controls to a specific dom element to get the input from. * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachControl(e) { e = ye.BackCompatCameraNoPreventDefault(arguments), !this._onCanvasBlurObserver && (this._scene = this.camera.getScene(), this._engine = this._scene.getEngine(), this._onCanvasBlurObserver = this._engine.onCanvasBlurObservable.add(() => { this._keys.length = 0; }), this._onKeyboardObserver = this._scene.onKeyboardObservable.add((t) => { const r = t.event; if (t.type === t4.KEYDOWN) (this.keysForward.indexOf(r.keyCode) !== -1 || this.keysBackward.indexOf(r.keyCode) !== -1 || this.keysUp.indexOf(r.keyCode) !== -1 || this.keysDown.indexOf(r.keyCode) !== -1 || this.keysLeft.indexOf(r.keyCode) !== -1 || this.keysRight.indexOf(r.keyCode) !== -1) && (this._keys.indexOf(r.keyCode) === -1 && this._keys.push(r.keyCode), e || r.preventDefault()); else if (this.keysForward.indexOf(r.keyCode) !== -1 || this.keysBackward.indexOf(r.keyCode) !== -1 || this.keysUp.indexOf(r.keyCode) !== -1 || this.keysDown.indexOf(r.keyCode) !== -1 || this.keysLeft.indexOf(r.keyCode) !== -1 || this.keysRight.indexOf(r.keyCode) !== -1) { const n = this._keys.indexOf(r.keyCode); n >= 0 && this._keys.splice(n, 1), e || r.preventDefault(); } })); } /** * Detach the current controls from the specified dom element. */ detachControl() { this._scene && (this._onKeyboardObserver && this._scene.onKeyboardObservable.remove(this._onKeyboardObserver), this._onCanvasBlurObserver && this._engine.onCanvasBlurObservable.remove(this._onCanvasBlurObserver), this._onKeyboardObserver = null, this._onCanvasBlurObserver = null), this._keys.length = 0; } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "FlyCameraKeyboardInput"; } /** * @internal */ _onLostFocus() { this._keys.length = 0; } /** * Get the friendly name associated with the input class. * @returns the input friendly name */ getSimpleName() { return "keyboard"; } /** * Update the current camera state depending on the inputs that have been used this frame. * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop. */ checkInputs() { if (this._onKeyboardObserver) { const e = this.camera; for (let t = 0; t < this._keys.length; t++) { const r = this._keys[t], n = e._computeLocalCameraSpeed(); this.keysForward.indexOf(r) !== -1 ? e._localDirection.copyFromFloats(0, 0, n) : this.keysBackward.indexOf(r) !== -1 ? e._localDirection.copyFromFloats(0, 0, -n) : this.keysUp.indexOf(r) !== -1 ? e._localDirection.copyFromFloats(0, n, 0) : this.keysDown.indexOf(r) !== -1 ? e._localDirection.copyFromFloats(0, -n, 0) : this.keysRight.indexOf(r) !== -1 ? e._localDirection.copyFromFloats(n, 0, 0) : this.keysLeft.indexOf(r) !== -1 && e._localDirection.copyFromFloats(-n, 0, 0), e.getScene().useRightHandedSystem && (e._localDirection.z *= -1), e.getViewMatrix().invertToRef(e._cameraTransformMatrix), S.TransformNormalToRef(e._localDirection, e._cameraTransformMatrix, e._transformedDirection), e.cameraDirection.addInPlace(e._transformedDirection); } } } } C([ M() ], jq.prototype, "keysForward", void 0); C([ M() ], jq.prototype, "keysBackward", void 0); C([ M() ], jq.prototype, "keysUp", void 0); C([ M() ], jq.prototype, "keysDown", void 0); C([ M() ], jq.prototype, "keysRight", void 0); C([ M() ], jq.prototype, "keysLeft", void 0); VA.FlyCameraKeyboardInput = jq; class CR { /** * Listen to mouse events to control the camera. * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs */ constructor() { this.buttons = [0, 1, 2], this.buttonsYaw = [-1, 0, 1], this.buttonsPitch = [-1, 0, 1], this.buttonsRoll = [2], this.activeButton = -1, this.angularSensibility = 1e3, this._previousPosition = null; } /** * Attach the mouse control to the HTML DOM element. * @param noPreventDefault Defines whether events caught by the controls should call preventdefault(). */ attachControl(e) { e = ye.BackCompatCameraNoPreventDefault(arguments), this._noPreventDefault = e, this._observer = this.camera.getScene()._inputManager._addCameraPointerObserver((t) => { this._pointerInput(t); }, ir.POINTERDOWN | ir.POINTERUP | ir.POINTERMOVE), this._rollObserver = this.camera.getScene().onBeforeRenderObservable.add(() => { this.camera.rollCorrect && this.camera.restoreRoll(this.camera.rollCorrect); }); } /** * Detach the current controls from the specified dom element. */ detachControl() { this._observer && (this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer), this.camera.getScene().onBeforeRenderObservable.remove(this._rollObserver), this._observer = null, this._rollObserver = null, this._previousPosition = null, this._noPreventDefault = void 0); } /** * Gets the class name of the current input. * @returns the class name. */ getClassName() { return "FlyCameraMouseInput"; } /** * Get the friendly name associated with the input class. * @returns the input's friendly name. */ getSimpleName() { return "mouse"; } // Track mouse movement, when the pointer is not locked. _pointerInput(e) { const t = e.event, n = this.camera.getEngine(); if (!this.touchEnabled && t.pointerType === "touch" || e.type !== ir.POINTERMOVE && this.buttons.indexOf(t.button) === -1) return; const i = t.target; if (e.type === ir.POINTERDOWN) { try { i == null || i.setPointerCapture(t.pointerId); } catch { } this._previousPosition = { x: t.clientX, y: t.clientY }, this.activeButton = t.button, this._noPreventDefault || (t.preventDefault(), this._element.focus()), n.isPointerLock && this._onMouseMove(e.event); } else if (e.type === ir.POINTERUP) { try { i == null || i.releasePointerCapture(t.pointerId); } catch { } this.activeButton = -1, this._previousPosition = null, this._noPreventDefault || t.preventDefault(); } else if (e.type === ir.POINTERMOVE) { if (!this._previousPosition) { n.isPointerLock && this._onMouseMove(e.event); return; } const s = t.clientX - this._previousPosition.x, a = t.clientY - this._previousPosition.y; this._rotateCamera(s, a), this._previousPosition = { x: t.clientX, y: t.clientY }, this._noPreventDefault || t.preventDefault(); } } // Track mouse movement, when pointer is locked. _onMouseMove(e) { if (!this.camera.getEngine().isPointerLock) return; const n = e.movementX, i = e.movementY; this._rotateCamera(n, i), this._previousPosition = null, this._noPreventDefault || e.preventDefault(); } /** * Rotate camera by mouse offset. * @param offsetX * @param offsetY */ _rotateCamera(e, t) { const r = this.camera, n = r._calculateHandednessMultiplier(); e *= n; const i = e / this.angularSensibility, s = t / this.angularSensibility, a = Ze.RotationYawPitchRoll(r.rotation.y, r.rotation.x, r.rotation.z); let f; if (this.buttonsPitch.some((o) => o === this.activeButton) && (f = Ze.RotationAxis(bf.X, s), a.multiplyInPlace(f)), this.buttonsYaw.some((o) => o === this.activeButton)) { f = Ze.RotationAxis(bf.Y, i), a.multiplyInPlace(f); const o = r.bankedTurnLimit + r._trackRoll; if (r.bankedTurn && -o < r.rotation.z && r.rotation.z < o) { const d = r.bankedTurnMultiplier * -i; f = Ze.RotationAxis(bf.Z, d), a.multiplyInPlace(f); } } this.buttonsRoll.some((o) => o === this.activeButton) && (f = Ze.RotationAxis(bf.Z, -i), r._trackRoll -= i, a.multiplyInPlace(f)), a.toEulerAnglesToRef(r.rotation); } } C([ M() ], CR.prototype, "buttons", void 0); C([ M() ], CR.prototype, "angularSensibility", void 0); VA.FlyCameraMouseInput = CR; class aA { constructor() { this.keysHeightOffsetIncr = [38], this.keysHeightOffsetDecr = [40], this.keysHeightOffsetModifierAlt = !1, this.keysHeightOffsetModifierCtrl = !1, this.keysHeightOffsetModifierShift = !1, this.keysRotationOffsetIncr = [37], this.keysRotationOffsetDecr = [39], this.keysRotationOffsetModifierAlt = !1, this.keysRotationOffsetModifierCtrl = !1, this.keysRotationOffsetModifierShift = !1, this.keysRadiusIncr = [40], this.keysRadiusDecr = [38], this.keysRadiusModifierAlt = !0, this.keysRadiusModifierCtrl = !1, this.keysRadiusModifierShift = !1, this.heightSensibility = 1, this.rotationSensibility = 1, this.radiusSensibility = 1, this._keys = new Array(); } /** * Attach the input controls to a specific dom element to get the input from. * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachControl(e) { e = ye.BackCompatCameraNoPreventDefault(arguments), !this._onCanvasBlurObserver && (this._scene = this.camera.getScene(), this._engine = this._scene.getEngine(), this._onCanvasBlurObserver = this._engine.onCanvasBlurObservable.add(() => { this._keys.length = 0; }), this._onKeyboardObserver = this._scene.onKeyboardObservable.add((t) => { const r = t.event; if (!r.metaKey) { if (t.type === t4.KEYDOWN) this._ctrlPressed = r.ctrlKey, this._altPressed = r.altKey, this._shiftPressed = r.shiftKey, (this.keysHeightOffsetIncr.indexOf(r.keyCode) !== -1 || this.keysHeightOffsetDecr.indexOf(r.keyCode) !== -1 || this.keysRotationOffsetIncr.indexOf(r.keyCode) !== -1 || this.keysRotationOffsetDecr.indexOf(r.keyCode) !== -1 || this.keysRadiusIncr.indexOf(r.keyCode) !== -1 || this.keysRadiusDecr.indexOf(r.keyCode) !== -1) && (this._keys.indexOf(r.keyCode) === -1 && this._keys.push(r.keyCode), r.preventDefault && (e || r.preventDefault())); else if (this.keysHeightOffsetIncr.indexOf(r.keyCode) !== -1 || this.keysHeightOffsetDecr.indexOf(r.keyCode) !== -1 || this.keysRotationOffsetIncr.indexOf(r.keyCode) !== -1 || this.keysRotationOffsetDecr.indexOf(r.keyCode) !== -1 || this.keysRadiusIncr.indexOf(r.keyCode) !== -1 || this.keysRadiusDecr.indexOf(r.keyCode) !== -1) { const n = this._keys.indexOf(r.keyCode); n >= 0 && this._keys.splice(n, 1), r.preventDefault && (e || r.preventDefault()); } } })); } /** * Detach the current controls from the specified dom element. */ detachControl() { this._scene && (this._onKeyboardObserver && this._scene.onKeyboardObservable.remove(this._onKeyboardObserver), this._onCanvasBlurObserver && this._engine.onCanvasBlurObservable.remove(this._onCanvasBlurObserver), this._onKeyboardObserver = null, this._onCanvasBlurObserver = null), this._keys.length = 0; } /** * Update the current camera state depending on the inputs that have been used this frame. * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop. */ checkInputs() { this._onKeyboardObserver && this._keys.forEach((e) => { this.keysHeightOffsetIncr.indexOf(e) !== -1 && this._modifierHeightOffset() ? this.camera.heightOffset += this.heightSensibility : this.keysHeightOffsetDecr.indexOf(e) !== -1 && this._modifierHeightOffset() ? this.camera.heightOffset -= this.heightSensibility : this.keysRotationOffsetIncr.indexOf(e) !== -1 && this._modifierRotationOffset() ? (this.camera.rotationOffset += this.rotationSensibility, this.camera.rotationOffset %= 360) : this.keysRotationOffsetDecr.indexOf(e) !== -1 && this._modifierRotationOffset() ? (this.camera.rotationOffset -= this.rotationSensibility, this.camera.rotationOffset %= 360) : this.keysRadiusIncr.indexOf(e) !== -1 && this._modifierRadius() ? this.camera.radius += this.radiusSensibility : this.keysRadiusDecr.indexOf(e) !== -1 && this._modifierRadius() && (this.camera.radius -= this.radiusSensibility); }); } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "FollowCameraKeyboardMoveInput"; } /** * Get the friendly name associated with the input class. * @returns the input friendly name */ getSimpleName() { return "keyboard"; } /** * Check if the pressed modifier keys (Alt/Ctrl/Shift) match those configured to * allow modification of the heightOffset value. */ _modifierHeightOffset() { return this.keysHeightOffsetModifierAlt === this._altPressed && this.keysHeightOffsetModifierCtrl === this._ctrlPressed && this.keysHeightOffsetModifierShift === this._shiftPressed; } /** * Check if the pressed modifier keys (Alt/Ctrl/Shift) match those configured to * allow modification of the rotationOffset value. */ _modifierRotationOffset() { return this.keysRotationOffsetModifierAlt === this._altPressed && this.keysRotationOffsetModifierCtrl === this._ctrlPressed && this.keysRotationOffsetModifierShift === this._shiftPressed; } /** * Check if the pressed modifier keys (Alt/Ctrl/Shift) match those configured to * allow modification of the radius value. */ _modifierRadius() { return this.keysRadiusModifierAlt === this._altPressed && this.keysRadiusModifierCtrl === this._ctrlPressed && this.keysRadiusModifierShift === this._shiftPressed; } } C([ M() ], aA.prototype, "keysHeightOffsetIncr", void 0); C([ M() ], aA.prototype, "keysHeightOffsetDecr", void 0); C([ M() ], aA.prototype, "keysHeightOffsetModifierAlt", void 0); C([ M() ], aA.prototype, "keysHeightOffsetModifierCtrl", void 0); C([ M() ], aA.prototype, "keysHeightOffsetModifierShift", void 0); C([ M() ], aA.prototype, "keysRotationOffsetIncr", void 0); C([ M() ], aA.prototype, "keysRotationOffsetDecr", void 0); C([ M() ], aA.prototype, "keysRotationOffsetModifierAlt", void 0); C([ M() ], aA.prototype, "keysRotationOffsetModifierCtrl", void 0); C([ M() ], aA.prototype, "keysRotationOffsetModifierShift", void 0); C([ M() ], aA.prototype, "keysRadiusIncr", void 0); C([ M() ], aA.prototype, "keysRadiusDecr", void 0); C([ M() ], aA.prototype, "keysRadiusModifierAlt", void 0); C([ M() ], aA.prototype, "keysRadiusModifierCtrl", void 0); C([ M() ], aA.prototype, "keysRadiusModifierShift", void 0); C([ M() ], aA.prototype, "heightSensibility", void 0); C([ M() ], aA.prototype, "rotationSensibility", void 0); C([ M() ], aA.prototype, "radiusSensibility", void 0); VA.FollowCameraKeyboardMoveInput = aA; class qD { constructor() { this.axisControlRadius = !0, this.axisControlHeight = !1, this.axisControlRotation = !1, this.wheelPrecision = 3, this.wheelDeltaPercentage = 0; } /** * Attach the input controls to a specific dom element to get the input from. * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachControl(e) { e = ye.BackCompatCameraNoPreventDefault(arguments), this._wheel = (t) => { if (t.type !== ir.POINTERWHEEL) return; const r = t.event; let n = 0; const i = Math.max(-1, Math.min(1, r.deltaY)); this.wheelDeltaPercentage ? (console.assert(this.axisControlRadius + this.axisControlHeight + this.axisControlRotation <= 1, "wheelDeltaPercentage only usable when mouse wheel controls ONE axis. Currently enabled: axisControlRadius: " + this.axisControlRadius + ", axisControlHeightOffset: " + this.axisControlHeight + ", axisControlRotationOffset: " + this.axisControlRotation), this.axisControlRadius ? n = i * 0.01 * this.wheelDeltaPercentage * this.camera.radius : this.axisControlHeight ? n = i * 0.01 * this.wheelDeltaPercentage * this.camera.heightOffset : this.axisControlRotation && (n = i * 0.01 * this.wheelDeltaPercentage * this.camera.rotationOffset)) : n = i * this.wheelPrecision, n && (this.axisControlRadius ? this.camera.radius += n : this.axisControlHeight ? this.camera.heightOffset -= n : this.axisControlRotation && (this.camera.rotationOffset -= n)), r.preventDefault && (e || r.preventDefault()); }, this._observer = this.camera.getScene()._inputManager._addCameraPointerObserver(this._wheel, ir.POINTERWHEEL); } /** * Detach the current controls from the specified dom element. */ detachControl() { this._observer && (this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer), this._observer = null, this._wheel = null); } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "ArcRotateCameraMouseWheelInput"; } /** * Get the friendly name associated with the input class. * @returns the input friendly name */ getSimpleName() { return "mousewheel"; } } C([ M() ], qD.prototype, "axisControlRadius", void 0); C([ M() ], qD.prototype, "axisControlHeight", void 0); C([ M() ], qD.prototype, "axisControlRotation", void 0); C([ M() ], qD.prototype, "wheelPrecision", void 0); C([ M() ], qD.prototype, "wheelDeltaPercentage", void 0); VA.FollowCameraMouseWheelInput = qD; class Eu extends VO { constructor() { super(...arguments), this.angularSensibilityX = 1, this.angularSensibilityY = 1, this.pinchPrecision = 1e4, this.pinchDeltaPercentage = 0, this.axisXControlRadius = !1, this.axisXControlHeight = !1, this.axisXControlRotation = !0, this.axisYControlRadius = !1, this.axisYControlHeight = !0, this.axisYControlRotation = !1, this.axisPinchControlRadius = !0, this.axisPinchControlHeight = !1, this.axisPinchControlRotation = !1, this.warningEnable = !0, this._warningCounter = 0; } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "FollowCameraPointersInput"; } onTouch(e, t, r) { this._warning(), this.axisXControlRotation ? this.camera.rotationOffset += t / this.angularSensibilityX : this.axisYControlRotation && (this.camera.rotationOffset += r / this.angularSensibilityX), this.axisXControlHeight ? this.camera.heightOffset += t / this.angularSensibilityY : this.axisYControlHeight && (this.camera.heightOffset += r / this.angularSensibilityY), this.axisXControlRadius ? this.camera.radius -= t / this.angularSensibilityY : this.axisYControlRadius && (this.camera.radius -= r / this.angularSensibilityY); } onMultiTouch(e, t, r, n, i, s) { if (r === 0 && i === null || n === 0 && s === null) return; let a = (n - r) / (this.pinchPrecision * (this.angularSensibilityX + this.angularSensibilityY) / 2); this.pinchDeltaPercentage ? (a *= 0.01 * this.pinchDeltaPercentage, this.axisPinchControlRotation && (this.camera.rotationOffset += a * this.camera.rotationOffset), this.axisPinchControlHeight && (this.camera.heightOffset += a * this.camera.heightOffset), this.axisPinchControlRadius && (this.camera.radius -= a * this.camera.radius)) : (this.axisPinchControlRotation && (this.camera.rotationOffset += a), this.axisPinchControlHeight && (this.camera.heightOffset += a), this.axisPinchControlRadius && (this.camera.radius -= a)); } _warning() { if (!this.warningEnable || this._warningCounter++ % 100 !== 0) return; const e = "It probably only makes sense to control ONE camera property with each pointer axis. Set 'warningEnable = false' if you are sure. Currently enabled: "; console.assert(this.axisXControlRotation + this.axisXControlHeight + this.axisXControlRadius <= 1, e + "axisXControlRotation: " + this.axisXControlRotation + ", axisXControlHeight: " + this.axisXControlHeight + ", axisXControlRadius: " + this.axisXControlRadius), console.assert(this.axisYControlRotation + this.axisYControlHeight + this.axisYControlRadius <= 1, e + "axisYControlRotation: " + this.axisYControlRotation + ", axisYControlHeight: " + this.axisYControlHeight + ", axisYControlRadius: " + this.axisYControlRadius), console.assert(this.axisPinchControlRotation + this.axisPinchControlHeight + this.axisPinchControlRadius <= 1, e + "axisPinchControlRotation: " + this.axisPinchControlRotation + ", axisPinchControlHeight: " + this.axisPinchControlHeight + ", axisPinchControlRadius: " + this.axisPinchControlRadius); } } C([ M() ], Eu.prototype, "angularSensibilityX", void 0); C([ M() ], Eu.prototype, "angularSensibilityY", void 0); C([ M() ], Eu.prototype, "pinchPrecision", void 0); C([ M() ], Eu.prototype, "pinchDeltaPercentage", void 0); C([ M() ], Eu.prototype, "axisXControlRadius", void 0); C([ M() ], Eu.prototype, "axisXControlHeight", void 0); C([ M() ], Eu.prototype, "axisXControlRotation", void 0); C([ M() ], Eu.prototype, "axisYControlRadius", void 0); C([ M() ], Eu.prototype, "axisYControlHeight", void 0); C([ M() ], Eu.prototype, "axisYControlRotation", void 0); C([ M() ], Eu.prototype, "axisPinchControlRadius", void 0); C([ M() ], Eu.prototype, "axisPinchControlHeight", void 0); C([ M() ], Eu.prototype, "axisPinchControlRotation", void 0); VA.FollowCameraPointersInput = Eu; class wc { constructor() { this.keysUp = [38], this.keysUpward = [33], this.keysDown = [40], this.keysDownward = [34], this.keysLeft = [37], this.keysRight = [39], this.rotationSpeed = 0.5, this.keysRotateLeft = [], this.keysRotateRight = [], this.keysRotateUp = [], this.keysRotateDown = [], this._keys = new Array(); } /** * Attach the input controls to a specific dom element to get the input from. * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachControl(e) { e = ye.BackCompatCameraNoPreventDefault(arguments), !this._onCanvasBlurObserver && (this._scene = this.camera.getScene(), this._engine = this._scene.getEngine(), this._onCanvasBlurObserver = this._engine.onCanvasBlurObservable.add(() => { this._keys.length = 0; }), this._onKeyboardObserver = this._scene.onKeyboardObservable.add((t) => { const r = t.event; if (!r.metaKey) { if (t.type === t4.KEYDOWN) (this.keysUp.indexOf(r.keyCode) !== -1 || this.keysDown.indexOf(r.keyCode) !== -1 || this.keysLeft.indexOf(r.keyCode) !== -1 || this.keysRight.indexOf(r.keyCode) !== -1 || this.keysUpward.indexOf(r.keyCode) !== -1 || this.keysDownward.indexOf(r.keyCode) !== -1 || this.keysRotateLeft.indexOf(r.keyCode) !== -1 || this.keysRotateRight.indexOf(r.keyCode) !== -1 || this.keysRotateUp.indexOf(r.keyCode) !== -1 || this.keysRotateDown.indexOf(r.keyCode) !== -1) && (this._keys.indexOf(r.keyCode) === -1 && this._keys.push(r.keyCode), e || r.preventDefault()); else if (this.keysUp.indexOf(r.keyCode) !== -1 || this.keysDown.indexOf(r.keyCode) !== -1 || this.keysLeft.indexOf(r.keyCode) !== -1 || this.keysRight.indexOf(r.keyCode) !== -1 || this.keysUpward.indexOf(r.keyCode) !== -1 || this.keysDownward.indexOf(r.keyCode) !== -1 || this.keysRotateLeft.indexOf(r.keyCode) !== -1 || this.keysRotateRight.indexOf(r.keyCode) !== -1 || this.keysRotateUp.indexOf(r.keyCode) !== -1 || this.keysRotateDown.indexOf(r.keyCode) !== -1) { const n = this._keys.indexOf(r.keyCode); n >= 0 && this._keys.splice(n, 1), e || r.preventDefault(); } } })); } /** * Detach the current controls from the specified dom element. */ detachControl() { this._scene && (this._onKeyboardObserver && this._scene.onKeyboardObservable.remove(this._onKeyboardObserver), this._onCanvasBlurObserver && this._engine.onCanvasBlurObservable.remove(this._onCanvasBlurObserver), this._onKeyboardObserver = null, this._onCanvasBlurObserver = null), this._keys.length = 0; } /** * Update the current camera state depending on the inputs that have been used this frame. * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop. */ checkInputs() { if (this._onKeyboardObserver) { const e = this.camera; for (let t = 0; t < this._keys.length; t++) { const r = this._keys[t], n = e._computeLocalCameraSpeed(); this.keysLeft.indexOf(r) !== -1 ? e._localDirection.copyFromFloats(-n, 0, 0) : this.keysUp.indexOf(r) !== -1 ? e._localDirection.copyFromFloats(0, 0, n) : this.keysRight.indexOf(r) !== -1 ? e._localDirection.copyFromFloats(n, 0, 0) : this.keysDown.indexOf(r) !== -1 ? e._localDirection.copyFromFloats(0, 0, -n) : this.keysUpward.indexOf(r) !== -1 ? e._localDirection.copyFromFloats(0, n, 0) : this.keysDownward.indexOf(r) !== -1 ? e._localDirection.copyFromFloats(0, -n, 0) : this.keysRotateLeft.indexOf(r) !== -1 ? (e._localDirection.copyFromFloats(0, 0, 0), e.cameraRotation.y -= this._getLocalRotation()) : this.keysRotateRight.indexOf(r) !== -1 ? (e._localDirection.copyFromFloats(0, 0, 0), e.cameraRotation.y += this._getLocalRotation()) : this.keysRotateUp.indexOf(r) !== -1 ? (e._localDirection.copyFromFloats(0, 0, 0), e.cameraRotation.x -= this._getLocalRotation()) : this.keysRotateDown.indexOf(r) !== -1 && (e._localDirection.copyFromFloats(0, 0, 0), e.cameraRotation.x += this._getLocalRotation()), e.getScene().useRightHandedSystem && (e._localDirection.z *= -1), e.getViewMatrix().invertToRef(e._cameraTransformMatrix), S.TransformNormalToRef(e._localDirection, e._cameraTransformMatrix, e._transformedDirection), e.cameraDirection.addInPlace(e._transformedDirection); } } } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "FreeCameraKeyboardMoveInput"; } /** @internal */ _onLostFocus() { this._keys.length = 0; } /** * Get the friendly name associated with the input class. * @returns the input friendly name */ getSimpleName() { return "keyboard"; } _getLocalRotation() { const e = this.camera._calculateHandednessMultiplier(); return this.rotationSpeed * this._engine.getDeltaTime() / 1e3 * e; } } C([ M() ], wc.prototype, "keysUp", void 0); C([ M() ], wc.prototype, "keysUpward", void 0); C([ M() ], wc.prototype, "keysDown", void 0); C([ M() ], wc.prototype, "keysDownward", void 0); C([ M() ], wc.prototype, "keysLeft", void 0); C([ M() ], wc.prototype, "keysRight", void 0); C([ M() ], wc.prototype, "rotationSpeed", void 0); C([ M() ], wc.prototype, "keysRotateLeft", void 0); C([ M() ], wc.prototype, "keysRotateRight", void 0); C([ M() ], wc.prototype, "keysRotateUp", void 0); C([ M() ], wc.prototype, "keysRotateDown", void 0); VA.FreeCameraKeyboardMoveInput = wc; class OR { /** * Manage the mouse inputs to control the movement of a free camera. * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs * @param touchEnabled Defines if touch is enabled or not */ constructor(e = !0) { this.touchEnabled = e, this.buttons = [0, 1, 2], this.angularSensibility = 2e3, this._previousPosition = null, this.onPointerMovedObservable = new Oe(), this._allowCameraRotation = !0, this._currentActiveButton = -1, this._activePointerId = -1; } /** * Attach the input controls to a specific dom element to get the input from. * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachControl(e) { e = ye.BackCompatCameraNoPreventDefault(arguments); const t = this.camera.getEngine(), r = t.getInputElement(); this._pointerInput || (this._pointerInput = (n) => { const i = n.event, s = i.pointerType === "touch"; if (!this.touchEnabled && s || n.type !== ir.POINTERMOVE && this.buttons.indexOf(i.button) === -1) return; const a = i.target; if (n.type === ir.POINTERDOWN) { if (s && this._activePointerId !== -1 || !s && this._currentActiveButton !== -1) return; this._activePointerId = i.pointerId; try { a == null || a.setPointerCapture(i.pointerId); } catch { } this._currentActiveButton === -1 && (this._currentActiveButton = i.button), this._previousPosition = { x: i.clientX, y: i.clientY }, e || (i.preventDefault(), r && r.focus()), t.isPointerLock && this._onMouseMove && this._onMouseMove(n.event); } else if (n.type === ir.POINTERUP) { if (s && this._activePointerId !== i.pointerId || !s && this._currentActiveButton !== i.button) return; try { a == null || a.releasePointerCapture(i.pointerId); } catch { } this._currentActiveButton = -1, this._previousPosition = null, e || i.preventDefault(), this._activePointerId = -1; } else if (n.type === ir.POINTERMOVE && (this._activePointerId === i.pointerId || !s)) { if (t.isPointerLock && this._onMouseMove) this._onMouseMove(n.event); else if (this._previousPosition) { const f = this.camera._calculateHandednessMultiplier(), o = (i.clientX - this._previousPosition.x) * f, d = i.clientY - this._previousPosition.y; this._allowCameraRotation && (this.camera.cameraRotation.y += o / this.angularSensibility, this.camera.cameraRotation.x += d / this.angularSensibility), this.onPointerMovedObservable.notifyObservers({ offsetX: o, offsetY: d }), this._previousPosition = { x: i.clientX, y: i.clientY }, e || i.preventDefault(); } } }), this._onMouseMove = (n) => { if (!t.isPointerLock) return; const i = this.camera._calculateHandednessMultiplier(), s = n.movementX * i; this.camera.cameraRotation.y += s / this.angularSensibility; const a = n.movementY; this.camera.cameraRotation.x += a / this.angularSensibility, this._previousPosition = null, e || n.preventDefault(); }, this._observer = this.camera.getScene()._inputManager._addCameraPointerObserver(this._pointerInput, ir.POINTERDOWN | ir.POINTERUP | ir.POINTERMOVE), r && (this._contextMenuBind = (n) => this.onContextMenu(n), r.addEventListener("contextmenu", this._contextMenuBind, !1)); } /** * Called on JS contextmenu event. * Override this method to provide functionality. * @param evt */ onContextMenu(e) { e.preventDefault(); } /** * Detach the current controls from the specified dom element. */ detachControl() { if (this._observer) { if (this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer), this._contextMenuBind) { const t = this.camera.getEngine().getInputElement(); t && t.removeEventListener("contextmenu", this._contextMenuBind); } this.onPointerMovedObservable && this.onPointerMovedObservable.clear(), this._observer = null, this._onMouseMove = null, this._previousPosition = null; } this._activePointerId = -1, this._currentActiveButton = -1; } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "FreeCameraMouseInput"; } /** * Get the friendly name associated with the input class. * @returns the input friendly name */ getSimpleName() { return "mouse"; } } C([ M() ], OR.prototype, "buttons", void 0); C([ M() ], OR.prototype, "angularSensibility", void 0); VA.FreeCameraMouseInput = OR; var Ja; (function(A) { A[A.MoveRelative = 0] = "MoveRelative", A[A.RotateRelative = 1] = "RotateRelative", A[A.MoveScene = 2] = "MoveScene"; })(Ja || (Ja = {})); class L0 extends IR { constructor() { super(...arguments), this._moveRelative = S.Zero(), this._rotateRelative = S.Zero(), this._moveScene = S.Zero(), this._wheelXAction = Ja.MoveRelative, this._wheelXActionCoordinate = Lx.X, this._wheelYAction = Ja.MoveRelative, this._wheelYActionCoordinate = Lx.Z, this._wheelZAction = null, this._wheelZActionCoordinate = null; } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "FreeCameraMouseWheelInput"; } /** * Set which movement axis (relative to camera's orientation) the mouse * wheel's X axis controls. * @param axis The axis to be moved. Set null to clear. */ set wheelXMoveRelative(e) { e === null && this._wheelXAction !== Ja.MoveRelative || (this._wheelXAction = Ja.MoveRelative, this._wheelXActionCoordinate = e); } /** * Get the configured movement axis (relative to camera's orientation) the * mouse wheel's X axis controls. * @returns The configured axis or null if none. */ get wheelXMoveRelative() { return this._wheelXAction !== Ja.MoveRelative ? null : this._wheelXActionCoordinate; } /** * Set which movement axis (relative to camera's orientation) the mouse * wheel's Y axis controls. * @param axis The axis to be moved. Set null to clear. */ set wheelYMoveRelative(e) { e === null && this._wheelYAction !== Ja.MoveRelative || (this._wheelYAction = Ja.MoveRelative, this._wheelYActionCoordinate = e); } /** * Get the configured movement axis (relative to camera's orientation) the * mouse wheel's Y axis controls. * @returns The configured axis or null if none. */ get wheelYMoveRelative() { return this._wheelYAction !== Ja.MoveRelative ? null : this._wheelYActionCoordinate; } /** * Set which movement axis (relative to camera's orientation) the mouse * wheel's Z axis controls. * @param axis The axis to be moved. Set null to clear. */ set wheelZMoveRelative(e) { e === null && this._wheelZAction !== Ja.MoveRelative || (this._wheelZAction = Ja.MoveRelative, this._wheelZActionCoordinate = e); } /** * Get the configured movement axis (relative to camera's orientation) the * mouse wheel's Z axis controls. * @returns The configured axis or null if none. */ get wheelZMoveRelative() { return this._wheelZAction !== Ja.MoveRelative ? null : this._wheelZActionCoordinate; } /** * Set which rotation axis (relative to camera's orientation) the mouse * wheel's X axis controls. * @param axis The axis to be moved. Set null to clear. */ set wheelXRotateRelative(e) { e === null && this._wheelXAction !== Ja.RotateRelative || (this._wheelXAction = Ja.RotateRelative, this._wheelXActionCoordinate = e); } /** * Get the configured rotation axis (relative to camera's orientation) the * mouse wheel's X axis controls. * @returns The configured axis or null if none. */ get wheelXRotateRelative() { return this._wheelXAction !== Ja.RotateRelative ? null : this._wheelXActionCoordinate; } /** * Set which rotation axis (relative to camera's orientation) the mouse * wheel's Y axis controls. * @param axis The axis to be moved. Set null to clear. */ set wheelYRotateRelative(e) { e === null && this._wheelYAction !== Ja.RotateRelative || (this._wheelYAction = Ja.RotateRelative, this._wheelYActionCoordinate = e); } /** * Get the configured rotation axis (relative to camera's orientation) the * mouse wheel's Y axis controls. * @returns The configured axis or null if none. */ get wheelYRotateRelative() { return this._wheelYAction !== Ja.RotateRelative ? null : this._wheelYActionCoordinate; } /** * Set which rotation axis (relative to camera's orientation) the mouse * wheel's Z axis controls. * @param axis The axis to be moved. Set null to clear. */ set wheelZRotateRelative(e) { e === null && this._wheelZAction !== Ja.RotateRelative || (this._wheelZAction = Ja.RotateRelative, this._wheelZActionCoordinate = e); } /** * Get the configured rotation axis (relative to camera's orientation) the * mouse wheel's Z axis controls. * @returns The configured axis or null if none. */ get wheelZRotateRelative() { return this._wheelZAction !== Ja.RotateRelative ? null : this._wheelZActionCoordinate; } /** * Set which movement axis (relative to the scene) the mouse wheel's X axis * controls. * @param axis The axis to be moved. Set null to clear. */ set wheelXMoveScene(e) { e === null && this._wheelXAction !== Ja.MoveScene || (this._wheelXAction = Ja.MoveScene, this._wheelXActionCoordinate = e); } /** * Get the configured movement axis (relative to the scene) the mouse wheel's * X axis controls. * @returns The configured axis or null if none. */ get wheelXMoveScene() { return this._wheelXAction !== Ja.MoveScene ? null : this._wheelXActionCoordinate; } /** * Set which movement axis (relative to the scene) the mouse wheel's Y axis * controls. * @param axis The axis to be moved. Set null to clear. */ set wheelYMoveScene(e) { e === null && this._wheelYAction !== Ja.MoveScene || (this._wheelYAction = Ja.MoveScene, this._wheelYActionCoordinate = e); } /** * Get the configured movement axis (relative to the scene) the mouse wheel's * Y axis controls. * @returns The configured axis or null if none. */ get wheelYMoveScene() { return this._wheelYAction !== Ja.MoveScene ? null : this._wheelYActionCoordinate; } /** * Set which movement axis (relative to the scene) the mouse wheel's Z axis * controls. * @param axis The axis to be moved. Set null to clear. */ set wheelZMoveScene(e) { e === null && this._wheelZAction !== Ja.MoveScene || (this._wheelZAction = Ja.MoveScene, this._wheelZActionCoordinate = e); } /** * Get the configured movement axis (relative to the scene) the mouse wheel's * Z axis controls. * @returns The configured axis or null if none. */ get wheelZMoveScene() { return this._wheelZAction !== Ja.MoveScene ? null : this._wheelZActionCoordinate; } /** * Called for each rendered frame. */ checkInputs() { if (this._wheelDeltaX === 0 && this._wheelDeltaY === 0 && this._wheelDeltaZ == 0) return; this._moveRelative.setAll(0), this._rotateRelative.setAll(0), this._moveScene.setAll(0), this._updateCamera(), this.camera.getScene().useRightHandedSystem && (this._moveRelative.z *= -1); const e = he.Zero(); this.camera.getViewMatrix().invertToRef(e); const t = S.Zero(); S.TransformNormalToRef(this._moveRelative, e, t), this.camera.cameraRotation.x += this._rotateRelative.x / 200, this.camera.cameraRotation.y += this._rotateRelative.y / 200, this.camera.cameraDirection.addInPlace(t), this.camera.cameraDirection.addInPlace(this._moveScene), super.checkInputs(); } /** * Update the camera according to any configured properties for the 3 * mouse-wheel axis. */ _updateCamera() { this._updateCameraProperty(this._wheelDeltaX, this._wheelXAction, this._wheelXActionCoordinate), this._updateCameraProperty(this._wheelDeltaY, this._wheelYAction, this._wheelYActionCoordinate), this._updateCameraProperty(this._wheelDeltaZ, this._wheelZAction, this._wheelZActionCoordinate); } /** * Update one property of the camera. * @param value * @param cameraProperty * @param coordinate */ _updateCameraProperty(e, t, r) { if (e === 0 || t === null || r === null) return; let n = null; switch (t) { case Ja.MoveRelative: n = this._moveRelative; break; case Ja.RotateRelative: n = this._rotateRelative; break; case Ja.MoveScene: n = this._moveScene; break; } switch (r) { case Lx.X: n.set(e, 0, 0); break; case Lx.Y: n.set(0, e, 0); break; case Lx.Z: n.set(0, 0, e); break; } } } C([ M() ], L0.prototype, "wheelXMoveRelative", null); C([ M() ], L0.prototype, "wheelYMoveRelative", null); C([ M() ], L0.prototype, "wheelZMoveRelative", null); C([ M() ], L0.prototype, "wheelXRotateRelative", null); C([ M() ], L0.prototype, "wheelYRotateRelative", null); C([ M() ], L0.prototype, "wheelZRotateRelative", null); C([ M() ], L0.prototype, "wheelXMoveScene", null); C([ M() ], L0.prototype, "wheelYMoveScene", null); C([ M() ], L0.prototype, "wheelZMoveScene", null); VA.FreeCameraMouseWheelInput = L0; class yR { /** * Manage the touch inputs to control the movement of a free camera. * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs * @param allowMouse Defines if mouse events can be treated as touch events */ constructor(e = !1) { this.allowMouse = e, this.touchAngularSensibility = 2e5, this.touchMoveSensibility = 250, this.singleFingerRotate = !1, this._offsetX = null, this._offsetY = null, this._pointerPressed = new Array(), this._isSafari = ye.IsSafari(); } /** * Attach the input controls to a specific dom element to get the input from. * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachControl(e) { e = ye.BackCompatCameraNoPreventDefault(arguments); let t = null; if (this._pointerInput === void 0 && (this._onLostFocus = () => { this._offsetX = null, this._offsetY = null; }, this._pointerInput = (r) => { const n = r.event, i = n.pointerType === "mouse" || this._isSafari && typeof n.pointerType > "u"; if (!(!this.allowMouse && i)) { if (r.type === ir.POINTERDOWN) { if (e || n.preventDefault(), this._pointerPressed.push(n.pointerId), this._pointerPressed.length !== 1) return; t = { x: n.clientX, y: n.clientY }; } else if (r.type === ir.POINTERUP) { e || n.preventDefault(); const s = this._pointerPressed.indexOf(n.pointerId); if (s === -1 || (this._pointerPressed.splice(s, 1), s != 0)) return; t = null, this._offsetX = null, this._offsetY = null; } else if (r.type === ir.POINTERMOVE) { if (e || n.preventDefault(), !t || this._pointerPressed.indexOf(n.pointerId) != 0) return; this._offsetX = n.clientX - t.x, this._offsetY = -(n.clientY - t.y); } } }), this._observer = this.camera.getScene()._inputManager._addCameraPointerObserver(this._pointerInput, ir.POINTERDOWN | ir.POINTERUP | ir.POINTERMOVE), this._onLostFocus) { const n = this.camera.getEngine().getInputElement(); n && n.addEventListener("blur", this._onLostFocus); } } /** * Detach the current controls from the specified dom element. */ detachControl() { if (this._pointerInput) { if (this._observer && (this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer), this._observer = null), this._onLostFocus) { const t = this.camera.getEngine().getInputElement(); t && t.removeEventListener("blur", this._onLostFocus), this._onLostFocus = null; } this._pointerPressed.length = 0, this._offsetX = null, this._offsetY = null; } } /** * Update the current camera state depending on the inputs that have been used this frame. * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop. */ checkInputs() { if (this._offsetX === null || this._offsetY === null || this._offsetX === 0 && this._offsetY === 0) return; const e = this.camera, t = e._calculateHandednessMultiplier(); if (e.cameraRotation.y = t * this._offsetX / this.touchAngularSensibility, this.singleFingerRotate && this._pointerPressed.length === 1 || !this.singleFingerRotate && this._pointerPressed.length > 1) e.cameraRotation.x = -this._offsetY / this.touchAngularSensibility; else { const n = e._computeLocalCameraSpeed(), i = new S(0, 0, this.touchMoveSensibility !== 0 ? n * this._offsetY / this.touchMoveSensibility : 0); he.RotationYawPitchRollToRef(e.rotation.y, e.rotation.x, 0, e._cameraRotationMatrix), e.cameraDirection.addInPlace(S.TransformCoordinates(i, e._cameraRotationMatrix)); } } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "FreeCameraTouchInput"; } /** * Get the friendly name associated with the input class. * @returns the input friendly name */ getSimpleName() { return "touch"; } } C([ M() ], yR.prototype, "touchAngularSensibility", void 0); C([ M() ], yR.prototype, "touchMoveSensibility", void 0); VA.FreeCameraTouchInput = yR; class kR extends RR { /** * Instantiates a new FreeCameraInputsManager. * @param camera Defines the camera the inputs belong to */ constructor(e) { super(e), this._mouseInput = null, this._mouseWheelInput = null; } /** * Add keyboard input support to the input manager. * @returns the current input manager */ addKeyboard() { return this.add(new wc()), this; } /** * Add mouse input support to the input manager. * @param touchEnabled if the FreeCameraMouseInput should support touch (default: true) * @returns the current input manager */ addMouse(e = !0) { return this._mouseInput || (this._mouseInput = new OR(e), this.add(this._mouseInput)), this; } /** * Removes the mouse input support from the manager * @returns the current input manager */ removeMouse() { return this._mouseInput && this.remove(this._mouseInput), this; } /** * Add mouse wheel input support to the input manager. * @returns the current input manager */ addMouseWheel() { return this._mouseWheelInput || (this._mouseWheelInput = new L0(), this.add(this._mouseWheelInput)), this; } /** * Removes the mouse wheel input support from the manager * @returns the current input manager */ removeMouseWheel() { return this._mouseWheelInput && this.remove(this._mouseWheelInput), this; } /** * Add touch input support to the input manager. * @returns the current input manager */ addTouch() { return this.add(new yR()), this; } /** * Remove all attached input methods from a camera */ clear() { super.clear(), this._mouseInput = null; } } kR.prototype.addDeviceOrientation = function(A) { return this._deviceOrientationInput || (this._deviceOrientationInput = new PQ(), A && (this._deviceOrientationInput.smoothFactor = A), this.add(this._deviceOrientationInput)), this; }; class PQ { /** * Can be used to detect if a device orientation sensor is available on a device * @param timeout amount of time in milliseconds to wait for a response from the sensor (default: infinite) * @returns a promise that will resolve on orientation change */ static WaitForOrientationChangeAsync(e) { return new Promise((t, r) => { let n = !1; const i = () => { window.removeEventListener("deviceorientation", i), n = !0, t(); }; e && setTimeout(() => { n || (window.removeEventListener("deviceorientation", i), r("WaitForOrientationChangeAsync timed out")); }, e), typeof DeviceOrientationEvent < "u" && typeof DeviceOrientationEvent.requestPermission == "function" ? DeviceOrientationEvent.requestPermission().then((s) => { s == "granted" ? window.addEventListener("deviceorientation", i) : ye.Warn("Permission not granted."); }).catch((s) => { ye.Error(s); }) : window.addEventListener("deviceorientation", i); }); } /** * Instantiates a new input * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs */ constructor() { this._screenOrientationAngle = 0, this._screenQuaternion = new Ze(), this._alpha = 0, this._beta = 0, this._gamma = 0, this.smoothFactor = 0, this._onDeviceOrientationChangedObservable = new Oe(), this._orientationChanged = () => { this._screenOrientationAngle = window.orientation !== void 0 ? +window.orientation : window.screen.orientation && window.screen.orientation.angle ? window.screen.orientation.angle : 0, this._screenOrientationAngle = -ye.ToRadians(this._screenOrientationAngle / 2), this._screenQuaternion.copyFromFloats(0, Math.sin(this._screenOrientationAngle), 0, Math.cos(this._screenOrientationAngle)); }, this._deviceOrientation = (e) => { this.smoothFactor ? (this._alpha = e.alpha !== null ? ye.SmoothAngleChange(this._alpha, e.alpha, this.smoothFactor) : 0, this._beta = e.beta !== null ? ye.SmoothAngleChange(this._beta, e.beta, this.smoothFactor) : 0, this._gamma = e.gamma !== null ? ye.SmoothAngleChange(this._gamma, e.gamma, this.smoothFactor) : 0) : (this._alpha = e.alpha !== null ? e.alpha : 0, this._beta = e.beta !== null ? e.beta : 0, this._gamma = e.gamma !== null ? e.gamma : 0), e.alpha !== null && this._onDeviceOrientationChangedObservable.notifyObservers(); }, this._constantTranform = new Ze(-Math.sqrt(0.5), 0, 0, Math.sqrt(0.5)), this._orientationChanged(); } /** * Define the camera controlled by the input. */ get camera() { return this._camera; } set camera(e) { this._camera = e, this._camera != null && !this._camera.rotationQuaternion && (this._camera.rotationQuaternion = new Ze()), this._camera && this._camera.onDisposeObservable.add(() => { this._onDeviceOrientationChangedObservable.clear(); }); } /** * Attach the input controls to a specific dom element to get the input from. */ attachControl() { const e = this.camera.getScene().getEngine().getHostWindow(); if (e) { const t = () => { e.addEventListener("orientationchange", this._orientationChanged), e.addEventListener("deviceorientation", this._deviceOrientation), this._orientationChanged(); }; typeof DeviceOrientationEvent < "u" && typeof DeviceOrientationEvent.requestPermission == "function" ? DeviceOrientationEvent.requestPermission().then((r) => { r === "granted" ? t() : ye.Warn("Permission not granted."); }).catch((r) => { ye.Error(r); }) : t(); } } /** * Detach the current controls from the specified dom element. */ detachControl() { window.removeEventListener("orientationchange", this._orientationChanged), window.removeEventListener("deviceorientation", this._deviceOrientation), this._alpha = 0; } /** * Update the current camera state depending on the inputs that have been used this frame. * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop. */ checkInputs() { this._alpha && (Ze.RotationYawPitchRollToRef(ye.ToRadians(this._alpha), ye.ToRadians(this._beta), -ye.ToRadians(this._gamma), this.camera.rotationQuaternion), this._camera.rotationQuaternion.multiplyInPlace(this._screenQuaternion), this._camera.rotationQuaternion.multiplyInPlace(this._constantTranform), this._camera.rotationQuaternion.z *= -1, this._camera.rotationQuaternion.w *= -1); } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "FreeCameraDeviceOrientationInput"; } /** * Get the friendly name associated with the input class. * @returns the input friendly name */ getSimpleName() { return "deviceOrientation"; } } VA.FreeCameraDeviceOrientationInput = PQ; class ER { constructor() { this.gamepadAngularSensibility = 200, this.gamepadMoveSensibility = 40, this.deadzoneDelta = 0.1, this._yAxisScale = 1, this._cameraTransform = he.Identity(), this._deltaTransform = S.Zero(), this._vector3 = S.Zero(), this._vector2 = at.Zero(); } /** * Gets or sets a boolean indicating that Yaxis (for right stick) should be inverted */ get invertYAxis() { return this._yAxisScale !== 1; } set invertYAxis(e) { this._yAxisScale = e ? -1 : 1; } /** * Attach the input controls to a specific dom element to get the input from. */ attachControl() { const e = this.camera.getScene().gamepadManager; this._onGamepadConnectedObserver = e.onGamepadConnectedObservable.add((t) => { t.type !== N9.POSE_ENABLED && (!this.gamepad || t.type === N9.XBOX) && (this.gamepad = t); }), this._onGamepadDisconnectedObserver = e.onGamepadDisconnectedObservable.add((t) => { this.gamepad === t && (this.gamepad = null); }), this.gamepad = e.getGamepadByType(N9.XBOX), !this.gamepad && e.gamepads.length && (this.gamepad = e.gamepads[0]); } /** * Detach the current controls from the specified dom element. */ detachControl() { this.camera.getScene().gamepadManager.onGamepadConnectedObservable.remove(this._onGamepadConnectedObserver), this.camera.getScene().gamepadManager.onGamepadDisconnectedObservable.remove(this._onGamepadDisconnectedObserver), this.gamepad = null; } /** * Update the current camera state depending on the inputs that have been used this frame. * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop. */ checkInputs() { if (this.gamepad && this.gamepad.leftStick) { const e = this.camera, t = this.gamepad.leftStick; this.gamepadMoveSensibility !== 0 && (t.x = Math.abs(t.x) > this.deadzoneDelta ? t.x / this.gamepadMoveSensibility : 0, t.y = Math.abs(t.y) > this.deadzoneDelta ? t.y / this.gamepadMoveSensibility : 0); let r = this.gamepad.rightStick; r && this.gamepadAngularSensibility !== 0 ? (r.x = Math.abs(r.x) > this.deadzoneDelta ? r.x / this.gamepadAngularSensibility : 0, r.y = (Math.abs(r.y) > this.deadzoneDelta ? r.y / this.gamepadAngularSensibility : 0) * this._yAxisScale) : r = { x: 0, y: 0 }, e.rotationQuaternion ? e.rotationQuaternion.toRotationMatrix(this._cameraTransform) : he.RotationYawPitchRollToRef(e.rotation.y, e.rotation.x, 0, this._cameraTransform); const n = e._computeLocalCameraSpeed() * 50; this._vector3.copyFromFloats(t.x * n, 0, -t.y * n), S.TransformCoordinatesToRef(this._vector3, this._cameraTransform, this._deltaTransform), e.cameraDirection.addInPlace(this._deltaTransform), this._vector2.copyFromFloats(r.y, r.x), e.cameraRotation.addInPlace(this._vector2); } } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "FreeCameraGamepadInput"; } /** * Get the friendly name associated with the input class. * @returns the input friendly name */ getSimpleName() { return "gamepad"; } } C([ M() ], ER.prototype, "gamepadAngularSensibility", void 0); C([ M() ], ER.prototype, "gamepadMoveSensibility", void 0); VA.FreeCameraGamepadInput = ER; var V9; (function(A) { A[A.X = 0] = "X", A[A.Y = 1] = "Y", A[A.Z = 2] = "Z"; })(V9 || (V9 = {})); class $r { static _GetDefaultOptions() { return { puckSize: 40, containerSize: 60, color: "cyan", puckImage: void 0, containerImage: void 0, position: void 0, alwaysVisible: !1, limitToContainer: !1 }; } /** * Creates a new virtual joystick * @param leftJoystick defines that the joystick is for left hand (false by default) * @param customizations Defines the options we want to customize the VirtualJoystick */ constructor(e, t) { this._released = !1; const r = Object.assign(Object.assign({}, $r._GetDefaultOptions()), t); if (e ? this._leftJoystick = !0 : this._leftJoystick = !1, $r._GlobalJoystickIndex++, this._axisTargetedByLeftAndRight = V9.X, this._axisTargetedByUpAndDown = V9.Y, this.reverseLeftRight = !1, this.reverseUpDown = !1, this._touches = new YC(), this.deltaPosition = S.Zero(), this._joystickSensibility = 25, this._inversedSensibility = 1 / (this._joystickSensibility / 1e3), this._onResize = () => { $r._VJCanvasWidth = window.innerWidth, $r._VJCanvasHeight = window.innerHeight, $r.Canvas && ($r.Canvas.width = $r._VJCanvasWidth, $r.Canvas.height = $r._VJCanvasHeight), $r._HalfWidth = $r._VJCanvasWidth / 2; }, !$r.Canvas) { window.addEventListener("resize", this._onResize, !1), $r.Canvas = document.createElement("canvas"), $r._VJCanvasWidth = window.innerWidth, $r._VJCanvasHeight = window.innerHeight, $r.Canvas.width = window.innerWidth, $r.Canvas.height = window.innerHeight, $r.Canvas.style.width = "100%", $r.Canvas.style.height = "100%", $r.Canvas.style.position = "absolute", $r.Canvas.style.backgroundColor = "transparent", $r.Canvas.style.top = "0px", $r.Canvas.style.left = "0px", $r.Canvas.style.zIndex = "5", $r.Canvas.style.touchAction = "none", $r.Canvas.setAttribute("touch-action", "none"); const n = $r.Canvas.getContext("2d"); if (!n) throw new Error("Unable to create canvas for virtual joystick"); $r._VJCanvasContext = n, $r._VJCanvasContext.strokeStyle = "#ffffff", $r._VJCanvasContext.lineWidth = 2, document.body.appendChild($r.Canvas); } $r._HalfWidth = $r.Canvas.width / 2, this.pressed = !1, this.limitToContainer = r.limitToContainer, this._joystickColor = r.color, this.containerSize = r.containerSize, this.puckSize = r.puckSize, r.position && this.setPosition(r.position.x, r.position.y), r.puckImage && this.setPuckImage(r.puckImage), r.containerImage && this.setContainerImage(r.containerImage), r.alwaysVisible && $r._AlwaysVisibleSticks++, this.alwaysVisible = r.alwaysVisible, this._joystickPointerId = -1, this._joystickPointerPos = new at(0, 0), this._joystickPreviousPointerPos = new at(0, 0), this._joystickPointerStartPos = new at(0, 0), this._deltaJoystickVector = new at(0, 0), this._onPointerDownHandlerRef = (n) => { this._onPointerDown(n); }, this._onPointerMoveHandlerRef = (n) => { this._onPointerMove(n); }, this._onPointerUpHandlerRef = (n) => { this._onPointerUp(n); }, $r.Canvas.addEventListener("pointerdown", this._onPointerDownHandlerRef, !1), $r.Canvas.addEventListener("pointermove", this._onPointerMoveHandlerRef, !1), $r.Canvas.addEventListener("pointerup", this._onPointerUpHandlerRef, !1), $r.Canvas.addEventListener("pointerout", this._onPointerUpHandlerRef, !1), $r.Canvas.addEventListener("contextmenu", (n) => { n.preventDefault(); }, !1), requestAnimationFrame(() => { this._drawVirtualJoystick(); }); } /** * Defines joystick sensibility (ie. the ratio between a physical move and virtual joystick position change) * @param newJoystickSensibility defines the new sensibility */ setJoystickSensibility(e) { this._joystickSensibility = e, this._inversedSensibility = 1 / (this._joystickSensibility / 1e3); } _onPointerDown(e) { let t; e.preventDefault(), this._leftJoystick === !0 ? t = e.clientX < $r._HalfWidth : t = e.clientX > $r._HalfWidth, t && this._joystickPointerId < 0 ? (this._joystickPointerId = e.pointerId, this._joystickPosition ? (this._joystickPointerStartPos = this._joystickPosition.clone(), this._joystickPointerPos = this._joystickPosition.clone(), this._joystickPreviousPointerPos = this._joystickPosition.clone(), this._onPointerMove(e)) : (this._joystickPointerStartPos.x = e.clientX, this._joystickPointerStartPos.y = e.clientY, this._joystickPointerPos = this._joystickPointerStartPos.clone(), this._joystickPreviousPointerPos = this._joystickPointerStartPos.clone()), this._deltaJoystickVector.x = 0, this._deltaJoystickVector.y = 0, this.pressed = !0, this._touches.add(e.pointerId.toString(), e)) : $r._GlobalJoystickIndex < 2 && this._action && (this._action(), this._touches.add(e.pointerId.toString(), { x: e.clientX, y: e.clientY, prevX: e.clientX, prevY: e.clientY })); } _onPointerMove(e) { if (this._joystickPointerId == e.pointerId) { if (this.limitToContainer) { const s = new at(e.clientX - this._joystickPointerStartPos.x, e.clientY - this._joystickPointerStartPos.y), a = s.length(); a > this.containerSize && s.scaleInPlace(this.containerSize / a), this._joystickPointerPos.x = this._joystickPointerStartPos.x + s.x, this._joystickPointerPos.y = this._joystickPointerStartPos.y + s.y; } else this._joystickPointerPos.x = e.clientX, this._joystickPointerPos.y = e.clientY; this._deltaJoystickVector = this._joystickPointerPos.clone(), this._deltaJoystickVector = this._deltaJoystickVector.subtract(this._joystickPointerStartPos), 0 < $r._AlwaysVisibleSticks && (this._leftJoystick ? this._joystickPointerPos.x = Math.min($r._HalfWidth, this._joystickPointerPos.x) : this._joystickPointerPos.x = Math.max($r._HalfWidth, this._joystickPointerPos.x)); const r = (this.reverseLeftRight ? -1 : 1) * this._deltaJoystickVector.x / this._inversedSensibility; switch (this._axisTargetedByLeftAndRight) { case V9.X: this.deltaPosition.x = Math.min(1, Math.max(-1, r)); break; case V9.Y: this.deltaPosition.y = Math.min(1, Math.max(-1, r)); break; case V9.Z: this.deltaPosition.z = Math.min(1, Math.max(-1, r)); break; } const i = (this.reverseUpDown ? 1 : -1) * this._deltaJoystickVector.y / this._inversedSensibility; switch (this._axisTargetedByUpAndDown) { case V9.X: this.deltaPosition.x = Math.min(1, Math.max(-1, i)); break; case V9.Y: this.deltaPosition.y = Math.min(1, Math.max(-1, i)); break; case V9.Z: this.deltaPosition.z = Math.min(1, Math.max(-1, i)); break; } } else { const t = this._touches.get(e.pointerId.toString()); t && (t.x = e.clientX, t.y = e.clientY); } } _onPointerUp(e) { if (this._joystickPointerId == e.pointerId) this._clearPreviousDraw(), this._joystickPointerId = -1, this.pressed = !1; else { const t = this._touches.get(e.pointerId.toString()); t && $r._VJCanvasContext.clearRect(t.prevX - 44, t.prevY - 44, 88, 88); } this._deltaJoystickVector.x = 0, this._deltaJoystickVector.y = 0, this._touches.remove(e.pointerId.toString()); } /** * Change the color of the virtual joystick * @param newColor a string that must be a CSS color value (like "red") or the hexa value (like "#FF0000") */ setJoystickColor(e) { this._joystickColor = e; } /** * Size of the joystick's container */ set containerSize(e) { this._joystickContainerSize = e, this._clearContainerSize = ~~(this._joystickContainerSize * 2.1), this._clearContainerSizeOffset = ~~(this._clearContainerSize / 2); } get containerSize() { return this._joystickContainerSize; } /** * Size of the joystick's puck */ set puckSize(e) { this._joystickPuckSize = e, this._clearPuckSize = ~~(this._joystickPuckSize * 2.1), this._clearPuckSizeOffset = ~~(this._clearPuckSize / 2); } get puckSize() { return this._joystickPuckSize; } /** * Clears the set position of the joystick */ clearPosition() { this.alwaysVisible = !1, this._joystickPosition = null; } /** * Defines whether or not the joystick container is always visible */ set alwaysVisible(e) { this._alwaysVisible !== e && (e && this._joystickPosition ? ($r._AlwaysVisibleSticks++, this._alwaysVisible = !0) : ($r._AlwaysVisibleSticks--, this._alwaysVisible = !1)); } get alwaysVisible() { return this._alwaysVisible; } /** * Sets the constant position of the Joystick container * @param x X axis coordinate * @param y Y axis coordinate */ setPosition(e, t) { this._joystickPointerStartPos && this._clearPreviousDraw(), this._joystickPosition = new at(e, t); } /** * Defines a callback to call when the joystick is touched * @param action defines the callback */ setActionOnTouch(e) { this._action = e; } /** * Defines which axis you'd like to control for left & right * @param axis defines the axis to use */ setAxisForLeftRight(e) { switch (e) { case V9.X: case V9.Y: case V9.Z: this._axisTargetedByLeftAndRight = e; break; default: this._axisTargetedByLeftAndRight = V9.X; break; } } /** * Defines which axis you'd like to control for up & down * @param axis defines the axis to use */ setAxisForUpDown(e) { switch (e) { case V9.X: case V9.Y: case V9.Z: this._axisTargetedByUpAndDown = e; break; default: this._axisTargetedByUpAndDown = V9.Y; break; } } /** * Clears the canvas from the previous puck / container draw */ _clearPreviousDraw() { const e = this._joystickPosition || this._joystickPointerStartPos; $r._VJCanvasContext.clearRect(e.x - this._clearContainerSizeOffset, e.y - this._clearContainerSizeOffset, this._clearContainerSize, this._clearContainerSize), $r._VJCanvasContext.clearRect(this._joystickPreviousPointerPos.x - this._clearPuckSizeOffset - 1, this._joystickPreviousPointerPos.y - this._clearPuckSizeOffset - 1, this._clearPuckSize + 2, this._clearPuckSize + 2); } /** * Loads `urlPath` to be used for the container's image * @param urlPath defines the urlPath of an image to use */ setContainerImage(e) { const t = new Image(); t.src = e, t.onload = () => this._containerImage = t; } /** * Loads `urlPath` to be used for the puck's image * @param urlPath defines the urlPath of an image to use */ setPuckImage(e) { const t = new Image(); t.src = e, t.onload = () => this._puckImage = t; } /** * Draws the Virtual Joystick's container */ _drawContainer() { const e = this._joystickPosition || this._joystickPointerStartPos; this._clearPreviousDraw(), this._containerImage ? $r._VJCanvasContext.drawImage(this._containerImage, e.x - this.containerSize, e.y - this.containerSize, this.containerSize * 2, this.containerSize * 2) : ($r._VJCanvasContext.beginPath(), $r._VJCanvasContext.strokeStyle = this._joystickColor, $r._VJCanvasContext.lineWidth = 2, $r._VJCanvasContext.arc(e.x, e.y, this.containerSize, 0, Math.PI * 2, !0), $r._VJCanvasContext.stroke(), $r._VJCanvasContext.closePath(), $r._VJCanvasContext.beginPath(), $r._VJCanvasContext.lineWidth = 6, $r._VJCanvasContext.strokeStyle = this._joystickColor, $r._VJCanvasContext.arc(e.x, e.y, this.puckSize, 0, Math.PI * 2, !0), $r._VJCanvasContext.stroke(), $r._VJCanvasContext.closePath()); } /** * Draws the Virtual Joystick's puck */ _drawPuck() { this._puckImage ? $r._VJCanvasContext.drawImage(this._puckImage, this._joystickPointerPos.x - this.puckSize, this._joystickPointerPos.y - this.puckSize, this.puckSize * 2, this.puckSize * 2) : ($r._VJCanvasContext.beginPath(), $r._VJCanvasContext.strokeStyle = this._joystickColor, $r._VJCanvasContext.lineWidth = 2, $r._VJCanvasContext.arc(this._joystickPointerPos.x, this._joystickPointerPos.y, this.puckSize, 0, Math.PI * 2, !0), $r._VJCanvasContext.stroke(), $r._VJCanvasContext.closePath()); } _drawVirtualJoystick() { this._released || (this.alwaysVisible && this._drawContainer(), this.pressed && this._touches.forEach((e, t) => { t.pointerId === this._joystickPointerId ? (this.alwaysVisible || this._drawContainer(), this._drawPuck(), this._joystickPreviousPointerPos = this._joystickPointerPos.clone()) : ($r._VJCanvasContext.clearRect(t.prevX - 44, t.prevY - 44, 88, 88), $r._VJCanvasContext.beginPath(), $r._VJCanvasContext.fillStyle = "white", $r._VJCanvasContext.beginPath(), $r._VJCanvasContext.strokeStyle = "red", $r._VJCanvasContext.lineWidth = 6, $r._VJCanvasContext.arc(t.x, t.y, 40, 0, Math.PI * 2, !0), $r._VJCanvasContext.stroke(), $r._VJCanvasContext.closePath(), t.prevX = t.x, t.prevY = t.y); }), requestAnimationFrame(() => { this._drawVirtualJoystick(); })); } /** * Release internal HTML canvas */ releaseCanvas() { $r.Canvas && ($r.Canvas.removeEventListener("pointerdown", this._onPointerDownHandlerRef), $r.Canvas.removeEventListener("pointermove", this._onPointerMoveHandlerRef), $r.Canvas.removeEventListener("pointerup", this._onPointerUpHandlerRef), $r.Canvas.removeEventListener("pointerout", this._onPointerUpHandlerRef), window.removeEventListener("resize", this._onResize), document.body.removeChild($r.Canvas), $r.Canvas = null), this._released = !0; } } $r._GlobalJoystickIndex = 0; $r._AlwaysVisibleSticks = 0; kR.prototype.addVirtualJoystick = function() { return this.add(new cQ()), this; }; class cQ { /** * Gets the left stick of the virtual joystick. * @returns The virtual Joystick */ getLeftJoystick() { return this._leftjoystick; } /** * Gets the right stick of the virtual joystick. * @returns The virtual Joystick */ getRightJoystick() { return this._rightjoystick; } /** * Update the current camera state depending on the inputs that have been used this frame. * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop. */ checkInputs() { if (this._leftjoystick) { const e = this.camera, t = e._computeLocalCameraSpeed() * 50, r = he.RotationYawPitchRoll(e.rotation.y, e.rotation.x, 0), n = S.TransformCoordinates(new S(this._leftjoystick.deltaPosition.x * t, this._leftjoystick.deltaPosition.y * t, this._leftjoystick.deltaPosition.z * t), r); e.cameraDirection = e.cameraDirection.add(n), e.cameraRotation = e.cameraRotation.addVector3(this._rightjoystick.deltaPosition), this._leftjoystick.pressed || (this._leftjoystick.deltaPosition = this._leftjoystick.deltaPosition.scale(0.9)), this._rightjoystick.pressed || (this._rightjoystick.deltaPosition = this._rightjoystick.deltaPosition.scale(0.9)); } } /** * Attach the input controls to a specific dom element to get the input from. */ attachControl() { this._leftjoystick = new $r(!0), this._leftjoystick.setAxisForUpDown(V9.Z), this._leftjoystick.setAxisForLeftRight(V9.X), this._leftjoystick.setJoystickSensibility(0.15), this._rightjoystick = new $r(!1), this._rightjoystick.setAxisForUpDown(V9.X), this._rightjoystick.setAxisForLeftRight(V9.Y), this._rightjoystick.reverseUpDown = !0, this._rightjoystick.setJoystickSensibility(0.05), this._rightjoystick.setJoystickColor("yellow"); } /** * Detach the current controls from the specified dom element. */ detachControl() { this._leftjoystick.releaseCanvas(), this._rightjoystick.releaseCanvas(); } /** * Gets the class name of the current input. * @returns the class name */ getClassName() { return "FreeCameraVirtualJoystickInput"; } /** * Get the friendly name associated with the input class. * @returns the input friendly name */ getSimpleName() { return "virtualJoystick"; } } VA.FreeCameraVirtualJoystickInput = cQ; class b1 extends Tr { /** * Instantiates a target camera that takes a mesh or position as a target and continues to look at it while it moves. * This is the base of the follow, arc rotate cameras and Free camera * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras * @param name Defines the name of the camera in the scene * @param position Defines the start position of the camera in the scene * @param scene Defines the scene the camera belongs to * @param setActiveOnSceneIfNoneActive Defines whether the camera should be marked as active if not other active cameras have been defined */ constructor(e, t, r, n = !0) { super(e, t, r, n), this._tmpUpVector = S.Zero(), this._tmpTargetVector = S.Zero(), this.cameraDirection = new S(0, 0, 0), this.cameraRotation = new at(0, 0), this.ignoreParentScaling = !1, this.updateUpVectorFromRotation = !1, this._tmpQuaternion = new Ze(), this.rotation = new S(0, 0, 0), this.speed = 2, this.noRotationConstraint = !1, this.invertRotation = !1, this.inverseRotationSpeed = 0.2, this.lockedTarget = null, this._currentTarget = S.Zero(), this._initialFocalDistance = 1, this._viewMatrix = he.Zero(), this._camMatrix = he.Zero(), this._cameraTransformMatrix = he.Zero(), this._cameraRotationMatrix = he.Zero(), this._referencePoint = new S(0, 0, 1), this._transformedReferencePoint = S.Zero(), this._deferredPositionUpdate = new S(), this._deferredRotationQuaternionUpdate = new Ze(), this._deferredRotationUpdate = new S(), this._deferredUpdated = !1, this._deferOnly = !1, this._defaultUp = S.Up(), this._cachedRotationZ = 0, this._cachedQuaternionRotationZ = 0; } /** * Gets the position in front of the camera at a given distance. * @param distance The distance from the camera we want the position to be * @returns the position */ getFrontPosition(e) { this.getWorldMatrix(); const t = this.getTarget().subtract(this.position); return t.normalize(), t.scaleInPlace(e), this.globalPosition.add(t); } /** @internal */ _getLockedTargetPosition() { if (!this.lockedTarget) return null; if (this.lockedTarget.absolutePosition) { const e = this.lockedTarget; e.computeWorldMatrix().getTranslationToRef(e.absolutePosition); } return this.lockedTarget.absolutePosition || this.lockedTarget; } /** * Store current camera state of the camera (fov, position, rotation, etc..) * @returns the camera */ storeState() { return this._storedPosition = this.position.clone(), this._storedRotation = this.rotation.clone(), this.rotationQuaternion && (this._storedRotationQuaternion = this.rotationQuaternion.clone()), super.storeState(); } /** * Restored camera state. You must call storeState() first * @returns whether it was successful or not * @internal */ _restoreStateValues() { return super._restoreStateValues() ? (this.position = this._storedPosition.clone(), this.rotation = this._storedRotation.clone(), this.rotationQuaternion && (this.rotationQuaternion = this._storedRotationQuaternion.clone()), this.cameraDirection.copyFromFloats(0, 0, 0), this.cameraRotation.copyFromFloats(0, 0), !0) : !1; } /** @internal */ _initCache() { super._initCache(), this._cache.lockedTarget = new S(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE), this._cache.rotation = new S(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE), this._cache.rotationQuaternion = new Ze(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE); } /** * @internal */ _updateCache(e) { e || super._updateCache(); const t = this._getLockedTargetPosition(); t ? this._cache.lockedTarget ? this._cache.lockedTarget.copyFrom(t) : this._cache.lockedTarget = t.clone() : this._cache.lockedTarget = null, this._cache.rotation.copyFrom(this.rotation), this.rotationQuaternion && this._cache.rotationQuaternion.copyFrom(this.rotationQuaternion); } // Synchronized /** @internal */ _isSynchronizedViewMatrix() { if (!super._isSynchronizedViewMatrix()) return !1; const e = this._getLockedTargetPosition(); return (this._cache.lockedTarget ? this._cache.lockedTarget.equals(e) : !e) && (this.rotationQuaternion ? this.rotationQuaternion.equals(this._cache.rotationQuaternion) : this._cache.rotation.equals(this.rotation)); } // Methods /** @internal */ _computeLocalCameraSpeed() { const e = this.getEngine(); return this.speed * Math.sqrt(e.getDeltaTime() / (e.getFps() * 100)); } // Target /** * Defines the target the camera should look at. * @param target Defines the new target as a Vector */ setTarget(e) { this.upVector.normalize(), this._initialFocalDistance = e.subtract(this.position).length(), this.position.z === e.z && (this.position.z += Dn), this._referencePoint.normalize().scaleInPlace(this._initialFocalDistance), he.LookAtLHToRef(this.position, e, this._defaultUp, this._camMatrix), this._camMatrix.invert(), this.rotation.x = Math.atan(this._camMatrix.m[6] / this._camMatrix.m[10]); const t = e.subtract(this.position); t.x >= 0 ? this.rotation.y = -Math.atan(t.z / t.x) + Math.PI / 2 : this.rotation.y = -Math.atan(t.z / t.x) - Math.PI / 2, this.rotation.z = 0, isNaN(this.rotation.x) && (this.rotation.x = 0), isNaN(this.rotation.y) && (this.rotation.y = 0), isNaN(this.rotation.z) && (this.rotation.z = 0), this.rotationQuaternion && Ze.RotationYawPitchRollToRef(this.rotation.y, this.rotation.x, this.rotation.z, this.rotationQuaternion); } /** * Defines the target point of the camera. * The camera looks towards it form the radius distance. */ get target() { return this.getTarget(); } set target(e) { this.setTarget(e); } /** * Return the current target position of the camera. This value is expressed in local space. * @returns the target position */ getTarget() { return this._currentTarget; } /** @internal */ _decideIfNeedsToMove() { return Math.abs(this.cameraDirection.x) > 0 || Math.abs(this.cameraDirection.y) > 0 || Math.abs(this.cameraDirection.z) > 0; } /** @internal */ _updatePosition() { if (this.parent) { this.parent.getWorldMatrix().invertToRef(ue.Matrix[0]), S.TransformNormalToRef(this.cameraDirection, ue.Matrix[0], ue.Vector3[0]), this._deferredPositionUpdate.addInPlace(ue.Vector3[0]), this._deferOnly ? this._deferredUpdated = !0 : this.position.copyFrom(this._deferredPositionUpdate); return; } this._deferredPositionUpdate.addInPlace(this.cameraDirection), this._deferOnly ? this._deferredUpdated = !0 : this.position.copyFrom(this._deferredPositionUpdate); } /** @internal */ _checkInputs() { const e = this.invertRotation ? -this.inverseRotationSpeed : 1, t = this._decideIfNeedsToMove(), r = this.cameraRotation.x || this.cameraRotation.y; this._deferredUpdated = !1, this._deferredRotationUpdate.copyFrom(this.rotation), this._deferredPositionUpdate.copyFrom(this.position), this.rotationQuaternion && this._deferredRotationQuaternionUpdate.copyFrom(this.rotationQuaternion), t && this._updatePosition(), r && (this.rotationQuaternion && this.rotationQuaternion.toEulerAnglesToRef(this._deferredRotationUpdate), this._deferredRotationUpdate.x += this.cameraRotation.x * e, this._deferredRotationUpdate.y += this.cameraRotation.y * e, this.noRotationConstraint || (this._deferredRotationUpdate.x > 1.570796 && (this._deferredRotationUpdate.x = 1.570796), this._deferredRotationUpdate.x < -1.570796 && (this._deferredRotationUpdate.x = -1.570796)), this._deferOnly ? this._deferredUpdated = !0 : this.rotation.copyFrom(this._deferredRotationUpdate), this.rotationQuaternion && this._deferredRotationUpdate.lengthSquared() && (Ze.RotationYawPitchRollToRef(this._deferredRotationUpdate.y, this._deferredRotationUpdate.x, this._deferredRotationUpdate.z, this._deferredRotationQuaternionUpdate), this._deferOnly ? this._deferredUpdated = !0 : this.rotationQuaternion.copyFrom(this._deferredRotationQuaternionUpdate))), t && (Math.abs(this.cameraDirection.x) < this.speed * Dn && (this.cameraDirection.x = 0), Math.abs(this.cameraDirection.y) < this.speed * Dn && (this.cameraDirection.y = 0), Math.abs(this.cameraDirection.z) < this.speed * Dn && (this.cameraDirection.z = 0), this.cameraDirection.scaleInPlace(this.inertia)), r && (Math.abs(this.cameraRotation.x) < this.speed * Dn && (this.cameraRotation.x = 0), Math.abs(this.cameraRotation.y) < this.speed * Dn && (this.cameraRotation.y = 0), this.cameraRotation.scaleInPlace(this.inertia)), super._checkInputs(); } _updateCameraRotationMatrix() { this.rotationQuaternion ? this.rotationQuaternion.toRotationMatrix(this._cameraRotationMatrix) : he.RotationYawPitchRollToRef(this.rotation.y, this.rotation.x, this.rotation.z, this._cameraRotationMatrix); } /** * Update the up vector to apply the rotation of the camera (So if you changed the camera rotation.z this will let you update the up vector as well) * @returns the current camera */ _rotateUpVectorWithCameraRotationMatrix() { return S.TransformNormalToRef(this._defaultUp, this._cameraRotationMatrix, this.upVector), this; } /** @internal */ _getViewMatrix() { return this.lockedTarget && this.setTarget(this._getLockedTargetPosition()), this._updateCameraRotationMatrix(), this.rotationQuaternion && this._cachedQuaternionRotationZ != this.rotationQuaternion.z ? (this._rotateUpVectorWithCameraRotationMatrix(), this._cachedQuaternionRotationZ = this.rotationQuaternion.z) : this._cachedRotationZ !== this.rotation.z && (this._rotateUpVectorWithCameraRotationMatrix(), this._cachedRotationZ = this.rotation.z), S.TransformCoordinatesToRef(this._referencePoint, this._cameraRotationMatrix, this._transformedReferencePoint), this.position.addToRef(this._transformedReferencePoint, this._currentTarget), this.updateUpVectorFromRotation && (this.rotationQuaternion ? bf.Y.rotateByQuaternionToRef(this.rotationQuaternion, this.upVector) : (Ze.FromEulerVectorToRef(this.rotation, this._tmpQuaternion), bf.Y.rotateByQuaternionToRef(this._tmpQuaternion, this.upVector))), this._computeViewMatrix(this.position, this._currentTarget, this.upVector), this._viewMatrix; } _computeViewMatrix(e, t, r) { if (this.ignoreParentScaling) { if (this.parent) { const n = this.parent.getWorldMatrix(); S.TransformCoordinatesToRef(e, n, this._globalPosition), S.TransformCoordinatesToRef(t, n, this._tmpTargetVector), S.TransformNormalToRef(r, n, this._tmpUpVector), this._markSyncedWithParent(); } else this._globalPosition.copyFrom(e), this._tmpTargetVector.copyFrom(t), this._tmpUpVector.copyFrom(r); this.getScene().useRightHandedSystem ? he.LookAtRHToRef(this._globalPosition, this._tmpTargetVector, this._tmpUpVector, this._viewMatrix) : he.LookAtLHToRef(this._globalPosition, this._tmpTargetVector, this._tmpUpVector, this._viewMatrix); return; } if (this.getScene().useRightHandedSystem ? he.LookAtRHToRef(e, t, r, this._viewMatrix) : he.LookAtLHToRef(e, t, r, this._viewMatrix), this.parent) { const n = this.parent.getWorldMatrix(); this._viewMatrix.invert(), this._viewMatrix.multiplyToRef(n, this._viewMatrix), this._viewMatrix.getTranslationToRef(this._globalPosition), this._viewMatrix.invert(), this._markSyncedWithParent(); } else this._globalPosition.copyFrom(e); } /** * @internal */ // eslint-disable-next-line @typescript-eslint/no-unused-vars createRigCamera(e, t) { if (this.cameraRigMode !== Tr.RIG_MODE_NONE) { const r = new b1(e, this.position.clone(), this.getScene()); return r.isRigCamera = !0, r.rigParent = this, this.cameraRigMode === Tr.RIG_MODE_VR && (this.rotationQuaternion || (this.rotationQuaternion = new Ze()), r._cameraRigParams = {}, r.rotationQuaternion = new Ze()), r.mode = this.mode, r.orthoLeft = this.orthoLeft, r.orthoRight = this.orthoRight, r.orthoTop = this.orthoTop, r.orthoBottom = this.orthoBottom, r; } return null; } /** * @internal */ _updateRigCameras() { const e = this._rigCameras[0], t = this._rigCameras[1]; switch (this.computeWorldMatrix(), this.cameraRigMode) { case Tr.RIG_MODE_STEREOSCOPIC_ANAGLYPH: case Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL: case Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED: case Tr.RIG_MODE_STEREOSCOPIC_OVERUNDER: case Tr.RIG_MODE_STEREOSCOPIC_INTERLACED: { const r = this.cameraRigMode === Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED ? 1 : -1, n = this.cameraRigMode === Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED ? -1 : 1; this._getRigCamPositionAndTarget(this._cameraRigParams.stereoHalfAngle * r, e), this._getRigCamPositionAndTarget(this._cameraRigParams.stereoHalfAngle * n, t); break; } case Tr.RIG_MODE_VR: e.rotationQuaternion ? (e.rotationQuaternion.copyFrom(this.rotationQuaternion), t.rotationQuaternion.copyFrom(this.rotationQuaternion)) : (e.rotation.copyFrom(this.rotation), t.rotation.copyFrom(this.rotation)), e.position.copyFrom(this.position), t.position.copyFrom(this.position); break; } super._updateRigCameras(); } _getRigCamPositionAndTarget(e, t) { this.getTarget().subtractToRef(this.position, b1._TargetFocalPoint), b1._TargetFocalPoint.normalize().scaleInPlace(this._initialFocalDistance); const n = b1._TargetFocalPoint.addInPlace(this.position); he.TranslationToRef(-n.x, -n.y, -n.z, b1._TargetTransformMatrix), b1._TargetTransformMatrix.multiplyToRef(he.RotationAxis(t.upVector, e), b1._RigCamTransformMatrix), he.TranslationToRef(n.x, n.y, n.z, b1._TargetTransformMatrix), b1._RigCamTransformMatrix.multiplyToRef(b1._TargetTransformMatrix, b1._RigCamTransformMatrix), S.TransformCoordinatesToRef(this.position, b1._RigCamTransformMatrix, t.position), t.setTarget(n); } /** * Gets the current object class name. * @returns the class name */ getClassName() { return "TargetCamera"; } } b1._RigCamTransformMatrix = new he(); b1._TargetTransformMatrix = new he(); b1._TargetFocalPoint = new S(); C([ fo() ], b1.prototype, "rotation", void 0); C([ M() ], b1.prototype, "speed", void 0); C([ tU("lockedTargetId") ], b1.prototype, "lockedTarget", void 0); class SA extends b1 { /** * Gets the input sensibility for a mouse input. (default is 2000.0) * Higher values reduce sensitivity. */ get angularSensibility() { const e = this.inputs.attached.mouse; return e ? e.angularSensibility : 0; } /** * Sets the input sensibility for a mouse input. (default is 2000.0) * Higher values reduce sensitivity. */ set angularSensibility(e) { const t = this.inputs.attached.mouse; t && (t.angularSensibility = e); } /** * Gets or Set the list of keyboard keys used to control the forward move of the camera. */ get keysUp() { const e = this.inputs.attached.keyboard; return e ? e.keysUp : []; } set keysUp(e) { const t = this.inputs.attached.keyboard; t && (t.keysUp = e); } /** * Gets or Set the list of keyboard keys used to control the upward move of the camera. */ get keysUpward() { const e = this.inputs.attached.keyboard; return e ? e.keysUpward : []; } set keysUpward(e) { const t = this.inputs.attached.keyboard; t && (t.keysUpward = e); } /** * Gets or Set the list of keyboard keys used to control the backward move of the camera. */ get keysDown() { const e = this.inputs.attached.keyboard; return e ? e.keysDown : []; } set keysDown(e) { const t = this.inputs.attached.keyboard; t && (t.keysDown = e); } /** * Gets or Set the list of keyboard keys used to control the downward move of the camera. */ get keysDownward() { const e = this.inputs.attached.keyboard; return e ? e.keysDownward : []; } set keysDownward(e) { const t = this.inputs.attached.keyboard; t && (t.keysDownward = e); } /** * Gets or Set the list of keyboard keys used to control the left strafe move of the camera. */ get keysLeft() { const e = this.inputs.attached.keyboard; return e ? e.keysLeft : []; } set keysLeft(e) { const t = this.inputs.attached.keyboard; t && (t.keysLeft = e); } /** * Gets or Set the list of keyboard keys used to control the right strafe move of the camera. */ get keysRight() { const e = this.inputs.attached.keyboard; return e ? e.keysRight : []; } set keysRight(e) { const t = this.inputs.attached.keyboard; t && (t.keysRight = e); } /** * Gets or Set the list of keyboard keys used to control the left rotation move of the camera. */ get keysRotateLeft() { const e = this.inputs.attached.keyboard; return e ? e.keysRotateLeft : []; } set keysRotateLeft(e) { const t = this.inputs.attached.keyboard; t && (t.keysRotateLeft = e); } /** * Gets or Set the list of keyboard keys used to control the right rotation move of the camera. */ get keysRotateRight() { const e = this.inputs.attached.keyboard; return e ? e.keysRotateRight : []; } set keysRotateRight(e) { const t = this.inputs.attached.keyboard; t && (t.keysRotateRight = e); } /** * Gets or Set the list of keyboard keys used to control the up rotation move of the camera. */ get keysRotateUp() { const e = this.inputs.attached.keyboard; return e ? e.keysRotateUp : []; } set keysRotateUp(e) { const t = this.inputs.attached.keyboard; t && (t.keysRotateUp = e); } /** * Gets or Set the list of keyboard keys used to control the down rotation move of the camera. */ get keysRotateDown() { const e = this.inputs.attached.keyboard; return e ? e.keysRotateDown : []; } set keysRotateDown(e) { const t = this.inputs.attached.keyboard; t && (t.keysRotateDown = e); } /** * Instantiates a Free Camera. * This represents a free type of camera. It can be useful in First Person Shooter game for instance. * Please consider using the new UniversalCamera instead as it adds more functionality like touch to this camera. * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera * @param name Define the name of the camera in the scene * @param position Define the start position of the camera in the scene * @param scene Define the scene the camera belongs to * @param setActiveOnSceneIfNoneActive Defines whether the camera should be marked as active if not other active cameras have been defined */ constructor(e, t, r, n = !0) { super(e, t, r, n), this.ellipsoid = new S(0.5, 1, 0.5), this.ellipsoidOffset = new S(0, 0, 0), this.checkCollisions = !1, this.applyGravity = !1, this._needMoveForGravity = !1, this._oldPosition = S.Zero(), this._diffPosition = S.Zero(), this._newPosition = S.Zero(), this._collisionMask = -1, this._onCollisionPositionChange = (i, s, a = null) => { this._newPosition.copyFrom(s), this._newPosition.subtractToRef(this._oldPosition, this._diffPosition), this._diffPosition.length() > Ge.CollisionsEpsilon && (this.position.addToRef(this._diffPosition, this._deferredPositionUpdate), this._deferOnly ? this._deferredUpdated = !0 : this.position.copyFrom(this._deferredPositionUpdate), this.onCollide && a && this.onCollide(a)); }, this.inputs = new kR(this), this.inputs.addKeyboard().addMouse(); } /** * Attached controls to the current camera. * @param ignored defines an ignored parameter kept for backward compatibility. * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachControl(e, t) { t = ye.BackCompatCameraNoPreventDefault(arguments), this.inputs.attachElement(t); } /** * Detach the current controls from the specified dom element. */ detachControl() { this.inputs.detachElement(), this.cameraDirection = new S(0, 0, 0), this.cameraRotation = new at(0, 0); } /** * Define a collision mask to limit the list of object the camera can collide with */ get collisionMask() { return this._collisionMask; } set collisionMask(e) { this._collisionMask = isNaN(e) ? -1 : e; } /** * @internal */ _collideWithWorld(e) { let t; this.parent ? t = S.TransformCoordinates(this.position, this.parent.getWorldMatrix()) : t = this.position, t.subtractFromFloatsToRef(0, this.ellipsoid.y, 0, this._oldPosition), this._oldPosition.addInPlace(this.ellipsoidOffset); const r = this.getScene().collisionCoordinator; this._collider || (this._collider = r.createCollider()), this._collider._radius = this.ellipsoid, this._collider.collisionMask = this._collisionMask; let n = e; this.applyGravity && (n = e.add(this.getScene().gravity)), r.getNewPosition(this._oldPosition, n, this._collider, 3, null, this._onCollisionPositionChange, this.uniqueId); } /** @internal */ _checkInputs() { this._localDirection || (this._localDirection = S.Zero(), this._transformedDirection = S.Zero()), this.inputs.checkInputs(), super._checkInputs(); } /** * Enable movement without a user input. This allows gravity to always be applied. */ set needMoveForGravity(e) { this._needMoveForGravity = e; } /** * When true, gravity is applied whether there is user input or not. */ get needMoveForGravity() { return this._needMoveForGravity; } /** @internal */ _decideIfNeedsToMove() { return this._needMoveForGravity || Math.abs(this.cameraDirection.x) > 0 || Math.abs(this.cameraDirection.y) > 0 || Math.abs(this.cameraDirection.z) > 0; } /** @internal */ _updatePosition() { this.checkCollisions && this.getScene().collisionsEnabled ? this._collideWithWorld(this.cameraDirection) : super._updatePosition(); } /** * Destroy the camera and release the current resources hold by it. */ dispose() { this.inputs.clear(), super.dispose(); } /** * Gets the current object class name. * @returns the class name */ getClassName() { return "FreeCamera"; } } C([ fo() ], SA.prototype, "ellipsoid", void 0); C([ fo() ], SA.prototype, "ellipsoidOffset", void 0); C([ M() ], SA.prototype, "checkCollisions", void 0); C([ M() ], SA.prototype, "applyGravity", void 0); Cs.AddNodeConstructor("TouchCamera", (A, e) => () => new pQ(A, S.Zero(), e)); class pQ extends SA { /** * Defines the touch sensibility for rotation. * The higher the faster. */ get touchAngularSensibility() { const e = this.inputs.attached.touch; return e ? e.touchAngularSensibility : 0; } set touchAngularSensibility(e) { const t = this.inputs.attached.touch; t && (t.touchAngularSensibility = e); } /** * Defines the touch sensibility for move. * The higher the faster. */ get touchMoveSensibility() { const e = this.inputs.attached.touch; return e ? e.touchMoveSensibility : 0; } set touchMoveSensibility(e) { const t = this.inputs.attached.touch; t && (t.touchMoveSensibility = e); } /** * Instantiates a new touch camera. * This represents a FPS type of camera controlled by touch. * This is like a universal camera minus the Gamepad controls. * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera * @param name Define the name of the camera in the scene * @param position Define the start position of the camera in the scene * @param scene Define the scene the camera belongs to */ constructor(e, t, r) { super(e, t, r), this.inputs.addTouch(), this._setupInputs(); } /** * Gets the current object class name. * @returns the class name */ getClassName() { return "TouchCamera"; } /** @internal */ _setupInputs() { const e = this.inputs.attached.touch, t = this.inputs.attached.mouse; t ? t.touchEnabled = !1 : e.allowMouse = !0; } } Cs.AddNodeConstructor("ArcRotateCamera", (A, e) => () => new ps(A, 0, 0, 1, S.Zero(), e)); class ps extends b1 { /** * Defines the target point of the camera. * The camera looks towards it from the radius distance. */ get target() { return this._target; } set target(e) { this.setTarget(e); } /** * Defines the target mesh of the camera. * The camera looks towards it from the radius distance. * Please note that setting a target host will disable panning. */ get targetHost() { return this._targetHost; } set targetHost(e) { e && this.setTarget(e); } /** * Return the current target position of the camera. This value is expressed in local space. * @returns the target position */ getTarget() { return this.target; } /** * Define the current local position of the camera in the scene */ get position() { return this._position; } set position(e) { this.setPosition(e); } /** * The vector the camera should consider as up. (default is Vector3(0, 1, 0) as returned by Vector3.Up()) * Setting this will copy the given vector to the camera's upVector, and set rotation matrices to and from Y up. * DO NOT set the up vector using copyFrom or copyFromFloats, as this bypasses setting the above matrices. */ set upVector(e) { this._upToYMatrix || (this._yToUpMatrix = new he(), this._upToYMatrix = new he(), this._upVector = S.Zero()), e.normalize(), this._upVector.copyFrom(e), this.setMatUp(); } get upVector() { return this._upVector; } /** * Sets the Y-up to camera up-vector rotation matrix, and the up-vector to Y-up rotation matrix. */ setMatUp() { he.RotationAlignToRef(S.UpReadOnly, this._upVector, this._yToUpMatrix), he.RotationAlignToRef(this._upVector, S.UpReadOnly, this._upToYMatrix); } //-- begin properties for backward compatibility for inputs /** * Gets or Set the pointer angular sensibility along the X axis or how fast is the camera rotating. */ get angularSensibilityX() { const e = this.inputs.attached.pointers; return e ? e.angularSensibilityX : 0; } set angularSensibilityX(e) { const t = this.inputs.attached.pointers; t && (t.angularSensibilityX = e); } /** * Gets or Set the pointer angular sensibility along the Y axis or how fast is the camera rotating. */ get angularSensibilityY() { const e = this.inputs.attached.pointers; return e ? e.angularSensibilityY : 0; } set angularSensibilityY(e) { const t = this.inputs.attached.pointers; t && (t.angularSensibilityY = e); } /** * Gets or Set the pointer pinch precision or how fast is the camera zooming. */ get pinchPrecision() { const e = this.inputs.attached.pointers; return e ? e.pinchPrecision : 0; } set pinchPrecision(e) { const t = this.inputs.attached.pointers; t && (t.pinchPrecision = e); } /** * Gets or Set the pointer pinch delta percentage or how fast is the camera zooming. * It will be used instead of pinchDeltaPrecision if different from 0. * It defines the percentage of current camera.radius to use as delta when pinch zoom is used. */ get pinchDeltaPercentage() { const e = this.inputs.attached.pointers; return e ? e.pinchDeltaPercentage : 0; } set pinchDeltaPercentage(e) { const t = this.inputs.attached.pointers; t && (t.pinchDeltaPercentage = e); } /** * Gets or Set the pointer use natural pinch zoom to override the pinch precision * and pinch delta percentage. * When useNaturalPinchZoom is true, multi touch zoom will zoom in such * that any object in the plane at the camera's target point will scale * perfectly with finger motion. */ get useNaturalPinchZoom() { const e = this.inputs.attached.pointers; return e ? e.useNaturalPinchZoom : !1; } set useNaturalPinchZoom(e) { const t = this.inputs.attached.pointers; t && (t.useNaturalPinchZoom = e); } /** * Gets or Set the pointer panning sensibility or how fast is the camera moving. */ get panningSensibility() { const e = this.inputs.attached.pointers; return e ? e.panningSensibility : 0; } set panningSensibility(e) { const t = this.inputs.attached.pointers; t && (t.panningSensibility = e); } /** * Gets or Set the list of keyboard keys used to control beta angle in a positive direction. */ get keysUp() { const e = this.inputs.attached.keyboard; return e ? e.keysUp : []; } set keysUp(e) { const t = this.inputs.attached.keyboard; t && (t.keysUp = e); } /** * Gets or Set the list of keyboard keys used to control beta angle in a negative direction. */ get keysDown() { const e = this.inputs.attached.keyboard; return e ? e.keysDown : []; } set keysDown(e) { const t = this.inputs.attached.keyboard; t && (t.keysDown = e); } /** * Gets or Set the list of keyboard keys used to control alpha angle in a negative direction. */ get keysLeft() { const e = this.inputs.attached.keyboard; return e ? e.keysLeft : []; } set keysLeft(e) { const t = this.inputs.attached.keyboard; t && (t.keysLeft = e); } /** * Gets or Set the list of keyboard keys used to control alpha angle in a positive direction. */ get keysRight() { const e = this.inputs.attached.keyboard; return e ? e.keysRight : []; } set keysRight(e) { const t = this.inputs.attached.keyboard; t && (t.keysRight = e); } /** * Gets or Set the mouse wheel precision or how fast is the camera zooming. */ get wheelPrecision() { const e = this.inputs.attached.mousewheel; return e ? e.wheelPrecision : 0; } set wheelPrecision(e) { const t = this.inputs.attached.mousewheel; t && (t.wheelPrecision = e); } /** * Gets or Set the boolean value that controls whether or not the mouse wheel * zooms to the location of the mouse pointer or not. The default is false. */ get zoomToMouseLocation() { const e = this.inputs.attached.mousewheel; return e ? e.zoomToMouseLocation : !1; } set zoomToMouseLocation(e) { const t = this.inputs.attached.mousewheel; t && (t.zoomToMouseLocation = e); } /** * Gets or Set the mouse wheel delta percentage or how fast is the camera zooming. * It will be used instead of pinchDeltaPrecision if different from 0. * It defines the percentage of current camera.radius to use as delta when pinch zoom is used. */ get wheelDeltaPercentage() { const e = this.inputs.attached.mousewheel; return e ? e.wheelDeltaPercentage : 0; } set wheelDeltaPercentage(e) { const t = this.inputs.attached.mousewheel; t && (t.wheelDeltaPercentage = e); } /** * Gets the bouncing behavior of the camera if it has been enabled. * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#bouncing-behavior */ get bouncingBehavior() { return this._bouncingBehavior; } /** * Defines if the bouncing behavior of the camera is enabled on the camera. * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#bouncing-behavior */ get useBouncingBehavior() { return this._bouncingBehavior != null; } set useBouncingBehavior(e) { e !== this.useBouncingBehavior && (e ? (this._bouncingBehavior = new fD(), this.addBehavior(this._bouncingBehavior)) : this._bouncingBehavior && (this.removeBehavior(this._bouncingBehavior), this._bouncingBehavior = null)); } /** * Gets the framing behavior of the camera if it has been enabled. * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#framing-behavior */ get framingBehavior() { return this._framingBehavior; } /** * Defines if the framing behavior of the camera is enabled on the camera. * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#framing-behavior */ get useFramingBehavior() { return this._framingBehavior != null; } set useFramingBehavior(e) { e !== this.useFramingBehavior && (e ? (this._framingBehavior = new Vu(), this.addBehavior(this._framingBehavior)) : this._framingBehavior && (this.removeBehavior(this._framingBehavior), this._framingBehavior = null)); } /** * Gets the auto rotation behavior of the camera if it has been enabled. * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#autorotation-behavior */ get autoRotationBehavior() { return this._autoRotationBehavior; } /** * Defines if the auto rotation behavior of the camera is enabled on the camera. * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#autorotation-behavior */ get useAutoRotationBehavior() { return this._autoRotationBehavior != null; } set useAutoRotationBehavior(e) { e !== this.useAutoRotationBehavior && (e ? (this._autoRotationBehavior = new hee(), this.addBehavior(this._autoRotationBehavior)) : this._autoRotationBehavior && (this.removeBehavior(this._autoRotationBehavior), this._autoRotationBehavior = null)); } /** * Instantiates a new ArcRotateCamera in a given scene * @param name Defines the name of the camera * @param alpha Defines the camera rotation along the longitudinal axis * @param beta Defines the camera rotation along the latitudinal axis * @param radius Defines the camera distance from its target * @param target Defines the camera target * @param scene Defines the scene the camera belongs to * @param setActiveOnSceneIfNoneActive Defines whether the camera should be marked as active if not other active cameras have been defined */ constructor(e, t, r, n, i, s, a = !0) { super(e, S.Zero(), s, a), this.inertialAlphaOffset = 0, this.inertialBetaOffset = 0, this.inertialRadiusOffset = 0, this.lowerAlphaLimit = null, this.upperAlphaLimit = null, this.lowerBetaLimit = 0.01, this.upperBetaLimit = Math.PI - 0.01, this.lowerRadiusLimit = null, this.upperRadiusLimit = null, this.inertialPanningX = 0, this.inertialPanningY = 0, this.pinchToPanMaxDistance = 20, this.panningDistanceLimit = null, this.panningOriginTarget = S.Zero(), this.panningInertia = 0.9, this.zoomOnFactor = 1, this.targetScreenOffset = at.Zero(), this.allowUpsideDown = !0, this.useInputToRestoreState = !0, this._viewMatrix = new he(), this.panningAxis = new S(1, 1, 0), this._transformedDirection = new S(), this.mapPanning = !1, this.onMeshTargetChangedObservable = new Oe(), this.checkCollisions = !1, this.collisionRadius = new S(0.5, 0.5, 0.5), this._previousPosition = S.Zero(), this._collisionVelocity = S.Zero(), this._newPosition = S.Zero(), this._computationVector = S.Zero(), this._onCollisionPositionChange = (f, o, d = null) => { d ? (this.setPosition(o), this.onCollide && this.onCollide(d)) : this._previousPosition.copyFrom(this._position); const v = Math.cos(this.alpha), u = Math.sin(this.alpha), l = Math.cos(this.beta); let P = Math.sin(this.beta); P === 0 && (P = 1e-4); const p = this._getTargetPosition(); this._computationVector.copyFromFloats(this.radius * v * P, this.radius * l, this.radius * u * P), p.addToRef(this._computationVector, this._newPosition), this._position.copyFrom(this._newPosition); let c = this.upVector; this.allowUpsideDown && this.beta < 0 && (c = c.clone(), c = c.negate()), this._computeViewMatrix(this._position, p, c), this._viewMatrix.addAtIndex(12, this.targetScreenOffset.x), this._viewMatrix.addAtIndex(13, this.targetScreenOffset.y), this._collisionTriggered = !1; }, this._target = S.Zero(), i && this.setTarget(i), this.alpha = t, this.beta = r, this.radius = n, this.getViewMatrix(), this.inputs = new CO(this), this.inputs.addKeyboard().addMouseWheel().addPointers(); } // Cache /** @internal */ _initCache() { super._initCache(), this._cache._target = new S(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE), this._cache.alpha = void 0, this._cache.beta = void 0, this._cache.radius = void 0, this._cache.targetScreenOffset = at.Zero(); } /** * @internal */ _updateCache(e) { e || super._updateCache(), this._cache._target.copyFrom(this._getTargetPosition()), this._cache.alpha = this.alpha, this._cache.beta = this.beta, this._cache.radius = this.radius, this._cache.targetScreenOffset.copyFrom(this.targetScreenOffset); } _getTargetPosition() { if (this._targetHost && this._targetHost.getAbsolutePosition) { const t = this._targetHost.getAbsolutePosition(); this._targetBoundingCenter ? t.addToRef(this._targetBoundingCenter, this._target) : this._target.copyFrom(t); } const e = this._getLockedTargetPosition(); return e || this._target; } /** * Stores the current state of the camera (alpha, beta, radius and target) * @returns the camera itself */ storeState() { return this._storedAlpha = this.alpha, this._storedBeta = this.beta, this._storedRadius = this.radius, this._storedTarget = this._getTargetPosition().clone(), this._storedTargetScreenOffset = this.targetScreenOffset.clone(), super.storeState(); } /** * @internal * Restored camera state. You must call storeState() first */ _restoreStateValues() { return super._restoreStateValues() ? (this.setTarget(this._storedTarget.clone()), this.alpha = this._storedAlpha, this.beta = this._storedBeta, this.radius = this._storedRadius, this.targetScreenOffset = this._storedTargetScreenOffset.clone(), this.inertialAlphaOffset = 0, this.inertialBetaOffset = 0, this.inertialRadiusOffset = 0, this.inertialPanningX = 0, this.inertialPanningY = 0, !0) : !1; } // Synchronized /** @internal */ _isSynchronizedViewMatrix() { return super._isSynchronizedViewMatrix() ? this._cache._target.equals(this._getTargetPosition()) && this._cache.alpha === this.alpha && this._cache.beta === this.beta && this._cache.radius === this.radius && this._cache.targetScreenOffset.equals(this.targetScreenOffset) : !1; } /** * Attached controls to the current camera. * @param ignored defines an ignored parameter kept for backward compatibility. * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) * @param useCtrlForPanning Defines whether ctrl is used for panning within the controls * @param panningMouseButton Defines whether panning is allowed through mouse click button */ attachControl(e, t, r = !0, n = 2) { const i = arguments; t = ye.BackCompatCameraNoPreventDefault(i), this._useCtrlForPanning = r, this._panningMouseButton = n, typeof i[0] == "boolean" && (i.length > 1 && (this._useCtrlForPanning = i[1]), i.length > 2 && (this._panningMouseButton = i[2])), this.inputs.attachElement(t), this._reset = () => { this.inertialAlphaOffset = 0, this.inertialBetaOffset = 0, this.inertialRadiusOffset = 0, this.inertialPanningX = 0, this.inertialPanningY = 0; }; } /** * Detach the current controls from the specified dom element. */ detachControl() { this.inputs.detachElement(), this._reset && this._reset(); } /** @internal */ _checkInputs() { if (!this._collisionTriggered) { if (this.inputs.checkInputs(), this.inertialAlphaOffset !== 0 || this.inertialBetaOffset !== 0 || this.inertialRadiusOffset !== 0) { const e = this.invertRotation ? -1 : 1, t = this._calculateHandednessMultiplier(); let r = this.inertialAlphaOffset * t; this.beta <= 0 && (r *= -1), this.alpha += r * e, this.beta += this.inertialBetaOffset * e, this.radius -= this.inertialRadiusOffset, this.inertialAlphaOffset *= this.inertia, this.inertialBetaOffset *= this.inertia, this.inertialRadiusOffset *= this.inertia, Math.abs(this.inertialAlphaOffset) < Dn && (this.inertialAlphaOffset = 0), Math.abs(this.inertialBetaOffset) < Dn && (this.inertialBetaOffset = 0), Math.abs(this.inertialRadiusOffset) < this.speed * Dn && (this.inertialRadiusOffset = 0); } if (this.inertialPanningX !== 0 || this.inertialPanningY !== 0) { const e = new S(this.inertialPanningX, this.inertialPanningY, this.inertialPanningY); if (this._viewMatrix.invertToRef(this._cameraTransformMatrix), e.multiplyInPlace(this.panningAxis), S.TransformNormalToRef(e, this._cameraTransformMatrix, this._transformedDirection), this.mapPanning) { const t = this.upVector, r = S.CrossToRef(this._transformedDirection, t, this._transformedDirection); S.CrossToRef(t, r, this._transformedDirection); } else this.panningAxis.y || (this._transformedDirection.y = 0); this._targetHost || (this.panningDistanceLimit ? (this._transformedDirection.addInPlace(this._target), S.DistanceSquared(this._transformedDirection, this.panningOriginTarget) <= this.panningDistanceLimit * this.panningDistanceLimit && this._target.copyFrom(this._transformedDirection)) : this._target.addInPlace(this._transformedDirection)), this.inertialPanningX *= this.panningInertia, this.inertialPanningY *= this.panningInertia, Math.abs(this.inertialPanningX) < this.speed * Dn && (this.inertialPanningX = 0), Math.abs(this.inertialPanningY) < this.speed * Dn && (this.inertialPanningY = 0); } this._checkLimits(), super._checkInputs(); } } _checkLimits() { this.lowerBetaLimit === null || this.lowerBetaLimit === void 0 ? this.allowUpsideDown && this.beta > Math.PI && (this.beta = this.beta - 2 * Math.PI) : this.beta < this.lowerBetaLimit && (this.beta = this.lowerBetaLimit), this.upperBetaLimit === null || this.upperBetaLimit === void 0 ? this.allowUpsideDown && this.beta < -Math.PI && (this.beta = this.beta + 2 * Math.PI) : this.beta > this.upperBetaLimit && (this.beta = this.upperBetaLimit), this.lowerAlphaLimit !== null && this.alpha < this.lowerAlphaLimit && (this.alpha = this.lowerAlphaLimit), this.upperAlphaLimit !== null && this.alpha > this.upperAlphaLimit && (this.alpha = this.upperAlphaLimit), this.lowerRadiusLimit !== null && this.radius < this.lowerRadiusLimit && (this.radius = this.lowerRadiusLimit, this.inertialRadiusOffset = 0), this.upperRadiusLimit !== null && this.radius > this.upperRadiusLimit && (this.radius = this.upperRadiusLimit, this.inertialRadiusOffset = 0); } /** * Rebuilds angles (alpha, beta) and radius from the give position and target */ rebuildAnglesAndRadius() { this._position.subtractToRef(this._getTargetPosition(), this._computationVector), (this._upVector.x !== 0 || this._upVector.y !== 1 || this._upVector.z !== 0) && S.TransformCoordinatesToRef(this._computationVector, this._upToYMatrix, this._computationVector), this.radius = this._computationVector.length(), this.radius === 0 && (this.radius = 1e-4); const e = this.alpha; this._computationVector.x === 0 && this._computationVector.z === 0 ? this.alpha = Math.PI / 2 : this.alpha = Math.acos(this._computationVector.x / Math.sqrt(Math.pow(this._computationVector.x, 2) + Math.pow(this._computationVector.z, 2))), this._computationVector.z < 0 && (this.alpha = 2 * Math.PI - this.alpha); const t = Math.round((e - this.alpha) / (2 * Math.PI)); this.alpha += t * 2 * Math.PI, this.beta = Math.acos(this._computationVector.y / this.radius), this._checkLimits(); } /** * Use a position to define the current camera related information like alpha, beta and radius * @param position Defines the position to set the camera at */ setPosition(e) { this._position.equals(e) || (this._position.copyFrom(e), this.rebuildAnglesAndRadius()); } /** * Defines the target the camera should look at. * This will automatically adapt alpha beta and radius to fit within the new target. * Please note that setting a target as a mesh will disable panning. * @param target Defines the new target as a Vector or a mesh * @param toBoundingCenter In case of a mesh target, defines whether to target the mesh position or its bounding information center * @param allowSamePosition If false, prevents reapplying the new computed position if it is identical to the current one (optim) * @param cloneAlphaBetaRadius If true, replicate the current setup (alpha, beta, radius) on the new target */ setTarget(e, t = !1, r = !1, n = !1) { var i; if (n = (i = this.overrideCloneAlphaBetaRadius) !== null && i !== void 0 ? i : n, e.getBoundingInfo) t ? this._targetBoundingCenter = e.getBoundingInfo().boundingBox.centerWorld.clone() : this._targetBoundingCenter = null, e.computeWorldMatrix(), this._targetHost = e, this._target = this._getTargetPosition(), this.onMeshTargetChangedObservable.notifyObservers(this._targetHost); else { const s = e, a = this._getTargetPosition(); if (a && !r && a.equals(s)) return; this._targetHost = null, this._target = s, this._targetBoundingCenter = null, this.onMeshTargetChangedObservable.notifyObservers(null); } n || this.rebuildAnglesAndRadius(); } /** @internal */ _getViewMatrix() { const e = Math.cos(this.alpha), t = Math.sin(this.alpha), r = Math.cos(this.beta); let n = Math.sin(this.beta); n === 0 && (n = 1e-4), this.radius === 0 && (this.radius = 1e-4); const i = this._getTargetPosition(); if (this._computationVector.copyFromFloats(this.radius * e * n, this.radius * r, this.radius * t * n), (this._upVector.x !== 0 || this._upVector.y !== 1 || this._upVector.z !== 0) && S.TransformCoordinatesToRef(this._computationVector, this._yToUpMatrix, this._computationVector), i.addToRef(this._computationVector, this._newPosition), this.getScene().collisionsEnabled && this.checkCollisions) { const s = this.getScene().collisionCoordinator; this._collider || (this._collider = s.createCollider()), this._collider._radius = this.collisionRadius, this._newPosition.subtractToRef(this._position, this._collisionVelocity), this._collisionTriggered = !0, s.getNewPosition(this._position, this._collisionVelocity, this._collider, 3, null, this._onCollisionPositionChange, this.uniqueId); } else { this._position.copyFrom(this._newPosition); let s = this.upVector; this.allowUpsideDown && n < 0 && (s = s.negate()), this._computeViewMatrix(this._position, i, s), this._viewMatrix.addAtIndex(12, this.targetScreenOffset.x), this._viewMatrix.addAtIndex(13, this.targetScreenOffset.y); } return this._currentTarget = i, this._viewMatrix; } /** * Zooms on a mesh to be at the min distance where we could see it fully in the current viewport. * @param meshes Defines the mesh to zoom on * @param doNotUpdateMaxZ Defines whether or not maxZ should be updated whilst zooming on the mesh (this can happen if the mesh is big and the maxradius pretty small for instance) */ zoomOn(e, t = !1) { e = e || this.getScene().meshes; const r = Ee.MinMax(e); let n = this._calculateLowerRadiusFromModelBoundingSphere(r.min, r.max); n = Math.max(Math.min(n, this.upperRadiusLimit || Number.MAX_VALUE), this.lowerRadiusLimit || 0), this.radius = n * this.zoomOnFactor, this.focusOn({ min: r.min, max: r.max, distance: n }, t); } /** * Focus on a mesh or a bounding box. This adapts the target and maxRadius if necessary but does not update the current radius. * The target will be changed but the radius * @param meshesOrMinMaxVectorAndDistance Defines the mesh or bounding info to focus on * @param doNotUpdateMaxZ Defines whether or not maxZ should be updated whilst zooming on the mesh (this can happen if the mesh is big and the maxradius pretty small for instance) */ focusOn(e, t = !1) { let r, n; if (e.min === void 0) { const i = e || this.getScene().meshes; r = Ee.MinMax(i), n = S.Distance(r.min, r.max); } else { const i = e; r = i, n = i.distance; } this._target = Ee.Center(r), t || (this.maxZ = n * 2); } /** * @override * Override Camera.createRigCamera */ createRigCamera(e, t) { let r = 0; switch (this.cameraRigMode) { case Tr.RIG_MODE_STEREOSCOPIC_ANAGLYPH: case Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL: case Tr.RIG_MODE_STEREOSCOPIC_OVERUNDER: case Tr.RIG_MODE_STEREOSCOPIC_INTERLACED: case Tr.RIG_MODE_VR: r = this._cameraRigParams.stereoHalfAngle * (t === 0 ? 1 : -1); break; case Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED: r = this._cameraRigParams.stereoHalfAngle * (t === 0 ? -1 : 1); break; } const n = new ps(e, this.alpha + r, this.beta, this.radius, this._target, this.getScene()); return n._cameraRigParams = {}, n.isRigCamera = !0, n.rigParent = this, n.upVector = this.upVector, n.mode = this.mode, n.orthoLeft = this.orthoLeft, n.orthoRight = this.orthoRight, n.orthoBottom = this.orthoBottom, n.orthoTop = this.orthoTop, n; } /** * @internal * @override * Override Camera._updateRigCameras */ _updateRigCameras() { const e = this._rigCameras[0], t = this._rigCameras[1]; switch (e.beta = t.beta = this.beta, this.cameraRigMode) { case Tr.RIG_MODE_STEREOSCOPIC_ANAGLYPH: case Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL: case Tr.RIG_MODE_STEREOSCOPIC_OVERUNDER: case Tr.RIG_MODE_STEREOSCOPIC_INTERLACED: case Tr.RIG_MODE_VR: e.alpha = this.alpha - this._cameraRigParams.stereoHalfAngle, t.alpha = this.alpha + this._cameraRigParams.stereoHalfAngle; break; case Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED: e.alpha = this.alpha + this._cameraRigParams.stereoHalfAngle, t.alpha = this.alpha - this._cameraRigParams.stereoHalfAngle; break; } super._updateRigCameras(); } /** * @internal */ _calculateLowerRadiusFromModelBoundingSphere(e, t, r = 1) { const n = S.Distance(e, t), s = this.getScene().getEngine().getAspectRatio(this), a = Math.tan(this.fov / 2), f = a * s, d = n * 0.5 * r, v = d * Math.sqrt(1 + 1 / (f * f)), u = d * Math.sqrt(1 + 1 / (a * a)); return Math.max(v, u); } /** * Destroy the camera and release the current resources hold by it. */ dispose() { this.inputs.clear(), super.dispose(); } /** * Gets the current object class name. * @returns the class name */ getClassName() { return "ArcRotateCamera"; } } C([ M() ], ps.prototype, "alpha", void 0); C([ M() ], ps.prototype, "beta", void 0); C([ M() ], ps.prototype, "radius", void 0); C([ M() ], ps.prototype, "overrideCloneAlphaBetaRadius", void 0); C([ fo("target") ], ps.prototype, "_target", void 0); C([ tU("targetHost") ], ps.prototype, "_targetHost", void 0); C([ M() ], ps.prototype, "inertialAlphaOffset", void 0); C([ M() ], ps.prototype, "inertialBetaOffset", void 0); C([ M() ], ps.prototype, "inertialRadiusOffset", void 0); C([ M() ], ps.prototype, "lowerAlphaLimit", void 0); C([ M() ], ps.prototype, "upperAlphaLimit", void 0); C([ M() ], ps.prototype, "lowerBetaLimit", void 0); C([ M() ], ps.prototype, "upperBetaLimit", void 0); C([ M() ], ps.prototype, "lowerRadiusLimit", void 0); C([ M() ], ps.prototype, "upperRadiusLimit", void 0); C([ M() ], ps.prototype, "inertialPanningX", void 0); C([ M() ], ps.prototype, "inertialPanningY", void 0); C([ M() ], ps.prototype, "pinchToPanMaxDistance", void 0); C([ M() ], ps.prototype, "panningDistanceLimit", void 0); C([ fo() ], ps.prototype, "panningOriginTarget", void 0); C([ M() ], ps.prototype, "panningInertia", void 0); C([ M() ], ps.prototype, "zoomToMouseLocation", null); C([ M() ], ps.prototype, "zoomOnFactor", void 0); C([ HR() ], ps.prototype, "targetScreenOffset", void 0); C([ M() ], ps.prototype, "allowUpsideDown", void 0); C([ M() ], ps.prototype, "useInputToRestoreState", void 0); Cs.AddNodeConstructor("DeviceOrientationCamera", (A, e) => () => new OO(A, S.Zero(), e)); class OO extends SA { /** * Creates a new device orientation camera * @param name The name of the camera * @param position The start position camera * @param scene The scene the camera belongs to */ constructor(e, t, r) { super(e, t, r), this._tmpDragQuaternion = new Ze(), this._disablePointerInputWhenUsingDeviceOrientation = !0, this._dragFactor = 0, this._quaternionCache = new Ze(), this.inputs.addDeviceOrientation(), this.inputs._deviceOrientationInput && this.inputs._deviceOrientationInput._onDeviceOrientationChangedObservable.addOnce(() => { this._disablePointerInputWhenUsingDeviceOrientation && this.inputs._mouseInput && (this.inputs._mouseInput._allowCameraRotation = !1, this.inputs._mouseInput.onPointerMovedObservable.add((n) => { this._dragFactor != 0 && (this._initialQuaternion || (this._initialQuaternion = new Ze()), Ze.FromEulerAnglesToRef(0, n.offsetX * this._dragFactor, 0, this._tmpDragQuaternion), this._initialQuaternion.multiplyToRef(this._tmpDragQuaternion, this._initialQuaternion)); })); }); } /** * Gets or sets a boolean indicating that pointer input must be disabled on first orientation sensor update (Default: true) */ get disablePointerInputWhenUsingDeviceOrientation() { return this._disablePointerInputWhenUsingDeviceOrientation; } set disablePointerInputWhenUsingDeviceOrientation(e) { this._disablePointerInputWhenUsingDeviceOrientation = e; } /** * Enabled turning on the y axis when the orientation sensor is active * @param dragFactor the factor that controls the turn speed (default: 1/300) */ enableHorizontalDragging(e = 1 / 300) { this._dragFactor = e; } /** * Gets the current instance class name ("DeviceOrientationCamera"). * This helps avoiding instanceof at run time. * @returns the class name */ getClassName() { return "DeviceOrientationCamera"; } /** * @internal * Checks and applies the current values of the inputs to the camera. (Internal use only) */ _checkInputs() { super._checkInputs(), this._quaternionCache.copyFrom(this.rotationQuaternion), this._initialQuaternion && this._initialQuaternion.multiplyToRef(this.rotationQuaternion, this.rotationQuaternion); } /** * Reset the camera to its default orientation on the specified axis only. * @param axis The axis to reset */ resetToCurrentRotation(e = bf.Y) { this.rotationQuaternion && (this._initialQuaternion || (this._initialQuaternion = new Ze()), this._initialQuaternion.copyFrom(this._quaternionCache || this.rotationQuaternion), ["x", "y", "z"].forEach((t) => { e[t] ? this._initialQuaternion[t] *= -1 : this._initialQuaternion[t] = 0; }), this._initialQuaternion.normalize(), this._initialQuaternion.multiplyToRef(this.rotationQuaternion, this.rotationQuaternion)); } } class Dee extends RR { /** * Instantiates a new FlyCameraInputsManager. * @param camera Defines the camera the inputs belong to. */ constructor(e) { super(e); } /** * Add keyboard input support to the input manager. * @returns the new FlyCameraKeyboardMoveInput(). */ addKeyboard() { return this.add(new jq()), this; } /** * Add mouse input support to the input manager. * @returns the new FlyCameraMouseInput(). */ addMouse() { return this.add(new CR()), this; } } class FR extends b1 { /** * Gets the input sensibility for mouse input. * Higher values reduce sensitivity. */ get angularSensibility() { const e = this.inputs.attached.mouse; return e ? e.angularSensibility : 0; } /** * Sets the input sensibility for a mouse input. * Higher values reduce sensitivity. */ set angularSensibility(e) { const t = this.inputs.attached.mouse; t && (t.angularSensibility = e); } /** * Get the keys for camera movement forward. */ get keysForward() { const e = this.inputs.attached.keyboard; return e ? e.keysForward : []; } /** * Set the keys for camera movement forward. */ set keysForward(e) { const t = this.inputs.attached.keyboard; t && (t.keysForward = e); } /** * Get the keys for camera movement backward. */ get keysBackward() { const e = this.inputs.attached.keyboard; return e ? e.keysBackward : []; } set keysBackward(e) { const t = this.inputs.attached.keyboard; t && (t.keysBackward = e); } /** * Get the keys for camera movement up. */ get keysUp() { const e = this.inputs.attached.keyboard; return e ? e.keysUp : []; } /** * Set the keys for camera movement up. */ set keysUp(e) { const t = this.inputs.attached.keyboard; t && (t.keysUp = e); } /** * Get the keys for camera movement down. */ get keysDown() { const e = this.inputs.attached.keyboard; return e ? e.keysDown : []; } /** * Set the keys for camera movement down. */ set keysDown(e) { const t = this.inputs.attached.keyboard; t && (t.keysDown = e); } /** * Get the keys for camera movement left. */ get keysLeft() { const e = this.inputs.attached.keyboard; return e ? e.keysLeft : []; } /** * Set the keys for camera movement left. */ set keysLeft(e) { const t = this.inputs.attached.keyboard; t && (t.keysLeft = e); } /** * Set the keys for camera movement right. */ get keysRight() { const e = this.inputs.attached.keyboard; return e ? e.keysRight : []; } /** * Set the keys for camera movement right. */ set keysRight(e) { const t = this.inputs.attached.keyboard; t && (t.keysRight = e); } /** * Instantiates a FlyCamera. * This is a flying camera, designed for 3D movement and rotation in all directions, * such as in a 3D Space Shooter or a Flight Simulator. * @param name Define the name of the camera in the scene. * @param position Define the starting position of the camera in the scene. * @param scene Define the scene the camera belongs to. * @param setActiveOnSceneIfNoneActive Defines whether the camera should be marked as active, if no other camera has been defined as active. */ constructor(e, t, r, n = !0) { super(e, t, r, n), this.ellipsoid = new S(1, 1, 1), this.ellipsoidOffset = new S(0, 0, 0), this.checkCollisions = !1, this.applyGravity = !1, this.cameraDirection = S.Zero(), this._trackRoll = 0, this.rollCorrect = 100, this.bankedTurn = !1, this.bankedTurnLimit = Math.PI / 2, this.bankedTurnMultiplier = 1, this._needMoveForGravity = !1, this._oldPosition = S.Zero(), this._diffPosition = S.Zero(), this._newPosition = S.Zero(), this._collisionMask = -1, this._onCollisionPositionChange = (i, s, a = null) => { ((o) => { this._newPosition.copyFrom(o), this._newPosition.subtractToRef(this._oldPosition, this._diffPosition), this._diffPosition.length() > Ge.CollisionsEpsilon && (this.position.addInPlace(this._diffPosition), this.onCollide && a && this.onCollide(a)); })(s); }, this.inputs = new Dee(this), this.inputs.addKeyboard().addMouse(); } /** * Attached controls to the current camera. * @param ignored defines an ignored parameter kept for backward compatibility. * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachControl(e, t) { t = ye.BackCompatCameraNoPreventDefault(arguments), this.inputs.attachElement(t); } /** * Detach a control from the HTML DOM element. * The camera will stop reacting to that input. */ detachControl() { this.inputs.detachElement(), this.cameraDirection = new S(0, 0, 0); } /** * Get the mask that the camera ignores in collision events. */ get collisionMask() { return this._collisionMask; } /** * Set the mask that the camera ignores in collision events. */ set collisionMask(e) { this._collisionMask = isNaN(e) ? -1 : e; } /** * @internal */ _collideWithWorld(e) { let t; this.parent ? t = S.TransformCoordinates(this.position, this.parent.getWorldMatrix()) : t = this.position, t.subtractFromFloatsToRef(0, this.ellipsoid.y, 0, this._oldPosition), this._oldPosition.addInPlace(this.ellipsoidOffset); const r = this.getScene().collisionCoordinator; this._collider || (this._collider = r.createCollider()), this._collider._radius = this.ellipsoid, this._collider.collisionMask = this._collisionMask; let n = e; this.applyGravity && (n = e.add(this.getScene().gravity)), r.getNewPosition(this._oldPosition, n, this._collider, 3, null, this._onCollisionPositionChange, this.uniqueId); } /** @internal */ _checkInputs() { this._localDirection || (this._localDirection = S.Zero(), this._transformedDirection = S.Zero()), this.inputs.checkInputs(), super._checkInputs(); } /** * Enable movement without a user input. This allows gravity to always be applied. */ set needMoveForGravity(e) { this._needMoveForGravity = e; } /** * When true, gravity is applied whether there is user input or not. */ get needMoveForGravity() { return this._needMoveForGravity; } /** @internal */ _decideIfNeedsToMove() { return this._needMoveForGravity || Math.abs(this.cameraDirection.x) > 0 || Math.abs(this.cameraDirection.y) > 0 || Math.abs(this.cameraDirection.z) > 0; } /** @internal */ _updatePosition() { this.checkCollisions && this.getScene().collisionsEnabled ? this._collideWithWorld(this.cameraDirection) : super._updatePosition(); } /** * Restore the Roll to its target value at the rate specified. * @param rate - Higher means slower restoring. * @internal */ restoreRoll(e) { const t = this._trackRoll, r = this.rotation.z, n = t - r, i = 1e-3; Math.abs(n) >= i && (this.rotation.z += n / e, Math.abs(t - this.rotation.z) <= i && (this.rotation.z = t)); } /** * Destroy the camera and release the current resources held by it. */ dispose() { this.inputs.clear(), super.dispose(); } /** * Get the current object class name. * @returns the class name. */ getClassName() { return "FlyCamera"; } } C([ fo() ], FR.prototype, "ellipsoid", void 0); C([ fo() ], FR.prototype, "ellipsoidOffset", void 0); C([ M() ], FR.prototype, "checkCollisions", void 0); C([ M() ], FR.prototype, "applyGravity", void 0); class jee extends RR { /** * Instantiates a new FollowCameraInputsManager. * @param camera Defines the camera the inputs belong to */ constructor(e) { super(e); } /** * Add keyboard input support to the input manager. * @returns the current input manager */ addKeyboard() { return this.add(new aA()), this; } /** * Add mouse wheel input support to the input manager. * @returns the current input manager */ addMouseWheel() { return this.add(new qD()), this; } /** * Add pointers input support to the input manager. * @returns the current input manager */ addPointers() { return this.add(new Eu()), this; } /** * Add orientation input support to the input manager. * @returns the current input manager */ addVRDeviceOrientation() { return console.warn("DeviceOrientation support not yet implemented for FollowCamera."), this; } } Cs.AddNodeConstructor("FollowCamera", (A, e) => () => new Fv(A, S.Zero(), e)); Cs.AddNodeConstructor("ArcFollowCamera", (A, e) => () => new wee(A, 0, 0, 1, null, e)); class Fv extends b1 { /** * Instantiates the follow camera. * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#followcamera * @param name Define the name of the camera in the scene * @param position Define the position of the camera * @param scene Define the scene the camera belong to * @param lockedTarget Define the target of the camera */ constructor(e, t, r, n = null) { super(e, t, r), this.radius = 12, this.lowerRadiusLimit = null, this.upperRadiusLimit = null, this.rotationOffset = 0, this.lowerRotationOffsetLimit = null, this.upperRotationOffsetLimit = null, this.heightOffset = 4, this.lowerHeightOffsetLimit = null, this.upperHeightOffsetLimit = null, this.cameraAcceleration = 0.05, this.maxCameraSpeed = 20, this.lockedTarget = n, this.inputs = new jee(this), this.inputs.addKeyboard().addMouseWheel().addPointers(); } _follow(e) { if (!e) return; const t = ue.Matrix[0]; e.absoluteRotationQuaternion.toRotationMatrix(t); const r = Math.atan2(t.m[8], t.m[10]), n = ye.ToRadians(this.rotationOffset) + r, i = e.getAbsolutePosition(), s = i.x + Math.sin(n) * this.radius, a = i.z + Math.cos(n) * this.radius, f = s - this.position.x, o = i.y + this.heightOffset - this.position.y, d = a - this.position.z; let v = f * this.cameraAcceleration * 2, u = o * this.cameraAcceleration, l = d * this.cameraAcceleration * 2; (v > this.maxCameraSpeed || v < -this.maxCameraSpeed) && (v = v < 1 ? -this.maxCameraSpeed : this.maxCameraSpeed), (u > this.maxCameraSpeed || u < -this.maxCameraSpeed) && (u = u < 1 ? -this.maxCameraSpeed : this.maxCameraSpeed), (l > this.maxCameraSpeed || l < -this.maxCameraSpeed) && (l = l < 1 ? -this.maxCameraSpeed : this.maxCameraSpeed), this.position = new S(this.position.x + v, this.position.y + u, this.position.z + l), this.setTarget(i); } /** * Attached controls to the current camera. * @param ignored defines an ignored parameter kept for backward compatibility. * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault) */ attachControl(e, t) { t = ye.BackCompatCameraNoPreventDefault(arguments), this.inputs.attachElement(t), this._reset = () => { }; } /** * Detach the current controls from the specified dom element. */ detachControl() { this.inputs.detachElement(), this._reset && this._reset(); } /** @internal */ _checkInputs() { this.inputs.checkInputs(), this._checkLimits(), super._checkInputs(), this.lockedTarget && this._follow(this.lockedTarget); } _checkLimits() { this.lowerRadiusLimit !== null && this.radius < this.lowerRadiusLimit && (this.radius = this.lowerRadiusLimit), this.upperRadiusLimit !== null && this.radius > this.upperRadiusLimit && (this.radius = this.upperRadiusLimit), this.lowerHeightOffsetLimit !== null && this.heightOffset < this.lowerHeightOffsetLimit && (this.heightOffset = this.lowerHeightOffsetLimit), this.upperHeightOffsetLimit !== null && this.heightOffset > this.upperHeightOffsetLimit && (this.heightOffset = this.upperHeightOffsetLimit), this.lowerRotationOffsetLimit !== null && this.rotationOffset < this.lowerRotationOffsetLimit && (this.rotationOffset = this.lowerRotationOffsetLimit), this.upperRotationOffsetLimit !== null && this.rotationOffset > this.upperRotationOffsetLimit && (this.rotationOffset = this.upperRotationOffsetLimit); } /** * Gets the camera class name. * @returns the class name */ getClassName() { return "FollowCamera"; } } C([ M() ], Fv.prototype, "radius", void 0); C([ M() ], Fv.prototype, "lowerRadiusLimit", void 0); C([ M() ], Fv.prototype, "upperRadiusLimit", void 0); C([ M() ], Fv.prototype, "rotationOffset", void 0); C([ M() ], Fv.prototype, "lowerRotationOffsetLimit", void 0); C([ M() ], Fv.prototype, "upperRotationOffsetLimit", void 0); C([ M() ], Fv.prototype, "heightOffset", void 0); C([ M() ], Fv.prototype, "lowerHeightOffsetLimit", void 0); C([ M() ], Fv.prototype, "upperHeightOffsetLimit", void 0); C([ M() ], Fv.prototype, "cameraAcceleration", void 0); C([ M() ], Fv.prototype, "maxCameraSpeed", void 0); C([ tU("lockedTargetId") ], Fv.prototype, "lockedTarget", void 0); class wee extends b1 { /** * Instantiates a new ArcFollowCamera * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#followcamera * @param name Define the name of the camera * @param alpha Define the rotation angle of the camera around the longitudinal axis * @param beta Define the rotation angle of the camera around the elevation axis * @param radius Define the radius of the camera from its target point * @param target Define the target of the camera * @param scene Define the scene the camera belongs to */ constructor(e, t, r, n, i, s) { super(e, S.Zero(), s), this.alpha = t, this.beta = r, this.radius = n, this._cartesianCoordinates = S.Zero(), this.setMeshTarget(i); } /** * Sets the mesh to follow with this camera. * @param target the target to follow */ setMeshTarget(e) { this._meshTarget = e, this._follow(); } _follow() { if (!this._meshTarget) return; this._cartesianCoordinates.x = this.radius * Math.cos(this.alpha) * Math.cos(this.beta), this._cartesianCoordinates.y = this.radius * Math.sin(this.beta), this._cartesianCoordinates.z = this.radius * Math.sin(this.alpha) * Math.cos(this.beta); const e = this._meshTarget.getAbsolutePosition(); this.position = e.add(this._cartesianCoordinates), this.setTarget(e); } /** @internal */ _checkInputs() { super._checkInputs(), this._follow(); } /** * Returns the class name of the object. * It is mostly used internally for serialization purposes. */ getClassName() { return "ArcFollowCamera"; } } var El; (function(A) { A[A.A = 0] = "A", A[A.B = 1] = "B", A[A.X = 2] = "X", A[A.Y = 3] = "Y", A[A.LB = 4] = "LB", A[A.RB = 5] = "RB", A[A.Back = 8] = "Back", A[A.Start = 9] = "Start", A[A.LeftStick = 10] = "LeftStick", A[A.RightStick = 11] = "RightStick"; })(El || (El = {})); var Lw; (function(A) { A[A.Up = 12] = "Up", A[A.Down = 13] = "Down", A[A.Left = 14] = "Left", A[A.Right = 15] = "Right"; })(Lw || (Lw = {})); class mee extends N9 { /** * Creates a new XBox360 gamepad object * @param id defines the id of this gamepad * @param index defines its index * @param gamepad defines the internal HTML gamepad object * @param xboxOne defines if it is a XBox One gamepad */ constructor(e, t, r, n = !1) { super(e, t, r, 0, 1, 2, 3), this._leftTrigger = 0, this._rightTrigger = 0, this.onButtonDownObservable = new Oe(), this.onButtonUpObservable = new Oe(), this.onPadDownObservable = new Oe(), this.onPadUpObservable = new Oe(), this._buttonA = 0, this._buttonB = 0, this._buttonX = 0, this._buttonY = 0, this._buttonBack = 0, this._buttonStart = 0, this._buttonLB = 0, this._buttonRB = 0, this._buttonLeftStick = 0, this._buttonRightStick = 0, this._dPadUp = 0, this._dPadDown = 0, this._dPadLeft = 0, this._dPadRight = 0, this._isXboxOnePad = !1, this.type = N9.XBOX, this._isXboxOnePad = n; } /** * Defines the callback to call when left trigger is pressed * @param callback defines the callback to use */ onlefttriggerchanged(e) { this._onlefttriggerchanged = e; } /** * Defines the callback to call when right trigger is pressed * @param callback defines the callback to use */ onrighttriggerchanged(e) { this._onrighttriggerchanged = e; } /** * Gets the left trigger value */ get leftTrigger() { return this._leftTrigger; } /** * Sets the left trigger value */ set leftTrigger(e) { this._onlefttriggerchanged && this._leftTrigger !== e && this._onlefttriggerchanged(e), this._leftTrigger = e; } /** * Gets the right trigger value */ get rightTrigger() { return this._rightTrigger; } /** * Sets the right trigger value */ set rightTrigger(e) { this._onrighttriggerchanged && this._rightTrigger !== e && this._onrighttriggerchanged(e), this._rightTrigger = e; } /** * Defines the callback to call when a button is pressed * @param callback defines the callback to use */ onbuttondown(e) { this._onbuttondown = e; } /** * Defines the callback to call when a button is released * @param callback defines the callback to use */ onbuttonup(e) { this._onbuttonup = e; } /** * Defines the callback to call when a pad is pressed * @param callback defines the callback to use */ ondpaddown(e) { this._ondpaddown = e; } /** * Defines the callback to call when a pad is released * @param callback defines the callback to use */ ondpadup(e) { this._ondpadup = e; } _setButtonValue(e, t, r) { return e !== t && (e === 1 && (this._onbuttondown && this._onbuttondown(r), this.onButtonDownObservable.notifyObservers(r)), e === 0 && (this._onbuttonup && this._onbuttonup(r), this.onButtonUpObservable.notifyObservers(r))), e; } _setDPadValue(e, t, r) { return e !== t && (e === 1 && (this._ondpaddown && this._ondpaddown(r), this.onPadDownObservable.notifyObservers(r)), e === 0 && (this._ondpadup && this._ondpadup(r), this.onPadUpObservable.notifyObservers(r))), e; } /** * Gets the value of the `A` button */ get buttonA() { return this._buttonA; } /** * Sets the value of the `A` button */ set buttonA(e) { this._buttonA = this._setButtonValue(e, this._buttonA, El.A); } /** * Gets the value of the `B` button */ get buttonB() { return this._buttonB; } /** * Sets the value of the `B` button */ set buttonB(e) { this._buttonB = this._setButtonValue(e, this._buttonB, El.B); } /** * Gets the value of the `X` button */ get buttonX() { return this._buttonX; } /** * Sets the value of the `X` button */ set buttonX(e) { this._buttonX = this._setButtonValue(e, this._buttonX, El.X); } /** * Gets the value of the `Y` button */ get buttonY() { return this._buttonY; } /** * Sets the value of the `Y` button */ set buttonY(e) { this._buttonY = this._setButtonValue(e, this._buttonY, El.Y); } /** * Gets the value of the `Start` button */ get buttonStart() { return this._buttonStart; } /** * Sets the value of the `Start` button */ set buttonStart(e) { this._buttonStart = this._setButtonValue(e, this._buttonStart, El.Start); } /** * Gets the value of the `Back` button */ get buttonBack() { return this._buttonBack; } /** * Sets the value of the `Back` button */ set buttonBack(e) { this._buttonBack = this._setButtonValue(e, this._buttonBack, El.Back); } /** * Gets the value of the `Left` button */ get buttonLB() { return this._buttonLB; } /** * Sets the value of the `Left` button */ set buttonLB(e) { this._buttonLB = this._setButtonValue(e, this._buttonLB, El.LB); } /** * Gets the value of the `Right` button */ get buttonRB() { return this._buttonRB; } /** * Sets the value of the `Right` button */ set buttonRB(e) { this._buttonRB = this._setButtonValue(e, this._buttonRB, El.RB); } /** * Gets the value of the Left joystick */ get buttonLeftStick() { return this._buttonLeftStick; } /** * Sets the value of the Left joystick */ set buttonLeftStick(e) { this._buttonLeftStick = this._setButtonValue(e, this._buttonLeftStick, El.LeftStick); } /** * Gets the value of the Right joystick */ get buttonRightStick() { return this._buttonRightStick; } /** * Sets the value of the Right joystick */ set buttonRightStick(e) { this._buttonRightStick = this._setButtonValue(e, this._buttonRightStick, El.RightStick); } /** * Gets the value of D-pad up */ get dPadUp() { return this._dPadUp; } /** * Sets the value of D-pad up */ set dPadUp(e) { this._dPadUp = this._setDPadValue(e, this._dPadUp, Lw.Up); } /** * Gets the value of D-pad down */ get dPadDown() { return this._dPadDown; } /** * Sets the value of D-pad down */ set dPadDown(e) { this._dPadDown = this._setDPadValue(e, this._dPadDown, Lw.Down); } /** * Gets the value of D-pad left */ get dPadLeft() { return this._dPadLeft; } /** * Sets the value of D-pad left */ set dPadLeft(e) { this._dPadLeft = this._setDPadValue(e, this._dPadLeft, Lw.Left); } /** * Gets the value of D-pad right */ get dPadRight() { return this._dPadRight; } /** * Sets the value of D-pad right */ set dPadRight(e) { this._dPadRight = this._setDPadValue(e, this._dPadRight, Lw.Right); } /** * Force the gamepad to synchronize with device values */ update() { super.update(), this._isXboxOnePad ? (this.buttonA = this.browserGamepad.buttons[0].value, this.buttonB = this.browserGamepad.buttons[1].value, this.buttonX = this.browserGamepad.buttons[2].value, this.buttonY = this.browserGamepad.buttons[3].value, this.buttonLB = this.browserGamepad.buttons[4].value, this.buttonRB = this.browserGamepad.buttons[5].value, this.leftTrigger = this.browserGamepad.buttons[6].value, this.rightTrigger = this.browserGamepad.buttons[7].value, this.buttonBack = this.browserGamepad.buttons[8].value, this.buttonStart = this.browserGamepad.buttons[9].value, this.buttonLeftStick = this.browserGamepad.buttons[10].value, this.buttonRightStick = this.browserGamepad.buttons[11].value, this.dPadUp = this.browserGamepad.buttons[12].value, this.dPadDown = this.browserGamepad.buttons[13].value, this.dPadLeft = this.browserGamepad.buttons[14].value, this.dPadRight = this.browserGamepad.buttons[15].value) : (this.buttonA = this.browserGamepad.buttons[0].value, this.buttonB = this.browserGamepad.buttons[1].value, this.buttonX = this.browserGamepad.buttons[2].value, this.buttonY = this.browserGamepad.buttons[3].value, this.buttonLB = this.browserGamepad.buttons[4].value, this.buttonRB = this.browserGamepad.buttons[5].value, this.leftTrigger = this.browserGamepad.buttons[6].value, this.rightTrigger = this.browserGamepad.buttons[7].value, this.buttonBack = this.browserGamepad.buttons[8].value, this.buttonStart = this.browserGamepad.buttons[9].value, this.buttonLeftStick = this.browserGamepad.buttons[10].value, this.buttonRightStick = this.browserGamepad.buttons[11].value, this.dPadUp = this.browserGamepad.buttons[12].value, this.dPadDown = this.browserGamepad.buttons[13].value, this.dPadLeft = this.browserGamepad.buttons[14].value, this.dPadRight = this.browserGamepad.buttons[15].value); } /** * Disposes the gamepad */ dispose() { super.dispose(), this.onButtonDownObservable.clear(), this.onButtonUpObservable.clear(), this.onPadDownObservable.clear(), this.onPadUpObservable.clear(); } } var dp; (function(A) { A[A.Cross = 0] = "Cross", A[A.Circle = 1] = "Circle", A[A.Square = 2] = "Square", A[A.Triangle = 3] = "Triangle", A[A.L1 = 4] = "L1", A[A.R1 = 5] = "R1", A[A.Share = 8] = "Share", A[A.Options = 9] = "Options", A[A.LeftStick = 10] = "LeftStick", A[A.RightStick = 11] = "RightStick"; })(dp || (dp = {})); var Kw; (function(A) { A[A.Up = 12] = "Up", A[A.Down = 13] = "Down", A[A.Left = 14] = "Left", A[A.Right = 15] = "Right"; })(Kw || (Kw = {})); class Bee extends N9 { /** * Creates a new DualShock gamepad object * @param id defines the id of this gamepad * @param index defines its index * @param gamepad defines the internal HTML gamepad object */ constructor(e, t, r) { super(e.replace("STANDARD GAMEPAD", "SONY PLAYSTATION DUALSHOCK"), t, r, 0, 1, 2, 3), this._leftTrigger = 0, this._rightTrigger = 0, this.onButtonDownObservable = new Oe(), this.onButtonUpObservable = new Oe(), this.onPadDownObservable = new Oe(), this.onPadUpObservable = new Oe(), this._buttonCross = 0, this._buttonCircle = 0, this._buttonSquare = 0, this._buttonTriangle = 0, this._buttonShare = 0, this._buttonOptions = 0, this._buttonL1 = 0, this._buttonR1 = 0, this._buttonLeftStick = 0, this._buttonRightStick = 0, this._dPadUp = 0, this._dPadDown = 0, this._dPadLeft = 0, this._dPadRight = 0, this.type = N9.DUALSHOCK; } /** * Defines the callback to call when left trigger is pressed * @param callback defines the callback to use */ onlefttriggerchanged(e) { this._onlefttriggerchanged = e; } /** * Defines the callback to call when right trigger is pressed * @param callback defines the callback to use */ onrighttriggerchanged(e) { this._onrighttriggerchanged = e; } /** * Gets the left trigger value */ get leftTrigger() { return this._leftTrigger; } /** * Sets the left trigger value */ set leftTrigger(e) { this._onlefttriggerchanged && this._leftTrigger !== e && this._onlefttriggerchanged(e), this._leftTrigger = e; } /** * Gets the right trigger value */ get rightTrigger() { return this._rightTrigger; } /** * Sets the right trigger value */ set rightTrigger(e) { this._onrighttriggerchanged && this._rightTrigger !== e && this._onrighttriggerchanged(e), this._rightTrigger = e; } /** * Defines the callback to call when a button is pressed * @param callback defines the callback to use */ onbuttondown(e) { this._onbuttondown = e; } /** * Defines the callback to call when a button is released * @param callback defines the callback to use */ onbuttonup(e) { this._onbuttonup = e; } /** * Defines the callback to call when a pad is pressed * @param callback defines the callback to use */ ondpaddown(e) { this._ondpaddown = e; } /** * Defines the callback to call when a pad is released * @param callback defines the callback to use */ ondpadup(e) { this._ondpadup = e; } _setButtonValue(e, t, r) { return e !== t && (e === 1 && (this._onbuttondown && this._onbuttondown(r), this.onButtonDownObservable.notifyObservers(r)), e === 0 && (this._onbuttonup && this._onbuttonup(r), this.onButtonUpObservable.notifyObservers(r))), e; } _setDPadValue(e, t, r) { return e !== t && (e === 1 && (this._ondpaddown && this._ondpaddown(r), this.onPadDownObservable.notifyObservers(r)), e === 0 && (this._ondpadup && this._ondpadup(r), this.onPadUpObservable.notifyObservers(r))), e; } /** * Gets the value of the `Cross` button */ get buttonCross() { return this._buttonCross; } /** * Sets the value of the `Cross` button */ set buttonCross(e) { this._buttonCross = this._setButtonValue(e, this._buttonCross, dp.Cross); } /** * Gets the value of the `Circle` button */ get buttonCircle() { return this._buttonCircle; } /** * Sets the value of the `Circle` button */ set buttonCircle(e) { this._buttonCircle = this._setButtonValue(e, this._buttonCircle, dp.Circle); } /** * Gets the value of the `Square` button */ get buttonSquare() { return this._buttonSquare; } /** * Sets the value of the `Square` button */ set buttonSquare(e) { this._buttonSquare = this._setButtonValue(e, this._buttonSquare, dp.Square); } /** * Gets the value of the `Triangle` button */ get buttonTriangle() { return this._buttonTriangle; } /** * Sets the value of the `Triangle` button */ set buttonTriangle(e) { this._buttonTriangle = this._setButtonValue(e, this._buttonTriangle, dp.Triangle); } /** * Gets the value of the `Options` button */ get buttonOptions() { return this._buttonOptions; } /** * Sets the value of the `Options` button */ set buttonOptions(e) { this._buttonOptions = this._setButtonValue(e, this._buttonOptions, dp.Options); } /** * Gets the value of the `Share` button */ get buttonShare() { return this._buttonShare; } /** * Sets the value of the `Share` button */ set buttonShare(e) { this._buttonShare = this._setButtonValue(e, this._buttonShare, dp.Share); } /** * Gets the value of the `L1` button */ get buttonL1() { return this._buttonL1; } /** * Sets the value of the `L1` button */ set buttonL1(e) { this._buttonL1 = this._setButtonValue(e, this._buttonL1, dp.L1); } /** * Gets the value of the `R1` button */ get buttonR1() { return this._buttonR1; } /** * Sets the value of the `R1` button */ set buttonR1(e) { this._buttonR1 = this._setButtonValue(e, this._buttonR1, dp.R1); } /** * Gets the value of the Left joystick */ get buttonLeftStick() { return this._buttonLeftStick; } /** * Sets the value of the Left joystick */ set buttonLeftStick(e) { this._buttonLeftStick = this._setButtonValue(e, this._buttonLeftStick, dp.LeftStick); } /** * Gets the value of the Right joystick */ get buttonRightStick() { return this._buttonRightStick; } /** * Sets the value of the Right joystick */ set buttonRightStick(e) { this._buttonRightStick = this._setButtonValue(e, this._buttonRightStick, dp.RightStick); } /** * Gets the value of D-pad up */ get dPadUp() { return this._dPadUp; } /** * Sets the value of D-pad up */ set dPadUp(e) { this._dPadUp = this._setDPadValue(e, this._dPadUp, Kw.Up); } /** * Gets the value of D-pad down */ get dPadDown() { return this._dPadDown; } /** * Sets the value of D-pad down */ set dPadDown(e) { this._dPadDown = this._setDPadValue(e, this._dPadDown, Kw.Down); } /** * Gets the value of D-pad left */ get dPadLeft() { return this._dPadLeft; } /** * Sets the value of D-pad left */ set dPadLeft(e) { this._dPadLeft = this._setDPadValue(e, this._dPadLeft, Kw.Left); } /** * Gets the value of D-pad right */ get dPadRight() { return this._dPadRight; } /** * Sets the value of D-pad right */ set dPadRight(e) { this._dPadRight = this._setDPadValue(e, this._dPadRight, Kw.Right); } /** * Force the gamepad to synchronize with device values */ update() { super.update(), this.buttonCross = this.browserGamepad.buttons[0].value, this.buttonCircle = this.browserGamepad.buttons[1].value, this.buttonSquare = this.browserGamepad.buttons[2].value, this.buttonTriangle = this.browserGamepad.buttons[3].value, this.buttonL1 = this.browserGamepad.buttons[4].value, this.buttonR1 = this.browserGamepad.buttons[5].value, this.leftTrigger = this.browserGamepad.buttons[6].value, this.rightTrigger = this.browserGamepad.buttons[7].value, this.buttonShare = this.browserGamepad.buttons[8].value, this.buttonOptions = this.browserGamepad.buttons[9].value, this.buttonLeftStick = this.browserGamepad.buttons[10].value, this.buttonRightStick = this.browserGamepad.buttons[11].value, this.dPadUp = this.browserGamepad.buttons[12].value, this.dPadDown = this.browserGamepad.buttons[13].value, this.dPadLeft = this.browserGamepad.buttons[14].value, this.dPadRight = this.browserGamepad.buttons[15].value; } /** * Disposes the gamepad */ dispose() { super.dispose(), this.onButtonDownObservable.clear(), this.onButtonUpObservable.clear(), this.onPadDownObservable.clear(), this.onPadUpObservable.clear(); } } class Wee { /** * Initializes the gamepad manager * @param _scene BabylonJS scene */ constructor(e) { if (this._scene = e, this._babylonGamepads = [], this._oneGamepadConnected = !1, this._isMonitoring = !1, this.onGamepadDisconnectedObservable = new Oe(), u9() ? (this._gamepadEventSupported = "GamepadEvent" in window, this._gamepadSupport = navigator && navigator.getGamepads) : this._gamepadEventSupported = !1, this.onGamepadConnectedObservable = new Oe((t) => { for (const r in this._babylonGamepads) { const n = this._babylonGamepads[r]; n && n._isConnected && this.onGamepadConnectedObservable.notifyObserver(t, n); } }), this._onGamepadConnectedEvent = (t) => { const r = t.gamepad; if (r.index in this._babylonGamepads && this._babylonGamepads[r.index].isConnected) return; let n; this._babylonGamepads[r.index] ? (n = this._babylonGamepads[r.index], n.browserGamepad = r, n._isConnected = !0) : n = this._addNewGamepad(r), this.onGamepadConnectedObservable.notifyObservers(n), this._startMonitoringGamepads(); }, this._onGamepadDisconnectedEvent = (t) => { const r = t.gamepad; for (const n in this._babylonGamepads) if (this._babylonGamepads[n].index === r.index) { const i = this._babylonGamepads[n]; i._isConnected = !1, this.onGamepadDisconnectedObservable.notifyObservers(i), i.dispose && i.dispose(); break; } }, this._gamepadSupport) if (this._updateGamepadObjects(), this._babylonGamepads.length && this._startMonitoringGamepads(), this._gamepadEventSupported) { const t = this._scene ? this._scene.getEngine().getHostWindow() : window; t && (t.addEventListener("gamepadconnected", this._onGamepadConnectedEvent, !1), t.addEventListener("gamepaddisconnected", this._onGamepadDisconnectedEvent, !1)); } else this._startMonitoringGamepads(); } /** * The gamepads in the game pad manager */ get gamepads() { return this._babylonGamepads; } /** * Get the gamepad controllers based on type * @param type The type of gamepad controller * @returns Nullable gamepad */ getGamepadByType(e = N9.XBOX) { for (const t of this._babylonGamepads) if (t && t.type === e) return t; return null; } /** * Disposes the gamepad manager */ dispose() { this._gamepadEventSupported && (this._onGamepadConnectedEvent && window.removeEventListener("gamepadconnected", this._onGamepadConnectedEvent), this._onGamepadDisconnectedEvent && window.removeEventListener("gamepaddisconnected", this._onGamepadDisconnectedEvent), this._onGamepadConnectedEvent = null, this._onGamepadDisconnectedEvent = null), this._babylonGamepads.forEach((e) => { e.dispose(); }), this.onGamepadConnectedObservable.clear(), this.onGamepadDisconnectedObservable.clear(), this._oneGamepadConnected = !1, this._stopMonitoringGamepads(), this._babylonGamepads = []; } _addNewGamepad(e) { this._oneGamepadConnected || (this._oneGamepadConnected = !0); let t; const r = e.id.search("054c") !== -1 && e.id.search("0ce6") === -1, n = e.id.search("Xbox One") !== -1; return n || e.id.search("Xbox 360") !== -1 || e.id.search("xinput") !== -1 || e.id.search("045e") !== -1 && e.id.search("Surface Dock") === -1 ? t = new mee(e.id, e.index, e, n) : r ? t = new Bee(e.id, e.index, e) : t = new xee(e.id, e.index, e), this._babylonGamepads[t.index] = t, t; } _startMonitoringGamepads() { this._isMonitoring || (this._isMonitoring = !0, this._checkGamepadsStatus()); } _stopMonitoringGamepads() { this._isMonitoring = !1; } /** @internal */ _checkGamepadsStatus() { this._updateGamepadObjects(); for (const e in this._babylonGamepads) { const t = this._babylonGamepads[e]; if (!(!t || !t.isConnected)) try { t.update(); } catch { this._loggedErrors.indexOf(t.index) === -1 && (ye.Warn(`Error updating gamepad ${t.id}`), this._loggedErrors.push(t.index)); } } this._isMonitoring && Ge.QueueNewFrame(() => { this._checkGamepadsStatus(); }); } // This function is called only on Chrome, which does not properly support // connection/disconnection events and forces you to recopy again the gamepad object _updateGamepadObjects() { const e = navigator.getGamepads ? navigator.getGamepads() : []; for (let t = 0; t < e.length; t++) { const r = e[t]; if (r) if (this._babylonGamepads[r.index]) this._babylonGamepads[t].browserGamepad = r, this._babylonGamepads[t].isConnected || (this._babylonGamepads[t]._isConnected = !0, this.onGamepadConnectedObservable.notifyObservers(this._babylonGamepads[t])); else { const n = this._addNewGamepad(r); this.onGamepadConnectedObservable.notifyObservers(n); } } } } Object.defineProperty(sr.prototype, "gamepadManager", { get: function() { if (!this._gamepadManager) { this._gamepadManager = new Wee(this); let A = this._getComponent(Ot.NAME_GAMEPAD); A || (A = new See(this), this._addComponent(A)); } return this._gamepadManager; }, enumerable: !0, configurable: !0 }); kR.prototype.addGamepad = function() { return this.add(new ER()), this; }; CO.prototype.addGamepad = function() { return this.add(new VR()), this; }; class See { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_GAMEPAD, this.scene = e; } /** * Registers the component in a given scene */ register() { this.scene._beforeCameraUpdateStage.registerStep(Ot.STEP_BEFORECAMERAUPDATE_GAMEPAD, this, this._beforeCameraUpdate); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { } /** * Disposes the component and the associated resources */ dispose() { const e = this.scene._gamepadManager; e && (e.dispose(), this.scene._gamepadManager = null); } _beforeCameraUpdate() { const e = this.scene._gamepadManager; e && e._isMonitoring && e._checkGamepadsStatus(); } } Cs.AddNodeConstructor("FreeCamera", (A, e) => () => new bD(A, S.Zero(), e)); class bD extends pQ { /** * Defines the gamepad rotation sensibility. * This is the threshold from when rotation starts to be accounted for to prevent jittering. */ get gamepadAngularSensibility() { const e = this.inputs.attached.gamepad; return e ? e.gamepadAngularSensibility : 0; } set gamepadAngularSensibility(e) { const t = this.inputs.attached.gamepad; t && (t.gamepadAngularSensibility = e); } /** * Defines the gamepad move sensibility. * This is the threshold from when moving starts to be accounted for to prevent jittering. */ get gamepadMoveSensibility() { const e = this.inputs.attached.gamepad; return e ? e.gamepadMoveSensibility : 0; } set gamepadMoveSensibility(e) { const t = this.inputs.attached.gamepad; t && (t.gamepadMoveSensibility = e); } /** * The Universal Camera is the one to choose for first person shooter type games, and works with all the keyboard, mouse, touch and gamepads. This replaces the earlier Free Camera, * which still works and will still be found in many Playgrounds. * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera * @param name Define the name of the camera in the scene * @param position Define the start position of the camera in the scene * @param scene Define the scene the camera belongs to */ constructor(e, t, r) { super(e, t, r), this.inputs.addGamepad(); } /** * Gets the current object class name. * @returns the class name */ getClassName() { return "UniversalCamera"; } } Tr._CreateDefaultParsedCamera = (A, e) => new bD(A, S.Zero(), e); Cs.AddNodeConstructor("GamepadCamera", (A, e) => () => new yO(A, S.Zero(), e)); class yO extends bD { /** * Instantiates a new Gamepad Camera * This represents a FPS type of camera. This is only here for back compat purpose. * Please use the UniversalCamera instead as both are identical. * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera * @param name Define the name of the camera in the scene * @param position Define the start position of the camera in the scene * @param scene Define the scene the camera belongs to */ constructor(e, t, r) { super(e, t, r); } /** * Gets the current object class name. * @returns the class name */ getClassName() { return "GamepadCamera"; } } const Q6e = "passCubePixelShader", Y6e = `varying vec2 vUV;uniform samplerCube textureSampler; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {vec2 uv=vUV*2.0-1.0; #ifdef POSITIVEX gl_FragColor=textureCube(textureSampler,vec3(1.001,uv.y,uv.x)); #endif #ifdef NEGATIVEX gl_FragColor=textureCube(textureSampler,vec3(-1.001,uv.y,uv.x)); #endif #ifdef POSITIVEY gl_FragColor=textureCube(textureSampler,vec3(uv.y,1.001,uv.x)); #endif #ifdef NEGATIVEY gl_FragColor=textureCube(textureSampler,vec3(uv.y,-1.001,uv.x)); #endif #ifdef POSITIVEZ gl_FragColor=textureCube(textureSampler,vec3(uv,1.001)); #endif #ifdef NEGATIVEZ gl_FragColor=textureCube(textureSampler,vec3(uv,-1.001)); #endif }`; Le.ShadersStore[Q6e] = Y6e; class lg extends kr { /** * Gets a string identifying the name of the class * @returns "PassPostProcess" string */ getClassName() { return "PassPostProcess"; } /** * Creates the PassPostProcess * @param name The name of the effect. * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) * @param textureType The type of texture to be used when performing the post processing. * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false) */ constructor(e, t, r = null, n, i, s, a = 0, f = !1) { super(e, "pass", null, null, t, r, n, i, s, void 0, a, void 0, null, f); } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new lg(e.name, e.options, t, e.renderTargetSamplingMode, e._engine, e.reusable), e, r, n); } } Ue("BABYLON.PassPostProcess", lg); class hQ extends kr { /** * Gets or sets the cube face to display. * * 0 is +X * * 1 is -X * * 2 is +Y * * 3 is -Y * * 4 is +Z * * 5 is -Z */ get face() { return this._face; } set face(e) { if (!(e < 0 || e > 5)) switch (this._face = e, this._face) { case 0: this.updateEffect("#define POSITIVEX"); break; case 1: this.updateEffect("#define NEGATIVEX"); break; case 2: this.updateEffect("#define POSITIVEY"); break; case 3: this.updateEffect("#define NEGATIVEY"); break; case 4: this.updateEffect("#define POSITIVEZ"); break; case 5: this.updateEffect("#define NEGATIVEZ"); break; } } /** * Gets a string identifying the name of the class * @returns "PassCubePostProcess" string */ getClassName() { return "PassCubePostProcess"; } /** * Creates the PassCubePostProcess * @param name The name of the effect. * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) * @param textureType The type of texture to be used when performing the post processing. * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false) */ constructor(e, t, r = null, n, i, s, a = 0, f = !1) { super(e, "passCube", null, null, t, r, n, i, s, "#define POSITIVEX", a, void 0, null, f), this._face = 0; } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new hQ(e.name, e.options, t, e.renderTargetSamplingMode, e._engine, e.reusable), e, r, n); } } Ge._RescalePostProcessFactory = (A) => new lg("rescale", 1, null, 2, A, !1, 0); const M6e = "anaglyphPixelShader", L6e = `varying vec2 vUV;uniform sampler2D textureSampler;uniform sampler2D leftSampler; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {vec4 leftFrag=texture2D(leftSampler,vUV);leftFrag=vec4(1.0,leftFrag.g,leftFrag.b,1.0);vec4 rightFrag=texture2D(textureSampler,vUV);rightFrag=vec4(rightFrag.r,1.0,1.0,1.0);gl_FragColor=vec4(rightFrag.rgb*leftFrag.rgb,1.0);}`; Le.ShadersStore[M6e] = L6e; class HQ extends kr { /** * Gets a string identifying the name of the class * @returns "AnaglyphPostProcess" string */ getClassName() { return "AnaglyphPostProcess"; } /** * Creates a new AnaglyphPostProcess * @param name defines postprocess name * @param options defines creation options or target ratio scale * @param rigCameras defines cameras using this postprocess * @param samplingMode defines required sampling mode (BABYLON.Texture.NEAREST_SAMPLINGMODE by default) * @param engine defines hosting engine * @param reusable defines if the postprocess will be reused multiple times per frame */ constructor(e, t, r, n, i, s) { super(e, "anaglyph", null, ["leftSampler"], t, r[1], n, i, s), this._passedProcess = r[0]._rigPostProcess, this.onApplyObservable.add((a) => { a.setTextureFromPostProcess("leftSampler", this._passedProcess); }); } } Ue("BABYLON.AnaglyphPostProcess", HQ); function NR(A) { A._rigCameras[0]._rigPostProcess = new lg(A.name + "_passthru", 1, A._rigCameras[0]), A._rigCameras[1]._rigPostProcess = new HQ(A.name + "_anaglyph", 1, A._rigCameras); } Cs.AddNodeConstructor("AnaglyphArcRotateCamera", (A, e, t) => () => new Uee(A, 0, 0, 1, S.Zero(), t.interaxial_distance, e)); class Uee extends ps { /** * Creates a new AnaglyphArcRotateCamera * @param name defines camera name * @param alpha defines alpha angle (in radians) * @param beta defines beta angle (in radians) * @param radius defines radius * @param target defines camera target * @param interaxialDistance defines distance between each color axis * @param scene defines the hosting scene */ constructor(e, t, r, n, i, s, a) { super(e, t, r, n, i, a), this._setRigMode = () => NR(this), this.interaxialDistance = s, this.setCameraRigMode(Tr.RIG_MODE_STEREOSCOPIC_ANAGLYPH, { interaxialDistance: s }); } /** * Gets camera class name * @returns AnaglyphArcRotateCamera */ getClassName() { return "AnaglyphArcRotateCamera"; } } Cs.AddNodeConstructor("AnaglyphFreeCamera", (A, e, t) => () => new Iee(A, S.Zero(), t.interaxial_distance, e)); class Iee extends SA { /** * Creates a new AnaglyphFreeCamera * @param name defines camera name * @param position defines initial position * @param interaxialDistance defines distance between each color axis * @param scene defines the hosting scene */ constructor(e, t, r, n) { super(e, t, n), this._setRigMode = () => NR(this), this.interaxialDistance = r, this.setCameraRigMode(Tr.RIG_MODE_STEREOSCOPIC_ANAGLYPH, { interaxialDistance: r }); } /** * Gets camera class name * @returns AnaglyphFreeCamera */ getClassName() { return "AnaglyphFreeCamera"; } } Cs.AddNodeConstructor("AnaglyphGamepadCamera", (A, e, t) => () => new Ree(A, S.Zero(), t.interaxial_distance, e)); class Ree extends yO { /** * Creates a new AnaglyphGamepadCamera * @param name defines camera name * @param position defines initial position * @param interaxialDistance defines distance between each color axis * @param scene defines the hosting scene */ constructor(e, t, r, n) { super(e, t, n), this._setRigMode = () => NR(this), this.interaxialDistance = r, this.setCameraRigMode(Tr.RIG_MODE_STEREOSCOPIC_ANAGLYPH, { interaxialDistance: r }); } /** * Gets camera class name * @returns AnaglyphGamepadCamera */ getClassName() { return "AnaglyphGamepadCamera"; } } Cs.AddNodeConstructor("AnaglyphUniversalCamera", (A, e, t) => () => new Vee(A, S.Zero(), t.interaxial_distance, e)); class Vee extends bD { /** * Creates a new AnaglyphUniversalCamera * @param name defines camera name * @param position defines initial position * @param interaxialDistance defines distance between each color axis * @param scene defines the hosting scene */ constructor(e, t, r, n) { super(e, t, n), this._setRigMode = () => NR(this), this.interaxialDistance = r, this.setCameraRigMode(Tr.RIG_MODE_STEREOSCOPIC_ANAGLYPH, { interaxialDistance: r }); } /** * Gets camera class name * @returns AnaglyphUniversalCamera */ getClassName() { return "AnaglyphUniversalCamera"; } } const K6e = "stereoscopicInterlacePixelShader", J6e = `const vec3 TWO=vec3(2.0,2.0,2.0);varying vec2 vUV;uniform sampler2D camASampler;uniform sampler2D textureSampler;uniform vec2 stepSize; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {bool useCamA;bool useCamB;vec2 texCoord1;vec2 texCoord2;vec3 frag1;vec3 frag2; #ifdef IS_STEREOSCOPIC_HORIZ useCamB=vUV.x>0.5;useCamA=!useCamB;texCoord1=vec2(useCamB ? (vUV.x-0.5)*2.0 : vUV.x*2.0,vUV.y);texCoord2=vec2(texCoord1.x+stepSize.x,vUV.y); #else #ifdef IS_STEREOSCOPIC_INTERLACED float rowNum=floor(vUV.y/stepSize.y);useCamA=mod(rowNum,2.0)==1.0;useCamB=mod(rowNum,2.0)==0.0;texCoord1=vec2(vUV.x,vUV.y);texCoord2=vec2(vUV.x,vUV.y); #else useCamB=vUV.y>0.5;useCamA=!useCamB;texCoord1=vec2(vUV.x,useCamB ? (vUV.y-0.5)*2.0 : vUV.y*2.0);texCoord2=vec2(vUV.x,texCoord1.y+stepSize.y); #endif #endif if (useCamB){frag1=texture2D(textureSampler,texCoord1).rgb;frag2=texture2D(textureSampler,texCoord2).rgb;}else if (useCamA){frag1=texture2D(camASampler ,texCoord1).rgb;frag2=texture2D(camASampler ,texCoord2).rgb;}else {discard;} gl_FragColor=vec4((frag1+frag2)/TWO,1.0);} `; Le.ShadersStore[K6e] = J6e; class Cee extends kr { /** * Gets a string identifying the name of the class * @returns "StereoscopicInterlacePostProcessI" string */ getClassName() { return "StereoscopicInterlacePostProcessI"; } /** * Initializes a StereoscopicInterlacePostProcessI * @param name The name of the effect. * @param rigCameras The rig cameras to be applied to the post process * @param isStereoscopicHoriz If the rendered results are horizontal or vertical * @param isStereoscopicInterlaced If the rendered results are alternate line interlaced * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) */ constructor(e, t, r, n, i, s, a) { super(e, "stereoscopicInterlace", ["stepSize"], ["camASampler"], 1, t[1], i, s, a, n ? "#define IS_STEREOSCOPIC_INTERLACED 1" : r ? "#define IS_STEREOSCOPIC_HORIZ 1" : void 0), this._passedProcess = t[0]._rigPostProcess, this._stepSize = new at(1 / this.width, 1 / this.height), this.onSizeChangedObservable.add(() => { this._stepSize = new at(1 / this.width, 1 / this.height); }), this.onApplyObservable.add((f) => { f.setTextureFromPostProcess("camASampler", this._passedProcess), f.setFloat2("stepSize", this._stepSize.x, this._stepSize.y); }); } } class z6e extends kr { /** * Gets a string identifying the name of the class * @returns "StereoscopicInterlacePostProcess" string */ getClassName() { return "StereoscopicInterlacePostProcess"; } /** * Initializes a StereoscopicInterlacePostProcess * @param name The name of the effect. * @param rigCameras The rig cameras to be applied to the post process * @param isStereoscopicHoriz If the rendered results are horizontal or vertical * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) */ constructor(e, t, r, n, i, s) { super(e, "stereoscopicInterlace", ["stepSize"], ["camASampler"], 1, t[1], n, i, s, r ? "#define IS_STEREOSCOPIC_HORIZ 1" : void 0), this._passedProcess = t[0]._rigPostProcess, this._stepSize = new at(1 / this.width, 1 / this.height), this.onSizeChangedObservable.add(() => { this._stepSize = new at(1 / this.width, 1 / this.height); }), this.onApplyObservable.add((a) => { a.setTextureFromPostProcess("camASampler", this._passedProcess), a.setFloat2("stepSize", this._stepSize.x, this._stepSize.y); }); } } function QR(A) { const e = A.cameraRigMode === Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL || A.cameraRigMode === Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED, t = A.cameraRigMode === Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED; A.cameraRigMode === Tr.RIG_MODE_STEREOSCOPIC_INTERLACED ? (A._rigCameras[0]._rigPostProcess = new lg(A.name + "_passthru", 1, A._rigCameras[0]), A._rigCameras[1]._rigPostProcess = new Cee(A.name + "_stereoInterlace", A._rigCameras, !1, !0)) : (A._rigCameras[t ? 1 : 0].viewport = new WA(0, 0, e ? 0.5 : 1, e ? 1 : 0.5), A._rigCameras[t ? 0 : 1].viewport = new WA(e ? 0.5 : 0, e ? 0 : 0.5, e ? 0.5 : 1, e ? 1 : 0.5)); } Cs.AddNodeConstructor("StereoscopicArcRotateCamera", (A, e, t) => () => new Oee(A, 0, 0, 1, S.Zero(), t.interaxial_distance, t.isStereoscopicSideBySide, e)); class Oee extends ps { /** * Creates a new StereoscopicArcRotateCamera * @param name defines camera name * @param alpha defines alpha angle (in radians) * @param beta defines beta angle (in radians) * @param radius defines radius * @param target defines camera target * @param interaxialDistance defines distance between each color axis * @param isStereoscopicSideBySide defines is stereoscopic is done side by side or over under * @param scene defines the hosting scene */ constructor(e, t, r, n, i, s, a, f) { super(e, t, r, n, i, f), this._setRigMode = () => QR(this), this.interaxialDistance = s, this.isStereoscopicSideBySide = a, this.setCameraRigMode(a ? Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL : Tr.RIG_MODE_STEREOSCOPIC_OVERUNDER, { interaxialDistance: s }); } /** * Gets camera class name * @returns StereoscopicArcRotateCamera */ getClassName() { return "StereoscopicArcRotateCamera"; } } Cs.AddNodeConstructor("StereoscopicFreeCamera", (A, e, t) => () => new yee(A, S.Zero(), t.interaxial_distance, t.isStereoscopicSideBySide, e)); class yee extends SA { /** * Creates a new StereoscopicFreeCamera * @param name defines camera name * @param position defines initial position * @param interaxialDistance defines distance between each color axis * @param isStereoscopicSideBySide defines is stereoscopic is done side by side or over under * @param scene defines the hosting scene */ constructor(e, t, r, n, i) { super(e, t, i), this._setRigMode = () => QR(this), this.interaxialDistance = r, this.isStereoscopicSideBySide = n, this.setCameraRigMode(n ? Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL : Tr.RIG_MODE_STEREOSCOPIC_OVERUNDER, { interaxialDistance: r }); } /** * Gets camera class name * @returns StereoscopicFreeCamera */ getClassName() { return "StereoscopicFreeCamera"; } } Cs.AddNodeConstructor("StereoscopicGamepadCamera", (A, e, t) => () => new kee(A, S.Zero(), t.interaxial_distance, t.isStereoscopicSideBySide, e)); class kee extends yO { /** * Creates a new StereoscopicGamepadCamera * @param name defines camera name * @param position defines initial position * @param interaxialDistance defines distance between each color axis * @param isStereoscopicSideBySide defines is stereoscopic is done side by side or over under * @param scene defines the hosting scene */ constructor(e, t, r, n, i) { super(e, t, i), this._setRigMode = () => QR(this), this.interaxialDistance = r, this.isStereoscopicSideBySide = n, this.setCameraRigMode(n ? Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL : Tr.RIG_MODE_STEREOSCOPIC_OVERUNDER, { interaxialDistance: r }); } /** * Gets camera class name * @returns StereoscopicGamepadCamera */ getClassName() { return "StereoscopicGamepadCamera"; } } Cs.AddNodeConstructor("StereoscopicFreeCamera", (A, e, t) => () => new Eee(A, S.Zero(), t.interaxial_distance, t.isStereoscopicSideBySide, e)); class Eee extends bD { /** * Creates a new StereoscopicUniversalCamera * @param name defines camera name * @param position defines initial position * @param interaxialDistance defines distance between each color axis * @param isStereoscopicSideBySide defines is stereoscopic is done side by side or over under * @param scene defines the hosting scene */ constructor(e, t, r, n, i) { super(e, t, i), this._setRigMode = () => QR(this), this.interaxialDistance = r, this.isStereoscopicSideBySide = n, this.setCameraRigMode(n ? Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL : Tr.RIG_MODE_STEREOSCOPIC_OVERUNDER, { interaxialDistance: r }); } /** * Gets camera class name * @returns StereoscopicUniversalCamera */ getClassName() { return "StereoscopicUniversalCamera"; } } class G6e extends bD { set distanceBetweenEyes(e) { this._distanceBetweenEyes = e; } /** * distance between the eyes */ get distanceBetweenEyes() { return this._distanceBetweenEyes; } set distanceToProjectionPlane(e) { this._distanceToProjectionPlane = e; } /** * Distance to projection plane (should be the same units the like distance between the eyes) */ get distanceToProjectionPlane() { return this._distanceToProjectionPlane; } /** * Creates a new StereoscopicScreenUniversalCamera * @param name defines camera name * @param position defines initial position * @param scene defines the hosting scene * @param distanceToProjectionPlane defines distance between each color axis. The rig cameras will receive this as their negative z position! * @param distanceBetweenEyes defines is stereoscopic is done side by side or over under */ constructor(e, t, r, n = 1, i = 0.065) { super(e, t, r), this._distanceBetweenEyes = i, this._distanceToProjectionPlane = n, this.setCameraRigMode(Tr.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL, { stereoHalfAngle: 0 }), this._cameraRigParams.stereoHalfAngle = 0, this._cameraRigParams.interaxialDistance = i; } /** * Gets camera class name * @returns StereoscopicScreenUniversalCamera */ getClassName() { return "StereoscopicUniversalCamera"; } /** * @internal */ createRigCamera(e) { const t = new b1(e, S.Zero(), this.getScene()), r = new Hr("tm_" + e, this.getScene()); return t.parent = r, r.setPivotMatrix(he.Identity(), !1), t.isRigCamera = !0, t.rigParent = this, t; } /** * @internal */ _updateRigCameras() { for (let e = 0; e < this._rigCameras.length; e++) { const t = this._rigCameras[e]; t.minZ = this.minZ, t.maxZ = this.maxZ, t.fov = this.fov, t.upVector.copyFrom(this.upVector), t.rotationQuaternion ? t.rotationQuaternion.copyFrom(this.rotationQuaternion) : t.rotation.copyFrom(this.rotation), this._updateCamera(this._rigCameras[e], e); } } _updateCamera(e, t) { const r = this.distanceBetweenEyes / 2, n = r / this.distanceToProjectionPlane; e.position.copyFrom(this.position), e.position.addInPlaceFromFloats(t === 0 ? -r : r, 0, -this._distanceToProjectionPlane); const i = e.parent, s = i.getPivotMatrix(); s.setTranslationFromFloats(t === 0 ? r : -r, 0, 0), s.setRowFromFloats(2, t === 0 ? n : -n, 0, 1, 0), i.setPivotMatrix(s, !1); } _setRigMode() { this._rigCameras[0].viewport = new WA(0, 0, 0.5, 1), this._rigCameras[1].viewport = new WA(0.5, 0, 0.5, 1); for (let e = 0; e < this._rigCameras.length; e++) this._updateCamera(this._rigCameras[e], e); } } Cs.AddNodeConstructor("VirtualJoysticksCamera", (A, e) => () => new Fee(A, S.Zero(), e)); class Fee extends SA { /** * Instantiates a VirtualJoysticksCamera. It can be useful in First Person Shooter game for instance. * It is identical to the Free Camera and simply adds by default a virtual joystick. * Virtual Joysticks are on-screen 2D graphics that are used to control the camera or other scene items. * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#virtual-joysticks-camera * @param name Define the name of the camera in the scene * @param position Define the start position of the camera in the scene * @param scene Define the scene the camera belongs to */ constructor(e, t, r) { super(e, t, r), this.inputs.addVirtualJoystick(); } /** * Gets the current object class name. * @returns the class name */ getClassName() { return "VirtualJoysticksCamera"; } } class Bm { constructor() { this.compensateDistortion = !0, this.multiviewEnabled = !1; } /** * Gets the rendering aspect ratio based on the provided resolutions. */ get aspectRatio() { return this.hResolution / (2 * this.vResolution); } /** * Gets the aspect ratio based on the FOV, scale factors, and real screen sizes. */ get aspectRatioFov() { return 2 * Math.atan(this.postProcessScaleFactor * this.vScreenSize / (2 * this.eyeToScreenDistance)); } /** * @internal */ get leftHMatrix() { const t = 4 * (this.hScreenSize / 4 - this.lensSeparationDistance / 2) / this.hScreenSize; return he.Translation(t, 0, 0); } /** * @internal */ get rightHMatrix() { const t = 4 * (this.hScreenSize / 4 - this.lensSeparationDistance / 2) / this.hScreenSize; return he.Translation(-t, 0, 0); } /** * @internal */ get leftPreViewMatrix() { return he.Translation(0.5 * this.interpupillaryDistance, 0, 0); } /** * @internal */ get rightPreViewMatrix() { return he.Translation(-0.5 * this.interpupillaryDistance, 0, 0); } /** * Get the default VRMetrics based on the most generic setup. * @returns the default vr metrics */ static GetDefault() { const e = new Bm(); return e.hResolution = 1280, e.vResolution = 800, e.hScreenSize = 0.149759993, e.vScreenSize = 0.0935999975, e.vScreenCenter = 0.0467999987, e.eyeToScreenDistance = 0.0410000011, e.lensSeparationDistance = 0.063500002, e.interpupillaryDistance = 0.064000003, e.distortionK = [1, 0.219999999, 0.239999995, 0], e.chromaAbCorrection = [0.995999992, -0.00400000019, 1.01400006, 0], e.postProcessScaleFactor = 1.714605507808412, e.lensCenterOffset = 0.151976421, e; } } const Z6e = "vrDistortionCorrectionPixelShader", _6e = `varying vec2 vUV;uniform sampler2D textureSampler;uniform vec2 LensCenter;uniform vec2 Scale;uniform vec2 ScaleIn;uniform vec4 HmdWarpParam;vec2 HmdWarp(vec2 in01) {vec2 theta=(in01-LensCenter)*ScaleIn; float rSq=theta.x*theta.x+theta.y*theta.y;vec2 rvector=theta*(HmdWarpParam.x+HmdWarpParam.y*rSq+HmdWarpParam.z*rSq*rSq+HmdWarpParam.w*rSq*rSq*rSq);return LensCenter+Scale*rvector;} #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {vec2 tc=HmdWarp(vUV);if (tc.x <0.0 || tc.x>1.0 || tc.y<0.0 || tc.y>1.0) gl_FragColor=vec4(0.0,0.0,0.0,0.0);else{gl_FragColor=texture2D(textureSampler,tc);}}`; Le.ShadersStore[Z6e] = _6e; class _5 extends kr { /** * Gets a string identifying the name of the class * @returns "VRDistortionCorrectionPostProcess" string */ getClassName() { return "VRDistortionCorrectionPostProcess"; } /** * Initializes the VRDistortionCorrectionPostProcess * @param name The name of the effect. * @param camera The camera to apply the render pass to. * @param isRightEye If this is for the right eye distortion * @param vrMetrics All the required metrics for the VR camera */ constructor(e, t, r, n) { super(e, "vrDistortionCorrection", ["LensCenter", "Scale", "ScaleIn", "HmdWarpParam"], null, n.postProcessScaleFactor, t, We.BILINEAR_SAMPLINGMODE), this._isRightEye = r, this._distortionFactors = n.distortionK, this._postProcessScaleFactor = n.postProcessScaleFactor, this._lensCenterOffset = n.lensCenterOffset, this.adaptScaleToCurrentViewport = !0, this.onSizeChangedObservable.add(() => { this._scaleIn = new at(2, 2 / this.aspectRatio), this._scaleFactor = new at(0.5 * (1 / this._postProcessScaleFactor), 0.5 * (1 / this._postProcessScaleFactor) * this.aspectRatio), this._lensCenter = new at(this._isRightEye ? 0.5 - this._lensCenterOffset * 0.5 : 0.5 + this._lensCenterOffset * 0.5, 0.5); }), this.onApplyObservable.add((i) => { i.setFloat2("LensCenter", this._lensCenter.x, this._lensCenter.y), i.setFloat2("Scale", this._scaleFactor.x, this._scaleFactor.y), i.setFloat2("ScaleIn", this._scaleIn.x, this._scaleIn.y), i.setFloat4("HmdWarpParam", this._distortionFactors[0], this._distortionFactors[1], this._distortionFactors[2], this._distortionFactors[3]); }); } } const $6e = "vrMultiviewToSingleviewPixelShader", eAe = `precision mediump sampler2DArray;varying vec2 vUV;uniform sampler2DArray multiviewSampler;uniform int imageIndex; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {gl_FragColor=texture2D(multiviewSampler,vec3(vUV,imageIndex));}`; Le.ShadersStore[$6e] = eAe; class $5 extends Ta { set samples(e) { this._samples = e; } get samples() { return this._samples; } /** * Creates a multiview render target * @param scene scene used with the render target * @param size the size of the render target (used for each view) */ constructor(e, t = 512) { super("multiview rtt", t, e, !1, !0, 0, !1, void 0, !1, !1, !0, void 0, !0), this._renderTarget = this.getScene().getEngine().createMultiviewRenderTargetTexture(this.getRenderWidth(), this.getRenderHeight()), this._texture = this._renderTarget.texture, this._texture.isMultiview = !0, this._texture.format = 5, this.samples = this._getEngine().getCaps().maxSamples || this.samples, this._texture.samples = this._samples; } /** * @internal */ _bindFrameBuffer() { this._renderTarget && this.getScene().getEngine().bindMultiviewFramebuffer(this._renderTarget); } /** * Gets the number of views the corresponding to the texture (eg. a MultiviewRenderTarget will have > 1) * @returns the view count */ getViewCount() { return 2; } } Ge.prototype.createMultiviewRenderTargetTexture = function(A, e, t, r) { const n = this._gl; if (!this.getCaps().multiview) throw "Multiview is not supported"; const i = this._createHardwareRenderTargetWrapper(!1, !1, { width: A, height: e }); i._framebuffer = n.createFramebuffer(); const s = new As(this, ri.Unknown, !0); return s.width = A, s.height = e, s.isMultiview = !0, t || (t = n.createTexture(), n.bindTexture(n.TEXTURE_2D_ARRAY, t), n.texStorage3D(n.TEXTURE_2D_ARRAY, 1, n.RGBA8, A, e, 2)), i._colorTextureArray = t, r || (r = n.createTexture(), n.bindTexture(n.TEXTURE_2D_ARRAY, r), n.texStorage3D(n.TEXTURE_2D_ARRAY, 1, n.DEPTH24_STENCIL8, A, e, 2)), i._depthStencilTextureArray = r, s.isReady = !0, i.setTextures(s), i._depthStencilTexture = s, i; }; Ge.prototype.bindMultiviewFramebuffer = function(A) { const e = A, t = this._gl, r = this.getCaps().oculusMultiview || this.getCaps().multiview; if (this.bindFramebuffer(e, void 0, void 0, void 0, !0), t.bindFramebuffer(t.DRAW_FRAMEBUFFER, e._framebuffer), e._colorTextureArray && e._depthStencilTextureArray) this.getCaps().oculusMultiview ? (r.framebufferTextureMultisampleMultiviewOVR(t.DRAW_FRAMEBUFFER, t.COLOR_ATTACHMENT0, e._colorTextureArray, 0, e.samples, 0, 2), r.framebufferTextureMultisampleMultiviewOVR(t.DRAW_FRAMEBUFFER, t.DEPTH_STENCIL_ATTACHMENT, e._depthStencilTextureArray, 0, e.samples, 0, 2)) : (r.framebufferTextureMultiviewOVR(t.DRAW_FRAMEBUFFER, t.COLOR_ATTACHMENT0, e._colorTextureArray, 0, 0, 2), r.framebufferTextureMultiviewOVR(t.DRAW_FRAMEBUFFER, t.DEPTH_STENCIL_ATTACHMENT, e._depthStencilTextureArray, 0, 0, 2)); else throw "Invalid multiview frame buffer"; }; Ge.prototype.bindSpaceWarpFramebuffer = function(A) { const e = A, t = this._gl, r = this.getCaps().oculusMultiview || this.getCaps().multiview; if (this.bindFramebuffer(e, void 0, void 0, void 0, !0), t.bindFramebuffer(t.DRAW_FRAMEBUFFER, e._framebuffer), e._colorTextureArray && e._depthStencilTextureArray) r.framebufferTextureMultiviewOVR(t.DRAW_FRAMEBUFFER, t.COLOR_ATTACHMENT0, e._colorTextureArray, 0, 0, 2), r.framebufferTextureMultiviewOVR(t.DRAW_FRAMEBUFFER, t.DEPTH_ATTACHMENT, e._depthStencilTextureArray, 0, 0, 2); else throw new Error("Invalid Space Warp framebuffer"); }; Tr.prototype._useMultiviewToSingleView = !1; Tr.prototype._multiviewTexture = null; Tr.prototype._resizeOrCreateMultiviewTexture = function(A, e) { this._multiviewTexture ? (this._multiviewTexture.getRenderWidth() != A || this._multiviewTexture.getRenderHeight() != e) && (this._multiviewTexture.dispose(), this._multiviewTexture = new $5(this.getScene(), { width: A, height: e })) : this._multiviewTexture = new $5(this.getScene(), { width: A, height: e }); }; function Nee(A, e) { const t = new yr(A, void 0, !0, e); return t.addUniform("viewProjection", 16), t.addUniform("viewProjectionR", 16), t.addUniform("view", 16), t.addUniform("projection", 16), t.addUniform("vEyePosition", 4), t; } const tAe = sr.prototype.createSceneUniformBuffer; sr.prototype._transformMatrixR = he.Zero(); sr.prototype._multiviewSceneUbo = null; sr.prototype._createMultiviewUbo = function() { this._multiviewSceneUbo = Nee(this.getEngine(), "scene_multiview"); }; sr.prototype.createSceneUniformBuffer = function(A) { return this._multiviewSceneUbo ? Nee(this.getEngine(), A) : tAe.bind(this)(A); }; sr.prototype._updateMultiviewUbo = function(A, e) { A && e && A.multiplyToRef(e, this._transformMatrixR), A && e && (A.multiplyToRef(e, ue.Matrix[0]), Xc.GetRightPlaneToRef(ue.Matrix[0], this._frustumPlanes[3])), this._multiviewSceneUbo && (this._multiviewSceneUbo.updateMatrix("viewProjection", this.getTransformMatrix()), this._multiviewSceneUbo.updateMatrix("viewProjectionR", this._transformMatrixR), this._multiviewSceneUbo.updateMatrix("view", this._viewMatrix), this._multiviewSceneUbo.updateMatrix("projection", this._projectionMatrix)); }; sr.prototype._renderMultiviewToSingleView = function(A) { A._resizeOrCreateMultiviewTexture(A._rigPostProcess && A._rigPostProcess && A._rigPostProcess.width > 0 ? A._rigPostProcess.width : this.getEngine().getRenderWidth(!0), A._rigPostProcess && A._rigPostProcess && A._rigPostProcess.height > 0 ? A._rigPostProcess.height : this.getEngine().getRenderHeight(!0)), this._multiviewSceneUbo || this._createMultiviewUbo(), A.outputRenderTarget = A._multiviewTexture, this._renderForCamera(A), A.outputRenderTarget = null; for (let e = 0; e < A._rigCameras.length; e++) { const t = this.getEngine(); this._activeCamera = A._rigCameras[e], t.setViewport(this._activeCamera.viewport), this.postProcessManager && (this.postProcessManager._prepareFrame(), this.postProcessManager._finalizeFrame(this._activeCamera.isIntermediate)); } }; class Qee extends kr { /** * Gets a string identifying the name of the class * @returns "VRMultiviewToSingleviewPostProcess" string */ getClassName() { return "VRMultiviewToSingleviewPostProcess"; } /** * Initializes a VRMultiviewToSingleview * @param name name of the post process * @param camera camera to be applied to * @param scaleFactor scaling factor to the size of the output texture */ constructor(e, t, r) { super(e, "vrMultiviewToSingleview", ["imageIndex"], ["multiviewSampler"], r, t, We.BILINEAR_SAMPLINGMODE); const n = t ?? this.getCamera(); this.onSizeChangedObservable.add(() => { }), this.onApplyObservable.add((i) => { n._scene.activeCamera && n._scene.activeCamera.isLeftCamera ? i.setInt("imageIndex", 0) : i.setInt("imageIndex", 1), i.setTexture("multiviewSampler", n._multiviewTexture); }); } } function kO(A, e) { const t = e.vrCameraMetrics || Bm.GetDefault(); A._rigCameras[0]._cameraRigParams.vrMetrics = t, A._rigCameras[0].viewport = new WA(0, 0, 0.5, 1), A._rigCameras[0]._cameraRigParams.vrWorkMatrix = new he(), A._rigCameras[0]._cameraRigParams.vrHMatrix = t.leftHMatrix, A._rigCameras[0]._cameraRigParams.vrPreViewMatrix = t.leftPreViewMatrix, A._rigCameras[0].getProjectionMatrix = A._rigCameras[0]._getVRProjectionMatrix, A._rigCameras[1]._cameraRigParams.vrMetrics = t, A._rigCameras[1].viewport = new WA(0.5, 0, 0.5, 1), A._rigCameras[1]._cameraRigParams.vrWorkMatrix = new he(), A._rigCameras[1]._cameraRigParams.vrHMatrix = t.rightHMatrix, A._rigCameras[1]._cameraRigParams.vrPreViewMatrix = t.rightPreViewMatrix, A._rigCameras[1].getProjectionMatrix = A._rigCameras[1]._getVRProjectionMatrix, t.multiviewEnabled && (A.getScene().getEngine().getCaps().multiview ? (A._useMultiviewToSingleView = !0, A._rigPostProcess = new Qee("VRMultiviewToSingleview", A, t.postProcessScaleFactor)) : (Se.Warn("Multiview is not supported, falling back to standard rendering"), t.multiviewEnabled = !1)), t.compensateDistortion && (A._rigCameras[0]._rigPostProcess = new _5("VR_Distort_Compensation_Left", A._rigCameras[0], !1, t), A._rigCameras[1]._rigPostProcess = new _5("VR_Distort_Compensation_Right", A._rigCameras[1], !0, t)); } Cs.AddNodeConstructor("VRDeviceOrientationArcRotateCamera", (A, e) => () => new Yee(A, 0, 0, 1, S.Zero(), e)); class Yee extends ps { /** * Creates a new VRDeviceOrientationArcRotateCamera * @param name defines camera name * @param alpha defines the camera rotation along the longitudinal axis * @param beta defines the camera rotation along the latitudinal axis * @param radius defines the camera distance from its target * @param target defines the camera target * @param scene defines the scene the camera belongs to * @param compensateDistortion defines if the camera needs to compensate the lens distortion * @param vrCameraMetrics defines the vr metrics associated to the camera */ constructor(e, t, r, n, i, s, a = !0, f = Bm.GetDefault()) { super(e, t, r, n, i, s), this._setRigMode = (o) => kO(this, o), f.compensateDistortion = a, this.setCameraRigMode(Tr.RIG_MODE_VR, { vrCameraMetrics: f }), this.inputs.addVRDeviceOrientation(); } /** * Gets camera class name * @returns VRDeviceOrientationArcRotateCamera */ getClassName() { return "VRDeviceOrientationArcRotateCamera"; } } Cs.AddNodeConstructor("VRDeviceOrientationFreeCamera", (A, e) => () => new EO(A, S.Zero(), e)); class EO extends OO { /** * Creates a new VRDeviceOrientationFreeCamera * @param name defines camera name * @param position defines the start position of the camera * @param scene defines the scene the camera belongs to * @param compensateDistortion defines if the camera needs to compensate the lens distortion * @param vrCameraMetrics defines the vr metrics associated to the camera */ constructor(e, t, r, n = !0, i = Bm.GetDefault()) { super(e, t, r), this._setRigMode = (s) => kO(this, s), i.compensateDistortion = n, this.setCameraRigMode(Tr.RIG_MODE_VR, { vrCameraMetrics: i }); } /** * Gets camera class name * @returns VRDeviceOrientationFreeCamera */ getClassName() { return "VRDeviceOrientationFreeCamera"; } } Cs.AddNodeConstructor("VRDeviceOrientationGamepadCamera", (A, e) => () => new Mee(A, S.Zero(), e)); class Mee extends EO { /** * Creates a new VRDeviceOrientationGamepadCamera * @param name defines camera name * @param position defines the start position of the camera * @param scene defines the scene the camera belongs to * @param compensateDistortion defines if the camera needs to compensate the lens distortion * @param vrCameraMetrics defines the vr metrics associated to the camera */ constructor(e, t, r, n = !0, i = Bm.GetDefault()) { super(e, t, r, n, i), this._setRigMode = (s) => kO(this, s), this.inputs.addGamepad(); } /** * Gets camera class name * @returns VRDeviceOrientationGamepadCamera */ getClassName() { return "VRDeviceOrientationGamepadCamera"; } } class GC { constructor() { this.previousWorldMatrices = {}, this.previousBones = {}; } /** * Add the required uniforms to the current list. * @param uniforms defines the current uniform list. */ static AddUniforms(e) { e.push("previousWorld", "previousViewProjection", "mPreviousBones"); } /** * Add the required samplers to the current list. * @param samplers defines the current sampler list. */ // eslint-disable-next-line @typescript-eslint/no-unused-vars static AddSamplers(e) { } /** * Binds the material data. * @param effect defines the effect to update * @param scene defines the scene the material belongs to. * @param mesh The mesh * @param world World matrix of this mesh * @param isFrozen Is the material frozen */ // eslint-disable-next-line @typescript-eslint/no-unused-vars bindForSubMesh(e, t, r, n, i) { if (t.prePassRenderer && t.prePassRenderer.enabled && t.prePassRenderer.currentRTisSceneRT && t.prePassRenderer.getIndex(2) !== -1) { this.previousWorldMatrices[r.uniqueId] || (this.previousWorldMatrices[r.uniqueId] = n.clone()), this.previousViewProjection || (this.previousViewProjection = t.getTransformMatrix().clone(), this.currentViewProjection = t.getTransformMatrix().clone()); const s = t.getEngine(); this.currentViewProjection.updateFlag !== t.getTransformMatrix().updateFlag ? (this._lastUpdateFrameId = s.frameId, this.previousViewProjection.copyFrom(this.currentViewProjection), this.currentViewProjection.copyFrom(t.getTransformMatrix())) : this._lastUpdateFrameId !== s.frameId && (this._lastUpdateFrameId = s.frameId, this.previousViewProjection.copyFrom(this.currentViewProjection)), e.setMatrix("previousWorld", this.previousWorldMatrices[r.uniqueId]), e.setMatrix("previousViewProjection", this.previousViewProjection), this.previousWorldMatrices[r.uniqueId] = n.clone(); } } } class Dt { /** * Are diffuse textures enabled in the application. */ static get DiffuseTextureEnabled() { return this._DiffuseTextureEnabled; } static set DiffuseTextureEnabled(e) { this._DiffuseTextureEnabled !== e && (this._DiffuseTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are detail textures enabled in the application. */ static get DetailTextureEnabled() { return this._DetailTextureEnabled; } static set DetailTextureEnabled(e) { this._DetailTextureEnabled !== e && (this._DetailTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are decal maps enabled in the application. */ static get DecalMapEnabled() { return this._DecalMapEnabled; } static set DecalMapEnabled(e) { this._DecalMapEnabled !== e && (this._DecalMapEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are ambient textures enabled in the application. */ static get AmbientTextureEnabled() { return this._AmbientTextureEnabled; } static set AmbientTextureEnabled(e) { this._AmbientTextureEnabled !== e && (this._AmbientTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are opacity textures enabled in the application. */ static get OpacityTextureEnabled() { return this._OpacityTextureEnabled; } static set OpacityTextureEnabled(e) { this._OpacityTextureEnabled !== e && (this._OpacityTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are reflection textures enabled in the application. */ static get ReflectionTextureEnabled() { return this._ReflectionTextureEnabled; } static set ReflectionTextureEnabled(e) { this._ReflectionTextureEnabled !== e && (this._ReflectionTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are emissive textures enabled in the application. */ static get EmissiveTextureEnabled() { return this._EmissiveTextureEnabled; } static set EmissiveTextureEnabled(e) { this._EmissiveTextureEnabled !== e && (this._EmissiveTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are specular textures enabled in the application. */ static get SpecularTextureEnabled() { return this._SpecularTextureEnabled; } static set SpecularTextureEnabled(e) { this._SpecularTextureEnabled !== e && (this._SpecularTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are bump textures enabled in the application. */ static get BumpTextureEnabled() { return this._BumpTextureEnabled; } static set BumpTextureEnabled(e) { this._BumpTextureEnabled !== e && (this._BumpTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are lightmap textures enabled in the application. */ static get LightmapTextureEnabled() { return this._LightmapTextureEnabled; } static set LightmapTextureEnabled(e) { this._LightmapTextureEnabled !== e && (this._LightmapTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are refraction textures enabled in the application. */ static get RefractionTextureEnabled() { return this._RefractionTextureEnabled; } static set RefractionTextureEnabled(e) { this._RefractionTextureEnabled !== e && (this._RefractionTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are color grading textures enabled in the application. */ static get ColorGradingTextureEnabled() { return this._ColorGradingTextureEnabled; } static set ColorGradingTextureEnabled(e) { this._ColorGradingTextureEnabled !== e && (this._ColorGradingTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are fresnels enabled in the application. */ static get FresnelEnabled() { return this._FresnelEnabled; } static set FresnelEnabled(e) { this._FresnelEnabled !== e && (this._FresnelEnabled = e, Ge.MarkAllMaterialsAsDirty(4)); } /** * Are clear coat textures enabled in the application. */ static get ClearCoatTextureEnabled() { return this._ClearCoatTextureEnabled; } static set ClearCoatTextureEnabled(e) { this._ClearCoatTextureEnabled !== e && (this._ClearCoatTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are clear coat bump textures enabled in the application. */ static get ClearCoatBumpTextureEnabled() { return this._ClearCoatBumpTextureEnabled; } static set ClearCoatBumpTextureEnabled(e) { this._ClearCoatBumpTextureEnabled !== e && (this._ClearCoatBumpTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are clear coat tint textures enabled in the application. */ static get ClearCoatTintTextureEnabled() { return this._ClearCoatTintTextureEnabled; } static set ClearCoatTintTextureEnabled(e) { this._ClearCoatTintTextureEnabled !== e && (this._ClearCoatTintTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are sheen textures enabled in the application. */ static get SheenTextureEnabled() { return this._SheenTextureEnabled; } static set SheenTextureEnabled(e) { this._SheenTextureEnabled !== e && (this._SheenTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are anisotropic textures enabled in the application. */ static get AnisotropicTextureEnabled() { return this._AnisotropicTextureEnabled; } static set AnisotropicTextureEnabled(e) { this._AnisotropicTextureEnabled !== e && (this._AnisotropicTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are thickness textures enabled in the application. */ static get ThicknessTextureEnabled() { return this._ThicknessTextureEnabled; } static set ThicknessTextureEnabled(e) { this._ThicknessTextureEnabled !== e && (this._ThicknessTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are refraction intensity textures enabled in the application. */ static get RefractionIntensityTextureEnabled() { return this._ThicknessTextureEnabled; } static set RefractionIntensityTextureEnabled(e) { this._RefractionIntensityTextureEnabled !== e && (this._RefractionIntensityTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are translucency intensity textures enabled in the application. */ static get TranslucencyIntensityTextureEnabled() { return this._ThicknessTextureEnabled; } static set TranslucencyIntensityTextureEnabled(e) { this._TranslucencyIntensityTextureEnabled !== e && (this._TranslucencyIntensityTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } /** * Are translucency intensity textures enabled in the application. */ static get IridescenceTextureEnabled() { return this._IridescenceTextureEnabled; } static set IridescenceTextureEnabled(e) { this._IridescenceTextureEnabled !== e && (this._IridescenceTextureEnabled = e, Ge.MarkAllMaterialsAsDirty(1)); } } Dt._DiffuseTextureEnabled = !0; Dt._DetailTextureEnabled = !0; Dt._DecalMapEnabled = !0; Dt._AmbientTextureEnabled = !0; Dt._OpacityTextureEnabled = !0; Dt._ReflectionTextureEnabled = !0; Dt._EmissiveTextureEnabled = !0; Dt._SpecularTextureEnabled = !0; Dt._BumpTextureEnabled = !0; Dt._LightmapTextureEnabled = !0; Dt._RefractionTextureEnabled = !0; Dt._ColorGradingTextureEnabled = !0; Dt._FresnelEnabled = !0; Dt._ClearCoatTextureEnabled = !0; Dt._ClearCoatBumpTextureEnabled = !0; Dt._ClearCoatTintTextureEnabled = !0; Dt._SheenTextureEnabled = !0; Dt._AnisotropicTextureEnabled = !0; Dt._ThicknessTextureEnabled = !0; Dt._RefractionIntensityTextureEnabled = !0; Dt._TranslucencyIntensityTextureEnabled = !0; Dt._IridescenceTextureEnabled = !0; const rAe = "decalFragmentDeclaration", nAe = `#ifdef DECAL uniform vec4 vDecalInfos; #endif `; Le.IncludesShadersStore[rAe] = nAe; const iAe = "defaultFragmentDeclaration", sAe = `uniform vec4 vEyePosition;uniform vec4 vDiffuseColor; #ifdef SPECULARTERM uniform vec4 vSpecularColor; #endif uniform vec3 vEmissiveColor;uniform vec3 vAmbientColor;uniform float visibility; #ifdef DIFFUSE uniform vec2 vDiffuseInfos; #endif #ifdef AMBIENT uniform vec2 vAmbientInfos; #endif #ifdef OPACITY uniform vec2 vOpacityInfos; #endif #ifdef EMISSIVE uniform vec2 vEmissiveInfos; #endif #ifdef LIGHTMAP uniform vec2 vLightmapInfos; #endif #ifdef BUMP uniform vec3 vBumpInfos;uniform vec2 vTangentSpaceParams; #endif #ifdef ALPHATEST uniform float alphaCutOff; #endif #if defined(REFLECTIONMAP_SPHERICAL) || defined(REFLECTIONMAP_PROJECTION) || defined(REFRACTION) || defined(PREPASS) uniform mat4 view; #endif #ifdef REFRACTION uniform vec4 vRefractionInfos; #ifndef REFRACTIONMAP_3D uniform mat4 refractionMatrix; #endif #ifdef REFRACTIONFRESNEL uniform vec4 refractionLeftColor;uniform vec4 refractionRightColor; #endif #if defined(USE_LOCAL_REFRACTIONMAP_CUBIC) && defined(REFRACTIONMAP_3D) uniform vec3 vRefractionPosition;uniform vec3 vRefractionSize; #endif #endif #if defined(SPECULAR) && defined(SPECULARTERM) uniform vec2 vSpecularInfos; #endif #ifdef DIFFUSEFRESNEL uniform vec4 diffuseLeftColor;uniform vec4 diffuseRightColor; #endif #ifdef OPACITYFRESNEL uniform vec4 opacityParts; #endif #ifdef EMISSIVEFRESNEL uniform vec4 emissiveLeftColor;uniform vec4 emissiveRightColor; #endif #ifdef REFLECTION uniform vec2 vReflectionInfos; #if defined(REFLECTIONMAP_PLANAR) || defined(REFLECTIONMAP_CUBIC) || defined(REFLECTIONMAP_PROJECTION) || defined(REFLECTIONMAP_EQUIRECTANGULAR) || defined(REFLECTIONMAP_SPHERICAL) || defined(REFLECTIONMAP_SKYBOX) uniform mat4 reflectionMatrix; #endif #ifndef REFLECTIONMAP_SKYBOX #if defined(USE_LOCAL_REFLECTIONMAP_CUBIC) && defined(REFLECTIONMAP_CUBIC) uniform vec3 vReflectionPosition;uniform vec3 vReflectionSize; #endif #endif #ifdef REFLECTIONFRESNEL uniform vec4 reflectionLeftColor;uniform vec4 reflectionRightColor; #endif #endif #ifdef DETAIL uniform vec4 vDetailInfos; #endif #include #define ADDITIONAL_FRAGMENT_DECLARATION `; Le.IncludesShadersStore[iAe] = sAe; const aAe = "sceneUboDeclaration", oAe = `layout(std140,column_major) uniform;uniform Scene {mat4 viewProjection; #ifdef MULTIVIEW mat4 viewProjectionR; #endif mat4 view;mat4 projection;vec4 vEyePosition;}; `; Le.IncludesShadersStore[aAe] = oAe; const fAe = "meshUboDeclaration", AAe = `#ifdef WEBGL2 uniform mat4 world;uniform float visibility; #else layout(std140,column_major) uniform;uniform Mesh {mat4 world;float visibility;}; #endif #define WORLD_UBO `; Le.IncludesShadersStore[fAe] = AAe; const dAe = "defaultUboDeclaration", vAe = `layout(std140,column_major) uniform;uniform Material {vec4 diffuseLeftColor;vec4 diffuseRightColor;vec4 opacityParts;vec4 reflectionLeftColor;vec4 reflectionRightColor;vec4 refractionLeftColor;vec4 refractionRightColor;vec4 emissiveLeftColor;vec4 emissiveRightColor;vec2 vDiffuseInfos;vec2 vAmbientInfos;vec2 vOpacityInfos;vec2 vReflectionInfos;vec3 vReflectionPosition;vec3 vReflectionSize;vec2 vEmissiveInfos;vec2 vLightmapInfos;vec2 vSpecularInfos;vec3 vBumpInfos;mat4 diffuseMatrix;mat4 ambientMatrix;mat4 opacityMatrix;mat4 reflectionMatrix;mat4 emissiveMatrix;mat4 lightmapMatrix;mat4 specularMatrix;mat4 bumpMatrix;vec2 vTangentSpaceParams;float pointSize;float alphaCutOff;mat4 refractionMatrix;vec4 vRefractionInfos;vec3 vRefractionPosition;vec3 vRefractionSize;vec4 vSpecularColor;vec3 vEmissiveColor;vec4 vDiffuseColor;vec3 vAmbientColor; #define ADDITIONAL_UBO_DECLARATION }; #include #include `; Le.IncludesShadersStore[dAe] = vAe; const uAe = "prePassDeclaration", lAe = `#ifdef PREPASS #extension GL_EXT_draw_buffers : require layout(location=0) out highp vec4 glFragData[{X}];highp vec4 gl_FragColor; #ifdef PREPASS_DEPTH varying highp vec3 vViewPos; #endif #ifdef PREPASS_VELOCITY varying highp vec4 vCurrentPosition;varying highp vec4 vPreviousPosition; #endif #endif `; Le.IncludesShadersStore[uAe] = lAe; const PAe = "oitDeclaration", cAe = `#ifdef ORDER_INDEPENDENT_TRANSPARENCY #extension GL_EXT_draw_buffers : require layout(location=0) out vec2 depth; layout(location=1) out vec4 frontColor;layout(location=2) out vec4 backColor; #define MAX_DEPTH 99999.0 highp vec4 gl_FragColor;uniform sampler2D oitDepthSampler;uniform sampler2D oitFrontColorSampler; #endif `; Le.IncludesShadersStore[PAe] = cAe; const pAe = "mainUVVaryingDeclaration", hAe = `#ifdef MAINUV{X} varying vec2 vMainUV{X}; #endif `; Le.IncludesShadersStore[pAe] = hAe; const HAe = "helperFunctions", gAe = `const float PI=3.1415926535897932384626433832795;const float RECIPROCAL_PI=0.3183098861837907;const float RECIPROCAL_PI2=0.15915494309189535;const float HALF_MIN=5.96046448e-08; const float LinearEncodePowerApprox=2.2;const float GammaEncodePowerApprox=1.0/LinearEncodePowerApprox;const vec3 LuminanceEncodeApprox=vec3(0.2126,0.7152,0.0722);const float Epsilon=0.0000001; #define saturate(x) clamp(x,0.0,1.0) #define absEps(x) abs(x)+Epsilon #define maxEps(x) max(x,Epsilon) #define saturateEps(x) clamp(x,Epsilon,1.0) mat3 transposeMat3(mat3 inMatrix) {vec3 i0=inMatrix[0];vec3 i1=inMatrix[1];vec3 i2=inMatrix[2];mat3 outMatrix=mat3( vec3(i0.x,i1.x,i2.x), vec3(i0.y,i1.y,i2.y), vec3(i0.z,i1.z,i2.z) );return outMatrix;} mat3 inverseMat3(mat3 inMatrix) {float a00=inMatrix[0][0],a01=inMatrix[0][1],a02=inMatrix[0][2];float a10=inMatrix[1][0],a11=inMatrix[1][1],a12=inMatrix[1][2];float a20=inMatrix[2][0],a21=inMatrix[2][1],a22=inMatrix[2][2];float b01=a22*a11-a12*a21;float b11=-a22*a10+a12*a20;float b21=a21*a10-a11*a20;float det=a00*b01+a01*b11+a02*b21;return mat3(b01,(-a22*a01+a02*a21),(a12*a01-a02*a11), b11,(a22*a00-a02*a20),(-a12*a00+a02*a10), b21,(-a21*a00+a01*a20),(a11*a00-a01*a10))/det;} #if USE_EXACT_SRGB_CONVERSIONS vec3 toLinearSpaceExact(vec3 color) {vec3 nearZeroSection=0.0773993808*color;vec3 remainingSection=pow(0.947867299*(color+vec3(0.055)),vec3(2.4)); #if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE) return mix(remainingSection,nearZeroSection,lessThanEqual(color,vec3(0.04045))); #else return vec3( color.r<=0.04045 ? nearZeroSection.r : remainingSection.r, color.g<=0.04045 ? nearZeroSection.g : remainingSection.g, color.b<=0.04045 ? nearZeroSection.b : remainingSection.b); #endif } vec3 toGammaSpaceExact(vec3 color) {vec3 nearZeroSection=12.92*color;vec3 remainingSection=1.055*pow(color,vec3(0.41666))-vec3(0.055); #if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE) return mix(remainingSection,nearZeroSection,lessThanEqual(color,vec3(0.0031308))); #else return vec3( color.r<=0.0031308 ? nearZeroSection.r : remainingSection.r, color.g<=0.0031308 ? nearZeroSection.g : remainingSection.g, color.b<=0.0031308 ? nearZeroSection.b : remainingSection.b); #endif } #endif float toLinearSpace(float color) { #if USE_EXACT_SRGB_CONVERSIONS float nearZeroSection=0.0773993808*color;float remainingSection=pow(0.947867299*(color+0.055),2.4);return color<=0.04045 ? nearZeroSection : remainingSection; #else return pow(color,LinearEncodePowerApprox); #endif } vec3 toLinearSpace(vec3 color) { #if USE_EXACT_SRGB_CONVERSIONS return toLinearSpaceExact(color); #else return pow(color,vec3(LinearEncodePowerApprox)); #endif } vec4 toLinearSpace(vec4 color) { #if USE_EXACT_SRGB_CONVERSIONS return vec4(toLinearSpaceExact(color.rgb),color.a); #else return vec4(pow(color.rgb,vec3(LinearEncodePowerApprox)),color.a); #endif } float toGammaSpace(float color) { #if USE_EXACT_SRGB_CONVERSIONS float nearZeroSection=12.92*color;float remainingSection=1.055*pow(color,0.41666)-0.055;return color<=0.0031308 ? nearZeroSection : remainingSection; #else return pow(color,GammaEncodePowerApprox); #endif } vec3 toGammaSpace(vec3 color) { #if USE_EXACT_SRGB_CONVERSIONS return toGammaSpaceExact(color); #else return pow(color,vec3(GammaEncodePowerApprox)); #endif } vec4 toGammaSpace(vec4 color) { #if USE_EXACT_SRGB_CONVERSIONS return vec4(toGammaSpaceExact(color.rgb),color.a); #else return vec4(pow(color.rgb,vec3(GammaEncodePowerApprox)),color.a); #endif } float square(float value) {return value*value;} vec3 square(vec3 value) {return value*value;} float pow5(float value) {float sq=value*value;return sq*sq*value;} float getLuminance(vec3 color) {return clamp(dot(color,LuminanceEncodeApprox),0.,1.);} float getRand(vec2 seed) {return fract(sin(dot(seed.xy ,vec2(12.9898,78.233)))*43758.5453);} float dither(vec2 seed,float varianceAmount) {float rand=getRand(seed);float normVariance=varianceAmount/255.0;float dither=mix(-normVariance,normVariance,rand);return dither;} const float rgbdMaxRange=255.0;vec4 toRGBD(vec3 color) {float maxRGB=maxEps(max(color.r,max(color.g,color.b)));float D =max(rgbdMaxRange/maxRGB,1.);D =clamp(floor(D)/255.0,0.,1.);vec3 rgb=color.rgb*D;rgb=toGammaSpace(rgb);return vec4(clamp(rgb,0.,1.),D); } vec3 fromRGBD(vec4 rgbd) {rgbd.rgb=toLinearSpace(rgbd.rgb);return rgbd.rgb/rgbd.a;} vec3 parallaxCorrectNormal( vec3 vertexPos,vec3 origVec,vec3 cubeSize,vec3 cubePos ) {vec3 invOrigVec=vec3(1.0,1.0,1.0)/origVec;vec3 halfSize=cubeSize*0.5;vec3 intersecAtMaxPlane=(cubePos+halfSize-vertexPos)*invOrigVec;vec3 intersecAtMinPlane=(cubePos-halfSize-vertexPos)*invOrigVec;vec3 largestIntersec=max(intersecAtMaxPlane,intersecAtMinPlane);float distance=min(min(largestIntersec.x,largestIntersec.y),largestIntersec.z);vec3 intersectPositionWS=vertexPos+origVec*distance;return intersectPositionWS-cubePos;} `; Le.IncludesShadersStore[HAe] = gAe; const XAe = "lightFragmentDeclaration", TAe = `#ifdef LIGHT{X} uniform vec4 vLightData{X};uniform vec4 vLightDiffuse{X}; #ifdef SPECULARTERM uniform vec4 vLightSpecular{X}; #else vec4 vLightSpecular{X}=vec4(0.); #endif #ifdef SHADOW{X} #ifdef SHADOWCSM{X} uniform mat4 lightMatrix{X}[SHADOWCSMNUM_CASCADES{X}];uniform float viewFrustumZ{X}[SHADOWCSMNUM_CASCADES{X}];uniform float frustumLengths{X}[SHADOWCSMNUM_CASCADES{X}];uniform float cascadeBlendFactor{X};varying vec4 vPositionFromLight{X}[SHADOWCSMNUM_CASCADES{X}];varying float vDepthMetric{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromCamera{X}; #if defined(SHADOWPCSS{X}) uniform highp sampler2DArrayShadow shadowSampler{X};uniform highp sampler2DArray depthSampler{X};uniform vec2 lightSizeUVCorrection{X}[SHADOWCSMNUM_CASCADES{X}];uniform float depthCorrection{X}[SHADOWCSMNUM_CASCADES{X}];uniform float penumbraDarkness{X}; #elif defined(SHADOWPCF{X}) uniform highp sampler2DArrayShadow shadowSampler{X}; #else uniform highp sampler2DArray shadowSampler{X}; #endif #ifdef SHADOWCSMDEBUG{X} const vec3 vCascadeColorsMultiplier{X}[8]=vec3[8] ( vec3 ( 1.5,0.0,0.0 ), vec3 ( 0.0,1.5,0.0 ), vec3 ( 0.0,0.0,5.5 ), vec3 ( 1.5,0.0,5.5 ), vec3 ( 1.5,1.5,0.0 ), vec3 ( 1.0,1.0,1.0 ), vec3 ( 0.0,1.0,5.5 ), vec3 ( 0.5,3.5,0.75 ) );vec3 shadowDebug{X}; #endif #ifdef SHADOWCSMUSESHADOWMAXZ{X} int index{X}=-1; #else int index{X}=SHADOWCSMNUM_CASCADES{X}-1; #endif float diff{X}=0.; #elif defined(SHADOWCUBE{X}) uniform samplerCube shadowSampler{X}; #else varying vec4 vPositionFromLight{X};varying float vDepthMetric{X}; #if defined(SHADOWPCSS{X}) uniform highp sampler2DShadow shadowSampler{X};uniform highp sampler2D depthSampler{X}; #elif defined(SHADOWPCF{X}) uniform highp sampler2DShadow shadowSampler{X}; #else uniform sampler2D shadowSampler{X}; #endif uniform mat4 lightMatrix{X}; #endif uniform vec4 shadowsInfo{X};uniform vec2 depthValues{X}; #endif #ifdef SPOTLIGHT{X} uniform vec4 vLightDirection{X};uniform vec4 vLightFalloff{X}; #elif defined(POINTLIGHT{X}) uniform vec4 vLightFalloff{X}; #elif defined(HEMILIGHT{X}) uniform vec3 vLightGround{X}; #endif #ifdef PROJECTEDLIGHTTEXTURE{X} uniform mat4 textureProjectionMatrix{X};uniform sampler2D projectionLightSampler{X}; #endif #endif `; Le.IncludesShadersStore[XAe] = TAe; const qAe = "lightUboDeclaration", bAe = `#ifdef LIGHT{X} uniform Light{X} {vec4 vLightData;vec4 vLightDiffuse;vec4 vLightSpecular; #ifdef SPOTLIGHT{X} vec4 vLightDirection;vec4 vLightFalloff; #elif defined(POINTLIGHT{X}) vec4 vLightFalloff; #elif defined(HEMILIGHT{X}) vec3 vLightGround; #endif vec4 shadowsInfo;vec2 depthValues;} light{X}; #ifdef PROJECTEDLIGHTTEXTURE{X} uniform mat4 textureProjectionMatrix{X};uniform sampler2D projectionLightSampler{X}; #endif #ifdef SHADOW{X} #ifdef SHADOWCSM{X} uniform mat4 lightMatrix{X}[SHADOWCSMNUM_CASCADES{X}];uniform float viewFrustumZ{X}[SHADOWCSMNUM_CASCADES{X}];uniform float frustumLengths{X}[SHADOWCSMNUM_CASCADES{X}];uniform float cascadeBlendFactor{X};varying vec4 vPositionFromLight{X}[SHADOWCSMNUM_CASCADES{X}];varying float vDepthMetric{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromCamera{X}; #if defined(SHADOWPCSS{X}) uniform highp sampler2DArrayShadow shadowSampler{X};uniform highp sampler2DArray depthSampler{X};uniform vec2 lightSizeUVCorrection{X}[SHADOWCSMNUM_CASCADES{X}];uniform float depthCorrection{X}[SHADOWCSMNUM_CASCADES{X}];uniform float penumbraDarkness{X}; #elif defined(SHADOWPCF{X}) uniform highp sampler2DArrayShadow shadowSampler{X}; #else uniform highp sampler2DArray shadowSampler{X}; #endif #ifdef SHADOWCSMDEBUG{X} const vec3 vCascadeColorsMultiplier{X}[8]=vec3[8] ( vec3 ( 1.5,0.0,0.0 ), vec3 ( 0.0,1.5,0.0 ), vec3 ( 0.0,0.0,5.5 ), vec3 ( 1.5,0.0,5.5 ), vec3 ( 1.5,1.5,0.0 ), vec3 ( 1.0,1.0,1.0 ), vec3 ( 0.0,1.0,5.5 ), vec3 ( 0.5,3.5,0.75 ) );vec3 shadowDebug{X}; #endif #ifdef SHADOWCSMUSESHADOWMAXZ{X} int index{X}=-1; #else int index{X}=SHADOWCSMNUM_CASCADES{X}-1; #endif float diff{X}=0.; #elif defined(SHADOWCUBE{X}) uniform samplerCube shadowSampler{X}; #else varying vec4 vPositionFromLight{X};varying float vDepthMetric{X}; #if defined(SHADOWPCSS{X}) uniform highp sampler2DShadow shadowSampler{X};uniform highp sampler2D depthSampler{X}; #elif defined(SHADOWPCF{X}) uniform highp sampler2DShadow shadowSampler{X}; #else uniform sampler2D shadowSampler{X}; #endif uniform mat4 lightMatrix{X}; #endif #endif #endif `; Le.IncludesShadersStore[qAe] = bAe; const xAe = "lightsFragmentFunctions", DAe = `struct lightingInfo {vec3 diffuse; #ifdef SPECULARTERM vec3 specular; #endif #ifdef NDOTL float ndl; #endif };lightingInfo computeLighting(vec3 viewDirectionW,vec3 vNormal,vec4 lightData,vec3 diffuseColor,vec3 specularColor,float range,float glossiness) {lightingInfo result;vec3 lightVectorW;float attenuation=1.0;if (lightData.w==0.) {vec3 direction=lightData.xyz-vPositionW;attenuation=max(0.,1.0-length(direction)/range);lightVectorW=normalize(direction);} else {lightVectorW=normalize(-lightData.xyz);} float ndl=max(0.,dot(vNormal,lightVectorW)); #ifdef NDOTL result.ndl=ndl; #endif result.diffuse=ndl*diffuseColor*attenuation; #ifdef SPECULARTERM vec3 angleW=normalize(viewDirectionW+lightVectorW);float specComp=max(0.,dot(vNormal,angleW));specComp=pow(specComp,max(1.,glossiness));result.specular=specComp*specularColor*attenuation; #endif return result;} lightingInfo computeSpotLighting(vec3 viewDirectionW,vec3 vNormal,vec4 lightData,vec4 lightDirection,vec3 diffuseColor,vec3 specularColor,float range,float glossiness) {lightingInfo result;vec3 direction=lightData.xyz-vPositionW;vec3 lightVectorW=normalize(direction);float attenuation=max(0.,1.0-length(direction)/range);float cosAngle=max(0.,dot(lightDirection.xyz,-lightVectorW));if (cosAngle>=lightDirection.w) {cosAngle=max(0.,pow(cosAngle,lightData.w));attenuation*=cosAngle;float ndl=max(0.,dot(vNormal,lightVectorW)); #ifdef NDOTL result.ndl=ndl; #endif result.diffuse=ndl*diffuseColor*attenuation; #ifdef SPECULARTERM vec3 angleW=normalize(viewDirectionW+lightVectorW);float specComp=max(0.,dot(vNormal,angleW));specComp=pow(specComp,max(1.,glossiness));result.specular=specComp*specularColor*attenuation; #endif return result;} result.diffuse=vec3(0.); #ifdef SPECULARTERM result.specular=vec3(0.); #endif #ifdef NDOTL result.ndl=0.; #endif return result;} lightingInfo computeHemisphericLighting(vec3 viewDirectionW,vec3 vNormal,vec4 lightData,vec3 diffuseColor,vec3 specularColor,vec3 groundColor,float glossiness) {lightingInfo result;float ndl=dot(vNormal,lightData.xyz)*0.5+0.5; #ifdef NDOTL result.ndl=ndl; #endif result.diffuse=mix(groundColor,diffuseColor,ndl); #ifdef SPECULARTERM vec3 angleW=normalize(viewDirectionW+lightData.xyz);float specComp=max(0.,dot(vNormal,angleW));specComp=pow(specComp,max(1.,glossiness));result.specular=specComp*specularColor; #endif return result;} #define inline vec3 computeProjectionTextureDiffuseLighting(sampler2D projectionLightSampler,mat4 textureProjectionMatrix){vec4 strq=textureProjectionMatrix*vec4(vPositionW,1.0);strq/=strq.w;vec3 textureColor=texture2D(projectionLightSampler,strq.xy).rgb;return textureColor;}`; Le.IncludesShadersStore[xAe] = DAe; const jAe = "shadowsFragmentFunctions", wAe = `#ifdef SHADOWS #if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE) #define TEXTUREFUNC(s,c,l) texture2DLodEXT(s,c,l) #else #define TEXTUREFUNC(s,c,b) texture2D(s,c,b) #endif #ifndef SHADOWFLOAT float unpack(vec4 color) {const vec4 bit_shift=vec4(1.0/(255.0*255.0*255.0),1.0/(255.0*255.0),1.0/255.0,1.0);return dot(color,bit_shift);} #endif float computeFallOff(float value,vec2 clipSpace,float frustumEdgeFalloff) {float mask=smoothstep(1.0-frustumEdgeFalloff,1.00000012,clamp(dot(clipSpace,clipSpace),0.,1.));return mix(value,1.0,mask);} #define inline float computeShadowCube(vec3 worldPos,vec3 lightPosition,samplerCube shadowSampler,float darkness,vec2 depthValues) {vec3 directionToLight=worldPos-lightPosition;float depth=length(directionToLight);depth=(depth+depthValues.x)/(depthValues.y);depth=clamp(depth,0.,1.0);directionToLight=normalize(directionToLight);directionToLight.y=-directionToLight.y; #ifndef SHADOWFLOAT float shadow=unpack(textureCube(shadowSampler,directionToLight)); #else float shadow=textureCube(shadowSampler,directionToLight).x; #endif return depth>shadow ? darkness : 1.0;} #define inline float computeShadowWithPoissonSamplingCube(vec3 worldPos,vec3 lightPosition,samplerCube shadowSampler,float mapSize,float darkness,vec2 depthValues) {vec3 directionToLight=worldPos-lightPosition;float depth=length(directionToLight);depth=(depth+depthValues.x)/(depthValues.y);depth=clamp(depth,0.,1.0);directionToLight=normalize(directionToLight);directionToLight.y=-directionToLight.y;float visibility=1.;vec3 poissonDisk[4];poissonDisk[0]=vec3(-1.0,1.0,-1.0);poissonDisk[1]=vec3(1.0,-1.0,-1.0);poissonDisk[2]=vec3(-1.0,-1.0,-1.0);poissonDisk[3]=vec3(1.0,-1.0,1.0); #ifndef SHADOWFLOAT if (unpack(textureCube(shadowSampler,directionToLight+poissonDisk[0]*mapSize))shadow ? computeFallOff(darkness,clipSpace.xy,frustumEdgeFalloff) : 1.;} #endif #define inline float computeShadow(vec4 vPositionFromLight,float depthMetric,sampler2D shadowSampler,float darkness,float frustumEdgeFalloff) {vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec2 uv=0.5*clipSpace.xy+vec2(0.5);if (uv.x<0. || uv.x>1.0 || uv.y<0. || uv.y>1.0) {return 1.0;} else {float shadowPixelDepth=clamp(depthMetric,0.,1.0); #ifndef SHADOWFLOAT float shadow=unpack(TEXTUREFUNC(shadowSampler,uv,0.)); #else float shadow=TEXTUREFUNC(shadowSampler,uv,0.).x; #endif return shadowPixelDepth>shadow ? computeFallOff(darkness,clipSpace.xy,frustumEdgeFalloff) : 1.;}} #define inline float computeShadowWithPoissonSampling(vec4 vPositionFromLight,float depthMetric,sampler2D shadowSampler,float mapSize,float darkness,float frustumEdgeFalloff) {vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec2 uv=0.5*clipSpace.xy+vec2(0.5);if (uv.x<0. || uv.x>1.0 || uv.y<0. || uv.y>1.0) {return 1.0;} else {float shadowPixelDepth=clamp(depthMetric,0.,1.0);float visibility=1.;vec2 poissonDisk[4];poissonDisk[0]=vec2(-0.94201624,-0.39906216);poissonDisk[1]=vec2(0.94558609,-0.76890725);poissonDisk[2]=vec2(-0.094184101,-0.92938870);poissonDisk[3]=vec2(0.34495938,0.29387760); #ifndef SHADOWFLOAT if (unpack(TEXTUREFUNC(shadowSampler,uv+poissonDisk[0]*mapSize,0.))1.0 || uv.y<0. || uv.y>1.0) {return 1.0;} else {float shadowPixelDepth=clamp(depthMetric,0.,1.0); #ifndef SHADOWFLOAT float shadowMapSample=unpack(TEXTUREFUNC(shadowSampler,uv,0.)); #else float shadowMapSample=TEXTUREFUNC(shadowSampler,uv,0.).x; #endif float esm=1.0-clamp(exp(min(87.,depthScale*shadowPixelDepth))*shadowMapSample,0.,1.-darkness);return computeFallOff(esm,clipSpace.xy,frustumEdgeFalloff);}} #define inline float computeShadowWithCloseESM(vec4 vPositionFromLight,float depthMetric,sampler2D shadowSampler,float darkness,float depthScale,float frustumEdgeFalloff) {vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec2 uv=0.5*clipSpace.xy+vec2(0.5);if (uv.x<0. || uv.x>1.0 || uv.y<0. || uv.y>1.0) {return 1.0;} else {float shadowPixelDepth=clamp(depthMetric,0.,1.0); #ifndef SHADOWFLOAT float shadowMapSample=unpack(TEXTUREFUNC(shadowSampler,uv,0.)); #else float shadowMapSample=TEXTUREFUNC(shadowSampler,uv,0.).x; #endif float esm=clamp(exp(min(87.,-depthScale*(shadowPixelDepth-shadowMapSample))),darkness,1.);return computeFallOff(esm,clipSpace.xy,frustumEdgeFalloff);}} #ifdef IS_NDC_HALF_ZRANGE #define ZINCLIP clipSpace.z #else #define ZINCLIP uvDepth.z #endif #if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE) #define GREATEST_LESS_THAN_ONE 0.99999994 /* disable_uniformity_analysis */ #define inline float computeShadowWithCSMPCF1(float layer,vec4 vPositionFromLight,float depthMetric,highp sampler2DArrayShadow shadowSampler,float darkness,float frustumEdgeFalloff) {vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=clamp(ZINCLIP,0.,GREATEST_LESS_THAN_ONE);vec4 uvDepthLayer=vec4(uvDepth.x,uvDepth.y,layer,uvDepth.z);float shadow=texture2D(shadowSampler,uvDepthLayer);shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);} #define inline float computeShadowWithCSMPCF3(float layer,vec4 vPositionFromLight,float depthMetric,highp sampler2DArrayShadow shadowSampler,vec2 shadowMapSizeAndInverse,float darkness,float frustumEdgeFalloff) {vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=clamp(ZINCLIP,0.,GREATEST_LESS_THAN_ONE);vec2 uv=uvDepth.xy*shadowMapSizeAndInverse.x; uv+=0.5; vec2 st=fract(uv); vec2 base_uv=floor(uv)-0.5; base_uv*=shadowMapSizeAndInverse.y; vec2 uvw0=3.-2.*st;vec2 uvw1=1.+2.*st;vec2 u=vec2((2.-st.x)/uvw0.x-1.,st.x/uvw1.x+1.)*shadowMapSizeAndInverse.y;vec2 v=vec2((2.-st.y)/uvw0.y-1.,st.y/uvw1.y+1.)*shadowMapSizeAndInverse.y;float shadow=0.;shadow+=uvw0.x*uvw0.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[0],v[0]),layer,uvDepth.z));shadow+=uvw1.x*uvw0.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[1],v[0]),layer,uvDepth.z));shadow+=uvw0.x*uvw1.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[0],v[1]),layer,uvDepth.z));shadow+=uvw1.x*uvw1.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[1],v[1]),layer,uvDepth.z));shadow=shadow/16.;shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);} #define inline float computeShadowWithCSMPCF5(float layer,vec4 vPositionFromLight,float depthMetric,highp sampler2DArrayShadow shadowSampler,vec2 shadowMapSizeAndInverse,float darkness,float frustumEdgeFalloff) {vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=clamp(ZINCLIP,0.,GREATEST_LESS_THAN_ONE);vec2 uv=uvDepth.xy*shadowMapSizeAndInverse.x; uv+=0.5; vec2 st=fract(uv); vec2 base_uv=floor(uv)-0.5; base_uv*=shadowMapSizeAndInverse.y; vec2 uvw0=4.-3.*st;vec2 uvw1=vec2(7.);vec2 uvw2=1.+3.*st;vec3 u=vec3((3.-2.*st.x)/uvw0.x-2.,(3.+st.x)/uvw1.x,st.x/uvw2.x+2.)*shadowMapSizeAndInverse.y;vec3 v=vec3((3.-2.*st.y)/uvw0.y-2.,(3.+st.y)/uvw1.y,st.y/uvw2.y+2.)*shadowMapSizeAndInverse.y;float shadow=0.;shadow+=uvw0.x*uvw0.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[0],v[0]),layer,uvDepth.z));shadow+=uvw1.x*uvw0.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[1],v[0]),layer,uvDepth.z));shadow+=uvw2.x*uvw0.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[2],v[0]),layer,uvDepth.z));shadow+=uvw0.x*uvw1.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[0],v[1]),layer,uvDepth.z));shadow+=uvw1.x*uvw1.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[1],v[1]),layer,uvDepth.z));shadow+=uvw2.x*uvw1.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[2],v[1]),layer,uvDepth.z));shadow+=uvw0.x*uvw2.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[0],v[2]),layer,uvDepth.z));shadow+=uvw1.x*uvw2.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[1],v[2]),layer,uvDepth.z));shadow+=uvw2.x*uvw2.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[2],v[2]),layer,uvDepth.z));shadow=shadow/144.;shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);} #define inline float computeShadowWithPCF1(vec4 vPositionFromLight,float depthMetric,highp sampler2DShadow shadowSampler,float darkness,float frustumEdgeFalloff) {if (depthMetric>1.0 || depthMetric<0.0) {return 1.0;} else {vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=ZINCLIP;float shadow=TEXTUREFUNC(shadowSampler,uvDepth,0.);shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);}} #define inline float computeShadowWithPCF3(vec4 vPositionFromLight,float depthMetric,highp sampler2DShadow shadowSampler,vec2 shadowMapSizeAndInverse,float darkness,float frustumEdgeFalloff) {if (depthMetric>1.0 || depthMetric<0.0) {return 1.0;} else {vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=ZINCLIP;vec2 uv=uvDepth.xy*shadowMapSizeAndInverse.x; uv+=0.5; vec2 st=fract(uv); vec2 base_uv=floor(uv)-0.5; base_uv*=shadowMapSizeAndInverse.y; vec2 uvw0=3.-2.*st;vec2 uvw1=1.+2.*st;vec2 u=vec2((2.-st.x)/uvw0.x-1.,st.x/uvw1.x+1.)*shadowMapSizeAndInverse.y;vec2 v=vec2((2.-st.y)/uvw0.y-1.,st.y/uvw1.y+1.)*shadowMapSizeAndInverse.y;float shadow=0.;shadow+=uvw0.x*uvw0.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[0],v[0]),uvDepth.z),0.);shadow+=uvw1.x*uvw0.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[1],v[0]),uvDepth.z),0.);shadow+=uvw0.x*uvw1.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[0],v[1]),uvDepth.z),0.);shadow+=uvw1.x*uvw1.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[1],v[1]),uvDepth.z),0.);shadow=shadow/16.;shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);}} #define inline float computeShadowWithPCF5(vec4 vPositionFromLight,float depthMetric,highp sampler2DShadow shadowSampler,vec2 shadowMapSizeAndInverse,float darkness,float frustumEdgeFalloff) {if (depthMetric>1.0 || depthMetric<0.0) {return 1.0;} else {vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=ZINCLIP;vec2 uv=uvDepth.xy*shadowMapSizeAndInverse.x; uv+=0.5; vec2 st=fract(uv); vec2 base_uv=floor(uv)-0.5; base_uv*=shadowMapSizeAndInverse.y; vec2 uvw0=4.-3.*st;vec2 uvw1=vec2(7.);vec2 uvw2=1.+3.*st;vec3 u=vec3((3.-2.*st.x)/uvw0.x-2.,(3.+st.x)/uvw1.x,st.x/uvw2.x+2.)*shadowMapSizeAndInverse.y;vec3 v=vec3((3.-2.*st.y)/uvw0.y-2.,(3.+st.y)/uvw1.y,st.y/uvw2.y+2.)*shadowMapSizeAndInverse.y;float shadow=0.;shadow+=uvw0.x*uvw0.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[0],v[0]),uvDepth.z),0.);shadow+=uvw1.x*uvw0.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[1],v[0]),uvDepth.z),0.);shadow+=uvw2.x*uvw0.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[2],v[0]),uvDepth.z),0.);shadow+=uvw0.x*uvw1.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[0],v[1]),uvDepth.z),0.);shadow+=uvw1.x*uvw1.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[1],v[1]),uvDepth.z),0.);shadow+=uvw2.x*uvw1.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[2],v[1]),uvDepth.z),0.);shadow+=uvw0.x*uvw2.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[0],v[2]),uvDepth.z),0.);shadow+=uvw1.x*uvw2.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[1],v[2]),uvDepth.z),0.);shadow+=uvw2.x*uvw2.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[2],v[2]),uvDepth.z),0.);shadow=shadow/144.;shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);}} const vec3 PoissonSamplers32[64]=vec3[64]( vec3(0.06407013,0.05409927,0.), vec3(0.7366577,0.5789394,0.), vec3(-0.6270542,-0.5320278,0.), vec3(-0.4096107,0.8411095,0.), vec3(0.6849564,-0.4990818,0.), vec3(-0.874181,-0.04579735,0.), vec3(0.9989998,0.0009880066,0.), vec3(-0.004920578,-0.9151649,0.), vec3(0.1805763,0.9747483,0.), vec3(-0.2138451,0.2635818,0.), vec3(0.109845,0.3884785,0.), vec3(0.06876755,-0.3581074,0.), vec3(0.374073,-0.7661266,0.), vec3(0.3079132,-0.1216763,0.), vec3(-0.3794335,-0.8271583,0.), vec3(-0.203878,-0.07715034,0.), vec3(0.5912697,0.1469799,0.), vec3(-0.88069,0.3031784,0.), vec3(0.5040108,0.8283722,0.), vec3(-0.5844124,0.5494877,0.), vec3(0.6017799,-0.1726654,0.), vec3(-0.5554981,0.1559997,0.), vec3(-0.3016369,-0.3900928,0.), vec3(-0.5550632,-0.1723762,0.), vec3(0.925029,0.2995041,0.), vec3(-0.2473137,0.5538505,0.), vec3(0.9183037,-0.2862392,0.), vec3(0.2469421,0.6718712,0.), vec3(0.3916397,-0.4328209,0.), vec3(-0.03576927,-0.6220032,0.), vec3(-0.04661255,0.7995201,0.), vec3(0.4402924,0.3640312,0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.), vec3(0.) );const vec3 PoissonSamplers64[64]=vec3[64]( vec3(-0.613392,0.617481,0.), vec3(0.170019,-0.040254,0.), vec3(-0.299417,0.791925,0.), vec3(0.645680,0.493210,0.), vec3(-0.651784,0.717887,0.), vec3(0.421003,0.027070,0.), vec3(-0.817194,-0.271096,0.), vec3(-0.705374,-0.668203,0.), vec3(0.977050,-0.108615,0.), vec3(0.063326,0.142369,0.), vec3(0.203528,0.214331,0.), vec3(-0.667531,0.326090,0.), vec3(-0.098422,-0.295755,0.), vec3(-0.885922,0.215369,0.), vec3(0.566637,0.605213,0.), vec3(0.039766,-0.396100,0.), vec3(0.751946,0.453352,0.), vec3(0.078707,-0.715323,0.), vec3(-0.075838,-0.529344,0.), vec3(0.724479,-0.580798,0.), vec3(0.222999,-0.215125,0.), vec3(-0.467574,-0.405438,0.), vec3(-0.248268,-0.814753,0.), vec3(0.354411,-0.887570,0.), vec3(0.175817,0.382366,0.), vec3(0.487472,-0.063082,0.), vec3(-0.084078,0.898312,0.), vec3(0.488876,-0.783441,0.), vec3(0.470016,0.217933,0.), vec3(-0.696890,-0.549791,0.), vec3(-0.149693,0.605762,0.), vec3(0.034211,0.979980,0.), vec3(0.503098,-0.308878,0.), vec3(-0.016205,-0.872921,0.), vec3(0.385784,-0.393902,0.), vec3(-0.146886,-0.859249,0.), vec3(0.643361,0.164098,0.), vec3(0.634388,-0.049471,0.), vec3(-0.688894,0.007843,0.), vec3(0.464034,-0.188818,0.), vec3(-0.440840,0.137486,0.), vec3(0.364483,0.511704,0.), vec3(0.034028,0.325968,0.), vec3(0.099094,-0.308023,0.), vec3(0.693960,-0.366253,0.), vec3(0.678884,-0.204688,0.), vec3(0.001801,0.780328,0.), vec3(0.145177,-0.898984,0.), vec3(0.062655,-0.611866,0.), vec3(0.315226,-0.604297,0.), vec3(-0.780145,0.486251,0.), vec3(-0.371868,0.882138,0.), vec3(0.200476,0.494430,0.), vec3(-0.494552,-0.711051,0.), vec3(0.612476,0.705252,0.), vec3(-0.578845,-0.768792,0.), vec3(-0.772454,-0.090976,0.), vec3(0.504440,0.372295,0.), vec3(0.155736,0.065157,0.), vec3(0.391522,0.849605,0.), vec3(-0.620106,-0.328104,0.), vec3(0.789239,-0.419965,0.), vec3(-0.545396,0.538133,0.), vec3(-0.178564,-0.596057,0.) ); #define inline float computeShadowWithCSMPCSS(float layer,vec4 vPositionFromLight,float depthMetric,highp sampler2DArray depthSampler,highp sampler2DArrayShadow shadowSampler,float shadowMapSizeInverse,float lightSizeUV,float darkness,float frustumEdgeFalloff,int searchTapCount,int pcfTapCount,vec3[64] poissonSamplers,vec2 lightSizeUVCorrection,float depthCorrection,float penumbraDarkness) {vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=clamp(ZINCLIP,0.,GREATEST_LESS_THAN_ONE);vec4 uvDepthLayer=vec4(uvDepth.x,uvDepth.y,layer,uvDepth.z);float blockerDepth=0.0;float sumBlockerDepth=0.0;float numBlocker=0.0;for (int i=0; i1.0 || depthMetric<0.0) {return 1.0;} else {vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=ZINCLIP;float blockerDepth=0.0;float sumBlockerDepth=0.0;float numBlocker=0.0;for (int i=0; i(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump,_SAMPLERNAME_,bump) #endif #if defined(DETAIL) #include(_DEFINENAME_,DETAIL,_VARYINGNAME_,Detail,_SAMPLERNAME_,detail) #endif #if defined(BUMP) && defined(PARALLAX) const float minSamples=4.;const float maxSamples=15.;const int iMaxSamples=15;vec2 parallaxOcclusion(vec3 vViewDirCoT,vec3 vNormalCoT,vec2 texCoord,float parallaxScale) {float parallaxLimit=length(vViewDirCoT.xy)/vViewDirCoT.z;parallaxLimit*=parallaxScale;vec2 vOffsetDir=normalize(vViewDirCoT.xy);vec2 vMaxOffset=vOffsetDir*parallaxLimit;float numSamples=maxSamples+(dot(vViewDirCoT,vNormalCoT)*(minSamples-maxSamples));float stepSize=1.0/numSamples;float currRayHeight=1.0;vec2 vCurrOffset=vec2(0,0);vec2 vLastOffset=vec2(0,0);float lastSampledHeight=1.0;float currSampledHeight=1.0;bool keepWorking=true;for (int i=0; icurrRayHeight) {float delta1=currSampledHeight-currRayHeight;float delta2=(currRayHeight+stepSize)-lastSampledHeight;float ratio=delta1/(delta1+delta2);vCurrOffset=(ratio)* vLastOffset+(1.0-ratio)*vCurrOffset;keepWorking=false;} else {currRayHeight-=stepSize;vLastOffset=vCurrOffset; #ifdef PARALLAX_RHS vCurrOffset-=stepSize*vMaxOffset; #else vCurrOffset+=stepSize*vMaxOffset; #endif lastSampledHeight=currSampledHeight;}} return vCurrOffset;} vec2 parallaxOffset(vec3 viewDir,float heightScale) {float height=texture2D(bumpSampler,vBumpUV).w;vec2 texCoordOffset=heightScale*viewDir.xy*height; #ifdef PARALLAX_RHS return texCoordOffset; #else return -texCoordOffset; #endif } #endif `; Le.IncludesShadersStore[EAe] = FAe; const NAe = "clipPlaneFragmentDeclaration", QAe = `#ifdef CLIPPLANE varying float fClipDistance; #endif #ifdef CLIPPLANE2 varying float fClipDistance2; #endif #ifdef CLIPPLANE3 varying float fClipDistance3; #endif #ifdef CLIPPLANE4 varying float fClipDistance4; #endif #ifdef CLIPPLANE5 varying float fClipDistance5; #endif #ifdef CLIPPLANE6 varying float fClipDistance6; #endif `; Le.IncludesShadersStore[NAe] = QAe; const YAe = "logDepthDeclaration", MAe = `#ifdef LOGARITHMICDEPTH uniform float logarithmicDepthConstant;varying float vFragmentDepth; #endif `; Le.IncludesShadersStore[YAe] = MAe; const LAe = "fogFragmentDeclaration", KAe = `#ifdef FOG #define FOGMODE_NONE 0. #define FOGMODE_EXP 1. #define FOGMODE_EXP2 2. #define FOGMODE_LINEAR 3. #define E 2.71828 uniform vec4 vFogInfos;uniform vec3 vFogColor;varying vec3 vFogDistance;float CalcFogFactor() {float fogCoeff=1.0;float fogStart=vFogInfos.y;float fogEnd=vFogInfos.z;float fogDensity=vFogInfos.w;float fogDistance=length(vFogDistance);if (FOGMODE_LINEAR==vFogInfos.x) {fogCoeff=(fogEnd-fogDistance)/(fogEnd-fogStart);} else if (FOGMODE_EXP==vFogInfos.x) {fogCoeff=1.0/pow(E,fogDistance*fogDensity);} else if (FOGMODE_EXP2==vFogInfos.x) {fogCoeff=1.0/pow(E,fogDistance*fogDistance*fogDensity*fogDensity);} return clamp(fogCoeff,0.0,1.0);} #endif `; Le.IncludesShadersStore[LAe] = KAe; const JAe = "clipPlaneFragment", zAe = `#if defined(CLIPPLANE) || defined(CLIPPLANE2) || defined(CLIPPLANE3) || defined(CLIPPLANE4) || defined(CLIPPLANE5) || defined(CLIPPLANE6) if (false) {} #endif #ifdef CLIPPLANE else if (fClipDistance>0.0) {discard;} #endif #ifdef CLIPPLANE2 else if (fClipDistance2>0.0) {discard;} #endif #ifdef CLIPPLANE3 else if (fClipDistance3>0.0) {discard;} #endif #ifdef CLIPPLANE4 else if (fClipDistance4>0.0) {discard;} #endif #ifdef CLIPPLANE5 else if (fClipDistance5>0.0) {discard;} #endif #ifdef CLIPPLANE6 else if (fClipDistance6>0.0) {discard;} #endif `; Le.IncludesShadersStore[JAe] = zAe; const GAe = "bumpFragment", ZAe = `vec2 uvOffset=vec2(0.0,0.0); #if defined(BUMP) || defined(PARALLAX) || defined(DETAIL) #ifdef NORMALXYSCALE float normalScale=1.0; #elif defined(BUMP) float normalScale=vBumpInfos.y; #else float normalScale=1.0; #endif #if defined(TANGENT) && defined(NORMAL) mat3 TBN=vTBN; #elif defined(BUMP) vec2 TBNUV=gl_FrontFacing ? vBumpUV : -vBumpUV;mat3 TBN=cotangent_frame(normalW*normalScale,vPositionW,TBNUV,vTangentSpaceParams); #else vec2 TBNUV=gl_FrontFacing ? vDetailUV : -vDetailUV;mat3 TBN=cotangent_frame(normalW*normalScale,vPositionW,TBNUV,vec2(1.,1.)); #endif #elif defined(ANISOTROPIC) #if defined(TANGENT) && defined(NORMAL) mat3 TBN=vTBN; #else vec2 TBNUV=gl_FrontFacing ? vMainUV1 : -vMainUV1;mat3 TBN=cotangent_frame(normalW,vPositionW,TBNUV,vec2(1.,1.)); #endif #endif #ifdef PARALLAX mat3 invTBN=transposeMat3(TBN); #ifdef PARALLAXOCCLUSION uvOffset=parallaxOcclusion(invTBN*-viewDirectionW,invTBN*normalW,vBumpUV,vBumpInfos.z); #else uvOffset=parallaxOffset(invTBN*viewDirectionW,vBumpInfos.z); #endif #endif #ifdef DETAIL vec4 detailColor=texture2D(detailSampler,vDetailUV+uvOffset);vec2 detailNormalRG=detailColor.wy*2.0-1.0;float detailNormalB=sqrt(1.-saturate(dot(detailNormalRG,detailNormalRG)));vec3 detailNormal=vec3(detailNormalRG,detailNormalB); #endif #ifdef BUMP #ifdef OBJECTSPACE_NORMALMAP #define CUSTOM_FRAGMENT_BUMP_FRAGMENT normalW=normalize(texture2D(bumpSampler,vBumpUV).xyz *2.0-1.0);normalW=normalize(mat3(normalMatrix)*normalW); #elif !defined(DETAIL) normalW=perturbNormal(TBN,texture2D(bumpSampler,vBumpUV+uvOffset).xyz,vBumpInfos.y); #else vec3 bumpNormal=texture2D(bumpSampler,vBumpUV+uvOffset).xyz*2.0-1.0; #if DETAIL_NORMALBLENDMETHOD==0 detailNormal.xy*=vDetailInfos.z;vec3 blendedNormal=normalize(vec3(bumpNormal.xy+detailNormal.xy,bumpNormal.z*detailNormal.z)); #elif DETAIL_NORMALBLENDMETHOD==1 detailNormal.xy*=vDetailInfos.z;bumpNormal+=vec3(0.0,0.0,1.0);detailNormal*=vec3(-1.0,-1.0,1.0);vec3 blendedNormal=bumpNormal*dot(bumpNormal,detailNormal)/bumpNormal.z-detailNormal; #endif normalW=perturbNormalBase(TBN,blendedNormal,vBumpInfos.y); #endif #elif defined(DETAIL) detailNormal.xy*=vDetailInfos.z;normalW=perturbNormalBase(TBN,detailNormal,vDetailInfos.z); #endif `; Le.IncludesShadersStore[GAe] = ZAe; const _Ae = "decalFragment", $Ae = `#ifdef DECAL #ifdef GAMMADECAL decalColor.rgb=toLinearSpace(decalColor.rgb); #endif #ifdef DECAL_SMOOTHALPHA decalColor.a*=decalColor.a; #endif surfaceAlbedo.rgb=mix(surfaceAlbedo.rgb,decalColor.rgb,decalColor.a); #endif `; Le.IncludesShadersStore[_Ae] = $Ae; const ede = "depthPrePass", tde = `#ifdef DEPTHPREPASS gl_FragColor=vec4(0.,0.,0.,1.0);return; #endif `; Le.IncludesShadersStore[ede] = tde; const rde = "lightFragment", nde = `#ifdef LIGHT{X} #if defined(SHADOWONLY) || defined(LIGHTMAP) && defined(LIGHTMAPEXCLUDED{X}) && defined(LIGHTMAPNOSPECULAR{X}) #else #ifdef PBR #ifdef SPOTLIGHT{X} preInfo=computePointAndSpotPreLightingInfo(light{X}.vLightData,viewDirectionW,normalW); #elif defined(POINTLIGHT{X}) preInfo=computePointAndSpotPreLightingInfo(light{X}.vLightData,viewDirectionW,normalW); #elif defined(HEMILIGHT{X}) preInfo=computeHemisphericPreLightingInfo(light{X}.vLightData,viewDirectionW,normalW); #elif defined(DIRLIGHT{X}) preInfo=computeDirectionalPreLightingInfo(light{X}.vLightData,viewDirectionW,normalW); #endif preInfo.NdotV=NdotV; #ifdef SPOTLIGHT{X} #ifdef LIGHT_FALLOFF_GLTF{X} preInfo.attenuation=computeDistanceLightFalloff_GLTF(preInfo.lightDistanceSquared,light{X}.vLightFalloff.y);preInfo.attenuation*=computeDirectionalLightFalloff_GLTF(light{X}.vLightDirection.xyz,preInfo.L,light{X}.vLightFalloff.z,light{X}.vLightFalloff.w); #elif defined(LIGHT_FALLOFF_PHYSICAL{X}) preInfo.attenuation=computeDistanceLightFalloff_Physical(preInfo.lightDistanceSquared);preInfo.attenuation*=computeDirectionalLightFalloff_Physical(light{X}.vLightDirection.xyz,preInfo.L,light{X}.vLightDirection.w); #elif defined(LIGHT_FALLOFF_STANDARD{X}) preInfo.attenuation=computeDistanceLightFalloff_Standard(preInfo.lightOffset,light{X}.vLightFalloff.x);preInfo.attenuation*=computeDirectionalLightFalloff_Standard(light{X}.vLightDirection.xyz,preInfo.L,light{X}.vLightDirection.w,light{X}.vLightData.w); #else preInfo.attenuation=computeDistanceLightFalloff(preInfo.lightOffset,preInfo.lightDistanceSquared,light{X}.vLightFalloff.x,light{X}.vLightFalloff.y);preInfo.attenuation*=computeDirectionalLightFalloff(light{X}.vLightDirection.xyz,preInfo.L,light{X}.vLightDirection.w,light{X}.vLightData.w,light{X}.vLightFalloff.z,light{X}.vLightFalloff.w); #endif #elif defined(POINTLIGHT{X}) #ifdef LIGHT_FALLOFF_GLTF{X} preInfo.attenuation=computeDistanceLightFalloff_GLTF(preInfo.lightDistanceSquared,light{X}.vLightFalloff.y); #elif defined(LIGHT_FALLOFF_PHYSICAL{X}) preInfo.attenuation=computeDistanceLightFalloff_Physical(preInfo.lightDistanceSquared); #elif defined(LIGHT_FALLOFF_STANDARD{X}) preInfo.attenuation=computeDistanceLightFalloff_Standard(preInfo.lightOffset,light{X}.vLightFalloff.x); #else preInfo.attenuation=computeDistanceLightFalloff(preInfo.lightOffset,preInfo.lightDistanceSquared,light{X}.vLightFalloff.x,light{X}.vLightFalloff.y); #endif #else preInfo.attenuation=1.0; #endif #ifdef HEMILIGHT{X} preInfo.roughness=roughness; #else preInfo.roughness=adjustRoughnessFromLightProperties(roughness,light{X}.vLightSpecular.a,preInfo.lightDistance); #endif #ifdef IRIDESCENCE preInfo.iridescenceIntensity=iridescenceIntensity; #endif #ifdef HEMILIGHT{X} info.diffuse=computeHemisphericDiffuseLighting(preInfo,light{X}.vLightDiffuse.rgb,light{X}.vLightGround); #elif defined(SS_TRANSLUCENCY) info.diffuse=computeDiffuseAndTransmittedLighting(preInfo,light{X}.vLightDiffuse.rgb,subSurfaceOut.transmittance); #else info.diffuse=computeDiffuseLighting(preInfo,light{X}.vLightDiffuse.rgb); #endif #ifdef SPECULARTERM #ifdef ANISOTROPIC info.specular=computeAnisotropicSpecularLighting(preInfo,viewDirectionW,normalW,anisotropicOut.anisotropicTangent,anisotropicOut.anisotropicBitangent,anisotropicOut.anisotropy,clearcoatOut.specularEnvironmentR0,specularEnvironmentR90,AARoughnessFactors.x,light{X}.vLightDiffuse.rgb); #else info.specular=computeSpecularLighting(preInfo,normalW,clearcoatOut.specularEnvironmentR0,specularEnvironmentR90,AARoughnessFactors.x,light{X}.vLightDiffuse.rgb); #endif #endif #ifdef SHEEN #ifdef SHEEN_LINKWITHALBEDO preInfo.roughness=sheenOut.sheenIntensity; #else #ifdef HEMILIGHT{X} preInfo.roughness=sheenOut.sheenRoughness; #else preInfo.roughness=adjustRoughnessFromLightProperties(sheenOut.sheenRoughness,light{X}.vLightSpecular.a,preInfo.lightDistance); #endif #endif info.sheen=computeSheenLighting(preInfo,normalW,sheenOut.sheenColor,specularEnvironmentR90,AARoughnessFactors.x,light{X}.vLightDiffuse.rgb); #endif #ifdef CLEARCOAT #ifdef HEMILIGHT{X} preInfo.roughness=clearcoatOut.clearCoatRoughness; #else preInfo.roughness=adjustRoughnessFromLightProperties(clearcoatOut.clearCoatRoughness,light{X}.vLightSpecular.a,preInfo.lightDistance); #endif info.clearCoat=computeClearCoatLighting(preInfo,clearcoatOut.clearCoatNormalW,clearcoatOut.clearCoatAARoughnessFactors.x,clearcoatOut.clearCoatIntensity,light{X}.vLightDiffuse.rgb); #ifdef CLEARCOAT_TINT absorption=computeClearCoatLightingAbsorption(clearcoatOut.clearCoatNdotVRefract,preInfo.L,clearcoatOut.clearCoatNormalW,clearcoatOut.clearCoatColor,clearcoatOut.clearCoatThickness,clearcoatOut.clearCoatIntensity);info.diffuse*=absorption; #ifdef SPECULARTERM info.specular*=absorption; #endif #endif info.diffuse*=info.clearCoat.w; #ifdef SPECULARTERM info.specular*=info.clearCoat.w; #endif #ifdef SHEEN info.sheen*=info.clearCoat.w; #endif #endif #else #ifdef SPOTLIGHT{X} info=computeSpotLighting(viewDirectionW,normalW,light{X}.vLightData,light{X}.vLightDirection,light{X}.vLightDiffuse.rgb,light{X}.vLightSpecular.rgb,light{X}.vLightDiffuse.a,glossiness); #elif defined(HEMILIGHT{X}) info=computeHemisphericLighting(viewDirectionW,normalW,light{X}.vLightData,light{X}.vLightDiffuse.rgb,light{X}.vLightSpecular.rgb,light{X}.vLightGround,glossiness); #elif defined(POINTLIGHT{X}) || defined(DIRLIGHT{X}) info=computeLighting(viewDirectionW,normalW,light{X}.vLightData,light{X}.vLightDiffuse.rgb,light{X}.vLightSpecular.rgb,light{X}.vLightDiffuse.a,glossiness); #endif #endif #ifdef PROJECTEDLIGHTTEXTURE{X} info.diffuse*=computeProjectionTextureDiffuseLighting(projectionLightSampler{X},textureProjectionMatrix{X}); #endif #endif #ifdef SHADOW{X} #ifdef SHADOWCSM{X} for (int i=0; i=0.) {index{X}=i;break;}} #ifdef SHADOWCSMUSESHADOWMAXZ{X} if (index{X}>=0) #endif { #if defined(SHADOWPCF{X}) #if defined(SHADOWLOWQUALITY{X}) shadow=computeShadowWithCSMPCF1(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #elif defined(SHADOWMEDIUMQUALITY{X}) shadow=computeShadowWithCSMPCF3(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #else shadow=computeShadowWithCSMPCF5(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #endif #elif defined(SHADOWPCSS{X}) #if defined(SHADOWLOWQUALITY{X}) shadow=computeShadowWithCSMPCSS16(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X}); #elif defined(SHADOWMEDIUMQUALITY{X}) shadow=computeShadowWithCSMPCSS32(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X}); #else shadow=computeShadowWithCSMPCSS64(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X}); #endif #else shadow=computeShadowCSM(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #endif #ifdef SHADOWCSMDEBUG{X} shadowDebug{X}=vec3(shadow)*vCascadeColorsMultiplier{X}[index{X}]; #endif #ifndef SHADOWCSMNOBLEND{X} float frustumLength=frustumLengths{X}[index{X}];float diffRatio=clamp(diff{X}/frustumLength,0.,1.)*cascadeBlendFactor{X};if (index{X}<(SHADOWCSMNUM_CASCADES{X}-1) && diffRatio<1.) {index{X}+=1;float nextShadow=0.; #if defined(SHADOWPCF{X}) #if defined(SHADOWLOWQUALITY{X}) nextShadow=computeShadowWithCSMPCF1(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #elif defined(SHADOWMEDIUMQUALITY{X}) nextShadow=computeShadowWithCSMPCF3(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #else nextShadow=computeShadowWithCSMPCF5(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #endif #elif defined(SHADOWPCSS{X}) #if defined(SHADOWLOWQUALITY{X}) nextShadow=computeShadowWithCSMPCSS16(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X}); #elif defined(SHADOWMEDIUMQUALITY{X}) nextShadow=computeShadowWithCSMPCSS32(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X}); #else nextShadow=computeShadowWithCSMPCSS64(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X}); #endif #else nextShadow=computeShadowCSM(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #endif shadow=mix(nextShadow,shadow,diffRatio); #ifdef SHADOWCSMDEBUG{X} shadowDebug{X}=mix(vec3(nextShadow)*vCascadeColorsMultiplier{X}[index{X}],shadowDebug{X},diffRatio); #endif } #endif } #elif defined(SHADOWCLOSEESM{X}) #if defined(SHADOWCUBE{X}) shadow=computeShadowWithCloseESMCube(vPositionW,light{X}.vLightData.xyz,shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.z,light{X}.depthValues); #else shadow=computeShadowWithCloseESM(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.z,light{X}.shadowsInfo.w); #endif #elif defined(SHADOWESM{X}) #if defined(SHADOWCUBE{X}) shadow=computeShadowWithESMCube(vPositionW,light{X}.vLightData.xyz,shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.z,light{X}.depthValues); #else shadow=computeShadowWithESM(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.z,light{X}.shadowsInfo.w); #endif #elif defined(SHADOWPOISSON{X}) #if defined(SHADOWCUBE{X}) shadow=computeShadowWithPoissonSamplingCube(vPositionW,light{X}.vLightData.xyz,shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.x,light{X}.depthValues); #else shadow=computeShadowWithPoissonSampling(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #endif #elif defined(SHADOWPCF{X}) #if defined(SHADOWLOWQUALITY{X}) shadow=computeShadowWithPCF1(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #elif defined(SHADOWMEDIUMQUALITY{X}) shadow=computeShadowWithPCF3(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #else shadow=computeShadowWithPCF5(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #endif #elif defined(SHADOWPCSS{X}) #if defined(SHADOWLOWQUALITY{X}) shadow=computeShadowWithPCSS16(vPositionFromLight{X},vDepthMetric{X},depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #elif defined(SHADOWMEDIUMQUALITY{X}) shadow=computeShadowWithPCSS32(vPositionFromLight{X},vDepthMetric{X},depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #else shadow=computeShadowWithPCSS64(vPositionFromLight{X},vDepthMetric{X},depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #endif #else #if defined(SHADOWCUBE{X}) shadow=computeShadowCube(vPositionW,light{X}.vLightData.xyz,shadowSampler{X},light{X}.shadowsInfo.x,light{X}.depthValues); #else shadow=computeShadow(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w); #endif #endif #ifdef SHADOWONLY #ifndef SHADOWINUSE #define SHADOWINUSE #endif globalShadow+=shadow;shadowLightCount+=1.0; #endif #else shadow=1.; #endif aggShadow+=shadow;numLights+=1.0; #ifndef SHADOWONLY #ifdef CUSTOMUSERLIGHTING diffuseBase+=computeCustomDiffuseLighting(info,diffuseBase,shadow); #ifdef SPECULARTERM specularBase+=computeCustomSpecularLighting(info,specularBase,shadow); #endif #elif defined(LIGHTMAP) && defined(LIGHTMAPEXCLUDED{X}) diffuseBase+=lightmapColor.rgb*shadow; #ifdef SPECULARTERM #ifndef LIGHTMAPNOSPECULAR{X} specularBase+=info.specular*shadow*lightmapColor.rgb; #endif #endif #ifdef CLEARCOAT #ifndef LIGHTMAPNOSPECULAR{X} clearCoatBase+=info.clearCoat.rgb*shadow*lightmapColor.rgb; #endif #endif #ifdef SHEEN #ifndef LIGHTMAPNOSPECULAR{X} sheenBase+=info.sheen.rgb*shadow; #endif #endif #else #ifdef SHADOWCSMDEBUG{X} diffuseBase+=info.diffuse*shadowDebug{X}; #else diffuseBase+=info.diffuse*shadow; #endif #ifdef SPECULARTERM specularBase+=info.specular*shadow; #endif #ifdef CLEARCOAT clearCoatBase+=info.clearCoat.rgb*shadow; #endif #ifdef SHEEN sheenBase+=info.sheen.rgb*shadow; #endif #endif #endif #endif `; Le.IncludesShadersStore[rde] = nde; const ide = "logDepthFragment", sde = `#ifdef LOGARITHMICDEPTH gl_FragDepthEXT=log2(vFragmentDepth)*logarithmicDepthConstant*0.5; #endif `; Le.IncludesShadersStore[ide] = sde; const ade = "fogFragment", ode = `#ifdef FOG float fog=CalcFogFactor(); #ifdef PBR fog=toLinearSpace(fog); #endif color.rgb=mix(vFogColor,color.rgb,fog); #endif `; Le.IncludesShadersStore[ade] = ode; const fde = "oitFragment", Ade = `#ifdef ORDER_INDEPENDENT_TRANSPARENCY float fragDepth=gl_FragCoord.z; #ifdef ORDER_INDEPENDENT_TRANSPARENCY_16BITS uint halfFloat=packHalf2x16(vec2(fragDepth));vec2 full=unpackHalf2x16(halfFloat);fragDepth=full.x; #endif ivec2 fragCoord=ivec2(gl_FragCoord.xy);vec2 lastDepth=texelFetch(oitDepthSampler,fragCoord,0).rg;vec4 lastFrontColor=texelFetch(oitFrontColorSampler,fragCoord,0);depth.rg=vec2(-MAX_DEPTH);frontColor=lastFrontColor;backColor=vec4(0.0); #ifdef USE_REVERSE_DEPTHBUFFER float furthestDepth=-lastDepth.x;float nearestDepth=lastDepth.y; #else float nearestDepth=-lastDepth.x;float furthestDepth=lastDepth.y; #endif float alphaMultiplier=1.0-lastFrontColor.a; #ifdef USE_REVERSE_DEPTHBUFFER if (fragDepth>nearestDepth || fragDepthfurthestDepth) { #endif return;} #ifdef USE_REVERSE_DEPTHBUFFER if (fragDepthfurthestDepth) { #else if (fragDepth>nearestDepth && fragDepth #if defined(BUMP) || !defined(NORMAL) #extension GL_OES_standard_derivatives : enable #endif #include[SCENE_MRT_COUNT] #include #define CUSTOM_FRAGMENT_BEGIN #ifdef LOGARITHMICDEPTH #extension GL_EXT_frag_depth : enable #endif varying vec3 vPositionW; #ifdef NORMAL varying vec3 vNormalW; #endif #if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES) varying vec4 vColor; #endif #include[1..7] #include #include<__decl__lightFragment>[0..maxSimultaneousLights] #include #include #include(_DEFINENAME_,DIFFUSE,_VARYINGNAME_,Diffuse,_SAMPLERNAME_,diffuse) #include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient,_SAMPLERNAME_,ambient) #include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity,_SAMPLERNAME_,opacity) #include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive,_SAMPLERNAME_,emissive) #include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap,_SAMPLERNAME_,lightmap) #include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal,_SAMPLERNAME_,decal) #ifdef REFRACTION #ifdef REFRACTIONMAP_3D uniform samplerCube refractionCubeSampler; #else uniform sampler2D refraction2DSampler; #endif #endif #if defined(SPECULARTERM) #include(_DEFINENAME_,SPECULAR,_VARYINGNAME_,Specular,_SAMPLERNAME_,specular) #endif #include #ifdef REFLECTION #ifdef REFLECTIONMAP_3D uniform samplerCube reflectionCubeSampler; #else uniform sampler2D reflection2DSampler; #endif #ifdef REFLECTIONMAP_SKYBOX varying vec3 vPositionUVW; #else #if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED) varying vec3 vDirectionW; #endif #endif #include #endif #include #include #include #include #include #include #include #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #define CUSTOM_FRAGMENT_MAIN_BEGIN #include vec3 viewDirectionW=normalize(vEyePosition.xyz-vPositionW);vec4 baseColor=vec4(1.,1.,1.,1.);vec3 diffuseColor=vDiffuseColor.rgb;float alpha=vDiffuseColor.a; #ifdef NORMAL vec3 normalW=normalize(vNormalW); #else vec3 normalW=normalize(-cross(dFdx(vPositionW),dFdy(vPositionW))); #endif #include #ifdef TWOSIDEDLIGHTING normalW=gl_FrontFacing ? normalW : -normalW; #endif #ifdef DIFFUSE baseColor=texture2D(diffuseSampler,vDiffuseUV+uvOffset); #if defined(ALPHATEST) && !defined(ALPHATEST_AFTERALLALPHACOMPUTATIONS) if (baseColor.a(surfaceAlbedo,baseColor,GAMMADECAL,_GAMMADECAL_NOTUSED_) #endif #include #if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES) baseColor.rgb*=vColor.rgb; #endif #ifdef DETAIL baseColor.rgb=baseColor.rgb*2.0*mix(0.5,detailColor.r,vDetailInfos.y); #endif #if defined(DECAL) && defined(DECAL_AFTER_DETAIL) vec4 decalColor=texture2D(decalSampler,vDecalUV+uvOffset); #include(surfaceAlbedo,baseColor,GAMMADECAL,_GAMMADECAL_NOTUSED_) #endif #define CUSTOM_FRAGMENT_UPDATE_DIFFUSE vec3 baseAmbientColor=vec3(1.,1.,1.); #ifdef AMBIENT baseAmbientColor=texture2D(ambientSampler,vAmbientUV+uvOffset).rgb*vAmbientInfos.y; #endif #define CUSTOM_FRAGMENT_BEFORE_LIGHTS #ifdef SPECULARTERM float glossiness=vSpecularColor.a;vec3 specularColor=vSpecularColor.rgb; #ifdef SPECULAR vec4 specularMapColor=texture2D(specularSampler,vSpecularUV+uvOffset);specularColor=specularMapColor.rgb; #ifdef GLOSSINESS glossiness=glossiness*specularMapColor.a; #endif #endif #else float glossiness=0.; #endif vec3 diffuseBase=vec3(0.,0.,0.);lightingInfo info; #ifdef SPECULARTERM vec3 specularBase=vec3(0.,0.,0.); #endif float shadow=1.;float aggShadow=0.;float numLights=0.; #ifdef LIGHTMAP vec4 lightmapColor=texture2D(lightmapSampler,vLightmapUV+uvOffset); #ifdef RGBDLIGHTMAP lightmapColor.rgb=fromRGBD(lightmapColor); #endif lightmapColor.rgb*=vLightmapInfos.y; #endif #include[0..maxSimultaneousLights] aggShadow=aggShadow/numLights;vec4 refractionColor=vec4(0.,0.,0.,1.); #ifdef REFRACTION vec3 refractionVector=normalize(refract(-viewDirectionW,normalW,vRefractionInfos.y)); #ifdef REFRACTIONMAP_3D #ifdef USE_LOCAL_REFRACTIONMAP_CUBIC refractionVector=parallaxCorrectNormal(vPositionW,refractionVector,vRefractionSize,vRefractionPosition); #endif refractionVector.y=refractionVector.y*vRefractionInfos.w;vec4 refractionLookup=textureCube(refractionCubeSampler,refractionVector);if (dot(refractionVector,viewDirectionW)<1.0) {refractionColor=refractionLookup;} #else vec3 vRefractionUVW=vec3(refractionMatrix*(view*vec4(vPositionW+refractionVector*vRefractionInfos.z,1.0)));vec2 refractionCoords=vRefractionUVW.xy/vRefractionUVW.z;refractionCoords.y=1.0-refractionCoords.y;refractionColor=texture2D(refraction2DSampler,refractionCoords); #endif #ifdef RGBDREFRACTION refractionColor.rgb=fromRGBD(refractionColor); #endif #ifdef IS_REFRACTION_LINEAR refractionColor.rgb=toGammaSpace(refractionColor.rgb); #endif refractionColor.rgb*=vRefractionInfos.x; #endif vec4 reflectionColor=vec4(0.,0.,0.,1.); #ifdef REFLECTION vec3 vReflectionUVW=computeReflectionCoords(vec4(vPositionW,1.0),normalW); #ifdef REFLECTIONMAP_OPPOSITEZ vReflectionUVW.z*=-1.0; #endif #ifdef REFLECTIONMAP_3D #ifdef ROUGHNESS float bias=vReflectionInfos.y; #ifdef SPECULARTERM #ifdef SPECULAR #ifdef GLOSSINESS bias*=(1.0-specularMapColor.a); #endif #endif #endif reflectionColor=textureCube(reflectionCubeSampler,vReflectionUVW,bias); #else reflectionColor=textureCube(reflectionCubeSampler,vReflectionUVW); #endif #else vec2 coords=vReflectionUVW.xy; #ifdef REFLECTIONMAP_PROJECTION coords/=vReflectionUVW.z; #endif coords.y=1.0-coords.y;reflectionColor=texture2D(reflection2DSampler,coords); #endif #ifdef RGBDREFLECTION reflectionColor.rgb=fromRGBD(reflectionColor); #endif #ifdef IS_REFLECTION_LINEAR reflectionColor.rgb=toGammaSpace(reflectionColor.rgb); #endif reflectionColor.rgb*=vReflectionInfos.x; #ifdef REFLECTIONFRESNEL float reflectionFresnelTerm=computeFresnelTerm(viewDirectionW,normalW,reflectionRightColor.a,reflectionLeftColor.a); #ifdef REFLECTIONFRESNELFROMSPECULAR #ifdef SPECULARTERM reflectionColor.rgb*=specularColor.rgb*(1.0-reflectionFresnelTerm)+reflectionFresnelTerm*reflectionRightColor.rgb; #else reflectionColor.rgb*=reflectionLeftColor.rgb*(1.0-reflectionFresnelTerm)+reflectionFresnelTerm*reflectionRightColor.rgb; #endif #else reflectionColor.rgb*=reflectionLeftColor.rgb*(1.0-reflectionFresnelTerm)+reflectionFresnelTerm*reflectionRightColor.rgb; #endif #endif #endif #ifdef REFRACTIONFRESNEL float refractionFresnelTerm=computeFresnelTerm(viewDirectionW,normalW,refractionRightColor.a,refractionLeftColor.a);refractionColor.rgb*=refractionLeftColor.rgb*(1.0-refractionFresnelTerm)+refractionFresnelTerm*refractionRightColor.rgb; #endif #ifdef OPACITY vec4 opacityMap=texture2D(opacitySampler,vOpacityUV+uvOffset); #ifdef OPACITYRGB opacityMap.rgb=opacityMap.rgb*vec3(0.3,0.59,0.11);alpha*=(opacityMap.x+opacityMap.y+opacityMap.z)* vOpacityInfos.y; #else alpha*=opacityMap.a*vOpacityInfos.y; #endif #endif #if defined(VERTEXALPHA) || defined(INSTANCESCOLOR) && defined(INSTANCES) alpha*=vColor.a; #endif #ifdef OPACITYFRESNEL float opacityFresnelTerm=computeFresnelTerm(viewDirectionW,normalW,opacityParts.z,opacityParts.w);alpha+=opacityParts.x*(1.0-opacityFresnelTerm)+opacityFresnelTerm*opacityParts.y; #endif #ifdef ALPHATEST #ifdef ALPHATEST_AFTERALLALPHACOMPUTATIONS if (alpha #include #ifdef IMAGEPROCESSINGPOSTPROCESS color.rgb=toLinearSpace(color.rgb); #else #ifdef IMAGEPROCESSING color.rgb=toLinearSpace(color.rgb);color=applyImageProcessing(color); #endif #endif color.a*=visibility; #ifdef PREMULTIPLYALPHA color.rgb*=color.a; #endif #define CUSTOM_FRAGMENT_BEFORE_FRAGCOLOR #ifdef PREPASS float writeGeometryInfo=color.a>0.4 ? 1.0 : 0.0;gl_FragData[0]=color; #ifdef PREPASS_POSITION gl_FragData[PREPASS_POSITION_INDEX]=vec4(vPositionW,writeGeometryInfo); #endif #ifdef PREPASS_VELOCITY vec2 a=(vCurrentPosition.xy/vCurrentPosition.w)*0.5+0.5;vec2 b=(vPreviousPosition.xy/vPreviousPosition.w)*0.5+0.5;vec2 velocity=abs(a-b);velocity=vec2(pow(velocity.x,1.0/3.0),pow(velocity.y,1.0/3.0))*sign(a-b)*0.5+0.5;gl_FragData[PREPASS_VELOCITY_INDEX]=vec4(velocity,0.0,writeGeometryInfo); #endif #ifdef PREPASS_IRRADIANCE gl_FragData[PREPASS_IRRADIANCE_INDEX]=vec4(0.0,0.0,0.0,writeGeometryInfo); #endif #ifdef PREPASS_DEPTH gl_FragData[PREPASS_DEPTH_INDEX]=vec4(vViewPos.z,0.0,0.0,writeGeometryInfo); #endif #ifdef PREPASS_NORMAL #ifdef PREPASS_NORMAL_WORLDSPACE gl_FragData[PREPASS_NORMAL_INDEX]=vec4(normalW,writeGeometryInfo); #else gl_FragData[PREPASS_NORMAL_INDEX]=vec4(normalize((view*vec4(normalW,0.0)).rgb),writeGeometryInfo); #endif #endif #ifdef PREPASS_ALBEDO_SQRT gl_FragData[PREPASS_ALBEDO_SQRT_INDEX]=vec4(0.0,0.0,0.0,writeGeometryInfo); #endif #ifdef PREPASS_REFLECTIVITY #if defined(SPECULARTERM) #if defined(SPECULAR) gl_FragData[PREPASS_REFLECTIVITY_INDEX]=vec4(toLinearSpace(specularMapColor))*writeGeometryInfo; #else gl_FragData[PREPASS_REFLECTIVITY_INDEX]=vec4(toLinearSpace(specularColor),1.0)*writeGeometryInfo; #endif #else gl_FragData[PREPASS_REFLECTIVITY_INDEX]=vec4(0.0,0.0,0.0,1.0)*writeGeometryInfo; #endif #endif #endif #if !defined(PREPASS) || defined(WEBGL2) gl_FragColor=color; #endif #include #if ORDER_INDEPENDENT_TRANSPARENCY if (fragDepth==nearestDepth) {frontColor.rgb+=color.rgb*color.a*alphaMultiplier;frontColor.a=1.0-alphaMultiplier*(1.0-color.a);} else {backColor+=color;} #endif #define CUSTOM_FRAGMENT_MAIN_END } `; Le.ShadersStore[dde] = vde; const ude = "decalVertexDeclaration", lde = `#ifdef DECAL uniform vec4 vDecalInfos;uniform mat4 decalMatrix; #endif `; Le.IncludesShadersStore[ude] = lde; const Pde = "defaultVertexDeclaration", cde = `uniform mat4 viewProjection;uniform mat4 view; #ifdef DIFFUSE uniform mat4 diffuseMatrix;uniform vec2 vDiffuseInfos; #endif #ifdef AMBIENT uniform mat4 ambientMatrix;uniform vec2 vAmbientInfos; #endif #ifdef OPACITY uniform mat4 opacityMatrix;uniform vec2 vOpacityInfos; #endif #ifdef EMISSIVE uniform vec2 vEmissiveInfos;uniform mat4 emissiveMatrix; #endif #ifdef LIGHTMAP uniform vec2 vLightmapInfos;uniform mat4 lightmapMatrix; #endif #if defined(SPECULAR) && defined(SPECULARTERM) uniform vec2 vSpecularInfos;uniform mat4 specularMatrix; #endif #ifdef BUMP uniform vec3 vBumpInfos;uniform mat4 bumpMatrix; #endif #ifdef REFLECTION uniform mat4 reflectionMatrix; #endif #ifdef POINTSIZE uniform float pointSize; #endif #ifdef DETAIL uniform vec4 vDetailInfos;uniform mat4 detailMatrix; #endif #include #define ADDITIONAL_VERTEX_DECLARATION `; Le.IncludesShadersStore[Pde] = cde; const pde = "uvAttributeDeclaration", hde = `#ifdef UV{X} attribute vec2 uv{X}; #endif `; Le.IncludesShadersStore[pde] = hde; const Hde = "bonesDeclaration", gde = `#if NUM_BONE_INFLUENCERS>0 attribute vec4 matricesIndices;attribute vec4 matricesWeights; #if NUM_BONE_INFLUENCERS>4 attribute vec4 matricesIndicesExtra;attribute vec4 matricesWeightsExtra; #endif #ifndef BAKED_VERTEX_ANIMATION_TEXTURE #ifdef BONETEXTURE uniform highp sampler2D boneSampler;uniform float boneTextureWidth; #else uniform mat4 mBones[BonesPerMesh]; #endif #ifdef BONES_VELOCITY_ENABLED uniform mat4 mPreviousBones[BonesPerMesh]; #endif #ifdef BONETEXTURE #define inline mat4 readMatrixFromRawSampler(sampler2D smp,float index) {float offset=index *4.0;float dx=1.0/boneTextureWidth;vec4 m0=texture2D(smp,vec2(dx*(offset+0.5),0.));vec4 m1=texture2D(smp,vec2(dx*(offset+1.5),0.));vec4 m2=texture2D(smp,vec2(dx*(offset+2.5),0.));vec4 m3=texture2D(smp,vec2(dx*(offset+3.5),0.));return mat4(m0,m1,m2,m3);} #endif #endif #endif `; Le.IncludesShadersStore[Hde] = gde; const Xde = "bakedVertexAnimationDeclaration", Tde = `#ifdef BAKED_VERTEX_ANIMATION_TEXTURE uniform float bakedVertexAnimationTime;uniform vec2 bakedVertexAnimationTextureSizeInverted;uniform vec4 bakedVertexAnimationSettings;uniform sampler2D bakedVertexAnimationTexture; #ifdef INSTANCES attribute vec4 bakedVertexAnimationSettingsInstanced; #endif #define inline mat4 readMatrixFromRawSamplerVAT(sampler2D smp,float index,float frame) {float offset=index*4.0;float frameUV=(frame+0.5)*bakedVertexAnimationTextureSizeInverted.y;float dx=bakedVertexAnimationTextureSizeInverted.x;vec4 m0=texture2D(smp,vec2(dx*(offset+0.5),frameUV));vec4 m1=texture2D(smp,vec2(dx*(offset+1.5),frameUV));vec4 m2=texture2D(smp,vec2(dx*(offset+2.5),frameUV));vec4 m3=texture2D(smp,vec2(dx*(offset+3.5),frameUV));return mat4(m0,m1,m2,m3);} #endif `; Le.IncludesShadersStore[Xde] = Tde; const qde = "instancesDeclaration", bde = `#ifdef INSTANCES attribute vec4 world0;attribute vec4 world1;attribute vec4 world2;attribute vec4 world3; #ifdef INSTANCESCOLOR attribute vec4 instanceColor; #endif #if defined(THIN_INSTANCES) && !defined(WORLD_UBO) uniform mat4 world; #endif #if defined(VELOCITY) || defined(PREPASS_VELOCITY) attribute vec4 previousWorld0;attribute vec4 previousWorld1;attribute vec4 previousWorld2;attribute vec4 previousWorld3; #ifdef THIN_INSTANCES uniform mat4 previousWorld; #endif #endif #else #if !defined(WORLD_UBO) uniform mat4 world; #endif #if defined(VELOCITY) || defined(PREPASS_VELOCITY) uniform mat4 previousWorld; #endif #endif `; Le.IncludesShadersStore[qde] = bde; const xde = "prePassVertexDeclaration", Dde = `#ifdef PREPASS #ifdef PREPASS_DEPTH varying vec3 vViewPos; #endif #ifdef PREPASS_VELOCITY uniform mat4 previousViewProjection;varying vec4 vCurrentPosition;varying vec4 vPreviousPosition; #endif #endif `; Le.IncludesShadersStore[xde] = Dde; const jde = "samplerVertexDeclaration", wde = `#if defined(_DEFINENAME_) && _DEFINENAME_DIRECTUV==0 varying vec2 v_VARYINGNAME_UV; #endif `; Le.IncludesShadersStore[jde] = wde; const mde = "bumpVertexDeclaration", Bde = `#if defined(BUMP) || defined(PARALLAX) || defined(CLEARCOAT_BUMP) || defined(ANISOTROPIC) #if defined(TANGENT) && defined(NORMAL) varying mat3 vTBN; #endif #endif `; Le.IncludesShadersStore[mde] = Bde; const Wde = "clipPlaneVertexDeclaration", Sde = `#ifdef CLIPPLANE uniform vec4 vClipPlane;varying float fClipDistance; #endif #ifdef CLIPPLANE2 uniform vec4 vClipPlane2;varying float fClipDistance2; #endif #ifdef CLIPPLANE3 uniform vec4 vClipPlane3;varying float fClipDistance3; #endif #ifdef CLIPPLANE4 uniform vec4 vClipPlane4;varying float fClipDistance4; #endif #ifdef CLIPPLANE5 uniform vec4 vClipPlane5;varying float fClipDistance5; #endif #ifdef CLIPPLANE6 uniform vec4 vClipPlane6;varying float fClipDistance6; #endif `; Le.IncludesShadersStore[Wde] = Sde; const Ude = "fogVertexDeclaration", Ide = `#ifdef FOG varying vec3 vFogDistance; #endif `; Le.IncludesShadersStore[Ude] = Ide; const Rde = "lightVxFragmentDeclaration", Vde = `#ifdef LIGHT{X} uniform vec4 vLightData{X};uniform vec4 vLightDiffuse{X}; #ifdef SPECULARTERM uniform vec4 vLightSpecular{X}; #else vec4 vLightSpecular{X}=vec4(0.); #endif #ifdef SHADOW{X} #ifdef SHADOWCSM{X} uniform mat4 lightMatrix{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromLight{X}[SHADOWCSMNUM_CASCADES{X}];varying float vDepthMetric{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromCamera{X}; #elif defined(SHADOWCUBE{X}) #else varying vec4 vPositionFromLight{X};varying float vDepthMetric{X};uniform mat4 lightMatrix{X}; #endif uniform vec4 shadowsInfo{X};uniform vec2 depthValues{X}; #endif #ifdef SPOTLIGHT{X} uniform vec4 vLightDirection{X};uniform vec4 vLightFalloff{X}; #elif defined(POINTLIGHT{X}) uniform vec4 vLightFalloff{X}; #elif defined(HEMILIGHT{X}) uniform vec3 vLightGround{X}; #endif #endif `; Le.IncludesShadersStore[Rde] = Vde; const Cde = "lightVxUboDeclaration", Ode = `#ifdef LIGHT{X} uniform Light{X} {vec4 vLightData;vec4 vLightDiffuse;vec4 vLightSpecular; #ifdef SPOTLIGHT{X} vec4 vLightDirection;vec4 vLightFalloff; #elif defined(POINTLIGHT{X}) vec4 vLightFalloff; #elif defined(HEMILIGHT{X}) vec3 vLightGround; #endif vec4 shadowsInfo;vec2 depthValues;} light{X}; #ifdef SHADOW{X} #ifdef SHADOWCSM{X} uniform mat4 lightMatrix{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromLight{X}[SHADOWCSMNUM_CASCADES{X}];varying float vDepthMetric{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromCamera{X}; #elif defined(SHADOWCUBE{X}) #else varying vec4 vPositionFromLight{X};varying float vDepthMetric{X};uniform mat4 lightMatrix{X}; #endif #endif #endif `; Le.IncludesShadersStore[Cde] = Ode; const yde = "morphTargetsVertexGlobalDeclaration", kde = `#ifdef MORPHTARGETS uniform float morphTargetInfluences[NUM_MORPH_INFLUENCERS]; #ifdef MORPHTARGETS_TEXTURE uniform float morphTargetTextureIndices[NUM_MORPH_INFLUENCERS];uniform vec3 morphTargetTextureInfo;uniform highp sampler2DArray morphTargets;vec3 readVector3FromRawSampler(int targetIndex,float vertexIndex) { float y=floor(vertexIndex/morphTargetTextureInfo.y);float x=vertexIndex-y*morphTargetTextureInfo.y;vec3 textureUV=vec3((x+0.5)/morphTargetTextureInfo.y,(y+0.5)/morphTargetTextureInfo.z,morphTargetTextureIndices[targetIndex]);return texture(morphTargets,textureUV).xyz;} #endif #endif `; Le.IncludesShadersStore[yde] = kde; const Ede = "morphTargetsVertexDeclaration", Fde = `#ifdef MORPHTARGETS #ifndef MORPHTARGETS_TEXTURE attribute vec3 position{X}; #ifdef MORPHTARGETS_NORMAL attribute vec3 normal{X}; #endif #ifdef MORPHTARGETS_TANGENT attribute vec3 tangent{X}; #endif #ifdef MORPHTARGETS_UV attribute vec2 uv_{X}; #endif #endif #endif `; Le.IncludesShadersStore[Ede] = Fde; const Nde = "morphTargetsVertexGlobal", Qde = `#ifdef MORPHTARGETS #ifdef MORPHTARGETS_TEXTURE float vertexID; #endif #endif `; Le.IncludesShadersStore[Nde] = Qde; const Yde = "morphTargetsVertex", Mde = `#ifdef MORPHTARGETS #ifdef MORPHTARGETS_TEXTURE vertexID=float(gl_VertexID)*morphTargetTextureInfo.x;positionUpdated+=(readVector3FromRawSampler({X},vertexID)-position)*morphTargetInfluences[{X}];vertexID+=1.0; #ifdef MORPHTARGETS_NORMAL normalUpdated+=(readVector3FromRawSampler({X},vertexID) -normal)*morphTargetInfluences[{X}];vertexID+=1.0; #endif #ifdef MORPHTARGETS_UV uvUpdated+=(readVector3FromRawSampler({X},vertexID).xy-uv)*morphTargetInfluences[{X}];vertexID+=1.0; #endif #ifdef MORPHTARGETS_TANGENT tangentUpdated.xyz+=(readVector3FromRawSampler({X},vertexID) -tangent.xyz)*morphTargetInfluences[{X}]; #endif #else positionUpdated+=(position{X}-position)*morphTargetInfluences[{X}]; #ifdef MORPHTARGETS_NORMAL normalUpdated+=(normal{X}-normal)*morphTargetInfluences[{X}]; #endif #ifdef MORPHTARGETS_TANGENT tangentUpdated.xyz+=(tangent{X}-tangent.xyz)*morphTargetInfluences[{X}]; #endif #ifdef MORPHTARGETS_UV uvUpdated+=(uv_{X}-uv)*morphTargetInfluences[{X}]; #endif #endif #endif `; Le.IncludesShadersStore[Yde] = Mde; const Lde = "instancesVertex", Kde = `#ifdef INSTANCES mat4 finalWorld=mat4(world0,world1,world2,world3); #if defined(PREPASS_VELOCITY) || defined(VELOCITY) mat4 finalPreviousWorld=mat4(previousWorld0,previousWorld1,previousWorld2,previousWorld3); #endif #ifdef THIN_INSTANCES finalWorld=world*finalWorld; #if defined(PREPASS_VELOCITY) || defined(VELOCITY) finalPreviousWorld=previousWorld*finalPreviousWorld; #endif #endif #else mat4 finalWorld=world; #if defined(PREPASS_VELOCITY) || defined(VELOCITY) mat4 finalPreviousWorld=previousWorld; #endif #endif `; Le.IncludesShadersStore[Lde] = Kde; const Jde = "bonesVertex", zde = `#ifndef BAKED_VERTEX_ANIMATION_TEXTURE #if NUM_BONE_INFLUENCERS>0 mat4 influence; #ifdef BONETEXTURE influence=readMatrixFromRawSampler(boneSampler,matricesIndices[0])*matricesWeights[0]; #if NUM_BONE_INFLUENCERS>1 influence+=readMatrixFromRawSampler(boneSampler,matricesIndices[1])*matricesWeights[1]; #endif #if NUM_BONE_INFLUENCERS>2 influence+=readMatrixFromRawSampler(boneSampler,matricesIndices[2])*matricesWeights[2]; #endif #if NUM_BONE_INFLUENCERS>3 influence+=readMatrixFromRawSampler(boneSampler,matricesIndices[3])*matricesWeights[3]; #endif #if NUM_BONE_INFLUENCERS>4 influence+=readMatrixFromRawSampler(boneSampler,matricesIndicesExtra[0])*matricesWeightsExtra[0]; #endif #if NUM_BONE_INFLUENCERS>5 influence+=readMatrixFromRawSampler(boneSampler,matricesIndicesExtra[1])*matricesWeightsExtra[1]; #endif #if NUM_BONE_INFLUENCERS>6 influence+=readMatrixFromRawSampler(boneSampler,matricesIndicesExtra[2])*matricesWeightsExtra[2]; #endif #if NUM_BONE_INFLUENCERS>7 influence+=readMatrixFromRawSampler(boneSampler,matricesIndicesExtra[3])*matricesWeightsExtra[3]; #endif #else influence=mBones[int(matricesIndices[0])]*matricesWeights[0]; #if NUM_BONE_INFLUENCERS>1 influence+=mBones[int(matricesIndices[1])]*matricesWeights[1]; #endif #if NUM_BONE_INFLUENCERS>2 influence+=mBones[int(matricesIndices[2])]*matricesWeights[2]; #endif #if NUM_BONE_INFLUENCERS>3 influence+=mBones[int(matricesIndices[3])]*matricesWeights[3]; #endif #if NUM_BONE_INFLUENCERS>4 influence+=mBones[int(matricesIndicesExtra[0])]*matricesWeightsExtra[0]; #endif #if NUM_BONE_INFLUENCERS>5 influence+=mBones[int(matricesIndicesExtra[1])]*matricesWeightsExtra[1]; #endif #if NUM_BONE_INFLUENCERS>6 influence+=mBones[int(matricesIndicesExtra[2])]*matricesWeightsExtra[2]; #endif #if NUM_BONE_INFLUENCERS>7 influence+=mBones[int(matricesIndicesExtra[3])]*matricesWeightsExtra[3]; #endif #endif finalWorld=finalWorld*influence; #endif #endif `; Le.IncludesShadersStore[Jde] = zde; const Gde = "bakedVertexAnimation", Zde = `#ifdef BAKED_VERTEX_ANIMATION_TEXTURE { #ifdef INSTANCES #define BVASNAME bakedVertexAnimationSettingsInstanced #else #define BVASNAME bakedVertexAnimationSettings #endif float VATStartFrame=BVASNAME.x;float VATEndFrame=BVASNAME.y;float VATOffsetFrame=BVASNAME.z;float VATSpeed=BVASNAME.w;float totalFrames=VATEndFrame-VATStartFrame+1.0;float time=bakedVertexAnimationTime*VATSpeed/totalFrames;float frameCorrection=time<1.0 ? 0.0 : 1.0;float numOfFrames=totalFrames-frameCorrection;float VATFrameNum=fract(time)*numOfFrames;VATFrameNum=mod(VATFrameNum+VATOffsetFrame,numOfFrames);VATFrameNum=floor(VATFrameNum);VATFrameNum+=VATStartFrame+frameCorrection;mat4 VATInfluence;VATInfluence=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndices[0],VATFrameNum)*matricesWeights[0]; #if NUM_BONE_INFLUENCERS>1 VATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndices[1],VATFrameNum)*matricesWeights[1]; #endif #if NUM_BONE_INFLUENCERS>2 VATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndices[2],VATFrameNum)*matricesWeights[2]; #endif #if NUM_BONE_INFLUENCERS>3 VATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndices[3],VATFrameNum)*matricesWeights[3]; #endif #if NUM_BONE_INFLUENCERS>4 VATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndicesExtra[0],VATFrameNum)*matricesWeightsExtra[0]; #endif #if NUM_BONE_INFLUENCERS>5 VATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndicesExtra[1],VATFrameNum)*matricesWeightsExtra[1]; #endif #if NUM_BONE_INFLUENCERS>6 VATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndicesExtra[2],VATFrameNum)*matricesWeightsExtra[2]; #endif #if NUM_BONE_INFLUENCERS>7 VATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndicesExtra[3],VATFrameNum)*matricesWeightsExtra[3]; #endif finalWorld=finalWorld*VATInfluence;} #endif `; Le.IncludesShadersStore[Gde] = Zde; const _de = "prePassVertex", $de = `#ifdef PREPASS_DEPTH vViewPos=(view*worldPos).rgb; #endif #if defined(PREPASS_VELOCITY) && defined(BONES_VELOCITY_ENABLED) vCurrentPosition=viewProjection*worldPos; #if NUM_BONE_INFLUENCERS>0 mat4 previousInfluence;previousInfluence=mPreviousBones[int(matricesIndices[0])]*matricesWeights[0]; #if NUM_BONE_INFLUENCERS>1 previousInfluence+=mPreviousBones[int(matricesIndices[1])]*matricesWeights[1]; #endif #if NUM_BONE_INFLUENCERS>2 previousInfluence+=mPreviousBones[int(matricesIndices[2])]*matricesWeights[2]; #endif #if NUM_BONE_INFLUENCERS>3 previousInfluence+=mPreviousBones[int(matricesIndices[3])]*matricesWeights[3]; #endif #if NUM_BONE_INFLUENCERS>4 previousInfluence+=mPreviousBones[int(matricesIndicesExtra[0])]*matricesWeightsExtra[0]; #endif #if NUM_BONE_INFLUENCERS>5 previousInfluence+=mPreviousBones[int(matricesIndicesExtra[1])]*matricesWeightsExtra[1]; #endif #if NUM_BONE_INFLUENCERS>6 previousInfluence+=mPreviousBones[int(matricesIndicesExtra[2])]*matricesWeightsExtra[2]; #endif #if NUM_BONE_INFLUENCERS>7 previousInfluence+=mPreviousBones[int(matricesIndicesExtra[3])]*matricesWeightsExtra[3]; #endif vPreviousPosition=previousViewProjection*finalPreviousWorld*previousInfluence*vec4(positionUpdated,1.0); #else vPreviousPosition=previousViewProjection*finalPreviousWorld*vec4(positionUpdated,1.0); #endif #endif `; Le.IncludesShadersStore[_de] = $de; const eve = "uvVariableDeclaration", tve = `#if !defined(UV{X}) && defined(MAINUV{X}) vec2 uv{X}=vec2(0.,0.); #endif #ifdef MAINUV{X} vMainUV{X}=uv{X}; #endif `; Le.IncludesShadersStore[eve] = tve; const rve = "samplerVertexImplementation", nve = `#if defined(_DEFINENAME_) && _DEFINENAME_DIRECTUV==0 if (v_INFONAME_==0.) {v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uvUpdated,1.0,0.0));} #ifdef UV2 else if (v_INFONAME_==1.) {v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uv2,1.0,0.0));} #endif #ifdef UV3 else if (v_INFONAME_==2.) {v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uv3,1.0,0.0));} #endif #ifdef UV4 else if (v_INFONAME_==3.) {v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uv4,1.0,0.0));} #endif #ifdef UV5 else if (v_INFONAME_==4.) {v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uv5,1.0,0.0));} #endif #ifdef UV6 else if (v_INFONAME_==5.) {v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uv6,1.0,0.0));} #endif #endif `; Le.IncludesShadersStore[rve] = nve; const ive = "bumpVertex", sve = `#if defined(BUMP) || defined(PARALLAX) || defined(CLEARCOAT_BUMP) || defined(ANISOTROPIC) #if defined(TANGENT) && defined(NORMAL) vec3 tbnNormal=normalize(normalUpdated);vec3 tbnTangent=normalize(tangentUpdated.xyz);vec3 tbnBitangent=cross(tbnNormal,tbnTangent)*tangentUpdated.w;vTBN=mat3(finalWorld)*mat3(tbnTangent,tbnBitangent,tbnNormal); #endif #endif `; Le.IncludesShadersStore[ive] = sve; const ave = "clipPlaneVertex", ove = `#ifdef CLIPPLANE fClipDistance=dot(worldPos,vClipPlane); #endif #ifdef CLIPPLANE2 fClipDistance2=dot(worldPos,vClipPlane2); #endif #ifdef CLIPPLANE3 fClipDistance3=dot(worldPos,vClipPlane3); #endif #ifdef CLIPPLANE4 fClipDistance4=dot(worldPos,vClipPlane4); #endif #ifdef CLIPPLANE5 fClipDistance5=dot(worldPos,vClipPlane5); #endif #ifdef CLIPPLANE6 fClipDistance6=dot(worldPos,vClipPlane6); #endif `; Le.IncludesShadersStore[ave] = ove; const fve = "fogVertex", Ave = `#ifdef FOG vFogDistance=(view*worldPos).xyz; #endif `; Le.IncludesShadersStore[fve] = Ave; const dve = "shadowsVertex", vve = `#ifdef SHADOWS #if defined(SHADOWCSM{X}) vPositionFromCamera{X}=view*worldPos;for (int i=0; i #define CUSTOM_VERTEX_BEGIN attribute vec3 position; #ifdef NORMAL attribute vec3 normal; #endif #ifdef TANGENT attribute vec4 tangent; #endif #ifdef UV1 attribute vec2 uv; #endif #include[2..7] #ifdef VERTEXCOLOR attribute vec4 color; #endif #include #include #include #include #include #include[1..7] #include(_DEFINENAME_,DIFFUSE,_VARYINGNAME_,Diffuse) #include(_DEFINENAME_,DETAIL,_VARYINGNAME_,Detail) #include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient) #include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity) #include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive) #include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap) #if defined(SPECULARTERM) #include(_DEFINENAME_,SPECULAR,_VARYINGNAME_,Specular) #endif #include(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump) #include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal) varying vec3 vPositionW; #ifdef NORMAL varying vec3 vNormalW; #endif #if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES) varying vec4 vColor; #endif #include #include #include #include<__decl__lightVxFragment>[0..maxSimultaneousLights] #include #include[0..maxSimultaneousMorphTargets] #ifdef REFLECTIONMAP_SKYBOX varying vec3 vPositionUVW; #endif #if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED) varying vec3 vDirectionW; #endif #include #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN vec3 positionUpdated=position; #ifdef NORMAL vec3 normalUpdated=normal; #endif #ifdef TANGENT vec4 tangentUpdated=tangent; #endif #ifdef UV1 vec2 uvUpdated=uv; #endif #include #include[0..maxSimultaneousMorphTargets] #ifdef REFLECTIONMAP_SKYBOX vPositionUVW=positionUpdated; #endif #define CUSTOM_VERTEX_UPDATE_POSITION #define CUSTOM_VERTEX_UPDATE_NORMAL #include #if defined(PREPASS) && defined(PREPASS_VELOCITY) && !defined(BONES_VELOCITY_ENABLED) vCurrentPosition=viewProjection*finalWorld*vec4(positionUpdated,1.0);vPreviousPosition=previousViewProjection*finalPreviousWorld*vec4(positionUpdated,1.0); #endif #include #include vec4 worldPos=finalWorld*vec4(positionUpdated,1.0); #ifdef NORMAL mat3 normalWorld=mat3(finalWorld); #if defined(INSTANCES) && defined(THIN_INSTANCES) vNormalW=normalUpdated/vec3(dot(normalWorld[0],normalWorld[0]),dot(normalWorld[1],normalWorld[1]),dot(normalWorld[2],normalWorld[2]));vNormalW=normalize(normalWorld*vNormalW); #else #ifdef NONUNIFORMSCALING normalWorld=transposeMat3(inverseMat3(normalWorld)); #endif vNormalW=normalize(normalWorld*normalUpdated); #endif #endif #define CUSTOM_VERTEX_UPDATE_WORLDPOS #ifdef MULTIVIEW if (gl_ViewID_OVR==0u) {gl_Position=viewProjection*worldPos;} else {gl_Position=viewProjectionR*worldPos;} #else gl_Position=viewProjection*worldPos; #endif vPositionW=vec3(worldPos); #include #if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED) vDirectionW=normalize(vec3(finalWorld*vec4(positionUpdated,0.0))); #endif #ifndef UV1 vec2 uvUpdated=vec2(0.,0.); #endif #ifdef MAINUV1 vMainUV1=uvUpdated; #endif #include[2..7] #include(_DEFINENAME_,DIFFUSE,_VARYINGNAME_,Diffuse,_MATRIXNAME_,diffuse,_INFONAME_,DiffuseInfos.x) #include(_DEFINENAME_,DETAIL,_VARYINGNAME_,Detail,_MATRIXNAME_,detail,_INFONAME_,DetailInfos.x) #include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient,_MATRIXNAME_,ambient,_INFONAME_,AmbientInfos.x) #include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity,_MATRIXNAME_,opacity,_INFONAME_,OpacityInfos.x) #include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive,_MATRIXNAME_,emissive,_INFONAME_,EmissiveInfos.x) #include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap,_MATRIXNAME_,lightmap,_INFONAME_,LightmapInfos.x) #if defined(SPECULARTERM) #include(_DEFINENAME_,SPECULAR,_VARYINGNAME_,Specular,_MATRIXNAME_,specular,_INFONAME_,SpecularInfos.x) #endif #include(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump,_MATRIXNAME_,bump,_INFONAME_,BumpInfos.x) #include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal,_MATRIXNAME_,decal,_INFONAME_,DecalInfos.x) #include #include #include #include[0..maxSimultaneousLights] #include #include #include #define CUSTOM_VERTEX_MAIN_END } `; Le.ShadersStore[Hve] = gve; const Xve = new RegExp("^([gimus]+)!"); class sq { /** * Creates a new instance of the plugin manager * @param material material that this manager will manage the plugins for */ constructor(e) { this._plugins = [], this._activePlugins = [], this._activePluginsForExtraEvents = [], this._material = e, this._scene = e.getScene(), this._engine = this._scene.getEngine(); } /** * @internal */ _addPlugin(e) { for (let n = 0; n < this._plugins.length; ++n) if (this._plugins[n].name === e.name) return !1; if (this._material._uniformBufferLayoutBuilt) throw `The plugin "${e.name}" can't be added to the material "${this._material.name}" because this material has already been used for rendering! Please add plugins to materials before any rendering with this material occurs.`; const t = e.getClassName(); sq._MaterialPluginClassToMainDefine[t] || (sq._MaterialPluginClassToMainDefine[t] = "MATERIALPLUGIN_" + ++sq._MaterialPluginCounter), this._material._callbackPluginEventGeneric = (n, i) => this._handlePluginEvent(n, i), this._plugins.push(e), this._plugins.sort((n, i) => n.priority - i.priority), this._codeInjectionPoints = {}; const r = {}; r[sq._MaterialPluginClassToMainDefine[t]] = { type: "boolean", default: !0 }; for (const n of this._plugins) n.collectDefines(r), this._collectPointNames("vertex", n.getCustomCode("vertex")), this._collectPointNames("fragment", n.getCustomCode("fragment")); return this._defineNamesFromPlugins = r, !0; } /** * @internal */ _activatePlugin(e) { this._activePlugins.indexOf(e) === -1 && (this._activePlugins.push(e), this._activePlugins.sort((t, r) => t.priority - r.priority), this._material._callbackPluginEventIsReadyForSubMesh = this._handlePluginEventIsReadyForSubMesh.bind(this), this._material._callbackPluginEventPrepareDefinesBeforeAttributes = this._handlePluginEventPrepareDefinesBeforeAttributes.bind(this), this._material._callbackPluginEventPrepareDefines = this._handlePluginEventPrepareDefines.bind(this), this._material._callbackPluginEventBindForSubMesh = this._handlePluginEventBindForSubMesh.bind(this), e.registerForExtraEvents && (this._activePluginsForExtraEvents.push(e), this._activePluginsForExtraEvents.sort((t, r) => t.priority - r.priority), this._material._callbackPluginEventHasRenderTargetTextures = this._handlePluginEventHasRenderTargetTextures.bind(this), this._material._callbackPluginEventFillRenderTargetTextures = this._handlePluginEventFillRenderTargetTextures.bind(this), this._material._callbackPluginEventHardBindForSubMesh = this._handlePluginEventHardBindForSubMesh.bind(this))); } /** * Gets a plugin from the list of plugins managed by this manager * @param name name of the plugin * @returns the plugin if found, else null */ getPlugin(e) { for (let t = 0; t < this._plugins.length; ++t) if (this._plugins[t].name === e) return this._plugins[t]; return null; } _handlePluginEventIsReadyForSubMesh(e) { let t = !0; for (const r of this._activePlugins) t = t && r.isReadyForSubMesh(e.defines, this._scene, this._engine, e.subMesh); e.isReadyForSubMesh = t; } _handlePluginEventPrepareDefinesBeforeAttributes(e) { for (const t of this._activePlugins) t.prepareDefinesBeforeAttributes(e.defines, this._scene, e.mesh); } _handlePluginEventPrepareDefines(e) { for (const t of this._activePlugins) t.prepareDefines(e.defines, this._scene, e.mesh); } _handlePluginEventHardBindForSubMesh(e) { for (const t of this._activePluginsForExtraEvents) t.hardBindForSubMesh(this._material._uniformBuffer, this._scene, this._engine, e.subMesh); } _handlePluginEventBindForSubMesh(e) { for (const t of this._activePlugins) t.bindForSubMesh(this._material._uniformBuffer, this._scene, this._engine, e.subMesh); } _handlePluginEventHasRenderTargetTextures(e) { let t = !1; for (const r of this._activePluginsForExtraEvents) if (t = r.hasRenderTargetTextures(), t) break; e.hasRenderTargetTextures = t; } _handlePluginEventFillRenderTargetTextures(e) { for (const t of this._activePluginsForExtraEvents) t.fillRenderTargetTextures(e.renderTargets); } _handlePluginEvent(e, t) { var r; switch (e) { case x6.GetActiveTextures: { const n = t; for (const i of this._activePlugins) i.getActiveTextures(n.activeTextures); break; } case x6.GetAnimatables: { const n = t; for (const i of this._activePlugins) i.getAnimatables(n.animatables); break; } case x6.HasTexture: { const n = t; let i = !1; for (const s of this._activePlugins) if (i = s.hasTexture(n.texture), i) break; n.hasTexture = i; break; } case x6.Disposed: { const n = t; for (const i of this._plugins) i.dispose(n.forceDisposeTextures); break; } case x6.GetDefineNames: { const n = t; n.defineNames = this._defineNamesFromPlugins; break; } case x6.PrepareEffect: { const n = t; for (const i of this._activePlugins) n.fallbackRank = i.addFallbacks(n.defines, n.fallbacks, n.fallbackRank), i.getAttributes(n.attributes, this._scene, n.mesh); this._uniformList.length > 0 && n.uniforms.push(...this._uniformList), this._samplerList.length > 0 && n.samplers.push(...this._samplerList), this._uboList.length > 0 && n.uniformBuffersNames.push(...this._uboList), n.customCode = this._injectCustomCode(n, n.customCode); break; } case x6.PrepareUniformBuffer: { const n = t; this._uboDeclaration = "", this._vertexDeclaration = "", this._fragmentDeclaration = "", this._uniformList = [], this._samplerList = [], this._uboList = []; for (const i of this._plugins) { const s = i.getUniforms(); if (s) { if (s.ubo) for (const a of s.ubo) { if (a.size && a.type) { const f = (r = a.arraySize) !== null && r !== void 0 ? r : 0; n.ubo.addUniform(a.name, a.size, f), this._uboDeclaration += `${a.type} ${a.name}${f > 0 ? `[${f}]` : ""}; `; } this._uniformList.push(a.name); } s.vertex && (this._vertexDeclaration += s.vertex + ` `), s.fragment && (this._fragmentDeclaration += s.fragment + ` `); } i.getSamplers(this._samplerList), i.getUniformBuffersNames(this._uboList); } break; } } } _collectPointNames(e, t) { if (t) for (const r in t) this._codeInjectionPoints[e] || (this._codeInjectionPoints[e] = {}), this._codeInjectionPoints[e][r] = !0; } _injectCustomCode(e, t) { return (r, n) => { var i, s; t && (n = t(r, n)), this._uboDeclaration && (n = n.replace("#define ADDITIONAL_UBO_DECLARATION", this._uboDeclaration)), this._vertexDeclaration && (n = n.replace("#define ADDITIONAL_VERTEX_DECLARATION", this._vertexDeclaration)), this._fragmentDeclaration && (n = n.replace("#define ADDITIONAL_FRAGMENT_DECLARATION", this._fragmentDeclaration)); const a = (i = this._codeInjectionPoints) === null || i === void 0 ? void 0 : i[r]; if (!a) return n; let f = null; for (let o in a) { let d = ""; for (const v of this._activePlugins) { let u = (s = v.getCustomCode(r)) === null || s === void 0 ? void 0 : s[o]; if (u) { if (v.resolveIncludes) { if (f === null) { const l = za.GLSL; f = { defines: [], indexParameters: e.indexParameters, isFragment: !1, shouldUseHighPrecisionShader: this._engine._shouldUseHighPrecisionShader, processor: void 0, supportsUniformBuffers: this._engine.supportsUniformBuffers, shadersRepository: Le.GetShadersRepository(l), includesShadersStore: Le.GetIncludesShadersStore(l), version: void 0, platformName: this._engine.shaderPlatformName, processingContext: void 0, isNDCHalfZRange: this._engine.isNDCHalfZRange, useReverseDepthBuffer: this._engine.useReverseDepthBuffer, processCodeAfterIncludes: void 0 // not used by _ProcessIncludes }; } f.isFragment = r === "fragment", hp._ProcessIncludes(u, f, (l) => u = l); } d += u + ` `; } } if (d.length > 0) if (o.charAt(0) === "!") { o = o.substring(1); let v = "g"; if (o.charAt(0) === "!") v = "", o = o.substring(1); else { const p = Xve.exec(o); p && p.length >= 2 && (v = p[1], o = o.substring(v.length + 1)); } v.indexOf("g") < 0 && (v += "g"); const u = n, l = new RegExp(o, v); let P = l.exec(u); for (; P !== null; ) { let p = d; for (let c = 0; c < P.length; ++c) p = p.replace("$" + c, P[c]); n = n.replace(P[0], p), P = l.exec(u); } } else { const v = "#define " + o; n = n.replace(v, ` ` + d + ` ` + v); } } return n; }; } } sq._MaterialPluginClassToMainDefine = {}; sq._MaterialPluginCounter = 0; gr.OnEnginesDisposedObservable.add(() => { gQ(); }); const tD = []; let eF = !1, tF = null; function Tve(A, e) { eF || (tF = gt.OnEventObservable.add((r) => { for (const [, n] of tD) n(r); }, x6.Created), eF = !0); const t = tD.filter(([r, n]) => r === A); t.length > 0 ? t[0][1] = e : tD.push([A, e]); } function qve(A) { for (let e = 0; e < tD.length; ++e) if (tD[e][0] === A) return tD.splice(e, 1), tD.length === 0 && gQ(), !0; return !1; } function gQ() { tD.length = 0, eF = !1, gt.OnEventObservable.remove(tF), tF = null; } class Gl { _enable(e) { e && this._pluginManager._activatePlugin(this); } /** * Creates a new material plugin * @param material parent material of the plugin * @param name name of the plugin * @param priority priority of the plugin * @param defines list of defines used by the plugin. The value of the property is the default value for this property * @param addToPluginList true to add the plugin to the list of plugins managed by the material plugin manager of the material (default: true) * @param enable true to enable the plugin (it is handy if the plugin does not handle properties to switch its current activation) * @param resolveIncludes Indicates that any #include directive in the plugin code must be replaced by the corresponding code (default: false) */ constructor(e, t, r, n, i = !0, s = !1, a = !1) { this.priority = 500, this.resolveIncludes = !1, this.registerForExtraEvents = !1, this._material = e, this.name = t, this.priority = r, this.resolveIncludes = a, e.pluginManager || (e.pluginManager = new sq(e), e.onDisposeObservable.add(() => { e.pluginManager = void 0; })), this._pluginDefineNames = n, this._pluginManager = e.pluginManager, i && this._pluginManager._addPlugin(this), s && this._enable(!0), this.markAllDefinesAsDirty = e._dirtyCallbacks[63]; } /** * Gets the current class name useful for serialization or dynamic coding. * @returns The class name. */ getClassName() { return "MaterialPluginBase"; } /** * Specifies that the submesh is ready to be used. * @param defines the list of "defines" to update. * @param scene defines the scene the material belongs to. * @param engine the engine this scene belongs to. * @param subMesh the submesh to check for readiness * @returns - boolean indicating that the submesh is ready or not. */ // eslint-disable-next-line @typescript-eslint/no-unused-vars isReadyForSubMesh(e, t, r, n) { return !0; } /** * Binds the material data (this function is called even if mustRebind() returns false) * @param uniformBuffer defines the Uniform buffer to fill in. * @param scene defines the scene the material belongs to. * @param engine defines the engine the material belongs to. * @param subMesh the submesh to bind data for */ // eslint-disable-next-line @typescript-eslint/no-unused-vars hardBindForSubMesh(e, t, r, n) { } /** * Binds the material data. * @param uniformBuffer defines the Uniform buffer to fill in. * @param scene defines the scene the material belongs to. * @param engine the engine this scene belongs to. * @param subMesh the submesh to bind data for */ // eslint-disable-next-line @typescript-eslint/no-unused-vars bindForSubMesh(e, t, r, n) { } /** * Disposes the resources of the material. * @param forceDisposeTextures - Forces the disposal of all textures. */ // eslint-disable-next-line @typescript-eslint/no-unused-vars dispose(e) { } /** * Returns a list of custom shader code fragments to customize the shader. * @param shaderType "vertex" or "fragment" * @returns null if no code to be added, or a list of pointName =\> code. * Note that `pointName` can also be a regular expression if it starts with a `!`. * In that case, the string found by the regular expression (if any) will be * replaced by the code provided. */ // eslint-disable-next-line @typescript-eslint/no-unused-vars getCustomCode(e) { return null; } /** * Collects all defines. * @param defines The object to append to. */ collectDefines(e) { if (this._pluginDefineNames) for (const t of Object.keys(this._pluginDefineNames)) { if (t[0] === "_") continue; const r = typeof this._pluginDefineNames[t]; e[t] = { type: r === "number" ? "number" : r === "string" ? "string" : r === "boolean" ? "boolean" : "object", default: this._pluginDefineNames[t] }; } } /** * Sets the defines for the next rendering. Called before MaterialHelper.PrepareDefinesForAttributes is called. * @param defines the list of "defines" to update. * @param scene defines the scene to the material belongs to. * @param mesh the mesh being rendered */ // eslint-disable-next-line @typescript-eslint/no-unused-vars prepareDefinesBeforeAttributes(e, t, r) { } /** * Sets the defines for the next rendering * @param defines the list of "defines" to update. * @param scene defines the scene to the material belongs to. * @param mesh the mesh being rendered */ // eslint-disable-next-line @typescript-eslint/no-unused-vars prepareDefines(e, t, r) { } /** * Checks to see if a texture is used in the material. * @param texture - Base texture to use. * @returns - Boolean specifying if a texture is used in the material. */ // eslint-disable-next-line @typescript-eslint/no-unused-vars hasTexture(e) { return !1; } /** * Gets a boolean indicating that current material needs to register RTT * @returns true if this uses a render target otherwise false. */ hasRenderTargetTextures() { return !1; } /** * Fills the list of render target textures. * @param renderTargets the list of render targets to update */ // eslint-disable-next-line @typescript-eslint/no-unused-vars fillRenderTargetTextures(e) { } /** * Returns an array of the actively used textures. * @param activeTextures Array of BaseTextures */ // eslint-disable-next-line @typescript-eslint/no-unused-vars getActiveTextures(e) { } /** * Returns the animatable textures. * @param animatables Array of animatable textures. */ // eslint-disable-next-line @typescript-eslint/no-unused-vars getAnimatables(e) { } /** * Add fallbacks to the effect fallbacks list. * @param defines defines the Base texture to use. * @param fallbacks defines the current fallback list. * @param currentRank defines the current fallback rank. * @returns the new fallback rank. */ addFallbacks(e, t, r) { return r; } /** * Gets the samplers used by the plugin. * @param samplers list that the sampler names should be added to. */ // eslint-disable-next-line @typescript-eslint/no-unused-vars getSamplers(e) { } /** * Gets the attributes used by the plugin. * @param attributes list that the attribute names should be added to. * @param scene the scene that the material belongs to. * @param mesh the mesh being rendered. */ // eslint-disable-next-line @typescript-eslint/no-unused-vars getAttributes(e, t, r) { } /** * Gets the uniform buffers names added by the plugin. * @param ubos list that the ubo names should be added to. */ // eslint-disable-next-line @typescript-eslint/no-unused-vars getUniformBuffersNames(e) { } /** * Gets the description of the uniforms to add to the ubo (if engine supports ubos) or to inject directly in the vertex/fragment shaders (if engine does not support ubos) * @returns the description of the uniforms */ getUniforms() { return {}; } /** * Makes a duplicate of the current configuration into another one. * @param plugin define the config where to copy the info */ copyTo(e) { jt.Clone(() => e, this); } /** * Serializes this plugin configuration. * @returns - An object with the serialized config. */ serialize() { return jt.Serialize(this); } /** * Parses a plugin configuration from a serialized object. * @param source - Serialized object. * @param scene Defines the scene we are parsing for * @param rootUrl Defines the rootUrl to load from */ parse(e, t, r) { jt.Parse(() => this, e, t, r); } } C([ M() ], Gl.prototype, "name", void 0); C([ M() ], Gl.prototype, "priority", void 0); C([ M() ], Gl.prototype, "resolveIncludes", void 0); C([ M() ], Gl.prototype, "registerForExtraEvents", void 0); class Lee extends na { constructor() { super(...arguments), this.DETAIL = !1, this.DETAILDIRECTUV = 0, this.DETAIL_NORMALBLENDMETHOD = 0; } } class l4 extends Gl { /** @internal */ _markAllSubMeshesAsTexturesDirty() { this._enable(this._isEnabled), this._internalMarkAllSubMeshesAsTexturesDirty(); } constructor(e, t = !0) { super(e, "DetailMap", 140, new Lee(), t), this._texture = null, this.diffuseBlendLevel = 1, this.roughnessBlendLevel = 1, this.bumpLevel = 1, this._normalBlendMethod = gt.MATERIAL_NORMALBLENDMETHOD_WHITEOUT, this._isEnabled = !1, this.isEnabled = !1, this._internalMarkAllSubMeshesAsTexturesDirty = e._dirtyCallbacks[1]; } isReadyForSubMesh(e, t, r) { return this._isEnabled ? !(e._areTexturesDirty && t.texturesEnabled && r.getCaps().standardDerivatives && this._texture && Dt.DetailTextureEnabled && !this._texture.isReady()) : !0; } prepareDefines(e, t) { if (this._isEnabled) { e.DETAIL_NORMALBLENDMETHOD = this._normalBlendMethod; const r = t.getEngine(); e._areTexturesDirty && (r.getCaps().standardDerivatives && this._texture && Dt.DetailTextureEnabled && this._isEnabled ? (Ye.PrepareDefinesForMergedUV(this._texture, e, "DETAIL"), e.DETAIL_NORMALBLENDMETHOD = this._normalBlendMethod) : e.DETAIL = !1); } else e.DETAIL = !1; } bindForSubMesh(e, t) { if (!this._isEnabled) return; const r = this._material.isFrozen; (!e.useUbo || !r || !e.isSync) && this._texture && Dt.DetailTextureEnabled && (e.updateFloat4("vDetailInfos", this._texture.coordinatesIndex, this.diffuseBlendLevel, this.bumpLevel, this.roughnessBlendLevel), Ye.BindTextureMatrix(this._texture, e, "detail")), t.texturesEnabled && this._texture && Dt.DetailTextureEnabled && e.setTexture("detailSampler", this._texture); } hasTexture(e) { return this._texture === e; } getActiveTextures(e) { this._texture && e.push(this._texture); } getAnimatables(e) { this._texture && this._texture.animations && this._texture.animations.length > 0 && e.push(this._texture); } dispose(e) { var t; e && ((t = this._texture) === null || t === void 0 || t.dispose()); } getClassName() { return "DetailMapConfiguration"; } getSamplers(e) { e.push("detailSampler"); } getUniforms() { return { ubo: [ { name: "vDetailInfos", size: 4, type: "vec4" }, { name: "detailMatrix", size: 16, type: "mat4" } ] }; } } C([ en("detailTexture"), At("_markAllSubMeshesAsTexturesDirty") ], l4.prototype, "texture", void 0); C([ M() ], l4.prototype, "diffuseBlendLevel", void 0); C([ M() ], l4.prototype, "roughnessBlendLevel", void 0); C([ M() ], l4.prototype, "bumpLevel", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], l4.prototype, "normalBlendMethod", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], l4.prototype, "isEnabled", void 0); const TE = { effect: null, subMesh: null }; class Kee extends na { /** * Initializes the Standard Material defines. * @param externalProperties The external properties */ constructor(e) { super(e), this.MAINUV1 = !1, this.MAINUV2 = !1, this.MAINUV3 = !1, this.MAINUV4 = !1, this.MAINUV5 = !1, this.MAINUV6 = !1, this.DIFFUSE = !1, this.DIFFUSEDIRECTUV = 0, this.BAKED_VERTEX_ANIMATION_TEXTURE = !1, this.AMBIENT = !1, this.AMBIENTDIRECTUV = 0, this.OPACITY = !1, this.OPACITYDIRECTUV = 0, this.OPACITYRGB = !1, this.REFLECTION = !1, this.EMISSIVE = !1, this.EMISSIVEDIRECTUV = 0, this.SPECULAR = !1, this.SPECULARDIRECTUV = 0, this.BUMP = !1, this.BUMPDIRECTUV = 0, this.PARALLAX = !1, this.PARALLAX_RHS = !1, this.PARALLAXOCCLUSION = !1, this.SPECULAROVERALPHA = !1, this.CLIPPLANE = !1, this.CLIPPLANE2 = !1, this.CLIPPLANE3 = !1, this.CLIPPLANE4 = !1, this.CLIPPLANE5 = !1, this.CLIPPLANE6 = !1, this.ALPHATEST = !1, this.DEPTHPREPASS = !1, this.ALPHAFROMDIFFUSE = !1, this.POINTSIZE = !1, this.FOG = !1, this.SPECULARTERM = !1, this.DIFFUSEFRESNEL = !1, this.OPACITYFRESNEL = !1, this.REFLECTIONFRESNEL = !1, this.REFRACTIONFRESNEL = !1, this.EMISSIVEFRESNEL = !1, this.FRESNEL = !1, this.NORMAL = !1, this.TANGENT = !1, this.UV1 = !1, this.UV2 = !1, this.UV3 = !1, this.UV4 = !1, this.UV5 = !1, this.UV6 = !1, this.VERTEXCOLOR = !1, this.VERTEXALPHA = !1, this.NUM_BONE_INFLUENCERS = 0, this.BonesPerMesh = 0, this.BONETEXTURE = !1, this.BONES_VELOCITY_ENABLED = !1, this.INSTANCES = !1, this.THIN_INSTANCES = !1, this.INSTANCESCOLOR = !1, this.GLOSSINESS = !1, this.ROUGHNESS = !1, this.EMISSIVEASILLUMINATION = !1, this.LINKEMISSIVEWITHDIFFUSE = !1, this.REFLECTIONFRESNELFROMSPECULAR = !1, this.LIGHTMAP = !1, this.LIGHTMAPDIRECTUV = 0, this.OBJECTSPACE_NORMALMAP = !1, this.USELIGHTMAPASSHADOWMAP = !1, this.REFLECTIONMAP_3D = !1, this.REFLECTIONMAP_SPHERICAL = !1, this.REFLECTIONMAP_PLANAR = !1, this.REFLECTIONMAP_CUBIC = !1, this.USE_LOCAL_REFLECTIONMAP_CUBIC = !1, this.USE_LOCAL_REFRACTIONMAP_CUBIC = !1, this.REFLECTIONMAP_PROJECTION = !1, this.REFLECTIONMAP_SKYBOX = !1, this.REFLECTIONMAP_EXPLICIT = !1, this.REFLECTIONMAP_EQUIRECTANGULAR = !1, this.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = !1, this.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = !1, this.REFLECTIONMAP_OPPOSITEZ = !1, this.INVERTCUBICMAP = !1, this.LOGARITHMICDEPTH = !1, this.REFRACTION = !1, this.REFRACTIONMAP_3D = !1, this.REFLECTIONOVERALPHA = !1, this.TWOSIDEDLIGHTING = !1, this.SHADOWFLOAT = !1, this.MORPHTARGETS = !1, this.MORPHTARGETS_NORMAL = !1, this.MORPHTARGETS_TANGENT = !1, this.MORPHTARGETS_UV = !1, this.NUM_MORPH_INFLUENCERS = 0, this.MORPHTARGETS_TEXTURE = !1, this.NONUNIFORMSCALING = !1, this.PREMULTIPLYALPHA = !1, this.ALPHATEST_AFTERALLALPHACOMPUTATIONS = !1, this.ALPHABLEND = !0, this.PREPASS = !1, this.PREPASS_IRRADIANCE = !1, this.PREPASS_IRRADIANCE_INDEX = -1, this.PREPASS_ALBEDO_SQRT = !1, this.PREPASS_ALBEDO_SQRT_INDEX = -1, this.PREPASS_DEPTH = !1, this.PREPASS_DEPTH_INDEX = -1, this.PREPASS_NORMAL = !1, this.PREPASS_NORMAL_INDEX = -1, this.PREPASS_NORMAL_WORLDSPACE = !1, this.PREPASS_POSITION = !1, this.PREPASS_POSITION_INDEX = -1, this.PREPASS_VELOCITY = !1, this.PREPASS_VELOCITY_INDEX = -1, this.PREPASS_REFLECTIVITY = !1, this.PREPASS_REFLECTIVITY_INDEX = -1, this.SCENE_MRT_COUNT = 0, this.RGBDLIGHTMAP = !1, this.RGBDREFLECTION = !1, this.RGBDREFRACTION = !1, this.IMAGEPROCESSING = !1, this.VIGNETTE = !1, this.VIGNETTEBLENDMODEMULTIPLY = !1, this.VIGNETTEBLENDMODEOPAQUE = !1, this.TONEMAPPING = !1, this.TONEMAPPING_ACES = !1, this.CONTRAST = !1, this.COLORCURVES = !1, this.COLORGRADING = !1, this.COLORGRADING3D = !1, this.SAMPLER3DGREENDEPTH = !1, this.SAMPLER3DBGRMAP = !1, this.DITHER = !1, this.IMAGEPROCESSINGPOSTPROCESS = !1, this.SKIPFINALCOLORCLAMP = !1, this.MULTIVIEW = !1, this.ORDER_INDEPENDENT_TRANSPARENCY = !1, this.ORDER_INDEPENDENT_TRANSPARENCY_16BITS = !1, this.CAMERA_ORTHOGRAPHIC = !1, this.CAMERA_PERSPECTIVE = !1, this.IS_REFLECTION_LINEAR = !1, this.IS_REFRACTION_LINEAR = !1, this.EXPOSURE = !1, this.DECAL_AFTER_DETAIL = !1, this.rebuild(); } setReflectionMode(e) { const t = [ "REFLECTIONMAP_CUBIC", "REFLECTIONMAP_EXPLICIT", "REFLECTIONMAP_PLANAR", "REFLECTIONMAP_PROJECTION", "REFLECTIONMAP_PROJECTION", "REFLECTIONMAP_SKYBOX", "REFLECTIONMAP_SPHERICAL", "REFLECTIONMAP_EQUIRECTANGULAR", "REFLECTIONMAP_EQUIRECTANGULAR_FIXED", "REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED" ]; for (const r of t) this[r] = r === e; } } class Wt extends P1 { /** * Gets the image processing configuration used either in this material. */ get imageProcessingConfiguration() { return this._imageProcessingConfiguration; } /** * Sets the Default image processing configuration used either in the this material. * * If sets to null, the scene one is in use. */ set imageProcessingConfiguration(e) { this._attachImageProcessingConfiguration(e), this._markAllSubMeshesAsTexturesDirty(); } /** * Attaches a new image processing configuration to the Standard Material. * @param configuration */ _attachImageProcessingConfiguration(e) { e !== this._imageProcessingConfiguration && (this._imageProcessingConfiguration && this._imageProcessingObserver && this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver), e ? this._imageProcessingConfiguration = e : this._imageProcessingConfiguration = this.getScene().imageProcessingConfiguration, this._imageProcessingConfiguration && (this._imageProcessingObserver = this._imageProcessingConfiguration.onUpdateParameters.add(() => { this._markAllSubMeshesAsImageProcessingDirty(); }))); } /** * Can this material render to prepass */ get isPrePassCapable() { return !this.disableDepthWrite; } /** * Gets whether the color curves effect is enabled. */ get cameraColorCurvesEnabled() { return this.imageProcessingConfiguration.colorCurvesEnabled; } /** * Sets whether the color curves effect is enabled. */ set cameraColorCurvesEnabled(e) { this.imageProcessingConfiguration.colorCurvesEnabled = e; } /** * Gets whether the color grading effect is enabled. */ get cameraColorGradingEnabled() { return this.imageProcessingConfiguration.colorGradingEnabled; } /** * Gets whether the color grading effect is enabled. */ set cameraColorGradingEnabled(e) { this.imageProcessingConfiguration.colorGradingEnabled = e; } /** * Gets whether tonemapping is enabled or not. */ get cameraToneMappingEnabled() { return this._imageProcessingConfiguration.toneMappingEnabled; } /** * Sets whether tonemapping is enabled or not */ set cameraToneMappingEnabled(e) { this._imageProcessingConfiguration.toneMappingEnabled = e; } /** * The camera exposure used on this material. * This property is here and not in the camera to allow controlling exposure without full screen post process. * This corresponds to a photographic exposure. */ get cameraExposure() { return this._imageProcessingConfiguration.exposure; } /** * The camera exposure used on this material. * This property is here and not in the camera to allow controlling exposure without full screen post process. * This corresponds to a photographic exposure. */ set cameraExposure(e) { this._imageProcessingConfiguration.exposure = e; } /** * Gets The camera contrast used on this material. */ get cameraContrast() { return this._imageProcessingConfiguration.contrast; } /** * Sets The camera contrast used on this material. */ set cameraContrast(e) { this._imageProcessingConfiguration.contrast = e; } /** * Gets the Color Grading 2D Lookup Texture. */ get cameraColorGradingTexture() { return this._imageProcessingConfiguration.colorGradingTexture; } /** * Sets the Color Grading 2D Lookup Texture. */ set cameraColorGradingTexture(e) { this._imageProcessingConfiguration.colorGradingTexture = e; } /** * The color grading curves provide additional color adjustmnent that is applied after any color grading transform (3D LUT). * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects. * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image; * corresponding to low luminance, medium luminance, and high luminance areas respectively. */ get cameraColorCurves() { return this._imageProcessingConfiguration.colorCurves; } /** * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT). * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects. * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image; * corresponding to low luminance, medium luminance, and high luminance areas respectively. */ set cameraColorCurves(e) { this._imageProcessingConfiguration.colorCurves = e; } /** * Can this material render to several textures at once */ get canRenderToMRT() { return !0; } /** * Instantiates a new standard material. * This is the default material used in Babylon. It is the best trade off between quality * and performances. * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/materials_introduction * @param name Define the name of the material in the scene * @param scene Define the scene the material belong to */ constructor(e, t) { super(e, t), this._diffuseTexture = null, this._ambientTexture = null, this._opacityTexture = null, this._reflectionTexture = null, this._emissiveTexture = null, this._specularTexture = null, this._bumpTexture = null, this._lightmapTexture = null, this._refractionTexture = null, this.ambientColor = new Ne(0, 0, 0), this.diffuseColor = new Ne(1, 1, 1), this.specularColor = new Ne(1, 1, 1), this.emissiveColor = new Ne(0, 0, 0), this.specularPower = 64, this._useAlphaFromDiffuseTexture = !1, this._useEmissiveAsIllumination = !1, this._linkEmissiveWithDiffuse = !1, this._useSpecularOverAlpha = !1, this._useReflectionOverAlpha = !1, this._disableLighting = !1, this._useObjectSpaceNormalMap = !1, this._useParallax = !1, this._useParallaxOcclusion = !1, this.parallaxScaleBias = 0.05, this._roughness = 0, this.indexOfRefraction = 0.98, this.invertRefractionY = !0, this.alphaCutOff = 0.4, this._useLightmapAsShadowmap = !1, this._useReflectionFresnelFromSpecular = !1, this._useGlossinessFromSpecularMapAlpha = !1, this._maxSimultaneousLights = 4, this._invertNormalMapX = !1, this._invertNormalMapY = !1, this._twoSidedLighting = !1, this._applyDecalMapAfterDetailMap = !1, this._renderTargets = new qf(16), this._worldViewProjectionMatrix = he.Zero(), this._globalAmbientColor = new Ne(0, 0, 0), this._cacheHasRenderTargetTextures = !1, this.detailMap = new l4(this), this._attachImageProcessingConfiguration(null), this.prePassConfiguration = new GC(), this.getRenderTargetTextures = () => (this._renderTargets.reset(), Wt.ReflectionTextureEnabled && this._reflectionTexture && this._reflectionTexture.isRenderTarget && this._renderTargets.push(this._reflectionTexture), Wt.RefractionTextureEnabled && this._refractionTexture && this._refractionTexture.isRenderTarget && this._renderTargets.push(this._refractionTexture), this._eventInfo.renderTargets = this._renderTargets, this._callbackPluginEventFillRenderTargetTextures(this._eventInfo), this._renderTargets); } /** * Gets a boolean indicating that current material needs to register RTT */ get hasRenderTargetTextures() { return Wt.ReflectionTextureEnabled && this._reflectionTexture && this._reflectionTexture.isRenderTarget || Wt.RefractionTextureEnabled && this._refractionTexture && this._refractionTexture.isRenderTarget ? !0 : this._cacheHasRenderTargetTextures; } /** * Gets the current class name of the material e.g. "StandardMaterial" * Mainly use in serialization. * @returns the class name */ getClassName() { return "StandardMaterial"; } /** * Specifies if the material will require alpha blending * @returns a boolean specifying if alpha blending is needed */ needAlphaBlending() { return this._disableAlphaBlending ? !1 : this.alpha < 1 || this._opacityTexture != null || this._shouldUseAlphaFromDiffuseTexture() || this._opacityFresnelParameters && this._opacityFresnelParameters.isEnabled; } /** * Specifies if this material should be rendered in alpha test mode * @returns a boolean specifying if an alpha test is needed. */ needAlphaTesting() { return this._forceAlphaTest ? !0 : this._hasAlphaChannel() && (this._transparencyMode == null || this._transparencyMode === gt.MATERIAL_ALPHATEST); } /** * Specifies whether or not the alpha value of the diffuse texture should be used for alpha blending. */ _shouldUseAlphaFromDiffuseTexture() { return this._diffuseTexture != null && this._diffuseTexture.hasAlpha && this._useAlphaFromDiffuseTexture && this._transparencyMode !== gt.MATERIAL_OPAQUE; } /** * Specifies whether or not there is a usable alpha channel for transparency. */ _hasAlphaChannel() { return this._diffuseTexture != null && this._diffuseTexture.hasAlpha || this._opacityTexture != null; } /** * Get the texture used for alpha test purpose. * @returns the diffuse texture in case of the standard material. */ getAlphaTestTexture() { return this._diffuseTexture; } /** * Get if the submesh is ready to be used and all its information available. * Child classes can use it to update shaders * @param mesh defines the mesh to check * @param subMesh defines which submesh to check * @param useInstances specifies that instances should be used * @returns a boolean indicating that the submesh is ready or not */ isReadyForSubMesh(e, t, r = !1) { if (this._uniformBufferLayoutBuilt || this.buildUniformLayout(), t.effect && this.isFrozen && t.effect._wasPreviouslyReady && t.effect._wasPreviouslyUsingInstances === r) return !0; t.materialDefines || (this._callbackPluginEventGeneric(x6.GetDefineNames, this._eventInfo), t.materialDefines = new Kee(this._eventInfo.defineNames)); const n = this.getScene(), i = t.materialDefines; if (this._isReadyForSubMesh(t)) return !0; const s = n.getEngine(); i._needNormals = Ye.PrepareDefinesForLights(n, e, i, !0, this._maxSimultaneousLights, this._disableLighting), Ye.PrepareDefinesForMultiview(n, i); const a = this.needAlphaBlendingForMesh(e) && this.getScene().useOrderIndependentTransparency; if (Ye.PrepareDefinesForPrePass(n, i, this.canRenderToMRT && !a), Ye.PrepareDefinesForOIT(n, i, a), i._areTexturesDirty) { this._eventInfo.hasRenderTargetTextures = !1, this._callbackPluginEventHasRenderTargetTextures(this._eventInfo), this._cacheHasRenderTargetTextures = this._eventInfo.hasRenderTargetTextures, i._needUVs = !1; for (let o = 1; o <= 6; ++o) i["MAINUV" + o] = !1; if (n.texturesEnabled) { if (i.DIFFUSEDIRECTUV = 0, i.BUMPDIRECTUV = 0, i.AMBIENTDIRECTUV = 0, i.OPACITYDIRECTUV = 0, i.EMISSIVEDIRECTUV = 0, i.SPECULARDIRECTUV = 0, i.LIGHTMAPDIRECTUV = 0, this._diffuseTexture && Wt.DiffuseTextureEnabled) if (this._diffuseTexture.isReadyOrNotBlocking()) Ye.PrepareDefinesForMergedUV(this._diffuseTexture, i, "DIFFUSE"); else return !1; else i.DIFFUSE = !1; if (this._ambientTexture && Wt.AmbientTextureEnabled) if (this._ambientTexture.isReadyOrNotBlocking()) Ye.PrepareDefinesForMergedUV(this._ambientTexture, i, "AMBIENT"); else return !1; else i.AMBIENT = !1; if (this._opacityTexture && Wt.OpacityTextureEnabled) if (this._opacityTexture.isReadyOrNotBlocking()) Ye.PrepareDefinesForMergedUV(this._opacityTexture, i, "OPACITY"), i.OPACITYRGB = this._opacityTexture.getAlphaFromRGB; else return !1; else i.OPACITY = !1; if (this._reflectionTexture && Wt.ReflectionTextureEnabled) if (this._reflectionTexture.isReadyOrNotBlocking()) { switch (i._needNormals = !0, i.REFLECTION = !0, i.ROUGHNESS = this._roughness > 0, i.REFLECTIONOVERALPHA = this._useReflectionOverAlpha, i.INVERTCUBICMAP = this._reflectionTexture.coordinatesMode === We.INVCUBIC_MODE, i.REFLECTIONMAP_3D = this._reflectionTexture.isCube, i.REFLECTIONMAP_OPPOSITEZ = i.REFLECTIONMAP_3D && this.getScene().useRightHandedSystem ? !this._reflectionTexture.invertZ : this._reflectionTexture.invertZ, i.RGBDREFLECTION = this._reflectionTexture.isRGBD, this._reflectionTexture.coordinatesMode) { case We.EXPLICIT_MODE: i.setReflectionMode("REFLECTIONMAP_EXPLICIT"); break; case We.PLANAR_MODE: i.setReflectionMode("REFLECTIONMAP_PLANAR"); break; case We.PROJECTION_MODE: i.setReflectionMode("REFLECTIONMAP_PROJECTION"); break; case We.SKYBOX_MODE: i.setReflectionMode("REFLECTIONMAP_SKYBOX"); break; case We.SPHERICAL_MODE: i.setReflectionMode("REFLECTIONMAP_SPHERICAL"); break; case We.EQUIRECTANGULAR_MODE: i.setReflectionMode("REFLECTIONMAP_EQUIRECTANGULAR"); break; case We.FIXED_EQUIRECTANGULAR_MODE: i.setReflectionMode("REFLECTIONMAP_EQUIRECTANGULAR_FIXED"); break; case We.FIXED_EQUIRECTANGULAR_MIRRORED_MODE: i.setReflectionMode("REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED"); break; case We.CUBIC_MODE: case We.INVCUBIC_MODE: default: i.setReflectionMode("REFLECTIONMAP_CUBIC"); break; } i.USE_LOCAL_REFLECTIONMAP_CUBIC = !!this._reflectionTexture.boundingBoxSize; } else return !1; else i.REFLECTION = !1, i.REFLECTIONMAP_OPPOSITEZ = !1; if (this._emissiveTexture && Wt.EmissiveTextureEnabled) if (this._emissiveTexture.isReadyOrNotBlocking()) Ye.PrepareDefinesForMergedUV(this._emissiveTexture, i, "EMISSIVE"); else return !1; else i.EMISSIVE = !1; if (this._lightmapTexture && Wt.LightmapTextureEnabled) if (this._lightmapTexture.isReadyOrNotBlocking()) Ye.PrepareDefinesForMergedUV(this._lightmapTexture, i, "LIGHTMAP"), i.USELIGHTMAPASSHADOWMAP = this._useLightmapAsShadowmap, i.RGBDLIGHTMAP = this._lightmapTexture.isRGBD; else return !1; else i.LIGHTMAP = !1; if (this._specularTexture && Wt.SpecularTextureEnabled) if (this._specularTexture.isReadyOrNotBlocking()) Ye.PrepareDefinesForMergedUV(this._specularTexture, i, "SPECULAR"), i.GLOSSINESS = this._useGlossinessFromSpecularMapAlpha; else return !1; else i.SPECULAR = !1; if (n.getEngine().getCaps().standardDerivatives && this._bumpTexture && Wt.BumpTextureEnabled) { if (this._bumpTexture.isReady()) Ye.PrepareDefinesForMergedUV(this._bumpTexture, i, "BUMP"), i.PARALLAX = this._useParallax, i.PARALLAX_RHS = n.useRightHandedSystem, i.PARALLAXOCCLUSION = this._useParallaxOcclusion; else return !1; i.OBJECTSPACE_NORMALMAP = this._useObjectSpaceNormalMap; } else i.BUMP = !1, i.PARALLAX = !1, i.PARALLAX_RHS = !1, i.PARALLAXOCCLUSION = !1; if (this._refractionTexture && Wt.RefractionTextureEnabled) if (this._refractionTexture.isReadyOrNotBlocking()) i._needUVs = !0, i.REFRACTION = !0, i.REFRACTIONMAP_3D = this._refractionTexture.isCube, i.RGBDREFRACTION = this._refractionTexture.isRGBD, i.USE_LOCAL_REFRACTIONMAP_CUBIC = !!this._refractionTexture.boundingBoxSize; else return !1; else i.REFRACTION = !1; i.TWOSIDEDLIGHTING = !this._backFaceCulling && this._twoSidedLighting; } else i.DIFFUSE = !1, i.AMBIENT = !1, i.OPACITY = !1, i.REFLECTION = !1, i.EMISSIVE = !1, i.LIGHTMAP = !1, i.BUMP = !1, i.REFRACTION = !1; i.ALPHAFROMDIFFUSE = this._shouldUseAlphaFromDiffuseTexture(), i.EMISSIVEASILLUMINATION = this._useEmissiveAsIllumination, i.LINKEMISSIVEWITHDIFFUSE = this._linkEmissiveWithDiffuse, i.SPECULAROVERALPHA = this._useSpecularOverAlpha, i.PREMULTIPLYALPHA = this.alphaMode === 7 || this.alphaMode === 8, i.ALPHATEST_AFTERALLALPHACOMPUTATIONS = this.transparencyMode !== null, i.ALPHABLEND = this.transparencyMode === null || this.needAlphaBlendingForMesh(e); } if (this._eventInfo.isReadyForSubMesh = !0, this._eventInfo.defines = i, this._eventInfo.subMesh = t, this._callbackPluginEventIsReadyForSubMesh(this._eventInfo), !this._eventInfo.isReadyForSubMesh) return !1; if (i._areImageProcessingDirty && this._imageProcessingConfiguration) { if (!this._imageProcessingConfiguration.isReady()) return !1; this._imageProcessingConfiguration.prepareDefines(i), i.IS_REFLECTION_LINEAR = this.reflectionTexture != null && !this.reflectionTexture.gammaSpace, i.IS_REFRACTION_LINEAR = this.refractionTexture != null && !this.refractionTexture.gammaSpace; } i._areFresnelDirty && (Wt.FresnelEnabled ? (this._diffuseFresnelParameters || this._opacityFresnelParameters || this._emissiveFresnelParameters || this._refractionFresnelParameters || this._reflectionFresnelParameters) && (i.DIFFUSEFRESNEL = this._diffuseFresnelParameters && this._diffuseFresnelParameters.isEnabled, i.OPACITYFRESNEL = this._opacityFresnelParameters && this._opacityFresnelParameters.isEnabled, i.REFLECTIONFRESNEL = this._reflectionFresnelParameters && this._reflectionFresnelParameters.isEnabled, i.REFLECTIONFRESNELFROMSPECULAR = this._useReflectionFresnelFromSpecular, i.REFRACTIONFRESNEL = this._refractionFresnelParameters && this._refractionFresnelParameters.isEnabled, i.EMISSIVEFRESNEL = this._emissiveFresnelParameters && this._emissiveFresnelParameters.isEnabled, i._needNormals = !0, i.FRESNEL = !0) : i.FRESNEL = !1), Ye.PrepareDefinesForMisc(e, n, this._useLogarithmicDepth, this.pointsCloud, this.fogEnabled, this._shouldTurnAlphaTestOn(e) || this._forceAlphaTest, i, this._applyDecalMapAfterDetailMap), Ye.PrepareDefinesForFrameBoundValues(n, s, this, i, r, null, t.getRenderingMesh().hasThinInstances), this._eventInfo.defines = i, this._eventInfo.mesh = e, this._callbackPluginEventPrepareDefinesBeforeAttributes(this._eventInfo), Ye.PrepareDefinesForAttributes(e, i, !0, !0, !0), this._callbackPluginEventPrepareDefines(this._eventInfo); let f = !1; if (i.isDirty) { const o = i._areLightsDisposed; i.markAsProcessed(); const d = new c1(); i.REFLECTION && d.addFallback(0, "REFLECTION"), i.SPECULAR && d.addFallback(0, "SPECULAR"), i.BUMP && d.addFallback(0, "BUMP"), i.PARALLAX && d.addFallback(1, "PARALLAX"), i.PARALLAX_RHS && d.addFallback(1, "PARALLAX_RHS"), i.PARALLAXOCCLUSION && d.addFallback(0, "PARALLAXOCCLUSION"), i.SPECULAROVERALPHA && d.addFallback(0, "SPECULAROVERALPHA"), i.FOG && d.addFallback(1, "FOG"), i.POINTSIZE && d.addFallback(0, "POINTSIZE"), i.LOGARITHMICDEPTH && d.addFallback(0, "LOGARITHMICDEPTH"), Ye.HandleFallbacksForShadows(i, d, this._maxSimultaneousLights), i.SPECULARTERM && d.addFallback(0, "SPECULARTERM"), i.DIFFUSEFRESNEL && d.addFallback(1, "DIFFUSEFRESNEL"), i.OPACITYFRESNEL && d.addFallback(2, "OPACITYFRESNEL"), i.REFLECTIONFRESNEL && d.addFallback(3, "REFLECTIONFRESNEL"), i.EMISSIVEFRESNEL && d.addFallback(4, "EMISSIVEFRESNEL"), i.FRESNEL && d.addFallback(4, "FRESNEL"), i.MULTIVIEW && d.addFallback(0, "MULTIVIEW"); const v = [J.PositionKind]; i.NORMAL && v.push(J.NormalKind), i.TANGENT && v.push(J.TangentKind); for (let j = 1; j <= 6; ++j) i["UV" + j] && v.push(`uv${j === 1 ? "" : j}`); i.VERTEXCOLOR && v.push(J.ColorKind), Ye.PrepareAttributesForBones(v, e, i, d), Ye.PrepareAttributesForInstances(v, i), Ye.PrepareAttributesForMorphTargets(v, e, i), Ye.PrepareAttributesForBakedVertexAnimation(v, e, i); let u = "default"; const l = [ "world", "view", "viewProjection", "vEyePosition", "vLightsType", "vAmbientColor", "vDiffuseColor", "vSpecularColor", "vEmissiveColor", "visibility", "vFogInfos", "vFogColor", "pointSize", "vDiffuseInfos", "vAmbientInfos", "vOpacityInfos", "vReflectionInfos", "vEmissiveInfos", "vSpecularInfos", "vBumpInfos", "vLightmapInfos", "vRefractionInfos", "mBones", "diffuseMatrix", "ambientMatrix", "opacityMatrix", "reflectionMatrix", "emissiveMatrix", "specularMatrix", "bumpMatrix", "normalMatrix", "lightmapMatrix", "refractionMatrix", "diffuseLeftColor", "diffuseRightColor", "opacityParts", "reflectionLeftColor", "reflectionRightColor", "emissiveLeftColor", "emissiveRightColor", "refractionLeftColor", "refractionRightColor", "vReflectionPosition", "vReflectionSize", "vRefractionPosition", "vRefractionSize", "logarithmicDepthConstant", "vTangentSpaceParams", "alphaCutOff", "boneTextureWidth", "morphTargetTextureInfo", "morphTargetTextureIndices" ], P = [ "diffuseSampler", "ambientSampler", "opacitySampler", "reflectionCubeSampler", "reflection2DSampler", "emissiveSampler", "specularSampler", "bumpSampler", "lightmapSampler", "refractionCubeSampler", "refraction2DSampler", "boneSampler", "morphTargets", "oitDepthSampler", "oitFrontColorSampler" ], p = ["Material", "Scene", "Mesh"], c = { maxSimultaneousLights: this._maxSimultaneousLights, maxSimultaneousMorphTargets: i.NUM_MORPH_INFLUENCERS }; this._eventInfo.fallbacks = d, this._eventInfo.fallbackRank = 0, this._eventInfo.defines = i, this._eventInfo.uniforms = l, this._eventInfo.attributes = v, this._eventInfo.samplers = P, this._eventInfo.uniformBuffersNames = p, this._eventInfo.customCode = void 0, this._eventInfo.mesh = e, this._eventInfo.indexParameters = c, this._callbackPluginEventGeneric(x6.PrepareEffect, this._eventInfo), GC.AddUniforms(l), Ui && (Ui.PrepareUniforms(l, i), Ui.PrepareSamplers(P, i)), Ye.PrepareUniformsAndSamplersList({ uniformsNames: l, uniformBuffersNames: p, samplers: P, defines: i, maxSimultaneousLights: this._maxSimultaneousLights }), Mf(l); const H = {}; this.customShaderNameResolve && (u = this.customShaderNameResolve(u, l, p, P, i, v, H)); const T = i.toString(), q = t.effect; let b = n.getEngine().createEffect(u, { attributes: v, uniformsNames: l, uniformBuffersNames: p, samplers: P, defines: T, fallbacks: d, onCompiled: this.onCompiled, onError: this.onError, indexParameters: c, processFinalCode: H.processFinalCode, processCodeAfterIncludes: this._eventInfo.customCode, multiTarget: i.PREPASS }, s); if (this._eventInfo.customCode = void 0, b) if (this._onEffectCreatedObservable && (TE.effect = b, TE.subMesh = t, this._onEffectCreatedObservable.notifyObservers(TE)), this.allowShaderHotSwapping && q && !b.isReady()) { if (b = q, i.markAsUnprocessed(), f = this.isFrozen, o) return i._areLightsDisposed = !0, !1; } else n.resetCachedMaterial(), t.setEffect(b, i, this._materialContext); } return !t.effect || !t.effect.isReady() ? !1 : (i._renderId = n.getRenderId(), t.effect._wasPreviouslyReady = !f, t.effect._wasPreviouslyUsingInstances = r, this._checkScenePerformancePriority(), !0); } /** * Builds the material UBO layouts. * Used internally during the effect preparation. */ buildUniformLayout() { const e = this._uniformBuffer; e.addUniform("diffuseLeftColor", 4), e.addUniform("diffuseRightColor", 4), e.addUniform("opacityParts", 4), e.addUniform("reflectionLeftColor", 4), e.addUniform("reflectionRightColor", 4), e.addUniform("refractionLeftColor", 4), e.addUniform("refractionRightColor", 4), e.addUniform("emissiveLeftColor", 4), e.addUniform("emissiveRightColor", 4), e.addUniform("vDiffuseInfos", 2), e.addUniform("vAmbientInfos", 2), e.addUniform("vOpacityInfos", 2), e.addUniform("vReflectionInfos", 2), e.addUniform("vReflectionPosition", 3), e.addUniform("vReflectionSize", 3), e.addUniform("vEmissiveInfos", 2), e.addUniform("vLightmapInfos", 2), e.addUniform("vSpecularInfos", 2), e.addUniform("vBumpInfos", 3), e.addUniform("diffuseMatrix", 16), e.addUniform("ambientMatrix", 16), e.addUniform("opacityMatrix", 16), e.addUniform("reflectionMatrix", 16), e.addUniform("emissiveMatrix", 16), e.addUniform("lightmapMatrix", 16), e.addUniform("specularMatrix", 16), e.addUniform("bumpMatrix", 16), e.addUniform("vTangentSpaceParams", 2), e.addUniform("pointSize", 1), e.addUniform("alphaCutOff", 1), e.addUniform("refractionMatrix", 16), e.addUniform("vRefractionInfos", 4), e.addUniform("vRefractionPosition", 3), e.addUniform("vRefractionSize", 3), e.addUniform("vSpecularColor", 4), e.addUniform("vEmissiveColor", 3), e.addUniform("vDiffuseColor", 4), e.addUniform("vAmbientColor", 3), super.buildUniformLayout(); } /** * Binds the submesh to this material by preparing the effect and shader to draw * @param world defines the world transformation matrix * @param mesh defines the mesh containing the submesh * @param subMesh defines the submesh to bind the material to */ bindForSubMesh(e, t, r) { var n; const i = this.getScene(), s = r.materialDefines; if (!s) return; const a = r.effect; if (!a) return; this._activeEffect = a, t.getMeshUniformBuffer().bindToEffect(a, "Mesh"), t.transferToEffect(e), this._uniformBuffer.bindToEffect(a, "Material"), this.prePassConfiguration.bindForSubMesh(this._activeEffect, i, t, e, this.isFrozen), this._eventInfo.subMesh = r, this._callbackPluginEventHardBindForSubMesh(this._eventInfo), s.OBJECTSPACE_NORMALMAP && (e.toNormalMatrix(this._normalMatrix), this.bindOnlyNormalMatrix(this._normalMatrix)); const f = a._forceRebindOnNextCall || this._mustRebind(i, a, t.visibility); Ye.BindBonesParameters(t, a); const o = this._uniformBuffer; if (f) { if (this.bindViewProjection(a), !o.useUbo || !this.isFrozen || !o.isSync || a._forceRebindOnNextCall) { if (Wt.FresnelEnabled && s.FRESNEL && (this.diffuseFresnelParameters && this.diffuseFresnelParameters.isEnabled && (o.updateColor4("diffuseLeftColor", this.diffuseFresnelParameters.leftColor, this.diffuseFresnelParameters.power), o.updateColor4("diffuseRightColor", this.diffuseFresnelParameters.rightColor, this.diffuseFresnelParameters.bias)), this.opacityFresnelParameters && this.opacityFresnelParameters.isEnabled && o.updateColor4("opacityParts", new Ne(this.opacityFresnelParameters.leftColor.toLuminance(), this.opacityFresnelParameters.rightColor.toLuminance(), this.opacityFresnelParameters.bias), this.opacityFresnelParameters.power), this.reflectionFresnelParameters && this.reflectionFresnelParameters.isEnabled && (o.updateColor4("reflectionLeftColor", this.reflectionFresnelParameters.leftColor, this.reflectionFresnelParameters.power), o.updateColor4("reflectionRightColor", this.reflectionFresnelParameters.rightColor, this.reflectionFresnelParameters.bias)), this.refractionFresnelParameters && this.refractionFresnelParameters.isEnabled && (o.updateColor4("refractionLeftColor", this.refractionFresnelParameters.leftColor, this.refractionFresnelParameters.power), o.updateColor4("refractionRightColor", this.refractionFresnelParameters.rightColor, this.refractionFresnelParameters.bias)), this.emissiveFresnelParameters && this.emissiveFresnelParameters.isEnabled && (o.updateColor4("emissiveLeftColor", this.emissiveFresnelParameters.leftColor, this.emissiveFresnelParameters.power), o.updateColor4("emissiveRightColor", this.emissiveFresnelParameters.rightColor, this.emissiveFresnelParameters.bias))), i.texturesEnabled) { if (this._diffuseTexture && Wt.DiffuseTextureEnabled && (o.updateFloat2("vDiffuseInfos", this._diffuseTexture.coordinatesIndex, this._diffuseTexture.level), Ye.BindTextureMatrix(this._diffuseTexture, o, "diffuse")), this._ambientTexture && Wt.AmbientTextureEnabled && (o.updateFloat2("vAmbientInfos", this._ambientTexture.coordinatesIndex, this._ambientTexture.level), Ye.BindTextureMatrix(this._ambientTexture, o, "ambient")), this._opacityTexture && Wt.OpacityTextureEnabled && (o.updateFloat2("vOpacityInfos", this._opacityTexture.coordinatesIndex, this._opacityTexture.level), Ye.BindTextureMatrix(this._opacityTexture, o, "opacity")), this._hasAlphaChannel() && o.updateFloat("alphaCutOff", this.alphaCutOff), this._reflectionTexture && Wt.ReflectionTextureEnabled && (o.updateFloat2("vReflectionInfos", this._reflectionTexture.level, this.roughness), o.updateMatrix("reflectionMatrix", this._reflectionTexture.getReflectionTextureMatrix()), this._reflectionTexture.boundingBoxSize)) { const d = this._reflectionTexture; o.updateVector3("vReflectionPosition", d.boundingBoxPosition), o.updateVector3("vReflectionSize", d.boundingBoxSize); } if (this._emissiveTexture && Wt.EmissiveTextureEnabled && (o.updateFloat2("vEmissiveInfos", this._emissiveTexture.coordinatesIndex, this._emissiveTexture.level), Ye.BindTextureMatrix(this._emissiveTexture, o, "emissive")), this._lightmapTexture && Wt.LightmapTextureEnabled && (o.updateFloat2("vLightmapInfos", this._lightmapTexture.coordinatesIndex, this._lightmapTexture.level), Ye.BindTextureMatrix(this._lightmapTexture, o, "lightmap")), this._specularTexture && Wt.SpecularTextureEnabled && (o.updateFloat2("vSpecularInfos", this._specularTexture.coordinatesIndex, this._specularTexture.level), Ye.BindTextureMatrix(this._specularTexture, o, "specular")), this._bumpTexture && i.getEngine().getCaps().standardDerivatives && Wt.BumpTextureEnabled && (o.updateFloat3("vBumpInfos", this._bumpTexture.coordinatesIndex, 1 / this._bumpTexture.level, this.parallaxScaleBias), Ye.BindTextureMatrix(this._bumpTexture, o, "bump"), i._mirroredCameraPosition ? o.updateFloat2("vTangentSpaceParams", this._invertNormalMapX ? 1 : -1, this._invertNormalMapY ? 1 : -1) : o.updateFloat2("vTangentSpaceParams", this._invertNormalMapX ? -1 : 1, this._invertNormalMapY ? -1 : 1)), this._refractionTexture && Wt.RefractionTextureEnabled) { let d = 1; if (this._refractionTexture.isCube || (o.updateMatrix("refractionMatrix", this._refractionTexture.getReflectionTextureMatrix()), this._refractionTexture.depth && (d = this._refractionTexture.depth)), o.updateFloat4("vRefractionInfos", this._refractionTexture.level, this.indexOfRefraction, d, this.invertRefractionY ? -1 : 1), this._refractionTexture.boundingBoxSize) { const v = this._refractionTexture; o.updateVector3("vRefractionPosition", v.boundingBoxPosition), o.updateVector3("vRefractionSize", v.boundingBoxSize); } } } this.pointsCloud && o.updateFloat("pointSize", this.pointSize), s.SPECULARTERM && o.updateColor4("vSpecularColor", this.specularColor, this.specularPower), o.updateColor3("vEmissiveColor", Wt.EmissiveTextureEnabled ? this.emissiveColor : Ne.BlackReadOnly), o.updateColor4("vDiffuseColor", this.diffuseColor, this.alpha), i.ambientColor.multiplyToRef(this.ambientColor, this._globalAmbientColor), o.updateColor3("vAmbientColor", this._globalAmbientColor); } i.texturesEnabled && (this._diffuseTexture && Wt.DiffuseTextureEnabled && a.setTexture("diffuseSampler", this._diffuseTexture), this._ambientTexture && Wt.AmbientTextureEnabled && a.setTexture("ambientSampler", this._ambientTexture), this._opacityTexture && Wt.OpacityTextureEnabled && a.setTexture("opacitySampler", this._opacityTexture), this._reflectionTexture && Wt.ReflectionTextureEnabled && (this._reflectionTexture.isCube ? a.setTexture("reflectionCubeSampler", this._reflectionTexture) : a.setTexture("reflection2DSampler", this._reflectionTexture)), this._emissiveTexture && Wt.EmissiveTextureEnabled && a.setTexture("emissiveSampler", this._emissiveTexture), this._lightmapTexture && Wt.LightmapTextureEnabled && a.setTexture("lightmapSampler", this._lightmapTexture), this._specularTexture && Wt.SpecularTextureEnabled && a.setTexture("specularSampler", this._specularTexture), this._bumpTexture && i.getEngine().getCaps().standardDerivatives && Wt.BumpTextureEnabled && a.setTexture("bumpSampler", this._bumpTexture), this._refractionTexture && Wt.RefractionTextureEnabled && (this._refractionTexture.isCube ? a.setTexture("refractionCubeSampler", this._refractionTexture) : a.setTexture("refraction2DSampler", this._refractionTexture))), this.getScene().useOrderIndependentTransparency && this.needAlphaBlendingForMesh(t) && this.getScene().depthPeelingRenderer.bind(a), this._eventInfo.subMesh = r, this._callbackPluginEventBindForSubMesh(this._eventInfo), Df(a, this, i), this.bindEyePosition(a); } else i.getEngine()._features.needToAlwaysBindUniformBuffers && (this._needToBindSceneUbo = !0); (f || !this.isFrozen) && (i.lightsEnabled && !this._disableLighting && Ye.BindLights(i, t, a, s, this._maxSimultaneousLights), (i.fogEnabled && t.applyFog && i.fogMode !== sr.FOGMODE_NONE || this._reflectionTexture || this._refractionTexture || t.receiveShadows || s.PREPASS) && this.bindView(a), Ye.BindFogParameters(i, t, a), s.NUM_MORPH_INFLUENCERS && Ye.BindMorphTargetParameters(t, a), s.BAKED_VERTEX_ANIMATION_TEXTURE && ((n = t.bakedVertexAnimationManager) === null || n === void 0 || n.bind(a, s.INSTANCES)), this.useLogarithmicDepth && Ye.BindLogDepth(s, a, i), this._imageProcessingConfiguration && !this._imageProcessingConfiguration.applyByPostProcess && this._imageProcessingConfiguration.bind(this._activeEffect)), this._afterBind(t, this._activeEffect), o.update(); } /** * Get the list of animatables in the material. * @returns the list of animatables object used in the material */ getAnimatables() { const e = super.getAnimatables(); return this._diffuseTexture && this._diffuseTexture.animations && this._diffuseTexture.animations.length > 0 && e.push(this._diffuseTexture), this._ambientTexture && this._ambientTexture.animations && this._ambientTexture.animations.length > 0 && e.push(this._ambientTexture), this._opacityTexture && this._opacityTexture.animations && this._opacityTexture.animations.length > 0 && e.push(this._opacityTexture), this._reflectionTexture && this._reflectionTexture.animations && this._reflectionTexture.animations.length > 0 && e.push(this._reflectionTexture), this._emissiveTexture && this._emissiveTexture.animations && this._emissiveTexture.animations.length > 0 && e.push(this._emissiveTexture), this._specularTexture && this._specularTexture.animations && this._specularTexture.animations.length > 0 && e.push(this._specularTexture), this._bumpTexture && this._bumpTexture.animations && this._bumpTexture.animations.length > 0 && e.push(this._bumpTexture), this._lightmapTexture && this._lightmapTexture.animations && this._lightmapTexture.animations.length > 0 && e.push(this._lightmapTexture), this._refractionTexture && this._refractionTexture.animations && this._refractionTexture.animations.length > 0 && e.push(this._refractionTexture), e; } /** * Gets the active textures from the material * @returns an array of textures */ getActiveTextures() { const e = super.getActiveTextures(); return this._diffuseTexture && e.push(this._diffuseTexture), this._ambientTexture && e.push(this._ambientTexture), this._opacityTexture && e.push(this._opacityTexture), this._reflectionTexture && e.push(this._reflectionTexture), this._emissiveTexture && e.push(this._emissiveTexture), this._specularTexture && e.push(this._specularTexture), this._bumpTexture && e.push(this._bumpTexture), this._lightmapTexture && e.push(this._lightmapTexture), this._refractionTexture && e.push(this._refractionTexture), e; } /** * Specifies if the material uses a texture * @param texture defines the texture to check against the material * @returns a boolean specifying if the material uses the texture */ hasTexture(e) { return !!(super.hasTexture(e) || this._diffuseTexture === e || this._ambientTexture === e || this._opacityTexture === e || this._reflectionTexture === e || this._emissiveTexture === e || this._specularTexture === e || this._bumpTexture === e || this._lightmapTexture === e || this._refractionTexture === e); } /** * Disposes the material * @param forceDisposeEffect specifies if effects should be forcefully disposed * @param forceDisposeTextures specifies if textures should be forcefully disposed */ dispose(e, t) { var r, n, i, s, a, f, o, d, v; t && ((r = this._diffuseTexture) === null || r === void 0 || r.dispose(), (n = this._ambientTexture) === null || n === void 0 || n.dispose(), (i = this._opacityTexture) === null || i === void 0 || i.dispose(), (s = this._reflectionTexture) === null || s === void 0 || s.dispose(), (a = this._emissiveTexture) === null || a === void 0 || a.dispose(), (f = this._specularTexture) === null || f === void 0 || f.dispose(), (o = this._bumpTexture) === null || o === void 0 || o.dispose(), (d = this._lightmapTexture) === null || d === void 0 || d.dispose(), (v = this._refractionTexture) === null || v === void 0 || v.dispose()), this._imageProcessingConfiguration && this._imageProcessingObserver && this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver), super.dispose(e, t); } /** * Makes a duplicate of the material, and gives it a new name * @param name defines the new name for the duplicated material * @param cloneTexturesOnlyOnce - if a texture is used in more than one channel (e.g diffuse and opacity), only clone it once and reuse it on the other channels. Default false. * @param rootUrl defines the root URL to use to load textures * @returns the cloned material */ clone(e, t = !0, r = "") { const n = jt.Clone(() => new Wt(e, this.getScene()), this, { cloneTexturesOnlyOnce: t }); return n.name = e, n.id = e, this.stencil.copyTo(n.stencil), this._clonePlugins(n, r), n; } /** * Creates a standard material from parsed material data * @param source defines the JSON representation of the material * @param scene defines the hosting scene * @param rootUrl defines the root URL to use to load textures and relative dependencies * @returns a new standard material */ static Parse(e, t, r) { const n = jt.Parse(() => new Wt(e.name, t), e, t, r); return e.stencil && n.stencil.parse(e.stencil, t, r), gt._parsePlugins(e, n, t, r), n; } // Flags used to enable or disable a type of texture for all Standard Materials /** * Are diffuse textures enabled in the application. */ static get DiffuseTextureEnabled() { return Dt.DiffuseTextureEnabled; } static set DiffuseTextureEnabled(e) { Dt.DiffuseTextureEnabled = e; } /** * Are detail textures enabled in the application. */ static get DetailTextureEnabled() { return Dt.DetailTextureEnabled; } static set DetailTextureEnabled(e) { Dt.DetailTextureEnabled = e; } /** * Are ambient textures enabled in the application. */ static get AmbientTextureEnabled() { return Dt.AmbientTextureEnabled; } static set AmbientTextureEnabled(e) { Dt.AmbientTextureEnabled = e; } /** * Are opacity textures enabled in the application. */ static get OpacityTextureEnabled() { return Dt.OpacityTextureEnabled; } static set OpacityTextureEnabled(e) { Dt.OpacityTextureEnabled = e; } /** * Are reflection textures enabled in the application. */ static get ReflectionTextureEnabled() { return Dt.ReflectionTextureEnabled; } static set ReflectionTextureEnabled(e) { Dt.ReflectionTextureEnabled = e; } /** * Are emissive textures enabled in the application. */ static get EmissiveTextureEnabled() { return Dt.EmissiveTextureEnabled; } static set EmissiveTextureEnabled(e) { Dt.EmissiveTextureEnabled = e; } /** * Are specular textures enabled in the application. */ static get SpecularTextureEnabled() { return Dt.SpecularTextureEnabled; } static set SpecularTextureEnabled(e) { Dt.SpecularTextureEnabled = e; } /** * Are bump textures enabled in the application. */ static get BumpTextureEnabled() { return Dt.BumpTextureEnabled; } static set BumpTextureEnabled(e) { Dt.BumpTextureEnabled = e; } /** * Are lightmap textures enabled in the application. */ static get LightmapTextureEnabled() { return Dt.LightmapTextureEnabled; } static set LightmapTextureEnabled(e) { Dt.LightmapTextureEnabled = e; } /** * Are refraction textures enabled in the application. */ static get RefractionTextureEnabled() { return Dt.RefractionTextureEnabled; } static set RefractionTextureEnabled(e) { Dt.RefractionTextureEnabled = e; } /** * Are color grading textures enabled in the application. */ static get ColorGradingTextureEnabled() { return Dt.ColorGradingTextureEnabled; } static set ColorGradingTextureEnabled(e) { Dt.ColorGradingTextureEnabled = e; } /** * Are fresnels enabled in the application. */ static get FresnelEnabled() { return Dt.FresnelEnabled; } static set FresnelEnabled(e) { Dt.FresnelEnabled = e; } } C([ en("diffuseTexture") ], Wt.prototype, "_diffuseTexture", void 0); C([ At("_markAllSubMeshesAsTexturesAndMiscDirty") ], Wt.prototype, "diffuseTexture", void 0); C([ en("ambientTexture") ], Wt.prototype, "_ambientTexture", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "ambientTexture", void 0); C([ en("opacityTexture") ], Wt.prototype, "_opacityTexture", void 0); C([ At("_markAllSubMeshesAsTexturesAndMiscDirty") ], Wt.prototype, "opacityTexture", void 0); C([ en("reflectionTexture") ], Wt.prototype, "_reflectionTexture", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "reflectionTexture", void 0); C([ en("emissiveTexture") ], Wt.prototype, "_emissiveTexture", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "emissiveTexture", void 0); C([ en("specularTexture") ], Wt.prototype, "_specularTexture", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "specularTexture", void 0); C([ en("bumpTexture") ], Wt.prototype, "_bumpTexture", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "bumpTexture", void 0); C([ en("lightmapTexture") ], Wt.prototype, "_lightmapTexture", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "lightmapTexture", void 0); C([ en("refractionTexture") ], Wt.prototype, "_refractionTexture", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "refractionTexture", void 0); C([ Oi("ambient") ], Wt.prototype, "ambientColor", void 0); C([ Oi("diffuse") ], Wt.prototype, "diffuseColor", void 0); C([ Oi("specular") ], Wt.prototype, "specularColor", void 0); C([ Oi("emissive") ], Wt.prototype, "emissiveColor", void 0); C([ M() ], Wt.prototype, "specularPower", void 0); C([ M("useAlphaFromDiffuseTexture") ], Wt.prototype, "_useAlphaFromDiffuseTexture", void 0); C([ At("_markAllSubMeshesAsTexturesAndMiscDirty") ], Wt.prototype, "useAlphaFromDiffuseTexture", void 0); C([ M("useEmissiveAsIllumination") ], Wt.prototype, "_useEmissiveAsIllumination", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "useEmissiveAsIllumination", void 0); C([ M("linkEmissiveWithDiffuse") ], Wt.prototype, "_linkEmissiveWithDiffuse", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "linkEmissiveWithDiffuse", void 0); C([ M("useSpecularOverAlpha") ], Wt.prototype, "_useSpecularOverAlpha", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "useSpecularOverAlpha", void 0); C([ M("useReflectionOverAlpha") ], Wt.prototype, "_useReflectionOverAlpha", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "useReflectionOverAlpha", void 0); C([ M("disableLighting") ], Wt.prototype, "_disableLighting", void 0); C([ At("_markAllSubMeshesAsLightsDirty") ], Wt.prototype, "disableLighting", void 0); C([ M("useObjectSpaceNormalMap") ], Wt.prototype, "_useObjectSpaceNormalMap", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "useObjectSpaceNormalMap", void 0); C([ M("useParallax") ], Wt.prototype, "_useParallax", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "useParallax", void 0); C([ M("useParallaxOcclusion") ], Wt.prototype, "_useParallaxOcclusion", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "useParallaxOcclusion", void 0); C([ M() ], Wt.prototype, "parallaxScaleBias", void 0); C([ M("roughness") ], Wt.prototype, "_roughness", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "roughness", void 0); C([ M() ], Wt.prototype, "indexOfRefraction", void 0); C([ M() ], Wt.prototype, "invertRefractionY", void 0); C([ M() ], Wt.prototype, "alphaCutOff", void 0); C([ M("useLightmapAsShadowmap") ], Wt.prototype, "_useLightmapAsShadowmap", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "useLightmapAsShadowmap", void 0); C([ eU("diffuseFresnelParameters") ], Wt.prototype, "_diffuseFresnelParameters", void 0); C([ At("_markAllSubMeshesAsFresnelDirty") ], Wt.prototype, "diffuseFresnelParameters", void 0); C([ eU("opacityFresnelParameters") ], Wt.prototype, "_opacityFresnelParameters", void 0); C([ At("_markAllSubMeshesAsFresnelAndMiscDirty") ], Wt.prototype, "opacityFresnelParameters", void 0); C([ eU("reflectionFresnelParameters") ], Wt.prototype, "_reflectionFresnelParameters", void 0); C([ At("_markAllSubMeshesAsFresnelDirty") ], Wt.prototype, "reflectionFresnelParameters", void 0); C([ eU("refractionFresnelParameters") ], Wt.prototype, "_refractionFresnelParameters", void 0); C([ At("_markAllSubMeshesAsFresnelDirty") ], Wt.prototype, "refractionFresnelParameters", void 0); C([ eU("emissiveFresnelParameters") ], Wt.prototype, "_emissiveFresnelParameters", void 0); C([ At("_markAllSubMeshesAsFresnelDirty") ], Wt.prototype, "emissiveFresnelParameters", void 0); C([ M("useReflectionFresnelFromSpecular") ], Wt.prototype, "_useReflectionFresnelFromSpecular", void 0); C([ At("_markAllSubMeshesAsFresnelDirty") ], Wt.prototype, "useReflectionFresnelFromSpecular", void 0); C([ M("useGlossinessFromSpecularMapAlpha") ], Wt.prototype, "_useGlossinessFromSpecularMapAlpha", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "useGlossinessFromSpecularMapAlpha", void 0); C([ M("maxSimultaneousLights") ], Wt.prototype, "_maxSimultaneousLights", void 0); C([ At("_markAllSubMeshesAsLightsDirty") ], Wt.prototype, "maxSimultaneousLights", void 0); C([ M("invertNormalMapX") ], Wt.prototype, "_invertNormalMapX", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "invertNormalMapX", void 0); C([ M("invertNormalMapY") ], Wt.prototype, "_invertNormalMapY", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "invertNormalMapY", void 0); C([ M("twoSidedLighting") ], Wt.prototype, "_twoSidedLighting", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Wt.prototype, "twoSidedLighting", void 0); C([ M("applyDecalMapAfterDetailMap") ], Wt.prototype, "_applyDecalMapAfterDetailMap", void 0); C([ At("_markAllSubMeshesAsMiscDirty") ], Wt.prototype, "applyDecalMapAfterDetailMap", void 0); Ue("BABYLON.StandardMaterial", Wt); sr.DefaultMaterialFactory = (A) => new Wt("default material", A); hr.prototype.createDynamicTexture = function(A, e, t, r) { const n = new As(this, ri.Dynamic); return n.baseWidth = A, n.baseHeight = e, t && (A = this.needPOTTextures ? hr.GetExponentOfTwo(A, this._caps.maxTextureSize) : A, e = this.needPOTTextures ? hr.GetExponentOfTwo(e, this._caps.maxTextureSize) : e), n.width = A, n.height = e, n.isReady = !1, n.generateMipMaps = t, n.samplingMode = r, this.updateTextureSamplingMode(r, n), this._internalTexturesCache.push(n), n; }; hr.prototype.updateDynamicTexture = function(A, e, t, r = !1, n, i = !1, s = !1) { if (!A) return; const a = this._gl, f = a.TEXTURE_2D, o = this._bindTextureDirectly(f, A, !0, i); this._unpackFlipY(t === void 0 ? A.invertY : t), r && a.pixelStorei(a.UNPACK_PREMULTIPLY_ALPHA_WEBGL, 1); const d = this._getWebGLTextureType(A.type), v = this._getInternalFormat(n || A.format), u = this._getRGBABufferInternalSizedFormat(A.type, v); a.texImage2D(f, 0, u, v, d, e), A.generateMipMaps && a.generateMipmap(f), o || this._bindTextureDirectly(f, null), r && a.pixelStorei(a.UNPACK_PREMULTIPLY_ALPHA_WEBGL, 0), n && (A.format = n), A._dynamicTextureSource = e, A._premulAlpha = r, A.invertY = t || !1, A.isReady = !0; }; class Xp extends We { /** * Creates a DynamicTexture * @param name defines the name of the texture * @param options provides 3 alternatives for width and height of texture, a canvas, object with width and height properties, number for both width and height * @param scene defines the scene where you want the texture * @param generateMipMaps defines the use of MinMaps or not (default is false) * @param samplingMode defines the sampling mode to use (default is Texture.TRILINEAR_SAMPLINGMODE) * @param format defines the texture format to use (default is Engine.TEXTUREFORMAT_RGBA) * @param invertY defines if the texture needs to be inverted on the y axis during loading */ constructor(e, t, r = null, n = !1, i = 3, s = 5, a) { super(null, r, !n, a, i, void 0, void 0, void 0, void 0, s), this.name = e, this.wrapU = We.CLAMP_ADDRESSMODE, this.wrapV = We.CLAMP_ADDRESSMODE, this._generateMipMaps = n; const f = this._getEngine(); if (!f) return; t.getContext ? (this._canvas = t, this._texture = f.createDynamicTexture(t.width, t.height, n, i)) : (this._canvas = f.createCanvas(1, 1), t.width || t.width === 0 ? this._texture = f.createDynamicTexture(t.width, t.height, n, i) : this._texture = f.createDynamicTexture(t, t, n, i)); const o = this.getSize(); this._canvas.width !== o.width && (this._canvas.width = o.width), this._canvas.height !== o.height && (this._canvas.height = o.height), this._context = this._canvas.getContext("2d"); } /** * Get the current class name of the texture useful for serialization or dynamic coding. * @returns "DynamicTexture" */ getClassName() { return "DynamicTexture"; } /** * Gets the current state of canRescale */ get canRescale() { return !0; } _recreate(e) { this._canvas.width = e.width, this._canvas.height = e.height, this.releaseInternalTexture(), this._texture = this._getEngine().createDynamicTexture(e.width, e.height, this._generateMipMaps, this.samplingMode); } /** * Scales the texture * @param ratio the scale factor to apply to both width and height */ scale(e) { const t = this.getSize(); t.width *= e, t.height *= e, this._recreate(t); } /** * Resizes the texture * @param width the new width * @param height the new height */ scaleTo(e, t) { const r = this.getSize(); r.width = e, r.height = t, this._recreate(r); } /** * Gets the context of the canvas used by the texture * @returns the canvas context of the dynamic texture */ getContext() { return this._context; } /** * Clears the texture * @param clearColor Defines the clear color to use */ clear(e) { const t = this.getSize(); e && (this._context.fillStyle = e), this._context.clearRect(0, 0, t.width, t.height); } /** * Updates the texture * @param invertY defines the direction for the Y axis (default is true - y increases downwards) * @param premulAlpha defines if alpha is stored as premultiplied (default is false) * @param allowGPUOptimization true to allow some specific GPU optimizations (subject to engine feature "allowGPUOptimizationsForGUI" being true) */ update(e, t = !1, r = !1) { this._getEngine().updateDynamicTexture(this._texture, this._canvas, e === void 0 ? !0 : e, t, this._format || void 0, void 0, r); } /** * Draws text onto the texture * @param text defines the text to be drawn * @param x defines the placement of the text from the left * @param y defines the placement of the text from the top when invertY is true and from the bottom when false * @param font defines the font to be used with font-style, font-size, font-name * @param color defines the color used for the text * @param fillColor defines the color for the canvas, use null to not overwrite canvas (this bleands with the background to replace, use the clear function) * @param invertY defines the direction for the Y axis (default is true - y increases downwards) * @param update defines whether texture is immediately update (default is true) */ drawText(e, t, r, n, i, s, a, f = !0) { const o = this.getSize(); if (s && (this._context.fillStyle = s, this._context.fillRect(0, 0, o.width, o.height)), this._context.font = n, t == null) { const d = this._context.measureText(e); t = (o.width - d.width) / 2; } if (r == null) { const d = parseInt(n.replace(/\D/g, "")); r = o.height / 2 + d / 3.65; } this._context.fillStyle = i || "", this._context.fillText(e, t, r), f && this.update(a); } /** * Clones the texture * @returns the clone of the texture. */ clone() { const e = this.getScene(); if (!e) return this; const t = this.getSize(), r = new Xp(this.name, t, e, this._generateMipMaps); return r.hasAlpha = this.hasAlpha, r.level = this.level, r.wrapU = this.wrapU, r.wrapV = this.wrapV, r; } /** * Serializes the dynamic texture. The scene should be ready before the dynamic texture is serialized * @returns a serialized dynamic texture object */ serialize() { const e = this.getScene(); e && !e.isReady() && Se.Warn("The scene must be ready before serializing the dynamic texture"); const t = super.serialize(); return Xp._IsCanvasElement(this._canvas) && (t.base64String = this._canvas.toDataURL()), t.invertY = this._invertY, t.samplingMode = this.samplingMode, t; } static _IsCanvasElement(e) { return e.toDataURL !== void 0; } /** @internal */ _rebuild() { this.update(); } } class XQ { /** * Check if fixed foveation is supported on this device */ get isFixedFoveationSupported() { return this.layerType == "XRWebGLLayer" && typeof this.layer.fixedFoveation == "number"; } /** * Get the fixed foveation currently set, as specified by the webxr specs * If this returns null, then fixed foveation is not supported */ get fixedFoveation() { return this.isFixedFoveationSupported ? this.layer.fixedFoveation : null; } /** * Set the fixed foveation to the specified value, as specified by the webxr specs * This value will be normalized to be between 0 and 1, 1 being max foveation, 0 being no foveation */ set fixedFoveation(e) { if (this.isFixedFoveationSupported) { const t = Math.max(0, Math.min(1, e || 0)); this.layer.fixedFoveation = t; } } constructor(e, t, r, n, i) { this.getWidth = e, this.getHeight = t, this.layer = r, this.layerType = n, this.createRenderTargetTextureProvider = i; } } class TQ { constructor(e, t) { this._scene = e, this.layerWrapper = t, this._renderTargetTextures = new Array(), this._engine = e.getEngine(); } _createInternalTexture(e, t) { const r = new As(this._engine, ri.Unknown, !0); return r.width = e.width, r.height = e.height, r._hardwareTexture = new jm(t, this._engine._gl), r.isReady = !0, r; } _createRenderTargetTexture(e, t, r, n, i, s) { if (!this._engine) throw new Error("Engine is disposed"); const a = { width: e, height: t }, f = s ? new $5(this._scene, a) : new Ta("XR renderTargetTexture", a, this._scene), o = f.renderTarget; if (o._samples = f.samples, (r || !n) && (o._framebuffer = r), n) if (s) o._colorTextureArray = n; else { const d = this._createInternalTexture(a, n); o.setTexture(d, 0), f._texture = d; } return i && (s ? o._depthStencilTextureArray = i : o._depthStencilTexture = this._createInternalTexture(a, i)), f.disableRescaling(), typeof XRWebGLBinding < "u" && (f.skipInitialClear = !0), this._renderTargetTextures.push(f), f; } _destroyRenderTargetTexture(e) { this._renderTargetTextures.splice(this._renderTargetTextures.indexOf(e), 1), e.dispose(); } getFramebufferDimensions() { return this._framebufferDimensions; } dispose() { this._renderTargetTextures.forEach((e) => e.dispose()), this._renderTargetTextures.length = 0; } } class qQ extends XQ { /** * @param layer is the layer to be wrapped. * @returns a new WebXRLayerWrapper wrapping the provided XRWebGLLayer. */ constructor(e) { super(() => e.framebufferWidth, () => e.framebufferHeight, e, "XRWebGLLayer", (t) => new bve(t.scene, this)), this.layer = e; } } class bve extends TQ { constructor(e, t) { super(e, t), this.layerWrapper = t, this._layer = t.layer, this._framebufferDimensions = { framebufferWidth: this._layer.framebufferWidth, framebufferHeight: this._layer.framebufferHeight }; } trySetViewportForView(e, t) { const r = this._layer.getViewport(t); if (!r) return !1; const n = this._framebufferDimensions.framebufferWidth, i = this._framebufferDimensions.framebufferHeight; return e.x = r.x / n, e.y = r.y / i, e.width = r.width / n, e.height = r.height / i, !0; } // eslint-disable-next-line @typescript-eslint/no-unused-vars getRenderTargetTextureForEye(e) { const t = this._layer.framebufferWidth, r = this._layer.framebufferHeight, n = this._layer.framebuffer; return (!this._rtt || t !== this._framebufferDimensions.framebufferWidth || r !== this._framebufferDimensions.framebufferHeight || n !== this._framebuffer) && (this._rtt = this._createRenderTargetTexture(t, r, n), this._framebufferDimensions.framebufferWidth = t, this._framebufferDimensions.framebufferHeight = r, this._framebuffer = n), this._rtt; } getRenderTargetTextureForView(e) { return this.getRenderTargetTextureForEye(e.eye); } } class YR { /** * Get the default values of the configuration object * @param engine defines the engine to use (can be null) * @returns default values of this configuration object */ static GetDefaults(e) { const t = new YR(); return t.canvasOptions = { antialias: !0, depth: !0, stencil: e ? e.isStencilEnable : !0, alpha: !0, framebufferScaleFactor: 1 }, t.newCanvasCssStyle = "position:absolute; bottom:0px;right:0px;z-index:10;width:90%;height:100%;background-color: #000000;", t; } } class Jee { /** * Initializes the canvas to be added/removed upon entering/exiting xr * @param _xrSessionManager The XR Session manager * @param _options optional configuration for this canvas output. defaults will be used if not provided */ constructor(e, t = YR.GetDefaults()) { if (this._options = t, this._canvas = null, this._engine = null, this.xrLayer = null, this._xrLayerWrapper = null, this.onXRLayerInitObservable = new Oe(), this._engine = e.scene.getEngine(), this._engine.onDisposeObservable.addOnce(() => { this._engine = null; }), t.canvasElement) this._setManagedOutputCanvas(t.canvasElement); else { const r = document.createElement("canvas"); r.style.cssText = this._options.newCanvasCssStyle || "position:absolute; bottom:0px;right:0px;", this._setManagedOutputCanvas(r); } e.onXRSessionInit.add(() => { this._addCanvas(); }), e.onXRSessionEnded.add(() => { this._removeCanvas(); }); } /** * Disposes of the object */ dispose() { this._removeCanvas(), this._setManagedOutputCanvas(null); } /** * Initializes a XRWebGLLayer to be used as the session's baseLayer. * @param xrSession xr session * @returns a promise that will resolve once the XR Layer has been created */ async initializeXRLayerAsync(e) { const t = () => (this.xrLayer = new XRWebGLLayer(e, this.canvasContext, this._options.canvasOptions), this._xrLayerWrapper = new qQ(this.xrLayer), this.onXRLayerInitObservable.notifyObservers(this.xrLayer), this.xrLayer); return this.canvasContext.makeXRCompatible ? this.canvasContext.makeXRCompatible().then( // catch any error and continue. When using the emulator is throws this error for no apparent reason. () => { }, () => { ye.Warn("Error executing makeXRCompatible. This does not mean that the session will work incorrectly."); } ).then(() => t()) : Promise.resolve(t()); } _addCanvas() { this._canvas && this._engine && this._canvas !== this._engine.getRenderingCanvas() && document.body.appendChild(this._canvas), this.xrLayer ? this._setCanvasSize(!0) : this.onXRLayerInitObservable.addOnce(() => { this._setCanvasSize(!0); }); } _removeCanvas() { this._canvas && this._engine && document.body.contains(this._canvas) && this._canvas !== this._engine.getRenderingCanvas() && document.body.removeChild(this._canvas), this._setCanvasSize(!1); } _setCanvasSize(e = !0, t = this._xrLayerWrapper) { !this._canvas || !this._engine || (e ? t && (this._canvas !== this._engine.getRenderingCanvas() ? (this._canvas.style.width = t.getWidth() + "px", this._canvas.style.height = t.getHeight() + "px") : this._engine.setSize(t.getWidth(), t.getHeight())) : this._originalCanvasSize && (this._canvas !== this._engine.getRenderingCanvas() ? (this._canvas.style.width = this._originalCanvasSize.width + "px", this._canvas.style.height = this._originalCanvasSize.height + "px") : this._engine.setSize(this._originalCanvasSize.width, this._originalCanvasSize.height))); } _setManagedOutputCanvas(e) { this._removeCanvas(), e ? (this._originalCanvasSize = { width: e.offsetWidth, height: e.offsetHeight }, this._canvas = e, this.canvasContext = this._canvas.getContext("webgl2"), this.canvasContext || (this.canvasContext = this._canvas.getContext("webgl"))) : (this._canvas = null, this.canvasContext = null); } } class zee extends XQ { constructor(e) { super(() => e.framebufferWidth, () => e.framebufferHeight, e, "XRWebGLLayer", (t) => new Gee(t, this)), this.layer = e; } } class Gee extends TQ { constructor(e, t) { super(e.scene, t), this.layerWrapper = t, this._nativeRTTProvider = navigator.xr.getNativeRenderTargetProvider(e.session, this._createRenderTargetTexture.bind(this), this._destroyRenderTargetTexture.bind(this)), this._nativeLayer = t.layer; } trySetViewportForView(e) { return e.x = 0, e.y = 0, e.width = 1, e.height = 1, !0; } getRenderTargetTextureForEye(e) { return this._nativeRTTProvider.getRenderTargetForEye(e); } getRenderTargetTextureForView(e) { return this._nativeRTTProvider.getRenderTargetForEye(e.eye); } getFramebufferDimensions() { return { framebufferWidth: this._nativeLayer.framebufferWidth, framebufferHeight: this._nativeLayer.framebufferHeight }; } } class Zee { constructor(e) { this._nativeRenderTarget = navigator.xr.getWebXRRenderTarget(e.scene.getEngine()); } async initializeXRLayerAsync(e) { return await this._nativeRenderTarget.initializeXRLayerAsync(e), this.xrLayer = this._nativeRenderTarget.xrLayer, this.xrLayer; } dispose() { } } class MR { /** * Constructs a WebXRSessionManager, this must be initialized within a user action before usage * @param scene The scene which the session should be created for */ constructor(e) { this.scene = e, this.currentTimestamp = -1, this.defaultHeightCompensation = 1.7, this.onXRFrameObservable = new Oe(), this.onXRReferenceSpaceChanged = new Oe(), this.onXRSessionEnded = new Oe(), this.onXRSessionInit = new Oe(), this.inXRFrameLoop = !1, this.inXRSession = !1, this._engine = e.getEngine(), this._onEngineDisposedObserver = this._engine.onDisposeObservable.addOnce(() => { this._engine = null; }), e.onDisposeObservable.addOnce(() => { this.dispose(); }); } /** * The current reference space used in this session. This reference space can constantly change! * It is mainly used to offset the camera's position. */ get referenceSpace() { return this._referenceSpace; } /** * Set a new reference space and triggers the observable */ set referenceSpace(e) { this._referenceSpace = e, this.onXRReferenceSpaceChanged.notifyObservers(this._referenceSpace); } /** * The mode for the managed XR session */ get sessionMode() { return this._sessionMode; } /** * Disposes of the session manager * This should be called explicitly by the dev, if required. */ dispose() { var e; this.inXRSession && this.exitXRAsync(), this.onXRFrameObservable.clear(), this.onXRSessionEnded.clear(), this.onXRReferenceSpaceChanged.clear(), this.onXRSessionInit.clear(), (e = this._engine) === null || e === void 0 || e.onDisposeObservable.remove(this._onEngineDisposedObserver), this._engine = null; } /** * Stops the xrSession and restores the render loop * @returns Promise which resolves after it exits XR */ exitXRAsync() { return this.session && this.inXRSession ? (this.inXRSession = !1, this.session.end().catch(() => { Se.Warn("Could not end XR session."); })) : Promise.resolve(); } /** * Attempts to set the framebuffer-size-normalized viewport to be rendered this frame for this view. * In the event of a failure, the supplied viewport is not updated. * @param viewport the viewport to which the view will be rendered * @param view the view for which to set the viewport * @returns whether the operation was successful */ trySetViewportForView(e, t) { var r; return ((r = this._baseLayerRTTProvider) === null || r === void 0 ? void 0 : r.trySetViewportForView(e, t)) || !1; } /** * Gets the correct render target texture to be rendered this frame for this eye * @param eye the eye for which to get the render target * @returns the render target for the specified eye or null if not available */ getRenderTargetTextureForEye(e) { var t; return ((t = this._baseLayerRTTProvider) === null || t === void 0 ? void 0 : t.getRenderTargetTextureForEye(e)) || null; } /** * Gets the correct render target texture to be rendered this frame for this view * @param view the view for which to get the render target * @returns the render target for the specified view or null if not available */ getRenderTargetTextureForView(e) { var t; return ((t = this._baseLayerRTTProvider) === null || t === void 0 ? void 0 : t.getRenderTargetTextureForView(e)) || null; } /** * Creates a WebXRRenderTarget object for the XR session * @param options optional options to provide when creating a new render target * @returns a WebXR render target to which the session can render */ getWebXRRenderTarget(e) { const t = this.scene.getEngine(); return this._xrNavigator.xr.native ? new Zee(this) : (e = e || YR.GetDefaults(t), e.canvasElement = e.canvasElement || t.getRenderingCanvas() || void 0, new Jee(this, e)); } /** * Initializes the manager * After initialization enterXR can be called to start an XR session * @returns Promise which resolves after it is initialized */ initializeAsync() { return this._xrNavigator = navigator, this._xrNavigator.xr ? Promise.resolve() : Promise.reject("WebXR not available"); } /** * Initializes an xr session * @param xrSessionMode mode to initialize * @param xrSessionInit defines optional and required values to pass to the session builder * @returns a promise which will resolve once the session has been initialized */ initializeSessionAsync(e = "immersive-vr", t = {}) { return this._xrNavigator.xr.requestSession(e, t).then((r) => (this.session = r, this._sessionMode = e, this.onXRSessionInit.notifyObservers(r), this.inXRSession = !0, this.session.addEventListener("end", () => { var n; this.inXRSession = !1, this.onXRSessionEnded.notifyObservers(null), this._engine && (this._engine.framebufferDimensionsObject = null, this._engine.restoreDefaultFramebuffer(), this._engine.customAnimationFrameRequester = null, this._engine._renderLoop()), this.isNative && ((n = this._baseLayerRTTProvider) === null || n === void 0 || n.dispose()), this._baseLayerRTTProvider = null, this._baseLayerWrapper = null; }, { once: !0 }), this.session)); } /** * Checks if a session would be supported for the creation options specified * @param sessionMode session mode to check if supported eg. immersive-vr * @returns A Promise that resolves to true if supported and false if not */ isSessionSupportedAsync(e) { return MR.IsSessionSupportedAsync(e); } /** * Resets the reference space to the one started the session */ resetReferenceSpace() { this.referenceSpace = this.baseReferenceSpace; } /** * Starts rendering to the xr layer */ runXRRenderLoop() { var e; !this.inXRSession || !this._engine || (this._engine.customAnimationFrameRequester = { requestAnimationFrame: (t) => this.session.requestAnimationFrame(t), renderFunction: (t, r) => { var n; !this.inXRSession || !this._engine || (this.currentFrame = r, this.currentTimestamp = t, r && (this.inXRFrameLoop = !0, this._engine.framebufferDimensionsObject = ((n = this._baseLayerRTTProvider) === null || n === void 0 ? void 0 : n.getFramebufferDimensions()) || null, this.onXRFrameObservable.notifyObservers(r), this._engine._renderLoop(), this._engine.framebufferDimensionsObject = null, this.inXRFrameLoop = !1)); } }, this._engine.framebufferDimensionsObject = ((e = this._baseLayerRTTProvider) === null || e === void 0 ? void 0 : e.getFramebufferDimensions()) || null, typeof window < "u" && window.cancelAnimationFrame && window.cancelAnimationFrame(this._engine._frameHandler), this._engine._renderLoop()); } /** * Sets the reference space on the xr session * @param referenceSpaceType space to set * @returns a promise that will resolve once the reference space has been set */ setReferenceSpaceTypeAsync(e = "local-floor") { return this.session.requestReferenceSpace(e).then((t) => t, (t) => (Se.Error("XR.requestReferenceSpace failed for the following reason: "), Se.Error(t), Se.Log('Defaulting to universally-supported "viewer" reference space type.'), this.session.requestReferenceSpace("viewer").then((r) => { const n = new XRRigidTransform({ x: 0, y: -this.defaultHeightCompensation, z: 0 }); return r.getOffsetReferenceSpace(n); }, (r) => { throw Se.Error(r), 'XR initialization failed: required "viewer" reference space type not supported.'; }))).then((t) => this.session.requestReferenceSpace("viewer").then((r) => (this.viewerReferenceSpace = r, t))).then((t) => (this.referenceSpace = this.baseReferenceSpace = t, this.referenceSpace)); } /** * Updates the render state of the session. * Note that this is deprecated in favor of WebXRSessionManager.updateRenderState(). * @param state state to set * @returns a promise that resolves once the render state has been updated * @deprecated */ updateRenderStateAsync(e) { return Promise.resolve(this.session.updateRenderState(e)); } /** * @internal */ _setBaseLayerWrapper(e) { var t, r; this.isNative && ((t = this._baseLayerRTTProvider) === null || t === void 0 || t.dispose()), this._baseLayerWrapper = e, this._baseLayerRTTProvider = ((r = this._baseLayerWrapper) === null || r === void 0 ? void 0 : r.createRenderTargetTextureProvider(this)) || null; } /** * @internal */ _getBaseLayerWrapper() { return this._baseLayerWrapper; } /** * Updates the render state of the session * @param state state to set */ updateRenderState(e) { e.baseLayer && this._setBaseLayerWrapper(this.isNative ? new zee(e.baseLayer) : new qQ(e.baseLayer)), this.session.updateRenderState(e); } /** * Returns a promise that resolves with a boolean indicating if the provided session mode is supported by this browser * @param sessionMode defines the session to test * @returns a promise with boolean as final value */ static IsSessionSupportedAsync(e) { if (!navigator.xr) return Promise.resolve(!1); const t = navigator.xr.isSessionSupported || navigator.xr.supportsSession; return t ? t.call(navigator.xr, e).then((r) => { const n = typeof r > "u" ? !0 : r; return Promise.resolve(n); }).catch((r) => (Se.Warn(r), Promise.resolve(!1))) : Promise.resolve(!1); } /** * Returns true if Babylon.js is using the BabylonNative backend, otherwise false */ get isNative() { var e; return (e = this._xrNavigator.xr.native) !== null && e !== void 0 ? e : !1; } /** * The current frame rate as reported by the device */ get currentFrameRate() { var e; return (e = this.session) === null || e === void 0 ? void 0 : e.frameRate; } /** * A list of supported frame rates (only available in-session! */ get supportedFrameRates() { var e; return (e = this.session) === null || e === void 0 ? void 0 : e.supportedFrameRates; } /** * Set the framerate of the session. * @param rate the new framerate. This value needs to be in the supportedFrameRates array * @returns a promise that resolves once the framerate has been set */ updateTargetFrameRate(e) { return this.session.updateTargetFrameRate(e); } /** * Run a callback in the xr render loop * @param callback the callback to call when in XR Frame * @param ignoreIfNotInSession if no session is currently running, run it first thing on the next session */ runInXRFrame(e, t = !0) { this.inXRFrameLoop ? e() : (this.inXRSession || !t) && this.onXRFrameObservable.addOnce(e); } /** * Check if fixed foveation is supported on this device */ get isFixedFoveationSupported() { var e; return ((e = this._baseLayerWrapper) === null || e === void 0 ? void 0 : e.isFixedFoveationSupported) || !1; } /** * Get the fixed foveation currently set, as specified by the webxr specs * If this returns null, then fixed foveation is not supported */ get fixedFoveation() { var e; return ((e = this._baseLayerWrapper) === null || e === void 0 ? void 0 : e.fixedFoveation) || null; } /** * Set the fixed foveation to the specified value, as specified by the webxr specs * This value will be normalized to be between 0 and 1, 1 being max foveation, 0 being no foveation */ set fixedFoveation(e) { const t = Math.max(0, Math.min(1, e || 0)); this._baseLayerWrapper && (this._baseLayerWrapper.fixedFoveation = t); } /** * Get the features enabled on the current session * This is only available in-session! * @see https://www.w3.org/TR/webxr/#dom-xrsession-enabledfeatures */ get enabledFeatures() { var e, t; return (t = (e = this.session) === null || e === void 0 ? void 0 : e.enabledFeatures) !== null && t !== void 0 ? t : null; } } var d9; (function(A) { A[A.ENTERING_XR = 0] = "ENTERING_XR", A[A.EXITING_XR = 1] = "EXITING_XR", A[A.IN_XR = 2] = "IN_XR", A[A.NOT_IN_XR = 3] = "NOT_IN_XR"; })(d9 || (d9 = {})); var Jw; (function(A) { A[A.NOT_TRACKING = 0] = "NOT_TRACKING", A[A.TRACKING_LOST = 1] = "TRACKING_LOST", A[A.TRACKING = 2] = "TRACKING"; })(Jw || (Jw = {})); Ee._GroundMeshParser = (A, e) => dU.Parse(A, e); class dU extends Ee { constructor(e, t) { super(e, t), this.generateOctree = !1; } /** * "GroundMesh" * @returns "GroundMesh" */ getClassName() { return "GroundMesh"; } /** * The minimum of x and y subdivisions */ get subdivisions() { return Math.min(this._subdivisionsX, this._subdivisionsY); } /** * X subdivisions */ get subdivisionsX() { return this._subdivisionsX; } /** * Y subdivisions */ get subdivisionsY() { return this._subdivisionsY; } /** * This function will divide the mesh into submeshes and update an octree to help to select the right submeshes * for rendering, picking and collision computations. Please note that you must have a decent number of submeshes * to get performance improvements when using an octree. * @param chunksCount the number of submeshes the mesh will be divided into * @param octreeBlocksSize the maximum size of the octree blocks (Default: 32) */ optimize(e, t = 32) { this._subdivisionsX = e, this._subdivisionsY = e, this.subdivide(e); const r = this; r.createOrUpdateSubmeshesOctree && r.createOrUpdateSubmeshesOctree(t); } /** * Returns a height (y) value in the World system : * the ground altitude at the coordinates (x, z) expressed in the World system. * @param x x coordinate * @param z z coordinate * @returns the ground y position if (x, z) are outside the ground surface. */ getHeightAtCoordinates(e, t) { const r = this.getWorldMatrix(), n = ue.Matrix[5]; r.invertToRef(n); const i = ue.Vector3[8]; if (S.TransformCoordinatesFromFloatsToRef(e, 0, t, n, i), e = i.x, t = i.z, e < this._minX || e >= this._maxX || t <= this._minZ || t > this._maxZ) return this.position.y; (!this._heightQuads || this._heightQuads.length == 0) && (this._initHeightQuads(), this._computeHeightQuads()); const s = this._getFacetAt(e, t), a = -(s.x * e + s.z * t + s.w) / s.y; return S.TransformCoordinatesFromFloatsToRef(0, a, 0, r, i), i.y; } /** * Returns a normalized vector (Vector3) orthogonal to the ground * at the ground coordinates (x, z) expressed in the World system. * @param x x coordinate * @param z z coordinate * @returns Vector3(0.0, 1.0, 0.0) if (x, z) are outside the ground surface. */ getNormalAtCoordinates(e, t) { const r = new S(0, 1, 0); return this.getNormalAtCoordinatesToRef(e, t, r), r; } /** * Updates the Vector3 passed a reference with a normalized vector orthogonal to the ground * at the ground coordinates (x, z) expressed in the World system. * Doesn't update the reference Vector3 if (x, z) are outside the ground surface. * @param x x coordinate * @param z z coordinate * @param ref vector to store the result * @returns the GroundMesh. */ getNormalAtCoordinatesToRef(e, t, r) { const n = this.getWorldMatrix(), i = ue.Matrix[5]; n.invertToRef(i); const s = ue.Vector3[8]; if (S.TransformCoordinatesFromFloatsToRef(e, 0, t, i, s), e = s.x, t = s.z, e < this._minX || e > this._maxX || t < this._minZ || t > this._maxZ) return this; (!this._heightQuads || this._heightQuads.length == 0) && (this._initHeightQuads(), this._computeHeightQuads()); const a = this._getFacetAt(e, t); return S.TransformNormalFromFloatsToRef(a.x, a.y, a.z, n, r), this; } /** * Force the heights to be recomputed for getHeightAtCoordinates() or getNormalAtCoordinates() * if the ground has been updated. * This can be used in the render loop. * @returns the GroundMesh. */ updateCoordinateHeights() { return (!this._heightQuads || this._heightQuads.length == 0) && this._initHeightQuads(), this._computeHeightQuads(), this; } // Returns the element "facet" from the heightQuads array relative to (x, z) local coordinates _getFacetAt(e, t) { const r = Math.floor((e + this._maxX) * this._subdivisionsX / this._width), n = Math.floor(-(t + this._maxZ) * this._subdivisionsY / this._height + this._subdivisionsY), i = this._heightQuads[n * this._subdivisionsX + r]; let s; return t < i.slope.x * e + i.slope.y ? s = i.facet1 : s = i.facet2, s; } // Creates and populates the heightMap array with "facet" elements : // a quad is two triangular facets separated by a slope, so a "facet" element is 1 slope + 2 facets // slope : Vector2(c, h) = 2D diagonal line equation setting apart two triangular facets in a quad : z = cx + h // facet1 : Vector4(a, b, c, d) = first facet 3D plane equation : ax + by + cz + d = 0 // facet2 : Vector4(a, b, c, d) = second facet 3D plane equation : ax + by + cz + d = 0 // Returns the GroundMesh. _initHeightQuads() { const e = this._subdivisionsX, t = this._subdivisionsY; this._heightQuads = new Array(); for (let r = 0; r < t; r++) for (let n = 0; n < e; n++) { const i = { slope: at.Zero(), facet1: new Ir(0, 0, 0, 0), facet2: new Ir(0, 0, 0, 0) }; this._heightQuads[r * e + n] = i; } return this; } // Compute each quad element values and update the heightMap array : // slope : Vector2(c, h) = 2D diagonal line equation setting apart two triangular facets in a quad : z = cx + h // facet1 : Vector4(a, b, c, d) = first facet 3D plane equation : ax + by + cz + d = 0 // facet2 : Vector4(a, b, c, d) = second facet 3D plane equation : ax + by + cz + d = 0 // Returns the GroundMesh. _computeHeightQuads() { const e = this.getVerticesData(J.PositionKind); if (!e) return this; const t = ue.Vector3[3], r = ue.Vector3[2], n = ue.Vector3[1], i = ue.Vector3[0], s = ue.Vector3[4], a = ue.Vector3[5], f = ue.Vector3[6], o = ue.Vector3[7], d = ue.Vector3[8]; let v = 0, u = 0, l = 0, P = 0, p = 0, c = 0, H = 0; const T = this._subdivisionsX, q = this._subdivisionsY; for (let b = 0; b < q; b++) for (let j = 0; j < T; j++) { v = j * 3, u = b * (T + 1) * 3, l = (b + 1) * (T + 1) * 3, t.x = e[u + v], t.y = e[u + v + 1], t.z = e[u + v + 2], r.x = e[u + v + 3], r.y = e[u + v + 4], r.z = e[u + v + 5], n.x = e[l + v], n.y = e[l + v + 1], n.z = e[l + v + 2], i.x = e[l + v + 3], i.y = e[l + v + 4], i.z = e[l + v + 5], P = (i.z - t.z) / (i.x - t.x), p = t.z - P * t.x, r.subtractToRef(t, s), n.subtractToRef(t, a), i.subtractToRef(t, f), S.CrossToRef(f, a, o), S.CrossToRef(s, f, d), o.normalize(), d.normalize(), c = -(o.x * t.x + o.y * t.y + o.z * t.z), H = -(d.x * r.x + d.y * r.y + d.z * r.z); const w = this._heightQuads[b * T + j]; w.slope.copyFromFloats(P, p), w.facet1.copyFromFloats(o.x, o.y, o.z, c), w.facet2.copyFromFloats(d.x, d.y, d.z, H); } return this; } /** * Serializes this ground mesh * @param serializationObject object to write serialization to */ serialize(e) { super.serialize(e), e.subdivisionsX = this._subdivisionsX, e.subdivisionsY = this._subdivisionsY, e.minX = this._minX, e.maxX = this._maxX, e.minZ = this._minZ, e.maxZ = this._maxZ, e.width = this._width, e.height = this._height; } /** * Parses a serialized ground mesh * @param parsedMesh the serialized mesh * @param scene the scene to create the ground mesh in * @returns the created ground mesh */ static Parse(e, t) { const r = new dU(e.name, t); return r._subdivisionsX = e.subdivisionsX || 1, r._subdivisionsY = e.subdivisionsY || 1, r._minX = e.minX, r._maxX = e.maxX, r._minZ = e.minZ, r._maxZ = e.maxZ, r._width = e.width, r._height = e.height, r; } } function y2(A) { const e = [], t = [], r = [], n = []; let i, s; const a = A.width || 1, f = A.height || 1, o = (A.subdivisionsX || A.subdivisions || 1) | 0, d = (A.subdivisionsY || A.subdivisions || 1) | 0; for (i = 0; i <= d; i++) for (s = 0; s <= o; s++) { const u = new S(s * a / o - a / 2, 0, (d - i) * f / d - f / 2), l = new S(0, 1, 0); t.push(u.x, u.y, u.z), r.push(l.x, l.y, l.z), n.push(s / o, us.UseOpenGLOrientationForUV ? i / d : 1 - i / d); } for (i = 0; i < d; i++) for (s = 0; s < o; s++) e.push(s + 1 + (i + 1) * (o + 1)), e.push(s + 1 + i * (o + 1)), e.push(s + i * (o + 1)), e.push(s + (i + 1) * (o + 1)), e.push(s + 1 + (i + 1) * (o + 1)), e.push(s + i * (o + 1)); const v = new Ut(); return v.indices = e, v.positions = t, v.normals = r, v.uvs = n, v; } function bQ(A) { const e = A.xmin !== void 0 && A.xmin !== null ? A.xmin : -1, t = A.zmin !== void 0 && A.zmin !== null ? A.zmin : -1, r = A.xmax !== void 0 && A.xmax !== null ? A.xmax : 1, n = A.zmax !== void 0 && A.zmax !== null ? A.zmax : 1, i = A.subdivisions || { w: 1, h: 1 }, s = A.precision || { w: 1, h: 1 }, a = [], f = [], o = [], d = []; let v, u, l, P; i.h = i.h < 1 ? 1 : i.h, i.w = i.w < 1 ? 1 : i.w, s.w = s.w < 1 ? 1 : s.w, s.h = s.h < 1 ? 1 : s.h; const p = { w: (r - e) / i.w, h: (n - t) / i.h }; function c(T, q, b, j) { const w = f.length / 3, m = s.w + 1; for (v = 0; v < s.h; v++) for (u = 0; u < s.w; u++) { const k = [w + u + v * m, w + (u + 1) + v * m, w + (u + 1) + (v + 1) * m, w + u + (v + 1) * m]; a.push(k[1]), a.push(k[2]), a.push(k[3]), a.push(k[0]), a.push(k[1]), a.push(k[3]); } const I = S.Zero(), N = new S(0, 1, 0); for (v = 0; v <= s.h; v++) for (I.z = v * (j - q) / s.h + q, u = 0; u <= s.w; u++) I.x = u * (b - T) / s.w + T, I.y = 0, f.push(I.x, I.y, I.z), o.push(N.x, N.y, N.z), d.push(u / s.w, v / s.h); } for (l = 0; l < i.h; l++) for (P = 0; P < i.w; P++) c(e + P * p.w, t + l * p.h, e + (P + 1) * p.w, t + (l + 1) * p.h); const H = new Ut(); return H.indices = a, H.positions = f, H.normals = o, H.uvs = d, H; } function xQ(A) { const e = [], t = [], r = [], n = []; let i, s; const a = A.colorFilter || new Ne(0.3, 0.59, 0.11), f = A.alphaFilter || 0; let o = !1; if (A.minHeight > A.maxHeight) { o = !0; const v = A.maxHeight; A.maxHeight = A.minHeight, A.minHeight = v; } for (i = 0; i <= A.subdivisions; i++) for (s = 0; s <= A.subdivisions; s++) { const v = new S(s * A.width / A.subdivisions - A.width / 2, 0, (A.subdivisions - i) * A.height / A.subdivisions - A.height / 2), u = (v.x + A.width / 2) / A.width * (A.bufferWidth - 1) | 0, l = (1 - (v.z + A.height / 2) / A.height) * (A.bufferHeight - 1) | 0, P = (u + l * A.bufferWidth) * 4; let p = A.buffer[P] / 255, c = A.buffer[P + 1] / 255, H = A.buffer[P + 2] / 255; const T = A.buffer[P + 3] / 255; o && (p = 1 - p, c = 1 - c, H = 1 - H); const q = p * a.r + c * a.g + H * a.b; T >= f ? v.y = A.minHeight + (A.maxHeight - A.minHeight) * q : v.y = A.minHeight - Dn, t.push(v.x, v.y, v.z), r.push(0, 0, 0), n.push(s / A.subdivisions, 1 - i / A.subdivisions); } for (i = 0; i < A.subdivisions; i++) for (s = 0; s < A.subdivisions; s++) { const v = s + 1 + (i + 1) * (A.subdivisions + 1), u = s + 1 + i * (A.subdivisions + 1), l = s + i * (A.subdivisions + 1), P = s + (i + 1) * (A.subdivisions + 1), p = t[v * 3 + 1] >= A.minHeight, c = t[u * 3 + 1] >= A.minHeight, H = t[l * 3 + 1] >= A.minHeight; p && c && H && (e.push(v), e.push(u), e.push(l)), t[P * 3 + 1] >= A.minHeight && p && H && (e.push(P), e.push(v), e.push(l)); } Ut.ComputeNormals(t, e, r); const d = new Ut(); return d.indices = e, d.positions = t, d.normals = r, d.uvs = n, d; } function Wm(A, e = {}, t) { const r = new dU(A, t); return r._setReady(!1), r._subdivisionsX = e.subdivisionsX || e.subdivisions || 1, r._subdivisionsY = e.subdivisionsY || e.subdivisions || 1, r._width = e.width || 1, r._height = e.height || 1, r._maxX = r._width / 2, r._maxZ = r._height / 2, r._minX = -r._maxX, r._minZ = -r._maxZ, y2(e).applyToMesh(r, e.updatable), r._setReady(!0), r; } function FO(A, e, t = null) { const r = new Ee(A, t); return bQ(e).applyToMesh(r, e.updatable), r; } function NO(A, e, t = {}, r = null) { const n = t.width || 10, i = t.height || 10, s = t.subdivisions || 1, a = t.minHeight || 0, f = t.maxHeight || 1, o = t.colorFilter || new Ne(0.3, 0.59, 0.11), d = t.alphaFilter || 0, v = t.updatable, u = t.onReady; r = r || gr.LastCreatedScene; const l = new dU(A, r); l._subdivisionsX = s, l._subdivisionsY = s, l._width = n, l._height = i, l._maxX = l._width / 2, l._maxZ = l._height / 2, l._minX = -l._maxX, l._minZ = -l._maxZ, l._setReady(!1); const P = (p) => { const c = p.width, H = p.height; if (r.isDisposed) return; const T = r == null ? void 0 : r.getEngine().resizeImageBitmap(p, c, H); xQ({ width: n, height: i, subdivisions: s, minHeight: a, maxHeight: f, colorFilter: o, buffer: T, bufferWidth: c, bufferHeight: H, alphaFilter: d }).applyToMesh(l, v), u && u(l), l._setReady(!0); }; return ye.LoadImage(e, P, () => { }, r.offlineProvider), l; } const xve = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateGround: Wm, // eslint-disable-next-line @typescript-eslint/naming-convention CreateGroundFromHeightMap: NO, // eslint-disable-next-line @typescript-eslint/naming-convention CreateTiledGround: FO }; Ut.CreateGround = y2; Ut.CreateTiledGround = bQ; Ut.CreateGroundFromHeightMap = xQ; Ee.CreateGround = (A, e, t, r, n, i) => Wm(A, { width: e, height: t, subdivisions: r, updatable: i }, n); Ee.CreateTiledGround = (A, e, t, r, n, i, s, a, f) => FO(A, { xmin: e, zmin: t, xmax: r, zmax: n, subdivisions: i, precision: s, updatable: f }, a); Ee.CreateGroundFromHeightMap = (A, e, t, r, n, i, s, a, f, o, d) => NO(A, e, { width: t, height: r, subdivisions: n, minHeight: i, maxHeight: s, updatable: f, onReady: o, alphaFilter: d }, a); function QO(A) { const e = [], t = [], r = [], n = [], i = A.diameter || 1, s = A.thickness || 0.5, a = (A.tessellation || 16) | 0, f = A.sideOrientation === 0 ? 0 : A.sideOrientation || Ut.DEFAULTSIDE, o = a + 1; for (let v = 0; v <= a; v++) { const u = v / a, l = v * Math.PI * 2 / a - Math.PI / 2, P = he.Translation(i / 2, 0, 0).multiply(he.RotationY(l)); for (let p = 0; p <= a; p++) { const c = 1 - p / a, H = p * Math.PI * 2 / a + Math.PI, T = Math.cos(H), q = Math.sin(H); let b = new S(T, q, 0), j = b.scale(s / 2); const w = new at(u, c); j = S.TransformCoordinates(j, P), b = S.TransformNormal(b, P), t.push(j.x, j.y, j.z), r.push(b.x, b.y, b.z), n.push(w.x, us.UseOpenGLOrientationForUV ? 1 - w.y : w.y); const m = (v + 1) % o, I = (p + 1) % o; e.push(v * o + p), e.push(v * o + I), e.push(m * o + p), e.push(v * o + I), e.push(m * o + I), e.push(m * o + p); } } Ut._ComputeSides(f, t, e, r, n, A.frontUVs, A.backUVs); const d = new Ut(); return d.indices = e, d.positions = t, d.normals = r, d.uvs = n, d; } function Ag(A, e = {}, t) { const r = new Ee(A, t); return e.sideOrientation = Ee._GetDefaultSideOrientation(e.sideOrientation), r._originalBuilderSideOrientation = e.sideOrientation, QO(e).applyToMesh(r, e.updatable), r; } const Dve = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateTorus: Ag }; Ut.CreateTorus = QO; Ee.CreateTorus = (A, e, t, r, n, i, s) => Ag(A, { diameter: e, thickness: t, tessellation: r, sideOrientation: s, updatable: i }, n); class YO { constructor(e, t = null) { if (this.scene = e, this._pointerDownOnMeshAsked = !1, this._isActionableMesh = !1, this._teleportationRequestInitiated = !1, this._teleportationBackRequestInitiated = !1, this._rotationRightAsked = !1, this._rotationLeftAsked = !1, this._dpadPressed = !0, this._activePointer = !1, this._id = YO._IdCounter++, t) this._gazeTracker = t.clone("gazeTracker"); else { this._gazeTracker = Ag("gazeTracker", { diameter: 35e-4, thickness: 25e-4, tessellation: 20, updatable: !1 }, e), this._gazeTracker.bakeCurrentTransformIntoVertices(), this._gazeTracker.isPickable = !1, this._gazeTracker.isVisible = !1; const r = new Wt("targetMat", e); r.specularColor = Ne.Black(), r.emissiveColor = new Ne(0.7, 0.7, 0.7), r.backFaceCulling = !1, this._gazeTracker.material = r; } } /** * @internal */ _getForwardRay(e) { return new Hi(S.Zero(), new S(0, 0, e)); } /** @internal */ _selectionPointerDown() { this._pointerDownOnMeshAsked = !0, this._currentHit && this.scene.simulatePointerDown(this._currentHit, { pointerId: this._id }); } /** @internal */ _selectionPointerUp() { this._currentHit && this.scene.simulatePointerUp(this._currentHit, { pointerId: this._id }), this._pointerDownOnMeshAsked = !1; } /** @internal */ _activatePointer() { this._activePointer = !0; } /** @internal */ _deactivatePointer() { this._activePointer = !1; } /** * @internal */ // eslint-disable-next-line @typescript-eslint/no-unused-vars _updatePointerDistance(e = 100) { } dispose() { this._interactionsEnabled = !1, this._teleportationEnabled = !1, this._gazeTracker && this._gazeTracker.dispose(); } } YO._IdCounter = 0; class pG extends YO { constructor(e, t) { super(t), this._getCamera = e; } _getForwardRay(e) { const t = this._getCamera(); return t ? t.getForwardRay(e) : new Hi(S.Zero(), S.Forward()); } } class jve { } class lm { /** Return this.onEnteringVRObservable * Note: This one is for backward compatibility. Please use onEnteringVRObservable directly */ get onEnteringVR() { return this.onEnteringVRObservable; } /** Return this.onExitingVRObservable * Note: This one is for backward compatibility. Please use onExitingVRObservable directly */ get onExitingVR() { return this.onExitingVRObservable; } /** * The mesh used to display where the user is going to teleport. */ get teleportationTarget() { return this._teleportationTarget; } /** * Sets the mesh to be used to display where the user is going to teleport. */ set teleportationTarget(e) { e && (e.name = "teleportationTarget", this._isDefaultTeleportationTarget = !1, this._teleportationTarget = e); } /** * The mesh used to display where the user is selecting, this mesh will be cloned and set as the gazeTracker for the left and right controller * when set bakeCurrentTransformIntoVertices will be called on the mesh. * See https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/bakingTransforms */ get gazeTrackerMesh() { return this._cameraGazer._gazeTracker; } set gazeTrackerMesh(e) { e && (this._cameraGazer._gazeTracker && this._cameraGazer._gazeTracker.dispose(), this._cameraGazer._gazeTracker = e, this._cameraGazer._gazeTracker.bakeCurrentTransformIntoVertices(), this._cameraGazer._gazeTracker.isPickable = !1, this._cameraGazer._gazeTracker.isVisible = !1, this._cameraGazer._gazeTracker.name = "gazeTracker"); } /** * If the ray of the gaze should be displayed. */ get displayGaze() { return this._displayGaze; } /** * Sets if the ray of the gaze should be displayed. */ set displayGaze(e) { this._displayGaze = e, e || (this._cameraGazer._gazeTracker.isVisible = !1); } /** * If the ray of the LaserPointer should be displayed. */ get displayLaserPointer() { return this._displayLaserPointer; } /** * Sets if the ray of the LaserPointer should be displayed. */ set displayLaserPointer(e) { this._displayLaserPointer = e; } /** * The deviceOrientationCamera used as the camera when not in VR. */ get deviceOrientationCamera() { return this._deviceOrientationCamera; } /** * Based on the current WebVR support, returns the current VR camera used. */ get currentVRCamera() { return this._scene.activeCamera; } /** * The deviceOrientationCamera that is used as a fallback when vr device is not connected. */ get vrDeviceOrientationCamera() { return this._vrDeviceOrientationCamera; } /** * The html button that is used to trigger entering into VR. */ get vrButton() { return this._btnVR; } get _teleportationRequestInitiated() { return this._cameraGazer._teleportationRequestInitiated; } /** * Instantiates a VRExperienceHelper. * Helps to quickly add VR support to an existing scene. * @param scene The scene the VRExperienceHelper belongs to. * @param webVROptions Options to modify the vr experience helper's behavior. */ constructor(e, t = {}) { if (this.webVROptions = t, this._fullscreenVRpresenting = !1, this.enableGazeEvenWhenNoPointerLock = !1, this.exitVROnDoubleTap = !0, this.onEnteringVRObservable = new Oe(), this.onAfterEnteringVRObservable = new Oe(), this.onExitingVRObservable = new Oe(), this._useCustomVRButton = !1, this._teleportActive = !1, this._floorMeshesCollection = [], this._teleportationMode = lm.TELEPORTATIONMODE_CONSTANTTIME, this._teleportationTime = 122, this._teleportationSpeed = 20, this._rotationAllowed = !0, this._teleportBackwardsVector = new S(0, -1, -1), this._isDefaultTeleportationTarget = !0, this._teleportationFillColor = "#444444", this._teleportationBorderColor = "#FFFFFF", this._rotationAngle = 0, this._haloCenter = new S(0, 0, 0), this._padSensibilityUp = 0.65, this._padSensibilityDown = 0.35, this._pickedLaserColor = new Ne(0.2, 0.2, 1), this._pickedGazeColor = new Ne(0, 0, 1), this.onNewMeshSelected = new Oe(), this.onNewMeshPicked = new Oe(), this.onBeforeCameraTeleport = new Oe(), this.onAfterCameraTeleport = new Oe(), this.onSelectedMeshUnselected = new Oe(), this.teleportationEnabled = !0, this._teleportationInitialized = !1, this._interactionsEnabled = !1, this._displayGaze = !0, this._displayLaserPointer = !0, this.updateGazeTrackerScale = !0, this.updateGazeTrackerColor = !0, this.updateControllerLaserColor = !0, this.requestPointerLockOnFullScreen = !0, this.xrTestDone = !1, this._onResize = () => { this._moveButtonToBottomRight(); }, this._onFullscreenChange = () => { this._fullscreenVRpresenting = !!document.fullscreenElement, !this._fullscreenVRpresenting && this._inputElement && (this.exitVR(), !this._useCustomVRButton && this._btnVR && (this._btnVR.style.top = this._inputElement.offsetTop + this._inputElement.offsetHeight - 70 + "px", this._btnVR.style.left = this._inputElement.offsetLeft + this._inputElement.offsetWidth - 100 + "px", this._updateButtonVisibility())); }, this._cachedAngularSensibility = { angularSensibilityX: null, angularSensibilityY: null, angularSensibility: null }, this._beforeRender = () => { this._scene.getEngine().isPointerLock || this.enableGazeEvenWhenNoPointerLock || (this._cameraGazer._gazeTracker.isVisible = !1); }, this._onNewGamepadConnected = (n) => { n.type !== N9.POSE_ENABLED && (n.leftStick && n.onleftstickchanged((i) => { this._teleportationInitialized && this.teleportationEnabled && (this._checkTeleportWithRay(i, this._cameraGazer), this._checkTeleportBackwards(i, this._cameraGazer)); }), n.rightStick && n.onrightstickchanged((i) => { this._teleportationInitialized && this._checkRotate(i, this._cameraGazer); }), n.type === N9.XBOX && (n.onbuttondown((i) => { this._interactionsEnabled && i === El.A && this._cameraGazer._selectionPointerDown(); }), n.onbuttonup((i) => { this._interactionsEnabled && i === El.A && this._cameraGazer._selectionPointerUp(); }))); }, this._workingVector = S.Zero(), this._workingQuaternion = Ze.Identity(), this._workingMatrix = he.Identity(), Se.Warn("WebVR is deprecated. Please avoid using this experience helper and use the WebXR experience helper instead"), this._scene = e, this._inputElement = e.getEngine().getInputElement(), !("getVRDisplays" in navigator) && t.useXR === void 0 && (t.useXR = !0), t.createFallbackVRDeviceOrientationFreeCamera === void 0 && (t.createFallbackVRDeviceOrientationFreeCamera = !0), t.createDeviceOrientationCamera === void 0 && (t.createDeviceOrientationCamera = !0), t.laserToggle === void 0 && (t.laserToggle = !0), this._hasEnteredVR = !1, this._scene.activeCamera ? this._position = this._scene.activeCamera.position.clone() : this._position = new S(0, this._defaultHeight, 0), t.createDeviceOrientationCamera || !this._scene.activeCamera) { if (this._deviceOrientationCamera = new OO("deviceOrientationVRHelper", this._position.clone(), e), this._scene.activeCamera && (this._deviceOrientationCamera.minZ = this._scene.activeCamera.minZ, this._deviceOrientationCamera.maxZ = this._scene.activeCamera.maxZ, this._scene.activeCamera instanceof b1 && this._scene.activeCamera.rotation)) { const n = this._scene.activeCamera; n.rotationQuaternion ? this._deviceOrientationCamera.rotationQuaternion.copyFrom(n.rotationQuaternion) : this._deviceOrientationCamera.rotationQuaternion.copyFrom(Ze.RotationYawPitchRoll(n.rotation.y, n.rotation.x, n.rotation.z)), this._deviceOrientationCamera.rotation = n.rotation.clone(); } this._scene.activeCamera = this._deviceOrientationCamera, this._inputElement && this._scene.activeCamera.attachControl(); } else this._existingCamera = this._scene.activeCamera; this.webVROptions.useXR && navigator.xr ? MR.IsSessionSupportedAsync("immersive-vr").then((n) => { n ? (Se.Log("Using WebXR. It is recommended to use the WebXRDefaultExperience directly"), e.createDefaultXRExperienceAsync({ floorMeshes: t.floorMeshes || [] }).then((i) => { this.xr = i, this.xrTestDone = !0, this._cameraGazer = new pG(() => this.xr.baseExperience.camera, e), this.xr.baseExperience.onStateChangedObservable.add((s) => { switch (s) { case d9.ENTERING_XR: this.onEnteringVRObservable.notifyObservers(this), this._interactionsEnabled || this.xr.pointerSelection.detach(), this.xr.pointerSelection.displayLaserPointer = this._displayLaserPointer; break; case d9.EXITING_XR: this.onExitingVRObservable.notifyObservers(this), this._scene.getEngine().resize(); break; case d9.IN_XR: this._hasEnteredVR = !0; break; case d9.NOT_IN_XR: this._hasEnteredVR = !1; break; } }); })) : this._completeVRInit(e, t); }) : this._completeVRInit(e, t); } _completeVRInit(e, t) { if (this.xrTestDone = !0, t.createFallbackVRDeviceOrientationFreeCamera && (this._vrDeviceOrientationCamera = new EO("VRDeviceOrientationVRHelper", this._position, this._scene, !0, t.vrDeviceOrientationCameraMetrics), this._vrDeviceOrientationCamera.angularSensibility = Number.MAX_VALUE), this._cameraGazer = new pG(() => this.currentVRCamera, e), !this._useCustomVRButton) { this._btnVR = document.createElement("BUTTON"), this._btnVR.className = "babylonVRicon", this._btnVR.id = "babylonVRiconbtn", this._btnVR.title = "Click to switch to VR"; let i = ".babylonVRicon { position: absolute; right: 20px; height: 50px; width: 80px; background-color: rgba(51,51,51,0.7); background-image: url(" + (window.SVGSVGElement ? "data:image/svg+xml;charset=UTF-8,%3Csvg%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%20width%3D%222048%22%20height%3D%221152%22%20viewBox%3D%220%200%202048%201152%22%20version%3D%221.1%22%3E%3Cpath%20transform%3D%22rotate%28180%201024%2C576.0000000000001%29%22%20d%3D%22m1109%2C896q17%2C0%2030%2C-12t13%2C-30t-12.5%2C-30.5t-30.5%2C-12.5l-170%2C0q-18%2C0%20-30.5%2C12.5t-12.5%2C30.5t13%2C30t30%2C12l170%2C0zm-85%2C256q59%2C0%20132.5%2C-1.5t154.5%2C-5.5t164.5%2C-11.5t163%2C-20t150%2C-30t124.5%2C-41.5q23%2C-11%2042%2C-24t38%2C-30q27%2C-25%2041%2C-61.5t14%2C-72.5l0%2C-257q0%2C-123%20-47%2C-232t-128%2C-190t-190%2C-128t-232%2C-47l-81%2C0q-37%2C0%20-68.5%2C14t-60.5%2C34.5t-55.5%2C45t-53%2C45t-53%2C34.5t-55.5%2C14t-55.5%2C-14t-53%2C-34.5t-53%2C-45t-55.5%2C-45t-60.5%2C-34.5t-68.5%2C-14l-81%2C0q-123%2C0%20-232%2C47t-190%2C128t-128%2C190t-47%2C232l0%2C257q0%2C68%2038%2C115t97%2C73q54%2C24%20124.5%2C41.5t150%2C30t163%2C20t164.5%2C11.5t154.5%2C5.5t132.5%2C1.5zm939%2C-298q0%2C39%20-24.5%2C67t-58.5%2C42q-54%2C23%20-122%2C39.5t-143.5%2C28t-155.5%2C19t-157%2C11t-148.5%2C5t-129.5%2C1.5q-59%2C0%20-130%2C-1.5t-148%2C-5t-157%2C-11t-155.5%2C-19t-143.5%2C-28t-122%2C-39.5q-34%2C-14%20-58.5%2C-42t-24.5%2C-67l0%2C-257q0%2C-106%2040.5%2C-199t110%2C-162.5t162.5%2C-109.5t199%2C-40l81%2C0q27%2C0%2052%2C14t50%2C34.5t51%2C44.5t55.5%2C44.5t63.5%2C34.5t74%2C14t74%2C-14t63.5%2C-34.5t55.5%2C-44.5t51%2C-44.5t50%2C-34.5t52%2C-14l14%2C0q37%2C0%2070%2C0.5t64.5%2C4.5t63.5%2C12t68%2C23q71%2C30%20128.5%2C78.5t98.5%2C110t63.5%2C133.5t22.5%2C149l0%2C257z%22%20fill%3D%22white%22%20/%3E%3C/svg%3E%0A" : "https://cdn.babylonjs.com/Assets/vrButton.png") + "); background-size: 80%; background-repeat:no-repeat; background-position: center; border: none; outline: none; transition: transform 0.125s ease-out } .babylonVRicon:hover { transform: scale(1.05) } .babylonVRicon:active {background-color: rgba(51,51,51,1) } .babylonVRicon:focus {background-color: rgba(51,51,51,1) }"; i += ".babylonVRicon.vrdisplaypresenting { display: none; }"; const s = document.createElement("style"); s.appendChild(document.createTextNode(i)), document.getElementsByTagName("head")[0].appendChild(s), this._moveButtonToBottomRight(); } this._btnVR && this._btnVR.addEventListener("click", () => { this.isInVRMode || this.enterVR(); }); const r = this._scene.getEngine().getHostWindow(); r && (r.addEventListener("resize", this._onResize), document.addEventListener("fullscreenchange", this._onFullscreenChange, !1), t.createFallbackVRDeviceOrientationFreeCamera && this._displayVRButton(), this._onKeyDown = (n) => { n.keyCode === 27 && this.isInVRMode && this.exitVR(); }, document.addEventListener("keydown", this._onKeyDown), this._scene.onPrePointerObservable.add(() => { this._hasEnteredVR && this.exitVROnDoubleTap && (this.exitVR(), this._fullscreenVRpresenting && this._scene.getEngine().exitFullscreen()); }, ir.POINTERDOUBLETAP, !1), e.onDisposeObservable.add(() => { this.dispose(); }), this._updateButtonVisibility(), this._circleEase = new Z$(), this._circleEase.setEasingMode(u1.EASINGMODE_EASEINOUT), this._teleportationEasing = this._circleEase, e.onPointerObservable.add((n) => { this._interactionsEnabled && e.activeCamera === this.vrDeviceOrientationCamera && n.event.pointerType === "mouse" && (n.type === ir.POINTERDOWN ? this._cameraGazer._selectionPointerDown() : n.type === ir.POINTERUP && this._cameraGazer._selectionPointerUp()); }), this.webVROptions.floorMeshes && this.enableTeleportation({ floorMeshes: this.webVROptions.floorMeshes })); } /** * Gets a value indicating if we are currently in VR mode. */ get isInVRMode() { return this.xr && this.webVROptions.useXR && this.xr.baseExperience.state === d9.IN_XR || this._fullscreenVRpresenting; } _moveButtonToBottomRight() { if (this._inputElement && !this._useCustomVRButton && this._btnVR) { const e = this._inputElement.getBoundingClientRect(); this._btnVR.style.top = e.top + e.height - 70 + "px", this._btnVR.style.left = e.left + e.width - 100 + "px"; } } _displayVRButton() { !this._useCustomVRButton && !this._btnVRDisplayed && this._btnVR && (document.body.appendChild(this._btnVR), this._btnVRDisplayed = !0); } _updateButtonVisibility() { !this._btnVR || this._useCustomVRButton || (this._btnVR.className = "babylonVRicon", this.isInVRMode && (this._btnVR.className += " vrdisplaypresenting")); } /** * Attempt to enter VR. If a headset is connected and ready, will request present on that. * Otherwise, will use the fullscreen API. */ enterVR() { if (this.xr) { this.xr.baseExperience.enterXRAsync("immersive-vr", "local-floor", this.xr.renderTarget); return; } if (this.onEnteringVRObservable) try { this.onEnteringVRObservable.notifyObservers(this); } catch (e) { Se.Warn("Error in your custom logic onEnteringVR: " + e); } this._scene.activeCamera && (this._position = this._scene.activeCamera.position.clone(), this.vrDeviceOrientationCamera && (this.vrDeviceOrientationCamera.rotation = Ze.FromRotationMatrix(this._scene.activeCamera.getWorldMatrix().getRotationMatrix()).toEulerAngles(), this.vrDeviceOrientationCamera.angularSensibility = 2e3), this._existingCamera = this._scene.activeCamera, this._existingCamera.angularSensibilityX && (this._cachedAngularSensibility.angularSensibilityX = this._existingCamera.angularSensibilityX, this._existingCamera.angularSensibilityX = Number.MAX_VALUE), this._existingCamera.angularSensibilityY && (this._cachedAngularSensibility.angularSensibilityY = this._existingCamera.angularSensibilityY, this._existingCamera.angularSensibilityY = Number.MAX_VALUE), this._existingCamera.angularSensibility && (this._cachedAngularSensibility.angularSensibility = this._existingCamera.angularSensibility, this._existingCamera.angularSensibility = Number.MAX_VALUE)), this._vrDeviceOrientationCamera && (this._vrDeviceOrientationCamera.position = this._position, this._scene.activeCamera && (this._vrDeviceOrientationCamera.minZ = this._scene.activeCamera.minZ), this._scene.activeCamera = this._vrDeviceOrientationCamera, this._scene.getEngine().enterFullscreen(this.requestPointerLockOnFullScreen), this._updateButtonVisibility(), this._vrDeviceOrientationCamera.onViewMatrixChangedObservable.addOnce(() => { this.onAfterEnteringVRObservable.notifyObservers({ success: !0 }); })), this._scene.activeCamera && this._inputElement && this._scene.activeCamera.attachControl(), this._interactionsEnabled && this._scene.registerBeforeRender(this._beforeRender), this._hasEnteredVR = !0; } /** * Attempt to exit VR, or fullscreen. */ exitVR() { if (this.xr) { this.xr.baseExperience.exitXRAsync(); return; } if (this._hasEnteredVR) { if (this.onExitingVRObservable) try { this.onExitingVRObservable.notifyObservers(this); } catch (e) { Se.Warn("Error in your custom logic onExitingVR: " + e); } this._scene.activeCamera && (this._position = this._scene.activeCamera.position.clone()), this.vrDeviceOrientationCamera && (this.vrDeviceOrientationCamera.angularSensibility = Number.MAX_VALUE), this._deviceOrientationCamera ? (this._deviceOrientationCamera.position = this._position, this._scene.activeCamera = this._deviceOrientationCamera, this._cachedAngularSensibility.angularSensibilityX && (this._deviceOrientationCamera.angularSensibilityX = this._cachedAngularSensibility.angularSensibilityX, this._cachedAngularSensibility.angularSensibilityX = null), this._cachedAngularSensibility.angularSensibilityY && (this._deviceOrientationCamera.angularSensibilityY = this._cachedAngularSensibility.angularSensibilityY, this._cachedAngularSensibility.angularSensibilityY = null), this._cachedAngularSensibility.angularSensibility && (this._deviceOrientationCamera.angularSensibility = this._cachedAngularSensibility.angularSensibility, this._cachedAngularSensibility.angularSensibility = null)) : this._existingCamera && (this._existingCamera.position = this._position, this._scene.activeCamera = this._existingCamera, this._inputElement && this._scene.activeCamera.attachControl(), this._cachedAngularSensibility.angularSensibilityX && (this._existingCamera.angularSensibilityX = this._cachedAngularSensibility.angularSensibilityX, this._cachedAngularSensibility.angularSensibilityX = null), this._cachedAngularSensibility.angularSensibilityY && (this._existingCamera.angularSensibilityY = this._cachedAngularSensibility.angularSensibilityY, this._cachedAngularSensibility.angularSensibilityY = null), this._cachedAngularSensibility.angularSensibility && (this._existingCamera.angularSensibility = this._cachedAngularSensibility.angularSensibility, this._cachedAngularSensibility.angularSensibility = null)), this._updateButtonVisibility(), this._interactionsEnabled && (this._scene.unregisterBeforeRender(this._beforeRender), this._cameraGazer._gazeTracker.isVisible = !1), this._scene.getEngine().resize(), this._hasEnteredVR = !1; } } /** * The position of the vr experience helper. */ get position() { return this._position; } /** * Sets the position of the vr experience helper. */ set position(e) { this._position = e, this._scene.activeCamera && (this._scene.activeCamera.position = e); } /** * Enables controllers and user interactions such as selecting and object or clicking on an object. */ enableInteractions() { if (!this._interactionsEnabled) { if (this.xr) { this.xr.baseExperience.state === d9.IN_XR && this.xr.pointerSelection.attach(); return; } this.raySelectionPredicate = (e) => e.isVisible && (e.isPickable || e.name === this._floorMeshName), this.meshSelectionPredicate = () => !0, this._raySelectionPredicate = (e) => this._isTeleportationFloor(e) || e.name.indexOf("gazeTracker") === -1 && e.name.indexOf("teleportationTarget") === -1 && e.name.indexOf("torusTeleportation") === -1 ? this.raySelectionPredicate(e) : !1, this._interactionsEnabled = !0; } } _isTeleportationFloor(e) { for (let t = 0; t < this._floorMeshesCollection.length; t++) if (this._floorMeshesCollection[t].id === e.id) return !0; return !!(this._floorMeshName && e.name === this._floorMeshName); } /** * Adds a floor mesh to be used for teleportation. * @param floorMesh the mesh to be used for teleportation. */ addFloorMesh(e) { this._floorMeshesCollection && (this._floorMeshesCollection.indexOf(e) > -1 || this._floorMeshesCollection.push(e)); } /** * Removes a floor mesh from being used for teleportation. * @param floorMesh the mesh to be removed. */ removeFloorMesh(e) { if (!this._floorMeshesCollection) return; const t = this._floorMeshesCollection.indexOf(e); t !== -1 && this._floorMeshesCollection.splice(t, 1); } /** * Enables interactions and teleportation using the VR controllers and gaze. * @param vrTeleportationOptions options to modify teleportation behavior. */ enableTeleportation(e = {}) { if (!this._teleportationInitialized) { if (this.enableInteractions(), this.webVROptions.useXR && (e.floorMeshes || e.floorMeshName)) { const r = e.floorMeshes || []; if (!r.length) { const n = this._scene.getMeshByName(e.floorMeshName); n && r.push(n); } if (this.xr) { r.forEach((n) => { this.xr.teleportation.addFloorMesh(n); }), this.xr.teleportation.attached || this.xr.teleportation.attach(); return; } else if (!this.xrTestDone) { const n = () => { this.xrTestDone && (this._scene.unregisterBeforeRender(n), this.xr ? this.xr.teleportation.attached || this.xr.teleportation.attach() : this.enableTeleportation(e)); }; this._scene.registerBeforeRender(n); return; } } e.floorMeshName && (this._floorMeshName = e.floorMeshName), e.floorMeshes && (this._floorMeshesCollection = e.floorMeshes), e.teleportationMode && (this._teleportationMode = e.teleportationMode), e.teleportationTime && e.teleportationTime > 0 && (this._teleportationTime = e.teleportationTime), e.teleportationSpeed && e.teleportationSpeed > 0 && (this._teleportationSpeed = e.teleportationSpeed), e.easingFunction !== void 0 && (this._teleportationEasing = e.easingFunction); const t = new Ui(); t.vignetteColor = new xt(0, 0, 0, 0), t.vignetteEnabled = !0, this._teleportationInitialized = !0, this._isDefaultTeleportationTarget && this._createTeleportationCircles(); } } _checkTeleportWithRay(e, t) { this._teleportationRequestInitiated && !t._teleportationRequestInitiated || (t._teleportationRequestInitiated ? Math.sqrt(e.y * e.y + e.x * e.x) < this._padSensibilityDown && (this._teleportActive && this.teleportCamera(this._haloCenter), t._teleportationRequestInitiated = !1) : e.y < -this._padSensibilityUp && t._dpadPressed && (t._activatePointer(), t._teleportationRequestInitiated = !0)); } _checkRotate(e, t) { t._teleportationRequestInitiated || (t._rotationLeftAsked ? e.x > -this._padSensibilityDown && (t._rotationLeftAsked = !1) : e.x < -this._padSensibilityUp && t._dpadPressed && (t._rotationLeftAsked = !0, this._rotationAllowed && this._rotateCamera(!1)), t._rotationRightAsked ? e.x < this._padSensibilityDown && (t._rotationRightAsked = !1) : e.x > this._padSensibilityUp && t._dpadPressed && (t._rotationRightAsked = !0, this._rotationAllowed && this._rotateCamera(!0))); } _checkTeleportBackwards(e, t) { if (!t._teleportationRequestInitiated) if (e.y > this._padSensibilityUp && t._dpadPressed) { if (!t._teleportationBackRequestInitiated) { if (!this.currentVRCamera) return; const r = Ze.FromRotationMatrix(this.currentVRCamera.getWorldMatrix().getRotationMatrix()), n = this.currentVRCamera.position; r.toEulerAnglesToRef(this._workingVector), this._workingVector.z = 0, this._workingVector.x = 0, Ze.RotationYawPitchRollToRef(this._workingVector.y, this._workingVector.x, this._workingVector.z, this._workingQuaternion), this._workingQuaternion.toRotationMatrix(this._workingMatrix), S.TransformCoordinatesToRef(this._teleportBackwardsVector, this._workingMatrix, this._workingVector); const i = new Hi(n, this._workingVector), s = this._scene.pickWithRay(i, this._raySelectionPredicate); s && s.pickedPoint && s.pickedMesh && this._isTeleportationFloor(s.pickedMesh) && s.distance < 5 && this.teleportCamera(s.pickedPoint), t._teleportationBackRequestInitiated = !0; } } else t._teleportationBackRequestInitiated = !1; } _createTeleportationCircles() { this._teleportationTarget = Wm("teleportationTarget", { width: 2, height: 2, subdivisions: 2 }, this._scene), this._teleportationTarget.isPickable = !1; const e = 512, t = new Xp("DynamicTexture", e, this._scene, !0); t.hasAlpha = !0; const r = t.getContext(), n = e / 2, i = e / 2, s = 200; r.beginPath(), r.arc(n, i, s, 0, 2 * Math.PI, !1), r.fillStyle = this._teleportationFillColor, r.fill(), r.lineWidth = 10, r.strokeStyle = this._teleportationBorderColor, r.stroke(), r.closePath(), t.update(); const a = new Wt("TextPlaneMaterial", this._scene); a.diffuseTexture = t, this._teleportationTarget.material = a; const f = Ag("torusTeleportation", { diameter: 0.75, thickness: 0.1, tessellation: 25, updatable: !1 }, this._scene); f.isPickable = !1, f.parent = this._teleportationTarget; const o = new st("animationInnerCircle", "position.y", 30, st.ANIMATIONTYPE_FLOAT, st.ANIMATIONLOOPMODE_CYCLE), d = []; d.push({ frame: 0, value: 0 }), d.push({ frame: 30, value: 0.4 }), d.push({ frame: 60, value: 0 }), o.setKeys(d); const v = new $N(); v.setEasingMode(u1.EASINGMODE_EASEINOUT), o.setEasingFunction(v), f.animations = [], f.animations.push(o), this._scene.beginAnimation(f, 0, 60, !0), this._hideTeleportationTarget(); } _hideTeleportationTarget() { this._teleportActive = !1, this._teleportationInitialized && (this._teleportationTarget.isVisible = !1, this._isDefaultTeleportationTarget && (this._teleportationTarget.getChildren()[0].isVisible = !1)); } _rotateCamera(e) { if (!(this.currentVRCamera instanceof SA)) return; e ? this._rotationAngle++ : this._rotationAngle--, this.currentVRCamera.animations = []; const t = Ze.FromRotationMatrix(he.RotationY(Math.PI / 4 * this._rotationAngle)), r = new st("animationRotation", "rotationQuaternion", 90, st.ANIMATIONTYPE_QUATERNION, st.ANIMATIONLOOPMODE_CONSTANT), n = []; n.push({ frame: 0, value: this.currentVRCamera.rotationQuaternion }), n.push({ frame: 6, value: t }), r.setKeys(n), r.setEasingFunction(this._circleEase), this.currentVRCamera.animations.push(r), this._postProcessMove.animations = []; const i = new st("animationPP", "vignetteWeight", 90, st.ANIMATIONTYPE_FLOAT, st.ANIMATIONLOOPMODE_CONSTANT), s = []; s.push({ frame: 0, value: 0 }), s.push({ frame: 3, value: 4 }), s.push({ frame: 6, value: 0 }), i.setKeys(s), i.setEasingFunction(this._circleEase), this._postProcessMove.animations.push(i); const a = new st("animationPP2", "vignetteStretch", 90, st.ANIMATIONTYPE_FLOAT, st.ANIMATIONLOOPMODE_CONSTANT), f = []; f.push({ frame: 0, value: 0 }), f.push({ frame: 3, value: 10 }), f.push({ frame: 6, value: 0 }), a.setKeys(f), a.setEasingFunction(this._circleEase), this._postProcessMove.animations.push(a), this._postProcessMove.imageProcessingConfiguration.vignetteWeight = 0, this._postProcessMove.imageProcessingConfiguration.vignetteStretch = 0, this._postProcessMove.samples = 4, this._scene.beginAnimation(this.currentVRCamera, 0, 6, !1, 1); } /** * Teleports the users feet to the desired location * @param location The location where the user's feet should be placed */ teleportCamera(e) { if (!(this.currentVRCamera instanceof SA)) return; this._workingVector.copyFrom(e), this.isInVRMode || (this._workingVector.y += this._defaultHeight), this.onBeforeCameraTeleport.notifyObservers(this._workingVector); const t = 90; let r, n; if (this._teleportationMode == lm.TELEPORTATIONMODE_CONSTANTSPEED) { n = t; const u = S.Distance(this.currentVRCamera.position, this._workingVector); r = this._teleportationSpeed / u; } else n = Math.round(this._teleportationTime * t / 1e3), r = 1; this.currentVRCamera.animations = []; const i = new st("animationCameraTeleportation", "position", t, st.ANIMATIONTYPE_VECTOR3, st.ANIMATIONLOOPMODE_CONSTANT), s = [ { frame: 0, value: this.currentVRCamera.position }, { frame: n, value: this._workingVector } ]; i.setKeys(s), i.setEasingFunction(this._teleportationEasing), this.currentVRCamera.animations.push(i), this._postProcessMove.animations = []; const a = Math.round(n / 2), f = new st("animationPP", "vignetteWeight", t, st.ANIMATIONTYPE_FLOAT, st.ANIMATIONLOOPMODE_CONSTANT), o = []; o.push({ frame: 0, value: 0 }), o.push({ frame: a, value: 8 }), o.push({ frame: n, value: 0 }), f.setKeys(o), this._postProcessMove.animations.push(f); const d = new st("animationPP2", "vignetteStretch", t, st.ANIMATIONTYPE_FLOAT, st.ANIMATIONLOOPMODE_CONSTANT), v = []; v.push({ frame: 0, value: 0 }), v.push({ frame: a, value: 10 }), v.push({ frame: n, value: 0 }), d.setKeys(v), this._postProcessMove.animations.push(d), this._postProcessMove.imageProcessingConfiguration.vignetteWeight = 0, this._postProcessMove.imageProcessingConfiguration.vignetteStretch = 0, this._scene.beginAnimation(this.currentVRCamera, 0, n, !1, r, () => { this.onAfterCameraTeleport.notifyObservers(this._workingVector); }), this._hideTeleportationTarget(); } /** * Permanently set new colors for the laser pointer * @param color the new laser color * @param pickedColor the new laser color when picked mesh detected */ setLaserColor(e, t = this._pickedLaserColor) { this._pickedLaserColor = t; } /** * Set lighting enabled / disabled on the laser pointer of both controllers * @param enabled should the lighting be enabled on the laser pointer */ setLaserLightingState(e = !0) { } /** * Permanently set new colors for the gaze pointer * @param color the new gaze color * @param pickedColor the new gaze color when picked mesh detected */ setGazeColor(e, t = this._pickedGazeColor) { this._pickedGazeColor = t; } /** * Sets the color of the laser ray from the vr controllers. * @param color new color for the ray. */ changeLaserColor(e) { this.updateControllerLaserColor; } /** * Sets the color of the ray from the vr headsets gaze. * @param color new color for the ray. */ changeGazeColor(e) { this.updateGazeTrackerColor && this._cameraGazer._gazeTracker.material && (this._cameraGazer._gazeTracker.material.emissiveColor = e); } /** * Exits VR and disposes of the vr experience helper */ dispose() { this.isInVRMode && this.exitVR(), this._postProcessMove && this._postProcessMove.dispose(), this._vrDeviceOrientationCamera && this._vrDeviceOrientationCamera.dispose(), !this._useCustomVRButton && this._btnVR && this._btnVR.parentNode && document.body.removeChild(this._btnVR), this._deviceOrientationCamera && this._scene.activeCamera != this._deviceOrientationCamera && this._deviceOrientationCamera.dispose(), this._cameraGazer && this._cameraGazer.dispose(), this._teleportationTarget && this._teleportationTarget.dispose(), this.xr && this.xr.dispose(), this._floorMeshesCollection.length = 0, document.removeEventListener("keydown", this._onKeyDown), window.removeEventListener("vrdisplaypresentchange", this._onVrDisplayPresentChangeBind), window.removeEventListener("resize", this._onResize), document.removeEventListener("fullscreenchange", this._onFullscreenChange), this._scene.gamepadManager.onGamepadConnectedObservable.removeCallback(this._onNewGamepadConnected), this._scene.unregisterBeforeRender(this._beforeRender); } /** * Gets the name of the VRExperienceHelper class * @returns "VRExperienceHelper" */ getClassName() { return "VRExperienceHelper"; } } lm.TELEPORTATIONMODE_CONSTANTTIME = 0; lm.TELEPORTATIONMODE_CONSTANTSPEED = 1; const wve = (A, e, t, r) => !(A.x > t.x + r || t.x - r > e.x || A.y > t.y + r || t.y - r > e.y || A.z > t.z + r || t.z - r > e.z), BW = /* @__PURE__ */ function() { const A = { root: 0, found: !1 }; return function(e, t, r, n) { A.root = 0, A.found = !1; const i = t * t - 4 * e * r; if (i < 0) return A; const s = Math.sqrt(i); let a = (-t - s) / (2 * e), f = (-t + s) / (2 * e); if (a > f) { const o = f; f = a, a = o; } return a > 0 && a < n ? (A.root = a, A.found = !0, A) : (f > 0 && f < n && (A.root = f, A.found = !0), A); }; }(); class LR { constructor() { this._collisionPoint = S.Zero(), this._planeIntersectionPoint = S.Zero(), this._tempVector = S.Zero(), this._tempVector2 = S.Zero(), this._tempVector3 = S.Zero(), this._tempVector4 = S.Zero(), this._edge = S.Zero(), this._baseToVertex = S.Zero(), this._destinationPoint = S.Zero(), this._slidePlaneNormal = S.Zero(), this._displacementVector = S.Zero(), this._radius = S.One(), this._retry = 0, this._basePointWorld = S.Zero(), this._velocityWorld = S.Zero(), this._normalizedVelocity = S.Zero(), this._collisionMask = -1; } get collisionMask() { return this._collisionMask; } set collisionMask(e) { this._collisionMask = isNaN(e) ? -1 : e; } /** * Gets the plane normal used to compute the sliding response (in local space) */ get slidePlaneNormal() { return this._slidePlaneNormal; } // Methods /** * @internal */ _initialize(e, t, r) { this._velocity = t, this._velocitySquaredLength = this._velocity.lengthSquared(); const n = Math.sqrt(this._velocitySquaredLength); n === 0 || n === 1 ? this._normalizedVelocity.copyFromFloats(t._x, t._y, t._z) : t.scaleToRef(1 / n, this._normalizedVelocity), this._basePoint = e, e.multiplyToRef(this._radius, this._basePointWorld), t.multiplyToRef(this._radius, this._velocityWorld), this._velocityWorldLength = this._velocityWorld.length(), this._epsilon = r, this.collisionFound = !1; } /** * @internal */ _checkPointInTriangle(e, t, r, n, i) { t.subtractToRef(e, this._tempVector), r.subtractToRef(e, this._tempVector2), S.CrossToRef(this._tempVector, this._tempVector2, this._tempVector4); let s = S.Dot(this._tempVector4, i); return s < 0 || (n.subtractToRef(e, this._tempVector3), S.CrossToRef(this._tempVector2, this._tempVector3, this._tempVector4), s = S.Dot(this._tempVector4, i), s < 0) ? !1 : (S.CrossToRef(this._tempVector3, this._tempVector, this._tempVector4), s = S.Dot(this._tempVector4, i), s >= 0); } /** * @internal */ _canDoCollision(e, t, r, n) { const i = S.Distance(this._basePointWorld, e), s = Math.max(this._radius.x, this._radius.y, this._radius.z); return !(i > this._velocityWorldLength + s + t || !wve(r, n, this._basePointWorld, this._velocityWorldLength + s)); } /** * @internal */ _testTriangle(e, t, r, n, i, s, a) { let f, o = !1; t || (t = []), t[e] || (t[e] = new BA(0, 0, 0, 0), t[e].copyFromPoints(r, n, i)); const d = t[e]; if (!s && !d.isFrontFacingTo(this._normalizedVelocity, 0)) return; const v = d.signedDistanceTo(this._basePoint), u = S.Dot(d.normal, this._velocity); if (LR.DoubleSidedCheck && u > 1e-4) return; if (u == 0) { if (Math.abs(v) >= 1) return; o = !0, f = 0; } else { f = (-1 - v) / u; let p = (1 - v) / u; if (f > p) { const c = p; p = f, f = c; } if (f > 1 || p < 0) return; f < 0 && (f = 0), f > 1 && (f = 1); } this._collisionPoint.copyFromFloats(0, 0, 0); let l = !1, P = 1; if (o || (this._basePoint.subtractToRef(d.normal, this._planeIntersectionPoint), this._velocity.scaleToRef(f, this._tempVector), this._planeIntersectionPoint.addInPlace(this._tempVector), this._checkPointInTriangle(this._planeIntersectionPoint, r, n, i, d.normal) && (l = !0, P = f, this._collisionPoint.copyFrom(this._planeIntersectionPoint))), !l) { let p = this._velocitySquaredLength; this._basePoint.subtractToRef(r, this._tempVector); let c = 2 * S.Dot(this._velocity, this._tempVector), H = this._tempVector.lengthSquared() - 1, T = BW(p, c, H, P); T.found && (P = T.root, l = !0, this._collisionPoint.copyFrom(r)), this._basePoint.subtractToRef(n, this._tempVector), c = 2 * S.Dot(this._velocity, this._tempVector), H = this._tempVector.lengthSquared() - 1, T = BW(p, c, H, P), T.found && (P = T.root, l = !0, this._collisionPoint.copyFrom(n)), this._basePoint.subtractToRef(i, this._tempVector), c = 2 * S.Dot(this._velocity, this._tempVector), H = this._tempVector.lengthSquared() - 1, T = BW(p, c, H, P), T.found && (P = T.root, l = !0, this._collisionPoint.copyFrom(i)), n.subtractToRef(r, this._edge), r.subtractToRef(this._basePoint, this._baseToVertex); let q = this._edge.lengthSquared(), b = S.Dot(this._edge, this._velocity), j = S.Dot(this._edge, this._baseToVertex); if (p = q * -this._velocitySquaredLength + b * b, c = 2 * (q * S.Dot(this._velocity, this._baseToVertex) - b * j), H = q * (1 - this._baseToVertex.lengthSquared()) + j * j, T = BW(p, c, H, P), T.found) { const w = (b * T.root - j) / q; w >= 0 && w <= 1 && (P = T.root, l = !0, this._edge.scaleInPlace(w), r.addToRef(this._edge, this._collisionPoint)); } if (i.subtractToRef(n, this._edge), n.subtractToRef(this._basePoint, this._baseToVertex), q = this._edge.lengthSquared(), b = S.Dot(this._edge, this._velocity), j = S.Dot(this._edge, this._baseToVertex), p = q * -this._velocitySquaredLength + b * b, c = 2 * (q * S.Dot(this._velocity, this._baseToVertex) - b * j), H = q * (1 - this._baseToVertex.lengthSquared()) + j * j, T = BW(p, c, H, P), T.found) { const w = (b * T.root - j) / q; w >= 0 && w <= 1 && (P = T.root, l = !0, this._edge.scaleInPlace(w), n.addToRef(this._edge, this._collisionPoint)); } if (r.subtractToRef(i, this._edge), i.subtractToRef(this._basePoint, this._baseToVertex), q = this._edge.lengthSquared(), b = S.Dot(this._edge, this._velocity), j = S.Dot(this._edge, this._baseToVertex), p = q * -this._velocitySquaredLength + b * b, c = 2 * (q * S.Dot(this._velocity, this._baseToVertex) - b * j), H = q * (1 - this._baseToVertex.lengthSquared()) + j * j, T = BW(p, c, H, P), T.found) { const w = (b * T.root - j) / q; w >= 0 && w <= 1 && (P = T.root, l = !0, this._edge.scaleInPlace(w), i.addToRef(this._edge, this._collisionPoint)); } } if (l) { const p = P * P * this._velocitySquaredLength; (!this.collisionFound || p < this._nearestDistanceSquared) && (a.collisionResponse && (this.intersectionPoint ? this.intersectionPoint.copyFrom(this._collisionPoint) : this.intersectionPoint = this._collisionPoint.clone(), this._nearestDistanceSquared = p, this._nearestDistance = Math.sqrt(p), this.collisionFound = !0), this.collidedMesh = a); } } /** * @internal */ _collide(e, t, r, n, i, s, a, f, o, d = !1) { if (d) if (!r || r.length === 0) for (let v = 0; v < t.length - 2; v += 1) { const u = t[v], l = t[v + 1], P = t[v + 2]; !u || !l || !P || ((o ? 1 : 0) ^ v % 2 ? this._testTriangle(v, e, u, l, P, a, f) : this._testTriangle(v, e, l, u, P, a, f)); } else for (let v = n; v < i - 2; v += 1) { const u = r[v], l = r[v + 1], P = r[v + 2]; if (P === 4294967295) { v += 2; continue; } const p = t[u], c = t[l], H = t[P]; !p || !c || !H || ((o ? 1 : 0) ^ v % 2 ? this._testTriangle(v, e, p, c, H, a, f) : this._testTriangle(v, e, c, p, H, a, f)); } else if (!r || r.length === 0) for (let v = 0; v < t.length; v += 3) { const u = t[v], l = t[v + 1], P = t[v + 2]; o ? this._testTriangle(v, e, u, l, P, a, f) : this._testTriangle(v, e, P, l, u, a, f); } else for (let v = n; v < i; v += 3) { const u = t[r[v] - s], l = t[r[v + 1] - s], P = t[r[v + 2] - s]; o ? this._testTriangle(v, e, u, l, P, a, f) : this._testTriangle(v, e, P, l, u, a, f); } } /** * @internal */ _getResponse(e, t) { e.addToRef(t, this._destinationPoint), t.scaleInPlace(this._nearestDistance / t.length()), this._basePoint.addToRef(t, e), e.subtractToRef(this.intersectionPoint, this._slidePlaneNormal), this._slidePlaneNormal.normalize(), this._slidePlaneNormal.scaleToRef(this._epsilon, this._displacementVector), e.addInPlace(this._displacementVector), this.intersectionPoint.addInPlace(this._displacementVector), this._slidePlaneNormal.scaleInPlace(BA.SignedDistanceToPlaneFromPositionAndNormal(this.intersectionPoint, this._slidePlaneNormal, this._destinationPoint)), this._destinationPoint.subtractInPlace(this._slidePlaneNormal), this._destinationPoint.subtractToRef(this.intersectionPoint, t); } } LR.DoubleSidedCheck = !1; class _ee { constructor() { this._scaledPosition = S.Zero(), this._scaledVelocity = S.Zero(), this._finalPosition = S.Zero(); } getNewPosition(e, t, r, n, i, s, a) { e.divideToRef(r._radius, this._scaledPosition), t.divideToRef(r._radius, this._scaledVelocity), r.collidedMesh = null, r._retry = 0, r._initialVelocity = this._scaledVelocity, r._initialPosition = this._scaledPosition, this._collideWithWorld(this._scaledPosition, this._scaledVelocity, r, n, this._finalPosition, i), this._finalPosition.multiplyInPlace(r._radius), s(a, this._finalPosition, r.collidedMesh); } createCollider() { return new LR(); } init(e) { this._scene = e; } _collideWithWorld(e, t, r, n, i, s = null) { const a = Ge.CollisionsEpsilon * 10; if (r._retry >= n) { i.copyFrom(e); return; } const f = s ? s.collisionMask : r.collisionMask; r._initialize(e, t, a); const o = s && s.surroundingMeshes || this._scene.meshes; for (let d = 0; d < o.length; d++) { const v = o[d]; v.isEnabled() && v.checkCollisions && v.subMeshes && v !== s && f & v.collisionGroup && v._checkCollision(r); } if (!r.collisionFound) { e.addToRef(t, i); return; } if ((t.x !== 0 || t.y !== 0 || t.z !== 0) && r._getResponse(e, t), t.length() <= a) { i.copyFrom(e); return; } r._retry++, this._collideWithWorld(e, t, r, n, i, s); } } sr.CollisionCoordinatorFactory = () => new _ee(); class Pm { /** * Creates a compute effect that can be used to execute a compute shader * @param baseName Name of the effect * @param options Set of all options to create the effect * @param engine The engine the effect is created for * @param key Effect Key identifying uniquely compiled shader variants */ constructor(e, t, r, n = "") { var i, s; this.name = null, this.defines = "", this.onCompiled = null, this.onError = null, this.uniqueId = 0, this.onCompileObservable = new Oe(), this.onErrorObservable = new Oe(), this.onBindObservable = new Oe(), this._wasPreviouslyReady = !1, this._isReady = !1, this._compilationError = "", this._key = "", this._computeSourceCodeOverride = "", this._pipelineContext = null, this._computeSourceCode = "", this._rawComputeSourceCode = "", this._shaderLanguage = za.WGSL, this.name = e, this._key = n, this._engine = r, this.uniqueId = Pm._UniqueIdSeed++, this.defines = (i = t.defines) !== null && i !== void 0 ? i : "", this.onError = t.onError, this.onCompiled = t.onCompiled, this._entryPoint = (s = t.entryPoint) !== null && s !== void 0 ? s : "main", this._shaderStore = Le.GetShadersStore(this._shaderLanguage), this._shaderRepository = Le.GetShadersRepository(this._shaderLanguage), this._includeShaderStore = Le.GetIncludesShadersStore(this._shaderLanguage); let a; const f = u9() ? this._engine.getHostDocument() : null; e.computeSource ? a = "source:" + e.computeSource : e.computeElement ? (a = f ? f.getElementById(e.computeElement) : null, a || (a = e.computeElement)) : a = e.compute || e; const o = { defines: this.defines.split(` `), indexParameters: void 0, isFragment: !1, shouldUseHighPrecisionShader: !1, processor: null, supportsUniformBuffers: this._engine.supportsUniformBuffers, shadersRepository: this._shaderRepository, includesShadersStore: this._includeShaderStore, version: (this._engine.version * 100).toString(), platformName: this._engine.shaderPlatformName, processingContext: null, isNDCHalfZRange: this._engine.isNDCHalfZRange, useReverseDepthBuffer: this._engine.useReverseDepthBuffer }; this._loadShader(a, "Compute", "", (d) => { hp.Initialize(o), hp.PreProcess(d, o, (v) => { this._rawComputeSourceCode = d, t.processFinalCode && (v = t.processFinalCode(v)); const u = hp.Finalize(v, "", o); this._useFinalCode(u.vertexCode, e); }, this._engine); }); } _useFinalCode(e, t) { if (t) { const r = t.computeElement || t.compute || t.spectorName || t; this._computeSourceCode = "//#define SHADER_NAME compute:" + r + ` ` + e; } else this._computeSourceCode = e; this._prepareEffect(); } /** * Unique key for this effect */ get key() { return this._key; } /** * If the effect has been compiled and prepared. * @returns if the effect is compiled and prepared. */ isReady() { try { return this._isReadyInternal(); } catch { return !1; } } _isReadyInternal() { return this._isReady ? !0 : this._pipelineContext ? this._pipelineContext.isReady : !1; } /** * The engine the effect was initialized with. * @returns the engine. */ getEngine() { return this._engine; } /** * The pipeline context for this effect * @returns the associated pipeline context */ getPipelineContext() { return this._pipelineContext; } /** * The error from the last compilation. * @returns the error string. */ getCompilationError() { return this._compilationError; } /** * Adds a callback to the onCompiled observable and call the callback immediately if already ready. * @param func The callback to be used. */ executeWhenCompiled(e) { if (this.isReady()) { e(this); return; } this.onCompileObservable.add((t) => { e(t); }), (!this._pipelineContext || this._pipelineContext.isAsync) && setTimeout(() => { this._checkIsReady(null); }, 16); } _checkIsReady(e) { try { if (this._isReadyInternal()) return; } catch (t) { this._processCompilationErrors(t, e); return; } setTimeout(() => { this._checkIsReady(e); }, 16); } _loadShader(e, t, r, n) { if (typeof HTMLElement < "u" && e instanceof HTMLElement) { const s = gR(e); n(s); return; } if (e.substr(0, 7) === "source:") { n(e.substr(7)); return; } if (e.substr(0, 7) === "base64:") { const s = window.atob(e.substr(7)); n(s); return; } if (this._shaderStore[e + t + "Shader"]) { n(this._shaderStore[e + t + "Shader"]); return; } if (r && this._shaderStore[e + r + "Shader"]) { n(this._shaderStore[e + r + "Shader"]); return; } let i; e[0] === "." || e[0] === "/" || e.indexOf("http") > -1 ? i = e : i = this._shaderRepository + e, this._engine._loadFile(i + "." + t.toLowerCase() + ".fx", n); } /** * Gets the compute shader source code of this effect */ get computeSourceCode() { var e, t; return this._computeSourceCodeOverride ? this._computeSourceCodeOverride : (t = (e = this._pipelineContext) === null || e === void 0 ? void 0 : e._getComputeShaderCode()) !== null && t !== void 0 ? t : this._computeSourceCode; } /** * Gets the compute shader source code before it has been processed by the preprocessor */ get rawComputeSourceCode() { return this._rawComputeSourceCode; } /** * Prepares the effect * @internal */ _prepareEffect() { const e = this.defines, t = this._pipelineContext; this._isReady = !1; try { const r = this._engine; this._pipelineContext = r.createComputePipelineContext(), this._pipelineContext._name = this._key, r._prepareComputePipelineContext(this._pipelineContext, this._computeSourceCodeOverride ? this._computeSourceCodeOverride : this._computeSourceCode, this._rawComputeSourceCode, this._computeSourceCodeOverride ? null : e, this._entryPoint), r._executeWhenComputeStateIsCompiled(this._pipelineContext, () => { this._compilationError = "", this._isReady = !0, this.onCompiled && this.onCompiled(this), this.onCompileObservable.notifyObservers(this), this.onCompileObservable.clear(), t && this.getEngine()._deleteComputePipelineContext(t); }), this._pipelineContext.isAsync && this._checkIsReady(t); } catch (r) { this._processCompilationErrors(r, t); } } _getShaderCodeAndErrorLine(e, t) { const r = /COMPUTE SHADER ERROR: 0:(\d+?):/; let n = null; if (t && e) { const i = t.match(r); if (i && i.length === 2) { const s = parseInt(i[1]), a = e.split(` `, -1); a.length >= s && (n = `Offending line [${s}] in compute code: ${a[s - 1]}`); } } return [e, n]; } _processCompilationErrors(e, t = null) { var r; if (this._compilationError = e.message, Se.Error("Unable to compile compute effect:"), Se.Error(`Defines: ` + this.defines), Pm.LogShaderCodeOnCompilationError) { let n = null, i = null; !((r = this._pipelineContext) === null || r === void 0) && r._getComputeShaderCode() && ([i, n] = this._getShaderCodeAndErrorLine(this._pipelineContext._getComputeShaderCode(), this._compilationError), i && (Se.Error("Compute code:"), Se.Error(i))), n && Se.Error(n); } Se.Error("Error: " + this._compilationError), t && (this._pipelineContext = t, this._isReady = !0, this.onError && this.onError(this, this._compilationError), this.onErrorObservable.notifyObservers(this)); } /** * Release all associated resources. **/ dispose() { this._pipelineContext && this._pipelineContext.dispose(), this._engine._releaseComputeEffect(this); } /** * This function will add a new compute shader to the shader store * @param name the name of the shader * @param computeShader compute shader content */ static RegisterShader(e, t) { Le.GetShadersStore(za.WGSL)[`${e}ComputeShader`] = t; } } Pm._UniqueIdSeed = 0; Pm.LogShaderCodeOnCompilationError = !0; var io; (function(A) { A[A.Texture = 0] = "Texture", A[A.StorageTexture = 1] = "StorageTexture", A[A.UniformBuffer = 2] = "UniformBuffer", A[A.StorageBuffer = 3] = "StorageBuffer", A[A.TextureWithoutSampler = 4] = "TextureWithoutSampler", A[A.Sampler = 5] = "Sampler", A[A.ExternalTexture = 6] = "ExternalTexture"; })(io || (io = {})); hr.prototype.createComputeEffect = function(A, e) { throw new Error("createComputeEffect: This engine does not support compute shaders!"); }; hr.prototype.createComputePipelineContext = function() { throw new Error("createComputePipelineContext: This engine does not support compute shaders!"); }; hr.prototype.createComputeContext = function() { }; hr.prototype.computeDispatch = function(A, e, t, r, n, i, s) { throw new Error("computeDispatch: This engine does not support compute shaders!"); }; hr.prototype.areAllComputeEffectsReady = function() { return !0; }; hr.prototype.releaseComputeEffects = function() { }; hr.prototype._prepareComputePipelineContext = function(A, e, t, r, n) { }; hr.prototype._rebuildComputeEffects = function() { }; hr.prototype._executeWhenComputeStateIsCompiled = function(A, e) { e(); }; hr.prototype._releaseComputeEffect = function(A) { }; hr.prototype._deleteComputePipelineContext = function(A) { }; class vU { /** * The options used to create the shader */ get options() { return this._options; } /** * The shaderPath used to create the shader */ get shaderPath() { return this._shaderPath; } /** * Instantiates a new compute shader. * @param name Defines the name of the compute shader in the scene * @param engine Defines the engine the compute shader belongs to * @param shaderPath Defines the route to the shader code in one of three ways: * * object: \{ compute: "custom" \}, used with ShaderStore.ShadersStoreWGSL["customComputeShader"] * * object: \{ computeElement: "HTMLElementId" \}, used with shader code in script tags * * object: \{ computeSource: "compute shader code string" \}, where the string contains the shader code * * string: try first to find the code in ShaderStore.ShadersStoreWGSL[shaderPath + "ComputeShader"]. If not, assumes it is a file with name shaderPath.compute.fx in index.html folder. * @param options Define the options used to create the shader */ constructor(e, t, r, n = {}) { if (this._bindings = {}, this._samplers = {}, this._contextIsDirty = !1, this.onCompiled = null, this.onError = null, this.name = e, this._engine = t, this.uniqueId = bR.UniqueId, !this._engine.getCaps().supportComputeShaders) { Se.Error("This engine does not support compute shaders!"); return; } if (!n.bindingsMapping) { Se.Error("You must provide the binding mappings as browsers don't support reflection for wgsl shaders yet!"); return; } this._context = t.createComputeContext(), this._shaderPath = r, this._options = Object.assign({ bindingsMapping: {}, defines: [] }, n); } /** * Gets the current class name of the material e.g. "ComputeShader" * Mainly use in serialization. * @returns the class name */ getClassName() { return "ComputeShader"; } /** * Binds a texture to the shader * @param name Binding name of the texture * @param texture Texture to bind * @param bindSampler Bind the sampler corresponding to the texture (default: true). The sampler will be bound just before the binding index of the texture */ setTexture(e, t, r = !0) { const n = this._bindings[e]; this._bindings[e] = { type: r ? io.Texture : io.TextureWithoutSampler, object: t, indexInGroupEntries: n == null ? void 0 : n.indexInGroupEntries }, this._contextIsDirty || (this._contextIsDirty = !n || n.object !== t || n.type !== this._bindings[e].type); } /** * Binds a storage texture to the shader * @param name Binding name of the texture * @param texture Texture to bind */ setStorageTexture(e, t) { const r = this._bindings[e]; this._contextIsDirty || (this._contextIsDirty = !r || r.object !== t), this._bindings[e] = { type: io.StorageTexture, object: t, indexInGroupEntries: r == null ? void 0 : r.indexInGroupEntries }; } /** * Binds an external texture to the shader * @param name Binding name of the texture * @param texture Texture to bind */ setExternalTexture(e, t) { const r = this._bindings[e]; this._contextIsDirty || (this._contextIsDirty = !r || r.object !== t), this._bindings[e] = { type: io.ExternalTexture, object: t, indexInGroupEntries: r == null ? void 0 : r.indexInGroupEntries }; } /** * Binds a video texture to the shader (by binding the external texture attached to this video) * @param name Binding name of the texture * @param texture Texture to bind * @returns true if the video texture was successfully bound, else false. false will be returned if the current engine does not support external textures */ setVideoTexture(e, t) { return t.externalTexture ? (this.setExternalTexture(e, t.externalTexture), !0) : !1; } /** * Binds a uniform buffer to the shader * @param name Binding name of the buffer * @param buffer Buffer to bind */ setUniformBuffer(e, t) { const r = this._bindings[e]; this._contextIsDirty || (this._contextIsDirty = !r || r.object !== t), this._bindings[e] = { type: io.UniformBuffer, object: t, indexInGroupEntries: r == null ? void 0 : r.indexInGroupEntries }; } /** * Binds a storage buffer to the shader * @param name Binding name of the buffer * @param buffer Buffer to bind */ setStorageBuffer(e, t) { const r = this._bindings[e]; this._contextIsDirty || (this._contextIsDirty = !r || r.object !== t), this._bindings[e] = { type: io.StorageBuffer, object: t, indexInGroupEntries: r == null ? void 0 : r.indexInGroupEntries }; } /** * Binds a texture sampler to the shader * @param name Binding name of the sampler * @param sampler Sampler to bind */ setTextureSampler(e, t) { const r = this._bindings[e]; this._contextIsDirty || (this._contextIsDirty = !r || !t.compareSampler(r.object)), this._bindings[e] = { type: io.Sampler, object: t, indexInGroupEntries: r == null ? void 0 : r.indexInGroupEntries }; } /** * Specifies that the compute shader is ready to be executed (the compute effect and all the resources are ready) * @returns true if the compute shader is ready to be executed */ isReady() { let e = this._effect; for (const i in this._bindings) { const s = this._bindings[i], a = s.type, f = s.object; switch (a) { case io.Texture: case io.TextureWithoutSampler: case io.StorageTexture: { if (!f.isReady()) return !1; break; } case io.ExternalTexture: { if (!f.isReady()) return !1; break; } } } const t = [], r = this._shaderPath; if (this._options.defines) for (let i = 0; i < this._options.defines.length; i++) t.push(this._options.defines[i]); const n = t.join(` `); return this._cachedDefines !== n && (this._cachedDefines = n, e = this._engine.createComputeEffect(r, { defines: n, entryPoint: this._options.entryPoint, onCompiled: this.onCompiled, onError: this.onError }), this._effect = e), !!e.isReady(); } /** * Dispatches (executes) the compute shader * @param x Number of workgroups to execute on the X dimension * @param y Number of workgroups to execute on the Y dimension (default: 1) * @param z Number of workgroups to execute on the Z dimension (default: 1) * @returns True if the dispatch could be done, else false (meaning either the compute effect or at least one of the bound resources was not ready) */ dispatch(e, t, r) { var n; if (!this.isReady()) return !1; for (const i in this._bindings) { const s = this._bindings[i]; if (!this._options.bindingsMapping[i]) throw new Error("ComputeShader ('" + this.name + "'): No binding mapping has been provided for the property '" + i + "'"); switch (s.type) { case io.Texture: { const a = this._samplers[i], f = s.object; (!a || !f._texture || !a.compareSampler(f._texture)) && (this._samplers[i] = new MN().setParameters(f.wrapU, f.wrapV, f.wrapR, f.anisotropicFilteringLevel, f._texture.samplingMode, (n = f._texture) === null || n === void 0 ? void 0 : n._comparisonFunction), this._contextIsDirty = !0); break; } case io.ExternalTexture: { this._contextIsDirty = !0; break; } case io.UniformBuffer: { const a = s.object; a.getBuffer() !== s.buffer && (s.buffer = a.getBuffer(), this._contextIsDirty = !0); break; } } } return this._contextIsDirty && (this._contextIsDirty = !1, this._context.clear()), this._engine.computeDispatch(this._effect, this._context, this._bindings, e, t, r, this._options.bindingsMapping), !0; } /** * Waits for the compute shader to be ready and executes it * @param x Number of workgroups to execute on the X dimension * @param y Number of workgroups to execute on the Y dimension (default: 1) * @param z Number of workgroups to execute on the Z dimension (default: 1) * @param delay Delay between the retries while the shader is not ready (in milliseconds - 10 by default) * @returns A promise that is resolved once the shader has been sent to the GPU. Note that it does not mean that the shader execution itself is finished! */ dispatchWhenReady(e, t, r, n = 10) { return new Promise((i) => { const s = () => { this.dispatch(e, t, r) ? i() : setTimeout(s, n); }; s(); }); } /** * Serializes this compute shader in a JSON representation * @returns the serialized compute shader object */ serialize() { const e = jt.Serialize(this); e.options = this._options, e.shaderPath = this._shaderPath, e.bindings = {}, e.textures = {}; for (const t in this._bindings) { const r = this._bindings[t], n = r.object; switch (r.type) { case io.Texture: case io.TextureWithoutSampler: case io.StorageTexture: { const i = n.serialize(); i && (e.textures[t] = i, e.bindings[t] = { type: r.type }); break; } case io.UniformBuffer: break; } } return e; } /** * Creates a compute shader from parsed compute shader data * @param source defines the JSON representation of the compute shader * @param scene defines the hosting scene * @param rootUrl defines the root URL to use to load textures and relative dependencies * @returns a new compute shader */ static Parse(e, t, r) { const n = jt.Parse(() => new vU(e.name, t.getEngine(), e.shaderPath, e.options), e, t, r); for (const i in e.textures) { const s = e.bindings[i], a = We.Parse(e.textures[i], t, r); s.type === io.Texture ? n.setTexture(i, a) : s.type === io.TextureWithoutSampler ? n.setTexture(i, a, !1) : n.setStorageTexture(i, a); } return n; } } C([ M() ], vU.prototype, "name", void 0); Ue("BABYLON.ComputeShader", vU); class NI { /** * Creates a new block * @param minPoint defines the minimum vector (in world space) of the block's bounding box * @param maxPoint defines the maximum vector (in world space) of the block's bounding box * @param capacity defines the maximum capacity of this block (if capacity is reached the block will be split into sub blocks) * @param depth defines the current depth of this block in the octree * @param maxDepth defines the maximal depth allowed (beyond this value, the capacity is ignored) * @param creationFunc defines a callback to call when an element is added to the block */ constructor(e, t, r, n, i, s) { this.entries = [], this._boundingVectors = new Array(), this._capacity = r, this._depth = n, this._maxDepth = i, this._creationFunc = s, this._minPoint = e, this._maxPoint = t, this._boundingVectors.push(e.clone()), this._boundingVectors.push(t.clone()), this._boundingVectors.push(e.clone()), this._boundingVectors[2].x = t.x, this._boundingVectors.push(e.clone()), this._boundingVectors[3].y = t.y, this._boundingVectors.push(e.clone()), this._boundingVectors[4].z = t.z, this._boundingVectors.push(t.clone()), this._boundingVectors[5].z = e.z, this._boundingVectors.push(t.clone()), this._boundingVectors[6].x = e.x, this._boundingVectors.push(t.clone()), this._boundingVectors[7].y = e.y; } // Property /** * Gets the maximum capacity of this block (if capacity is reached the block will be split into sub blocks) */ get capacity() { return this._capacity; } /** * Gets the minimum vector (in world space) of the block's bounding box */ get minPoint() { return this._minPoint; } /** * Gets the maximum vector (in world space) of the block's bounding box */ get maxPoint() { return this._maxPoint; } // Methods /** * Add a new element to this block * @param entry defines the element to add */ addEntry(e) { if (this.blocks) { for (let t = 0; t < this.blocks.length; t++) this.blocks[t].addEntry(e); return; } this._creationFunc(e, this), this.entries.length > this.capacity && this._depth < this._maxDepth && this.createInnerBlocks(); } /** * Remove an element from this block * @param entry defines the element to remove */ removeEntry(e) { if (this.blocks) { for (let r = 0; r < this.blocks.length; r++) this.blocks[r].removeEntry(e); return; } const t = this.entries.indexOf(e); t > -1 && this.entries.splice(t, 1); } /** * Add an array of elements to this block * @param entries defines the array of elements to add */ addEntries(e) { for (let t = 0; t < e.length; t++) { const r = e[t]; this.addEntry(r); } } /** * Test if the current block intersects the frustum planes and if yes, then add its content to the selection array * @param frustumPlanes defines the frustum planes to test * @param selection defines the array to store current content if selection is positive * @param allowDuplicate defines if the selection array can contains duplicated entries */ select(e, t, r) { if (cp.IsInFrustum(this._boundingVectors, e)) { if (this.blocks) { for (let n = 0; n < this.blocks.length; n++) this.blocks[n].select(e, t, r); return; } r ? t.concat(this.entries) : t.concatWithNoDuplicate(this.entries); } } /** * Test if the current block intersect with the given bounding sphere and if yes, then add its content to the selection array * @param sphereCenter defines the bounding sphere center * @param sphereRadius defines the bounding sphere radius * @param selection defines the array to store current content if selection is positive * @param allowDuplicate defines if the selection array can contains duplicated entries */ intersects(e, t, r, n) { if (cp.IntersectsSphere(this._minPoint, this._maxPoint, e, t)) { if (this.blocks) { for (let i = 0; i < this.blocks.length; i++) this.blocks[i].intersects(e, t, r, n); return; } n ? r.concat(this.entries) : r.concatWithNoDuplicate(this.entries); } } /** * Test if the current block intersect with the given ray and if yes, then add its content to the selection array * @param ray defines the ray to test with * @param selection defines the array to store current content if selection is positive */ intersectsRay(e, t) { if (e.intersectsBoxMinMax(this._minPoint, this._maxPoint)) { if (this.blocks) { for (let r = 0; r < this.blocks.length; r++) this.blocks[r].intersectsRay(e, t); return; } t.concatWithNoDuplicate(this.entries); } } /** * Subdivide the content into child blocks (this block will then be empty) */ createInnerBlocks() { NI._CreateBlocks(this._minPoint, this._maxPoint, this.entries, this._capacity, this._depth, this._maxDepth, this, this._creationFunc), this.entries.splice(0); } /** * @internal */ static _CreateBlocks(e, t, r, n, i, s, a, f) { a.blocks = new Array(); const o = new S((t.x - e.x) / 2, (t.y - e.y) / 2, (t.z - e.z) / 2); for (let d = 0; d < 2; d++) for (let v = 0; v < 2; v++) for (let u = 0; u < 2; u++) { const l = e.add(o.multiplyByFloats(d, v, u)), P = e.add(o.multiplyByFloats(d + 1, v + 1, u + 1)), p = new NI(l, P, n, i + 1, s, f); p.addEntries(r), a.blocks.push(p); } } } class cm { /** * Creates a octree * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimizeOctrees * @param creationFunc function to be used to instantiate the octree * @param maxBlockCapacity defines the maximum number of meshes you want on your octree's leaves (default: 64) * @param maxDepth defines the maximum depth (sub-levels) for your octree. Default value is 2, which means 8 8 8 = 512 blocks :) (This parameter takes precedence over capacity.) */ constructor(e, t, r = 2) { this.maxDepth = r, this.dynamicContent = [], this._maxBlockCapacity = t || 64, this._selectionContent = new K8(1024), this._creationFunc = e; } // Methods /** * Updates the octree by adding blocks for the passed in meshes within the min and max world parameters * @param worldMin worldMin for the octree blocks var blockSize = new Vector3((worldMax.x - worldMin.x) / 2, (worldMax.y - worldMin.y) / 2, (worldMax.z - worldMin.z) / 2); * @param worldMax worldMax for the octree blocks var blockSize = new Vector3((worldMax.x - worldMin.x) / 2, (worldMax.y - worldMin.y) / 2, (worldMax.z - worldMin.z) / 2); * @param entries meshes to be added to the octree blocks */ update(e, t, r) { NI._CreateBlocks(e, t, r, this._maxBlockCapacity, 0, this.maxDepth, this, this._creationFunc); } /** * Adds a mesh to the octree * @param entry Mesh to add to the octree */ addMesh(e) { for (let t = 0; t < this.blocks.length; t++) this.blocks[t].addEntry(e); } /** * Remove an element from the octree * @param entry defines the element to remove */ removeMesh(e) { for (let t = 0; t < this.blocks.length; t++) this.blocks[t].removeEntry(e); } /** * Selects an array of meshes within the frustum * @param frustumPlanes The frustum planes to use which will select all meshes within it * @param allowDuplicate If duplicate objects are allowed in the resulting object array * @returns array of meshes within the frustum */ select(e, t) { this._selectionContent.reset(); for (let r = 0; r < this.blocks.length; r++) this.blocks[r].select(e, this._selectionContent, t); return t ? this._selectionContent.concat(this.dynamicContent) : this._selectionContent.concatWithNoDuplicate(this.dynamicContent), this._selectionContent; } /** * Test if the octree intersect with the given bounding sphere and if yes, then add its content to the selection array * @param sphereCenter defines the bounding sphere center * @param sphereRadius defines the bounding sphere radius * @param allowDuplicate defines if the selection array can contains duplicated entries * @returns an array of objects that intersect the sphere */ intersects(e, t, r) { this._selectionContent.reset(); for (let n = 0; n < this.blocks.length; n++) this.blocks[n].intersects(e, t, this._selectionContent, r); return r ? this._selectionContent.concat(this.dynamicContent) : this._selectionContent.concatWithNoDuplicate(this.dynamicContent), this._selectionContent; } /** * Test if the octree intersect with the given ray and if yes, then add its content to resulting array * @param ray defines the ray to test with * @returns array of intersected objects */ intersectsRay(e) { this._selectionContent.reset(); for (let t = 0; t < this.blocks.length; t++) this.blocks[t].intersectsRay(e, this._selectionContent); return this._selectionContent.concatWithNoDuplicate(this.dynamicContent), this._selectionContent; } } cm.CreationFuncForMeshes = (A, e) => { const t = A.getBoundingInfo(); !A.isBlocked && t.boundingBox.intersectsMinMax(e.minPoint, e.maxPoint) && e.entries.push(A); }; cm.CreationFuncForSubMeshes = (A, e) => { A.getBoundingInfo().boundingBox.intersectsMinMax(e.minPoint, e.maxPoint) && e.entries.push(A); }; sr.prototype.createOrUpdateSelectionOctree = function(A = 64, e = 2) { let t = this._getComponent(Ot.NAME_OCTREE); t || (t = new DQ(this), this._addComponent(t)), this._selectionOctree || (this._selectionOctree = new cm(cm.CreationFuncForMeshes, A, e)); const r = this.getWorldExtends(); return this._selectionOctree.update(r.min, r.max, this.meshes), this._selectionOctree; }; Object.defineProperty(sr.prototype, "selectionOctree", { get: function() { return this._selectionOctree; }, enumerable: !0, configurable: !0 }); jn.prototype.createOrUpdateSubmeshesOctree = function(A = 64, e = 2) { const t = this.getScene(); let r = t._getComponent(Ot.NAME_OCTREE); r || (r = new DQ(t), t._addComponent(r)), this._submeshesOctree || (this._submeshesOctree = new cm(cm.CreationFuncForSubMeshes, A, e)), this.computeWorldMatrix(!0); const i = this.getBoundingInfo().boundingBox; return this._submeshesOctree.update(i.minimumWorld, i.maximumWorld, this.subMeshes), this._submeshesOctree; }; class DQ { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_OCTREE, this.checksIsEnabled = !0, this._tempRay = new Hi(S.Zero(), new S(1, 1, 1)), e = e || gr.LastCreatedScene, e && (this.scene = e, this.scene.getActiveMeshCandidates = () => this.getActiveMeshCandidates(), this.scene.getActiveSubMeshCandidates = (t) => this.getActiveSubMeshCandidates(t), this.scene.getCollidingSubMeshCandidates = (t, r) => this.getCollidingSubMeshCandidates(t, r), this.scene.getIntersectingSubMeshCandidates = (t, r) => this.getIntersectingSubMeshCandidates(t, r)); } /** * Registers the component in a given scene */ register() { this.scene.onMeshRemovedObservable.add((e) => { const t = this.scene.selectionOctree; if (t != null) { const r = t.dynamicContent.indexOf(e); r !== -1 && t.dynamicContent.splice(r, 1); } }), this.scene.onMeshImportedObservable.add((e) => { const t = this.scene.selectionOctree; t != null && t.addMesh(e); }); } /** * Return the list of active meshes * @returns the list of active meshes */ getActiveMeshCandidates() { var e; return ((e = this.scene._selectionOctree) === null || e === void 0 ? void 0 : e.select(this.scene.frustumPlanes)) || this.scene._getDefaultMeshCandidates(); } /** * Return the list of active sub meshes * @param mesh The mesh to get the candidates sub meshes from * @returns the list of active sub meshes */ getActiveSubMeshCandidates(e) { return e._submeshesOctree && e.useOctreeForRenderingSelection ? e._submeshesOctree.select(this.scene.frustumPlanes) : this.scene._getDefaultSubMeshCandidates(e); } /** * Return the list of sub meshes intersecting with a given local ray * @param mesh defines the mesh to find the submesh for * @param localRay defines the ray in local space * @returns the list of intersecting sub meshes */ getIntersectingSubMeshCandidates(e, t) { return e._submeshesOctree && e.useOctreeForPicking ? (Hi.TransformToRef(t, e.getWorldMatrix(), this._tempRay), e._submeshesOctree.intersectsRay(this._tempRay)) : this.scene._getDefaultSubMeshCandidates(e); } /** * Return the list of sub meshes colliding with a collider * @param mesh defines the mesh to find the submesh for * @param collider defines the collider to evaluate the collision against * @returns the list of colliding sub meshes */ getCollidingSubMeshCandidates(e, t) { if (e._submeshesOctree && e.useOctreeForCollisions) { const r = t._velocityWorldLength + Math.max(t._radius.x, t._radius.y, t._radius.z); return e._submeshesOctree.intersects(t._basePointWorld, r); } return this.scene._getDefaultSubMeshCandidates(e); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { } /** * Disposes the component and the associated resources. */ dispose() { } } function MO(A) { const e = A.height || 2; let t = A.diameterTop === 0 ? 0 : A.diameterTop || A.diameter || 1, r = A.diameterBottom === 0 ? 0 : A.diameterBottom || A.diameter || 1; t = t || 1e-5, r = r || 1e-5; const n = (A.tessellation || 24) | 0, i = (A.subdivisions || 1) | 0, s = !!A.hasRings, a = !!A.enclose, f = A.cap === 0 ? 0 : A.cap || Ee.CAP_ALL, o = A.arc && (A.arc <= 0 || A.arc > 1) ? 1 : A.arc || 1, d = A.sideOrientation === 0 ? 0 : A.sideOrientation || Ut.DEFAULTSIDE, v = A.faceUV || new Array(3), u = A.faceColors, l = o !== 1 && a ? 2 : 0, P = s ? i : 1, p = 2 + (1 + l) * P; let c; for (c = 0; c < p; c++) u && u[c] === void 0 && (u[c] = new xt(1, 1, 1, 1)); for (c = 0; c < p; c++) v && v[c] === void 0 && (v[c] = new Ir(0, 0, 1, 1)); const H = [], T = [], q = [], b = [], j = [], w = Math.PI * 2 * o / n; let m, I, N; const k = (r - t) / 2 / e, R = S.Zero(), y = S.Zero(), O = S.Zero(), Y = S.Zero(), ee = S.Zero(), Z = bf.Y; let te, fe, _, G = 1, L = 1, $ = 0, ae = 0; for (te = 0; te <= i; te++) for (I = te / i, N = (I * (t - r) + r) / 2, G = s && te !== 0 && te !== i ? 2 : 1, _ = 0; _ < G; _++) { for (s && (L += _), a && (L += 2 * _), fe = 0; fe <= n; fe++) m = fe * w, R.x = Math.cos(-m) * N, R.y = -e / 2 + I * e, R.z = Math.sin(-m) * N, t === 0 && te === i ? (y.x = q[q.length - (n + 1) * 3], y.y = q[q.length - (n + 1) * 3 + 1], y.z = q[q.length - (n + 1) * 3 + 2]) : (y.x = R.x, y.z = R.z, y.y = Math.sqrt(y.x * y.x + y.z * y.z) * k, y.normalize()), fe === 0 && (O.copyFrom(R), Y.copyFrom(y)), T.push(R.x, R.y, R.z), q.push(y.x, y.y, y.z), s ? ae = $ !== L ? v[L].y : v[L].w : ae = v[L].y + (v[L].w - v[L].y) * I, b.push(v[L].x + (v[L].z - v[L].x) * fe / n, us.UseOpenGLOrientationForUV ? 1 - ae : ae), u && j.push(u[L].r, u[L].g, u[L].b, u[L].a); o !== 1 && a && (T.push(R.x, R.y, R.z), T.push(0, R.y, 0), T.push(0, R.y, 0), T.push(O.x, O.y, O.z), S.CrossToRef(Z, y, ee), ee.normalize(), q.push(ee.x, ee.y, ee.z, ee.x, ee.y, ee.z), S.CrossToRef(Y, Z, ee), ee.normalize(), q.push(ee.x, ee.y, ee.z, ee.x, ee.y, ee.z), s ? ae = $ !== L ? v[L + 1].y : v[L + 1].w : ae = v[L + 1].y + (v[L + 1].w - v[L + 1].y) * I, b.push(v[L + 1].x, us.UseOpenGLOrientationForUV ? 1 - ae : ae), b.push(v[L + 1].z, us.UseOpenGLOrientationForUV ? 1 - ae : ae), s ? ae = $ !== L ? v[L + 2].y : v[L + 2].w : ae = v[L + 2].y + (v[L + 2].w - v[L + 2].y) * I, b.push(v[L + 2].x, us.UseOpenGLOrientationForUV ? 1 - ae : ae), b.push(v[L + 2].z, us.UseOpenGLOrientationForUV ? 1 - ae : ae), u && (j.push(u[L + 1].r, u[L + 1].g, u[L + 1].b, u[L + 1].a), j.push(u[L + 1].r, u[L + 1].g, u[L + 1].b, u[L + 1].a), j.push(u[L + 2].r, u[L + 2].g, u[L + 2].b, u[L + 2].a), j.push(u[L + 2].r, u[L + 2].g, u[L + 2].b, u[L + 2].a))), $ !== L && ($ = L); } const Pe = o !== 1 && a ? n + 4 : n; for (te = 0, L = 0; L < i; L++) { let Xe = 0, De = 0, ne = 0, re = 0; for (fe = 0; fe < n; fe++) Xe = te * (Pe + 1) + fe, De = (te + 1) * (Pe + 1) + fe, ne = te * (Pe + 1) + (fe + 1), re = (te + 1) * (Pe + 1) + (fe + 1), H.push(Xe, De, ne), H.push(re, ne, De); o !== 1 && a && (H.push(Xe + 2, De + 2, ne + 2), H.push(re + 2, ne + 2, De + 2), H.push(Xe + 4, De + 4, ne + 4), H.push(re + 4, ne + 4, De + 4)), te = s ? te + 2 : te + 1; } const ge = (Xe) => { const De = Xe ? t / 2 : r / 2; if (De === 0) return; let ne, re, ve; const qe = Xe ? v[p - 1] : v[0]; let ke = null; u && (ke = Xe ? u[p - 1] : u[0]); const be = T.length / 3, Fe = Xe ? e / 2 : -e / 2, Ke = new S(0, Fe, 0); T.push(Ke.x, Ke.y, Ke.z), q.push(0, Xe ? 1 : -1, 0); const nt = qe.y + (qe.w - qe.y) * 0.5; b.push(qe.x + (qe.z - qe.x) * 0.5, us.UseOpenGLOrientationForUV ? 1 - nt : nt), ke && j.push(ke.r, ke.g, ke.b, ke.a); const ut = new at(0.5, 0.5); for (ve = 0; ve <= n; ve++) { ne = Math.PI * 2 * ve * o / n; const bt = Math.cos(-ne), wt = Math.sin(-ne); re = new S(bt * De, Fe, wt * De); const Tt = new at(bt * ut.x + 0.5, wt * ut.y + 0.5); T.push(re.x, re.y, re.z), q.push(0, Xe ? 1 : -1, 0); const lr = qe.y + (qe.w - qe.y) * Tt.y; b.push(qe.x + (qe.z - qe.x) * Tt.x, us.UseOpenGLOrientationForUV ? 1 - lr : lr), ke && j.push(ke.r, ke.g, ke.b, ke.a); } for (ve = 0; ve < n; ve++) Xe ? (H.push(be), H.push(be + (ve + 2)), H.push(be + (ve + 1))) : (H.push(be), H.push(be + (ve + 1)), H.push(be + (ve + 2))); }; (f === Ee.CAP_START || f === Ee.CAP_ALL) && ge(!1), (f === Ee.CAP_END || f === Ee.CAP_ALL) && ge(!0), Ut._ComputeSides(d, T, H, q, b, A.frontUVs, A.backUVs); const me = new Ut(); return me.indices = H, me.positions = T, me.normals = q, me.uvs = b, u && (me.colors = j), me; } function Ld(A, e = {}, t) { const r = new Ee(A, t); return e.sideOrientation = Ee._GetDefaultSideOrientation(e.sideOrientation), r._originalBuilderSideOrientation = e.sideOrientation, MO(e).applyToMesh(r, e.updatable), r; } const mve = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateCylinder: Ld }; Ut.CreateCylinder = MO; Ee.CreateCylinder = (A, e, t, r, n, i, s, a, f) => ((s === void 0 || !(s instanceof sr)) && (s !== void 0 && (f = a || Ee.DEFAULTSIDE, a = s), s = i, i = 1), Ld(A, { height: e, diameterTop: t, diameterBottom: r, tessellation: n, subdivisions: i, sideOrientation: f, updatable: a }, s)); Cs.AddNodeConstructor("Light_Type_3", (A, e) => () => new y0(A, S.Zero(), e)); class y0 extends ci { /** * Creates a HemisphericLight object in the scene according to the passed direction (Vector3). * The HemisphericLight simulates the ambient environment light, so the passed direction is the light reflection direction, not the incoming direction. * The HemisphericLight can't cast shadows. * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction * @param name The friendly name of the light * @param direction The direction of the light reflection * @param scene The scene the light belongs to */ constructor(e, t, r) { super(e, r), this.groundColor = new Ne(0, 0, 0), this.direction = t || S.Up(); } _buildUniformLayout() { this._uniformBuffer.addUniform("vLightData", 4), this._uniformBuffer.addUniform("vLightDiffuse", 4), this._uniformBuffer.addUniform("vLightSpecular", 4), this._uniformBuffer.addUniform("vLightGround", 3), this._uniformBuffer.addUniform("shadowsInfo", 3), this._uniformBuffer.addUniform("depthValues", 2), this._uniformBuffer.create(); } /** * Returns the string "HemisphericLight". * @returns The class name */ getClassName() { return "HemisphericLight"; } /** * Sets the HemisphericLight direction towards the passed target (Vector3). * Returns the updated direction. * @param target The target the direction should point to * @returns The computed direction */ setDirectionToTarget(e) { return this.direction = S.Normalize(e.subtract(S.Zero())), this.direction; } /** * Returns the shadow generator associated to the light. * @returns Always null for hemispheric lights because it does not support shadows. */ getShadowGenerator() { return null; } /** * Sets the passed Effect object with the HemisphericLight normalized direction and color and the passed name (string). * @param _effect The effect to update * @param lightIndex The index of the light in the effect to update * @returns The hemispheric light */ transferToEffect(e, t) { const r = S.Normalize(this.direction); return this._uniformBuffer.updateFloat4("vLightData", r.x, r.y, r.z, 0, t), this._uniformBuffer.updateColor3("vLightGround", this.groundColor.scale(this.intensity), t), this; } transferToNodeMaterialEffect(e, t) { const r = S.Normalize(this.direction); return e.setFloat3(t, r.x, r.y, r.z), this; } /** * Computes the world matrix of the node * @returns the world matrix */ computeWorldMatrix() { return this._worldMatrix || (this._worldMatrix = he.Identity()), this._worldMatrix; } /** * Returns the integer 3. * @returns The light Type id as a constant defines in Light.LIGHTTYPEID_x */ getTypeID() { return ci.LIGHTTYPEID_HEMISPHERICLIGHT; } /** * Prepares the list of defines specific to the light type. * @param defines the list of defines * @param lightIndex defines the index of the light for the effect */ prepareLightSpecificDefines(e, t) { e["HEMILIGHT" + t] = !0; } } C([ Oi() ], y0.prototype, "groundColor", void 0); C([ fo() ], y0.prototype, "direction", void 0); class Ds { /** * Gets the camera that is used to render the utility layer (when not set, this will be the last active camera) * @param getRigParentIfPossible if the current active camera is a rig camera, should its parent camera be returned * @returns the camera that is used when rendering the utility layer */ getRenderCamera(e) { if (this._renderCamera) return this._renderCamera; { let t; return this.originalScene.activeCameras && this.originalScene.activeCameras.length > 1 ? t = this.originalScene.activeCameras[this.originalScene.activeCameras.length - 1] : t = this.originalScene.activeCamera, e && t && t.isRigCamera ? t.rigParent : t; } } /** * Sets the camera that should be used when rendering the utility layer (If set to null the last active camera will be used) * @param cam the camera that should be used when rendering the utility layer */ setRenderCamera(e) { this._renderCamera = e; } /** * @internal * Light which used by gizmos to get light shading */ _getSharedGizmoLight() { return this._sharedGizmoLight || (this._sharedGizmoLight = new y0("shared gizmo light", new S(0, 1, 0), this.utilityLayerScene), this._sharedGizmoLight.intensity = 2, this._sharedGizmoLight.groundColor = Ne.Gray()), this._sharedGizmoLight; } /** * A shared utility layer that can be used to overlay objects into a scene (Depth map of the previous scene is cleared before drawing on top of it) */ static get DefaultUtilityLayer() { return Ds._DefaultUtilityLayer == null ? Ds._CreateDefaultUtilityLayerFromScene(gr.LastCreatedScene) : Ds._DefaultUtilityLayer; } /** * Creates an utility layer, and set it as a default utility layer * @param scene associated scene * @internal */ static _CreateDefaultUtilityLayerFromScene(e) { return Ds._DefaultUtilityLayer = new Ds(e), Ds._DefaultUtilityLayer.originalScene.onDisposeObservable.addOnce(() => { Ds._DefaultUtilityLayer = null; }), Ds._DefaultUtilityLayer; } /** * A shared utility layer that can be used to embed objects into a scene (Depth map of the previous scene is not cleared before drawing on top of it) */ static get DefaultKeepDepthUtilityLayer() { return Ds._DefaultKeepDepthUtilityLayer == null && (Ds._DefaultKeepDepthUtilityLayer = new Ds(gr.LastCreatedScene), Ds._DefaultKeepDepthUtilityLayer.utilityLayerScene.autoClearDepthAndStencil = !1, Ds._DefaultKeepDepthUtilityLayer.originalScene.onDisposeObservable.addOnce(() => { Ds._DefaultKeepDepthUtilityLayer = null; })), Ds._DefaultKeepDepthUtilityLayer; } /** * Instantiates a UtilityLayerRenderer * @param originalScene the original scene that will be rendered on top of * @param handleEvents boolean indicating if the utility layer should handle events */ constructor(e, t = !0) { this.originalScene = e, this._pointerCaptures = {}, this._lastPointerEvents = {}, this._sharedGizmoLight = null, this._renderCamera = null, this.pickUtilitySceneFirst = !0, this.shouldRender = !0, this.onlyCheckPointerDownEvents = !0, this.processAllEvents = !1, this.pickingEnabled = !0, this.onPointerOutObservable = new Oe(), this.utilityLayerScene = new sr(e.getEngine(), { virtual: !0 }), this.utilityLayerScene.useRightHandedSystem = e.useRightHandedSystem, this.utilityLayerScene._allowPostProcessClearColor = !1, this.utilityLayerScene.postProcessesEnabled = !1, this.utilityLayerScene.detachControl(), t && (this._originalPointerObserver = e.onPrePointerObservable.add((r) => { if (!this.utilityLayerScene.activeCamera || !this.pickingEnabled || !this.processAllEvents && r.type !== ir.POINTERMOVE && r.type !== ir.POINTERUP && r.type !== ir.POINTERDOWN && r.type !== ir.POINTERDOUBLETAP) return; this.utilityLayerScene.pointerX = e.pointerX, this.utilityLayerScene.pointerY = e.pointerY; const n = r.event; if (e.isPointerCaptured(n.pointerId)) { this._pointerCaptures[n.pointerId] = !1; return; } const i = (a) => { let f = null; if (r.nearInteractionPickingInfo) r.nearInteractionPickingInfo.pickedMesh.getScene() == a ? f = r.nearInteractionPickingInfo : f = new F9(); else if (a !== this.utilityLayerScene && r.originalPickingInfo) f = r.originalPickingInfo; else { let o = null; this._renderCamera && (o = a._activeCamera, a._activeCamera = this._renderCamera, r.ray = null), f = r.ray ? a.pickWithRay(r.ray) : a.pick(e.pointerX, e.pointerY), o && (a._activeCamera = o); } return f; }, s = i(this.utilityLayerScene); if (!r.ray && s && (r.ray = s.ray), this.utilityLayerScene.onPrePointerObservable.notifyObservers(r), this.onlyCheckPointerDownEvents && r.type != ir.POINTERDOWN) { r.skipOnPointerObservable || this.utilityLayerScene.onPointerObservable.notifyObservers(new vp(r.type, r.event, s), r.type), r.type === ir.POINTERUP && this._pointerCaptures[n.pointerId] && (this._pointerCaptures[n.pointerId] = !1); return; } if (this.utilityLayerScene.autoClearDepthAndStencil || this.pickUtilitySceneFirst) s && s.hit && (r.skipOnPointerObservable || this.utilityLayerScene.onPointerObservable.notifyObservers(new vp(r.type, r.event, s), r.type), r.skipOnPointerObservable = !0); else { const a = i(e), f = r.event; a && s && (s.distance === 0 && a.pickedMesh ? this.mainSceneTrackerPredicate && this.mainSceneTrackerPredicate(a.pickedMesh) ? (this._notifyObservers(r, a, f), r.skipOnPointerObservable = !0) : r.type === ir.POINTERDOWN ? this._pointerCaptures[f.pointerId] = !0 : (r.type === ir.POINTERMOVE || r.type === ir.POINTERUP) && (this._lastPointerEvents[f.pointerId] && (this.onPointerOutObservable.notifyObservers(f.pointerId), delete this._lastPointerEvents[f.pointerId]), this._notifyObservers(r, a, f)) : !this._pointerCaptures[f.pointerId] && (s.distance < a.distance || a.distance === 0) ? (this._notifyObservers(r, s, f), r.skipOnPointerObservable || (r.skipOnPointerObservable = s.distance > 0)) : !this._pointerCaptures[f.pointerId] && s.distance >= a.distance && (this.mainSceneTrackerPredicate && this.mainSceneTrackerPredicate(a.pickedMesh) ? (this._notifyObservers(r, a, f), r.skipOnPointerObservable = !0) : ((r.type === ir.POINTERMOVE || r.type === ir.POINTERUP) && this._lastPointerEvents[f.pointerId] && (this.onPointerOutObservable.notifyObservers(f.pointerId), delete this._lastPointerEvents[f.pointerId]), this._notifyObservers(r, s, f))), r.type === ir.POINTERUP && this._pointerCaptures[f.pointerId] && (this._pointerCaptures[f.pointerId] = !1)); } }), this._originalPointerObserver && e.onPrePointerObservable.makeObserverTopPriority(this._originalPointerObserver)), this.utilityLayerScene.autoClear = !1, this._afterRenderObserver = this.originalScene.onAfterRenderCameraObservable.add((r) => { this.shouldRender && r == this.getRenderCamera() && this.render(); }), this._sceneDisposeObserver = this.originalScene.onDisposeObservable.add(() => { this.dispose(); }), this._updateCamera(); } _notifyObservers(e, t, r) { e.skipOnPointerObservable || (this.utilityLayerScene.onPointerObservable.notifyObservers(new vp(e.type, e.event, t), e.type), this._lastPointerEvents[r.pointerId] = !0); } /** * Renders the utility layers scene on top of the original scene */ render() { if (this._updateCamera(), this.utilityLayerScene.activeCamera) { const e = this.utilityLayerScene.activeCamera.getScene(), t = this.utilityLayerScene.activeCamera; t._scene = this.utilityLayerScene, t.leftCamera && (t.leftCamera._scene = this.utilityLayerScene), t.rightCamera && (t.rightCamera._scene = this.utilityLayerScene), this.utilityLayerScene.render(!1), t._scene = e, t.leftCamera && (t.leftCamera._scene = e), t.rightCamera && (t.rightCamera._scene = e); } } /** * Disposes of the renderer */ dispose() { this.onPointerOutObservable.clear(), this._afterRenderObserver && this.originalScene.onAfterCameraRenderObservable.remove(this._afterRenderObserver), this._sceneDisposeObserver && this.originalScene.onDisposeObservable.remove(this._sceneDisposeObserver), this._originalPointerObserver && this.originalScene.onPrePointerObservable.remove(this._originalPointerObserver), this.utilityLayerScene.dispose(); } _updateCamera() { this.utilityLayerScene.cameraToUseForPointers = this.getRenderCamera(), this.utilityLayerScene.activeCamera = this.getRenderCamera(); } } Ds._DefaultUtilityLayer = null; Ds._DefaultKeepDepthUtilityLayer = null; var QI; (function(A) { A[A.Origin = 0] = "Origin", A[A.Pivot = 1] = "Pivot"; })(QI || (QI = {})); var pD; (function(A) { A[A.World = 0] = "World", A[A.Local = 1] = "Local"; })(pD || (pD = {})); class Lo { /** * Ratio for the scale of the gizmo (Default: 1) */ set scaleRatio(e) { this._scaleRatio = e; } get scaleRatio() { return this._scaleRatio; } /** * True when the mouse pointer is hovered a gizmo mesh */ get isHovered() { return this._isHovered; } /** * Mesh that the gizmo will be attached to. (eg. on a drag gizmo the mesh that will be dragged) * * When set, interactions will be enabled */ get attachedMesh() { return this._attachedMesh; } set attachedMesh(e) { this._attachedMesh = e, e && (this._attachedNode = e), this._rootMesh.setEnabled(!!e), this._attachedNodeChanged(e); } /** * Node that the gizmo will be attached to. (eg. on a drag gizmo the mesh, bone or NodeTransform that will be dragged) * * When set, interactions will be enabled */ get attachedNode() { return this._attachedNode; } set attachedNode(e) { this._attachedNode = e, this._attachedMesh = null, this._rootMesh.setEnabled(!!e), this._attachedNodeChanged(e); } /** * Disposes and replaces the current meshes in the gizmo with the specified mesh * @param mesh The mesh to replace the default mesh of the gizmo */ setCustomMesh(e) { if (e.getScene() != this.gizmoLayer.utilityLayerScene) throw "When setting a custom mesh on a gizmo, the custom meshes scene must be the same as the gizmos (eg. gizmo.gizmoLayer.utilityLayerScene)"; this._rootMesh.getChildMeshes().forEach((t) => { t.dispose(); }), e.parent = this._rootMesh, this._customMeshSet = !0; } /** * If set the gizmo's rotation will be updated to match the attached mesh each frame (Default: true) * NOTE: This is only possible for meshes with uniform scaling, as otherwise it's not possible to decompose the rotation */ set updateGizmoRotationToMatchAttachedMesh(e) { this._updateGizmoRotationToMatchAttachedMesh = e; } get updateGizmoRotationToMatchAttachedMesh() { return this._updateGizmoRotationToMatchAttachedMesh; } /** * If set the gizmo's position will be updated to match the attached mesh each frame (Default: true) */ set updateGizmoPositionToMatchAttachedMesh(e) { this._updateGizmoPositionToMatchAttachedMesh = e; } get updateGizmoPositionToMatchAttachedMesh() { return this._updateGizmoPositionToMatchAttachedMesh; } /** * Defines where the gizmo will be positioned if `updateGizmoPositionToMatchAttachedMesh` is enabled. * (Default: GizmoAnchorPoint.Origin) */ set anchorPoint(e) { this._anchorPoint = e; } get anchorPoint() { return this._anchorPoint; } /** * Set the coordinate system to use. By default it's local. * But it's possible for a user to tweak so its local for translation and world for rotation. * In that case, setting the coordinate system will change `updateGizmoRotationToMatchAttachedMesh` and `updateGizmoPositionToMatchAttachedMesh` */ set coordinatesMode(e) { this._coordinatesMode = e; const t = e == pD.Local; this.updateGizmoRotationToMatchAttachedMesh = t, this.updateGizmoPositionToMatchAttachedMesh = !0; } get coordinatesMode() { return this._coordinatesMode; } /** * When set, the gizmo will always appear the same size no matter where the camera is (default: true) */ set updateScale(e) { this._updateScale = e; } get updateScale() { return this._updateScale; } // eslint-disable-next-line @typescript-eslint/no-unused-vars _attachedNodeChanged(e) { } /** * Creates a gizmo * @param gizmoLayer The utility layer the gizmo will be added to */ constructor(e = Ds.DefaultUtilityLayer) { this.gizmoLayer = e, this._attachedMesh = null, this._attachedNode = null, this._customRotationQuaternion = null, this._scaleRatio = 1, this._isHovered = !1, this._customMeshSet = !1, this._updateGizmoRotationToMatchAttachedMesh = !0, this._updateGizmoPositionToMatchAttachedMesh = !0, this._anchorPoint = QI.Origin, this._updateScale = !0, this._coordinatesMode = pD.Local, this._interactionsEnabled = !0, this._rightHandtoLeftHandMatrix = he.RotationY(Math.PI), this._rootMesh = new Ee("gizmoRootNode", e.utilityLayerScene), this._rootMesh.rotationQuaternion = Ze.Identity(), this._beforeRenderObserver = this.gizmoLayer.utilityLayerScene.onBeforeRenderObservable.add(() => { this._update(); }); } /** * posture that the gizmo will be display * When set null, default value will be used (Quaternion(0, 0, 0, 1)) */ get customRotationQuaternion() { return this._customRotationQuaternion; } set customRotationQuaternion(e) { this._customRotationQuaternion = e; } /** * Updates the gizmo to match the attached mesh's position/rotation */ _update() { if (this.attachedNode) { let e = this.attachedNode; if (this.attachedMesh && (e = this.attachedMesh || this.attachedNode), this.updateGizmoPositionToMatchAttachedMesh) if (this.anchorPoint == QI.Pivot && e.getAbsolutePivotPoint) { const t = e.getAbsolutePivotPoint(); this._rootMesh.position.copyFrom(t); } else { const t = e.getWorldMatrix().getRow(3), r = t ? t.toVector3() : new S(0, 0, 0); this._rootMesh.position.copyFrom(r); } if (this.updateGizmoRotationToMatchAttachedMesh) { const r = e._isMesh || e.getClassName() === "AbstractMesh" || e.getClassName() === "TransformNode" || e.getClassName() === "InstancedMesh" ? e : void 0; e.getWorldMatrix().decompose(void 0, this._rootMesh.rotationQuaternion, void 0, Lo.PreserveScaling ? r : void 0), this._rootMesh.rotationQuaternion.normalize(); } else this._customRotationQuaternion ? this._rootMesh.rotationQuaternion.copyFrom(this._customRotationQuaternion) : this._rootMesh.rotationQuaternion.set(0, 0, 0, 1); if (this.updateScale) { const t = this.gizmoLayer.utilityLayerScene.activeCamera, r = t.globalPosition; this._rootMesh.position.subtractToRef(r, ue.Vector3[0]); let n = this.scaleRatio; if (t.mode == Tr.ORTHOGRAPHIC_CAMERA) { if (t.orthoTop && t.orthoBottom) { const i = t.orthoTop - t.orthoBottom; n *= i; } } else { const i = t.getScene().useRightHandedSystem ? S.RightHandedForwardReadOnly : S.LeftHandedForwardReadOnly, s = t.getDirection(i); n *= S.Dot(ue.Vector3[0], s); } this._rootMesh.scaling.setAll(n), e._getWorldMatrixDeterminant() < 0 && !Lo.PreserveScaling && (this._rootMesh.scaling.y *= -1); } else this._rootMesh.scaling.setAll(this.scaleRatio); } } /** * if transform has a pivot and is not using PostMultiplyPivotMatrix, then the worldMatrix contains the pivot matrix (it's not cancelled at the end) * so, when extracting the world matrix component, the translation (and other components) is containing the pivot translation. * And the pivot is applied each frame. Removing it anyway here makes it applied only in computeWorldMatrix. * @param transform local transform that needs to be transform by the pivot inverse matrix * @param localMatrix local matrix that needs to be transform by the pivot inverse matrix * @param result resulting matrix transformed by pivot inverse if the transform node is using pivot without using post Multiply Pivot Matrix */ _handlePivotMatrixInverse(e, t, r) { if (e.isUsingPivotMatrix() && !e.isUsingPostMultiplyPivotMatrix()) { e.getPivotMatrix().invertToRef(ue.Matrix[5]), ue.Matrix[5].multiplyToRef(t, r); return; } r.copyFrom(t); } /** * computes the rotation/scaling/position of the transform once the Node world matrix has changed. */ _matrixChanged() { if (this._attachedNode) if (this._attachedNode._isCamera) { const e = this._attachedNode; let t, r; if (e.parent) { const i = ue.Matrix[1]; e.parent._worldMatrix.invertToRef(i), this._attachedNode._worldMatrix.multiplyToRef(i, ue.Matrix[0]), t = ue.Matrix[0]; } else t = this._attachedNode._worldMatrix; if (e.getScene().useRightHandedSystem ? (this._rightHandtoLeftHandMatrix.multiplyToRef(t, ue.Matrix[1]), r = ue.Matrix[1]) : r = t, r.decompose(ue.Vector3[1], ue.Quaternion[0], ue.Vector3[0]), this._attachedNode.getClassName() === "FreeCamera" || this._attachedNode.getClassName() === "FlyCamera" || this._attachedNode.getClassName() === "ArcFollowCamera" || this._attachedNode.getClassName() === "TargetCamera" || this._attachedNode.getClassName() === "TouchCamera" || this._attachedNode.getClassName() === "UniversalCamera") { const i = this._attachedNode; i.rotation = ue.Quaternion[0].toEulerAngles(), i.rotationQuaternion && (i.rotationQuaternion.copyFrom(ue.Quaternion[0]), i.rotationQuaternion.normalize()); } e.position.copyFrom(ue.Vector3[0]); } else if (this._attachedNode._isMesh || this._attachedNode.getClassName() === "AbstractMesh" || this._attachedNode.getClassName() === "TransformNode" || this._attachedNode.getClassName() === "InstancedMesh") { const e = this._attachedNode; if (e.parent) { const t = ue.Matrix[0], r = ue.Matrix[1]; e.parent.getWorldMatrix().invertToRef(t), this._attachedNode.getWorldMatrix().multiplyToRef(t, r); const n = ue.Matrix[4]; if (this._handlePivotMatrixInverse(e, r, n), n.decompose(ue.Vector3[0], ue.Quaternion[0], e.position, Lo.PreserveScaling ? e : void 0, Lo.UseAbsoluteScaling), ue.Quaternion[0].normalize(), e.isUsingPivotMatrix()) { const i = ue.Quaternion[1]; Ze.RotationYawPitchRollToRef(e.rotation.y, e.rotation.x, e.rotation.z, i); const s = ue.Matrix[2]; he.ScalingToRef(e.scaling.x, e.scaling.y, e.scaling.z, s); const a = ue.Matrix[2]; i.toRotationMatrix(a); const f = e.getPivotMatrix(), o = ue.Matrix[3]; f.invertToRef(o), f.multiplyToRef(s, ue.Matrix[4]), ue.Matrix[4].multiplyToRef(a, ue.Matrix[5]), ue.Matrix[5].multiplyToRef(o, ue.Matrix[6]), ue.Matrix[6].getTranslationToRef(ue.Vector3[1]), e.position.subtractInPlace(ue.Vector3[1]); } } else { const t = ue.Matrix[4]; this._handlePivotMatrixInverse(e, this._attachedNode._worldMatrix, t), t.decompose(ue.Vector3[0], ue.Quaternion[0], e.position, Lo.PreserveScaling ? e : void 0, Lo.UseAbsoluteScaling); } ue.Vector3[0].scaleInPlace(1 / e.scalingDeterminant), e.scaling.copyFrom(ue.Vector3[0]), e.billboardMode || (e.rotationQuaternion ? (e.rotationQuaternion.copyFrom(ue.Quaternion[0]), e.rotationQuaternion.normalize()) : e.rotation = ue.Quaternion[0].toEulerAngles()); } else if (this._attachedNode.getClassName() === "Bone") { const e = this._attachedNode, t = e.getParent(); if (t) { const r = ue.Matrix[0], n = ue.Matrix[1]; t.getFinalMatrix().invertToRef(r), e.getFinalMatrix().multiplyToRef(r, n), e.getLocalMatrix().copyFrom(n); } else e.getLocalMatrix().copyFrom(e.getFinalMatrix()); e.markAsDirty(); } else { const e = this._attachedNode; if (e.getTypeID) { const t = e.getTypeID(); if (t === ci.LIGHTTYPEID_DIRECTIONALLIGHT || t === ci.LIGHTTYPEID_SPOTLIGHT || t === ci.LIGHTTYPEID_POINTLIGHT) { const r = e.parent; if (r) { const n = ue.Matrix[0], i = ue.Matrix[1]; r.getWorldMatrix().invertToRef(n), e.getWorldMatrix().multiplyToRef(n, i), i.decompose(void 0, ue.Quaternion[0], ue.Vector3[0]); } else this._attachedNode._worldMatrix.decompose(void 0, ue.Quaternion[0], ue.Vector3[0]); e.position = new S(ue.Vector3[0].x, ue.Vector3[0].y, ue.Vector3[0].z), e.direction && (e.direction = new S(e.direction.x, e.direction.y, e.direction.z)); } } } } /** * refresh gizmo mesh material * @param gizmoMeshes * @param material material to apply */ _setGizmoMeshMaterial(e, t) { e && e.forEach((r) => { r.material = t, r.color && (r.color = t.diffuseColor); }); } /** * Subscribes to pointer up, down, and hover events. Used for responsive gizmos. * @param gizmoLayer The utility layer the gizmo will be added to * @param gizmoAxisCache Gizmo axis definition used for reactive gizmo UI * @returns {Observer} pointerObserver */ static GizmoAxisPointerObserver(e, t) { let r = !1; return e.utilityLayerScene.onPointerObservable.add((i) => { var s, a; if (i.pickInfo) { if (i.type === ir.POINTERMOVE) { if (r) return; t.forEach((f) => { var o, d; if (f.colliderMeshes && f.gizmoMeshes) { const v = ((o = f.colliderMeshes) === null || o === void 0 ? void 0 : o.indexOf((d = i == null ? void 0 : i.pickInfo) === null || d === void 0 ? void 0 : d.pickedMesh)) != -1, u = f.dragBehavior.enabled ? v || f.active ? f.hoverMaterial : f.material : f.disableMaterial; f.gizmoMeshes.forEach((l) => { l.material = u, l.color && (l.color = u.diffuseColor); }); } }); } if (i.type === ir.POINTERDOWN && t.has((s = i.pickInfo.pickedMesh) === null || s === void 0 ? void 0 : s.parent)) { r = !0; const f = t.get((a = i.pickInfo.pickedMesh) === null || a === void 0 ? void 0 : a.parent); f.active = !0, t.forEach((o) => { var d, v; const l = (((d = o.colliderMeshes) === null || d === void 0 ? void 0 : d.indexOf((v = i == null ? void 0 : i.pickInfo) === null || v === void 0 ? void 0 : v.pickedMesh)) != -1 || o.active) && o.dragBehavior.enabled ? o.hoverMaterial : o.disableMaterial; o.gizmoMeshes.forEach((P) => { P.material = l, P.color && (P.color = l.diffuseColor); }); }); } i.type === ir.POINTERUP && t.forEach((f) => { f.active = !1, r = !1, f.gizmoMeshes.forEach((o) => { o.material = f.dragBehavior.enabled ? f.material : f.disableMaterial, o.color && (o.color = f.material.diffuseColor); }); }); } }); } /** * Disposes of the gizmo */ dispose() { this._rootMesh.dispose(), this._beforeRenderObserver && this.gizmoLayer.utilityLayerScene.onBeforeRenderObservable.remove(this._beforeRenderObserver); } } Lo.PreserveScaling = !1; Lo.UseAbsoluteScaling = !0; class lp extends Lo { /** Default material used to render when gizmo is not disabled or hovered */ get coloredMaterial() { return this._coloredMaterial; } /** Material used to render when gizmo is hovered with mouse*/ get hoverMaterial() { return this._hoverMaterial; } /** Material used to render when gizmo is disabled. typically grey.*/ get disableMaterial() { return this._disableMaterial; } /** * @internal */ static _CreateArrow(e, t, r = 1, n = !1) { const i = new Hr("arrow", e), s = Ld("cylinder", { diameterTop: 0, height: 0.075, diameterBottom: 0.0375 * (1 + (r - 1) / 4), tessellation: 96 }, e), a = Ld("cylinder", { diameterTop: 5e-3 * r, height: 0.275, diameterBottom: 5e-3 * r, tessellation: 96 }, e); return s.parent = i, s.material = t, s.rotation.x = Math.PI / 2, s.position.z += 0.3, a.parent = i, a.material = t, a.position.z += 0.275 / 2, a.rotation.x = Math.PI / 2, n && (a.visibility = 0, s.visibility = 0), i; } /** * @internal */ static _CreateArrowInstance(e, t) { const r = new Hr("arrow", e); for (const n of t.getChildMeshes()) { const i = n.createInstance(n.name); i.parent = r; } return r; } /** * Creates an AxisDragGizmo * @param dragAxis The axis which the gizmo will be able to drag on * @param color The color of the gizmo * @param gizmoLayer The utility layer the gizmo will be added to * @param parent * @param thickness display gizmo axis thickness * @param hoverColor The color of the gizmo when hovering over and dragging * @param disableColor The Color of the gizmo when its disabled */ constructor(e, t = Ne.Gray(), r = Ds.DefaultUtilityLayer, n = null, i = 1, s = Ne.Yellow(), a = Ne.Gray()) { var f; super(r), this._pointerObserver = null, this.snapDistance = 0, this.onSnapObservable = new Oe(), this._isEnabled = !0, this._parent = null, this._dragging = !1, this._parent = n, this._coloredMaterial = new Wt("", r.utilityLayerScene), this._coloredMaterial.diffuseColor = t, this._coloredMaterial.specularColor = t.subtract(new Ne(0.1, 0.1, 0.1)), this._hoverMaterial = new Wt("", r.utilityLayerScene), this._hoverMaterial.diffuseColor = s, this._disableMaterial = new Wt("", r.utilityLayerScene), this._disableMaterial.diffuseColor = a, this._disableMaterial.alpha = 0.4; const o = lp._CreateArrow(r.utilityLayerScene, this._coloredMaterial, i), d = lp._CreateArrow(r.utilityLayerScene, this._coloredMaterial, i + 4, !0); this._gizmoMesh = new Ee("", r.utilityLayerScene), this._gizmoMesh.addChild(o), this._gizmoMesh.addChild(d), this._gizmoMesh.lookAt(this._rootMesh.position.add(e)), this._gizmoMesh.scaling.scaleInPlace(1 / 3), this._gizmoMesh.parent = this._rootMesh; let v = 0; const u = { snapDistance: 0 }; this.dragBehavior = new O9({ dragAxis: e }), this.dragBehavior.moveAttached = !1, this.dragBehavior.updateDragPlane = !1, this._rootMesh.addBehavior(this.dragBehavior), this.dragBehavior.onDragObservable.add((p) => { if (this.attachedNode) { let c = !1; if (this.snapDistance == 0) this.attachedNode.getWorldMatrix().getTranslationToRef(ue.Vector3[2]), ue.Vector3[2].addInPlace(p.delta), this.dragBehavior.validateDrag(ue.Vector3[2]) && (this.attachedNode.position && this.attachedNode.position.addInPlaceFromFloats(p.delta.x, p.delta.y, p.delta.z), this.attachedNode.getWorldMatrix().addTranslationFromFloats(p.delta.x, p.delta.y, p.delta.z), this.attachedNode.updateCache(), c = !0); else if (v += p.dragDistance, Math.abs(v) > this.snapDistance) { const H = Math.floor(Math.abs(v) / this.snapDistance); v = v % this.snapDistance, p.delta.normalizeToRef(ue.Vector3[1]), ue.Vector3[1].scaleInPlace(this.snapDistance * H), this.attachedNode.getWorldMatrix().getTranslationToRef(ue.Vector3[2]), ue.Vector3[2].addInPlace(ue.Vector3[1]), this.dragBehavior.validateDrag(ue.Vector3[2]) && (this.attachedNode.getWorldMatrix().addTranslationFromFloats(ue.Vector3[1].x, ue.Vector3[1].y, ue.Vector3[1].z), this.attachedNode.updateCache(), u.snapDistance = this.snapDistance * H * Math.sign(v), this.onSnapObservable.notifyObservers(u), c = !0); } c && this._matrixChanged(); } }), this.dragBehavior.onDragStartObservable.add(() => { this._dragging = !0; }), this.dragBehavior.onDragEndObservable.add(() => { this._dragging = !1; }); const l = r._getSharedGizmoLight(); l.includedOnlyMeshes = l.includedOnlyMeshes.concat(this._rootMesh.getChildMeshes(!1)); const P = { gizmoMeshes: o.getChildMeshes(), colliderMeshes: d.getChildMeshes(), material: this._coloredMaterial, hoverMaterial: this._hoverMaterial, disableMaterial: this._disableMaterial, active: !1, dragBehavior: this.dragBehavior }; (f = this._parent) === null || f === void 0 || f.addToAxisCache(d, P), this._pointerObserver = r.utilityLayerScene.onPointerObservable.add((p) => { var c; if (!this._customMeshSet && (this._isHovered = P.colliderMeshes.indexOf((c = p == null ? void 0 : p.pickInfo) === null || c === void 0 ? void 0 : c.pickedMesh) != -1, !this._parent)) { const H = this.dragBehavior.enabled ? this._isHovered || this._dragging ? this._hoverMaterial : this._coloredMaterial : this._disableMaterial; this._setGizmoMeshMaterial(P.gizmoMeshes, H); } }), this.dragBehavior.onEnabledObservable.add((p) => { this._setGizmoMeshMaterial(P.gizmoMeshes, p ? P.material : P.disableMaterial); }); } _attachedNodeChanged(e) { this.dragBehavior && (this.dragBehavior.enabled = !!e); } /** * If the gizmo is enabled */ set isEnabled(e) { this._isEnabled = e, e ? this._parent && (this.attachedMesh = this._parent.attachedMesh, this.attachedNode = this._parent.attachedNode) : (this.attachedMesh = null, this.attachedNode = null); } get isEnabled() { return this._isEnabled; } /** * Disposes of the gizmo */ dispose() { this.onSnapObservable.clear(), this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(this._pointerObserver), this.dragBehavior.detach(), this._gizmoMesh && this._gizmoMesh.dispose(), [this._coloredMaterial, this._hoverMaterial, this._disableMaterial].forEach((e) => { e && e.dispose(); }), super.dispose(); } } class aq { /** * Gets or sets a number used to scale line length */ get scaleLines() { return this._scaleLines; } set scaleLines(e) { this._scaleLines = e, this._xAxis.scaling.setAll(this._scaleLines * this._scaleLinesFactor), this._yAxis.scaling.setAll(this._scaleLines * this._scaleLinesFactor), this._zAxis.scaling.setAll(this._scaleLines * this._scaleLinesFactor); } /** Gets the node hierarchy used to render x-axis */ get xAxis() { return this._xAxis; } /** Gets the node hierarchy used to render y-axis */ get yAxis() { return this._yAxis; } /** Gets the node hierarchy used to render z-axis */ get zAxis() { return this._zAxis; } /** * Creates a new AxesViewer * @param scene defines the hosting scene * @param scaleLines defines a number used to scale line length (1 by default) * @param renderingGroupId defines a number used to set the renderingGroupId of the meshes (2 by default) * @param xAxis defines the node hierarchy used to render the x-axis * @param yAxis defines the node hierarchy used to render the y-axis * @param zAxis defines the node hierarchy used to render the z-axis * @param lineThickness The line thickness to use when creating the arrow. defaults to 1. */ constructor(e, t = 1, r = 2, n, i, s, a = 1) { if (this._scaleLinesFactor = 4, this._instanced = !1, this.scene = null, this._scaleLines = 1, e = e || gr.LastCreatedScene, !!e) { if (!n) { const f = new Wt("xAxisMaterial", e); f.disableLighting = !0, f.emissiveColor = Ne.Red().scale(0.5), n = lp._CreateArrow(e, f, a); } if (!i) { const f = new Wt("yAxisMaterial", e); f.disableLighting = !0, f.emissiveColor = Ne.Green().scale(0.5), i = lp._CreateArrow(e, f, a); } if (!s) { const f = new Wt("zAxisMaterial", e); f.disableLighting = !0, f.emissiveColor = Ne.Blue().scale(0.5), s = lp._CreateArrow(e, f, a); } this._xAxis = n, this._yAxis = i, this._zAxis = s, this.scaleLines = t, r != null && (aq._SetRenderingGroupId(this._xAxis, r), aq._SetRenderingGroupId(this._yAxis, r), aq._SetRenderingGroupId(this._zAxis, r)), this.scene = e, this.update(new S(), S.Right(), S.Up(), S.Forward()); } } /** * Force the viewer to update * @param position defines the position of the viewer * @param xaxis defines the x axis of the viewer * @param yaxis defines the y axis of the viewer * @param zaxis defines the z axis of the viewer */ update(e, t, r, n) { this._xAxis.position.copyFrom(e), this._xAxis.setDirection(t), this._yAxis.position.copyFrom(e), this._yAxis.setDirection(r), this._zAxis.position.copyFrom(e), this._zAxis.setDirection(n); } /** * Creates an instance of this axes viewer. * @returns a new axes viewer with instanced meshes */ createInstance() { const e = lp._CreateArrowInstance(this.scene, this._xAxis), t = lp._CreateArrowInstance(this.scene, this._yAxis), r = lp._CreateArrowInstance(this.scene, this._zAxis), n = new aq(this.scene, this.scaleLines, null, e, t, r); return n._instanced = !0, n; } /** Releases resources */ dispose() { this._xAxis && this._xAxis.dispose(!1, !this._instanced), this._yAxis && this._yAxis.dispose(!1, !this._instanced), this._zAxis && this._zAxis.dispose(!1, !this._instanced), this.scene = null; } static _SetRenderingGroupId(e, t) { e.getChildMeshes().forEach((r) => { r.renderingGroupId = t; }); } } class Bve extends aq { /** * Creates a new BoneAxesViewer * @param scene defines the hosting scene * @param bone defines the target bone * @param mesh defines the target mesh * @param scaleLines defines a scaling factor for line length (1 by default) */ constructor(e, t, r, n = 1) { super(e, n), this.pos = S.Zero(), this.xaxis = S.Zero(), this.yaxis = S.Zero(), this.zaxis = S.Zero(), this.mesh = r, this.bone = t; } /** * Force the viewer to update */ update() { if (!this.mesh || !this.bone) return; const e = this.bone; e.getAbsolutePositionToRef(this.mesh, this.pos), e.getDirectionToRef(bf.X, this.mesh, this.xaxis), e.getDirectionToRef(bf.Y, this.mesh, this.yaxis), e.getDirectionToRef(bf.Z, this.mesh, this.zaxis), super.update(this.pos, this.xaxis, this.yaxis, this.zaxis); } /** Releases resources */ dispose() { this.mesh && (this.mesh = null, this.bone = null, super.dispose()); } } Object.defineProperty(sr.prototype, "debugLayer", { get: function() { return this._debugLayer || (this._debugLayer = new pm(this)), this._debugLayer; }, enumerable: !0, configurable: !0 }); var rF; (function(A) { A[A.Properties = 0] = "Properties", A[A.Debug = 1] = "Debug", A[A.Statistics = 2] = "Statistics", A[A.Tools = 3] = "Tools", A[A.Settings = 4] = "Settings"; })(rF || (rF = {})); class pm { /** * Observable triggered when a property is changed through the inspector. */ get onPropertyChangedObservable() { return this.BJSINSPECTOR && this.BJSINSPECTOR.Inspector ? this.BJSINSPECTOR.Inspector.OnPropertyChangedObservable : (this._onPropertyChangedObservable || (this._onPropertyChangedObservable = new Oe()), this._onPropertyChangedObservable); } /** * Observable triggered when the selection is changed through the inspector. */ get onSelectionChangedObservable() { return this.BJSINSPECTOR && this.BJSINSPECTOR.Inspector ? this.BJSINSPECTOR.Inspector.OnSelectionChangeObservable : (this._onSelectionChangedObservable || (this._onSelectionChangedObservable = new Oe()), this._onSelectionChangedObservable); } /** * Instantiates a new debug layer. * The debug layer (aka Inspector) is the go to tool in order to better understand * what is happening in your scene * @see https://doc.babylonjs.com/toolsAndResources/inspector * @param scene Defines the scene to inspect */ constructor(e) { this.BJSINSPECTOR = this._getGlobalInspector(), this._scene = e || gr.LastCreatedScene, this._scene && this._scene.onDisposeObservable.add(() => { this._scene._debugLayer && this._scene._debugLayer.hide(); }); } /** * Creates the inspector window. * @param config */ _createInspector(e) { if (this.isVisible()) return; if (this._onPropertyChangedObservable) { for (const r of this._onPropertyChangedObservable.observers) this.BJSINSPECTOR.Inspector.OnPropertyChangedObservable.add(r); this._onPropertyChangedObservable.clear(), this._onPropertyChangedObservable = void 0; } if (this._onSelectionChangedObservable) { for (const r of this._onSelectionChangedObservable.observers) this.BJSINSPECTOR.Inspector.OnSelectionChangedObservable.add(r); this._onSelectionChangedObservable.clear(), this._onSelectionChangedObservable = void 0; } const t = Object.assign(Object.assign({}, pm.Config), e); this.BJSINSPECTOR = this.BJSINSPECTOR || this._getGlobalInspector(), this.BJSINSPECTOR.Inspector.Show(this._scene, t); } /** * Select a specific entity in the scene explorer and highlight a specific block in that entity property grid * @param entity defines the entity to select * @param lineContainerTitles defines the specific blocks to highlight (could be a string or an array of strings) */ select(e, t) { this.BJSINSPECTOR && (t && (Object.prototype.toString.call(t) == "[object String]" ? this.BJSINSPECTOR.Inspector.MarkLineContainerTitleForHighlighting(t) : this.BJSINSPECTOR.Inspector.MarkMultipleLineContainerTitlesForHighlighting(t)), this.BJSINSPECTOR.Inspector.OnSelectionChangeObservable.notifyObservers(e)); } /** Get the inspector from bundle or global */ _getGlobalInspector() { if (typeof INSPECTOR < "u") return INSPECTOR; if (typeof BABYLON < "u" && typeof BABYLON.Inspector < "u") return BABYLON; } /** * Get if the inspector is visible or not. * @returns true if visible otherwise, false */ isVisible() { return this.BJSINSPECTOR && this.BJSINSPECTOR.Inspector.IsVisible; } /** * Hide the inspector and close its window. */ hide() { this.BJSINSPECTOR && this.BJSINSPECTOR.Inspector.Hide(); } /** * Update the scene in the inspector */ setAsActiveScene() { this.BJSINSPECTOR && this.BJSINSPECTOR.Inspector._SetNewScene(this._scene); } /** * Launch the debugLayer. * @param config Define the configuration of the inspector * @returns a promise fulfilled when the debug layer is visible */ show(e) { return new Promise((t) => { if (typeof this.BJSINSPECTOR > "u") { const r = e && e.inspectorURL ? e.inspectorURL : pm.InspectorURL; ye.LoadBabylonScript(r, () => { this._createInspector(e), t(this); }); } else this._createInspector(e), t(this); }); } } pm.InspectorURL = `${ye._DefaultCdnUrl}/v${Ge.Version}/inspector/babylon.inspector.bundle.js`; pm.Config = { overlay: !1, showExplorer: !0, showInspector: !0, embedMode: !1, handleResize: !0, enablePopup: !0 }; function LO(A) { let t = [0, 1, 2, 0, 2, 3, 4, 5, 6, 4, 6, 7, 8, 9, 10, 8, 10, 11, 12, 13, 14, 12, 14, 15, 16, 17, 18, 16, 18, 19, 20, 21, 22, 20, 22, 23]; const r = [ 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0 ], n = []; let i = []; const s = A.width || A.size || 1, a = A.height || A.size || 1, f = A.depth || A.size || 1, o = A.wrap || !1; let d = A.topBaseAt === void 0 ? 1 : A.topBaseAt, v = A.bottomBaseAt === void 0 ? 0 : A.bottomBaseAt; d = (d + 4) % 4, v = (v + 4) % 4; const u = [2, 0, 3, 1], l = [2, 0, 1, 3]; let P = u[d], p = l[v], c = [ 1, -1, 1, -1, -1, 1, -1, 1, 1, 1, 1, 1, 1, 1, -1, -1, 1, -1, -1, -1, -1, 1, -1, -1, 1, 1, -1, 1, -1, -1, 1, -1, 1, 1, 1, 1, -1, 1, 1, -1, -1, 1, -1, -1, -1, -1, 1, -1, -1, 1, 1, -1, 1, -1, 1, 1, -1, 1, 1, 1, 1, -1, 1, 1, -1, -1, -1, -1, -1, -1, -1, 1 ]; if (o) { t = [2, 3, 0, 2, 0, 1, 4, 5, 6, 4, 6, 7, 9, 10, 11, 9, 11, 8, 12, 14, 15, 12, 13, 14], c = [ -1, 1, 1, 1, 1, 1, 1, -1, 1, -1, -1, 1, 1, 1, -1, -1, 1, -1, -1, -1, -1, 1, -1, -1, 1, 1, 1, 1, 1, -1, 1, -1, -1, 1, -1, 1, -1, 1, -1, -1, 1, 1, -1, -1, 1, -1, -1, -1 ]; let m = [ [1, 1, 1], [-1, 1, 1], [-1, 1, -1], [1, 1, -1] ], I = [ [-1, -1, 1], [1, -1, 1], [1, -1, -1], [-1, -1, -1] ]; const N = [17, 18, 19, 16], k = [22, 23, 20, 21]; for (; P > 0; ) m.unshift(m.pop()), N.unshift(N.pop()), P--; for (; p > 0; ) I.unshift(I.pop()), k.unshift(k.pop()), p--; m = m.flat(), I = I.flat(), c = c.concat(m).concat(I), t.push(N[0], N[2], N[3], N[0], N[1], N[2]), t.push(k[0], k[2], k[3], k[0], k[1], k[2]); } const H = [s / 2, a / 2, f / 2]; i = c.reduce((m, I, N) => m.concat(I * H[N % 3]), []); const T = A.sideOrientation === 0 ? 0 : A.sideOrientation || Ut.DEFAULTSIDE, q = A.faceUV || new Array(6), b = A.faceColors, j = []; for (let m = 0; m < 6; m++) q[m] === void 0 && (q[m] = new Ir(0, 0, 1, 1)), b && b[m] === void 0 && (b[m] = new xt(1, 1, 1, 1)); for (let m = 0; m < 6; m++) if (n.push(q[m].z, us.UseOpenGLOrientationForUV ? 1 - q[m].w : q[m].w), n.push(q[m].x, us.UseOpenGLOrientationForUV ? 1 - q[m].w : q[m].w), n.push(q[m].x, us.UseOpenGLOrientationForUV ? 1 - q[m].y : q[m].y), n.push(q[m].z, us.UseOpenGLOrientationForUV ? 1 - q[m].y : q[m].y), b) for (let I = 0; I < 4; I++) j.push(b[m].r, b[m].g, b[m].b, b[m].a); Ut._ComputeSides(T, i, t, r, n, A.frontUVs, A.backUVs); const w = new Ut(); if (w.indices = t, w.positions = i, w.normals = r, w.uvs = n, b) { const m = T === Ut.DOUBLESIDE ? j.concat(j) : j; w.colors = m; } return w; } function $ee(A) { const e = A.width || A.size || 1, t = A.height || A.size || 1, r = A.depth || A.size || 1, n = (A.widthSegments || A.segments || 1) | 0, i = (A.heightSegments || A.segments || 1) | 0, s = (A.depthSegments || A.segments || 1) | 0, a = new he(), f = new he(), o = new he(), d = y2({ width: e, height: r, subdivisionsX: n, subdivisionsY: s }); he.TranslationToRef(0, -t / 2, 0, f), he.RotationZToRef(Math.PI, a), a.multiplyToRef(f, o), d.transform(o); const v = y2({ width: e, height: r, subdivisionsX: n, subdivisionsY: s }); he.TranslationToRef(0, t / 2, 0, o), v.transform(o); const u = y2({ width: t, height: r, subdivisionsX: i, subdivisionsY: s }); he.TranslationToRef(-e / 2, 0, 0, f), he.RotationZToRef(Math.PI / 2, a), a.multiplyToRef(f, o), u.transform(o); const l = y2({ width: t, height: r, subdivisionsX: i, subdivisionsY: s }); he.TranslationToRef(e / 2, 0, 0, f), he.RotationZToRef(-Math.PI / 2, a), a.multiplyToRef(f, o), l.transform(o); const P = y2({ width: e, height: t, subdivisionsX: n, subdivisionsY: i }); he.TranslationToRef(0, 0, -r / 2, f), he.RotationXToRef(-Math.PI / 2, a), a.multiplyToRef(f, o), P.transform(o); const p = y2({ width: e, height: t, subdivisionsX: n, subdivisionsY: i }); return he.TranslationToRef(0, 0, r / 2, f), he.RotationXToRef(Math.PI / 2, a), a.multiplyToRef(f, o), p.transform(o), d.merge([v, l, u, P, p], !0), d; } function k0(A, e = {}, t = null) { const r = new Ee(A, t); return e.sideOrientation = Ee._GetDefaultSideOrientation(e.sideOrientation), r._originalBuilderSideOrientation = e.sideOrientation, LO(e).applyToMesh(r, e.updatable), r; } const Wve = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateBox: k0 }; Ut.CreateBox = LO; Ee.CreateBox = (A, e, t = null, r, n) => k0(A, { size: e, sideOrientation: n, updatable: r }, t); function KO(A) { const e = (A.segments || 32) | 0, t = A.diameterX || A.diameter || 1, r = A.diameterY || A.diameter || 1, n = A.diameterZ || A.diameter || 1, i = A.arc && (A.arc <= 0 || A.arc > 1) ? 1 : A.arc || 1, s = A.slice && A.slice <= 0 ? 1 : A.slice || 1, a = A.sideOrientation === 0 ? 0 : A.sideOrientation || Ut.DEFAULTSIDE, f = !!A.dedupTopBottomIndices, o = new S(t / 2, r / 2, n / 2), d = 2 + e, v = 2 * d, u = [], l = [], P = [], p = []; for (let H = 0; H <= d; H++) { const T = H / d, q = T * Math.PI * s; for (let b = 0; b <= v; b++) { const j = b / v, w = j * Math.PI * 2 * i, m = he.RotationZ(-q), I = he.RotationY(w), N = S.TransformCoordinates(S.Up(), m), k = S.TransformCoordinates(N, I), R = k.multiply(o), y = k.divide(o).normalize(); l.push(R.x, R.y, R.z), P.push(y.x, y.y, y.z), p.push(j, us.UseOpenGLOrientationForUV ? 1 - T : T); } if (H > 0) { const b = l.length / 3; for (let j = b - 2 * (v + 1); j + v + 2 < b; j++) f ? (H > 1 && (u.push(j), u.push(j + 1), u.push(j + v + 1)), (H < d || s < 1) && (u.push(j + v + 1), u.push(j + 1), u.push(j + v + 2))) : (u.push(j), u.push(j + 1), u.push(j + v + 1), u.push(j + v + 1), u.push(j + 1), u.push(j + v + 2)); } } Ut._ComputeSides(a, l, u, P, p, A.frontUVs, A.backUVs); const c = new Ut(); return c.indices = u, c.positions = l, c.normals = P, c.uvs = p, c; } function UA(A, e = {}, t = null) { const r = new Ee(A, t); return e.sideOrientation = Ee._GetDefaultSideOrientation(e.sideOrientation), r._originalBuilderSideOrientation = e.sideOrientation, KO(e).applyToMesh(r, e.updatable), r; } const Sve = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateSphere: UA }; Ut.CreateSphere = KO; Ee.CreateSphere = (A, e, t, r, n, i) => UA(A, { segments: e, diameterX: t, diameterY: t, diameterZ: t, sideOrientation: i, updatable: n }, r); function JO(A = { subdivisions: 2, tessellation: 16, height: 1, radius: 0.25, capSubdivisions: 6 }) { const e = Math.max(A.subdivisions ? A.subdivisions : 2, 1) | 0, t = Math.max(A.tessellation ? A.tessellation : 16, 3) | 0, r = Math.max(A.height ? A.height : 1, 0), n = Math.max(A.radius ? A.radius : 0.25, 0), i = Math.max(A.capSubdivisions ? A.capSubdivisions : 6, 1) | 0, s = t, a = e, f = Math.max(A.radiusTop ? A.radiusTop : n, 0), o = Math.max(A.radiusBottom ? A.radiusBottom : n, 0), d = r - (f + o), v = 0, u = 2 * Math.PI, l = Math.max(A.topCapSubdivisions ? A.topCapSubdivisions : i, 1), P = Math.max(A.bottomCapSubdivisions ? A.bottomCapSubdivisions : i, 1), p = Math.acos((o - f) / r); let c = []; const H = [], T = [], q = []; let b = 0; const j = [], w = d * 0.5, m = Math.PI * 0.5; let I, N; const k = S.Zero(), R = S.Zero(), y = Math.cos(p), O = Math.sin(p), Y = new at(f * O, w + f * y).subtract(new at(o * O, -w + o * y)).length(), ee = f * p + Y + o * (m - p); let Z = 0; for (N = 0; N <= l; N++) { const G = [], L = m - p * (N / l); Z += f * p / l; const $ = Math.cos(L), ae = Math.sin(L), Pe = $ * f; for (I = 0; I <= s; I++) { const ge = I / s, me = ge * u + v, Xe = Math.sin(me), De = Math.cos(me); R.x = Pe * Xe, R.y = w + ae * f, R.z = Pe * De, H.push(R.x, R.y, R.z), k.set($ * Xe, ae, $ * De), T.push(k.x, k.y, k.z), q.push(ge, us.UseOpenGLOrientationForUV ? Z / ee : 1 - Z / ee), G.push(b), b++; } j.push(G); } const te = r - f - o + y * f - y * o, fe = O * (o - f) / te; for (N = 1; N <= a; N++) { const G = []; Z += Y / a; const L = O * (N * (o - f) / a + f); for (I = 0; I <= s; I++) { const $ = I / s, ae = $ * u + v, Pe = Math.sin(ae), ge = Math.cos(ae); R.x = L * Pe, R.y = w + y * f - N * te / a, R.z = L * ge, H.push(R.x, R.y, R.z), k.set(Pe, fe, ge).normalize(), T.push(k.x, k.y, k.z), q.push($, us.UseOpenGLOrientationForUV ? Z / ee : 1 - Z / ee), G.push(b), b++; } j.push(G); } for (N = 1; N <= P; N++) { const G = [], L = m - p - (Math.PI - p) * (N / P); Z += o * p / P; const $ = Math.cos(L), ae = Math.sin(L), Pe = $ * o; for (I = 0; I <= s; I++) { const ge = I / s, me = ge * u + v, Xe = Math.sin(me), De = Math.cos(me); R.x = Pe * Xe, R.y = -w + ae * o, R.z = Pe * De, H.push(R.x, R.y, R.z), k.set($ * Xe, ae, $ * De), T.push(k.x, k.y, k.z), q.push(ge, us.UseOpenGLOrientationForUV ? Z / ee : 1 - Z / ee), G.push(b), b++; } j.push(G); } for (I = 0; I < s; I++) for (N = 0; N < l + a + P; N++) { const G = j[N][I], L = j[N + 1][I], $ = j[N + 1][I + 1], ae = j[N][I + 1]; c.push(G), c.push(L), c.push(ae), c.push(L), c.push($), c.push(ae); } if (c = c.reverse(), A.orientation && !A.orientation.equals(S.Up())) { const G = new he(); A.orientation.clone().scale(Math.PI * 0.5).cross(S.Up()).toQuaternion().toRotationMatrix(G); const L = S.Zero(); for (let $ = 0; $ < H.length; $ += 3) L.set(H[$], H[$ + 1], H[$ + 2]), S.TransformCoordinatesToRef(L.clone(), G, L), H[$] = L.x, H[$ + 1] = L.y, H[$ + 2] = L.z; } const _ = new Ut(); return _.positions = H, _.normals = T, _.uvs = q, _.indices = c, _; } function KR(A, e = { orientation: S.Up(), subdivisions: 2, tessellation: 16, height: 1, radius: 0.25, capSubdivisions: 6, updatable: !1 }, t = null) { const r = new Ee(A, t); return JO(e).applyToMesh(r, e.updatable), r; } const Uve = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateCapsule: KR }; Ee.CreateCapsule = (A, e, t) => KR(A, e, t); Ut.CreateCapsule = JO; function jQ(A) { let e = A.pathArray; const t = A.closeArray || !1, r = A.closePath || !1, n = A.invertUV || !1, i = Math.floor(e[0].length / 2); let s = A.offset || i; s = s > i ? i : Math.floor(s); const a = A.sideOrientation === 0 ? 0 : A.sideOrientation || Ut.DEFAULTSIDE, f = A.uvs, o = A.colors, d = [], v = [], u = [], l = [], P = [], p = [], c = [], H = []; let T; const q = [], b = []; let j, w, m; if (e.length < 2) { const ve = [], qe = []; for (w = 0; w < e[0].length - s; w++) ve.push(e[0][w]), qe.push(e[0][w + s]); e = [ve, qe]; } let I = 0; const N = r ? 1 : 0; let k, R; T = e[0].length; let y, O; for (j = 0; j < e.length; j++) { for (c[j] = 0, P[j] = [0], k = e[j], R = k.length, T = T < R ? T : R, m = 0; m < R; ) d.push(k[m].x, k[m].y, k[m].z), m > 0 && (y = k[m].subtract(k[m - 1]).length(), O = y + c[j], P[j].push(O), c[j] = O), m++; r && (m--, d.push(k[0].x, k[0].y, k[0].z), y = k[m].subtract(k[0]).length(), O = y + c[j], P[j].push(O), c[j] = O), q[j] = R + N, b[j] = I, I += R + N; } let Y, ee, Z = null, te = null; for (w = 0; w < T + N; w++) { for (H[w] = 0, p[w] = [0], j = 0; j < e.length - 1; j++) Y = e[j], ee = e[j + 1], w === T ? (Z = Y[0], te = ee[0]) : (Z = Y[w], te = ee[w]), y = te.subtract(Z).length(), O = y + H[w], p[w].push(O), H[w] = O; t && te && Z && (Y = e[j], ee = e[0], w === T && (te = ee[0]), y = te.subtract(Z).length(), O = y + H[w], H[w] = O); } let fe, _; if (f) for (j = 0; j < f.length; j++) l.push(f[j].x, us.UseOpenGLOrientationForUV ? 1 - f[j].y : f[j].y); else for (j = 0; j < e.length; j++) for (w = 0; w < T + N; w++) fe = c[j] != 0 ? P[j][w] / c[j] : 0, _ = H[w] != 0 ? p[w][j] / H[w] : 0, n ? l.push(_, fe) : l.push(fe, us.UseOpenGLOrientationForUV ? 1 - _ : _); j = 0; let G = 0, L = q[j] - 1, $ = q[j + 1] - 1, ae = L < $ ? L : $, Pe = b[1] - b[0]; const ge = t ? q.length : q.length - 1; for (; G <= ae && j < ge; ) v.push(G, G + Pe, G + 1), v.push(G + Pe + 1, G + 1, G + Pe), G += 1, G === ae && (j++, j === q.length - 1 ? (Pe = b[0] - b[j], L = q[j] - 1, $ = q[0] - 1) : (Pe = b[j + 1] - b[j], L = q[j] - 1, $ = q[j + 1] - 1), G = b[j], ae = L < $ ? L + G : $ + G); if (Ut.ComputeNormals(d, v, u), r) { let ve = 0, qe = 0; for (j = 0; j < e.length; j++) ve = b[j] * 3, j + 1 < e.length ? qe = (b[j + 1] - 1) * 3 : qe = u.length - 3, u[ve] = (u[ve] + u[qe]) * 0.5, u[ve + 1] = (u[ve + 1] + u[qe + 1]) * 0.5, u[ve + 2] = (u[ve + 2] + u[qe + 2]) * 0.5, u[qe] = u[ve], u[qe + 1] = u[ve + 1], u[qe + 2] = u[ve + 2]; } Ut._ComputeSides(a, d, v, u, l, A.frontUVs, A.backUVs); let me = null; if (o) { me = new Float32Array(o.length * 4); for (let ve = 0; ve < o.length; ve++) me[ve * 4] = o[ve].r, me[ve * 4 + 1] = o[ve].g, me[ve * 4 + 2] = o[ve].b, me[ve * 4 + 3] = o[ve].a; } const Xe = new Ut(), De = new Float32Array(d), ne = new Float32Array(u), re = new Float32Array(l); return Xe.indices = v, Xe.positions = De, Xe.normals = ne, Xe.uvs = re, me && Xe.set(me, J.ColorKind), r && (Xe._idx = b), Xe; } function n4(A, e, t = null) { const r = e.pathArray, n = e.closeArray, i = e.closePath, s = Ee._GetDefaultSideOrientation(e.sideOrientation), a = e.instance, f = e.updatable; if (a) { const o = ue.Vector3[0].setAll(Number.MAX_VALUE), d = ue.Vector3[1].setAll(-Number.MAX_VALUE), v = (l) => { let P = r[0].length; const p = a; let c = 0; const H = p._originalBuilderSideOrientation === Ee.DOUBLESIDE ? 2 : 1; for (let T = 1; T <= H; ++T) for (let q = 0; q < r.length; ++q) { const b = r[q], j = b.length; P = P < j ? P : j; for (let w = 0; w < P; ++w) { const m = b[w]; l[c] = m.x, l[c + 1] = m.y, l[c + 2] = m.z, o.minimizeInPlaceFromFloats(m.x, m.y, m.z), d.maximizeInPlaceFromFloats(m.x, m.y, m.z), c += 3; } if (p._creationDataStorage && p._creationDataStorage.closePath) { const w = b[0]; l[c] = w.x, l[c + 1] = w.y, l[c + 2] = w.z, c += 3; } } }, u = a.getVerticesData(J.PositionKind); if (v(u), a.hasBoundingInfo ? a.getBoundingInfo().reConstruct(o, d, a._worldMatrix) : a.buildBoundingInfo(o, d, a._worldMatrix), a.updateVerticesData(J.PositionKind, u, !1, !1), e.colors) { const l = a.getVerticesData(J.ColorKind); for (let P = 0, p = 0; P < e.colors.length; P++, p += 4) { const c = e.colors[P]; l[p] = c.r, l[p + 1] = c.g, l[p + 2] = c.b, l[p + 3] = c.a; } a.updateVerticesData(J.ColorKind, l, !1, !1); } if (e.uvs) { const l = a.getVerticesData(J.UVKind); for (let P = 0; P < e.uvs.length; P++) l[P * 2] = e.uvs[P].x, l[P * 2 + 1] = us.UseOpenGLOrientationForUV ? 1 - e.uvs[P].y : e.uvs[P].y; a.updateVerticesData(J.UVKind, l, !1, !1); } if (!a.areNormalsFrozen || a.isFacetDataEnabled) { const l = a.getIndices(), P = a.getVerticesData(J.NormalKind), p = a.isFacetDataEnabled ? a.getFacetDataParameters() : null; if (Ut.ComputeNormals(u, l, P, p), a._creationDataStorage && a._creationDataStorage.closePath) { let c = 0, H = 0; for (let T = 0; T < r.length; T++) c = a._creationDataStorage.idx[T] * 3, T + 1 < r.length ? H = (a._creationDataStorage.idx[T + 1] - 1) * 3 : H = P.length - 3, P[c] = (P[c] + P[H]) * 0.5, P[c + 1] = (P[c + 1] + P[H + 1]) * 0.5, P[c + 2] = (P[c + 2] + P[H + 2]) * 0.5, P[H] = P[c], P[H + 1] = P[c + 1], P[H + 2] = P[c + 2]; } a.areNormalsFrozen || a.updateVerticesData(J.NormalKind, P, !1, !1); } return a; } else { const o = new Ee(A, t); o._originalBuilderSideOrientation = s, o._creationDataStorage = new tQ(); const d = jQ(e); return i && (o._creationDataStorage.idx = d._idx), o._creationDataStorage.closePath = i, o._creationDataStorage.closeArray = n, d.applyToMesh(o, f), o; } } const Ive = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateRibbon: n4 }; Ut.CreateRibbon = jQ; Ee.CreateRibbon = (A, e, t = !1, r, n, i, s = !1, a, f) => n4(A, { pathArray: e, closeArray: t, closePath: r, offset: n, updatable: s, sideOrientation: a, instance: f }, i); function zO(A) { const e = [], t = [], r = [], n = [], i = A.radius || 0.5, s = A.tessellation || 64, a = A.arc && (A.arc <= 0 || A.arc > 1) ? 1 : A.arc || 1, f = A.sideOrientation === 0 ? 0 : A.sideOrientation || Ut.DEFAULTSIDE; e.push(0, 0, 0), n.push(0.5, 0.5); const o = Math.PI * 2 * a, d = a === 1 ? o / s : o / (s - 1); let v = 0; for (let P = 0; P < s; P++) { const p = Math.cos(v), c = Math.sin(v), H = (p + 1) / 2, T = (1 - c) / 2; e.push(i * p, i * c, 0), n.push(H, us.UseOpenGLOrientationForUV ? 1 - T : T), v += d; } a === 1 && (e.push(e[3], e[4], e[5]), n.push(n[2], us.UseOpenGLOrientationForUV ? 1 - n[3] : n[3])); const u = e.length / 3; for (let P = 1; P < u - 1; P++) t.push(P + 1, 0, P); Ut.ComputeNormals(e, t, r), Ut._ComputeSides(f, e, t, r, n, A.frontUVs, A.backUVs); const l = new Ut(); return l.indices = t, l.positions = e, l.normals = r, l.uvs = n, l; } function uU(A, e = {}, t = null) { const r = new Ee(A, t); return e.sideOrientation = Ee._GetDefaultSideOrientation(e.sideOrientation), r._originalBuilderSideOrientation = e.sideOrientation, zO(e).applyToMesh(r, e.updatable), r; } const Rve = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateDisc: uU }; Ut.CreateDisc = zO; Ee.CreateDisc = (A, e, t, r = null, n, i) => uU(A, { radius: e, tessellation: t, sideOrientation: i, updatable: n }, r); function $W(A) { const e = A.pattern || Ee.NO_FLIP, t = A.tileWidth || A.tileSize || 1, r = A.tileHeight || A.tileSize || 1, n = A.alignHorizontal || 0, i = A.alignVertical || 0, s = A.width || A.size || 1, a = Math.floor(s / t); let f = s - a * t; const o = A.height || A.size || 1, d = Math.floor(o / r); let v = o - d * r; const u = t * a / 2, l = r * d / 2; let P = 0, p = 0, c = 0, H = 0, T = 0, q = 0; if (f > 0 || v > 0) { switch (c = -u, H = -l, T = u, q = l, n) { case Ee.CENTER: f /= 2, c -= f, T += f; break; case Ee.LEFT: T += f, P = -f / 2; break; case Ee.RIGHT: c -= f, P = f / 2; break; } switch (i) { case Ee.CENTER: v /= 2, H -= v, q += v; break; case Ee.BOTTOM: q += v, p = -v / 2; break; case Ee.TOP: H -= v, p = v / 2; break; } } const b = [], j = [], w = []; w[0] = [0, 0, 1, 0, 1, 1, 0, 1], w[1] = [0, 0, 1, 0, 1, 1, 0, 1], (e === Ee.ROTATE_TILE || e === Ee.ROTATE_ROW) && (w[1] = [1, 1, 0, 1, 0, 0, 1, 0]), (e === Ee.FLIP_TILE || e === Ee.FLIP_ROW) && (w[1] = [1, 0, 0, 0, 0, 1, 1, 1]), (e === Ee.FLIP_N_ROTATE_TILE || e === Ee.FLIP_N_ROTATE_ROW) && (w[1] = [0, 1, 1, 1, 1, 0, 0, 0]); let m = []; const I = [], N = []; let k = 0; for (let Y = 0; Y < d; Y++) for (let ee = 0; ee < a; ee++) b.push(-u + ee * t + P, -l + Y * r + p, 0), b.push(-u + (ee + 1) * t + P, -l + Y * r + p, 0), b.push(-u + (ee + 1) * t + P, -l + (Y + 1) * r + p, 0), b.push(-u + ee * t + P, -l + (Y + 1) * r + p, 0), N.push(k, k + 1, k + 3, k + 1, k + 2, k + 3), e === Ee.FLIP_TILE || e === Ee.ROTATE_TILE || e === Ee.FLIP_N_ROTATE_TILE ? m = m.concat(w[(ee % 2 + Y % 2) % 2]) : e === Ee.FLIP_ROW || e === Ee.ROTATE_ROW || e === Ee.FLIP_N_ROTATE_ROW ? m = m.concat(w[Y % 2]) : m = m.concat(w[0]), I.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1), j.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1), k += 4; if (f > 0 || v > 0) { const Y = v > 0 && (i === Ee.CENTER || i === Ee.TOP), ee = v > 0 && (i === Ee.CENTER || i === Ee.BOTTOM), Z = f > 0 && (n === Ee.CENTER || n === Ee.RIGHT), te = f > 0 && (n === Ee.CENTER || n === Ee.LEFT); let fe = [], _, G, L, $; if (Y && Z && (b.push(c + P, H + p, 0), b.push(-u + P, H + p, 0), b.push(-u + P, H + v + p, 0), b.push(c + P, H + v + p, 0), N.push(k, k + 1, k + 3, k + 1, k + 2, k + 3), k += 4, _ = 1 - f / t, G = 1 - v / r, L = 1, $ = 1, fe = [_, G, L, G, L, $, _, $], e === Ee.ROTATE_ROW && (fe = [1 - _, 1 - G, 1 - L, 1 - G, 1 - L, 1 - $, 1 - _, 1 - $]), e === Ee.FLIP_ROW && (fe = [1 - _, G, 1 - L, G, 1 - L, $, 1 - _, $]), e === Ee.FLIP_N_ROTATE_ROW && (fe = [_, 1 - G, L, 1 - G, L, 1 - $, _, 1 - $]), m = m.concat(fe), I.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1), j.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1)), Y && te && (b.push(u + P, H + p, 0), b.push(T + P, H + p, 0), b.push(T + P, H + v + p, 0), b.push(u + P, H + v + p, 0), N.push(k, k + 1, k + 3, k + 1, k + 2, k + 3), k += 4, _ = 0, G = 1 - v / r, L = f / t, $ = 1, fe = [_, G, L, G, L, $, _, $], (e === Ee.ROTATE_ROW || e === Ee.ROTATE_TILE && a % 2 === 0) && (fe = [1 - _, 1 - G, 1 - L, 1 - G, 1 - L, 1 - $, 1 - _, 1 - $]), (e === Ee.FLIP_ROW || e === Ee.FLIP_TILE && a % 2 === 0) && (fe = [1 - _, G, 1 - L, G, 1 - L, $, 1 - _, $]), (e === Ee.FLIP_N_ROTATE_ROW || e === Ee.FLIP_N_ROTATE_TILE && a % 2 === 0) && (fe = [_, 1 - G, L, 1 - G, L, 1 - $, _, 1 - $]), m = m.concat(fe), I.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1), j.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1)), ee && Z && (b.push(c + P, l + p, 0), b.push(-u + P, l + p, 0), b.push(-u + P, q + p, 0), b.push(c + P, q + p, 0), N.push(k, k + 1, k + 3, k + 1, k + 2, k + 3), k += 4, _ = 1 - f / t, G = 0, L = 1, $ = v / r, fe = [_, G, L, G, L, $, _, $], (e === Ee.ROTATE_ROW && d % 2 === 1 || e === Ee.ROTATE_TILE && d % 1 === 0) && (fe = [1 - _, 1 - G, 1 - L, 1 - G, 1 - L, 1 - $, 1 - _, 1 - $]), (e === Ee.FLIP_ROW && d % 2 === 1 || e === Ee.FLIP_TILE && d % 2 === 0) && (fe = [1 - _, G, 1 - L, G, 1 - L, $, 1 - _, $]), (e === Ee.FLIP_N_ROTATE_ROW && d % 2 === 1 || e === Ee.FLIP_N_ROTATE_TILE && d % 2 === 0) && (fe = [_, 1 - G, L, 1 - G, L, 1 - $, _, 1 - $]), m = m.concat(fe), I.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1), j.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1)), ee && te && (b.push(u + P, l + p, 0), b.push(T + P, l + p, 0), b.push(T + P, q + p, 0), b.push(u + P, q + p, 0), N.push(k, k + 1, k + 3, k + 1, k + 2, k + 3), k += 4, _ = 0, G = 0, L = f / t, $ = v / r, fe = [_, G, L, G, L, $, _, $], (e === Ee.ROTATE_ROW && d % 2 === 1 || e === Ee.ROTATE_TILE && (d + a) % 2 === 1) && (fe = [1 - _, 1 - G, 1 - L, 1 - G, 1 - L, 1 - $, 1 - _, 1 - $]), (e === Ee.FLIP_ROW && d % 2 === 1 || e === Ee.FLIP_TILE && (d + a) % 2 === 1) && (fe = [1 - _, G, 1 - L, G, 1 - L, $, 1 - _, $]), (e === Ee.FLIP_N_ROTATE_ROW && d % 2 === 1 || e === Ee.FLIP_N_ROTATE_TILE && (d + a) % 2 === 1) && (fe = [_, 1 - G, L, 1 - G, L, 1 - $, _, 1 - $]), m = m.concat(fe), I.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1), j.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1)), Y) { const ae = []; _ = 0, G = 1 - v / r, L = 1, $ = 1, ae[0] = [_, G, L, G, L, $, _, $], ae[1] = [_, G, L, G, L, $, _, $], (e === Ee.ROTATE_TILE || e === Ee.ROTATE_ROW) && (ae[1] = [1 - _, 1 - G, 1 - L, 1 - G, 1 - L, 1 - $, 1 - _, 1 - $]), (e === Ee.FLIP_TILE || e === Ee.FLIP_ROW) && (ae[1] = [1 - _, G, 1 - L, G, 1 - L, $, 1 - _, $]), (e === Ee.FLIP_N_ROTATE_TILE || e === Ee.FLIP_N_ROTATE_ROW) && (ae[1] = [_, 1 - G, L, 1 - G, L, 1 - $, _, 1 - $]); for (let Pe = 0; Pe < a; Pe++) b.push(-u + Pe * t + P, H + p, 0), b.push(-u + (Pe + 1) * t + P, H + p, 0), b.push(-u + (Pe + 1) * t + P, H + v + p, 0), b.push(-u + Pe * t + P, H + v + p, 0), N.push(k, k + 1, k + 3, k + 1, k + 2, k + 3), k += 4, e === Ee.FLIP_TILE || e === Ee.ROTATE_TILE || e === Ee.FLIP_N_ROTATE_TILE ? m = m.concat(ae[(Pe + 1) % 2]) : e === Ee.FLIP_ROW || e === Ee.ROTATE_ROW || e === Ee.FLIP_N_ROTATE_ROW ? m = m.concat(ae[1]) : m = m.concat(ae[0]), I.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1), j.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1); } if (ee) { const ae = []; _ = 0, G = 0, L = 1, $ = v / r, ae[0] = [_, G, L, G, L, $, _, $], ae[1] = [_, G, L, G, L, $, _, $], (e === Ee.ROTATE_TILE || e === Ee.ROTATE_ROW) && (ae[1] = [1 - _, 1 - G, 1 - L, 1 - G, 1 - L, 1 - $, 1 - _, 1 - $]), (e === Ee.FLIP_TILE || e === Ee.FLIP_ROW) && (ae[1] = [1 - _, G, 1 - L, G, 1 - L, $, 1 - _, $]), (e === Ee.FLIP_N_ROTATE_TILE || e === Ee.FLIP_N_ROTATE_ROW) && (ae[1] = [_, 1 - G, L, 1 - G, L, 1 - $, _, 1 - $]); for (let Pe = 0; Pe < a; Pe++) b.push(-u + Pe * t + P, q - v + p, 0), b.push(-u + (Pe + 1) * t + P, q - v + p, 0), b.push(-u + (Pe + 1) * t + P, q + p, 0), b.push(-u + Pe * t + P, q + p, 0), N.push(k, k + 1, k + 3, k + 1, k + 2, k + 3), k += 4, e === Ee.FLIP_TILE || e === Ee.ROTATE_TILE || e === Ee.FLIP_N_ROTATE_TILE ? m = m.concat(ae[(Pe + d) % 2]) : e === Ee.FLIP_ROW || e === Ee.ROTATE_ROW || e === Ee.FLIP_N_ROTATE_ROW ? m = m.concat(ae[d % 2]) : m = m.concat(ae[0]), I.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1), j.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1); } if (Z) { const ae = []; _ = 1 - f / t, G = 0, L = 1, $ = 1, ae[0] = [_, G, L, G, L, $, _, $], ae[1] = [_, G, L, G, L, $, _, $], (e === Ee.ROTATE_TILE || e === Ee.ROTATE_ROW) && (ae[1] = [1 - _, 1 - G, 1 - L, 1 - G, 1 - L, 1 - $, 1 - _, 1 - $]), (e === Ee.FLIP_TILE || e === Ee.FLIP_ROW) && (ae[1] = [1 - _, G, 1 - L, G, 1 - L, $, 1 - _, $]), (e === Ee.FLIP_N_ROTATE_TILE || e === Ee.FLIP_N_ROTATE_ROW) && (ae[1] = [_, 1 - G, L, 1 - G, L, 1 - $, _, 1 - $]); for (let Pe = 0; Pe < d; Pe++) b.push(c + P, -l + Pe * r + p, 0), b.push(c + f + P, -l + Pe * r + p, 0), b.push(c + f + P, -l + (Pe + 1) * r + p, 0), b.push(c + P, -l + (Pe + 1) * r + p, 0), N.push(k, k + 1, k + 3, k + 1, k + 2, k + 3), k += 4, e === Ee.FLIP_TILE || e === Ee.ROTATE_TILE || e === Ee.FLIP_N_ROTATE_TILE ? m = m.concat(ae[(Pe + 1) % 2]) : e === Ee.FLIP_ROW || e === Ee.ROTATE_ROW || e === Ee.FLIP_N_ROTATE_ROW ? m = m.concat(ae[Pe % 2]) : m = m.concat(ae[0]), I.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1), j.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1); } if (te) { const ae = []; _ = 0, G = 0, L = f / r, $ = 1, ae[0] = [_, G, L, G, L, $, _, $], ae[1] = [_, G, L, G, L, $, _, $], (e === Ee.ROTATE_TILE || e === Ee.ROTATE_ROW) && (ae[1] = [1 - _, 1 - G, 1 - L, 1 - G, 1 - L, 1 - $, 1 - _, 1 - $]), (e === Ee.FLIP_TILE || e === Ee.FLIP_ROW) && (ae[1] = [1 - _, G, 1 - L, G, 1 - L, $, 1 - _, $]), (e === Ee.FLIP_N_ROTATE_TILE || e === Ee.FLIP_N_ROTATE_ROW) && (ae[1] = [_, 1 - G, L, 1 - G, L, 1 - $, _, 1 - $]); for (let Pe = 0; Pe < d; Pe++) b.push(T - f + P, -l + Pe * r + p, 0), b.push(T + P, -l + Pe * r + p, 0), b.push(T + P, -l + (Pe + 1) * r + p, 0), b.push(T - f + P, -l + (Pe + 1) * r + p, 0), N.push(k, k + 1, k + 3, k + 1, k + 2, k + 3), k += 4, e === Ee.FLIP_TILE || e === Ee.ROTATE_TILE || e === Ee.FLIP_N_ROTATE_TILE ? m = m.concat(ae[(Pe + a) % 2]) : e === Ee.FLIP_ROW || e === Ee.ROTATE_ROW || e === Ee.FLIP_N_ROTATE_ROW ? m = m.concat(ae[Pe % 2]) : m = m.concat(ae[0]), I.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1), j.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1); } } const R = A.sideOrientation === 0 ? 0 : A.sideOrientation || Ut.DEFAULTSIDE; Ut._ComputeSides(R, b, N, j, m, A.frontUVs, A.backUVs); const y = new Ut(); y.indices = N, y.positions = b, y.normals = j, y.uvs = m; const O = R === Ut.DOUBLESIDE ? I.concat(I) : I; return y.colors = O, y; } function wQ(A, e, t = null) { const r = new Ee(A, t); return e.sideOrientation = Ee._GetDefaultSideOrientation(e.sideOrientation), r._originalBuilderSideOrientation = e.sideOrientation, $W(e).applyToMesh(r, e.updatable), r; } const Vve = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateTiledPlane: wQ }; Ut.CreateTiledPlane = $W; function mQ(A) { const t = A.faceUV || new Array(6), r = A.faceColors, n = A.pattern || Ee.NO_FLIP, i = A.width || A.size || 1, s = A.height || A.size || 1, a = A.depth || A.size || 1, f = A.tileWidth || A.tileSize || 1, o = A.tileHeight || A.tileSize || 1, d = A.alignHorizontal || 0, v = A.alignVertical || 0, u = A.sideOrientation === 0 ? 0 : A.sideOrientation || Ut.DEFAULTSIDE; for (let L = 0; L < 6; L++) t[L] === void 0 && (t[L] = new Ir(0, 0, 1, 1)), r && r[L] === void 0 && (r[L] = new xt(1, 1, 1, 1)); const l = i / 2, P = s / 2, p = a / 2, c = []; for (let L = 0; L < 2; L++) c[L] = $W({ pattern: n, tileWidth: f, tileHeight: o, width: i, height: s, alignVertical: v, alignHorizontal: d, sideOrientation: u }); for (let L = 2; L < 4; L++) c[L] = $W({ pattern: n, tileWidth: f, tileHeight: o, width: a, height: s, alignVertical: v, alignHorizontal: d, sideOrientation: u }); let H = v; v === Ee.BOTTOM ? H = Ee.TOP : v === Ee.TOP && (H = Ee.BOTTOM); for (let L = 4; L < 6; L++) c[L] = $W({ pattern: n, tileWidth: f, tileHeight: o, width: i, height: a, alignVertical: H, alignHorizontal: d, sideOrientation: u }); let T = [], q = [], b = [], j = []; const w = [], m = [], I = [], N = []; let k = 0, R = 0; for (let L = 0; L < 6; L++) { const $ = c[L].positions.length; m[L] = [], I[L] = []; for (let ae = 0; ae < $ / 3; ae++) m[L].push(new S(c[L].positions[3 * ae], c[L].positions[3 * ae + 1], c[L].positions[3 * ae + 2])), I[L].push(new S(c[L].normals[3 * ae], c[L].normals[3 * ae + 1], c[L].normals[3 * ae + 2])); k = c[L].uvs.length, N[L] = []; for (let ae = 0; ae < k; ae += 2) N[L][ae] = t[L].x + (t[L].z - t[L].x) * c[L].uvs[ae], N[L][ae + 1] = t[L].y + (t[L].w - t[L].y) * c[L].uvs[ae + 1], us.UseOpenGLOrientationForUV && (N[L][ae + 1] = 1 - N[L][ae + 1]); if (b = b.concat(N[L]), j = j.concat(c[L].indices.map((ae) => ae + R)), R += m[L].length, r) for (let ae = 0; ae < 4; ae++) w.push(r[L].r, r[L].g, r[L].b, r[L].a); } const y = new S(0, 0, p), O = he.RotationY(Math.PI); T = m[0].map((L) => S.TransformNormal(L, O).add(y)).map((L) => [L.x, L.y, L.z]).reduce((L, $) => L.concat($), []), q = I[0].map((L) => S.TransformNormal(L, O)).map((L) => [L.x, L.y, L.z]).reduce((L, $) => L.concat($), []), T = T.concat(m[1].map((L) => L.subtract(y)).map((L) => [L.x, L.y, L.z]).reduce((L, $) => L.concat($), [])), q = q.concat(I[1].map((L) => [L.x, L.y, L.z]).reduce((L, $) => L.concat($), [])); const Y = new S(l, 0, 0), ee = he.RotationY(-Math.PI / 2); T = T.concat(m[2].map((L) => S.TransformNormal(L, ee).add(Y)).map((L) => [L.x, L.y, L.z]).reduce((L, $) => L.concat($), [])), q = q.concat(I[2].map((L) => S.TransformNormal(L, ee)).map((L) => [L.x, L.y, L.z]).reduce((L, $) => L.concat($), [])); const Z = he.RotationY(Math.PI / 2); T = T.concat(m[3].map((L) => S.TransformNormal(L, Z).subtract(Y)).map((L) => [L.x, L.y, L.z]).reduce((L, $) => L.concat($), [])), q = q.concat(I[3].map((L) => S.TransformNormal(L, Z)).map((L) => [L.x, L.y, L.z]).reduce((L, $) => L.concat($), [])); const te = new S(0, P, 0), fe = he.RotationX(Math.PI / 2); T = T.concat(m[4].map((L) => S.TransformNormal(L, fe).add(te)).map((L) => [L.x, L.y, L.z]).reduce((L, $) => L.concat($), [])), q = q.concat(I[4].map((L) => S.TransformNormal(L, fe)).map((L) => [L.x, L.y, L.z]).reduce((L, $) => L.concat($), [])); const _ = he.RotationX(-Math.PI / 2); T = T.concat(m[5].map((L) => S.TransformNormal(L, _).subtract(te)).map((L) => [L.x, L.y, L.z]).reduce((L, $) => L.concat($), [])), q = q.concat(I[5].map((L) => S.TransformNormal(L, _)).map((L) => [L.x, L.y, L.z]).reduce((L, $) => L.concat($), [])), Ut._ComputeSides(u, T, j, q, b); const G = new Ut(); if (G.indices = j, G.positions = T, G.normals = q, G.uvs = b, r) { const L = u === Ut.DOUBLESIDE ? w.concat(w) : w; G.colors = L; } return G; } function BQ(A, e, t = null) { const r = new Ee(A, t); return e.sideOrientation = Ee._GetDefaultSideOrientation(e.sideOrientation), r._originalBuilderSideOrientation = e.sideOrientation, mQ(e).applyToMesh(r, e.updatable), r; } const Cve = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateTiledBox: BQ }; Ut.CreateTiledBox = mQ; function WQ(A) { const e = [], t = [], r = [], n = [], i = A.radius || 2, s = A.tube || 0.5, a = A.radialSegments || 32, f = A.tubularSegments || 32, o = A.p || 2, d = A.q || 3, v = A.sideOrientation === 0 ? 0 : A.sideOrientation || Ut.DEFAULTSIDE, u = (c) => { const H = Math.cos(c), T = Math.sin(c), q = d / o * c, b = Math.cos(q), j = i * (2 + b) * 0.5 * H, w = i * (2 + b) * T * 0.5, m = i * Math.sin(q) * 0.5; return new S(j, w, m); }; let l, P; for (l = 0; l <= a; l++) { const H = l % a / a * 2 * o * Math.PI, T = u(H), q = u(H + 0.01), b = q.subtract(T); let j = q.add(T); const w = S.Cross(b, j); for (j = S.Cross(w, b), w.normalize(), j.normalize(), P = 0; P < f; P++) { const I = P % f / f * 2 * Math.PI, N = -s * Math.cos(I), k = s * Math.sin(I); t.push(T.x + N * j.x + k * w.x), t.push(T.y + N * j.y + k * w.y), t.push(T.z + N * j.z + k * w.z), n.push(l / a), n.push(us.UseOpenGLOrientationForUV ? 1 - P / f : P / f); } } for (l = 0; l < a; l++) for (P = 0; P < f; P++) { const c = (P + 1) % f, H = l * f + P, T = (l + 1) * f + P, q = (l + 1) * f + c, b = l * f + c; e.push(b), e.push(T), e.push(H), e.push(b), e.push(q), e.push(T); } Ut.ComputeNormals(t, e, r), Ut._ComputeSides(v, t, e, r, n, A.frontUVs, A.backUVs); const p = new Ut(); return p.indices = e, p.positions = t, p.normals = r, p.uvs = n, p; } function GO(A, e = {}, t) { const r = new Ee(A, t); return e.sideOrientation = Ee._GetDefaultSideOrientation(e.sideOrientation), r._originalBuilderSideOrientation = e.sideOrientation, WQ(e).applyToMesh(r, e.updatable), r; } const Ove = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateTorusKnot: GO }; Ut.CreateTorusKnot = WQ; Ee.CreateTorusKnot = (A, e, t, r, n, i, s, a, f, o) => GO(A, { radius: e, tube: t, radialSegments: r, tubularSegments: n, p: i, q: s, sideOrientation: o, updatable: f }, a); const qE = { effect: null, subMesh: null }; class Zo extends P1 { /** * Instantiate a new shader material. * The ShaderMaterial object has the necessary methods to pass data from your scene to the Vertex and Fragment Shaders and returns a material that can be applied to any mesh. * This returned material effects how the mesh will look based on the code in the shaders. * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/shaders/shaderMaterial * @param name Define the name of the material in the scene * @param scene Define the scene the material belongs to * @param shaderPath Defines the route to the shader code in one of three ways: * * object: \{ vertex: "custom", fragment: "custom" \}, used with Effect.ShadersStore["customVertexShader"] and Effect.ShadersStore["customFragmentShader"] * * object: \{ vertexElement: "vertexShaderCode", fragmentElement: "fragmentShaderCode" \}, used with shader code in script tags * * object: \{ vertexSource: "vertex shader code string", fragmentSource: "fragment shader code string" \} using with strings containing the shaders code * * string: "./COMMON_NAME", used with external files COMMON_NAME.vertex.fx and COMMON_NAME.fragment.fx in index.html folder. * @param options Define the options used to create the shader * @param storeEffectOnSubMeshes true to store effect on submeshes, false to store the effect directly in the material class. */ constructor(e, t, r, n = {}, i = !0) { super(e, t, i), this._textures = {}, this._textureArrays = {}, this._externalTextures = {}, this._floats = {}, this._ints = {}, this._uints = {}, this._floatsArrays = {}, this._colors3 = {}, this._colors3Arrays = {}, this._colors4 = {}, this._colors4Arrays = {}, this._vectors2 = {}, this._vectors3 = {}, this._vectors4 = {}, this._quaternions = {}, this._quaternionsArrays = {}, this._matrices = {}, this._matrixArrays = {}, this._matrices3x3 = {}, this._matrices2x2 = {}, this._vectors2Arrays = {}, this._vectors3Arrays = {}, this._vectors4Arrays = {}, this._uniformBuffers = {}, this._textureSamplers = {}, this._storageBuffers = {}, this._cachedWorldViewMatrix = new he(), this._cachedWorldViewProjectionMatrix = new he(), this._multiview = !1, this._materialHelperNeedsPreviousMatrices = !1, this._shaderPath = r, this._options = Object.assign({ needAlphaBlending: !1, needAlphaTesting: !1, attributes: ["position", "normal", "uv"], uniforms: ["worldViewProjection"], uniformBuffers: [], samplers: [], externalTextures: [], samplerObjects: [], storageBuffers: [], defines: [], useClipPlane: !1 }, n); } /** * Gets the shader path used to define the shader code * It can be modified to trigger a new compilation */ get shaderPath() { return this._shaderPath; } /** * Sets the shader path used to define the shader code * It can be modified to trigger a new compilation */ set shaderPath(e) { this._shaderPath = e; } /** * Gets the options used to compile the shader. * They can be modified to trigger a new compilation */ get options() { return this._options; } /** * is multiview set to true? */ get isMultiview() { return this._multiview; } /** * Gets the current class name of the material e.g. "ShaderMaterial" * Mainly use in serialization. * @returns the class name */ getClassName() { return "ShaderMaterial"; } /** * Specifies if the material will require alpha blending * @returns a boolean specifying if alpha blending is needed */ needAlphaBlending() { return this.alpha < 1 || this._options.needAlphaBlending; } /** * Specifies if this material should be rendered in alpha test mode * @returns a boolean specifying if an alpha test is needed. */ needAlphaTesting() { return this._options.needAlphaTesting; } _checkUniform(e) { this._options.uniforms.indexOf(e) === -1 && this._options.uniforms.push(e); } /** * Set a texture in the shader. * @param name Define the name of the uniform samplers as defined in the shader * @param texture Define the texture to bind to this sampler * @returns the material itself allowing "fluent" like uniform updates */ setTexture(e, t) { return this._options.samplers.indexOf(e) === -1 && this._options.samplers.push(e), this._textures[e] = t, this; } /** * Set a texture array in the shader. * @param name Define the name of the uniform sampler array as defined in the shader * @param textures Define the list of textures to bind to this sampler * @returns the material itself allowing "fluent" like uniform updates */ setTextureArray(e, t) { return this._options.samplers.indexOf(e) === -1 && this._options.samplers.push(e), this._checkUniform(e), this._textureArrays[e] = t, this; } /** * Set an internal texture in the shader. * @param name Define the name of the uniform samplers as defined in the shader * @param texture Define the texture to bind to this sampler * @returns the material itself allowing "fluent" like uniform updates */ setExternalTexture(e, t) { return this._options.externalTextures.indexOf(e) === -1 && this._options.externalTextures.push(e), this._externalTextures[e] = t, this; } /** * Set a float in the shader. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setFloat(e, t) { return this._checkUniform(e), this._floats[e] = t, this; } /** * Set a int in the shader. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setInt(e, t) { return this._checkUniform(e), this._ints[e] = t, this; } /** * Set a unsigned int in the shader. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setUInt(e, t) { return this._checkUniform(e), this._uints[e] = t, this; } /** * Set an array of floats in the shader. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setFloats(e, t) { return this._checkUniform(e), this._floatsArrays[e] = t, this; } /** * Set a vec3 in the shader from a Color3. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setColor3(e, t) { return this._checkUniform(e), this._colors3[e] = t, this; } /** * Set a vec3 array in the shader from a Color3 array. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setColor3Array(e, t) { return this._checkUniform(e), this._colors3Arrays[e] = t.reduce((r, n) => (n.toArray(r, r.length), r), []), this; } /** * Set a vec4 in the shader from a Color4. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setColor4(e, t) { return this._checkUniform(e), this._colors4[e] = t, this; } /** * Set a vec4 array in the shader from a Color4 array. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setColor4Array(e, t) { return this._checkUniform(e), this._colors4Arrays[e] = t.reduce((r, n) => (n.toArray(r, r.length), r), []), this; } /** * Set a vec2 in the shader from a Vector2. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setVector2(e, t) { return this._checkUniform(e), this._vectors2[e] = t, this; } /** * Set a vec3 in the shader from a Vector3. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setVector3(e, t) { return this._checkUniform(e), this._vectors3[e] = t, this; } /** * Set a vec4 in the shader from a Vector4. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setVector4(e, t) { return this._checkUniform(e), this._vectors4[e] = t, this; } /** * Set a vec4 in the shader from a Quaternion. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setQuaternion(e, t) { return this._checkUniform(e), this._quaternions[e] = t, this; } /** * Set a vec4 array in the shader from a Quaternion array. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setQuaternionArray(e, t) { return this._checkUniform(e), this._quaternionsArrays[e] = t.reduce((r, n) => (n.toArray(r, r.length), r), []), this; } /** * Set a mat4 in the shader from a Matrix. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setMatrix(e, t) { return this._checkUniform(e), this._matrices[e] = t, this; } /** * Set a float32Array in the shader from a matrix array. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setMatrices(e, t) { this._checkUniform(e); const r = new Float32Array(t.length * 16); for (let n = 0; n < t.length; n++) t[n].copyToArray(r, n * 16); return this._matrixArrays[e] = r, this; } /** * Set a mat3 in the shader from a Float32Array. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setMatrix3x3(e, t) { return this._checkUniform(e), this._matrices3x3[e] = t, this; } /** * Set a mat2 in the shader from a Float32Array. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setMatrix2x2(e, t) { return this._checkUniform(e), this._matrices2x2[e] = t, this; } /** * Set a vec2 array in the shader from a number array. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setArray2(e, t) { return this._checkUniform(e), this._vectors2Arrays[e] = t, this; } /** * Set a vec3 array in the shader from a number array. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setArray3(e, t) { return this._checkUniform(e), this._vectors3Arrays[e] = t, this; } /** * Set a vec4 array in the shader from a number array. * @param name Define the name of the uniform as defined in the shader * @param value Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setArray4(e, t) { return this._checkUniform(e), this._vectors4Arrays[e] = t, this; } /** * Set a uniform buffer in the shader * @param name Define the name of the uniform as defined in the shader * @param buffer Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setUniformBuffer(e, t) { return this._options.uniformBuffers.indexOf(e) === -1 && this._options.uniformBuffers.push(e), this._uniformBuffers[e] = t, this; } /** * Set a texture sampler in the shader * @param name Define the name of the uniform as defined in the shader * @param sampler Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setTextureSampler(e, t) { return this._options.samplerObjects.indexOf(e) === -1 && this._options.samplerObjects.push(e), this._textureSamplers[e] = t, this; } /** * Set a storage buffer in the shader * @param name Define the name of the storage buffer as defined in the shader * @param buffer Define the value to give to the uniform * @returns the material itself allowing "fluent" like uniform updates */ setStorageBuffer(e, t) { return this._options.storageBuffers.indexOf(e) === -1 && this._options.storageBuffers.push(e), this._storageBuffers[e] = t, this; } /** * Adds, removes, or replaces the specified shader define and value. * * setDefine("MY_DEFINE", true); // enables a boolean define * * setDefine("MY_DEFINE", "0.5"); // adds "#define MY_DEFINE 0.5" to the shader (or sets and replaces the value of any existing define with that name) * * setDefine("MY_DEFINE", false); // disables and removes the define * Note if the active defines do change, the shader will be recompiled and this can be expensive. * @param define the define name e.g., "OUTPUT_TO_SRGB" or "#define OUTPUT_TO_SRGB". If the define was passed into the constructor already, the version used should match that, and in either case, it should not include any appended value. * @param value either the value of the define (e.g. a numerical value) or for booleans, true if the define should be enabled or false if it should be disabled * @returns the material itself allowing "fluent" like uniform updates */ setDefine(e, t) { const r = e.trimEnd() + " ", n = this.options.defines.findIndex((i) => i === e || i.startsWith(r)); return n >= 0 && this.options.defines.splice(n, 1), (typeof t != "boolean" || t) && this.options.defines.push(r + t), this; } /** * Specifies that the submesh is ready to be used * @param mesh defines the mesh to check * @param subMesh defines which submesh to check * @param useInstances specifies that instances should be used * @returns a boolean indicating that the submesh is ready or not */ isReadyForSubMesh(e, t, r) { return this.isReady(e, r, t); } /** * Checks if the material is ready to render the requested mesh * @param mesh Define the mesh to render * @param useInstances Define whether or not the material is used with instances * @param subMesh defines which submesh to render * @returns true if ready, otherwise false */ isReady(e, t, r) { var n, i, s, a; const f = r && this._storeEffectOnSubMeshes; if (this.isFrozen) if (f) { if (r.effect && r.effect._wasPreviouslyReady) return !0; } else { const N = this._drawWrapper.effect; if (N && N._wasPreviouslyReady && N._wasPreviouslyUsingInstances === t) return !0; } const o = this.getScene(), d = o.getEngine(), v = [], u = [], l = new c1(); let P = this._shaderPath, p = this._options.uniforms, c = this._options.uniformBuffers, H = this._options.samplers; d.getCaps().multiview && o.activeCamera && o.activeCamera.outputRenderTarget && o.activeCamera.outputRenderTarget.getViewCount() > 1 && (this._multiview = !0, v.push("#define MULTIVIEW"), this._options.uniforms.indexOf("viewProjection") !== -1 && this._options.uniforms.indexOf("viewProjectionR") === -1 && this._options.uniforms.push("viewProjectionR")); for (let N = 0; N < this._options.defines.length; N++) { const k = this._options.defines[N].indexOf("#define") === 0 ? this._options.defines[N] : `#define ${this._options.defines[N]}`; v.push(k); } for (let N = 0; N < this._options.attributes.length; N++) u.push(this._options.attributes[N]); if (e && e.isVerticesDataPresent(J.ColorKind) && (u.indexOf(J.ColorKind) === -1 && u.push(J.ColorKind), v.push("#define VERTEXCOLOR")), t && (v.push("#define INSTANCES"), Ye.PushAttributesForInstances(u, this._materialHelperNeedsPreviousMatrices), e != null && e.hasThinInstances && (v.push("#define THIN_INSTANCES"), e && e.isVerticesDataPresent(J.ColorInstanceKind) && (u.push(J.ColorInstanceKind), v.push("#define INSTANCESCOLOR")))), e && e.useBones && e.computeBonesUsingShaders && e.skeleton) { u.push(J.MatricesIndicesKind), u.push(J.MatricesWeightsKind), e.numBoneInfluencers > 4 && (u.push(J.MatricesIndicesExtraKind), u.push(J.MatricesWeightsExtraKind)); const N = e.skeleton; v.push("#define NUM_BONE_INFLUENCERS " + e.numBoneInfluencers), l.addCPUSkinningFallback(0, e), N.isUsingTextureForMatrices ? (v.push("#define BONETEXTURE"), this._options.uniforms.indexOf("boneTextureWidth") === -1 && this._options.uniforms.push("boneTextureWidth"), this._options.samplers.indexOf("boneSampler") === -1 && this._options.samplers.push("boneSampler")) : (v.push("#define BonesPerMesh " + (N.bones.length + 1)), this._options.uniforms.indexOf("mBones") === -1 && this._options.uniforms.push("mBones")); } else v.push("#define NUM_BONE_INFLUENCERS 0"); let T = 0; const q = e ? e.morphTargetManager : null; if (q) { const N = q.supportsUVs && v.indexOf("#define UV1") !== -1, k = q.supportsTangents && v.indexOf("#define TANGENT") !== -1, R = q.supportsNormals && v.indexOf("#define NORMAL") !== -1; T = q.numInfluencers, N && v.push("#define MORPHTARGETS_UV"), k && v.push("#define MORPHTARGETS_TANGENT"), R && v.push("#define MORPHTARGETS_NORMAL"), T > 0 && v.push("#define MORPHTARGETS"), q.isUsingTextureForTargets && (v.push("#define MORPHTARGETS_TEXTURE"), this._options.uniforms.indexOf("morphTargetTextureIndices") === -1 && this._options.uniforms.push("morphTargetTextureIndices"), this._options.samplers.indexOf("morphTargets") === -1 && this._options.samplers.push("morphTargets")), v.push("#define NUM_MORPH_INFLUENCERS " + T); for (let y = 0; y < T; y++) u.push(J.PositionKind + y), R && u.push(J.NormalKind + y), k && u.push(J.TangentKind + y), N && u.push(J.UVKind + "_" + y); T > 0 && (p = p.slice(), p.push("morphTargetInfluences"), p.push("morphTargetTextureInfo"), p.push("morphTargetTextureIndices")); } else v.push("#define NUM_MORPH_INFLUENCERS 0"); if (e) { const N = e.bakedVertexAnimationManager; N && N.isEnabled && (v.push("#define BAKED_VERTEX_ANIMATION_TEXTURE"), this._options.uniforms.indexOf("bakedVertexAnimationSettings") === -1 && this._options.uniforms.push("bakedVertexAnimationSettings"), this._options.uniforms.indexOf("bakedVertexAnimationTextureSizeInverted") === -1 && this._options.uniforms.push("bakedVertexAnimationTextureSizeInverted"), this._options.uniforms.indexOf("bakedVertexAnimationTime") === -1 && this._options.uniforms.push("bakedVertexAnimationTime"), this._options.samplers.indexOf("bakedVertexAnimationTexture") === -1 && this._options.samplers.push("bakedVertexAnimationTexture")), Ye.PrepareAttributesForBakedVertexAnimation(u, e, v); } for (const N in this._textures) if (!this._textures[N].isReady()) return !1; e && this._shouldTurnAlphaTestOn(e) && v.push("#define ALPHATEST"), this._options.useClipPlane !== !1 && (Mf(p), xq(this, o, v)), this._useLogarithmicDepth && (v.push("#define LOGARITHMICDEPTH"), this._options.uniforms.indexOf("logarithmicDepthConstant") === -1 && this._options.uniforms.push("logarithmicDepthConstant")), this.customShaderNameResolve && (p = p.slice(), c = c.slice(), H = H.slice(), P = this.customShaderNameResolve(P, p, c, H, v, u)); const b = f ? r._getDrawWrapper() : this._drawWrapper, j = (n = b == null ? void 0 : b.effect) !== null && n !== void 0 ? n : null, w = (i = b == null ? void 0 : b.defines) !== null && i !== void 0 ? i : null, m = v.join(` `); let I = j; return w !== m && (I = d.createEffect(P, { attributes: u, uniformsNames: p, uniformBuffersNames: c, samplers: H, defines: m, fallbacks: l, onCompiled: this.onCompiled, onError: this.onError, indexParameters: { maxSimultaneousMorphTargets: T }, shaderLanguage: this._options.shaderLanguage }, d), f ? r.setEffect(I, m, this._materialContext) : b && b.setEffect(I, m), this._onEffectCreatedObservable && (qE.effect = I, qE.subMesh = (s = r ?? (e == null ? void 0 : e.subMeshes[0])) !== null && s !== void 0 ? s : null, this._onEffectCreatedObservable.notifyObservers(qE))), I._wasPreviouslyUsingInstances = !!t, !((a = !(I != null && I.isReady())) !== null && a !== void 0) || a ? !1 : (j !== I && o.resetCachedMaterial(), I._wasPreviouslyReady = !0, !0); } /** * Binds the world matrix to the material * @param world defines the world transformation matrix * @param effectOverride - If provided, use this effect instead of internal effect */ bindOnlyWorldMatrix(e, t) { const r = this.getScene(), n = t ?? this.getEffect(); n && (this._options.uniforms.indexOf("world") !== -1 && n.setMatrix("world", e), this._options.uniforms.indexOf("worldView") !== -1 && (e.multiplyToRef(r.getViewMatrix(), this._cachedWorldViewMatrix), n.setMatrix("worldView", this._cachedWorldViewMatrix)), this._options.uniforms.indexOf("worldViewProjection") !== -1 && (e.multiplyToRef(r.getTransformMatrix(), this._cachedWorldViewProjectionMatrix), n.setMatrix("worldViewProjection", this._cachedWorldViewProjectionMatrix))); } /** * Binds the submesh to this material by preparing the effect and shader to draw * @param world defines the world transformation matrix * @param mesh defines the mesh containing the submesh * @param subMesh defines the submesh to bind the material to */ bindForSubMesh(e, t, r) { var n; this.bind(e, t, (n = r._drawWrapperOverride) === null || n === void 0 ? void 0 : n.effect, r); } /** * Binds the material to the mesh * @param world defines the world transformation matrix * @param mesh defines the mesh to bind the material to * @param effectOverride - If provided, use this effect instead of internal effect * @param subMesh defines the submesh to bind the material to */ bind(e, t, r, n) { var i; const s = n && this._storeEffectOnSubMeshes, a = r ?? (s ? n.effect : this.getEffect()); if (!a) return; const f = this.getScene(); this._activeEffect = a, this.bindOnlyWorldMatrix(e, r); const o = this._options.uniformBuffers; let d = !1; if (a && o && o.length > 0 && f.getEngine().supportsUniformBuffers) for (let u = 0; u < o.length; ++u) switch (o[u]) { case "Mesh": t && (t.getMeshUniformBuffer().bindToEffect(a, "Mesh"), t.transferToEffect(e)); break; case "Scene": Ye.BindSceneUniformBuffer(a, f.getSceneUniformBuffer()), f.finalizeSceneUbo(), d = !0; break; } const v = t && s ? this._mustRebind(f, a, t.visibility) : f.getCachedMaterial() !== this; if (a && v) { !d && this._options.uniforms.indexOf("view") !== -1 && a.setMatrix("view", f.getViewMatrix()), !d && this._options.uniforms.indexOf("projection") !== -1 && a.setMatrix("projection", f.getProjectionMatrix()), !d && this._options.uniforms.indexOf("viewProjection") !== -1 && (a.setMatrix("viewProjection", f.getTransformMatrix()), this._multiview && a.setMatrix("viewProjectionR", f._transformMatrixR)), f.activeCamera && this._options.uniforms.indexOf("cameraPosition") !== -1 && a.setVector3("cameraPosition", f.activeCamera.globalPosition), Ye.BindBonesParameters(t, a), Df(a, this, f), this._useLogarithmicDepth && Ye.BindLogDepth(s ? n.materialDefines : a.defines, a, f); let u; for (u in this._textures) a.setTexture(u, this._textures[u]); for (u in this._textureArrays) a.setTextureArray(u, this._textureArrays[u]); for (u in this._externalTextures) a.setExternalTexture(u, this._externalTextures[u]); for (u in this._ints) a.setInt(u, this._ints[u]); for (u in this._uints) a.setUInt(u, this._uints[u]); for (u in this._floats) a.setFloat(u, this._floats[u]); for (u in this._floatsArrays) a.setArray(u, this._floatsArrays[u]); for (u in this._colors3) a.setColor3(u, this._colors3[u]); for (u in this._colors3Arrays) a.setArray3(u, this._colors3Arrays[u]); for (u in this._colors4) { const l = this._colors4[u]; a.setFloat4(u, l.r, l.g, l.b, l.a); } for (u in this._colors4Arrays) a.setArray4(u, this._colors4Arrays[u]); for (u in this._vectors2) a.setVector2(u, this._vectors2[u]); for (u in this._vectors3) a.setVector3(u, this._vectors3[u]); for (u in this._vectors4) a.setVector4(u, this._vectors4[u]); for (u in this._quaternions) a.setQuaternion(u, this._quaternions[u]); for (u in this._matrices) a.setMatrix(u, this._matrices[u]); for (u in this._matrixArrays) a.setMatrices(u, this._matrixArrays[u]); for (u in this._matrices3x3) a.setMatrix3x3(u, this._matrices3x3[u]); for (u in this._matrices2x2) a.setMatrix2x2(u, this._matrices2x2[u]); for (u in this._vectors2Arrays) a.setArray2(u, this._vectors2Arrays[u]); for (u in this._vectors3Arrays) a.setArray3(u, this._vectors3Arrays[u]); for (u in this._vectors4Arrays) a.setArray4(u, this._vectors4Arrays[u]); for (u in this._quaternionsArrays) a.setArray4(u, this._quaternionsArrays[u]); for (u in this._uniformBuffers) { const l = this._uniformBuffers[u].getBuffer(); l && a.bindUniformBuffer(l, u); } for (u in this._textureSamplers) a.setTextureSampler(u, this._textureSamplers[u]); for (u in this._storageBuffers) a.setStorageBuffer(u, this._storageBuffers[u]); } if (a && t && (v || !this.isFrozen)) { const u = t.morphTargetManager; u && u.numInfluencers > 0 && Ye.BindMorphTargetParameters(t, a); const l = t.bakedVertexAnimationManager; l && l.isEnabled && ((i = t.bakedVertexAnimationManager) === null || i === void 0 || i.bind(a, !!a._wasPreviouslyUsingInstances)); } this._afterBind(t, a); } /** * Gets the active textures from the material * @returns an array of textures */ getActiveTextures() { const e = super.getActiveTextures(); for (const t in this._textures) e.push(this._textures[t]); for (const t in this._textureArrays) { const r = this._textureArrays[t]; for (let n = 0; n < r.length; n++) e.push(r[n]); } return e; } /** * Specifies if the material uses a texture * @param texture defines the texture to check against the material * @returns a boolean specifying if the material uses the texture */ hasTexture(e) { if (super.hasTexture(e)) return !0; for (const t in this._textures) if (this._textures[t] === e) return !0; for (const t in this._textureArrays) { const r = this._textureArrays[t]; for (let n = 0; n < r.length; n++) if (r[n] === e) return !0; } return !1; } /** * Makes a duplicate of the material, and gives it a new name * @param name defines the new name for the duplicated material * @returns the cloned material */ clone(e) { const t = jt.Clone(() => new Zo(e, this.getScene(), this._shaderPath, this._options, this._storeEffectOnSubMeshes), this); t.name = e, t.id = e, typeof t._shaderPath == "object" && (t._shaderPath = Object.assign({}, t._shaderPath)), this._options = Object.assign({}, this._options), Object.keys(this._options).forEach((r) => { const n = this._options[r]; Array.isArray(n) && (this._options[r] = n.slice(0)); }), this.stencil.copyTo(t.stencil); for (const r in this._textures) t.setTexture(r, this._textures[r]); for (const r in this._textureArrays) t.setTextureArray(r, this._textureArrays[r]); for (const r in this._externalTextures) t.setExternalTexture(r, this._externalTextures[r]); for (const r in this._ints) t.setInt(r, this._ints[r]); for (const r in this._uints) t.setUInt(r, this._uints[r]); for (const r in this._floats) t.setFloat(r, this._floats[r]); for (const r in this._floatsArrays) t.setFloats(r, this._floatsArrays[r]); for (const r in this._colors3) t.setColor3(r, this._colors3[r]); for (const r in this._colors3Arrays) t._colors3Arrays[r] = this._colors3Arrays[r]; for (const r in this._colors4) t.setColor4(r, this._colors4[r]); for (const r in this._colors4Arrays) t._colors4Arrays[r] = this._colors4Arrays[r]; for (const r in this._vectors2) t.setVector2(r, this._vectors2[r]); for (const r in this._vectors3) t.setVector3(r, this._vectors3[r]); for (const r in this._vectors4) t.setVector4(r, this._vectors4[r]); for (const r in this._quaternions) t.setQuaternion(r, this._quaternions[r]); for (const r in this._quaternionsArrays) t._quaternionsArrays[r] = this._quaternionsArrays[r]; for (const r in this._matrices) t.setMatrix(r, this._matrices[r]); for (const r in this._matrixArrays) t._matrixArrays[r] = this._matrixArrays[r].slice(); for (const r in this._matrices3x3) t.setMatrix3x3(r, this._matrices3x3[r]); for (const r in this._matrices2x2) t.setMatrix2x2(r, this._matrices2x2[r]); for (const r in this._vectors2Arrays) t.setArray2(r, this._vectors2Arrays[r]); for (const r in this._vectors3Arrays) t.setArray3(r, this._vectors3Arrays[r]); for (const r in this._vectors4Arrays) t.setArray4(r, this._vectors4Arrays[r]); for (const r in this._uniformBuffers) t.setUniformBuffer(r, this._uniformBuffers[r]); for (const r in this._textureSamplers) t.setTextureSampler(r, this._textureSamplers[r]); for (const r in this._storageBuffers) t.setStorageBuffer(r, this._storageBuffers[r]); return t; } /** * Disposes the material * @param forceDisposeEffect specifies if effects should be forcefully disposed * @param forceDisposeTextures specifies if textures should be forcefully disposed * @param notBoundToMesh specifies if the material that is being disposed is known to be not bound to any mesh */ dispose(e, t, r) { if (t) { let n; for (n in this._textures) this._textures[n].dispose(); for (n in this._textureArrays) { const i = this._textureArrays[n]; for (let s = 0; s < i.length; s++) i[s].dispose(); } } this._textures = {}, super.dispose(e, t, r); } /** * Serializes this material in a JSON representation * @returns the serialized material object */ serialize() { const e = jt.Serialize(this); e.customType = "BABYLON.ShaderMaterial", e.uniqueId = this.uniqueId, e.options = this._options, e.shaderPath = this._shaderPath, e.storeEffectOnSubMeshes = this._storeEffectOnSubMeshes; let t; e.stencil = this.stencil.serialize(), e.textures = {}; for (t in this._textures) e.textures[t] = this._textures[t].serialize(); e.textureArrays = {}; for (t in this._textureArrays) { e.textureArrays[t] = []; const r = this._textureArrays[t]; for (let n = 0; n < r.length; n++) e.textureArrays[t].push(r[n].serialize()); } e.ints = {}; for (t in this._ints) e.ints[t] = this._ints[t]; e.uints = {}; for (t in this._uints) e.uints[t] = this._uints[t]; e.floats = {}; for (t in this._floats) e.floats[t] = this._floats[t]; e.FloatArrays = {}; for (t in this._floatsArrays) e.FloatArrays[t] = this._floatsArrays[t]; e.colors3 = {}; for (t in this._colors3) e.colors3[t] = this._colors3[t].asArray(); e.colors3Arrays = {}; for (t in this._colors3Arrays) e.colors3Arrays[t] = this._colors3Arrays[t]; e.colors4 = {}; for (t in this._colors4) e.colors4[t] = this._colors4[t].asArray(); e.colors4Arrays = {}; for (t in this._colors4Arrays) e.colors4Arrays[t] = this._colors4Arrays[t]; e.vectors2 = {}; for (t in this._vectors2) e.vectors2[t] = this._vectors2[t].asArray(); e.vectors3 = {}; for (t in this._vectors3) e.vectors3[t] = this._vectors3[t].asArray(); e.vectors4 = {}; for (t in this._vectors4) e.vectors4[t] = this._vectors4[t].asArray(); e.quaternions = {}; for (t in this._quaternions) e.quaternions[t] = this._quaternions[t].asArray(); e.matrices = {}; for (t in this._matrices) e.matrices[t] = this._matrices[t].asArray(); e.matrixArray = {}; for (t in this._matrixArrays) e.matrixArray[t] = this._matrixArrays[t]; e.matrices3x3 = {}; for (t in this._matrices3x3) e.matrices3x3[t] = this._matrices3x3[t]; e.matrices2x2 = {}; for (t in this._matrices2x2) e.matrices2x2[t] = this._matrices2x2[t]; e.vectors2Arrays = {}; for (t in this._vectors2Arrays) e.vectors2Arrays[t] = this._vectors2Arrays[t]; e.vectors3Arrays = {}; for (t in this._vectors3Arrays) e.vectors3Arrays[t] = this._vectors3Arrays[t]; e.vectors4Arrays = {}; for (t in this._vectors4Arrays) e.vectors4Arrays[t] = this._vectors4Arrays[t]; e.quaternionsArrays = {}; for (t in this._quaternionsArrays) e.quaternionsArrays[t] = this._quaternionsArrays[t]; return e; } /** * Creates a shader material from parsed shader material data * @param source defines the JSON representation of the material * @param scene defines the hosting scene * @param rootUrl defines the root URL to use to load textures and relative dependencies * @returns a new material */ static Parse(e, t, r) { const n = jt.Parse(() => new Zo(e.name, t, e.shaderPath, e.options, e.storeEffectOnSubMeshes), e, t, r); let i; e.stencil && n.stencil.parse(e.stencil, t, r); for (i in e.textures) n.setTexture(i, We.Parse(e.textures[i], t, r)); for (i in e.textureArrays) { const s = e.textureArrays[i], a = []; for (let f = 0; f < s.length; f++) a.push(We.Parse(s[f], t, r)); n.setTextureArray(i, a); } for (i in e.ints) n.setInt(i, e.ints[i]); for (i in e.uints) n.setUInt(i, e.uints[i]); for (i in e.floats) n.setFloat(i, e.floats[i]); for (i in e.floatsArrays) n.setFloats(i, e.floatsArrays[i]); for (i in e.colors3) n.setColor3(i, Ne.FromArray(e.colors3[i])); for (i in e.colors3Arrays) { const s = e.colors3Arrays[i].reduce((a, f, o) => (o % 3 === 0 ? a.push([f]) : a[a.length - 1].push(f), a), []).map((a) => Ne.FromArray(a)); n.setColor3Array(i, s); } for (i in e.colors4) n.setColor4(i, xt.FromArray(e.colors4[i])); for (i in e.colors4Arrays) { const s = e.colors4Arrays[i].reduce((a, f, o) => (o % 4 === 0 ? a.push([f]) : a[a.length - 1].push(f), a), []).map((a) => xt.FromArray(a)); n.setColor4Array(i, s); } for (i in e.vectors2) n.setVector2(i, at.FromArray(e.vectors2[i])); for (i in e.vectors3) n.setVector3(i, S.FromArray(e.vectors3[i])); for (i in e.vectors4) n.setVector4(i, Ir.FromArray(e.vectors4[i])); for (i in e.quaternions) n.setQuaternion(i, Ze.FromArray(e.quaternions[i])); for (i in e.matrices) n.setMatrix(i, he.FromArray(e.matrices[i])); for (i in e.matrixArray) n._matrixArrays[i] = new Float32Array(e.matrixArray[i]); for (i in e.matrices3x3) n.setMatrix3x3(i, e.matrices3x3[i]); for (i in e.matrices2x2) n.setMatrix2x2(i, e.matrices2x2[i]); for (i in e.vectors2Arrays) n.setArray2(i, e.vectors2Arrays[i]); for (i in e.vectors3Arrays) n.setArray3(i, e.vectors3Arrays[i]); for (i in e.vectors4Arrays) n.setArray4(i, e.vectors4Arrays[i]); for (i in e.quaternionsArrays) n.setArray4(i, e.quaternionsArrays[i]); return n; } /** * Creates a new ShaderMaterial from a snippet saved in a remote file * @param name defines the name of the ShaderMaterial to create (can be null or empty to use the one from the json data) * @param url defines the url to load from * @param scene defines the hosting scene * @param rootUrl defines the root URL to use to load textures and relative dependencies * @returns a promise that will resolve to the new ShaderMaterial */ static ParseFromFileAsync(e, t, r, n = "") { return new Promise((i, s) => { const a = new ho(); a.addEventListener("readystatechange", () => { if (a.readyState == 4) if (a.status == 200) { const f = JSON.parse(a.responseText), o = this.Parse(f, r || gr.LastCreatedScene, n); e && (o.name = e), i(o); } else s("Unable to load the ShaderMaterial"); }), a.open("GET", t), a.send(); }); } /** * Creates a ShaderMaterial from a snippet saved by the Inspector * @param snippetId defines the snippet to load * @param scene defines the hosting scene * @param rootUrl defines the root URL to use to load textures and relative dependencies * @returns a promise that will resolve to the new ShaderMaterial */ static ParseFromSnippetAsync(e, t, r = "") { return new Promise((n, i) => { const s = new ho(); s.addEventListener("readystatechange", () => { if (s.readyState == 4) if (s.status == 200) { const a = JSON.parse(JSON.parse(s.responseText).jsonPayload), f = JSON.parse(a.shaderMaterial), o = this.Parse(f, t || gr.LastCreatedScene, r); o.snippetId = e, n(o); } else i("Unable to load the snippet " + e); }), s.open("GET", this.SnippetUrl + "/" + e.replace(/#/g, "/")), s.send(); }); } } Zo.SnippetUrl = "https://snippet.babylonjs.com"; Zo.CreateFromSnippetAsync = Zo.ParseFromSnippetAsync; Ue("BABYLON.ShaderMaterial", Zo); const yve = "colorPixelShader", kve = `#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES) #define VERTEXCOLOR varying vec4 vColor; #else uniform vec4 color; #endif #include #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #define CUSTOM_FRAGMENT_MAIN_BEGIN #include #if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES) gl_FragColor=vColor; #else gl_FragColor=color; #endif #define CUSTOM_FRAGMENT_MAIN_END }`; Le.ShadersStore[yve] = kve; const Eve = "colorVertexShader", Fve = `attribute vec3 position; #ifdef VERTEXCOLOR attribute vec4 color; #endif #include #include #include #include uniform mat4 viewProjection; #ifdef MULTIVIEW uniform mat4 viewProjectionR; #endif #if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES) varying vec4 vColor; #endif #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN #include #include #include vec4 worldPos=finalWorld*vec4(position,1.0); #ifdef MULTIVIEW if (gl_ViewID_OVR==0u) {gl_Position=viewProjection*worldPos;} else {gl_Position=viewProjectionR*worldPos;} #else gl_Position=viewProjection*worldPos; #endif #include #include #define CUSTOM_VERTEX_MAIN_END }`; Le.ShadersStore[Eve] = Fve; Ee._LinesMeshParser = (A, e) => Tp.Parse(A, e); class Tp extends Ee { _isShaderMaterial(e) { return e.getClassName() === "ShaderMaterial"; } /** * Creates a new LinesMesh * @param name defines the name * @param scene defines the hosting scene * @param parent defines the parent mesh if any * @param source defines the optional source LinesMesh used to clone data from * @param doNotCloneChildren When cloning, skip cloning child meshes of source, default False. * When false, achieved by calling a clone(), also passing False. * This will make creation of children, recursive. * @param useVertexColor defines if this LinesMesh supports vertex color * @param useVertexAlpha defines if this LinesMesh supports vertex alpha * @param material material to use to draw the line. If not provided, will create a new one */ constructor(e, t = null, r = null, n = null, i, s, a, f) { super(e, t, r, n, i), this.useVertexColor = s, this.useVertexAlpha = a, this.color = new Ne(1, 1, 1), this.alpha = 1, n && (this.color = n.color.clone(), this.alpha = n.alpha, this.useVertexColor = n.useVertexColor, this.useVertexAlpha = n.useVertexAlpha), this.intersectionThreshold = 0.1; const o = [], d = { attributes: [J.PositionKind], uniforms: ["world", "viewProjection"], needAlphaBlending: !0, defines: o, useClipPlane: null }; a === !1 ? d.needAlphaBlending = !1 : d.defines.push("#define VERTEXALPHA"), s ? (d.defines.push("#define VERTEXCOLOR"), d.attributes.push(J.ColorKind)) : (d.uniforms.push("color"), this._color4 = new xt()), f ? this.material = f : (this.material = new Zo("colorShader", this.getScene(), "color", d, !1), this.material.doNotSerialize = !0); } isReady() { return this._lineMaterial.isReady(this, !!this._userInstancedBuffersStorage || this.hasThinInstances) ? super.isReady() : !1; } /** * Returns the string "LineMesh" */ getClassName() { return "LinesMesh"; } /** * @internal */ get material() { return this._lineMaterial; } /** * @internal */ set material(e) { this._lineMaterial = e, this._lineMaterial.fillMode = gt.LineListDrawMode; } /** * @internal */ get checkCollisions() { return !1; } set checkCollisions(e) { } /** * @internal */ _bind(e, t) { if (!this._geometry) return this; const r = this.isUnIndexed ? null : this._geometry.getIndexBuffer(); if (!this._userInstancedBuffersStorage || this.hasThinInstances ? this._geometry._bind(t, r) : this._geometry._bind(t, r, this._userInstancedBuffersStorage.vertexBuffers, this._userInstancedBuffersStorage.vertexArrayObjects), !this.useVertexColor && this._isShaderMaterial(this._lineMaterial)) { const { r: n, g: i, b: s } = this.color; this._color4.set(n, i, s, this.alpha), this._lineMaterial.setColor4("color", this._color4); } return this; } /** * @internal */ _draw(e, t, r) { if (!this._geometry || !this._geometry.getVertexBuffers() || !this._unIndexed && !this._geometry.getIndexBuffer()) return this; const n = this.getScene().getEngine(); return this._unIndexed ? n.drawArraysType(gt.LineListDrawMode, e.verticesStart, e.verticesCount, r) : n.drawElementsType(gt.LineListDrawMode, e.indexStart, e.indexCount, r), this; } /** * Disposes of the line mesh * @param doNotRecurse If children should be disposed * @param disposeMaterialAndTextures This parameter is not used by the LineMesh class * @param doNotDisposeMaterial If the material should not be disposed (default: false, meaning the material is disposed) */ // eslint-disable-next-line @typescript-eslint/no-unused-vars dispose(e, t = !1, r) { r || this._lineMaterial.dispose(!1, !1, !0), super.dispose(e); } /** * Returns a new LineMesh object cloned from the current one. * @param name * @param newParent * @param doNotCloneChildren */ clone(e, t = null, r) { return new Tp(e, this.getScene(), t, this, r); } /** * Creates a new InstancedLinesMesh object from the mesh model. * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/copies/instances * @param name defines the name of the new instance * @returns a new InstancedLinesMesh */ createInstance(e) { const t = new SQ(e, this); if (this.instancedBuffers) { t.instancedBuffers = {}; for (const r in this.instancedBuffers) t.instancedBuffers[r] = this.instancedBuffers[r]; } return t; } /** * Serializes this ground mesh * @param serializationObject object to write serialization to */ serialize(e) { super.serialize(e), e.color = this.color.asArray(), e.alpha = this.alpha; } /** * Parses a serialized ground mesh * @param parsedMesh the serialized mesh * @param scene the scene to create the ground mesh in * @returns the created ground mesh */ static Parse(e, t) { const r = new Tp(e.name, t); return r.color = Ne.FromArray(e.color), r.alpha = e.alpha, r; } } class SQ extends bp { constructor(e, t) { super(e, t), this.intersectionThreshold = t.intersectionThreshold; } /** * Returns the string "InstancedLinesMesh". */ getClassName() { return "InstancedLinesMesh"; } } function UQ(A) { const e = [], t = [], r = A.lines, n = A.colors, i = []; let s = 0; for (let f = 0; f < r.length; f++) { const o = r[f]; for (let d = 0; d < o.length; d++) { const { x: v, y: u, z: l } = o[d]; if (t.push(v, u, l), n) { const P = n[f], { r: p, g: c, b: H, a: T } = P[d]; i.push(p, c, H, T); } d > 0 && (e.push(s - 1), e.push(s)), s++; } } const a = new Ut(); return a.indices = e, a.positions = t, n && (a.colors = i), a; } function IQ(A) { const e = A.dashSize || 3, t = A.gapSize || 1, r = A.dashNb || 200, n = A.points, i = [], s = [], a = S.Zero(); let f = 0, o = 0, d = 0, v = 0, u = 0, l = 0, P = 0; for (P = 0; P < n.length - 1; P++) n[P + 1].subtractToRef(n[P], a), f += a.length(); for (d = f / r, v = e * d / (e + t), P = 0; P < n.length - 1; P++) { n[P + 1].subtractToRef(n[P], a), o = Math.floor(a.length() / d), a.normalize(); for (let c = 0; c < o; c++) u = d * c, i.push(n[P].x + u * a.x, n[P].y + u * a.y, n[P].z + u * a.z), i.push(n[P].x + (u + v) * a.x, n[P].y + (u + v) * a.y, n[P].z + (u + v) * a.z), s.push(l, l + 1), l += 2; } const p = new Ut(); return p.positions = i, p.indices = s, p; } function rm(A, e, t = null) { const r = e.instance, n = e.lines, i = e.colors; if (r) { const o = r.getVerticesData(J.PositionKind); let d, v; i && (d = r.getVerticesData(J.ColorKind)); let u = 0, l = 0; for (let P = 0; P < n.length; P++) { const p = n[P]; for (let c = 0; c < p.length; c++) o[u] = p[c].x, o[u + 1] = p[c].y, o[u + 2] = p[c].z, i && d && (v = i[P], d[l] = v[c].r, d[l + 1] = v[c].g, d[l + 2] = v[c].b, d[l + 3] = v[c].a, l += 4), u += 3; } return r.updateVerticesData(J.PositionKind, o, !1, !1), i && d && r.updateVerticesData(J.ColorKind, d, !1, !1), r; } const s = !!i, a = new Tp(A, t, null, void 0, void 0, s, e.useVertexAlpha, e.material); return UQ(e).applyToMesh(a, e.updatable), a; } function ka(A, e, t = null) { const r = e.colors ? [e.colors] : null; return rm(A, { lines: [e.points], updatable: e.updatable, instance: e.instance, colors: r, useVertexAlpha: e.useVertexAlpha, material: e.material }, t); } function ZO(A, e, t = null) { const r = e.points, n = e.instance, i = e.gapSize || 1, s = e.dashSize || 3; if (n) { const o = (d) => { const v = S.Zero(), u = d.length / 6; let l = 0, P = 0, p = 0, c = 0, H = 0, T = 0, q = 0, b = 0; for (q = 0; q < r.length - 1; q++) r[q + 1].subtractToRef(r[q], v), l += v.length(); p = l / u; const j = n._creationDataStorage.dashSize, w = n._creationDataStorage.gapSize; for (c = j * p / (j + w), q = 0; q < r.length - 1; q++) for (r[q + 1].subtractToRef(r[q], v), P = Math.floor(v.length() / p), v.normalize(), b = 0; b < P && T < d.length; ) H = p * b, d[T] = r[q].x + H * v.x, d[T + 1] = r[q].y + H * v.y, d[T + 2] = r[q].z + H * v.z, d[T + 3] = r[q].x + (H + c) * v.x, d[T + 4] = r[q].y + (H + c) * v.y, d[T + 5] = r[q].z + (H + c) * v.z, T += 6, b++; for (; T < d.length; ) d[T] = r[q].x, d[T + 1] = r[q].y, d[T + 2] = r[q].z, T += 3; }; return (e.dashNb || e.dashSize || e.gapSize || e.useVertexAlpha || e.material) && Se.Warn("You have used an option other than points with the instance option. Please be aware that these other options will be ignored."), n.updateMeshPositions(o, !1), n; } const a = new Tp(A, t, null, void 0, void 0, void 0, e.useVertexAlpha, e.material); return IQ(e).applyToMesh(a, e.updatable), a._creationDataStorage = new tQ(), a._creationDataStorage.dashSize = s, a._creationDataStorage.gapSize = i, a; } const Nve = { CreateDashedLines: ZO, CreateLineSystem: rm, CreateLines: ka }; Ut.CreateLineSystem = UQ; Ut.CreateDashedLines = IQ; Ee.CreateLines = (A, e, t = null, r = !1, n = null) => ka(A, { points: e, updatable: r, instance: n }, t); Ee.CreateDashedLines = (A, e, t, r, n, i = null, s, a) => ZO(A, { points: e, dashSize: t, gapSize: r, dashNb: n, updatable: s, instance: a }, i); class Qve extends at { constructor(e, t) { super(e.x, e.y), this.index = t; } } class bE { constructor() { this.elements = []; } add(e) { const t = []; return e.forEach((r) => { const n = new Qve(r, this.elements.length); t.push(n), this.elements.push(n); }), t; } computeBounds() { const e = new at(this.elements[0].x, this.elements[0].y), t = new at(this.elements[0].x, this.elements[0].y); return this.elements.forEach((r) => { r.x < e.x ? e.x = r.x : r.x > t.x && (t.x = r.x), r.y < e.y ? e.y = r.y : r.y > t.y && (t.y = r.y); }), { min: e, max: t, width: t.x - e.x, height: t.y - e.y }; } } class Yve { /** * Creates a rectangle * @param xmin bottom X coord * @param ymin bottom Y coord * @param xmax top X coord * @param ymax top Y coord * @returns points that make the resulting rectangle */ static Rectangle(e, t, r, n) { return [new at(e, t), new at(r, t), new at(r, n), new at(e, n)]; } /** * Creates a circle * @param radius radius of circle * @param cx scale in x * @param cy scale in y * @param numberOfSides number of sides that make up the circle * @returns points that make the resulting circle */ static Circle(e, t = 0, r = 0, n = 32) { const i = []; let s = 0; const a = Math.PI * 2 / n; for (let f = 0; f < n; f++) i.push(new at(t + Math.cos(s) * e, r + Math.sin(s) * e)), s -= a; return i; } /** * Creates a polygon from input string * @param input Input polygon data * @returns the parsed points */ static Parse(e) { const t = e.split(/[^-+eE.\d]+/).map(parseFloat).filter((i) => !isNaN(i)); let r; const n = []; for (r = 0; r < (t.length & 2147483646); r += 2) n.push(new at(t[r], t[r + 1])); return n; } /** * Starts building a polygon from x and y coordinates * @param x x coordinate * @param y y coordinate * @returns the started path2 */ static StartingAt(e, t) { return sU.StartingAt(e, t); } } class ete { _addToepoint(e) { for (const t of e) this._epoints.push(t.x, t.y); } /** * Creates a PolygonMeshBuilder * @param name name of the builder * @param contours Path of the polygon * @param scene scene to add to when creating the mesh * @param earcutInjection can be used to inject your own earcut reference */ constructor(e, t, r, n = earcut) { this._points = new bE(), this._outlinepoints = new bE(), this._holes = new Array(), this._epoints = new Array(), this._eholes = new Array(), this.bjsEarcut = n, this._name = e, this._scene = r || gr.LastCreatedScene; let i; t instanceof sU ? i = t.getPoints() : i = t, this._addToepoint(i), this._points.add(i), this._outlinepoints.add(i), typeof this.bjsEarcut > "u" && Se.Warn("Earcut was not found, the polygon will not be built."); } /** * Adds a hole within the polygon * @param hole Array of points defining the hole * @returns this */ addHole(e) { this._points.add(e); const t = new bE(); return t.add(e), this._holes.push(t), this._eholes.push(this._epoints.length / 2), this._addToepoint(e), this; } /** * Creates the polygon * @param updatable If the mesh should be updatable * @param depth The depth of the mesh created * @param smoothingThreshold Dot product threshold for smoothed normals * @returns the created mesh */ build(e = !1, t = 0, r = 2) { const n = new Ee(this._name, this._scene), i = this.buildVertexData(t, r); return n.setVerticesData(J.PositionKind, i.positions, e), n.setVerticesData(J.NormalKind, i.normals, e), n.setVerticesData(J.UVKind, i.uvs, e), n.setIndices(i.indices), n; } /** * Creates the polygon * @param depth The depth of the mesh created * @param smoothingThreshold Dot product threshold for smoothed normals * @returns the created VertexData */ buildVertexData(e = 0, t = 2) { const r = new Ut(), n = [], i = [], s = [], a = this._points.computeBounds(); this._points.elements.forEach((d) => { n.push(0, 1, 0), i.push(d.x, 0, d.y), s.push((d.x - a.min.x) / a.width, (d.y - a.min.y) / a.height); }); const f = [], o = this.bjsEarcut(this._epoints, this._eholes, 2); for (let d = 0; d < o.length; d++) f.push(o[d]); if (e > 0) { const d = i.length / 3; this._points.elements.forEach((u) => { n.push(0, -1, 0), i.push(u.x, -e, u.y), s.push(1 - (u.x - a.min.x) / a.width, 1 - (u.y - a.min.y) / a.height); }); const v = f.length; for (let u = 0; u < v; u += 3) { const l = f[u + 0], P = f[u + 1], p = f[u + 2]; f.push(p + d), f.push(P + d), f.push(l + d); } this._addSide(i, n, s, f, a, this._outlinepoints, e, !1, t), this._holes.forEach((u) => { this._addSide(i, n, s, f, a, u, e, !0, t); }); } return r.indices = f, r.positions = i, r.normals = n, r.uvs = s, r; } /** * Adds a side to the polygon * @param positions points that make the polygon * @param normals normals of the polygon * @param uvs uvs of the polygon * @param indices indices of the polygon * @param bounds bounds of the polygon * @param points points of the polygon * @param depth depth of the polygon * @param flip flip of the polygon * @param smoothingThreshold */ _addSide(e, t, r, n, i, s, a, f, o) { let d = e.length / 3, v = 0; for (let u = 0; u < s.elements.length; u++) { const l = s.elements[u], P = s.elements[(u + 1) % s.elements.length]; e.push(l.x, 0, l.y), e.push(l.x, -a, l.y), e.push(P.x, 0, P.y), e.push(P.x, -a, P.y); const p = s.elements[(u + s.elements.length - 1) % s.elements.length], c = s.elements[(u + 2) % s.elements.length]; let H = new S(-(P.y - l.y), 0, P.x - l.x), T = new S(-(l.y - p.y), 0, l.x - p.x), q = new S(-(c.y - P.y), 0, c.x - P.x); f || (H = H.scale(-1), T = T.scale(-1), q = q.scale(-1)); const b = H.normalizeToNew(); let j = T.normalizeToNew(), w = q.normalizeToNew(); const m = S.Dot(j, b); m > o ? m < Dn - 1 ? j = new S(l.x, 0, l.y).subtract(new S(P.x, 0, P.y)).normalize() : j = T.add(H).normalize() : j = b; const I = S.Dot(q, H); I > o ? I < Dn - 1 ? w = new S(P.x, 0, P.y).subtract(new S(l.x, 0, l.y)).normalize() : w = q.add(H).normalize() : w = b, r.push(v / i.width, 0), r.push(v / i.width, 1), v += H.length(), r.push(v / i.width, 0), r.push(v / i.width, 1), t.push(j.x, j.y, j.z), t.push(j.x, j.y, j.z), t.push(w.x, w.y, w.z), t.push(w.x, w.y, w.z), f ? (n.push(d), n.push(d + 2), n.push(d + 1), n.push(d + 1), n.push(d + 2), n.push(d + 3)) : (n.push(d), n.push(d + 1), n.push(d + 2), n.push(d + 1), n.push(d + 3), n.push(d + 2)), d += 4; } } } function RQ(A, e, t, r, n, i, s) { const a = t || new Array(3), f = r, o = [], d = s || !1; for (let N = 0; N < 3; N++) a[N] === void 0 && (a[N] = new Ir(0, 0, 1, 1)), f && f[N] === void 0 && (f[N] = new xt(1, 1, 1, 1)); const v = A.getVerticesData(J.PositionKind), u = A.getVerticesData(J.NormalKind), l = A.getVerticesData(J.UVKind), P = A.getIndices(), p = v.length / 9; let c = 0, H = 0, T = 0, q = 0, b = 0; const j = [0]; if (d) for (let N = p; N < v.length / 3; N += 4) H = v[3 * (N + 2)] - v[3 * N], T = v[3 * (N + 2) + 2] - v[3 * N + 2], q = Math.sqrt(H * H + T * T), b += q, j.push(b); let w = 0, m = 0; for (let N = 0; N < u.length; N += 3) Math.abs(u[N + 1]) < 1e-3 && (m = 1), Math.abs(u[N + 1] - 1) < 1e-3 && (m = 0), Math.abs(u[N + 1] + 1) < 1e-3 && (m = 2), w = N / 3, m === 1 ? (c = w - p, c % 4 < 1.5 ? d ? l[2 * w] = a[m].x + (a[m].z - a[m].x) * j[Math.floor(c / 4)] / b : l[2 * w] = a[m].x : d ? l[2 * w] = a[m].x + (a[m].z - a[m].x) * j[Math.floor(c / 4) + 1] / b : l[2 * w] = a[m].z, c % 2 === 0 ? l[2 * w + 1] = us.UseOpenGLOrientationForUV ? 1 - a[m].w : a[m].w : l[2 * w + 1] = us.UseOpenGLOrientationForUV ? 1 - a[m].y : a[m].y) : (l[2 * w] = (1 - l[2 * w]) * a[m].x + l[2 * w] * a[m].z, l[2 * w + 1] = (1 - l[2 * w + 1]) * a[m].y + l[2 * w + 1] * a[m].w, us.UseOpenGLOrientationForUV && (l[2 * w + 1] = 1 - l[2 * w + 1])), f && o.push(f[m].r, f[m].g, f[m].b, f[m].a); Ut._ComputeSides(e, v, P, u, l, n, i); const I = new Ut(); if (I.indices = P, I.positions = v, I.normals = u, I.uvs = l, f) { const N = e === Ut.DOUBLESIDE ? o.concat(o) : o; I.colors = N; } return I; } function JR(A, e, t = null, r = earcut) { e.sideOrientation = Ee._GetDefaultSideOrientation(e.sideOrientation); const n = e.shape, i = e.holes || [], s = e.depth || 0, a = e.smoothingThreshold || 2, f = []; let o = []; for (let P = 0; P < n.length; P++) f[P] = new at(n[P].x, n[P].z); const d = 1e-8; f[0].equalsWithEpsilon(f[f.length - 1], d) && f.pop(); const v = new ete(A, f, t || gr.LastCreatedScene, r); for (let P = 0; P < i.length; P++) { o = []; for (let p = 0; p < i[P].length; p++) o.push(new at(i[P][p].x, i[P][p].z)); v.addHole(o); } const u = v.build(!1, s, a); return u._originalBuilderSideOrientation = e.sideOrientation, RQ(u, e.sideOrientation, e.faceUV, e.faceColors, e.frontUVs, e.backUVs, e.wrap).applyToMesh(u, e.updatable), u; } function zR(A, e, t = null, r = earcut) { return JR(A, e, t, r); } const Mve = { ExtrudePolygon: zR, CreatePolygon: JR }; Ut.CreatePolygon = RQ; Ee.CreatePolygon = (A, e, t, r, n, i, s = earcut) => JR(A, { shape: e, holes: r, updatable: n, sideOrientation: i }, t, s); Ee.ExtrudePolygon = (A, e, t, r, n, i, s, a = earcut) => zR(A, { shape: e, holes: n, depth: t, updatable: i, sideOrientation: s }, r, a); function GR(A, e, t = null) { const r = e.path, n = e.shape, i = e.scale || 1, s = e.rotation || 0, a = e.cap === 0 ? 0 : e.cap || Ee.NO_CAP, f = e.updatable, o = Ee._GetDefaultSideOrientation(e.sideOrientation), d = e.instance || null, v = e.invertUV || !1, u = e.closeShape || !1, l = e.closePath || !1; return tte(A, n, r, i, s, null, null, l, u, a, !1, t, !!f, o, d, v, e.frontUVs || null, e.backUVs || null, e.firstNormal || null, !!e.adjustFrame); } function ZR(A, e, t = null) { const r = e.path, n = e.shape, i = e.scaleFunction || (() => 1), s = e.rotationFunction || (() => 0), a = e.closePath || e.ribbonCloseArray || !1, f = e.closeShape || e.ribbonClosePath || !1, o = e.cap === 0 ? 0 : e.cap || Ee.NO_CAP, d = e.updatable, v = e.firstNormal || null, u = e.adjustFrame || !1, l = Ee._GetDefaultSideOrientation(e.sideOrientation), P = e.instance, p = e.invertUV || !1; return tte(A, n, r, null, null, i, s, a, f, o, !0, t, !!d, l, P || null, p, e.frontUVs || null, e.backUVs || null, v, u); } function tte(A, e, t, r, n, i, s, a, f, o, d, v, u, l, P, p, c, H, T, q) { const b = (N, k, R, y, O, Y, ee, Z, te, fe, _) => { const G = R.getTangents(), L = R.getNormals(), $ = R.getBinormals(), ae = R.getDistances(); if (_) { for (let qe = 0; qe < G.length; qe++) if (G[qe].x == 0 && G[qe].y == 0 && G[qe].z == 0 && G[qe].copyFrom(G[qe - 1]), L[qe].x == 0 && L[qe].y == 0 && L[qe].z == 0 && L[qe].copyFrom(L[qe - 1]), $[qe].x == 0 && $[qe].y == 0 && $[qe].z == 0 && $[qe].copyFrom($[qe - 1]), qe > 0) { let ke = G[qe - 1]; S.Dot(ke, G[qe]) < 0 && G[qe].scaleInPlace(-1), ke = L[qe - 1], S.Dot(ke, L[qe]) < 0 && L[qe].scaleInPlace(-1), ke = $[qe - 1], S.Dot(ke, $[qe]) < 0 && $[qe].scaleInPlace(-1); } } let Pe = 0; const ge = () => O !== null ? O : 1, Xe = fe && Z ? Z : () => Y !== null ? Y : 0, De = fe && ee ? ee : ge; let ne = te === Ee.NO_CAP || te === Ee.CAP_END ? 0 : 2; const re = ue.Matrix[0]; for (let qe = 0; qe < k.length; qe++) { const ke = [], be = Xe(qe, ae[qe]), Fe = De(qe, ae[qe]); he.RotationAxisToRef(G[qe], Pe, re); for (let Ke = 0; Ke < N.length; Ke++) { const nt = G[qe].scale(N[Ke].z).add(L[qe].scale(N[Ke].x)).add($[qe].scale(N[Ke].y)), ut = S.Zero(); S.TransformCoordinatesToRef(nt, re, ut), ut.scaleInPlace(Fe).addInPlace(k[qe]), ke[Ke] = ut; } y[ne] = ke, Pe += be, ne++; } const ve = (qe) => { const ke = Array(), be = S.Zero(); let Fe; for (Fe = 0; Fe < qe.length; Fe++) be.addInPlace(qe[Fe]); for (be.scaleInPlace(1 / qe.length), Fe = 0; Fe < qe.length; Fe++) ke.push(be); return ke; }; switch (te) { case Ee.NO_CAP: break; case Ee.CAP_START: y[0] = ve(y[2]), y[1] = y[2]; break; case Ee.CAP_END: y[ne] = y[ne - 1], y[ne + 1] = ve(y[ne - 1]); break; case Ee.CAP_ALL: y[0] = ve(y[2]), y[1] = y[2], y[ne] = y[ne - 1], y[ne + 1] = ve(y[ne - 1]); break; } return y; }; let j, w; if (P) { const N = P._creationDataStorage; return j = T ? N.path3D.update(t, T) : N.path3D.update(t), w = b(e, t, N.path3D, N.pathArray, r, n, i, s, N.cap, d, q), P = n4("", { pathArray: w, closeArray: !1, closePath: !1, offset: 0, updatable: !1, sideOrientation: 0, instance: P }, v || void 0), P; } j = T ? new vm(t, T) : new vm(t); const m = new Array(); o = o < 0 || o > 3 ? 0 : o, w = b(e, t, j, m, r, n, i, s, o, d, q); const I = n4(A, { pathArray: w, closeArray: a, closePath: f, updatable: u, sideOrientation: l, invertUV: p, frontUVs: c || void 0, backUVs: H || void 0 }, v); return I._creationDataStorage.pathArray = w, I._creationDataStorage.path3D = j, I._creationDataStorage.cap = o, I; } const Lve = { // eslint-disable-next-line @typescript-eslint/naming-convention ExtrudeShape: GR, // eslint-disable-next-line @typescript-eslint/naming-convention ExtrudeShapeCustom: ZR }; Ee.ExtrudeShape = (A, e, t, r, n, i, s = null, a, f, o) => { const d = { shape: e, path: t, scale: r, rotation: n, cap: i === 0 ? 0 : i || Ee.NO_CAP, sideOrientation: f, instance: o, updatable: a }; return GR(A, d, s); }; Ee.ExtrudeShapeCustom = (A, e, t, r, n, i, s, a, f, o, d, v) => { const u = { shape: e, path: t, scaleFunction: r, rotationFunction: n, ribbonCloseArray: i, ribbonClosePath: s, cap: a === 0 ? 0 : a || Ee.NO_CAP, sideOrientation: d, instance: v, updatable: o }; return ZR(A, u, f); }; function _O(A, e, t = null) { const r = e.arc ? e.arc <= 0 || e.arc > 1 ? 1 : e.arc : 1, n = e.closed === void 0 ? !0 : e.closed, i = e.shape, s = e.radius || 1, a = e.tessellation || 64, f = e.clip || 0, o = e.updatable, d = Ee._GetDefaultSideOrientation(e.sideOrientation), v = e.cap || Ee.NO_CAP, u = Math.PI * 2, l = [], P = e.invertUV || !1; let p = 0, c = 0; const H = u / a * r; let T, q; for (p = 0; p <= a - f; p++) { for (q = [], (v == Ee.CAP_START || v == Ee.CAP_ALL) && (q.push(new S(0, i[0].y, 0)), q.push(new S(Math.cos(p * H) * i[0].x * s, i[0].y, Math.sin(p * H) * i[0].x * s))), c = 0; c < i.length; c++) T = new S(Math.cos(p * H) * i[c].x * s, i[c].y, Math.sin(p * H) * i[c].x * s), q.push(T); (v == Ee.CAP_END || v == Ee.CAP_ALL) && (q.push(new S(Math.cos(p * H) * i[i.length - 1].x * s, i[i.length - 1].y, Math.sin(p * H) * i[i.length - 1].x * s)), q.push(new S(0, i[i.length - 1].y, 0))), l.push(q); } return n4(A, { pathArray: l, closeArray: n, sideOrientation: d, updatable: o, invertUV: P, frontUVs: e.frontUVs, backUVs: e.backUVs }, t); } const Kve = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateLathe: _O }; Ee.CreateLathe = (A, e, t, r, n, i, s) => _O(A, { shape: e, radius: t, tessellation: r, sideOrientation: s, updatable: i }, n); function $O(A, e, t = null) { const r = e.path; let n = e.instance, i = 1; e.radius !== void 0 ? i = e.radius : n && (i = n._creationDataStorage.radius); const s = e.tessellation || 64, a = e.radiusFunction || null; let f = e.cap || Ee.NO_CAP; const o = e.invertUV || !1, d = e.updatable, v = Ee._GetDefaultSideOrientation(e.sideOrientation); e.arc = e.arc && (e.arc <= 0 || e.arc > 1) ? 1 : e.arc || 1; const u = (H, T, q, b, j, w, m, I) => { const N = T.getTangents(), k = T.getNormals(), R = T.getDistances(), O = Math.PI * 2 / j * I, ee = w || (() => b); let Z, te, fe, _; const G = ue.Matrix[0]; let L = m === Ee.NO_CAP || m === Ee.CAP_END ? 0 : 2; for (let ae = 0; ae < H.length; ae++) { te = ee(ae, R[ae]), Z = Array(), fe = k[ae]; for (let Pe = 0; Pe < j; Pe++) he.RotationAxisToRef(N[ae], O * Pe, G), _ = Z[Pe] ? Z[Pe] : S.Zero(), S.TransformCoordinatesToRef(fe, G, _), _.scaleInPlace(te).addInPlace(H[ae]), Z[Pe] = _; q[L] = Z, L++; } const $ = (ae, Pe) => { const ge = Array(); for (let me = 0; me < ae; me++) ge.push(H[Pe]); return ge; }; switch (m) { case Ee.NO_CAP: break; case Ee.CAP_START: q[0] = $(j, 0), q[1] = q[2].slice(0); break; case Ee.CAP_END: q[L] = q[L - 1].slice(0), q[L + 1] = $(j, H.length - 1); break; case Ee.CAP_ALL: q[0] = $(j, 0), q[1] = q[2].slice(0), q[L] = q[L - 1].slice(0), q[L + 1] = $(j, H.length - 1); break; } return q; }; let l, P; if (n) { const H = n._creationDataStorage, T = e.arc || H.arc; return l = H.path3D.update(r), P = u(r, l, H.pathArray, i, H.tessellation, a, H.cap, T), n = n4("", { pathArray: P, instance: n }), H.path3D = l, H.pathArray = P, H.arc = T, H.radius = i, n; } l = new vm(r); const p = new Array(); f = f < 0 || f > 3 ? 0 : f, P = u(r, l, p, i, s, a, f, e.arc); const c = n4(A, { pathArray: P, closePath: !0, closeArray: !1, updatable: d, sideOrientation: v, invertUV: o, frontUVs: e.frontUVs, backUVs: e.backUVs }, t); return c._creationDataStorage.pathArray = P, c._creationDataStorage.path3D = l, c._creationDataStorage.tessellation = s, c._creationDataStorage.cap = f, c._creationDataStorage.arc = e.arc, c._creationDataStorage.radius = i, c; } const Jve = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateTube: $O }; Ee.CreateTube = (A, e, t, r, n, i, s, a, f, o) => $O(A, { path: e, radius: t, tessellation: r, radiusFunction: n, arc: 1, cap: i, updatable: a, sideOrientation: f, instance: o }, s); function VQ(A) { const e = []; e[0] = { vertex: [ [0, 0, 1.732051], [1.632993, 0, -0.5773503], [-0.8164966, 1.414214, -0.5773503], [-0.8164966, -1.414214, -0.5773503] ], face: [ [0, 1, 2], [0, 2, 3], [0, 3, 1], [1, 3, 2] ] }, e[1] = { vertex: [ [0, 0, 1.414214], [1.414214, 0, 0], [0, 1.414214, 0], [-1.414214, 0, 0], [0, -1.414214, 0], [0, 0, -1.414214] ], face: [ [0, 1, 2], [0, 2, 3], [0, 3, 4], [0, 4, 1], [1, 4, 5], [1, 5, 2], [2, 5, 3], [3, 5, 4] ] }, e[2] = { vertex: [ [0, 0, 1.070466], [0.7136442, 0, 0.7978784], [-0.3568221, 0.618034, 0.7978784], [-0.3568221, -0.618034, 0.7978784], [0.7978784, 0.618034, 0.3568221], [0.7978784, -0.618034, 0.3568221], [-0.9341724, 0.381966, 0.3568221], [0.1362939, 1, 0.3568221], [0.1362939, -1, 0.3568221], [-0.9341724, -0.381966, 0.3568221], [0.9341724, 0.381966, -0.3568221], [0.9341724, -0.381966, -0.3568221], [-0.7978784, 0.618034, -0.3568221], [-0.1362939, 1, -0.3568221], [-0.1362939, -1, -0.3568221], [-0.7978784, -0.618034, -0.3568221], [0.3568221, 0.618034, -0.7978784], [0.3568221, -0.618034, -0.7978784], [-0.7136442, 0, -0.7978784], [0, 0, -1.070466] ], face: [ [0, 1, 4, 7, 2], [0, 2, 6, 9, 3], [0, 3, 8, 5, 1], [1, 5, 11, 10, 4], [2, 7, 13, 12, 6], [3, 9, 15, 14, 8], [4, 10, 16, 13, 7], [5, 8, 14, 17, 11], [6, 12, 18, 15, 9], [10, 11, 17, 19, 16], [12, 13, 16, 19, 18], [14, 15, 18, 19, 17] ] }, e[3] = { vertex: [ [0, 0, 1.175571], [1.051462, 0, 0.5257311], [0.3249197, 1, 0.5257311], [-0.8506508, 0.618034, 0.5257311], [-0.8506508, -0.618034, 0.5257311], [0.3249197, -1, 0.5257311], [0.8506508, 0.618034, -0.5257311], [0.8506508, -0.618034, -0.5257311], [-0.3249197, 1, -0.5257311], [-1.051462, 0, -0.5257311], [-0.3249197, -1, -0.5257311], [0, 0, -1.175571] ], face: [ [0, 1, 2], [0, 2, 3], [0, 3, 4], [0, 4, 5], [0, 5, 1], [1, 5, 7], [1, 7, 6], [1, 6, 2], [2, 6, 8], [2, 8, 3], [3, 8, 9], [3, 9, 4], [4, 9, 10], [4, 10, 5], [5, 10, 7], [6, 7, 11], [6, 11, 8], [7, 10, 11], [8, 11, 9], [9, 11, 10] ] }, e[4] = { vertex: [ [0, 0, 1.070722], [0.7148135, 0, 0.7971752], [-0.104682, 0.7071068, 0.7971752], [-0.6841528, 0.2071068, 0.7971752], [-0.104682, -0.7071068, 0.7971752], [0.6101315, 0.7071068, 0.5236279], [1.04156, 0.2071068, 0.1367736], [0.6101315, -0.7071068, 0.5236279], [-0.3574067, 1, 0.1367736], [-0.7888348, -0.5, 0.5236279], [-0.9368776, 0.5, 0.1367736], [-0.3574067, -1, 0.1367736], [0.3574067, 1, -0.1367736], [0.9368776, -0.5, -0.1367736], [0.7888348, 0.5, -0.5236279], [0.3574067, -1, -0.1367736], [-0.6101315, 0.7071068, -0.5236279], [-1.04156, -0.2071068, -0.1367736], [-0.6101315, -0.7071068, -0.5236279], [0.104682, 0.7071068, -0.7971752], [0.6841528, -0.2071068, -0.7971752], [0.104682, -0.7071068, -0.7971752], [-0.7148135, 0, -0.7971752], [0, 0, -1.070722] ], face: [ [0, 2, 3], [1, 6, 5], [4, 9, 11], [7, 15, 13], [8, 16, 10], [12, 14, 19], [17, 22, 18], [20, 21, 23], [0, 1, 5, 2], [0, 3, 9, 4], [0, 4, 7, 1], [1, 7, 13, 6], [2, 5, 12, 8], [2, 8, 10, 3], [3, 10, 17, 9], [4, 11, 15, 7], [5, 6, 14, 12], [6, 13, 20, 14], [8, 12, 19, 16], [9, 17, 18, 11], [10, 16, 22, 17], [11, 18, 21, 15], [13, 15, 21, 20], [14, 20, 23, 19], [16, 19, 23, 22], [18, 22, 23, 21] ] }, e[5] = { vertex: [ [0, 0, 1.322876], [1.309307, 0, 0.1889822], [-0.9819805, 0.8660254, 0.1889822], [0.1636634, -1.299038, 0.1889822], [0.3273268, 0.8660254, -0.9449112], [-0.8183171, -0.4330127, -0.9449112] ], face: [ [0, 3, 1], [2, 4, 5], [0, 1, 4, 2], [0, 2, 5, 3], [1, 3, 5, 4] ] }, e[6] = { vertex: [ [0, 0, 1.159953], [1.013464, 0, 0.5642542], [-0.3501431, 0.9510565, 0.5642542], [-0.7715208, -0.6571639, 0.5642542], [0.6633206, 0.9510565, -0.03144481], [0.8682979, -0.6571639, -0.3996071], [-1.121664, 0.2938926, -0.03144481], [-0.2348831, -1.063314, -0.3996071], [0.5181548, 0.2938926, -0.9953061], [-0.5850262, -0.112257, -0.9953061] ], face: [ [0, 1, 4, 2], [0, 2, 6, 3], [1, 5, 8, 4], [3, 6, 9, 7], [5, 7, 9, 8], [0, 3, 7, 5, 1], [2, 4, 8, 9, 6] ] }, e[7] = { vertex: [ [0, 0, 1.118034], [0.8944272, 0, 0.6708204], [-0.2236068, 0.8660254, 0.6708204], [-0.7826238, -0.4330127, 0.6708204], [0.6708204, 0.8660254, 0.2236068], [1.006231, -0.4330127, -0.2236068], [-1.006231, 0.4330127, 0.2236068], [-0.6708204, -0.8660254, -0.2236068], [0.7826238, 0.4330127, -0.6708204], [0.2236068, -0.8660254, -0.6708204], [-0.8944272, 0, -0.6708204], [0, 0, -1.118034] ], face: [ [0, 1, 4, 2], [0, 2, 6, 3], [1, 5, 8, 4], [3, 6, 10, 7], [5, 9, 11, 8], [7, 10, 11, 9], [0, 3, 7, 9, 5, 1], [2, 4, 8, 11, 10, 6] ] }, e[8] = { vertex: [ [-0.729665, 0.670121, 0.319155], [-0.655235, -0.29213, -0.754096], [-0.093922, -0.607123, 0.537818], [0.702196, 0.595691, 0.485187], [0.776626, -0.36656, -0.588064] ], face: [ [1, 4, 2], [0, 1, 2], [3, 0, 2], [4, 3, 2], [4, 1, 0, 3] ] }, e[9] = { vertex: [ [-0.868849, -0.100041, 0.61257], [-0.329458, 0.976099, 0.28078], [-0.26629, -0.013796, -0.477654], [-0.13392, -1.034115, 0.229829], [0.738834, 0.707117, -0.307018], [0.859683, -0.535264, -0.338508] ], face: [ [3, 0, 2], [5, 3, 2], [4, 5, 2], [1, 4, 2], [0, 1, 2], [0, 3, 5, 4, 1] ] }, e[10] = { vertex: [ [-0.610389, 0.243975, 0.531213], [-0.187812, -0.48795, -0.664016], [-0.187812, 0.9759, -0.664016], [0.187812, -0.9759, 0.664016], [0.798201, 0.243975, 0.132803] ], face: [ [1, 3, 0], [3, 4, 0], [3, 1, 4], [0, 2, 1], [0, 4, 2], [2, 4, 1] ] }, e[11] = { vertex: [ [-1.028778, 0.392027, -0.048786], [-0.640503, -0.646161, 0.621837], [-0.125162, -0.395663, -0.540059], [4683e-6, 0.888447, -0.651988], [0.125161, 0.395663, 0.540059], [0.632925, -0.791376, 0.433102], [1.031672, 0.157063, -0.354165] ], face: [ [3, 2, 0], [2, 1, 0], [2, 5, 1], [0, 4, 3], [0, 1, 4], [4, 1, 5], [2, 3, 6], [3, 4, 6], [5, 2, 6], [4, 5, 6] ] }, e[12] = { vertex: [ [-0.669867, 0.334933, -0.529576], [-0.669867, 0.334933, 0.529577], [-0.4043, 1.212901, 0], [-0.334933, -0.669867, -0.529576], [-0.334933, -0.669867, 0.529577], [0.334933, 0.669867, -0.529576], [0.334933, 0.669867, 0.529577], [0.4043, -1.212901, 0], [0.669867, -0.334933, -0.529576], [0.669867, -0.334933, 0.529577] ], face: [ [8, 9, 7], [6, 5, 2], [3, 8, 7], [5, 0, 2], [4, 3, 7], [0, 1, 2], [9, 4, 7], [1, 6, 2], [9, 8, 5, 6], [8, 3, 0, 5], [3, 4, 1, 0], [4, 9, 6, 1] ] }, e[13] = { vertex: [ [-0.931836, 0.219976, -0.264632], [-0.636706, 0.318353, 0.692816], [-0.613483, -0.735083, -0.264632], [-0.326545, 0.979634, 0], [-0.318353, -0.636706, 0.692816], [-0.159176, 0.477529, -0.856368], [0.159176, -0.477529, -0.856368], [0.318353, 0.636706, 0.692816], [0.326545, -0.979634, 0], [0.613482, 0.735082, -0.264632], [0.636706, -0.318353, 0.692816], [0.931835, -0.219977, -0.264632] ], face: [ [11, 10, 8], [7, 9, 3], [6, 11, 8], [9, 5, 3], [2, 6, 8], [5, 0, 3], [4, 2, 8], [0, 1, 3], [10, 4, 8], [1, 7, 3], [10, 11, 9, 7], [11, 6, 5, 9], [6, 2, 0, 5], [2, 4, 1, 0], [4, 10, 7, 1] ] }, e[14] = { vertex: [ [-0.93465, 0.300459, -0.271185], [-0.838689, -0.260219, -0.516017], [-0.711319, 0.717591, 0.128359], [-0.710334, -0.156922, 0.080946], [-0.599799, 0.556003, -0.725148], [-0.503838, -4675e-6, -0.969981], [-0.487004, 0.26021, 0.48049], [-0.460089, -0.750282, -0.512622], [-0.376468, 0.973135, -0.325605], [-0.331735, -0.646985, 0.084342], [-0.254001, 0.831847, 0.530001], [-0.125239, -0.494738, -0.966586], [0.029622, 0.027949, 0.730817], [0.056536, -0.982543, -0.262295], [0.08085, 1.087391, 0.076037], [0.125583, -0.532729, 0.485984], [0.262625, 0.599586, 0.780328], [0.391387, -0.726999, -0.716259], [0.513854, -0.868287, 0.139347], [0.597475, 0.85513, 0.326364], [0.641224, 0.109523, 0.783723], [0.737185, -0.451155, 0.538891], [0.848705, -0.612742, -0.314616], [0.976075, 0.365067, 0.32976], [1.072036, -0.19561, 0.084927] ], face: [ [15, 18, 21], [12, 20, 16], [6, 10, 2], [3, 0, 1], [9, 7, 13], [2, 8, 4, 0], [0, 4, 5, 1], [1, 5, 11, 7], [7, 11, 17, 13], [13, 17, 22, 18], [18, 22, 24, 21], [21, 24, 23, 20], [20, 23, 19, 16], [16, 19, 14, 10], [10, 14, 8, 2], [15, 9, 13, 18], [12, 15, 21, 20], [6, 12, 16, 10], [3, 6, 2, 0], [9, 3, 1, 7], [9, 15, 12, 6, 3], [22, 17, 11, 5, 4, 8, 14, 19, 23, 24] ] }; const t = A.type && (A.type < 0 || A.type >= e.length) ? 0 : A.type || 0, r = A.size, n = A.sizeX || r || 1, i = A.sizeY || r || 1, s = A.sizeZ || r || 1, a = A.custom || e[t], f = a.face.length, o = A.faceUV || new Array(f), d = A.faceColors, v = A.flat === void 0 ? !0 : A.flat, u = A.sideOrientation === 0 ? 0 : A.sideOrientation || Ut.DEFAULTSIDE, l = [], P = [], p = [], c = [], H = []; let T = 0, q = 0; const b = []; let j = 0, w = 0, m, I, N, k, R, y; if (v) for (w = 0; w < f; w++) d && d[w] === void 0 && (d[w] = new xt(1, 1, 1, 1)), o && o[w] === void 0 && (o[w] = new Ir(0, 0, 1, 1)); if (v) for (w = 0; w < f; w++) { const Y = a.face[w].length; for (N = 2 * Math.PI / Y, k = 0.5 * Math.tan(N / 2), R = 0.5, j = 0; j < Y; j++) l.push(a.vertex[a.face[w][j]][0] * n, a.vertex[a.face[w][j]][1] * i, a.vertex[a.face[w][j]][2] * s), b.push(T), T++, m = o[w].x + (o[w].z - o[w].x) * (0.5 + k), I = o[w].y + (o[w].w - o[w].y) * (R - 0.5), c.push(m, us.UseOpenGLOrientationForUV ? 1 - I : I), y = k * Math.cos(N) - R * Math.sin(N), R = k * Math.sin(N) + R * Math.cos(N), k = y, d && H.push(d[w].r, d[w].g, d[w].b, d[w].a); for (j = 0; j < Y - 2; j++) P.push(b[0 + q], b[j + 2 + q], b[j + 1 + q]); q += Y; } else { for (j = 0; j < a.vertex.length; j++) l.push(a.vertex[j][0] * n, a.vertex[j][1] * i, a.vertex[j][2] * s), c.push(0, us.UseOpenGLOrientationForUV ? 1 : 0); for (w = 0; w < f; w++) for (j = 0; j < a.face[w].length - 2; j++) P.push(a.face[w][0], a.face[w][j + 2], a.face[w][j + 1]); } Ut.ComputeNormals(l, P, p), Ut._ComputeSides(u, l, P, p, c, A.frontUVs, A.backUVs); const O = new Ut(); return O.positions = l, O.indices = P, O.normals = p, O.uvs = c, d && v && (O.colors = H), O; } function hm(A, e = {}, t = null) { const r = new Ee(A, t); return e.sideOrientation = Ee._GetDefaultSideOrientation(e.sideOrientation), r._originalBuilderSideOrientation = e.sideOrientation, VQ(e).applyToMesh(r, e.updatable), r; } const zve = { // eslint-disable-next-line @typescript-eslint/naming-convention CreatePolyhedron: hm }; Ut.CreatePolyhedron = VQ; Ee.CreatePolyhedron = (A, e, t) => hm(A, e, t); const Gve = new S(1, 0, 0), Zve = new S(-1, 0, 0), _ve = new S(0, 1, 0), $ve = new S(0, -1, 0), eue = new S(0, 0, 1), tue = new S(0, 0, -1); class ZC { constructor(e = S.Zero(), t = S.Up(), r = at.Zero(), n = 0, i = 0, s = null, a = null, f = null, o = null) { this.position = e, this.normal = t, this.uv = r, this.vertexIdx = n, this.vertexIdxForBones = i, this.localPositionOverride = s, this.localNormalOverride = a, this.matrixIndicesOverride = f, this.matrixWeightsOverride = o; } clone() { var e, t, r, n; return new ZC(this.position.clone(), this.normal.clone(), this.uv.clone(), this.vertexIdx, this.vertexIdxForBones, (e = this.localPositionOverride) === null || e === void 0 ? void 0 : e.slice(), (t = this.localNormalOverride) === null || t === void 0 ? void 0 : t.slice(), (r = this.matrixIndicesOverride) === null || r === void 0 ? void 0 : r.slice(), (n = this.matrixWeightsOverride) === null || n === void 0 ? void 0 : n.slice()); } } function ey(A, e, t) { var r, n, i, s; const a = !!e.skeleton, f = t.localMode || a, o = e.overrideMaterialSideOrientation !== null && e.overrideMaterialSideOrientation !== void 0, d = e.getIndices(), v = a ? e.getPositionData(!0, !0) : e.getVerticesData(J.PositionKind), u = a ? e.getNormalsData(!0, !0) : e.getVerticesData(J.NormalKind), l = f ? a ? e.getVerticesData(J.PositionKind) : v : null, P = f ? a ? e.getVerticesData(J.NormalKind) : u : null, p = e.getVerticesData(J.UVKind), c = a ? e.getVerticesData(J.MatricesIndicesKind) : null, H = a ? e.getVerticesData(J.MatricesWeightsKind) : null, T = a ? e.getVerticesData(J.MatricesIndicesExtraKind) : null, q = a ? e.getVerticesData(J.MatricesWeightsExtraKind) : null, b = t.position || S.Zero(); let j = t.normal || S.Up(); const w = t.size || S.One(), m = t.angle || 0; if (!j) { const L = new S(0, 0, 1), $ = e.getScene().activeCamera, ae = S.TransformCoordinates(L, $.getWorldMatrix()); j = $.globalPosition.subtract(ae); } const I = -Math.atan2(j.z, j.x) - Math.PI / 2, N = Math.sqrt(j.x * j.x + j.z * j.z), k = Math.atan2(j.y, N), R = new Ut(); R.indices = [], R.positions = [], R.normals = [], R.uvs = [], R.matricesIndices = a ? [] : null, R.matricesWeights = a ? [] : null, R.matricesIndicesExtra = T ? [] : null, R.matricesWeightsExtra = q ? [] : null; let y = 0; const O = (L, $) => { const ae = new ZC(); if (!d || !v || !u) return ae; const Pe = d[L]; if (ae.vertexIdx = Pe * 3, ae.vertexIdxForBones = Pe * 4, ae.position = new S(v[Pe * 3], v[Pe * 3 + 1], v[Pe * 3 + 2]), S.TransformCoordinatesToRef(ae.position, $, ae.position), ae.normal = new S(u[Pe * 3], u[Pe * 3 + 1], u[Pe * 3 + 2]), S.TransformNormalToRef(ae.normal, $, ae.normal), t.captureUVS && p) { const ge = p[Pe * 2 + 1]; ae.uv = new at(p[Pe * 2], us.UseOpenGLOrientationForUV ? 1 - ge : ge); } return ae; }, Y = [0, 0, 0, 0], ee = (L, $) => { if (L.length === 0) return L; const ae = 0.5 * Math.abs(S.Dot(w, $)), Pe = (Xe, De, ne, re) => { for (let ve = 0; ve < re; ++ve) if (Xe[ne + ve] === De) return ne + ve; return -1; }, ge = (Xe, De) => { var ne, re, ve, qe, ke, be, Fe, Ke, nt, ut, bt, wt, Tt, lr, Qt, tr; const br = S.GetClipFactor(Xe.position, De.position, $, ae); let Xn = Y, qr = Y; if (c && H) { const Vn = Xe.matrixIndicesOverride ? 0 : Xe.vertexIdxForBones, ki = (ne = Xe.matrixIndicesOverride) !== null && ne !== void 0 ? ne : c, as = (re = Xe.matrixWeightsOverride) !== null && re !== void 0 ? re : H, Rr = De.matrixIndicesOverride ? 0 : De.vertexIdxForBones, Ii = (ve = De.matrixIndicesOverride) !== null && ve !== void 0 ? ve : c, is = (qe = De.matrixWeightsOverride) !== null && qe !== void 0 ? qe : H; Xn = [0, 0, 0, 0], qr = [0, 0, 0, 0]; let ji = 0; for (let oi = 0; oi < 4; ++oi) if (as[Vn + oi] > 0) { const vr = Pe(Ii, ki[Vn + oi], Rr, 4); Xn[ji] = ki[Vn + oi], qr[ji] = Xt.Lerp(as[Vn + oi], vr >= 0 ? is[vr] : 0, br), ji++; } for (let oi = 0; oi < 4 && ji < 4; ++oi) { const vr = Ii[Rr + oi]; Pe(ki, vr, Vn, 4) === -1 && (Xn[ji] = vr, qr[ji] = Xt.Lerp(0, is[Rr + oi], br), ji++); } const _i = qr[0] + qr[1] + qr[2] + qr[3]; qr[0] /= _i, qr[1] /= _i, qr[2] /= _i, qr[3] /= _i; } const En = Xe.localPositionOverride ? Xe.localPositionOverride[0] : (ke = l == null ? void 0 : l[Xe.vertexIdx]) !== null && ke !== void 0 ? ke : 0, Bn = Xe.localPositionOverride ? Xe.localPositionOverride[1] : (be = l == null ? void 0 : l[Xe.vertexIdx + 1]) !== null && be !== void 0 ? be : 0, Fi = Xe.localPositionOverride ? Xe.localPositionOverride[2] : (Fe = l == null ? void 0 : l[Xe.vertexIdx + 2]) !== null && Fe !== void 0 ? Fe : 0, Gt = De.localPositionOverride ? De.localPositionOverride[0] : (Ke = l == null ? void 0 : l[De.vertexIdx]) !== null && Ke !== void 0 ? Ke : 0, xr = De.localPositionOverride ? De.localPositionOverride[1] : (nt = l == null ? void 0 : l[De.vertexIdx + 1]) !== null && nt !== void 0 ? nt : 0, Br = De.localPositionOverride ? De.localPositionOverride[2] : (ut = l == null ? void 0 : l[De.vertexIdx + 2]) !== null && ut !== void 0 ? ut : 0, nn = Xe.localNormalOverride ? Xe.localNormalOverride[0] : (bt = P == null ? void 0 : P[Xe.vertexIdx]) !== null && bt !== void 0 ? bt : 0, sn = Xe.localNormalOverride ? Xe.localNormalOverride[1] : (wt = P == null ? void 0 : P[Xe.vertexIdx + 1]) !== null && wt !== void 0 ? wt : 0, Pn = Xe.localNormalOverride ? Xe.localNormalOverride[2] : (Tt = P == null ? void 0 : P[Xe.vertexIdx + 2]) !== null && Tt !== void 0 ? Tt : 0, an = De.localNormalOverride ? De.localNormalOverride[0] : (lr = P == null ? void 0 : P[De.vertexIdx]) !== null && lr !== void 0 ? lr : 0, Wn = De.localNormalOverride ? De.localNormalOverride[1] : (Qt = P == null ? void 0 : P[De.vertexIdx + 1]) !== null && Qt !== void 0 ? Qt : 0, rr = De.localNormalOverride ? De.localNormalOverride[2] : (tr = P == null ? void 0 : P[De.vertexIdx + 2]) !== null && tr !== void 0 ? tr : 0, Sr = nn + (an - nn) * br, nr = sn + (Wn - sn) * br, Er = Pn + (rr - Pn) * br, Rn = Math.sqrt(Sr * Sr + nr * nr + Er * Er); return new ZC(S.Lerp(Xe.position, De.position, br), S.Lerp(Xe.normal, De.normal, br).normalize(), at.Lerp(Xe.uv, De.uv, br), -1, -1, l ? [ En + (Gt - En) * br, Bn + (xr - Bn) * br, Fi + (Br - Fi) * br ] : null, P ? [Sr / Rn, nr / Rn, Er / Rn] : null, Xn, qr); }; let me = null; L.length > 3 && (me = []); for (let Xe = 0; Xe < L.length; Xe += 3) { let De = 0, ne = null, re = null, ve = null, qe = null; const ke = S.Dot(L[Xe].position, $) - ae, be = S.Dot(L[Xe + 1].position, $) - ae, Fe = S.Dot(L[Xe + 2].position, $) - ae, Ke = ke > 0, nt = be > 0, ut = Fe > 0; switch (De = (Ke ? 1 : 0) + (nt ? 1 : 0) + (ut ? 1 : 0), De) { case 0: L.length > 3 ? (me.push(L[Xe]), me.push(L[Xe + 1]), me.push(L[Xe + 2])) : me = L; break; case 1: if (me = me ?? new Array(), Ke && (ne = L[Xe + 1], re = L[Xe + 2], ve = ge(L[Xe], ne), qe = ge(L[Xe], re)), nt) { ne = L[Xe], re = L[Xe + 2], ve = ge(L[Xe + 1], ne), qe = ge(L[Xe + 1], re), me.push(ve), me.push(re.clone()), me.push(ne.clone()), me.push(re.clone()), me.push(ve.clone()), me.push(qe); break; } ut && (ne = L[Xe], re = L[Xe + 1], ve = ge(L[Xe + 2], ne), qe = ge(L[Xe + 2], re)), ne && re && ve && qe && (me.push(ne.clone()), me.push(re.clone()), me.push(ve), me.push(qe), me.push(ve.clone()), me.push(re.clone())); break; case 2: me = me ?? new Array(), Ke || (ne = L[Xe].clone(), re = ge(ne, L[Xe + 1]), ve = ge(ne, L[Xe + 2]), me.push(ne), me.push(re), me.push(ve)), nt || (ne = L[Xe + 1].clone(), re = ge(ne, L[Xe + 2]), ve = ge(ne, L[Xe]), me.push(ne), me.push(re), me.push(ve)), ut || (ne = L[Xe + 2].clone(), re = ge(ne, L[Xe]), ve = ge(ne, L[Xe + 1]), me.push(ne), me.push(re), me.push(ve)); break; } } return me; }, Z = e instanceof Ee ? e : null, te = Z == null ? void 0 : Z._thinInstanceDataStorage.matrixData, fe = (Z == null ? void 0 : Z.thinInstanceCount) || 1, _ = ue.Matrix[0]; _.copyFrom(he.IdentityReadOnly); for (let L = 0; L < fe; ++L) { if (Z != null && Z.hasThinInstances && te) { const Xe = L * 16; _.setRowFromFloats(0, te[Xe + 0], te[Xe + 1], te[Xe + 2], te[Xe + 3]), _.setRowFromFloats(1, te[Xe + 4], te[Xe + 5], te[Xe + 6], te[Xe + 7]), _.setRowFromFloats(2, te[Xe + 8], te[Xe + 9], te[Xe + 10], te[Xe + 11]), _.setRowFromFloats(3, te[Xe + 12], te[Xe + 13], te[Xe + 14], te[Xe + 15]); } const $ = he.RotationYawPitchRoll(I, k, m).multiply(he.Translation(b.x, b.y, b.z)), ae = he.Invert($), Pe = e.getWorldMatrix(), ge = _.multiply(Pe).multiply(ae), me = new Array(3); for (let Xe = 0; Xe < d.length; Xe += 3) { let De = me; if (De[0] = O(Xe, ge), o && f ? (De[1] = O(Xe + 2, ge), De[2] = O(Xe + 1, ge)) : (De[1] = O(Xe + 1, ge), De[2] = O(Xe + 2, ge)), !(t.cullBackFaces && -De[0].normal.z <= 0 && -De[1].normal.z <= 0 && -De[2].normal.z <= 0) && (De = ee(De, Gve), !!De && (De = ee(De, Zve), !!De && (De = ee(De, _ve), !!De && (De = ee(De, $ve), !!De && (De = ee(De, eue), !!De && (De = ee(De, tue), !!De))))))) for (let ne = 0; ne < De.length; ne++) { const re = De[ne]; if (R.indices.push(y), f ? (re.localPositionOverride ? (R.positions[y * 3] = re.localPositionOverride[0], R.positions[y * 3 + 1] = re.localPositionOverride[1], R.positions[y * 3 + 2] = re.localPositionOverride[2]) : l && (R.positions[y * 3] = l[re.vertexIdx], R.positions[y * 3 + 1] = l[re.vertexIdx + 1], R.positions[y * 3 + 2] = l[re.vertexIdx + 2]), re.localNormalOverride ? (R.normals[y * 3] = re.localNormalOverride[0], R.normals[y * 3 + 1] = re.localNormalOverride[1], R.normals[y * 3 + 2] = re.localNormalOverride[2]) : P && (R.normals[y * 3] = P[re.vertexIdx], R.normals[y * 3 + 1] = P[re.vertexIdx + 1], R.normals[y * 3 + 2] = P[re.vertexIdx + 2])) : (re.position.toArray(R.positions, y * 3), re.normal.toArray(R.normals, y * 3)), R.matricesIndices && R.matricesWeights && (re.matrixIndicesOverride ? (R.matricesIndices[y * 4] = re.matrixIndicesOverride[0], R.matricesIndices[y * 4 + 1] = re.matrixIndicesOverride[1], R.matricesIndices[y * 4 + 2] = re.matrixIndicesOverride[2], R.matricesIndices[y * 4 + 3] = re.matrixIndicesOverride[3]) : (c && (R.matricesIndices[y * 4] = c[re.vertexIdxForBones], R.matricesIndices[y * 4 + 1] = c[re.vertexIdxForBones + 1], R.matricesIndices[y * 4 + 2] = c[re.vertexIdxForBones + 2], R.matricesIndices[y * 4 + 3] = c[re.vertexIdxForBones + 3]), T && R.matricesIndicesExtra && (R.matricesIndicesExtra[y * 4] = T[re.vertexIdxForBones], R.matricesIndicesExtra[y * 4 + 1] = T[re.vertexIdxForBones + 1], R.matricesIndicesExtra[y * 4 + 2] = T[re.vertexIdxForBones + 2], R.matricesIndicesExtra[y * 4 + 3] = T[re.vertexIdxForBones + 3])), re.matrixWeightsOverride ? (R.matricesWeights[y * 4] = re.matrixWeightsOverride[0], R.matricesWeights[y * 4 + 1] = re.matrixWeightsOverride[1], R.matricesWeights[y * 4 + 2] = re.matrixWeightsOverride[2], R.matricesWeights[y * 4 + 3] = re.matrixWeightsOverride[3]) : (H && (R.matricesWeights[y * 4] = H[re.vertexIdxForBones], R.matricesWeights[y * 4 + 1] = H[re.vertexIdxForBones + 1], R.matricesWeights[y * 4 + 2] = H[re.vertexIdxForBones + 2], R.matricesWeights[y * 4 + 3] = H[re.vertexIdxForBones + 3]), q && R.matricesWeightsExtra && (R.matricesWeightsExtra[y * 4] = q[re.vertexIdxForBones], R.matricesWeightsExtra[y * 4 + 1] = q[re.vertexIdxForBones + 1], R.matricesWeightsExtra[y * 4 + 2] = q[re.vertexIdxForBones + 2], R.matricesWeightsExtra[y * 4 + 3] = q[re.vertexIdxForBones + 3]))), t.captureUVS) re.uv.toArray(R.uvs, y * 2); else { R.uvs.push(0.5 + re.position.x / w.x); const ve = 0.5 + re.position.y / w.y; R.uvs.push(us.UseOpenGLOrientationForUV ? 1 - ve : ve); } y++; } } } R.indices.length === 0 && (R.indices = null), R.positions.length === 0 && (R.positions = null), R.normals.length === 0 && (R.normals = null), R.uvs.length === 0 && (R.uvs = null), ((r = R.matricesIndices) === null || r === void 0 ? void 0 : r.length) === 0 && (R.matricesIndices = null), ((n = R.matricesWeights) === null || n === void 0 ? void 0 : n.length) === 0 && (R.matricesWeights = null), ((i = R.matricesIndicesExtra) === null || i === void 0 ? void 0 : i.length) === 0 && (R.matricesIndicesExtra = null), ((s = R.matricesWeightsExtra) === null || s === void 0 ? void 0 : s.length) === 0 && (R.matricesWeightsExtra = null); const G = new Ee(A, e.getScene()); return R.applyToMesh(G), f ? (G.skeleton = e.skeleton, G.parent = e) : (G.position = b.clone(), G.rotation = new S(k, I, m)), G.computeWorldMatrix(!0), G.refreshBoundingInfo(!0, !0), G; } const rue = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateDecal: ey }; Ee.CreateDecal = (A, e, t, r, n, i) => ey(A, e, { position: t, normal: r, size: n, angle: i }); class f1 { /** * Creates a new isovector from the given x and y coordinates * @param x defines the first coordinate, must be an integer * @param y defines the second coordinate, must be an integer */ constructor(e = 0, t = 0) { this.x = e, this.y = t, e !== Math.floor(e) && Se.Warn("x is not an integer, floor(x) used"), t !== Math.floor(t) && Se.Warn("y is not an integer, floor(y) used"); } // Operators /** * Gets a new IsoVector copied from the IsoVector * @returns a new IsoVector */ clone() { return new f1(this.x, this.y); } /** * Rotates one IsoVector 60 degrees counter clockwise about another * Please note that this is an in place operation * @param other an IsoVector a center of rotation * @returns the rotated IsoVector */ rotate60About(e) { const t = this.x; return this.x = e.x + e.y - this.y, this.y = t + this.y - e.x, this; } /** * Rotates one IsoVector 60 degrees clockwise about another * Please note that this is an in place operation * @param other an IsoVector as center of rotation * @returns the rotated IsoVector */ rotateNeg60About(e) { const t = this.x; return this.x = t + this.y - e.y, this.y = e.x + e.y - t, this; } /** * For an equilateral triangle OAB with O at isovector (0, 0) and A at isovector (m, n) * Rotates one IsoVector 120 degrees counter clockwise about the center of the triangle * Please note that this is an in place operation * @param m integer a measure a Primary triangle of order (m, n) m > n * @param n >= 0 integer a measure for a Primary triangle of order (m, n) * @returns the rotated IsoVector */ rotate120(e, t) { e !== Math.floor(e) && Se.Warn("m not an integer only floor(m) used"), t !== Math.floor(t) && Se.Warn("n not an integer only floor(n) used"); const r = this.x; return this.x = e - r - this.y, this.y = t + r, this; } /** * For an equilateral triangle OAB with O at isovector (0, 0) and A at isovector (m, n) * Rotates one IsoVector 120 degrees clockwise about the center of the triangle * Please note that this is an in place operation * @param m integer a measure a Primary triangle of order (m, n) m > n * @param n >= 0 integer a measure for a Primary triangle of order (m, n) * @returns the rotated IsoVector */ rotateNeg120(e, t) { e !== Math.floor(e) && Se.Warn("m is not an integer, floor(m) used"), t !== Math.floor(t) && Se.Warn("n is not an integer, floor(n) used"); const r = this.x; return this.x = this.y - t, this.y = e + t - r - this.y, this; } /** * Transforms an IsoVector to one in Cartesian 3D space based on an isovector * @param origin an IsoVector * @param isoGridSize * @returns Point as a Vector3 */ toCartesianOrigin(e, t) { const r = S.Zero(); return r.x = e.x + 2 * this.x * t + this.y * t, r.y = e.y + Math.sqrt(3) * this.y * t, r; } // Statics /** * Gets a new IsoVector(0, 0) * @returns a new IsoVector */ static Zero() { return new f1(0, 0); } } class CQ { constructor() { this.cartesian = [], this.vertices = [], this.max = [], this.min = [], this.closestTo = [], this.innerFacets = [], this.isoVecsABOB = [], this.isoVecsOBOA = [], this.isoVecsBAOA = [], this.vertexTypes = [], this.IDATA = new _C("icosahedron", "Regular", [ [0, kf, -1], [-kf, 1, 0], [-1, 0, -kf], [1, 0, -kf], [kf, 1, 0], [0, kf, 1], [-1, 0, kf], [-kf, -1, 0], [0, -kf, -1], [kf, -1, 0], [1, 0, kf], [0, -kf, 1] ], [ [0, 2, 1], [0, 3, 2], [0, 4, 3], [0, 5, 4], [0, 1, 5], [7, 6, 1], [8, 7, 2], [9, 8, 3], [10, 9, 4], [6, 10, 5], [2, 7, 1], [3, 8, 2], [4, 9, 3], [5, 10, 4], [1, 6, 5], [11, 6, 7], [11, 7, 8], [11, 8, 9], [11, 9, 10], [11, 10, 6] ]); } /** * Creates the PrimaryIsoTriangle Triangle OAB * @param m an integer * @param n an integer */ //operators setIndices() { let e = 12; const t = {}, r = this.m, n = this.n; let i = r, s = 1, a = 0; n !== 0 && (i = Xt.HCF(r, n)), s = r / i, a = n / i; let f, o, d, v, u; const l = f1.Zero(), P = new f1(r, n), p = new f1(-n, r + n), c = f1.Zero(), H = f1.Zero(), T = f1.Zero(); let q = [], b, j, w, m; const I = [], N = this.vertByDist, k = (R, y, O, Y) => { b = R + "|" + O, j = y + "|" + Y, b in t || j in t ? b in t && !(j in t) ? t[j] = t[b] : j in t && !(b in t) && (t[b] = t[j]) : (t[b] = e, t[j] = e, e++), N[O][0] > 2 ? I[t[b]] = [-N[O][0], N[O][1], t[b]] : I[t[b]] = [q[N[O][0]], N[O][1], t[b]]; }; this.IDATA.edgematch = [ [1, "B"], [2, "B"], [3, "B"], [4, "B"], [0, "B"], [10, "O", 14, "A"], [11, "O", 10, "A"], [12, "O", 11, "A"], [13, "O", 12, "A"], [14, "O", 13, "A"], [0, "O"], [1, "O"], [2, "O"], [3, "O"], [4, "O"], [19, "B", 5, "A"], [15, "B", 6, "A"], [16, "B", 7, "A"], [17, "B", 8, "A"], [18, "B", 9, "A"] ]; for (let R = 0; R < 20; R++) { if (q = this.IDATA.face[R], d = q[2], v = q[1], u = q[0], w = l.x + "|" + l.y, b = R + "|" + w, b in t || (t[b] = d, I[d] = [q[N[w][0]], N[w][1]]), w = P.x + "|" + P.y, b = R + "|" + w, b in t || (t[b] = v, I[v] = [q[N[w][0]], N[w][1]]), w = p.x + "|" + p.y, b = R + "|" + w, b in t || (t[b] = u, I[u] = [q[N[w][0]], N[w][1]]), f = this.IDATA.edgematch[R][0], o = this.IDATA.edgematch[R][1], o === "B") for (let y = 1; y < i; y++) H.x = r - y * (s + a), H.y = n + y * s, T.x = -y * a, T.y = y * (s + a), w = H.x + "|" + H.y, m = T.x + "|" + T.y, k(R, f, w, m); if (o === "O") for (let y = 1; y < i; y++) T.x = -y * a, T.y = y * (s + a), c.x = y * s, c.y = y * a, w = T.x + "|" + T.y, m = c.x + "|" + c.y, k(R, f, w, m); if (f = this.IDATA.edgematch[R][2], o = this.IDATA.edgematch[R][3], o && o === "A") for (let y = 1; y < i; y++) c.x = y * s, c.y = y * a, H.x = r - (i - y) * (s + a), H.y = n + (i - y) * s, w = c.x + "|" + c.y, m = H.x + "|" + H.y, k(R, f, w, m); for (let y = 0; y < this.vertices.length; y++) w = this.vertices[y].x + "|" + this.vertices[y].y, b = R + "|" + w, b in t || (t[b] = e++, N[w][0] > 2 ? I[t[b]] = [-N[w][0], N[w][1], t[b]] : I[t[b]] = [q[N[w][0]], N[w][1], t[b]]); } this.closestTo = I, this.vecToidx = t; } calcCoeffs() { const e = this.m, t = this.n, r = Math.sqrt(3) / 3, n = e * e + t * t + e * t; this.coau = (e + t) / n, this.cobu = -t / n, this.coav = -r * (e - t) / n, this.cobv = r * (2 * e + t) / n; } createInnerFacets() { const e = this.m, t = this.n; for (let r = 0; r < t + e + 1; r++) for (let n = this.min[r]; n < this.max[r] + 1; n++) n < this.max[r] && n < this.max[r + 1] + 1 && this.innerFacets.push(["|" + n + "|" + r, "|" + n + "|" + (r + 1), "|" + (n + 1) + "|" + r]), r > 0 && n < this.max[r - 1] && n + 1 < this.max[r] + 1 && this.innerFacets.push(["|" + n + "|" + r, "|" + (n + 1) + "|" + r, "|" + (n + 1) + "|" + (r - 1)]); } edgeVecsABOB() { const e = this.m, t = this.n, r = new f1(-t, e + t); for (let n = 1; n < e + t; n++) { const i = new f1(this.min[n], n), s = new f1(this.min[n - 1], n - 1), a = new f1(this.min[n + 1], n + 1), f = i.clone(), o = s.clone(), d = a.clone(); f.rotate60About(r), o.rotate60About(r), d.rotate60About(r); const v = new f1(this.max[f.y], f.y), u = new f1(this.max[f.y - 1], f.y - 1), l = new f1(this.max[f.y - 1] - 1, f.y - 1); (f.x !== v.x || f.y !== v.y) && (f.x !== u.x ? (this.vertexTypes.push([1, 0, 0]), this.isoVecsABOB.push([i, u, l]), this.vertexTypes.push([1, 0, 0]), this.isoVecsABOB.push([i, l, v])) : f.y === d.y ? (this.vertexTypes.push([1, 1, 0]), this.isoVecsABOB.push([i, s, u]), this.vertexTypes.push([1, 0, 1]), this.isoVecsABOB.push([i, u, a])) : (this.vertexTypes.push([1, 1, 0]), this.isoVecsABOB.push([i, s, u]), this.vertexTypes.push([1, 0, 0]), this.isoVecsABOB.push([i, u, v]))); } } mapABOBtoOBOA() { const e = new f1(0, 0); for (let t = 0; t < this.isoVecsABOB.length; t++) { const r = []; for (let n = 0; n < 3; n++) e.x = this.isoVecsABOB[t][n].x, e.y = this.isoVecsABOB[t][n].y, this.vertexTypes[t][n] === 0 && e.rotateNeg120(this.m, this.n), r.push(e.clone()); this.isoVecsOBOA.push(r); } } mapABOBtoBAOA() { const e = new f1(0, 0); for (let t = 0; t < this.isoVecsABOB.length; t++) { const r = []; for (let n = 0; n < 3; n++) e.x = this.isoVecsABOB[t][n].x, e.y = this.isoVecsABOB[t][n].y, this.vertexTypes[t][n] === 1 && e.rotate120(this.m, this.n), r.push(e.clone()); this.isoVecsBAOA.push(r); } } // eslint-disable-next-line @typescript-eslint/naming-convention MapToFace(e, t) { const r = this.IDATA.face[e], n = r[2], i = r[1], s = r[0], a = S.FromArray(this.IDATA.vertex[n]), f = S.FromArray(this.IDATA.vertex[i]), o = S.FromArray(this.IDATA.vertex[s]), d = f.subtract(a), v = o.subtract(a), u = d.scale(this.coau).add(v.scale(this.cobu)), l = d.scale(this.coav).add(v.scale(this.cobv)); let P, p = ue.Vector3[0]; for (let c = 0; c < this.cartesian.length; c++) p = u.scale(this.cartesian[c].x).add(l.scale(this.cartesian[c].y)).add(a), p.x, p.y, p.z, P = e + "|" + this.vertices[c].x + "|" + this.vertices[c].y, t.vertex[this.vecToidx[P]] = [p.x, p.y, p.z]; } //statics /**Creates a primary triangle * @internal */ build(e, t) { const r = [], n = f1.Zero(), i = new f1(e, t), s = new f1(-t, e + t); r.push(n, i, s); for (let j = t; j < e + 1; j++) for (let w = 0; w < e + 1 - j; w++) r.push(new f1(w, j)); if (t > 0) { const j = Xt.HCF(e, t), w = e / j, m = t / j; for (let N = 1; N < j; N++) r.push(new f1(N * w, N * m)), r.push(new f1(-N * m, N * (w + m))), r.push(new f1(e - N * (w + m), t + N * w)); const I = e / t; for (let N = 1; N < t; N++) for (let k = 0; k < N * I; k++) r.push(new f1(k, N)), r.push(new f1(k, N).rotate120(e, t)), r.push(new f1(k, N).rotateNeg120(e, t)); } r.sort((j, w) => j.x - w.x), r.sort((j, w) => j.y - w.y); const a = new Array(e + t + 1), f = new Array(e + t + 1); for (let j = 0; j < a.length; j++) a[j] = 1 / 0, f[j] = -1 / 0; let o = 0, d = 0; const v = r.length; for (let j = 0; j < v; j++) d = r[j].x, o = r[j].y, a[o] = Math.min(d, a[o]), f[o] = Math.max(d, f[o]); const u = (j, w) => { const m = j.clone(); return w === "A" && m.rotateNeg120(e, t), w === "B" && m.rotate120(e, t), m.x < 0 ? m.y : m.x + m.y; }, l = [], P = [], p = [], c = [], H = {}, T = []; let q = -1, b = -1; for (let j = 0; j < v; j++) l[j] = r[j].toCartesianOrigin(new f1(0, 0), 0.5), P[j] = u(r[j], "O"), p[j] = u(r[j], "A"), c[j] = u(r[j], "B"), P[j] === p[j] && p[j] === c[j] ? (q = 3, b = P[j]) : P[j] === p[j] ? (q = 4, b = P[j]) : p[j] === c[j] ? (q = 5, b = p[j]) : c[j] === P[j] && (q = 6, b = P[j]), P[j] < p[j] && P[j] < c[j] && (q = 2, b = P[j]), p[j] < P[j] && p[j] < c[j] && (q = 1, b = p[j]), c[j] < p[j] && c[j] < P[j] && (q = 0, b = c[j]), T.push([q, b, r[j].x, r[j].y]); T.sort((j, w) => j[2] - w[2]), T.sort((j, w) => j[3] - w[3]), T.sort((j, w) => j[1] - w[1]), T.sort((j, w) => j[0] - w[0]); for (let j = 0; j < T.length; j++) H[T[j][2] + "|" + T[j][3]] = [T[j][0], T[j][1], j]; return this.m = e, this.n = t, this.vertices = r, this.vertByDist = H, this.cartesian = l, this.min = a, this.max = f, this; } } class _C { constructor(e, t, r, n) { this.name = e, this.category = t, this.vertex = r, this.face = n; } } class _R extends _C { /** * @internal */ innerToData(e, t) { for (let r = 0; r < t.innerFacets.length; r++) this.face.push(t.innerFacets[r].map((n) => t.vecToidx[e + n])); } /** * @internal */ mapABOBtoDATA(e, t) { const r = t.IDATA.edgematch[e][0]; for (let n = 0; n < t.isoVecsABOB.length; n++) { const i = []; for (let s = 0; s < 3; s++) t.vertexTypes[n][s] === 0 ? i.push(e + "|" + t.isoVecsABOB[n][s].x + "|" + t.isoVecsABOB[n][s].y) : i.push(r + "|" + t.isoVecsABOB[n][s].x + "|" + t.isoVecsABOB[n][s].y); this.face.push([t.vecToidx[i[0]], t.vecToidx[i[1]], t.vecToidx[i[2]]]); } } /** * @internal */ mapOBOAtoDATA(e, t) { const r = t.IDATA.edgematch[e][0]; for (let n = 0; n < t.isoVecsOBOA.length; n++) { const i = []; for (let s = 0; s < 3; s++) t.vertexTypes[n][s] === 1 ? i.push(e + "|" + t.isoVecsOBOA[n][s].x + "|" + t.isoVecsOBOA[n][s].y) : i.push(r + "|" + t.isoVecsOBOA[n][s].x + "|" + t.isoVecsOBOA[n][s].y); this.face.push([t.vecToidx[i[0]], t.vecToidx[i[1]], t.vecToidx[i[2]]]); } } /** * @internal */ mapBAOAtoDATA(e, t) { const r = t.IDATA.edgematch[e][2]; for (let n = 0; n < t.isoVecsBAOA.length; n++) { const i = []; for (let s = 0; s < 3; s++) t.vertexTypes[n][s] === 1 ? i.push(e + "|" + t.isoVecsBAOA[n][s].x + "|" + t.isoVecsBAOA[n][s].y) : i.push(r + "|" + t.isoVecsBAOA[n][s].x + "|" + t.isoVecsBAOA[n][s].y); this.face.push([t.vecToidx[i[0]], t.vecToidx[i[1]], t.vecToidx[i[2]]]); } } /** * @internal */ orderData(e) { const t = []; for (let s = 0; s < 13; s++) t[s] = []; const r = e.closestTo; for (let s = 0; s < r.length; s++) r[s][0] > -1 ? r[s][1] > 0 && t[r[s][0]].push([s, r[s][1]]) : t[12].push([s, r[s][0]]); const n = []; for (let s = 0; s < 12; s++) n[s] = s; let i = 12; for (let s = 0; s < 12; s++) { t[s].sort((a, f) => a[1] - f[1]); for (let a = 0; a < t[s].length; a++) n[t[s][a][0]] = i++; } for (let s = 0; s < t[12].length; s++) n[t[12][s][0]] = i++; for (let s = 0; s < this.vertex.length; s++) this.vertex[s].push(n[s]); this.vertex.sort((s, a) => s[3] - a[3]); for (let s = 0; s < this.vertex.length; s++) this.vertex[s].pop(); for (let s = 0; s < this.face.length; s++) for (let a = 0; a < this.face[s].length; a++) this.face[s][a] = n[this.face[s][a]]; this.sharedNodes = t[12].length, this.poleNodes = this.vertex.length - this.sharedNodes; } /** * @internal */ setOrder(e, t) { const r = [], n = []; let i = t.pop(); n.push(i); let s = this.face[i].indexOf(e); s = (s + 2) % 3; let a = this.face[i][s]; r.push(a); let f = 0; for (; t.length > 0; ) i = t[f], this.face[i].indexOf(a) > -1 ? (s = (this.face[i].indexOf(a) + 1) % 3, a = this.face[i][s], r.push(a), n.push(i), t.splice(f, 1), f = 0) : f++; return this.adjacentFaces.push(r), n; } /** * @internal */ toGoldbergPolyhedronData() { const e = new _C("GeoDual", "Goldberg", [], []); e.name = "GD dual"; const t = this.vertex.length, r = new Array(t); for (let o = 0; o < t; o++) r[o] = []; for (let o = 0; o < this.face.length; o++) for (let d = 0; d < 3; d++) r[this.face[o][d]].push(o); let n = 0, i = 0, s = 0, a = [], f = []; this.adjacentFaces = []; for (let o = 0; o < r.length; o++) e.face[o] = this.setOrder(o, r[o].concat([])), r[o].forEach((d) => { n = 0, i = 0, s = 0, a = this.face[d]; for (let v = 0; v < 3; v++) f = this.vertex[a[v]], n += f[0], i += f[1], s += f[2]; e.vertex[d] = [n / 3, i / 3, s / 3]; }); return e; } //statics /**Builds the data for a Geodesic Polyhedron from a primary triangle * @param primTri the primary triangle * @internal */ static BuildGeodesicData(e) { const t = new _R("Geodesic-m-n", "Geodesic", [ [0, kf, -1], [-kf, 1, 0], [-1, 0, -kf], [1, 0, -kf], [kf, 1, 0], [0, kf, 1], [-1, 0, kf], [-kf, -1, 0], [0, -kf, -1], [kf, -1, 0], [1, 0, kf], [0, -kf, 1] ], []); e.setIndices(), e.calcCoeffs(), e.createInnerFacets(), e.edgeVecsABOB(), e.mapABOBtoOBOA(), e.mapABOBtoBAOA(); for (let n = 0; n < e.IDATA.face.length; n++) e.MapToFace(n, t), t.innerToData(n, e), e.IDATA.edgematch[n][1] === "B" && t.mapABOBtoDATA(n, e), e.IDATA.edgematch[n][1] === "O" && t.mapOBOAtoDATA(n, e), e.IDATA.edgematch[n][3] === "A" && t.mapBAOAtoDATA(n, e); t.orderData(e); const r = 1; return t.vertex = t.vertex.map(function(n) { const i = n[0], s = n[1], a = n[2], f = Math.sqrt(i * i + s * s + a * a); return n[0] *= r / f, n[1] *= r / f, n[2] *= r / f, n; }), t; } } function rte(A, e, t = null) { let r = e.m || 1; r !== Math.floor(r) && Se.Warn("m not an integer only floor(m) used"); let n = e.n || 0; if (n !== Math.floor(n) && Se.Warn("n not an integer only floor(n) used"), n > r) { const o = n; n = r, r = o, Se.Warn("n > m therefore m and n swapped"); } const i = new CQ(); i.build(r, n); const a = { custom: _R.BuildGeodesicData(i), size: e.size, sizeX: e.sizeX, sizeY: e.sizeY, sizeZ: e.sizeZ, faceUV: e.faceUV, faceColors: e.faceColors, flat: e.flat, updatable: e.updatable, sideOrientation: e.sideOrientation, frontUVs: e.frontUVs, backUVs: e.backUVs }; return hm(A, a, t); } Ee._GoldbergMeshParser = (A, e) => $R.Parse(A, e); class $R extends Ee { constructor() { super(...arguments), this.goldbergData = { faceColors: [], faceCenters: [], faceZaxis: [], faceXaxis: [], faceYaxis: [], nbSharedFaces: 0, nbUnsharedFaces: 0, nbFaces: 0, nbFacesAtPole: 0, adjacentFaces: [] }; } /** * Gets the related Goldberg face from pole infos * @param poleOrShared Defines the pole index or the shared face index if the fromPole parameter is passed in * @param fromPole Defines an optional pole index to find the related info from * @returns the goldberg face number */ relatedGoldbergFace(e, t) { return t === void 0 ? (e > this.goldbergData.nbUnsharedFaces - 1 && (Se.Warn("Maximum number of unshared faces used"), e = this.goldbergData.nbUnsharedFaces - 1), this.goldbergData.nbUnsharedFaces + e) : (e > 11 && (Se.Warn("Last pole used"), e = 11), t > this.goldbergData.nbFacesAtPole - 1 && (Se.Warn("Maximum number of faces at a pole used"), t = this.goldbergData.nbFacesAtPole - 1), 12 + e * this.goldbergData.nbFacesAtPole + t); } _changeGoldbergFaceColors(e) { for (let r = 0; r < e.length; r++) { const n = e[r][0], i = e[r][1], s = e[r][2]; for (let a = n; a < i + 1; a++) this.goldbergData.faceColors[a] = s; } const t = []; for (let r = 0; r < 12; r++) for (let n = 0; n < 5; n++) t.push(this.goldbergData.faceColors[r].r, this.goldbergData.faceColors[r].g, this.goldbergData.faceColors[r].b, this.goldbergData.faceColors[r].a); for (let r = 12; r < this.goldbergData.faceColors.length; r++) for (let n = 0; n < 6; n++) t.push(this.goldbergData.faceColors[r].r, this.goldbergData.faceColors[r].g, this.goldbergData.faceColors[r].b, this.goldbergData.faceColors[r].a); return t; } /** * Set new goldberg face colors * @param colorRange the new color to apply to the mesh */ setGoldbergFaceColors(e) { const t = this._changeGoldbergFaceColors(e); this.setVerticesData(J.ColorKind, t); } /** * Updates new goldberg face colors * @param colorRange the new color to apply to the mesh */ updateGoldbergFaceColors(e) { const t = this._changeGoldbergFaceColors(e); this.updateVerticesData(J.ColorKind, t); } _changeGoldbergFaceUVs(e) { const t = this.getVerticesData(J.UVKind); for (let r = 0; r < e.length; r++) { const n = e[r][0], i = e[r][1], s = e[r][2], a = e[r][3], f = e[r][4], o = [], d = []; let v, u; for (let l = 0; l < 5; l++) v = s.x + a * Math.cos(f + l * Math.PI / 2.5), u = s.y + a * Math.sin(f + l * Math.PI / 2.5), v < 0 && (v = 0), v > 1 && (v = 1), o.push(v, u); for (let l = 0; l < 6; l++) v = s.x + a * Math.cos(f + l * Math.PI / 3), u = s.y + a * Math.sin(f + l * Math.PI / 3), v < 0 && (v = 0), v > 1 && (v = 1), d.push(v, u); for (let l = n; l < Math.min(12, i + 1); l++) for (let P = 0; P < 5; P++) t[10 * l + 2 * P] = o[2 * P], t[10 * l + 2 * P + 1] = o[2 * P + 1]; for (let l = Math.max(12, n); l < i + 1; l++) for (let P = 0; P < 6; P++) t[12 * l - 24 + 2 * P] = d[2 * P], t[12 * l - 23 + 2 * P] = d[2 * P + 1]; } return t; } /** * set new goldberg face UVs * @param uvRange the new UVs to apply to the mesh */ setGoldbergFaceUVs(e) { const t = this._changeGoldbergFaceUVs(e); this.setVerticesData(J.UVKind, t); } /** * Updates new goldberg face UVs * @param uvRange the new UVs to apply to the mesh */ updateGoldbergFaceUVs(e) { const t = this._changeGoldbergFaceUVs(e); this.updateVerticesData(J.UVKind, t); } /** * Places a mesh on a particular face of the goldberg polygon * @param mesh Defines the mesh to position * @param face Defines the face to position onto * @param position Defines the position relative to the face we are positioning the mesh onto */ placeOnGoldbergFaceAt(e, t, r) { const n = S.RotationFromAxis(this.goldbergData.faceXaxis[t], this.goldbergData.faceYaxis[t], this.goldbergData.faceZaxis[t]); e.rotation = n, e.position = this.goldbergData.faceCenters[t].add(this.goldbergData.faceXaxis[t].scale(r.x)).add(this.goldbergData.faceYaxis[t].scale(r.y)).add(this.goldbergData.faceZaxis[t].scale(r.z)); } /** * Serialize current mesh * @param serializationObject defines the object which will receive the serialization data */ serialize(e) { super.serialize(e), e.type = "GoldbergMesh"; const t = {}; if (t.adjacentFaces = this.goldbergData.adjacentFaces, t.nbSharedFaces = this.goldbergData.nbSharedFaces, t.nbUnsharedFaces = this.goldbergData.nbUnsharedFaces, t.nbFaces = this.goldbergData.nbFaces, t.nbFacesAtPole = this.goldbergData.nbFacesAtPole, this.goldbergData.faceColors) { t.faceColors = []; for (const r of this.goldbergData.faceColors) t.faceColors.push(r.asArray()); } if (this.goldbergData.faceCenters) { t.faceCenters = []; for (const r of this.goldbergData.faceCenters) t.faceCenters.push(r.asArray()); } if (this.goldbergData.faceZaxis) { t.faceZaxis = []; for (const r of this.goldbergData.faceZaxis) t.faceZaxis.push(r.asArray()); } if (this.goldbergData.faceYaxis) { t.faceYaxis = []; for (const r of this.goldbergData.faceYaxis) t.faceYaxis.push(r.asArray()); } if (this.goldbergData.faceXaxis) { t.faceXaxis = []; for (const r of this.goldbergData.faceXaxis) t.faceXaxis.push(r.asArray()); } e.goldbergData = t; } /** * Parses a serialized goldberg mesh * @param parsedMesh the serialized mesh * @param scene the scene to create the goldberg mesh in * @returns the created goldberg mesh */ static Parse(e, t) { const r = e.goldbergData; r.faceColors = r.faceColors.map((i) => xt.FromArray(i)), r.faceCenters = r.faceCenters.map((i) => S.FromArray(i)), r.faceZaxis = r.faceZaxis.map((i) => S.FromArray(i)), r.faceXaxis = r.faceXaxis.map((i) => S.FromArray(i)), r.faceYaxis = r.faceYaxis.map((i) => S.FromArray(i)); const n = new $R(e.name, t); return n.goldbergData = r, n; } } function nte(A, e) { const t = A.size, r = A.sizeX || t || 1, n = A.sizeY || t || 1, i = A.sizeZ || t || 1, s = A.sideOrientation === 0 ? 0 : A.sideOrientation || Ut.DEFAULTSIDE, a = [], f = [], o = [], d = []; let v = 1 / 0, u = -1 / 0, l = 1 / 0, P = -1 / 0; for (let H = 0; H < e.vertex.length; H++) v = Math.min(v, e.vertex[H][0] * r), u = Math.max(u, e.vertex[H][0] * r), l = Math.min(l, e.vertex[H][1] * n), P = Math.max(P, e.vertex[H][1] * n); let p = 0; for (let H = 0; H < e.face.length; H++) { const T = e.face[H], q = S.FromArray(e.vertex[T[0]]), b = S.FromArray(e.vertex[T[2]]), j = S.FromArray(e.vertex[T[1]]), w = b.subtract(q), m = j.subtract(q), I = S.Cross(m, w).normalize(); for (let N = 0; N < T.length; N++) { o.push(I.x, I.y, I.z); const k = e.vertex[T[N]]; a.push(k[0] * r, k[1] * n, k[2] * i); const R = (k[1] * n - l) / (P - l); d.push((k[0] * r - v) / (u - v), us.UseOpenGLOrientationForUV ? 1 - R : R); } for (let N = 0; N < T.length - 2; N++) f.push(p, p + N + 2, p + N + 1); p += T.length; } Ut._ComputeSides(s, a, f, o, d); const c = new Ut(); return c.positions = a, c.indices = f, c.normals = o, c.uvs = d, c; } function ite(A, e, t = null) { const r = e.size, n = e.sizeX || r || 1, i = e.sizeY || r || 1, s = e.sizeZ || r || 1; let a = e.m || 1; a !== Math.floor(a) && Se.Warn("m not an integer only floor(m) used"); let f = e.n || 0; if (f !== Math.floor(f) && Se.Warn("n not an integer only floor(n) used"), f > a) { const P = f; f = a, a = P, Se.Warn("n > m therefore m and n swapped"); } const o = new CQ(); o.build(a, f); const d = _R.BuildGeodesicData(o), v = d.toGoldbergPolyhedronData(), u = new $R(A, t); e.sideOrientation = Ee._GetDefaultSideOrientation(e.sideOrientation), u._originalBuilderSideOrientation = e.sideOrientation, nte(e, v).applyToMesh(u, e.updatable), u.goldbergData.nbSharedFaces = d.sharedNodes, u.goldbergData.nbUnsharedFaces = d.poleNodes, u.goldbergData.adjacentFaces = d.adjacentFaces, u.goldbergData.nbFaces = u.goldbergData.nbSharedFaces + u.goldbergData.nbUnsharedFaces, u.goldbergData.nbFacesAtPole = (u.goldbergData.nbUnsharedFaces - 12) / 12; for (let P = 0; P < d.vertex.length; P++) u.goldbergData.faceCenters.push(S.FromArray(d.vertex[P])), u.goldbergData.faceCenters[P].x *= n, u.goldbergData.faceCenters[P].y *= i, u.goldbergData.faceCenters[P].z *= s, u.goldbergData.faceColors.push(new xt(1, 1, 1, 1)); for (let P = 0; P < v.face.length; P++) { const p = v.face[P], c = S.FromArray(v.vertex[p[0]]), H = S.FromArray(v.vertex[p[2]]), T = S.FromArray(v.vertex[p[1]]), q = H.subtract(c), b = T.subtract(c), j = S.Cross(b, q).normalize(), w = S.Cross(b, j).normalize(); u.goldbergData.faceXaxis.push(b.normalize()), u.goldbergData.faceYaxis.push(j), u.goldbergData.faceZaxis.push(w); } return u; } class nue { /** Create the ShapePath used to support glyphs */ constructor(e) { this._paths = [], this._tempPaths = [], this._holes = [], this._resolution = e; } /** Move the virtual cursor to a coordinate */ moveTo(e, t) { this._currentPath = new sU(e, t), this._tempPaths.push(this._currentPath); } /** Draw a line from the virtual cursor to a given coordinate */ lineTo(e, t) { this._currentPath.addLineTo(e, t); } /** Create a quadratic curve from the virtual cursor to a given coordinate */ quadraticCurveTo(e, t, r, n) { this._currentPath.addQuadraticCurveTo(e, t, r, n, this._resolution); } /** Create a bezier curve from the virtual cursor to a given coordinate */ bezierCurveTo(e, t, r, n, i, s) { this._currentPath.addBezierCurveTo(e, t, r, n, i, s, this._resolution); } /** Extract holes based on CW / CCW */ extractHoles() { for (const e of this._tempPaths) e.area() > 0 ? this._holes.push(e) : this._paths.push(e); if (!this._paths.length && this._holes.length) { const e = this._holes; this._holes = this._paths, this._paths = e; } this._tempPaths.length = 0; } /** Gets the list of paths */ get paths() { return this._paths; } /** Gets the list of holes */ get holes() { return this._holes; } } function iue(A, e, t, r, n, i) { const s = i.glyphs[A] || i.glyphs["?"]; if (!s) return null; const a = new nue(n); if (s.o) { const f = s.o.split(" "); for (let o = 0, d = f.length; o < d; ) switch (f[o++]) { case "m": { const u = parseInt(f[o++]) * e + t, l = parseInt(f[o++]) * e + r; a.moveTo(u, l); break; } case "l": { const u = parseInt(f[o++]) * e + t, l = parseInt(f[o++]) * e + r; a.lineTo(u, l); break; } case "q": { const u = parseInt(f[o++]) * e + t, l = parseInt(f[o++]) * e + r, P = parseInt(f[o++]) * e + t, p = parseInt(f[o++]) * e + r; a.quadraticCurveTo(P, p, u, l); break; } case "b": { const u = parseInt(f[o++]) * e + t, l = parseInt(f[o++]) * e + r, P = parseInt(f[o++]) * e + t, p = parseInt(f[o++]) * e + r, c = parseInt(f[o++]) * e + t, H = parseInt(f[o++]) * e + r; a.bezierCurveTo(P, p, c, H, u, l); break; } } } return a.extractHoles(), { offsetX: s.ha * e, shapePath: a }; } function OQ(A, e, t, r) { const n = Array.from(A), i = e / r.resolution, s = (r.boundingBox.yMax - r.boundingBox.yMin + r.underlineThickness) * i, a = []; let f = 0, o = 0; for (let d = 0; d < n.length; d++) { const v = n[d]; if (v === ` `) f = 0, o -= s; else { const u = iue(v, i, f, o, t, r); u && (f += u.offsetX, a.push(u.shapePath)); } } return a; } function ste(A, e, t, r = { size: 50, resolution: 8, depth: 1 }, n = null, i = earcut) { var s, a; const f = OQ(e, r.size || 50, r.resolution || 8, t), o = []; let d = 0; for (const u of f) { if (!u.paths.length) continue; const l = u.holes.slice(); for (const P of u.paths) { const p = [], c = [], H = P.getPoints(); for (const b of H) c.push(new S(b.x, 0, b.y)); const T = l.slice(); for (const b of T) { const j = b.getPoints(); let w = !1; for (const I of j) if (P.isPointInside(I)) { w = !0; break; } if (!w) continue; const m = []; for (const I of j) m.push(new S(I.x, 0, I.y)); p.push(m), l.splice(l.indexOf(b), 1); } if (!p.length && l.length) for (const b of l) { const j = b.getPoints(), w = []; for (const m of j) w.push(new S(m.x, 0, m.y)); p.push(w); } const q = zR(A, { shape: c, holes: p.length ? p : void 0, depth: r.depth || 1, faceUV: r.faceUV || ((s = r.perLetterFaceUV) === null || s === void 0 ? void 0 : s.call(r, d)), faceColors: r.faceColors || ((a = r.perLetterFaceColors) === null || a === void 0 ? void 0 : a.call(r, d)), sideOrientation: Ee._GetDefaultSideOrientation(r.sideOrientation || Ee.DOUBLESIDE) }, n, i); o.push(q), d++; } } const v = Ee.MergeMeshes(o, !0, !0); if (v) { const u = v.getBoundingInfo().boundingBox; v.position.x += -(u.minimumWorld.x + u.maximumWorld.x) / 2, v.position.y += -(u.minimumWorld.y + u.maximumWorld.y) / 2, v.position.z += -(u.minimumWorld.z + u.maximumWorld.z) / 2 + u.extendSize.z, v.name = A; const l = new Hr("pivot", n); l.rotation.x = -Math.PI / 2, v.parent = l, v.bakeCurrentTransformIntoVertices(), v.parent = null, l.dispose(); } return v; } const Ef = { CreateBox: k0, CreateTiledBox: BQ, CreateSphere: UA, CreateDisc: uU, CreateIcoSphere: UR, CreateRibbon: n4, CreateCylinder: Ld, CreateTorus: Ag, CreateTorusKnot: GO, CreateLineSystem: rm, CreateLines: ka, CreateDashedLines: ZO, ExtrudeShape: GR, ExtrudeShapeCustom: ZR, CreateLathe: _O, CreateTiledPlane: wQ, CreatePlane: u4, CreateGround: Wm, CreateTiledGround: FO, CreateGroundFromHeightMap: NO, CreatePolygon: JR, ExtrudePolygon: zR, CreateTube: $O, CreatePolyhedron: hm, CreateGeodesic: rte, CreateGoldberg: ite, CreateDecal: ey, CreateCapsule: KR, CreateText: ste }; class sue { /** * Creates a new PhysicsViewer * @param scene defines the hosting scene */ constructor(e) { if (this._impostors = [], this._meshes = [], this._bodies = [], this._inertiaBodies = [], this._constraints = [], this._bodyMeshes = [], this._inertiaMeshes = [], this._constraintMeshes = [], this._numMeshes = 0, this._numBodies = 0, this._numInertiaBodies = 0, this._numConstraints = 0, this._debugMeshMeshes = new Array(), this._constraintAxesSize = 0.4, this._scene = e || gr.LastCreatedScene, !this._scene) return; const t = this._scene.getPhysicsEngine(); t && (this._physicsEnginePlugin = t.getPhysicsPlugin()), this._utilityLayer = new Ds(this._scene, !1), this._utilityLayer.pickUtilitySceneFirst = !1, this._utilityLayer.utilityLayerScene.autoClearDepthAndStencil = !0; } /** * Updates the debug meshes of the physics engine. * * This code is useful for synchronizing the debug meshes of the physics engine with the physics impostor and mesh. * It checks if the impostor is disposed and if the plugin version is 1, then it syncs the mesh with the impostor. * This ensures that the debug meshes are up to date with the physics engine. */ _updateDebugMeshes() { const e = this._physicsEnginePlugin; (e == null ? void 0 : e.getPluginVersion()) === 1 ? this._updateDebugMeshesV1() : this._updateDebugMeshesV2(); } /** * Updates the debug meshes of the physics engine. * * This method is useful for synchronizing the debug meshes with the physics impostors. * It iterates through the impostors and meshes, and if the plugin version is 1, it syncs the mesh with the impostor. * This ensures that the debug meshes accurately reflect the physics impostors, which is important for debugging the physics engine. */ _updateDebugMeshesV1() { const e = this._physicsEnginePlugin; for (let t = 0; t < this._numMeshes; t++) { const r = this._impostors[t]; if (r) if (r.isDisposed) this.hideImpostor(this._impostors[t--]); else { if (r.type === tn.MeshImpostor) continue; const n = this._meshes[t]; n && e && e.syncMeshWithImpostor(n, r); } } } /** * Updates the debug meshes of the physics engine for V2 plugin. * * This method is useful for synchronizing the debug meshes of the physics engine with the current state of the bodies. * It iterates through the bodies array and updates the debug meshes with the current transform of each body. * This ensures that the debug meshes accurately reflect the current state of the physics engine. */ _updateDebugMeshesV2() { const e = this._physicsEnginePlugin; for (let t = 0; t < this._numBodies; t++) { const r = this._bodies[t], n = this._bodyMeshes[t]; r && n && e.syncTransform(r, n); } } _updateInertiaMeshes() { for (let e = 0; e < this._numInertiaBodies; e++) { const t = this._inertiaBodies[e], r = this._inertiaMeshes[e]; t && r && this._updateDebugInertia(t, r); } } _updateDebugInertia(e, t) { var r; const n = he.Identity(), i = he.Identity(), s = he.Identity(); if (e._pluginDataInstances.length) { const a = t, f = a._thinInstanceDataStorage.matrixData, o = e.transformNode._thinInstanceDataStorage.matrixData; for (let d = 0; d < e._pluginDataInstances.length; d++) { const v = e.getMassProperties(d); this._getMeshDebugInertiaMatrixToRef(v, n), he.FromArrayToRef(o, d * 16, i), n.multiplyToRef(i, s), s.copyToArray(f, d * 16); } a.thinInstanceBufferUpdated("matrix"); } else { const a = e.getMassProperties(); if (this._getMeshDebugInertiaMatrixToRef(a, n), (r = e.transformNode.rotationQuaternion) === null || r === void 0 || r.toRotationMatrix(i), i.setTranslation(e.transformNode.position), e.transformNode.parent) { const f = e.transformNode.parent.computeWorldMatrix(!0); i.multiplyToRef(f, i); } n.multiplyToRef(i, n), n.decomposeToTransformNode(t); } } _updateDebugConstraints() { for (let e = 0; e < this._numConstraints; e++) { const t = this._constraints[e], r = this._constraintMeshes[e]; t && r && this._updateDebugConstraint(t, r); } } /** * Given a scaling vector, make all of its components * 1, preserving the sign * @param scaling */ _makeScalingUnitInPlace(e) { Math.abs(e.x - 1) > Dn && (e.x = 1 * Math.sign(e.x)), Math.abs(e.y - 1) > Dn && (e.y = 1 * Math.sign(e.y)), Math.abs(e.z - 1) > Dn && (e.z = 1 * Math.sign(e.z)); } _updateDebugConstraint(e, t) { if (!e._initOptions) return; const { pivotA: r, pivotB: n, axisA: i, axisB: s, perpAxisA: a, perpAxisB: f } = e._initOptions; !r || !n || !i || !s || !a || !f || t.getDescendants(!0).forEach((o) => { const d = o.getDescendants(!0)[0], v = o.getDescendants(!0)[1], { parentBody: u, parentBodyIndex: l } = d.metadata, { childBody: P, childBodyIndex: p } = v.metadata, c = this._getTransformFromBodyToRef(u, ue.Matrix[0], l), H = this._getTransformFromBodyToRef(P, ue.Matrix[1], p); c.decomposeToTransformNode(d), this._makeScalingUnitInPlace(d.scaling), H.decomposeToTransformNode(v), this._makeScalingUnitInPlace(v.scaling); const T = d.getDescendants(!0)[0]; T.position.copyFrom(r); const q = v.getDescendants(!0)[0]; q.position.copyFrom(n), Ze.FromRotationMatrixToRef(he.FromXYZAxesToRef(i, a, S.CrossToRef(i, a, ue.Vector3[0]), ue.Matrix[0]), T.rotationQuaternion), Ze.FromRotationMatrixToRef(he.FromXYZAxesToRef(s, f, S.CrossToRef(s, f, ue.Vector3[1]), ue.Matrix[1]), q.rotationQuaternion); }); } /** * Renders a specified physic impostor * @param impostor defines the impostor to render * @param targetMesh defines the mesh represented by the impostor * @returns the new debug mesh used to render the impostor */ showImpostor(e, t) { if (!this._scene) return null; for (let n = 0; n < this._numMeshes; n++) if (this._impostors[n] == e) return null; const r = this._getDebugMesh(e, t); return r && (this._impostors[this._numMeshes] = e, this._meshes[this._numMeshes] = r, this._numMeshes === 0 && (this._renderFunction = () => this._updateDebugMeshes(), this._scene.registerBeforeRender(this._renderFunction)), this._numMeshes++), r; } /** * Shows a debug mesh for a given physics body. * @param body The physics body to show. * @returns The debug mesh, or null if the body is already shown. * * This function is useful for visualizing the physics body in the scene. * It creates a debug mesh for the given body and adds it to the scene. * It also registers a before render function to update the debug mesh position and rotation. */ showBody(e) { if (!this._scene) return null; for (let r = 0; r < this._numBodies; r++) if (this._bodies[r] == e) return null; const t = this._getDebugBodyMesh(e); return t && (this._bodies[this._numBodies] = e, this._bodyMeshes[this._numBodies] = t, this._numBodies === 0 && (this._renderFunction = () => this._updateDebugMeshes(), this._scene.registerBeforeRender(this._renderFunction)), this._numBodies++), t; } /** * Shows a debug box corresponding to the inertia of a given body * @param body */ showInertia(e) { if (!this._scene) return null; for (let r = 0; r < this._numInertiaBodies; r++) if (this._inertiaBodies[r] == e) return null; const t = this._getDebugInertiaMesh(e); return t && (this._inertiaBodies[this._numInertiaBodies] = e, this._inertiaMeshes[this._numInertiaBodies] = t, this._numInertiaBodies === 0 && (this._inertiaRenderFunction = () => this._updateInertiaMeshes(), this._scene.registerBeforeRender(this._inertiaRenderFunction)), this._numInertiaBodies++), t; } /** * Shows a debug mesh for a given physics constraint. * @param constraint the physics constraint to show * @returns the debug mesh, or null if the constraint is already shown */ showConstraint(e) { if (!this._scene) return null; for (let r = 0; r < this._numConstraints; r++) if (this._constraints[r] == e) return null; const t = this._getDebugConstraintMesh(e); return t && (this._constraints[this._numConstraints] = e, this._constraintMeshes[this._numConstraints] = t, this._numConstraints === 0 && (this._constraintRenderFunction = () => this._updateDebugConstraints(), this._scene.registerBeforeRender(this._constraintRenderFunction)), this._numConstraints++), t; } /** * Hides an impostor from the scene. * @param impostor - The impostor to hide. * * This method is useful for hiding an impostor from the scene. It removes the * impostor from the utility layer scene, disposes the mesh, and removes the * impostor from the list of impostors. If the impostor is the last one in the * list, it also unregisters the render function. */ hideImpostor(e) { if (!e || !this._scene || !this._utilityLayer) return; let t = !1; const r = this._utilityLayer.utilityLayerScene; for (let n = 0; n < this._numMeshes; n++) if (this._impostors[n] == e) { const i = this._meshes[n]; if (!i) continue; r.removeMesh(i), i.dispose(); const s = this._debugMeshMeshes.indexOf(i); s > -1 && this._debugMeshMeshes.splice(s, 1), this._numMeshes--, this._numMeshes > 0 ? (this._meshes[n] = this._meshes[this._numMeshes], this._impostors[n] = this._impostors[this._numMeshes], this._meshes[this._numMeshes] = null, this._impostors[this._numMeshes] = null) : (this._meshes[0] = null, this._impostors[0] = null), t = !0; break; } t && this._numMeshes === 0 && this._scene.unregisterBeforeRender(this._renderFunction); } /** * Hides a body from the physics engine. * @param body - The body to hide. * * This function is useful for hiding a body from the physics engine. * It removes the body from the utility layer scene and disposes the mesh associated with it. * It also unregisters the render function if the number of bodies is 0. * This is useful for hiding a body from the physics engine without deleting it. */ hideBody(e) { if (!e || !this._scene || !this._utilityLayer) return; let t = !1; const r = this._utilityLayer.utilityLayerScene; for (let n = 0; n < this._numBodies; n++) if (this._bodies[n] === e) { const i = this._bodyMeshes[n]; if (!i) continue; r.removeMesh(i), i.dispose(), this._numBodies--, this._numBodies > 0 ? (this._bodyMeshes[n] = this._bodyMeshes[this._numBodies], this._bodies[n] = this._bodies[this._numBodies], this._bodyMeshes[this._numBodies] = null, this._bodies[this._numBodies] = null) : (this._bodyMeshes[0] = null, this._bodies[0] = null), t = !0; break; } t && this._numBodies === 0 && this._scene.unregisterBeforeRender(this._renderFunction); } hideInertia(e) { if (!e || !this._scene || !this._utilityLayer) return; let t = !1; const r = this._utilityLayer.utilityLayerScene; for (let n = 0; n < this._numInertiaBodies; n++) if (this._inertiaBodies[n] === e) { const i = this._inertiaMeshes[n]; if (!i) continue; r.removeMesh(i), i.dispose(), this._inertiaBodies.splice(n, 1), this._inertiaMeshes.splice(n, 1), this._numInertiaBodies--, t = !0; break; } t && this._numInertiaBodies === 0 && this._scene.unregisterBeforeRender(this._inertiaRenderFunction); } /** * Hide a physics constraint from the viewer utility layer * @param constraint the constraint to hide */ hideConstraint(e) { if (!e || !this._scene || !this._utilityLayer) return; let t = !1; const r = this._utilityLayer.utilityLayerScene; for (let n = 0; n < this._numConstraints; n++) if (this._constraints[n] === e) { const i = this._constraintMeshes[n]; if (!i) continue; r.removeMesh(i), i.dispose(), this._constraints.splice(n, 1), this._constraintMeshes.splice(n, 1), this._numConstraints--, this._numConstraints > 0 ? (this._constraints[n] = this._constraints[this._numConstraints], this._constraintMeshes[n] = this._constraintMeshes[this._numConstraints], this._constraints[this._numConstraints] = null, this._constraintMeshes[this._numConstraints] = null) : (this._constraints[0] = null, this._constraintMeshes[0] = null), t = !0; break; } t && this._numConstraints === 0 && this._scene.unregisterBeforeRender(this._constraintRenderFunction); } _getDebugMaterial(e) { return this._debugMaterial || (this._debugMaterial = new Wt("", e), this._debugMaterial.wireframe = !0, this._debugMaterial.emissiveColor = Ne.White(), this._debugMaterial.disableLighting = !0), this._debugMaterial; } _getDebugInertiaMaterial(e) { return this._debugInertiaMaterial || (this._debugInertiaMaterial = new Wt("", e), this._debugInertiaMaterial.disableLighting = !0, this._debugInertiaMaterial.alpha = 0), this._debugInertiaMaterial; } _getDebugBoxMesh(e) { return this._debugBoxMesh || (this._debugBoxMesh = k0("physicsBodyBoxViewMesh", { size: 1 }, e), this._debugBoxMesh.rotationQuaternion = Ze.Identity(), this._debugBoxMesh.material = this._getDebugMaterial(e), this._debugBoxMesh.setEnabled(!1)), this._debugBoxMesh.createInstance("physicsBodyBoxViewInstance"); } _getDebugSphereMesh(e) { return this._debugSphereMesh || (this._debugSphereMesh = UA("physicsBodySphereViewMesh", { diameter: 1 }, e), this._debugSphereMesh.rotationQuaternion = Ze.Identity(), this._debugSphereMesh.material = this._getDebugMaterial(e), this._debugSphereMesh.setEnabled(!1)), this._debugSphereMesh.createInstance("physicsBodySphereViewInstance"); } _getDebugCapsuleMesh(e) { return this._debugCapsuleMesh || (this._debugCapsuleMesh = KR("physicsBodyCapsuleViewMesh", { height: 1 }, e), this._debugCapsuleMesh.rotationQuaternion = Ze.Identity(), this._debugCapsuleMesh.material = this._getDebugMaterial(e), this._debugCapsuleMesh.setEnabled(!1)), this._debugCapsuleMesh.createInstance("physicsBodyCapsuleViewInstance"); } _getDebugCylinderMesh(e) { return this._debugCylinderMesh || (this._debugCylinderMesh = Ld("physicsBodyCylinderViewMesh", { diameterTop: 1, diameterBottom: 1, height: 1 }, e), this._debugCylinderMesh.rotationQuaternion = Ze.Identity(), this._debugCylinderMesh.material = this._getDebugMaterial(e), this._debugCylinderMesh.setEnabled(!1)), this._debugCylinderMesh.createInstance("physicsBodyCylinderViewInstance"); } _getDebugMeshMesh(e, t) { const r = new Ee(e.name, t, null, e); return r.setParent(e), r.position = S.Zero(), r.material = this._getDebugMaterial(t), this._debugMeshMeshes.push(r), r; } _getDebugMesh(e, t) { if (!this._utilityLayer || t && t.parent && t.parent.physicsImpostor) return null; let r = null; const n = this._utilityLayer.utilityLayerScene; if (!e.physicsBody) return Se.Warn("Unable to get physicsBody of impostor. It might be initialized later by its parent's impostor."), null; switch (e.type) { case tn.BoxImpostor: r = this._getDebugBoxMesh(n), e.getBoxSizeToRef(r.scaling); break; case tn.SphereImpostor: { r = this._getDebugSphereMesh(n); const i = e.getRadius(); r.scaling.x = i * 2, r.scaling.y = i * 2, r.scaling.z = i * 2; break; } case tn.CapsuleImpostor: { r = this._getDebugCapsuleMesh(n); const i = e.object.getBoundingInfo(); r.scaling.x = (i.boundingBox.maximum.x - i.boundingBox.minimum.x) * 2 * e.object.scaling.x, r.scaling.y = (i.boundingBox.maximum.y - i.boundingBox.minimum.y) * e.object.scaling.y, r.scaling.z = (i.boundingBox.maximum.z - i.boundingBox.minimum.z) * 2 * e.object.scaling.z; break; } case tn.MeshImpostor: t && (r = this._getDebugMeshMesh(t, n)); break; case tn.NoImpostor: t ? t.getChildMeshes().filter((s) => s.physicsImpostor ? 1 : 0).forEach((s) => { if (s.physicsImpostor && s.getClassName() === "Mesh") { const a = s.getBoundingInfo(), f = a.boundingBox.minimum, o = a.boundingBox.maximum; switch (s.physicsImpostor.type) { case tn.BoxImpostor: r = this._getDebugBoxMesh(n), r.position.copyFrom(f), r.position.addInPlace(o), r.position.scaleInPlace(0.5); break; case tn.SphereImpostor: r = this._getDebugSphereMesh(n); break; case tn.CylinderImpostor: r = this._getDebugCylinderMesh(n); break; default: r = null; break; } r && (r.scaling.x = o.x - f.x, r.scaling.y = o.y - f.y, r.scaling.z = o.z - f.z, r.parent = s); } }) : Se.Warn("No target mesh parameter provided for NoImpostor. Skipping."), r = null; break; case tn.CylinderImpostor: { r = this._getDebugCylinderMesh(n); const i = e.object.getBoundingInfo(); r.scaling.x = (i.boundingBox.maximum.x - i.boundingBox.minimum.x) * e.object.scaling.x, r.scaling.y = (i.boundingBox.maximum.y - i.boundingBox.minimum.y) * e.object.scaling.y, r.scaling.z = (i.boundingBox.maximum.z - i.boundingBox.minimum.z) * e.object.scaling.z; break; } } return r; } /** * Creates a debug mesh for a given physics body * @param body The physics body to create the debug mesh for * @returns The created debug mesh or null if the utility layer is not available * * This code is useful for creating a debug mesh for a given physics body. * It creates a Mesh object with a VertexData object containing the positions and indices * of the geometry of the body. The mesh is then assigned a debug material from the utility layer scene. * This allows for visualizing the physics body in the scene. */ _getDebugBodyMesh(e) { if (!this._utilityLayer) return null; const t = this._utilityLayer.utilityLayerScene, r = new Ee("custom", t), n = new Ut(), i = e.getGeometry(); if (n.positions = i.positions, n.indices = i.indices, n.applyToMesh(r), e._pluginDataInstances) { const s = new Float32Array(e._pluginDataInstances.length * 16); r.thinInstanceSetBuffer("matrix", s, 16); } return r.material = this._getDebugMaterial(t), r; } _getMeshDebugInertiaMatrixToRef(e, t) { var r, n, i; const s = (r = e.inertiaOrientation) !== null && r !== void 0 ? r : Ze.Identity(), a = (n = e.inertia) !== null && n !== void 0 ? n : S.Zero(), f = (i = e.centerOfMass) !== null && i !== void 0 ? i : S.Zero(), o = (a.x - a.y + a.z) * 6, d = Math.sqrt(Math.max(o, 0)), v = a.x * 12 - o, u = Math.sqrt(Math.max(v, 0)), l = a.z * 12 - o, P = Math.sqrt(Math.max(l, 0)), p = ue.Vector3[0]; p.set(P, d, u); const c = he.ScalingToRef(p.x, p.y, p.z, ue.Matrix[0]), H = s.toRotationMatrix(ue.Matrix[1]), T = he.TranslationToRef(f.x, f.y, f.z, ue.Matrix[2]); return c.multiplyToRef(H, t), t.multiplyToRef(T, t), t; } _getDebugInertiaMesh(e) { if (!this._utilityLayer) return null; const t = this._utilityLayer.utilityLayerScene, r = Ef.CreateBox("custom", { size: 1 }, t), n = he.Identity(); if (e._pluginDataInstances.length) { const i = new Float32Array(e._pluginDataInstances.length * 16); for (let s = 0; s < e._pluginDataInstances.length; ++s) { const a = e.getMassProperties(s); this._getMeshDebugInertiaMatrixToRef(a, n), n.copyToArray(i, s * 16); } r.thinInstanceSetBuffer("matrix", i, 16); } else { const i = e.getMassProperties(); this._getMeshDebugInertiaMatrixToRef(i, n), n.decomposeToTransformNode(r); } return r.enableEdgesRendering(), r.edgesWidth = 2, r.edgesColor = new xt(1, 0, 1, 1), r.material = this._getDebugInertiaMaterial(t), r; } _getTransformFromBodyToRef(e, t, r) { const n = e.transformNode; return r && r >= 0 ? he.FromArrayToRef(n._thinInstanceDataStorage.matrixData, r, t) : t.copyFrom(n.getWorldMatrix()); } _getDebugConstraintMesh(e) { if (!this._utilityLayer) return null; const t = this._utilityLayer.utilityLayerScene; if (!e._initOptions) return null; const { pivotA: r, pivotB: n, axisA: i, axisB: s, perpAxisA: a, perpAxisB: f } = e._initOptions; if (!r || !n || !i || !s || !a || !f) return null; const o = new Ee("parentingDebugConstraint", t), d = e.getBodiesUsingConstraint(); for (const v of d) { const u = new Hr("parentOfPair", t); u.parent = o; const { parentBody: l, parentBodyIndex: P, childBody: p, childBodyIndex: c } = v, H = this._getTransformFromBodyToRef(l, ue.Matrix[0], P), T = this._getTransformFromBodyToRef(p, ue.Matrix[1], c), q = new Hr("parentCoordSystem", t); q.parent = u, q.metadata = { parentBody: l, parentBodyIndex: P }, H.decomposeToTransformNode(q); const b = new Hr("childCoordSystem", t); b.parent = u, b.metadata = { childBody: p, childBodyIndex: c }, T.decomposeToTransformNode(b); const j = Ze.FromRotationMatrix(he.FromXYZAxesToRef(i, a, i.cross(a), ue.Matrix[0])), w = Ze.FromRotationMatrix(he.FromXYZAxesToRef(s, f, s.cross(f), ue.Matrix[0])), m = r, I = n, N = new Hr("constraint_parent", t); N.position.copyFrom(m), N.rotationQuaternion = j, N.parent = q; const k = new Hr("constraint_child", t); k.parent = b, k.position.copyFrom(I), k.rotationQuaternion = w; const R = new aq(t, this._constraintAxesSize); R.xAxis.parent = N, R.yAxis.parent = N, R.zAxis.parent = N; const y = new aq(t, this._constraintAxesSize); y.xAxis.parent = k, y.yAxis.parent = k, y.zAxis.parent = k; } return o; } /** * Clean up physics debug display */ dispose() { for (let e = this._numMeshes - 1; e >= 0; e--) this.hideImpostor(this._impostors[0]); for (let e = this._numBodies - 1; e >= 0; e--) this.hideBody(this._bodies[0]); for (let e = this._numInertiaBodies - 1; e >= 0; e--) this.hideInertia(this._inertiaBodies[0]); this._debugBoxMesh && this._debugBoxMesh.dispose(), this._debugSphereMesh && this._debugSphereMesh.dispose(), this._debugCylinderMesh && this._debugCylinderMesh.dispose(), this._debugMaterial && this._debugMaterial.dispose(), this._impostors.length = 0, this._scene = null, this._physicsEnginePlugin = null, this._utilityLayer && (this._utilityLayer.dispose(), this._utilityLayer = null); } } class yQ { /** * Helper function to create a colored helper in a scene in one line. * @param ray Defines the ray we are currently trying to visualize * @param scene Defines the scene the ray is used in * @param color Defines the color we want to see the ray in * @returns The newly created ray helper. */ static CreateAndShow(e, t, r) { const n = new yQ(e); return n.show(t, r), n; } /** * Instantiate a new ray helper. * As raycast might be hard to debug, the RayHelper can help rendering the different rays * in order to better appreciate the issue one might have. * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/interactions/picking_collisions#debugging * @param ray Defines the ray we are currently trying to visualize */ constructor(e) { this.ray = e; } /** * Shows the ray we are willing to debug. * @param scene Defines the scene the ray needs to be rendered in * @param color Defines the color the ray needs to be rendered in */ show(e, t) { if (!this._renderFunction && this.ray) { const r = this.ray; this._renderFunction = () => this._render(), this._scene = e, this._renderPoints = [r.origin, r.origin.add(r.direction.scale(r.length))], this._renderLine = ka("ray", { points: this._renderPoints, updatable: !0 }, e), this._renderLine.isPickable = !1, this._renderFunction && this._scene.registerBeforeRender(this._renderFunction); } t && this._renderLine && this._renderLine.color.copyFrom(t); } /** * Hides the ray we are debugging. */ hide() { this._renderFunction && this._scene && (this._scene.unregisterBeforeRender(this._renderFunction), this._scene = null, this._renderFunction = null, this._renderLine && (this._renderLine.dispose(), this._renderLine = null), this._renderPoints = []); } _render() { var e; const t = this.ray; if (!t) return; const r = this._renderPoints[1], n = Math.min(t.length, 1e6); r.copyFrom(t.direction), r.scaleInPlace(n), r.addInPlace(t.origin), this._renderPoints[0].copyFrom(t.origin), ka("ray", { points: this._renderPoints, updatable: !0, instance: this._renderLine }, this._scene), (e = this._renderLine) === null || e === void 0 || e.refreshBoundingInfo(); } /** * Attach a ray helper to a mesh so that we can easily see its orientation for instance or information like its normals. * @param mesh Defines the mesh we want the helper attached to * @param meshSpaceDirection Defines the direction of the Ray in mesh space (local space of the mesh node) * @param meshSpaceOrigin Defines the origin of the Ray in mesh space (local space of the mesh node) * @param length Defines the length of the ray */ attachToMesh(e, t, r, n) { this._attachedToMesh = e; const i = this.ray; i && (i.direction || (i.direction = S.Zero()), i.origin || (i.origin = S.Zero()), n && (i.length = n), r || (r = S.Zero()), t || (t = new S(0, 0, -1)), this._scene || (this._scene = e.getScene()), this._meshSpaceDirection ? (this._meshSpaceDirection.copyFrom(t), this._meshSpaceOrigin.copyFrom(r)) : (this._meshSpaceDirection = t.clone(), this._meshSpaceOrigin = r.clone()), this._onAfterRenderObserver || (this._onAfterRenderObserver = this._scene.onBeforeRenderObservable.add(() => this._updateToMesh()), this._onAfterStepObserver = this._scene.onAfterStepObservable.add(() => this._updateToMesh())), this._attachedToMesh.computeWorldMatrix(!0), this._updateToMesh()); } /** * Detach the ray helper from the mesh it has previously been attached to. */ detachFromMesh() { this._attachedToMesh && this._scene && (this._onAfterRenderObserver && (this._scene.onBeforeRenderObservable.remove(this._onAfterRenderObserver), this._scene.onAfterStepObservable.remove(this._onAfterStepObserver)), this._attachedToMesh = null, this._onAfterRenderObserver = null, this._onAfterStepObserver = null, this._scene = null); } _updateToMesh() { const e = this.ray; if (!(!this._attachedToMesh || !e)) { if (this._attachedToMesh.isDisposed()) { this.detachFromMesh(); return; } this._attachedToMesh.getDirectionToRef(this._meshSpaceDirection, e.direction), S.TransformCoordinatesToRef(this._meshSpaceOrigin, this._attachedToMesh.getWorldMatrix(), e.origin); } } /** * Dispose the helper and release its associated resources. */ dispose() { this.hide(), this.detachFromMesh(), this.ray = null; } } class Ru { /** public static method to create a BoneWeight Shader * @param options The constructor options * @param scene The scene that the shader is scoped to * @returns The created ShaderMaterial * @see http://www.babylonjs-playground.com/#1BZJVJ#395 */ static CreateBoneWeightShader(e, t) { var r, n, i, s, a, f; const o = e.skeleton, d = (r = e.colorBase) !== null && r !== void 0 ? r : Ne.Black(), v = (n = e.colorZero) !== null && n !== void 0 ? n : Ne.Blue(), u = (i = e.colorQuarter) !== null && i !== void 0 ? i : Ne.Green(), l = (s = e.colorHalf) !== null && s !== void 0 ? s : Ne.Yellow(), P = (a = e.colorFull) !== null && a !== void 0 ? a : Ne.Red(), p = (f = e.targetBoneIndex) !== null && f !== void 0 ? f : 0; An.ShadersStore["boneWeights:" + o.name + "VertexShader"] = `precision highp float; attribute vec3 position; attribute vec2 uv; uniform mat4 view; uniform mat4 projection; uniform mat4 worldViewProjection; #include #if NUM_BONE_INFLUENCERS == 0 attribute vec4 matricesIndices; attribute vec4 matricesWeights; #endif #include #include varying vec3 vColor; uniform vec3 colorBase; uniform vec3 colorZero; uniform vec3 colorQuarter; uniform vec3 colorHalf; uniform vec3 colorFull; uniform float targetBoneIndex; void main() { vec3 positionUpdated = position; #include #include #include vec4 worldPos = finalWorld * vec4(positionUpdated, 1.0); vec3 color = colorBase; float totalWeight = 0.; if(matricesIndices[0] == targetBoneIndex && matricesWeights[0] > 0.){ totalWeight += matricesWeights[0]; } if(matricesIndices[1] == targetBoneIndex && matricesWeights[1] > 0.){ totalWeight += matricesWeights[1]; } if(matricesIndices[2] == targetBoneIndex && matricesWeights[2] > 0.){ totalWeight += matricesWeights[2]; } if(matricesIndices[3] == targetBoneIndex && matricesWeights[3] > 0.){ totalWeight += matricesWeights[3]; } color = mix(color, colorZero, smoothstep(0., 0.25, totalWeight)); color = mix(color, colorQuarter, smoothstep(0.25, 0.5, totalWeight)); color = mix(color, colorHalf, smoothstep(0.5, 0.75, totalWeight)); color = mix(color, colorFull, smoothstep(0.75, 1.0, totalWeight)); vColor = color; gl_Position = projection * view * worldPos; }`, An.ShadersStore["boneWeights:" + o.name + "FragmentShader"] = ` precision highp float; varying vec3 vPosition; varying vec3 vColor; void main() { vec4 color = vec4(vColor, 1.0); gl_FragColor = color; } `; const c = new Zo("boneWeight:" + o.name, t, { vertex: "boneWeights:" + o.name, fragment: "boneWeights:" + o.name }, { attributes: ["position", "normal", "matricesIndices", "matricesWeights"], uniforms: [ "world", "worldView", "worldViewProjection", "view", "projection", "viewProjection", "colorBase", "colorZero", "colorQuarter", "colorHalf", "colorFull", "targetBoneIndex" ] }); return c.setColor3("colorBase", d), c.setColor3("colorZero", v), c.setColor3("colorQuarter", u), c.setColor3("colorHalf", l), c.setColor3("colorFull", P), c.setFloat("targetBoneIndex", p), c.getClassName = () => "BoneWeightShader", c.transparencyMode = gt.MATERIAL_OPAQUE, c; } /** public static method to create a BoneWeight Shader * @param options The constructor options * @param scene The scene that the shader is scoped to * @returns The created ShaderMaterial */ static CreateSkeletonMapShader(e, t) { var r; const n = e.skeleton, i = (r = e.colorMap) !== null && r !== void 0 ? r : [ { color: new Ne(1, 0.38, 0.18), location: 0 }, { color: new Ne(0.59, 0.18, 1), location: 0.2 }, { color: new Ne(0.59, 1, 0.18), location: 0.4 }, { color: new Ne(1, 0.87, 0.17), location: 0.6 }, { color: new Ne(1, 0.17, 0.42), location: 0.8 }, { color: new Ne(0.17, 0.68, 1), location: 1 } ], s = n.bones.length + 1, a = Ru._CreateBoneMapColorBuffer(s, i, t), f = new Zo("boneWeights:" + n.name, t, { vertexSource: `precision highp float; attribute vec3 position; attribute vec2 uv; uniform mat4 view; uniform mat4 projection; uniform mat4 worldViewProjection; uniform float colorMap[` + n.bones.length * 4 + `]; #include #if NUM_BONE_INFLUENCERS == 0 attribute vec4 matricesIndices; attribute vec4 matricesWeights; #endif #include #include varying vec3 vColor; void main() { vec3 positionUpdated = position; #include #include #include vec3 color = vec3(0.); bool first = true; for (int i = 0; i < 4; i++) { int boneIdx = int(matricesIndices[i]); float boneWgt = matricesWeights[i]; vec3 c = vec3(colorMap[boneIdx * 4 + 0], colorMap[boneIdx * 4 + 1], colorMap[boneIdx * 4 + 2]); if (boneWgt > 0.) { if (first) { first = false; color = c; } else { color = mix(color, c, boneWgt); } } } vColor = color; vec4 worldPos = finalWorld * vec4(positionUpdated, 1.0); gl_Position = projection * view * worldPos; }`, fragmentSource: ` precision highp float; varying vec3 vColor; void main() { vec4 color = vec4( vColor, 1.0 ); gl_FragColor = color; } ` }, { attributes: ["position", "normal", "matricesIndices", "matricesWeights"], uniforms: ["world", "worldView", "worldViewProjection", "view", "projection", "viewProjection", "colorMap"] }); return f.setFloats("colorMap", a), f.getClassName = () => "SkeletonMapShader", f.transparencyMode = gt.MATERIAL_OPAQUE, f; } /** private static method to create a BoneWeight Shader * @param size The size of the buffer to create (usually the bone count) * @param colorMap The gradient data to generate * @param scene The scene that the shader is scoped to * @returns an Array of floats from the color gradient values */ static _CreateBoneMapColorBuffer(e, t, r) { const n = new Xp("temp", { width: e, height: 1 }, r, !1), i = n.getContext(), s = i.createLinearGradient(0, 0, e, 0); t.forEach((d) => { s.addColorStop(d.location, d.color.toHexString()); }), i.fillStyle = s, i.fillRect(0, 0, e, 1), n.update(); const a = [], f = i.getImageData(0, 0, e, 1).data, o = 1 / 255; for (let d = 0; d < f.length; d++) a.push(f[d] * o); return n.dispose(), a; } /** Gets the Scene. */ get scene() { return this._scene; } /** Gets the utilityLayer. */ get utilityLayer() { return this._utilityLayer; } /** Checks Ready Status. */ get isReady() { return this._ready; } /** Sets Ready Status. */ set ready(e) { this._ready = e; } /** Gets the debugMesh */ get debugMesh() { return this._debugMesh; } /** Sets the debugMesh */ set debugMesh(e) { this._debugMesh = e; } /** Gets the displayMode */ get displayMode() { return this.options.displayMode || Ru.DISPLAY_LINES; } /** Sets the displayMode */ set displayMode(e) { e > Ru.DISPLAY_SPHERE_AND_SPURS && (e = Ru.DISPLAY_LINES), this.options.displayMode = e; } /** * Creates a new SkeletonViewer * @param skeleton defines the skeleton to render * @param mesh defines the mesh attached to the skeleton * @param scene defines the hosting scene * @param autoUpdateBonesMatrices defines a boolean indicating if bones matrices must be forced to update before rendering (true by default) * @param renderingGroupId defines the rendering group id to use with the viewer * @param options All of the extra constructor options for the SkeletonViewer */ constructor(e, t, r, n = !0, i = 3, s = {}) { var a, f, o, d, v, u, l, P, p, c, H, T, q, b; if (this.skeleton = e, this.mesh = t, this.autoUpdateBonesMatrices = n, this.renderingGroupId = i, this.options = s, this.color = Ne.White(), this._debugLines = new Array(), this._localAxes = null, this._isEnabled = !0, this._obs = null, this._scene = r, this._ready = !1, s.pauseAnimations = (a = s.pauseAnimations) !== null && a !== void 0 ? a : !0, s.returnToRest = (f = s.returnToRest) !== null && f !== void 0 ? f : !1, s.displayMode = (o = s.displayMode) !== null && o !== void 0 ? o : Ru.DISPLAY_LINES, s.displayOptions = (d = s.displayOptions) !== null && d !== void 0 ? d : {}, s.displayOptions.midStep = (v = s.displayOptions.midStep) !== null && v !== void 0 ? v : 0.235, s.displayOptions.midStepFactor = (u = s.displayOptions.midStepFactor) !== null && u !== void 0 ? u : 0.155, s.displayOptions.sphereBaseSize = (l = s.displayOptions.sphereBaseSize) !== null && l !== void 0 ? l : 0.15, s.displayOptions.sphereScaleUnit = (P = s.displayOptions.sphereScaleUnit) !== null && P !== void 0 ? P : 2, s.displayOptions.sphereFactor = (p = s.displayOptions.sphereFactor) !== null && p !== void 0 ? p : 0.865, s.displayOptions.spurFollowsChild = (c = s.displayOptions.spurFollowsChild) !== null && c !== void 0 ? c : !1, s.displayOptions.showLocalAxes = (H = s.displayOptions.showLocalAxes) !== null && H !== void 0 ? H : !1, s.displayOptions.localAxesSize = (T = s.displayOptions.localAxesSize) !== null && T !== void 0 ? T : 0.075, s.computeBonesUsingShaders = (q = s.computeBonesUsingShaders) !== null && q !== void 0 ? q : !0, s.useAllBones = (b = s.useAllBones) !== null && b !== void 0 ? b : !0, this._boneIndices = /* @__PURE__ */ new Set(), !s.useAllBones) { const w = t == null ? void 0 : t.getVerticesData(J.MatricesIndicesKind), m = t == null ? void 0 : t.getVerticesData(J.MatricesWeightsKind); if (w && m) for (let I = 0; I < w.length; ++I) { const N = w[I]; m[I] !== 0 && this._boneIndices.add(N); } } this._utilityLayer = new Ds(this._scene, !1), this._utilityLayer.pickUtilitySceneFirst = !1, this._utilityLayer.utilityLayerScene.autoClearDepthAndStencil = !0; let j = this.options.displayMode || 0; j > Ru.DISPLAY_SPHERE_AND_SPURS && (j = Ru.DISPLAY_LINES), this.displayMode = j, this.update(), this._bindObs(); } /** The Dynamic bindings for the update functions */ _bindObs() { switch (this.displayMode) { case Ru.DISPLAY_LINES: { this._obs = this.scene.onBeforeRenderObservable.add(() => { this._displayLinesUpdate(); }); break; } } } /** Update the viewer to sync with current skeleton state, only used to manually update. */ update() { switch (this.displayMode) { case Ru.DISPLAY_LINES: { this._displayLinesUpdate(); break; } case Ru.DISPLAY_SPHERES: { this._buildSpheresAndSpurs(!0); break; } case Ru.DISPLAY_SPHERE_AND_SPURS: { this._buildSpheresAndSpurs(!1); break; } } this._buildLocalAxes(); } /** Gets or sets a boolean indicating if the viewer is enabled */ set isEnabled(e) { this.isEnabled !== e && (this._isEnabled = e, this.debugMesh && this.debugMesh.setEnabled(e), e && !this._obs ? this._bindObs() : !e && this._obs && (this.scene.onBeforeRenderObservable.remove(this._obs), this._obs = null)); } get isEnabled() { return this._isEnabled; } _getBonePosition(e, t, r, n = 0, i = 0, s = 0) { const a = ue.Matrix[0], f = t.getParent(); if (a.copyFrom(t.getLocalMatrix()), n !== 0 || i !== 0 || s !== 0) { const o = ue.Matrix[1]; he.IdentityToRef(o), o.setTranslationFromFloats(n, i, s), o.multiplyToRef(a, a); } f && a.multiplyToRef(f.getAbsoluteMatrix(), a), a.multiplyToRef(r, a), e.x = a.m[12], e.y = a.m[13], e.z = a.m[14]; } _getLinesForBonesWithLength(e, t) { const r = e.length; let n, i; t ? (n = t.getWorldMatrix(), i = t.position) : (n = new he(), i = e[0].position); let s = 0; for (let a = 0; a < r; a++) { const f = e[a]; let o = this._debugLines[s]; f._index === -1 || !this._boneIndices.has(f.getIndex()) && !this.options.useAllBones || (o || (o = [S.Zero(), S.Zero()], this._debugLines[s] = o), this._getBonePosition(o[0], f, n), this._getBonePosition(o[1], f, n, 0, f.length, 0), o[0].subtractInPlace(i), o[1].subtractInPlace(i), s++); } } _getLinesForBonesNoLength(e) { const t = e.length; let r = 0; const n = this.mesh; let i, s; n ? (i = n, s = n.position) : (i = new Hr(""), s = e[0].position); for (let a = t - 1; a >= 0; a--) { const f = e[a], o = f.getParent(); if (!o || !this._boneIndices.has(f.getIndex()) && !this.options.useAllBones) continue; let d = this._debugLines[r]; d || (d = [S.Zero(), S.Zero()], this._debugLines[r] = d), f.getAbsolutePositionToRef(i, d[0]), o.getAbsolutePositionToRef(i, d[1]), d[0].subtractInPlace(s), d[1].subtractInPlace(s), r++; } n || i.dispose(); } /** * function to revert the mesh and scene back to the initial state. * @param animationState */ _revert(e) { this.options.pauseAnimations && (this.scene.animationsEnabled = e, this.utilityLayer.utilityLayerScene.animationsEnabled = e); } /** * function to get the absolute bind pose of a bone by accumulating transformations up the bone hierarchy. * @param bone * @param matrix */ _getAbsoluteBindPoseToRef(e, t) { if (e === null || e._index === -1) { t.copyFrom(he.Identity()); return; } this._getAbsoluteBindPoseToRef(e.getParent(), t), e.getBindMatrix().multiplyToRef(t, t); } /** * function to build and bind sphere joint points and spur bone representations. * @param spheresOnly */ _buildSpheresAndSpurs(e = !0) { var t, r; this._debugMesh && (this._debugMesh.dispose(), this._debugMesh = null, this.ready = !1), this._ready = !1; const n = (t = this.utilityLayer) === null || t === void 0 ? void 0 : t.utilityLayerScene, i = this.skeleton.bones, s = [], a = [], f = this.scene.animationsEnabled; try { this.options.pauseAnimations && (this.scene.animationsEnabled = !1, n.animationsEnabled = !1), this.options.returnToRest && this.skeleton.returnToRest(), this.autoUpdateBonesMatrices && this.skeleton.computeAbsoluteMatrices(); let o = Number.NEGATIVE_INFINITY; const d = this.options.displayOptions || {}; for (let p = 0; p < i.length; p++) { const c = i[p]; if (c._index === -1 || !this._boneIndices.has(c.getIndex()) && !this.options.useAllBones) continue; const H = new he(); this._getAbsoluteBindPoseToRef(c, H); const T = new S(); H.decompose(void 0, void 0, T), c.children.forEach((I) => { const N = new he(); I.getLocalMatrix().multiplyToRef(H, N); const k = new S(); N.decompose(void 0, void 0, k); const R = S.Distance(T, k); if (R > o && (o = R), e) return; const y = k.clone().subtract(T.clone()), O = y.length(), Y = y.normalize().scale(O), ee = d.midStep || 0.165, Z = d.midStepFactor || 0.215, te = Y.scale(ee), fe = ZR("skeletonViewer", { shape: [new S(1, -1, 0), new S(1, 1, 0), new S(-1, 1, 0), new S(-1, -1, 0), new S(1, -1, 0)], path: [S.Zero(), te, Y], scaleFunction: ($) => { switch ($) { case 0: case 2: return 0; case 1: return O * Z; } return 0; }, sideOrientation: Ee.DEFAULTSIDE, updatable: !1 }, n), _ = fe.getTotalVertices(), G = [], L = []; for (let $ = 0; $ < _; $++) G.push(1, 0, 0, 0), d.spurFollowsChild && $ > 9 ? L.push(I.getIndex(), 0, 0, 0) : L.push(c.getIndex(), 0, 0, 0); fe.position = T.clone(), fe.setVerticesData(J.MatricesWeightsKind, G, !1), fe.setVerticesData(J.MatricesIndicesKind, L, !1), fe.convertToFlatShadedMesh(), a.push(fe); }); const q = d.sphereBaseSize || 0.2, b = UA("skeletonViewer", { segments: 6, diameter: q, updatable: !0 }, n), j = b.getTotalVertices(), w = [], m = []; for (let I = 0; I < j; I++) w.push(1, 0, 0, 0), m.push(c.getIndex(), 0, 0, 0); b.setVerticesData(J.MatricesWeightsKind, w, !1), b.setVerticesData(J.MatricesIndicesKind, m, !1), b.position = T.clone(), s.push([b, c]); } const v = d.sphereScaleUnit || 2, u = d.sphereFactor || 0.85, l = []; for (let p = 0; p < s.length; p++) { const [c, H] = s[p], T = 1 / (v / o); let q = 0, b = H; for (; b.getParent() && b.getParent().getIndex() !== -1; ) q++, b = b.getParent(); c.scaling.scaleInPlace(T * Math.pow(u, q)), l.push(c); } this.debugMesh = Ee.MergeMeshes(l.concat(a), !0, !0), this.debugMesh && (this.debugMesh.renderingGroupId = this.renderingGroupId, this.debugMesh.skeleton = this.skeleton, this.debugMesh.parent = this.mesh, this.debugMesh.computeBonesUsingShaders = (r = this.options.computeBonesUsingShaders) !== null && r !== void 0 ? r : !0, this.debugMesh.alwaysSelectAsActiveMesh = !0); const P = this.utilityLayer._getSharedGizmoLight(); P.intensity = 0.7, this._revert(f), this.ready = !0; } catch (o) { console.error(o), this._revert(f), this.dispose(); } } _buildLocalAxes() { var e; this._localAxes && this._localAxes.dispose(), this._localAxes = null; const t = this.options.displayOptions || {}; if (!t.showLocalAxes) return; const r = this._utilityLayer.utilityLayerScene, n = t.localAxesSize || 0.075, i = [], s = [], a = new xt(1, 0, 0, 1), f = new xt(0, 1, 0, 1), o = new xt(0, 0, 1, 1), d = [], v = [], u = 6; for (const l in this.skeleton.bones) { const P = this.skeleton.bones[l]; if (P._index === -1 || !this._boneIndices.has(P.getIndex()) && !this.options.useAllBones) continue; const p = new he(), c = new S(); this._getAbsoluteBindPoseToRef(P, p), p.decompose(void 0, ue.Quaternion[0], c); const H = new he(); ue.Quaternion[0].toRotationMatrix(H); const T = S.TransformCoordinates(new S(0 + n, 0, 0), H), q = S.TransformCoordinates(new S(0, 0 + n, 0), H), b = S.TransformCoordinates(new S(0, 0, 0 + n), H), j = [c, c.add(T)], w = [c, c.add(q)], m = [c, c.add(b)], I = [j, w, m], N = [ [a, a], [f, f], [o, o] ]; i.push(...I), s.push(...N); for (let k = 0; k < u; k++) d.push(1, 0, 0, 0), v.push(P.getIndex(), 0, 0, 0); } this._localAxes = rm("localAxes", { lines: i, colors: s, updatable: !0 }, r), this._localAxes.setVerticesData(J.MatricesWeightsKind, d, !1), this._localAxes.setVerticesData(J.MatricesIndicesKind, v, !1), this._localAxes.skeleton = this.skeleton, this._localAxes.renderingGroupId = this.renderingGroupId + 1, this._localAxes.parent = this.mesh, this._localAxes.computeBonesUsingShaders = (e = this.options.computeBonesUsingShaders) !== null && e !== void 0 ? e : !0; } /** Update the viewer to sync with current skeleton state, only used for the line display. */ _displayLinesUpdate() { if (!this._utilityLayer) return; this.autoUpdateBonesMatrices && this.skeleton.computeAbsoluteMatrices(), this.skeleton.bones[0].length === void 0 ? this._getLinesForBonesNoLength(this.skeleton.bones) : this._getLinesForBonesWithLength(this.skeleton.bones, this.mesh); const e = this._utilityLayer.utilityLayerScene; e && (this._debugMesh ? rm("", { lines: this._debugLines, updatable: !0, instance: this._debugMesh }, e) : (this._debugMesh = rm("", { lines: this._debugLines, updatable: !0, instance: null }, e), this._debugMesh.renderingGroupId = this.renderingGroupId), this.mesh ? this._debugMesh.position.copyFrom(this.mesh.position) : this._debugMesh.position.copyFrom(this.skeleton.bones[0].position), this._debugMesh.color = this.color); } /** Changes the displayMode of the skeleton viewer * @param mode The displayMode numerical value */ changeDisplayMode(e) { const t = !!this.isEnabled; this.displayMode !== e && (this.isEnabled = !1, this._debugMesh && (this._debugMesh.dispose(), this._debugMesh = null, this.ready = !1), this.displayMode = e, this.update(), this._bindObs(), this.isEnabled = t); } /** Sets a display option of the skeleton viewer * * | Option | Type | Default | Description | * | ---------------- | ------- | ------- | ----------- | * | midStep | float | 0.235 | A percentage between a bone and its child that determines the widest part of a spur. Only used when `displayMode` is set to `DISPLAY_SPHERE_AND_SPURS`. | * | midStepFactor | float | 0.15 | Mid step width expressed as a factor of the length. A value of 0.5 makes the spur width half of the spur length. Only used when `displayMode` is set to `DISPLAY_SPHERE_AND_SPURS`. | * | sphereBaseSize | float | 2 | Sphere base size. Only used when `displayMode` is set to `DISPLAY_SPHERE_AND_SPURS`. | * | sphereScaleUnit | float | 0.865 | Sphere scale factor used to scale spheres in relation to the longest bone. Only used when `displayMode` is set to `DISPLAY_SPHERE_AND_SPURS`. | * | spurFollowsChild | boolean | false | Whether a spur should attach its far end to the child bone. | * | showLocalAxes | boolean | false | Displays local axes on all bones. | * | localAxesSize | float | 0.075 | Determines the length of each local axis. | * * @param option String of the option name * @param value The numerical option value */ changeDisplayOptions(e, t) { const r = !!this.isEnabled; this.options.displayOptions[e] = t, this.isEnabled = !1, this._debugMesh && (this._debugMesh.dispose(), this._debugMesh = null, this.ready = !1), this.update(), this._bindObs(), this.isEnabled = r; } /** Release associated resources */ dispose() { this.isEnabled = !1, this._debugMesh && (this._debugMesh.dispose(), this._debugMesh = null), this._utilityLayer && (this._utilityLayer.dispose(), this._utilityLayer = null), this.ready = !1; } } Ru.DISPLAY_LINES = 0; Ru.DISPLAY_SPHERES = 1; Ru.DISPLAY_SPHERE_AND_SPURS = 2; class aue { /** * Gets or sets the transparency of the frustum planes */ get transparency() { return this._transparency; } set transparency(e) { this._transparency = e; for (let t = 6; t < 12; ++t) this._lightHelperFrustumMeshes[t].material.alpha = e; } /** * true to display the edges of the frustum */ get showLines() { return this._showLines; } set showLines(e) { if (this._showLines !== e) { this._showLines = e; for (let t = 0; t < 6; ++t) this._lightHelperFrustumMeshes[t].setEnabled(e); } } /** * true to display the planes of the frustum */ get showPlanes() { return this._showPlanes; } set showPlanes(e) { if (this._showPlanes !== e) { this._showPlanes = e; for (let t = 6; t < 12; ++t) this._lightHelperFrustumMeshes[t].setEnabled(e); } } /** * Creates a new frustum viewer * @param light directional light to display the frustum for * @param camera camera used to retrieve the minZ / maxZ values if the shadowMinZ/shadowMaxZ values of the light are not setup */ constructor(e, t) { this._oldPosition = new S(Number.NaN, Number.NaN, Number.NaN), this._oldDirection = new S(Number.NaN, Number.NaN, Number.NaN), this._transparency = 0.3, this._showLines = !0, this._showPlanes = !0, this._scene = e.getScene(), this._light = e, this._camera = t, this._inverseViewMatrix = he.Identity(), this._lightHelperFrustumMeshes = [], this._createGeometry(), this.show(), this.update(); } /** * Shows the frustum */ show() { this._lightHelperFrustumMeshes.forEach((e, t) => { e.setEnabled(t < 6 && this._showLines || t >= 6 && this._showPlanes); }), this._oldPosition.set(Number.NaN, Number.NaN, Number.NaN), this._visible = !0; } /** * Hides the frustum */ hide() { this._lightHelperFrustumMeshes.forEach((e) => { e.setEnabled(!1); }), this._visible = !1; } /** * Updates the frustum. * Call this method to update the frustum view if the light has changed position/direction */ update() { var e, t, r, n, i, s; if (!this._visible || this._oldPosition.equals(this._light.position) && this._oldDirection.equals(this._light.direction) && this._oldAutoCalc === this._light.autoCalcShadowZBounds && this._oldMinZ === this._light.shadowMinZ && this._oldMaxZ === this._light.shadowMaxZ) return; this._oldPosition.copyFrom(this._light.position), this._oldDirection.copyFrom(this._light.direction), this._oldAutoCalc = this._light.autoCalcShadowZBounds, this._oldMinZ = this._light.shadowMinZ, this._oldMaxZ = this._light.shadowMaxZ, ue.Vector3[0].set(this._light.orthoLeft, this._light.orthoBottom, this._light.shadowMinZ !== void 0 ? this._light.shadowMinZ : this._camera.minZ), ue.Vector3[1].set(this._light.orthoRight, this._light.orthoTop, this._light.shadowMaxZ !== void 0 ? this._light.shadowMaxZ : this._camera.maxZ); const a = this._getInvertViewMatrix(); ue.Vector3[2].copyFromFloats(ue.Vector3[1].x, ue.Vector3[1].y, ue.Vector3[0].z), ue.Vector3[3].copyFromFloats(ue.Vector3[1].x, ue.Vector3[0].y, ue.Vector3[0].z), ue.Vector3[4].copyFromFloats(ue.Vector3[0].x, ue.Vector3[0].y, ue.Vector3[0].z), ue.Vector3[5].copyFromFloats(ue.Vector3[0].x, ue.Vector3[1].y, ue.Vector3[0].z), S.TransformCoordinatesToRef(ue.Vector3[2], a, ue.Vector3[2]), S.TransformCoordinatesToRef(ue.Vector3[3], a, ue.Vector3[3]), S.TransformCoordinatesToRef(ue.Vector3[4], a, ue.Vector3[4]), S.TransformCoordinatesToRef(ue.Vector3[5], a, ue.Vector3[5]), ue.Vector3[6].copyFromFloats(ue.Vector3[1].x, ue.Vector3[1].y, ue.Vector3[1].z), ue.Vector3[7].copyFromFloats(ue.Vector3[1].x, ue.Vector3[0].y, ue.Vector3[1].z), ue.Vector3[8].copyFromFloats(ue.Vector3[0].x, ue.Vector3[0].y, ue.Vector3[1].z), ue.Vector3[9].copyFromFloats(ue.Vector3[0].x, ue.Vector3[1].y, ue.Vector3[1].z), S.TransformCoordinatesToRef(ue.Vector3[6], a, ue.Vector3[6]), S.TransformCoordinatesToRef(ue.Vector3[7], a, ue.Vector3[7]), S.TransformCoordinatesToRef(ue.Vector3[8], a, ue.Vector3[8]), S.TransformCoordinatesToRef(ue.Vector3[9], a, ue.Vector3[9]), ka("nearlines", { updatable: !0, points: this._nearLinesPoints, instance: this._lightHelperFrustumMeshes[0] }, this._scene), ka("farlines", { updatable: !0, points: this._farLinesPoints, instance: this._lightHelperFrustumMeshes[1] }, this._scene), ka("trlines", { updatable: !0, points: this._trLinesPoints, instance: this._lightHelperFrustumMeshes[2] }, this._scene), ka("brlines", { updatable: !0, points: this._brLinesPoints, instance: this._lightHelperFrustumMeshes[3] }, this._scene), ka("tllines", { updatable: !0, points: this._tlLinesPoints, instance: this._lightHelperFrustumMeshes[4] }, this._scene), ka("bllines", { updatable: !0, points: this._blLinesPoints, instance: this._lightHelperFrustumMeshes[5] }, this._scene), ue.Vector3[2].toArray(this._nearPlaneVertices, 0), ue.Vector3[3].toArray(this._nearPlaneVertices, 3), ue.Vector3[4].toArray(this._nearPlaneVertices, 6), ue.Vector3[5].toArray(this._nearPlaneVertices, 9), (e = this._lightHelperFrustumMeshes[6].geometry) === null || e === void 0 || e.updateVerticesDataDirectly("position", this._nearPlaneVertices, 0), ue.Vector3[6].toArray(this._farPlaneVertices, 0), ue.Vector3[7].toArray(this._farPlaneVertices, 3), ue.Vector3[8].toArray(this._farPlaneVertices, 6), ue.Vector3[9].toArray(this._farPlaneVertices, 9), (t = this._lightHelperFrustumMeshes[7].geometry) === null || t === void 0 || t.updateVerticesDataDirectly("position", this._farPlaneVertices, 0), ue.Vector3[2].toArray(this._rightPlaneVertices, 0), ue.Vector3[6].toArray(this._rightPlaneVertices, 3), ue.Vector3[7].toArray(this._rightPlaneVertices, 6), ue.Vector3[3].toArray(this._rightPlaneVertices, 9), (r = this._lightHelperFrustumMeshes[8].geometry) === null || r === void 0 || r.updateVerticesDataDirectly("position", this._rightPlaneVertices, 0), ue.Vector3[5].toArray(this._leftPlaneVertices, 0), ue.Vector3[9].toArray(this._leftPlaneVertices, 3), ue.Vector3[8].toArray(this._leftPlaneVertices, 6), ue.Vector3[4].toArray(this._leftPlaneVertices, 9), (n = this._lightHelperFrustumMeshes[9].geometry) === null || n === void 0 || n.updateVerticesDataDirectly("position", this._leftPlaneVertices, 0), ue.Vector3[2].toArray(this._topPlaneVertices, 0), ue.Vector3[6].toArray(this._topPlaneVertices, 3), ue.Vector3[9].toArray(this._topPlaneVertices, 6), ue.Vector3[5].toArray(this._topPlaneVertices, 9), (i = this._lightHelperFrustumMeshes[10].geometry) === null || i === void 0 || i.updateVerticesDataDirectly("position", this._topPlaneVertices, 0), ue.Vector3[3].toArray(this._bottomPlaneVertices, 0), ue.Vector3[7].toArray(this._bottomPlaneVertices, 3), ue.Vector3[8].toArray(this._bottomPlaneVertices, 6), ue.Vector3[4].toArray(this._bottomPlaneVertices, 9), (s = this._lightHelperFrustumMeshes[11].geometry) === null || s === void 0 || s.updateVerticesDataDirectly("position", this._bottomPlaneVertices, 0); } /** * Dispose of the class / remove the frustum view */ dispose() { this._lightHelperFrustumMeshes.forEach((e) => { var t; (t = e.material) === null || t === void 0 || t.dispose(), e.dispose(); }), this._rootNode.dispose(); } _createGeometry() { this._rootNode = new Hr("directionalLightHelperRoot_" + this._light.name, this._scene), this._rootNode.parent = this._light.parent, this._nearLinesPoints = [S.ZeroReadOnly, S.ZeroReadOnly, S.ZeroReadOnly, S.ZeroReadOnly, S.ZeroReadOnly]; const e = ka("nearlines", { updatable: !0, points: this._nearLinesPoints }, this._scene); e.parent = this._rootNode, e.alwaysSelectAsActiveMesh = !0, this._farLinesPoints = [S.ZeroReadOnly, S.ZeroReadOnly, S.ZeroReadOnly, S.ZeroReadOnly, S.ZeroReadOnly]; const t = ka("farlines", { updatable: !0, points: this._farLinesPoints }, this._scene); t.parent = this._rootNode, t.alwaysSelectAsActiveMesh = !0, this._trLinesPoints = [S.ZeroReadOnly, S.ZeroReadOnly]; const r = ka("trlines", { updatable: !0, points: this._trLinesPoints }, this._scene); r.parent = this._rootNode, r.alwaysSelectAsActiveMesh = !0, this._brLinesPoints = [S.ZeroReadOnly, S.ZeroReadOnly]; const n = ka("brlines", { updatable: !0, points: this._brLinesPoints }, this._scene); n.parent = this._rootNode, n.alwaysSelectAsActiveMesh = !0, this._tlLinesPoints = [S.ZeroReadOnly, S.ZeroReadOnly]; const i = ka("tllines", { updatable: !0, points: this._tlLinesPoints }, this._scene); i.parent = this._rootNode, i.alwaysSelectAsActiveMesh = !0, this._blLinesPoints = [S.ZeroReadOnly, S.ZeroReadOnly]; const s = ka("bllines", { updatable: !0, points: this._blLinesPoints }, this._scene); s.parent = this._rootNode, s.alwaysSelectAsActiveMesh = !0, this._lightHelperFrustumMeshes.push(e, t, r, n, i, s); const a = (f, o, d) => { const v = new Ee(f + "plane", this._scene), u = new Wt(f + "PlaneMat", this._scene); v.material = u, v.parent = this._rootNode, v.alwaysSelectAsActiveMesh = !0, u.emissiveColor = o, u.alpha = this.transparency, u.backFaceCulling = !1, u.disableLighting = !0; const l = [0, 1, 2, 0, 2, 3], P = new Ut(); P.positions = d, P.indices = l, P.applyToMesh(v, !0), this._lightHelperFrustumMeshes.push(v); }; this._nearPlaneVertices = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0], this._farPlaneVertices = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0], this._rightPlaneVertices = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0], this._leftPlaneVertices = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0], this._topPlaneVertices = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0], this._bottomPlaneVertices = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0], a("near", new Ne(1, 0, 0), this._nearPlaneVertices), a("far", new Ne(0.3, 0, 0), this._farPlaneVertices), a("right", new Ne(0, 1, 0), this._rightPlaneVertices), a("left", new Ne(0, 0.3, 0), this._leftPlaneVertices), a("top", new Ne(0, 0, 1), this._topPlaneVertices), a("bottom", new Ne(0, 0, 0.3), this._bottomPlaneVertices), this._nearLinesPoints[0] = ue.Vector3[2], this._nearLinesPoints[1] = ue.Vector3[3], this._nearLinesPoints[2] = ue.Vector3[4], this._nearLinesPoints[3] = ue.Vector3[5], this._nearLinesPoints[4] = ue.Vector3[2], this._farLinesPoints[0] = ue.Vector3[6], this._farLinesPoints[1] = ue.Vector3[7], this._farLinesPoints[2] = ue.Vector3[8], this._farLinesPoints[3] = ue.Vector3[9], this._farLinesPoints[4] = ue.Vector3[6], this._trLinesPoints[0] = ue.Vector3[2], this._trLinesPoints[1] = ue.Vector3[6], this._brLinesPoints[0] = ue.Vector3[3], this._brLinesPoints[1] = ue.Vector3[7], this._tlLinesPoints[0] = ue.Vector3[4], this._tlLinesPoints[1] = ue.Vector3[8], this._blLinesPoints[0] = ue.Vector3[5], this._blLinesPoints[1] = ue.Vector3[9]; } _getInvertViewMatrix() { return he.LookAtLHToRef(this._light.position, this._light.position.add(this._light.direction), S.UpReadOnly, this._inverseViewMatrix), this._inverseViewMatrix.invertToRef(this._inverseViewMatrix), this._inverseViewMatrix; } } class et { } et.ALPHA_DISABLE = 0; et.ALPHA_ADD = 1; et.ALPHA_COMBINE = 2; et.ALPHA_SUBTRACT = 3; et.ALPHA_MULTIPLY = 4; et.ALPHA_MAXIMIZED = 5; et.ALPHA_ONEONE = 6; et.ALPHA_PREMULTIPLIED = 7; et.ALPHA_PREMULTIPLIED_PORTERDUFF = 8; et.ALPHA_INTERPOLATE = 9; et.ALPHA_SCREENMODE = 10; et.ALPHA_ONEONE_ONEONE = 11; et.ALPHA_ALPHATOCOLOR = 12; et.ALPHA_REVERSEONEMINUS = 13; et.ALPHA_SRC_DSTONEMINUSSRCALPHA = 14; et.ALPHA_ONEONE_ONEZERO = 15; et.ALPHA_EXCLUSION = 16; et.ALPHA_LAYER_ACCUMULATE = 17; et.ALPHA_EQUATION_ADD = 0; et.ALPHA_EQUATION_SUBSTRACT = 1; et.ALPHA_EQUATION_REVERSE_SUBTRACT = 2; et.ALPHA_EQUATION_MAX = 3; et.ALPHA_EQUATION_MIN = 4; et.ALPHA_EQUATION_DARKEN = 5; et.DELAYLOADSTATE_NONE = 0; et.DELAYLOADSTATE_LOADED = 1; et.DELAYLOADSTATE_LOADING = 2; et.DELAYLOADSTATE_NOTLOADED = 4; et.NEVER = 512; et.ALWAYS = 519; et.LESS = 513; et.EQUAL = 514; et.LEQUAL = 515; et.GREATER = 516; et.GEQUAL = 518; et.NOTEQUAL = 517; et.KEEP = 7680; et.ZERO = 0; et.REPLACE = 7681; et.INCR = 7682; et.DECR = 7683; et.INVERT = 5386; et.INCR_WRAP = 34055; et.DECR_WRAP = 34056; et.TEXTURE_CLAMP_ADDRESSMODE = 0; et.TEXTURE_WRAP_ADDRESSMODE = 1; et.TEXTURE_MIRROR_ADDRESSMODE = 2; et.TEXTURE_CREATIONFLAG_STORAGE = 1; et.TEXTUREFORMAT_ALPHA = 0; et.TEXTUREFORMAT_LUMINANCE = 1; et.TEXTUREFORMAT_LUMINANCE_ALPHA = 2; et.TEXTUREFORMAT_RGB = 4; et.TEXTUREFORMAT_RGBA = 5; et.TEXTUREFORMAT_RED = 6; et.TEXTUREFORMAT_R = 6; et.TEXTUREFORMAT_RG = 7; et.TEXTUREFORMAT_RED_INTEGER = 8; et.TEXTUREFORMAT_R_INTEGER = 8; et.TEXTUREFORMAT_RG_INTEGER = 9; et.TEXTUREFORMAT_RGB_INTEGER = 10; et.TEXTUREFORMAT_RGBA_INTEGER = 11; et.TEXTUREFORMAT_BGRA = 12; et.TEXTUREFORMAT_DEPTH24_STENCIL8 = 13; et.TEXTUREFORMAT_DEPTH32_FLOAT = 14; et.TEXTUREFORMAT_DEPTH16 = 15; et.TEXTUREFORMAT_DEPTH24 = 16; et.TEXTUREFORMAT_DEPTH24UNORM_STENCIL8 = 17; et.TEXTUREFORMAT_DEPTH32FLOAT_STENCIL8 = 18; et.TEXTUREFORMAT_STENCIL8 = 19; et.TEXTUREFORMAT_UNDEFINED = 4294967295; et.TEXTUREFORMAT_COMPRESSED_RGBA_BPTC_UNORM = 36492; et.TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_BPTC_UNORM = 36493; et.TEXTUREFORMAT_COMPRESSED_RGB_BPTC_UNSIGNED_FLOAT = 36495; et.TEXTUREFORMAT_COMPRESSED_RGB_BPTC_SIGNED_FLOAT = 36494; et.TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT5 = 33779; et.TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT = 35919; et.TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT3 = 33778; et.TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_S3TC_DXT3_EXT = 35918; et.TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT1 = 33777; et.TEXTUREFORMAT_COMPRESSED_RGB_S3TC_DXT1 = 33776; et.TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT = 35917; et.TEXTUREFORMAT_COMPRESSED_SRGB_S3TC_DXT1_EXT = 35916; et.TEXTUREFORMAT_COMPRESSED_RGBA_ASTC_4x4 = 37808; et.TEXTUREFORMAT_COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR = 37840; et.TEXTUREFORMAT_COMPRESSED_RGB_ETC1_WEBGL = 36196; et.TEXTUREFORMAT_COMPRESSED_RGB8_ETC2 = 37492; et.TEXTUREFORMAT_COMPRESSED_SRGB8_ETC2 = 37493; et.TEXTUREFORMAT_COMPRESSED_RGB8_PUNCHTHROUGH_ALPHA1_ETC2 = 37494; et.TEXTUREFORMAT_COMPRESSED_SRGB8_PUNCHTHROUGH_ALPHA1_ETC2 = 37495; et.TEXTUREFORMAT_COMPRESSED_RGBA8_ETC2_EAC = 37496; et.TEXTUREFORMAT_COMPRESSED_SRGB8_ALPHA8_ETC2_EAC = 37497; et.TEXTURETYPE_UNSIGNED_BYTE = 0; et.TEXTURETYPE_UNSIGNED_INT = 0; et.TEXTURETYPE_FLOAT = 1; et.TEXTURETYPE_HALF_FLOAT = 2; et.TEXTURETYPE_BYTE = 3; et.TEXTURETYPE_SHORT = 4; et.TEXTURETYPE_UNSIGNED_SHORT = 5; et.TEXTURETYPE_INT = 6; et.TEXTURETYPE_UNSIGNED_INTEGER = 7; et.TEXTURETYPE_UNSIGNED_SHORT_4_4_4_4 = 8; et.TEXTURETYPE_UNSIGNED_SHORT_5_5_5_1 = 9; et.TEXTURETYPE_UNSIGNED_SHORT_5_6_5 = 10; et.TEXTURETYPE_UNSIGNED_INT_2_10_10_10_REV = 11; et.TEXTURETYPE_UNSIGNED_INT_24_8 = 12; et.TEXTURETYPE_UNSIGNED_INT_10F_11F_11F_REV = 13; et.TEXTURETYPE_UNSIGNED_INT_5_9_9_9_REV = 14; et.TEXTURETYPE_FLOAT_32_UNSIGNED_INT_24_8_REV = 15; et.TEXTURETYPE_UNDEFINED = 16; et.TEXTURE_2D = 3553; et.TEXTURE_2D_ARRAY = 35866; et.TEXTURE_CUBE_MAP = 34067; et.TEXTURE_CUBE_MAP_ARRAY = 3735928559; et.TEXTURE_3D = 32879; et.TEXTURE_NEAREST_SAMPLINGMODE = 1; et.TEXTURE_NEAREST_NEAREST = 1; et.TEXTURE_BILINEAR_SAMPLINGMODE = 2; et.TEXTURE_LINEAR_LINEAR = 2; et.TEXTURE_TRILINEAR_SAMPLINGMODE = 3; et.TEXTURE_LINEAR_LINEAR_MIPLINEAR = 3; et.TEXTURE_NEAREST_NEAREST_MIPNEAREST = 4; et.TEXTURE_NEAREST_LINEAR_MIPNEAREST = 5; et.TEXTURE_NEAREST_LINEAR_MIPLINEAR = 6; et.TEXTURE_NEAREST_LINEAR = 7; et.TEXTURE_NEAREST_NEAREST_MIPLINEAR = 8; et.TEXTURE_LINEAR_NEAREST_MIPNEAREST = 9; et.TEXTURE_LINEAR_NEAREST_MIPLINEAR = 10; et.TEXTURE_LINEAR_LINEAR_MIPNEAREST = 11; et.TEXTURE_LINEAR_NEAREST = 12; et.TEXTURE_EXPLICIT_MODE = 0; et.TEXTURE_SPHERICAL_MODE = 1; et.TEXTURE_PLANAR_MODE = 2; et.TEXTURE_CUBIC_MODE = 3; et.TEXTURE_PROJECTION_MODE = 4; et.TEXTURE_SKYBOX_MODE = 5; et.TEXTURE_INVCUBIC_MODE = 6; et.TEXTURE_EQUIRECTANGULAR_MODE = 7; et.TEXTURE_FIXED_EQUIRECTANGULAR_MODE = 8; et.TEXTURE_FIXED_EQUIRECTANGULAR_MIRRORED_MODE = 9; et.TEXTURE_FILTERING_QUALITY_OFFLINE = 4096; et.TEXTURE_FILTERING_QUALITY_HIGH = 64; et.TEXTURE_FILTERING_QUALITY_MEDIUM = 16; et.TEXTURE_FILTERING_QUALITY_LOW = 8; et.SCALEMODE_FLOOR = 1; et.SCALEMODE_NEAREST = 2; et.SCALEMODE_CEILING = 3; et.MATERIAL_TextureDirtyFlag = 1; et.MATERIAL_LightDirtyFlag = 2; et.MATERIAL_FresnelDirtyFlag = 4; et.MATERIAL_AttributesDirtyFlag = 8; et.MATERIAL_MiscDirtyFlag = 16; et.MATERIAL_PrePassDirtyFlag = 32; et.MATERIAL_AllDirtyFlag = 63; et.MATERIAL_TriangleFillMode = 0; et.MATERIAL_WireFrameFillMode = 1; et.MATERIAL_PointFillMode = 2; et.MATERIAL_PointListDrawMode = 3; et.MATERIAL_LineListDrawMode = 4; et.MATERIAL_LineLoopDrawMode = 5; et.MATERIAL_LineStripDrawMode = 6; et.MATERIAL_TriangleStripDrawMode = 7; et.MATERIAL_TriangleFanDrawMode = 8; et.MATERIAL_ClockWiseSideOrientation = 0; et.MATERIAL_CounterClockWiseSideOrientation = 1; et.ACTION_NothingTrigger = 0; et.ACTION_OnPickTrigger = 1; et.ACTION_OnLeftPickTrigger = 2; et.ACTION_OnRightPickTrigger = 3; et.ACTION_OnCenterPickTrigger = 4; et.ACTION_OnPickDownTrigger = 5; et.ACTION_OnDoublePickTrigger = 6; et.ACTION_OnPickUpTrigger = 7; et.ACTION_OnPickOutTrigger = 16; et.ACTION_OnLongPressTrigger = 8; et.ACTION_OnPointerOverTrigger = 9; et.ACTION_OnPointerOutTrigger = 10; et.ACTION_OnEveryFrameTrigger = 11; et.ACTION_OnIntersectionEnterTrigger = 12; et.ACTION_OnIntersectionExitTrigger = 13; et.ACTION_OnKeyDownTrigger = 14; et.ACTION_OnKeyUpTrigger = 15; et.PARTICLES_BILLBOARDMODE_Y = 2; et.PARTICLES_BILLBOARDMODE_ALL = 7; et.PARTICLES_BILLBOARDMODE_STRETCHED = 8; et.PARTICLES_BILLBOARDMODE_STRETCHED_LOCAL = 9; et.MESHES_CULLINGSTRATEGY_STANDARD = 0; et.MESHES_CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY = 1; et.MESHES_CULLINGSTRATEGY_OPTIMISTIC_INCLUSION = 2; et.MESHES_CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY = 3; et.SCENELOADER_NO_LOGGING = 0; et.SCENELOADER_MINIMAL_LOGGING = 1; et.SCENELOADER_SUMMARY_LOGGING = 2; et.SCENELOADER_DETAILED_LOGGING = 3; et.PREPASS_IRRADIANCE_TEXTURE_TYPE = 0; et.PREPASS_POSITION_TEXTURE_TYPE = 1; et.PREPASS_VELOCITY_TEXTURE_TYPE = 2; et.PREPASS_REFLECTIVITY_TEXTURE_TYPE = 3; et.PREPASS_COLOR_TEXTURE_TYPE = 4; et.PREPASS_DEPTH_TEXTURE_TYPE = 5; et.PREPASS_NORMAL_TEXTURE_TYPE = 6; et.PREPASS_ALBEDO_SQRT_TEXTURE_TYPE = 7; et.BUFFER_CREATIONFLAG_READ = 1; et.BUFFER_CREATIONFLAG_WRITE = 2; et.BUFFER_CREATIONFLAG_READWRITE = 3; et.BUFFER_CREATIONFLAG_UNIFORM = 4; et.BUFFER_CREATIONFLAG_VERTEX = 8; et.BUFFER_CREATIONFLAG_INDEX = 16; et.BUFFER_CREATIONFLAG_STORAGE = 32; et.RENDERPASS_MAIN = 0; et.INPUT_ALT_KEY = 18; et.INPUT_CTRL_KEY = 17; et.INPUT_META_KEY1 = 91; et.INPUT_META_KEY2 = 92; et.INPUT_META_KEY3 = 93; et.INPUT_SHIFT_KEY = 16; et.SNAPSHOTRENDERING_STANDARD = 0; et.SNAPSHOTRENDERING_FAST = 1; et.PERSPECTIVE_CAMERA = 0; et.ORTHOGRAPHIC_CAMERA = 1; et.FOVMODE_VERTICAL_FIXED = 0; et.FOVMODE_HORIZONTAL_FIXED = 1; et.RIG_MODE_NONE = 0; et.RIG_MODE_STEREOSCOPIC_ANAGLYPH = 10; et.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL = 11; et.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED = 12; et.RIG_MODE_STEREOSCOPIC_OVERUNDER = 13; et.RIG_MODE_STEREOSCOPIC_INTERLACED = 14; et.RIG_MODE_VR = 20; et.RIG_MODE_CUSTOM = 22; et.MAX_SUPPORTED_UV_SETS = 6; et.GL_ALPHA_EQUATION_ADD = 32774; et.GL_ALPHA_EQUATION_MIN = 32775; et.GL_ALPHA_EQUATION_MAX = 32776; et.GL_ALPHA_EQUATION_SUBTRACT = 32778; et.GL_ALPHA_EQUATION_REVERSE_SUBTRACT = 32779; et.GL_ALPHA_FUNCTION_SRC = 768; et.GL_ALPHA_FUNCTION_ONE_MINUS_SRC_COLOR = 769; et.GL_ALPHA_FUNCTION_SRC_ALPHA = 770; et.GL_ALPHA_FUNCTION_ONE_MINUS_SRC_ALPHA = 771; et.GL_ALPHA_FUNCTION_DST_ALPHA = 772; et.GL_ALPHA_FUNCTION_ONE_MINUS_DST_ALPHA = 773; et.GL_ALPHA_FUNCTION_DST_COLOR = 774; et.GL_ALPHA_FUNCTION_ONE_MINUS_DST_COLOR = 775; et.GL_ALPHA_FUNCTION_SRC_ALPHA_SATURATED = 776; et.GL_ALPHA_FUNCTION_CONSTANT_COLOR = 32769; et.GL_ALPHA_FUNCTION_ONE_MINUS_CONSTANT_COLOR = 32770; et.GL_ALPHA_FUNCTION_CONSTANT_ALPHA = 32771; et.GL_ALPHA_FUNCTION_ONE_MINUS_CONSTANT_ALPHA = 32772; et.SnippetUrl = "https://snippet.babylonjs.com"; class ate { constructor() { this.renderWidth = 512, this.renderHeight = 256, this.textureSize = 512, this.deterministicLockstep = !1, this.lockstepMaxSteps = 4; } } class ote extends Ge { /** * Gets a boolean indicating that the engine is running in deterministic lock step mode * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep * @returns true if engine is in deterministic lock step mode */ isDeterministicLockStep() { return this._options.deterministicLockstep; } /** * Gets the max steps when engine is running in deterministic lock step * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep * @returns the max steps */ getLockstepMaxSteps() { return this._options.lockstepMaxSteps; } /** * Gets the current hardware scaling level. * By default the hardware scaling level is computed from the window device ratio. * if level = 1 then the engine will render at the exact resolution of the canvas. If level = 0.5 then the engine will render at twice the size of the canvas. * @returns a number indicating the current hardware scaling level */ getHardwareScalingLevel() { return 1; } constructor(e = new ate()) { super(null), Ge.Instances.push(this), e.deterministicLockstep === void 0 && (e.deterministicLockstep = !1), e.lockstepMaxSteps === void 0 && (e.lockstepMaxSteps = 4), this._options = e, k9.SetMatrixPrecision(!!e.useHighPrecisionMatrix), this._caps = { maxTexturesImageUnits: 16, maxVertexTextureImageUnits: 16, maxCombinedTexturesImageUnits: 32, maxTextureSize: 512, maxCubemapTextureSize: 512, maxRenderTextureSize: 512, maxVertexAttribs: 16, maxVaryingVectors: 16, maxFragmentUniformVectors: 16, maxVertexUniformVectors: 16, standardDerivatives: !1, astc: null, pvrtc: null, etc1: null, etc2: null, bptc: null, maxAnisotropy: 0, uintIndices: !1, fragmentDepthSupported: !1, highPrecisionShaderSupported: !0, colorBufferFloat: !1, supportFloatTexturesResolve: !1, textureFloat: !1, textureFloatLinearFiltering: !1, textureFloatRender: !1, textureHalfFloat: !1, textureHalfFloatLinearFiltering: !1, textureHalfFloatRender: !1, textureLOD: !1, texelFetch: !1, drawBuffersExtension: !1, depthTextureExtension: !1, vertexArrayObject: !1, instancedArrays: !1, supportOcclusionQuery: !1, canUseTimestampForTimerQuery: !1, maxMSAASamples: 1, blendMinMax: !1, canUseGLInstanceID: !1, canUseGLVertexID: !1, supportComputeShaders: !1, supportSRGBBuffers: !1, supportTransformFeedbacks: !1, textureMaxLevel: !1, texture2DArrayMaxLayerCount: 128, disableMorphTargetTexture: !1 }, this._features = { forceBitmapOverHTMLImageElement: !1, supportRenderAndCopyToLodForFloatTextures: !1, supportDepthStencilTexture: !1, supportShadowSamplers: !1, uniformBufferHardCheckMatrix: !1, allowTexturePrefiltering: !1, trackUbosInFrame: !1, checkUbosContentBeforeUpload: !1, supportCSM: !1, basisNeedsPOT: !1, support3DTextures: !1, needTypeSuffixInShaderConstants: !1, supportMSAA: !1, supportSSAO2: !1, supportExtendedTextureFormats: !1, supportSwitchCaseInShader: !1, supportSyncTextureRead: !1, needsInvertingBitmap: !1, useUBOBindingCache: !1, needShaderCodeInlining: !1, needToAlwaysBindUniformBuffers: !1, supportRenderPasses: !0, supportSpriteInstancing: !1, forceVertexBufferStrideMultiple4Bytes: !1, _collectUbosUpdatedInFrame: !1 }, Se.Log(`Babylon.js v${Ge.Version} - Null engine`); const t = typeof self < "u" ? self : typeof global < "u" ? global : window; typeof URL > "u" && (t.URL = { createObjectURL: function() { }, revokeObjectURL: function() { } }), typeof Blob > "u" && (t.Blob = function() { }); } /** * Creates a vertex buffer * @param vertices the data for the vertex buffer * @returns the new WebGL static buffer */ createVertexBuffer(e) { const t = new tg(); return t.references = 1, t; } /** * Creates a new index buffer * @param indices defines the content of the index buffer * @returns a new webGL buffer */ createIndexBuffer(e) { const t = new tg(); return t.references = 1, t; } /** * Clear the current render buffer or the current render target (if any is set up) * @param color defines the color to use * @param backBuffer defines if the back buffer must be cleared * @param depth defines if the depth buffer must be cleared * @param stencil defines if the stencil buffer must be cleared */ clear(e, t, r, n = !1) { } /** * Gets the current render width * @param useScreen defines if screen size must be used (or the current render target if any) * @returns a number defining the current render width */ getRenderWidth(e = !1) { return !e && this._currentRenderTarget ? this._currentRenderTarget.width : this._options.renderWidth; } /** * Gets the current render height * @param useScreen defines if screen size must be used (or the current render target if any) * @returns a number defining the current render height */ getRenderHeight(e = !1) { return !e && this._currentRenderTarget ? this._currentRenderTarget.height : this._options.renderHeight; } /** * Set the WebGL's viewport * @param viewport defines the viewport element to be used * @param requiredWidth defines the width required for rendering. If not provided the rendering canvas' width is used * @param requiredHeight defines the height required for rendering. If not provided the rendering canvas' height is used */ setViewport(e, t, r) { this._cachedViewport = e; } createShaderProgram(e, t, r, n, i) { return { // eslint-disable-next-line @typescript-eslint/naming-convention __SPECTOR_rebuildProgram: null }; } /** * Gets the list of webGL uniform locations associated with a specific program based on a list of uniform names * @param pipelineContext defines the pipeline context to use * @param uniformsNames defines the list of uniform names * @returns an array of webGL uniform locations */ getUniforms(e, t) { return []; } /** * Gets the lsit of active attributes for a given webGL program * @param pipelineContext defines the pipeline context to use * @param attributesNames defines the list of attribute names to get * @returns an array of indices indicating the offset of each attribute */ getAttributes(e, t) { return []; } /** * Binds an effect to the webGL context * @param effect defines the effect to bind */ bindSamplers(e) { this._currentEffect = null; } /** * Activates an effect, making it the current one (ie. the one used for rendering) * @param effect defines the effect to activate */ enableEffect(e) { e = e !== null && zo.IsWrapper(e) ? e.effect : e, this._currentEffect = e, e && (e.onBind && e.onBind(e), e._onBindObservable && e._onBindObservable.notifyObservers(e)); } /** * Set various states to the webGL context * @param culling defines culling state: true to enable culling, false to disable it * @param zOffset defines the value to apply to zOffset (0 by default) * @param force defines if states must be applied even if cache is up to date * @param reverseSide defines if culling must be reversed (CCW if false, CW if true) * @param cullBackFaces true to cull back faces, false to cull front faces (if culling is enabled) * @param stencil stencil states to set * @param zOffsetUnits defines the value to apply to zOffsetUnits (0 by default) */ setState(e, t = 0, r, n = !1, i, s, a = 0) { } /** * Set the value of an uniform to an array of int32 * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of int32 to store * @returns true if value was set */ setIntArray(e, t) { return !0; } /** * Set the value of an uniform to an array of int32 (stored as vec2) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of int32 to store * @returns true if value was set */ setIntArray2(e, t) { return !0; } /** * Set the value of an uniform to an array of int32 (stored as vec3) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of int32 to store * @returns true if value was set */ setIntArray3(e, t) { return !0; } /** * Set the value of an uniform to an array of int32 (stored as vec4) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of int32 to store * @returns true if value was set */ setIntArray4(e, t) { return !0; } /** * Set the value of an uniform to an array of float32 * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of float32 to store * @returns true if value was set */ setFloatArray(e, t) { return !0; } /** * Set the value of an uniform to an array of float32 (stored as vec2) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of float32 to store * @returns true if value was set */ setFloatArray2(e, t) { return !0; } /** * Set the value of an uniform to an array of float32 (stored as vec3) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of float32 to store * @returns true if value was set */ setFloatArray3(e, t) { return !0; } /** * Set the value of an uniform to an array of float32 (stored as vec4) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of float32 to store * @returns true if value was set */ setFloatArray4(e, t) { return !0; } /** * Set the value of an uniform to an array of number * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of number to store * @returns true if value was set */ setArray(e, t) { return !0; } /** * Set the value of an uniform to an array of number (stored as vec2) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of number to store * @returns true if value was set */ setArray2(e, t) { return !0; } /** * Set the value of an uniform to an array of number (stored as vec3) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of number to store * @returns true if value was set */ setArray3(e, t) { return !0; } /** * Set the value of an uniform to an array of number (stored as vec4) * @param uniform defines the webGL uniform location where to store the value * @param array defines the array of number to store * @returns true if value was set */ setArray4(e, t) { return !0; } /** * Set the value of an uniform to an array of float32 (stored as matrices) * @param uniform defines the webGL uniform location where to store the value * @param matrices defines the array of float32 to store * @returns true if value was set */ setMatrices(e, t) { return !0; } /** * Set the value of an uniform to a matrix (3x3) * @param uniform defines the webGL uniform location where to store the value * @param matrix defines the Float32Array representing the 3x3 matrix to store * @returns true if value was set */ setMatrix3x3(e, t) { return !0; } /** * Set the value of an uniform to a matrix (2x2) * @param uniform defines the webGL uniform location where to store the value * @param matrix defines the Float32Array representing the 2x2 matrix to store * @returns true if value was set */ setMatrix2x2(e, t) { return !0; } /** * Set the value of an uniform to a number (float) * @param uniform defines the webGL uniform location where to store the value * @param value defines the float number to store * @returns true if value was set */ setFloat(e, t) { return !0; } /** * Set the value of an uniform to a vec2 * @param uniform defines the webGL uniform location where to store the value * @param x defines the 1st component of the value * @param y defines the 2nd component of the value * @returns true if value was set */ setFloat2(e, t, r) { return !0; } /** * Set the value of an uniform to a vec3 * @param uniform defines the webGL uniform location where to store the value * @param x defines the 1st component of the value * @param y defines the 2nd component of the value * @param z defines the 3rd component of the value * @returns true if value was set */ setFloat3(e, t, r, n) { return !0; } /** * Set the value of an uniform to a boolean * @param uniform defines the webGL uniform location where to store the value * @param bool defines the boolean to store * @returns true if value was set */ setBool(e, t) { return !0; } /** * Set the value of an uniform to a vec4 * @param uniform defines the webGL uniform location where to store the value * @param x defines the 1st component of the value * @param y defines the 2nd component of the value * @param z defines the 3rd component of the value * @param w defines the 4th component of the value * @returns true if value was set */ setFloat4(e, t, r, n, i) { return !0; } /** * Sets the current alpha mode * @param mode defines the mode to use (one of the Engine.ALPHA_XXX) * @param noDepthWriteChange defines if depth writing state should remains unchanged (false by default) * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/advanced/transparent_rendering */ setAlphaMode(e, t = !1) { this._alphaMode !== e && (this.alphaState.alphaBlend = e !== 0, t || this.setDepthWrite(e === 0), this._alphaMode = e); } /** * Bind webGl buffers directly to the webGL context * @param vertexBuffers defines the vertex buffer to bind * @param indexBuffer defines the index buffer to bind * @param effect defines the effect associated with the vertex buffer */ bindBuffers(e, t, r) { } /** * Force the entire cache to be cleared * You should not have to use this function unless your engine needs to share the webGL context with another engine * @param bruteForce defines a boolean to force clearing ALL caches (including stencil, detoh and alpha states) */ wipeCaches(e) { this.preventCacheWipeBetweenFrames || (this.resetTextureCache(), this._currentEffect = null, e && (this._currentProgram = null, this._stencilStateComposer.reset(), this.depthCullingState.reset(), this.alphaState.reset()), this._cachedVertexBuffers = null, this._cachedIndexBuffer = null, this._cachedEffectForVertexBuffers = null); } /** * Send a draw order * @param useTriangles defines if triangles must be used to draw (else wireframe will be used) * @param indexStart defines the starting index * @param indexCount defines the number of index to draw * @param instancesCount defines the number of instances to draw (if instantiation is enabled) */ draw(e, t, r, n) { } /** * Draw a list of indexed primitives * @param fillMode defines the primitive to use * @param indexStart defines the starting index * @param indexCount defines the number of index to draw * @param instancesCount defines the number of instances to draw (if instantiation is enabled) */ drawElementsType(e, t, r, n) { } /** * Draw a list of unindexed primitives * @param fillMode defines the primitive to use * @param verticesStart defines the index of first vertex to draw * @param verticesCount defines the count of vertices to draw * @param instancesCount defines the number of instances to draw (if instantiation is enabled) */ drawArraysType(e, t, r, n) { } /** @internal */ _createTexture() { return {}; } /** * @internal */ _releaseTexture(e) { } /** * Usually called from Texture.ts. * Passed information to create a WebGLTexture * @param urlArg defines a value which contains one of the following: * * A conventional http URL, e.g. 'http://...' or 'file://...' * * A base64 string of in-line texture data, e.g. '...' * * An indicator that data being passed using the buffer parameter, e.g. 'data:mytexture.jpg' * @param noMipmap defines a boolean indicating that no mipmaps shall be generated. Ignored for compressed textures. They must be in the file * @param invertY when true, image is flipped when loaded. You probably want true. Certain compressed textures may invert this if their default is inverted (eg. ktx) * @param scene needed for loading to the correct scene * @param samplingMode mode with should be used sample / access the texture (Default: Texture.TRILINEAR_SAMPLINGMODE) * @param onLoad optional callback to be called upon successful completion * @param onError optional callback to be called upon failure * @param buffer a source of a file previously fetched as either a base64 string, an ArrayBuffer (compressed or image format), HTMLImageElement (image format), or a Blob * @param fallback an internal argument in case the function must be called again, due to etc1 not having alpha capabilities * @param format internal format. Default: RGB when extension is '.jpg' else RGBA. Ignored for compressed textures * @param forcedExtension defines the extension to use to pick the right loader * @param mimeType defines an optional mime type * @returns a InternalTexture for assignment back into BABYLON.Texture */ createTexture(e, t, r, n, i = 3, s = null, a = null, f = null, o = null, d = null, v = null, u) { const l = new As(this, ri.Url), P = String(e); return l.url = P, l.generateMipMaps = !t, l.samplingMode = i, l.invertY = r, l.baseWidth = this._options.textureSize, l.baseHeight = this._options.textureSize, l.width = this._options.textureSize, l.height = this._options.textureSize, d && (l.format = d), l.isReady = !0, s && setTimeout(() => { s(l); }), this._internalTexturesCache.push(l), l; } /** * @internal */ _createHardwareRenderTargetWrapper(e, t, r) { const n = new DR(e, t, r, this); return this._renderTargetWrapperCache.push(n), n; } /** * Creates a new render target wrapper * @param size defines the size of the texture * @param options defines the options used to create the texture * @returns a new render target wrapper */ createRenderTargetTexture(e, t) { const r = this._createHardwareRenderTargetWrapper(!1, !1, e), n = {}; t !== void 0 && typeof t == "object" ? (n.generateMipMaps = t.generateMipMaps, n.generateDepthBuffer = t.generateDepthBuffer === void 0 ? !0 : t.generateDepthBuffer, n.generateStencilBuffer = n.generateDepthBuffer && t.generateStencilBuffer, n.type = t.type === void 0 ? 0 : t.type, n.samplingMode = t.samplingMode === void 0 ? 3 : t.samplingMode) : (n.generateMipMaps = t, n.generateDepthBuffer = !0, n.generateStencilBuffer = !1, n.type = 0, n.samplingMode = 3); const i = new As(this, ri.RenderTarget), s = e.width || e, a = e.height || e; return r._generateDepthBuffer = n.generateDepthBuffer, r._generateStencilBuffer = !!n.generateStencilBuffer, i.baseWidth = s, i.baseHeight = a, i.width = s, i.height = a, i.isReady = !0, i.samples = 1, i.generateMipMaps = !!n.generateMipMaps, i.samplingMode = n.samplingMode, i.type = n.type, this._internalTexturesCache.push(i), r; } /** * Creates a new render target wrapper * @param size defines the size of the texture * @param options defines the options used to create the texture * @returns a new render target wrapper */ createRenderTargetCubeTexture(e, t) { const r = this._createHardwareRenderTargetWrapper(!1, !0, e), n = Object.assign({ generateMipMaps: !0, generateDepthBuffer: !0, generateStencilBuffer: !1, type: 0, samplingMode: 3, format: 5 }, t); n.generateStencilBuffer = n.generateDepthBuffer && n.generateStencilBuffer, (n.type === 1 && !this._caps.textureFloatLinearFiltering || n.type === 2 && !this._caps.textureHalfFloatLinearFiltering) && (n.samplingMode = 1), r._generateDepthBuffer = n.generateDepthBuffer, r._generateStencilBuffer = !!n.generateStencilBuffer; const i = new As(this, ri.RenderTarget); return i.baseWidth = e, i.baseHeight = e, i.width = e, i.height = e, i.isReady = !0, i.isCube = !0, i.samples = 1, i.generateMipMaps = !!n.generateMipMaps, i.samplingMode = n.samplingMode, i.type = n.type, this._internalTexturesCache.push(i), r; } /** * Update the sampling mode of a given texture * @param samplingMode defines the required sampling mode * @param texture defines the texture to update */ updateTextureSamplingMode(e, t) { t.samplingMode = e; } /** * Creates a raw texture * @param data defines the data to store in the texture * @param width defines the width of the texture * @param height defines the height of the texture * @param format defines the format of the data * @param generateMipMaps defines if the engine should generate the mip levels * @param invertY defines if data must be stored with Y axis inverted * @param samplingMode defines the required sampling mode (Texture.NEAREST_SAMPLINGMODE by default) * @param compression defines the compression used (null by default) * @param type defines the type fo the data (Engine.TEXTURETYPE_UNSIGNED_INT by default) * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg) * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU). * @returns the raw texture inside an InternalTexture */ createRawTexture(e, t, r, n, i, s, a, f = null, o = 0, d = 0, v = !1) { const u = new As(this, ri.Raw); return u.baseWidth = t, u.baseHeight = r, u.width = t, u.height = r, u.format = n, u.generateMipMaps = i, u.samplingMode = a, u.invertY = s, u._compression = f, u.type = o, u._useSRGBBuffer = v, this._doNotHandleContextLost || (u._bufferView = e), u; } /** * Update a raw texture * @param texture defines the texture to update * @param data defines the data to store in the texture * @param format defines the format of the data * @param invertY defines if data must be stored with Y axis inverted * @param compression defines the compression used (null by default) * @param type defines the type fo the data (Engine.TEXTURETYPE_UNSIGNED_INT by default) * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU). */ updateRawTexture(e, t, r, n, i = null, s = 0, a = !1) { e && (e._bufferView = t, e.format = r, e.invertY = n, e._compression = i, e.type = s, e._useSRGBBuffer = a); } /** * Binds the frame buffer to the specified texture. * @param rtWrapper The render target wrapper to render to * @param faceIndex The face of the texture to render to in case of cube texture * @param requiredWidth The width of the target to render to * @param requiredHeight The height of the target to render to * @param forceFullscreenViewport Forces the viewport to be the entire texture/screen if true */ bindFramebuffer(e, t, r, n, i) { this._currentRenderTarget && this.unBindFramebuffer(this._currentRenderTarget), this._currentRenderTarget = e, this._currentFramebuffer = null, this._cachedViewport && !i && this.setViewport(this._cachedViewport, r, n); } /** * Unbind the current render target texture from the webGL context * @param rtWrapper defines the render target wrapper to unbind * @param disableGenerateMipMaps defines a boolean indicating that mipmaps must not be generated * @param onBeforeUnbind defines a function which will be called before the effective unbind */ unBindFramebuffer(e, t = !1, r) { this._currentRenderTarget = null, r && r(), this._currentFramebuffer = null; } /** * Creates a dynamic vertex buffer * @param vertices the data for the dynamic vertex buffer * @returns the new WebGL dynamic buffer */ createDynamicVertexBuffer(e) { const t = new tg(); return t.references = 1, t.capacity = 1, t; } /** * Update the content of a dynamic texture * @param texture defines the texture to update * @param canvas defines the canvas containing the source * @param invertY defines if data must be stored with Y axis inverted * @param premulAlpha defines if alpha is stored as premultiplied * @param format defines the format of the data */ updateDynamicTexture(e, t, r, n = !1, i) { } /** * Gets a boolean indicating if all created effects are ready * @returns true if all effects are ready */ areAllEffectsReady() { return !0; } /** * @internal * Get the current error code of the webGL context * @returns the error code * @see https://developer.mozilla.org/en-US/docs/Web/API/WebGLRenderingContext/getError */ getError() { return 0; } /** @internal */ _getUnpackAlignement() { return 1; } /** * @internal */ _unpackFlipY(e) { } /** * Update a dynamic index buffer * @param indexBuffer defines the target index buffer * @param indices defines the data to update * @param offset defines the offset in the target index buffer where update should start */ updateDynamicIndexBuffer(e, t, r = 0) { } /** * Updates a dynamic vertex buffer. * @param vertexBuffer the vertex buffer to update * @param vertices the data used to update the vertex buffer * @param byteOffset the byte offset of the data (optional) * @param byteLength the byte length of the data (optional) */ updateDynamicVertexBuffer(e, t, r, n) { } /** * @internal */ _bindTextureDirectly(e, t) { return this._boundTexturesCache[this._activeChannel] !== t ? (this._boundTexturesCache[this._activeChannel] = t, !0) : !1; } /** * @internal */ _bindTexture(e, t) { e < 0 || this._bindTextureDirectly(0, t); } _deleteBuffer(e) { } /** * Force the engine to release all cached effects. This means that next effect compilation will have to be done completely even if a similar effect was already compiled */ releaseEffects() { } displayLoadingUI() { } hideLoadingUI() { } set loadingUIText(e) { } /** * @internal */ _uploadCompressedDataToTextureDirectly(e, t, r, n, i, s = 0, a = 0) { } /** * @internal */ _uploadDataToTextureDirectly(e, t, r = 0, n = 0) { } /** * @internal */ _uploadArrayBufferViewToTexture(e, t, r = 0, n = 0) { } /** * @internal */ _uploadImageToTexture(e, t, r = 0, n = 0) { } } hr.prototype._debugPushGroup = function(A, e) { }; hr.prototype._debugPopGroup = function(A) { }; hr.prototype._debugInsertMarker = function(A, e) { }; hr.prototype._debugFlushPendingCommands = function() { }; class fte { constructor() { this._timeElapsedQueryEnded = !1; } } class Ate { constructor() { this.occlusionInternalRetryCounter = 0, this.isOcclusionQueryInProgress = !1, this.isOccluded = !1, this.occlusionRetryCount = -1, this.occlusionType = jn.OCCLUSION_TYPE_NONE, this.occlusionQueryAlgorithmType = jn.OCCLUSION_ALGORITHM_TYPE_CONSERVATIVE, this.forceRenderingWhenOccluded = !1; } } Ge.prototype.createQuery = function() { const A = this._gl.createQuery(); if (!A) throw new Error("Unable to create Occlusion Query"); return A; }; Ge.prototype.deleteQuery = function(A) { return this._gl.deleteQuery(A), this; }; Ge.prototype.isQueryResultAvailable = function(A) { return this._gl.getQueryParameter(A, this._gl.QUERY_RESULT_AVAILABLE); }; Ge.prototype.getQueryResult = function(A) { return this._gl.getQueryParameter(A, this._gl.QUERY_RESULT); }; Ge.prototype.beginOcclusionQuery = function(A, e) { const t = this._getGlAlgorithmType(A); return this._gl.beginQuery(t, e), !0; }; Ge.prototype.endOcclusionQuery = function(A) { const e = this._getGlAlgorithmType(A); return this._gl.endQuery(e), this; }; Ge.prototype._createTimeQuery = function() { const A = this.getCaps().timerQuery; return A.createQueryEXT ? A.createQueryEXT() : this.createQuery(); }; Ge.prototype._deleteTimeQuery = function(A) { const e = this.getCaps().timerQuery; if (e.deleteQueryEXT) { e.deleteQueryEXT(A); return; } this.deleteQuery(A); }; Ge.prototype._getTimeQueryResult = function(A) { const e = this.getCaps().timerQuery; return e.getQueryObjectEXT ? e.getQueryObjectEXT(A, e.QUERY_RESULT_EXT) : this.getQueryResult(A); }; Ge.prototype._getTimeQueryAvailability = function(A) { const e = this.getCaps().timerQuery; return e.getQueryObjectEXT ? e.getQueryObjectEXT(A, e.QUERY_RESULT_AVAILABLE_EXT) : this.isQueryResultAvailable(A); }; Ge.prototype.startTimeQuery = function() { const A = this.getCaps(), e = A.timerQuery; if (!e) return null; const t = new fte(); if (this._gl.getParameter(e.GPU_DISJOINT_EXT), A.canUseTimestampForTimerQuery) t._startTimeQuery = this._createTimeQuery(), e.queryCounterEXT(t._startTimeQuery, e.TIMESTAMP_EXT); else { if (this._currentNonTimestampToken) return this._currentNonTimestampToken; t._timeElapsedQuery = this._createTimeQuery(), e.beginQueryEXT ? e.beginQueryEXT(e.TIME_ELAPSED_EXT, t._timeElapsedQuery) : this._gl.beginQuery(e.TIME_ELAPSED_EXT, t._timeElapsedQuery), this._currentNonTimestampToken = t; } return t; }; Ge.prototype.endTimeQuery = function(A) { const e = this.getCaps(), t = e.timerQuery; if (!t || !A) return -1; if (e.canUseTimestampForTimerQuery) { if (!A._startTimeQuery) return -1; A._endTimeQuery || (A._endTimeQuery = this._createTimeQuery(), t.queryCounterEXT(A._endTimeQuery, t.TIMESTAMP_EXT)); } else if (!A._timeElapsedQueryEnded) { if (!A._timeElapsedQuery) return -1; t.endQueryEXT ? t.endQueryEXT(t.TIME_ELAPSED_EXT) : (this._gl.endQuery(t.TIME_ELAPSED_EXT), this._currentNonTimestampToken = null), A._timeElapsedQueryEnded = !0; } const r = this._gl.getParameter(t.GPU_DISJOINT_EXT); let n = !1; if (A._endTimeQuery ? n = this._getTimeQueryAvailability(A._endTimeQuery) : A._timeElapsedQuery && (n = this._getTimeQueryAvailability(A._timeElapsedQuery)), n && !r) { let i = 0; if (e.canUseTimestampForTimerQuery) { if (!A._startTimeQuery || !A._endTimeQuery) return -1; const s = this._getTimeQueryResult(A._startTimeQuery); i = this._getTimeQueryResult(A._endTimeQuery) - s, this._deleteTimeQuery(A._startTimeQuery), this._deleteTimeQuery(A._endTimeQuery), A._startTimeQuery = null, A._endTimeQuery = null; } else { if (!A._timeElapsedQuery) return -1; i = this._getTimeQueryResult(A._timeElapsedQuery), this._deleteTimeQuery(A._timeElapsedQuery), A._timeElapsedQuery = null, A._timeElapsedQueryEnded = !1; } return i; } return -1; }; Ge.prototype._captureGPUFrameTime = !1; Ge.prototype._gpuFrameTime = new v9(); Ge.prototype.getGPUFrameTimeCounter = function() { return this._gpuFrameTime; }; Ge.prototype.captureGPUFrameTime = function(A) { A !== this._captureGPUFrameTime && (this._captureGPUFrameTime = A, A ? (this._onBeginFrameObserver = this.onBeginFrameObservable.add(() => { this._gpuFrameTimeToken || (this._gpuFrameTimeToken = this.startTimeQuery()); }), this._onEndFrameObserver = this.onEndFrameObservable.add(() => { if (!this._gpuFrameTimeToken) return; const e = this.endTimeQuery(this._gpuFrameTimeToken); e > -1 && (this._gpuFrameTimeToken = null, this._gpuFrameTime.fetchNewFrame(), this._gpuFrameTime.addCount(e, !0)); })) : (this.onBeginFrameObservable.remove(this._onBeginFrameObserver), this._onBeginFrameObserver = null, this.onEndFrameObservable.remove(this._onEndFrameObserver), this._onEndFrameObserver = null)); }; Ge.prototype._getGlAlgorithmType = function(A) { return A === jn.OCCLUSION_ALGORITHM_TYPE_CONSERVATIVE ? this._gl.ANY_SAMPLES_PASSED_CONSERVATIVE : this._gl.ANY_SAMPLES_PASSED; }; Object.defineProperty(jn.prototype, "isOcclusionQueryInProgress", { get: function() { return this._occlusionDataStorage.isOcclusionQueryInProgress; }, set: function(A) { this._occlusionDataStorage.isOcclusionQueryInProgress = A; }, enumerable: !1, configurable: !0 }); Object.defineProperty(jn.prototype, "_occlusionDataStorage", { get: function() { return this.__occlusionDataStorage || (this.__occlusionDataStorage = new Ate()), this.__occlusionDataStorage; }, enumerable: !1, configurable: !0 }); Object.defineProperty(jn.prototype, "isOccluded", { get: function() { return this._occlusionDataStorage.isOccluded; }, set: function(A) { this._occlusionDataStorage.isOccluded = A; }, enumerable: !0, configurable: !0 }); Object.defineProperty(jn.prototype, "occlusionQueryAlgorithmType", { get: function() { return this._occlusionDataStorage.occlusionQueryAlgorithmType; }, set: function(A) { this._occlusionDataStorage.occlusionQueryAlgorithmType = A; }, enumerable: !0, configurable: !0 }); Object.defineProperty(jn.prototype, "occlusionType", { get: function() { return this._occlusionDataStorage.occlusionType; }, set: function(A) { this._occlusionDataStorage.occlusionType = A; }, enumerable: !0, configurable: !0 }); Object.defineProperty(jn.prototype, "occlusionRetryCount", { get: function() { return this._occlusionDataStorage.occlusionRetryCount; }, set: function(A) { this._occlusionDataStorage.occlusionRetryCount = A; }, enumerable: !0, configurable: !0 }); Object.defineProperty(jn.prototype, "forceRenderingWhenOccluded", { get: function() { return this._occlusionDataStorage.forceRenderingWhenOccluded; }, set: function(A) { this._occlusionDataStorage.forceRenderingWhenOccluded = A; }, enumerable: !0, configurable: !0 }); jn.prototype._checkOcclusionQuery = function() { const A = this._occlusionDataStorage; if (A.occlusionType === jn.OCCLUSION_TYPE_NONE) return A.isOccluded = !1, !1; const e = this.getEngine(); if (!e.getCaps().supportOcclusionQuery || !e.isQueryResultAvailable) return A.isOccluded = !1, !1; if (this.isOcclusionQueryInProgress && this._occlusionQuery !== null && this._occlusionQuery !== void 0) if (e.isQueryResultAvailable(this._occlusionQuery)) { const n = e.getQueryResult(this._occlusionQuery); A.isOcclusionQueryInProgress = !1, A.occlusionInternalRetryCounter = 0, A.isOccluded = !(n > 0); } else if (A.occlusionInternalRetryCounter++, A.occlusionRetryCount !== -1 && A.occlusionInternalRetryCounter > A.occlusionRetryCount) A.isOcclusionQueryInProgress = !1, A.occlusionInternalRetryCounter = 0, A.isOccluded = A.occlusionType === jn.OCCLUSION_TYPE_OPTIMISTIC ? !1 : A.isOccluded; else return A.occlusionType === jn.OCCLUSION_TYPE_OPTIMISTIC ? !1 : A.isOccluded; const t = this.getScene(); if (t.getBoundingBoxRenderer) { const r = t.getBoundingBoxRenderer(); this._occlusionQuery === null && (this._occlusionQuery = e.createQuery()), e.beginOcclusionQuery(A.occlusionQueryAlgorithmType, this._occlusionQuery) && (r.renderOcclusionBoundingBox(this), e.endOcclusionQuery(A.occlusionQueryAlgorithmType), this._occlusionDataStorage.isOcclusionQueryInProgress = !0); } return A.isOccluded; }; var oue = !0; Ge.prototype.createTransformFeedback = function() { const A = this._gl.createTransformFeedback(); if (!A) throw new Error("Unable to create Transform Feedback"); return A; }; Ge.prototype.deleteTransformFeedback = function(A) { this._gl.deleteTransformFeedback(A); }; Ge.prototype.bindTransformFeedback = function(A) { this._gl.bindTransformFeedback(this._gl.TRANSFORM_FEEDBACK, A); }; Ge.prototype.beginTransformFeedback = function(A = !0) { this._gl.beginTransformFeedback(A ? this._gl.POINTS : this._gl.TRIANGLES); }; Ge.prototype.endTransformFeedback = function() { this._gl.endTransformFeedback(); }; Ge.prototype.setTranformFeedbackVaryings = function(A, e) { this._gl.transformFeedbackVaryings(A, e, this._gl.INTERLEAVED_ATTRIBS); }; Ge.prototype.bindTransformFeedbackBuffer = function(A) { this._gl.bindBufferBase(this._gl.TRANSFORM_FEEDBACK_BUFFER, 0, A ? A.underlyingResource : null); }; hr.prototype.createExternalTexture = function(A) { return null; }; hr.prototype.setExternalTexture = function(A, e) { throw new Error("setExternalTexture: This engine does not support external textures!"); }; hr.prototype.updateVideoTexture = function(A, e, t) { if (!A || A._isDisabled) return; const r = this._getInternalFormat(A.format), n = this._getRGBABufferInternalSizedFormat(0, A.format), i = this._bindTextureDirectly(this._gl.TEXTURE_2D, A, !0); this._unpackFlipY(!t); try { if (this._videoTextureSupported === void 0 && (this._gl.getError(), this._gl.texImage2D(this._gl.TEXTURE_2D, 0, n, r, this._gl.UNSIGNED_BYTE, e), this._gl.getError() !== 0 ? this._videoTextureSupported = !1 : this._videoTextureSupported = !0), this._videoTextureSupported) this._gl.texImage2D(this._gl.TEXTURE_2D, 0, n, r, this._gl.UNSIGNED_BYTE, e); else { if (!A._workingCanvas) { A._workingCanvas = this.createCanvas(A.width, A.height); const s = A._workingCanvas.getContext("2d"); if (!s) throw new Error("Unable to get 2d context"); A._workingContext = s, A._workingCanvas.width = A.width, A._workingCanvas.height = A.height; } A._workingContext.clearRect(0, 0, A.width, A.height), A._workingContext.drawImage(e, 0, 0, e.videoWidth, e.videoHeight, 0, 0, A.width, A.height), this._gl.texImage2D(this._gl.TEXTURE_2D, 0, n, r, this._gl.UNSIGNED_BYTE, A._workingCanvas); } A.generateMipMaps && this._gl.generateMipmap(this._gl.TEXTURE_2D), i || this._bindTextureDirectly(this._gl.TEXTURE_2D, null), A.isReady = !0; } catch { A._isDisabled = !0; } }; hr.prototype.restoreSingleAttachment = function() { const A = this._gl; this.bindAttachments([A.BACK]); }; hr.prototype.restoreSingleAttachmentForRenderTarget = function() { const A = this._gl; this.bindAttachments([A.COLOR_ATTACHMENT0]); }; hr.prototype.buildTextureLayout = function(A) { const e = this._gl, t = []; for (let r = 0; r < A.length; r++) A[r] ? t.push(e["COLOR_ATTACHMENT" + r]) : t.push(e.NONE); return t; }; hr.prototype.bindAttachments = function(A) { this._gl.drawBuffers(A); }; hr.prototype.unBindMultiColorAttachmentFramebuffer = function(A, e = !1, t) { this._currentRenderTarget = null; const r = this._gl, n = A._attachments, i = n.length; if (A._MSAAFramebuffer) { r.bindFramebuffer(r.READ_FRAMEBUFFER, A._MSAAFramebuffer), r.bindFramebuffer(r.DRAW_FRAMEBUFFER, A._framebuffer); for (let s = 0; s < i; s++) { const a = A.textures[s]; for (let f = 0; f < i; f++) n[f] = r.NONE; n[s] = r[this.webGLVersion > 1 ? "COLOR_ATTACHMENT" + s : "COLOR_ATTACHMENT" + s + "_WEBGL"], r.readBuffer(n[s]), r.drawBuffers(n), r.blitFramebuffer(0, 0, a.width, a.height, 0, 0, a.width, a.height, r.COLOR_BUFFER_BIT, r.NEAREST); } for (let s = 0; s < i; s++) n[s] = r[this.webGLVersion > 1 ? "COLOR_ATTACHMENT" + s : "COLOR_ATTACHMENT" + s + "_WEBGL"]; r.drawBuffers(n); } for (let s = 0; s < i; s++) { const a = A.textures[s]; a != null && a.generateMipMaps && !e && !a.isCube && (this._bindTextureDirectly(r.TEXTURE_2D, a, !0), r.generateMipmap(r.TEXTURE_2D), this._bindTextureDirectly(r.TEXTURE_2D, null)); } t && (A._MSAAFramebuffer && this._bindUnboundFramebuffer(A._framebuffer), t()), this._bindUnboundFramebuffer(null); }; hr.prototype.createMultipleRenderTarget = function(A, e, t = !0) { var r; let n = !1, i = !0, s = !1, a = !1, f = 15, o = 1; const d = 0, v = 3, u = !1, l = 5, P = 3553; let p = [], c = [], H = [], T = [], q = [], b = [], j = [], w = []; const m = this._createHardwareRenderTargetWrapper(!0, !1, A); e !== void 0 && (n = e.generateMipMaps === void 0 ? !1 : e.generateMipMaps, i = e.generateDepthBuffer === void 0 ? !0 : e.generateDepthBuffer, s = e.generateStencilBuffer === void 0 ? !1 : e.generateStencilBuffer, a = e.generateDepthTexture === void 0 ? !1 : e.generateDepthTexture, o = e.textureCount || 1, e.types && (p = e.types), e.samplingModes && (c = e.samplingModes), e.useSRGBBuffers && (H = e.useSRGBBuffers), e.formats && (T = e.formats), e.targetTypes && (q = e.targetTypes), e.faceIndex && (b = e.faceIndex), e.layerIndex && (j = e.layerIndex), e.layerCounts && (w = e.layerCounts), this.webGLVersion > 1 && (e.depthTextureFormat === 13 || e.depthTextureFormat === 17 || e.depthTextureFormat === 16 || e.depthTextureFormat === 14 || e.depthTextureFormat === 18) && (f = e.depthTextureFormat)); const I = this._gl, N = I.createFramebuffer(); this._bindUnboundFramebuffer(N); const k = A.width || A, R = A.height || A, y = [], O = [], Y = this.webGLVersion > 1 && a && (e.depthTextureFormat === 13 || e.depthTextureFormat === 17 || e.depthTextureFormat === 18), ee = this._setupFramebufferDepthAttachments(!Y && s, !a && i, k, R); m._framebuffer = N, m._depthStencilBuffer = ee, m._generateDepthBuffer = !a && i, m._generateStencilBuffer = !Y && s, m._attachments = O; for (let Z = 0; Z < o; Z++) { let te = c[Z] || v, fe = p[Z] || d, _ = H[Z] || u; const G = T[Z] || l, L = q[Z] || P, $ = (r = w[Z]) !== null && r !== void 0 ? r : 1; (fe === 1 && !this._caps.textureFloatLinearFiltering || fe === 2 && !this._caps.textureHalfFloatLinearFiltering) && (te = 1); const ae = this._getSamplingParameters(te, n); fe === 1 && !this._caps.textureFloat && (fe = 0, Se.Warn("Float textures are not supported. Render target forced to TEXTURETYPE_UNSIGNED_BYTE type")), _ = _ && this._caps.supportSRGBBuffers && (this.webGLVersion > 1 || this.isWebGPU); const Pe = this.webGLVersion > 1, ge = I[Pe ? "COLOR_ATTACHMENT" + Z : "COLOR_ATTACHMENT" + Z + "_WEBGL"]; if (O.push(ge), L === -1) continue; const me = new As(this, ri.MultiRenderTarget); y[Z] = me, I.activeTexture(I["TEXTURE" + Z]), I.bindTexture(L, me._hardwareTexture.underlyingResource), I.texParameteri(L, I.TEXTURE_MAG_FILTER, ae.mag), I.texParameteri(L, I.TEXTURE_MIN_FILTER, ae.min), I.texParameteri(L, I.TEXTURE_WRAP_S, I.CLAMP_TO_EDGE), I.texParameteri(L, I.TEXTURE_WRAP_T, I.CLAMP_TO_EDGE); const Xe = this._getRGBABufferInternalSizedFormat(fe, G, _), De = this._getInternalFormat(G), ne = this._getWebGLTextureType(fe); if (Pe && (L === 35866 || L === 32879)) L === 35866 ? me.is2DArray = !0 : me.is3D = !0, me.baseDepth = me.depth = $, I.texImage3D(L, 0, Xe, k, R, $, 0, De, ne, null); else if (L === 34067) { for (let re = 0; re < 6; re++) I.texImage2D(I.TEXTURE_CUBE_MAP_POSITIVE_X + re, 0, Xe, k, R, 0, De, ne, null); me.isCube = !0; } else I.texImage2D(I.TEXTURE_2D, 0, Xe, k, R, 0, De, ne, null); n && I.generateMipmap(L), this._bindTextureDirectly(L, null), me.baseWidth = k, me.baseHeight = R, me.width = k, me.height = R, me.isReady = !0, me.samples = 1, me.generateMipMaps = n, me.samplingMode = te, me.type = fe, me._useSRGBBuffer = _, me.format = G, this._internalTexturesCache.push(me); } if (a && this._caps.depthTextureExtension) { const Z = new As(this, ri.Depth); let te = 5, fe = I.DEPTH_COMPONENT16, _ = I.DEPTH_COMPONENT, G = I.UNSIGNED_SHORT, L = I.DEPTH_ATTACHMENT; this.webGLVersion < 2 ? fe = I.DEPTH_COMPONENT : f === 14 ? (te = 1, G = I.FLOAT, fe = I.DEPTH_COMPONENT32F) : f === 18 ? (te = 0, G = I.FLOAT_32_UNSIGNED_INT_24_8_REV, fe = I.DEPTH32F_STENCIL8, _ = I.DEPTH_STENCIL, L = I.DEPTH_STENCIL_ATTACHMENT) : f === 16 ? (te = 0, G = I.UNSIGNED_INT, fe = I.DEPTH_COMPONENT24, L = I.DEPTH_ATTACHMENT) : (f === 13 || f === 17) && (te = 12, G = I.UNSIGNED_INT_24_8, fe = I.DEPTH24_STENCIL8, _ = I.DEPTH_STENCIL, L = I.DEPTH_STENCIL_ATTACHMENT), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, Z._hardwareTexture.underlyingResource), I.texParameteri(I.TEXTURE_2D, I.TEXTURE_MAG_FILTER, I.NEAREST), I.texParameteri(I.TEXTURE_2D, I.TEXTURE_MIN_FILTER, I.NEAREST), I.texParameteri(I.TEXTURE_2D, I.TEXTURE_WRAP_S, I.CLAMP_TO_EDGE), I.texParameteri(I.TEXTURE_2D, I.TEXTURE_WRAP_T, I.CLAMP_TO_EDGE), I.texImage2D(I.TEXTURE_2D, 0, fe, k, R, 0, _, G, null), I.framebufferTexture2D(I.FRAMEBUFFER, L, I.TEXTURE_2D, Z._hardwareTexture.underlyingResource, 0), Z.baseWidth = k, Z.baseHeight = R, Z.width = k, Z.height = R, Z.isReady = !0, Z.samples = 1, Z.generateMipMaps = n, Z.samplingMode = 1, Z.format = f, Z.type = te, y[o] = Z, this._internalTexturesCache.push(Z); } return m.setTextures(y), t && I.drawBuffers(O), this._bindUnboundFramebuffer(null), m.setLayerAndFaceIndices(j, b), this.resetTextureCache(), m; }; hr.prototype.updateMultipleRenderTargetTextureSampleCount = function(A, e, t = !0) { if (this.webGLVersion < 2 || !A || !A.texture) return 1; if (A.samples === e) return e; const r = A._attachments.length; if (r === 0) return 1; const n = this._gl; e = Math.min(e, this.getCaps().maxMSAASamples); const i = !!A._depthStencilBuffer; if (i && (n.deleteRenderbuffer(A._depthStencilBuffer), A._depthStencilBuffer = null), A._MSAAFramebuffer && (n.deleteFramebuffer(A._MSAAFramebuffer), A._MSAAFramebuffer = null), e > 1 && typeof n.renderbufferStorageMultisample == "function") { const s = n.createFramebuffer(); if (!s) throw new Error("Unable to create multi sampled framebuffer"); A._MSAAFramebuffer = s, this._bindUnboundFramebuffer(s); const a = []; for (let f = 0; f < r; f++) A.textures[f]._hardwareTexture.releaseMSAARenderBuffers(); for (let f = 0; f < r; f++) { const o = A.textures[f], d = o._hardwareTexture, v = n[this.webGLVersion > 1 ? "COLOR_ATTACHMENT" + f : "COLOR_ATTACHMENT" + f + "_WEBGL"], u = this._createRenderBuffer(o.width, o.height, e, -1, this._getRGBABufferInternalSizedFormat(o.type, o.format, o._useSRGBBuffer), v); if (!u) throw new Error("Unable to create multi sampled framebuffer"); d.addMSAARenderBuffer(u), o.samples = e, a.push(v); } t && n.drawBuffers(a); } else this._bindUnboundFramebuffer(A._framebuffer); return i && (A._depthStencilBuffer = this._setupFramebufferDepthAttachments(A._generateStencilBuffer, A._generateDepthBuffer, A.texture.width, A.texture.height, e)), this._bindUnboundFramebuffer(null), e; }; hr.prototype._createDepthStencilCubeTexture = function(A, e, t) { const r = new As(this, ri.DepthStencil); if (r.isCube = !0, this.webGLVersion === 1) return Se.Error("Depth cube texture is not supported by WebGL 1."), r; const n = Object.assign({ bilinearFiltering: !1, comparisonFunction: 0, generateStencil: !1 }, e), i = this._gl; this._bindTextureDirectly(i.TEXTURE_CUBE_MAP, r, !0), this._setupDepthStencilTexture(r, A, n.generateStencil, n.bilinearFiltering, n.comparisonFunction), t._depthStencilTexture = r, t._depthStencilTextureWithStencil = n.generateStencil; for (let s = 0; s < 6; s++) n.generateStencil ? i.texImage2D(i.TEXTURE_CUBE_MAP_POSITIVE_X + s, 0, i.DEPTH24_STENCIL8, A, A, 0, i.DEPTH_STENCIL, i.UNSIGNED_INT_24_8, null) : i.texImage2D(i.TEXTURE_CUBE_MAP_POSITIVE_X + s, 0, i.DEPTH_COMPONENT24, A, A, 0, i.DEPTH_COMPONENT, i.UNSIGNED_INT, null); return this._bindTextureDirectly(i.TEXTURE_CUBE_MAP, null), this._internalTexturesCache.push(r), r; }; hr.prototype._partialLoadFile = function(A, e, t, r, n = null) { const i = (a) => { t[e] = a, t._internalCount++, t._internalCount === 6 && r(t); }, s = (a, f) => { n && a && n(a.status + " " + a.statusText, f); }; this._loadFile(A, i, void 0, void 0, !0, s); }; hr.prototype._cascadeLoadFiles = function(A, e, t, r = null) { const n = []; n._internalCount = 0; for (let i = 0; i < 6; i++) this._partialLoadFile(t[i], i, n, e, r); }; hr.prototype._cascadeLoadImgs = function(A, e, t, r, n = null, i) { const s = []; s._internalCount = 0; for (let a = 0; a < 6; a++) this._partialLoadImg(r[a], a, s, A, e, t, n, i); }; hr.prototype._partialLoadImg = function(A, e, t, r, n, i, s = null, a) { const f = v4(); nU(A, (v) => { t[e] = v, t._internalCount++, r && r.removePendingData(f), t._internalCount === 6 && i && i(n, t); }, (v, u) => { r && r.removePendingData(f), s && s(v, u); }, r ? r.offlineProvider : null, a), r && r.addPendingData(f); }; hr.prototype._setCubeMapTextureParams = function(A, e, t) { const r = this._gl; r.texParameteri(r.TEXTURE_CUBE_MAP, r.TEXTURE_MAG_FILTER, r.LINEAR), r.texParameteri(r.TEXTURE_CUBE_MAP, r.TEXTURE_MIN_FILTER, e ? r.LINEAR_MIPMAP_LINEAR : r.LINEAR), r.texParameteri(r.TEXTURE_CUBE_MAP, r.TEXTURE_WRAP_S, r.CLAMP_TO_EDGE), r.texParameteri(r.TEXTURE_CUBE_MAP, r.TEXTURE_WRAP_T, r.CLAMP_TO_EDGE), A.samplingMode = e ? 3 : 2, e && this.getCaps().textureMaxLevel && t !== void 0 && t > 0 && (r.texParameteri(r.TEXTURE_CUBE_MAP, r.TEXTURE_MAX_LEVEL, t), A._maxLodLevel = t), this._bindTextureDirectly(r.TEXTURE_CUBE_MAP, null); }; hr.prototype.createCubeTextureBase = function(A, e, t, r, n = null, i = null, s, a = null, f = !1, o = 0, d = 0, v = null, u = null, l = null, P = !1) { const p = v || new As(this, ri.Cube); p.isCube = !0, p.url = A, p.generateMipMaps = !r, p._lodGenerationScale = o, p._lodGenerationOffset = d, p._useSRGBBuffer = !!P && this._caps.supportSRGBBuffers && (this.webGLVersion > 1 || this.isWebGPU || !!r), p !== v && (p.label = A.substring(0, 60)), this._doNotHandleContextLost || (p._extension = a, p._files = t); const c = A; this._transformTextureUrl && !v && (A = this._transformTextureUrl(A)); const H = A.split("?")[0], T = H.lastIndexOf("."), q = a || (T > -1 ? H.substring(T).toLowerCase() : ""); let b = null; for (const w of hr._TextureLoaders) if (w.canLoad(q)) { b = w; break; } const j = (w, m) => { A === c ? i && w && i(w.status + " " + w.statusText, m) : (Se.Warn(`Failed to load ${A}, falling back to the ${c}`), this.createCubeTextureBase(c, e, t, !!r, n, i, s, a, f, o, d, p, u, l, P)); }; if (b) { const w = (m) => { u && u(p, m), b.loadCubeData(m, p, f, n, i); }; t && t.length === 6 ? b.supportCascades ? this._cascadeLoadFiles(e, (m) => w(m.map((I) => new Uint8Array(I))), t, i) : i ? i("Textures type does not support cascades.") : Se.Warn("Texture loader does not support cascades.") : this._loadFile(A, (m) => w(new Uint8Array(m)), void 0, void 0, !0, j); } else { if (!t) throw new Error("Cannot load cubemap because files were not defined"); this._cascadeLoadImgs(e, p, (w, m) => { l && l(w, m); }, t, i); } return this._internalTexturesCache.push(p), p; }; hr.prototype.createCubeTexture = function(A, e, t, r, n = null, i = null, s, a = null, f = !1, o = 0, d = 0, v = null, u, l = !1) { const P = this._gl; return this.createCubeTextureBase(A, e, t, !!r, n, i, s, a, f, o, d, v, (p) => this._bindTextureDirectly(P.TEXTURE_CUBE_MAP, p, !0), (p, c) => { const H = this.needPOTTextures ? hr.GetExponentOfTwo(c[0].width, this._caps.maxCubemapTextureSize) : c[0].width, T = H, q = [ P.TEXTURE_CUBE_MAP_POSITIVE_X, P.TEXTURE_CUBE_MAP_POSITIVE_Y, P.TEXTURE_CUBE_MAP_POSITIVE_Z, P.TEXTURE_CUBE_MAP_NEGATIVE_X, P.TEXTURE_CUBE_MAP_NEGATIVE_Y, P.TEXTURE_CUBE_MAP_NEGATIVE_Z ]; this._bindTextureDirectly(P.TEXTURE_CUBE_MAP, p, !0), this._unpackFlipY(!1); const b = s ? this._getInternalFormat(s, p._useSRGBBuffer) : p._useSRGBBuffer ? this._glSRGBExtensionValues.SRGB8_ALPHA8 : P.RGBA; let j = s ? this._getInternalFormat(s) : P.RGBA; p._useSRGBBuffer && this.webGLVersion === 1 && (j = b); for (let w = 0; w < q.length; w++) if (c[w].width !== H || c[w].height !== T) { if (this._prepareWorkingCanvas(), !this._workingCanvas || !this._workingContext) { Se.Warn("Cannot create canvas to resize texture."); return; } this._workingCanvas.width = H, this._workingCanvas.height = T, this._workingContext.drawImage(c[w], 0, 0, c[w].width, c[w].height, 0, 0, H, T), P.texImage2D(q[w], 0, b, j, P.UNSIGNED_BYTE, this._workingCanvas); } else P.texImage2D(q[w], 0, b, j, P.UNSIGNED_BYTE, c[w]); r || P.generateMipmap(P.TEXTURE_CUBE_MAP), this._setCubeMapTextureParams(p, !r), p.width = H, p.height = T, p.isReady = !0, s && (p.format = s), p.onLoadedObservable.notifyObservers(p), p.onLoadedObservable.clear(), n && n(); }, !!l); }; hr.prototype.setTextureSampler = function(A, e) { throw new Error("setTextureSampler: This engine does not support separate texture sampler objects!"); }; class fue { } const dte = new Oe(), vte = new Oe(); Object.defineProperty(Ge.prototype, "onBeforeViewRenderObservable", { get: function() { return dte; } }); Object.defineProperty(Ge.prototype, "onAfterViewRenderObservable", { get: function() { return vte; } }); Object.defineProperty(Ge.prototype, "inputElement", { get: function() { return this._inputElement; }, set: function(A) { var e; this._inputElement !== A && (this._inputElement = A, (e = this._onEngineViewChanged) === null || e === void 0 || e.call(this)); } }); Ge.prototype.getInputElement = function() { return this.inputElement || this.getRenderingCanvas(); }; Ge.prototype.registerView = function(A, e, t) { this.views || (this.views = []); for (const i of this.views) if (i.target === A) return i; const r = this.getRenderingCanvas(); r && (A.width = r.width, A.height = r.height); const n = { target: A, camera: e, clearBeforeCopy: t, enabled: !0, id: (Math.random() * 1e5).toFixed() }; return this.views.push(n), e && !Array.isArray(e) && e.onDisposeObservable.add(() => { this.unRegisterView(A); }), n; }; Ge.prototype.unRegisterView = function(A) { if (!this.views || this.views.length === 0) return this; for (const e of this.views) if (e.target === A) { const t = this.views.indexOf(e); t !== -1 && this.views.splice(t, 1); break; } return this; }; Ge.prototype._renderViewStep = function(A) { const e = A.target, t = e.getContext("2d"); if (!t) return !0; const r = this.getRenderingCanvas(); dte.notifyObservers(A); const n = A.camera; let i = null, s = null, a = null; if (n && (a = Array.isArray(n) ? n[0].getScene() : n.getScene(), i = a.activeCamera, s = a.activeCameras, this.activeView = A, Array.isArray(n) ? a.activeCameras = n : (a.activeCamera = n, a.activeCameras = null)), A.customResize) A.customResize(e); else { const f = Math.floor(e.clientWidth / this._hardwareScalingLevel), o = Math.floor(e.clientHeight / this._hardwareScalingLevel), d = f !== e.width || r.width !== e.width || o !== e.height || r.height !== e.height; e.clientWidth && e.clientHeight && d && (e.width = f, e.height = o, this.setSize(f, o)); } return !r.width || !r.height ? !1 : (this._renderFrame(), this.flushFramebuffer(), A.clearBeforeCopy && t.clearRect(0, 0, r.width, r.height), t.drawImage(r, 0, 0), a && (a.activeCameras = s, a.activeCamera = i), vte.notifyObservers(A), !0); }; Ge.prototype._renderViews = function() { if (!this.views || this.views.length === 0 || !this.getRenderingCanvas()) return !1; let e; for (const t of this.views) { if (!t.enabled) continue; if (t.target === this.inputElement) { e = t; continue; } if (!this._renderViewStep(t)) return !1; } return e && !this._renderViewStep(e) ? !1 : (this.activeView = null, !0); }; hr.prototype.createStorageBuffer = function(A, e) { throw new Error("createStorageBuffer: Unsupported method in this engine!"); }; hr.prototype.updateStorageBuffer = function(A, e, t, r) { }; hr.prototype.readFromStorageBuffer = function(A, e, t, r) { throw new Error("readFromStorageBuffer: Unsupported method in this engine!"); }; hr.prototype.setStorageBuffer = function(A, e) { throw new Error("setStorageBuffer: Unsupported method in this engine!"); }; function Aue(A) { const e = (i) => { const s = "\\b" + i + "\\b"; return A && (A === i || A.match(new RegExp(s, "g"))); }; if (this._excludedCompressedTextures && this._excludedCompressedTextures.some(e)) return A; const t = A.lastIndexOf("."), r = A.lastIndexOf("?"), n = r > -1 ? A.substring(r, A.length) : ""; return (t > -1 ? A.substring(0, t) : A) + this._textureFormatInUse + n; } Object.defineProperty(Ge.prototype, "texturesSupported", { get: function() { const A = []; return this._caps.astc && A.push("-astc.ktx"), this._caps.s3tc && A.push("-dxt.ktx"), this._caps.pvrtc && A.push("-pvrtc.ktx"), this._caps.etc2 && A.push("-etc2.ktx"), this._caps.etc1 && A.push("-etc1.ktx"), A; }, enumerable: !0, configurable: !0 }); Object.defineProperty(Ge.prototype, "textureFormatInUse", { get: function() { return this._textureFormatInUse || null; }, enumerable: !0, configurable: !0 }); Ge.prototype.setCompressedTextureExclusions = function(A) { this._excludedCompressedTextures = A; }; Ge.prototype.setTextureFormatToUse = function(A) { const e = this.texturesSupported; for (let t = 0, r = e.length; t < r; t++) for (let n = 0, i = A.length; n < i; n++) if (e[t] === A[n].toLowerCase()) return this._transformTextureUrl = Aue.bind(this), this._textureFormatInUse = e[t]; return this._textureFormatInUse = "", this._transformTextureUrl = null, null; }; class hD { constructor() { const e = new ArrayBuffer(hD.DEFAULT_BUFFER_SIZE); this._uint32s = new Uint32Array(e), this._int32s = new Int32Array(e), this._float32s = new Float32Array(e), this._length = hD.DEFAULT_BUFFER_SIZE / 4, this._position = 0, this._nativeDataStream = new _native.NativeDataStream(() => { this._flush(); }); } writeUint32(e) { this._flushIfNecessary(1), this._uint32s[this._position++] = e; } writeInt32(e) { this._flushIfNecessary(1), this._int32s[this._position++] = e; } writeFloat32(e) { this._flushIfNecessary(1), this._float32s[this._position++] = e; } writeUint32Array(e) { this._flushIfNecessary(1 + e.length), this._uint32s[this._position++] = e.length, this._uint32s.set(e, this._position), this._position += e.length; } writeInt32Array(e) { this._flushIfNecessary(1 + e.length), this._uint32s[this._position++] = e.length, this._int32s.set(e, this._position), this._position += e.length; } writeFloat32Array(e) { this._flushIfNecessary(1 + e.length), this._uint32s[this._position++] = e.length, this._float32s.set(e, this._position), this._position += e.length; } writeNativeData(e) { this._flushIfNecessary(e.length), this._uint32s.set(e, this._position), this._position += e.length; } writeBoolean(e) { this.writeUint32(e ? 1 : 0); } _flushIfNecessary(e) { this._position + e > this._length && this._flush(); } _flush() { this._nativeDataStream.writeBuffer(this._uint32s.buffer, this._position), this._position = 0; } } hD.DEFAULT_BUFFER_SIZE = 65536; const B2 = [ Math.sqrt(1 / (4 * Math.PI)), -Math.sqrt(3 / (4 * Math.PI)), Math.sqrt(3 / (4 * Math.PI)), -Math.sqrt(3 / (4 * Math.PI)), Math.sqrt(15 / (4 * Math.PI)), -Math.sqrt(15 / (4 * Math.PI)), Math.sqrt(5 / (16 * Math.PI)), -Math.sqrt(15 / (4 * Math.PI)), Math.sqrt(15 / (16 * Math.PI)) // l22 ], due = [ () => 1, (A) => A.y, (A) => A.z, (A) => A.x, (A) => A.x * A.y, (A) => A.y * A.z, (A) => 3 * A.z * A.z - 1, (A) => A.x * A.z, (A) => A.x * A.x - A.y * A.y // l22 ], F8 = (A, e) => B2[A] * due[A](e), N8 = [Math.PI, 2 * Math.PI / 3, 2 * Math.PI / 3, 2 * Math.PI / 3, Math.PI / 4, Math.PI / 4, Math.PI / 4, Math.PI / 4, Math.PI / 4]; class HD { constructor() { this.preScaled = !1, this.l00 = S.Zero(), this.l1_1 = S.Zero(), this.l10 = S.Zero(), this.l11 = S.Zero(), this.l2_2 = S.Zero(), this.l2_1 = S.Zero(), this.l20 = S.Zero(), this.l21 = S.Zero(), this.l22 = S.Zero(); } /** * Adds a light to the spherical harmonics * @param direction the direction of the light * @param color the color of the light * @param deltaSolidAngle the delta solid angle of the light */ addLight(e, t, r) { ue.Vector3[0].set(t.r, t.g, t.b); const n = ue.Vector3[0], i = ue.Vector3[1]; n.scaleToRef(r, i), i.scaleToRef(F8(0, e), ue.Vector3[2]), this.l00.addInPlace(ue.Vector3[2]), i.scaleToRef(F8(1, e), ue.Vector3[2]), this.l1_1.addInPlace(ue.Vector3[2]), i.scaleToRef(F8(2, e), ue.Vector3[2]), this.l10.addInPlace(ue.Vector3[2]), i.scaleToRef(F8(3, e), ue.Vector3[2]), this.l11.addInPlace(ue.Vector3[2]), i.scaleToRef(F8(4, e), ue.Vector3[2]), this.l2_2.addInPlace(ue.Vector3[2]), i.scaleToRef(F8(5, e), ue.Vector3[2]), this.l2_1.addInPlace(ue.Vector3[2]), i.scaleToRef(F8(6, e), ue.Vector3[2]), this.l20.addInPlace(ue.Vector3[2]), i.scaleToRef(F8(7, e), ue.Vector3[2]), this.l21.addInPlace(ue.Vector3[2]), i.scaleToRef(F8(8, e), ue.Vector3[2]), this.l22.addInPlace(ue.Vector3[2]); } /** * Scales the spherical harmonics by the given amount * @param scale the amount to scale */ scaleInPlace(e) { this.l00.scaleInPlace(e), this.l1_1.scaleInPlace(e), this.l10.scaleInPlace(e), this.l11.scaleInPlace(e), this.l2_2.scaleInPlace(e), this.l2_1.scaleInPlace(e), this.l20.scaleInPlace(e), this.l21.scaleInPlace(e), this.l22.scaleInPlace(e); } /** * Convert from incident radiance (Li) to irradiance (E) by applying convolution with the cosine-weighted hemisphere. * * ``` * E_lm = A_l * L_lm * ``` * * In spherical harmonics this convolution amounts to scaling factors for each frequency band. * This corresponds to equation 5 in "An Efficient Representation for Irradiance Environment Maps", where * the scaling factors are given in equation 9. */ convertIncidentRadianceToIrradiance() { this.l00.scaleInPlace(N8[0]), this.l1_1.scaleInPlace(N8[1]), this.l10.scaleInPlace(N8[2]), this.l11.scaleInPlace(N8[3]), this.l2_2.scaleInPlace(N8[4]), this.l2_1.scaleInPlace(N8[5]), this.l20.scaleInPlace(N8[6]), this.l21.scaleInPlace(N8[7]), this.l22.scaleInPlace(N8[8]); } /** * Convert from irradiance to outgoing radiance for Lambertian BDRF, suitable for efficient shader evaluation. * * ``` * L = (1/pi) * E * rho * ``` * * This is done by an additional scale by 1/pi, so is a fairly trivial operation but important conceptually. */ convertIrradianceToLambertianRadiance() { this.scaleInPlace(1 / Math.PI); } /** * Integrates the reconstruction coefficients directly in to the SH preventing further * required operations at run time. * * This is simply done by scaling back the SH with Ylm constants parameter. * The trigonometric part being applied by the shader at run time. */ preScaleForRendering() { this.preScaled = !0, this.l00.scaleInPlace(B2[0]), this.l1_1.scaleInPlace(B2[1]), this.l10.scaleInPlace(B2[2]), this.l11.scaleInPlace(B2[3]), this.l2_2.scaleInPlace(B2[4]), this.l2_1.scaleInPlace(B2[5]), this.l20.scaleInPlace(B2[6]), this.l21.scaleInPlace(B2[7]), this.l22.scaleInPlace(B2[8]); } /** * update the spherical harmonics coefficients from the given array * @param data defines the 9x3 coefficients (l00, l1-1, l10, l11, l2-2, l2-1, l20, l21, l22) * @returns the spherical harmonics (this) */ updateFromArray(e) { return S.FromArrayToRef(e[0], 0, this.l00), S.FromArrayToRef(e[1], 0, this.l1_1), S.FromArrayToRef(e[2], 0, this.l10), S.FromArrayToRef(e[3], 0, this.l11), S.FromArrayToRef(e[4], 0, this.l2_2), S.FromArrayToRef(e[5], 0, this.l2_1), S.FromArrayToRef(e[6], 0, this.l20), S.FromArrayToRef(e[7], 0, this.l21), S.FromArrayToRef(e[8], 0, this.l22), this; } /** * update the spherical harmonics coefficients from the given floats array * @param data defines the 9x3 coefficients (l00, l1-1, l10, l11, l2-2, l2-1, l20, l21, l22) * @returns the spherical harmonics (this) */ updateFromFloatsArray(e) { return S.FromFloatsToRef(e[0], e[1], e[2], this.l00), S.FromFloatsToRef(e[3], e[4], e[5], this.l1_1), S.FromFloatsToRef(e[6], e[7], e[8], this.l10), S.FromFloatsToRef(e[9], e[10], e[11], this.l11), S.FromFloatsToRef(e[12], e[13], e[14], this.l2_2), S.FromFloatsToRef(e[15], e[16], e[17], this.l2_1), S.FromFloatsToRef(e[18], e[19], e[20], this.l20), S.FromFloatsToRef(e[21], e[22], e[23], this.l21), S.FromFloatsToRef(e[24], e[25], e[26], this.l22), this; } /** * Constructs a spherical harmonics from an array. * @param data defines the 9x3 coefficients (l00, l1-1, l10, l11, l2-2, l2-1, l20, l21, l22) * @returns the spherical harmonics */ static FromArray(e) { return new HD().updateFromArray(e); } // Keep for references. /** * Gets the spherical harmonics from polynomial * @param polynomial the spherical polynomial * @returns the spherical harmonics */ static FromPolynomial(e) { const t = new HD(); return t.l00 = e.xx.scale(0.376127).add(e.yy.scale(0.376127)).add(e.zz.scale(0.376126)), t.l1_1 = e.y.scale(0.977204), t.l10 = e.z.scale(0.977204), t.l11 = e.x.scale(0.977204), t.l2_2 = e.xy.scale(1.16538), t.l2_1 = e.yz.scale(1.16538), t.l20 = e.zz.scale(1.34567).subtract(e.xx.scale(0.672834)).subtract(e.yy.scale(0.672834)), t.l21 = e.zx.scale(1.16538), t.l22 = e.xx.scale(1.16538).subtract(e.yy.scale(1.16538)), t.l1_1.scaleInPlace(-1), t.l11.scaleInPlace(-1), t.l2_1.scaleInPlace(-1), t.l21.scaleInPlace(-1), t.scaleInPlace(Math.PI), t; } } class i4 { constructor() { this.x = S.Zero(), this.y = S.Zero(), this.z = S.Zero(), this.xx = S.Zero(), this.yy = S.Zero(), this.zz = S.Zero(), this.xy = S.Zero(), this.yz = S.Zero(), this.zx = S.Zero(); } /** * The spherical harmonics used to create the polynomials. */ get preScaledHarmonics() { return this._harmonics || (this._harmonics = HD.FromPolynomial(this)), this._harmonics.preScaled || this._harmonics.preScaleForRendering(), this._harmonics; } /** * Adds an ambient color to the spherical polynomial * @param color the color to add */ addAmbient(e) { ue.Vector3[0].copyFromFloats(e.r, e.g, e.b); const t = ue.Vector3[0]; this.xx.addInPlace(t), this.yy.addInPlace(t), this.zz.addInPlace(t); } /** * Scales the spherical polynomial by the given amount * @param scale the amount to scale */ scaleInPlace(e) { this.x.scaleInPlace(e), this.y.scaleInPlace(e), this.z.scaleInPlace(e), this.xx.scaleInPlace(e), this.yy.scaleInPlace(e), this.zz.scaleInPlace(e), this.yz.scaleInPlace(e), this.zx.scaleInPlace(e), this.xy.scaleInPlace(e); } /** * Updates the spherical polynomial from harmonics * @param harmonics the spherical harmonics * @returns the spherical polynomial */ updateFromHarmonics(e) { return this._harmonics = e, this.x.copyFrom(e.l11), this.x.scaleInPlace(1.02333).scaleInPlace(-1), this.y.copyFrom(e.l1_1), this.y.scaleInPlace(1.02333).scaleInPlace(-1), this.z.copyFrom(e.l10), this.z.scaleInPlace(1.02333), this.xx.copyFrom(e.l00), ue.Vector3[0].copyFrom(e.l20).scaleInPlace(0.247708), ue.Vector3[1].copyFrom(e.l22).scaleInPlace(0.429043), this.xx.scaleInPlace(0.886277).subtractInPlace(ue.Vector3[0]).addInPlace(ue.Vector3[1]), this.yy.copyFrom(e.l00), this.yy.scaleInPlace(0.886277).subtractInPlace(ue.Vector3[0]).subtractInPlace(ue.Vector3[1]), this.zz.copyFrom(e.l00), ue.Vector3[0].copyFrom(e.l20).scaleInPlace(0.495417), this.zz.scaleInPlace(0.886277).addInPlace(ue.Vector3[0]), this.yz.copyFrom(e.l2_1), this.yz.scaleInPlace(0.858086).scaleInPlace(-1), this.zx.copyFrom(e.l21), this.zx.scaleInPlace(0.858086).scaleInPlace(-1), this.xy.copyFrom(e.l2_2), this.xy.scaleInPlace(0.858086), this.scaleInPlace(1 / Math.PI), this; } /** * Gets the spherical polynomial from harmonics * @param harmonics the spherical harmonics * @returns the spherical polynomial */ static FromHarmonics(e) { return new i4().updateFromHarmonics(e); } /** * Constructs a spherical polynomial from an array. * @param data defines the 9x3 coefficients (x, y, z, xx, yy, zz, yz, zx, xy) * @returns the spherical polynomial */ static FromArray(e) { const t = new i4(); return S.FromArrayToRef(e[0], 0, t.x), S.FromArrayToRef(e[1], 0, t.y), S.FromArrayToRef(e[2], 0, t.z), S.FromArrayToRef(e[3], 0, t.xx), S.FromArrayToRef(e[4], 0, t.yy), S.FromArrayToRef(e[5], 0, t.zz), S.FromArrayToRef(e[6], 0, t.yz), S.FromArrayToRef(e[7], 0, t.zx), S.FromArrayToRef(e[8], 0, t.xy), t; } } const vue = "rgbdDecodePixelShader", uue = `varying vec2 vUV;uniform sampler2D textureSampler; #include #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {gl_FragColor=vec4(fromRGBD(texture2D(textureSampler,vUV)),1.0);}`; Le.ShadersStore[vue] = uue; function ute(A, e, t, r = !0) { const n = A.getScene(), i = n.getEngine(), s = new Ta("resized" + A.name, { width: e, height: t }, n, !A.noMipmap, !0, A._texture.type, !1, A.samplingMode, !1); s.wrapU = A.wrapU, s.wrapV = A.wrapV, s.uOffset = A.uOffset, s.vOffset = A.vOffset, s.uScale = A.uScale, s.vScale = A.vScale, s.uAng = A.uAng, s.vAng = A.vAng, s.wAng = A.wAng, s.coordinatesIndex = A.coordinatesIndex, s.level = A.level, s.anisotropicFilteringLevel = A.anisotropicFilteringLevel, s._texture.isReady = !1, A.wrapU = We.CLAMP_ADDRESSMODE, A.wrapV = We.CLAMP_ADDRESSMODE; const a = new lg("pass", 1, null, r ? We.BILINEAR_SAMPLINGMODE : We.NEAREST_SAMPLINGMODE, i, !1, 0); return a.externalTextureSamplerBinding = !0, a.getEffect().executeWhenCompiled(() => { a.onApply = function(o) { o.setTexture("textureSampler", A); }; const f = s.renderTarget; f && (n.postProcessManager.directRender([a], f), i.unBindFramebuffer(f), s.disposeFramebufferObjects(), a.dispose(), s.getInternalTexture().isReady = !0); }), s; } function ty(A, e, t, r, n, i, s, a) { const f = e.getEngine(); return e.isReady = !1, n = n ?? e.samplingMode, r = r ?? e.type, i = i ?? e.format, s = s ?? e.width, a = a ?? e.height, r === -1 && (r = 0), new Promise((o) => { const d = new kr("postprocess", A, null, null, 1, null, n, f, !1, void 0, r, void 0, null, !1, i); d.externalTextureSamplerBinding = !0; const v = f.createRenderTargetTexture({ width: s, height: a }, { generateDepthBuffer: !1, generateMipMaps: !1, generateStencilBuffer: !1, samplingMode: n, type: r, format: i }); d.getEffect().executeWhenCompiled(() => { d.onApply = (u) => { u._bindTexture("textureSampler", e), u.setFloat2("scale", 1, 1); }, t.postProcessManager.directRender([d], v, !0), f.restoreDefaultFramebuffer(), f._releaseTexture(e), d && d.dispose(), v._swapAndDie(e), e.type = r, e.format = 5, e.isReady = !0, o(e); }); }); } let PC, hG; function LH(A) { PC || (PC = new Float32Array(1), hG = new Int32Array(PC.buffer)), PC[0] = A; const e = hG[0]; let t = e >> 16 & 32768, r = e >> 12 & 2047; const n = e >> 23 & 255; return n < 103 ? t : n > 142 ? (t |= 31744, t |= (n == 255 ? 0 : 1) && e & 8388607, t) : n < 113 ? (r |= 2048, t |= (r >> 114 - n) + (r >> 113 - n & 1), t) : (t |= n - 112 << 10 | r >> 1, t += r & 1, t); } function QH(A) { const e = (A & 32768) >> 15, t = (A & 31744) >> 10, r = A & 1023; return t === 0 ? (e ? -1 : 1) * Math.pow(2, -14) * (r / Math.pow(2, 10)) : t == 31 ? r ? NaN : (e ? -1 : 1) * (1 / 0) : (e ? -1 : 1) * Math.pow(2, t - 15) * (1 + r / Math.pow(2, 10)); } const lue = async (A, e, t, r, n) => { const i = A.getScene(), s = i.getEngine(); let a; if (!A.isCube) a = new kr("lod", "lod", ["lod", "gamma"], null, 1, null, We.NEAREST_NEAREST_MIPNEAREST, s); else { const d = ["#define POSITIVEX", "#define NEGATIVEX", "#define POSITIVEY", "#define NEGATIVEY", "#define POSITIVEZ", "#define NEGATIVEZ"]; a = new kr("lodCube", "lodCube", ["lod", "gamma"], null, 1, null, We.NEAREST_NEAREST_MIPNEAREST, s, !1, d[r]); } await new Promise((d) => { a.getEffect().executeWhenCompiled(() => { d(0); }); }); const f = new Ta("temp", { width: e, height: t }, i, !1); a.onApply = function(d) { d.setTexture("textureSampler", A), d.setFloat("lod", n), d.setBool("gamma", A.gammaSpace); }; const o = A.getInternalTexture(); try { if (f.renderTarget && o) { const d = o.samplingMode; n !== 0 ? A.updateSamplingMode(We.NEAREST_NEAREST_MIPNEAREST) : A.updateSamplingMode(We.NEAREST_NEAREST), i.postProcessManager.directRender([a], f.renderTarget, !0), A.updateSamplingMode(d); const v = await s.readPixels(0, 0, e, t), u = new Uint8Array(v.buffer, 0, v.byteLength); return s.unBindFramebuffer(f.renderTarget), u; } else throw Error("Render to texture failed."); } finally { f.dispose(), a.dispose(); } }; async function lte(A, e, t, r = 0, n = 0) { return !A.isReady() && A._texture && await new Promise((i, s) => { if (A._texture === null) { s(0); return; } A._texture.onLoadedObservable.addOnce(() => { i(0); }); }), await lue(A, e, t, r, n); } const Pte = { /** * Uses the GPU to create a copy texture rescaled at a given size * @param texture Texture to copy from * @param width defines the desired width * @param height defines the desired height * @param useBilinearMode defines if bilinear mode has to be used * @returns the generated texture */ CreateResizedCopy: ute, /** * Apply a post process to a texture * @param postProcessName name of the fragment post process * @param internalTexture the texture to encode * @param scene the scene hosting the texture * @param type type of the output texture. If not provided, use the one from internalTexture * @param samplingMode sampling mode to use to sample the source texture. If not provided, use the one from internalTexture * @param format format of the output texture. If not provided, use the one from internalTexture * @returns a promise with the internalTexture having its texture replaced by the result of the processing */ ApplyPostProcess: ty, /** * Converts a number to half float * @param value number to convert * @returns converted number */ ToHalfFloat: LH, /** * Converts a half float to a number * @param value half float to convert * @returns converted half float */ FromHalfFloat: QH, /** * Gets the data of the specified texture by rendering it to an intermediate RGBA texture and retrieving the bytes from it. * This is convienent to get 8-bit RGBA values for a texture in a GPU compressed format. * @param texture the source texture * @param width the width of the result, which does not have to match the source texture width * @param height the height of the result, which does not have to match the source texture height * @param face if the texture has multiple faces, the face index to use for the source * @param channels a filter for which of the RGBA channels to return in the result * @param lod if the texture has multiple LODs, the lod index to use for the source * @returns the 8-bit texture data */ GetTextureDataAsync: lte }; class $C { /** * Expand the RGBD Texture from RGBD to Half Float if possible. * @param texture the texture to expand. */ static ExpandRGBDTexture(e) { const t = e._texture; if (!t || !e.isRGBD) return; const r = t.getEngine(), n = r.getCaps(), i = t.isReady; let s = !1; n.textureHalfFloatRender && n.textureHalfFloatLinearFiltering ? (s = !0, t.type = 2) : n.textureFloatRender && n.textureFloatLinearFiltering && (s = !0, t.type = 1), s && (t.isReady = !1, t._isRGBD = !1, t.invertY = !1); const a = () => { if (s) { const f = new kr("rgbdDecode", "rgbdDecode", null, null, 1, null, 3, r, !1, void 0, t.type, void 0, null, !1); f.externalTextureSamplerBinding = !0; const o = r.createRenderTargetTexture(t.width, { generateDepthBuffer: !1, generateMipMaps: !1, generateStencilBuffer: !1, samplingMode: t.samplingMode, type: t.type, format: 5 }); f.getEffect().executeWhenCompiled(() => { f.onApply = (d) => { d._bindTexture("textureSampler", t), d.setFloat2("scale", 1, 1); }, e.getScene().postProcessManager.directRender([f], o, !0), r.restoreDefaultFramebuffer(), r._releaseTexture(t), f && f.dispose(), o._swapAndDie(t), t.isReady = !0; }); } }; i ? a() : e.onLoadObservable.addOnce(a); } /** * Encode the texture to RGBD if possible. * @param internalTexture the texture to encode * @param scene the scene hosting the texture * @param outputTextureType type of the texture in which the encoding is performed * @returns a promise with the internalTexture having its texture replaced by the result of the processing */ static EncodeTextureToRGBD(e, t, r = 0) { return ty("rgbdEncode", e, t, r, 1, 5); } } class WW { constructor(e, t, r, n) { this.name = e, this.worldAxisForNormal = t, this.worldAxisForFileX = r, this.worldAxisForFileY = n; } } class Sm { /** * Converts a texture to the according Spherical Polynomial data. * This extracts the first 3 orders only as they are the only one used in the lighting. * * @param texture The texture to extract the information from. * @returns The Spherical Polynomial data. */ static ConvertCubeMapTextureToSphericalPolynomial(e) { var t; if (!e.isCube) return null; (t = e.getScene()) === null || t === void 0 || t.getEngine().flushFramebuffer(); const r = e.getSize().width, n = e.readPixels(0, void 0, void 0, !1), i = e.readPixels(1, void 0, void 0, !1); let s, a; e.isRenderTarget ? (s = e.readPixels(3, void 0, void 0, !1), a = e.readPixels(2, void 0, void 0, !1)) : (s = e.readPixels(2, void 0, void 0, !1), a = e.readPixels(3, void 0, void 0, !1)); const f = e.readPixels(4, void 0, void 0, !1), o = e.readPixels(5, void 0, void 0, !1), d = e.gammaSpace, v = 5; let u = 0; return (e.textureType == 1 || e.textureType == 2) && (u = 1), new Promise((l) => { Promise.all([i, n, s, a, f, o]).then(([P, p, c, H, T, q]) => { const b = { size: r, right: p, left: P, up: c, down: H, front: T, back: q, format: v, type: u, gammaSpace: d }; l(this.ConvertCubeMapToSphericalPolynomial(b)); }); }); } /** * Compute the area on the unit sphere of the rectangle defined by (x,y) and the origin * See https://www.rorydriscoll.com/2012/01/15/cubemap-texel-solid-angle/ * @param x * @param y */ static _AreaElement(e, t) { return Math.atan2(e * t, Math.sqrt(e * e + t * t + 1)); } /** * Converts a cubemap to the according Spherical Polynomial data. * This extracts the first 3 orders only as they are the only one used in the lighting. * * @param cubeInfo The Cube map to extract the information from. * @returns The Spherical Polynomial data. */ static ConvertCubeMapToSphericalPolynomial(e) { const t = new HD(); let r = 0; const n = 2 / e.size, i = n, s = 0.5 * n, a = s - 1; for (let u = 0; u < 6; u++) { const l = this._FileFaces[u], P = e[l.name]; let p = a; const c = e.format === 5 ? 4 : 3; for (let H = 0; H < e.size; H++) { let T = a; for (let q = 0; q < e.size; q++) { const b = l.worldAxisForFileX.scale(T).add(l.worldAxisForFileY.scale(p)).add(l.worldAxisForNormal); b.normalize(); const j = this._AreaElement(T - s, p - s) - this._AreaElement(T - s, p + s) - this._AreaElement(T + s, p - s) + this._AreaElement(T + s, p + s); let w = P[H * e.size * c + q * c + 0], m = P[H * e.size * c + q * c + 1], I = P[H * e.size * c + q * c + 2]; isNaN(w) && (w = 0), isNaN(m) && (m = 0), isNaN(I) && (I = 0), e.type === 0 && (w /= 255, m /= 255, I /= 255), e.gammaSpace && (w = Math.pow(Xt.Clamp(w), xI), m = Math.pow(Xt.Clamp(m), xI), I = Math.pow(Xt.Clamp(I), xI)); const N = this.MAX_HDRI_VALUE; if (this.PRESERVE_CLAMPED_COLORS) { const R = Math.max(w, m, I); if (R > N) { const y = N / R; w *= y, m *= y, I *= y; } } else w = Xt.Clamp(w, 0, N), m = Xt.Clamp(m, 0, N), I = Xt.Clamp(I, 0, N); const k = new Ne(w, m, I); t.addLight(b, k, j), r += j, T += n; } p += i; } } const v = 4 * Math.PI * 6 / 6 / r; return t.scaleInPlace(v), t.convertIncidentRadianceToIrradiance(), t.convertIrradianceToLambertianRadiance(), i4.FromHarmonics(t); } } Sm._FileFaces = [ new WW("right", new S(1, 0, 0), new S(0, 0, -1), new S(0, -1, 0)), new WW("left", new S(-1, 0, 0), new S(0, 0, 1), new S(0, -1, 0)), new WW("up", new S(0, 1, 0), new S(1, 0, 0), new S(0, 0, 1)), new WW("down", new S(0, -1, 0), new S(1, 0, 0), new S(0, 0, -1)), new WW("front", new S(0, 0, 1), new S(1, 0, 0), new S(0, -1, 0)), new WW("back", new S(0, 0, -1), new S(-1, 0, 0), new S(0, -1, 0)) // -Z bottom ]; Sm.MAX_HDRI_VALUE = 4096; Sm.PRESERVE_CLAMPED_COLORS = !1; ls.prototype.forceSphericalPolynomialsRecompute = function() { this._texture && (this._texture._sphericalPolynomial = null, this._texture._sphericalPolynomialPromise = null, this._texture._sphericalPolynomialComputed = !1); }; Object.defineProperty(ls.prototype, "sphericalPolynomial", { get: function() { if (this._texture) { if (this._texture._sphericalPolynomial || this._texture._sphericalPolynomialComputed) return this._texture._sphericalPolynomial; if (this._texture.isReady) return this._texture._sphericalPolynomialPromise || (this._texture._sphericalPolynomialPromise = Sm.ConvertCubeMapTextureToSphericalPolynomial(this), this._texture._sphericalPolynomialPromise === null ? this._texture._sphericalPolynomialComputed = !0 : this._texture._sphericalPolynomialPromise.then((A) => { this._texture._sphericalPolynomial = A, this._texture._sphericalPolynomialComputed = !0; })), null; } return null; }, set: function(A) { this._texture && (this._texture._sphericalPolynomial = A); }, enumerable: !0, configurable: !0 }); const Pue = "rgbdEncodePixelShader", cue = `varying vec2 vUV;uniform sampler2D textureSampler; #include #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {gl_FragColor=toRGBD(texture2D(textureSampler,vUV).rgb);}`; Le.ShadersStore[Pue] = cue; const kQ = "image/png", nF = 2, WI = [134, 22, 135, 150, 246, 214, 150, 54]; function ry(A) { const e = new DataView(A.buffer, A.byteOffset, A.byteLength); let t = 0; for (let s = 0; s < WI.length; s++) if (e.getUint8(t++) !== WI[s]) return Se.Error("Not a babylon environment map"), null; let r = "", n = 0; for (; n = e.getUint8(t++); ) r += String.fromCharCode(n); let i = JSON.parse(r); return i = eV(i), i.specular && (i.specular.specularDataPosition = t, i.specular.lodGenerationScale = i.specular.lodGenerationScale || 0.8), i; } function eV(A) { if (A.version > nF) throw new Error(`Unsupported babylon environment map version "${A.version}". Latest supported version is "${nF}".`); return A.version === 2 || (A = Object.assign(Object.assign({}, A), { version: 2, imageType: kQ })), A; } async function cte(A, e = {}) { var t, r; const n = A.getInternalTexture(); if (!n) return Promise.reject("The cube texture is invalid."); const i = (t = e.imageType) !== null && t !== void 0 ? t : kQ, s = n.getEngine(); if (A.textureType !== 2 && A.textureType !== 1 && A.textureType !== 0 && A.textureType !== 0 && A.textureType !== 7 && A.textureType !== -1) return Promise.reject("The cube texture should allow HDR (Full Float or Half Float)."); let a = 1; if (!s.getCaps().textureFloatRender && (a = 2, !s.getCaps().textureHalfFloatRender)) return Promise.reject("Env texture can only be created when the browser supports half float or full float rendering."); A.sphericalPolynomial; const f = (r = A.getInternalTexture()) === null || r === void 0 ? void 0 : r._sphericalPolynomialPromise, o = n.width, d = new sr(s), v = {}; s.flushFramebuffer(); const u = Xt.ILog2(n.width); for (let m = 0; m <= u; m++) { const I = Math.pow(2, u - m); for (let N = 0; N < 6; N++) { let k = await A.readPixels(N, m, void 0, !1); if (k && k.byteLength === k.length) { const Y = new Float32Array(k.byteLength * 4); for (let ee = 0; ee < k.byteLength; ee++) Y[ee] = k[ee] / 255, Y[ee] = Math.pow(Y[ee], 2.2); k = Y; } else if (k && A.gammaSpace) { const Y = k; for (let ee = 0; ee < Y.length; ee++) Y[ee] = Math.pow(Y[ee], 2.2); } const R = s.createRawTexture(k, I, I, 5, !1, !0, 1, null, a); await $C.EncodeTextureToRGBD(R, d, a); const y = await s._readTexturePixels(R, I, I), O = await eA.DumpDataAsync(I, I, y, i, void 0, !1, !0, e.imageQuality); v[m * 6 + N] = O, R.dispose(); } } d.dispose(), f && await f; const l = { version: nF, width: o, imageType: i, irradiance: pue(A), specular: { mipmaps: [], lodGenerationScale: A.lodGenerationScale } }; let P = 0; for (let m = 0; m <= u; m++) for (let I = 0; I < 6; I++) { const N = v[m * 6 + I].byteLength; l.specular.mipmaps.push({ length: N, position: P }), P += N; } const p = JSON.stringify(l), c = new ArrayBuffer(p.length + 1), H = new Uint8Array(c); for (let m = 0, I = p.length; m < I; m++) H[m] = p.charCodeAt(m); H[p.length] = 0; const T = WI.length + P + c.byteLength, q = new ArrayBuffer(T), b = new Uint8Array(q), j = new DataView(q); let w = 0; for (let m = 0; m < WI.length; m++) j.setUint8(w++, WI[m]); b.set(new Uint8Array(c), w), w += c.byteLength; for (let m = 0; m <= u; m++) for (let I = 0; I < 6; I++) { const N = v[m * 6 + I]; b.set(new Uint8Array(N), w), w += N.byteLength; } return q; } function pue(A) { const e = A.sphericalPolynomial; return e == null ? null : { x: [e.x.x, e.x.y, e.x.z], y: [e.y.x, e.y.y, e.y.z], z: [e.z.x, e.z.y, e.z.z], xx: [e.xx.x, e.xx.y, e.xx.z], yy: [e.yy.x, e.yy.y, e.yy.z], zz: [e.zz.x, e.zz.y, e.zz.z], yz: [e.yz.x, e.yz.y, e.yz.z], zx: [e.zx.x, e.zx.y, e.zx.z], xy: [e.xy.x, e.xy.y, e.xy.z] }; } function ny(A, e) { e = eV(e); const t = e.specular; let r = Xt.Log2(e.width); if (r = Math.round(r) + 1, t.mipmaps.length !== 6 * r) throw new Error(`Unsupported specular mipmaps number "${t.mipmaps.length}"`); const n = new Array(r); for (let i = 0; i < r; i++) { n[i] = new Array(6); for (let s = 0; s < 6; s++) { const a = t.mipmaps[i * 6 + s]; n[i][s] = new Uint8Array(A.buffer, A.byteOffset + t.specularDataPosition + a.position, a.length); } } return n; } function EQ(A, e, t) { t = eV(t); const r = t.specular; if (!r) return Promise.resolve(); A._lodGenerationScale = r.lodGenerationScale; const n = ny(e, t); return YI(A, n, t.imageType); } function HG(A, e, t, r, n, i, s, a, f, o, d) { return new Promise((v, u) => { if (t) { const l = e.createTexture(null, !0, !0, null, 1, null, (P) => { u(P); }, A); r.getEffect().executeWhenCompiled(() => { r.externalTextureSamplerBinding = !0, r.onApply = (P) => { P._bindTexture("textureSampler", l), P.setFloat2("scale", 1, e._features.needsInvertingBitmap && A instanceof ImageBitmap ? -1 : 1); }, e.scenes.length && (e.scenes[0].postProcessManager.directRender([r], o, !0, i, s), e.restoreDefaultFramebuffer(), l.dispose(), URL.revokeObjectURL(n), v()); }); } else { if (e._uploadImageToTexture(d, A, i, s), a) { const l = f[s]; l && e._uploadImageToTexture(l._texture, A, i, 0); } v(); } }); } function YI(A, e, t = kQ) { if (!ye.IsExponentOfTwo(A.width)) throw new Error("Texture size must be a power of two"); const r = Xt.ILog2(A.width) + 1, n = A.getEngine(); let i = !1, s = !1, a = null, f = null, o = null; const d = n.getCaps(); if (A.format = 5, A.type = 0, A.generateMipMaps = !0, A._cachedAnisotropicFilteringLevel = null, n.updateTextureSamplingMode(3, A), d.textureLOD ? n._features.supportRenderAndCopyToLodForFloatTextures ? d.textureHalfFloatRender && d.textureHalfFloatLinearFiltering ? (i = !0, A.type = 2) : d.textureFloatRender && d.textureFloatLinearFiltering && (i = !0, A.type = 1) : i = !1 : (i = !1, s = !0, o = {}), i) a = new kr("rgbdDecode", "rgbdDecode", null, null, 1, null, 3, n, !1, void 0, A.type, void 0, null, !1), A._isRGBD = !1, A.invertY = !1, f = n.createRenderTargetCubeTexture(A.width, { generateDepthBuffer: !1, generateMipMaps: !0, generateStencilBuffer: !1, samplingMode: 3, type: A.type, format: 5 }); else if (A._isRGBD = !0, A.invertY = !0, s) { const l = A._lodGenerationScale, P = A._lodGenerationOffset; for (let p = 0; p < 3; p++) { const H = 1 - p / 2, T = P, q = (r - 1) * l + P, b = T + (q - T) * H, j = Math.round(Math.min(Math.max(b, 0), q)), w = new As(n, ri.Temp); w.isCube = !0, w.invertY = !0, w.generateMipMaps = !1, n.updateTextureSamplingMode(2, w); const m = new ls(null); switch (m._isCube = !0, m._texture = w, o[j] = m, p) { case 0: A._lodTextureLow = m; break; case 1: A._lodTextureMid = m; break; case 2: A._lodTextureHigh = m; break; } } } const v = []; for (let u = 0; u < e.length; u++) for (let l = 0; l < 6; l++) { const P = e[u][l], p = new Blob([P], { type: t }), c = URL.createObjectURL(p); let H; if (typeof Image > "u" || n._features.forceBitmapOverHTMLImageElement) H = n.createImageBitmap(p, { premultiplyAlpha: "none" }).then((T) => HG(T, n, i, a, c, l, u, s, o, f, A)); else { const T = new Image(); T.src = c, H = new Promise((q, b) => { T.onload = () => { HG(T, n, i, a, c, l, u, s, o, f, A).then(() => q()).catch((j) => { b(j); }); }, T.onerror = (j) => { b(j); }; }); } v.push(H); } if (e.length < r) { let u; const l = Math.pow(2, r - 1 - e.length), P = l * l * 4; switch (A.type) { case 0: { u = new Uint8Array(P); break; } case 2: { u = new Uint16Array(P); break; } case 1: { u = new Float32Array(P); break; } } for (let p = e.length; p < r; p++) for (let c = 0; c < 6; c++) n._uploadArrayBufferViewToTexture(A, u, c, p); } return Promise.all(v).then(() => { f && (n._releaseTexture(A), f._swapAndDie(A)), a && a.dispose(), s && (A._lodTextureHigh && A._lodTextureHigh._texture && (A._lodTextureHigh._texture.isReady = !0), A._lodTextureMid && A._lodTextureMid._texture && (A._lodTextureMid._texture.isReady = !0), A._lodTextureLow && A._lodTextureLow._texture && (A._lodTextureLow._texture.isReady = !0)); }); } function iy(A, e) { e = eV(e); const t = e.irradiance; if (!t) return; const r = new i4(); S.FromArrayToRef(t.x, 0, r.x), S.FromArrayToRef(t.y, 0, r.y), S.FromArrayToRef(t.z, 0, r.z), S.FromArrayToRef(t.xx, 0, r.xx), S.FromArrayToRef(t.yy, 0, r.yy), S.FromArrayToRef(t.zz, 0, r.zz), S.FromArrayToRef(t.yz, 0, r.yz), S.FromArrayToRef(t.zx, 0, r.zx), S.FromArrayToRef(t.xy, 0, r.xy), A._sphericalPolynomial = r; } function pte(A, e, t, r, n) { const i = A.getEngine().createRawCubeTexture(null, A.width, A.format, A.type, A.generateMipMaps, A.invertY, A.samplingMode, A._compression), s = YI(i, e).then(() => A); return A.onRebuildCallback = (a) => ({ proxy: s, isReady: !0, isAsync: !0 }), A._source = ri.CubeRawRGBD, A._bufferViewArrayArray = e, A._lodGenerationScale = r, A._lodGenerationOffset = n, A._sphericalPolynomial = t, YI(A, e).then(() => (A.isReady = !0, A)); } const hue = { /** * Gets the environment info from an env file. * @param data The array buffer containing the .env bytes. * @returns the environment file info (the json header) if successfully parsed, normalized to the latest supported version. */ GetEnvInfo: ry, /** * Creates an environment texture from a loaded cube texture. * @param texture defines the cube texture to convert in env file * @param options options for the conversion process * @param options.imageType the mime type for the encoded images, with support for "image/png" (default) and "image/webp" * @param options.imageQuality the image quality of encoded WebP images. * @returns a promise containing the environment data if successful. */ CreateEnvTextureAsync: cte, /** * Creates the ArrayBufferViews used for initializing environment texture image data. * @param data the image data * @param info parameters that determine what views will be created for accessing the underlying buffer * @returns the views described by info providing access to the underlying buffer */ CreateImageDataArrayBufferViews: ny, /** * Uploads the texture info contained in the env file to the GPU. * @param texture defines the internal texture to upload to * @param data defines the data to load * @param info defines the texture info retrieved through the GetEnvInfo method * @returns a promise */ UploadEnvLevelsAsync: EQ, /** * Uploads the levels of image data to the GPU. * @param texture defines the internal texture to upload to * @param imageData defines the array buffer views of image data [mipmap][face] * @param imageType the mime type of the image data * @returns a promise */ UploadLevelsAsync: YI, /** * Uploads spherical polynomials information to the texture. * @param texture defines the texture we are trying to upload the information to * @param info defines the environment texture info retrieved through the GetEnvInfo method */ UploadEnvSpherical: iy }; function cC(A, e, t, r) { let n = r, i = 0, s = ""; for (; n < t.length; ) { const a = t.charAt(n); if (s) a === s ? s === '"' || s === "'" ? t.charAt(n - 1) !== "\\" && (s = "") : s = "" : s === "*/" && a === "*" && n + 1 < t.length && (t.charAt(n + 1) === "/" && (s = ""), s === "" && n++); else switch (a) { case A: i++; break; case e: i--; break; case '"': case "'": case "`": s = a; break; case "/": if (n + 1 < t.length) { const f = t.charAt(n + 1); f === "/" ? s = ` ` : f === "*" && (s = "*/"); } break; } if (n++, i === 0) break; } return i === 0 ? n - 1 : -1; } function gG(A, e) { for (; e < A.length; ) { const t = A[e]; if (t !== " " && t !== ` ` && t !== "\r" && t !== " " && t !== ` ` && t !== " ") break; e++; } return e; } function xE(A) { const e = A.charCodeAt(0); return e >= 48 && e <= 57 || // 0-9 e >= 65 && e <= 90 || // A-Z e >= 97 && e <= 122 || // a-z e == 95; } function iF(A) { let e = 0, t = "", r = !1; const n = []; for (; e < A.length; ) { const i = A.charAt(e); if (t) i === t ? t === '"' || t === "'" ? (A.charAt(e - 1) !== "\\" && (t = ""), n.push(i)) : (t = "", r = !1) : t === "*/" && i === "*" && e + 1 < A.length ? (A.charAt(e + 1) === "/" && (t = ""), t === "" && (r = !1, e++)) : r || n.push(i); else { switch (i) { case '"': case "'": case "`": t = i; break; case "/": if (e + 1 < A.length) { const s = A.charAt(e + 1); s === "/" ? (t = ` `, r = !0) : s === "*" && (t = "*/", r = !0); } break; } r || n.push(i); } e++; } return n.join(""); } function Hue(A, e, t, r) { for (; e >= 0 && A.charAt(e) !== t && (!r || A.charAt(e) !== r); ) e--; return e; } function gue(A) { return A.replace(/[.*+?^${}()|[\]\\]/g, "\\$&"); } class Aq { /** Gets the code after the inlining process */ get code() { return this._sourceCode; } /** * Initializes the inliner * @param sourceCode shader code source to inline * @param numMaxIterations maximum number of iterations (used to detect recursive calls) */ constructor(e, t = 20) { this.debug = !1, this._sourceCode = e, this._numMaxIterations = t, this._functionDescr = [], this.inlineToken = "#define inline"; } /** * Start the processing of the shader code */ processCode() { this.debug && console.log(`Start inlining process (code size=${this._sourceCode.length})...`), this._collectFunctions(), this._processInlining(this._numMaxIterations), this.debug && console.log("End of inlining process."); } _collectFunctions() { let e = 0; for (; e < this._sourceCode.length; ) { const t = this._sourceCode.indexOf(this.inlineToken, e); if (t < 0) break; const r = this._sourceCode.indexOf("(", t + this.inlineToken.length); if (r < 0) { this.debug && console.warn(`Could not find the opening parenthesis after the token. startIndex=${e}`), e = t + this.inlineToken.length; continue; } const n = Aq._RegexpFindFunctionNameAndType.exec(this._sourceCode.substring(t + this.inlineToken.length, r)); if (!n) { this.debug && console.warn(`Could not extract the name/type of the function from: ${this._sourceCode.substring(t + this.inlineToken.length, r)}`), e = t + this.inlineToken.length; continue; } const [i, s] = [n[3], n[4]], a = cC("(", ")", this._sourceCode, r); if (a < 0) { this.debug && console.warn(`Could not extract the parameters the function '${s}' (type=${i}). funcParamsStartIndex=${r}`), e = t + this.inlineToken.length; continue; } const f = this._sourceCode.substring(r + 1, a), o = gG(this._sourceCode, a + 1); if (o === this._sourceCode.length) { this.debug && console.warn(`Could not extract the body of the function '${s}' (type=${i}). funcParamsEndIndex=${a}`), e = t + this.inlineToken.length; continue; } const d = cC("{", "}", this._sourceCode, o); if (d < 0) { this.debug && console.warn(`Could not extract the body of the function '${s}' (type=${i}). funcBodyStartIndex=${o}`), e = t + this.inlineToken.length; continue; } const v = this._sourceCode.substring(o, d + 1), u = iF(f).split(","), l = []; for (let c = 0; c < u.length; ++c) { const H = u[c].trim(), T = H.lastIndexOf(" "); T >= 0 && l.push(H.substring(T + 1)); } i !== "void" && l.push("return"), this._functionDescr.push({ name: s, type: i, parameters: l, body: v, callIndex: 0 }), e = d + 1; const P = t > 0 ? this._sourceCode.substring(0, t) : "", p = d + 1 < this._sourceCode.length - 1 ? this._sourceCode.substring(d + 1) : ""; this._sourceCode = P + p, e -= d + 1 - t; } this.debug && console.log(`Collect functions: ${this._functionDescr.length} functions found. functionDescr=`, this._functionDescr); } _processInlining(e = 20) { for (; e-- >= 0 && this._replaceFunctionCallsByCode(); ) ; return this.debug && console.log(`numMaxIterations is ${e} after inlining process`), e >= 0; } _replaceFunctionCallsByCode() { let e = !1; for (const t of this._functionDescr) { const { name: r, type: n, parameters: i, body: s } = t; let a = 0; for (; a < this._sourceCode.length; ) { const f = this._sourceCode.indexOf(r, a); if (f < 0) break; if (f === 0 || xE(this._sourceCode.charAt(f - 1))) { a = f + r.length; continue; } const o = gG(this._sourceCode, f + r.length); if (o === this._sourceCode.length || this._sourceCode.charAt(o) !== "(") { a = f + r.length; continue; } const d = cC("(", ")", this._sourceCode, o); if (d < 0) { this.debug && console.warn(`Could not extract the parameters of the function call. Function '${r}' (type=${n}). callParamsStartIndex=${o}`), a = f + r.length; continue; } const v = this._sourceCode.substring(o + 1, d), l = ((q) => { const b = []; let j = 0, w = 0; for (; j < q.length; ) { if (q.charAt(j) === "(") { const m = cC("(", ")", q, j); if (m < 0) return null; j = m; } else q.charAt(j) === "," && (b.push(q.substring(w, j)), w = j + 1); j++; } return w < j && b.push(q.substring(w, j)), b; })(iF(v)); if (l === null) { this.debug && console.warn(`Invalid function call: can't extract the parameters of the function call. Function '${r}' (type=${n}). callParamsStartIndex=${o}, callParams=` + v), a = f + r.length; continue; } const P = []; for (let q = 0; q < l.length; ++q) { const b = l[q].trim(); P.push(b); } const p = n !== "void" ? r + "_" + t.callIndex++ : null; if (p && P.push(p + " ="), P.length !== i.length) { this.debug && console.warn(`Invalid function call: not the same number of parameters for the call than the number expected by the function. Function '${r}' (type=${n}). function parameters=${i}, call parameters=${P}`), a = f + r.length; continue; } a = d + 1; const c = this._replaceNames(s, i, P); let H = f > 0 ? this._sourceCode.substring(0, f) : ""; const T = d + 1 < this._sourceCode.length - 1 ? this._sourceCode.substring(d + 1) : ""; if (p) { const q = Hue(this._sourceCode, f - 1, ` `, "{"); H = this._sourceCode.substring(0, q + 1); const b = this._sourceCode.substring(q + 1, f); this._sourceCode = H + n + " " + p + `; ` + c + ` ` + b + p + T, this.debug && console.log(`Replace function call by code. Function '${r}' (type=${n}). injectDeclarationIndex=${q}, call parameters=${P}`); } else this._sourceCode = H + c + T, a += c.length - (d + 1 - f), this.debug && console.log(`Replace function call by code. Function '${r}' (type=${n}). functionCallIndex=${f}, call parameters=${P}`); e = !0; } } return e; } _replaceNames(e, t, r) { for (let n = 0; n < t.length; ++n) { const i = new RegExp(gue(t[n]), "g"), s = t[n].length, a = r[n]; e = e.replace(i, (f, ...o) => { const d = o[0]; return xE(e.charAt(d - 1)) || xE(e.charAt(d + s)) ? t[n] : a; }); } return e; } } Aq._RegexpFindFunctionNameAndType = /((\s+?)(\w+)\s+(\w+)\s*?)$/; class Xue { get isAsync() { return this.isParallelCompiled; } get isReady() { if (this.compilationError) { const e = this.compilationError.message; throw new Error("SHADER ERROR" + (typeof e == "string" ? ` ` + e : "")); } return this.isCompiled; } _getVertexShaderCode() { return null; } _getFragmentShaderCode() { return null; } // TODO: what should this do? _handlesSpectorRebuildCallback(e) { throw new Error("Not implemented"); } constructor(e) { this.isParallelCompiled = !0, this.isCompiled = !1, this._valueCache = {}, this._engine = e; } _fillEffectInformation(e, t, r, n, i, s, a, f) { const o = this._engine; if (o.supportsUniformBuffers) for (const u in t) e.bindUniformBlock(u, t[u]); this._engine.getUniforms(this, r).forEach((u, l) => { n[r[l]] = u; }), this._uniforms = n; let v; for (v = 0; v < i.length; v++) e.getUniform(i[v]) == null && (i.splice(v, 1), v--); i.forEach((u, l) => { s[u] = l; }), f.push(...o.getAttributes(this, a)); } /** * Release all associated resources. **/ dispose() { this._uniforms = {}; } /** * @internal */ _cacheMatrix(e, t) { const r = this._valueCache[e], n = t.updateFlag; return r !== void 0 && r === n ? !1 : (this._valueCache[e] = n, !0); } /** * @internal */ _cacheFloat2(e, t, r) { let n = this._valueCache[e]; if (!n) return n = [t, r], this._valueCache[e] = n, !0; let i = !1; return n[0] !== t && (n[0] = t, i = !0), n[1] !== r && (n[1] = r, i = !0), i; } /** * @internal */ _cacheFloat3(e, t, r, n) { let i = this._valueCache[e]; if (!i) return i = [t, r, n], this._valueCache[e] = i, !0; let s = !1; return i[0] !== t && (i[0] = t, s = !0), i[1] !== r && (i[1] = r, s = !0), i[2] !== n && (i[2] = n, s = !0), s; } /** * @internal */ _cacheFloat4(e, t, r, n, i) { let s = this._valueCache[e]; if (!s) return s = [t, r, n, i], this._valueCache[e] = s, !0; let a = !1; return s[0] !== t && (s[0] = t, a = !0), s[1] !== r && (s[1] = r, a = !0), s[2] !== n && (s[2] = n, a = !0), s[3] !== i && (s[3] = i, a = !0), a; } /** * Sets an integer value on a uniform variable. * @param uniformName Name of the variable. * @param value Value to be set. */ setInt(e, t) { const r = this._valueCache[e]; r !== void 0 && r === t || this._engine.setInt(this._uniforms[e], t) && (this._valueCache[e] = t); } /** * Sets a int2 on a uniform variable. * @param uniformName Name of the variable. * @param x First int in int2. * @param y Second int in int2. */ setInt2(e, t, r) { this._cacheFloat2(e, t, r) && (this._engine.setInt2(this._uniforms[e], t, r) || (this._valueCache[e] = null)); } /** * Sets a int3 on a uniform variable. * @param uniformName Name of the variable. * @param x First int in int3. * @param y Second int in int3. * @param z Third int in int3. */ setInt3(e, t, r, n) { this._cacheFloat3(e, t, r, n) && (this._engine.setInt3(this._uniforms[e], t, r, n) || (this._valueCache[e] = null)); } /** * Sets a int4 on a uniform variable. * @param uniformName Name of the variable. * @param x First int in int4. * @param y Second int in int4. * @param z Third int in int4. * @param w Fourth int in int4. */ setInt4(e, t, r, n, i) { this._cacheFloat4(e, t, r, n, i) && (this._engine.setInt4(this._uniforms[e], t, r, n, i) || (this._valueCache[e] = null)); } /** * Sets an int array on a uniform variable. * @param uniformName Name of the variable. * @param array array to be set. */ setIntArray(e, t) { this._valueCache[e] = null, this._engine.setIntArray(this._uniforms[e], t); } /** * Sets an int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setIntArray2(e, t) { this._valueCache[e] = null, this._engine.setIntArray2(this._uniforms[e], t); } /** * Sets an int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setIntArray3(e, t) { this._valueCache[e] = null, this._engine.setIntArray3(this._uniforms[e], t); } /** * Sets an int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setIntArray4(e, t) { this._valueCache[e] = null, this._engine.setIntArray4(this._uniforms[e], t); } /** * Sets an unsigned integer value on a uniform variable. * @param uniformName Name of the variable. * @param value Value to be set. */ setUInt(e, t) { const r = this._valueCache[e]; r !== void 0 && r === t || this._engine.setUInt(this._uniforms[e], t) && (this._valueCache[e] = t); } /** * Sets a unsigned int2 on a uniform variable. * @param uniformName Name of the variable. * @param x First unsigned int in uint2. * @param y Second unsigned int in uint2. */ setUInt2(e, t, r) { this._cacheFloat2(e, t, r) && (this._engine.setUInt2(this._uniforms[e], t, r) || (this._valueCache[e] = null)); } /** * Sets a unsigned int3 on a uniform variable. * @param uniformName Name of the variable. * @param x First unsigned int in uint3. * @param y Second unsigned int in uint3. * @param z Third unsigned int in uint3. */ setUInt3(e, t, r, n) { this._cacheFloat3(e, t, r, n) && (this._engine.setUInt3(this._uniforms[e], t, r, n) || (this._valueCache[e] = null)); } /** * Sets a unsigned int4 on a uniform variable. * @param uniformName Name of the variable. * @param x First unsigned int in uint4. * @param y Second unsigned int in uint4. * @param z Third unsigned int in uint4. * @param w Fourth unsigned int in uint4. */ setUInt4(e, t, r, n, i) { this._cacheFloat4(e, t, r, n, i) && (this._engine.setUInt4(this._uniforms[e], t, r, n, i) || (this._valueCache[e] = null)); } /** * Sets an unsigned int array on a uniform variable. * @param uniformName Name of the variable. * @param array array to be set. */ setUIntArray(e, t) { this._valueCache[e] = null, this._engine.setUIntArray(this._uniforms[e], t); } /** * Sets an unsigned int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setUIntArray2(e, t) { this._valueCache[e] = null, this._engine.setUIntArray2(this._uniforms[e], t); } /** * Sets an unsigned int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setUIntArray3(e, t) { this._valueCache[e] = null, this._engine.setUIntArray3(this._uniforms[e], t); } /** * Sets an unsigned int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setUIntArray4(e, t) { this._valueCache[e] = null, this._engine.setUIntArray4(this._uniforms[e], t); } /** * Sets an float array on a uniform variable. * @param uniformName Name of the variable. * @param array array to be set. */ setFloatArray(e, t) { this._valueCache[e] = null, this._engine.setFloatArray(this._uniforms[e], t); } /** * Sets an float array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setFloatArray2(e, t) { this._valueCache[e] = null, this._engine.setFloatArray2(this._uniforms[e], t); } /** * Sets an float array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setFloatArray3(e, t) { this._valueCache[e] = null, this._engine.setFloatArray3(this._uniforms[e], t); } /** * Sets an float array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setFloatArray4(e, t) { this._valueCache[e] = null, this._engine.setFloatArray4(this._uniforms[e], t); } /** * Sets an array on a uniform variable. * @param uniformName Name of the variable. * @param array array to be set. */ setArray(e, t) { this._valueCache[e] = null, this._engine.setArray(this._uniforms[e], t); } /** * Sets an array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setArray2(e, t) { this._valueCache[e] = null, this._engine.setArray2(this._uniforms[e], t); } /** * Sets an array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setArray3(e, t) { this._valueCache[e] = null, this._engine.setArray3(this._uniforms[e], t); } /** * Sets an array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setArray4(e, t) { this._valueCache[e] = null, this._engine.setArray4(this._uniforms[e], t); } /** * Sets matrices on a uniform variable. * @param uniformName Name of the variable. * @param matrices matrices to be set. */ setMatrices(e, t) { t && (this._valueCache[e] = null, this._engine.setMatrices(this._uniforms[e], t)); } /** * Sets matrix on a uniform variable. * @param uniformName Name of the variable. * @param matrix matrix to be set. */ setMatrix(e, t) { this._cacheMatrix(e, t) && (this._engine.setMatrices(this._uniforms[e], t.toArray()) || (this._valueCache[e] = null)); } /** * Sets a 3x3 matrix on a uniform variable. (Specified as [1,2,3,4,5,6,7,8,9] will result in [1,2,3][4,5,6][7,8,9] matrix) * @param uniformName Name of the variable. * @param matrix matrix to be set. */ setMatrix3x3(e, t) { this._valueCache[e] = null, this._engine.setMatrix3x3(this._uniforms[e], t); } /** * Sets a 2x2 matrix on a uniform variable. (Specified as [1,2,3,4] will result in [1,2][3,4] matrix) * @param uniformName Name of the variable. * @param matrix matrix to be set. */ setMatrix2x2(e, t) { this._valueCache[e] = null, this._engine.setMatrix2x2(this._uniforms[e], t); } /** * Sets a float on a uniform variable. * @param uniformName Name of the variable. * @param value value to be set. * @returns this effect. */ setFloat(e, t) { const r = this._valueCache[e]; r !== void 0 && r === t || this._engine.setFloat(this._uniforms[e], t) && (this._valueCache[e] = t); } /** * Sets a boolean on a uniform variable. * @param uniformName Name of the variable. * @param bool value to be set. */ setBool(e, t) { const r = this._valueCache[e]; r !== void 0 && r === t || this._engine.setInt(this._uniforms[e], t ? 1 : 0) && (this._valueCache[e] = t ? 1 : 0); } /** * Sets a Vector2 on a uniform variable. * @param uniformName Name of the variable. * @param vector2 vector2 to be set. */ setVector2(e, t) { this._cacheFloat2(e, t.x, t.y) && (this._engine.setFloat2(this._uniforms[e], t.x, t.y) || (this._valueCache[e] = null)); } /** * Sets a float2 on a uniform variable. * @param uniformName Name of the variable. * @param x First float in float2. * @param y Second float in float2. */ setFloat2(e, t, r) { this._cacheFloat2(e, t, r) && (this._engine.setFloat2(this._uniforms[e], t, r) || (this._valueCache[e] = null)); } /** * Sets a Vector3 on a uniform variable. * @param uniformName Name of the variable. * @param vector3 Value to be set. */ setVector3(e, t) { this._cacheFloat3(e, t.x, t.y, t.z) && (this._engine.setFloat3(this._uniforms[e], t.x, t.y, t.z) || (this._valueCache[e] = null)); } /** * Sets a float3 on a uniform variable. * @param uniformName Name of the variable. * @param x First float in float3. * @param y Second float in float3. * @param z Third float in float3. */ setFloat3(e, t, r, n) { this._cacheFloat3(e, t, r, n) && (this._engine.setFloat3(this._uniforms[e], t, r, n) || (this._valueCache[e] = null)); } /** * Sets a Vector4 on a uniform variable. * @param uniformName Name of the variable. * @param vector4 Value to be set. */ setVector4(e, t) { this._cacheFloat4(e, t.x, t.y, t.z, t.w) && (this._engine.setFloat4(this._uniforms[e], t.x, t.y, t.z, t.w) || (this._valueCache[e] = null)); } /** * Sets a Quaternion on a uniform variable. * @param uniformName Name of the variable. * @param quaternion Value to be set. */ setQuaternion(e, t) { this._cacheFloat4(e, t.x, t.y, t.z, t.w) && (this._engine.setFloat4(this._uniforms[e], t.x, t.y, t.z, t.w) || (this._valueCache[e] = null)); } /** * Sets a float4 on a uniform variable. * @param uniformName Name of the variable. * @param x First float in float4. * @param y Second float in float4. * @param z Third float in float4. * @param w Fourth float in float4. * @returns this effect. */ setFloat4(e, t, r, n, i) { this._cacheFloat4(e, t, r, n, i) && (this._engine.setFloat4(this._uniforms[e], t, r, n, i) || (this._valueCache[e] = null)); } /** * Sets a Color3 on a uniform variable. * @param uniformName Name of the variable. * @param color3 Value to be set. */ setColor3(e, t) { this._cacheFloat3(e, t.r, t.g, t.b) && (this._engine.setFloat3(this._uniforms[e], t.r, t.g, t.b) || (this._valueCache[e] = null)); } /** * Sets a Color4 on a uniform variable. * @param uniformName Name of the variable. * @param color3 Value to be set. * @param alpha Alpha value to be set. */ setColor4(e, t, r) { this._cacheFloat4(e, t.r, t.g, t.b, r) && (this._engine.setFloat4(this._uniforms[e], t.r, t.g, t.b, r) || (this._valueCache[e] = null)); } /** * Sets a Color4 on a uniform variable * @param uniformName defines the name of the variable * @param color4 defines the value to be set */ setDirectColor4(e, t) { this._cacheFloat4(e, t.r, t.g, t.b, t.a) && (this._engine.setFloat4(this._uniforms[e], t.r, t.g, t.b, t.a) || (this._valueCache[e] = null)); } } class Tue extends DR { get _framebuffer() { return this.__framebuffer; } set _framebuffer(e) { this.__framebuffer && this._engine._releaseFramebufferObjects(this.__framebuffer), this.__framebuffer = e; } get _framebufferDepthStencil() { return this.__framebufferDepthStencil; } set _framebufferDepthStencil(e) { this.__framebufferDepthStencil && this._engine._releaseFramebufferObjects(this.__framebufferDepthStencil), this.__framebufferDepthStencil = e; } constructor(e, t, r, n) { super(e, t, r, n), this.__framebuffer = null, this.__framebufferDepthStencil = null, this._engine = n; } dispose(e = !1) { this._framebuffer = null, this._framebufferDepthStencil = null, super.dispose(e); } } class XG { get underlyingResource() { return this._nativeTexture; } constructor(e, t) { this._engine = t, this.set(e); } setUsage() { } set(e) { this._nativeTexture = e; } reset() { this._nativeTexture = null; } release() { this._nativeTexture && this._engine.deleteTexture(this._nativeTexture), this.reset(); } } function DE(A, e) { switch (A) { case 15: return _native.Engine.TEXTURE_FORMAT_D16; case 16: return _native.Engine.TEXTURE_FORMAT_D24; case 13: return _native.Engine.TEXTURE_FORMAT_D24S8; case 14: return _native.Engine.TEXTURE_FORMAT_D32F; case 36492: return _native.Engine.TEXTURE_FORMAT_BC7; case 36494: return _native.Engine.TEXTURE_FORMAT_BC6H; case 33779: return _native.Engine.TEXTURE_FORMAT_BC3; case 33778: return _native.Engine.TEXTURE_FORMAT_BC2; case 33777: return _native.Engine.TEXTURE_FORMAT_BC1; case 33776: return _native.Engine.TEXTURE_FORMAT_BC1; case 37808: return _native.Engine.TEXTURE_FORMAT_ASTC4x4; case 36196: return _native.Engine.TEXTURE_FORMAT_ETC1; case 37492: return _native.Engine.TEXTURE_FORMAT_ETC2; case 37496: return _native.Engine.TEXTURE_FORMAT_ETC2A; case 4: { switch (e) { case 0: return _native.Engine.TEXTURE_FORMAT_RGB8; case 3: return _native.Engine.TEXTURE_FORMAT_RGB8S; case 6: return _native.Engine.TEXTURE_FORMAT_RGB8I; case 7: return _native.Engine.TEXTURE_FORMAT_RGB8U; } break; } case 5: { switch (e) { case 0: return _native.Engine.TEXTURE_FORMAT_RGBA8; case 1: return _native.Engine.TEXTURE_FORMAT_RGBA32F; case 2: return _native.Engine.TEXTURE_FORMAT_RGBA16F; case 3: return _native.Engine.TEXTURE_FORMAT_RGBA8S; case 4: return _native.Engine.TEXTURE_FORMAT_RGBA16I; case 5: return _native.Engine.TEXTURE_FORMAT_RGBA16U; case 6: return _native.Engine.TEXTURE_FORMAT_RGBA32I; case 7: return _native.Engine.TEXTURE_FORMAT_RGBA32U; } break; } case 6: { switch (e) { case 0: return _native.Engine.TEXTURE_FORMAT_R8; case 1: return _native.Engine.TEXTURE_FORMAT_R32F; case 2: return _native.Engine.TEXTURE_FORMAT_R16F; case 3: return _native.Engine.TEXTURE_FORMAT_R8S; case 4: return _native.Engine.TEXTURE_FORMAT_R16S; case 5: return _native.Engine.TEXTURE_FORMAT_R16U; case 6: return _native.Engine.TEXTURE_FORMAT_R32I; case 7: return _native.Engine.TEXTURE_FORMAT_R32U; } break; } case 7: { switch (e) { case 0: return _native.Engine.TEXTURE_FORMAT_RG8; case 1: return _native.Engine.TEXTURE_FORMAT_RG32F; case 2: return _native.Engine.TEXTURE_FORMAT_RG16F; case 3: return _native.Engine.TEXTURE_FORMAT_RG8S; case 4: return _native.Engine.TEXTURE_FORMAT_RG16S; case 5: return _native.Engine.TEXTURE_FORMAT_RG16U; case 6: return _native.Engine.TEXTURE_FORMAT_RG32I; case 7: return _native.Engine.TEXTURE_FORMAT_RG32U; } break; } case 12: { switch (e) { case 0: return _native.Engine.TEXTURE_FORMAT_BGRA8; } break; } } throw new O0(`Unsupported texture format or type: format ${A}, type ${e}.`, Z2.UnsupportedTextureError); } function AI(A) { switch (A) { case 1: return _native.Engine.TEXTURE_NEAREST_NEAREST; case 2: return _native.Engine.TEXTURE_LINEAR_LINEAR; case 3: return _native.Engine.TEXTURE_LINEAR_LINEAR_MIPLINEAR; case 4: return _native.Engine.TEXTURE_NEAREST_NEAREST_MIPNEAREST; case 5: return _native.Engine.TEXTURE_NEAREST_LINEAR_MIPNEAREST; case 6: return _native.Engine.TEXTURE_NEAREST_LINEAR_MIPLINEAR; case 7: return _native.Engine.TEXTURE_NEAREST_LINEAR; case 8: return _native.Engine.TEXTURE_NEAREST_NEAREST_MIPLINEAR; case 9: return _native.Engine.TEXTURE_LINEAR_NEAREST_MIPNEAREST; case 10: return _native.Engine.TEXTURE_LINEAR_NEAREST_MIPLINEAR; case 11: return _native.Engine.TEXTURE_LINEAR_LINEAR_MIPNEAREST; case 12: return _native.Engine.TEXTURE_LINEAR_NEAREST; default: throw new Error(`Unsupported sampling mode: ${A}.`); } } function jE(A) { switch (A) { case 1: return _native.Engine.ADDRESS_MODE_WRAP; case 0: return _native.Engine.ADDRESS_MODE_CLAMP; case 2: return _native.Engine.ADDRESS_MODE_MIRROR; default: throw new Error("Unexpected wrap mode: " + A + "."); } } function que(A) { switch (A) { case 513: return _native.Engine.STENCIL_TEST_LESS; case 515: return _native.Engine.STENCIL_TEST_LEQUAL; case 514: return _native.Engine.STENCIL_TEST_EQUAL; case 518: return _native.Engine.STENCIL_TEST_GEQUAL; case 516: return _native.Engine.STENCIL_TEST_GREATER; case 517: return _native.Engine.STENCIL_TEST_NOTEQUAL; case 512: return _native.Engine.STENCIL_TEST_NEVER; case 519: return _native.Engine.STENCIL_TEST_ALWAYS; default: throw new Error(`Unsupported stencil func mode: ${A}.`); } } function bue(A) { switch (A) { case 7680: return _native.Engine.STENCIL_OP_FAIL_S_KEEP; case 0: return _native.Engine.STENCIL_OP_FAIL_S_ZERO; case 7681: return _native.Engine.STENCIL_OP_FAIL_S_REPLACE; case 7682: return _native.Engine.STENCIL_OP_FAIL_S_INCR; case 7683: return _native.Engine.STENCIL_OP_FAIL_S_DECR; case 5386: return _native.Engine.STENCIL_OP_FAIL_S_INVERT; case 34055: return _native.Engine.STENCIL_OP_FAIL_S_INCRSAT; case 34056: return _native.Engine.STENCIL_OP_FAIL_S_DECRSAT; default: throw new Error(`Unsupported stencil OpFail mode: ${A}.`); } } function xue(A) { switch (A) { case 7680: return _native.Engine.STENCIL_OP_FAIL_Z_KEEP; case 0: return _native.Engine.STENCIL_OP_FAIL_Z_ZERO; case 7681: return _native.Engine.STENCIL_OP_FAIL_Z_REPLACE; case 7682: return _native.Engine.STENCIL_OP_FAIL_Z_INCR; case 7683: return _native.Engine.STENCIL_OP_FAIL_Z_DECR; case 5386: return _native.Engine.STENCIL_OP_FAIL_Z_INVERT; case 34055: return _native.Engine.STENCIL_OP_FAIL_Z_INCRSAT; case 34056: return _native.Engine.STENCIL_OP_FAIL_Z_DECRSAT; default: throw new Error(`Unsupported stencil depthFail mode: ${A}.`); } } function Due(A) { switch (A) { case 7680: return _native.Engine.STENCIL_OP_PASS_Z_KEEP; case 0: return _native.Engine.STENCIL_OP_PASS_Z_ZERO; case 7681: return _native.Engine.STENCIL_OP_PASS_Z_REPLACE; case 7682: return _native.Engine.STENCIL_OP_PASS_Z_INCR; case 7683: return _native.Engine.STENCIL_OP_PASS_Z_DECR; case 5386: return _native.Engine.STENCIL_OP_PASS_Z_INVERT; case 34055: return _native.Engine.STENCIL_OP_PASS_Z_INCRSAT; case 34056: return _native.Engine.STENCIL_OP_PASS_Z_DECRSAT; default: throw new Error(`Unsupported stencil opPass mode: ${A}.`); } } function jue(A) { switch (A) { case 0: return _native.Engine.ALPHA_DISABLE; case 1: return _native.Engine.ALPHA_ADD; case 2: return _native.Engine.ALPHA_COMBINE; case 3: return _native.Engine.ALPHA_SUBTRACT; case 4: return _native.Engine.ALPHA_MULTIPLY; case 5: return _native.Engine.ALPHA_MAXIMIZED; case 6: return _native.Engine.ALPHA_ONEONE; case 7: return _native.Engine.ALPHA_PREMULTIPLIED; case 8: return _native.Engine.ALPHA_PREMULTIPLIED_PORTERDUFF; case 9: return _native.Engine.ALPHA_INTERPOLATE; case 10: return _native.Engine.ALPHA_SCREENMODE; default: throw new Error(`Unsupported alpha mode: ${A}.`); } } function wue(A) { switch (A) { case J.BYTE: return _native.Engine.ATTRIB_TYPE_INT8; case J.UNSIGNED_BYTE: return _native.Engine.ATTRIB_TYPE_UINT8; case J.SHORT: return _native.Engine.ATTRIB_TYPE_INT16; case J.UNSIGNED_SHORT: return _native.Engine.ATTRIB_TYPE_UINT16; case J.FLOAT: return _native.Engine.ATTRIB_TYPE_FLOAT; default: throw new Error(`Unsupported attribute type: ${A}.`); } } const hte = new Oe(); if (typeof self < "u" && !Object.prototype.hasOwnProperty.call(self, "_native")) { let A; Object.defineProperty(self, "_native", { get: () => A, set: (e) => { A = e, A && hte.notifyObservers(A); } }); } function Hte() { return new Promise((A) => { typeof _native > "u" ? hte.addOnce((e) => A(e)) : A(_native); }); } async function gte(A, e) { (await Hte())[A] = e; } class TG extends tg { } class mue { constructor(e) { this._engine = e, this._pending = new Array(), this._isCommandBufferScopeActive = !1, this._commandStream = Hm._createNativeDataStream(), this._engine.setCommandDataStream(this._commandStream); } beginCommandScope() { if (this._isCommandBufferScopeActive) throw new Error("Command scope already active."); this._isCommandBufferScopeActive = !0; } endCommandScope() { if (!this._isCommandBufferScopeActive) throw new Error("Command scope is not active."); this._isCommandBufferScopeActive = !1, this._submit(); } startEncodingCommand(e) { this._commandStream.writeNativeData(e); } encodeCommandArgAsUInt32(e) { this._commandStream.writeUint32(e); } encodeCommandArgAsUInt32s(e) { this._commandStream.writeUint32Array(e); } encodeCommandArgAsInt32(e) { this._commandStream.writeInt32(e); } encodeCommandArgAsInt32s(e) { this._commandStream.writeInt32Array(e); } encodeCommandArgAsFloat32(e) { this._commandStream.writeFloat32(e); } encodeCommandArgAsFloat32s(e) { this._commandStream.writeFloat32Array(e); } encodeCommandArgAsNativeData(e) { this._commandStream.writeNativeData(e), this._pending.push(e); } finishEncodingCommand() { this._isCommandBufferScopeActive || this._submit(); } _submit() { this._engine.submitCommands(), this._pending.length = 0; } } class Hm extends Ge { setHardwareScalingLevel(e) { super.setHardwareScalingLevel(e), this._engine.setHardwareScalingLevel(e); } constructor(e = {}) { if (super(null, !1, void 0, e.adaptToDeviceRatio), this._engine = new _native.Engine(), this._camera = _native.Camera ? new _native.Camera() : null, this._commandBufferEncoder = new mue(this._engine), this._boundBuffersVertexArray = null, this._currentDepthTest = _native.Engine.DEPTH_TEST_LEQUAL, this._stencilTest = !1, this._stencilMask = 255, this._stencilFunc = 519, this._stencilFuncRef = 0, this._stencilFuncMask = 255, this._stencilOpStencilFail = 7680, this._stencilOpDepthFail = 7680, this._stencilOpStencilDepthPass = 7681, this._zOffset = 0, this._zOffsetUnits = 0, this._depthWrite = !0, _native.Engine.PROTOCOL_VERSION !== Hm.PROTOCOL_VERSION) throw new Error(`Protocol version mismatch: ${_native.Engine.PROTOCOL_VERSION} (Native) !== ${Hm.PROTOCOL_VERSION} (JS)`); this._webGLVersion = 2, this.disableUniformBuffers = !0, this._shaderPlatformName = "NATIVE", this._caps = { maxTexturesImageUnits: 16, maxVertexTextureImageUnits: 16, maxCombinedTexturesImageUnits: 32, maxTextureSize: _native.Engine.CAPS_LIMITS_MAX_TEXTURE_SIZE, maxCubemapTextureSize: 512, maxRenderTextureSize: 512, maxVertexAttribs: 16, maxVaryingVectors: 16, maxFragmentUniformVectors: 16, maxVertexUniformVectors: 16, standardDerivatives: !0, astc: null, pvrtc: null, etc1: null, etc2: null, bptc: null, maxAnisotropy: 16, uintIndices: !0, fragmentDepthSupported: !1, highPrecisionShaderSupported: !0, colorBufferFloat: !1, supportFloatTexturesResolve: !1, textureFloat: !0, textureFloatLinearFiltering: !1, textureFloatRender: !0, textureHalfFloat: !0, textureHalfFloatLinearFiltering: !1, textureHalfFloatRender: !0, textureLOD: !0, texelFetch: !1, drawBuffersExtension: !1, depthTextureExtension: !1, vertexArrayObject: !0, instancedArrays: !0, supportOcclusionQuery: !1, canUseTimestampForTimerQuery: !1, blendMinMax: !1, maxMSAASamples: 16, canUseGLInstanceID: !0, canUseGLVertexID: !0, supportComputeShaders: !1, supportSRGBBuffers: !0, supportTransformFeedbacks: !1, textureMaxLevel: !1, texture2DArrayMaxLayerCount: _native.Engine.CAPS_LIMITS_MAX_TEXTURE_LAYERS, disableMorphTargetTexture: !1 }, this._features = { forceBitmapOverHTMLImageElement: !1, supportRenderAndCopyToLodForFloatTextures: !1, supportDepthStencilTexture: !1, supportShadowSamplers: !1, uniformBufferHardCheckMatrix: !1, allowTexturePrefiltering: !1, trackUbosInFrame: !1, checkUbosContentBeforeUpload: !1, supportCSM: !1, basisNeedsPOT: !1, support3DTextures: !1, needTypeSuffixInShaderConstants: !1, supportMSAA: !0, supportSSAO2: !1, supportExtendedTextureFormats: !1, supportSwitchCaseInShader: !1, supportSyncTextureRead: !1, needsInvertingBitmap: !0, useUBOBindingCache: !0, needShaderCodeInlining: !0, needToAlwaysBindUniformBuffers: !1, supportRenderPasses: !0, supportSpriteInstancing: !1, forceVertexBufferStrideMultiple4Bytes: !1, _collectUbosUpdatedInFrame: !1 }, ye.Log("Babylon Native (v" + Ge.Version + ") launched"), ye.LoadScript = function(n, i, s, a) { ye.LoadFile(n, (f) => { Function(f).apply(null), i && i(); }, void 0, void 0, !1, (f, o) => { s && s("LoadScript Error", o); }); }, typeof URL > "u" && (window.URL = { createObjectURL: function() { }, revokeObjectURL: function() { } }), typeof Blob > "u" && (window.Blob = function(n) { return n; }), Array.prototype.flat || Object.defineProperty(Array.prototype, "flat", { configurable: !0, value: function n() { const i = isNaN(arguments[0]) ? 1 : Number(arguments[0]); return i ? Array.prototype.reduce.call(this, function(s, a) { return Array.isArray(a) ? s.push.apply(s, n.call(a, i - 1)) : s.push(a), s; }, []) : Array.prototype.slice.call(this); }, writable: !0 }); const t = window && window.devicePixelRatio || 1; this._hardwareScalingLevel = e.adaptToDeviceRatio ? 1 / t : 1, this._engine.setHardwareScalingLevel(this._hardwareScalingLevel), this._lastDevicePixelRatio = t, this.resize(); const r = this.getDepthFunction(); r && this.setDepthFunction(r), this._shaderProcessor = new LN(), this.onNewSceneAddedObservable.add((n) => { const i = n.render; n.render = (...s) => { this._commandBufferEncoder.beginCommandScope(), i.apply(n, s), this._commandBufferEncoder.endCommandScope(); }; }); } dispose() { super.dispose(), this._boundBuffersVertexArray && this._deleteVertexArray(this._boundBuffersVertexArray), this._engine.dispose(); } /** @internal */ static _createNativeDataStream() { return new hD(); } /** * Can be used to override the current requestAnimationFrame requester. * @internal */ _queueNewFrame(e, t) { return t.requestAnimationFrame && t !== window ? t.requestAnimationFrame(e) : this._engine.requestAnimationFrame(e), 0; } /** * Override default engine behavior. * @param framebuffer */ _bindUnboundFramebuffer(e) { this._currentFramebuffer !== e && (this._currentFramebuffer && (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_UNBINDFRAMEBUFFER), this._commandBufferEncoder.encodeCommandArgAsNativeData(this._currentFramebuffer), this._commandBufferEncoder.finishEncodingCommand()), e && (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_BINDFRAMEBUFFER), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.finishEncodingCommand()), this._currentFramebuffer = e); } /** * Gets host document * @returns the host document object */ getHostDocument() { return null; } clear(e, t, r, n = !1) { if (this.useReverseDepthBuffer) throw new Error("reverse depth buffer is not currently implemented"); this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_CLEAR), this._commandBufferEncoder.encodeCommandArgAsUInt32(t && e ? 1 : 0), this._commandBufferEncoder.encodeCommandArgAsFloat32(e ? e.r : 0), this._commandBufferEncoder.encodeCommandArgAsFloat32(e ? e.g : 0), this._commandBufferEncoder.encodeCommandArgAsFloat32(e ? e.b : 0), this._commandBufferEncoder.encodeCommandArgAsFloat32(e ? e.a : 1), this._commandBufferEncoder.encodeCommandArgAsUInt32(r ? 1 : 0), this._commandBufferEncoder.encodeCommandArgAsFloat32(1), this._commandBufferEncoder.encodeCommandArgAsUInt32(n ? 1 : 0), this._commandBufferEncoder.encodeCommandArgAsUInt32(0), this._commandBufferEncoder.finishEncodingCommand(); } createIndexBuffer(e, t, r) { const n = this._normalizeIndexData(e), i = new TG(); return i.references = 1, i.is32Bits = n.BYTES_PER_ELEMENT === 4, n.byteLength && (i.nativeIndexBuffer = this._engine.createIndexBuffer(n.buffer, n.byteOffset, n.byteLength, i.is32Bits, t ?? !1)), i; } createVertexBuffer(e, t, r) { const n = ArrayBuffer.isView(e) ? e : new Float32Array(e), i = new TG(); return i.references = 1, n.byteLength && (i.nativeVertexBuffer = this._engine.createVertexBuffer(n.buffer, n.byteOffset, n.byteLength, t ?? !1)), i; } _recordVertexArrayObject(e, t, r, n, i) { r && this._engine.recordIndexBuffer(e, r.nativeIndexBuffer); const s = n.getAttributesNames(); for (let a = 0; a < s.length; a++) { const f = n.getAttributeLocation(a); if (f >= 0) { const o = s[a]; let d = null; if (i && (d = i[o]), d || (d = t[o]), d) { const v = d.getBuffer(); v && v.nativeVertexBuffer && this._engine.recordVertexBuffer(e, v.nativeVertexBuffer, f, d.byteOffset, d.byteStride, d.getSize(), wue(d.type), d.normalized, d.getInstanceDivisor()); } } } } bindBuffers(e, t, r) { this._boundBuffersVertexArray && this._deleteVertexArray(this._boundBuffersVertexArray), this._boundBuffersVertexArray = this._engine.createVertexArray(), this._recordVertexArrayObject(this._boundBuffersVertexArray, e, t, r), this.bindVertexArrayObject(this._boundBuffersVertexArray); } recordVertexArrayObject(e, t, r, n) { const i = this._engine.createVertexArray(); return this._recordVertexArrayObject(i, e, t, r, n), i; } _deleteVertexArray(e) { this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_DELETEVERTEXARRAY), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.finishEncodingCommand(); } bindVertexArrayObject(e) { this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_BINDVERTEXARRAY), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.finishEncodingCommand(); } releaseVertexArrayObject(e) { this._deleteVertexArray(e); } getAttributes(e, t) { const r = e; return this._engine.getAttributes(r.nativeProgram, t); } /** * Draw a list of indexed primitives * @param fillMode defines the primitive to use * @param indexStart defines the starting index * @param indexCount defines the number of index to draw * @param instancesCount defines the number of instances to draw (if instantiation is enabled) */ drawElementsType(e, t, r, n) { this._drawCalls.addCount(1, !1), n && _native.Engine.COMMAND_DRAWINDEXEDINSTANCED ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_DRAWINDEXEDINSTANCED), this._commandBufferEncoder.encodeCommandArgAsUInt32(e), this._commandBufferEncoder.encodeCommandArgAsUInt32(t), this._commandBufferEncoder.encodeCommandArgAsUInt32(r), this._commandBufferEncoder.encodeCommandArgAsUInt32(n)) : (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_DRAWINDEXED), this._commandBufferEncoder.encodeCommandArgAsUInt32(e), this._commandBufferEncoder.encodeCommandArgAsUInt32(t), this._commandBufferEncoder.encodeCommandArgAsUInt32(r)), this._commandBufferEncoder.finishEncodingCommand(); } /** * Draw a list of unindexed primitives * @param fillMode defines the primitive to use * @param verticesStart defines the index of first vertex to draw * @param verticesCount defines the count of vertices to draw * @param instancesCount defines the number of instances to draw (if instantiation is enabled) */ drawArraysType(e, t, r, n) { this._drawCalls.addCount(1, !1), n && _native.Engine.COMMAND_DRAWINSTANCED ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_DRAWINSTANCED), this._commandBufferEncoder.encodeCommandArgAsUInt32(e), this._commandBufferEncoder.encodeCommandArgAsUInt32(t), this._commandBufferEncoder.encodeCommandArgAsUInt32(r), this._commandBufferEncoder.encodeCommandArgAsUInt32(n)) : (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_DRAW), this._commandBufferEncoder.encodeCommandArgAsUInt32(e), this._commandBufferEncoder.encodeCommandArgAsUInt32(t), this._commandBufferEncoder.encodeCommandArgAsUInt32(r)), this._commandBufferEncoder.finishEncodingCommand(); } createPipelineContext() { return new Xue(this); } createMaterialContext() { } createDrawContext() { } _preparePipelineContext(e, t, r, n, i, s, a, f) { const o = e; n ? o.nativeProgram = this.createRawShaderProgram() : o.nativeProgram = this.createShaderProgram(e, t, r, f); } isAsync(e) { return !!(e.isAsync && this._engine.createProgramAsync); } /** * @internal */ _executeWhenRenderingStateIsCompiled(e, t) { const r = e; if (!this.isAsync(e)) { t(); return; } const n = r.onCompiled; n ? r.onCompiled = () => { n(), t(); } : r.onCompiled = t; } createRawShaderProgram() { throw new Error("Not Supported"); } createShaderProgram(e, t, r, n) { const i = e; if (i.nativeProgram) throw new Error("Tried to create a second program in the same NativePipelineContext"); this.onBeforeShaderCompilationObservable.notifyObservers(this); const s = new Aq(t); s.processCode(), t = s.code; const a = new Aq(r); a.processCode(), r = a.code, t = hr._ConcatenateShader(t, n), r = hr._ConcatenateShader(r, n); const f = () => { var o; i.isCompiled = !0, (o = i.onCompiled) === null || o === void 0 || o.call(i), this.onAfterShaderCompilationObservable.notifyObservers(this); }; if (this.isAsync(e)) return this._engine.createProgramAsync(t, r, f, (o) => { i.compilationError = o; }); try { const o = i.nativeProgram = this._engine.createProgram(t, r); return f(), o; } catch (o) { const d = o == null ? void 0 : o.message; throw new Error("SHADER ERROR" + (typeof d == "string" ? ` ` + d : "")); } } /** * Inline functions in shader code that are marked to be inlined * @param code code to inline * @returns inlined code */ inlineShaderCode(e) { const t = new Aq(e); return t.debug = !1, t.processCode(), t.code; } _setProgram(e) { this._currentProgram !== e && (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETPROGRAM), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.finishEncodingCommand(), this._currentProgram = e); } _deletePipelineContext(e) { const t = e; t && t.nativeProgram && (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_DELETEPROGRAM), this._commandBufferEncoder.encodeCommandArgAsNativeData(t.nativeProgram), this._commandBufferEncoder.finishEncodingCommand()); } getUniforms(e, t) { const r = e; return this._engine.getUniforms(r.nativeProgram, t); } bindUniformBlock(e, t, r) { throw new Error("Not Implemented"); } bindSamplers(e) { const t = e.getPipelineContext(); this._setProgram(t.nativeProgram); const r = e.getSamplers(); for (let n = 0; n < r.length; n++) { const i = e.getUniform(r[n]); i && (this._boundUniforms[n] = i); } this._currentEffect = null; } getRenderWidth(e = !1) { return !e && this._currentRenderTarget ? this._currentRenderTarget.width : this._engine.getRenderWidth(); } getRenderHeight(e = !1) { return !e && this._currentRenderTarget ? this._currentRenderTarget.height : this._engine.getRenderHeight(); } setViewport(e, t, r) { this._cachedViewport = e, this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETVIEWPORT), this._commandBufferEncoder.encodeCommandArgAsFloat32(e.x), this._commandBufferEncoder.encodeCommandArgAsFloat32(e.y), this._commandBufferEncoder.encodeCommandArgAsFloat32(e.width), this._commandBufferEncoder.encodeCommandArgAsFloat32(e.height), this._commandBufferEncoder.finishEncodingCommand(); } enableScissor(e, t, r, n) { this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETSCISSOR), this._commandBufferEncoder.encodeCommandArgAsFloat32(e), this._commandBufferEncoder.encodeCommandArgAsFloat32(t), this._commandBufferEncoder.encodeCommandArgAsFloat32(r), this._commandBufferEncoder.encodeCommandArgAsFloat32(n), this._commandBufferEncoder.finishEncodingCommand(); } disableScissor() { this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETSCISSOR), this._commandBufferEncoder.encodeCommandArgAsFloat32(0), this._commandBufferEncoder.encodeCommandArgAsFloat32(0), this._commandBufferEncoder.encodeCommandArgAsFloat32(0), this._commandBufferEncoder.encodeCommandArgAsFloat32(0), this._commandBufferEncoder.finishEncodingCommand(); } setState(e, t = 0, r, n = !1, i, s, a = 0) { var f, o; this._zOffset = t, this._zOffsetUnits = a, this._zOffset !== 0 && ye.Warn("zOffset is not supported in Native engine."), this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETSTATE), this._commandBufferEncoder.encodeCommandArgAsUInt32(e ? 1 : 0), this._commandBufferEncoder.encodeCommandArgAsFloat32(t), this._commandBufferEncoder.encodeCommandArgAsFloat32(a), this._commandBufferEncoder.encodeCommandArgAsUInt32(!((o = (f = this.cullBackFaces) !== null && f !== void 0 ? f : i) !== null && o !== void 0) || o ? 1 : 0), this._commandBufferEncoder.encodeCommandArgAsUInt32(n ? 1 : 0), this._commandBufferEncoder.finishEncodingCommand(); } /** * Gets the client rect of native canvas. Needed for InputManager. * @returns a client rectangle */ getInputElementClientRect() { return { bottom: this.getRenderHeight(), height: this.getRenderHeight(), left: 0, right: this.getRenderWidth(), top: 0, width: this.getRenderWidth(), x: 0, y: 0, toJSON: () => { } }; } /** * Set the z offset Factor to apply to current rendering * @param value defines the offset to apply */ setZOffset(e) { e !== this._zOffset && (this._zOffset = e, this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETZOFFSET), this._commandBufferEncoder.encodeCommandArgAsFloat32(this.useReverseDepthBuffer ? -e : e), this._commandBufferEncoder.finishEncodingCommand()); } /** * Gets the current value of the zOffset Factor * @returns the current zOffset Factor state */ getZOffset() { return this._zOffset; } /** * Set the z offset Units to apply to current rendering * @param value defines the offset to apply */ setZOffsetUnits(e) { e !== this._zOffsetUnits && (this._zOffsetUnits = e, this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETZOFFSETUNITS), this._commandBufferEncoder.encodeCommandArgAsFloat32(this.useReverseDepthBuffer ? -e : e), this._commandBufferEncoder.finishEncodingCommand()); } /** * Gets the current value of the zOffset Units * @returns the current zOffset Units state */ getZOffsetUnits() { return this._zOffsetUnits; } /** * Enable or disable depth buffering * @param enable defines the state to set */ setDepthBuffer(e) { this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETDEPTHTEST), this._commandBufferEncoder.encodeCommandArgAsUInt32(e ? this._currentDepthTest : _native.Engine.DEPTH_TEST_ALWAYS), this._commandBufferEncoder.finishEncodingCommand(); } /** * Gets a boolean indicating if depth writing is enabled * @returns the current depth writing state */ getDepthWrite() { return this._depthWrite; } getDepthFunction() { switch (this._currentDepthTest) { case _native.Engine.DEPTH_TEST_NEVER: return 512; case _native.Engine.DEPTH_TEST_ALWAYS: return 519; case _native.Engine.DEPTH_TEST_GREATER: return 516; case _native.Engine.DEPTH_TEST_GEQUAL: return 518; case _native.Engine.DEPTH_TEST_NOTEQUAL: return 517; case _native.Engine.DEPTH_TEST_EQUAL: return 514; case _native.Engine.DEPTH_TEST_LESS: return 513; case _native.Engine.DEPTH_TEST_LEQUAL: return 515; } return null; } setDepthFunction(e) { let t = 0; switch (e) { case 512: t = _native.Engine.DEPTH_TEST_NEVER; break; case 519: t = _native.Engine.DEPTH_TEST_ALWAYS; break; case 516: t = _native.Engine.DEPTH_TEST_GREATER; break; case 518: t = _native.Engine.DEPTH_TEST_GEQUAL; break; case 517: t = _native.Engine.DEPTH_TEST_NOTEQUAL; break; case 514: t = _native.Engine.DEPTH_TEST_EQUAL; break; case 513: t = _native.Engine.DEPTH_TEST_LESS; break; case 515: t = _native.Engine.DEPTH_TEST_LEQUAL; break; } this._currentDepthTest = t, this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETDEPTHTEST), this._commandBufferEncoder.encodeCommandArgAsUInt32(this._currentDepthTest), this._commandBufferEncoder.finishEncodingCommand(); } /** * Enable or disable depth writing * @param enable defines the state to set */ setDepthWrite(e) { this._depthWrite = e, this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETDEPTHWRITE), this._commandBufferEncoder.encodeCommandArgAsUInt32(Number(e)), this._commandBufferEncoder.finishEncodingCommand(); } /** * Enable or disable color writing * @param enable defines the state to set */ setColorWrite(e) { this._colorWrite = e, this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETCOLORWRITE), this._commandBufferEncoder.encodeCommandArgAsUInt32(Number(e)), this._commandBufferEncoder.finishEncodingCommand(); } /** * Gets a boolean indicating if color writing is enabled * @returns the current color writing state */ getColorWrite() { return this._colorWrite; } applyStencil() { this._setStencil(this._stencilMask, bue(this._stencilOpStencilFail), xue(this._stencilOpDepthFail), Due(this._stencilOpStencilDepthPass), que(this._stencilFunc), this._stencilFuncRef); } _setStencil(e, t, r, n, i, s) { this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETSTENCIL), this._commandBufferEncoder.encodeCommandArgAsUInt32(e), this._commandBufferEncoder.encodeCommandArgAsUInt32(t), this._commandBufferEncoder.encodeCommandArgAsUInt32(r), this._commandBufferEncoder.encodeCommandArgAsUInt32(n), this._commandBufferEncoder.encodeCommandArgAsUInt32(i), this._commandBufferEncoder.encodeCommandArgAsUInt32(s), this._commandBufferEncoder.finishEncodingCommand(); } /** * Enable or disable the stencil buffer * @param enable defines if the stencil buffer must be enabled or disabled */ setStencilBuffer(e) { this._stencilTest = e, e ? this.applyStencil() : this._setStencil(255, _native.Engine.STENCIL_OP_FAIL_S_KEEP, _native.Engine.STENCIL_OP_FAIL_Z_KEEP, _native.Engine.STENCIL_OP_PASS_Z_KEEP, _native.Engine.STENCIL_TEST_ALWAYS, 0); } /** * Gets a boolean indicating if stencil buffer is enabled * @returns the current stencil buffer state */ getStencilBuffer() { return this._stencilTest; } /** * Gets the current stencil operation when stencil passes * @returns a number defining stencil operation to use when stencil passes */ getStencilOperationPass() { return this._stencilOpStencilDepthPass; } /** * Sets the stencil operation to use when stencil passes * @param operation defines the stencil operation to use when stencil passes */ setStencilOperationPass(e) { this._stencilOpStencilDepthPass = e, this.applyStencil(); } /** * Sets the current stencil mask * @param mask defines the new stencil mask to use */ setStencilMask(e) { this._stencilMask = e, this.applyStencil(); } /** * Sets the current stencil function * @param stencilFunc defines the new stencil function to use */ setStencilFunction(e) { this._stencilFunc = e, this.applyStencil(); } /** * Sets the current stencil reference * @param reference defines the new stencil reference to use */ setStencilFunctionReference(e) { this._stencilFuncRef = e, this.applyStencil(); } /** * Sets the current stencil mask * @param mask defines the new stencil mask to use */ setStencilFunctionMask(e) { this._stencilFuncMask = e; } /** * Sets the stencil operation to use when stencil fails * @param operation defines the stencil operation to use when stencil fails */ setStencilOperationFail(e) { this._stencilOpStencilFail = e, this.applyStencil(); } /** * Sets the stencil operation to use when depth fails * @param operation defines the stencil operation to use when depth fails */ setStencilOperationDepthFail(e) { this._stencilOpDepthFail = e, this.applyStencil(); } /** * Gets the current stencil mask * @returns a number defining the new stencil mask to use */ getStencilMask() { return this._stencilMask; } /** * Gets the current stencil function * @returns a number defining the stencil function to use */ getStencilFunction() { return this._stencilFunc; } /** * Gets the current stencil reference value * @returns a number defining the stencil reference value to use */ getStencilFunctionReference() { return this._stencilFuncRef; } /** * Gets the current stencil mask * @returns a number defining the stencil mask to use */ getStencilFunctionMask() { return this._stencilFuncMask; } /** * Gets the current stencil operation when stencil fails * @returns a number defining stencil operation to use when stencil fails */ getStencilOperationFail() { return this._stencilOpStencilFail; } /** * Gets the current stencil operation when depth fails * @returns a number defining stencil operation to use when depth fails */ getStencilOperationDepthFail() { return this._stencilOpDepthFail; } /** * Sets alpha constants used by some alpha blending modes * @param r defines the red component * @param g defines the green component * @param b defines the blue component * @param a defines the alpha component */ setAlphaConstants(e, t, r, n) { throw new Error("Setting alpha blend constant color not yet implemented."); } /** * Sets the current alpha mode * @param mode defines the mode to use (one of the BABYLON.undefined) * @param noDepthWriteChange defines if depth writing state should remains unchanged (false by default) * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/advanced/transparent_rendering */ setAlphaMode(e, t = !1) { if (this._alphaMode === e) return; const r = jue(e); this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETBLENDMODE), this._commandBufferEncoder.encodeCommandArgAsUInt32(r), this._commandBufferEncoder.finishEncodingCommand(), t || this.setDepthWrite(e === 0), this._alphaMode = e; } /** * Gets the current alpha mode * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/advanced/transparent_rendering * @returns the current alpha mode */ getAlphaMode() { return this._alphaMode; } setInt(e, t) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETINT), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsInt32(t), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setIntArray(e, t) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETINTARRAY), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsInt32s(t), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setIntArray2(e, t) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETINTARRAY2), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsInt32s(t), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setIntArray3(e, t) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETINTARRAY3), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsInt32s(t), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setIntArray4(e, t) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETINTARRAY4), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsInt32s(t), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setFloatArray(e, t) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETFLOATARRAY), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsFloat32s(t), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setFloatArray2(e, t) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETFLOATARRAY2), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsFloat32s(t), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setFloatArray3(e, t) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETFLOATARRAY3), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsFloat32s(t), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setFloatArray4(e, t) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETFLOATARRAY4), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsFloat32s(t), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setArray(e, t) { return e ? this.setFloatArray(e, new Float32Array(t)) : !1; } setArray2(e, t) { return e ? this.setFloatArray2(e, new Float32Array(t)) : !1; } setArray3(e, t) { return e ? this.setFloatArray3(e, new Float32Array(t)) : !1; } setArray4(e, t) { return e ? this.setFloatArray4(e, new Float32Array(t)) : !1; } setMatrices(e, t) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETMATRICES), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsFloat32s(t), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setMatrix3x3(e, t) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETMATRIX3X3), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsFloat32s(t), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setMatrix2x2(e, t) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETMATRIX2X2), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsFloat32s(t), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setFloat(e, t) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETFLOAT), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsFloat32(t), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setFloat2(e, t, r) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETFLOAT2), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsFloat32(t), this._commandBufferEncoder.encodeCommandArgAsFloat32(r), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setFloat3(e, t, r, n) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETFLOAT3), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsFloat32(t), this._commandBufferEncoder.encodeCommandArgAsFloat32(r), this._commandBufferEncoder.encodeCommandArgAsFloat32(n), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setFloat4(e, t, r, n, i) { return e ? (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETFLOAT4), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsFloat32(t), this._commandBufferEncoder.encodeCommandArgAsFloat32(r), this._commandBufferEncoder.encodeCommandArgAsFloat32(n), this._commandBufferEncoder.encodeCommandArgAsFloat32(i), this._commandBufferEncoder.finishEncodingCommand(), !0) : !1; } setColor3(e, t) { return e ? (this.setFloat3(e, t.r, t.g, t.b), !0) : !1; } setColor4(e, t, r) { return e ? (this.setFloat4(e, t.r, t.g, t.b, r), !0) : !1; } wipeCaches(e) { this.preventCacheWipeBetweenFrames || (this.resetTextureCache(), this._currentEffect = null, e && (this._currentProgram = null, this._stencilStateComposer.reset(), this._depthCullingState.reset(), this._alphaState.reset()), this._cachedVertexBuffers = null, this._cachedIndexBuffer = null, this._cachedEffectForVertexBuffers = null); } _createTexture() { return this._engine.createTexture(); } _deleteTexture(e) { e && this._engine.deleteTexture(e); } /** * Update the content of a dynamic texture * @param texture defines the texture to update * @param canvas defines the canvas containing the source * @param invertY defines if data must be stored with Y axis inverted * @param premulAlpha defines if alpha is stored as premultiplied * @param format defines the format of the data */ updateDynamicTexture(e, t, r, n = !1, i) { if (n === void 0 && (n = !1), e && e._hardwareTexture) { const s = t.getCanvasTexture(), a = e._hardwareTexture.underlyingResource; this._engine.copyTexture(a, s), e.isReady = !0; } } createDynamicTexture(e, t, r, n) { return e = Math.max(e, 1), t = Math.max(t, 1), this.createRawTexture(new Uint8Array(e * t * 4), e, t, 5, !1, !1, n); } createVideoElement(e) { return this._camera ? this._camera.createVideo(e) : null; } updateVideoTexture(e, t, r) { if (e && e._hardwareTexture && this._camera) { const n = e._hardwareTexture.underlyingResource; this._camera.updateVideoTexture(n, t, r); } } createRawTexture(e, t, r, n, i, s, a, f = null, o = 0, d = 0, v = !1) { const u = new As(this, ri.Raw); if (u.format = n, u.generateMipMaps = i, u.samplingMode = a, u.invertY = s, u.baseWidth = t, u.baseHeight = r, u.width = u.baseWidth, u.height = u.baseHeight, u._compression = f, u.type = o, u._useSRGBBuffer = this._getUseSRGBBuffer(v, !i), this.updateRawTexture(u, e, n, s, f, o, u._useSRGBBuffer), u._hardwareTexture) { const l = u._hardwareTexture.underlyingResource, P = AI(a); this._setTextureSampling(l, P); } return this._internalTexturesCache.push(u), u; } createRawTexture2DArray(e, t, r, n, i, s, a, f, o = null, d = 0) { const v = new As(this, ri.Raw2DArray); if (v.baseWidth = t, v.baseHeight = r, v.baseDepth = n, v.width = t, v.height = r, v.depth = n, v.format = i, v.type = d, v.generateMipMaps = s, v.samplingMode = f, v.is2DArray = !0, v._hardwareTexture) { const u = v._hardwareTexture.underlyingResource; this._engine.loadRawTexture2DArray(u, e, t, r, n, DE(i, d), s, a); const l = AI(f); this._setTextureSampling(u, l); } return v.isReady = !0, this._internalTexturesCache.push(v), v; } updateRawTexture(e, t, r, n, i = null, s = 0, a = !1) { if (e) { if (t && e._hardwareTexture) { const f = e._hardwareTexture.underlyingResource; this._engine.loadRawTexture(f, t, e.width, e.height, DE(r, s), e.generateMipMaps, e.invertY); } e.isReady = !0; } } // TODO: Refactor to share more logic with babylon.engine.ts version. /** * Usually called from Texture.ts. * Passed information to create a NativeTexture * @param url defines a value which contains one of the following: * * A conventional http URL, e.g. 'http://...' or 'file://...' * * A base64 string of in-line texture data, e.g. '...' * * An indicator that data being passed using the buffer parameter, e.g. 'data:mytexture.jpg' * @param noMipmap defines a boolean indicating that no mipmaps shall be generated. Ignored for compressed textures. They must be in the file * @param invertY when true, image is flipped when loaded. You probably want true. Certain compressed textures may invert this if their default is inverted (eg. ktx) * @param scene needed for loading to the correct scene * @param samplingMode mode with should be used sample / access the texture (Default: Texture.TRILINEAR_SAMPLINGMODE) * @param onLoad optional callback to be called upon successful completion * @param onError optional callback to be called upon failure * @param buffer a source of a file previously fetched as either a base64 string, an ArrayBuffer (compressed or image format), HTMLImageElement (image format), or a Blob * @param fallback an internal argument in case the function must be called again, due to etc1 not having alpha capabilities * @param format internal format. Default: RGB when extension is '.jpg' else RGBA. Ignored for compressed textures * @param forcedExtension defines the extension to use to pick the right loader * @param mimeType defines an optional mime type * @param loaderOptions options to be passed to the loader * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg) * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU). * @returns a InternalTexture for assignment back into BABYLON.Texture */ createTexture(e, t, r, n, i = 3, s = null, a = null, f = null, o = null, d = null, v = null, u, l, P, p = !1) { e = e || ""; const c = e.substr(0, 5) === "data:", H = c && e.indexOf(";base64,") !== -1, T = o || new As(this, ri.Url), q = e; this._transformTextureUrl && !H && !o && !f && (e = this._transformTextureUrl(e)); const b = e.lastIndexOf("."), j = v || (b > -1 ? e.substring(b).toLowerCase() : ""); let w = null; for (const N of Ge._TextureLoaders) if (N.canLoad(j)) { w = N; break; } n && n.addPendingData(T), T.url = e, T.generateMipMaps = !t, T.samplingMode = i, T.invertY = r, T._useSRGBBuffer = this._getUseSRGBBuffer(p, t), this.doNotHandleContextLost || (T._buffer = f); let m = null; s && !o && (m = T.onLoadedObservable.add(s)), o || this._internalTexturesCache.push(T); const I = (N, k) => { n && n.removePendingData(T), e === q ? (m && T.onLoadedObservable.remove(m), gr.UseFallbackTexture && this.createTexture(gr.FallbackTexture, t, T.invertY, n, i, null, a, f, T), a && a((N || "Unknown error") + (gr.UseFallbackTexture ? " - Fallback texture was used" : ""), k)) : (Se.Warn(`Failed to load ${e}, falling back to ${q}`), this.createTexture(q, t, T.invertY, n, i, s, a, f, T, d, v, u, l)); }; if (w) throw new Error("Loading textures from IInternalTextureLoader not yet implemented."); { const N = (k) => { if (!T._hardwareTexture) { n && n.removePendingData(T); return; } const R = T._hardwareTexture.underlyingResource; this._engine.loadTexture(R, k, !t, r, T._useSRGBBuffer, () => { T.baseWidth = this._engine.getTextureWidth(R), T.baseHeight = this._engine.getTextureHeight(R), T.width = T.baseWidth, T.height = T.baseHeight, T.isReady = !0; const y = AI(i); this._setTextureSampling(R, y), n && n.removePendingData(T), T.onLoadedObservable.notifyObservers(T), T.onLoadedObservable.clear(); }, () => { throw new Error("Could not load a native texture."); }); }; if (c && f) if (f instanceof ArrayBuffer) N(new Uint8Array(f)); else if (ArrayBuffer.isView(f)) N(f); else if (typeof f == "string") N(new Uint8Array(ye.DecodeBase64(f))); else throw new Error("Unsupported buffer type"); else H ? N(new Uint8Array(ye.DecodeBase64(e))) : this._loadFile(e, (k) => N(new Uint8Array(k)), void 0, void 0, !0, (k, R) => { I("Unable to load " + (k && k.responseURL, R)); }); } return T; } /** * Wraps an external native texture in a Babylon texture. * @param texture defines the external texture * @param hasMipMaps defines whether the external texture has mip maps * @param samplingMode defines the sampling mode for the external texture (default: 3) * @returns the babylon internal texture */ wrapNativeTexture(e, t = !1, r = 3) { const n = new XG(e, this._engine), i = new As(this, ri.Unknown, !0); return i._hardwareTexture = n, i.baseWidth = this._engine.getTextureWidth(e), i.baseHeight = this._engine.getTextureHeight(e), i.width = i.baseWidth, i.height = i.baseHeight, i.isReady = !0, i.useMipMaps = t, this.updateTextureSamplingMode(r, i), i; } /** * Wraps an external web gl texture in a Babylon texture. * @returns the babylon internal texture */ wrapWebGLTexture() { throw new Error("wrapWebGLTexture is not supported, use wrapNativeTexture instead."); } _createDepthStencilTexture(e, t, r) { var n, i; const s = t.generateStencil || !1, a = t.samples || 1, f = r, o = new As(this, ri.DepthStencil), d = (n = e.width) !== null && n !== void 0 ? n : e, v = (i = e.height) !== null && i !== void 0 ? i : e, u = this._engine.createFrameBuffer(o._hardwareTexture.underlyingResource, d, v, s, !0, a); return f._framebufferDepthStencil = u, o; } /** * @internal */ _releaseFramebufferObjects(e) { e && (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_DELETEFRAMEBUFFER), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.finishEncodingCommand()); } /** * @internal Engine abstraction for loading and creating an image bitmap from a given source string. * @param imageSource source to load the image from. * @param options An object that sets options for the image's extraction. * @returns ImageBitmap */ _createImageBitmapFromSource(e, t) { return new Promise((n, i) => { const s = this.createCanvasImage(); s.onload = () => { try { const a = this._engine.createImageBitmap(s); n(a); } catch (a) { i(`Error loading image ${s.src} with exception: ${a}`); } }, s.onerror = (a) => { i(`Error loading image ${s.src} with exception: ${a}`); }, s.src = e; }); } /** * Engine abstraction for createImageBitmap * @param image source for image * @param options An object that sets options for the image's extraction. * @returns ImageBitmap */ createImageBitmap(e, t) { return new Promise((r, n) => { if (Array.isArray(e)) { const i = e; if (i.length) { const s = this._engine.createImageBitmap(i[0]); if (s) { r(s); return; } } } n("Unsupported data for createImageBitmap."); }); } /** * Resize an image and returns the image data as an uint8array * @param image image to resize * @param bufferWidth destination buffer width * @param bufferHeight destination buffer height * @returns an uint8array containing RGBA values of bufferWidth * bufferHeight size */ resizeImageBitmap(e, t, r) { return this._engine.resizeImageBitmap(e, t, r); } /** * Creates a cube texture * @param rootUrl defines the url where the files to load is located * @param scene defines the current scene * @param files defines the list of files to load (1 per face) * @param noMipmap defines a boolean indicating that no mipmaps shall be generated (false by default) * @param onLoad defines an optional callback raised when the texture is loaded * @param onError defines an optional callback raised if there is an issue to load the texture * @param format defines the format of the data * @param forcedExtension defines the extension to use to pick the right loader * @param createPolynomials if a polynomial sphere should be created for the cube texture * @param lodScale defines the scale applied to environment texture. This manages the range of LOD level used for IBL according to the roughness * @param lodOffset defines the offset applied to environment texture. This manages first LOD level used for IBL according to the roughness * @param fallback defines texture to use while falling back when (compressed) texture file not found. * @param loaderOptions options to be passed to the loader * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU). * @returns the cube texture as an InternalTexture */ createCubeTexture(e, t, r, n, i = null, s = null, a, f = null, o = !1, d = 0, v = 0, u = null, l, P = !1) { const p = u || new As(this, ri.Cube); p.isCube = !0, p.url = e, p.generateMipMaps = !n, p._lodGenerationScale = d, p._lodGenerationOffset = v, p._useSRGBBuffer = this._getUseSRGBBuffer(P, !!n), this._doNotHandleContextLost || (p._extension = f, p._files = r); const c = e.lastIndexOf("."); if ((f || (c > -1 ? e.substring(c).toLowerCase() : "")) === ".env") { const T = (q) => { const b = ry(q); p.width = b.width, p.height = b.width, iy(p, b); const j = b.specular; if (!j) throw new Error("Nothing else parsed so far"); p._lodGenerationScale = j.lodGenerationScale; const w = ny(q, b); p.format = 5, p.type = 0, p.generateMipMaps = !0, p.getEngine().updateTextureSamplingMode(We.TRILINEAR_SAMPLINGMODE, p), p._isRGBD = !0, p.invertY = !0, this._engine.loadCubeTextureWithMips(p._hardwareTexture.underlyingResource, w, !1, p._useSRGBBuffer, () => { p.isReady = !0, i && i(); }, () => { throw new Error("Could not load a native cube texture."); }); }; if (r && r.length === 6) throw new Error("Multi-file loading not allowed on env files."); { const q = (b, j) => { s && b && s(b.status + " " + b.statusText, j); }; this._loadFile(e, (b) => T(new Uint8Array(b)), void 0, void 0, !0, q); } } else { if (!r || r.length !== 6) throw new Error("Cannot load cubemap because 6 files were not defined"); const T = [r[0], r[3], r[1], r[4], r[2], r[5]]; Promise.all(T.map((q) => ye.LoadFileAsync(q).then((b) => new Uint8Array(b)))).then((q) => new Promise((b, j) => { this._engine.loadCubeTexture(p._hardwareTexture.underlyingResource, q, !n, !0, p._useSRGBBuffer, b, j); })).then(() => { p.isReady = !0, i && i(); }, (q) => { s && s(`Failed to load cubemap: ${q.message}`, q); }); } return this._internalTexturesCache.push(p), p; } /** @internal */ _createHardwareTexture() { return new XG(this._createTexture(), this._engine); } /** @internal */ _createHardwareRenderTargetWrapper(e, t, r) { const n = new Tue(e, t, r, this); return this._renderTargetWrapperCache.push(n), n; } /** @internal */ _createInternalTexture(e, t, r = !0, n = ri.Unknown) { var i, s, a; let f = !1, o = 0, d = 3, v = 5, u = !1, l = 1, P; t !== void 0 && typeof t == "object" ? (f = !!t.generateMipMaps, o = t.type === void 0 ? 0 : t.type, d = t.samplingMode === void 0 ? 3 : t.samplingMode, v = t.format === void 0 ? 5 : t.format, u = t.useSRGBBuffer === void 0 ? !1 : t.useSRGBBuffer, l = (i = t.samples) !== null && i !== void 0 ? i : 1, P = t.label) : f = !!t, u = this._getUseSRGBBuffer(u, !f), (o === 1 && !this._caps.textureFloatLinearFiltering || o === 2 && !this._caps.textureHalfFloatLinearFiltering) && (d = 1), o === 1 && !this._caps.textureFloat && (o = 0, Se.Warn("Float textures are not supported. Type forced to TEXTURETYPE_UNSIGNED_BYTE")); const p = new As(this, n), c = (s = e.width) !== null && s !== void 0 ? s : e, H = (a = e.height) !== null && a !== void 0 ? a : e, T = e.layers || 0; if (T !== 0) throw new Error("Texture layers are not supported in Babylon Native"); const q = p._hardwareTexture.underlyingResource, b = DE(v, o); return this._engine.initializeTexture(q, c, H, f, b, !0, u, l), this._setTextureSampling(q, AI(d)), p._useSRGBBuffer = u, p.baseWidth = c, p.baseHeight = H, p.width = c, p.height = H, p.depth = T, p.isReady = !0, p.samples = l, p.generateMipMaps = f, p.samplingMode = d, p.type = o, p.format = v, p.label = P, this._internalTexturesCache.push(p), p; } createRenderTargetTexture(e, t) { var r, n, i, s; const a = this._createHardwareRenderTargetWrapper(!1, !1, e); let f = !0, o = !1, d = !1, v, u = 1; t !== void 0 && typeof t == "object" && (f = (r = t.generateDepthBuffer) !== null && r !== void 0 ? r : !0, o = !!t.generateStencilBuffer, d = !!t.noColorAttachment, v = t.colorAttachment, u = (n = t.samples) !== null && n !== void 0 ? n : 1); const l = v || (d ? null : this._createInternalTexture(e, t, !0, ri.RenderTarget)), P = (i = e.width) !== null && i !== void 0 ? i : e, p = (s = e.height) !== null && s !== void 0 ? s : e, c = this._engine.createFrameBuffer(l ? l._hardwareTexture.underlyingResource : null, P, p, o, f, u); return a._framebuffer = c, a._generateDepthBuffer = f, a._generateStencilBuffer = o, a._samples = u, a.setTextures(l), a; } updateRenderTargetTextureSampleCount(e, t) { return Se.Warn("Updating render target sample count is not currently supported"), e.samples; } updateTextureSamplingMode(e, t) { if (t._hardwareTexture) { const r = AI(e); this._setTextureSampling(t._hardwareTexture.underlyingResource, r); } t.samplingMode = e; } bindFramebuffer(e, t, r, n, i) { const s = e; if (this._currentRenderTarget && this.unBindFramebuffer(this._currentRenderTarget), this._currentRenderTarget = e, t) throw new Error("Cuboid frame buffers are not yet supported in NativeEngine."); if (r || n) throw new Error("Required width/height for frame buffers not yet supported in NativeEngine."); s._framebufferDepthStencil ? this._bindUnboundFramebuffer(s._framebufferDepthStencil) : this._bindUnboundFramebuffer(s._framebuffer); } unBindFramebuffer(e, t = !1, r) { this._currentRenderTarget = null, r && r(), this._bindUnboundFramebuffer(null); } createDynamicVertexBuffer(e) { return this.createVertexBuffer(e, !0); } updateDynamicIndexBuffer(e, t, r = 0) { const n = e, i = this._normalizeIndexData(t); n.is32Bits = i.BYTES_PER_ELEMENT === 4, this._engine.updateDynamicIndexBuffer(n.nativeIndexBuffer, i.buffer, i.byteOffset, i.byteLength, r); } updateDynamicVertexBuffer(e, t, r, n) { const i = e, s = ArrayBuffer.isView(t) ? t : new Float32Array(t); this._engine.updateDynamicVertexBuffer(i.nativeVertexBuffer, s.buffer, s.byteOffset + (r ?? 0), n ?? s.byteLength); } // TODO: Refactor to share more logic with base Engine implementation. _setTexture(e, t, r = !1, n = !1) { const i = this._boundUniforms[e]; if (!i) return !1; if (!t) return this._boundTexturesCache[e] != null && (this._activeChannel = e, this._boundTexturesCache[e] = null), !1; if (t.video) this._activeChannel = e, t.update(); else if (t.delayLoadState === 4) return t.delayLoad(), !1; let s; return n ? s = t.depthStencilTexture : t.isReady() ? s = t.getInternalTexture() : t.isCube ? s = this.emptyCubeTexture : t.is3D ? s = this.emptyTexture3D : t.is2DArray ? s = this.emptyTexture2DArray : s = this.emptyTexture, this._activeChannel = e, !s || !s._hardwareTexture ? !1 : (this._setTextureWrapMode(s._hardwareTexture.underlyingResource, jE(t.wrapU), jE(t.wrapV), jE(t.wrapR)), this._updateAnisotropicLevel(t), this._setTextureCore(i, s._hardwareTexture.underlyingResource), !0); } // filter is a NativeFilter.XXXX value. _setTextureSampling(e, t) { this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETTEXTURESAMPLING), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsUInt32(t), this._commandBufferEncoder.finishEncodingCommand(); } // addressModes are NativeAddressMode.XXXX values. _setTextureWrapMode(e, t, r, n) { this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETTEXTUREWRAPMODE), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsUInt32(t), this._commandBufferEncoder.encodeCommandArgAsUInt32(r), this._commandBufferEncoder.encodeCommandArgAsUInt32(n), this._commandBufferEncoder.finishEncodingCommand(); } _setTextureCore(e, t) { this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETTEXTURE), this._commandBufferEncoder.encodeCommandArgAsNativeData(e), this._commandBufferEncoder.encodeCommandArgAsNativeData(t), this._commandBufferEncoder.finishEncodingCommand(); } // TODO: Share more of this logic with the base implementation. // TODO: Rename to match naming in base implementation once refactoring allows different parameters. _updateAnisotropicLevel(e) { const t = e.getInternalTexture(), r = e.anisotropicFilteringLevel; !t || !t._hardwareTexture || t._cachedAnisotropicFilteringLevel !== r && (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_SETTEXTUREANISOTROPICLEVEL), this._commandBufferEncoder.encodeCommandArgAsNativeData(t._hardwareTexture.underlyingResource), this._commandBufferEncoder.encodeCommandArgAsUInt32(r), this._commandBufferEncoder.finishEncodingCommand(), t._cachedAnisotropicFilteringLevel = r); } /** * @internal */ _bindTexture(e, t) { const r = this._boundUniforms[e]; if (r && t && t._hardwareTexture) { const n = t._hardwareTexture.underlyingResource; this._setTextureCore(r, n); } } _deleteBuffer(e) { e.nativeIndexBuffer && (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_DELETEINDEXBUFFER), this._commandBufferEncoder.encodeCommandArgAsNativeData(e.nativeIndexBuffer), this._commandBufferEncoder.finishEncodingCommand(), delete e.nativeIndexBuffer), e.nativeVertexBuffer && (this._commandBufferEncoder.startEncodingCommand(_native.Engine.COMMAND_DELETEVERTEXBUFFER), this._commandBufferEncoder.encodeCommandArgAsNativeData(e.nativeVertexBuffer), this._commandBufferEncoder.finishEncodingCommand(), delete e.nativeVertexBuffer); } /** * Create a canvas * @param width width * @param height height * @returns ICanvas interface */ createCanvas(e, t) { if (!_native.Canvas) throw new Error("Native Canvas plugin not available."); const r = new _native.Canvas(); return r.width = e, r.height = t, r; } /** * Create an image to use with canvas * @returns IImage interface */ createCanvasImage() { if (!_native.Canvas) throw new Error("Native Canvas plugin not available."); return new _native.Image(); } /** * Update a portion of an internal texture * @param texture defines the texture to update * @param imageData defines the data to store into the texture * @param xOffset defines the x coordinates of the update rectangle * @param yOffset defines the y coordinates of the update rectangle * @param width defines the width of the update rectangle * @param height defines the height of the update rectangle * @param faceIndex defines the face index if texture is a cube (0 by default) * @param lod defines the lod level to update (0 by default) * @param generateMipMaps defines whether to generate mipmaps or not */ updateTextureData(e, t, r, n, i, s, a = 0, f = 0, o = !1) { throw new Error("updateTextureData not implemented."); } /** * @internal */ _uploadCompressedDataToTextureDirectly(e, t, r, n, i, s = 0, a = 0) { throw new Error("_uploadCompressedDataToTextureDirectly not implemented."); } /** * @internal */ _uploadDataToTextureDirectly(e, t, r = 0, n = 0) { throw new Error("_uploadDataToTextureDirectly not implemented."); } /** * @internal */ _uploadArrayBufferViewToTexture(e, t, r = 0, n = 0) { throw new Error("_uploadArrayBufferViewToTexture not implemented."); } /** * @internal */ _uploadImageToTexture(e, t, r = 0, n = 0) { throw new Error("_uploadArrayBufferViewToTexture not implemented."); } getFontOffset(e) { return { ascent: 0, height: 0, descent: 0 }; } _readTexturePixels(e, t, r, n, i, s, a, f, o, d) { var v, u, l, P; if (n !== void 0 && n !== -1) throw new Error(`Reading cubemap faces is not supported, but faceIndex is ${n}.`); return this._engine.readTexture((v = e._hardwareTexture) === null || v === void 0 ? void 0 : v.underlyingResource, i ?? 0, o ?? 0, d ?? 0, t, r, (u = s == null ? void 0 : s.buffer) !== null && u !== void 0 ? u : null, (l = s == null ? void 0 : s.byteOffset) !== null && l !== void 0 ? l : 0, (P = s == null ? void 0 : s.byteLength) !== null && P !== void 0 ? P : 0).then((p) => (s || (s = new Uint8Array(p)), s)); } } Hm.PROTOCOL_VERSION = 8; Hm._createNativeDataStream = function() { return _native.NativeDataStream.VALIDATION_ENABLED ? new Xte() : new hD(); }; class Xte extends hD { constructor() { super(); } writeUint32(e) { super.writeUint32(_native.NativeDataStream.VALIDATION_UINT_32), super.writeUint32(e); } writeInt32(e) { super.writeUint32(_native.NativeDataStream.VALIDATION_INT_32), super.writeInt32(e); } writeFloat32(e) { super.writeUint32(_native.NativeDataStream.VALIDATION_FLOAT_32), super.writeFloat32(e); } writeUint32Array(e) { super.writeUint32(_native.NativeDataStream.VALIDATION_UINT_32_ARRAY), super.writeUint32Array(e); } writeInt32Array(e) { super.writeUint32(_native.NativeDataStream.VALIDATION_INT_32_ARRAY), super.writeInt32Array(e); } writeFloat32Array(e) { super.writeUint32(_native.NativeDataStream.VALIDATION_FLOAT_32_ARRAY), super.writeFloat32Array(e); } writeNativeData(e) { super.writeUint32(_native.NativeDataStream.VALIDATION_NATIVE_DATA), super.writeNativeData(e); } writeBoolean(e) { super.writeUint32(_native.NativeDataStream.VALIDATION_BOOLEAN), super.writeBoolean(e); } } var sF; (function(A) { A.LowPower = "low-power", A.HighPerformance = "high-performance"; })(sF || (sF = {})); var k2; (function(A) { A.DepthClipControl = "depth-clip-control", A.Depth32FloatStencil8 = "depth32float-stencil8", A.TextureCompressionBC = "texture-compression-bc", A.TextureCompressionETC2 = "texture-compression-etc2", A.TextureCompressionASTC = "texture-compression-astc", A.TimestampQuery = "timestamp-query", A.IndirectFirstInstance = "indirect-first-instance", A.ShaderF16 = "shader-f16", A.RG11B10UFloatRenderable = "rg11b10ufloat-renderable", A.BGRA8UnormStorage = "bgra8unorm-storage", A.Float32Filterable = "float32-filterable"; })(k2 || (k2 = {})); var aF; (function(A) { A.Unmapped = "unmapped", A.Pending = "pending", A.Mapped = "mapped"; })(aF || (aF = {})); var ga; (function(A) { A[A.MapRead = 1] = "MapRead", A[A.MapWrite = 2] = "MapWrite", A[A.CopySrc = 4] = "CopySrc", A[A.CopyDst = 8] = "CopyDst", A[A.Index = 16] = "Index", A[A.Vertex = 32] = "Vertex", A[A.Uniform = 64] = "Uniform", A[A.Storage = 128] = "Storage", A[A.Indirect = 256] = "Indirect", A[A.QueryResolve = 512] = "QueryResolve"; })(ga || (ga = {})); var vD; (function(A) { A[A.Read = 1] = "Read", A[A.Write = 2] = "Write"; })(vD || (vD = {})); var Hp; (function(A) { A.E1d = "1d", A.E2d = "2d", A.E3d = "3d"; })(Hp || (Hp = {})); var Po; (function(A) { A[A.CopySrc = 1] = "CopySrc", A[A.CopyDst = 2] = "CopyDst", A[A.TextureBinding = 4] = "TextureBinding", A[A.StorageBinding = 8] = "StorageBinding", A[A.RenderAttachment = 16] = "RenderAttachment"; })(Po || (Po = {})); var Da; (function(A) { A.E1d = "1d", A.E2d = "2d", A.E2dArray = "2d-array", A.Cube = "cube", A.CubeArray = "cube-array", A.E3d = "3d"; })(Da || (Da = {})); var zH; (function(A) { A.All = "all", A.StencilOnly = "stencil-only", A.DepthOnly = "depth-only"; })(zH || (zH = {})); var we; (function(A) { A.R8Unorm = "r8unorm", A.R8Snorm = "r8snorm", A.R8Uint = "r8uint", A.R8Sint = "r8sint", A.R16Uint = "r16uint", A.R16Sint = "r16sint", A.R16Float = "r16float", A.RG8Unorm = "rg8unorm", A.RG8Snorm = "rg8snorm", A.RG8Uint = "rg8uint", A.RG8Sint = "rg8sint", A.R32Uint = "r32uint", A.R32Sint = "r32sint", A.R32Float = "r32float", A.RG16Uint = "rg16uint", A.RG16Sint = "rg16sint", A.RG16Float = "rg16float", A.RGBA8Unorm = "rgba8unorm", A.RGBA8UnormSRGB = "rgba8unorm-srgb", A.RGBA8Snorm = "rgba8snorm", A.RGBA8Uint = "rgba8uint", A.RGBA8Sint = "rgba8sint", A.BGRA8Unorm = "bgra8unorm", A.BGRA8UnormSRGB = "bgra8unorm-srgb", A.RGB9E5UFloat = "rgb9e5ufloat", A.RGB10A2UINT = "rgb10a2uint", A.RGB10A2Unorm = "rgb10a2unorm", A.RG11B10UFloat = "rg11b10ufloat", A.RG32Uint = "rg32uint", A.RG32Sint = "rg32sint", A.RG32Float = "rg32float", A.RGBA16Uint = "rgba16uint", A.RGBA16Sint = "rgba16sint", A.RGBA16Float = "rgba16float", A.RGBA32Uint = "rgba32uint", A.RGBA32Sint = "rgba32sint", A.RGBA32Float = "rgba32float", A.Stencil8 = "stencil8", A.Depth16Unorm = "depth16unorm", A.Depth24Plus = "depth24plus", A.Depth24PlusStencil8 = "depth24plus-stencil8", A.Depth32Float = "depth32float", A.BC1RGBAUnorm = "bc1-rgba-unorm", A.BC1RGBAUnormSRGB = "bc1-rgba-unorm-srgb", A.BC2RGBAUnorm = "bc2-rgba-unorm", A.BC2RGBAUnormSRGB = "bc2-rgba-unorm-srgb", A.BC3RGBAUnorm = "bc3-rgba-unorm", A.BC3RGBAUnormSRGB = "bc3-rgba-unorm-srgb", A.BC4RUnorm = "bc4-r-unorm", A.BC4RSnorm = "bc4-r-snorm", A.BC5RGUnorm = "bc5-rg-unorm", A.BC5RGSnorm = "bc5-rg-snorm", A.BC6HRGBUFloat = "bc6h-rgb-ufloat", A.BC6HRGBFloat = "bc6h-rgb-float", A.BC7RGBAUnorm = "bc7-rgba-unorm", A.BC7RGBAUnormSRGB = "bc7-rgba-unorm-srgb", A.ETC2RGB8Unorm = "etc2-rgb8unorm", A.ETC2RGB8UnormSRGB = "etc2-rgb8unorm-srgb", A.ETC2RGB8A1Unorm = "etc2-rgb8a1unorm", A.ETC2RGB8A1UnormSRGB = "etc2-rgb8a1unorm-srgb", A.ETC2RGBA8Unorm = "etc2-rgba8unorm", A.ETC2RGBA8UnormSRGB = "etc2-rgba8unorm-srgb", A.EACR11Unorm = "eac-r11unorm", A.EACR11Snorm = "eac-r11snorm", A.EACRG11Unorm = "eac-rg11unorm", A.EACRG11Snorm = "eac-rg11snorm", A.ASTC4x4Unorm = "astc-4x4-unorm", A.ASTC4x4UnormSRGB = "astc-4x4-unorm-srgb", A.ASTC5x4Unorm = "astc-5x4-unorm", A.ASTC5x4UnormSRGB = "astc-5x4-unorm-srgb", A.ASTC5x5Unorm = "astc-5x5-unorm", A.ASTC5x5UnormSRGB = "astc-5x5-unorm-srgb", A.ASTC6x5Unorm = "astc-6x5-unorm", A.ASTC6x5UnormSRGB = "astc-6x5-unorm-srgb", A.ASTC6x6Unorm = "astc-6x6-unorm", A.ASTC6x6UnormSRGB = "astc-6x6-unorm-srgb", A.ASTC8x5Unorm = "astc-8x5-unorm", A.ASTC8x5UnormSRGB = "astc-8x5-unorm-srgb", A.ASTC8x6Unorm = "astc-8x6-unorm", A.ASTC8x6UnormSRGB = "astc-8x6-unorm-srgb", A.ASTC8x8Unorm = "astc-8x8-unorm", A.ASTC8x8UnormSRGB = "astc-8x8-unorm-srgb", A.ASTC10x5Unorm = "astc-10x5-unorm", A.ASTC10x5UnormSRGB = "astc-10x5-unorm-srgb", A.ASTC10x6Unorm = "astc-10x6-unorm", A.ASTC10x6UnormSRGB = "astc-10x6-unorm-srgb", A.ASTC10x8Unorm = "astc-10x8-unorm", A.ASTC10x8UnormSRGB = "astc-10x8-unorm-srgb", A.ASTC10x10Unorm = "astc-10x10-unorm", A.ASTC10x10UnormSRGB = "astc-10x10-unorm-srgb", A.ASTC12x10Unorm = "astc-12x10-unorm", A.ASTC12x10UnormSRGB = "astc-12x10-unorm-srgb", A.ASTC12x12Unorm = "astc-12x12-unorm", A.ASTC12x12UnormSRGB = "astc-12x12-unorm-srgb", A.Depth32FloatStencil8 = "depth32float-stencil8"; })(we || (we = {})); var zw; (function(A) { A.ClampToEdge = "clamp-to-edge", A.Repeat = "repeat", A.MirrorRepeat = "mirror-repeat"; })(zw || (zw = {})); var Ji; (function(A) { A.Nearest = "nearest", A.Linear = "linear"; })(Ji || (Ji = {})); var oF; (function(A) { A.Nearest = "nearest", A.Linear = "linear"; })(oF || (oF = {})); var q6; (function(A) { A.Never = "never", A.Less = "less", A.Equal = "equal", A.LessEqual = "less-equal", A.Greater = "greater", A.NotEqual = "not-equal", A.GreaterEqual = "greater-equal", A.Always = "always"; })(q6 || (q6 = {})); var G8; (function(A) { A[A.Vertex = 1] = "Vertex", A[A.Fragment = 2] = "Fragment", A[A.Compute = 4] = "Compute"; })(G8 || (G8 = {})); var dq; (function(A) { A.Uniform = "uniform", A.Storage = "storage", A.ReadOnlyStorage = "read-only-storage"; })(dq || (dq = {})); var Xq; (function(A) { A.Filtering = "filtering", A.NonFiltering = "non-filtering", A.Comparison = "comparison"; })(Xq || (Xq = {})); var Ll; (function(A) { A.Float = "float", A.UnfilterableFloat = "unfilterable-float", A.Depth = "depth", A.Sint = "sint", A.Uint = "uint"; })(Ll || (Ll = {})); var eO; (function(A) { A.WriteOnly = "write-only"; })(eO || (eO = {})); var fF; (function(A) { A.Error = "error", A.Warning = "warning", A.Info = "info"; })(fF || (fF = {})); var AF; (function(A) { A.Validation = "validation", A.Internal = "internal"; })(AF || (AF = {})); var SS; (function(A) { A.Auto = "auto"; })(SS || (SS = {})); var Fl; (function(A) { A.PointList = "point-list", A.LineList = "line-list", A.LineStrip = "line-strip", A.TriangleList = "triangle-list", A.TriangleStrip = "triangle-strip"; })(Fl || (Fl = {})); var MI; (function(A) { A.CCW = "ccw", A.CW = "cw"; })(MI || (MI = {})); var eS; (function(A) { A.None = "none", A.Front = "front", A.Back = "back"; })(eS || (eS = {})); var dF; (function(A) { A[A.Red = 1] = "Red", A[A.Green = 2] = "Green", A[A.Blue = 4] = "Blue", A[A.Alpha = 8] = "Alpha", A[A.All = 15] = "All"; })(dF || (dF = {})); var ad; (function(A) { A.Zero = "zero", A.One = "one", A.Src = "src", A.OneMinusSrc = "one-minus-src", A.SrcAlpha = "src-alpha", A.OneMinusSrcAlpha = "one-minus-src-alpha", A.Dst = "dst", A.OneMinusDst = "one-minus-dst", A.DstAlpha = "dst-alpha", A.OneMinusDstAlpha = "one-minus-dst-alpha", A.SrcAlphaSaturated = "src-alpha-saturated", A.Constant = "constant", A.OneMinusConstant = "one-minus-constant"; })(ad || (ad = {})); var Z8; (function(A) { A.Add = "add", A.Subtract = "subtract", A.ReverseSubtract = "reverse-subtract", A.Min = "min", A.Max = "max"; })(Z8 || (Z8 = {})); var T0; (function(A) { A.Keep = "keep", A.Zero = "zero", A.Replace = "replace", A.Invert = "invert", A.IncrementClamp = "increment-clamp", A.DecrementClamp = "decrement-clamp", A.IncrementWrap = "increment-wrap", A.DecrementWrap = "decrement-wrap"; })(T0 || (T0 = {})); var Tq; (function(A) { A.Uint16 = "uint16", A.Uint32 = "uint32"; })(Tq || (Tq = {})); var jo; (function(A) { A.Uint8x2 = "uint8x2", A.Uint8x4 = "uint8x4", A.Sint8x2 = "sint8x2", A.Sint8x4 = "sint8x4", A.Unorm8x2 = "unorm8x2", A.Unorm8x4 = "unorm8x4", A.Snorm8x2 = "snorm8x2", A.Snorm8x4 = "snorm8x4", A.Uint16x2 = "uint16x2", A.Uint16x4 = "uint16x4", A.Sint16x2 = "sint16x2", A.Sint16x4 = "sint16x4", A.Unorm16x2 = "unorm16x2", A.Unorm16x4 = "unorm16x4", A.Snorm16x2 = "snorm16x2", A.Snorm16x4 = "snorm16x4", A.Float16x2 = "float16x2", A.Float16x4 = "float16x4", A.Float32 = "float32", A.Float32x2 = "float32x2", A.Float32x3 = "float32x3", A.Float32x4 = "float32x4", A.Uint32 = "uint32", A.Uint32x2 = "uint32x2", A.Uint32x3 = "uint32x3", A.Uint32x4 = "uint32x4", A.Sint32 = "sint32", A.Sint32x2 = "sint32x2", A.Sint32x3 = "sint32x3", A.Sint32x4 = "sint32x4", A.UNORM10x10x10x2 = "unorm10-10-10-2"; })(jo || (jo = {})); var LI; (function(A) { A.Vertex = "vertex", A.Instance = "instance"; })(LI || (LI = {})); var vF; (function(A) { A.Beginning = "beginning", A.End = "end"; })(vF || (vF = {})); var uF; (function(A) { A.Beginning = "beginning", A.End = "end"; })(uF || (uF = {})); var f9; (function(A) { A.Load = "load", A.Clear = "clear"; })(f9 || (f9 = {})); var Hc; (function(A) { A.Store = "store", A.Discard = "discard"; })(Hc || (Hc = {})); var KI; (function(A) { A.Occlusion = "occlusion", A.Timestamp = "timestamp"; })(KI || (KI = {})); var JI; (function(A) { A.Opaque = "opaque", A.Premultiplied = "premultiplied"; })(JI || (JI = {})); var lF; (function(A) { A.Unknown = "unknown", A.Destroyed = "destroyed"; })(lF || (lF = {})); var PF; (function(A) { A.Validation = "validation", A.OutOfMemory = "out-of-memory", A.Internal = "internal"; })(PF || (PF = {})); class Wo { constructor() { this.shaderLanguage = za.GLSL, this.vertexBufferKindToNumberOfComponents = {}; } _addUniformToLeftOverUBO(e, t, r) { let n = 0; [e, t, n] = this._getArraySize(e, t, r); for (let i = 0; i < this._webgpuProcessingContext.leftOverUniforms.length; i++) if (this._webgpuProcessingContext.leftOverUniforms[i].name === e) return; this._webgpuProcessingContext.leftOverUniforms.push({ name: e, type: t, length: n }); } _buildLeftOverUBO() { if (!this._webgpuProcessingContext.leftOverUniforms.length) return ""; const e = Wo.LeftOvertUBOName; let t = this._webgpuProcessingContext.availableBuffers[e]; return t || (t = { binding: this._webgpuProcessingContext.getNextFreeUBOBinding() }, this._webgpuProcessingContext.availableBuffers[e] = t, this._addBufferBindingDescription(e, t, dq.Uniform, !0), this._addBufferBindingDescription(e, t, dq.Uniform, !1)), this._generateLeftOverUBOCode(e, t); } _collectBindingNames() { for (let e = 0; e < this._webgpuProcessingContext.bindGroupLayoutEntries.length; e++) { const t = this._webgpuProcessingContext.bindGroupLayoutEntries[e]; if (t === void 0) { this._webgpuProcessingContext.bindGroupLayoutEntries[e] = []; continue; } for (let r = 0; r < t.length; r++) { const n = this._webgpuProcessingContext.bindGroupLayoutEntries[e][r], i = this._webgpuProcessingContext.bindGroupLayoutEntryInfo[e][n.binding].name, s = this._webgpuProcessingContext.bindGroupLayoutEntryInfo[e][n.binding].nameInArrayOfTexture; n && (n.texture || n.externalTexture || n.storageTexture ? this._webgpuProcessingContext.textureNames.push(s) : n.sampler ? this._webgpuProcessingContext.samplerNames.push(i) : n.buffer && this._webgpuProcessingContext.bufferNames.push(i)); } } } _preCreateBindGroupEntries() { const e = this._webgpuProcessingContext.bindGroupEntries; for (let t = 0; t < this._webgpuProcessingContext.bindGroupLayoutEntries.length; t++) { const r = this._webgpuProcessingContext.bindGroupLayoutEntries[t], n = []; for (let i = 0; i < r.length; i++) { const s = this._webgpuProcessingContext.bindGroupLayoutEntries[t][i]; s.sampler || s.texture || s.storageTexture || s.externalTexture ? n.push({ binding: s.binding, resource: void 0 }) : s.buffer && n.push({ binding: s.binding, resource: { buffer: void 0, offset: 0, size: 0 } }); } e[t] = n; } } _addTextureBindingDescription(e, t, r, n, i, s) { let { groupIndex: a, bindingIndex: f } = t.textures[r]; if (this._webgpuProcessingContext.bindGroupLayoutEntries[a] || (this._webgpuProcessingContext.bindGroupLayoutEntries[a] = [], this._webgpuProcessingContext.bindGroupLayoutEntryInfo[a] = []), !this._webgpuProcessingContext.bindGroupLayoutEntryInfo[a][f]) { let o; n === null ? o = this._webgpuProcessingContext.bindGroupLayoutEntries[a].push({ binding: f, visibility: 0, externalTexture: {} }) : i ? o = this._webgpuProcessingContext.bindGroupLayoutEntries[a].push({ binding: f, visibility: 0, storageTexture: { access: eO.WriteOnly, format: i, viewDimension: n } }) : o = this._webgpuProcessingContext.bindGroupLayoutEntries[a].push({ binding: f, visibility: 0, texture: { sampleType: t.sampleType, viewDimension: n, multisampled: !1 } }); const d = t.isTextureArray ? e + r : e; this._webgpuProcessingContext.bindGroupLayoutEntryInfo[a][f] = { name: e, index: o - 1, nameInArrayOfTexture: d }; } f = this._webgpuProcessingContext.bindGroupLayoutEntryInfo[a][f].index, s ? this._webgpuProcessingContext.bindGroupLayoutEntries[a][f].visibility |= G8.Vertex : this._webgpuProcessingContext.bindGroupLayoutEntries[a][f].visibility |= G8.Fragment; } _addSamplerBindingDescription(e, t, r) { let { groupIndex: n, bindingIndex: i } = t.binding; if (this._webgpuProcessingContext.bindGroupLayoutEntries[n] || (this._webgpuProcessingContext.bindGroupLayoutEntries[n] = [], this._webgpuProcessingContext.bindGroupLayoutEntryInfo[n] = []), !this._webgpuProcessingContext.bindGroupLayoutEntryInfo[n][i]) { const s = this._webgpuProcessingContext.bindGroupLayoutEntries[n].push({ binding: i, visibility: 0, sampler: { type: t.type } }); this._webgpuProcessingContext.bindGroupLayoutEntryInfo[n][i] = { name: e, index: s - 1 }; } i = this._webgpuProcessingContext.bindGroupLayoutEntryInfo[n][i].index, r ? this._webgpuProcessingContext.bindGroupLayoutEntries[n][i].visibility |= G8.Vertex : this._webgpuProcessingContext.bindGroupLayoutEntries[n][i].visibility |= G8.Fragment; } _addBufferBindingDescription(e, t, r, n) { let { groupIndex: i, bindingIndex: s } = t.binding; if (this._webgpuProcessingContext.bindGroupLayoutEntries[i] || (this._webgpuProcessingContext.bindGroupLayoutEntries[i] = [], this._webgpuProcessingContext.bindGroupLayoutEntryInfo[i] = []), !this._webgpuProcessingContext.bindGroupLayoutEntryInfo[i][s]) { const a = this._webgpuProcessingContext.bindGroupLayoutEntries[i].push({ binding: s, visibility: 0, buffer: { type: r } }); this._webgpuProcessingContext.bindGroupLayoutEntryInfo[i][s] = { name: e, index: a - 1 }; } s = this._webgpuProcessingContext.bindGroupLayoutEntryInfo[i][s].index, n ? this._webgpuProcessingContext.bindGroupLayoutEntries[i][s].visibility |= G8.Vertex : this._webgpuProcessingContext.bindGroupLayoutEntries[i][s].visibility |= G8.Fragment; } _injectStartingAndEndingCode(e, t, r, n) { let i = e.indexOf(t); if (i < 0) return console.error('No "main" function found in shader code! Processing aborted.'), e; if (r) { for (; i++ < e.length && e.charAt(i) != "{"; ) ; if (i < e.length) { const s = e.substring(0, i + 1), a = e.substring(i + 1); e = s + r + a; } } if (n) { const s = e.lastIndexOf("}"); e = e.substring(0, s), e += n + ` }`; } return e; } } Wo.AutoSamplerSuffix = "Sampler"; Wo.LeftOvertUBOName = "LeftOver"; Wo.InternalsUBOName = "Internals"; Wo.UniformSizes = { // GLSL types bool: 1, int: 1, float: 1, vec2: 2, ivec2: 2, uvec2: 2, vec3: 3, ivec3: 3, uvec3: 3, vec4: 4, ivec4: 4, uvec4: 4, mat2: 4, mat3: 12, mat4: 16, // WGSL types i32: 1, u32: 1, f32: 1, mat2x2: 4, mat3x3: 12, mat4x4: 16 }; Wo._SamplerFunctionByWebGLSamplerType = { sampler2D: "sampler2D", sampler2DArray: "sampler2DArray", sampler2DShadow: "sampler2DShadow", sampler2DArrayShadow: "sampler2DArrayShadow", samplerCube: "samplerCube", sampler3D: "sampler3D" }; Wo._TextureTypeByWebGLSamplerType = { sampler2D: "texture2D", sampler2DArray: "texture2DArray", sampler2DShadow: "texture2D", sampler2DArrayShadow: "texture2DArray", samplerCube: "textureCube", samplerCubeArray: "textureCubeArray", sampler3D: "texture3D" }; Wo._GpuTextureViewDimensionByWebGPUTextureType = { textureCube: Da.Cube, textureCubeArray: Da.CubeArray, texture2D: Da.E2d, texture2DArray: Da.E2dArray, texture3D: Da.E3d }; Wo._SamplerTypeByWebGLSamplerType = { sampler2DShadow: "samplerShadow", sampler2DArrayShadow: "samplerShadow" }; Wo._IsComparisonSamplerByWebGPUSamplerType = { samplerShadow: !0, samplerArrayShadow: !0, sampler: !1 }; class Bue { get isAsync() { return !1; } get isReady() { return !!this.stages; } constructor(e, t) { this.bindGroupLayouts = {}, this._name = "unnamed", this.shaderProcessingContext = e, this._leftOverUniformsByName = {}, this.engine = t, this.vertexBufferKindToType = {}; } _handlesSpectorRebuildCallback() { } _fillEffectInformation(e, t, r, n, i, s, a, f) { const o = this.engine; e._fragmentSourceCode = "", e._vertexSourceCode = ""; const d = this.shaderProcessingContext.availableTextures; let v; for (v = 0; v < i.length; v++) { const P = i[v], p = d[i[v]]; p == null || p == null ? (i.splice(v, 1), v--) : s[P] = v; } for (const P of o.getAttributes(this, a)) f.push(P); this.buildUniformLayout(); const u = [], l = []; for (v = 0; v < a.length; v++) { const P = f[v]; P >= 0 && (u.push(a[v]), l.push(P)); } this.shaderProcessingContext.attributeNamesFromEffect = u, this.shaderProcessingContext.attributeLocationsFromEffect = l; } /** @internal */ /** * Build the uniform buffer used in the material. */ buildUniformLayout() { if (this.shaderProcessingContext.leftOverUniforms.length) { this.uniformBuffer = new yr(this.engine, void 0, void 0, "leftOver-" + this._name); for (const e of this.shaderProcessingContext.leftOverUniforms) { const t = e.type.replace(/^(.*?)(<.*>)?$/, "$1"), r = Wo.UniformSizes[t]; this.uniformBuffer.addUniform(e.name, r, e.length), this._leftOverUniformsByName[e.name] = e.type; } this.uniformBuffer.create(); } } /** * Release all associated resources. **/ dispose() { this.uniformBuffer && this.uniformBuffer.dispose(); } /** * Sets an integer value on a uniform variable. * @param uniformName Name of the variable. * @param value Value to be set. */ setInt(e, t) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateInt(e, t); } /** * Sets an int2 value on a uniform variable. * @param uniformName Name of the variable. * @param x First int in int2. * @param y Second int in int2. */ setInt2(e, t, r) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateInt2(e, t, r); } /** * Sets an int3 value on a uniform variable. * @param uniformName Name of the variable. * @param x First int in int3. * @param y Second int in int3. * @param z Third int in int3. */ setInt3(e, t, r, n) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateInt3(e, t, r, n); } /** * Sets an int4 value on a uniform variable. * @param uniformName Name of the variable. * @param x First int in int4. * @param y Second int in int4. * @param z Third int in int4. * @param w Fourth int in int4. */ setInt4(e, t, r, n, i) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateInt4(e, t, r, n, i); } /** * Sets an int array on a uniform variable. * @param uniformName Name of the variable. * @param array array to be set. */ setIntArray(e, t) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateIntArray(e, t); } /** * Sets an int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setIntArray2(e, t) { this.setIntArray(e, t); } /** * Sets an int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setIntArray3(e, t) { this.setIntArray(e, t); } /** * Sets an int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setIntArray4(e, t) { this.setIntArray(e, t); } /** * Sets an unsigned integer value on a uniform variable. * @param uniformName Name of the variable. * @param value Value to be set. */ setUInt(e, t) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateUInt(e, t); } /** * Sets an unsigned int2 value on a uniform variable. * @param uniformName Name of the variable. * @param x First unsigned int in uint2. * @param y Second unsigned int in uint2. */ setUInt2(e, t, r) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateUInt2(e, t, r); } /** * Sets an unsigned int3 value on a uniform variable. * @param uniformName Name of the variable. * @param x First unsigned int in uint3. * @param y Second unsigned int in uint3. * @param z Third unsigned int in uint3. */ setUInt3(e, t, r, n) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateUInt3(e, t, r, n); } /** * Sets an unsigned int4 value on a uniform variable. * @param uniformName Name of the variable. * @param x First unsigned int in uint4. * @param y Second unsigned int in uint4. * @param z Third unsigned int in uint4. * @param w Fourth unsigned int in uint4. */ setUInt4(e, t, r, n, i) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateUInt4(e, t, r, n, i); } /** * Sets an unsigned int array on a uniform variable. * @param uniformName Name of the variable. * @param array array to be set. */ setUIntArray(e, t) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateUIntArray(e, t); } /** * Sets an unsigned int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setUIntArray2(e, t) { this.setUIntArray(e, t); } /** * Sets an unsigned int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setUIntArray3(e, t) { this.setUIntArray(e, t); } /** * Sets an unsigned int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setUIntArray4(e, t) { this.setUIntArray(e, t); } /** * Sets an array on a uniform variable. * @param uniformName Name of the variable. * @param array array to be set. */ setArray(e, t) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateArray(e, t); } /** * Sets an array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setArray2(e, t) { this.setArray(e, t); } /** * Sets an array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. * @returns this effect. */ setArray3(e, t) { this.setArray(e, t); } /** * Sets an array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader) * @param uniformName Name of the variable. * @param array array to be set. */ setArray4(e, t) { this.setArray(e, t); } /** * Sets matrices on a uniform variable. * @param uniformName Name of the variable. * @param matrices matrices to be set. */ setMatrices(e, t) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateMatrices(e, t); } /** * Sets matrix on a uniform variable. * @param uniformName Name of the variable. * @param matrix matrix to be set. */ setMatrix(e, t) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateMatrix(e, t); } /** * Sets a 3x3 matrix on a uniform variable. (Specified as [1,2,3,4,5,6,7,8,9] will result in [1,2,3][4,5,6][7,8,9] matrix) * @param uniformName Name of the variable. * @param matrix matrix to be set. */ setMatrix3x3(e, t) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateMatrix3x3(e, t); } /** * Sets a 2x2 matrix on a uniform variable. (Specified as [1,2,3,4] will result in [1,2][3,4] matrix) * @param uniformName Name of the variable. * @param matrix matrix to be set. */ setMatrix2x2(e, t) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateMatrix2x2(e, t); } /** * Sets a float on a uniform variable. * @param uniformName Name of the variable. * @param value value to be set. * @returns this effect. */ setFloat(e, t) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateFloat(e, t); } /** * Sets a Vector2 on a uniform variable. * @param uniformName Name of the variable. * @param vector2 vector2 to be set. */ setVector2(e, t) { this.setFloat2(e, t.x, t.y); } /** * Sets a float2 on a uniform variable. * @param uniformName Name of the variable. * @param x First float in float2. * @param y Second float in float2. */ setFloat2(e, t, r) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateFloat2(e, t, r); } /** * Sets a Vector3 on a uniform variable. * @param uniformName Name of the variable. * @param vector3 Value to be set. */ setVector3(e, t) { this.setFloat3(e, t.x, t.y, t.z); } /** * Sets a float3 on a uniform variable. * @param uniformName Name of the variable. * @param x First float in float3. * @param y Second float in float3. * @param z Third float in float3. */ setFloat3(e, t, r, n) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateFloat3(e, t, r, n); } /** * Sets a Vector4 on a uniform variable. * @param uniformName Name of the variable. * @param vector4 Value to be set. */ setVector4(e, t) { this.setFloat4(e, t.x, t.y, t.z, t.w); } /** * Sets a Quaternion on a uniform variable. * @param uniformName Name of the variable. * @param quaternion Value to be set. */ setQuaternion(e, t) { this.setFloat4(e, t.x, t.y, t.z, t.w); } /** * Sets a float4 on a uniform variable. * @param uniformName Name of the variable. * @param x First float in float4. * @param y Second float in float4. * @param z Third float in float4. * @param w Fourth float in float4. * @returns this effect. */ setFloat4(e, t, r, n, i) { !this.uniformBuffer || !this._leftOverUniformsByName[e] || this.uniformBuffer.updateFloat4(e, t, r, n, i); } /** * Sets a Color3 on a uniform variable. * @param uniformName Name of the variable. * @param color3 Value to be set. */ setColor3(e, t) { this.setFloat3(e, t.r, t.g, t.b); } /** * Sets a Color4 on a uniform variable. * @param uniformName Name of the variable. * @param color3 Value to be set. * @param alpha Alpha value to be set. */ setColor4(e, t, r) { this.setFloat4(e, t.r, t.g, t.b, r); } /** * Sets a Color4 on a uniform variable * @param uniformName defines the name of the variable * @param color4 defines the value to be set */ setDirectColor4(e, t) { this.setFloat4(e, t.r, t.g, t.b, t.a); } _getVertexShaderCode() { var e; return (e = this.sources) === null || e === void 0 ? void 0 : e.vertex; } _getFragmentShaderCode() { var e; return (e = this.sources) === null || e === void 0 ? void 0 : e.fragment; } } const Wue = 4, Sue = 65536, qG = { // GLSL types mat2: 2, mat3: 3, mat4: 4, // WGSL types mat2x2: 2, mat3x3: 3, mat4x4: 4 }; class gp { static get KnownUBOs() { return gp._SimplifiedKnownBindings ? gp._SimplifiedKnownUBOs : gp._KnownUBOs; } constructor(e) { this.shaderLanguage = e, this._attributeNextLocation = 0, this._varyingNextLocation = 0, this.freeGroupIndex = 0, this.freeBindingIndex = 0, this.availableVaryings = {}, this.availableAttributes = {}, this.availableBuffers = {}, this.availableTextures = {}, this.availableSamplers = {}, this.orderedAttributes = [], this.bindGroupLayoutEntries = [], this.bindGroupLayoutEntryInfo = [], this.bindGroupEntries = [], this.bufferNames = [], this.textureNames = [], this.samplerNames = [], this.leftOverUniforms = [], this._findStartingGroupBinding(); } _findStartingGroupBinding() { const e = gp.KnownUBOs, t = []; for (const r in e) { const n = e[r].binding; n.groupIndex !== -1 && (t[n.groupIndex] === void 0 ? t[n.groupIndex] = n.bindingIndex : t[n.groupIndex] = Math.max(t[n.groupIndex], n.bindingIndex)); } this.freeGroupIndex = t.length - 1, this.freeGroupIndex === 0 ? (this.freeGroupIndex++, this.freeBindingIndex = 0) : this.freeBindingIndex = t[t.length - 1] + 1; } getAttributeNextLocation(e, t = 0) { var r; const n = this._attributeNextLocation; return this._attributeNextLocation += ((r = qG[e]) !== null && r !== void 0 ? r : 1) * (t || 1), n; } getVaryingNextLocation(e, t = 0) { var r; const n = this._varyingNextLocation; return this._varyingNextLocation += ((r = qG[e]) !== null && r !== void 0 ? r : 1) * (t || 1), n; } getNextFreeUBOBinding() { return this._getNextFreeBinding(1); } _getNextFreeBinding(e) { if (this.freeBindingIndex > Sue - e && (this.freeGroupIndex++, this.freeBindingIndex = 0), this.freeGroupIndex === Wue) throw "Too many textures or UBOs have been declared and it is not supported in WebGPU."; const t = { groupIndex: this.freeGroupIndex, bindingIndex: this.freeBindingIndex }; return this.freeBindingIndex += e, t; } } gp._SimplifiedKnownBindings = !0; gp._SimplifiedKnownUBOs = { Scene: { binding: { groupIndex: 0, bindingIndex: 0 } }, Light0: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light1: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light2: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light3: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light4: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light5: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light6: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light7: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light8: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light9: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light10: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light11: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light12: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light13: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light14: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light15: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light16: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light17: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light18: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light19: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light20: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light21: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light22: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light23: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light24: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light25: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light26: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light27: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light28: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light29: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light30: { binding: { groupIndex: -1, bindingIndex: -1 } }, Light31: { binding: { groupIndex: -1, bindingIndex: -1 } }, Material: { binding: { groupIndex: -1, bindingIndex: -1 } }, Mesh: { binding: { groupIndex: -1, bindingIndex: -1 } }, Internals: { binding: { groupIndex: -1, bindingIndex: -1 } } }; gp._KnownUBOs = { Scene: { binding: { groupIndex: 0, bindingIndex: 0 } }, Light0: { binding: { groupIndex: 1, bindingIndex: 0 } }, Light1: { binding: { groupIndex: 1, bindingIndex: 1 } }, Light2: { binding: { groupIndex: 1, bindingIndex: 2 } }, Light3: { binding: { groupIndex: 1, bindingIndex: 3 } }, Light4: { binding: { groupIndex: 1, bindingIndex: 4 } }, Light5: { binding: { groupIndex: 1, bindingIndex: 5 } }, Light6: { binding: { groupIndex: 1, bindingIndex: 6 } }, Light7: { binding: { groupIndex: 1, bindingIndex: 7 } }, Light8: { binding: { groupIndex: 1, bindingIndex: 8 } }, Light9: { binding: { groupIndex: 1, bindingIndex: 9 } }, Light10: { binding: { groupIndex: 1, bindingIndex: 10 } }, Light11: { binding: { groupIndex: 1, bindingIndex: 11 } }, Light12: { binding: { groupIndex: 1, bindingIndex: 12 } }, Light13: { binding: { groupIndex: 1, bindingIndex: 13 } }, Light14: { binding: { groupIndex: 1, bindingIndex: 14 } }, Light15: { binding: { groupIndex: 1, bindingIndex: 15 } }, Light16: { binding: { groupIndex: 1, bindingIndex: 16 } }, Light17: { binding: { groupIndex: 1, bindingIndex: 17 } }, Light18: { binding: { groupIndex: 1, bindingIndex: 18 } }, Light19: { binding: { groupIndex: 1, bindingIndex: 19 } }, Light20: { binding: { groupIndex: 1, bindingIndex: 20 } }, Light21: { binding: { groupIndex: 1, bindingIndex: 21 } }, Light22: { binding: { groupIndex: 1, bindingIndex: 22 } }, Light23: { binding: { groupIndex: 1, bindingIndex: 23 } }, Light24: { binding: { groupIndex: 1, bindingIndex: 24 } }, Light25: { binding: { groupIndex: 1, bindingIndex: 25 } }, Light26: { binding: { groupIndex: 1, bindingIndex: 26 } }, Light27: { binding: { groupIndex: 1, bindingIndex: 27 } }, Light28: { binding: { groupIndex: 1, bindingIndex: 28 } }, Light29: { binding: { groupIndex: 1, bindingIndex: 29 } }, Light30: { binding: { groupIndex: 1, bindingIndex: 30 } }, Light31: { binding: { groupIndex: 1, bindingIndex: 31 } }, Material: { binding: { groupIndex: 2, bindingIndex: 0 } }, Mesh: { binding: { groupIndex: 2, bindingIndex: 1 } }, Internals: { binding: { groupIndex: 2, bindingIndex: 2 } } }; class Uue extends Wo { constructor() { super(...arguments), this._missingVaryings = [], this._textureArrayProcessing = [], this._vertexIsGLES3 = !1, this._fragmentIsGLES3 = !1, this.shaderLanguage = za.GLSL, this.parseGLES3 = !0; } _getArraySize(e, t, r) { let n = 0; const i = e.indexOf("["), s = e.indexOf("]"); if (i > 0 && s > 0) { const a = e.substring(i + 1, s); n = +a, isNaN(n) && (n = +r[a.trim()]), e = e.substr(0, i); } return [e, t, n]; } initializeShaders(e) { this._webgpuProcessingContext = e, this._missingVaryings.length = 0, this._textureArrayProcessing.length = 0, this.attributeKeywordName = void 0, this.varyingVertexKeywordName = void 0, this.varyingFragmentKeywordName = void 0; } preProcessShaderCode(e, t) { const r = `// Internals UBO uniform ${Wo.InternalsUBOName} { float yFactor_; float textureOutputHeight_; }; `, n = e.indexOf("// Internals UBO") !== -1; return t ? (this._fragmentIsGLES3 = e.indexOf("#version 3") !== -1, this._fragmentIsGLES3 && (this.varyingFragmentKeywordName = "in"), n ? e : r + `##INJECTCODE## ` + e) : (this._vertexIsGLES3 = e.indexOf("#version 3") !== -1, this._vertexIsGLES3 && (this.attributeKeywordName = "in", this.varyingVertexKeywordName = "out"), n ? e : r + e); } varyingCheck(e, t) { const r = /(flat\s)?\s*\bout\b/, n = /(flat\s)?\s*\bin\b/, i = /(flat\s)?\s*\bvarying\b/; return (t && this._fragmentIsGLES3 ? n : !t && this._vertexIsGLES3 ? r : i).test(e); } varyingProcessor(e, t, r) { var n; this._preProcessors = r; const i = /\s*(flat)?\s*out\s+(?:(?:highp)?|(?:lowp)?)\s*(\S+)\s+(\S+)\s*;/gm, s = /\s*(flat)?\s*in\s+(?:(?:highp)?|(?:lowp)?)\s*(\S+)\s+(\S+)\s*;/gm, a = /\s*(flat)?\s*varying\s+(?:(?:highp)?|(?:lowp)?)\s*(\S+)\s+(\S+)\s*;/gm, o = (t && this._fragmentIsGLES3 ? s : !t && this._vertexIsGLES3 ? i : a).exec(e); if (o !== null) { const d = (n = o[1]) !== null && n !== void 0 ? n : "", v = o[2], u = o[3]; let l; t ? (l = this._webgpuProcessingContext.availableVaryings[u], this._missingVaryings[l] = "", l === void 0 && Se.Warn(`Invalid fragment shader: The varying named "${u}" is not declared in the vertex shader! This declaration will be ignored.`)) : (l = this._webgpuProcessingContext.getVaryingNextLocation(v, this._getArraySize(u, v, r)[2]), this._webgpuProcessingContext.availableVaryings[u] = l, this._missingVaryings[l] = `layout(location = ${l}) ${d} in ${v} ${u};`), e = e.replace(o[0], l === void 0 ? "" : `layout(location = ${l}) ${d} ${t ? "in" : "out"} ${v} ${u};`); } return e; } attributeProcessor(e, t) { this._preProcessors = t; const r = /\s*in\s+(\S+)\s+(\S+)\s*;/gm, n = /\s*attribute\s+(\S+)\s+(\S+)\s*;/gm, s = (this._vertexIsGLES3 ? r : n).exec(e); if (s !== null) { const a = s[1], f = s[2], o = this._webgpuProcessingContext.getAttributeNextLocation(a, this._getArraySize(f, a, t)[2]); this._webgpuProcessingContext.availableAttributes[f] = o, this._webgpuProcessingContext.orderedAttributes[o] = f; const d = this.vertexBufferKindToNumberOfComponents[f]; if (d !== void 0) { const v = d < 0 ? d === -1 ? "int" : "ivec" + -d : d === 1 ? "uint" : "uvec" + d, u = `_int_${f}_`; e = e.replace(s[0], `layout(location = ${o}) in ${v} ${u}; ${a} ${f} = ${a}(${u});`); } else e = e.replace(s[0], `layout(location = ${o}) in ${a} ${f};`); } return e; } uniformProcessor(e, t, r) { var n; this._preProcessors = r; const s = /\s*uniform\s+(?:(?:highp)?|(?:lowp)?)\s*(\S+)\s+(\S+)\s*;/gm.exec(e); if (s !== null) { let a = s[1], f = s[2]; if (a.indexOf("sampler") === 0 || a.indexOf("sampler") === 1) { let o = 0; [f, a, o] = this._getArraySize(f, a, r); let d = this._webgpuProcessingContext.availableTextures[f]; if (!d) { d = { autoBindSampler: !0, isTextureArray: o > 0, isStorageTexture: !1, textures: [], sampleType: Ll.Float }; for (let I = 0; I < (o || 1); ++I) d.textures.push(this._webgpuProcessingContext.getNextFreeUBOBinding()); } const v = (n = Wo._SamplerTypeByWebGLSamplerType[a]) !== null && n !== void 0 ? n : "sampler", u = !!Wo._IsComparisonSamplerByWebGPUSamplerType[v], l = u ? Xq.Comparison : Xq.Filtering, P = f + Wo.AutoSamplerSuffix; let p = this._webgpuProcessingContext.availableSamplers[P]; p || (p = { binding: this._webgpuProcessingContext.getNextFreeUBOBinding(), type: l }); const c = a.charAt(0) === "u" ? "u" : a.charAt(0) === "i" ? "i" : ""; c && (a = a.substr(1)); const H = u ? Ll.Depth : c === "u" ? Ll.Uint : c === "i" ? Ll.Sint : Ll.Float; d.sampleType = H; const T = o > 0, q = p.binding.groupIndex, b = p.binding.bindingIndex, j = Wo._SamplerFunctionByWebGLSamplerType[a], w = Wo._TextureTypeByWebGLSamplerType[a], m = Wo._GpuTextureViewDimensionByWebGPUTextureType[w]; if (!T) o = 1, e = `layout(set = ${q}, binding = ${b}) uniform ${v} ${P}; layout(set = ${d.textures[0].groupIndex}, binding = ${d.textures[0].bindingIndex}) uniform ${c}${w} ${f}Texture; #define ${f} ${c}${j}(${f}Texture, ${P})`; else { const I = []; I.push(`layout(set = ${q}, binding = ${b}) uniform ${c}${v} ${P};`), e = ` `; for (let N = 0; N < o; ++N) { const k = d.textures[N].groupIndex, R = d.textures[N].bindingIndex; I.push(`layout(set = ${k}, binding = ${R}) uniform ${w} ${f}Texture${N};`), e += `${N > 0 ? ` ` : ""}#define ${f}${N} ${c}${j}(${f}Texture${N}, ${P})`; } e = I.join(` `) + e, this._textureArrayProcessing.push(f); } this._webgpuProcessingContext.availableTextures[f] = d, this._webgpuProcessingContext.availableSamplers[P] = p, this._addSamplerBindingDescription(P, p, !t); for (let I = 0; I < o; ++I) this._addTextureBindingDescription(f, d, I, m, null, !t); } else this._addUniformToLeftOverUBO(f, a, r), e = ""; } return e; } uniformBufferProcessor(e, t) { const n = /uniform\s+(\w+)/gm.exec(e); if (n !== null) { const i = n[1]; let s = this._webgpuProcessingContext.availableBuffers[i]; if (!s) { const a = gp.KnownUBOs[i]; let f; a && a.binding.groupIndex !== -1 ? f = a.binding : f = this._webgpuProcessingContext.getNextFreeUBOBinding(), s = { binding: f }, this._webgpuProcessingContext.availableBuffers[i] = s; } this._addBufferBindingDescription(i, s, dq.Uniform, !t), e = e.replace("uniform", `layout(set = ${s.binding.groupIndex}, binding = ${s.binding.bindingIndex}) uniform`); } return e; } postProcessor(e, t, r, n, i) { const s = e.search(/#extension.+GL_EXT_draw_buffers.+require/) !== -1, a = /#extension.+(GL_OVR_multiview2|GL_OES_standard_derivatives|GL_EXT_shader_texture_lod|GL_EXT_frag_depth|GL_EXT_draw_buffers).+(enable|require)/g; if (e = e.replace(a, ""), e = e.replace(/texture2D\s*\(/g, "texture("), r) { const f = e.indexOf("gl_FragCoord") >= 0, o = ` glFragCoord_ = gl_FragCoord; if (yFactor_ == 1.) { glFragCoord_.y = textureOutputHeight_ - glFragCoord_.y; } `, d = f ? `vec4 glFragCoord_; ` : "", v = e.search(/layout *\(location *= *0\) *out/g) !== -1; if (e = e.replace(/texture2DLodEXT\s*\(/g, "textureLod("), e = e.replace(/textureCubeLodEXT\s*\(/g, "textureLod("), e = e.replace(/textureCube\s*\(/g, "texture("), e = e.replace(/gl_FragDepthEXT/g, "gl_FragDepth"), e = e.replace(/gl_FragColor/g, "glFragColor"), e = e.replace(/gl_FragData/g, "glFragData"), e = e.replace(/gl_FragCoord/g, "glFragCoord_"), !this._fragmentIsGLES3) e = e.replace(/void\s+?main\s*\(/g, (s || v ? "" : `layout(location = 0) out vec4 glFragColor; `) + "void main("); else { const u = /^\s*out\s+\S+\s+\S+\s*;/gm.exec(e); u !== null && (e = e.substring(0, u.index) + "layout(location = 0) " + e.substring(u.index)); } e = e.replace(/dFdy/g, "(-yFactor_)*dFdy"), e = e.replace("##INJECTCODE##", d), f && (e = this._injectStartingAndEndingCode(e, "void main", o)); } else if (e = e.replace(/gl_InstanceID/g, "gl_InstanceIndex"), e = e.replace(/gl_VertexID/g, "gl_VertexIndex"), t.indexOf("#define MULTIVIEW") !== -1) return `#extension GL_OVR_multiview2 : require layout (num_views = 2) in; ` + e; if (!r) { const f = e.lastIndexOf("}"); e = e.substring(0, f), e += `gl_Position.y *= yFactor_; `, i.isNDCHalfZRange || (e += `gl_Position.z = (gl_Position.z + gl_Position.w) / 2.0; `), e += "}"; } return e; } _applyTextureArrayProcessing(e, t) { const r = new RegExp(t + "\\s*\\[(.+)?\\]", "gm"); let n = r.exec(e); for (; n !== null; ) { const i = n[1]; let s = +i; this._preProcessors && isNaN(s) && (s = +this._preProcessors[i.trim()]), e = e.replace(n[0], t + s), n = r.exec(e); } return e; } _generateLeftOverUBOCode(e, t) { let r = `layout(set = ${t.binding.groupIndex}, binding = ${t.binding.bindingIndex}) uniform ${e} { `; for (const n of this._webgpuProcessingContext.leftOverUniforms) n.length > 0 ? r += ` ${n.type} ${n.name}[${n.length}]; ` : r += ` ${n.type} ${n.name}; `; return r += `}; `, r; } finalizeShaders(e, t) { for (let n = 0; n < this._textureArrayProcessing.length; ++n) { const i = this._textureArrayProcessing[n]; e = this._applyTextureArrayProcessing(e, i), t = this._applyTextureArrayProcessing(t, i); } for (let n = 0; n < this._missingVaryings.length; ++n) { const i = this._missingVaryings[n]; i && i.length > 0 && (t = i + ` ` + t); } const r = this._buildLeftOverUBO(); return e = r + e, t = r + t, this._collectBindingNames(), this._preCreateBindGroupEntries(), this._preProcessors = null, this.vertexBufferKindToNumberOfComponents = {}, { vertexCode: e, fragmentCode: t }; } } const Iue = "bonesDeclaration", Rue = `#if NUM_BONE_INFLUENCERS>0 attribute matricesIndices : vec4;attribute matricesWeights : vec4; #if NUM_BONE_INFLUENCERS>4 attribute matricesIndicesExtra : vec4;attribute matricesWeightsExtra : vec4; #endif #ifndef BAKED_VERTEX_ANIMATION_TEXTURE #ifdef BONETEXTURE var boneSampler : texture_2d;uniform boneTextureWidth : f32; #else uniform mBones : array; #ifdef BONES_VELOCITY_ENABLED uniform mPreviousBones : array; #endif #endif #ifdef BONETEXTURE fn readMatrixFromRawSampler(smp : texture_2d,index : f32)->mat4x4 {let offset=i32(index) *4; let m0=textureLoad(smp,vec2(offset+0,0),0);let m1=textureLoad(smp,vec2(offset+1,0),0);let m2=textureLoad(smp,vec2(offset+2,0),0);let m3=textureLoad(smp,vec2(offset+3,0),0);return mat4x4(m0,m1,m2,m3);} #endif #endif #endif `; Le.IncludesShadersStoreWGSL[Iue] = Rue; const Vue = "bonesVertex", Cue = `#ifndef BAKED_VERTEX_ANIMATION_TEXTURE #if NUM_BONE_INFLUENCERS>0 var influence : mat4x4; #ifdef BONETEXTURE influence=readMatrixFromRawSampler(boneSampler,vertexInputs.matricesIndices[0])*vertexInputs.matricesWeights[0]; #if NUM_BONE_INFLUENCERS>1 influence=influence+readMatrixFromRawSampler(boneSampler,vertexInputs.matricesIndices[1])*vertexInputs.matricesWeights[1]; #endif #if NUM_BONE_INFLUENCERS>2 influence=influence+readMatrixFromRawSampler(boneSampler,vertexInputs.matricesIndices[2])*vertexInputs.matricesWeights[2]; #endif #if NUM_BONE_INFLUENCERS>3 influence=influence+readMatrixFromRawSampler(boneSampler,vertexInputs.matricesIndices[3])*vertexInputs.matricesWeights[3]; #endif #if NUM_BONE_INFLUENCERS>4 influence=influence+readMatrixFromRawSampler(boneSampler,vertexInputs.matricesIndicesExtra[0])*vertexInputs.matricesWeightsExtra[0]; #endif #if NUM_BONE_INFLUENCERS>5 influence=influence+readMatrixFromRawSampler(boneSampler,vertexInputs.matricesIndicesExtra[1])*vertexInputs.matricesWeightsExtra[1]; #endif #if NUM_BONE_INFLUENCERS>6 influence=influence+readMatrixFromRawSampler(boneSampler,vertexInputs.matricesIndicesExtra[2])*vertexInputs.matricesWeightsExtra[2]; #endif #if NUM_BONE_INFLUENCERS>7 influence=influence+readMatrixFromRawSampler(boneSampler,vertexInputs.matricesIndicesExtra[3])*vertexInputs.matricesWeightsExtra[3]; #endif #else influence=uniforms.mBones[int(vertexInputs.matricesIndices[0])]*vertexInputs.matricesWeights[0]; #if NUM_BONE_INFLUENCERS>1 influence=influence+uniforms.mBones[int(vertexInputs.matricesIndices[1])]*vertexInputs.matricesWeights[1]; #endif #if NUM_BONE_INFLUENCERS>2 influence=influence+uniforms.mBones[int(vertexInputs.matricesIndices[2])]*vertexInputs.matricesWeights[2]; #endif #if NUM_BONE_INFLUENCERS>3 influence=influence+uniforms.mBones[int(vertexInputs.matricesIndices[3])]*vertexInputs.matricesWeights[3]; #endif #if NUM_BONE_INFLUENCERS>4 influence=influence+uniforms.mBones[int(vertexInputs.matricesIndicesExtra[0])]*vertexInputs.matricesWeightsExtra[0]; #endif #if NUM_BONE_INFLUENCERS>5 influence=influence+uniforms.mBones[int(vertexInputs.matricesIndicesExtra[1])]*vertexInputs.matricesWeightsExtra[1]; #endif #if NUM_BONE_INFLUENCERS>6 influence=influence+uniforms.mBones[int(vertexInputs.matricesIndicesExtra[2])]*vertexInputs.matricesWeightsExtra[2]; #endif #if NUM_BONE_INFLUENCERS>7 influence=influence+uniforms.mBones[int(vertexInputs.matricesIndicesExtra[3])]*vertexInputs.matricesWeightsExtra[3]; #endif #endif finalWorld=finalWorld*influence; #endif #endif `; Le.IncludesShadersStoreWGSL[Vue] = Cue; const Oue = "bakedVertexAnimationDeclaration", yue = `#ifdef BAKED_VERTEX_ANIMATION_TEXTURE uniform bakedVertexAnimationTime: f32;uniform bakedVertexAnimationTextureSizeInverted: vec2;uniform bakedVertexAnimationSettings: vec4;var bakedVertexAnimationTexture : texture_2d; #ifdef INSTANCES attribute bakedVertexAnimationSettingsInstanced : vec4; #endif fn readMatrixFromRawSamplerVAT(smp : texture_2d,index : f32,frame : f32)->mat4x4 {let offset=i32(index)*4;let frameUV=i32(frame);let m0=textureLoad(smp,vec2(offset+0,frameUV),0);let m1=textureLoad(smp,vec2(offset+1,frameUV),0);let m2=textureLoad(smp,vec2(offset+2,frameUV),0);let m3=textureLoad(smp,vec2(offset+3,frameUV),0);return mat4x4(m0,m1,m2,m3);} #endif `; Le.IncludesShadersStoreWGSL[Oue] = yue; const kue = "bakedVertexAnimation", Eue = `#ifdef BAKED_VERTEX_ANIMATION_TEXTURE { #ifdef INSTANCES let VATStartFrame: f32=vertexInputs.bakedVertexAnimationSettingsInstanced.x;let VATEndFrame: f32=vertexInputs.bakedVertexAnimationSettingsInstanced.y;let VATOffsetFrame: f32=vertexInputs.bakedVertexAnimationSettingsInstanced.z;let VATSpeed: f32=vertexInputs.bakedVertexAnimationSettingsInstanced.w; #else let VATStartFrame: f32=uniforms.bakedVertexAnimationSettings.x;let VATEndFrame: f32=uniforms.bakedVertexAnimationSettings.y;let VATOffsetFrame: f32=uniforms.bakedVertexAnimationSettings.z;let VATSpeed: f32=uniforms.bakedVertexAnimationSettings.w; #endif let totalFrames: f32=VATEndFrame-VATStartFrame+1.0;let time: f32=uniforms.bakedVertexAnimationTime*VATSpeed/totalFrames;let frameCorrection: f32=select(1.0,0.0,time<1.0);let numOfFrames: f32=totalFrames-frameCorrection;var VATFrameNum: f32=fract(time)*numOfFrames;VATFrameNum=(VATFrameNum+VATOffsetFrame) % numOfFrames;VATFrameNum=floor(VATFrameNum);VATFrameNum=VATFrameNum+VATStartFrame+frameCorrection;var VATInfluence : mat4x4;VATInfluence=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,vertexInputs.matricesIndices[0],VATFrameNum)*vertexInputs.matricesWeights[0]; #if NUM_BONE_INFLUENCERS>1 VATInfluence=VATInfluence+readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,vertexInputs.matricesIndices[1],VATFrameNum)*vertexInputs.matricesWeights[1]; #endif #if NUM_BONE_INFLUENCERS>2 VATInfluence=VATInfluence+readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,vertexInputs.matricesIndices[2],VATFrameNum)*vertexInputs.matricesWeights[2]; #endif #if NUM_BONE_INFLUENCERS>3 VATInfluence=VATInfluence+readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,vertexInputs.matricesIndices[3],VATFrameNum)*vertexInputs.matricesWeights[3]; #endif #if NUM_BONE_INFLUENCERS>4 VATInfluence=VATInfluence+readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,vertexInputs.matricesIndicesExtra[0],VATFrameNum)*vertexInputs.matricesWeightsExtra[0]; #endif #if NUM_BONE_INFLUENCERS>5 VATInfluence=VATInfluence+readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,vertexInputs.matricesIndicesExtra[1],VATFrameNum)*vertexInputs.matricesWeightsExtra[1]; #endif #if NUM_BONE_INFLUENCERS>6 VATInfluence=VATInfluence+readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,vertexInputs.matricesIndicesExtra[2],VATFrameNum)*vertexInputs.matricesWeightsExtra[2]; #endif #if NUM_BONE_INFLUENCERS>7 VATInfluence=VATInfluence+readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,vertexInputs.matricesIndicesExtra[3],VATFrameNum)*vertexInputs.matricesWeightsExtra[3]; #endif finalWorld=finalWorld*VATInfluence;} #endif `; Le.IncludesShadersStoreWGSL[kue] = Eue; const Fue = "clipPlaneFragment", Nue = `#if defined(CLIPPLANE) || defined(CLIPPLANE2) || defined(CLIPPLANE3) || defined(CLIPPLANE4) || defined(CLIPPLANE5) || defined(CLIPPLANE6) if (false) {} #endif #ifdef CLIPPLANE else if (fragmentInputs.fClipDistance>0.0) {discard;} #endif #ifdef CLIPPLANE2 else if (fragmentInputs.fClipDistance2>0.0) {discard;} #endif #ifdef CLIPPLANE3 else if (fragmentInputs.fClipDistance3>0.0) {discard;} #endif #ifdef CLIPPLANE4 else if (fragmentInputs.fClipDistance4>0.0) {discard;} #endif #ifdef CLIPPLANE5 else if (fragmentInputs.fClipDistance5>0.0) {discard;} #endif #ifdef CLIPPLANE6 else if (fragmentInputs.fClipDistance6>0.0) {discard;} #endif `; Le.IncludesShadersStoreWGSL[Fue] = Nue; const Que = "clipPlaneFragmentDeclaration", Yue = `#ifdef CLIPPLANE varying fClipDistance: f32; #endif #ifdef CLIPPLANE2 varying fClipDistance2: f32; #endif #ifdef CLIPPLANE3 varying fClipDistance3: f32; #endif #ifdef CLIPPLANE4 varying fClipDistance4: f32; #endif #ifdef CLIPPLANE5 varying fClipDistance5: f32; #endif #ifdef CLIPPLANE6 varying fClipDistance6: f32; #endif `; Le.IncludesShadersStoreWGSL[Que] = Yue; const Mue = "clipPlaneVertex", Lue = `#ifdef CLIPPLANE vertexOutputs.fClipDistance=dot(worldPos,uniforms.vClipPlane); #endif #ifdef CLIPPLANE2 vertexOutputs.fClipDistance2=dot(worldPos,uniforms.vClipPlane2); #endif #ifdef CLIPPLANE3 vertexOutputs.fClipDistance3=dot(worldPos,uniforms.vClipPlane3); #endif #ifdef CLIPPLANE4 vertexOutputs.fClipDistance4=dot(worldPos,uniforms.vClipPlane4); #endif #ifdef CLIPPLANE5 vertexOutputs.fClipDistance5=dot(worldPos,uniforms.vClipPlane5); #endif #ifdef CLIPPLANE6 vertexOutputs.fClipDistance6=dot(worldPos,uniforms.vClipPlane6); #endif `; Le.IncludesShadersStoreWGSL[Mue] = Lue; const Kue = "clipPlaneVertexDeclaration", Jue = `#ifdef CLIPPLANE uniform vClipPlane: vec4;varying fClipDistance: f32; #endif #ifdef CLIPPLANE2 uniform vClipPlane2: vec4;varying fClipDistance2: f32; #endif #ifdef CLIPPLANE3 uniform vClipPlane3: vec4;varying fClipDistance3: f32; #endif #ifdef CLIPPLANE4 uniform vClipPlane4: vec4;varying fClipDistance4: f32; #endif #ifdef CLIPPLANE5 uniform vClipPlane5: vec4;varying fClipDistance5: f32; #endif #ifdef CLIPPLANE6 uniform vClipPlane6: vec4;varying fClipDistance6: f32; #endif `; Le.IncludesShadersStoreWGSL[Kue] = Jue; const zue = "instancesDeclaration", Gue = `#ifdef INSTANCES attribute world0 : vec4;attribute world1 : vec4;attribute world2 : vec4;attribute world3 : vec4; #ifdef INSTANCESCOLOR attribute instanceColor : vec4; #endif #if defined(THIN_INSTANCES) && !defined(WORLD_UBO) uniform world : mat4x4; #endif #if defined(VELOCITY) || defined(PREPASS_VELOCITY) attribute previousWorld0 : vec4;attribute previousWorld1 : vec4;attribute previousWorld2 : vec4;attribute previousWorld3 : vec4; #ifdef THIN_INSTANCES uniform previousWorld : mat4x4; #endif #endif #else #if !defined(WORLD_UBO) uniform world : mat4x4; #endif #if defined(VELOCITY) || defined(PREPASS_VELOCITY) uniform previousWorld : mat4x4; #endif #endif `; Le.IncludesShadersStoreWGSL[zue] = Gue; const Zue = "instancesVertex", _ue = `#ifdef INSTANCES var finalWorld=mat4x4(vertexInputs.world0,vertexInputs.world1,vertexInputs.world2,vertexInputs.world3); #if defined(PREPASS_VELOCITY) || defined(VELOCITY) var finalPreviousWorld=mat4x4(previousWorld0,previousWorld1,previousWorld2,previousWorld3); #endif #ifdef THIN_INSTANCES #if !defined(WORLD_UBO) finalWorld=uniforms.world*finalWorld; #else finalWorld=mesh.world*finalWorld; #endif #if defined(PREPASS_VELOCITY) || defined(VELOCITY) finalPreviousWorld=previousWorld*finalPreviousWorld; #endif #endif #else #if !defined(WORLD_UBO) var finalWorld=uniforms.world; #else var finalWorld=mesh.world; #endif #if defined(PREPASS_VELOCITY) || defined(VELOCITY) var finalPreviousWorld=previousWorld; #endif #endif `; Le.IncludesShadersStoreWGSL[Zue] = _ue; const $ue = "meshUboDeclaration", ele = `struct Mesh {world : mat4x4, visibility : f32,};var mesh : Mesh; #define WORLD_UBO `; Le.IncludesShadersStoreWGSL[$ue] = ele; const tle = "morphTargetsVertex", rle = `#ifdef MORPHTARGETS #ifdef MORPHTARGETS_TEXTURE vertexID=f32(vertexInputs.vertexIndex)*uniforms.morphTargetTextureInfo.x;positionUpdated=positionUpdated+(readVector3FromRawSampler({X},vertexID)-vertexInputs.position)*uniforms.morphTargetInfluences[{X}];vertexID=vertexID+1.0; #ifdef MORPHTARGETS_NORMAL normalUpdated=normalUpdated+(readVector3FromRawSampler({X},vertexID) -vertexInputs.normal)*uniforms.morphTargetInfluences[{X}];vertexID=vertexID+1.0; #endif #ifdef MORPHTARGETS_UV uvUpdated=uvUpdated+(readVector3FromRawSampler({X},vertexID).xy-vertexInputs.uv)*uniforms.morphTargetInfluences[{X}];vertexID=vertexID+1.0; #endif #ifdef MORPHTARGETS_TANGENT tangentUpdated.xyz=tangentUpdated.xyz+(readVector3FromRawSampler({X},vertexID) -vertexInputs.tangent.xyz)*uniforms.morphTargetInfluences[{X}]; #endif #else positionUpdated=positionUpdated+(position{X}-vertexInputs.position)*uniforms.morphTargetInfluences[{X}]; #ifdef MORPHTARGETS_NORMAL normalUpdated+=(normal{X}-vertexInputs.normal)*uniforms.morphTargetInfluences[{X}]; #endif #ifdef MORPHTARGETS_TANGENT tangentUpdated.xyz=tangentUpdated.xyz+(tangent{X}-vertexInputs.tangent.xyz)*uniforms.morphTargetInfluences[{X}]; #endif #ifdef MORPHTARGETS_UV uvUpdated=uvUpdated+(uv_{X}-vertexInputs.uv)*uniforms.morphTargetInfluences[{X}]; #endif #endif #endif `; Le.IncludesShadersStoreWGSL[tle] = rle; const nle = "morphTargetsVertexDeclaration", ile = `#ifdef MORPHTARGETS #ifndef MORPHTARGETS_TEXTURE attribute position{X} : vec3; #ifdef MORPHTARGETS_NORMAL attribute normal{X} : vec3; #endif #ifdef MORPHTARGETS_TANGENT attribute tangent{X} : vec3; #endif #ifdef MORPHTARGETS_UV attribute uv_{X} : vec2; #endif #endif #endif `; Le.IncludesShadersStoreWGSL[nle] = ile; const sle = "morphTargetsVertexGlobal", ale = `#ifdef MORPHTARGETS #ifdef MORPHTARGETS_TEXTURE var vertexID : f32; #endif #endif `; Le.IncludesShadersStoreWGSL[sle] = ale; const ole = "morphTargetsVertexGlobalDeclaration", fle = `#ifdef MORPHTARGETS uniform morphTargetInfluences : array; #ifdef MORPHTARGETS_TEXTURE uniform morphTargetTextureIndices : array;uniform morphTargetTextureInfo : vec3;var morphTargets : texture_2d_array;var morphTargetsSampler : sampler;fn readVector3FromRawSampler(targetIndex : i32,vertexIndex : f32)->vec3 { let y=floor(vertexIndex/uniforms.morphTargetTextureInfo.y);let x=vertexIndex-y*uniforms.morphTargetTextureInfo.y;let textureUV=vec2((x+0.5)/uniforms.morphTargetTextureInfo.y,(y+0.5)/uniforms.morphTargetTextureInfo.z);return textureSampleLevel(morphTargets,morphTargetsSampler,textureUV,i32(uniforms.morphTargetTextureIndices[targetIndex]),0.0).xyz;} #endif #endif `; Le.IncludesShadersStoreWGSL[ole] = fle; const Ale = "sceneUboDeclaration", dle = `struct Scene {viewProjection : mat4x4, #ifdef MULTIVIEW viewProjectionR : mat4x4, #endif view : mat4x4, projection : mat4x4, vEyePosition : vec4,};var scene : Scene; `; Le.IncludesShadersStoreWGSL[Ale] = dle; const bG = "fragmentOutputs.fragDepth", vle = "uniforms", ule = "internals", lle = { texture_1d: Da.E1d, texture_2d: Da.E2d, texture_2d_array: Da.E2dArray, texture_3d: Da.E3d, texture_cube: Da.Cube, texture_cube_array: Da.CubeArray, texture_multisampled_2d: Da.E2d, texture_depth_2d: Da.E2d, texture_depth_2d_array: Da.E2dArray, texture_depth_cube: Da.Cube, texture_depth_cube_array: Da.CubeArray, texture_depth_multisampled_2d: Da.E2d, texture_storage_1d: Da.E1d, texture_storage_2d: Da.E2d, texture_storage_2d_array: Da.E2dArray, texture_storage_3d: Da.E3d, texture_external: null }; class Ple extends Wo { constructor() { super(...arguments), this.shaderLanguage = za.WGSL, this.uniformRegexp = /uniform\s+(\w+)\s*:\s*(.+)\s*;/, this.textureRegexp = /var\s+(\w+)\s*:\s*((array<\s*)?(texture_\w+)\s*(<\s*(.+)\s*>)?\s*(,\s*\w+\s*>\s*)?);/, this.noPrecision = !0; } _getArraySize(e, t, r) { let n = 0; const i = t.lastIndexOf(">"); if (t.indexOf("array") >= 0 && i > 0) { let s = i; for (; s > 0 && t.charAt(s) !== " " && t.charAt(s) !== ","; ) s--; const a = t.substring(s + 1, i); for (n = +a, isNaN(n) && (n = +r[a.trim()]); s > 0 && (t.charAt(s) === " " || t.charAt(s) === ","); ) s--; t = t.substring(t.indexOf("<") + 1, s + 1); } return [e, t, n]; } initializeShaders(e) { this._webgpuProcessingContext = e, this._attributesInputWGSL = [], this._attributesWGSL = [], this._attributesConversionCodeWGSL = [], this._hasNonFloatAttribute = !1, this._varyingsWGSL = [], this._varyingNamesWGSL = [], this._stridedUniformArrays = []; } preProcessShaderCode(e) { return `struct ${Wo.InternalsUBOName} { yFactor_: f32, textureOutputHeight_: f32, }; var ${ule} : ${Wo.InternalsUBOName}; ` + iF(e); } varyingProcessor(e, t, r) { const i = /\s*varying\s+(?:(?:highp)?|(?:lowp)?)\s*(\S+)\s*:\s*(.+)\s*;/gm.exec(e); if (i !== null) { const s = i[2], a = i[1]; let f; t ? (f = this._webgpuProcessingContext.availableVaryings[a], f === void 0 && Se.Warn(`Invalid fragment shader: The varying named "${a}" is not declared in the vertex shader! This declaration will be ignored.`)) : (f = this._webgpuProcessingContext.getVaryingNextLocation(s, this._getArraySize(a, s, r)[2]), this._webgpuProcessingContext.availableVaryings[a] = f, this._varyingsWGSL.push(` @location(${f}) ${a} : ${s},`), this._varyingNamesWGSL.push(a)), e = ""; } return e; } attributeProcessor(e, t) { const n = /\s*attribute\s+(\S+)\s*:\s*(.+)\s*;/gm.exec(e); if (n !== null) { const i = n[2], s = n[1], a = this._webgpuProcessingContext.getAttributeNextLocation(i, this._getArraySize(s, i, t)[2]); this._webgpuProcessingContext.availableAttributes[s] = a, this._webgpuProcessingContext.orderedAttributes[a] = s; const f = this.vertexBufferKindToNumberOfComponents[s]; if (f !== void 0) { const o = f < 0 ? f === -1 ? "i32" : "vec" + -f + "" : f === 1 ? "u32" : "vec" + f + "", d = `_int_${s}_`; this._attributesInputWGSL.push(`@location(${a}) ${d} : ${o},`), this._attributesWGSL.push(`${s} : ${i},`), this._attributesConversionCodeWGSL.push(`vertexInputs.${s} = ${i}(vertexInputs_.${d});`), this._hasNonFloatAttribute = !0; } else this._attributesInputWGSL.push(`@location(${a}) ${s} : ${i},`), this._attributesWGSL.push(`${s} : ${i},`), this._attributesConversionCodeWGSL.push(`vertexInputs.${s} = vertexInputs_.${s};`); e = ""; } return e; } uniformProcessor(e, t, r) { const n = this.uniformRegexp.exec(e); if (n !== null) { const i = n[2], s = n[1]; this._addUniformToLeftOverUBO(s, i, r), e = ""; } return e; } textureProcessor(e, t, r) { const n = this.textureRegexp.exec(e); if (n !== null) { const i = n[1], s = n[2], a = !!n[3], f = n[4], o = f.indexOf("storage") > 0, d = n[6], v = o ? d.substring(0, d.indexOf(",")).trim() : null; let u = a ? this._getArraySize(i, s, r)[2] : 0, l = this._webgpuProcessingContext.availableTextures[i]; if (l) u = l.textures.length; else { l = { isTextureArray: u > 0, isStorageTexture: o, textures: [], sampleType: Ll.Float }, u = u || 1; for (let H = 0; H < u; ++H) l.textures.push(this._webgpuProcessingContext.getNextFreeUBOBinding()); } this._webgpuProcessingContext.availableTextures[i] = l; const P = f.indexOf("depth") > 0, p = lle[f], c = P ? Ll.Depth : d === "u32" ? Ll.Uint : d === "i32" ? Ll.Sint : Ll.Float; if (l.sampleType = c, p === void 0) throw `Can't get the texture dimension corresponding to the texture function "${f}"!`; for (let H = 0; H < u; ++H) { const { groupIndex: T, bindingIndex: q } = l.textures[H]; H === 0 && (e = `@group(${T}) @binding(${q}) ${e}`), this._addTextureBindingDescription(i, l, H, p, v, !t); } } return e; } postProcessor(e) { return e; } finalizeShaders(e, t) { const r = t.indexOf("fragmentInputs.position") >= 0 ? ` if (internals.yFactor_ == 1.) { fragmentInputs.position.y = internals.textureOutputHeight_ - fragmentInputs.position.y; } ` : ""; e = this._processSamplers(e, !0), t = this._processSamplers(t, !1), e = this._processCustomBuffers(e, !0), t = this._processCustomBuffers(t, !1); const n = this._buildLeftOverUBO(); e = n + e, t = n + t, e = e.replace(/#define /g, "//#define "), e = this._processStridedUniformArrays(e); let i = `struct VertexInputs { @builtin(vertex_index) vertexIndex : u32, @builtin(instance_index) instanceIndex : u32, `; this._attributesInputWGSL.length > 0 && (i += this._attributesInputWGSL.join(` `)), i += ` }; var vertexInputs` + (this._hasNonFloatAttribute ? "_" : "") + ` : VertexInputs; `, this._hasNonFloatAttribute && (i += `struct VertexInputs_ { vertexIndex : u32, instanceIndex : u32, `, i += this._attributesWGSL.join(` `), i += ` }; var vertexInputs : VertexInputs_; `); let s = `struct FragmentInputs { @builtin(position) position : vec4, `; this._varyingsWGSL.length > 0 && (s += this._varyingsWGSL.join(` `)), s += ` }; var vertexOutputs : FragmentInputs; `, e = i + s + e; let a = ` vertexInputs${this._hasNonFloatAttribute ? "_" : ""} = input; `; this._hasNonFloatAttribute && (a += `vertexInputs.vertexIndex = vertexInputs_.vertexIndex; vertexInputs.instanceIndex = vertexInputs_.instanceIndex; `, a += this._attributesConversionCodeWGSL.join(` `), a += ` `); const f = ` vertexOutputs.position.y = vertexOutputs.position.y * internals.yFactor_; return vertexOutputs;`; e = this._injectStartingAndEndingCode(e, "fn main", a, f), t = t.replace(/#define /g, "//#define "), t = this._processStridedUniformArrays(t), t = t.replace(/dpdy/g, "(-internals.yFactor_)*dpdy"); let o = `struct FragmentInputs { @builtin(position) position : vec4, @builtin(front_facing) frontFacing : bool, `; this._varyingsWGSL.length > 0 && (o += this._varyingsWGSL.join(` `)), o += ` }; var fragmentInputs : FragmentInputs; `; let d = `struct FragmentOutputs { @location(0) color : vec4, `, v = !1, u = 0; for (; !v && (u = t.indexOf(bG, u), !(u < 0)); ) { const p = u; for (v = !0; u > 1 && t.charAt(u) !== ` `; ) { if (t.charAt(u) === "/" && t.charAt(u - 1) === "/") { v = !1; break; } u--; } u = p + bG.length; } v && (d += ` @builtin(frag_depth) fragDepth: f32, `), d += `}; var fragmentOutputs : FragmentOutputs; `, t = o + d + t; const l = ` fragmentInputs = input; ` + r, P = " return fragmentOutputs;"; return t = this._injectStartingAndEndingCode(t, "fn main", l, P), this._collectBindingNames(), this._preCreateBindGroupEntries(), this.vertexBufferKindToNumberOfComponents = {}, { vertexCode: e, fragmentCode: t }; } _generateLeftOverUBOCode(e, t) { let r = "", n = `struct ${e} { `; for (const i of this._webgpuProcessingContext.leftOverUniforms) { const s = i.type.replace(/^(.*?)(<.*>)?$/, "$1"), a = Wo.UniformSizes[s]; if (i.length > 0) if (a <= 2) { const f = `${e}_${this._stridedUniformArrays.length}_strided_arr`; r += `struct ${f} { @size(16) el: ${s}, }`, this._stridedUniformArrays.push(i.name), n += ` @align(16) ${i.name} : array<${f}, ${i.length}>, `; } else n += ` ${i.name} : array<${i.type}, ${i.length}>, `; else n += ` ${i.name} : ${i.type}, `; } return n += `}; `, n = `${r} ${n}`, n += `@group(${t.binding.groupIndex}) @binding(${t.binding.bindingIndex}) var ${vle} : ${e}; `, n; } _processSamplers(e, t) { const r = /var\s+(\w+Sampler)\s*:\s*(sampler|sampler_comparison)\s*;/gm; for (; ; ) { const n = r.exec(e); if (n === null) break; const i = n[1], s = n[2], a = i.indexOf(Wo.AutoSamplerSuffix) === i.length - Wo.AutoSamplerSuffix.length ? i.substring(0, i.indexOf(Wo.AutoSamplerSuffix)) : null, f = s === "sampler_comparison" ? Xq.Comparison : Xq.Filtering; if (a) { const l = this._webgpuProcessingContext.availableTextures[a]; l && (l.autoBindSampler = !0); } let o = this._webgpuProcessingContext.availableSamplers[i]; o || (o = { binding: this._webgpuProcessingContext.getNextFreeUBOBinding(), type: f }, this._webgpuProcessingContext.availableSamplers[i] = o), this._addSamplerBindingDescription(i, o, t); const d = e.substring(0, n.index), v = `@group(${o.binding.groupIndex}) @binding(${o.binding.bindingIndex}) `, u = e.substring(n.index); e = d + v + u, r.lastIndex += v.length; } return e; } _processCustomBuffers(e, t) { const r = /var<\s*(uniform|storage)\s*(,\s*(read|read_write)\s*)?>\s+(\S+)\s*:\s*(\S+)\s*;/gm; for (; ; ) { const n = r.exec(e); if (n === null) break; const i = n[1], s = n[3]; let a = n[4]; const f = n[5]; let o = this._webgpuProcessingContext.availableBuffers[a]; if (!o) { const p = i === "uniform" ? gp.KnownUBOs[f] : null; let c; p ? (a = f, c = p.binding, c.groupIndex === -1 && (c = this._webgpuProcessingContext.getNextFreeUBOBinding())) : c = this._webgpuProcessingContext.getNextFreeUBOBinding(), o = { binding: c }, this._webgpuProcessingContext.availableBuffers[a] = o; } this._addBufferBindingDescription(a, this._webgpuProcessingContext.availableBuffers[a], s === "read_write" ? dq.Storage : i === "storage" ? dq.ReadOnlyStorage : dq.Uniform, t); const d = o.binding.groupIndex, v = o.binding.bindingIndex, u = e.substring(0, n.index), l = `@group(${d}) @binding(${v}) `, P = e.substring(n.index); e = u + l + P, r.lastIndex += l.length; } return e; } _processStridedUniformArrays(e) { for (const t of this._stridedUniformArrays) e = e.replace(new RegExp(`${t}\\s*\\[(.*)\\]`, "g"), `${t}[$1].el`); return e; } } class DC { get underlyingResource() { return this._webgpuTexture; } getMSAATexture(e = 0) { var t, r; return (r = (t = this._webgpuMSAATexture) === null || t === void 0 ? void 0 : t[e]) !== null && r !== void 0 ? r : null; } setMSAATexture(e, t = -1) { this._webgpuMSAATexture || (this._webgpuMSAATexture = []), t === -1 && (t = this._webgpuMSAATexture.length), this._webgpuMSAATexture[t] = e; } releaseMSAATexture() { if (this._webgpuMSAATexture) { for (const e of this._webgpuMSAATexture) e == null || e.destroy(); this._webgpuMSAATexture = null; } } constructor(e = null) { this.format = we.RGBA8Unorm, this.textureUsages = 0, this.textureAdditionalUsages = 0, this._webgpuTexture = e, this._webgpuMSAATexture = null, this.view = null, this.viewForWriting = null; } set(e) { this._webgpuTexture = e; } setUsage(e, t, r, n, i) { this.createView({ format: this.format, dimension: r ? Da.Cube : Da.E2d, mipLevelCount: t ? Xt.ILog2(Math.max(n, i)) + 1 : 1, baseArrayLayer: 0, baseMipLevel: 0, arrayLayerCount: r ? 6 : 1, aspect: zH.All }); } createView(e, t = !1) { if (this.view = this._webgpuTexture.createView(e), t && e) { const r = e.mipLevelCount; e.mipLevelCount = 1, this.viewForWriting = this._webgpuTexture.createView(e), e.mipLevelCount = r; } } reset() { this._webgpuTexture = null, this._webgpuMSAATexture = null, this.view = null, this.viewForWriting = null; } release() { var e, t; (e = this._webgpuTexture) === null || e === void 0 || e.destroy(), this.releaseMSAATexture(), (t = this._copyInvertYTempTexture) === null || t === void 0 || t.destroy(), this.reset(); } } const cle = ` const vec2 pos[4] = vec2[4](vec2(-1.0f, 1.0f), vec2(1.0f, 1.0f), vec2(-1.0f, -1.0f), vec2(1.0f, -1.0f)); const vec2 tex[4] = vec2[4](vec2(0.0f, 0.0f), vec2(1.0f, 0.0f), vec2(0.0f, 1.0f), vec2(1.0f, 1.0f)); layout(location = 0) out vec2 vTex; void main() { vTex = tex[gl_VertexIndex]; gl_Position = vec4(pos[gl_VertexIndex], 0.0, 1.0); } `, ple = ` layout(set = 0, binding = 0) uniform sampler imgSampler; layout(set = 0, binding = 1) uniform texture2D img; layout(location = 0) in vec2 vTex; layout(location = 0) out vec4 outColor; void main() { outColor = texture(sampler2D(img, imgSampler), vTex); } `, Tte = ` #extension GL_EXT_samplerless_texture_functions : enable const vec2 pos[4] = vec2[4](vec2(-1.0f, 1.0f), vec2(1.0f, 1.0f), vec2(-1.0f, -1.0f), vec2(1.0f, -1.0f)); const vec2 tex[4] = vec2[4](vec2(0.0f, 0.0f), vec2(1.0f, 0.0f), vec2(0.0f, 1.0f), vec2(1.0f, 1.0f)); layout(set = 0, binding = 0) uniform texture2D img; #ifdef INVERTY layout(location = 0) out flat ivec2 vTextureSize; #endif void main() { #ifdef INVERTY vTextureSize = textureSize(img, 0); #endif gl_Position = vec4(pos[gl_VertexIndex], 0.0, 1.0); } `, hle = ` #extension GL_EXT_samplerless_texture_functions : enable layout(set = 0, binding = 0) uniform texture2D img; #ifdef INVERTY layout(location = 0) in flat ivec2 vTextureSize; #endif layout(location = 0) out vec4 outColor; void main() { #ifdef INVERTY vec4 color = texelFetch(img, ivec2(gl_FragCoord.x, vTextureSize.y - gl_FragCoord.y), 0); #else vec4 color = texelFetch(img, ivec2(gl_FragCoord.xy), 0); #endif #ifdef PREMULTIPLYALPHA color.rgb *= color.a; #endif outColor = color; } `, Hle = Tte, gle = ` #extension GL_EXT_samplerless_texture_functions : enable layout(set = 0, binding = 0) uniform texture2D img; layout(set = 0, binding = 1) uniform Params { float ofstX; float ofstY; float width; float height; }; #ifdef INVERTY layout(location = 0) in flat ivec2 vTextureSize; #endif layout(location = 0) out vec4 outColor; void main() { if (gl_FragCoord.x < ofstX || gl_FragCoord.x >= ofstX + width) { discard; } if (gl_FragCoord.y < ofstY || gl_FragCoord.y >= ofstY + height) { discard; } #ifdef INVERTY vec4 color = texelFetch(img, ivec2(gl_FragCoord.x, ofstY + height - (gl_FragCoord.y - ofstY)), 0); #else vec4 color = texelFetch(img, ivec2(gl_FragCoord.xy), 0); #endif #ifdef PREMULTIPLYALPHA color.rgb *= color.a; #endif outColor = color; } `, Xle = ` const vec2 pos[4] = vec2[4](vec2(-1.0f, 1.0f), vec2(1.0f, 1.0f), vec2(-1.0f, -1.0f), vec2(1.0f, -1.0f)); void main() { gl_Position = vec4(pos[gl_VertexIndex], 0.0, 1.0); } `, Tle = ` layout(set = 0, binding = 0) uniform Uniforms { uniform vec4 color; }; layout(location = 0) out vec4 outColor; void main() { outColor = color; } `, qle = ` struct VertexOutput { @builtin(position) Position : vec4, @location(0) fragUV : vec2 } @vertex fn main( @builtin(vertex_index) VertexIndex : u32 ) -> VertexOutput { var pos = array, 4>( vec2(-1.0, 1.0), vec2( 1.0, 1.0), vec2(-1.0, -1.0), vec2( 1.0, -1.0) ); var tex = array, 4>( vec2(0.0, 0.0), vec2(1.0, 0.0), vec2(0.0, 1.0), vec2(1.0, 1.0) ); var output: VertexOutput; output.Position = vec4(pos[VertexIndex], 0.0, 1.0); output.fragUV = tex[VertexIndex]; return output; } `, ble = ` @group(0) @binding(0) var videoSampler: sampler; @group(0) @binding(1) var videoTexture: texture_external; @fragment fn main( @location(0) fragUV: vec2 ) -> @location(0) vec4 { return textureSampleBaseClampToEdge(videoTexture, videoSampler, fragUV); } `, xle = ` @group(0) @binding(0) var videoSampler: sampler; @group(0) @binding(1) var videoTexture: texture_external; @fragment fn main( @location(0) fragUV: vec2 ) -> @location(0) vec4 { return textureSampleBaseClampToEdge(videoTexture, videoSampler, vec2(fragUV.x, 1.0 - fragUV.y)); } `; var H0; (function(A) { A[A.MipMap = 0] = "MipMap", A[A.InvertYPremultiplyAlpha = 1] = "InvertYPremultiplyAlpha", A[A.Clear = 2] = "Clear", A[A.InvertYPremultiplyAlphaWithOfst = 3] = "InvertYPremultiplyAlphaWithOfst"; })(H0 || (H0 = {})); var QW; (function(A) { A[A.DontInvertY = 0] = "DontInvertY", A[A.InvertY = 1] = "InvertY"; })(QW || (QW = {})); const xG = [ { vertex: cle, fragment: ple }, { vertex: Tte, fragment: hle }, { vertex: Xle, fragment: Tle }, { vertex: Hle, fragment: gle } ], F2 = { "": 0, r8unorm: 1, r8uint: 2, r8sint: 3, r16uint: 4, r16sint: 5, r16float: 6, rg8unorm: 7, rg8uint: 8, rg8sint: 9, r32uint: 10, r32sint: 11, r32float: 12, rg16uint: 13, rg16sint: 14, rg16float: 15, rgba8unorm: 16, "rgba8unorm-srgb": 17, rgba8uint: 18, rgba8sint: 19, bgra8unorm: 20, "bgra8unorm-srgb": 21, rgb10a2uint: 22, rgb10a2unorm: 23, /* rg11b10ufloat: this entry is dynamically added if the "RG11B10UFloatRenderable" extension is supported */ rg32uint: 24, rg32sint: 25, rg32float: 26, rgba16uint: 27, rgba16sint: 28, rgba16float: 29, rgba32uint: 30, rgba32sint: 31, rgba32float: 32, stencil8: 33, depth16unorm: 34, depth24plus: 35, "depth24plus-stencil8": 36, depth32float: 37, "depth32float-stencil8": 38 }; class Ns { static ComputeNumMipmapLevels(e, t) { return Xt.ILog2(Math.max(e, t)) + 1; } //------------------------------------------------------------------------------ // Initialization / Helpers //------------------------------------------------------------------------------ constructor(e, t, r, n, i) { if (this._pipelines = {}, this._compiledShaders = [], this._videoPipelines = {}, this._videoCompiledShaders = [], this._deferredReleaseTextures = [], this._device = e, this._glslang = t, this._tintWASM = r, this._bufferManager = n, i.indexOf(k2.RG11B10UFloatRenderable) !== -1) { const s = Object.keys(F2); F2[we.RG11B10UFloat] = F2[s[s.length - 1]] + 1; } this._mipmapSampler = e.createSampler({ minFilter: Ji.Linear }), this._videoSampler = e.createSampler({ minFilter: Ji.Linear }), this._ubCopyWithOfst = this._bufferManager.createBuffer(4 * 4, ga.Uniform | ga.CopyDst, "UBCopyWithOffset").underlyingResource, this._getPipeline(we.RGBA8Unorm), this._getVideoPipeline(we.RGBA8Unorm); } _getPipeline(e, t = H0.MipMap, r) { const n = t === H0.MipMap ? 1 : t === H0.InvertYPremultiplyAlpha ? ((r.invertY ? 1 : 0) << 1) + ((r.premultiplyAlpha ? 1 : 0) << 2) : t === H0.Clear ? 8 : t === H0.InvertYPremultiplyAlphaWithOfst ? ((r.invertY ? 1 : 0) << 4) + ((r.premultiplyAlpha ? 1 : 0) << 5) : 0; this._pipelines[e] || (this._pipelines[e] = []); let i = this._pipelines[e][n]; if (!i) { let s = `#version 450 `; (t === H0.InvertYPremultiplyAlpha || t === H0.InvertYPremultiplyAlphaWithOfst) && (r.invertY && (s += `#define INVERTY `), r.premultiplyAlpha && (s += `#define PREMULTIPLYALPHA `)); let a = this._compiledShaders[n]; if (!a) { let o = this._glslang.compileGLSL(s + xG[t].vertex, "vertex"), d = this._glslang.compileGLSL(s + xG[t].fragment, "fragment"); this._tintWASM && (o = this._tintWASM.convertSpirV2WGSL(o), d = this._tintWASM.convertSpirV2WGSL(d)); const v = this._device.createShaderModule({ code: o }), u = this._device.createShaderModule({ code: d }); a = this._compiledShaders[n] = [v, u]; } const f = this._device.createRenderPipeline({ layout: SS.Auto, vertex: { module: a[0], entryPoint: "main" }, fragment: { module: a[1], entryPoint: "main", targets: [ { format: e } ] }, primitive: { topology: Fl.TriangleStrip, stripIndexFormat: Tq.Uint16 } }); i = this._pipelines[e][n] = [f, f.getBindGroupLayout(0)]; } return i; } _getVideoPipeline(e, t = QW.DontInvertY) { const r = t === QW.InvertY ? 1 : 0; this._videoPipelines[e] || (this._videoPipelines[e] = []); let n = this._videoPipelines[e][r]; if (!n) { let i = this._videoCompiledShaders[r]; if (!i) { const a = this._device.createShaderModule({ code: qle }), f = this._device.createShaderModule({ code: r === 0 ? ble : xle }); i = this._videoCompiledShaders[r] = [a, f]; } const s = this._device.createRenderPipeline({ label: `CopyVideoToTexture_${e}_${r === 0 ? "DontInvertY" : "InvertY"}`, layout: SS.Auto, vertex: { module: i[0], entryPoint: "main" }, fragment: { module: i[1], entryPoint: "main", targets: [ { format: e } ] }, primitive: { topology: Fl.TriangleStrip, stripIndexFormat: Tq.Uint16 } }); n = this._videoPipelines[e][r] = [s, s.getBindGroupLayout(0)]; } return n; } static GetTextureTypeFromFormat(e) { switch (e) { case we.R8Unorm: case we.R8Snorm: case we.R8Uint: case we.R8Sint: case we.RG8Unorm: case we.RG8Snorm: case we.RG8Uint: case we.RG8Sint: case we.RGBA8Unorm: case we.RGBA8UnormSRGB: case we.RGBA8Snorm: case we.RGBA8Uint: case we.RGBA8Sint: case we.BGRA8Unorm: case we.BGRA8UnormSRGB: case we.RGB10A2UINT: case we.RGB10A2Unorm: case we.RGB9E5UFloat: case we.RG11B10UFloat: case we.BC7RGBAUnorm: case we.BC7RGBAUnormSRGB: case we.BC6HRGBUFloat: case we.BC6HRGBFloat: case we.BC5RGUnorm: case we.BC5RGSnorm: case we.BC3RGBAUnorm: case we.BC3RGBAUnormSRGB: case we.BC2RGBAUnorm: case we.BC2RGBAUnormSRGB: case we.BC4RUnorm: case we.BC4RSnorm: case we.BC1RGBAUnorm: case we.BC1RGBAUnormSRGB: case we.ETC2RGB8Unorm: case we.ETC2RGB8UnormSRGB: case we.ETC2RGB8A1Unorm: case we.ETC2RGB8A1UnormSRGB: case we.ETC2RGBA8Unorm: case we.ETC2RGBA8UnormSRGB: case we.EACR11Unorm: case we.EACR11Snorm: case we.EACRG11Unorm: case we.EACRG11Snorm: case we.ASTC4x4Unorm: case we.ASTC4x4UnormSRGB: case we.ASTC5x4Unorm: case we.ASTC5x4UnormSRGB: case we.ASTC5x5Unorm: case we.ASTC5x5UnormSRGB: case we.ASTC6x5Unorm: case we.ASTC6x5UnormSRGB: case we.ASTC6x6Unorm: case we.ASTC6x6UnormSRGB: case we.ASTC8x5Unorm: case we.ASTC8x5UnormSRGB: case we.ASTC8x6Unorm: case we.ASTC8x6UnormSRGB: case we.ASTC8x8Unorm: case we.ASTC8x8UnormSRGB: case we.ASTC10x5Unorm: case we.ASTC10x5UnormSRGB: case we.ASTC10x6Unorm: case we.ASTC10x6UnormSRGB: case we.ASTC10x8Unorm: case we.ASTC10x8UnormSRGB: case we.ASTC10x10Unorm: case we.ASTC10x10UnormSRGB: case we.ASTC12x10Unorm: case we.ASTC12x10UnormSRGB: case we.ASTC12x12Unorm: case we.ASTC12x12UnormSRGB: case we.Stencil8: return 0; case we.R16Uint: case we.R16Sint: case we.RG16Uint: case we.RG16Sint: case we.RGBA16Uint: case we.RGBA16Sint: case we.Depth16Unorm: return 5; case we.R16Float: case we.RG16Float: case we.RGBA16Float: return 2; case we.R32Uint: case we.R32Sint: case we.RG32Uint: case we.RG32Sint: case we.RGBA32Uint: case we.RGBA32Sint: return 7; case we.R32Float: case we.RG32Float: case we.RGBA32Float: case we.Depth32Float: case we.Depth32FloatStencil8: case we.Depth24Plus: case we.Depth24PlusStencil8: return 1; } return 0; } static _GetBlockInformationFromFormat(e) { switch (e) { case we.R8Unorm: case we.R8Snorm: case we.R8Uint: case we.R8Sint: return { width: 1, height: 1, length: 1 }; case we.R16Uint: case we.R16Sint: case we.R16Float: case we.RG8Unorm: case we.RG8Snorm: case we.RG8Uint: case we.RG8Sint: return { width: 1, height: 1, length: 2 }; case we.R32Uint: case we.R32Sint: case we.R32Float: case we.RG16Uint: case we.RG16Sint: case we.RG16Float: case we.RGBA8Unorm: case we.RGBA8UnormSRGB: case we.RGBA8Snorm: case we.RGBA8Uint: case we.RGBA8Sint: case we.BGRA8Unorm: case we.BGRA8UnormSRGB: case we.RGB9E5UFloat: case we.RGB10A2UINT: case we.RGB10A2Unorm: case we.RG11B10UFloat: return { width: 1, height: 1, length: 4 }; case we.RG32Uint: case we.RG32Sint: case we.RG32Float: case we.RGBA16Uint: case we.RGBA16Sint: case we.RGBA16Float: return { width: 1, height: 1, length: 8 }; case we.RGBA32Uint: case we.RGBA32Sint: case we.RGBA32Float: return { width: 1, height: 1, length: 16 }; case we.Stencil8: throw "No fixed size for Stencil8 format!"; case we.Depth16Unorm: return { width: 1, height: 1, length: 2 }; case we.Depth24Plus: throw "No fixed size for Depth24Plus format!"; case we.Depth24PlusStencil8: throw "No fixed size for Depth24PlusStencil8 format!"; case we.Depth32Float: return { width: 1, height: 1, length: 4 }; case we.Depth32FloatStencil8: return { width: 1, height: 1, length: 5 }; case we.BC7RGBAUnorm: case we.BC7RGBAUnormSRGB: case we.BC6HRGBUFloat: case we.BC6HRGBFloat: case we.BC5RGUnorm: case we.BC5RGSnorm: case we.BC3RGBAUnorm: case we.BC3RGBAUnormSRGB: case we.BC2RGBAUnorm: case we.BC2RGBAUnormSRGB: return { width: 4, height: 4, length: 16 }; case we.BC4RUnorm: case we.BC4RSnorm: case we.BC1RGBAUnorm: case we.BC1RGBAUnormSRGB: return { width: 4, height: 4, length: 8 }; case we.ETC2RGB8Unorm: case we.ETC2RGB8UnormSRGB: case we.ETC2RGB8A1Unorm: case we.ETC2RGB8A1UnormSRGB: case we.EACR11Unorm: case we.EACR11Snorm: return { width: 4, height: 4, length: 8 }; case we.ETC2RGBA8Unorm: case we.ETC2RGBA8UnormSRGB: case we.EACRG11Unorm: case we.EACRG11Snorm: return { width: 4, height: 4, length: 16 }; case we.ASTC4x4Unorm: case we.ASTC4x4UnormSRGB: return { width: 4, height: 4, length: 16 }; case we.ASTC5x4Unorm: case we.ASTC5x4UnormSRGB: return { width: 5, height: 4, length: 16 }; case we.ASTC5x5Unorm: case we.ASTC5x5UnormSRGB: return { width: 5, height: 5, length: 16 }; case we.ASTC6x5Unorm: case we.ASTC6x5UnormSRGB: return { width: 6, height: 5, length: 16 }; case we.ASTC6x6Unorm: case we.ASTC6x6UnormSRGB: return { width: 6, height: 6, length: 16 }; case we.ASTC8x5Unorm: case we.ASTC8x5UnormSRGB: return { width: 8, height: 5, length: 16 }; case we.ASTC8x6Unorm: case we.ASTC8x6UnormSRGB: return { width: 8, height: 6, length: 16 }; case we.ASTC8x8Unorm: case we.ASTC8x8UnormSRGB: return { width: 8, height: 8, length: 16 }; case we.ASTC10x5Unorm: case we.ASTC10x5UnormSRGB: return { width: 10, height: 5, length: 16 }; case we.ASTC10x6Unorm: case we.ASTC10x6UnormSRGB: return { width: 10, height: 6, length: 16 }; case we.ASTC10x8Unorm: case we.ASTC10x8UnormSRGB: return { width: 10, height: 8, length: 16 }; case we.ASTC10x10Unorm: case we.ASTC10x10UnormSRGB: return { width: 10, height: 10, length: 16 }; case we.ASTC12x10Unorm: case we.ASTC12x10UnormSRGB: return { width: 12, height: 10, length: 16 }; case we.ASTC12x12Unorm: case we.ASTC12x12UnormSRGB: return { width: 12, height: 12, length: 16 }; } return { width: 1, height: 1, length: 4 }; } static _IsHardwareTexture(e) { return !!e.release; } static _IsInternalTexture(e) { return !!e.dispose; } static IsImageBitmap(e) { return e.close !== void 0; } static IsImageBitmapArray(e) { return Array.isArray(e) && e[0].close !== void 0; } setCommandEncoder(e) { this._commandEncoderForCreation = e; } static IsCompressedFormat(e) { switch (e) { case we.BC7RGBAUnormSRGB: case we.BC7RGBAUnorm: case we.BC6HRGBFloat: case we.BC6HRGBUFloat: case we.BC5RGSnorm: case we.BC5RGUnorm: case we.BC4RSnorm: case we.BC4RUnorm: case we.BC3RGBAUnormSRGB: case we.BC3RGBAUnorm: case we.BC2RGBAUnormSRGB: case we.BC2RGBAUnorm: case we.BC1RGBAUnormSRGB: case we.BC1RGBAUnorm: case we.ETC2RGB8Unorm: case we.ETC2RGB8UnormSRGB: case we.ETC2RGB8A1Unorm: case we.ETC2RGB8A1UnormSRGB: case we.ETC2RGBA8Unorm: case we.ETC2RGBA8UnormSRGB: case we.EACR11Unorm: case we.EACR11Snorm: case we.EACRG11Unorm: case we.EACRG11Snorm: case we.ASTC4x4Unorm: case we.ASTC4x4UnormSRGB: case we.ASTC5x4Unorm: case we.ASTC5x4UnormSRGB: case we.ASTC5x5Unorm: case we.ASTC5x5UnormSRGB: case we.ASTC6x5Unorm: case we.ASTC6x5UnormSRGB: case we.ASTC6x6Unorm: case we.ASTC6x6UnormSRGB: case we.ASTC8x5Unorm: case we.ASTC8x5UnormSRGB: case we.ASTC8x6Unorm: case we.ASTC8x6UnormSRGB: case we.ASTC8x8Unorm: case we.ASTC8x8UnormSRGB: case we.ASTC10x5Unorm: case we.ASTC10x5UnormSRGB: case we.ASTC10x6Unorm: case we.ASTC10x6UnormSRGB: case we.ASTC10x8Unorm: case we.ASTC10x8UnormSRGB: case we.ASTC10x10Unorm: case we.ASTC10x10UnormSRGB: case we.ASTC12x10Unorm: case we.ASTC12x10UnormSRGB: case we.ASTC12x12Unorm: case we.ASTC12x12UnormSRGB: return !0; } return !1; } static GetWebGPUTextureFormat(e, t, r = !1) { switch (t) { case 15: return we.Depth16Unorm; case 16: return we.Depth24Plus; case 13: return we.Depth24PlusStencil8; case 14: return we.Depth32Float; case 18: return we.Depth32FloatStencil8; case 19: return we.Stencil8; case 36492: return r ? we.BC7RGBAUnormSRGB : we.BC7RGBAUnorm; case 36495: return we.BC6HRGBUFloat; case 36494: return we.BC6HRGBFloat; case 33779: return r ? we.BC3RGBAUnormSRGB : we.BC3RGBAUnorm; case 33778: return r ? we.BC2RGBAUnormSRGB : we.BC2RGBAUnorm; case 33777: case 33776: return r ? we.BC1RGBAUnormSRGB : we.BC1RGBAUnorm; case 37808: return r ? we.ASTC4x4UnormSRGB : we.ASTC4x4Unorm; case 36196: case 37492: return r ? we.ETC2RGB8UnormSRGB : we.ETC2RGB8Unorm; case 37496: return r ? we.ETC2RGBA8UnormSRGB : we.ETC2RGBA8Unorm; } switch (e) { case 3: switch (t) { case 6: return we.R8Snorm; case 7: return we.RG8Snorm; case 4: throw "RGB format not supported in WebGPU"; case 8: return we.R8Sint; case 9: return we.RG8Sint; case 10: throw "RGB_INTEGER format not supported in WebGPU"; case 11: return we.RGBA8Sint; default: return we.RGBA8Snorm; } case 0: switch (t) { case 6: return we.R8Unorm; case 7: return we.RG8Unorm; case 4: throw "TEXTUREFORMAT_RGB format not supported in WebGPU"; case 5: return r ? we.RGBA8UnormSRGB : we.RGBA8Unorm; case 12: return r ? we.BGRA8UnormSRGB : we.BGRA8Unorm; case 8: return we.R8Uint; case 9: return we.RG8Uint; case 10: throw "RGB_INTEGER format not supported in WebGPU"; case 11: return we.RGBA8Uint; case 0: throw "TEXTUREFORMAT_ALPHA format not supported in WebGPU"; case 1: throw "TEXTUREFORMAT_LUMINANCE format not supported in WebGPU"; case 2: throw "TEXTUREFORMAT_LUMINANCE_ALPHA format not supported in WebGPU"; default: return we.RGBA8Unorm; } case 4: switch (t) { case 8: return we.R16Sint; case 9: return we.RG16Sint; case 10: throw "TEXTUREFORMAT_RGB_INTEGER format not supported in WebGPU"; case 11: return we.RGBA16Sint; default: return we.RGBA16Sint; } case 5: switch (t) { case 8: return we.R16Uint; case 9: return we.RG16Uint; case 10: throw "TEXTUREFORMAT_RGB_INTEGER format not supported in WebGPU"; case 11: return we.RGBA16Uint; default: return we.RGBA16Uint; } case 6: switch (t) { case 8: return we.R32Sint; case 9: return we.RG32Sint; case 10: throw "TEXTUREFORMAT_RGB_INTEGER format not supported in WebGPU"; case 11: return we.RGBA32Sint; default: return we.RGBA32Sint; } case 7: switch (t) { case 8: return we.R32Uint; case 9: return we.RG32Uint; case 10: throw "TEXTUREFORMAT_RGB_INTEGER format not supported in WebGPU"; case 11: return we.RGBA32Uint; default: return we.RGBA32Uint; } case 1: switch (t) { case 6: return we.R32Float; case 7: return we.RG32Float; case 4: throw "TEXTUREFORMAT_RGB format not supported in WebGPU"; case 5: return we.RGBA32Float; default: return we.RGBA32Float; } case 2: switch (t) { case 6: return we.R16Float; case 7: return we.RG16Float; case 4: throw "TEXTUREFORMAT_RGB format not supported in WebGPU"; case 5: return we.RGBA16Float; default: return we.RGBA16Float; } case 10: throw "TEXTURETYPE_UNSIGNED_SHORT_5_6_5 format not supported in WebGPU"; case 13: switch (t) { case 5: return we.RG11B10UFloat; case 11: throw "TEXTUREFORMAT_RGBA_INTEGER format not supported in WebGPU when type is TEXTURETYPE_UNSIGNED_INT_10F_11F_11F_REV"; default: return we.RG11B10UFloat; } case 14: switch (t) { case 5: return we.RGB9E5UFloat; case 11: throw "TEXTUREFORMAT_RGBA_INTEGER format not supported in WebGPU when type is TEXTURETYPE_UNSIGNED_INT_5_9_9_9_REV"; default: return we.RGB9E5UFloat; } case 8: throw "TEXTURETYPE_UNSIGNED_SHORT_4_4_4_4 format not supported in WebGPU"; case 9: throw "TEXTURETYPE_UNSIGNED_SHORT_5_5_5_1 format not supported in WebGPU"; case 11: switch (t) { case 5: return we.RGB10A2Unorm; case 11: return we.RGB10A2UINT; default: return we.RGB10A2Unorm; } } return r ? we.RGBA8UnormSRGB : we.RGBA8Unorm; } static GetNumChannelsFromWebGPUTextureFormat(e) { switch (e) { case we.R8Unorm: case we.R8Snorm: case we.R8Uint: case we.R8Sint: case we.BC4RUnorm: case we.BC4RSnorm: case we.R16Uint: case we.R16Sint: case we.Depth16Unorm: case we.R16Float: case we.R32Uint: case we.R32Sint: case we.R32Float: case we.Depth32Float: case we.Stencil8: case we.Depth24Plus: case we.EACR11Unorm: case we.EACR11Snorm: return 1; case we.RG8Unorm: case we.RG8Snorm: case we.RG8Uint: case we.RG8Sint: case we.Depth32FloatStencil8: case we.BC5RGUnorm: case we.BC5RGSnorm: case we.RG16Uint: case we.RG16Sint: case we.RG16Float: case we.RG32Uint: case we.RG32Sint: case we.RG32Float: case we.Depth24PlusStencil8: case we.EACRG11Unorm: case we.EACRG11Snorm: return 2; case we.RGB9E5UFloat: case we.RG11B10UFloat: case we.BC6HRGBUFloat: case we.BC6HRGBFloat: case we.ETC2RGB8Unorm: case we.ETC2RGB8UnormSRGB: return 3; case we.RGBA8Unorm: case we.RGBA8UnormSRGB: case we.RGBA8Snorm: case we.RGBA8Uint: case we.RGBA8Sint: case we.BGRA8Unorm: case we.BGRA8UnormSRGB: case we.RGB10A2UINT: case we.RGB10A2Unorm: case we.BC7RGBAUnorm: case we.BC7RGBAUnormSRGB: case we.BC3RGBAUnorm: case we.BC3RGBAUnormSRGB: case we.BC2RGBAUnorm: case we.BC2RGBAUnormSRGB: case we.BC1RGBAUnorm: case we.BC1RGBAUnormSRGB: case we.RGBA16Uint: case we.RGBA16Sint: case we.RGBA16Float: case we.RGBA32Uint: case we.RGBA32Sint: case we.RGBA32Float: case we.ETC2RGB8A1Unorm: case we.ETC2RGB8A1UnormSRGB: case we.ETC2RGBA8Unorm: case we.ETC2RGBA8UnormSRGB: case we.ASTC4x4Unorm: case we.ASTC4x4UnormSRGB: case we.ASTC5x4Unorm: case we.ASTC5x4UnormSRGB: case we.ASTC5x5Unorm: case we.ASTC5x5UnormSRGB: case we.ASTC6x5Unorm: case we.ASTC6x5UnormSRGB: case we.ASTC6x6Unorm: case we.ASTC6x6UnormSRGB: case we.ASTC8x5Unorm: case we.ASTC8x5UnormSRGB: case we.ASTC8x6Unorm: case we.ASTC8x6UnormSRGB: case we.ASTC8x8Unorm: case we.ASTC8x8UnormSRGB: case we.ASTC10x5Unorm: case we.ASTC10x5UnormSRGB: case we.ASTC10x6Unorm: case we.ASTC10x6UnormSRGB: case we.ASTC10x8Unorm: case we.ASTC10x8UnormSRGB: case we.ASTC10x10Unorm: case we.ASTC10x10UnormSRGB: case we.ASTC12x10Unorm: case we.ASTC12x10UnormSRGB: case we.ASTC12x12Unorm: case we.ASTC12x12UnormSRGB: return 4; } throw `Unknown format ${e}!`; } static HasStencilAspect(e) { switch (e) { case we.Stencil8: case we.Depth32FloatStencil8: case we.Depth24PlusStencil8: return !0; } return !1; } static HasDepthAndStencilAspects(e) { switch (e) { case we.Depth32FloatStencil8: case we.Depth24PlusStencil8: return !0; } return !1; } static GetDepthFormatOnly(e) { switch (e) { case we.Depth16Unorm: return we.Depth16Unorm; case we.Depth24Plus: return we.Depth24Plus; case we.Depth24PlusStencil8: return we.Depth24Plus; case we.Depth32Float: return we.Depth32Float; case we.Depth32FloatStencil8: return we.Depth32Float; } return e; } static GetSample(e) { return e > 1 ? 4 : 1; } copyVideoToTexture(e, t, r, n = !1, i) { var s, a, f, o; const d = i === void 0, [v, u] = this._getVideoPipeline(r, n ? QW.InvertY : QW.DontInvertY); d && (i = this._device.createCommandEncoder({})), (a = (s = i).pushDebugGroup) === null || a === void 0 || a.call(s, `copy video to texture - invertY=${n}`); const P = { colorAttachments: [ { view: t._hardwareTexture.underlyingResource.createView({ format: r, dimension: Da.E2d, mipLevelCount: 1, baseArrayLayer: 0, baseMipLevel: 0, arrayLayerCount: 1, aspect: zH.All }), loadOp: f9.Load, storeOp: Hc.Store } ] }, p = i.beginRenderPass(P), c = { layout: u, entries: [ { binding: 0, resource: this._videoSampler }, { binding: 1, resource: this._device.importExternalTexture({ source: e.underlyingResource }) } ] }, H = this._device.createBindGroup(c); p.setPipeline(v), p.setBindGroup(0, H), p.draw(4, 1, 0, 0), p.end(), (o = (f = i).popDebugGroup) === null || o === void 0 || o.call(f), d && (this._device.queue.submit([i.finish()]), i = null); } invertYPreMultiplyAlpha(e, t, r, n, i = !1, s = !1, a = 0, f = 0, o = 1, d = 0, v = 0, u = 0, l = 0, P, p) { var c, H, T, q, b, j; const w = u !== 0, m = P === void 0, [I, N] = this._getPipeline(n, w ? H0.InvertYPremultiplyAlphaWithOfst : H0.InvertYPremultiplyAlpha, { invertY: i, premultiplyAlpha: s }); a = Math.max(a, 0), m && (P = this._device.createCommandEncoder({})), (H = (c = P).pushDebugGroup) === null || H === void 0 || H.call(c, `internal process texture - invertY=${i} premultiplyAlpha=${s}`); let k; if (Ns._IsHardwareTexture(e) ? (k = e.underlyingResource, i && !s && o === 1 && a === 0 || (e = void 0)) : (k = e, e = void 0), !k) return; w && this._bufferManager.setRawData(this._ubCopyWithOfst, 0, new Float32Array([d, v, u, l]), 0, 4 * 4); const R = e, y = (T = R == null ? void 0 : R._copyInvertYTempTexture) !== null && T !== void 0 ? T : this.createTexture({ width: t, height: r, layers: 1 }, !1, !1, !1, !1, !1, n, 1, P, Po.CopySrc | Po.RenderAttachment | Po.TextureBinding, void 0, "TempTextureForCopyWithInvertY"), O = (q = R == null ? void 0 : R._copyInvertYRenderPassDescr) !== null && q !== void 0 ? q : { colorAttachments: [ { view: y.createView({ format: n, dimension: Da.E2d, baseMipLevel: 0, mipLevelCount: 1, arrayLayerCount: 1, baseArrayLayer: 0 }), loadOp: f9.Load, storeOp: Hc.Store } ] }, Y = P.beginRenderPass(O); let ee = w ? R == null ? void 0 : R._copyInvertYBindGroupWithOfst : R == null ? void 0 : R._copyInvertYBindGroup; if (!ee) { const Z = { layout: N, entries: [ { binding: 0, resource: k.createView({ format: n, dimension: Da.E2d, baseMipLevel: f, mipLevelCount: 1, arrayLayerCount: o, baseArrayLayer: a }) } ] }; w && Z.entries.push({ binding: 1, resource: { buffer: this._ubCopyWithOfst } }), ee = this._device.createBindGroup(Z); } Y.setPipeline(I), Y.setBindGroup(0, ee), Y.draw(4, 1, 0, 0), Y.end(), P.copyTextureToTexture({ texture: y }, { texture: k, mipLevel: f, origin: { x: 0, y: 0, z: a } }, { width: t, height: r, depthOrArrayLayers: 1 }), R ? (R._copyInvertYTempTexture = y, R._copyInvertYRenderPassDescr = O, w ? R._copyInvertYBindGroupWithOfst = ee : R._copyInvertYBindGroup = ee) : this._deferredReleaseTextures.push([y, null]), (j = (b = P).popDebugGroup) === null || j === void 0 || j.call(b), m && (this._device.queue.submit([P.finish()]), P = null); } copyWithInvertY(e, t, r, n) { var i, s, a, f; const o = n === void 0, [d, v] = this._getPipeline(t, H0.InvertYPremultiplyAlpha, { invertY: !0, premultiplyAlpha: !1 }); o && (n = this._device.createCommandEncoder({})), (s = (i = n).pushDebugGroup) === null || s === void 0 || s.call(i, "internal copy texture with invertY"); const u = n.beginRenderPass(r), l = this._device.createBindGroup({ layout: v, entries: [ { binding: 0, resource: e } ] }); u.setPipeline(d), u.setBindGroup(0, l), u.draw(4, 1, 0, 0), u.end(), (f = (a = n).popDebugGroup) === null || f === void 0 || f.call(a), o && (this._device.queue.submit([n.finish()]), n = null); } //------------------------------------------------------------------------------ // Creation //------------------------------------------------------------------------------ createTexture(e, t = !1, r = !1, n = !1, i = !1, s = !1, a = we.RGBA8Unorm, f = 1, o, d = -1, v = 0, u) { f = Ns.GetSample(f); const l = e.layers || 1, P = { width: e.width, height: e.height, depthOrArrayLayers: l }, p = F2[a] ? Po.RenderAttachment : 0, c = Ns.IsCompressedFormat(a), H = t ? Ns.ComputeNumMipmapLevels(e.width, e.height) : 1, T = d >= 0 ? d : Po.CopySrc | Po.CopyDst | Po.TextureBinding; v |= t && !c ? Po.CopySrc | p : 0, !c && !s && (v |= p | Po.CopyDst); const q = this._device.createTexture({ label: `Texture${s ? "3D" : "2D"}_${u ? u + "_" : ""}${P.width}x${P.height}x${P.depthOrArrayLayers}_${t ? "wmips" : "womips"}_${a}_samples${f}`, size: P, dimension: s ? Hp.E3d : Hp.E2d, format: a, usage: T | v, sampleCount: f, mipLevelCount: H }); return Ns.IsImageBitmap(e) && (this.updateTexture(e, q, e.width, e.height, l, a, 0, 0, n, i, 0, 0), t && r && this.generateMipmaps(q, a, H, 0, o)), q; } createCubeTexture(e, t = !1, r = !1, n = !1, i = !1, s = we.RGBA8Unorm, a = 1, f, o = -1, d = 0, v) { a = Ns.GetSample(a); const u = Ns.IsImageBitmapArray(e) ? e[0].width : e.width, l = Ns.IsImageBitmapArray(e) ? e[0].height : e.height, P = F2[s] ? Po.RenderAttachment : 0, p = Ns.IsCompressedFormat(s), c = t ? Ns.ComputeNumMipmapLevels(u, l) : 1, H = o >= 0 ? o : Po.CopySrc | Po.CopyDst | Po.TextureBinding; d |= t && !p ? Po.CopySrc | P : 0, p || (d |= P | Po.CopyDst); const T = this._device.createTexture({ label: `TextureCube_${v ? v + "_" : ""}${u}x${l}x6_${t ? "wmips" : "womips"}_${s}_samples${a}`, size: { width: u, height: l, depthOrArrayLayers: 6 }, dimension: Hp.E2d, format: s, usage: H | d, sampleCount: a, mipLevelCount: c }); return Ns.IsImageBitmapArray(e) && (this.updateCubeTextures(e, T, u, l, s, n, i, 0, 0), t && r && this.generateCubeMipmaps(T, s, c, f)), T; } generateCubeMipmaps(e, t, r, n) { var i, s, a, f; const o = n === void 0; o && (n = this._device.createCommandEncoder({})), (s = (i = n).pushDebugGroup) === null || s === void 0 || s.call(i, `create cube mipmaps - ${r} levels`); for (let d = 0; d < 6; ++d) this.generateMipmaps(e, t, r, d, n); (f = (a = n).popDebugGroup) === null || f === void 0 || f.call(a), o && (this._device.queue.submit([n.finish()]), n = null); } generateMipmaps(e, t, r, n = 0, i) { var s, a, f, o, d, v, u, l; const P = i === void 0, [p, c] = this._getPipeline(t); n = Math.max(n, 0), P && (i = this._device.createCommandEncoder({})), (a = (s = i).pushDebugGroup) === null || a === void 0 || a.call(s, `create mipmaps for face #${n} - ${r} levels`); let H; if (Ns._IsHardwareTexture(e) ? (H = e.underlyingResource, e._mipmapGenRenderPassDescr = e._mipmapGenRenderPassDescr || [], e._mipmapGenBindGroup = e._mipmapGenBindGroup || []) : (H = e, e = void 0), !H) return; const T = e; for (let q = 1; q < r; ++q) { const b = (o = (f = T == null ? void 0 : T._mipmapGenRenderPassDescr[n]) === null || f === void 0 ? void 0 : f[q - 1]) !== null && o !== void 0 ? o : { colorAttachments: [ { view: H.createView({ format: t, dimension: Da.E2d, baseMipLevel: q, mipLevelCount: 1, arrayLayerCount: 1, baseArrayLayer: n }), loadOp: f9.Load, storeOp: Hc.Store } ] }; T && (T._mipmapGenRenderPassDescr[n] = T._mipmapGenRenderPassDescr[n] || [], T._mipmapGenRenderPassDescr[n][q - 1] = b); const j = i.beginRenderPass(b), w = (v = (d = T == null ? void 0 : T._mipmapGenBindGroup[n]) === null || d === void 0 ? void 0 : d[q - 1]) !== null && v !== void 0 ? v : this._device.createBindGroup({ layout: c, entries: [ { binding: 0, resource: this._mipmapSampler }, { binding: 1, resource: H.createView({ format: t, dimension: Da.E2d, baseMipLevel: q - 1, mipLevelCount: 1, arrayLayerCount: 1, baseArrayLayer: n }) } ] }); T && (T._mipmapGenBindGroup[n] = T._mipmapGenBindGroup[n] || [], T._mipmapGenBindGroup[n][q - 1] = w), j.setPipeline(p), j.setBindGroup(0, w), j.draw(4, 1, 0, 0), j.end(); } (l = (u = i).popDebugGroup) === null || l === void 0 || l.call(u), P && (this._device.queue.submit([i.finish()]), i = null); } createGPUTextureForInternalTexture(e, t, r, n, i) { e._hardwareTexture || (e._hardwareTexture = new DC()), t === void 0 && (t = e.width), r === void 0 && (r = e.height), n === void 0 && (n = e.depth); const s = e._hardwareTexture, a = ((i ?? 0) & 1) !== 0; s.format = Ns.GetWebGPUTextureFormat(e.type, e.format, e._useSRGBBuffer), s.textureUsages = e._source === ri.RenderTarget || e.source === ri.MultiRenderTarget ? Po.TextureBinding | Po.CopySrc | Po.RenderAttachment : e._source === ri.DepthStencil ? Po.TextureBinding | Po.RenderAttachment : -1, s.textureAdditionalUsages = a ? Po.StorageBinding : 0; const f = e.generateMipMaps, o = n || 1; let d; if (e._maxLodLevel !== null ? d = e._maxLodLevel : d = f ? Ns.ComputeNumMipmapLevels(t, r) : 1, e.isCube) { const v = this.createCubeTexture({ width: t, height: r }, e.generateMipMaps, e.generateMipMaps, e.invertY, !1, s.format, 1, this._commandEncoderForCreation, s.textureUsages, s.textureAdditionalUsages, e.label); s.set(v), s.createView({ format: Ns.GetDepthFormatOnly(s.format), dimension: Da.Cube, mipLevelCount: d, baseArrayLayer: 0, baseMipLevel: 0, arrayLayerCount: 6, aspect: Ns.HasDepthAndStencilAspects(s.format) ? zH.DepthOnly : zH.All }, a); } else { const v = this.createTexture({ width: t, height: r, layers: o }, e.generateMipMaps, e.generateMipMaps, e.invertY, !1, e.is3D, s.format, 1, this._commandEncoderForCreation, s.textureUsages, s.textureAdditionalUsages, e.label); s.set(v), s.createView({ format: Ns.GetDepthFormatOnly(s.format), dimension: e.is2DArray ? Da.E2dArray : e.is3D ? Hp.E3d : Da.E2d, mipLevelCount: d, baseArrayLayer: 0, baseMipLevel: 0, arrayLayerCount: e.is3D ? 1 : o, aspect: Ns.HasDepthAndStencilAspects(s.format) ? zH.DepthOnly : zH.All }, a); } return e.width = e.baseWidth = t, e.height = e.baseHeight = r, e.depth = e.baseDepth = n, this.createMSAATexture(e, e.samples), s; } createMSAATexture(e, t, r = !0, n = -1) { const i = e._hardwareTexture; if (r && (i == null || i.releaseMSAATexture()), !i || (t ?? 1) <= 1) return; const s = e.width, a = e.height, f = this.createTexture({ width: s, height: a, layers: 1 }, !1, !1, !1, !1, !1, i.format, t, this._commandEncoderForCreation, Po.RenderAttachment, 0, e.label ? "MSAA" + e.label : void 0); i.setMSAATexture(f, n); } //------------------------------------------------------------------------------ // Update //------------------------------------------------------------------------------ updateCubeTextures(e, t, r, n, i, s = !1, a = !1, f = 0, o = 0) { const d = [0, 3, 1, 4, 2, 5]; for (let v = 0; v < d.length; ++v) { const u = e[d[v]]; this.updateTexture(u, t, r, n, 1, i, v, 0, s, a, f, o); } } // TODO WEBGPU handle data source not being in the same format than the destination texture? updateTexture(e, t, r, n, i, s, a = 0, f = 0, o = !1, d = !1, v = 0, u = 0, l) { const P = Ns._IsInternalTexture(t) ? t._hardwareTexture.underlyingResource : t, p = Ns._GetBlockInformationFromFormat(s), c = Ns._IsInternalTexture(t) ? t._hardwareTexture : t, H = { texture: P, origin: { x: v, y: u, z: Math.max(a, 0) }, mipLevel: f, premultipliedAlpha: d }, T = { width: Math.ceil(r / p.width) * p.width, height: Math.ceil(n / p.height) * p.height, depthOrArrayLayers: i || 1 }; if (e.byteLength !== void 0) { e = e; const q = Math.ceil(r / p.width) * p.length; if (Math.ceil(q / 256) * 256 === q) { const j = this._device.createCommandEncoder({}), w = this._bufferManager.createRawBuffer(e.byteLength, ga.MapWrite | ga.CopySrc, !0, "TempBufferForUpdateTexture" + (P ? "_" + P.label : "")), m = w.getMappedRange(); new Uint8Array(m).set(e), w.unmap(), j.copyBufferToTexture({ buffer: w, offset: 0, bytesPerRow: q, rowsPerImage: n }, H, T), this._device.queue.submit([j.finish()]), this._bufferManager.releaseBuffer(w); } else this._device.queue.writeTexture(H, e, { offset: 0, bytesPerRow: q, rowsPerImage: n }, T); if (o || d) if (Ns._IsInternalTexture(t)) { const j = v === 0 && u === 0 && r === t.width && n === t.height; this.invertYPreMultiplyAlpha(c, t.width, t.height, s, o, d, a, f, i || 1, v, u, j ? 0 : r, j ? 0 : n, void 0, l); } else throw "updateTexture: Can't process the texture data because a GPUTexture was provided instead of an InternalTexture!"; } else if (e = e, o) if (H.premultipliedAlpha = !1, Ns._IsInternalTexture(t) && v === 0 && u === 0 && r === t.width && n === t.height) this._device.queue.copyExternalImageToTexture({ source: e }, H, T), this.invertYPreMultiplyAlpha(c, r, n, s, o, d, a, f, i || 1, 0, 0, 0, 0, void 0, l); else { const q = this._device.createCommandEncoder({}), b = this.createTexture({ width: r, height: n, layers: 1 }, !1, !1, !1, !1, !1, s, 1, q, Po.CopySrc | Po.TextureBinding, void 0, "TempTextureForUpdateTexture"); this._deferredReleaseTextures.push([b, null]), T.depthOrArrayLayers = 1, this._device.queue.copyExternalImageToTexture({ source: e }, { texture: b }, T), T.depthOrArrayLayers = i || 1, this.invertYPreMultiplyAlpha(b, r, n, s, o, d, a, f, i || 1, 0, 0, 0, 0, q, l), q.copyTextureToTexture({ texture: b }, H, T), this._device.queue.submit([q.finish()]); } else this._device.queue.copyExternalImageToTexture({ source: e }, H, T); } readPixels(e, t, r, n, i, s, a = 0, f = 0, o = null, d = !1) { const v = Ns._GetBlockInformationFromFormat(s), u = Math.ceil(n / v.width) * v.length, l = Math.ceil(u / 256) * 256, P = l * i, p = this._bufferManager.createRawBuffer(P, ga.MapRead | ga.CopyDst, void 0, "TempBufferForReadPixels" + (e.label ? "_" + e.label : "")), c = this._device.createCommandEncoder({}); return c.copyTextureToBuffer({ texture: e, mipLevel: f, origin: { x: t, y: r, z: Math.max(a, 0) } }, { buffer: p, offset: 0, bytesPerRow: l }, { width: n, height: i, depthOrArrayLayers: 1 }), this._device.queue.submit([c.finish()]), this._bufferManager.readDataFromBuffer(p, P, n, i, u, l, Ns.GetTextureTypeFromFormat(s), 0, o, !0, d); } //------------------------------------------------------------------------------ // Dispose //------------------------------------------------------------------------------ releaseTexture(e) { if (Ns._IsInternalTexture(e)) { const t = e._hardwareTexture, r = e._irradianceTexture; this._deferredReleaseTextures.push([t, r]); } else this._deferredReleaseTextures.push([e, null]); } destroyDeferredTextures() { for (let e = 0; e < this._deferredReleaseTextures.length; ++e) { const [t, r] = this._deferredReleaseTextures[e]; t && (Ns._IsHardwareTexture(t) ? t.release() : t.destroy()), r == null || r.dispose(); } this._deferredReleaseTextures.length = 0; } } class qte extends tg { constructor(e, t = 0) { super(), this.capacity = t, this._buffer = e; } get underlyingResource() { return this._buffer; } } class tO { static _IsGPUBuffer(e) { return e.underlyingResource === void 0; } static _FlagsToString(e, t = "") { let r = t; for (let n = 0; n <= 9; ++n) e & 1 << n && (r && (r += "_"), r += ga[1 << n]); return r; } constructor(e) { this._deferredReleaseBuffers = [], this._device = e; } createRawBuffer(e, t, r = !1, n) { const i = e.byteLength !== void 0 ? e.byteLength + 3 & -4 : e + 3 & -4, s = { label: tO._FlagsToString(t, n ?? "Buffer") + "_size" + i, mappedAtCreation: r, size: i, usage: t }; return this._device.createBuffer(s); } createBuffer(e, t, r) { const n = e.byteLength !== void 0, i = this.createRawBuffer(e, t, void 0, r), s = new qte(i); return s.references = 1, s.capacity = n ? e.byteLength : e, n && this.setSubData(s, 0, e), s; } setRawData(e, t, r, n, i) { this._device.queue.writeBuffer(e, t, r.buffer, n, i); } setSubData(e, t, r, n = 0, i = 0) { const s = e.underlyingResource; i = i || r.byteLength, i = Math.min(i, e.capacity - t); let a = r.byteOffset + n, f = a + i; const o = i + 3 & -4; if (o !== i) { const u = new Uint8Array(r.buffer.slice(a, f)); r = new Uint8Array(o), r.set(u), n = 0, a = 0, f = o, i = o; } const d = 1024 * 1024 * 15; let v = 0; for (; f - (a + v) > d; ) this._device.queue.writeBuffer(s, t + v, r.buffer, a + v, d), v += d; this._device.queue.writeBuffer(s, t + v, r.buffer, a + v, i - v); } _getHalfFloatAsFloatRGBAArrayBuffer(e, t, r) { r || (r = new Float32Array(e)); const n = new Uint16Array(t); for (; e--; ) r[e] = QH(n[e]); return r; } readDataFromBuffer(e, t, r, n, i, s, a = 0, f = 0, o = null, d = !0, v = !1) { const u = a === 1 ? 2 : a === 2 ? 1 : 0; return new Promise((l, P) => { e.mapAsync(vD.Read, f, t).then(() => { const p = e.getMappedRange(f, t); let c = o; if (v) c === null ? c = KC(a, t, !0, p) : c = KC(a, c.buffer, void 0, p); else if (c === null) switch (u) { case 0: c = new Uint8Array(t), c.set(new Uint8Array(p)); break; case 1: c = this._getHalfFloatAsFloatRGBAArrayBuffer(t / 2, p); break; case 2: c = new Float32Array(t / 4), c.set(new Float32Array(p)); break; } else switch (u) { case 0: c = new Uint8Array(c.buffer), c.set(new Uint8Array(p)); break; case 1: c = this._getHalfFloatAsFloatRGBAArrayBuffer(t / 2, p, o); break; case 2: c = new Float32Array(c.buffer), c.set(new Float32Array(p)); break; } if (i !== s) { u === 1 && !v && (i *= 2, s *= 2); const H = new Uint8Array(c.buffer); let T = i, q = 0; for (let b = 1; b < n; ++b) { q = b * s; for (let j = 0; j < i; ++j) H[T++] = H[q++]; } u !== 0 && !v ? c = new Float32Array(H.buffer, 0, T / 4) : c = new Uint8Array(H.buffer, 0, T); } e.unmap(), d && this.releaseBuffer(e), l(c); }, (p) => P(p)); }); } releaseBuffer(e) { return tO._IsGPUBuffer(e) ? (this._deferredReleaseBuffers.push(e), !0) : (e.references--, e.references === 0 ? (this._deferredReleaseBuffers.push(e.underlyingResource), !0) : !1); } destroyDeferredBuffers() { for (let e = 0; e < this._deferredReleaseBuffers.length; ++e) this._deferredReleaseBuffers[e].destroy(); this._deferredReleaseBuffers.length = 0; } } const Dle = [ 0, 0, 3, 7, 0, 2, 6, 2, 4, 1, 5, 3, 1 // TEXTURE_LINEAR_NEAREST ], jle = [ 0, 64, 32, 96, 16, 80, 48, 112, 8 // ALWAYS ], wle = [ 0, 128, 128, 0, 0, 0, 0, 128, 0, 0, 0, 0, 128 // TEXTURE_LINEAR_NEAREST ]; class rD { constructor(e) { this._samplers = {}, this._device = e, this.disabled = !1; } static GetSamplerHashCode(e) { var t, r, n; const i = e._cachedAnisotropicFilteringLevel && e._cachedAnisotropicFilteringLevel > 1 ? 4 : 1; return Dle[e.samplingMode] + jle[(e._comparisonFunction || 514) - 512 + 1] + wle[e.samplingMode] + // handle the lodMinClamp = lodMaxClamp = 0 case when no filter used for mip mapping (((t = e._cachedWrapU) !== null && t !== void 0 ? t : 1) << 8) + (((r = e._cachedWrapV) !== null && r !== void 0 ? r : 1) << 10) + (((n = e._cachedWrapR) !== null && n !== void 0 ? n : 1) << 12) + ((e.useMipMaps ? 1 : 0) << 14) + // need to factor this in because _getSamplerFilterDescriptor depends on samplingMode AND useMipMaps! (i << 15); } static _GetSamplerFilterDescriptor(e, t) { let r, n, i, s, a; const f = e.useMipMaps; switch (e.samplingMode) { case 11: r = Ji.Linear, n = Ji.Linear, i = Ji.Nearest, f || (s = a = 0); break; case 3: case 3: r = Ji.Linear, n = Ji.Linear, f ? i = Ji.Linear : (i = Ji.Nearest, s = a = 0); break; case 8: r = Ji.Nearest, n = Ji.Nearest, f ? i = Ji.Linear : (i = Ji.Nearest, s = a = 0); break; case 4: r = Ji.Nearest, n = Ji.Nearest, i = Ji.Nearest, f || (s = a = 0); break; case 5: r = Ji.Nearest, n = Ji.Linear, i = Ji.Nearest, f || (s = a = 0); break; case 6: r = Ji.Nearest, n = Ji.Linear, f ? i = Ji.Linear : (i = Ji.Nearest, s = a = 0); break; case 7: r = Ji.Nearest, n = Ji.Linear, i = Ji.Nearest, s = a = 0; break; case 1: case 1: r = Ji.Nearest, n = Ji.Nearest, i = Ji.Nearest, s = a = 0; break; case 9: r = Ji.Linear, n = Ji.Nearest, i = Ji.Nearest, f || (s = a = 0); break; case 10: r = Ji.Linear, n = Ji.Nearest, f ? i = Ji.Linear : (i = Ji.Nearest, s = a = 0); break; case 2: case 2: r = Ji.Linear, n = Ji.Linear, i = Ji.Nearest, s = a = 0; break; case 12: r = Ji.Linear, n = Ji.Nearest, i = Ji.Nearest, s = a = 0; break; default: r = Ji.Nearest, n = Ji.Nearest, i = Ji.Nearest, s = a = 0; break; } return t > 1 && (s !== 0 || a !== 0) && i !== Ji.Nearest ? { magFilter: Ji.Linear, minFilter: Ji.Linear, mipmapFilter: Ji.Linear, anisotropyEnabled: !0 } : { magFilter: r, minFilter: n, mipmapFilter: i, lodMinClamp: s, lodMaxClamp: a }; } static _GetWrappingMode(e) { switch (e) { case 1: return zw.Repeat; case 0: return zw.ClampToEdge; case 2: return zw.MirrorRepeat; } return zw.Repeat; } static _GetSamplerWrappingDescriptor(e) { return { addressModeU: this._GetWrappingMode(e._cachedWrapU), addressModeV: this._GetWrappingMode(e._cachedWrapV), addressModeW: this._GetWrappingMode(e._cachedWrapR) }; } static _GetSamplerDescriptor(e, t) { const r = e.useMipMaps && e._cachedAnisotropicFilteringLevel && e._cachedAnisotropicFilteringLevel > 1 ? 4 : 1, n = this._GetSamplerFilterDescriptor(e, r); return Object.assign(Object.assign(Object.assign({ label: t }, n), this._GetSamplerWrappingDescriptor(e)), { compare: e._comparisonFunction ? rD.GetCompareFunction(e._comparisonFunction) : void 0, maxAnisotropy: n.anisotropyEnabled ? r : 1 }); } static GetCompareFunction(e) { switch (e) { case 519: return q6.Always; case 514: return q6.Equal; case 516: return q6.Greater; case 518: return q6.GreaterEqual; case 513: return q6.Less; case 515: return q6.LessEqual; case 512: return q6.Never; case 517: return q6.NotEqual; default: return q6.Less; } } getSampler(e, t = !1, r = 0, n) { if (this.disabled) return this._device.createSampler(rD._GetSamplerDescriptor(e, n)); t ? r = 0 : r === 0 && (r = rD.GetSamplerHashCode(e)); let i = t ? void 0 : this._samplers[r]; return i || (i = this._device.createSampler(rD._GetSamplerDescriptor(e, n)), t || (this._samplers[r] = i)), i; } } var hf; (function(A) { A[A.StencilReadMask = 0] = "StencilReadMask", A[A.StencilWriteMask = 1] = "StencilWriteMask", A[A.DepthBias = 2] = "DepthBias", A[A.DepthBiasSlopeScale = 3] = "DepthBiasSlopeScale", A[A.DepthStencilState = 4] = "DepthStencilState", A[A.MRTAttachments1 = 5] = "MRTAttachments1", A[A.MRTAttachments2 = 6] = "MRTAttachments2", A[A.RasterizationState = 7] = "RasterizationState", A[A.ColorStates = 8] = "ColorStates", A[A.ShaderStage = 9] = "ShaderStage", A[A.TextureStage = 10] = "TextureStage", A[A.VertexState = 11] = "VertexState", A[A.NumStates = 12] = "NumStates"; })(hf || (hf = {})); const pC = { 0: 1, 1: 2, 768: 3, 769: 4, 770: 5, 771: 6, 772: 7, 773: 8, 774: 9, 775: 10, 776: 11, 32769: 12, 32770: 13, 32771: 12, 32772: 13 // OneMinusBlendColor (alpha) }, SW = { 0: 0, 7680: 1, 7681: 2, 7682: 3, 7683: 4, 5386: 5, 34055: 6, 34056: 7 // DECR_WRAP }, mle = { [J.PositionKind]: !0, [J.NormalKind]: !0, [J.TangentKind]: !0, [J.UVKind]: !0, [J.UV2Kind]: !0, [J.UV3Kind]: !0, [J.UV4Kind]: !0, [J.UV5Kind]: !0, [J.UV6Kind]: !0, [J.ColorKind]: !0, [J.ColorInstanceKind]: !0, [J.MatricesIndicesKind]: !0, [J.MatricesWeightsKind]: !0, [J.MatricesIndicesExtraKind]: !0, [J.MatricesWeightsExtraKind]: !0 }; class co { static _IsSignedType(e) { switch (e) { case J.BYTE: case J.SHORT: case J.INT: case J.FLOAT: return !0; case J.UNSIGNED_BYTE: case J.UNSIGNED_SHORT: case J.UNSIGNED_INT: return !1; default: throw new Error(`Invalid type '${e}'`); } } constructor(e, t) { this.mrtTextureCount = 0, this._device = e, this._useTextureStage = !0, this._states = new Array(30), this._statesLength = 0, this._stateDirtyLowestIndex = 0, this._emptyVertexBuffer = t, this._mrtFormats = [], this._parameter = { token: void 0, pipeline: null }, this.disabled = !1, this.vertexBuffers = [], this._kMaxVertexBufferStride = e.limits.maxVertexBufferArrayStride || 2048, this.reset(); } reset() { this._isDirty = !0, this.vertexBuffers.length = 0, this.setAlphaToCoverage(!1), this.resetDepthCullingState(), this.setClampDepth(!1), this.setDepthBias(0), this._webgpuColorFormat = [we.BGRA8Unorm], this.setColorFormat(we.BGRA8Unorm), this.setMRT([]), this.setAlphaBlendEnabled(!1), this.setAlphaBlendFactors([null, null, null, null], [null, null]), this.setWriteMask(15), this.setDepthStencilFormat(we.Depth24PlusStencil8), this.setStencilEnabled(!1), this.resetStencilState(), this.setBuffers(null, null, null), this._setTextureState(0); } get colorFormats() { return this._mrtAttachments1 > 0 ? this._mrtFormats : this._webgpuColorFormat; } getRenderPipeline(e, t, r, n = 0) { if (r = Ns.GetSample(r), this.disabled) { const s = co._GetTopology(e); return this._setVertexState(t), this._setTextureState(n), this._parameter.pipeline = this._createRenderPipeline(t, s, r), co.NumCacheMiss++, co._NumPipelineCreationCurrentFrame++, this._parameter.pipeline; } if (this._setShaderStage(t.uniqueId), this._setRasterizationState(e, r), this._setColorStates(), this._setDepthStencilState(), this._setVertexState(t), this._setTextureState(n), this.lastStateDirtyLowestIndex = this._stateDirtyLowestIndex, !this._isDirty && this._parameter.pipeline) return this._stateDirtyLowestIndex = this._statesLength, co.NumCacheHitWithoutHash++, this._parameter.pipeline; if (this._getRenderPipeline(this._parameter), this._isDirty = !1, this._stateDirtyLowestIndex = this._statesLength, this._parameter.pipeline) return co.NumCacheHitWithHash++, this._parameter.pipeline; const i = co._GetTopology(e); return this._parameter.pipeline = this._createRenderPipeline(t, i, r), this._setRenderPipeline(this._parameter), co.NumCacheMiss++, co._NumPipelineCreationCurrentFrame++, this._parameter.pipeline; } endFrame() { co.NumPipelineCreationLastFrame = co._NumPipelineCreationCurrentFrame, co._NumPipelineCreationCurrentFrame = 0; } setAlphaToCoverage(e) { this._alphaToCoverageEnabled = e; } setFrontFace(e) { this._frontFace = e; } setCullEnabled(e) { this._cullEnabled = e; } setCullFace(e) { this._cullFace = e; } setClampDepth(e) { this._clampDepth = e; } resetDepthCullingState() { this.setDepthCullingState(!1, 2, 1, 0, 0, !0, !0, 519); } setDepthCullingState(e, t, r, n, i, s, a, f) { this._depthWriteEnabled = a, this._depthTestEnabled = s, this._depthCompare = (f ?? 519) - 512, this._cullFace = r, this._cullEnabled = e, this._frontFace = t, this.setDepthBiasSlopeScale(n), this.setDepthBias(i); } setDepthBias(e) { this._depthBias !== e && (this._depthBias = e, this._states[hf.DepthBias] = e, this._isDirty = !0, this._stateDirtyLowestIndex = Math.min(this._stateDirtyLowestIndex, hf.DepthBias)); } /*public setDepthBiasClamp(depthBiasClamp: number): void { if (this._depthBiasClamp !== depthBiasClamp) { this._depthBiasClamp = depthBiasClamp; this._states[StatePosition.DepthBiasClamp] = depthBiasClamp.toString(); this._isDirty = true; } }*/ setDepthBiasSlopeScale(e) { this._depthBiasSlopeScale !== e && (this._depthBiasSlopeScale = e, this._states[hf.DepthBiasSlopeScale] = e, this._isDirty = !0, this._stateDirtyLowestIndex = Math.min(this._stateDirtyLowestIndex, hf.DepthBiasSlopeScale)); } setColorFormat(e) { this._webgpuColorFormat[0] = e, this._colorFormat = F2[e ?? ""]; } setMRTAttachments(e) { this.mrtAttachments = e; let t = 0; for (let r = 0; r < e.length; ++r) e[r] !== 0 && (t += 1 << r); this._mrtEnabledMask !== t && (this._mrtEnabledMask = t, this._isDirty = !0, this._stateDirtyLowestIndex = Math.min(this._stateDirtyLowestIndex, hf.MRTAttachments1)); } setMRT(e, t) { var r, n; if (t = t ?? e.length, t > 10) throw "Can't handle more than 10 attachments for a MRT in cache render pipeline!"; this.mrtTextureArray = e, this.mrtTextureCount = t, this._mrtEnabledMask = 65535; const i = [0, 0]; let s = 0, a = 0, f = 0; for (let o = 0; o < t; ++o) { const d = e[o], v = d == null ? void 0 : d._hardwareTexture; this._mrtFormats[f] = (r = v == null ? void 0 : v.format) !== null && r !== void 0 ? r : this._webgpuColorFormat[0], i[s] += F2[(n = this._mrtFormats[f]) !== null && n !== void 0 ? n : ""] << a, a += 6, f++, a >= 32 && (a = 0, s++); } this._mrtFormats.length = f, (this._mrtAttachments1 !== i[0] || this._mrtAttachments2 !== i[1]) && (this._mrtAttachments1 = i[0], this._mrtAttachments2 = i[1], this._states[hf.MRTAttachments1] = i[0], this._states[hf.MRTAttachments2] = i[1], this._isDirty = !0, this._stateDirtyLowestIndex = Math.min(this._stateDirtyLowestIndex, hf.MRTAttachments1)); } setAlphaBlendEnabled(e) { this._alphaBlendEnabled = e; } setAlphaBlendFactors(e, t) { this._alphaBlendFuncParams = e, this._alphaBlendEqParams = t; } setWriteMask(e) { this._writeMask = e; } setDepthStencilFormat(e) { this._webgpuDepthStencilFormat = e, this._depthStencilFormat = e === void 0 ? 0 : F2[e]; } setDepthTestEnabled(e) { this._depthTestEnabled = e; } setDepthWriteEnabled(e) { this._depthWriteEnabled = e; } setDepthCompare(e) { this._depthCompare = (e ?? 519) - 512; } setStencilEnabled(e) { this._stencilEnabled = e; } setStencilCompare(e) { this._stencilFrontCompare = (e ?? 519) - 512; } setStencilDepthFailOp(e) { this._stencilFrontDepthFailOp = e === null ? 1 : SW[e]; } setStencilPassOp(e) { this._stencilFrontPassOp = e === null ? 2 : SW[e]; } setStencilFailOp(e) { this._stencilFrontFailOp = e === null ? 1 : SW[e]; } setStencilReadMask(e) { this._stencilReadMask !== e && (this._stencilReadMask = e, this._states[hf.StencilReadMask] = e, this._isDirty = !0, this._stateDirtyLowestIndex = Math.min(this._stateDirtyLowestIndex, hf.StencilReadMask)); } setStencilWriteMask(e) { this._stencilWriteMask !== e && (this._stencilWriteMask = e, this._states[hf.StencilWriteMask] = e, this._isDirty = !0, this._stateDirtyLowestIndex = Math.min(this._stateDirtyLowestIndex, hf.StencilWriteMask)); } resetStencilState() { this.setStencilState(!1, 519, 7680, 7681, 7680, 255, 255); } setStencilState(e, t, r, n, i, s, a) { this._stencilEnabled = e, this._stencilFrontCompare = (t ?? 519) - 512, this._stencilFrontDepthFailOp = r === null ? 1 : SW[r], this._stencilFrontPassOp = n === null ? 2 : SW[n], this._stencilFrontFailOp = i === null ? 1 : SW[i], this.setStencilReadMask(s), this.setStencilWriteMask(a); } setBuffers(e, t, r) { this._vertexBuffers = e, this._overrideVertexBuffers = r, this._indexBuffer = t; } static _GetTopology(e) { switch (e) { case 0: return Fl.TriangleList; case 2: return Fl.PointList; case 1: return Fl.LineList; case 3: return Fl.PointList; case 4: return Fl.LineList; case 5: throw "LineLoop is an unsupported fillmode in WebGPU"; case 6: return Fl.LineStrip; case 7: return Fl.TriangleStrip; case 8: throw "TriangleFan is an unsupported fillmode in WebGPU"; default: return Fl.TriangleList; } } static _GetAphaBlendOperation(e) { switch (e) { case 32774: return Z8.Add; case 32778: return Z8.Subtract; case 32779: return Z8.ReverseSubtract; case 32775: return Z8.Min; case 32776: return Z8.Max; default: return Z8.Add; } } static _GetAphaBlendFactor(e) { switch (e) { case 0: return ad.Zero; case 1: return ad.One; case 768: return ad.Src; case 769: return ad.OneMinusSrc; case 770: return ad.SrcAlpha; case 771: return ad.OneMinusSrcAlpha; case 772: return ad.DstAlpha; case 773: return ad.OneMinusDstAlpha; case 774: return ad.Dst; case 775: return ad.OneMinusDst; case 776: return ad.SrcAlphaSaturated; case 32769: return ad.Constant; case 32770: return ad.OneMinusConstant; case 32771: return ad.Constant; case 32772: return ad.OneMinusConstant; default: return ad.One; } } static _GetCompareFunction(e) { switch (e) { case 0: return q6.Never; case 1: return q6.Less; case 2: return q6.Equal; case 3: return q6.LessEqual; case 4: return q6.Greater; case 5: return q6.NotEqual; case 6: return q6.GreaterEqual; case 7: return q6.Always; } return q6.Never; } static _GetStencilOpFunction(e) { switch (e) { case 0: return T0.Zero; case 1: return T0.Keep; case 2: return T0.Replace; case 3: return T0.IncrementClamp; case 4: return T0.DecrementClamp; case 5: return T0.Invert; case 6: return T0.IncrementWrap; case 7: return T0.DecrementWrap; } return T0.Keep; } static _GetVertexInputDescriptorFormat(e) { const t = e.type, r = e.normalized, n = e.getSize(); switch (t) { case J.BYTE: switch (n) { case 1: case 2: return r ? jo.Snorm8x2 : jo.Sint8x2; case 3: case 4: return r ? jo.Snorm8x4 : jo.Sint8x4; } break; case J.UNSIGNED_BYTE: switch (n) { case 1: case 2: return r ? jo.Unorm8x2 : jo.Uint8x2; case 3: case 4: return r ? jo.Unorm8x4 : jo.Uint8x4; } break; case J.SHORT: switch (n) { case 1: case 2: return r ? jo.Snorm16x2 : jo.Sint16x2; case 3: case 4: return r ? jo.Snorm16x4 : jo.Sint16x4; } break; case J.UNSIGNED_SHORT: switch (n) { case 1: case 2: return r ? jo.Unorm16x2 : jo.Uint16x2; case 3: case 4: return r ? jo.Unorm16x4 : jo.Uint16x4; } break; case J.INT: switch (n) { case 1: return jo.Sint32; case 2: return jo.Sint32x2; case 3: return jo.Sint32x3; case 4: return jo.Sint32x4; } break; case J.UNSIGNED_INT: switch (n) { case 1: return jo.Uint32; case 2: return jo.Uint32x2; case 3: return jo.Uint32x3; case 4: return jo.Uint32x4; } break; case J.FLOAT: switch (n) { case 1: return jo.Float32; case 2: return jo.Float32x2; case 3: return jo.Float32x3; case 4: return jo.Float32x4; } break; } throw new Error(`Invalid Format '${e.getKind()}' - type=${t}, normalized=${r}, size=${n}`); } _getAphaBlendState() { return this._alphaBlendEnabled ? { srcFactor: co._GetAphaBlendFactor(this._alphaBlendFuncParams[2]), dstFactor: co._GetAphaBlendFactor(this._alphaBlendFuncParams[3]), operation: co._GetAphaBlendOperation(this._alphaBlendEqParams[1]) } : null; } _getColorBlendState() { return this._alphaBlendEnabled ? { srcFactor: co._GetAphaBlendFactor(this._alphaBlendFuncParams[0]), dstFactor: co._GetAphaBlendFactor(this._alphaBlendFuncParams[1]), operation: co._GetAphaBlendOperation(this._alphaBlendEqParams[0]) } : null; } _setShaderStage(e) { this._shaderId !== e && (this._shaderId = e, this._states[hf.ShaderStage] = e, this._isDirty = !0, this._stateDirtyLowestIndex = Math.min(this._stateDirtyLowestIndex, hf.ShaderStage)); } _setRasterizationState(e, t) { const r = this._frontFace, n = this._cullEnabled ? this._cullFace : 0, i = this._clampDepth ? 1 : 0, s = this._alphaToCoverageEnabled ? 1 : 0, a = r - 1 + (n << 1) + (i << 3) + (s << 4) + (e << 5) + (t << 8); this._rasterizationState !== a && (this._rasterizationState = a, this._states[hf.RasterizationState] = this._rasterizationState, this._isDirty = !0, this._stateDirtyLowestIndex = Math.min(this._stateDirtyLowestIndex, hf.RasterizationState)); } _setColorStates() { let e = ((this._writeMask ? 1 : 0) << 22) + (this._colorFormat << 23) + ((this._depthWriteEnabled ? 1 : 0) << 29); this._alphaBlendEnabled && (e += ((this._alphaBlendFuncParams[0] === null ? 2 : pC[this._alphaBlendFuncParams[0]]) << 0) + ((this._alphaBlendFuncParams[1] === null ? 2 : pC[this._alphaBlendFuncParams[1]]) << 4) + ((this._alphaBlendFuncParams[2] === null ? 2 : pC[this._alphaBlendFuncParams[2]]) << 8) + ((this._alphaBlendFuncParams[3] === null ? 2 : pC[this._alphaBlendFuncParams[3]]) << 12) + ((this._alphaBlendEqParams[0] === null ? 1 : this._alphaBlendEqParams[0] - 32773) << 16) + ((this._alphaBlendEqParams[1] === null ? 1 : this._alphaBlendEqParams[1] - 32773) << 19)), e !== this._colorStates && (this._colorStates = e, this._states[hf.ColorStates] = this._colorStates, this._isDirty = !0, this._stateDirtyLowestIndex = Math.min(this._stateDirtyLowestIndex, hf.ColorStates)); } _setDepthStencilState() { const e = this._stencilEnabled ? this._stencilFrontCompare + (this._stencilFrontDepthFailOp << 3) + (this._stencilFrontPassOp << 6) + (this._stencilFrontFailOp << 9) : 591, t = this._depthStencilFormat + ((this._depthTestEnabled ? this._depthCompare : 7) << 6) + (e << 10); this._depthStencilState !== t && (this._depthStencilState = t, this._states[hf.DepthStencilState] = this._depthStencilState, this._isDirty = !0, this._stateDirtyLowestIndex = Math.min(this._stateDirtyLowestIndex, hf.DepthStencilState)); } _setVertexState(e) { var t, r; const n = this._statesLength; let i = hf.VertexState; const s = e._pipelineContext, a = s.shaderProcessingContext.attributeNamesFromEffect, f = s.shaderProcessingContext.attributeLocationsFromEffect; let o, d = 0; for (let v = 0; v < a.length; v++) { const u = f[v]; let l = (t = this._overrideVertexBuffers && this._overrideVertexBuffers[a[v]]) !== null && t !== void 0 ? t : this._vertexBuffers[a[v]]; l || (l = this._emptyVertexBuffer); const P = (r = l.effectiveBuffer) === null || r === void 0 ? void 0 : r.underlyingResource; if (l._validOffsetRange === void 0) { const c = l.effectiveByteOffset, H = l.getSize(!0), T = l.effectiveByteStride; l._validOffsetRange = c + H <= this._kMaxVertexBufferStride && T === 0 || T !== 0 && c + H <= T; } o && o === P && l._validOffsetRange || (this.vertexBuffers[d++] = l, o = l._validOffsetRange ? P : null); const p = l.hashCode + (u << 7); this._isDirty = this._isDirty || this._states[i] !== p, this._states[i++] = p; } this.vertexBuffers.length = d, this._statesLength = i, this._isDirty = this._isDirty || i !== n, this._isDirty && (this._stateDirtyLowestIndex = Math.min(this._stateDirtyLowestIndex, hf.VertexState)); } _setTextureState(e) { this._textureState !== e && (this._textureState = e, this._states[hf.TextureStage] = this._textureState, this._isDirty = !0, this._stateDirtyLowestIndex = Math.min(this._stateDirtyLowestIndex, hf.TextureStage)); } _createPipelineLayout(e) { if (this._useTextureStage) return this._createPipelineLayoutWithTextureStage(e); const t = [], r = e.shaderProcessingContext.bindGroupLayoutEntries; for (let n = 0; n < r.length; n++) { const i = r[n]; t[n] = this._device.createBindGroupLayout({ entries: i }); } return e.bindGroupLayouts[0] = t, this._device.createPipelineLayout({ bindGroupLayouts: t }); } _createPipelineLayoutWithTextureStage(e) { var t; const r = e.shaderProcessingContext, n = r.bindGroupLayoutEntries; let i = 1; for (let a = 0; a < n.length; a++) { const f = n[a]; for (let o = 0; o < f.length; o++) { const d = n[a][o]; if (d.texture) { const v = r.bindGroupLayoutEntryInfo[a][d.binding].name, u = r.availableTextures[v], l = u.autoBindSampler ? r.availableSamplers[v + Wo.AutoSamplerSuffix] : null; let P = u.sampleType, p = (t = l == null ? void 0 : l.type) !== null && t !== void 0 ? t : Xq.Filtering; if (this._textureState & i && P !== Ll.Depth && (u.autoBindSampler && (p = Xq.NonFiltering), P = Ll.UnfilterableFloat), d.texture.sampleType = P, l) { const c = r.bindGroupLayoutEntryInfo[l.binding.groupIndex][l.binding.bindingIndex].index; n[l.binding.groupIndex][c].sampler.type = p; } i = i << 1; } } } const s = []; for (let a = 0; a < n.length; ++a) s[a] = this._device.createBindGroupLayout({ entries: n[a] }); return e.bindGroupLayouts[this._textureState] = s, this._device.createPipelineLayout({ bindGroupLayouts: s }); } _getVertexInputDescriptor(e) { var t, r; const n = [], i = e._pipelineContext, s = i.shaderProcessingContext.attributeNamesFromEffect, a = i.shaderProcessingContext.attributeLocationsFromEffect; let f, o; for (let d = 0; d < s.length; d++) { const v = a[d]; let u = (t = this._overrideVertexBuffers && this._overrideVertexBuffers[s[d]]) !== null && t !== void 0 ? t : this._vertexBuffers[s[d]]; u || (u = this._emptyVertexBuffer); let l = (r = u.effectiveBuffer) === null || r === void 0 ? void 0 : r.underlyingResource, P = u.effectiveByteOffset; const p = !u._validOffsetRange; if (!(f && o && f === l) || p) { const c = { arrayStride: u.effectiveByteStride, stepMode: u.getIsInstanced() ? LI.Instance : LI.Vertex, attributes: [] }; n.push(c), o = c.attributes, p && (P = 0, l = null); } o.push({ shaderLocation: v, offset: P, format: co._GetVertexInputDescriptorFormat(u) }), f = l; } return n; } _processNonFloatVertexBuffers(e, t) { const r = e.engine._getShaderProcessor(e.shaderProcessingContext.shaderLanguage); let n = !1; for (const i in this._vertexBuffers) { const s = this._vertexBuffers[i]; if (!s || !mle[i]) continue; const a = s.normalized ? J.FLOAT : s.type, f = e.vertexBufferKindToType[i]; (a !== J.FLOAT && f === void 0 || f !== void 0 && f !== a) && (n = !0, e.vertexBufferKindToType[i] = a, a !== J.FLOAT && (r.vertexBufferKindToNumberOfComponents[i] = J.DeduceStride(i), co._IsSignedType(a) && (r.vertexBufferKindToNumberOfComponents[i] *= -1))); } n && t._processShaderCode(r, !0); } _createRenderPipeline(e, t, r) { var n, i, s; const a = e._pipelineContext, f = this._getVertexInputDescriptor(e), o = this._createPipelineLayout(a), d = [], v = this._getAphaBlendState(), u = this._getColorBlendState(); if (this._processNonFloatVertexBuffers(a, e), this._mrtAttachments1 > 0) for (let c = 0; c < this._mrtFormats.length; ++c) { const H = this._mrtFormats[c]; if (H) { const T = { format: H, writeMask: this._mrtEnabledMask & 1 << c ? this._writeMask : 0 }; v && u && (T.blend = { alpha: v, color: u }), d.push(T); } else d.push(null); } else if (this._webgpuColorFormat[0]) { const c = { format: this._webgpuColorFormat[0], writeMask: this._writeMask }; v && u && (c.blend = { alpha: v, color: u }), d.push(c); } else d.push(null); const l = { compare: co._GetCompareFunction( this._stencilEnabled ? this._stencilFrontCompare : 7 /* ALWAYS */ ), depthFailOp: co._GetStencilOpFunction( this._stencilEnabled ? this._stencilFrontDepthFailOp : 1 /* KEEP */ ), failOp: co._GetStencilOpFunction( this._stencilEnabled ? this._stencilFrontFailOp : 1 /* KEEP */ ), passOp: co._GetStencilOpFunction( this._stencilEnabled ? this._stencilFrontPassOp : 1 /* KEEP */ ) }; let P; (t === Fl.LineStrip || t === Fl.TriangleStrip) && (P = !this._indexBuffer || this._indexBuffer.is32Bits ? Tq.Uint32 : Tq.Uint16); const p = this._webgpuDepthStencilFormat ? Ns.HasStencilAspect(this._webgpuDepthStencilFormat) : !1; return this._device.createRenderPipeline({ label: `RenderPipeline_${(i = (n = d[0]) === null || n === void 0 ? void 0 : n.format) !== null && i !== void 0 ? i : "nooutput"}_${(s = this._webgpuDepthStencilFormat) !== null && s !== void 0 ? s : "nodepth"}_samples${r}_textureState${this._textureState}`, layout: o, vertex: { module: a.stages.vertexStage.module, entryPoint: a.stages.vertexStage.entryPoint, buffers: f }, primitive: { topology: t, stripIndexFormat: P, frontFace: this._frontFace === 1 ? MI.CCW : MI.CW, cullMode: this._cullEnabled ? this._cullFace === 2 ? eS.Front : eS.Back : eS.None }, fragment: a.stages.fragmentStage ? { module: a.stages.fragmentStage.module, entryPoint: a.stages.fragmentStage.entryPoint, targets: d } : void 0, multisample: { count: r /*mask, alphaToCoverageEnabled,*/ }, depthStencil: this._webgpuDepthStencilFormat === void 0 ? void 0 : { depthWriteEnabled: this._depthWriteEnabled, depthCompare: this._depthTestEnabled ? co._GetCompareFunction(this._depthCompare) : q6.Always, format: this._webgpuDepthStencilFormat, stencilFront: this._stencilEnabled && p ? l : void 0, stencilBack: this._stencilEnabled && p ? l : void 0, stencilReadMask: this._stencilEnabled && p ? this._stencilReadMask : void 0, stencilWriteMask: this._stencilEnabled && p ? this._stencilWriteMask : void 0, depthBias: this._depthBias, depthBiasClamp: this._depthBiasClamp, depthBiasSlopeScale: this._depthBiasSlopeScale } }); } } co.NumCacheHitWithoutHash = 0; co.NumCacheHitWithHash = 0; co.NumCacheMiss = 0; co.NumPipelineCreationLastFrame = 0; co._NumPipelineCreationCurrentFrame = 0; class bte { constructor() { this.values = {}; } count() { let e = 0, t = this.pipeline ? 1 : 0; for (const r in this.values) { const n = this.values[r], [i, s] = n.count(); e += i, t += s, e++; } return [e, t]; } } class N2 extends co { static GetNodeCounts() { const e = N2._Cache.count(); return { nodeCount: e[0], pipelineCount: e[1] }; } static _GetPipelines(e, t, r, n) { if (e.pipeline) { const i = r.slice(); i.length = n, t.push(i); } for (const i in e.values) { const s = e.values[i]; r[n] = parseInt(i), N2._GetPipelines(s, t, r, n + 1); } } static GetPipelines() { const e = []; return N2._GetPipelines(N2._Cache, e, [], 0), e; } constructor(e, t) { super(e, t), this._nodeStack = [], this._nodeStack[0] = N2._Cache; } _getRenderPipeline(e) { let t = this._nodeStack[this._stateDirtyLowestIndex]; for (let r = this._stateDirtyLowestIndex; r < this._statesLength; ++r) { let n = t.values[this._states[r]]; n || (n = new bte(), t.values[this._states[r]] = n), t = n, this._nodeStack[r + 1] = t; } e.token = t, e.pipeline = t.pipeline; } _setRenderPipeline(e) { e.token.pipeline = e.pipeline; } } N2._Cache = new bte(); class Ble extends KN { constructor(e) { super(!1), this._cache = e, this.reset(); } get func() { return this._func; } set func(e) { this._func !== e && (this._func = e, this._cache.setStencilCompare(e)); } get funcMask() { return this._funcMask; } set funcMask(e) { this._funcMask !== e && (this._funcMask = e, this._cache.setStencilReadMask(e)); } get opStencilFail() { return this._opStencilFail; } set opStencilFail(e) { this._opStencilFail !== e && (this._opStencilFail = e, this._cache.setStencilFailOp(e)); } get opDepthFail() { return this._opDepthFail; } set opDepthFail(e) { this._opDepthFail !== e && (this._opDepthFail = e, this._cache.setStencilDepthFailOp(e)); } get opStencilDepthPass() { return this._opStencilDepthPass; } set opStencilDepthPass(e) { this._opStencilDepthPass !== e && (this._opStencilDepthPass = e, this._cache.setStencilPassOp(e)); } get mask() { return this._mask; } set mask(e) { this._mask !== e && (this._mask = e, this._cache.setStencilWriteMask(e)); } get enabled() { return this._enabled; } set enabled(e) { this._enabled !== e && (this._enabled = e, this._cache.setStencilEnabled(e)); } reset() { super.reset(), this._cache.resetStencilState(); } apply() { var e; const t = (e = this.stencilMaterial) === null || e === void 0 ? void 0 : e.enabled; this.enabled = t ? this.stencilMaterial.enabled : this.stencilGlobal.enabled, this.enabled && (this.func = t ? this.stencilMaterial.func : this.stencilGlobal.func, this.funcRef = t ? this.stencilMaterial.funcRef : this.stencilGlobal.funcRef, this.funcMask = t ? this.stencilMaterial.funcMask : this.stencilGlobal.funcMask, this.opStencilFail = t ? this.stencilMaterial.opStencilFail : this.stencilGlobal.opStencilFail, this.opDepthFail = t ? this.stencilMaterial.opDepthFail : this.stencilGlobal.opDepthFail, this.opStencilDepthPass = t ? this.stencilMaterial.opStencilDepthPass : this.stencilGlobal.opStencilDepthPass, this.mask = t ? this.stencilMaterial.mask : this.stencilGlobal.mask); } } class Wle extends YN { /** * Initializes the state. * @param cache */ constructor(e) { super(!1), this._cache = e, this.reset(); } get zOffset() { return this._zOffset; } set zOffset(e) { this._zOffset !== e && (this._zOffset = e, this._isZOffsetDirty = !0, this._cache.setDepthBiasSlopeScale(e)); } get zOffsetUnits() { return this._zOffsetUnits; } set zOffsetUnits(e) { this._zOffsetUnits !== e && (this._zOffsetUnits = e, this._isZOffsetDirty = !0, this._cache.setDepthBias(e)); } get cullFace() { return this._cullFace; } set cullFace(e) { this._cullFace !== e && (this._cullFace = e, this._isCullFaceDirty = !0, this._cache.setCullFace(e ?? 1)); } get cull() { return this._cull; } set cull(e) { this._cull !== e && (this._cull = e, this._isCullDirty = !0, this._cache.setCullEnabled(!!e)); } get depthFunc() { return this._depthFunc; } set depthFunc(e) { this._depthFunc !== e && (this._depthFunc = e, this._isDepthFuncDirty = !0, this._cache.setDepthCompare(e)); } get depthMask() { return this._depthMask; } set depthMask(e) { this._depthMask !== e && (this._depthMask = e, this._isDepthMaskDirty = !0, this._cache.setDepthWriteEnabled(e)); } get depthTest() { return this._depthTest; } set depthTest(e) { this._depthTest !== e && (this._depthTest = e, this._isDepthTestDirty = !0, this._cache.setDepthTestEnabled(e)); } get frontFace() { return this._frontFace; } set frontFace(e) { this._frontFace !== e && (this._frontFace = e, this._isFrontFaceDirty = !0, this._cache.setFrontFace(e ?? 2)); } reset() { super.reset(), this._cache.resetDepthCullingState(); } apply() { } } class FQ { /** * Checks if a texture is an external or internal texture * @param texture the external or internal texture * @returns true if the texture is an external texture, else false */ static IsExternalTexture(e) { return e.underlyingResource !== void 0; } /** * Get the class name of the texture. * @returns "ExternalTexture" */ getClassName() { return "ExternalTexture"; } /** * Gets the underlying texture object */ get underlyingResource() { return this._video; } /** * Constructs the texture * @param video The video the texture should be wrapped around */ constructor(e) { this.useMipMaps = !1, this.type = 16, this.format = 4294967295, this._video = e, this.uniqueId = As._Counter++; } /** * Get if the texture is ready to be used (downloaded, converted, mip mapped...). * @returns true if fully ready */ isReady() { return this._video.readyState >= this._video.HAVE_CURRENT_DATA; } /** * Dispose the texture and release its associated resources. */ dispose() { } } class sy { get forceBindGroupCreation() { return this._numExternalTextures > 0; } get hasFloatOrDepthTextures() { return this._numFloatOrDepthTextures > 0; } constructor() { this.uniqueId = sy._Counter++, this.updateId = 0, this.textureState = 0, this.reset(); } reset() { this.samplers = {}, this.textures = {}, this.isDirty = !0, this._numFloatOrDepthTextures = 0, this._numExternalTextures = 0; } setSampler(e, t) { let r = this.samplers[e], n = -1; r ? n = r.hashCode : this.samplers[e] = r = { sampler: t, hashCode: 0 }, r.sampler = t, r.hashCode = t ? rD.GetSamplerHashCode(t) : 0; const i = n !== r.hashCode; i && this.updateId++, this.isDirty || (this.isDirty = i); } setTexture(e, t) { var r, n, i; let s = this.textures[e], a = -1; s ? a = (n = (r = s.texture) === null || r === void 0 ? void 0 : r.uniqueId) !== null && n !== void 0 ? n : -1 : this.textures[e] = s = { texture: t, isFloatOrDepthTexture: !1, isExternalTexture: !1 }, s.isExternalTexture && this._numExternalTextures--, s.isFloatOrDepthTexture && this._numFloatOrDepthTextures--, t ? (s.isFloatOrDepthTexture = t.type === 1 || t.format >= 13 && t.format <= 18, s.isExternalTexture = FQ.IsExternalTexture(t), s.isFloatOrDepthTexture && this._numFloatOrDepthTextures++, s.isExternalTexture && this._numExternalTextures++) : (s.isFloatOrDepthTexture = !1, s.isExternalTexture = !1), s.texture = t; const f = a !== ((i = t == null ? void 0 : t.uniqueId) !== null && i !== void 0 ? i : -1); f && this.updateId++, this.isDirty || (this.isDirty = f); } } sy._Counter = 0; class tV { isDirty(e) { return this._isDirty || this._materialContextUpdateId !== e; } resetIsDirty(e) { this._isDirty = !1, this._materialContextUpdateId = e; } get useInstancing() { return this._useInstancing; } set useInstancing(e) { this._useInstancing !== e && (e ? (this.indirectDrawBuffer = this._bufferManager.createRawBuffer(20, ga.CopyDst | ga.Indirect | ga.Storage, void 0, "IndirectDrawBuffer"), this._indirectDrawData = new Uint32Array(5), this._indirectDrawData[3] = 0, this._indirectDrawData[4] = 0) : (this.indirectDrawBuffer && this._bufferManager.releaseBuffer(this.indirectDrawBuffer), this.indirectDrawBuffer = void 0, this._indirectDrawData = void 0), this._useInstancing = e, this._currentInstanceCount = -1); } constructor(e) { this._bufferManager = e, this.uniqueId = tV._Counter++, this._useInstancing = !1, this._currentInstanceCount = 0, this.reset(); } reset() { this.buffers = {}, this._isDirty = !0, this._materialContextUpdateId = 0, this.fastBundle = void 0, this.bindGroups = void 0; } setBuffer(e, t) { var r; this._isDirty || (this._isDirty = (t == null ? void 0 : t.uniqueId) !== ((r = this.buffers[e]) === null || r === void 0 ? void 0 : r.uniqueId)), this.buffers[e] = t; } setIndirectData(e, t, r) { t === this._currentInstanceCount || !this.indirectDrawBuffer || !this._indirectDrawData || (this._currentInstanceCount = t, this._indirectDrawData[0] = e, this._indirectDrawData[1] = t, this._indirectDrawData[2] = r, this._bufferManager.setRawData(this.indirectDrawBuffer, 0, this._indirectDrawData, 0, 20)); } dispose() { this.indirectDrawBuffer && (this._bufferManager.releaseBuffer(this.indirectDrawBuffer), this.indirectDrawBuffer = void 0, this._indirectDrawData = void 0), this.fastBundle = void 0, this.bindGroups = void 0, this.buffers = void 0; } } tV._Counter = 0; class jC { constructor() { this.values = {}; } } class d1 { static get Statistics() { return { totalCreated: d1.NumBindGroupsCreatedTotal, lastFrameCreated: d1.NumBindGroupsCreatedLastFrame, lookupLastFrame: d1.NumBindGroupsLookupLastFrame, noLookupLastFrame: d1.NumBindGroupsNoLookupLastFrame }; } constructor(e, t, r) { this.disabled = !1, this._device = e, this._cacheSampler = t, this._engine = r; } endFrame() { d1.NumBindGroupsCreatedLastFrame = d1._NumBindGroupsCreatedCurrentFrame, d1.NumBindGroupsLookupLastFrame = d1._NumBindGroupsLookupCurrentFrame, d1.NumBindGroupsNoLookupLastFrame = d1._NumBindGroupsNoLookupCurrentFrame, d1._NumBindGroupsCreatedCurrentFrame = 0, d1._NumBindGroupsLookupCurrentFrame = 0, d1._NumBindGroupsNoLookupCurrentFrame = 0; } /** * Cache is currently based on the uniform/storage buffers, samplers and textures used by the binding groups. * Note that all uniform buffers have an offset of 0 in Babylon and we don't have a use case where we would have the same buffer used with different capacity values: * that means we don't need to factor in the offset/size of the buffer in the cache, only the id * @param webgpuPipelineContext * @param drawContext * @param materialContext */ getBindGroups(e, t, r) { var n, i, s, a, f, o, d, v, u, l; let P, p = d1._Cache; const c = this.disabled || r.forceBindGroupCreation; if (!c) { if (!t.isDirty(r.updateId) && !r.isDirty) return d1._NumBindGroupsNoLookupCurrentFrame++, t.bindGroups; for (const T of e.shaderProcessingContext.bufferNames) { const q = (i = (n = t.buffers[T]) === null || n === void 0 ? void 0 : n.uniqueId) !== null && i !== void 0 ? i : 0; let b = p.values[q]; b || (b = new jC(), p.values[q] = b), p = b; } for (const T of e.shaderProcessingContext.samplerNames) { const q = (a = (s = r.samplers[T]) === null || s === void 0 ? void 0 : s.hashCode) !== null && a !== void 0 ? a : 0; let b = p.values[q]; b || (b = new jC(), p.values[q] = b), p = b; } for (const T of e.shaderProcessingContext.textureNames) { const q = (d = (o = (f = r.textures[T]) === null || f === void 0 ? void 0 : f.texture) === null || o === void 0 ? void 0 : o.uniqueId) !== null && d !== void 0 ? d : 0; let b = p.values[q]; b || (b = new jC(), p.values[q] = b), p = b; } P = p.bindGroups; } if (t.resetIsDirty(r.updateId), r.isDirty = !1, P) return t.bindGroups = P, d1._NumBindGroupsLookupCurrentFrame++, P; P = [], t.bindGroups = P, c || (p.bindGroups = P), d1.NumBindGroupsCreatedTotal++, d1._NumBindGroupsCreatedCurrentFrame++; const H = e.bindGroupLayouts[r.textureState]; for (let T = 0; T < e.shaderProcessingContext.bindGroupLayoutEntries.length; T++) { const q = e.shaderProcessingContext.bindGroupLayoutEntries[T], b = e.shaderProcessingContext.bindGroupEntries[T]; for (let w = 0; w < q.length; w++) { const m = e.shaderProcessingContext.bindGroupLayoutEntries[T][w], I = e.shaderProcessingContext.bindGroupLayoutEntryInfo[T][m.binding], N = (v = I.nameInArrayOfTexture) !== null && v !== void 0 ? v : I.name; if (m.sampler) { const k = r.samplers[N]; if (k) { const R = k.sampler; if (!R) { this._engine.dbgSanityChecks && Se.Error(`Trying to bind a null sampler! entry=${JSON.stringify(m)}, name=${N}, bindingInfo=${JSON.stringify(k, (y, O) => y === "texture" ? "" : O)}, materialContext.uniqueId=${r.uniqueId}`, 50); continue; } b[w].resource = this._cacheSampler.getSampler(R, !1, k.hashCode, R.label); } else Se.Error(`Sampler "${N}" could not be bound. entry=${JSON.stringify(m)}, materialContext=${JSON.stringify(r, (R, y) => R === "texture" || R === "sampler" ? "" : y)}`, 50); } else if (m.texture || m.storageTexture) { const k = r.textures[N]; if (k) { if (this._engine.dbgSanityChecks && k.texture === null) { Se.Error(`Trying to bind a null texture! entry=${JSON.stringify(m)}, bindingInfo=${JSON.stringify(k, (y, O) => y === "texture" ? "" : O)}, materialContext.uniqueId=${r.uniqueId}`, 50); continue; } const R = k.texture._hardwareTexture; if (this._engine.dbgSanityChecks && (!R || m.texture && !R.view || m.storageTexture && !R.viewForWriting)) { Se.Error(`Trying to bind a null gpu texture or view! entry=${JSON.stringify(m)}, name=${N}, bindingInfo=${JSON.stringify(k, (y, O) => y === "texture" ? "" : O)}, isReady=${(u = k.texture) === null || u === void 0 ? void 0 : u.isReady}, materialContext.uniqueId=${r.uniqueId}`, 50); continue; } b[w].resource = m.storageTexture ? R.viewForWriting : R.view; } else Se.Error(`Texture "${N}" could not be bound. entry=${JSON.stringify(m)}, materialContext=${JSON.stringify(r, (R, y) => R === "texture" || R === "sampler" ? "" : y)}`, 50); } else if (m.externalTexture) { const k = r.textures[N]; if (k) { if (this._engine.dbgSanityChecks && k.texture === null) { Se.Error(`Trying to bind a null external texture! entry=${JSON.stringify(m)}, name=${N}, bindingInfo=${JSON.stringify(k, (y, O) => y === "texture" ? "" : O)}, materialContext.uniqueId=${r.uniqueId}`, 50); continue; } const R = k.texture.underlyingResource; if (this._engine.dbgSanityChecks && !R) { Se.Error(`Trying to bind a null gpu external texture! entry=${JSON.stringify(m)}, name=${N}, bindingInfo=${JSON.stringify(k, (y, O) => y === "texture" ? "" : O)}, isReady=${(l = k.texture) === null || l === void 0 ? void 0 : l.isReady}, materialContext.uniqueId=${r.uniqueId}`, 50); continue; } b[w].resource = this._device.importExternalTexture({ source: R }); } else Se.Error(`Texture "${N}" could not be bound. entry=${JSON.stringify(m)}, materialContext=${JSON.stringify(r, (R, y) => R === "texture" || R === "sampler" ? "" : y)}`, 50); } else if (m.buffer) { const k = t.buffers[N]; if (k) { const R = k.underlyingResource; b[w].resource.buffer = R, b[w].resource.size = k.capacity; } else Se.Error(`Can't find buffer "${N}". entry=${JSON.stringify(m)}, buffers=${JSON.stringify(t.buffers)}, drawContext.uniqueId=${t.uniqueId}`, 50); } } const j = H[T]; P[T] = this._device.createBindGroup({ layout: j, entries: b }); } return P; } } d1.NumBindGroupsCreatedTotal = 0; d1.NumBindGroupsCreatedLastFrame = 0; d1.NumBindGroupsLookupLastFrame = 0; d1.NumBindGroupsNoLookupLastFrame = 0; d1._Cache = new jC(); d1._NumBindGroupsCreatedCurrentFrame = 0; d1._NumBindGroupsLookupCurrentFrame = 0; d1._NumBindGroupsNoLookupCurrentFrame = 0; const Sle = "clearQuadVertexShader", Ule = `uniform float depthValue;const vec2 pos[4]={vec2(-1.0,1.0), vec2(1.0,1.0), vec2(-1.0,-1.0), vec2(1.0,-1.0)}; #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN gl_Position=vec4(pos[gl_VertexID],depthValue,1.0); #define CUSTOM_VERTEX_MAIN_END } `; Le.ShadersStore[Sle] = Ule; const Ile = "clearQuadPixelShader", Rle = `uniform vec4 color;void main() {gl_FragColor=color;} `; Le.ShadersStore[Ile] = Rle; class Vle { setDepthStencilFormat(e) { this._depthTextureFormat = e, this._cacheRenderPipeline.setDepthStencilFormat(e); } setColorFormat(e) { this._cacheRenderPipeline.setColorFormat(e); } setMRTAttachments(e, t, r) { this._cacheRenderPipeline.setMRT(t, r), this._cacheRenderPipeline.setMRTAttachments(e); } constructor(e, t, r) { this._bindGroups = {}, this._bundleCache = {}, this._keyTemp = [], this._device = e, this._engine = t, this._cacheRenderPipeline = new N2(this._device, r), this._cacheRenderPipeline.setDepthTestEnabled(!1), this._cacheRenderPipeline.setStencilReadMask(255), this._effect = t.createEffect("clearQuad", [], ["color", "depthValue"]); } clear(e, t, r, n, i = 1) { var s, a; let f, o = null, d; const v = !!this._engine._currentRenderTarget; if (e) f = e; else { let T = 0; this._keyTemp.length = 0; for (let b = 0; b < this._cacheRenderPipeline.colorFormats.length; ++b) this._keyTemp[T++] = F2[(s = this._cacheRenderPipeline.colorFormats[b]) !== null && s !== void 0 ? s : ""]; const q = F2[(a = this._depthTextureFormat) !== null && a !== void 0 ? a : 0]; if (this._keyTemp[T] = (t ? t.r + t.g * 256 + t.b * 256 * 256 + t.a * 256 * 256 * 256 : 0) + (r ? 2 ** 32 : 0) + (n ? 2 ** 33 : 0) + (this._engine.useReverseDepthBuffer ? 2 ** 34 : 0) + (v ? 2 ** 35 : 0) + (i > 1 ? 2 ** 36 : 0) + q * 2 ** 37, d = this._keyTemp.join("_"), o = this._bundleCache[d], o) return o; f = this._device.createRenderBundleEncoder({ colorFormats: this._cacheRenderPipeline.colorFormats, depthStencilFormat: this._depthTextureFormat, sampleCount: Ns.GetSample(i) }); } this._cacheRenderPipeline.setDepthWriteEnabled(!!r), this._cacheRenderPipeline.setStencilEnabled(!!n && !!this._depthTextureFormat && Ns.HasStencilAspect(this._depthTextureFormat)), this._cacheRenderPipeline.setStencilWriteMask(n ? 255 : 0), this._cacheRenderPipeline.setStencilCompare(n ? 519 : 512), this._cacheRenderPipeline.setStencilPassOp(n ? 7681 : 7680), this._cacheRenderPipeline.setWriteMask(t ? 15 : 0); const u = this._cacheRenderPipeline.getRenderPipeline(7, this._effect, i), l = this._effect._pipelineContext; t && this._effect.setDirectColor4("color", t), this._effect.setFloat("depthValue", this._engine.useReverseDepthBuffer ? this._engine._clearReverseDepthValue : this._engine._clearDepthValue), l.uniformBuffer.update(); const P = v ? this._engine._ubInvertY : this._engine._ubDontInvertY, p = l.uniformBuffer.getBuffer(), c = p.uniqueId + "-" + P.uniqueId; let H = this._bindGroups[c]; if (!H) { const T = l.bindGroupLayouts[0]; H = this._bindGroups[c] = [], H.push(this._device.createBindGroup({ layout: T[0], entries: [] })), gp._SimplifiedKnownBindings || H.push(this._device.createBindGroup({ layout: T[1], entries: [] })), H.push(this._device.createBindGroup({ layout: T[gp._SimplifiedKnownBindings ? 1 : 2], entries: [ { binding: 0, resource: { buffer: P.underlyingResource, size: P.capacity } }, { binding: 1, resource: { buffer: p.underlyingResource, size: p.capacity } } ] })); } f.setPipeline(u); for (let T = 0; T < H.length; ++T) f.setBindGroup(T, H[T]); return f.draw(4, 1, 0, 0), e || (o = f.finish(), this._bundleCache[d] = o), o; } } class NQ { constructor(e, t, r, n) { this.x = Math.floor(e), this.y = Math.floor(t), this.w = Math.floor(r), this.h = Math.floor(n); } run(e) { e.setViewport(this.x, this.y, this.w, this.h, 0, 1); } clone() { return new NQ(this.x, this.y, this.w, this.h); } } class QQ { constructor(e, t, r, n) { this.x = e, this.y = t, this.w = r, this.h = n; } run(e) { e.setScissorRect(this.x, this.y, this.w, this.h); } clone() { return new QQ(this.x, this.y, this.w, this.h); } } class rO { constructor(e) { this.ref = e; } run(e) { e.setStencilReference(this.ref); } clone() { return new rO(this.ref); } } class YQ { constructor(e) { this.color = e; } run(e) { e.setBlendConstant(this.color); } clone() { return new YQ(this.color); } } class MQ { constructor(e) { this.query = e; } run(e) { e.beginOcclusionQuery(this.query); } clone() { return new MQ(this.query); } } class LQ { constructor() { } run(e) { e.endOcclusionQuery(); } clone() { return new LQ(); } } class KQ { constructor() { this.bundles = []; } run(e) { e.executeBundles(this.bundles); } clone() { const e = new KQ(); return e.bundles = this.bundles, e; } } class JQ { constructor(e) { this.numDrawCalls = 0, this._device = e, this._list = new Array(10), this._listLength = 0; } addBundle(e) { if (!this._currentItemIsBundle) { const t = new KQ(); this._list[this._listLength++] = t, this._currentBundleList = t.bundles, this._currentItemIsBundle = !0; } e && this._currentBundleList.push(e); } _finishBundle() { this._currentItemIsBundle && this._bundleEncoder && (this._currentBundleList.push(this._bundleEncoder.finish()), this._bundleEncoder = void 0, this._currentItemIsBundle = !1); } addItem(e) { this._finishBundle(), this._list[this._listLength++] = e, this._currentItemIsBundle = !1; } getBundleEncoder(e, t, r) { return this._currentItemIsBundle || (this.addBundle(), this._bundleEncoder = this._device.createRenderBundleEncoder({ colorFormats: e, depthStencilFormat: t, sampleCount: Ns.GetSample(r) })), this._bundleEncoder; } close() { this._finishBundle(); } run(e) { this.close(); for (let t = 0; t < this._listLength; ++t) this._list[t].run(e); } reset() { this._listLength = 0, this._currentItemIsBundle = !1, this.numDrawCalls = 0; } clone() { this.close(); const e = new JQ(this._device); e._list = new Array(this._listLength), e._listLength = this._listLength, e.numDrawCalls = this.numDrawCalls; for (let t = 0; t < this._listLength; ++t) e._list[t] = this._list[t].clone(); return e; } } class xte { get querySet() { return this._querySet; } constructor(e, t, r, n, i = !0, s) { this._dstBuffers = [], this._device = r, this._bufferManager = n, this._count = e, this._canUseMultipleBuffers = i, this._querySet = r.createQuerySet({ label: s, type: t, count: e }), this._queryBuffer = n.createRawBuffer(8 * e, ga.QueryResolve | ga.CopySrc, void 0, "QueryBuffer"), i || this._dstBuffers.push(this._bufferManager.createRawBuffer(8 * this._count, ga.MapRead | ga.CopyDst, void 0, "QueryBufferNoMultipleBuffers")); } _getBuffer(e, t) { if (!this._canUseMultipleBuffers && this._dstBuffers.length === 0) return null; const r = this._device.createCommandEncoder(); let n; return this._dstBuffers.length === 0 ? n = this._bufferManager.createRawBuffer(8 * this._count, ga.MapRead | ga.CopyDst, void 0, "QueryBufferAdditionalBuffer") : (n = this._dstBuffers[this._dstBuffers.length - 1], this._dstBuffers.length--), r.resolveQuerySet(this._querySet, e, t, this._queryBuffer, 0), r.copyBufferToBuffer(this._queryBuffer, 0, n, 0, 8 * t), this._device.queue.submit([r.finish()]), n; } async readValues(e = 0, t = 1) { const r = this._getBuffer(e, t); if (r === null) return null; await r.mapAsync(vD.Read); const n = new BigUint64Array(r.getMappedRange()).slice(); return r.unmap(), this._dstBuffers[this._dstBuffers.length] = r, n; } async readValue(e = 0) { const t = this._getBuffer(e, 1); if (t === null) return null; await t.mapAsync(vD.Read); const r = new BigUint64Array(t.getMappedRange()), n = Number(r[0]); return t.unmap(), this._dstBuffers[this._dstBuffers.length] = t, n; } async readTwoValuesAndSubtract(e = 0) { const t = this._getBuffer(e, 2); if (t === null) return null; await t.mapAsync(vD.Read); const r = new BigUint64Array(t.getMappedRange()), n = Number(r[1] - r[0]); return t.unmap(), this._dstBuffers[this._dstBuffers.length] = t, n; } dispose() { this._querySet.destroy(), this._bufferManager.releaseBuffer(this._queryBuffer); for (let e = 0; e < this._dstBuffers.length; ++e) this._bufferManager.releaseBuffer(this._dstBuffers[e]); } } class Cle { get gpuFrameTimeCounter() { return this._gpuFrameTimeCounter; } constructor(e, t) { this._enabled = !1, this._gpuFrameTimeCounter = new v9(), this._measureDurationState = 0, this._device = e, this._bufferManager = t; } get enable() { return this._enabled; } set enable(e) { this._enabled !== e && (this._enabled = e, this._measureDurationState = 0, e ? this._measureDuration = new Ole(this._device, this._bufferManager) : this._measureDuration.dispose()); } startFrame(e) { this._enabled && this._measureDurationState === 0 && (this._measureDuration.start(e), this._measureDurationState = 1); } endFrame(e) { this._measureDurationState === 1 && (this._measureDurationState = 2, this._measureDuration.stop(e).then((t) => { t !== null && t >= 0 && (this._gpuFrameTimeCounter.fetchNewFrame(), this._gpuFrameTimeCounter.addCount(t, !0)), this._measureDurationState = 0; })); } } class Ole { constructor(e, t) { this._querySet = new xte(2, KI.Timestamp, e, t); } start(e) { e.writeTimestamp(this._querySet.querySet, 0); } async stop(e) { return e.writeTimestamp(this._querySet.querySet, 1), this._querySet.readTwoValuesAndSubtract(0); } dispose() { this._querySet.dispose(); } } class yle { get querySet() { return this._querySet.querySet; } get hasQueries() { return this._currentTotalIndices !== this._availableIndices.length; } canBeginQuery(e) { if (this._frameQuerySetIsDirty === this._engine.frameId || this._queryFrameId[e] === this._engine.frameId) return !1; const t = this._engine._getCurrentRenderPassWrapper().renderPassDescriptor.occlusionQuerySet !== void 0; return t && (this._queryFrameId[e] = this._engine.frameId), t; } constructor(e, t, r, n = 50, i = 100) { this._availableIndices = [], this._frameQuerySetIsDirty = -1, this._queryFrameId = [], this._engine = e, this._device = t, this._bufferManager = r, this._frameLastBuffer = -1, this._currentTotalIndices = 0, this._countIncrement = i, this._allocateNewIndices(n); } createQuery() { this._availableIndices.length === 0 && this._allocateNewIndices(); const e = this._availableIndices[this._availableIndices.length - 1]; return this._availableIndices.length--, e; } deleteQuery(e) { this._availableIndices[this._availableIndices.length] = e; } isQueryResultAvailable(e) { return this._retrieveQueryBuffer(), !!this._lastBuffer && e < this._lastBuffer.length; } getQueryResult(e) { var t, r; return Number((r = (t = this._lastBuffer) === null || t === void 0 ? void 0 : t[e]) !== null && r !== void 0 ? r : -1); } _retrieveQueryBuffer() { this._lastBuffer && this._frameLastBuffer === this._engine.frameId || this._frameLastBuffer !== this._engine.frameId && (this._frameLastBuffer = this._engine.frameId, this._querySet.readValues(0, this._currentTotalIndices).then((e) => { this._lastBuffer = e; })); } _allocateNewIndices(e) { e = e ?? this._countIncrement, this._delayQuerySetDispose(); for (let t = 0; t < e; ++t) this._availableIndices.push(this._currentTotalIndices + t); this._currentTotalIndices += e, this._querySet = new xte(this._currentTotalIndices, KI.Occlusion, this._device, this._bufferManager, !1, "QuerySet_OcclusionQuery_count_" + this._currentTotalIndices), this._frameQuerySetIsDirty = this._engine.frameId; } _delayQuerySetDispose() { const e = this._querySet; e && setTimeout(() => e.dispose, 1e3); } dispose() { var e; (e = this._querySet) === null || e === void 0 || e.dispose(), this._availableIndices.length = 0; } } class Nl { async initTwgsl(e) { if (!Nl._twgsl) return e = e || {}, e = Object.assign(Object.assign({}, Nl._TWgslDefaultOptions), e), e.twgsl ? (Nl._twgsl = e.twgsl, Promise.resolve()) : (e.jsPath && e.wasmPath && await ye.LoadBabylonScriptAsync(e.jsPath), self.twgsl ? (Nl._twgsl = await self.twgsl(ye.GetBabylonScriptURL(e.wasmPath)), Promise.resolve()) : Promise.reject("twgsl is not available.")); } convertSpirV2WGSL(e, t = !1) { const r = Nl._twgsl.convertSpirV2WGSL(e, Nl.DisableUniformityAnalysis || t); return Nl.ShowWGSLShaderCode && (console.log(r), console.log("***********************************************")), Nl.DisableUniformityAnalysis || t ? `diagnostic(off, derivative_uniformity); ` + r : r; } } Nl._TWgslDefaultOptions = { jsPath: `${ye._DefaultCdnUrl}/twgsl/twgsl.js`, wasmPath: `${ye._DefaultCdnUrl}/twgsl/twgsl.wasm` }; Nl.ShowWGSLShaderCode = !1; Nl.DisableUniformityAnalysis = !1; Nl._twgsl = null; class kle { constructor(e, t, r) { this._record = !1, this._play = !1, this._playBundleListIndex = 0, this._allBundleLists = [], this._enabled = !1, this._engine = e, this._mode = t, this._bundleList = r; } get enabled() { return this._enabled; } get play() { return this._play; } get record() { return this._record; } set enabled(e) { this._allBundleLists.length = 0, this._record = this._enabled = e, this._play = !1, e && (this._modeSaved = this._mode, this._mode = 0); } get mode() { return this._mode; } set mode(e) { this._record ? this._modeSaved = e : this._mode = e; } endRenderPass(e) { if (!this._record && !this._play) return !1; let t; if (this._record) t = this._bundleList.clone(), this._allBundleLists.push(t), this._bundleList.reset(); else { if (this._playBundleListIndex >= this._allBundleLists.length) throw new Error(`Invalid playBundleListIndex! Your snapshot is no longer valid for the current frame, you should recreate a new one. playBundleListIndex=${this._playBundleListIndex}, allBundleLists.length=${this._allBundleLists.length}}`); t = this._allBundleLists[this._playBundleListIndex++]; } return t.run(e), this._mode === 1 && this._engine._reportDrawCall(t.numDrawCalls), !0; } endFrame() { this._record && (this._record = !1, this._play = !0, this._mode = this._modeSaved), this._playBundleListIndex = 0; } reset() { this.enabled = !1, this.enabled = !0; } } const Ele = "postprocessVertexShader", Fle = `attribute position: vec2;uniform scale: vec2;varying vUV: vec2;const madd=vec2(0.5,0.5); #define CUSTOM_VERTEX_DEFINITIONS @vertex fn main(input : VertexInputs)->FragmentInputs { #define CUSTOM_VERTEX_MAIN_BEGIN vertexOutputs.vUV=(vertexInputs.position*madd+madd)*uniforms.scale;vertexOutputs.position=vec4(vertexInputs.position,0.0,1.0); #define CUSTOM_VERTEX_MAIN_END } `; Le.ShadersStoreWGSL[Ele] = Fle; const DG = { label: "TextureView_SwapChain_ResolveTarget", dimension: Hp.E2d, format: void 0, mipLevelCount: 1, arrayLayerCount: 1 }, jG = { label: "TextureView_SwapChain", dimension: Hp.E2d, format: void 0, mipLevelCount: 1, arrayLayerCount: 1 }, hC = "/* disable_uniformity_analysis */", Nle = new xt(); class wn extends Ge { /** * Gets or sets the snapshot rendering mode */ get snapshotRenderingMode() { return this._snapshotRendering.mode; } set snapshotRenderingMode(e) { this._snapshotRendering.mode = e; } /** * Creates a new snapshot at the next frame using the current snapshotRenderingMode */ snapshotRenderingReset() { this._snapshotRendering.reset(); } /** * Enables or disables the snapshot rendering mode * Note that the WebGL engine does not support snapshot rendering so setting the value won't have any effect for this engine */ get snapshotRendering() { return this._snapshotRendering.enabled; } set snapshotRendering(e) { this._snapshotRendering.enabled = e; } /** * Sets this to true to disable the cache for the samplers. You should do it only for testing purpose! */ get disableCacheSamplers() { return this._cacheSampler ? this._cacheSampler.disabled : !1; } set disableCacheSamplers(e) { this._cacheSampler && (this._cacheSampler.disabled = e); } /** * Sets this to true to disable the cache for the render pipelines. You should do it only for testing purpose! */ get disableCacheRenderPipelines() { return this._cacheRenderPipeline ? this._cacheRenderPipeline.disabled : !1; } set disableCacheRenderPipelines(e) { this._cacheRenderPipeline && (this._cacheRenderPipeline.disabled = e); } /** * Sets this to true to disable the cache for the bind groups. You should do it only for testing purpose! */ get disableCacheBindGroups() { return this._cacheBindGroups ? this._cacheBindGroups.disabled : !1; } set disableCacheBindGroups(e) { this._cacheBindGroups && (this._cacheBindGroups.disabled = e); } /** * Gets a Promise indicating if the engine can be instantiated (ie. if a WebGPU context can be found) */ static get IsSupportedAsync() { return navigator.gpu ? navigator.gpu.requestAdapter().then((e) => !!e, () => !1).catch(() => !1) : Promise.resolve(!1); } /** * Not supported by WebGPU, you should call IsSupportedAsync instead! */ static get IsSupported() { return Se.Warn("You must call IsSupportedAsync for WebGPU!"), !1; } /** * Gets a boolean indicating that the engine supports uniform buffers */ get supportsUniformBuffers() { return !0; } /** Gets the supported extensions by the WebGPU adapter */ get supportedExtensions() { return this._adapterSupportedExtensions; } /** Gets the currently enabled extensions on the WebGPU device */ get enabledExtensions() { return this._deviceEnabledExtensions; } /** Gets the supported limits by the WebGPU adapter */ get supportedLimits() { return this._adapterSupportedLimits; } /** Gets the current limits of the WebGPU device */ get currentLimits() { return this._deviceLimits; } /** * Returns a string describing the current engine */ get description() { return this.name + this.version; } /** * Returns the version of the engine */ get version() { return 1; } /** * Gets an object containing information about the current engine context * @returns an object containing the vendor, the renderer and the version of the current engine context */ getInfo() { return { vendor: this._adapterInfo.vendor || "unknown vendor", renderer: this._adapterInfo.architecture || "unknown renderer", version: this._adapterInfo.description || "unknown version" }; } /** * (WebGPU only) True (default) to be in compatibility mode, meaning rendering all existing scenes without artifacts (same rendering than WebGL). * Setting the property to false will improve performances but may not work in some scenes if some precautions are not taken. * See https://doc.babylonjs.com/setup/support/webGPU/webGPUOptimization/webGPUNonCompatibilityMode for more details */ get compatibilityMode() { return this._compatibilityMode; } set compatibilityMode(e) { this._compatibilityMode = e; } /** @internal */ get currentSampleCount() { return this._currentRenderTarget ? this._currentRenderTarget.samples : this._mainPassSampleCount; } /** * Create a new instance of the gpu engine asynchronously * @param canvas Defines the canvas to use to display the result * @param options Defines the options passed to the engine to create the GPU context dependencies * @returns a promise that resolves with the created engine */ static CreateAsync(e, t = {}) { const r = new wn(e, t); return new Promise((n) => { r.initAsync(t.glslangOptions, t.twgslOptions).then(() => n(r)); }); } /** * Create a new instance of the gpu engine. * @param canvas Defines the canvas to use to display the result * @param options Defines the options passed to the engine to create the GPU context dependencies */ constructor(e, t = {}) { var r, n; if (super(null, (r = t.antialias) !== null && r !== void 0 ? r : !0, t), this._uploadEncoderDescriptor = { label: "upload" }, this._renderEncoderDescriptor = { label: "render" }, this._clearDepthValue = 1, this._clearReverseDepthValue = 0, this._clearStencilValue = 0, this._defaultSampleCount = 4, this._glslang = null, this._tintWASM = null, this._adapterInfo = { vendor: "", architecture: "", device: "", description: "" }, this._compiledComputeEffects = {}, this._counters = { numEnableEffects: 0, numEnableDrawWrapper: 0, numBundleCreationNonCompatMode: 0, numBundleReuseNonCompatMode: 0 }, this.countersLastFrame = { numEnableEffects: 0, numEnableDrawWrapper: 0, numBundleCreationNonCompatMode: 0, numBundleReuseNonCompatMode: 0 }, this.numMaxUncapturedErrors = 20, this._commandBuffers = [null, null], this._currentRenderPass = null, this._mainRenderPassWrapper = { renderPassDescriptor: null, colorAttachmentViewDescriptor: null, depthAttachmentViewDescriptor: null, colorAttachmentGPUTextures: [], depthTextureFormat: void 0 }, this._rttRenderPassWrapper = { renderPassDescriptor: null, colorAttachmentViewDescriptor: null, depthAttachmentViewDescriptor: null, colorAttachmentGPUTextures: [], depthTextureFormat: void 0 }, this._pendingDebugCommands = [], this._currentOverrideVertexBuffers = null, this._currentIndexBuffer = null, this._colorWriteLocal = !0, this._forceEnableEffect = !1, this.dbgShowShaderCode = !1, this.dbgSanityChecks = !0, this.dbgVerboseLogsForFirstFrames = !1, this.dbgVerboseLogsNumFrames = 10, this.dbgLogIfNotDrawWrapper = !0, this.dbgShowEmptyEnableEffectCalls = !0, this.isNDCHalfZRange = !0, this.hasOriginBottomLeft = !1, this._viewportsCurrent = { x: 0, y: 0, w: 0, h: 0 }, this._scissorsCurrent = { x: 0, y: 0, w: 0, h: 0 }, this._scissorCached = { x: 0, y: 0, z: 0, w: 0 }, this._stencilRefsCurrent = -1, this._blendColorsCurrent = [null, null, null, null], this._name = "WebGPU", t.deviceDescriptor = t.deviceDescriptor || {}, t.enableGPUDebugMarkers = (n = t.enableGPUDebugMarkers) !== null && n !== void 0 ? n : !1, Se.Log(`Babylon.js v${Ge.Version} - ${this.description} engine`), !navigator.gpu) { Se.Error("WebGPU is not supported by your browser."); return; } t.swapChainFormat = t.swapChainFormat || navigator.gpu.getPreferredCanvasFormat(), this._isWebGPU = !0, this._shaderPlatformName = "WEBGPU", this._renderingCanvas = e, this._options = t, this._mainPassSampleCount = t.antialias ? this._defaultSampleCount : 1, this._setupMobileChecks(), this._sharedInit(e), this._shaderProcessor = new Uue(), this._shaderProcessorWGSL = new Ple(); } //------------------------------------------------------------------------------ // Initialization //------------------------------------------------------------------------------ /** * Initializes the WebGPU context and dependencies. * @param glslangOptions Defines the GLSLang compiler options if necessary * @param twgslOptions Defines the Twgsl compiler options if necessary * @returns a promise notifying the readiness of the engine. */ initAsync(e, t) { var r; return this._initGlslang(e ?? ((r = this._options) === null || r === void 0 ? void 0 : r.glslangOptions)).then((n) => { var i; return this._glslang = n, this._tintWASM = wn.UseTWGSL ? new Nl() : null, this._tintWASM ? this._tintWASM.initTwgsl(t ?? ((i = this._options) === null || i === void 0 ? void 0 : i.twgslOptions)).then(() => navigator.gpu.requestAdapter(this._options), (s) => { throw Se.Error("Can not initialize twgsl!"), Se.Error(s), Error("WebGPU initializations stopped."); }) : navigator.gpu.requestAdapter(this._options); }, (n) => { throw Se.Error("Can not initialize glslang!"), Se.Error(n), Error("WebGPU initializations stopped."); }).then((n) => { var i, s, a; if (n) { this._adapter = n, this._adapterSupportedExtensions = [], (i = this._adapter.features) === null || i === void 0 || i.forEach((d) => this._adapterSupportedExtensions.push(d)), this._adapterSupportedLimits = this._adapter.limits, this._adapter.requestAdapterInfo().then((d) => { this._adapterInfo = d; }); const f = (s = this._options.deviceDescriptor) !== null && s !== void 0 ? s : {}, o = (a = f == null ? void 0 : f.requiredFeatures) !== null && a !== void 0 ? a : this._options.enableAllFeatures ? this._adapterSupportedExtensions : void 0; if (o) { const d = o, v = []; for (const u of d) this._adapterSupportedExtensions.indexOf(u) !== -1 && v.push(u); f.requiredFeatures = v; } if (this._options.setMaximumLimits && !f.requiredLimits) { f.requiredLimits = {}; for (const d in this._adapterSupportedLimits) f.requiredLimits[d] = this._adapterSupportedLimits[d]; } return this._adapter.requestDevice(f); } else throw "Could not retrieve a WebGPU adapter (adapter is null)."; }).then((n) => { var i, s; this._device = n, this._deviceEnabledExtensions = [], (i = this._device.features) === null || i === void 0 || i.forEach((f) => this._deviceEnabledExtensions.push(f)), this._deviceLimits = n.limits; let a = -1; this._device.addEventListener("uncapturederror", (f) => { ++a < this.numMaxUncapturedErrors ? Se.Warn(`WebGPU uncaptured error (${a + 1}): ${f.error} - ${f.error.message}`) : a++ === this.numMaxUncapturedErrors && Se.Warn(`WebGPU uncaptured error: too many warnings (${this.numMaxUncapturedErrors}), no more warnings will be reported to the console for this engine.`); }), this._doNotHandleContextLost || (s = this._device.lost) === null || s === void 0 || s.then((f) => { this._isDisposed || (this._contextWasLost = !0, Se.Warn("WebGPU context lost. " + f), this.onContextLostObservable.notifyObservers(this), this._restoreEngineAfterContextLost(() => this.initAsync())); }); }, (n) => { Se.Error("Could not retrieve a WebGPU device."), Se.Error(n); }).then(() => { this._bufferManager = new tO(this._device), this._textureHelper = new Ns(this._device, this._glslang, this._tintWASM, this._bufferManager, this._deviceEnabledExtensions), this._cacheSampler = new rD(this._device), this._cacheBindGroups = new d1(this._device, this._cacheSampler, this), this._timestampQuery = new Cle(this._device, this._bufferManager), this._occlusionQuery = this._device.createQuerySet ? new yle(this, this._device, this._bufferManager) : void 0, this._bundleList = new JQ(this._device), this._snapshotRendering = new kle(this, this._snapshotRenderingMode, this._bundleList), this._ubInvertY = this._bufferManager.createBuffer(new Float32Array([-1, 0]), ga.Uniform | ga.CopyDst, "UBInvertY"), this._ubDontInvertY = this._bufferManager.createBuffer(new Float32Array([1, 0]), ga.Uniform | ga.CopyDst, "UBDontInvertY"), this.dbgVerboseLogsForFirstFrames && this._count === void 0 && (this._count = 0, console.log("%c frame #" + this._count + " - begin", "background: #ffff00")), this._uploadEncoder = this._device.createCommandEncoder(this._uploadEncoderDescriptor), this._renderEncoder = this._device.createCommandEncoder(this._renderEncoderDescriptor), this._initializeLimits(), this._emptyVertexBuffer = new J(this, [0], "", !1, !1, 1, !1, 0, 1), this._cacheRenderPipeline = new N2(this._device, this._emptyVertexBuffer), this._depthCullingState = new Wle(this._cacheRenderPipeline), this._stencilStateComposer = new Ble(this._cacheRenderPipeline), this._stencilStateComposer.stencilGlobal = this._stencilState, this._depthCullingState.depthTest = !0, this._depthCullingState.depthFunc = 515, this._depthCullingState.depthMask = !0, this._textureHelper.setCommandEncoder(this._uploadEncoder), this._clearQuad = new Vle(this._device, this, this._emptyVertexBuffer), this._defaultDrawContext = this.createDrawContext(), this._currentDrawContext = this._defaultDrawContext, this._defaultMaterialContext = this.createMaterialContext(), this._currentMaterialContext = this._defaultMaterialContext, this._initializeContextAndSwapChain(), this._initializeMainAttachments(), this.resize(); }).catch((n) => { Se.Error("Can not create WebGPU Device and/or context."), Se.Error(n), console.trace && console.trace(); }); } _initGlslang(e) { return e = e || {}, e = Object.assign(Object.assign({}, wn._GLSLslangDefaultOptions), e), e.glslang ? Promise.resolve(e.glslang) : self.glslang ? self.glslang(e.wasmPath) : e.jsPath && e.wasmPath ? ye.LoadBabylonScriptAsync(e.jsPath).then(() => self.glslang(ye.GetBabylonScriptURL(e.wasmPath))) : Promise.reject("gslang is not available."); } _initializeLimits() { this._caps = { maxTexturesImageUnits: this._deviceLimits.maxSampledTexturesPerShaderStage, maxVertexTextureImageUnits: this._deviceLimits.maxSampledTexturesPerShaderStage, maxCombinedTexturesImageUnits: this._deviceLimits.maxSampledTexturesPerShaderStage * 2, maxTextureSize: this._deviceLimits.maxTextureDimension2D, maxCubemapTextureSize: this._deviceLimits.maxTextureDimension2D, maxRenderTextureSize: this._deviceLimits.maxTextureDimension2D, maxVertexAttribs: this._deviceLimits.maxVertexAttributes, maxVaryingVectors: this._deviceLimits.maxInterStageShaderVariables, maxFragmentUniformVectors: Math.floor(this._deviceLimits.maxUniformBufferBindingSize / 4), maxVertexUniformVectors: Math.floor(this._deviceLimits.maxUniformBufferBindingSize / 4), standardDerivatives: !0, astc: this._deviceEnabledExtensions.indexOf(k2.TextureCompressionASTC) >= 0 ? !0 : void 0, s3tc: this._deviceEnabledExtensions.indexOf(k2.TextureCompressionBC) >= 0 ? !0 : void 0, pvrtc: null, etc1: null, etc2: this._deviceEnabledExtensions.indexOf(k2.TextureCompressionETC2) >= 0 ? !0 : void 0, bptc: this._deviceEnabledExtensions.indexOf(k2.TextureCompressionBC) >= 0 ? !0 : void 0, maxAnisotropy: 16, uintIndices: !0, fragmentDepthSupported: !0, highPrecisionShaderSupported: !0, colorBufferFloat: !0, supportFloatTexturesResolve: !1, textureFloat: !0, textureFloatLinearFiltering: this._deviceEnabledExtensions.indexOf(k2.Float32Filterable) >= 0, textureFloatRender: !0, textureHalfFloat: !0, textureHalfFloatLinearFiltering: !0, textureHalfFloatRender: !0, textureLOD: !0, texelFetch: !0, drawBuffersExtension: !0, depthTextureExtension: !0, vertexArrayObject: !1, instancedArrays: !0, timerQuery: typeof BigUint64Array < "u" && this._deviceEnabledExtensions.indexOf(k2.TimestampQuery) !== -1 ? !0 : void 0, supportOcclusionQuery: typeof BigUint64Array < "u", canUseTimestampForTimerQuery: !0, multiview: !1, oculusMultiview: !1, parallelShaderCompile: void 0, blendMinMax: !0, maxMSAASamples: 4, canUseGLInstanceID: !0, canUseGLVertexID: !0, supportComputeShaders: !0, supportSRGBBuffers: !0, supportTransformFeedbacks: !1, textureMaxLevel: !0, texture2DArrayMaxLayerCount: this._deviceLimits.maxTextureArrayLayers, disableMorphTargetTexture: !1 }, this._caps.parallelShaderCompile = null, this._features = { forceBitmapOverHTMLImageElement: !0, supportRenderAndCopyToLodForFloatTextures: !0, supportDepthStencilTexture: !0, supportShadowSamplers: !0, uniformBufferHardCheckMatrix: !1, allowTexturePrefiltering: !0, trackUbosInFrame: !0, checkUbosContentBeforeUpload: !0, supportCSM: !0, basisNeedsPOT: !1, support3DTextures: !0, needTypeSuffixInShaderConstants: !0, supportMSAA: !0, supportSSAO2: !0, supportExtendedTextureFormats: !0, supportSwitchCaseInShader: !0, supportSyncTextureRead: !1, needsInvertingBitmap: !1, useUBOBindingCache: !1, needShaderCodeInlining: !0, needToAlwaysBindUniformBuffers: !0, supportRenderPasses: !0, supportSpriteInstancing: !0, forceVertexBufferStrideMultiple4Bytes: !0, _collectUbosUpdatedInFrame: !1 }; } _initializeContextAndSwapChain() { if (!this._renderingCanvas) throw "The rendering canvas has not been set!"; this._context = this._renderingCanvas.getContext("webgpu"), this._configureContext(), this._colorFormat = this._options.swapChainFormat, this._mainRenderPassWrapper.colorAttachmentGPUTextures = [new DC()], this._mainRenderPassWrapper.colorAttachmentGPUTextures[0].format = this._colorFormat, this._setColorFormat(this._mainRenderPassWrapper); } // Set default values as WebGL with depth and stencil attachment for the broadest Compat. _initializeMainAttachments() { if (!this._bufferManager) return; this.flushFramebuffer(), this._mainTextureExtends = { width: this.getRenderWidth(!0), height: this.getRenderHeight(!0), depthOrArrayLayers: 1 }; const e = new Float32Array([this.getRenderHeight(!0)]); this._bufferManager.setSubData(this._ubInvertY, 4, e), this._bufferManager.setSubData(this._ubDontInvertY, 4, e); let t; if (this._options.antialias) { const i = { label: `Texture_MainColor_${this._mainTextureExtends.width}x${this._mainTextureExtends.height}_antialiasing`, size: this._mainTextureExtends, mipLevelCount: 1, sampleCount: this._mainPassSampleCount, dimension: Hp.E2d, format: this._options.swapChainFormat, usage: Po.RenderAttachment }; this._mainTexture && this._textureHelper.releaseTexture(this._mainTexture), this._mainTexture = this._device.createTexture(i), t = [ { view: this._mainTexture.createView({ label: "TextureView_MainColor_antialiasing", dimension: Hp.E2d, format: this._options.swapChainFormat, mipLevelCount: 1, arrayLayerCount: 1 }), clearValue: new xt(0, 0, 0, 1), loadOp: f9.Clear, storeOp: Hc.Store // don't use StoreOp.Discard, else using several cameras with different viewports or using scissors will fail because we call beginRenderPass / endPass several times for the same color attachment! } ]; } else t = [ { view: void 0, clearValue: new xt(0, 0, 0, 1), loadOp: f9.Clear, storeOp: Hc.Store } ]; this._mainRenderPassWrapper.depthTextureFormat = this.isStencilEnable ? we.Depth24PlusStencil8 : we.Depth32Float, this._setDepthTextureFormat(this._mainRenderPassWrapper), this._setColorFormat(this._mainRenderPassWrapper); const r = { label: `Texture_MainDepthStencil_${this._mainTextureExtends.width}x${this._mainTextureExtends.height}`, size: this._mainTextureExtends, mipLevelCount: 1, sampleCount: this._mainPassSampleCount, dimension: Hp.E2d, format: this._mainRenderPassWrapper.depthTextureFormat, usage: Po.RenderAttachment }; this._depthTexture && this._textureHelper.releaseTexture(this._depthTexture), this._depthTexture = this._device.createTexture(r); const n = { view: this._depthTexture.createView({ label: `TextureView_MainDepthStencil_${this._mainTextureExtends.width}x${this._mainTextureExtends.height}`, dimension: Hp.E2d, format: this._depthTexture.format, mipLevelCount: 1, arrayLayerCount: 1 }), depthClearValue: this._clearDepthValue, depthLoadOp: f9.Clear, depthStoreOp: Hc.Store, stencilClearValue: this._clearStencilValue, stencilLoadOp: this.isStencilEnable ? f9.Clear : void 0, stencilStoreOp: this.isStencilEnable ? Hc.Store : void 0 }; this._mainRenderPassWrapper.renderPassDescriptor = { label: "MainRenderPass", colorAttachments: t, depthStencilAttachment: n }; } _configureContext() { this._context.configure({ device: this._device, format: this._options.swapChainFormat, usage: Po.RenderAttachment | Po.CopySrc, alphaMode: this.premultipliedAlpha ? JI.Premultiplied : JI.Opaque }); } /** * Force a specific size of the canvas * @param width defines the new canvas' width * @param height defines the new canvas' height * @param forceSetSize true to force setting the sizes of the underlying canvas * @returns true if the size was changed */ setSize(e, t, r = !1) { return super.setSize(e, t, r) ? (this.dbgVerboseLogsForFirstFrames && (this._count === void 0 && (this._count = 0), (!this._count || this._count < this.dbgVerboseLogsNumFrames) && console.log("frame #" + this._count + " - setSize -", e, t)), this._initializeMainAttachments(), this.snapshotRendering && this.snapshotRenderingReset(), !0) : !1; } /** * @internal */ _getShaderProcessor(e) { return e === za.WGSL ? this._shaderProcessorWGSL : this._shaderProcessor; } /** * @internal */ _getShaderProcessingContext(e) { return new gp(e); } _currentPassIsMainPass() { return this._currentRenderTarget === null; } _getCurrentRenderPass() { return this._currentRenderTarget && !this._currentRenderPass ? this._startRenderTargetRenderPass(this._currentRenderTarget, !1, null, !1, !1) : this._currentRenderPass || this._startMainRenderPass(!1), this._currentRenderPass; } /** @internal */ _getCurrentRenderPassWrapper() { return this._currentRenderTarget ? this._rttRenderPassWrapper : this._mainRenderPassWrapper; } //------------------------------------------------------------------------------ // Static Pipeline WebGPU States //------------------------------------------------------------------------------ /** @internal */ applyStates() { this._stencilStateComposer.apply(), this._cacheRenderPipeline.setAlphaBlendEnabled(this._alphaState.alphaBlend); } /** * Force the entire cache to be cleared * You should not have to use this function unless your engine needs to share the WebGPU context with another engine * @param bruteForce defines a boolean to force clearing ALL caches (including stencil, detoh and alpha states) */ wipeCaches(e) { this.preventCacheWipeBetweenFrames && !e || (this._forceEnableEffect = !0, this._currentIndexBuffer = null, this._currentOverrideVertexBuffers = null, this._cacheRenderPipeline.setBuffers(null, null, null), e && (this._stencilStateComposer.reset(), this._depthCullingState.reset(), this._depthCullingState.depthFunc = 515, this._alphaState.reset(), this._alphaMode = 1, this._alphaEquation = 0, this._cacheRenderPipeline.setAlphaBlendFactors(this._alphaState._blendFunctionParameters, this._alphaState._blendEquationParameters), this._cacheRenderPipeline.setAlphaBlendEnabled(!1), this.setColorWrite(!0)), this._cachedVertexBuffers = null, this._cachedIndexBuffer = null, this._cachedEffectForVertexBuffers = null); } /** * Enable or disable color writing * @param enable defines the state to set */ setColorWrite(e) { this._colorWriteLocal = e, this._cacheRenderPipeline.setWriteMask(e ? 15 : 0); } /** * Gets a boolean indicating if color writing is enabled * @returns the current color writing state */ getColorWrite() { return this._colorWriteLocal; } _mustUpdateViewport() { const e = this._viewportCached.x, t = this._viewportCached.y, r = this._viewportCached.z, n = this._viewportCached.w, i = this._viewportsCurrent.x !== e || this._viewportsCurrent.y !== t || this._viewportsCurrent.w !== r || this._viewportsCurrent.h !== n; return i && (this._viewportsCurrent.x = this._viewportCached.x, this._viewportsCurrent.y = this._viewportCached.y, this._viewportsCurrent.w = this._viewportCached.z, this._viewportsCurrent.h = this._viewportCached.w), i; } _applyViewport(e) { const t = Math.floor(this._viewportCached.x), r = Math.floor(this._viewportCached.z), n = Math.floor(this._viewportCached.w); let i = Math.floor(this._viewportCached.y); this._currentRenderTarget || (i = this.getRenderHeight(!0) - i - n), e ? e.addItem(new NQ(t, i, r, n)) : this._getCurrentRenderPass().setViewport(t, i, r, n, 0, 1), this.dbgVerboseLogsForFirstFrames && (this._count === void 0 && (this._count = 0), (!this._count || this._count < this.dbgVerboseLogsNumFrames) && console.log("frame #" + this._count + " - viewport applied - (", this._viewportCached.x, this._viewportCached.y, this._viewportCached.z, this._viewportCached.w, ") current pass is main pass=" + this._currentPassIsMainPass())); } /** * @internal */ _viewport(e, t, r, n) { this._viewportCached.x = e, this._viewportCached.y = t, this._viewportCached.z = r, this._viewportCached.w = n; } _mustUpdateScissor() { const e = this._scissorCached.x, t = this._scissorCached.y, r = this._scissorCached.z, n = this._scissorCached.w, i = this._scissorsCurrent.x !== e || this._scissorsCurrent.y !== t || this._scissorsCurrent.w !== r || this._scissorsCurrent.h !== n; return i && (this._scissorsCurrent.x = this._scissorCached.x, this._scissorsCurrent.y = this._scissorCached.y, this._scissorsCurrent.w = this._scissorCached.z, this._scissorsCurrent.h = this._scissorCached.w), i; } _applyScissor(e) { const t = this._currentRenderTarget ? this._scissorCached.y : this.getRenderHeight() - this._scissorCached.w - this._scissorCached.y; e ? e.addItem(new QQ(this._scissorCached.x, t, this._scissorCached.z, this._scissorCached.w)) : this._getCurrentRenderPass().setScissorRect(this._scissorCached.x, t, this._scissorCached.z, this._scissorCached.w), this.dbgVerboseLogsForFirstFrames && (this._count === void 0 && (this._count = 0), (!this._count || this._count < this.dbgVerboseLogsNumFrames) && console.log("frame #" + this._count + " - scissor applied - (", this._scissorCached.x, this._scissorCached.y, this._scissorCached.z, this._scissorCached.w, ") current pass is main pass=" + this._currentPassIsMainPass())); } _scissorIsActive() { return this._scissorCached.x !== 0 || this._scissorCached.y !== 0 || this._scissorCached.z !== 0 || this._scissorCached.w !== 0; } enableScissor(e, t, r, n) { this._scissorCached.x = e, this._scissorCached.y = t, this._scissorCached.z = r, this._scissorCached.w = n; } disableScissor() { this._scissorCached.x = this._scissorCached.y = this._scissorCached.z = this._scissorCached.w = 0, this._scissorsCurrent.x = this._scissorsCurrent.y = this._scissorsCurrent.w = this._scissorsCurrent.h = 0; } _mustUpdateStencilRef() { const e = this._stencilStateComposer.funcRef !== this._stencilRefsCurrent; return e && (this._stencilRefsCurrent = this._stencilStateComposer.funcRef), e; } _applyStencilRef(e) { var t, r; e ? e.addItem(new rO((t = this._stencilStateComposer.funcRef) !== null && t !== void 0 ? t : 0)) : this._getCurrentRenderPass().setStencilReference((r = this._stencilStateComposer.funcRef) !== null && r !== void 0 ? r : 0); } _mustUpdateBlendColor() { const e = this._alphaState._blendConstants, t = e[0] !== this._blendColorsCurrent[0] || e[1] !== this._blendColorsCurrent[1] || e[2] !== this._blendColorsCurrent[2] || e[3] !== this._blendColorsCurrent[3]; return t && (this._blendColorsCurrent[0] = e[0], this._blendColorsCurrent[1] = e[1], this._blendColorsCurrent[2] = e[2], this._blendColorsCurrent[3] = e[3]), t; } _applyBlendColor(e) { e ? e.addItem(new YQ(this._alphaState._blendConstants.slice())) : this._getCurrentRenderPass().setBlendConstant(this._alphaState._blendConstants); } _resetRenderPassStates() { this._viewportsCurrent.x = this._viewportsCurrent.y = this._viewportsCurrent.w = this._viewportsCurrent.h = 0, this._scissorsCurrent.x = this._scissorsCurrent.y = this._scissorsCurrent.w = this._scissorsCurrent.h = 0, this._stencilRefsCurrent = -1, this._blendColorsCurrent[0] = this._blendColorsCurrent[1] = this._blendColorsCurrent[2] = this._blendColorsCurrent[3] = null; } /** * Clear the current render buffer or the current render target (if any is set up) * @param color defines the color to use * @param backBuffer defines if the back buffer must be cleared * @param depth defines if the depth buffer must be cleared * @param stencil defines if the stencil buffer must be cleared */ clear(e, t, r, n = !1) { e && e.a === void 0 && (e.a = 1); const i = this._scissorIsActive(); this.dbgVerboseLogsForFirstFrames && (this._count === void 0 && (this._count = 0), (!this._count || this._count < this.dbgVerboseLogsNumFrames) && console.log("frame #" + this._count + " - clear - backBuffer=", t, " depth=", r, " stencil=", n, " scissor is active=", i)), this._currentRenderTarget ? i ? (this._currentRenderPass || this._startRenderTargetRenderPass(this._currentRenderTarget, !1, t ? e : null, r, n), this._applyScissor(this.compatibilityMode ? null : this._bundleList), this._clearFullQuad(t ? e : null, r, n)) : (this._currentRenderPass && this._endCurrentRenderPass(), this._startRenderTargetRenderPass(this._currentRenderTarget, !0, t ? e : null, r, n)) : ((!this._currentRenderPass || !i) && this._startMainRenderPass(!i, t ? e : null, r, n), i && (this._applyScissor(this.compatibilityMode ? null : this._bundleList), this._clearFullQuad(t ? e : null, r, n))); } _clearFullQuad(e, t, r) { var n, i; const s = this.compatibilityMode ? this._getCurrentRenderPass() : null; this._clearQuad.setColorFormat(this._colorFormat), this._clearQuad.setDepthStencilFormat(this._depthTextureFormat), this._clearQuad.setMRTAttachments((n = this._cacheRenderPipeline.mrtAttachments) !== null && n !== void 0 ? n : [], (i = this._cacheRenderPipeline.mrtTextureArray) !== null && i !== void 0 ? i : [], this._cacheRenderPipeline.mrtTextureCount), this.compatibilityMode ? s.setStencilReference(this._clearStencilValue) : this._bundleList.addItem(new rO(this._clearStencilValue)); const a = this._clearQuad.clear(s, e, t, r, this.currentSampleCount); this.compatibilityMode ? this._applyStencilRef(null) : (this._bundleList.addBundle(a), this._applyStencilRef(this._bundleList), this._reportDrawCall()); } //------------------------------------------------------------------------------ // Vertex/Index/Storage Buffers //------------------------------------------------------------------------------ /** * Creates a vertex buffer * @param data the data for the vertex buffer * @param _updatable whether the buffer should be created as updatable * @param label defines the label of the buffer (for debug purpose) * @returns the new buffer */ createVertexBuffer(e, t, r) { let n; return e instanceof Array ? n = new Float32Array(e) : e instanceof ArrayBuffer ? n = new Uint8Array(e) : n = e, this._bufferManager.createBuffer(n, ga.Vertex | ga.CopyDst, r); } /** * Creates a vertex buffer * @param data the data for the dynamic vertex buffer * @param label defines the label of the buffer (for debug purpose) * @returns the new buffer */ createDynamicVertexBuffer(e, t) { return this.createVertexBuffer(e, void 0, t); } /** * Creates a new index buffer * @param indices defines the content of the index buffer * @param updatable defines if the index buffer must be updatable * @param label defines the label of the buffer (for debug purpose) * @returns a new buffer */ createIndexBuffer(e, t, r) { let n = !0, i; e instanceof Uint32Array || e instanceof Int32Array ? i = e : e instanceof Uint16Array ? (i = e, n = !1) : e.length > 65535 ? i = new Uint32Array(e) : (i = new Uint16Array(e), n = !1); const s = this._bufferManager.createBuffer(i, ga.Index | ga.CopyDst, r); return s.is32Bits = n, s; } /** * @internal */ _createBuffer(e, t, r) { let n; e instanceof Array ? n = new Float32Array(e) : e instanceof ArrayBuffer ? n = new Uint8Array(e) : n = e; let i = 0; return t & 1 && (i |= ga.CopySrc), t & 2 && (i |= ga.CopyDst), t & 4 && (i |= ga.Uniform), t & 8 && (i |= ga.Vertex), t & 16 && (i |= ga.Index), t & 32 && (i |= ga.Storage), this._bufferManager.createBuffer(n, i, r); } /** * @internal */ bindBuffersDirectly() { throw "Not implemented on WebGPU"; } /** * @internal */ updateAndBindInstancesBuffer() { throw "Not implemented on WebGPU"; } /** * Bind a list of vertex buffers with the engine * @param vertexBuffers defines the list of vertex buffers to bind * @param indexBuffer defines the index buffer to bind * @param effect defines the effect associated with the vertex buffers * @param overrideVertexBuffers defines optional list of avertex buffers that overrides the entries in vertexBuffers */ bindBuffers(e, t, r, n) { this._currentIndexBuffer = t, this._currentOverrideVertexBuffers = n ?? null, this._cacheRenderPipeline.setBuffers(e, t, this._currentOverrideVertexBuffers); } /** * @internal */ _releaseBuffer(e) { return this._bufferManager.releaseBuffer(e); } //------------------------------------------------------------------------------ // Effects //------------------------------------------------------------------------------ /** * Create a new effect (used to store vertex/fragment shaders) * @param baseName defines the base name of the effect (The name of file without .fragment.fx or .vertex.fx) * @param attributesNamesOrOptions defines either a list of attribute names or an IEffectCreationOptions object * @param uniformsNamesOrEngine defines either a list of uniform names or the engine to use * @param samplers defines an array of string used to represent textures * @param defines defines the string containing the defines to use to compile the shaders * @param fallbacks defines the list of potential fallbacks to use if shader compilation fails * @param onCompiled defines a function to call when the effect creation is successful * @param onError defines a function to call when the effect creation has failed * @param indexParameters defines an object containing the index values to use to compile shaders (like the maximum number of simultaneous lights) * @param shaderLanguage the language the shader is written in (default: GLSL) * @returns the new Effect */ createEffect(e, t, r, n, i, s, a, f, o, d = za.GLSL) { var v; const u = e.vertexElement || e.vertex || e.vertexToken || e.vertexSource || e, l = e.fragmentElement || e.fragment || e.fragmentToken || e.fragmentSource || e, P = this._getGlobalDefines(); let p = (v = i ?? t.defines) !== null && v !== void 0 ? v : ""; P && (p += ` ` + P); const c = u + "+" + l + "@" + p; if (this._compiledEffects[c]) { const T = this._compiledEffects[c]; return a && T.isReady() && a(T), T; } const H = new An(e, t, r, n, this, i, s, a, f, o, c, d); return this._compiledEffects[c] = H, H; } _compileRawShaderToSpirV(e, t) { return this._glslang.compileGLSL(e, t); } _compileShaderToSpirV(e, t, r, n) { return this._compileRawShaderToSpirV(n + (r ? r + ` ` : "") + e, t); } _getWGSLShader(e, t, r) { return r ? r = "//" + r.split(` `).join(` //`) + ` ` : r = "", r + e; } _createPipelineStageDescriptor(e, t, r, n, i) { return this._tintWASM && r === za.GLSL && (e = this._tintWASM.convertSpirV2WGSL(e, n), t = this._tintWASM.convertSpirV2WGSL(t, i)), { vertexStage: { module: this._device.createShaderModule({ code: e }), entryPoint: "main" }, fragmentStage: { module: this._device.createShaderModule({ code: t }), entryPoint: "main" } }; } _compileRawPipelineStageDescriptor(e, t, r) { const n = e.indexOf(hC) >= 0, i = t.indexOf(hC) >= 0, s = r === za.GLSL ? this._compileRawShaderToSpirV(e, "vertex") : e, a = r === za.GLSL ? this._compileRawShaderToSpirV(t, "fragment") : t; return this._createPipelineStageDescriptor(s, a, r, n, i); } _compilePipelineStageDescriptor(e, t, r, n) { this.onBeforeShaderCompilationObservable.notifyObservers(this); const i = e.indexOf(hC) >= 0, s = t.indexOf(hC) >= 0, a = `#version 450 `, f = n === za.GLSL ? this._compileShaderToSpirV(e, "vertex", r, a) : this._getWGSLShader(e, "vertex", r), o = n === za.GLSL ? this._compileShaderToSpirV(t, "fragment", r, a) : this._getWGSLShader(t, "fragment", r), d = this._createPipelineStageDescriptor(f, o, n, i, s); return this.onAfterShaderCompilationObservable.notifyObservers(this), d; } /** * @internal */ createRawShaderProgram() { throw "Not available on WebGPU"; } /** * @internal */ createShaderProgram() { throw "Not available on WebGPU"; } /** * Inline functions in shader code that are marked to be inlined * @param code code to inline * @returns inlined code */ inlineShaderCode(e) { const t = new Aq(e); return t.debug = !1, t.processCode(), t.code; } /** * Creates a new pipeline context * @param shaderProcessingContext defines the shader processing context used during the processing if available * @returns the new pipeline */ createPipelineContext(e) { return new Bue(e, this); } /** * Creates a new material context * @returns the new context */ createMaterialContext() { return new sy(); } /** * Creates a new draw context * @returns the new context */ createDrawContext() { return new tV(this._bufferManager); } /** * @internal */ _preparePipelineContext(e, t, r, n, i, s, a, f) { const o = e, d = o.shaderProcessingContext.shaderLanguage; this.dbgShowShaderCode && (console.log(f), console.log(t), console.log(r), console.log("***********************************************")), o.sources = { fragment: r, vertex: t, rawVertex: i, rawFragment: s }, n ? o.stages = this._compileRawPipelineStageDescriptor(t, r, d) : o.stages = this._compilePipelineStageDescriptor(t, r, f, d); } /** * Gets the list of active attributes for a given WebGPU program * @param pipelineContext defines the pipeline context to use * @param attributesNames defines the list of attribute names to get * @returns an array of indices indicating the offset of each attribute */ getAttributes(e, t) { const r = new Array(t.length), n = e; for (let i = 0; i < t.length; i++) { const s = t[i], a = n.shaderProcessingContext.availableAttributes[s]; a !== void 0 && (r[i] = a); } return r; } /** * Activates an effect, making it the current one (ie. the one used for rendering) * @param effect defines the effect to activate */ enableEffect(e) { if (!e) return; let t = !0; if (!zo.IsWrapper(e)) t = e !== this._currentEffect, this._currentEffect = e, this._currentMaterialContext = this._defaultMaterialContext, this._currentDrawContext = this._defaultDrawContext, this._counters.numEnableEffects++, this.dbgLogIfNotDrawWrapper && Se.Warn(`enableEffect has been called with an Effect and not a Wrapper! effect.uniqueId=${e.uniqueId}, effect.name=${e.name}, effect.name.vertex=${e.name.vertex}, effect.name.fragment=${e.name.fragment}`, 10); else if (!e.effect || e.effect === this._currentEffect && e.materialContext === this._currentMaterialContext && e.drawContext === this._currentDrawContext && !this._forceEnableEffect) { if (!e.effect && this.dbgShowEmptyEnableEffectCalls) throw console.error("drawWrapper=", e), "Invalid call to enableEffect: the effect property is empty!"; return; } else if (t = e.effect !== this._currentEffect, this._currentEffect = e.effect, this._currentMaterialContext = e.materialContext, this._currentDrawContext = e.drawContext, this._counters.numEnableDrawWrapper++, !this._currentMaterialContext) throw console.error("drawWrapper=", e), "Invalid call to enableEffect: the materialContext property is empty!"; this._stencilStateComposer.stencilMaterial = void 0, this._forceEnableEffect = t || this._forceEnableEffect ? !1 : this._forceEnableEffect, t && (this._currentEffect.onBind && this._currentEffect.onBind(this._currentEffect), this._currentEffect._onBindObservable && this._currentEffect._onBindObservable.notifyObservers(this._currentEffect)); } /** * @internal */ _releaseEffect(e) { this._compiledEffects[e._key] && (delete this._compiledEffects[e._key], this._deletePipelineContext(e.getPipelineContext())); } /** * Force the engine to release all cached effects. This means that next effect compilation will have to be done completely even if a similar effect was already compiled */ releaseEffects() { for (const e in this._compiledEffects) { const t = this._compiledEffects[e].getPipelineContext(); this._deletePipelineContext(t); } this._compiledEffects = {}; } _deletePipelineContext(e) { e && e.dispose(); } //------------------------------------------------------------------------------ // Textures //------------------------------------------------------------------------------ /** * Gets a boolean indicating that only power of 2 textures are supported * Please note that you can still use non power of 2 textures but in this case the engine will forcefully convert them */ get needPOTTextures() { return !1; } /** @internal */ _createHardwareTexture() { return new DC(); } /** * @internal */ _releaseTexture(e) { const t = this._internalTexturesCache.indexOf(e); t !== -1 && this._internalTexturesCache.splice(t, 1), this._textureHelper.releaseTexture(e); } /** * @internal */ _getRGBABufferInternalSizedFormat() { return 5; } updateTextureComparisonFunction(e, t) { e._comparisonFunction = t; } /** * Creates an internal texture without binding it to a framebuffer * @internal * @param size defines the size of the texture * @param options defines the options used to create the texture * @param delayGPUTextureCreation true to delay the texture creation the first time it is really needed. false to create it right away * @param source source type of the texture * @returns a new internal texture */ _createInternalTexture(e, t, r = !0, n = ri.Unknown) { var i, s, a; const f = {}; t !== void 0 && typeof t == "object" ? (f.generateMipMaps = t.generateMipMaps, f.type = t.type === void 0 ? 0 : t.type, f.samplingMode = t.samplingMode === void 0 ? 3 : t.samplingMode, f.format = t.format === void 0 ? 5 : t.format, f.samples = (i = t.samples) !== null && i !== void 0 ? i : 1, f.creationFlags = (s = t.creationFlags) !== null && s !== void 0 ? s : 0, f.useSRGBBuffer = (a = t.useSRGBBuffer) !== null && a !== void 0 ? a : !1, f.label = t.label) : (f.generateMipMaps = t, f.type = 0, f.samplingMode = 3, f.format = 5, f.samples = 1, f.creationFlags = 0, f.useSRGBBuffer = !1), (f.type === 1 && !this._caps.textureFloatLinearFiltering || f.type === 2 && !this._caps.textureHalfFloatLinearFiltering) && (f.samplingMode = 1), f.type === 1 && !this._caps.textureFloat && (f.type = 0, Se.Warn("Float textures are not supported. Type forced to TEXTURETYPE_UNSIGNED_BYTE")); const o = new As(this, n), d = e.width || e, v = e.height || e, u = e.layers || 0; return o.baseWidth = d, o.baseHeight = v, o.width = d, o.height = v, o.depth = u, o.isReady = !0, o.samples = f.samples, o.generateMipMaps = !!f.generateMipMaps, o.samplingMode = f.samplingMode, o.type = f.type, o.format = f.format, o.is2DArray = u > 0, o._cachedWrapU = 0, o._cachedWrapV = 0, o._useSRGBBuffer = f.useSRGBBuffer, o.label = f.label, this._internalTexturesCache.push(o), r || this._textureHelper.createGPUTextureForInternalTexture(o, d, v, u || 1, f.creationFlags), o; } /** * Usually called from Texture.ts. * Passed information to create a hardware texture * @param url defines a value which contains one of the following: * * A conventional http URL, e.g. 'http://...' or 'file://...' * * A base64 string of in-line texture data, e.g. '...' * * An indicator that data being passed using the buffer parameter, e.g. 'data:mytexture.jpg' * @param noMipmap defines a boolean indicating that no mipmaps shall be generated. Ignored for compressed textures. They must be in the file * @param invertY when true, image is flipped when loaded. You probably want true. Certain compressed textures may invert this if their default is inverted (eg. ktx) * @param scene needed for loading to the correct scene * @param samplingMode mode with should be used sample / access the texture (Default: Texture.TRILINEAR_SAMPLINGMODE) * @param onLoad optional callback to be called upon successful completion * @param onError optional callback to be called upon failure * @param buffer a source of a file previously fetched as either a base64 string, an ArrayBuffer (compressed or image format), HTMLImageElement (image format), or a Blob * @param fallback an internal argument in case the function must be called again, due to etc1 not having alpha capabilities * @param format internal format. Default: RGB when extension is '.jpg' else RGBA. Ignored for compressed textures * @param forcedExtension defines the extension to use to pick the right loader * @param mimeType defines an optional mime type * @param loaderOptions options to be passed to the loader * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg) * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU). * @returns a InternalTexture for assignment back into BABYLON.Texture */ createTexture(e, t, r, n, i = 3, s = null, a = null, f = null, o = null, d = null, v = null, u, l, P, p) { return this._createTextureBase(e, t, r, n, i, s, a, (c, H, T, q, b, j, w, m) => { var I; const N = q; if (c.baseWidth = N.width, c.baseHeight = N.height, c.width = N.width, c.height = N.height, c.format = c.format !== -1 ? c.format : d ?? 5, c.type = c.type !== -1 ? c.type : 0, m(c.width, c.height, N, H, c, () => { }), !((I = c._hardwareTexture) === null || I === void 0) && I.underlyingResource) !j && !w && this._generateMipmaps(c, this._uploadEncoder); else { const k = this._textureHelper.createGPUTextureForInternalTexture(c, N.width, N.height, void 0, P); Ns.IsImageBitmap(N) && (this._textureHelper.updateTexture(N, c, N.width, N.height, c.depth, k.format, 0, 0, b, !1, 0, 0), !j && !w && this._generateMipmaps(c, this._uploadEncoder)); } T && T.removePendingData(c), c.isReady = !0, c.onLoadedObservable.notifyObservers(c), c.onLoadedObservable.clear(); }, () => !1, f, o, d, v, u, l, p); } /** * Wraps an external web gpu texture in a Babylon texture. * @param texture defines the external texture * @returns the babylon internal texture */ wrapWebGPUTexture(e) { const t = new DC(e), r = new As(this, ri.Unknown, !0); return r._hardwareTexture = t, r.isReady = !0, r; } /** * Wraps an external web gl texture in a Babylon texture. * @returns the babylon internal texture */ wrapWebGLTexture() { throw new Error("wrapWebGLTexture is not supported, use wrapWebGPUTexture instead."); } generateMipMapsForCubemap(e) { var t; e.generateMipMaps && (!((t = e._hardwareTexture) === null || t === void 0) && t.underlyingResource || this._textureHelper.createGPUTextureForInternalTexture(e), this._generateMipmaps(e)); } /** * Update the sampling mode of a given texture * @param samplingMode defines the required sampling mode * @param texture defines the texture to update * @param generateMipMaps defines whether to generate mipmaps for the texture */ updateTextureSamplingMode(e, t, r = !1) { r && (t.generateMipMaps = !0, this._generateMipmaps(t)), t.samplingMode = e; } /** * Update the sampling mode of a given texture * @param texture defines the texture to update * @param wrapU defines the texture wrap mode of the u coordinates * @param wrapV defines the texture wrap mode of the v coordinates * @param wrapR defines the texture wrap mode of the r coordinates */ updateTextureWrappingMode(e, t, r = null, n = null) { t !== null && (e._cachedWrapU = t), r !== null && (e._cachedWrapV = r), (e.is2DArray || e.is3D) && n !== null && (e._cachedWrapR = n); } /** * Update the dimensions of a texture * @param texture texture to update * @param width new width of the texture * @param height new height of the texture * @param depth new depth of the texture */ updateTextureDimensions(e, t, r, n = 1) { if (!e._hardwareTexture || e.width === t && e.height === r && e.depth === n) return; const i = e._hardwareTexture.textureAdditionalUsages; e._hardwareTexture.release(), this._textureHelper.createGPUTextureForInternalTexture(e, t, r, n, i); } /** * @internal */ _setInternalTexture(e, t, r) { if (r = r ?? e, this._currentEffect) { const i = this._currentEffect._pipelineContext.shaderProcessingContext.availableTextures[r]; if (this._currentMaterialContext.setTexture(e, t), i && i.autoBindSampler) { const s = r + Wo.AutoSamplerSuffix; this._currentMaterialContext.setSampler(s, t); } } } /** * Sets a texture to the according uniform. * @param channel The texture channel * @param unused unused parameter * @param texture The texture to apply * @param name The name of the uniform in the effect */ setTexture(e, t, r, n) { this._setTexture(e, r, !1, !1, n, n); } /** * Sets an array of texture to the WebGPU context * @param channel defines the channel where the texture array must be set * @param unused unused parameter * @param textures defines the array of textures to bind * @param name name of the channel */ setTextureArray(e, t, r, n) { for (let i = 0; i < r.length; i++) this._setTexture(-1, r[i], !0, !1, n + i.toString(), n); } _setTexture(e, t, r = !1, n = !1, i = "", s) { if (s = s ?? i, this._currentEffect) { if (!t) return this._currentMaterialContext.setTexture(i, null), !1; if (t.video) t.update(); else if (t.delayLoadState === 4) return t.delayLoad(), !1; let a = null; if (n ? a = t.depthStencilTexture : t.isReady() ? a = t.getInternalTexture() : t.isCube ? a = this.emptyCubeTexture : t.is3D ? a = this.emptyTexture3D : t.is2DArray ? a = this.emptyTexture2DArray : a = this.emptyTexture, a && !a.isMultiview) { if (a.isCube && a._cachedCoordinatesMode !== t.coordinatesMode) { a._cachedCoordinatesMode = t.coordinatesMode; const f = t.coordinatesMode !== 3 && t.coordinatesMode !== 5 ? 1 : 0; t.wrapU = f, t.wrapV = f; } a._cachedWrapU = t.wrapU, a._cachedWrapV = t.wrapV, a.is3D && (a._cachedWrapR = t.wrapR), this._setAnisotropicLevel(0, a, t.anisotropicFilteringLevel); } this._setInternalTexture(i, a, s); } else this.dbgVerboseLogsForFirstFrames && (this._count === void 0 && (this._count = 0), (!this._count || this._count < this.dbgVerboseLogsNumFrames) && console.log("frame #" + this._count + " - _setTexture called with a null _currentEffect! texture=", t)); return !0; } /** * @internal */ _setAnisotropicLevel(e, t, r) { t._cachedAnisotropicFilteringLevel !== r && (t._cachedAnisotropicFilteringLevel = Math.min(r, this._caps.maxAnisotropy)); } /** * @internal */ _bindTexture(e, t, r) { e !== void 0 && this._setInternalTexture(r, t); } /** * Generates the mipmaps for a texture * @param texture texture to generate the mipmaps for */ generateMipmaps(e) { this._generateMipmaps(e); } /** * @internal */ _generateMipmaps(e, t) { t = t ?? this._renderEncoder; const r = e._hardwareTexture; if (!r) return; t === this._renderEncoder && this._endCurrentRenderPass(); const n = e._hardwareTexture.format, i = Ns.ComputeNumMipmapLevels(e.width, e.height); this.dbgVerboseLogsForFirstFrames && (this._count === void 0 && (this._count = 0), (!this._count || this._count < this.dbgVerboseLogsNumFrames) && console.log("frame #" + this._count + " - generate mipmaps - width=" + e.width + ", height=" + e.height + ", isCube=" + e.isCube + ", command encoder=" + (t === this._renderEncoder ? "render" : "copy"))), e.isCube ? this._textureHelper.generateCubeMipmaps(r, n, i, t) : this._textureHelper.generateMipmaps(r, n, i, 0, t); } /** * Update a portion of an internal texture * @param texture defines the texture to update * @param imageData defines the data to store into the texture * @param xOffset defines the x coordinates of the update rectangle * @param yOffset defines the y coordinates of the update rectangle * @param width defines the width of the update rectangle * @param height defines the height of the update rectangle * @param faceIndex defines the face index if texture is a cube (0 by default) * @param lod defines the lod level to update (0 by default) * @param generateMipMaps defines whether to generate mipmaps or not */ updateTextureData(e, t, r, n, i, s, a = 0, f = 0, o = !1) { var d; let v = e._hardwareTexture; !((d = e._hardwareTexture) === null || d === void 0) && d.underlyingResource || (v = this._textureHelper.createGPUTextureForInternalTexture(e)); const u = new Uint8Array(t.buffer, t.byteOffset, t.byteLength); this._textureHelper.updateTexture(u, e, i, s, e.depth, v.format, a, f, e.invertY, !1, r, n), o && this._generateMipmaps(e); } /** * @internal */ _uploadCompressedDataToTextureDirectly(e, t, r, n, i, s = 0, a = 0) { var f; let o = e._hardwareTexture; !((f = e._hardwareTexture) === null || f === void 0) && f.underlyingResource || (e.format = t, o = this._textureHelper.createGPUTextureForInternalTexture(e, r, n)); const d = new Uint8Array(i.buffer, i.byteOffset, i.byteLength); this._textureHelper.updateTexture(d, e, r, n, e.depth, o.format, s, a, !1, !1, 0, 0); } /** * @internal */ _uploadDataToTextureDirectly(e, t, r = 0, n = 0, i, s = !1) { var a; const f = Math.round(Math.log(e.width) * Math.LOG2E), o = Math.round(Math.log(e.height) * Math.LOG2E), d = s ? e.width : Math.pow(2, Math.max(f - n, 0)), v = s ? e.height : Math.pow(2, Math.max(o - n, 0)); let u = e._hardwareTexture; !((a = e._hardwareTexture) === null || a === void 0) && a.underlyingResource || (u = this._textureHelper.createGPUTextureForInternalTexture(e, d, v)); const l = new Uint8Array(t.buffer, t.byteOffset, t.byteLength); this._textureHelper.updateTexture(l, e, d, v, e.depth, u.format, r, n, e.invertY, !1, 0, 0); } /** * @internal */ _uploadArrayBufferViewToTexture(e, t, r = 0, n = 0) { this._uploadDataToTextureDirectly(e, t, r, n); } /** * @internal */ _uploadImageToTexture(e, t, r = 0, n = 0) { var i; let s = e._hardwareTexture; if (!((i = e._hardwareTexture) === null || i === void 0) && i.underlyingResource || (s = this._textureHelper.createGPUTextureForInternalTexture(e)), t instanceof HTMLImageElement) throw "WebGPU engine: HTMLImageElement not supported in _uploadImageToTexture!"; const a = t, f = Math.ceil(e.width / (1 << n)), o = Math.ceil(e.height / (1 << n)); this._textureHelper.updateTexture(a, e, f, o, e.depth, s.format, r, n, e.invertY, !1, 0, 0); } /** * Reads pixels from the current frame buffer. Please note that this function can be slow * @param x defines the x coordinate of the rectangle where pixels must be read * @param y defines the y coordinate of the rectangle where pixels must be read * @param width defines the width of the rectangle where pixels must be read * @param height defines the height of the rectangle where pixels must be read * @param hasAlpha defines whether the output should have alpha or not (defaults to true) * @param flushRenderer true to flush the renderer from the pending commands before reading the pixels * @returns a ArrayBufferView promise (Uint8Array) containing RGBA colors */ // eslint-disable-next-line @typescript-eslint/no-unused-vars readPixels(e, t, r, n, i = !0, s = !0) { const f = this._getCurrentRenderPassWrapper().colorAttachmentGPUTextures[0]; if (!f) return Promise.resolve(new Uint8Array(0)); const o = f.underlyingResource, d = f.format; return o ? (s && this.flushFramebuffer(), this._textureHelper.readPixels(o, e, t, r, n, d)) : Promise.resolve(new Uint8Array(0)); } //------------------------------------------------------------------------------ // Frame management //------------------------------------------------------------------------------ /** * Begin a new frame */ beginFrame() { super.beginFrame(); } /** * End the current frame */ endFrame() { if (this._endCurrentRenderPass(), this._snapshotRendering.endFrame(), this._timestampQuery.endFrame(this._renderEncoder), this.flushFramebuffer(), this._textureHelper.destroyDeferredTextures(), this._bufferManager.destroyDeferredBuffers(), this._features._collectUbosUpdatedInFrame) { if (this.dbgVerboseLogsForFirstFrames && (this._count === void 0 && (this._count = 0), !this._count || this._count < this.dbgVerboseLogsNumFrames)) { const e = []; for (const t in yr._UpdatedUbosInFrame) e.push(t + ":" + yr._UpdatedUbosInFrame[t]); console.log("frame #" + this._count + " - updated ubos -", e.join(", ")); } yr._UpdatedUbosInFrame = {}; } this.countersLastFrame.numEnableEffects = this._counters.numEnableEffects, this.countersLastFrame.numEnableDrawWrapper = this._counters.numEnableDrawWrapper, this.countersLastFrame.numBundleCreationNonCompatMode = this._counters.numBundleCreationNonCompatMode, this.countersLastFrame.numBundleReuseNonCompatMode = this._counters.numBundleReuseNonCompatMode, this._counters.numEnableEffects = 0, this._counters.numEnableDrawWrapper = 0, this._counters.numBundleCreationNonCompatMode = 0, this._counters.numBundleReuseNonCompatMode = 0, this._cacheRenderPipeline.endFrame(), this._cacheBindGroups.endFrame(), this._pendingDebugCommands.length = 0, super.endFrame(), this.dbgVerboseLogsForFirstFrames && (this._count === void 0 && (this._count = 0), this._count < this.dbgVerboseLogsNumFrames && console.log("%c frame #" + this._count + " - end", "background: #ffff00"), this._count < this.dbgVerboseLogsNumFrames && (this._count++, this._count !== this.dbgVerboseLogsNumFrames && console.log("%c frame #" + this._count + " - begin", "background: #ffff00"))); } /** * Force a WebGPU flush (ie. a flush of all waiting commands) */ flushFramebuffer() { this._endCurrentRenderPass(), this._commandBuffers[0] = this._uploadEncoder.finish(), this._commandBuffers[1] = this._renderEncoder.finish(), this._device.queue.submit(this._commandBuffers), this._uploadEncoder = this._device.createCommandEncoder(this._uploadEncoderDescriptor), this._renderEncoder = this._device.createCommandEncoder(this._renderEncoderDescriptor), this._timestampQuery.startFrame(this._uploadEncoder), this._textureHelper.setCommandEncoder(this._uploadEncoder), this._bundleList.reset(); } /** @internal */ _currentFrameBufferIsDefaultFrameBuffer() { return this._currentPassIsMainPass(); } //------------------------------------------------------------------------------ // Render Pass //------------------------------------------------------------------------------ _startRenderTargetRenderPass(e, t, r, n, i) { var s, a, f, o, d, v, u, l; this._endCurrentRenderPass(); const P = e, p = P._depthStencilTexture, c = p == null ? void 0 : p._hardwareTexture, H = c == null ? void 0 : c.underlyingResource, T = c == null ? void 0 : c.getMSAATexture(), q = H == null ? void 0 : H.createView(this._rttRenderPassWrapper.depthAttachmentViewDescriptor), b = T == null ? void 0 : T.createView(this._rttRenderPassWrapper.depthAttachmentViewDescriptor), j = c ? Ns.HasStencilAspect(c.format) : !1, w = []; this.useReverseDepthBuffer && this.setDepthFunctionToGreaterOrEqual(); const m = Nle; r && (m.r = r.r * 255, m.g = r.g * 255, m.b = r.b * 255, m.a = r.a * 255); const I = t && r, N = t && n, k = t && i; if (P._attachments && P.isMulti) { (!this._mrtAttachments || this._mrtAttachments.length === 0) && (this._mrtAttachments = P._defaultAttachments); for (let R = 0; R < this._mrtAttachments.length; ++R) { const y = this._mrtAttachments[R], O = P.textures[R], Y = O == null ? void 0 : O._hardwareTexture, ee = Y == null ? void 0 : Y.underlyingResource; if (Y && ee) { const Z = Y.getMSAATexture(R), te = (a = (s = P.layerIndices) === null || s === void 0 ? void 0 : s[R]) !== null && a !== void 0 ? a : 0, fe = (o = (f = P.faceIndices) === null || f === void 0 ? void 0 : f[R]) !== null && o !== void 0 ? o : 0, _ = Object.assign(Object.assign({}, this._rttRenderPassWrapper.colorAttachmentViewDescriptor), { format: Y.format, baseArrayLayer: O.isCube ? te * 6 + fe : te }), G = Object.assign(Object.assign({}, this._rttRenderPassWrapper.colorAttachmentViewDescriptor), { format: Y.format, baseArrayLayer: 0 }), L = O.type === 7 || O.type === 5, $ = ee.createView(_), ae = Z == null ? void 0 : Z.createView(G); w.push({ view: ae || $, resolveTarget: Z ? $ : void 0, clearValue: y !== 0 && I ? L ? m : r : void 0, loadOp: y !== 0 && I ? f9.Clear : f9.Load, storeOp: Hc.Store }); } } this._cacheRenderPipeline.setMRT(P.textures, this._mrtAttachments.length), this._cacheRenderPipeline.setMRTAttachments(this._mrtAttachments); } else { const R = P.texture; if (R) { const y = R._hardwareTexture, O = y.underlyingResource, Y = y.getMSAATexture(), ee = O.createView(this._rttRenderPassWrapper.colorAttachmentViewDescriptor), Z = Y == null ? void 0 : Y.createView(this._rttRenderPassWrapper.colorAttachmentViewDescriptor), te = R.type === 7 || R.type === 5; w.push({ view: Z || ee, resolveTarget: Y ? ee : void 0, clearValue: I ? te ? m : r : void 0, loadOp: I ? f9.Clear : f9.Load, storeOp: Hc.Store }); } else w.push(null); } if ((d = this._debugPushGroup) === null || d === void 0 || d.call(this, "render target pass", 1), this._rttRenderPassWrapper.renderPassDescriptor = { label: ((v = e.label) !== null && v !== void 0 ? v : "RTT") + "RenderPass", colorAttachments: w, depthStencilAttachment: p && H ? { view: b || q, depthClearValue: N ? this.useReverseDepthBuffer ? this._clearReverseDepthValue : this._clearDepthValue : void 0, depthLoadOp: N ? f9.Clear : f9.Load, depthStoreOp: Hc.Store, stencilClearValue: P._depthStencilTextureWithStencil && k ? this._clearStencilValue : void 0, stencilLoadOp: j ? P._depthStencilTextureWithStencil && k ? f9.Clear : f9.Load : void 0, stencilStoreOp: j ? Hc.Store : void 0 } : void 0, occlusionQuerySet: !((u = this._occlusionQuery) === null || u === void 0) && u.hasQueries ? this._occlusionQuery.querySet : void 0 }, this._currentRenderPass = this._renderEncoder.beginRenderPass(this._rttRenderPassWrapper.renderPassDescriptor), this.dbgVerboseLogsForFirstFrames && (this._count === void 0 && (this._count = 0), !this._count || this._count < this.dbgVerboseLogsNumFrames)) { const R = P.texture; console.log("frame #" + this._count + " - render target begin pass - rtt name=" + e.label + ", internalTexture.uniqueId=" + R.uniqueId + ", width=" + R.width + ", height=" + R.height + ", setClearStates=" + t, "renderPassDescriptor=", this._rttRenderPassWrapper.renderPassDescriptor); } (l = this._debugFlushPendingCommands) === null || l === void 0 || l.call(this), this._resetRenderPassStates(), (!c || !Ns.HasStencilAspect(c.format)) && (this._stencilStateComposer.enabled = !1); } _startMainRenderPass(e, t, r, n) { var i, s, a; this._endCurrentRenderPass(), this.useReverseDepthBuffer && this.setDepthFunctionToGreaterOrEqual(); const f = e && t, o = e && r, d = e && n; this._mainRenderPassWrapper.renderPassDescriptor.colorAttachments[0].clearValue = f ? t : void 0, this._mainRenderPassWrapper.renderPassDescriptor.colorAttachments[0].loadOp = f ? f9.Clear : f9.Load, this._mainRenderPassWrapper.renderPassDescriptor.depthStencilAttachment.depthClearValue = o ? this.useReverseDepthBuffer ? this._clearReverseDepthValue : this._clearDepthValue : void 0, this._mainRenderPassWrapper.renderPassDescriptor.depthStencilAttachment.depthLoadOp = o ? f9.Clear : f9.Load, this._mainRenderPassWrapper.renderPassDescriptor.depthStencilAttachment.stencilClearValue = d ? this._clearStencilValue : void 0, this._mainRenderPassWrapper.renderPassDescriptor.depthStencilAttachment.stencilLoadOp = this.isStencilEnable ? d ? f9.Clear : f9.Load : void 0, this._mainRenderPassWrapper.renderPassDescriptor.occlusionQuerySet = !((i = this._occlusionQuery) === null || i === void 0) && i.hasQueries ? this._occlusionQuery.querySet : void 0; const v = this._context.getCurrentTexture(); this._mainRenderPassWrapper.colorAttachmentGPUTextures[0].set(v), this._options.antialias ? (DG.format = v.format, this._mainRenderPassWrapper.renderPassDescriptor.colorAttachments[0].resolveTarget = v.createView(DG)) : (jG.format = v.format, this._mainRenderPassWrapper.renderPassDescriptor.colorAttachments[0].view = v.createView(jG)), this.dbgVerboseLogsForFirstFrames && (this._count === void 0 && (this._count = 0), (!this._count || this._count < this.dbgVerboseLogsNumFrames) && console.log("frame #" + this._count + " - main begin pass - texture width=" + this._mainTextureExtends.width, " height=" + this._mainTextureExtends.height + ", setClearStates=" + e, "renderPassDescriptor=", this._mainRenderPassWrapper.renderPassDescriptor)), (s = this._debugPushGroup) === null || s === void 0 || s.call(this, "main pass", 0), this._currentRenderPass = this._renderEncoder.beginRenderPass(this._mainRenderPassWrapper.renderPassDescriptor), this._setDepthTextureFormat(this._mainRenderPassWrapper), this._setColorFormat(this._mainRenderPassWrapper), (a = this._debugFlushPendingCommands) === null || a === void 0 || a.call(this), this._resetRenderPassStates(), this._isStencilEnable || (this._stencilStateComposer.enabled = !1); } /** @internal */ _endCurrentRenderPass() { var e, t, r; if (!this._currentRenderPass) return 0; const n = this._currentPassIsMainPass() ? 2 : 1; return !this._snapshotRendering.endRenderPass(this._currentRenderPass) && !this.compatibilityMode && (this._bundleList.run(this._currentRenderPass), this._bundleList.reset()), this._currentRenderPass.end(), this.dbgVerboseLogsForFirstFrames && (this._count === void 0 && (this._count = 0), (!this._count || this._count < this.dbgVerboseLogsNumFrames) && console.log("frame #" + this._count + " - " + (n === 2 ? "main" : "render target") + " end pass" + (n === 1 ? " - internalTexture.uniqueId=" + ((t = (e = this._currentRenderTarget) === null || e === void 0 ? void 0 : e.texture) === null || t === void 0 ? void 0 : t.uniqueId) : ""))), (r = this._debugPopGroup) === null || r === void 0 || r.call(this, 0), this._currentRenderPass = null, n; } /** * Binds the frame buffer to the specified texture. * @param texture The render target wrapper to render to * @param faceIndex The face of the texture to render to in case of cube texture * @param requiredWidth The width of the target to render to * @param requiredHeight The height of the target to render to * @param forceFullscreenViewport Forces the viewport to be the entire texture/screen if true * @param lodLevel defines the lod level to bind to the frame buffer * @param layer defines the 2d array index to bind to frame buffer to */ bindFramebuffer(e, t = 0, r, n, i, s = 0, a = 0) { var f, o; const d = (f = e.texture) === null || f === void 0 ? void 0 : f._hardwareTexture; this._currentRenderTarget ? this.unBindFramebuffer(this._currentRenderTarget) : this._endCurrentRenderPass(), this._currentRenderTarget = e, this._rttRenderPassWrapper.colorAttachmentGPUTextures[0] = d, this._rttRenderPassWrapper.depthTextureFormat = this._currentRenderTarget._depthStencilTexture ? Ns.GetWebGPUTextureFormat(-1, this._currentRenderTarget._depthStencilTexture.format) : void 0, this._setDepthTextureFormat(this._rttRenderPassWrapper), this._setColorFormat(this._rttRenderPassWrapper), this._rttRenderPassWrapper.colorAttachmentViewDescriptor = { format: this._colorFormat, dimension: Da.E2d, mipLevelCount: 1, baseArrayLayer: e.isCube ? a * 6 + t : a, baseMipLevel: s, arrayLayerCount: 1, aspect: zH.All }, this._rttRenderPassWrapper.depthAttachmentViewDescriptor = { format: this._depthTextureFormat, dimension: Da.E2d, mipLevelCount: 1, baseArrayLayer: e.isCube ? a * 6 + t : a, baseMipLevel: 0, arrayLayerCount: 1, aspect: zH.All }, this.dbgVerboseLogsForFirstFrames && (this._count === void 0 && (this._count = 0), (!this._count || this._count < this.dbgVerboseLogsNumFrames) && console.log("frame #" + this._count + " - bindFramebuffer - rtt name=" + e.label + ", internalTexture.uniqueId=" + ((o = e.texture) === null || o === void 0 ? void 0 : o.uniqueId) + ", face=" + t + ", lodLevel=" + s + ", layer=" + a, "colorAttachmentViewDescriptor=", this._rttRenderPassWrapper.colorAttachmentViewDescriptor, "depthAttachmentViewDescriptor=", this._rttRenderPassWrapper.depthAttachmentViewDescriptor)), this._cachedViewport && !i ? this.setViewport(this._cachedViewport, r, n) : (r || (r = e.width, s && (r = r / Math.pow(2, s))), n || (n = e.height, s && (n = n / Math.pow(2, s))), this._viewport(0, 0, r, n)), this.wipeCaches(); } /** * Unbind the current render target texture from the WebGPU context * @param texture defines the render target wrapper to unbind * @param disableGenerateMipMaps defines a boolean indicating that mipmaps must not be generated * @param onBeforeUnbind defines a function which will be called before the effective unbind */ unBindFramebuffer(e, t = !1, r) { var n, i; const s = this._currentRenderTarget; this._currentRenderTarget = null, r && r(), this._currentRenderTarget = s, this._endCurrentRenderPass(), !((n = e.texture) === null || n === void 0) && n.generateMipMaps && !t && !e.isCube && this._generateMipmaps(e.texture), this._currentRenderTarget = null, this.dbgVerboseLogsForFirstFrames && (this._count === void 0 && (this._count = 0), (!this._count || this._count < this.dbgVerboseLogsNumFrames) && console.log("frame #" + this._count + " - unBindFramebuffer - rtt name=" + e.label + ", internalTexture.uniqueId=", (i = e.texture) === null || i === void 0 ? void 0 : i.uniqueId)), this._mrtAttachments = [], this._cacheRenderPipeline.setMRT([]), this._cacheRenderPipeline.setMRTAttachments(this._mrtAttachments); } /** * Unbind the current render target and bind the default framebuffer */ restoreDefaultFramebuffer() { this._currentRenderTarget ? this.unBindFramebuffer(this._currentRenderTarget) : this._currentRenderPass || this._startMainRenderPass(!1), this._cachedViewport && this.setViewport(this._cachedViewport), this.wipeCaches(); } //------------------------------------------------------------------------------ // Render //------------------------------------------------------------------------------ /** * @internal */ _setColorFormat(e) { var t, r; const n = (r = (t = e.colorAttachmentGPUTextures[0]) === null || t === void 0 ? void 0 : t.format) !== null && r !== void 0 ? r : null; this._cacheRenderPipeline.setColorFormat(n), this._colorFormat !== n && (this._colorFormat = n); } /** * @internal */ _setDepthTextureFormat(e) { this._cacheRenderPipeline.setDepthStencilFormat(e.depthTextureFormat), this._depthTextureFormat !== e.depthTextureFormat && (this._depthTextureFormat = e.depthTextureFormat); } setDitheringState() { } setRasterizerState() { } /** * Set various states to the webGL context * @param culling defines culling state: true to enable culling, false to disable it * @param zOffset defines the value to apply to zOffset (0 by default) * @param force defines if states must be applied even if cache is up to date * @param reverseSide defines if culling must be reversed (CCW if false, CW if true) * @param cullBackFaces true to cull back faces, false to cull front faces (if culling is enabled) * @param stencil stencil states to set * @param zOffsetUnits defines the value to apply to zOffsetUnits (0 by default) */ setState(e, t = 0, r, n = !1, i, s, a = 0) { var f, o; (this._depthCullingState.cull !== e || r) && (this._depthCullingState.cull = e); const d = !((o = (f = this.cullBackFaces) !== null && f !== void 0 ? f : i) !== null && o !== void 0) || o ? 1 : 2; (this._depthCullingState.cullFace !== d || r) && (this._depthCullingState.cullFace = d), this.setZOffset(t), this.setZOffsetUnits(a); const v = n ? this._currentRenderTarget ? 1 : 2 : this._currentRenderTarget ? 2 : 1; (this._depthCullingState.frontFace !== v || r) && (this._depthCullingState.frontFace = v), this._stencilStateComposer.stencilMaterial = s; } _applyRenderPassChanges(e) { const t = this._stencilStateComposer.enabled ? this._mustUpdateStencilRef() : !1, r = this._alphaState.alphaBlend ? this._mustUpdateBlendColor() : !1; this._mustUpdateViewport() && this._applyViewport(e), this._mustUpdateScissor() && this._applyScissor(e), t && this._applyStencilRef(e), r && this._applyBlendColor(e); } _draw(e, t, r, n, i) { var s; const a = this._getCurrentRenderPass(), f = this._bundleList; this.applyStates(); const o = this._currentEffect._pipelineContext; if (this.bindUniformBufferBase(this._currentRenderTarget ? this._ubInvertY : this._ubDontInvertY, 0, Wo.InternalsUBOName), o.uniformBuffer && (o.uniformBuffer.update(), this.bindUniformBufferBase(o.uniformBuffer.getBuffer(), 0, Wo.LeftOvertUBOName)), this._snapshotRendering.play) { this._reportDrawCall(); return; } !this.compatibilityMode && (this._currentDrawContext.isDirty(this._currentMaterialContext.updateId) || this._currentMaterialContext.isDirty || this._currentMaterialContext.forceBindGroupCreation) && (this._currentDrawContext.fastBundle = void 0); const d = !this.compatibilityMode && this._currentDrawContext.fastBundle; let v = a; if (d || this._snapshotRendering.record) { if (this._applyRenderPassChanges(f), !this._snapshotRendering.record) { this._counters.numBundleReuseNonCompatMode++, this._currentDrawContext.indirectDrawBuffer && this._currentDrawContext.setIndirectData(n, i || 1, r), f.addBundle(this._currentDrawContext.fastBundle), this._reportDrawCall(); return; } v = f.getBundleEncoder(this._cacheRenderPipeline.colorFormats, this._depthTextureFormat, this.currentSampleCount), f.numDrawCalls++; } let u = 0; if (this._currentMaterialContext.hasFloatOrDepthTextures) { let H = 1; for (let T = 0; T < o.shaderProcessingContext.textureNames.length; ++T) { const q = o.shaderProcessingContext.textureNames[T], b = (s = this._currentMaterialContext.textures[q]) === null || s === void 0 ? void 0 : s.texture, j = b && b.format >= 13 && b.format <= 18; ((b == null ? void 0 : b.type) === 1 && !this._caps.textureFloatLinearFiltering || j) && (u |= H), H = H << 1; } } this._currentMaterialContext.textureState = u; const l = this._cacheRenderPipeline.getRenderPipeline(t, this._currentEffect, this.currentSampleCount, u), P = this._cacheBindGroups.getBindGroups(o, this._currentDrawContext, this._currentMaterialContext); this._snapshotRendering.record || (this._applyRenderPassChanges(this.compatibilityMode ? null : f), this.compatibilityMode || (this._counters.numBundleCreationNonCompatMode++, v = this._device.createRenderBundleEncoder({ colorFormats: this._cacheRenderPipeline.colorFormats, depthStencilFormat: this._depthTextureFormat, sampleCount: Ns.GetSample(this.currentSampleCount) }))), v.setPipeline(l), this._currentIndexBuffer && v.setIndexBuffer(this._currentIndexBuffer.underlyingResource, this._currentIndexBuffer.is32Bits ? Tq.Uint32 : Tq.Uint16, 0); const p = this._cacheRenderPipeline.vertexBuffers; for (let H = 0; H < p.length; H++) { const T = p[H], q = T.effectiveBuffer; q && v.setVertexBuffer(H, q.underlyingResource, T._validOffsetRange ? 0 : T.byteOffset); } for (let H = 0; H < P.length; H++) v.setBindGroup(H, P[H]); const c = !this.compatibilityMode && !this._snapshotRendering.record; c && this._currentDrawContext.indirectDrawBuffer ? (this._currentDrawContext.setIndirectData(n, i || 1, r), e === 0 ? v.drawIndexedIndirect(this._currentDrawContext.indirectDrawBuffer, 0) : v.drawIndirect(this._currentDrawContext.indirectDrawBuffer, 0)) : e === 0 ? v.drawIndexed(n, i || 1, r, 0, 0) : v.draw(n, i || 1, r, 0), c && (this._currentDrawContext.fastBundle = v.finish(), f.addBundle(this._currentDrawContext.fastBundle)), this._reportDrawCall(); } /** * Draw a list of indexed primitives * @param fillMode defines the primitive to use * @param indexStart defines the starting index * @param indexCount defines the number of index to draw * @param instancesCount defines the number of instances to draw (if instantiation is enabled) */ drawElementsType(e, t, r, n = 1) { this._draw(0, e, t, r, n); } /** * Draw a list of unindexed primitives * @param fillMode defines the primitive to use * @param verticesStart defines the index of first vertex to draw * @param verticesCount defines the count of vertices to draw * @param instancesCount defines the number of instances to draw (if instantiation is enabled) */ drawArraysType(e, t, r, n = 1) { this._currentIndexBuffer = null, this._draw(1, e, t, r, n); } //------------------------------------------------------------------------------ // Dispose //------------------------------------------------------------------------------ /** * Dispose and release all associated resources */ dispose() { var e, t; this._isDisposed = !0, (e = this._mainTexture) === null || e === void 0 || e.destroy(), (t = this._depthTexture) === null || t === void 0 || t.destroy(), this._device.destroy(), super.dispose(); } //------------------------------------------------------------------------------ // Misc //------------------------------------------------------------------------------ /** * Gets the current render width * @param useScreen defines if screen size must be used (or the current render target if any) * @returns a number defining the current render width */ getRenderWidth(e = !1) { var t, r; return !e && this._currentRenderTarget ? this._currentRenderTarget.width : (r = (t = this._renderingCanvas) === null || t === void 0 ? void 0 : t.width) !== null && r !== void 0 ? r : 0; } /** * Gets the current render height * @param useScreen defines if screen size must be used (or the current render target if any) * @returns a number defining the current render height */ getRenderHeight(e = !1) { var t, r; return !e && this._currentRenderTarget ? this._currentRenderTarget.height : (r = (t = this._renderingCanvas) === null || t === void 0 ? void 0 : t.height) !== null && r !== void 0 ? r : 0; } //------------------------------------------------------------------------------ // Errors //------------------------------------------------------------------------------ /** * Get the current error code of the WebGPU context * @returns the error code */ getError() { return 0; } //------------------------------------------------------------------------------ // Unused WebGPU //------------------------------------------------------------------------------ /** * @internal */ bindSamplers() { } /** * @internal */ _bindTextureDirectly() { return !1; } /** * Gets a boolean indicating if all created effects are ready * @returns always true - No parallel shader compilation */ areAllEffectsReady() { return !0; } /** * @internal */ _executeWhenRenderingStateIsCompiled(e, t) { t(); } /** * @internal */ _isRenderingStateCompiled() { return !0; } /** @internal */ _getUnpackAlignement() { return 1; } /** * @internal */ _unpackFlipY() { } /** * @internal */ _bindUnboundFramebuffer() { throw "_bindUnboundFramebuffer is not implementedin WebGPU! You probably want to use restoreDefaultFramebuffer or unBindFramebuffer instead"; } // TODO WEBGPU. All of the below should go once engine split with baseEngine. /** * @internal */ _getSamplingParameters() { throw "_getSamplingParameters is not available in WebGPU"; } /** * @internal */ getUniforms() { return []; } /** * @internal */ setIntArray() { return !1; } /** * @internal */ setIntArray2() { return !1; } /** * @internal */ setIntArray3() { return !1; } /** * @internal */ setIntArray4() { return !1; } /** * @internal */ setArray() { return !1; } /** * @internal */ setArray2() { return !1; } /** * @internal */ setArray3() { return !1; } /** * @internal */ setArray4() { return !1; } /** * @internal */ setMatrices() { return !1; } /** * @internal */ setMatrix3x3() { return !1; } /** * @internal */ setMatrix2x2() { return !1; } /** * @internal */ setFloat() { return !1; } /** * @internal */ setFloat2() { return !1; } /** * @internal */ setFloat3() { return !1; } /** * @internal */ setFloat4() { return !1; } } wn._GLSLslangDefaultOptions = { jsPath: `${ye._DefaultCdnUrl}/glslang/glslang.js`, wasmPath: `${ye._DefaultCdnUrl}/glslang/glslang.wasm` }; wn.UseTWGSL = !0; wn.prototype.setAlphaMode = function(A, e = !1) { if (this._alphaMode === A && (A === 0 && !this._alphaState.alphaBlend || A !== 0 && this._alphaState.alphaBlend)) { if (!e) { const t = A === 0; this.depthCullingState.depthMask !== t && (this.setDepthWrite(t), this._cacheRenderPipeline.setDepthWriteEnabled(t)); } return; } switch (A) { case 0: this._alphaState.alphaBlend = !1; break; case 7: this._alphaState.setAlphaBlendFunctionParameters(1, 771, 1, 1), this._alphaState.alphaBlend = !0; break; case 8: this._alphaState.setAlphaBlendFunctionParameters(1, 771, 1, 771), this._alphaState.alphaBlend = !0; break; case 2: this._alphaState.setAlphaBlendFunctionParameters(770, 771, 1, 1), this._alphaState.alphaBlend = !0; break; case 6: this._alphaState.setAlphaBlendFunctionParameters(1, 1, 0, 1), this._alphaState.alphaBlend = !0; break; case 1: this._alphaState.setAlphaBlendFunctionParameters(770, 1, 0, 1), this._alphaState.alphaBlend = !0; break; case 3: this._alphaState.setAlphaBlendFunctionParameters(0, 769, 1, 1), this._alphaState.alphaBlend = !0; break; case 4: this._alphaState.setAlphaBlendFunctionParameters(774, 0, 1, 1), this._alphaState.alphaBlend = !0; break; case 5: this._alphaState.setAlphaBlendFunctionParameters(770, 769, 1, 1), this._alphaState.alphaBlend = !0; break; case 9: this._alphaState.setAlphaBlendFunctionParameters(32769, 32770, 32771, 32772), this._alphaState.alphaBlend = !0; break; case 10: this._alphaState.setAlphaBlendFunctionParameters(1, 769, 1, 771), this._alphaState.alphaBlend = !0; break; case 11: this._alphaState.setAlphaBlendFunctionParameters(1, 1, 1, 1), this._alphaState.alphaBlend = !0; break; case 12: this._alphaState.setAlphaBlendFunctionParameters(772, 1, 0, 0), this._alphaState.alphaBlend = !0; break; case 13: this._alphaState.setAlphaBlendFunctionParameters(775, 769, 773, 771), this._alphaState.alphaBlend = !0; break; case 14: this._alphaState.setAlphaBlendFunctionParameters(1, 771, 1, 771), this._alphaState.alphaBlend = !0; break; case 15: this._alphaState.setAlphaBlendFunctionParameters(1, 1, 1, 0), this._alphaState.alphaBlend = !0; break; case 16: this._alphaState.setAlphaBlendFunctionParameters(775, 769, 0, 1), this._alphaState.alphaBlend = !0; break; case 17: this._alphaState.setAlphaBlendFunctionParameters(770, 771, 1, 771), this._alphaState.alphaBlend = !0; break; } e || (this.setDepthWrite(A === Ge.ALPHA_DISABLE), this._cacheRenderPipeline.setDepthWriteEnabled(A === Ge.ALPHA_DISABLE)), this._alphaMode = A, this._cacheRenderPipeline.setAlphaBlendEnabled(this._alphaState.alphaBlend), this._cacheRenderPipeline.setAlphaBlendFactors(this._alphaState._blendFunctionParameters, this._alphaState._blendEquationParameters); }; wn.prototype.setAlphaEquation = function(A) { Ge.prototype.setAlphaEquation.call(this, A), this._cacheRenderPipeline.setAlphaBlendFactors(this._alphaState._blendFunctionParameters, this._alphaState._blendEquationParameters); }; class ay { getBindGroups(e, t, r) { if (!r) throw new Error("WebGPUComputeContext.getBindGroups: bindingsMapping is required until browsers support reflection for wgsl shaders!"); if (this._bindGroups.length === 0) { const n = this._bindGroupEntries.length > 0; for (const i in e) { const s = e[i], a = r[i], f = a.group, o = a.binding, d = s.type, v = s.object; let u = s.indexInGroupEntries, l = this._bindGroupEntries[f]; switch (l || (l = this._bindGroupEntries[f] = []), d) { case io.Sampler: { const P = v; u !== void 0 && n ? l[u].resource = this._cacheSampler.getSampler(P) : (s.indexInGroupEntries = l.length, l.push({ binding: o, resource: this._cacheSampler.getSampler(P) })); break; } case io.Texture: case io.TextureWithoutSampler: { const P = v, p = P._texture._hardwareTexture; u !== void 0 && n ? (d === io.Texture && (l[u++].resource = this._cacheSampler.getSampler(P._texture)), l[u].resource = p.view) : (s.indexInGroupEntries = l.length, d === io.Texture && l.push({ binding: o - 1, resource: this._cacheSampler.getSampler(P._texture) }), l.push({ binding: o, resource: p.view })); break; } case io.StorageTexture: { const P = v, p = P._texture._hardwareTexture; p.textureAdditionalUsages & Po.StorageBinding || Se.Error(`computeDispatch: The texture (name=${P.name}, uniqueId=${P.uniqueId}) is not a storage texture!`, 50), u !== void 0 && n ? l[u].resource = p.viewForWriting : (s.indexInGroupEntries = l.length, l.push({ binding: o, resource: p.viewForWriting })); break; } case io.ExternalTexture: { const p = v.underlyingResource; u !== void 0 && n ? l[u].resource = this._device.importExternalTexture({ source: p }) : (s.indexInGroupEntries = l.length, l.push({ binding: o, resource: this._device.importExternalTexture({ source: p }) })); break; } case io.UniformBuffer: case io.StorageBuffer: { const p = (d === io.UniformBuffer, v).getBuffer(), c = p.underlyingResource; u !== void 0 && n ? (l[u].resource.buffer = c, l[u].resource.size = p.capacity) : (s.indexInGroupEntries = l.length, l.push({ binding: o, resource: { buffer: c, offset: 0, size: p.capacity } })); break; } } } for (let i = 0; i < this._bindGroupEntries.length; ++i) { const s = this._bindGroupEntries[i]; if (!s) { this._bindGroups[i] = void 0; continue; } this._bindGroups[i] = this._device.createBindGroup({ layout: t.getBindGroupLayout(i), entries: s }); } this._bindGroups.length = this._bindGroupEntries.length; } return this._bindGroups; } constructor(e, t) { this._device = e, this._cacheSampler = t, this.uniqueId = ay._Counter++, this._bindGroupEntries = [], this.clear(); } clear() { this._bindGroups = []; } } ay._Counter = 0; class Qle { get isAsync() { return !1; } get isReady() { return !!this.stage; } constructor(e) { this._name = "unnamed", this.engine = e; } _getComputeShaderCode() { var e; return (e = this.sources) === null || e === void 0 ? void 0 : e.compute; } dispose() { } } wn.prototype.createComputeContext = function() { return new ay(this._device, this._cacheSampler); }; wn.prototype.createComputeEffect = function(A, e) { const r = (A.computeElement || A.compute || A.computeToken || A.computeSource || A) + "@" + e.defines; if (this._compiledComputeEffects[r]) { const i = this._compiledComputeEffects[r]; return e.onCompiled && i.isReady() && e.onCompiled(i), i; } const n = new Pm(A, e, this, r); return this._compiledComputeEffects[r] = n, n; }; wn.prototype.createComputePipelineContext = function() { return new Qle(this); }; wn.prototype.areAllComputeEffectsReady = function() { for (const A in this._compiledComputeEffects) if (!this._compiledComputeEffects[A].isReady()) return !1; return !0; }; wn.prototype.computeDispatch = function(A, e, t, r, n, i, s) { this._endCurrentRenderPass(); const a = A._pipelineContext, f = e; a.computePipeline || (a.computePipeline = this._device.createComputePipeline({ layout: SS.Auto, compute: a.stage })); const o = this._renderEncoder.beginComputePass(); o.setPipeline(a.computePipeline); const d = f.getBindGroups(t, a.computePipeline, s); for (let v = 0; v < d.length; ++v) { const u = d[v]; u && o.setBindGroup(v, u); } o.dispatchWorkgroups(r, n, i), o.end(); }; wn.prototype.releaseComputeEffects = function() { for (const A in this._compiledComputeEffects) { const e = this._compiledComputeEffects[A].getPipelineContext(); this._deleteComputePipelineContext(e); } this._compiledComputeEffects = {}; }; wn.prototype._prepareComputePipelineContext = function(A, e, t, r, n) { const i = A; this.dbgShowShaderCode && (console.log(r), console.log(e)), i.sources = { compute: e, rawCompute: t }, i.stage = this._createComputePipelineStageDescriptor(e, r, n); }; wn.prototype._releaseComputeEffect = function(A) { this._compiledComputeEffects[A._key] && (delete this._compiledComputeEffects[A._key], this._deleteComputePipelineContext(A.getPipelineContext())); }; wn.prototype._rebuildComputeEffects = function() { for (const A in this._compiledComputeEffects) { const e = this._compiledComputeEffects[A]; e._pipelineContext = null, e._wasPreviouslyReady = !1, e._prepareEffect(); } }; wn.prototype._deleteComputePipelineContext = function(A) { A && A.dispose(); }; wn.prototype._createComputePipelineStageDescriptor = function(A, e, t) { return e ? e = "//" + e.split(` `).join(` //`) + ` ` : e = "", { module: this._device.createShaderModule({ code: e + A }), entryPoint: t }; }; wn.prototype._createDepthStencilCubeTexture = function(A, e) { const t = new As(this, ri.DepthStencil); t.isCube = !0; const r = Object.assign({ bilinearFiltering: !1, comparisonFunction: 0, generateStencil: !1, samples: 1 }, e); return t.format = r.generateStencil ? 13 : 14, this._setupDepthStencilTexture(t, A, r.generateStencil, r.bilinearFiltering, r.comparisonFunction, r.samples), this._textureHelper.createGPUTextureForInternalTexture(t), this._internalTexturesCache.push(t), t; }; wn.prototype.createCubeTexture = function(A, e, t, r, n = null, i = null, s, a = null, f = !1, o = 0, d = 0, v = null, u = !1) { return this.createCubeTextureBase(A, e, t, !!r, n, i, s, a, f, o, d, v, null, (l, P) => { const p = P, c = p[0].width, H = c; this._setCubeMapTextureParams(l, !r), l.format = s ?? -1; const T = this._textureHelper.createGPUTextureForInternalTexture(l, c, H); this._textureHelper.updateCubeTextures(p, T.underlyingResource, c, H, T.format, !1, !1, 0, 0), r || this._generateMipmaps(l, this._uploadEncoder), l.isReady = !0, l.onLoadedObservable.notifyObservers(l), l.onLoadedObservable.clear(), n && n(); }, !!u); }; wn.prototype._setCubeMapTextureParams = function(A, e, t) { A.samplingMode = e ? 3 : 2, A._cachedWrapU = 0, A._cachedWrapV = 0, t && (A._maxLodLevel = t); }; wn.prototype._debugPushGroup = function(A, e) { this._options.enableGPUDebugMarkers && (e === 0 || e === 1 ? this._renderEncoder.pushDebugGroup(A) : this._currentRenderPass ? this._currentRenderPass.pushDebugGroup(A) : this._pendingDebugCommands.push(["push", A])); }; wn.prototype._debugPopGroup = function(A) { this._options.enableGPUDebugMarkers && (A === 0 || A === 1 ? this._renderEncoder.popDebugGroup() : this._currentRenderPass ? this._currentRenderPass.popDebugGroup() : this._pendingDebugCommands.push(["pop", null])); }; wn.prototype._debugInsertMarker = function(A, e) { this._options.enableGPUDebugMarkers && (e === 0 || e === 1 ? this._renderEncoder.insertDebugMarker(A) : this._currentRenderPass ? this._currentRenderPass.insertDebugMarker(A) : this._pendingDebugCommands.push(["insert", A])); }; wn.prototype._debugFlushPendingCommands = function() { for (let A = 0; A < this._pendingDebugCommands.length; ++A) { const [e, t] = this._pendingDebugCommands[A]; switch (e) { case "push": this._debugPushGroup(t); break; case "pop": this._debugPopGroup(); break; case "insert": this._debugInsertMarker(t); break; } } this._pendingDebugCommands.length = 0; }; wn.prototype.updateDynamicIndexBuffer = function(A, e, t = 0) { const r = A; let n; A.is32Bits ? n = e instanceof Uint32Array ? e : new Uint32Array(e) : n = e instanceof Uint16Array ? e : new Uint16Array(e), this._bufferManager.setSubData(r, t, n); }; wn.prototype.updateDynamicVertexBuffer = function(A, e, t, r) { const n = A; t === void 0 && (t = 0); let i; r === void 0 ? (e instanceof Array ? i = new Float32Array(e) : e instanceof ArrayBuffer ? i = new Uint8Array(e) : i = e, r = i.byteLength) : e instanceof Array ? i = new Float32Array(e) : e instanceof ArrayBuffer ? i = new Uint8Array(e) : i = e, this._bufferManager.setSubData(n, t, i, 0, r); }; wn.prototype.updateDynamicTexture = function(A, e, t, r = !1, n, i, s) { var a; if (!A) return; const f = e.width, o = e.height; let d = A._hardwareTexture; !((a = A._hardwareTexture) === null || a === void 0) && a.underlyingResource || (d = this._textureHelper.createGPUTextureForInternalTexture(A, f, o)), this._textureHelper.updateTexture(e, A, f, o, A.depth, d.format, 0, 0, t, r, 0, 0, s), A.generateMipMaps && this._generateMipmaps(A), A.isReady = !0; }; class Yle extends FQ { constructor(e) { super(e); } } An.prototype.setExternalTexture = function(A, e) { this._engine.setExternalTexture(A, e); }; wn.prototype.createExternalTexture = function(A) { return new Yle(A); }; wn.prototype.setExternalTexture = function(A, e) { if (!e) { this._currentMaterialContext.setTexture(A, null); return; } this._setInternalTexture(A, e); }; wn.prototype.unBindMultiColorAttachmentFramebuffer = function(A, e = !1, t) { t && t(); const n = A._attachments.length; this._endCurrentRenderPass(); for (let i = 0; i < n; i++) { const s = A.textures[i]; s.generateMipMaps && !e && !s.isCube && this._generateMipmaps(s); } this._currentRenderTarget = null, this._mrtAttachments = [], this._cacheRenderPipeline.setMRT([]), this._cacheRenderPipeline.setMRTAttachments(this._mrtAttachments); }; wn.prototype.createMultipleRenderTarget = function(A, e, t) { var r, n, i; let s = !1, a = !0, f = !1, o = !1, d = 15, v = 1; const u = 0, l = 3, P = !1, p = 5, c = 3553; let H = [], T = [], q = [], b = [], j = [], w = [], m = [], I = [], N = []; const k = this._createHardwareRenderTargetWrapper(!0, !1, A); e !== void 0 && (s = e.generateMipMaps === void 0 ? !1 : e.generateMipMaps, a = e.generateDepthBuffer === void 0 ? !0 : e.generateDepthBuffer, f = e.generateStencilBuffer === void 0 ? !1 : e.generateStencilBuffer, o = e.generateDepthTexture === void 0 ? !1 : e.generateDepthTexture, v = e.textureCount || 1, d = (r = e.depthTextureFormat) !== null && r !== void 0 ? r : 15, e.types && (H = e.types), e.samplingModes && (T = e.samplingModes), e.useSRGBBuffers && (q = e.useSRGBBuffers), e.formats && (b = e.formats), e.targetTypes && (j = e.targetTypes), e.faceIndex && (w = e.faceIndex), e.layerIndex && (m = e.layerIndex), e.layerCounts && (I = e.layerCounts), N = (n = e.labels) !== null && n !== void 0 ? n : N); const R = A.width || A, y = A.height || A; let O = null; (a || f || o) && (o || (a && f ? d = 13 : a ? d = 14 : d = 19), O = k.createDepthStencilTexture(0, !1, f, 1, d, "MultipleRenderTargetDepthStencil")); const Y = [], ee = [], Z = []; k._generateDepthBuffer = a, k._generateStencilBuffer = f, k._attachments = ee, k._defaultAttachments = Z; for (let te = 0; te < v; te++) { let fe = T[te] || l, _ = H[te] || u; const G = b[te] || p, L = (q[te] || P) && this._caps.supportSRGBBuffers, $ = j[te] || c, ae = (i = I[te]) !== null && i !== void 0 ? i : 1; if ((_ === 1 && !this._caps.textureFloatLinearFiltering || _ === 2 && !this._caps.textureHalfFloatLinearFiltering) && (fe = 1), _ === 1 && !this._caps.textureFloat && (_ = 0, Se.Warn("Float textures are not supported. Render target forced to TEXTURETYPE_UNSIGNED_BYTE type")), ee.push(te + 1), Z.push(t ? te + 1 : te === 0 ? 1 : 0), $ === -1) continue; const Pe = new As(this, ri.MultiRenderTarget); switch (Y[te] = Pe, $) { case 34067: Pe.isCube = !0; break; case 32879: Pe.is3D = !0, Pe.baseDepth = Pe.depth = ae; break; case 35866: Pe.is2DArray = !0, Pe.baseDepth = Pe.depth = ae; break; } Pe.baseWidth = R, Pe.baseHeight = y, Pe.width = R, Pe.height = y, Pe.isReady = !0, Pe.samples = 1, Pe.generateMipMaps = s, Pe.samplingMode = fe, Pe.type = _, Pe._cachedWrapU = 0, Pe._cachedWrapV = 0, Pe._useSRGBBuffer = L, Pe.format = G, Pe.label = N[te], this._internalTexturesCache.push(Pe), this._textureHelper.createGPUTextureForInternalTexture(Pe); } return O && (O.incrementReferences(), Y[v] = O, this._internalTexturesCache.push(O)), k.setTextures(Y), k.setLayerAndFaceIndices(m, w), k; }; wn.prototype.updateMultipleRenderTargetTextureSampleCount = function(A, e) { if (!A || !A.textures || A.textures[0].samples === e) return e; const t = A.textures.length; if (t === 0) return 1; e = Math.min(e, this.getCaps().maxMSAASamples); for (let n = 0; n < t; ++n) { const s = A.textures[n]._hardwareTexture; s == null || s.releaseMSAATexture(); } const r = A._depthStencilTexture === A.textures[t - 1]; for (let n = 0; n < t; ++n) { const i = A.textures[n]; this._textureHelper.createMSAATexture(i, e, !1, n === t - 1 && r ? 0 : n), i.samples = e; } return A._depthStencilTexture && !r && (this._textureHelper.createMSAATexture(A._depthStencilTexture, e), A._depthStencilTexture.samples = e), e; }; wn.prototype.bindAttachments = function(A) { A.length === 0 || !this._currentRenderTarget || (this._mrtAttachments = A, this._currentRenderPass && this._cacheRenderPipeline.setMRTAttachments(A)); }; wn.prototype.buildTextureLayout = function(A) { const e = []; for (let t = 0; t < A.length; t++) A[t] ? e.push(t + 1) : e.push(0); return e; }; wn.prototype.restoreSingleAttachment = function() { }; wn.prototype.restoreSingleAttachmentForRenderTarget = function() { }; wn.prototype.getGPUFrameTimeCounter = function() { return this._timestampQuery.gpuFrameTimeCounter; }; wn.prototype.captureGPUFrameTime = function(A) { this._timestampQuery.enable = A && !!this._caps.timerQuery; }; wn.prototype.createQuery = function() { return this._occlusionQuery.createQuery(); }; wn.prototype.deleteQuery = function(A) { return this._occlusionQuery.deleteQuery(A), this; }; wn.prototype.isQueryResultAvailable = function(A) { return this._occlusionQuery.isQueryResultAvailable(A); }; wn.prototype.getQueryResult = function(A) { return this._occlusionQuery.getQueryResult(A); }; wn.prototype.beginOcclusionQuery = function(A, e) { var t; if (this.compatibilityMode) { if (this._occlusionQuery.canBeginQuery(e)) return (t = this._currentRenderPass) === null || t === void 0 || t.beginOcclusionQuery(e), !0; } else return this._bundleList.addItem(new MQ(e)), !0; return !1; }; wn.prototype.endOcclusionQuery = function() { var A; return this.compatibilityMode ? (A = this._currentRenderPass) === null || A === void 0 || A.endOcclusionQuery() : this._bundleList.addItem(new LQ()), this; }; wn.prototype.createRawTexture = function(A, e, t, r, n, i, s, a = null, f = 0, o = 0, d = !1) { const v = new As(this, ri.Raw); return v.baseWidth = e, v.baseHeight = t, v.width = e, v.height = t, v.format = r, v.generateMipMaps = n, v.samplingMode = s, v.invertY = i, v._compression = a, v.type = f, v._useSRGBBuffer = d, this._doNotHandleContextLost || (v._bufferView = A), this._textureHelper.createGPUTextureForInternalTexture(v, e, t, void 0, o), this.updateRawTexture(v, A, r, i, a, f, d), this._internalTexturesCache.push(v), v; }; wn.prototype.updateRawTexture = function(A, e, t, r, n = null, i = 0, s = !1) { if (A) { if (this._doNotHandleContextLost || (A._bufferView = e, A.invertY = r, A._compression = n, A._useSRGBBuffer = s), e) { const a = A._hardwareTexture; t === 4 && (e = rV(e, A.width, A.height, i)); const o = new Uint8Array(e.buffer, e.byteOffset, e.byteLength); this._textureHelper.updateTexture(o, A, A.width, A.height, A.depth, a.format, 0, 0, r, !1, 0, 0), A.generateMipMaps && this._generateMipmaps(A, this._uploadEncoder); } A.isReady = !0; } }; wn.prototype.createRawCubeTexture = function(A, e, t, r, n, i, s, a = null) { const f = new As(this, ri.CubeRaw); return r === 1 && !this._caps.textureFloatLinearFiltering ? (n = !1, s = 1, Se.Warn("Float texture filtering is not supported. Mipmap generation and sampling mode are forced to false and TEXTURE_NEAREST_SAMPLINGMODE, respectively.")) : r === 2 && !this._caps.textureHalfFloatLinearFiltering ? (n = !1, s = 1, Se.Warn("Half float texture filtering is not supported. Mipmap generation and sampling mode are forced to false and TEXTURE_NEAREST_SAMPLINGMODE, respectively.")) : r === 1 && !this._caps.textureFloatRender ? (n = !1, Se.Warn("Render to float textures is not supported. Mipmap generation forced to false.")) : r === 2 && !this._caps.colorBufferFloat && (n = !1, Se.Warn("Render to half float textures is not supported. Mipmap generation forced to false.")), f.isCube = !0, f.format = t === 4 ? 5 : t, f.type = r, f.generateMipMaps = n, f.width = e, f.height = e, f.samplingMode = s, this._doNotHandleContextLost || (f._bufferViewArray = A), f.invertY = i, f._compression = a, f._cachedWrapU = 0, f._cachedWrapV = 0, this._textureHelper.createGPUTextureForInternalTexture(f), A && this.updateRawCubeTexture(f, A, t, r, i, a), f.isReady = !0, f; }; wn.prototype.updateRawCubeTexture = function(A, e, t, r, n, i = null) { A._bufferViewArray = e, A.invertY = n, A._compression = i; const s = A._hardwareTexture, a = t === 4, f = []; for (let o = 0; o < e.length; ++o) { let d = e[o]; a && (d = rV(e[o], A.width, A.height, r)), f.push(new Uint8Array(d.buffer, d.byteOffset, d.byteLength)); } this._textureHelper.updateCubeTextures(f, s.underlyingResource, A.width, A.height, s.format, n, !1, 0, 0), A.generateMipMaps && this._generateMipmaps(A, this._uploadEncoder), A.isReady = !0; }; wn.prototype.createRawCubeTextureFromUrl = function(A, e, t, r, n, i, s, a, f = null, o = null, d = 3, v = !1) { const u = this.createRawCubeTexture(null, t, r, n, !i, v, d, null); e == null || e.addPendingData(u), u.url = A, this._internalTexturesCache.push(u); const l = (p, c) => { e == null || e.removePendingData(u), o && p && o(p.status + " " + p.statusText, c); }, P = (p) => { const c = u.width, H = s(p); if (!H) return; const T = [0, 2, 4, 1, 3, 5]; if (a) { const q = r === 4, b = a(H), j = u._hardwareTexture, w = [0, 1, 2, 3, 4, 5]; for (let m = 0; m < b.length; m++) { const I = c >> m, N = []; for (let k = 0; k < 6; k++) { let R = b[m][w[k]]; q && (R = rV(R, I, I, n)), N.push(new Uint8Array(R.buffer, R.byteOffset, R.byteLength)); } this._textureHelper.updateCubeTextures(N, j.underlyingResource, I, I, j.format, v, !1, 0, 0); } } else { const q = []; for (let b = 0; b < 6; b++) q.push(H[T[b]]); this.updateRawCubeTexture(u, q, r, n, v); } u.isReady = !0, e == null || e.removePendingData(u), f && f(); }; return this._loadFile(A, (p) => { P(p); }, void 0, e == null ? void 0 : e.offlineProvider, !0, l), u; }; wn.prototype.createRawTexture3D = function(A, e, t, r, n, i, s, a, f = null, o = 0, d = 0) { const v = ri.Raw3D, u = new As(this, v); return u.baseWidth = e, u.baseHeight = t, u.baseDepth = r, u.width = e, u.height = t, u.depth = r, u.format = n, u.type = o, u.generateMipMaps = i, u.samplingMode = a, u.is3D = !0, this._doNotHandleContextLost || (u._bufferView = A), this._textureHelper.createGPUTextureForInternalTexture(u, e, t, void 0, d), this.updateRawTexture3D(u, A, n, s, f, o), this._internalTexturesCache.push(u), u; }; wn.prototype.updateRawTexture3D = function(A, e, t, r, n = null, i = 0) { if (this._doNotHandleContextLost || (A._bufferView = e, A.format = t, A.invertY = r, A._compression = n), e) { const s = A._hardwareTexture; t === 4 && (e = rV(e, A.width, A.height, i)); const f = new Uint8Array(e.buffer, e.byteOffset, e.byteLength); this._textureHelper.updateTexture(f, A, A.width, A.height, A.depth, s.format, 0, 0, r, !1, 0, 0), A.generateMipMaps && this._generateMipmaps(A, this._uploadEncoder); } A.isReady = !0; }; wn.prototype.createRawTexture2DArray = function(A, e, t, r, n, i, s, a, f = null, o = 0, d = 0) { const v = ri.Raw2DArray, u = new As(this, v); return u.baseWidth = e, u.baseHeight = t, u.baseDepth = r, u.width = e, u.height = t, u.depth = r, u.format = n, u.type = o, u.generateMipMaps = i, u.samplingMode = a, u.is2DArray = !0, this._doNotHandleContextLost || (u._bufferView = A), this._textureHelper.createGPUTextureForInternalTexture(u, e, t, r, d), this.updateRawTexture2DArray(u, A, n, s, f, o), this._internalTexturesCache.push(u), u; }; wn.prototype.updateRawTexture2DArray = function(A, e, t, r, n = null, i = 0) { if (this._doNotHandleContextLost || (A._bufferView = e, A.format = t, A.invertY = r, A._compression = n), e) { const s = A._hardwareTexture; t === 4 && (e = rV(e, A.width, A.height, i)); const f = new Uint8Array(e.buffer, e.byteOffset, e.byteLength); this._textureHelper.updateTexture(f, A, A.width, A.height, A.depth, s.format, 0, 0, r, !1, 0, 0), A.generateMipMaps && this._generateMipmaps(A, this._uploadEncoder); } A.isReady = !0; }; function rV(A, e, t, r) { let n, i = 1; r === 1 ? n = new Float32Array(e * t * 4) : r === 2 ? (n = new Uint16Array(e * t * 4), i = 15360) : r === 7 ? n = new Uint32Array(e * t * 4) : n = new Uint8Array(e * t * 4); for (let s = 0; s < e; s++) for (let a = 0; a < t; a++) { const f = (a * e + s) * 3, o = (a * e + s) * 4; n[o + 0] = A[f + 0], n[o + 1] = A[f + 1], n[o + 2] = A[f + 2], n[o + 3] = i; } return n; } wn.prototype._readTexturePixels = function(A, e, t, r = -1, n = 0, i = null, s = !0, a = !1, f = 0, o = 0) { const d = A._hardwareTexture; return s && this.flushFramebuffer(), this._textureHelper.readPixels(d.underlyingResource, f, o, e, t, d.format, r, n, i, a); }; wn.prototype._readTexturePixelsSync = function() { throw "_readTexturePixelsSync is unsupported in WebGPU!"; }; class Mle extends DR { } wn.prototype._createHardwareRenderTargetWrapper = function(A, e, t) { const r = new Mle(A, e, t, this); return this._renderTargetWrapperCache.push(r), r; }; wn.prototype.createRenderTargetTexture = function(A, e) { var t, r, n; const i = this._createHardwareRenderTargetWrapper(!1, !1, A), s = {}; e !== void 0 && typeof e == "object" ? (s.generateMipMaps = e.generateMipMaps, s.generateDepthBuffer = e.generateDepthBuffer === void 0 ? !0 : e.generateDepthBuffer, s.generateStencilBuffer = s.generateDepthBuffer && e.generateStencilBuffer, s.samplingMode = e.samplingMode === void 0 ? 3 : e.samplingMode, s.creationFlags = (t = e.creationFlags) !== null && t !== void 0 ? t : 0, s.noColorAttachment = !!e.noColorAttachment, s.samples = e.samples, s.label = e.label) : (s.generateMipMaps = e, s.generateDepthBuffer = !0, s.generateStencilBuffer = !1, s.samplingMode = 3, s.creationFlags = 0, s.noColorAttachment = !1); const a = s.noColorAttachment ? null : this._createInternalTexture(A, e, !0, ri.RenderTarget); return i.label = (r = s.label) !== null && r !== void 0 ? r : "RenderTargetWrapper", i._samples = (n = s.samples) !== null && n !== void 0 ? n : 1, i._generateDepthBuffer = s.generateDepthBuffer, i._generateStencilBuffer = !!s.generateStencilBuffer, i.setTextures(a), (i._generateDepthBuffer || i._generateStencilBuffer) && i.createDepthStencilTexture( 0, !1, // force false as filtering is not supported for depth textures i._generateStencilBuffer, i.samples, s.generateStencilBuffer ? 13 : 14, s.label ? s.label + "-DepthStencil" : void 0 ), a && (e !== void 0 && typeof e == "object" && e.createMipMaps && !s.generateMipMaps && (a.generateMipMaps = !0), this._textureHelper.createGPUTextureForInternalTexture(a, void 0, void 0, void 0, s.creationFlags), e !== void 0 && typeof e == "object" && e.createMipMaps && !s.generateMipMaps && (a.generateMipMaps = !1)), i; }; wn.prototype._createDepthStencilTexture = function(A, e) { const t = new As(this, ri.DepthStencil); t.label = e.label; const r = Object.assign({ bilinearFiltering: !1, comparisonFunction: 0, generateStencil: !1, samples: 1, depthTextureFormat: e.generateStencil ? 13 : 14 }, e); t.format = r.depthTextureFormat, this._setupDepthStencilTexture(t, A, r.generateStencil, r.bilinearFiltering, r.comparisonFunction, r.samples), this._textureHelper.createGPUTextureForInternalTexture(t); const n = t._hardwareTexture; return t.type = Ns.GetTextureTypeFromFormat(n.format), this._internalTexturesCache.push(t), t; }; wn.prototype._setupDepthStencilTexture = function(A, e, t, r, n, i = 1) { const s = e.width || e, a = e.height || e, f = e.layers || 0; A.baseWidth = s, A.baseHeight = a, A.width = s, A.height = a, A.is2DArray = f > 0, A.depth = f, A.isReady = !0, A.samples = i, A.generateMipMaps = !1, A.samplingMode = r ? 2 : 1, A.type = 1, A._comparisonFunction = n, A._cachedWrapU = 0, A._cachedWrapV = 0; }; wn.prototype.updateRenderTargetTextureSampleCount = function(A, e) { return !A || !A.texture || A.samples === e || (e = Math.min(e, this.getCaps().maxMSAASamples), this._textureHelper.createMSAATexture(A.texture, e), A._depthStencilTexture && (this._textureHelper.createMSAATexture(A._depthStencilTexture, e), A._depthStencilTexture.samples = e), A._samples = e, A.texture.samples = e), e; }; wn.prototype.createRenderTargetCubeTexture = function(A, e) { var t; const r = this._createHardwareRenderTargetWrapper(!1, !0, A), n = Object.assign({ generateMipMaps: !0, generateDepthBuffer: !0, generateStencilBuffer: !1, type: 0, samplingMode: 3, format: 5, samples: 1 }, e); n.generateStencilBuffer = n.generateDepthBuffer && n.generateStencilBuffer, r.label = (t = n.label) !== null && t !== void 0 ? t : "RenderTargetWrapper", r._generateDepthBuffer = n.generateDepthBuffer, r._generateStencilBuffer = n.generateStencilBuffer; const i = new As(this, ri.RenderTarget); return i.width = A, i.height = A, i.depth = 0, i.isReady = !0, i.isCube = !0, i.samples = n.samples, i.generateMipMaps = n.generateMipMaps, i.samplingMode = n.samplingMode, i.type = n.type, i.format = n.format, this._internalTexturesCache.push(i), r.setTextures(i), (r._generateDepthBuffer || r._generateStencilBuffer) && r.createDepthStencilTexture(0, n.samplingMode === void 0 || n.samplingMode === 2 || n.samplingMode === 2 || n.samplingMode === 3 || n.samplingMode === 3 || n.samplingMode === 5 || n.samplingMode === 6 || n.samplingMode === 7 || n.samplingMode === 11, r._generateStencilBuffer, r.samples), e && e.createMipMaps && !n.generateMipMaps && (i.generateMipMaps = !0), this._textureHelper.createGPUTextureForInternalTexture(i), e && e.createMipMaps && !n.generateMipMaps && (i.generateMipMaps = !1), r; }; An.prototype.setTextureSampler = function(A, e) { this._engine.setTextureSampler(A, e); }; wn.prototype.setTextureSampler = function(A, e) { var t; (t = this._currentMaterialContext) === null || t === void 0 || t.setSampler(A, e); }; An.prototype.setStorageBuffer = function(A, e) { this._engine.setStorageBuffer(A, e); }; wn.prototype.createStorageBuffer = function(A, e, t) { return this._createBuffer(A, e | 32, t); }; wn.prototype.updateStorageBuffer = function(A, e, t, r) { const n = A; t === void 0 && (t = 0); let i; r === void 0 ? (e instanceof Array ? i = new Float32Array(e) : e instanceof ArrayBuffer ? i = new Uint8Array(e) : i = e, r = i.byteLength) : e instanceof Array ? i = new Float32Array(e) : e instanceof ArrayBuffer ? i = new Uint8Array(e) : i = e, this._bufferManager.setSubData(n, t, i, 0, r); }; wn.prototype.readFromStorageBuffer = function(A, e, t, r) { t = t || A.capacity; const n = this._bufferManager.createRawBuffer(t, ga.MapRead | ga.CopyDst, void 0, "TempReadFromStorageBuffer"); return this._renderEncoder.copyBufferToBuffer(A.underlyingResource, e ?? 0, n, 0, t), new Promise((i, s) => { this.onEndFrameObservable.addOnce(() => { n.mapAsync(vD.Read, 0, t).then(() => { const a = n.getMappedRange(0, t); let f = r; if (f === void 0) f = new Uint8Array(t), f.set(new Uint8Array(a)); else { const o = f.constructor; f = new o(f.buffer), f.set(new o(a)); } n.unmap(), this._bufferManager.releaseBuffer(n), i(f); }, (a) => s(a)); }); }); }; wn.prototype.setStorageBuffer = function(A, e) { var t, r; (t = this._currentDrawContext) === null || t === void 0 || t.setBuffer(A, (r = e == null ? void 0 : e.getBuffer()) !== null && r !== void 0 ? r : null); }; wn.prototype.createUniformBuffer = function(A, e) { let t; return A instanceof Array ? t = new Float32Array(A) : t = A, this._bufferManager.createBuffer(t, ga.Uniform | ga.CopyDst, e); }; wn.prototype.createDynamicUniformBuffer = function(A, e) { return this.createUniformBuffer(A, e); }; wn.prototype.updateUniformBuffer = function(A, e, t, r) { t === void 0 && (t = 0); const n = A; let i; r === void 0 ? (e instanceof Float32Array ? i = e : i = new Float32Array(e), r = i.byteLength) : e instanceof Float32Array ? i = e : i = new Float32Array(e), this._bufferManager.setSubData(n, t, i, 0, r); }; wn.prototype.bindUniformBufferBase = function(A, e, t) { this._currentDrawContext.setBuffer(t, A); }; wn.prototype.bindUniformBlock = function() { }; function Lle(A) { return !!(A && A.underlyingResource !== void 0); } wn.prototype.updateVideoTexture = function(A, e, t) { var r; if (!A || A._isDisabled) return; this._videoTextureSupported === void 0 && (this._videoTextureSupported = !0); let n = A._hardwareTexture; !((r = A._hardwareTexture) === null || r === void 0) && r.underlyingResource || (n = this._textureHelper.createGPUTextureForInternalTexture(A)), Lle(e) ? (this._textureHelper.copyVideoToTexture(e, A, n.format, !t), A.generateMipMaps && this._generateMipmaps(A), A.isReady = !0) : e && this.createImageBitmap(e).then((i) => { this._textureHelper.updateTexture(i, A, A.width, A.height, A.depth, n.format, 0, 0, !t, !1, 0, 0), A.generateMipMaps && this._generateMipmaps(A), A.isReady = !0; }).catch(() => { A.isReady = !0; }); }; class Kle { /** * Creates an engine based on the capabilities of the underlying hardware * @param canvas Defines the canvas to use to display the result * @param options Defines the options passed to the engine to create the context dependencies * @returns a promise that resolves with the created engine */ static async CreateAsync(e, t) { return await wn.IsSupportedAsync ? wn.CreateAsync(e, t) : Ge.IsSupported ? new Ge(e, void 0, t) : new ote(t); } } class qp { } qp.COPY = 1; qp.CUT = 2; qp.PASTE = 3; class wC { /** *Creates an instance of ClipboardInfo. * @param type Defines the type of event (BABYLON.ClipboardEventTypes) * @param event Defines the related dom event */ constructor(e, t) { this.type = e, this.event = t; } /** * Get the clipboard event's type from the keycode. * @param keyCode Defines the keyCode for the current keyboard event. * @returns {number} */ static GetTypeFromCharacter(e) { switch (e) { case 67: return qp.COPY; case 86: return qp.PASTE; case 88: return qp.CUT; default: return -1; } } } class Pp extends Lo { /** Default material used to render when gizmo is not disabled or hovered */ get coloredMaterial() { return this._coloredMaterial; } /** Material used to render when gizmo is hovered with mouse*/ get hoverMaterial() { return this._hoverMaterial; } /** Material used to render when gizmo is disabled. typically grey.*/ get disableMaterial() { return this._disableMaterial; } /** * Creates an AxisScaleGizmo * @param dragAxis The axis which the gizmo will be able to scale on * @param color The color of the gizmo * @param gizmoLayer The utility layer the gizmo will be added to * @param parent * @param thickness display gizmo axis thickness * @param hoverColor The color of the gizmo when hovering over and dragging * @param disableColor The Color of the gizmo when its disabled */ constructor(e, t = Ne.Gray(), r = Ds.DefaultUtilityLayer, n = null, i = 1, s = Ne.Yellow(), a = Ne.Gray()) { var f, o, d, v, u, l, P; super(r), this._pointerObserver = null, this.snapDistance = 0, this.onSnapObservable = new Oe(), this.uniformScaling = !1, this.sensitivity = 1, this.dragScale = 1, this.incrementalSnap = !1, this._isEnabled = !0, this._parent = null, this._dragging = !1, this._tmpVector = new S(0, 0, 0), this._parent = n, this._coloredMaterial = new Wt("", r.utilityLayerScene), this._coloredMaterial.diffuseColor = t, this._coloredMaterial.specularColor = t.subtract(new Ne(0.1, 0.1, 0.1)), this._hoverMaterial = new Wt("", r.utilityLayerScene), this._hoverMaterial.diffuseColor = s, this._disableMaterial = new Wt("", r.utilityLayerScene), this._disableMaterial.diffuseColor = a, this._disableMaterial.alpha = 0.4, this._gizmoMesh = new Ee("axis", r.utilityLayerScene); const { arrowMesh: p, arrowTail: c } = this._createGizmoMesh(this._gizmoMesh, i), H = this._createGizmoMesh(this._gizmoMesh, i + 4, !0); this._gizmoMesh.lookAt(this._rootMesh.position.add(e)), this._rootMesh.addChild(this._gizmoMesh, Lo.PreserveScaling), this._gizmoMesh.scaling.scaleInPlace(1 / 3); const T = p.position.clone(), q = c.position.clone(), b = c.scaling.clone(), j = (y) => { const O = y * (3 / this._rootMesh.scaling.length()) * 6; p.position.z += O / 3.5, c.scaling.y += O, this.dragScale = c.scaling.y, c.position.z = p.position.z / 2; }, w = () => { p.position.set(T.x, T.y, T.z), c.position.set(q.x, q.y, q.z), c.scaling.set(b.x, b.y, b.z), this.dragScale = c.scaling.y, this._dragging = !1; }; this.dragBehavior = new O9({ dragAxis: e }), this.dragBehavior.moveAttached = !1, this.dragBehavior.updateDragPlane = !1, this._rootMesh.addBehavior(this.dragBehavior); let m = 0, I = 0; const N = { snapDistance: 0 }; this.dragBehavior.onDragObservable.add((y) => { if (this.attachedNode) { const O = this.sensitivity * y.dragDistance * (this.scaleRatio * 3 / this._rootMesh.scaling.length()), Y = this._tmpVector; let ee = !1, Z = 0; if (this.uniformScaling ? Y.setAll(0.57735) : Y.copyFrom(e), this.snapDistance == 0) Y.scaleToRef(O, Y); else { m += O, I += O; const _ = this.incrementalSnap ? I : m; Math.abs(_) > this.snapDistance ? (Z = Math.floor(Math.abs(_) / this.snapDistance), _ < 0 && (Z *= -1), m = m % this.snapDistance, Y.scaleToRef(this.snapDistance * Z, Y), ee = !0) : Y.scaleInPlace(0); } Y.addInPlaceFromFloats(1, 1, 1), Y.x = Math.abs(Y.x) < Pp.MinimumAbsoluteScale ? Pp.MinimumAbsoluteScale * (Y.x < 0 ? -1 : 1) : Y.x, Y.y = Math.abs(Y.y) < Pp.MinimumAbsoluteScale ? Pp.MinimumAbsoluteScale * (Y.y < 0 ? -1 : 1) : Y.y, Y.z = Math.abs(Y.z) < Pp.MinimumAbsoluteScale ? Pp.MinimumAbsoluteScale * (Y.z < 0 ? -1 : 1) : Y.z; const te = this.attachedNode._isMesh ? this.attachedNode : void 0; Math.abs(this.snapDistance) > 0 && this.incrementalSnap ? (this.attachedNode.getWorldMatrix().decompose(void 0, ue.Quaternion[0], ue.Vector3[2], Lo.PreserveScaling ? te : void 0), he.ComposeToRef(Y, ue.Quaternion[0], ue.Vector3[2], ue.Matrix[1])) : (he.ScalingToRef(Y.x, Y.y, Y.z, ue.Matrix[2]), ue.Matrix[2].multiplyToRef(this.attachedNode.getWorldMatrix(), ue.Matrix[1])), ue.Matrix[1].decompose(ue.Vector3[1], void 0, void 0, Lo.PreserveScaling ? te : void 0); const fe = 1e5; Math.abs(ue.Vector3[1].x) < fe && Math.abs(ue.Vector3[1].y) < fe && Math.abs(ue.Vector3[1].z) < fe && this.attachedNode.getWorldMatrix().copyFrom(ue.Matrix[1]), ee && (N.snapDistance = this.snapDistance * Z, this.onSnapObservable.notifyObservers(N)), this._matrixChanged(); } }), this.dragBehavior.onDragStartObservable.add(() => { this._dragging = !0; }), this.dragBehavior.onDragObservable.add((y) => j(y.dragDistance)), this.dragBehavior.onDragEndObservable.add(w), (d = (o = (f = n == null ? void 0 : n.uniformScaleGizmo) === null || f === void 0 ? void 0 : f.dragBehavior) === null || o === void 0 ? void 0 : o.onDragObservable) === null || d === void 0 || d.add((y) => j(y.delta.y)), (l = (u = (v = n == null ? void 0 : n.uniformScaleGizmo) === null || v === void 0 ? void 0 : v.dragBehavior) === null || u === void 0 ? void 0 : u.onDragEndObservable) === null || l === void 0 || l.add(w); const k = { gizmoMeshes: [p, c], colliderMeshes: [H.arrowMesh, H.arrowTail], material: this._coloredMaterial, hoverMaterial: this._hoverMaterial, disableMaterial: this._disableMaterial, active: !1, dragBehavior: this.dragBehavior }; (P = this._parent) === null || P === void 0 || P.addToAxisCache(this._gizmoMesh, k), this._pointerObserver = r.utilityLayerScene.onPointerObservable.add((y) => { var O; if (!this._customMeshSet && (this._isHovered = k.colliderMeshes.indexOf((O = y == null ? void 0 : y.pickInfo) === null || O === void 0 ? void 0 : O.pickedMesh) != -1, !this._parent)) { const Y = this.dragBehavior.enabled ? this._isHovered || this._dragging ? this._hoverMaterial : this._coloredMaterial : this._disableMaterial; this._setGizmoMeshMaterial(k.gizmoMeshes, Y); } }), this.dragBehavior.onEnabledObservable.add((y) => { this._setGizmoMeshMaterial(k.gizmoMeshes, y ? this._coloredMaterial : this._disableMaterial); }); const R = r._getSharedGizmoLight(); R.includedOnlyMeshes = R.includedOnlyMeshes.concat(this._rootMesh.getChildMeshes()); } /** * Create Geometry for Gizmo * @param parentMesh * @param thickness * @param isCollider */ _createGizmoMesh(e, t, r = !1) { const n = k0("yPosMesh", { size: 0.4 * (1 + (t - 1) / 4) }, this.gizmoLayer.utilityLayerScene), i = Ld("cylinder", { diameterTop: 5e-3 * t, height: 0.275, diameterBottom: 5e-3 * t, tessellation: 96 }, this.gizmoLayer.utilityLayerScene); return n.scaling.scaleInPlace(0.1), n.material = this._coloredMaterial, n.rotation.x = Math.PI / 2, n.position.z += 0.3, i.material = this._coloredMaterial, i.position.z += 0.275 / 2, i.rotation.x = Math.PI / 2, r && (n.visibility = 0, i.visibility = 0), e.addChild(n), e.addChild(i), { arrowMesh: n, arrowTail: i }; } _attachedNodeChanged(e) { this.dragBehavior && (this.dragBehavior.enabled = !!e); } /** * If the gizmo is enabled */ set isEnabled(e) { this._isEnabled = e, e ? this._parent && (this.attachedMesh = this._parent.attachedMesh, this.attachedNode = this._parent.attachedNode) : (this.attachedMesh = null, this.attachedNode = null); } get isEnabled() { return this._isEnabled; } /** * Disposes of the gizmo */ dispose() { this.onSnapObservable.clear(), this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(this._pointerObserver), this.dragBehavior.detach(), this._gizmoMesh && this._gizmoMesh.dispose(), [this._coloredMaterial, this._hoverMaterial, this._disableMaterial].forEach((e) => { e && e.dispose(); }), super.dispose(); } /** * Disposes and replaces the current meshes in the gizmo with the specified mesh * @param mesh The mesh to replace the default mesh of the gizmo * @param useGizmoMaterial If the gizmo's default material should be used (default: false) */ setCustomMesh(e, t = !1) { super.setCustomMesh(e), t && (this._rootMesh.getChildMeshes().forEach((r) => { r.material = this._coloredMaterial, r.color && (r.color = this._coloredMaterial.diffuseColor); }), this._customMeshSet = !1); } } Pp.MinimumAbsoluteScale = Dn; class zQ extends Lo { /** * Sets the axis factor * @param factor the Vector3 value */ set axisFactor(e) { this._axisFactor = e; const t = this._scaleBoxesParent.getChildMeshes(); let r = 0; for (let n = 0; n < 3; n++) for (let i = 0; i < 3; i++) for (let s = 0; s < 3; s++) { const a = (n === 1 ? 1 : 0) + (i === 1 ? 1 : 0) + (s === 1 ? 1 : 0); if (!(a === 1 || a === 3)) { if (t[r]) { const f = new S(n - 1, i - 1, s - 1); f.multiplyInPlace(this._axisFactor), t[r].setEnabled(f.lengthSquared() > Dn); } r++; } } } /** * Gets the axis factor * @returns the Vector3 factor value */ get axisFactor() { return this._axisFactor; } /** * Sets scale drag speed value * @param value the new speed value */ set scaleDragSpeed(e) { this._scaleDragSpeed = e; } /** * Gets scale drag speed * @returns the scale speed number */ get scaleDragSpeed() { return this._scaleDragSpeed; } /** Default material used to render when gizmo is not disabled or hovered */ get coloredMaterial() { return this._coloredMaterial; } /** Material used to render when gizmo is hovered with mouse*/ get hoverMaterial() { return this._hoverColoredMaterial; } /** * Get the pointerDragBehavior */ get pointerDragBehavior() { return this._pointerDragBehavior; } /** * Sets the color of the bounding box gizmo * @param color the color to set */ setColor(e) { this._coloredMaterial.emissiveColor = e, this._hoverColoredMaterial.emissiveColor = e.clone().add(new Ne(0.3, 0.3, 0.3)), this._lineBoundingBox.getChildren().forEach((t) => { t.color && (t.color = e); }); } /** * Creates an BoundingBoxGizmo * @param color The color of the gizmo * @param gizmoLayer The utility layer the gizmo will be added to */ constructor(e = Ne.Gray(), t = Ds.DefaultKeepDepthUtilityLayer) { super(t), this._boundingDimensions = new S(1, 1, 1), this._renderObserver = null, this._pointerObserver = null, this._scaleDragSpeed = 0.2, this._tmpQuaternion = new Ze(), this._tmpVector = new S(0, 0, 0), this._tmpRotationMatrix = new he(), this.ignoreChildren = !1, this.includeChildPredicate = null, this.rotationSphereSize = 0.1, this.scaleBoxSize = 0.1, this.fixedDragMeshScreenSize = !1, this.fixedDragMeshBoundsSize = !1, this.fixedDragMeshScreenSizeDistanceFactor = 10, this.scalingSnapDistance = 0, this.rotationSnapDistance = 0, this.onDragStartObservable = new Oe(), this.onScaleBoxDragObservable = new Oe(), this.onScaleBoxDragEndObservable = new Oe(), this.onRotationSphereDragObservable = new Oe(), this.onRotationSphereDragEndObservable = new Oe(), this.scalePivot = null, this._axisFactor = new S(1, 1, 1), this._existingMeshScale = new S(), this._dragMesh = null, this._pointerDragBehavior = new O9(), this.updateScale = !1, this._anchorMesh = new jn("anchor", t.utilityLayerScene), this._coloredMaterial = new Wt("", t.utilityLayerScene), this._coloredMaterial.disableLighting = !0, this._hoverColoredMaterial = new Wt("", t.utilityLayerScene), this._hoverColoredMaterial.disableLighting = !0, this._lineBoundingBox = new jn("", t.utilityLayerScene), this._lineBoundingBox.rotationQuaternion = new Ze(); const r = []; r.push(ka("lines", { points: [new S(0, 0, 0), new S(this._boundingDimensions.x, 0, 0)] }, t.utilityLayerScene)), r.push(ka("lines", { points: [new S(0, 0, 0), new S(0, this._boundingDimensions.y, 0)] }, t.utilityLayerScene)), r.push(ka("lines", { points: [new S(0, 0, 0), new S(0, 0, this._boundingDimensions.z)] }, t.utilityLayerScene)), r.push(ka("lines", { points: [new S(this._boundingDimensions.x, 0, 0), new S(this._boundingDimensions.x, this._boundingDimensions.y, 0)] }, t.utilityLayerScene)), r.push(ka("lines", { points: [new S(this._boundingDimensions.x, 0, 0), new S(this._boundingDimensions.x, 0, this._boundingDimensions.z)] }, t.utilityLayerScene)), r.push(ka("lines", { points: [new S(0, this._boundingDimensions.y, 0), new S(this._boundingDimensions.x, this._boundingDimensions.y, 0)] }, t.utilityLayerScene)), r.push(ka("lines", { points: [new S(0, this._boundingDimensions.y, 0), new S(0, this._boundingDimensions.y, this._boundingDimensions.z)] }, t.utilityLayerScene)), r.push(ka("lines", { points: [new S(0, 0, this._boundingDimensions.z), new S(this._boundingDimensions.x, 0, this._boundingDimensions.z)] }, t.utilityLayerScene)), r.push(ka("lines", { points: [new S(0, 0, this._boundingDimensions.z), new S(0, this._boundingDimensions.y, this._boundingDimensions.z)] }, t.utilityLayerScene)), r.push(ka("lines", { points: [ new S(this._boundingDimensions.x, this._boundingDimensions.y, this._boundingDimensions.z), new S(0, this._boundingDimensions.y, this._boundingDimensions.z) ] }, t.utilityLayerScene)), r.push(ka("lines", { points: [ new S(this._boundingDimensions.x, this._boundingDimensions.y, this._boundingDimensions.z), new S(this._boundingDimensions.x, 0, this._boundingDimensions.z) ] }, t.utilityLayerScene)), r.push(ka("lines", { points: [ new S(this._boundingDimensions.x, this._boundingDimensions.y, this._boundingDimensions.z), new S(this._boundingDimensions.x, this._boundingDimensions.y, 0) ] }, t.utilityLayerScene)), r.forEach((i) => { i.color = e, i.position.addInPlace(new S(-this._boundingDimensions.x / 2, -this._boundingDimensions.y / 2, -this._boundingDimensions.z / 2)), i.isPickable = !1, this._lineBoundingBox.addChild(i); }), this._rootMesh.addChild(this._lineBoundingBox), this.setColor(e), this._rotateSpheresParent = new jn("", t.utilityLayerScene), this._rotateSpheresParent.rotationQuaternion = new Ze(); for (let i = 0; i < 12; i++) { const s = UA("", { diameter: 1 }, t.utilityLayerScene); s.rotationQuaternion = new Ze(), s.material = this._coloredMaterial, s.isNearGrabbable = !0; const a = new O9({}); a.moveAttached = !1, a.updateDragPlane = !1, s.addBehavior(a); const f = new S(1, 0, 0); let o = 0, d = 0; a.onDragStartObservable.add(() => { f.copyFrom(s.forward), o = 0, d = 0; }), a.onDragObservable.add((v) => { if (this.onRotationSphereDragObservable.notifyObservers({}), this.attachedMesh) { const u = this.attachedMesh.parent; if (u && u.scaling && u.scaling.isNonUniformWithinEpsilon(1e-3)) { Se.Warn("BoundingBoxGizmo controls are not supported on child meshes with non-uniform parent scaling"); return; } Vs._RemoveAndStorePivotPoint(this.attachedMesh); const l = f, P = v.dragPlaneNormal.scale(S.Dot(v.dragPlaneNormal, l)), p = l.subtract(P).normalizeToNew(); let c = S.Dot(p, v.delta) < 0 ? Math.abs(v.delta.length()) : -Math.abs(v.delta.length()); if (c = c / this._boundingDimensions.length() * this._anchorMesh.scaling.length(), this.attachedMesh.rotationQuaternion || (this.attachedMesh.rotationQuaternion = Ze.RotationYawPitchRoll(this.attachedMesh.rotation.y, this.attachedMesh.rotation.x, this.attachedMesh.rotation.z)), this._anchorMesh.rotationQuaternion || (this._anchorMesh.rotationQuaternion = Ze.RotationYawPitchRoll(this._anchorMesh.rotation.y, this._anchorMesh.rotation.x, this._anchorMesh.rotation.z)), o += c, Math.abs(o) <= 2 * Math.PI) { if (this.rotationSnapDistance > 0) { const H = Math.floor(Math.abs(o) / this.rotationSnapDistance) * (o < 0 ? -1 : 1), T = this.rotationSnapDistance * H; c = T - d, d = T; } i >= 8 ? Ze.RotationYawPitchRollToRef(0, 0, c, this._tmpQuaternion) : i >= 4 ? Ze.RotationYawPitchRollToRef(c, 0, 0, this._tmpQuaternion) : Ze.RotationYawPitchRollToRef(0, c, 0, this._tmpQuaternion), this.attachedMesh.isUsingPivotMatrix() && this._anchorMesh.position.copyFrom(this.attachedMesh.position), this._anchorMesh.addChild(this.attachedMesh), this._anchorMesh.getScene().useRightHandedSystem && this._tmpQuaternion.conjugateInPlace(), this._tmpQuaternion.normalize(), this._anchorMesh.rotationQuaternion.multiplyToRef(this._tmpQuaternion, this._anchorMesh.rotationQuaternion), this._anchorMesh.rotationQuaternion.normalize(), this._anchorMesh.removeChild(this.attachedMesh), this.attachedMesh.setParent(u); } this.updateBoundingBox(), Vs._RestorePivotPoint(this.attachedMesh); } this._updateDummy(); }), a.onDragStartObservable.add(() => { this.onDragStartObservable.notifyObservers({}), this._selectNode(s); }), a.onDragEndObservable.add((v) => { this.onRotationSphereDragEndObservable.notifyObservers({}), this._selectNode(null), this._updateDummy(), this._unhoverMeshOnTouchUp(v.pointerInfo, s); }), this._rotateSpheresParent.addChild(s); } this._rootMesh.addChild(this._rotateSpheresParent), this._scaleBoxesParent = new jn("", t.utilityLayerScene), this._scaleBoxesParent.rotationQuaternion = new Ze(); for (let i = 0; i < 3; i++) for (let s = 0; s < 3; s++) for (let a = 0; a < 3; a++) { const f = (i === 1 ? 1 : 0) + (s === 1 ? 1 : 0) + (a === 1 ? 1 : 0); if (f === 1 || f === 3) continue; const o = k0("", { size: 1 }, t.utilityLayerScene); o.material = this._coloredMaterial, o._internalMetadata = f === 2, o.isNearGrabbable = !0; const d = new S(i - 1, s - 1, a - 1).normalize(), v = new O9({ dragAxis: d }); v.updateDragPlane = !1, v.moveAttached = !1; let u = 0, l = 0; o.addBehavior(v), v.onDragObservable.add((P) => { if (this.onScaleBoxDragObservable.notifyObservers({}), this.attachedMesh) { const p = this.attachedMesh.parent; if (p && p.scaling && p.scaling.isNonUniformWithinEpsilon(1e-3)) { Se.Warn("BoundingBoxGizmo controls are not supported on child meshes with non-uniform parent scaling"); return; } Vs._RemoveAndStorePivotPoint(this.attachedMesh); let c = P.dragDistance / this._boundingDimensions.length() * this._anchorMesh.scaling.length(); if (u += c, this.scalingSnapDistance > 0) { const T = Math.floor(Math.abs(u) / this.scalingSnapDistance) * (u < 0 ? -1 : 1), q = this.scalingSnapDistance * T; c = q - l, l = q; } const H = new S(c, c, c); f === 2 && (H.x *= Math.abs(d.x), H.y *= Math.abs(d.y), H.z *= Math.abs(d.z)), H.scaleInPlace(this._scaleDragSpeed), H.multiplyInPlace(this._axisFactor), this.updateBoundingBox(), this.scalePivot ? (this.attachedMesh.getWorldMatrix().getRotationMatrixToRef(this._tmpRotationMatrix), this._boundingDimensions.scaleToRef(0.5, this._tmpVector), S.TransformCoordinatesToRef(this._tmpVector, this._tmpRotationMatrix, this._tmpVector), this._anchorMesh.position.subtractInPlace(this._tmpVector), this._boundingDimensions.multiplyToRef(this.scalePivot, this._tmpVector), S.TransformCoordinatesToRef(this._tmpVector, this._tmpRotationMatrix, this._tmpVector), this._anchorMesh.position.addInPlace(this._tmpVector)) : (o.absolutePosition.subtractToRef(this._anchorMesh.position, this._tmpVector), this._anchorMesh.position.subtractInPlace(this._tmpVector), this.attachedMesh.isUsingPivotMatrix() && this._anchorMesh.position.subtractInPlace(this.attachedMesh.getPivotPoint())), this._anchorMesh.addChild(this.attachedMesh), this._anchorMesh.scaling.addInPlace(H), (this._anchorMesh.scaling.x < 0 || this._anchorMesh.scaling.y < 0 || this._anchorMesh.scaling.z < 0) && this._anchorMesh.scaling.subtractInPlace(H), this._anchorMesh.removeChild(this.attachedMesh), this.attachedMesh.setParent(p), Vs._RestorePivotPoint(this.attachedMesh); } this._updateDummy(); }), v.onDragStartObservable.add(() => { this.onDragStartObservable.notifyObservers({}), this._selectNode(o), u = 0, l = 0; }), v.onDragEndObservable.add((P) => { this.onScaleBoxDragEndObservable.notifyObservers({}), this._selectNode(null), this._updateDummy(), this._unhoverMeshOnTouchUp(P.pointerInfo, o); }), this._scaleBoxesParent.addChild(o); } this._rootMesh.addChild(this._scaleBoxesParent); const n = []; this._pointerObserver = t.utilityLayerScene.onPointerObservable.add((i) => { n[i.event.pointerId] ? i.pickInfo && i.pickInfo.pickedMesh != n[i.event.pointerId] && (n[i.event.pointerId].material = this._coloredMaterial, delete n[i.event.pointerId], this._isHovered = !1) : this._rotateSpheresParent.getChildMeshes().concat(this._scaleBoxesParent.getChildMeshes()).forEach((s) => { i.pickInfo && i.pickInfo.pickedMesh == s && (n[i.event.pointerId] = s, s.material = this._hoverColoredMaterial, this._isHovered = !0); }); }), this._renderObserver = this.gizmoLayer.originalScene.onBeforeRenderObservable.add(() => { this.attachedMesh && !this._existingMeshScale.equals(this.attachedMesh.scaling) ? this.updateBoundingBox() : (this.fixedDragMeshScreenSize || this.fixedDragMeshBoundsSize) && (this._updateRotationSpheres(), this._updateScaleBoxes()), this._dragMesh && this.attachedMesh && this._pointerDragBehavior.dragging && (this._lineBoundingBox.position.rotateByQuaternionToRef(this._rootMesh.rotationQuaternion, this._tmpVector), this.attachedMesh.setAbsolutePosition(this._dragMesh.position.add(this._tmpVector.scale(-1)))); }), this.updateBoundingBox(); } _attachedNodeChanged(e) { if (e) { this._anchorMesh.scaling.setAll(1), Vs._RemoveAndStorePivotPoint(e); const t = e.parent; this._anchorMesh.addChild(e), this._anchorMesh.removeChild(e), e.setParent(t), Vs._RestorePivotPoint(e), this.updateBoundingBox(), e.getChildMeshes(!1).forEach((r) => { r.markAsDirty("scaling"); }), this.gizmoLayer.utilityLayerScene.onAfterRenderObservable.addOnce(() => { this._updateDummy(); }); } } _selectNode(e) { this._rotateSpheresParent.getChildMeshes().concat(this._scaleBoxesParent.getChildMeshes()).forEach((t) => { t.isVisible = !e || t == e; }); } _unhoverMeshOnTouchUp(e, t) { (e == null ? void 0 : e.event) instanceof PointerEvent && (e == null ? void 0 : e.event.pointerType) === "touch" && (t.material = this._coloredMaterial); } /** * returns an array containing all boxes used for scaling (in increasing x, y and z orders) */ getScaleBoxes() { return this._scaleBoxesParent.getChildMeshes(); } /** * Updates the bounding box information for the Gizmo */ updateBoundingBox() { if (this.attachedMesh) { Vs._RemoveAndStorePivotPoint(this.attachedMesh); const e = this.attachedMesh.parent; this.attachedMesh.setParent(null), this._update(), this.attachedMesh.rotationQuaternion || (this.attachedMesh.rotationQuaternion = Ze.RotationYawPitchRoll(this.attachedMesh.rotation.y, this.attachedMesh.rotation.x, this.attachedMesh.rotation.z)), this._anchorMesh.rotationQuaternion || (this._anchorMesh.rotationQuaternion = Ze.RotationYawPitchRoll(this._anchorMesh.rotation.y, this._anchorMesh.rotation.x, this._anchorMesh.rotation.z)), this._anchorMesh.rotationQuaternion.copyFrom(this.attachedMesh.rotationQuaternion), this._tmpQuaternion.copyFrom(this.attachedMesh.rotationQuaternion), this._tmpVector.copyFrom(this.attachedMesh.position), this.attachedMesh.rotationQuaternion.set(0, 0, 0, 1), this.attachedMesh.position.set(0, 0, 0); const t = this.attachedMesh.getHierarchyBoundingVectors(!this.ignoreChildren, this.includeChildPredicate); t.max.subtractToRef(t.min, this._boundingDimensions), this._lineBoundingBox.scaling.copyFrom(this._boundingDimensions), this._lineBoundingBox.position.set((t.max.x + t.min.x) / 2, (t.max.y + t.min.y) / 2, (t.max.z + t.min.z) / 2), this._rotateSpheresParent.position.copyFrom(this._lineBoundingBox.position), this._scaleBoxesParent.position.copyFrom(this._lineBoundingBox.position), this._lineBoundingBox.computeWorldMatrix(), this._anchorMesh.position.copyFrom(this._lineBoundingBox.absolutePosition), this.attachedMesh.rotationQuaternion.copyFrom(this._tmpQuaternion), this.attachedMesh.position.copyFrom(this._tmpVector), this.attachedMesh.setParent(e); } this._updateRotationSpheres(), this._updateScaleBoxes(), this.attachedMesh && (this._existingMeshScale.copyFrom(this.attachedMesh.scaling), Vs._RestorePivotPoint(this.attachedMesh)); } _updateRotationSpheres() { const e = this._rotateSpheresParent.getChildMeshes(); for (let t = 0; t < 3; t++) for (let r = 0; r < 2; r++) for (let n = 0; n < 2; n++) { const i = t * 4 + r * 2 + n; if (t == 0 && (e[i].position.set(this._boundingDimensions.x / 2, this._boundingDimensions.y * r, this._boundingDimensions.z * n), e[i].position.addInPlace(new S(-this._boundingDimensions.x / 2, -this._boundingDimensions.y / 2, -this._boundingDimensions.z / 2)), e[i].lookAt(S.Cross(e[i].position.normalizeToNew(), S.Right()).normalizeToNew().add(e[i].position))), t == 1 && (e[i].position.set(this._boundingDimensions.x * r, this._boundingDimensions.y / 2, this._boundingDimensions.z * n), e[i].position.addInPlace(new S(-this._boundingDimensions.x / 2, -this._boundingDimensions.y / 2, -this._boundingDimensions.z / 2)), e[i].lookAt(S.Cross(e[i].position.normalizeToNew(), S.Up()).normalizeToNew().add(e[i].position))), t == 2 && (e[i].position.set(this._boundingDimensions.x * r, this._boundingDimensions.y * n, this._boundingDimensions.z / 2), e[i].position.addInPlace(new S(-this._boundingDimensions.x / 2, -this._boundingDimensions.y / 2, -this._boundingDimensions.z / 2)), e[i].lookAt(S.Cross(e[i].position.normalizeToNew(), S.Forward()).normalizeToNew().add(e[i].position))), this.fixedDragMeshScreenSize && this.gizmoLayer.utilityLayerScene.activeCamera) { e[i].absolutePosition.subtractToRef(this.gizmoLayer.utilityLayerScene.activeCamera.position, this._tmpVector); const s = this.rotationSphereSize * this._tmpVector.length() / this.fixedDragMeshScreenSizeDistanceFactor; e[i].scaling.set(s, s, s); } else this.fixedDragMeshBoundsSize ? e[i].scaling.set(this.rotationSphereSize * this._boundingDimensions.x, this.rotationSphereSize * this._boundingDimensions.y, this.rotationSphereSize * this._boundingDimensions.z) : e[i].scaling.set(this.rotationSphereSize, this.rotationSphereSize, this.rotationSphereSize); } } _updateScaleBoxes() { const e = this._scaleBoxesParent.getChildMeshes(); let t = 0; for (let r = 0; r < 3; r++) for (let n = 0; n < 3; n++) for (let i = 0; i < 3; i++) { const s = (r === 1 ? 1 : 0) + (n === 1 ? 1 : 0) + (i === 1 ? 1 : 0); if (!(s === 1 || s === 3)) { if (e[t]) if (e[t].position.set(this._boundingDimensions.x * (r / 2), this._boundingDimensions.y * (n / 2), this._boundingDimensions.z * (i / 2)), e[t].position.addInPlace(new S(-this._boundingDimensions.x / 2, -this._boundingDimensions.y / 2, -this._boundingDimensions.z / 2)), this.fixedDragMeshScreenSize && this.gizmoLayer.utilityLayerScene.activeCamera) { e[t].absolutePosition.subtractToRef(this.gizmoLayer.utilityLayerScene.activeCamera.position, this._tmpVector); const a = this.scaleBoxSize * this._tmpVector.length() / this.fixedDragMeshScreenSizeDistanceFactor; e[t].scaling.set(a, a, a); } else this.fixedDragMeshBoundsSize ? e[t].scaling.set(this.scaleBoxSize * this._boundingDimensions.x, this.scaleBoxSize * this._boundingDimensions.y, this.scaleBoxSize * this._boundingDimensions.z) : e[t].scaling.set(this.scaleBoxSize, this.scaleBoxSize, this.scaleBoxSize); t++; } } } /** * Enables rotation on the specified axis and disables rotation on the others * @param axis The list of axis that should be enabled (eg. "xy" or "xyz") */ setEnabledRotationAxis(e) { this._rotateSpheresParent.getChildMeshes().forEach((t, r) => { r < 4 ? t.setEnabled(e.indexOf("x") != -1) : r < 8 ? t.setEnabled(e.indexOf("y") != -1) : t.setEnabled(e.indexOf("z") != -1); }); } /** * Enables/disables scaling * @param enable if scaling should be enabled * @param homogeneousScaling defines if scaling should only be homogeneous */ setEnabledScaling(e, t = !1) { this._scaleBoxesParent.getChildMeshes().forEach((r) => { let n = e; t && r._internalMetadata === !0 && (n = !1), r.setEnabled(n); }); } _updateDummy() { this._dragMesh && (this._dragMesh.position.copyFrom(this._lineBoundingBox.getAbsolutePosition()), this._dragMesh.scaling.copyFrom(this._lineBoundingBox.scaling), this._dragMesh.rotationQuaternion.copyFrom(this._rootMesh.rotationQuaternion)); } /** * Enables a pointer drag behavior on the bounding box of the gizmo */ enableDragBehavior() { this._dragMesh = k0("dummy", { size: 1 }, this.gizmoLayer.utilityLayerScene), this._dragMesh.visibility = 0, this._dragMesh.rotationQuaternion = new Ze(), this._pointerDragBehavior.useObjectOrientationForDragging = !1, this._dragMesh.addBehavior(this._pointerDragBehavior); } /** * Disposes of the gizmo */ dispose() { this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(this._pointerObserver), this.gizmoLayer.originalScene.onBeforeRenderObservable.remove(this._renderObserver), this._lineBoundingBox.dispose(), this._rotateSpheresParent.dispose(), this._scaleBoxesParent.dispose(), this._dragMesh && this._dragMesh.dispose(), super.dispose(); } /** * Makes a mesh not pickable and wraps the mesh inside of a bounding box mesh that is pickable. (This is useful to avoid picking within complex geometry) * @param mesh the mesh to wrap in the bounding box mesh and make not pickable * @returns the bounding box mesh with the passed in mesh as a child */ static MakeNotPickableAndWrapInBoundingBox(e) { const t = (a) => { a.isPickable = !1, a.getChildMeshes().forEach((f) => { t(f); }); }; t(e), e.rotationQuaternion || (e.rotationQuaternion = Ze.RotationYawPitchRoll(e.rotation.y, e.rotation.x, e.rotation.z)); const r = e.position.clone(), n = e.rotationQuaternion.clone(); e.rotationQuaternion.set(0, 0, 0, 1), e.position.set(0, 0, 0); const i = k0("box", { size: 1 }, e.getScene()), s = e.getHierarchyBoundingVectors(); return s.max.subtractToRef(s.min, i.scaling), i.scaling.y === 0 && (i.scaling.y = Dn), i.scaling.x === 0 && (i.scaling.x = Dn), i.scaling.z === 0 && (i.scaling.z = Dn), i.position.set((s.max.x + s.min.x) / 2, (s.max.y + s.min.y) / 2, (s.max.z + s.min.z) / 2), e.addChild(i), e.rotationQuaternion.copyFrom(n), e.position.copyFrom(r), e.removeChild(i), i.addChild(e), i.visibility = 0, i; } /** * CustomMeshes are not supported by this gizmo */ setCustomMesh() { Se.Error("Custom meshes are not supported on this gizmo"); } } class S0 extends Lo { /** Default material used to render when gizmo is not disabled or hovered */ get coloredMaterial() { return this._coloredMaterial; } /** Material used to render when gizmo is hovered with mouse */ get hoverMaterial() { return this._hoverMaterial; } /** Color used to render the drag angle sector when gizmo is rotated with mouse */ set rotationColor(e) { this._rotationShaderMaterial.setColor3("rotationColor", e); } /** Material used to render when gizmo is disabled. typically grey. */ get disableMaterial() { return this._disableMaterial; } /** * Creates a PlaneRotationGizmo * @param planeNormal The normal of the plane which the gizmo will be able to rotate on * @param color The color of the gizmo * @param gizmoLayer The utility layer the gizmo will be added to * @param tessellation Amount of tessellation to be used when creating rotation circles * @param parent * @param useEulerRotation Use and update Euler angle instead of quaternion * @param thickness display gizmo axis thickness * @param hoverColor The color of the gizmo when hovering over and dragging * @param disableColor The Color of the gizmo when its disabled */ constructor(e, t = Ne.Gray(), r = Ds.DefaultUtilityLayer, n = 32, i = null, s = !1, a = 1, f = Ne.Yellow(), o = Ne.Gray()) { var d; super(r), this._pointerObserver = null, this.snapDistance = 0, this.onSnapObservable = new Oe(), this.angle = 0, this.sensitivity = 1, this._isEnabled = !0, this._parent = null, this._dragging = !1, this._angles = new S(), this._parent = i, this._coloredMaterial = new Wt("", r.utilityLayerScene), this._coloredMaterial.diffuseColor = t, this._coloredMaterial.specularColor = t.subtract(new Ne(0.1, 0.1, 0.1)), this._hoverMaterial = new Wt("", r.utilityLayerScene), this._hoverMaterial.diffuseColor = f, this._hoverMaterial.specularColor = f, this._disableMaterial = new Wt("", r.utilityLayerScene), this._disableMaterial.diffuseColor = o, this._disableMaterial.alpha = 0.4, this._gizmoMesh = new Ee("", r.utilityLayerScene); const { rotationMesh: v, collider: u } = this._createGizmoMesh(this._gizmoMesh, a, n); this._rotationDisplayPlane = u4("rotationDisplay", { size: 0.6, updatable: !1 }, this.gizmoLayer.utilityLayerScene), this._rotationDisplayPlane.rotation.z = Math.PI * 0.5, this._rotationDisplayPlane.parent = this._gizmoMesh, this._rotationDisplayPlane.setEnabled(!1), An.ShadersStore.rotationGizmoVertexShader = S0._RotationGizmoVertexShader, An.ShadersStore.rotationGizmoFragmentShader = S0._RotationGizmoFragmentShader, this._rotationShaderMaterial = new Zo("shader", this.gizmoLayer.utilityLayerScene, { vertex: "rotationGizmo", fragment: "rotationGizmo" }, { attributes: ["position", "uv"], uniforms: ["worldViewProjection", "angles", "rotationColor"] }), this._rotationShaderMaterial.backFaceCulling = !1, this.rotationColor = f, this._rotationDisplayPlane.material = this._rotationShaderMaterial, this._rotationDisplayPlane.visibility = 0.999, this._gizmoMesh.lookAt(this._rootMesh.position.add(e)), this._rootMesh.addChild(this._gizmoMesh, Lo.PreserveScaling), this._gizmoMesh.scaling.scaleInPlace(1 / 3), this.dragBehavior = new O9({ dragPlaneNormal: e }), this.dragBehavior.moveAttached = !1, this.dragBehavior.maxDragAngle = S0.MaxDragAngle, this.dragBehavior._useAlternatePickedPointAboveMaxDragAngle = !0, this._rootMesh.addBehavior(this.dragBehavior); const l = new S(), P = new he(), p = new S(); let c = new S(); this.dragBehavior.onDragStartObservable.add((m) => { this.attachedNode && (l.copyFrom(m.dragPlanePoint), this._rotationDisplayPlane.setEnabled(!0), this._rotationDisplayPlane.getWorldMatrix().invertToRef(P), S.TransformCoordinatesToRef(m.dragPlanePoint, P, l), this._angles.x = Math.atan2(l.y, l.x) + Math.PI, this._angles.y = 0, this._angles.z = this.updateGizmoRotationToMatchAttachedMesh ? 1 : 0, this._dragging = !0, l.copyFrom(m.dragPlanePoint), this._rotationShaderMaterial.setVector3("angles", this._angles), this.angle = 0); }), this.dragBehavior.onDragEndObservable.add(() => { this._dragging = !1, this._rotationDisplayPlane.setEnabled(!1); }); const H = { snapDistance: 0 }; let T = 0; const q = new he(), b = new Ze(); this.dragBehavior.onDragObservable.add((m) => { if (this.attachedNode) { const I = new S(1, 1, 1), N = new Ze(0, 0, 0, 1), k = new S(0, 0, 0); if (this.attachedNode.getWorldMatrix().decompose(I, N, k), !(Math.abs(Math.abs(I.x) - Math.abs(I.y)) <= Dn && Math.abs(Math.abs(I.x) - Math.abs(I.z)) <= Dn) && this.updateGizmoRotationToMatchAttachedMesh) { Se.Warn("Unable to use a rotation gizmo matching mesh rotation with non uniform scaling. Use uniform scaling or set updateGizmoRotationToMatchAttachedMesh to false."); return; } N.normalize(); const y = this.updateGizmoPositionToMatchAttachedMesh ? k : this._rootMesh.absolutePosition, O = m.dragPlanePoint.subtract(y).normalize(), Y = l.subtract(y).normalize(), ee = S.Cross(O, Y), Z = S.Dot(O, Y); let te = Math.atan2(ee.length(), Z) * this.sensitivity; p.copyFrom(e), c.copyFrom(e), this.updateGizmoRotationToMatchAttachedMesh && (N.toRotationMatrix(P), c = S.TransformCoordinates(p, P)); let fe = !1; if (r.utilityLayerScene.activeCamera) { const $ = r.utilityLayerScene.activeCamera.position.subtract(y).normalize(); S.Dot($, c) > 0 && (p.scaleInPlace(-1), c.scaleInPlace(-1), fe = !0); } S.Dot(c, ee) > 0 && (te = -te), ue.Vector3[0].set(te, 0, 0), this.dragBehavior.validateDrag(ue.Vector3[0]) || (te = 0); let G = !1; if (this.snapDistance != 0) if (T += te, Math.abs(T) > this.snapDistance) { let $ = Math.floor(Math.abs(T) / this.snapDistance); T < 0 && ($ *= -1), T = T % this.snapDistance, te = this.snapDistance * $, G = !0; } else te = 0; const L = Math.sin(te / 2); if (b.set(p.x * L, p.y * L, p.z * L, Math.cos(te / 2)), q.determinant() > 0) { const $ = new S(); b.toEulerAnglesToRef($), Ze.RotationYawPitchRollToRef($.y, -$.x, -$.z, b); } if (this.updateGizmoRotationToMatchAttachedMesh) N.multiplyToRef(b, N), N.normalize(), he.ComposeToRef(I, N, k, this.attachedNode.getWorldMatrix()); else { b.toRotationMatrix(ue.Matrix[0]); const $ = this.attachedNode.getWorldMatrix().getTranslation(); this.attachedNode.getWorldMatrix().multiplyToRef(ue.Matrix[0], this.attachedNode.getWorldMatrix()), this.attachedNode.getWorldMatrix().setTranslation($); } l.copyFrom(m.dragPlanePoint), G && (H.snapDistance = te, this.onSnapObservable.notifyObservers(H)), this._angles.y += te, this.angle += fe ? -te : te, this._rotationShaderMaterial.setVector3("angles", this._angles), this._matrixChanged(); } }); const j = r._getSharedGizmoLight(); j.includedOnlyMeshes = j.includedOnlyMeshes.concat(this._rootMesh.getChildMeshes(!1)); const w = { colliderMeshes: [u], gizmoMeshes: [v], material: this._coloredMaterial, hoverMaterial: this._hoverMaterial, disableMaterial: this._disableMaterial, active: !1, dragBehavior: this.dragBehavior }; (d = this._parent) === null || d === void 0 || d.addToAxisCache(this._gizmoMesh, w), this._pointerObserver = r.utilityLayerScene.onPointerObservable.add((m) => { var I; if (!this._customMeshSet && (this.dragBehavior.maxDragAngle = S0.MaxDragAngle, this._isHovered = w.colliderMeshes.indexOf((I = m == null ? void 0 : m.pickInfo) === null || I === void 0 ? void 0 : I.pickedMesh) != -1, !this._parent)) { const N = w.dragBehavior.enabled ? this._isHovered || this._dragging ? this._hoverMaterial : this._coloredMaterial : this._disableMaterial; this._setGizmoMeshMaterial(w.gizmoMeshes, N); } }), this.dragBehavior.onEnabledObservable.add((m) => { this._setGizmoMeshMaterial(w.gizmoMeshes, m ? this._coloredMaterial : this._disableMaterial); }); } /** * Create Geometry for Gizmo * @param parentMesh * @param thickness * @param tessellation */ _createGizmoMesh(e, t, r) { const n = Ag("ignore", { diameter: 0.6, thickness: 0.03 * t, tessellation: r }, this.gizmoLayer.utilityLayerScene); n.visibility = 0; const i = Ag("", { diameter: 0.6, thickness: 5e-3 * t, tessellation: r }, this.gizmoLayer.utilityLayerScene); return i.material = this._coloredMaterial, i.rotation.x = Math.PI / 2, n.rotation.x = Math.PI / 2, e.addChild(i, Lo.PreserveScaling), e.addChild(n, Lo.PreserveScaling), { rotationMesh: i, collider: n }; } _attachedNodeChanged(e) { this.dragBehavior && (this.dragBehavior.enabled = !!e); } /** * If the gizmo is enabled */ set isEnabled(e) { this._isEnabled = e, e ? this._parent && (this.attachedMesh = this._parent.attachedMesh) : this.attachedMesh = null; } get isEnabled() { return this._isEnabled; } /** * Disposes of the gizmo */ dispose() { this.onSnapObservable.clear(), this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(this._pointerObserver), this.dragBehavior.detach(), this._gizmoMesh && this._gizmoMesh.dispose(), this._rotationDisplayPlane && this._rotationDisplayPlane.dispose(), this._rotationShaderMaterial && this._rotationShaderMaterial.dispose(), [this._coloredMaterial, this._hoverMaterial, this._disableMaterial].forEach((e) => { e && e.dispose(); }), super.dispose(); } } S0.MaxDragAngle = Math.PI * 9 / 20; S0._RotationGizmoVertexShader = ` precision highp float; attribute vec3 position; attribute vec2 uv; uniform mat4 worldViewProjection; varying vec3 vPosition; varying vec2 vUV; void main(void) { gl_Position = worldViewProjection * vec4(position, 1.0); vUV = uv; }`; S0._RotationGizmoFragmentShader = ` precision highp float; varying vec2 vUV; varying vec3 vPosition; uniform vec3 angles; uniform vec3 rotationColor; #define twopi 6.283185307 void main(void) { vec2 uv = vUV - vec2(0.5); float angle = atan(uv.y, uv.x) + 3.141592; float delta = gl_FrontFacing ? angles.y : -angles.y; float begin = angles.x - delta * angles.z; float start = (begin < (begin + delta)) ? begin : (begin + delta); float end = (begin > (begin + delta)) ? begin : (begin + delta); float len = sqrt(dot(uv,uv)); float opacity = 1. - step(0.5, len); float base = abs(floor(start / twopi)) * twopi; start += base; end += base; float intensity = 0.; for (int i = 0; i < 5; i++) { intensity += max(step(start, angle) - step(end, angle), 0.); angle += twopi; } gl_FragColor = vec4(rotationColor, min(intensity * 0.25, 0.8)) * opacity; } `; class GQ extends Lo { get attachedMesh() { return this._meshAttached; } set attachedMesh(e) { this._meshAttached = e, this._nodeAttached = e, this._checkBillboardTransform(), [this.xGizmo, this.yGizmo, this.zGizmo].forEach((t) => { t.isEnabled ? t.attachedMesh = e : t.attachedMesh = null; }); } get attachedNode() { return this._nodeAttached; } set attachedNode(e) { this._meshAttached = null, this._nodeAttached = e, this._checkBillboardTransform(), [this.xGizmo, this.yGizmo, this.zGizmo].forEach((t) => { t.isEnabled ? t.attachedNode = e : t.attachedNode = null; }); } _checkBillboardTransform() { this._nodeAttached && this._nodeAttached.billboardMode && console.log("Rotation Gizmo will not work with transforms in billboard mode."); } /** * Sensitivity factor for dragging (Default: 1) */ set sensitivity(e) { this._sensitivity = e, [this.xGizmo, this.yGizmo, this.zGizmo].forEach((t) => { t && (t.sensitivity = e); }); } get sensitivity() { return this._sensitivity; } /** * True when the mouse pointer is hovering a gizmo mesh */ get isHovered() { let e = !1; return [this.xGizmo, this.yGizmo, this.zGizmo].forEach((t) => { e = e || t.isHovered; }), e; } /** * Creates a RotationGizmo * @param gizmoLayer The utility layer the gizmo will be added to * @param tessellation Amount of tessellation to be used when creating rotation circles * @param useEulerRotation Use and update Euler angle instead of quaternion * @param thickness display gizmo axis thickness * @param gizmoManager Gizmo manager * @param options More options */ constructor(e = Ds.DefaultUtilityLayer, t = 32, r = !1, n = 1, i, s) { super(e), this.onDragStartObservable = new Oe(), this.onDragObservable = new Oe(), this.onDragEndObservable = new Oe(), this._observables = [], this._sensitivity = 1, this._gizmoAxisCache = /* @__PURE__ */ new Map(); const a = s && s.xOptions && s.xOptions.color ? s.xOptions.color : Ne.Red().scale(0.5), f = s && s.yOptions && s.yOptions.color ? s.yOptions.color : Ne.Green().scale(0.5), o = s && s.zOptions && s.zOptions.color ? s.zOptions.color : Ne.Blue().scale(0.5); this.xGizmo = new S0(new S(1, 0, 0), a, e, t, this, r, n), this.yGizmo = new S0(new S(0, 1, 0), f, e, t, this, r, n), this.zGizmo = new S0(new S(0, 0, 1), o, e, t, this, r, n), [this.xGizmo, this.yGizmo, this.zGizmo].forEach((d) => { s && s.updateScale != null && (d.updateScale = s.updateScale), d.dragBehavior.onDragStartObservable.add(() => { this.onDragStartObservable.notifyObservers({}); }), d.dragBehavior.onDragObservable.add(() => { this.onDragObservable.notifyObservers({}); }), d.dragBehavior.onDragEndObservable.add(() => { this.onDragEndObservable.notifyObservers({}); }); }), this.attachedMesh = null, this.attachedNode = null, i ? i.addToAxisCache(this._gizmoAxisCache) : Lo.GizmoAxisPointerObserver(e, this._gizmoAxisCache); } /** * If set the gizmo's rotation will be updated to match the attached mesh each frame (Default: true) * NOTE: This is only possible for meshes with uniform scaling, as otherwise it's not possible to decompose the rotation */ set updateGizmoRotationToMatchAttachedMesh(e) { this.xGizmo && (this.xGizmo.updateGizmoRotationToMatchAttachedMesh = e, this.yGizmo.updateGizmoRotationToMatchAttachedMesh = e, this.zGizmo.updateGizmoRotationToMatchAttachedMesh = e); } get updateGizmoRotationToMatchAttachedMesh() { return this.xGizmo.updateGizmoRotationToMatchAttachedMesh; } set updateGizmoPositionToMatchAttachedMesh(e) { this.xGizmo && (this.xGizmo.updateGizmoPositionToMatchAttachedMesh = e, this.yGizmo.updateGizmoPositionToMatchAttachedMesh = e, this.zGizmo.updateGizmoPositionToMatchAttachedMesh = e); } get updateGizmoPositionToMatchAttachedMesh() { return this.xGizmo.updateGizmoPositionToMatchAttachedMesh; } set anchorPoint(e) { this._anchorPoint = e, [this.xGizmo, this.yGizmo, this.zGizmo].forEach((t) => { t.anchorPoint = e; }); } get anchorPoint() { return this._anchorPoint; } /** * Set the coordinate system to use. By default it's local. * But it's possible for a user to tweak so its local for translation and world for rotation. * In that case, setting the coordinate system will change `updateGizmoRotationToMatchAttachedMesh` and `updateGizmoPositionToMatchAttachedMesh` */ set coordinatesMode(e) { [this.xGizmo, this.yGizmo, this.zGizmo].forEach((t) => { t.coordinatesMode = e; }); } set updateScale(e) { this.xGizmo && (this.xGizmo.updateScale = e, this.yGizmo.updateScale = e, this.zGizmo.updateScale = e); } get updateScale() { return this.xGizmo.updateScale; } /** * Drag distance in babylon units that the gizmo will snap to when dragged (Default: 0) */ set snapDistance(e) { this.xGizmo && (this.xGizmo.snapDistance = e, this.yGizmo.snapDistance = e, this.zGizmo.snapDistance = e); } get snapDistance() { return this.xGizmo.snapDistance; } /** * Ratio for the scale of the gizmo (Default: 1) */ set scaleRatio(e) { this.xGizmo && (this.xGizmo.scaleRatio = e, this.yGizmo.scaleRatio = e, this.zGizmo.scaleRatio = e); } get scaleRatio() { return this.xGizmo.scaleRatio; } /** * posture that the gizmo will be display * When set null, default value will be used (Quaternion(0, 0, 0, 1)) */ get customRotationQuaternion() { return this._customRotationQuaternion; } set customRotationQuaternion(e) { this._customRotationQuaternion = e, [this.xGizmo, this.yGizmo, this.zGizmo].forEach((t) => { t && (t.customRotationQuaternion = e); }); } /** * Builds Gizmo Axis Cache to enable features such as hover state preservation and graying out other axis during manipulation * @param mesh Axis gizmo mesh * @param cache Gizmo axis definition used for reactive gizmo UI */ addToAxisCache(e, t) { this._gizmoAxisCache.set(e, t); } /** * Disposes of the gizmo */ dispose() { this.xGizmo.dispose(), this.yGizmo.dispose(), this.zGizmo.dispose(), this.onDragStartObservable.clear(), this.onDragObservable.clear(), this.onDragEndObservable.clear(), this._observables.forEach((e) => { this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(e); }); } /** * CustomMeshes are not supported by this gizmo */ setCustomMesh() { Se.Error("Custom meshes are not supported on this gizmo, please set the custom meshes on the gizmos contained within this one (gizmo.xGizmo, gizmo.yGizmo, gizmo.zGizmo)"); } } class tS extends Lo { /** Default material used to render when gizmo is not disabled or hovered */ get coloredMaterial() { return this._coloredMaterial; } /** Material used to render when gizmo is hovered with mouse*/ get hoverMaterial() { return this._hoverMaterial; } /** Material used to render when gizmo is disabled. typically grey.*/ get disableMaterial() { return this._disableMaterial; } /** * @internal */ static _CreatePlane(e, t) { const r = new Hr("plane", e), n = u4("dragPlane", { width: 0.1375, height: 0.1375, sideOrientation: 2 }, e); return n.material = t, n.parent = r, r; } /** * Creates a PlaneDragGizmo * @param dragPlaneNormal The axis normal to which the gizmo will be able to drag on * @param color The color of the gizmo * @param gizmoLayer The utility layer the gizmo will be added to * @param parent * @param hoverColor The color of the gizmo when hovering over and dragging * @param disableColor The Color of the gizmo when its disabled */ constructor(e, t = Ne.Gray(), r = Ds.DefaultUtilityLayer, n = null, i = Ne.Yellow(), s = Ne.Gray()) { var a; super(r), this._pointerObserver = null, this.snapDistance = 0, this.onSnapObservable = new Oe(), this._isEnabled = !1, this._parent = null, this._dragging = !1, this._parent = n, this._coloredMaterial = new Wt("", r.utilityLayerScene), this._coloredMaterial.diffuseColor = t, this._coloredMaterial.specularColor = t.subtract(new Ne(0.1, 0.1, 0.1)), this._hoverMaterial = new Wt("", r.utilityLayerScene), this._hoverMaterial.diffuseColor = i, this._disableMaterial = new Wt("", r.utilityLayerScene), this._disableMaterial.diffuseColor = s, this._disableMaterial.alpha = 0.4, this._gizmoMesh = tS._CreatePlane(r.utilityLayerScene, this._coloredMaterial), this._gizmoMesh.lookAt(this._rootMesh.position.add(e)), this._gizmoMesh.scaling.scaleInPlace(1 / 3), this._gizmoMesh.parent = this._rootMesh; let f = 0; const o = new S(), d = { snapDistance: 0 }; this.dragBehavior = new O9({ dragPlaneNormal: e }), this.dragBehavior.moveAttached = !1, this._rootMesh.addBehavior(this.dragBehavior), this.dragBehavior.onDragObservable.add((l) => { if (this.attachedNode) { if (this.snapDistance == 0) this.attachedNode.getWorldMatrix().getTranslationToRef(ue.Vector3[0]), ue.Vector3[0].addToRef(l.delta, ue.Vector3[0]), this.dragBehavior.validateDrag(ue.Vector3[0]) && this.attachedNode.getWorldMatrix().addTranslationFromFloats(l.delta.x, l.delta.y, l.delta.z); else if (f += l.dragDistance, Math.abs(f) > this.snapDistance) { const P = Math.floor(Math.abs(f) / this.snapDistance); f = f % this.snapDistance, l.delta.normalizeToRef(o), o.scaleInPlace(this.snapDistance * P), this.attachedNode.getWorldMatrix().getTranslationToRef(ue.Vector3[0]), ue.Vector3[0].addToRef(o, ue.Vector3[0]), this.dragBehavior.validateDrag(ue.Vector3[0]) && (this.attachedNode.getWorldMatrix().addTranslationFromFloats(o.x, o.y, o.z), d.snapDistance = this.snapDistance * P, this.onSnapObservable.notifyObservers(d)); } this._matrixChanged(); } }), this.dragBehavior.onDragStartObservable.add(() => { this._dragging = !0; }), this.dragBehavior.onDragEndObservable.add(() => { this._dragging = !1; }); const v = r._getSharedGizmoLight(); v.includedOnlyMeshes = v.includedOnlyMeshes.concat(this._rootMesh.getChildMeshes(!1)); const u = { gizmoMeshes: this._gizmoMesh.getChildMeshes(), colliderMeshes: this._gizmoMesh.getChildMeshes(), material: this._coloredMaterial, hoverMaterial: this._hoverMaterial, disableMaterial: this._disableMaterial, active: !1, dragBehavior: this.dragBehavior }; (a = this._parent) === null || a === void 0 || a.addToAxisCache(this._gizmoMesh, u), this._pointerObserver = r.utilityLayerScene.onPointerObservable.add((l) => { var P; if (!this._customMeshSet && (this._isHovered = u.colliderMeshes.indexOf((P = l == null ? void 0 : l.pickInfo) === null || P === void 0 ? void 0 : P.pickedMesh) != -1, !this._parent)) { const p = u.dragBehavior.enabled ? this._isHovered || this._dragging ? this._hoverMaterial : this._coloredMaterial : this._disableMaterial; this._setGizmoMeshMaterial(u.gizmoMeshes, p); } }), this.dragBehavior.onEnabledObservable.add((l) => { this._setGizmoMeshMaterial(u.gizmoMeshes, l ? this._coloredMaterial : this._disableMaterial); }); } _attachedNodeChanged(e) { this.dragBehavior && (this.dragBehavior.enabled = !!e); } /** * If the gizmo is enabled */ set isEnabled(e) { this._isEnabled = e, e ? this._parent && (this.attachedNode = this._parent.attachedNode) : this.attachedNode = null; } get isEnabled() { return this._isEnabled; } /** * Disposes of the gizmo */ dispose() { this.onSnapObservable.clear(), this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(this._pointerObserver), this.dragBehavior.detach(), super.dispose(), this._gizmoMesh && this._gizmoMesh.dispose(), [this._coloredMaterial, this._hoverMaterial, this._disableMaterial].forEach((e) => { e && e.dispose(); }); } } class oy extends Lo { get attachedMesh() { return this._meshAttached; } set attachedMesh(e) { this._meshAttached = e, this._nodeAttached = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.xPlaneGizmo, this.yPlaneGizmo, this.zPlaneGizmo].forEach((t) => { t.isEnabled ? t.attachedMesh = e : t.attachedMesh = null; }); } get attachedNode() { return this._nodeAttached; } set attachedNode(e) { this._meshAttached = null, this._nodeAttached = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.xPlaneGizmo, this.yPlaneGizmo, this.zPlaneGizmo].forEach((t) => { t.isEnabled ? t.attachedNode = e : t.attachedNode = null; }); } /** * True when the mouse pointer is hovering a gizmo mesh */ get isHovered() { let e = !1; return [this.xGizmo, this.yGizmo, this.zGizmo, this.xPlaneGizmo, this.yPlaneGizmo, this.zPlaneGizmo].forEach((t) => { e = e || t.isHovered; }), e; } /** * Creates a PositionGizmo * @param gizmoLayer The utility layer the gizmo will be added to @param thickness display gizmo axis thickness * @param gizmoManager */ constructor(e = Ds.DefaultUtilityLayer, t = 1, r) { super(e), this._meshAttached = null, this._nodeAttached = null, this._observables = [], this._gizmoAxisCache = /* @__PURE__ */ new Map(), this.onDragStartObservable = new Oe(), this.onDragObservable = new Oe(), this.onDragEndObservable = new Oe(), this._planarGizmoEnabled = !1, this.xGizmo = new lp(new S(1, 0, 0), Ne.Red().scale(0.5), e, this, t), this.yGizmo = new lp(new S(0, 1, 0), Ne.Green().scale(0.5), e, this, t), this.zGizmo = new lp(new S(0, 0, 1), Ne.Blue().scale(0.5), e, this, t), this.xPlaneGizmo = new tS(new S(1, 0, 0), Ne.Red().scale(0.5), this.gizmoLayer, this), this.yPlaneGizmo = new tS(new S(0, 1, 0), Ne.Green().scale(0.5), this.gizmoLayer, this), this.zPlaneGizmo = new tS(new S(0, 0, 1), Ne.Blue().scale(0.5), this.gizmoLayer, this), [this.xGizmo, this.yGizmo, this.zGizmo, this.xPlaneGizmo, this.yPlaneGizmo, this.zPlaneGizmo].forEach((n) => { n.dragBehavior.onDragStartObservable.add(() => { this.onDragStartObservable.notifyObservers({}); }), n.dragBehavior.onDragObservable.add(() => { this.onDragObservable.notifyObservers({}); }), n.dragBehavior.onDragEndObservable.add(() => { this.onDragEndObservable.notifyObservers({}); }); }), this.attachedMesh = null, r ? r.addToAxisCache(this._gizmoAxisCache) : Lo.GizmoAxisPointerObserver(e, this._gizmoAxisCache); } /** * If the planar drag gizmo is enabled * setting this will enable/disable XY, XZ and YZ planes regardless of individual gizmo settings. */ set planarGizmoEnabled(e) { this._planarGizmoEnabled = e, [this.xPlaneGizmo, this.yPlaneGizmo, this.zPlaneGizmo].forEach((t) => { t && (t.isEnabled = e, e && (t.attachedMesh ? t.attachedMesh = this.attachedMesh : t.attachedNode = this.attachedNode)); }, this); } get planarGizmoEnabled() { return this._planarGizmoEnabled; } /** * posture that the gizmo will be display * When set null, default value will be used (Quaternion(0, 0, 0, 1)) */ get customRotationQuaternion() { return this._customRotationQuaternion; } set customRotationQuaternion(e) { this._customRotationQuaternion = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.xPlaneGizmo, this.yPlaneGizmo, this.zPlaneGizmo].forEach((t) => { t && (t.customRotationQuaternion = e); }); } /** * If set the gizmo's rotation will be updated to match the attached mesh each frame (Default: true) * NOTE: This is only possible for meshes with uniform scaling, as otherwise it's not possible to decompose the rotation */ set updateGizmoRotationToMatchAttachedMesh(e) { this._updateGizmoRotationToMatchAttachedMesh = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.xPlaneGizmo, this.yPlaneGizmo, this.zPlaneGizmo].forEach((t) => { t && (t.updateGizmoRotationToMatchAttachedMesh = e); }); } get updateGizmoRotationToMatchAttachedMesh() { return this._updateGizmoRotationToMatchAttachedMesh; } set updateGizmoPositionToMatchAttachedMesh(e) { this._updateGizmoPositionToMatchAttachedMesh = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.xPlaneGizmo, this.yPlaneGizmo, this.zPlaneGizmo].forEach((t) => { t && (t.updateGizmoPositionToMatchAttachedMesh = e); }); } get updateGizmoPositionToMatchAttachedMesh() { return this._updateGizmoPositionToMatchAttachedMesh; } set anchorPoint(e) { this._anchorPoint = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.xPlaneGizmo, this.yPlaneGizmo, this.zPlaneGizmo].forEach((t) => { t.anchorPoint = e; }); } get anchorPoint() { return this._anchorPoint; } /** * Set the coordinate system to use. By default it's local. * But it's possible for a user to tweak so its local for translation and world for rotation. * In that case, setting the coordinate system will change `updateGizmoRotationToMatchAttachedMesh` and `updateGizmoPositionToMatchAttachedMesh` */ set coordinatesMode(e) { [this.xGizmo, this.yGizmo, this.zGizmo, this.xPlaneGizmo, this.yPlaneGizmo, this.zPlaneGizmo].forEach((t) => { t.coordinatesMode = e; }); } set updateScale(e) { this.xGizmo && (this.xGizmo.updateScale = e, this.yGizmo.updateScale = e, this.zGizmo.updateScale = e); } get updateScale() { return this.xGizmo.updateScale; } /** * Drag distance in babylon units that the gizmo will snap to when dragged (Default: 0) */ set snapDistance(e) { this._snapDistance = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.xPlaneGizmo, this.yPlaneGizmo, this.zPlaneGizmo].forEach((t) => { t && (t.snapDistance = e); }); } get snapDistance() { return this._snapDistance; } /** * Ratio for the scale of the gizmo (Default: 1) */ set scaleRatio(e) { this._scaleRatio = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.xPlaneGizmo, this.yPlaneGizmo, this.zPlaneGizmo].forEach((t) => { t && (t.scaleRatio = e); }); } get scaleRatio() { return this._scaleRatio; } /** * Builds Gizmo Axis Cache to enable features such as hover state preservation and graying out other axis during manipulation * @param mesh Axis gizmo mesh * @param cache Gizmo axis definition used for reactive gizmo UI */ addToAxisCache(e, t) { this._gizmoAxisCache.set(e, t); } /** * Disposes of the gizmo */ dispose() { [this.xGizmo, this.yGizmo, this.zGizmo, this.xPlaneGizmo, this.yPlaneGizmo, this.zPlaneGizmo].forEach((e) => { e && e.dispose(); }), this._observables.forEach((e) => { this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(e); }), this.onDragStartObservable.clear(), this.onDragObservable.clear(), this.onDragEndObservable.clear(); } /** * CustomMeshes are not supported by this gizmo */ setCustomMesh() { Se.Error("Custom meshes are not supported on this gizmo, please set the custom meshes on the gizmos contained within this one (gizmo.xGizmo, gizmo.yGizmo, gizmo.zGizmo,gizmo.xPlaneGizmo, gizmo.yPlaneGizmo, gizmo.zPlaneGizmo)"); } } class ZQ extends Lo { /** Default material used to render when gizmo is not disabled or hovered */ get coloredMaterial() { return this._coloredMaterial; } /** Material used to render when gizmo is hovered with mouse*/ get hoverMaterial() { return this._hoverMaterial; } /** Material used to render when gizmo is disabled. typically grey.*/ get disableMaterial() { return this._disableMaterial; } get attachedMesh() { return this._meshAttached; } set attachedMesh(e) { this._meshAttached = e, this._nodeAttached = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.uniformScaleGizmo].forEach((t) => { t.isEnabled ? t.attachedMesh = e : t.attachedMesh = null; }); } get attachedNode() { return this._nodeAttached; } set attachedNode(e) { this._meshAttached = null, this._nodeAttached = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.uniformScaleGizmo].forEach((t) => { t.isEnabled ? t.attachedNode = e : t.attachedNode = null; }); } set updateScale(e) { this.xGizmo && (this.xGizmo.updateScale = e, this.yGizmo.updateScale = e, this.zGizmo.updateScale = e); } get updateScale() { return this.xGizmo.updateScale; } /** * True when the mouse pointer is hovering a gizmo mesh */ get isHovered() { let e = !1; return [this.xGizmo, this.yGizmo, this.zGizmo].forEach((t) => { e = e || t.isHovered; }), e; } /** * Creates a ScaleGizmo * @param gizmoLayer The utility layer the gizmo will be added to * @param thickness display gizmo axis thickness * @param gizmoManager */ constructor(e = Ds.DefaultUtilityLayer, t = 1, r) { super(e), this._meshAttached = null, this._nodeAttached = null, this._incrementalSnap = !1, this._sensitivity = 1, this._observables = [], this._gizmoAxisCache = /* @__PURE__ */ new Map(), this.onDragStartObservable = new Oe(), this.onDragObservable = new Oe(), this.onDragEndObservable = new Oe(), this.uniformScaleGizmo = this._createUniformScaleMesh(), this.xGizmo = new Pp(new S(1, 0, 0), Ne.Red().scale(0.5), e, this, t), this.yGizmo = new Pp(new S(0, 1, 0), Ne.Green().scale(0.5), e, this, t), this.zGizmo = new Pp(new S(0, 0, 1), Ne.Blue().scale(0.5), e, this, t), [this.xGizmo, this.yGizmo, this.zGizmo, this.uniformScaleGizmo].forEach((n) => { n.dragBehavior.onDragStartObservable.add(() => { this.onDragStartObservable.notifyObservers({}); }), n.dragBehavior.onDragObservable.add(() => { this.onDragObservable.notifyObservers({}); }), n.dragBehavior.onDragEndObservable.add(() => { this.onDragEndObservable.notifyObservers({}); }); }), this.attachedMesh = null, this.attachedNode = null, r ? r.addToAxisCache(this._gizmoAxisCache) : Lo.GizmoAxisPointerObserver(e, this._gizmoAxisCache); } /** Create Geometry for Gizmo */ _createUniformScaleMesh() { this._coloredMaterial = new Wt("", this.gizmoLayer.utilityLayerScene), this._coloredMaterial.diffuseColor = Ne.Gray(), this._hoverMaterial = new Wt("", this.gizmoLayer.utilityLayerScene), this._hoverMaterial.diffuseColor = Ne.Yellow(), this._disableMaterial = new Wt("", this.gizmoLayer.utilityLayerScene), this._disableMaterial.diffuseColor = Ne.Gray(), this._disableMaterial.alpha = 0.4; const e = new Pp(new S(0, 1, 0), Ne.Gray().scale(0.5), this.gizmoLayer, this); e.updateGizmoRotationToMatchAttachedMesh = !1, e.uniformScaling = !0, this._uniformScalingMesh = hm("uniform", { type: 1 }, e.gizmoLayer.utilityLayerScene), this._uniformScalingMesh.scaling.scaleInPlace(0.01), this._uniformScalingMesh.visibility = 0, this._octahedron = hm("", { type: 1 }, e.gizmoLayer.utilityLayerScene), this._octahedron.scaling.scaleInPlace(7e-3), this._uniformScalingMesh.addChild(this._octahedron), e.setCustomMesh(this._uniformScalingMesh, !0); const t = this.gizmoLayer._getSharedGizmoLight(); t.includedOnlyMeshes = t.includedOnlyMeshes.concat(this._octahedron); const r = { gizmoMeshes: [this._octahedron, this._uniformScalingMesh], colliderMeshes: [this._uniformScalingMesh], material: this._coloredMaterial, hoverMaterial: this._hoverMaterial, disableMaterial: this._disableMaterial, active: !1, dragBehavior: e.dragBehavior }; return this.addToAxisCache(e._rootMesh, r), e; } set updateGizmoRotationToMatchAttachedMesh(e) { e ? (this._updateGizmoRotationToMatchAttachedMesh = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.uniformScaleGizmo].forEach((t) => { t && (t.updateGizmoRotationToMatchAttachedMesh = e); })) : Se.Warn("Setting updateGizmoRotationToMatchAttachedMesh = false on scaling gizmo is not supported."); } get updateGizmoRotationToMatchAttachedMesh() { return this._updateGizmoRotationToMatchAttachedMesh; } set anchorPoint(e) { this._anchorPoint = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.uniformScaleGizmo].forEach((t) => { t && (t.anchorPoint = e); }); } get anchorPoint() { return this._anchorPoint; } /** * posture that the gizmo will be display * When set null, default value will be used (Quaternion(0, 0, 0, 1)) */ get customRotationQuaternion() { return this._customRotationQuaternion; } set customRotationQuaternion(e) { this._customRotationQuaternion = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.uniformScaleGizmo].forEach((t) => { t && (t.customRotationQuaternion = e); }); } /** * Set the coordinate system to use. By default it's local. * But it's possible for a user to tweak so its local for translation and world for rotation. * In that case, setting the coordinate system will change `updateGizmoRotationToMatchAttachedMesh` and `updateGizmoPositionToMatchAttachedMesh` */ set coordinatesMode(e) { e == pD.World && Se.Warn("Setting coordinates Mode to world on scaling gizmo is not supported."), [this.xGizmo, this.yGizmo, this.zGizmo, this.uniformScaleGizmo].forEach((t) => { t.coordinatesMode = pD.Local; }); } /** * Drag distance in babylon units that the gizmo will snap to when dragged (Default: 0) */ set snapDistance(e) { this._snapDistance = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.uniformScaleGizmo].forEach((t) => { t && (t.snapDistance = e); }); } get snapDistance() { return this._snapDistance; } /** * Incremental snap scaling (default is false). When true, with a snapDistance of 0.1, scaling will be 1.1,1.2,1.3 instead of, when false: 1.1,1.21,1.33,... */ set incrementalSnap(e) { this._incrementalSnap = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.uniformScaleGizmo].forEach((t) => { t && (t.incrementalSnap = e); }); } get incrementalSnap() { return this._incrementalSnap; } /** * Ratio for the scale of the gizmo (Default: 1) */ set scaleRatio(e) { this._scaleRatio = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.uniformScaleGizmo].forEach((t) => { t && (t.scaleRatio = e); }); } get scaleRatio() { return this._scaleRatio; } /** * Sensitivity factor for dragging (Default: 1) */ set sensitivity(e) { this._sensitivity = e, [this.xGizmo, this.yGizmo, this.zGizmo, this.uniformScaleGizmo].forEach((t) => { t && (t.sensitivity = e); }); } get sensitivity() { return this._sensitivity; } /** * Builds Gizmo Axis Cache to enable features such as hover state preservation and graying out other axis during manipulation * @param mesh Axis gizmo mesh * @param cache Gizmo axis definition used for reactive gizmo UI */ addToAxisCache(e, t) { this._gizmoAxisCache.set(e, t); } /** * Disposes of the gizmo */ dispose() { [this.xGizmo, this.yGizmo, this.zGizmo, this.uniformScaleGizmo].forEach((e) => { e && e.dispose(); }), this._observables.forEach((e) => { this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(e); }), this.onDragStartObservable.clear(), this.onDragObservable.clear(), this.onDragEndObservable.clear(), [this._uniformScalingMesh, this._octahedron].forEach((e) => { e && e.dispose(); }), [this._coloredMaterial, this._hoverMaterial, this._disableMaterial].forEach((e) => { e && e.dispose(); }); } } class Jle { /** * Utility layer that the bounding box gizmo belongs to */ get keepDepthUtilityLayer() { return this._defaultKeepDepthUtilityLayer; } /** * Utility layer that all gizmos besides bounding box belong to */ get utilityLayer() { return this._defaultUtilityLayer; } /** * True when the mouse pointer is hovering a gizmo mesh */ get isHovered() { let e = !1; for (const t in this.gizmos) { const r = this.gizmos[t]; if (r && r.isHovered) { e = !0; break; } } return e; } /** * Ratio for the scale of the gizmo (Default: 1) */ set scaleRatio(e) { this._scaleRatio = e, [this.gizmos.positionGizmo, this.gizmos.rotationGizmo, this.gizmos.scaleGizmo].forEach((t) => { t && (t.scaleRatio = e); }); } get scaleRatio() { return this._scaleRatio; } /** * Set the coordinate system to use. By default it's local. * But it's possible for a user to tweak so its local for translation and world for rotation. * In that case, setting the coordinate system will change `updateGizmoRotationToMatchAttachedMesh` and `updateGizmoPositionToMatchAttachedMesh` */ set coordinatesMode(e) { this._coordinatesMode = e, [this.gizmos.positionGizmo, this.gizmos.rotationGizmo, this.gizmos.scaleGizmo].forEach((t) => { t && (t.coordinatesMode = e); }); } get coordinatesMode() { return this._coordinatesMode; } /** * Instantiates a gizmo manager * @param _scene the scene to overlay the gizmos on top of * @param thickness display gizmo axis thickness * @param utilityLayer the layer where gizmos are rendered * @param keepDepthUtilityLayer the layer where occluded gizmos are rendered */ constructor(e, t = 1, r = Ds.DefaultUtilityLayer, n = Ds.DefaultKeepDepthUtilityLayer) { this._scene = e, this.clearGizmoOnEmptyPointerEvent = !1, this.enableAutoPicking = !0, this.onAttachedToMeshObservable = new Oe(), this.onAttachedToNodeObservable = new Oe(), this._gizmosEnabled = { positionGizmo: !1, rotationGizmo: !1, scaleGizmo: !1, boundingBoxGizmo: !1 }, this._pointerObservers = [], this._attachedMesh = null, this._attachedNode = null, this._boundingBoxColor = Ne.FromHexString("#0984e3"), this._thickness = 1, this._scaleRatio = 1, this._coordinatesMode = pD.Local, this._gizmoAxisCache = /* @__PURE__ */ new Map(), this.boundingBoxDragBehavior = new Hee(), this.attachableMeshes = null, this.attachableNodes = null, this.usePointerToAttachGizmos = !0, this._defaultUtilityLayer = r, this._defaultKeepDepthUtilityLayer = n, this._defaultKeepDepthUtilityLayer.utilityLayerScene.autoClearDepthAndStencil = !1, this._thickness = t, this.gizmos = { positionGizmo: null, rotationGizmo: null, scaleGizmo: null, boundingBoxGizmo: null }; const i = this._attachToMeshPointerObserver(e), s = Lo.GizmoAxisPointerObserver(this._defaultUtilityLayer, this._gizmoAxisCache); this._pointerObservers = [i, s]; } /** * Subscribes to pointer down events, for attaching and detaching mesh * @param scene The scene layer the observer will be added to */ _attachToMeshPointerObserver(e) { return e.onPointerObservable.add((r) => { if (this.usePointerToAttachGizmos && r.type == ir.POINTERDOWN) if (r.pickInfo && r.pickInfo.pickedMesh) { if (this.enableAutoPicking) { let n = r.pickInfo.pickedMesh; if (this.attachableMeshes == null) for (; n && n.parent != null; ) n = n.parent; else { let i = !1; this.attachableMeshes.forEach((s) => { n && (n == s || n.isDescendantOf(s)) && (n = s, i = !0); }), i || (n = null); } n instanceof jn ? this._attachedMesh != n && this.attachToMesh(n) : this.clearGizmoOnEmptyPointerEvent && this.attachToMesh(null); } } else this.clearGizmoOnEmptyPointerEvent && this.attachToMesh(null); }); } /** * Attaches a set of gizmos to the specified mesh * @param mesh The mesh the gizmo's should be attached to */ attachToMesh(e) { this._attachedMesh && this._attachedMesh.removeBehavior(this.boundingBoxDragBehavior), this._attachedNode && this._attachedNode.removeBehavior(this.boundingBoxDragBehavior), this._attachedMesh = e, this._attachedNode = null; for (const t in this.gizmos) { const r = this.gizmos[t]; r && this._gizmosEnabled[t] && (r.attachedMesh = e); } this.boundingBoxGizmoEnabled && this._attachedMesh && this._attachedMesh.addBehavior(this.boundingBoxDragBehavior), this.onAttachedToMeshObservable.notifyObservers(e); } /** * Attaches a set of gizmos to the specified node * @param node The node the gizmo's should be attached to */ attachToNode(e) { this._attachedMesh && this._attachedMesh.removeBehavior(this.boundingBoxDragBehavior), this._attachedNode && this._attachedNode.removeBehavior(this.boundingBoxDragBehavior), this._attachedMesh = null, this._attachedNode = e; for (const t in this.gizmos) { const r = this.gizmos[t]; r && this._gizmosEnabled[t] && (r.attachedNode = e); } this.boundingBoxGizmoEnabled && this._attachedNode && this._attachedNode.addBehavior(this.boundingBoxDragBehavior), this.onAttachedToNodeObservable.notifyObservers(e); } /** * If the position gizmo is enabled */ set positionGizmoEnabled(e) { e ? (this.gizmos.positionGizmo || (this.gizmos.positionGizmo = new oy(this._defaultUtilityLayer, this._thickness, this)), this._attachedNode ? this.gizmos.positionGizmo.attachedNode = this._attachedNode : this.gizmos.positionGizmo.attachedMesh = this._attachedMesh) : this.gizmos.positionGizmo && (this.gizmos.positionGizmo.attachedNode = null), this._gizmosEnabled.positionGizmo = e; } get positionGizmoEnabled() { return this._gizmosEnabled.positionGizmo; } /** * If the rotation gizmo is enabled */ set rotationGizmoEnabled(e) { e ? (this.gizmos.rotationGizmo || (this.gizmos.rotationGizmo = new GQ(this._defaultUtilityLayer, 32, !1, this._thickness, this)), this._attachedNode ? this.gizmos.rotationGizmo.attachedNode = this._attachedNode : this.gizmos.rotationGizmo.attachedMesh = this._attachedMesh) : this.gizmos.rotationGizmo && (this.gizmos.rotationGizmo.attachedNode = null), this._gizmosEnabled.rotationGizmo = e; } get rotationGizmoEnabled() { return this._gizmosEnabled.rotationGizmo; } /** * If the scale gizmo is enabled */ set scaleGizmoEnabled(e) { e ? (this.gizmos.scaleGizmo = this.gizmos.scaleGizmo || new ZQ(this._defaultUtilityLayer, this._thickness, this), this._attachedNode ? this.gizmos.scaleGizmo.attachedNode = this._attachedNode : this.gizmos.scaleGizmo.attachedMesh = this._attachedMesh) : this.gizmos.scaleGizmo && (this.gizmos.scaleGizmo.attachedNode = null), this._gizmosEnabled.scaleGizmo = e; } get scaleGizmoEnabled() { return this._gizmosEnabled.scaleGizmo; } /** * If the boundingBox gizmo is enabled */ set boundingBoxGizmoEnabled(e) { e ? (this.gizmos.boundingBoxGizmo = this.gizmos.boundingBoxGizmo || new zQ(this._boundingBoxColor, this._defaultKeepDepthUtilityLayer), this._attachedMesh ? this.gizmos.boundingBoxGizmo.attachedMesh = this._attachedMesh : this.gizmos.boundingBoxGizmo.attachedNode = this._attachedNode, this._attachedMesh ? (this._attachedMesh.removeBehavior(this.boundingBoxDragBehavior), this._attachedMesh.addBehavior(this.boundingBoxDragBehavior)) : this._attachedNode && (this._attachedNode.removeBehavior(this.boundingBoxDragBehavior), this._attachedNode.addBehavior(this.boundingBoxDragBehavior))) : this.gizmos.boundingBoxGizmo && (this._attachedMesh ? this._attachedMesh.removeBehavior(this.boundingBoxDragBehavior) : this._attachedNode && this._attachedNode.removeBehavior(this.boundingBoxDragBehavior), this.gizmos.boundingBoxGizmo.attachedNode = null), this._gizmosEnabled.boundingBoxGizmo = e; } get boundingBoxGizmoEnabled() { return this._gizmosEnabled.boundingBoxGizmo; } /** * Builds Gizmo Axis Cache to enable features such as hover state preservation and graying out other axis during manipulation * @param gizmoAxisCache Gizmo axis definition used for reactive gizmo UI */ addToAxisCache(e) { e.size > 0 && e.forEach((t, r) => { this._gizmoAxisCache.set(r, t); }); } /** * Disposes of the gizmo manager */ dispose() { var e, t; this._pointerObservers.forEach((r) => { this._scene.onPointerObservable.remove(r); }); for (const r in this.gizmos) { const n = this.gizmos[r]; n && n.dispose(); } this._defaultKeepDepthUtilityLayer !== Ds._DefaultKeepDepthUtilityLayer && ((e = this._defaultKeepDepthUtilityLayer) === null || e === void 0 || e.dispose()), this._defaultUtilityLayer !== Ds._DefaultUtilityLayer && ((t = this._defaultUtilityLayer) === null || t === void 0 || t.dispose()), this.boundingBoxDragBehavior.detach(), this.onAttachedToMeshObservable.clear(); } } class xD extends ci { constructor() { super(...arguments), this._needProjectionMatrixCompute = !0; } _setPosition(e) { this._position = e; } /** * Sets the position the shadow will be casted from. Also use as the light position for both * point and spot lights. */ get position() { return this._position; } /** * Sets the position the shadow will be casted from. Also use as the light position for both * point and spot lights. */ set position(e) { this._setPosition(e); } _setDirection(e) { this._direction = e; } /** * In 2d mode (needCube being false), gets the direction used to cast the shadow. * Also use as the light direction on spot and directional lights. */ get direction() { return this._direction; } /** * In 2d mode (needCube being false), sets the direction used to cast the shadow. * Also use as the light direction on spot and directional lights. */ set direction(e) { this._setDirection(e); } /** * Gets the shadow projection clipping minimum z value. */ get shadowMinZ() { return this._shadowMinZ; } /** * Sets the shadow projection clipping minimum z value. */ set shadowMinZ(e) { this._shadowMinZ = e, this.forceProjectionMatrixCompute(); } /** * Sets the shadow projection clipping maximum z value. */ get shadowMaxZ() { return this._shadowMaxZ; } /** * Gets the shadow projection clipping maximum z value. */ set shadowMaxZ(e) { this._shadowMaxZ = e, this.forceProjectionMatrixCompute(); } /** * Computes the transformed information (transformedPosition and transformedDirection in World space) of the current light * @returns true if the information has been computed, false if it does not need to (no parenting) */ computeTransformedInformation() { return this.parent && this.parent.getWorldMatrix ? (this.transformedPosition || (this.transformedPosition = S.Zero()), S.TransformCoordinatesToRef(this.position, this.parent.getWorldMatrix(), this.transformedPosition), this.direction && (this.transformedDirection || (this.transformedDirection = S.Zero()), S.TransformNormalToRef(this.direction, this.parent.getWorldMatrix(), this.transformedDirection)), !0) : !1; } /** * Return the depth scale used for the shadow map. * @returns the depth scale. */ getDepthScale() { return 50; } /** * Get the direction to use to render the shadow map. In case of cube texture, the face index can be passed. * @param faceIndex The index of the face we are computed the direction to generate shadow * @returns The set direction in 2d mode otherwise the direction to the cubemap face if needCube() is true */ // eslint-disable-next-line @typescript-eslint/no-unused-vars getShadowDirection(e) { return this.transformedDirection ? this.transformedDirection : this.direction; } /** * Returns the ShadowLight absolute position in the World. * @returns the position vector in world space */ getAbsolutePosition() { return this.transformedPosition ? this.transformedPosition : this.position; } /** * Sets the ShadowLight direction toward the passed target. * @param target The point to target in local space * @returns the updated ShadowLight direction */ setDirectionToTarget(e) { return this.direction = S.Normalize(e.subtract(this.position)), this.direction; } /** * Returns the light rotation in euler definition. * @returns the x y z rotation in local space. */ getRotation() { this.direction.normalize(); const e = S.Cross(this.direction, bf.Y), t = S.Cross(e, this.direction); return S.RotationFromAxis(e, t, this.direction); } /** * Returns whether or not the shadow generation require a cube texture or a 2d texture. * @returns true if a cube texture needs to be use */ needCube() { return !1; } /** * Detects if the projection matrix requires to be recomputed this frame. * @returns true if it requires to be recomputed otherwise, false. */ needProjectionMatrixCompute() { return this._needProjectionMatrixCompute; } /** * Forces the shadow generator to recompute the projection matrix even if position and direction did not changed. */ forceProjectionMatrixCompute() { this._needProjectionMatrixCompute = !0; } /** @internal */ _initCache() { super._initCache(), this._cache.position = S.Zero(); } /** @internal */ _isSynchronized() { return !!this._cache.position.equals(this.position); } /** * Computes the world matrix of the node * @param force defines if the cache version should be invalidated forcing the world matrix to be created from scratch * @returns the world matrix */ computeWorldMatrix(e) { return !e && this.isSynchronized() ? (this._currentRenderId = this.getScene().getRenderId(), this._worldMatrix) : (this._updateCache(), this._cache.position.copyFrom(this.position), this._worldMatrix || (this._worldMatrix = he.Identity()), he.TranslationToRef(this.position.x, this.position.y, this.position.z, this._worldMatrix), this.parent && this.parent.getWorldMatrix && (this._worldMatrix.multiplyToRef(this.parent.getWorldMatrix(), this._worldMatrix), this._markSyncedWithParent()), this._worldMatrixDeterminantIsDirty = !0, this._worldMatrix); } /** * Gets the minZ used for shadow according to both the scene and the light. * @param activeCamera The camera we are returning the min for * @returns the depth min z */ getDepthMinZ(e) { return this.shadowMinZ !== void 0 ? this.shadowMinZ : e.minZ; } /** * Gets the maxZ used for shadow according to both the scene and the light. * @param activeCamera The camera we are returning the max for * @returns the depth max z */ getDepthMaxZ(e) { return this.shadowMaxZ !== void 0 ? this.shadowMaxZ : e.maxZ; } /** * Sets the shadow projection matrix in parameter to the generated projection matrix. * @param matrix The matrix to updated with the projection information * @param viewMatrix The transform matrix of the light * @param renderList The list of mesh to render in the map * @returns The current light */ setShadowProjectionMatrix(e, t, r) { return this.customProjectionMatrixBuilder ? this.customProjectionMatrixBuilder(t, r, e) : this._setDefaultShadowProjectionMatrix(e, t, r), this; } /** @internal */ _syncParentEnabledState() { super._syncParentEnabledState(), (!this.parent || !this.parent.getWorldMatrix) && (this.transformedPosition = null, this.transformedDirection = null); } } C([ fo() ], xD.prototype, "position", null); C([ fo() ], xD.prototype, "direction", null); C([ M() ], xD.prototype, "shadowMinZ", null); C([ M() ], xD.prototype, "shadowMaxZ", null); Cs.AddNodeConstructor("Light_Type_1", (A, e) => () => new IA(A, S.Zero(), e)); class IA extends xD { /** * Fix frustum size for the shadow generation. This is disabled if the value is 0. */ get shadowFrustumSize() { return this._shadowFrustumSize; } /** * Specifies a fix frustum size for the shadow generation. */ set shadowFrustumSize(e) { this._shadowFrustumSize = e, this.forceProjectionMatrixCompute(); } /** * Gets the shadow projection scale against the optimal computed one. * 0.1 by default which means that the projection window is increase by 10% from the optimal size. * This does not impact in fixed frustum size (shadowFrustumSize being set) */ get shadowOrthoScale() { return this._shadowOrthoScale; } /** * Sets the shadow projection scale against the optimal computed one. * 0.1 by default which means that the projection window is increase by 10% from the optimal size. * This does not impact in fixed frustum size (shadowFrustumSize being set) */ set shadowOrthoScale(e) { this._shadowOrthoScale = e, this.forceProjectionMatrixCompute(); } /** * Gets or sets the orthoLeft property used to build the light frustum */ get orthoLeft() { return this._orthoLeft; } set orthoLeft(e) { this._orthoLeft = e; } /** * Gets or sets the orthoRight property used to build the light frustum */ get orthoRight() { return this._orthoRight; } set orthoRight(e) { this._orthoRight = e; } /** * Gets or sets the orthoTop property used to build the light frustum */ get orthoTop() { return this._orthoTop; } set orthoTop(e) { this._orthoTop = e; } /** * Gets or sets the orthoBottom property used to build the light frustum */ get orthoBottom() { return this._orthoBottom; } set orthoBottom(e) { this._orthoBottom = e; } /** * Creates a DirectionalLight object in the scene, oriented towards the passed direction (Vector3). * The directional light is emitted from everywhere in the given direction. * It can cast shadows. * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction * @param name The friendly name of the light * @param direction The direction of the light * @param scene The scene the light belongs to */ constructor(e, t, r) { super(e, r), this._shadowFrustumSize = 0, this._shadowOrthoScale = 0.1, this.autoUpdateExtends = !0, this.autoCalcShadowZBounds = !1, this._orthoLeft = Number.MAX_VALUE, this._orthoRight = Number.MIN_VALUE, this._orthoTop = Number.MIN_VALUE, this._orthoBottom = Number.MAX_VALUE, this.position = t.scale(-1), this.direction = t; } /** * Returns the string "DirectionalLight". * @returns The class name */ getClassName() { return "DirectionalLight"; } /** * Returns the integer 1. * @returns The light Type id as a constant defines in Light.LIGHTTYPEID_x */ getTypeID() { return ci.LIGHTTYPEID_DIRECTIONALLIGHT; } /** * Sets the passed matrix "matrix" as projection matrix for the shadows cast by the light according to the passed view matrix. * Returns the DirectionalLight Shadow projection matrix. * @param matrix * @param viewMatrix * @param renderList */ _setDefaultShadowProjectionMatrix(e, t, r) { this.shadowFrustumSize > 0 ? this._setDefaultFixedFrustumShadowProjectionMatrix(e) : this._setDefaultAutoExtendShadowProjectionMatrix(e, t, r); } /** * Sets the passed matrix "matrix" as fixed frustum projection matrix for the shadows cast by the light according to the passed view matrix. * Returns the DirectionalLight Shadow projection matrix. * @param matrix */ _setDefaultFixedFrustumShadowProjectionMatrix(e) { const t = this.getScene().activeCamera; t && he.OrthoLHToRef(this.shadowFrustumSize, this.shadowFrustumSize, this.shadowMinZ !== void 0 ? this.shadowMinZ : t.minZ, this.shadowMaxZ !== void 0 ? this.shadowMaxZ : t.maxZ, e, this.getScene().getEngine().isNDCHalfZRange); } /** * Sets the passed matrix "matrix" as auto extend projection matrix for the shadows cast by the light according to the passed view matrix. * Returns the DirectionalLight Shadow projection matrix. * @param matrix * @param viewMatrix * @param renderList */ _setDefaultAutoExtendShadowProjectionMatrix(e, t, r) { const n = this.getScene().activeCamera; if (!n) return; if (this.autoUpdateExtends || this._orthoLeft === Number.MAX_VALUE) { const d = S.Zero(); this._orthoLeft = Number.MAX_VALUE, this._orthoRight = -Number.MAX_VALUE, this._orthoTop = -Number.MAX_VALUE, this._orthoBottom = Number.MAX_VALUE; let v = Number.MAX_VALUE, u = -Number.MAX_VALUE; for (let l = 0; l < r.length; l++) { const P = r[l]; if (!P) continue; const c = P.getBoundingInfo().boundingBox; for (let H = 0; H < c.vectorsWorld.length; H++) S.TransformCoordinatesToRef(c.vectorsWorld[H], t, d), d.x < this._orthoLeft && (this._orthoLeft = d.x), d.y < this._orthoBottom && (this._orthoBottom = d.y), d.x > this._orthoRight && (this._orthoRight = d.x), d.y > this._orthoTop && (this._orthoTop = d.y), this.autoCalcShadowZBounds && (d.z < v && (v = d.z), d.z > u && (u = d.z)); } this.autoCalcShadowZBounds && (this._shadowMinZ = v, this._shadowMaxZ = u); } const i = this._orthoRight - this._orthoLeft, s = this._orthoTop - this._orthoBottom, a = this.shadowMinZ !== void 0 ? this.shadowMinZ : n.minZ, f = this.shadowMaxZ !== void 0 ? this.shadowMaxZ : n.maxZ, o = this.getScene().getEngine().useReverseDepthBuffer; he.OrthoOffCenterLHToRef(this._orthoLeft - i * this.shadowOrthoScale, this._orthoRight + i * this.shadowOrthoScale, this._orthoBottom - s * this.shadowOrthoScale, this._orthoTop + s * this.shadowOrthoScale, o ? f : a, o ? a : f, e, this.getScene().getEngine().isNDCHalfZRange); } _buildUniformLayout() { this._uniformBuffer.addUniform("vLightData", 4), this._uniformBuffer.addUniform("vLightDiffuse", 4), this._uniformBuffer.addUniform("vLightSpecular", 4), this._uniformBuffer.addUniform("shadowsInfo", 3), this._uniformBuffer.addUniform("depthValues", 2), this._uniformBuffer.create(); } /** * Sets the passed Effect object with the DirectionalLight transformed position (or position if not parented) and the passed name. * @param effect The effect to update * @param lightIndex The index of the light in the effect to update * @returns The directional light */ transferToEffect(e, t) { return this.computeTransformedInformation() ? (this._uniformBuffer.updateFloat4("vLightData", this.transformedDirection.x, this.transformedDirection.y, this.transformedDirection.z, 1, t), this) : (this._uniformBuffer.updateFloat4("vLightData", this.direction.x, this.direction.y, this.direction.z, 1, t), this); } transferToNodeMaterialEffect(e, t) { return this.computeTransformedInformation() ? (e.setFloat3(t, this.transformedDirection.x, this.transformedDirection.y, this.transformedDirection.z), this) : (e.setFloat3(t, this.direction.x, this.direction.y, this.direction.z), this); } /** * Gets the minZ used for shadow according to both the scene and the light. * * Values are fixed on directional lights as it relies on an ortho projection hence the need to convert being * -1 and 1 to 0 and 1 doing (depth + min) / (min + max) -> (depth + 1) / (1 + 1) -> (depth * 0.5) + 0.5. * (when not using reverse depth buffer / NDC half Z range) * @param activeCamera The camera we are returning the min for * @returns the depth min z */ // eslint-disable-next-line @typescript-eslint/no-unused-vars getDepthMinZ(e) { const t = this._scene.getEngine(); return !t.useReverseDepthBuffer && t.isNDCHalfZRange ? 0 : 1; } /** * Gets the maxZ used for shadow according to both the scene and the light. * * Values are fixed on directional lights as it relies on an ortho projection hence the need to convert being * -1 and 1 to 0 and 1 doing (depth + min) / (min + max) -> (depth + 1) / (1 + 1) -> (depth * 0.5) + 0.5. * (when not using reverse depth buffer / NDC half Z range) * @param activeCamera The camera we are returning the max for * @returns the depth max z */ // eslint-disable-next-line @typescript-eslint/no-unused-vars getDepthMaxZ(e) { const t = this._scene.getEngine(); return t.useReverseDepthBuffer && t.isNDCHalfZRange ? 0 : 1; } /** * Prepares the list of defines specific to the light type. * @param defines the list of defines * @param lightIndex defines the index of the light for the effect */ prepareLightSpecificDefines(e, t) { e["DIRLIGHT" + t] = !0; } } C([ M() ], IA.prototype, "shadowFrustumSize", null); C([ M() ], IA.prototype, "shadowOrthoScale", null); C([ M() ], IA.prototype, "autoUpdateExtends", void 0); C([ M() ], IA.prototype, "autoCalcShadowZBounds", void 0); C([ M("orthoLeft") ], IA.prototype, "_orthoLeft", void 0); C([ M("orthoRight") ], IA.prototype, "_orthoRight", void 0); C([ M("orthoTop") ], IA.prototype, "_orthoTop", void 0); C([ M("orthoBottom") ], IA.prototype, "_orthoBottom", void 0); function zI(A, e = {}, t) { e.diameter || (e.diameter = 1), e.segments || (e.segments = 16); const r = UA("", { slice: 0.5, diameter: e.diameter, segments: e.segments }, t), n = uU("", { radius: e.diameter / 2, tessellation: e.segments * 3 + (4 - e.segments) }, t); n.rotation.x = -Math.PI / 2, n.parent = r; const i = Ee.MergeMeshes([n, r], !0); return i.name = A, i; } const zle = { // eslint-disable-next-line @typescript-eslint/naming-convention CreateHemisphere: zI }; Ee.CreateHemisphere = (A, e, t, r) => zI(A, { segments: e, diameter: t }, r); Cs.AddNodeConstructor("Light_Type_2", (A, e) => () => new nA(A, S.Zero(), S.Zero(), 0, 0, e)); class nA extends xD { /** * Gets the cone angle of the spot light in Radians. */ get angle() { return this._angle; } /** * Sets the cone angle of the spot light in Radians. */ set angle(e) { this._angle = e, this._cosHalfAngle = Math.cos(e * 0.5), this._projectionTextureProjectionLightDirty = !0, this.forceProjectionMatrixCompute(), this._computeAngleValues(); } /** * Only used in gltf falloff mode, this defines the angle where * the directional falloff will start before cutting at angle which could be seen * as outer angle. */ get innerAngle() { return this._innerAngle; } /** * Only used in gltf falloff mode, this defines the angle where * the directional falloff will start before cutting at angle which could be seen * as outer angle. */ set innerAngle(e) { this._innerAngle = e, this._computeAngleValues(); } /** * Allows scaling the angle of the light for shadow generation only. */ get shadowAngleScale() { return this._shadowAngleScale; } /** * Allows scaling the angle of the light for shadow generation only. */ set shadowAngleScale(e) { this._shadowAngleScale = e, this.forceProjectionMatrixCompute(); } /** * Allows reading the projection texture */ get projectionTextureMatrix() { return this._projectionTextureMatrix; } /** * Gets the near clip of the Spotlight for texture projection. */ get projectionTextureLightNear() { return this._projectionTextureLightNear; } /** * Sets the near clip of the Spotlight for texture projection. */ set projectionTextureLightNear(e) { this._projectionTextureLightNear = e, this._projectionTextureProjectionLightDirty = !0; } /** * Gets the far clip of the Spotlight for texture projection. */ get projectionTextureLightFar() { return this._projectionTextureLightFar; } /** * Sets the far clip of the Spotlight for texture projection. */ set projectionTextureLightFar(e) { this._projectionTextureLightFar = e, this._projectionTextureProjectionLightDirty = !0; } /** * Gets the Up vector of the Spotlight for texture projection. */ get projectionTextureUpDirection() { return this._projectionTextureUpDirection; } /** * Sets the Up vector of the Spotlight for texture projection. */ set projectionTextureUpDirection(e) { this._projectionTextureUpDirection = e, this._projectionTextureProjectionLightDirty = !0; } /** * Gets the projection texture of the light. */ get projectionTexture() { return this._projectionTexture; } /** * Sets the projection texture of the light. */ set projectionTexture(e) { this._projectionTexture !== e && (this._projectionTexture = e, this._projectionTextureDirty = !0, this._projectionTexture && !this._projectionTexture.isReady() && (nA._IsProceduralTexture(this._projectionTexture) ? this._projectionTexture.getEffect().executeWhenCompiled(() => { this._markMeshesAsLightDirty(); }) : nA._IsTexture(this._projectionTexture) && this._projectionTexture.onLoadObservable.addOnce(() => { this._markMeshesAsLightDirty(); }))); } static _IsProceduralTexture(e) { return e.onGeneratedObservable !== void 0; } static _IsTexture(e) { return e.onLoadObservable !== void 0; } /** * Gets or sets the light projection matrix as used by the projection texture */ get projectionTextureProjectionLightMatrix() { return this._projectionTextureProjectionLightMatrix; } set projectionTextureProjectionLightMatrix(e) { this._projectionTextureProjectionLightMatrix = e, this._projectionTextureProjectionLightDirty = !1, this._projectionTextureDirty = !0; } /** * Creates a SpotLight object in the scene. A spot light is a simply light oriented cone. * It can cast shadows. * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction * @param name The light friendly name * @param position The position of the spot light in the scene * @param direction The direction of the light in the scene * @param angle The cone angle of the light in Radians * @param exponent The light decay speed with the distance from the emission spot * @param scene The scene the lights belongs to */ constructor(e, t, r, n, i, s) { super(e, s), this._innerAngle = 0, this._projectionTextureMatrix = he.Zero(), this._projectionTextureLightNear = 1e-6, this._projectionTextureLightFar = 1e3, this._projectionTextureUpDirection = S.Up(), this._projectionTextureViewLightDirty = !0, this._projectionTextureProjectionLightDirty = !0, this._projectionTextureDirty = !0, this._projectionTextureViewTargetVector = S.Zero(), this._projectionTextureViewLightMatrix = he.Zero(), this._projectionTextureProjectionLightMatrix = he.Zero(), this._projectionTextureScalingMatrix = he.FromValues(0.5, 0, 0, 0, 0, 0.5, 0, 0, 0, 0, 0.5, 0, 0.5, 0.5, 0.5, 1), this.position = t, this.direction = r, this.angle = n, this.exponent = i; } /** * Returns the string "SpotLight". * @returns the class name */ getClassName() { return "SpotLight"; } /** * Returns the integer 2. * @returns The light Type id as a constant defines in Light.LIGHTTYPEID_x */ getTypeID() { return ci.LIGHTTYPEID_SPOTLIGHT; } /** * Overrides the direction setter to recompute the projection texture view light Matrix. * @param value */ _setDirection(e) { super._setDirection(e), this._projectionTextureViewLightDirty = !0; } /** * Overrides the position setter to recompute the projection texture view light Matrix. * @param value */ _setPosition(e) { super._setPosition(e), this._projectionTextureViewLightDirty = !0; } /** * Sets the passed matrix "matrix" as perspective projection matrix for the shadows and the passed view matrix with the fov equal to the SpotLight angle and and aspect ratio of 1.0. * Returns the SpotLight. * @param matrix * @param viewMatrix * @param renderList */ // eslint-disable-next-line @typescript-eslint/no-unused-vars _setDefaultShadowProjectionMatrix(e, t, r) { const n = this.getScene().activeCamera; if (!n) return; this._shadowAngleScale = this._shadowAngleScale || 1; const i = this._shadowAngleScale * this._angle, s = this.shadowMinZ !== void 0 ? this.shadowMinZ : n.minZ, a = this.shadowMaxZ !== void 0 ? this.shadowMaxZ : n.maxZ, f = this.getScene().getEngine().useReverseDepthBuffer; he.PerspectiveFovLHToRef(i, 1, f ? a : s, f ? s : a, e, !0, this._scene.getEngine().isNDCHalfZRange, void 0, f); } _computeProjectionTextureViewLightMatrix() { this._projectionTextureViewLightDirty = !1, this._projectionTextureDirty = !0, this.getAbsolutePosition().addToRef(this.direction, this._projectionTextureViewTargetVector), he.LookAtLHToRef(this.getAbsolutePosition(), this._projectionTextureViewTargetVector, this._projectionTextureUpDirection, this._projectionTextureViewLightMatrix); } _computeProjectionTextureProjectionLightMatrix() { this._projectionTextureProjectionLightDirty = !1, this._projectionTextureDirty = !0; const e = this.projectionTextureLightFar, t = this.projectionTextureLightNear, r = e / (e - t), n = -r * t, i = 1 / Math.tan(this._angle / 2), s = 1; he.FromValuesToRef(i / s, 0, 0, 0, 0, i, 0, 0, 0, 0, r, 1, 0, 0, n, 0, this._projectionTextureProjectionLightMatrix); } /** * Main function for light texture projection matrix computing. */ _computeProjectionTextureMatrix() { if (this._projectionTextureDirty = !1, this._projectionTextureViewLightMatrix.multiplyToRef(this._projectionTextureProjectionLightMatrix, this._projectionTextureMatrix), this._projectionTexture instanceof We) { const e = this._projectionTexture.uScale / 2, t = this._projectionTexture.vScale / 2; he.FromValuesToRef(e, 0, 0, 0, 0, t, 0, 0, 0, 0, 0.5, 0, 0.5, 0.5, 0.5, 1, this._projectionTextureScalingMatrix); } this._projectionTextureMatrix.multiplyToRef(this._projectionTextureScalingMatrix, this._projectionTextureMatrix); } _buildUniformLayout() { this._uniformBuffer.addUniform("vLightData", 4), this._uniformBuffer.addUniform("vLightDiffuse", 4), this._uniformBuffer.addUniform("vLightSpecular", 4), this._uniformBuffer.addUniform("vLightDirection", 3), this._uniformBuffer.addUniform("vLightFalloff", 4), this._uniformBuffer.addUniform("shadowsInfo", 3), this._uniformBuffer.addUniform("depthValues", 2), this._uniformBuffer.create(); } _computeAngleValues() { this._lightAngleScale = 1 / Math.max(1e-3, Math.cos(this._innerAngle * 0.5) - this._cosHalfAngle), this._lightAngleOffset = -this._cosHalfAngle * this._lightAngleScale; } /** * Sets the passed Effect "effect" with the Light textures. * @param effect The effect to update * @param lightIndex The index of the light in the effect to update * @returns The light */ transferTexturesToEffect(e, t) { return this.projectionTexture && this.projectionTexture.isReady() && (this._projectionTextureViewLightDirty && this._computeProjectionTextureViewLightMatrix(), this._projectionTextureProjectionLightDirty && this._computeProjectionTextureProjectionLightMatrix(), this._projectionTextureDirty && this._computeProjectionTextureMatrix(), e.setMatrix("textureProjectionMatrix" + t, this._projectionTextureMatrix), e.setTexture("projectionLightSampler" + t, this.projectionTexture)), this; } /** * Sets the passed Effect object with the SpotLight transformed position (or position if not parented) and normalized direction. * @param effect The effect to update * @param lightIndex The index of the light in the effect to update * @returns The spot light */ transferToEffect(e, t) { let r; return this.computeTransformedInformation() ? (this._uniformBuffer.updateFloat4("vLightData", this.transformedPosition.x, this.transformedPosition.y, this.transformedPosition.z, this.exponent, t), r = S.Normalize(this.transformedDirection)) : (this._uniformBuffer.updateFloat4("vLightData", this.position.x, this.position.y, this.position.z, this.exponent, t), r = S.Normalize(this.direction)), this._uniformBuffer.updateFloat4("vLightDirection", r.x, r.y, r.z, this._cosHalfAngle, t), this._uniformBuffer.updateFloat4("vLightFalloff", this.range, this._inverseSquaredRange, this._lightAngleScale, this._lightAngleOffset, t), this; } transferToNodeMaterialEffect(e, t) { let r; return this.computeTransformedInformation() ? r = S.Normalize(this.transformedDirection) : r = S.Normalize(this.direction), this.getScene().useRightHandedSystem ? e.setFloat3(t, -r.x, -r.y, -r.z) : e.setFloat3(t, r.x, r.y, r.z), this; } /** * Disposes the light and the associated resources. */ dispose() { super.dispose(), this._projectionTexture && this._projectionTexture.dispose(); } /** * Gets the minZ used for shadow according to both the scene and the light. * @param activeCamera The camera we are returning the min for * @returns the depth min z */ getDepthMinZ(e) { const t = this._scene.getEngine(), r = this.shadowMinZ !== void 0 ? this.shadowMinZ : e.minZ; return t.useReverseDepthBuffer && t.isNDCHalfZRange ? r : this._scene.getEngine().isNDCHalfZRange ? 0 : r; } /** * Gets the maxZ used for shadow according to both the scene and the light. * @param activeCamera The camera we are returning the max for * @returns the depth max z */ getDepthMaxZ(e) { const t = this._scene.getEngine(), r = this.shadowMaxZ !== void 0 ? this.shadowMaxZ : e.maxZ; return t.useReverseDepthBuffer && t.isNDCHalfZRange ? 0 : r; } /** * Prepares the list of defines specific to the light type. * @param defines the list of defines * @param lightIndex defines the index of the light for the effect */ prepareLightSpecificDefines(e, t) { e["SPOTLIGHT" + t] = !0, e["PROJECTEDLIGHTTEXTURE" + t] = !!(this.projectionTexture && this.projectionTexture.isReady()); } } C([ M() ], nA.prototype, "angle", null); C([ M() ], nA.prototype, "innerAngle", null); C([ M() ], nA.prototype, "shadowAngleScale", null); C([ M() ], nA.prototype, "exponent", void 0); C([ M() ], nA.prototype, "projectionTextureLightNear", null); C([ M() ], nA.prototype, "projectionTextureLightFar", null); C([ M() ], nA.prototype, "projectionTextureUpDirection", null); C([ en("projectedLightTexture") ], nA.prototype, "_projectionTexture", void 0); class D0 extends Lo { /** * Creates a LightGizmo * @param gizmoLayer The utility layer the gizmo will be added to */ constructor(e = Ds.DefaultUtilityLayer) { super(e), this._cachedPosition = new S(), this._cachedForward = new S(0, 0, 1), this._pointerObserver = null, this.onClickedObservable = new Oe(), this._light = null, this.attachedMesh = new jn("", this.gizmoLayer.utilityLayerScene), this._attachedMeshParent = new Hr("parent", this.gizmoLayer.utilityLayerScene), this.attachedMesh.parent = this._attachedMeshParent, this._material = new Wt("light", this.gizmoLayer.utilityLayerScene), this._material.diffuseColor = new Ne(0.5, 0.5, 0.5), this._material.specularColor = new Ne(0.1, 0.1, 0.1), this._pointerObserver = e.utilityLayerScene.onPointerObservable.add((t) => { this._light && (this._isHovered = !!(t.pickInfo && this._rootMesh.getChildMeshes().indexOf(t.pickInfo.pickedMesh) != -1), this._isHovered && t.event.button === 0 && this.onClickedObservable.notifyObservers(this._light)); }, ir.POINTERDOWN); } /** * Override attachedNode because lightgizmo only support attached mesh * It will return the attached mesh (if any) and setting an attached node will log * a warning */ get attachedNode() { return this.attachedMesh; } set attachedNode(e) { console.warn("Nodes cannot be attached to LightGizmo. Attach to a mesh instead."); } /** * The light that the gizmo is attached to */ set light(e) { if (this._light = e, e) { this._lightMesh && this._lightMesh.dispose(), e instanceof y0 ? this._lightMesh = D0._CreateHemisphericLightMesh(this.gizmoLayer.utilityLayerScene) : e instanceof IA ? this._lightMesh = D0._CreateDirectionalLightMesh(this.gizmoLayer.utilityLayerScene) : e instanceof nA ? this._lightMesh = D0._CreateSpotLightMesh(this.gizmoLayer.utilityLayerScene) : this._lightMesh = D0._CreatePointLightMesh(this.gizmoLayer.utilityLayerScene), this._lightMesh.getChildMeshes(!1).forEach((r) => { r.material = this._material; }), this._lightMesh.parent = this._rootMesh; const t = this.gizmoLayer._getSharedGizmoLight(); if (t.includedOnlyMeshes = t.includedOnlyMeshes.concat(this._lightMesh.getChildMeshes(!1)), this._lightMesh.rotationQuaternion = new Ze(), this.attachedMesh.reservedDataStore || (this.attachedMesh.reservedDataStore = {}), this.attachedMesh.reservedDataStore.lightGizmo = this, e.parent && this._attachedMeshParent.freezeWorldMatrix(e.parent.getWorldMatrix()), e.position && (this.attachedMesh.position.copyFrom(e.position), this.attachedMesh.computeWorldMatrix(!0), this._cachedPosition.copyFrom(this.attachedMesh.position)), e.direction) { this.attachedMesh.setDirection(e.direction), this.attachedMesh.computeWorldMatrix(!0); const r = this._getMeshForward(); this._cachedForward.copyFrom(r); } this._update(); } } get light() { return this._light; } /** * Gets the material used to render the light gizmo */ get material() { return this._material; } /** * @internal * returns mesh forward */ _getMeshForward() { let e = this.attachedMesh.forward; return this.attachedMesh.getScene().useRightHandedSystem && (e.negateToRef(ue.Vector3[0]), e = ue.Vector3[0]), e; } /** * @internal * Updates the gizmo to match the attached mesh's position/rotation */ _update() { if (super._update(), !!this._light) { if (this._light.parent && this._attachedMeshParent.freezeWorldMatrix(this._light.parent.getWorldMatrix()), this._light.position) if (this.attachedMesh.position.equals(this._cachedPosition)) this.attachedMesh.position.copyFrom(this._light.position), this.attachedMesh.computeWorldMatrix(!0), this._cachedPosition.copyFrom(this.attachedMesh.position); else { const e = this.attachedMesh.position; this._light.position = new S(e.x, e.y, e.z), this._cachedPosition.copyFrom(this.attachedMesh.position); } if (this._light.direction) { const e = this._getMeshForward(); if (S.DistanceSquared(e, this._cachedForward) > 1e-4) { const t = e; this._light.direction = new S(t.x, t.y, t.z), this._cachedForward.copyFrom(e); } else S.DistanceSquared(e, this._light.direction) > 1e-4 && (this.attachedMesh.setDirection(this._light.direction), this.attachedMesh.computeWorldMatrix(!0), this._cachedForward.copyFrom(e)); } } } /** * Disposes of the light gizmo */ dispose() { this.onClickedObservable.clear(), this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(this._pointerObserver), this._material.dispose(), super.dispose(), this._attachedMeshParent.dispose(); } static _CreateHemisphericLightMesh(e) { const t = new Ee("hemisphereLight", e), r = zI(t.name, { segments: 10, diameter: 1 }, e); r.position.z = -0.15, r.rotation.x = Math.PI / 2, r.parent = t; const n = this._CreateLightLines(3, e); return n.parent = t, t.scaling.scaleInPlace(D0._Scale), t.rotation.x = Math.PI / 2, t; } static _CreatePointLightMesh(e) { const t = new Ee("pointLight", e), r = UA(t.name, { segments: 10, diameter: 1 }, e); r.rotation.x = Math.PI / 2, r.parent = t; const n = this._CreateLightLines(5, e); return n.parent = t, t.scaling.scaleInPlace(D0._Scale), t.rotation.x = Math.PI / 2, t; } static _CreateSpotLightMesh(e) { const t = new Ee("spotLight", e), r = UA(t.name, { segments: 10, diameter: 1 }, e); r.parent = t; const n = zI(t.name, { segments: 10, diameter: 2 }, e); n.parent = t, n.rotation.x = -Math.PI / 2; const i = this._CreateLightLines(2, e); return i.parent = t, t.scaling.scaleInPlace(D0._Scale), t.rotation.x = Math.PI / 2, t; } static _CreateDirectionalLightMesh(e) { const t = new Ee("directionalLight", e), r = new Ee(t.name, e); r.parent = t; const n = UA(t.name, { diameter: 1.2, segments: 10 }, e); n.parent = r; const i = Ld(t.name, { updatable: !1, height: 6, diameterTop: 0.3, diameterBottom: 0.3, tessellation: 6, subdivisions: 1 }, e); i.parent = r; let s = i.clone(t.name); s.scaling.y = 0.5, s.position.x += 1.25; let a = i.clone(t.name); a.scaling.y = 0.5, a.position.x += -1.25; const f = Ld(t.name, { updatable: !1, height: 1, diameterTop: 0, diameterBottom: 0.6, tessellation: 6, subdivisions: 1 }, e); return f.position.y += 3, f.parent = r, s = f.clone(t.name), s.position.y = 1.5, s.position.x += 1.25, a = f.clone(t.name), a.position.y = 1.5, a.position.x += -1.25, r.scaling.scaleInPlace(D0._Scale), r.rotation.z = Math.PI / 2, r.rotation.y = Math.PI / 2, t; } } D0._Scale = 7e-3; D0._CreateLightLines = (A, e) => { const r = new Ee("root", e); r.rotation.x = Math.PI / 2; const n = new Ee("linePivot", e); n.parent = r; const i = Ld("line", { updatable: !1, height: 2, diameterTop: 0.2, diameterBottom: 0.3, tessellation: 6, subdivisions: 1 }, e); if (i.position.y = i.scaling.y / 2 + 1.2, i.parent = n, A < 2) return n; for (let a = 0; a < 4; a++) { const f = n.clone("lineParentClone"); f.rotation.z = Math.PI / 4, f.rotation.y = Math.PI / 2 + Math.PI / 2 * a, f.getChildMeshes()[0].scaling.y = 0.5, f.getChildMeshes()[0].scaling.x = f.getChildMeshes()[0].scaling.z = 0.8, f.getChildMeshes()[0].position.y = f.getChildMeshes()[0].scaling.y / 2 + 1.2; } if (A < 3) return r; for (let a = 0; a < 4; a++) { const f = n.clone("linePivotClone"); f.rotation.z = Math.PI / 2, f.rotation.y = Math.PI / 2 * a; } if (A < 4) return r; for (let a = 0; a < 4; a++) { const f = n.clone("linePivotClone"); f.rotation.z = Math.PI + Math.PI / 4, f.rotation.y = Math.PI / 2 + Math.PI / 2 * a, f.getChildMeshes()[0].scaling.y = 0.5, f.getChildMeshes()[0].scaling.x = f.getChildMeshes()[0].scaling.z = 0.8, f.getChildMeshes()[0].position.y = f.getChildMeshes()[0].scaling.y / 2 + 1.2; } if (A < 5) return r; const s = n.clone("linePivotClone"); return s.rotation.z = Math.PI, r; }; class rS extends Lo { /** * Creates a CameraGizmo * @param gizmoLayer The utility layer the gizmo will be added to */ constructor(e = Ds.DefaultUtilityLayer, t) { super(e), this._pointerObserver = null, this.onClickedObservable = new Oe(), this._camera = null, this._invProjection = new he(), this._material = new Wt("cameraGizmoMaterial", this.gizmoLayer.utilityLayerScene), this._gizmoColor = t, this._material.diffuseColor = t ?? new Ne(0.5, 0.5, 0.5), this._material.specularColor = new Ne(0.1, 0.1, 0.1), this._pointerObserver = e.utilityLayerScene.onPointerObservable.add((r) => { this._camera && (this._isHovered = !!(r.pickInfo && this._rootMesh.getChildMeshes().indexOf(r.pickInfo.pickedMesh) != -1), this._isHovered && r.event.button === 0 && this.onClickedObservable.notifyObservers(this._camera)); }, ir.POINTERDOWN); } /** Gets or sets a boolean indicating if frustum lines must be rendered (true by default)) */ get displayFrustum() { return this._cameraLinesMesh.isEnabled(); } set displayFrustum(e) { this._cameraLinesMesh.setEnabled(e); } /** * The camera that the gizmo is attached to */ set camera(e) { var t, r; if (this._camera = e, this.attachedNode = e, e) { this._cameraMesh && this._cameraMesh.dispose(), this._cameraLinesMesh && this._cameraLinesMesh.dispose(), this._cameraMesh = rS._CreateCameraMesh(this.gizmoLayer.utilityLayerScene); const n = (r = (t = this._gizmoColor) === null || t === void 0 ? void 0 : t.toColor4(1)) !== null && r !== void 0 ? r : new xt(1, 1, 1, 1); this._cameraLinesMesh = rS._CreateCameraFrustum(this.gizmoLayer.utilityLayerScene, n), this._cameraMesh.getChildMeshes(!1).forEach((s) => { s.material = this._material; }), this._cameraMesh.parent = this._rootMesh, this._cameraLinesMesh.parent = this._rootMesh, this.gizmoLayer.utilityLayerScene.activeCamera && this.gizmoLayer.utilityLayerScene.activeCamera.maxZ < e.maxZ * 1.5 && (this.gizmoLayer.utilityLayerScene.activeCamera.maxZ = e.maxZ * 1.5), this.attachedNode.reservedDataStore || (this.attachedNode.reservedDataStore = {}), this.attachedNode.reservedDataStore.cameraGizmo = this; const i = this.gizmoLayer._getSharedGizmoLight(); i.includedOnlyMeshes = i.includedOnlyMeshes.concat(this._cameraMesh.getChildMeshes(!1)), this._update(); } } get camera() { return this._camera; } /** * Gets the material used to render the camera gizmo */ get material() { return this._material; } /** * @internal * Updates the gizmo to match the attached mesh's position/rotation */ _update() { super._update(), this._camera && (this._camera.getProjectionMatrix().invertToRef(this._invProjection), this._cameraLinesMesh.setPivotMatrix(this._invProjection, !1), this._cameraLinesMesh.scaling.x = 1 / this._rootMesh.scaling.x, this._cameraLinesMesh.scaling.y = 1 / this._rootMesh.scaling.y, this._cameraLinesMesh.scaling.z = 1 / this._rootMesh.scaling.z, this._cameraMesh.parent = null, this._cameraMesh.rotation.y = Math.PI * 0.5 * (this._camera.getScene().useRightHandedSystem ? 1 : -1), this._cameraMesh.parent = this._rootMesh); } /** * Disposes of the camera gizmo */ dispose() { this.onClickedObservable.clear(), this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(this._pointerObserver), this._cameraMesh && this._cameraMesh.dispose(), this._cameraLinesMesh && this._cameraLinesMesh.dispose(), this._material.dispose(), super.dispose(); } static _CreateCameraMesh(e) { const t = new Ee("rootCameraGizmo", e), r = new Ee(t.name, e); r.parent = t; const n = k0(t.name, { width: 1, height: 0.8, depth: 0.5 }, e); n.parent = r; const i = Ld(t.name, { height: 0.5, diameterTop: 0.8, diameterBottom: 0.8 }, e); i.parent = r, i.position.y = 0.3, i.position.x = -0.6, i.rotation.x = Math.PI * 0.5; const s = Ld(t.name, { height: 0.5, diameterTop: 0.6, diameterBottom: 0.6 }, e); s.parent = r, s.position.y = 0.5, s.position.x = 0.4, s.rotation.x = Math.PI * 0.5; const a = Ld(t.name, { height: 0.5, diameterTop: 0.5, diameterBottom: 0.5 }, e); return a.parent = r, a.position.y = 0, a.position.x = 0.6, a.rotation.z = Math.PI * 0.5, t.scaling.scaleInPlace(rS._Scale), r.position.x = -0.9, t; } static _CreateCameraFrustum(e, t) { const r = new Ee("rootCameraGizmo", e), n = new Ee(r.name, e); n.parent = r; for (let i = 0; i < 4; i += 2) for (let s = 0; s < 4; s += 2) { let a = ka("lines", { points: [new S(-1 + s, -1 + i, -1), new S(-1 + s, -1 + i, 1)], colors: [t, t] }, e); a.parent = n, a.alwaysSelectAsActiveMesh = !0, a.isPickable = !1, a = ka("lines", { points: [new S(-1, -1 + s, -1 + i), new S(1, -1 + s, -1 + i)], colors: [t, t] }, e), a.parent = n, a.alwaysSelectAsActiveMesh = !0, a.isPickable = !1, a = ka("lines", { points: [new S(-1 + s, -1, -1 + i), new S(-1 + s, 1, -1 + i)], colors: [t, t] }, e), a.parent = n, a.alwaysSelectAsActiveMesh = !0, a.isPickable = !1; } return r; } } rS._Scale = 0.05; const Gle = "kernelBlurVaryingDeclaration", Zle = "varying vec2 sampleCoord{X};"; Le.IncludesShadersStore[Gle] = Zle; const _le = "packingFunctions", $le = `vec4 pack(float depth) {const vec4 bit_shift=vec4(255.0*255.0*255.0,255.0*255.0,255.0,1.0);const vec4 bit_mask=vec4(0.0,1.0/255.0,1.0/255.0,1.0/255.0);vec4 res=fract(depth*bit_shift);res-=res.xxyz*bit_mask;return res;} float unpack(vec4 color) {const vec4 bit_shift=vec4(1.0/(255.0*255.0*255.0),1.0/(255.0*255.0),1.0/255.0,1.0);return dot(color,bit_shift);}`; Le.IncludesShadersStore[_le] = $le; const ePe = "kernelBlurFragment", tPe = `#ifdef DOF factor=sampleCoC(sampleCoord{X}); computedWeight=KERNEL_WEIGHT{X}*factor;sumOfWeights+=computedWeight; #else computedWeight=KERNEL_WEIGHT{X}; #endif #ifdef PACKEDFLOAT blend+=unpack(texture2D(textureSampler,sampleCoord{X}))*computedWeight; #else blend+=texture2D(textureSampler,sampleCoord{X})*computedWeight; #endif `; Le.IncludesShadersStore[ePe] = tPe; const rPe = "kernelBlurFragment2", nPe = `#ifdef DOF factor=sampleCoC(sampleCenter+delta*KERNEL_DEP_OFFSET{X});computedWeight=KERNEL_DEP_WEIGHT{X}*factor;sumOfWeights+=computedWeight; #else computedWeight=KERNEL_DEP_WEIGHT{X}; #endif #ifdef PACKEDFLOAT blend+=unpack(texture2D(textureSampler,sampleCenter+delta*KERNEL_DEP_OFFSET{X}))*computedWeight; #else blend+=texture2D(textureSampler,sampleCenter+delta*KERNEL_DEP_OFFSET{X})*computedWeight; #endif `; Le.IncludesShadersStore[rPe] = nPe; const iPe = "kernelBlurPixelShader", sPe = `uniform sampler2D textureSampler;uniform vec2 delta;varying vec2 sampleCenter; #ifdef DOF uniform sampler2D circleOfConfusionSampler;float sampleCoC(in vec2 offset) {float coc=texture2D(circleOfConfusionSampler,offset).r;return coc; } #endif #include[0..varyingCount] #ifdef PACKEDFLOAT #include #endif #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {float computedWeight=0.0; #ifdef PACKEDFLOAT float blend=0.; #else vec4 blend=vec4(0.); #endif #ifdef DOF float sumOfWeights=CENTER_WEIGHT; float factor=0.0; #ifdef PACKEDFLOAT blend+=unpack(texture2D(textureSampler,sampleCenter))*CENTER_WEIGHT; #else blend+=texture2D(textureSampler,sampleCenter)*CENTER_WEIGHT; #endif #endif #include[0..varyingCount] #include[0..depCount] #ifdef PACKEDFLOAT gl_FragColor=pack(blend); #else gl_FragColor=blend; #endif #ifdef DOF gl_FragColor/=sumOfWeights; #endif }`; Le.ShadersStore[iPe] = sPe; const aPe = "kernelBlurVertex", oPe = "sampleCoord{X}=sampleCenter+delta*KERNEL_OFFSET{X};"; Le.IncludesShadersStore[aPe] = oPe; const fPe = "kernelBlurVertexShader", APe = `attribute vec2 position;uniform vec2 delta;varying vec2 sampleCenter; #include[0..varyingCount] const vec2 madd=vec2(0.5,0.5); #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN sampleCenter=(position*madd+madd); #include[0..varyingCount] gl_Position=vec4(position,0.0,1.0); #define CUSTOM_VERTEX_MAIN_END }`; Le.ShadersStore[fPe] = APe; class c9 extends kr { /** * Sets the length in pixels of the blur sample region */ set kernel(e) { this._idealKernel !== e && (e = Math.max(e, 1), this._idealKernel = e, this._kernel = this._nearestBestKernel(e), this._blockCompilation || this._updateParameters()); } /** * Gets the length in pixels of the blur sample region */ get kernel() { return this._idealKernel; } /** * Sets whether or not the blur needs to unpack/repack floats */ set packedFloat(e) { this._packedFloat !== e && (this._packedFloat = e, this._blockCompilation || this._updateParameters()); } /** * Gets whether or not the blur is unpacking/repacking floats */ get packedFloat() { return this._packedFloat; } /** * Gets a string identifying the name of the class * @returns "BlurPostProcess" string */ getClassName() { return "BlurPostProcess"; } /** * Creates a new instance BlurPostProcess * @param name The name of the effect. * @param direction The direction in which to blur the image. * @param kernel The size of the kernel to be used when computing the blur. eg. Size of 3 will blur the center pixel by 2 pixels surrounding it. * @param options The required width/height ratio to downsize to before computing the render pass. (Use 1.0 for full size) * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) * @param textureType Type of textures used when performing the post process. (default: 0) * @param defines * @param _blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false) * @param textureFormat Format of textures used when performing the post process. (default: TEXTUREFORMAT_RGBA) */ constructor(e, t, r, n, i, s = We.BILINEAR_SAMPLINGMODE, a, f, o = 0, d = "", v = !1, u = 5) { super(e, "kernelBlur", ["delta", "direction"], ["circleOfConfusionSampler"], n, i, s, a, f, null, o, "kernelBlur", { varyingCount: 0, depCount: 0 }, !0, u), this._blockCompilation = v, this._packedFloat = !1, this._staticDefines = "", this._staticDefines = d, this.direction = t, this.onApplyObservable.add((l) => { this._outputTexture ? l.setFloat2("delta", 1 / this._outputTexture.width * this.direction.x, 1 / this._outputTexture.height * this.direction.y) : l.setFloat2("delta", 1 / this.width * this.direction.x, 1 / this.height * this.direction.y); }), this.kernel = r; } /** * Updates the effect with the current post process compile time values and recompiles the shader. * @param defines Define statements that should be added at the beginning of the shader. (default: null) * @param uniforms Set of uniform variables that will be passed to the shader. (default: null) * @param samplers Set of Texture2D variables that will be passed to the shader. (default: null) * @param indexParameters The index parameters to be used for babylons include syntax "#include[0..varyingCount]". (default: undefined) See usage in babylon.blurPostProcess.ts and kernelBlur.vertex.fx * @param onCompiled Called when the shader has been compiled. * @param onError Called if there is an error when compiling a shader. */ updateEffect(e = null, t = null, r = null, n, i, s) { this._updateParameters(i, s); } _updateParameters(e, t) { const r = this._kernel, n = (r - 1) / 2; let i = [], s = [], a = 0; for (let c = 0; c < r; c++) { const H = c / (r - 1), T = this._gaussianWeight(H * 2 - 1); i[c] = c - n, s[c] = T, a += T; } for (let c = 0; c < s.length; c++) s[c] /= a; const f = [], o = [], d = []; for (let c = 0; c <= n; c += 2) { const H = Math.min(c + 1, Math.floor(n)); if (c === H) d.push({ o: i[c], w: s[c] }); else { const q = H === n, b = s[c] + s[H] * (q ? 0.5 : 1), j = i[c] + 1 / (1 + s[c] / s[H]); j === 0 ? (d.push({ o: i[c], w: s[c] }), d.push({ o: i[c + 1], w: s[c + 1] })) : (d.push({ o: j, w: b }), d.push({ o: -j, w: b })); } } for (let c = 0; c < d.length; c++) o[c] = d[c].o, f[c] = d[c].w; i = o, s = f; const v = this.getEngine().getCaps().maxVaryingVectors, u = Math.max(v, 0) - 1; let l = Math.min(i.length, u), P = ""; P += this._staticDefines, this._staticDefines.indexOf("DOF") != -1 && (P += `#define CENTER_WEIGHT ${this._glslFloat(s[l - 1])} `, l--); for (let c = 0; c < l; c++) P += `#define KERNEL_OFFSET${c} ${this._glslFloat(i[c])} `, P += `#define KERNEL_WEIGHT${c} ${this._glslFloat(s[c])} `; let p = 0; for (let c = u; c < i.length; c++) P += `#define KERNEL_DEP_OFFSET${p} ${this._glslFloat(i[c])} `, P += `#define KERNEL_DEP_WEIGHT${p} ${this._glslFloat(s[c])} `, p++; this.packedFloat && (P += "#define PACKEDFLOAT 1"), this._blockCompilation = !1, super.updateEffect(P, null, null, { varyingCount: l, depCount: p }, e, t); } /** * Best kernels are odd numbers that when divided by 2, their integer part is even, so 5, 9 or 13. * Other odd kernels optimize correctly but require proportionally more samples, even kernels are * possible but will produce minor visual artifacts. Since each new kernel requires a new shader we * want to minimize kernel changes, having gaps between physical kernels is helpful in that regard. * The gaps between physical kernels are compensated for in the weighting of the samples * @param idealKernel Ideal blur kernel. * @returns Nearest best kernel. */ _nearestBestKernel(e) { const t = Math.round(e); for (const r of [t, t - 1, t + 1, t - 2, t + 2]) if (r % 2 !== 0 && Math.floor(r / 2) % 2 === 0 && r > 0) return Math.max(r, 3); return Math.max(t, 3); } /** * Calculates the value of a Gaussian distribution with sigma 3 at a given point. * @param x The point on the Gaussian distribution to sample. * @returns the value of the Gaussian function at x. */ _gaussianWeight(e) { const t = 0.3333333333333333, r = Math.sqrt(2 * Math.PI) * t, n = -(e * e / (2 * t * t)); return 1 / r * Math.exp(n); } /** * Generates a string that can be used as a floating point number in GLSL. * @param x Value to print. * @param decimalFigures Number of decimal places to print the number to (excluding trailing 0s). * @returns GLSL float string. */ _glslFloat(e, t = 8) { return e.toFixed(t).replace(/0+$/, ""); } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new c9(e.name, e.direction, e.kernel, e.options, t, e.renderTargetSamplingMode, r.getEngine(), e.reusable, e.textureType, void 0, !1), e, r, n); } } C([ M("kernel") ], c9.prototype, "_kernel", void 0); C([ M("packedFloat") ], c9.prototype, "_packedFloat", void 0); C([ HR() ], c9.prototype, "direction", void 0); Ue("BABYLON.BlurPostProcess", c9); class uD extends Ta { /** * Define the blur ratio used to blur the reflection if needed. */ set blurRatio(e) { this._blurRatio !== e && (this._blurRatio = e, this._preparePostProcesses()); } get blurRatio() { return this._blurRatio; } /** * Define the adaptive blur kernel used to blur the reflection if needed. * This will autocompute the closest best match for the `blurKernel` */ set adaptiveBlurKernel(e) { this._adaptiveBlurKernel = e, this._autoComputeBlurKernel(); } /** * Define the blur kernel used to blur the reflection if needed. * Please consider using `adaptiveBlurKernel` as it could find the closest best value for you. */ set blurKernel(e) { this.blurKernelX = e, this.blurKernelY = e; } /** * Define the blur kernel on the X Axis used to blur the reflection if needed. * Please consider using `adaptiveBlurKernel` as it could find the closest best value for you. */ set blurKernelX(e) { this._blurKernelX !== e && (this._blurKernelX = e, this._preparePostProcesses()); } get blurKernelX() { return this._blurKernelX; } /** * Define the blur kernel on the Y Axis used to blur the reflection if needed. * Please consider using `adaptiveBlurKernel` as it could find the closest best value for you. */ set blurKernelY(e) { this._blurKernelY !== e && (this._blurKernelY = e, this._preparePostProcesses()); } get blurKernelY() { return this._blurKernelY; } _autoComputeBlurKernel() { const e = this.getScene().getEngine(), t = this.getRenderWidth() / e.getRenderWidth(), r = this.getRenderHeight() / e.getRenderHeight(); this.blurKernelX = this._adaptiveBlurKernel * t, this.blurKernelY = this._adaptiveBlurKernel * r; } _onRatioRescale() { this._sizeRatio && (this.resize(this._initialSizeParameter), this._adaptiveBlurKernel || this._preparePostProcesses()), this._adaptiveBlurKernel && this._autoComputeBlurKernel(); } _updateGammaSpace() { const e = this.getScene(); e && (this.gammaSpace = !e.imageProcessingConfiguration.isEnabled || !e.imageProcessingConfiguration.applyByPostProcess); } /** * Instantiates a Mirror Texture. * Mirror texture can be used to simulate the view from a mirror in a scene. * It will dynamically be rendered every frame to adapt to the camera point of view. * You can then easily use it as a reflectionTexture on a flat surface. * In case the surface is not a plane, please consider relying on reflection probes. * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#mirrors * @param name * @param size * @param scene * @param generateMipMaps * @param type * @param samplingMode * @param generateDepthBuffer */ constructor(e, t, r, n, i = 0, s = We.BILINEAR_SAMPLINGMODE, a = !0) { if (super(e, t, r, n, !0, i, !1, s, a), this.mirrorPlane = new BA(0, 1, 0, 1), this._transformMatrix = he.Zero(), this._mirrorMatrix = he.Zero(), this._adaptiveBlurKernel = 0, this._blurKernelX = 0, this._blurKernelY = 0, this._blurRatio = 1, r = this.getScene(), !r) return this; this.ignoreCameraViewport = !0, this._updateGammaSpace(), this._imageProcessingConfigChangeObserver = r.imageProcessingConfiguration.onUpdateParameters.add(() => { this._updateGammaSpace(); }); const f = r.getEngine(); f.supportsUniformBuffers && (this._sceneUBO = r.createSceneUniformBuffer(`Scene for Mirror Texture (name "${e}")`)), this.onBeforeBindObservable.add(() => { var d; (d = f._debugPushGroup) === null || d === void 0 || d.call(f, `mirror generation for ${e}`, 1); }), this.onAfterUnbindObservable.add(() => { var d; (d = f._debugPopGroup) === null || d === void 0 || d.call(f, 1); }); let o; this.onBeforeRenderObservable.add(() => { this._sceneUBO && (this._currentSceneUBO = r.getSceneUniformBuffer(), r.setSceneUniformBuffer(this._sceneUBO), r.getSceneUniformBuffer().unbindEffect()), he.ReflectionToRef(this.mirrorPlane, this._mirrorMatrix), this._mirrorMatrix.multiplyToRef(r.getViewMatrix(), this._transformMatrix), r.setTransformMatrix(this._transformMatrix, r.getProjectionMatrix()), o = r.clipPlane, r.clipPlane = this.mirrorPlane, r._mirroredCameraPosition = S.TransformCoordinates(r.activeCamera.globalPosition, this._mirrorMatrix); }), this.onAfterRenderObservable.add(() => { this._sceneUBO && r.setSceneUniformBuffer(this._currentSceneUBO), r.updateTransformMatrix(), r._mirroredCameraPosition = null, r.clipPlane = o; }); } _preparePostProcesses() { if (this.clearPostProcesses(!0), this._blurKernelX && this._blurKernelY) { const e = this.getScene().getEngine(), t = e.getCaps().textureFloatRender && e.getCaps().textureFloatLinearFiltering ? 1 : 2; this._blurX = new c9("horizontal blur", new at(1, 0), this._blurKernelX, this._blurRatio, null, We.BILINEAR_SAMPLINGMODE, e, !1, t), this._blurX.autoClear = !1, this._blurRatio === 1 && this.samples < 2 && this._texture ? this._blurX.inputTexture = this._renderTarget : this._blurX.alwaysForcePOT = !0, this._blurY = new c9("vertical blur", new at(0, 1), this._blurKernelY, this._blurRatio, null, We.BILINEAR_SAMPLINGMODE, e, !1, t), this._blurY.autoClear = !1, this._blurY.alwaysForcePOT = this._blurRatio !== 1, this.addPostProcess(this._blurX), this.addPostProcess(this._blurY); } else this._blurY && (this.removePostProcess(this._blurY), this._blurY.dispose(), this._blurY = null), this._blurX && (this.removePostProcess(this._blurX), this._blurX.dispose(), this._blurX = null); } /** * Clone the mirror texture. * @returns the cloned texture */ clone() { const e = this.getScene(); if (!e) return this; const t = this.getSize(), r = new uD(this.name, t.width, e, this._renderTargetOptions.generateMipMaps, this._renderTargetOptions.type, this._renderTargetOptions.samplingMode, this._renderTargetOptions.generateDepthBuffer); return r.hasAlpha = this.hasAlpha, r.level = this.level, r.mirrorPlane = this.mirrorPlane.clone(), this.renderList && (r.renderList = this.renderList.slice(0)), r; } /** * Serialize the texture to a JSON representation you could use in Parse later on * @returns the serialized JSON representation */ serialize() { if (!this.name) return null; const e = super.serialize(); return e.mirrorPlane = this.mirrorPlane.asArray(), e; } /** * Dispose the texture and release its associated resources. */ dispose() { var e; super.dispose(); const t = this.getScene(); t && t.imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingConfigChangeObserver), (e = this._sceneUBO) === null || e === void 0 || e.dispose(); } } We._CreateMirror = (A, e, t, r) => new uD(A, e, t, r); class v1 extends ls { /** * Gets or sets the size of the bounding box associated with the cube texture * When defined, the cubemap will switch to local mode * @see https://community.arm.com/graphics/b/blog/posts/reflections-based-on-local-cubemaps-in-unity * @example https://www.babylonjs-playground.com/#RNASML */ set boundingBoxSize(e) { if (this._boundingBoxSize && this._boundingBoxSize.equals(e)) return; this._boundingBoxSize = e; const t = this.getScene(); t && t.markAllMaterialsAsDirty(1); } /** * Returns the bounding box size * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#using-local-cubemap-mode */ get boundingBoxSize() { return this._boundingBoxSize; } /** * Sets texture matrix rotation angle around Y axis in radians. */ set rotationY(e) { this._rotationY = e, this.setReflectionTextureMatrix(he.RotationY(this._rotationY)); } /** * Gets texture matrix rotation angle around Y axis radians. */ get rotationY() { return this._rotationY; } /** * Are mip maps generated for this texture or not. */ get noMipmap() { return this._noMipmap; } /** * Gets the forced extension (if any) */ get forcedExtension() { return this._forcedExtension; } /** * Creates a cube texture from an array of image urls * @param files defines an array of image urls * @param scene defines the hosting scene * @param noMipmap specifies if mip maps are not used * @returns a cube texture */ static CreateFromImages(e, t, r) { let n = ""; return e.forEach((i) => n += i), new v1(n, t, null, r, e); } /** * Creates and return a texture created from prefilterd data by tools like IBL Baker or Lys. * @param url defines the url of the prefiltered texture * @param scene defines the scene the texture is attached to * @param forcedExtension defines the extension of the file if different from the url * @param createPolynomials defines whether or not to create polynomial harmonics from the texture data if necessary * @returns the prefiltered texture */ static CreateFromPrefilteredData(e, t, r = null, n = !0) { const i = t.useDelayedTextureLoading; t.useDelayedTextureLoading = !1; const s = new v1(e, t, null, !1, null, null, null, void 0, !0, r, n); return t.useDelayedTextureLoading = i, s; } /** * Creates a cube texture to use with reflection for instance. It can be based upon dds or six images as well * as prefiltered data. * @param rootUrl defines the url of the texture or the root name of the six images * @param sceneOrEngine defines the scene or engine the texture is attached to * @param extensions defines the suffixes add to the picture name in case six images are in use like _px.jpg... * @param noMipmap defines if mipmaps should be created or not * @param files defines the six files to load for the different faces in that order: px, py, pz, nx, ny, nz * @param onLoad defines a callback triggered at the end of the file load if no errors occurred * @param onError defines a callback triggered in case of error during load * @param format defines the internal format to use for the texture once loaded * @param prefiltered defines whether or not the texture is created from prefiltered data * @param forcedExtension defines the extensions to use (force a special type of file to load) in case it is different from the file name * @param createPolynomials defines whether or not to create polynomial harmonics from the texture data if necessary * @param lodScale defines the scale applied to environment texture. This manages the range of LOD level used for IBL according to the roughness * @param lodOffset defines the offset applied to environment texture. This manages first LOD level used for IBL according to the roughness * @param loaderOptions options to be passed to the loader * @param useSRGBBuffer Defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU) (default: false) * @returns the cube texture */ constructor(e, t, r = null, n = !1, i = null, s = null, a = null, f = 5, o = !1, d = null, v = !1, u = 0.8, l = 0, P, p) { var c; super(t), this._lodScale = 0.8, this._lodOffset = 0, this.onLoadObservable = new Oe(), this.boundingBoxPosition = S.Zero(), this._rotationY = 0, this._files = null, this._forcedExtension = null, this._extensions = null, this._textureMatrixRefraction = new he(), this.name = e, this.url = e, this._noMipmap = n, this.hasAlpha = !1, this._format = f, this.isCube = !0, this._textureMatrix = he.Identity(), this._createPolynomials = v, this.coordinatesMode = We.CUBIC_MODE, this._extensions = r, this._files = i, this._forcedExtension = d, this._loaderOptions = P, this._useSRGBBuffer = p, this._lodScale = u, this._lodOffset = l, !(!e && !i) && this.updateURL(e, d, s, o, a, r, (c = this.getScene()) === null || c === void 0 ? void 0 : c.useDelayedTextureLoading, i); } /** * Get the current class name of the texture useful for serialization or dynamic coding. * @returns "CubeTexture" */ getClassName() { return "CubeTexture"; } /** * Update the url (and optional buffer) of this texture if url was null during construction. * @param url the url of the texture * @param forcedExtension defines the extension to use * @param onLoad callback called when the texture is loaded (defaults to null) * @param prefiltered Defines whether the updated texture is prefiltered or not * @param onError callback called if there was an error during the loading process (defaults to null) * @param extensions defines the suffixes add to the picture name in case six images are in use like _px.jpg... * @param delayLoad defines if the texture should be loaded now (false by default) * @param files defines the six files to load for the different faces in that order: px, py, pz, nx, ny, nz */ updateURL(e, t, r = null, n = !1, i = null, s = null, a = !1, f = null) { (!this.name || this.name.startsWith("data:")) && (this.name = e), this.url = e, t && (this._forcedExtension = t); const o = e.lastIndexOf("."), d = t || (o > -1 ? e.substring(o).toLowerCase() : ""), v = d.indexOf(".dds") === 0, u = d.indexOf(".env") === 0, l = d.indexOf(".basis") === 0; if (u ? (this.gammaSpace = !1, this._prefiltered = !1, this.anisotropicFilteringLevel = 1) : (this._prefiltered = n, n && (this.gammaSpace = !1, this.anisotropicFilteringLevel = 1)), f) this._files = f; else if (!l && !u && !v && !s && (s = ["_px.jpg", "_py.jpg", "_pz.jpg", "_nx.jpg", "_ny.jpg", "_nz.jpg"]), this._files = this._files || [], this._files.length = 0, s) { for (let P = 0; P < s.length; P++) this._files.push(e + s[P]); this._extensions = s; } a ? (this.delayLoadState = 4, this._delayedOnLoad = r, this._delayedOnError = i) : this._loadTexture(r, i); } /** * Delays loading of the cube texture * @param forcedExtension defines the extension to use */ delayLoad(e) { this.delayLoadState === 4 && (e && (this._forcedExtension = e), this.delayLoadState = 1, this._loadTexture(this._delayedOnLoad, this._delayedOnError)); } /** * Returns the reflection texture matrix * @returns the reflection texture matrix */ getReflectionTextureMatrix() { return this._textureMatrix; } /** * Sets the reflection texture matrix * @param value Reflection texture matrix */ setReflectionTextureMatrix(e) { var t, r; if (e.updateFlag === this._textureMatrix.updateFlag || (e.isIdentity() !== this._textureMatrix.isIdentity() && ((t = this.getScene()) === null || t === void 0 || t.markAllMaterialsAsDirty(1, (a) => a.getActiveTextures().indexOf(this) !== -1)), this._textureMatrix = e, !(!((r = this.getScene()) === null || r === void 0) && r.useRightHandedSystem))) return; const n = ue.Vector3[0], i = ue.Quaternion[0], s = ue.Vector3[1]; this._textureMatrix.decompose(n, i, s), i.z *= -1, i.w *= -1, he.ComposeToRef(n, i, s, this._textureMatrixRefraction); } /** * Gets a suitable rotate/transform matrix when the texture is used for refraction. * There's a separate function from getReflectionTextureMatrix because refraction requires a special configuration of the matrix in right-handed mode. * @returns The refraction matrix */ getRefractionTextureMatrix() { var e; return !((e = this.getScene()) === null || e === void 0) && e.useRightHandedSystem ? this._textureMatrixRefraction : this._textureMatrix; } _loadTexture(e = null, t = null) { var r; const n = this.getScene(), i = this._texture; this._texture = this._getFromCache(this.url, this._noMipmap, void 0, void 0, this._useSRGBBuffer, this.isCube); const s = () => { var f; this.onLoadObservable.notifyObservers(this), i && (i.dispose(), (f = this.getScene()) === null || f === void 0 || f.markAllMaterialsAsDirty(1)), e && e(); }, a = (f, o) => { this._loadingError = !0, this._errorObject = { message: f, exception: o }, t && t(f, o), We.OnTextureLoadErrorObservable.notifyObservers(this); }; this._texture ? this._texture.isReady ? ye.SetImmediate(() => s()) : this._texture.onLoadedObservable.add(() => s()) : (this._prefiltered ? this._texture = this._getEngine().createPrefilteredCubeTexture(this.url, n, this._lodScale, this._lodOffset, e, a, this._format, this._forcedExtension, this._createPolynomials) : this._texture = this._getEngine().createCubeTexture(this.url, n, this._files, this._noMipmap, e, a, this._format, this._forcedExtension, !1, this._lodScale, this._lodOffset, null, this._loaderOptions, !!this._useSRGBBuffer), (r = this._texture) === null || r === void 0 || r.onLoadedObservable.add(() => this.onLoadObservable.notifyObservers(this))); } /** * Parses text to create a cube texture * @param parsedTexture define the serialized text to read from * @param scene defines the hosting scene * @param rootUrl defines the root url of the cube texture * @returns a cube texture */ static Parse(e, t, r) { const n = jt.Parse(() => { var i; let s = !1; return e.prefiltered && (s = e.prefiltered), new v1(r + ((i = e.url) !== null && i !== void 0 ? i : e.name), t, e.extensions, !1, e.files || null, null, null, void 0, s, e.forcedExtension); }, e, t); if (e.boundingBoxPosition && (n.boundingBoxPosition = S.FromArray(e.boundingBoxPosition)), e.boundingBoxSize && (n.boundingBoxSize = S.FromArray(e.boundingBoxSize)), e.animations) for (let i = 0; i < e.animations.length; i++) { const s = e.animations[i], a = Jo("BABYLON.Animation"); a && n.animations.push(a.Parse(s)); } return n; } /** * Makes a clone, or deep copy, of the cube texture * @returns a new cube texture */ clone() { let e = 0; const t = jt.Clone(() => { const r = new v1(this.url, this.getScene() || this._getEngine(), this._extensions, this._noMipmap, this._files); return e = r.uniqueId, r; }, this); return t.uniqueId = e, t; } } C([ M() ], v1.prototype, "url", void 0); C([ fo() ], v1.prototype, "boundingBoxPosition", void 0); C([ fo() ], v1.prototype, "boundingBoxSize", null); C([ M("rotationY") ], v1.prototype, "rotationY", null); C([ M("files") ], v1.prototype, "_files", void 0); C([ M("forcedExtension") ], v1.prototype, "_forcedExtension", void 0); C([ M("extensions") ], v1.prototype, "_extensions", void 0); C([ qO("textureMatrix") ], v1.prototype, "_textureMatrix", void 0); C([ qO("textureMatrixRefraction") ], v1.prototype, "_textureMatrixRefraction", void 0); We._CubeTextureParser = v1.Parse; Ue("BABYLON.CubeTexture", v1); const dPe = "backgroundFragmentDeclaration", vPe = `uniform vec4 vEyePosition;uniform vec4 vPrimaryColor; #ifdef USEHIGHLIGHTANDSHADOWCOLORS uniform vec4 vPrimaryColorShadow; #endif uniform float shadowLevel;uniform float alpha; #ifdef DIFFUSE uniform vec2 vDiffuseInfos; #endif #ifdef REFLECTION uniform vec2 vReflectionInfos;uniform mat4 reflectionMatrix;uniform vec3 vReflectionMicrosurfaceInfos; #endif #if defined(REFLECTIONFRESNEL) || defined(OPACITYFRESNEL) uniform vec3 vBackgroundCenter; #endif #ifdef REFLECTIONFRESNEL uniform vec4 vReflectionControl; #endif #if defined(REFLECTIONMAP_SPHERICAL) || defined(REFLECTIONMAP_PROJECTION) || defined(REFRACTION) uniform mat4 view; #endif #ifdef PROJECTED_GROUND uniform vec2 projectedGroundInfos; #endif `; Le.IncludesShadersStore[dPe] = vPe; const uPe = "backgroundUboDeclaration", lPe = `layout(std140,column_major) uniform;uniform Material {uniform vec4 vPrimaryColor;uniform vec4 vPrimaryColorShadow;uniform vec2 vDiffuseInfos;uniform vec2 vReflectionInfos;uniform mat4 diffuseMatrix;uniform mat4 reflectionMatrix;uniform vec3 vReflectionMicrosurfaceInfos;uniform float fFovMultiplier;uniform float pointSize;uniform float shadowLevel;uniform float alpha;uniform vec3 vBackgroundCenter;uniform vec4 vReflectionControl;uniform vec2 projectedGroundInfos;}; #include `; Le.IncludesShadersStore[uPe] = lPe; const PPe = "backgroundPixelShader", cPe = `#ifdef TEXTURELODSUPPORT #extension GL_EXT_shader_texture_lod : enable #endif precision highp float; #include<__decl__backgroundFragment> #include varying vec3 vPositionW; #ifdef MAINUV1 varying vec2 vMainUV1; #endif #ifdef MAINUV2 varying vec2 vMainUV2; #endif #ifdef NORMAL varying vec3 vNormalW; #endif #ifdef DIFFUSE #if DIFFUSEDIRECTUV==1 #define vDiffuseUV vMainUV1 #elif DIFFUSEDIRECTUV==2 #define vDiffuseUV vMainUV2 #else varying vec2 vDiffuseUV; #endif uniform sampler2D diffuseSampler; #endif #ifdef REFLECTION #ifdef REFLECTIONMAP_3D #define sampleReflection(s,c) textureCube(s,c) uniform samplerCube reflectionSampler; #ifdef TEXTURELODSUPPORT #define sampleReflectionLod(s,c,l) textureCubeLodEXT(s,c,l) #else uniform samplerCube reflectionSamplerLow;uniform samplerCube reflectionSamplerHigh; #endif #else #define sampleReflection(s,c) texture2D(s,c) uniform sampler2D reflectionSampler; #ifdef TEXTURELODSUPPORT #define sampleReflectionLod(s,c,l) texture2DLodEXT(s,c,l) #else uniform samplerCube reflectionSamplerLow;uniform samplerCube reflectionSamplerHigh; #endif #endif #ifdef REFLECTIONMAP_SKYBOX varying vec3 vPositionUVW; #else #if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED) varying vec3 vDirectionW; #endif #endif #include #endif #ifndef FROMLINEARSPACE #define FROMLINEARSPACE; #endif #ifndef SHADOWONLY #define SHADOWONLY; #endif #include #include<__decl__lightFragment>[0..maxSimultaneousLights] #include #include #include #ifdef LOGARITHMICDEPTH #extension GL_EXT_frag_depth : enable #endif #include #include #include #ifdef REFLECTIONFRESNEL #define FRESNEL_MAXIMUM_ON_ROUGH 0.25 vec3 fresnelSchlickEnvironmentGGX(float VdotN,vec3 reflectance0,vec3 reflectance90,float smoothness) {float weight=mix(FRESNEL_MAXIMUM_ON_ROUGH,1.0,smoothness);return reflectance0+weight*(reflectance90-reflectance0)*pow5(saturate(1.0-VdotN));} #endif #ifdef PROJECTED_GROUND float diskIntersectWithBackFaceCulling(vec3 ro,vec3 rd,vec3 c,float r) {float d=rd.y;if(d>0.0) { return 1e6; } vec3 o=ro-c;float t=-o.y/d;vec3 q=o+rd*t;return (dot(q,q) vec3 viewDirectionW=normalize(vEyePosition.xyz-vPositionW); #ifdef NORMAL vec3 normalW=normalize(vNormalW); #else vec3 normalW=vec3(0.0,1.0,0.0); #endif float shadow=1.;float globalShadow=0.;float shadowLightCount=0.;float aggShadow=0.;float numLights=0.; #include[0..maxSimultaneousLights] #ifdef SHADOWINUSE globalShadow/=shadowLightCount; #else globalShadow=1.0; #endif #ifndef BACKMAT_SHADOWONLY vec4 reflectionColor=vec4(1.,1.,1.,1.); #ifdef REFLECTION #ifdef PROJECTED_GROUND vec3 reflectionVector=project(viewDirectionW,vEyePosition.xyz);reflectionVector=vec3(reflectionMatrix*vec4(reflectionVector,1.)); #else vec3 reflectionVector=computeReflectionCoords(vec4(vPositionW,1.0),normalW); #endif #ifdef REFLECTIONMAP_OPPOSITEZ reflectionVector.z*=-1.0; #endif #ifdef REFLECTIONMAP_3D vec3 reflectionCoords=reflectionVector; #else vec2 reflectionCoords=reflectionVector.xy; #ifdef REFLECTIONMAP_PROJECTION reflectionCoords/=reflectionVector.z; #endif reflectionCoords.y=1.0-reflectionCoords.y; #endif #ifdef REFLECTIONBLUR float reflectionLOD=vReflectionInfos.y; #ifdef TEXTURELODSUPPORT reflectionLOD=reflectionLOD*log2(vReflectionMicrosurfaceInfos.x)*vReflectionMicrosurfaceInfos.y+vReflectionMicrosurfaceInfos.z;reflectionColor=sampleReflectionLod(reflectionSampler,reflectionCoords,reflectionLOD); #else float lodReflectionNormalized=saturate(reflectionLOD);float lodReflectionNormalizedDoubled=lodReflectionNormalized*2.0;vec4 reflectionSpecularMid=sampleReflection(reflectionSampler,reflectionCoords);if(lodReflectionNormalizedDoubled<1.0){reflectionColor=mix( sampleReflection(reflectionSamplerHigh,reflectionCoords), reflectionSpecularMid, lodReflectionNormalizedDoubled );} else {reflectionColor=mix( reflectionSpecularMid, sampleReflection(reflectionSamplerLow,reflectionCoords), lodReflectionNormalizedDoubled-1.0 );} #endif #else vec4 reflectionSample=sampleReflection(reflectionSampler,reflectionCoords);reflectionColor=reflectionSample; #endif #ifdef RGBDREFLECTION reflectionColor.rgb=fromRGBD(reflectionColor); #endif #ifdef GAMMAREFLECTION reflectionColor.rgb=toLinearSpace(reflectionColor.rgb); #endif #ifdef REFLECTIONBGR reflectionColor.rgb=reflectionColor.bgr; #endif reflectionColor.rgb*=vReflectionInfos.x; #endif vec3 diffuseColor=vec3(1.,1.,1.);float finalAlpha=alpha; #ifdef DIFFUSE vec4 diffuseMap=texture2D(diffuseSampler,vDiffuseUV); #ifdef GAMMADIFFUSE diffuseMap.rgb=toLinearSpace(diffuseMap.rgb); #endif diffuseMap.rgb*=vDiffuseInfos.y; #ifdef DIFFUSEHASALPHA finalAlpha*=diffuseMap.a; #endif diffuseColor=diffuseMap.rgb; #endif #ifdef REFLECTIONFRESNEL vec3 colorBase=diffuseColor; #else vec3 colorBase=reflectionColor.rgb*diffuseColor; #endif colorBase=max(colorBase,0.0); #ifdef USERGBCOLOR vec3 finalColor=colorBase; #else #ifdef USEHIGHLIGHTANDSHADOWCOLORS vec3 mainColor=mix(vPrimaryColorShadow.rgb,vPrimaryColor.rgb,colorBase); #else vec3 mainColor=vPrimaryColor.rgb; #endif vec3 finalColor=colorBase*mainColor; #endif #ifdef REFLECTIONFRESNEL vec3 reflectionAmount=vReflectionControl.xxx;vec3 reflectionReflectance0=vReflectionControl.yyy;vec3 reflectionReflectance90=vReflectionControl.zzz;float VdotN=dot(normalize(vEyePosition.xyz),normalW);vec3 planarReflectionFresnel=fresnelSchlickEnvironmentGGX(saturate(VdotN),reflectionReflectance0,reflectionReflectance90,1.0);reflectionAmount*=planarReflectionFresnel; #ifdef REFLECTIONFALLOFF float reflectionDistanceFalloff=1.0-saturate(length(vPositionW.xyz-vBackgroundCenter)*vReflectionControl.w);reflectionDistanceFalloff*=reflectionDistanceFalloff;reflectionAmount*=reflectionDistanceFalloff; #endif finalColor=mix(finalColor,reflectionColor.rgb,saturate(reflectionAmount)); #endif #ifdef OPACITYFRESNEL float viewAngleToFloor=dot(normalW,normalize(vEyePosition.xyz-vBackgroundCenter));const float startAngle=0.1;float fadeFactor=saturate(viewAngleToFloor/startAngle);finalAlpha*=fadeFactor*fadeFactor; #endif #ifdef SHADOWINUSE finalColor=mix(finalColor*shadowLevel,finalColor,globalShadow); #endif vec4 color=vec4(finalColor,finalAlpha); #else vec4 color=vec4(vPrimaryColor.rgb,(1.0-clamp(globalShadow,0.,1.))*alpha); #endif #include #include #ifdef IMAGEPROCESSINGPOSTPROCESS #if !defined(SKIPFINALCOLORCLAMP) color.rgb=clamp(color.rgb,0.,30.0); #endif #else color=applyImageProcessing(color); #endif #ifdef PREMULTIPLYALPHA color.rgb*=color.a; #endif #ifdef NOISE color.rgb+=dither(vPositionW.xy,0.5);color=max(color,0.0); #endif gl_FragColor=color; #define CUSTOM_FRAGMENT_MAIN_END } `; Le.ShadersStore[PPe] = cPe; const pPe = "backgroundVertexDeclaration", hPe = `uniform mat4 view;uniform mat4 viewProjection;uniform float shadowLevel; #ifdef DIFFUSE uniform mat4 diffuseMatrix;uniform vec2 vDiffuseInfos; #endif #ifdef REFLECTION uniform vec2 vReflectionInfos;uniform mat4 reflectionMatrix;uniform vec3 vReflectionMicrosurfaceInfos;uniform float fFovMultiplier; #endif #ifdef POINTSIZE uniform float pointSize; #endif `; Le.IncludesShadersStore[pPe] = hPe; const HPe = "backgroundVertexShader", gPe = `precision highp float; #include<__decl__backgroundVertex> #include attribute vec3 position; #ifdef NORMAL attribute vec3 normal; #endif #include #include #include varying vec3 vPositionW; #ifdef NORMAL varying vec3 vNormalW; #endif #ifdef UV1 attribute vec2 uv; #endif #ifdef UV2 attribute vec2 uv2; #endif #ifdef MAINUV1 varying vec2 vMainUV1; #endif #ifdef MAINUV2 varying vec2 vMainUV2; #endif #if defined(DIFFUSE) && DIFFUSEDIRECTUV==0 varying vec2 vDiffuseUV; #endif #include #include #include<__decl__lightVxFragment>[0..maxSimultaneousLights] #ifdef REFLECTIONMAP_SKYBOX varying vec3 vPositionUVW; #endif #if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED) varying vec3 vDirectionW; #endif #include #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN #ifdef REFLECTIONMAP_SKYBOX vPositionUVW=position; #endif #include #include #include #ifdef MULTIVIEW if (gl_ViewID_OVR==0u) {gl_Position=viewProjection*finalWorld*vec4(position,1.0);} else {gl_Position=viewProjectionR*finalWorld*vec4(position,1.0);} #else gl_Position=viewProjection*finalWorld*vec4(position,1.0); #endif vec4 worldPos=finalWorld*vec4(position,1.0);vPositionW=vec3(worldPos); #ifdef NORMAL mat3 normalWorld=mat3(finalWorld); #ifdef NONUNIFORMSCALING normalWorld=transposeMat3(inverseMat3(normalWorld)); #endif vNormalW=normalize(normalWorld*normal); #endif #if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED) vDirectionW=normalize(vec3(finalWorld*vec4(position,0.0))); #ifdef EQUIRECTANGULAR_RELFECTION_FOV mat3 screenToWorld=inverseMat3(mat3(finalWorld*viewProjection));vec3 segment=mix(vDirectionW,screenToWorld*vec3(0.0,0.0,1.0),abs(fFovMultiplier-1.0));if (fFovMultiplier<=1.0) {vDirectionW=normalize(segment);} else {vDirectionW=normalize(vDirectionW+(vDirectionW-segment));} #endif #endif #ifndef UV1 vec2 uv=vec2(0.,0.); #endif #ifndef UV2 vec2 uv2=vec2(0.,0.); #endif #ifdef MAINUV1 vMainUV1=uv; #endif #ifdef MAINUV2 vMainUV2=uv2; #endif #if defined(DIFFUSE) && DIFFUSEDIRECTUV==0 if (vDiffuseInfos.x==0.) {vDiffuseUV=vec2(diffuseMatrix*vec4(uv,1.0,0.0));} else {vDiffuseUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0));} #endif #include #include #include[0..maxSimultaneousLights] #ifdef VERTEXCOLOR vColor=color; #endif #if defined(POINTSIZE) && !defined(WEBGPU) gl_PointSize=pointSize; #endif #include #define CUSTOM_VERTEX_MAIN_END } `; Le.ShadersStore[HPe] = gPe; class XPe extends na { /** * Constructor of the defines. */ constructor() { super(), this.DIFFUSE = !1, this.DIFFUSEDIRECTUV = 0, this.GAMMADIFFUSE = !1, this.DIFFUSEHASALPHA = !1, this.OPACITYFRESNEL = !1, this.REFLECTIONBLUR = !1, this.REFLECTIONFRESNEL = !1, this.REFLECTIONFALLOFF = !1, this.TEXTURELODSUPPORT = !1, this.PREMULTIPLYALPHA = !1, this.USERGBCOLOR = !1, this.USEHIGHLIGHTANDSHADOWCOLORS = !1, this.BACKMAT_SHADOWONLY = !1, this.NOISE = !1, this.REFLECTIONBGR = !1, this.PROJECTED_GROUND = !1, this.IMAGEPROCESSING = !1, this.VIGNETTE = !1, this.VIGNETTEBLENDMODEMULTIPLY = !1, this.VIGNETTEBLENDMODEOPAQUE = !1, this.TONEMAPPING = !1, this.TONEMAPPING_ACES = !1, this.CONTRAST = !1, this.COLORCURVES = !1, this.COLORGRADING = !1, this.COLORGRADING3D = !1, this.SAMPLER3DGREENDEPTH = !1, this.SAMPLER3DBGRMAP = !1, this.DITHER = !1, this.IMAGEPROCESSINGPOSTPROCESS = !1, this.SKIPFINALCOLORCLAMP = !1, this.EXPOSURE = !1, this.MULTIVIEW = !1, this.REFLECTION = !1, this.REFLECTIONMAP_3D = !1, this.REFLECTIONMAP_SPHERICAL = !1, this.REFLECTIONMAP_PLANAR = !1, this.REFLECTIONMAP_CUBIC = !1, this.REFLECTIONMAP_PROJECTION = !1, this.REFLECTIONMAP_SKYBOX = !1, this.REFLECTIONMAP_EXPLICIT = !1, this.REFLECTIONMAP_EQUIRECTANGULAR = !1, this.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = !1, this.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = !1, this.INVERTCUBICMAP = !1, this.REFLECTIONMAP_OPPOSITEZ = !1, this.LODINREFLECTIONALPHA = !1, this.GAMMAREFLECTION = !1, this.RGBDREFLECTION = !1, this.EQUIRECTANGULAR_RELFECTION_FOV = !1, this.MAINUV1 = !1, this.MAINUV2 = !1, this.UV1 = !1, this.UV2 = !1, this.CLIPPLANE = !1, this.CLIPPLANE2 = !1, this.CLIPPLANE3 = !1, this.CLIPPLANE4 = !1, this.CLIPPLANE5 = !1, this.CLIPPLANE6 = !1, this.POINTSIZE = !1, this.FOG = !1, this.NORMAL = !1, this.NUM_BONE_INFLUENCERS = 0, this.BonesPerMesh = 0, this.INSTANCES = !1, this.SHADOWFLOAT = !1, this.LOGARITHMICDEPTH = !1, this.NONUNIFORMSCALING = !1, this.ALPHATEST = !1, this.rebuild(); } } class Vi extends P1 { /** * Experimental Internal Use Only. * * Key light Color in "perceptual value" meaning the color you would like to see on screen. * This acts as a helper to set the primary color to a more "human friendly" value. * Conversion to linear space as well as exposure and tone mapping correction will be applied to keep the * output color as close as possible from the chosen value. * (This does not account for contrast color grading and color curves as they are considered post effect and not directly * part of lighting setup.) */ get _perceptualColor() { return this.__perceptualColor; } set _perceptualColor(e) { this.__perceptualColor = e, this._computePrimaryColorFromPerceptualColor(), this._markAllSubMeshesAsLightsDirty(); } /** * Defines the level of the shadows (dark area of the reflection map) in order to help scaling the colors. * The color opposite to the primary color is used at the level chosen to define what the black area would look. */ get primaryColorShadowLevel() { return this._primaryColorShadowLevel; } set primaryColorShadowLevel(e) { this._primaryColorShadowLevel = e, this._computePrimaryColors(), this._markAllSubMeshesAsLightsDirty(); } /** * Defines the level of the highlights (highlight area of the reflection map) in order to help scaling the colors. * The primary color is used at the level chosen to define what the white area would look. */ get primaryColorHighlightLevel() { return this._primaryColorHighlightLevel; } set primaryColorHighlightLevel(e) { this._primaryColorHighlightLevel = e, this._computePrimaryColors(), this._markAllSubMeshesAsLightsDirty(); } /** * Sets the reflection reflectance fresnel values according to the default standard * empirically know to work well :-) */ set reflectionStandardFresnelWeight(e) { let t = e; t < 0.5 ? (t = t * 2, this.reflectionReflectance0 = Vi.StandardReflectance0 * t, this.reflectionReflectance90 = Vi.StandardReflectance90 * t) : (t = t * 2 - 1, this.reflectionReflectance0 = Vi.StandardReflectance0 + (1 - Vi.StandardReflectance0) * t, this.reflectionReflectance90 = Vi.StandardReflectance90 + (1 - Vi.StandardReflectance90) * t); } /** * The current fov(field of view) multiplier, 0.0 - 2.0. Defaults to 1.0. Lower values "zoom in" and higher values "zoom out". * Best used when trying to implement visual zoom effects like fish-eye or binoculars while not adjusting camera fov. * Recommended to be keep at 1.0 except for special cases. */ get fovMultiplier() { return this._fovMultiplier; } set fovMultiplier(e) { isNaN(e) && (e = 1), this._fovMultiplier = Math.max(0, Math.min(2, e)); } /** * Attaches a new image processing configuration to the PBR Material. * @param configuration (if null the scene configuration will be use) */ _attachImageProcessingConfiguration(e) { e !== this._imageProcessingConfiguration && (this._imageProcessingConfiguration && this._imageProcessingObserver && this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver), e ? this._imageProcessingConfiguration = e : this._imageProcessingConfiguration = this.getScene().imageProcessingConfiguration, this._imageProcessingConfiguration && (this._imageProcessingObserver = this._imageProcessingConfiguration.onUpdateParameters.add(() => { this._computePrimaryColorFromPerceptualColor(), this._markAllSubMeshesAsImageProcessingDirty(); }))); } /** * Gets the image processing configuration used either in this material. */ get imageProcessingConfiguration() { return this._imageProcessingConfiguration; } /** * Sets the Default image processing configuration used either in the this material. * * If sets to null, the scene one is in use. */ set imageProcessingConfiguration(e) { this._attachImageProcessingConfiguration(e), this._markAllSubMeshesAsTexturesDirty(); } /** * Gets whether the color curves effect is enabled. */ get cameraColorCurvesEnabled() { return this.imageProcessingConfiguration.colorCurvesEnabled; } /** * Sets whether the color curves effect is enabled. */ set cameraColorCurvesEnabled(e) { this.imageProcessingConfiguration.colorCurvesEnabled = e; } /** * Gets whether the color grading effect is enabled. */ get cameraColorGradingEnabled() { return this.imageProcessingConfiguration.colorGradingEnabled; } /** * Gets whether the color grading effect is enabled. */ set cameraColorGradingEnabled(e) { this.imageProcessingConfiguration.colorGradingEnabled = e; } /** * Gets whether tonemapping is enabled or not. */ get cameraToneMappingEnabled() { return this._imageProcessingConfiguration.toneMappingEnabled; } /** * Sets whether tonemapping is enabled or not */ set cameraToneMappingEnabled(e) { this._imageProcessingConfiguration.toneMappingEnabled = e; } /** * The camera exposure used on this material. * This property is here and not in the camera to allow controlling exposure without full screen post process. * This corresponds to a photographic exposure. */ get cameraExposure() { return this._imageProcessingConfiguration.exposure; } /** * The camera exposure used on this material. * This property is here and not in the camera to allow controlling exposure without full screen post process. * This corresponds to a photographic exposure. */ set cameraExposure(e) { this._imageProcessingConfiguration.exposure = e; } /** * Gets The camera contrast used on this material. */ get cameraContrast() { return this._imageProcessingConfiguration.contrast; } /** * Sets The camera contrast used on this material. */ set cameraContrast(e) { this._imageProcessingConfiguration.contrast = e; } /** * Gets the Color Grading 2D Lookup Texture. */ get cameraColorGradingTexture() { return this._imageProcessingConfiguration.colorGradingTexture; } /** * Sets the Color Grading 2D Lookup Texture. */ set cameraColorGradingTexture(e) { this.imageProcessingConfiguration.colorGradingTexture = e; } /** * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT). * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects. * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image; * corresponding to low luminance, medium luminance, and high luminance areas respectively. */ get cameraColorCurves() { return this.imageProcessingConfiguration.colorCurves; } /** * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT). * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects. * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image; * corresponding to low luminance, medium luminance, and high luminance areas respectively. */ set cameraColorCurves(e) { this.imageProcessingConfiguration.colorCurves = e; } /** * Instantiates a Background Material in the given scene * @param name The friendly name of the material * @param scene The scene to add the material to */ constructor(e, t) { super(e, t), this.primaryColor = Ne.White(), this._primaryColorShadowLevel = 0, this._primaryColorHighlightLevel = 0, this.reflectionTexture = null, this.reflectionBlur = 0, this.diffuseTexture = null, this._shadowLights = null, this.shadowLights = null, this.shadowLevel = 0, this.sceneCenter = S.Zero(), this.opacityFresnel = !0, this.reflectionFresnel = !1, this.reflectionFalloffDistance = 0, this.reflectionAmount = 1, this.reflectionReflectance0 = 0.05, this.reflectionReflectance90 = 0.5, this.useRGBColor = !0, this.enableNoise = !1, this._fovMultiplier = 1, this.useEquirectangularFOV = !1, this._maxSimultaneousLights = 4, this.maxSimultaneousLights = 4, this._shadowOnly = !1, this.shadowOnly = !1, this._imageProcessingObserver = null, this.switchToBGR = !1, this._enableGroundProjection = !1, this.enableGroundProjection = !1, this.projectedGroundRadius = 1e3, this.projectedGroundHeight = 10, this._renderTargets = new qf(16), this._reflectionControls = Ir.Zero(), this._white = Ne.White(), this._primaryShadowColor = Ne.Black(), this._primaryHighlightColor = Ne.Black(), this._attachImageProcessingConfiguration(null), this.getRenderTargetTextures = () => (this._renderTargets.reset(), this._diffuseTexture && this._diffuseTexture.isRenderTarget && this._renderTargets.push(this._diffuseTexture), this._reflectionTexture && this._reflectionTexture.isRenderTarget && this._renderTargets.push(this._reflectionTexture), this._renderTargets); } /** * Gets a boolean indicating that current material needs to register RTT */ get hasRenderTargetTextures() { return !!(this._diffuseTexture && this._diffuseTexture.isRenderTarget || this._reflectionTexture && this._reflectionTexture.isRenderTarget); } /** * The entire material has been created in order to prevent overdraw. * @returns false */ needAlphaTesting() { return !0; } /** * The entire material has been created in order to prevent overdraw. * @returns true if blending is enable */ needAlphaBlending() { return this.alpha < 1 || this._diffuseTexture != null && this._diffuseTexture.hasAlpha || this._shadowOnly; } /** * Checks whether the material is ready to be rendered for a given mesh. * @param mesh The mesh to render * @param subMesh The submesh to check against * @param useInstances Specify wether or not the material is used with instances * @returns true if all the dependencies are ready (Textures, Effects...) */ isReadyForSubMesh(e, t, r = !1) { if (t.effect && this.isFrozen && t.effect._wasPreviouslyReady && t.effect._wasPreviouslyUsingInstances === r) return !0; t.materialDefines || (t.materialDefines = new XPe()); const n = this.getScene(), i = t.materialDefines; if (this._isReadyForSubMesh(t)) return !0; const s = n.getEngine(); if (Ye.PrepareDefinesForLights(n, e, i, !1, this._maxSimultaneousLights), i._needNormals = !0, Ye.PrepareDefinesForMultiview(n, i), i._areTexturesDirty) { if (i._needUVs = !1, n.texturesEnabled) { if (n.getEngine().getCaps().textureLOD && (i.TEXTURELODSUPPORT = !0), this._diffuseTexture && Dt.DiffuseTextureEnabled) { if (!this._diffuseTexture.isReadyOrNotBlocking()) return !1; Ye.PrepareDefinesForMergedUV(this._diffuseTexture, i, "DIFFUSE"), i.DIFFUSEHASALPHA = this._diffuseTexture.hasAlpha, i.GAMMADIFFUSE = this._diffuseTexture.gammaSpace, i.OPACITYFRESNEL = this._opacityFresnel; } else i.DIFFUSE = !1, i.DIFFUSEDIRECTUV = 0, i.DIFFUSEHASALPHA = !1, i.GAMMADIFFUSE = !1, i.OPACITYFRESNEL = !1; const a = this._reflectionTexture; if (a && Dt.ReflectionTextureEnabled) { if (!a.isReadyOrNotBlocking()) return !1; switch (i.REFLECTION = !0, i.GAMMAREFLECTION = a.gammaSpace, i.RGBDREFLECTION = a.isRGBD, i.REFLECTIONBLUR = this._reflectionBlur > 0, i.LODINREFLECTIONALPHA = a.lodLevelInAlpha, i.EQUIRECTANGULAR_RELFECTION_FOV = this.useEquirectangularFOV, i.REFLECTIONBGR = this.switchToBGR, a.coordinatesMode === We.INVCUBIC_MODE && (i.INVERTCUBICMAP = !0), i.REFLECTIONMAP_3D = a.isCube, i.REFLECTIONMAP_OPPOSITEZ = i.REFLECTIONMAP_3D && this.getScene().useRightHandedSystem ? !a.invertZ : a.invertZ, a.coordinatesMode) { case We.EXPLICIT_MODE: i.REFLECTIONMAP_EXPLICIT = !0; break; case We.PLANAR_MODE: i.REFLECTIONMAP_PLANAR = !0; break; case We.PROJECTION_MODE: i.REFLECTIONMAP_PROJECTION = !0; break; case We.SKYBOX_MODE: i.REFLECTIONMAP_SKYBOX = !0; break; case We.SPHERICAL_MODE: i.REFLECTIONMAP_SPHERICAL = !0; break; case We.EQUIRECTANGULAR_MODE: i.REFLECTIONMAP_EQUIRECTANGULAR = !0; break; case We.FIXED_EQUIRECTANGULAR_MODE: i.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = !0; break; case We.FIXED_EQUIRECTANGULAR_MIRRORED_MODE: i.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = !0; break; case We.CUBIC_MODE: case We.INVCUBIC_MODE: default: i.REFLECTIONMAP_CUBIC = !0; break; } this.reflectionFresnel ? (i.REFLECTIONFRESNEL = !0, i.REFLECTIONFALLOFF = this.reflectionFalloffDistance > 0, this._reflectionControls.x = this.reflectionAmount, this._reflectionControls.y = this.reflectionReflectance0, this._reflectionControls.z = this.reflectionReflectance90, this._reflectionControls.w = 1 / this.reflectionFalloffDistance) : (i.REFLECTIONFRESNEL = !1, i.REFLECTIONFALLOFF = !1); } else i.REFLECTION = !1, i.REFLECTIONFRESNEL = !1, i.REFLECTIONFALLOFF = !1, i.REFLECTIONBLUR = !1, i.REFLECTIONMAP_3D = !1, i.REFLECTIONMAP_SPHERICAL = !1, i.REFLECTIONMAP_PLANAR = !1, i.REFLECTIONMAP_CUBIC = !1, i.REFLECTIONMAP_PROJECTION = !1, i.REFLECTIONMAP_SKYBOX = !1, i.REFLECTIONMAP_EXPLICIT = !1, i.REFLECTIONMAP_EQUIRECTANGULAR = !1, i.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = !1, i.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = !1, i.INVERTCUBICMAP = !1, i.REFLECTIONMAP_OPPOSITEZ = !1, i.LODINREFLECTIONALPHA = !1, i.GAMMAREFLECTION = !1, i.RGBDREFLECTION = !1; } i.PREMULTIPLYALPHA = this.alphaMode === 7 || this.alphaMode === 8, i.USERGBCOLOR = this._useRGBColor, i.NOISE = this._enableNoise; } if (i._areLightsDirty && (i.USEHIGHLIGHTANDSHADOWCOLORS = !this._useRGBColor && (this._primaryColorShadowLevel !== 0 || this._primaryColorHighlightLevel !== 0), i.BACKMAT_SHADOWONLY = this._shadowOnly), i._areImageProcessingDirty && this._imageProcessingConfiguration) { if (!this._imageProcessingConfiguration.isReady()) return !1; this._imageProcessingConfiguration.prepareDefines(i); } if (i._areMiscDirty && (i.REFLECTIONMAP_3D && this._enableGroundProjection ? (i.PROJECTED_GROUND = !0, i.REFLECTIONMAP_SKYBOX = !0) : i.PROJECTED_GROUND = !1), Ye.PrepareDefinesForMisc(e, n, this._useLogarithmicDepth, this.pointsCloud, this.fogEnabled, this._shouldTurnAlphaTestOn(e), i), Ye.PrepareDefinesForFrameBoundValues(n, s, this, i, r, null, t.getRenderingMesh().hasThinInstances), Ye.PrepareDefinesForAttributes(e, i, !1, !0, !1) && e && !n.getEngine().getCaps().standardDerivatives && !e.isVerticesDataPresent(J.NormalKind) && (e.createNormals(!0), Se.Warn("BackgroundMaterial: Normals have been created for the mesh: " + e.name)), i.isDirty) { i.markAsProcessed(), n.resetCachedMaterial(); const a = new c1(); i.FOG && a.addFallback(0, "FOG"), i.POINTSIZE && a.addFallback(1, "POINTSIZE"), i.MULTIVIEW && a.addFallback(0, "MULTIVIEW"), Ye.HandleFallbacksForShadows(i, a, this._maxSimultaneousLights); const f = [J.PositionKind]; i.NORMAL && f.push(J.NormalKind), i.UV1 && f.push(J.UVKind), i.UV2 && f.push(J.UV2Kind), Ye.PrepareAttributesForBones(f, e, i, a), Ye.PrepareAttributesForInstances(f, i); const o = [ "world", "view", "viewProjection", "vEyePosition", "vLightsType", "vFogInfos", "vFogColor", "pointSize", "mBones", "vPrimaryColor", "vPrimaryColorShadow", "vReflectionInfos", "reflectionMatrix", "vReflectionMicrosurfaceInfos", "fFovMultiplier", "shadowLevel", "alpha", "vBackgroundCenter", "vReflectionControl", "vDiffuseInfos", "diffuseMatrix", "projectedGroundInfos", "logarithmicDepthConstant" ]; Mf(o); const d = ["diffuseSampler", "reflectionSampler", "reflectionSamplerLow", "reflectionSamplerHigh"], v = ["Material", "Scene"]; Ui && (Ui.PrepareUniforms(o, i), Ui.PrepareSamplers(d, i)), Ye.PrepareUniformsAndSamplersList({ uniformsNames: o, uniformBuffersNames: v, samplers: d, defines: i, maxSimultaneousLights: this._maxSimultaneousLights }); const u = i.toString(), l = n.getEngine().createEffect("background", { attributes: f, uniformsNames: o, uniformBuffersNames: v, samplers: d, defines: u, fallbacks: a, onCompiled: this.onCompiled, onError: this.onError, indexParameters: { maxSimultaneousLights: this._maxSimultaneousLights } }, s); t.setEffect(l, i, this._materialContext), this.buildUniformLayout(); } return !t.effect || !t.effect.isReady() ? !1 : (i._renderId = n.getRenderId(), t.effect._wasPreviouslyReady = !0, t.effect._wasPreviouslyUsingInstances = r, this._checkScenePerformancePriority(), !0); } /** * Compute the primary color according to the chosen perceptual color. */ _computePrimaryColorFromPerceptualColor() { this.__perceptualColor && (this._primaryColor.copyFrom(this.__perceptualColor), this._primaryColor.toLinearSpaceToRef(this._primaryColor, this.getScene().getEngine().useExactSrgbConversions), this._imageProcessingConfiguration && this._primaryColor.scaleToRef(1 / this._imageProcessingConfiguration.exposure, this._primaryColor), this._computePrimaryColors()); } /** * Compute the highlights and shadow colors according to their chosen levels. */ _computePrimaryColors() { this._primaryColorShadowLevel === 0 && this._primaryColorHighlightLevel === 0 || (this._primaryColor.scaleToRef(this._primaryColorShadowLevel, this._primaryShadowColor), this._primaryColor.subtractToRef(this._primaryShadowColor, this._primaryShadowColor), this._white.subtractToRef(this._primaryColor, this._primaryHighlightColor), this._primaryHighlightColor.scaleToRef(this._primaryColorHighlightLevel, this._primaryHighlightColor), this._primaryColor.addToRef(this._primaryHighlightColor, this._primaryHighlightColor)); } /** * Build the uniform buffer used in the material. */ buildUniformLayout() { this._uniformBuffer.addUniform("vPrimaryColor", 4), this._uniformBuffer.addUniform("vPrimaryColorShadow", 4), this._uniformBuffer.addUniform("vDiffuseInfos", 2), this._uniformBuffer.addUniform("vReflectionInfos", 2), this._uniformBuffer.addUniform("diffuseMatrix", 16), this._uniformBuffer.addUniform("reflectionMatrix", 16), this._uniformBuffer.addUniform("vReflectionMicrosurfaceInfos", 3), this._uniformBuffer.addUniform("fFovMultiplier", 1), this._uniformBuffer.addUniform("pointSize", 1), this._uniformBuffer.addUniform("shadowLevel", 1), this._uniformBuffer.addUniform("alpha", 1), this._uniformBuffer.addUniform("vBackgroundCenter", 3), this._uniformBuffer.addUniform("vReflectionControl", 4), this._uniformBuffer.addUniform("projectedGroundInfos", 2), this._uniformBuffer.create(); } /** * Unbind the material. */ unbind() { this._diffuseTexture && this._diffuseTexture.isRenderTarget && this._uniformBuffer.setTexture("diffuseSampler", null), this._reflectionTexture && this._reflectionTexture.isRenderTarget && this._uniformBuffer.setTexture("reflectionSampler", null), super.unbind(); } /** * Bind only the world matrix to the material. * @param world The world matrix to bind. */ bindOnlyWorldMatrix(e) { this._activeEffect.setMatrix("world", e); } /** * Bind the material for a dedicated submeh (every used meshes will be considered opaque). * @param world The world matrix to bind. * @param mesh * @param subMesh The submesh to bind for. */ bindForSubMesh(e, t, r) { const n = this.getScene(), i = r.materialDefines; if (!i) return; const s = r.effect; if (!s) return; this._activeEffect = s, this.bindOnlyWorldMatrix(e), Ye.BindBonesParameters(t, this._activeEffect); const a = this._mustRebind(n, s, t.visibility); if (a) { this._uniformBuffer.bindToEffect(s, "Material"), this.bindViewProjection(s); const f = this._reflectionTexture; (!this._uniformBuffer.useUbo || !this.isFrozen || !this._uniformBuffer.isSync) && (n.texturesEnabled && (this._diffuseTexture && Dt.DiffuseTextureEnabled && (this._uniformBuffer.updateFloat2("vDiffuseInfos", this._diffuseTexture.coordinatesIndex, this._diffuseTexture.level), Ye.BindTextureMatrix(this._diffuseTexture, this._uniformBuffer, "diffuse")), f && Dt.ReflectionTextureEnabled && (this._uniformBuffer.updateMatrix("reflectionMatrix", f.getReflectionTextureMatrix()), this._uniformBuffer.updateFloat2("vReflectionInfos", f.level, this._reflectionBlur), this._uniformBuffer.updateFloat3("vReflectionMicrosurfaceInfos", f.getSize().width, f.lodGenerationScale, f.lodGenerationOffset))), this.shadowLevel > 0 && this._uniformBuffer.updateFloat("shadowLevel", this.shadowLevel), this._uniformBuffer.updateFloat("alpha", this.alpha), this.pointsCloud && this._uniformBuffer.updateFloat("pointSize", this.pointSize), i.USEHIGHLIGHTANDSHADOWCOLORS ? (this._uniformBuffer.updateColor4("vPrimaryColor", this._primaryHighlightColor, 1), this._uniformBuffer.updateColor4("vPrimaryColorShadow", this._primaryShadowColor, 1)) : this._uniformBuffer.updateColor4("vPrimaryColor", this._primaryColor, 1)), this._uniformBuffer.updateFloat("fFovMultiplier", this._fovMultiplier), n.texturesEnabled && (this._diffuseTexture && Dt.DiffuseTextureEnabled && this._uniformBuffer.setTexture("diffuseSampler", this._diffuseTexture), f && Dt.ReflectionTextureEnabled && (i.REFLECTIONBLUR && i.TEXTURELODSUPPORT ? this._uniformBuffer.setTexture("reflectionSampler", f) : i.REFLECTIONBLUR ? (this._uniformBuffer.setTexture("reflectionSampler", f._lodTextureMid || f), this._uniformBuffer.setTexture("reflectionSamplerLow", f._lodTextureLow || f), this._uniformBuffer.setTexture("reflectionSamplerHigh", f._lodTextureHigh || f)) : this._uniformBuffer.setTexture("reflectionSampler", f), i.REFLECTIONFRESNEL && (this._uniformBuffer.updateFloat3("vBackgroundCenter", this.sceneCenter.x, this.sceneCenter.y, this.sceneCenter.z), this._uniformBuffer.updateFloat4("vReflectionControl", this._reflectionControls.x, this._reflectionControls.y, this._reflectionControls.z, this._reflectionControls.w))), i.PROJECTED_GROUND && this._uniformBuffer.updateFloat2("projectedGroundInfos", this.projectedGroundRadius, this.projectedGroundHeight)), Df(this._activeEffect, this, n), n.bindEyePosition(s); } else n.getEngine()._features.needToAlwaysBindUniformBuffers && (this._uniformBuffer.bindToEffect(s, "Material"), this._needToBindSceneUbo = !0); (a || !this.isFrozen) && (n.lightsEnabled && Ye.BindLights(n, t, this._activeEffect, i, this._maxSimultaneousLights), this.bindView(s), Ye.BindFogParameters(n, t, this._activeEffect, !0), this._useLogarithmicDepth && Ye.BindLogDepth(i, s, n), this._imageProcessingConfiguration && this._imageProcessingConfiguration.bind(this._activeEffect)), this._afterBind(t, this._activeEffect), this._uniformBuffer.update(); } /** * Checks to see if a texture is used in the material. * @param texture - Base texture to use. * @returns - Boolean specifying if a texture is used in the material. */ hasTexture(e) { return !!(super.hasTexture(e) || this._reflectionTexture === e || this._diffuseTexture === e); } /** * Dispose the material. * @param forceDisposeEffect Force disposal of the associated effect. * @param forceDisposeTextures Force disposal of the associated textures. */ dispose(e = !1, t = !1) { t && (this.diffuseTexture && this.diffuseTexture.dispose(), this.reflectionTexture && this.reflectionTexture.dispose()), this._renderTargets.dispose(), this._imageProcessingConfiguration && this._imageProcessingObserver && this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver), super.dispose(e); } /** * Clones the material. * @param name The cloned name. * @returns The cloned material. */ clone(e) { return jt.Clone(() => new Vi(e, this.getScene()), this); } /** * Serializes the current material to its JSON representation. * @returns The JSON representation. */ serialize() { const e = super.serialize(); return e.customType = "BABYLON.BackgroundMaterial", e; } /** * Gets the class name of the material * @returns "BackgroundMaterial" */ getClassName() { return "BackgroundMaterial"; } /** * Parse a JSON input to create back a background material. * @param source The JSON data to parse * @param scene The scene to create the parsed material in * @param rootUrl The root url of the assets the material depends upon * @returns the instantiated BackgroundMaterial. */ static Parse(e, t, r) { return jt.Parse(() => new Vi(e.name, t), e, t, r); } } Vi.StandardReflectance0 = 0.05; Vi.StandardReflectance90 = 0.5; C([ Oi() ], Vi.prototype, "_primaryColor", void 0); C([ At("_markAllSubMeshesAsLightsDirty") ], Vi.prototype, "primaryColor", void 0); C([ Oi() ], Vi.prototype, "__perceptualColor", void 0); C([ M() ], Vi.prototype, "_primaryColorShadowLevel", void 0); C([ M() ], Vi.prototype, "_primaryColorHighlightLevel", void 0); C([ At("_markAllSubMeshesAsLightsDirty") ], Vi.prototype, "primaryColorHighlightLevel", null); C([ en() ], Vi.prototype, "_reflectionTexture", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Vi.prototype, "reflectionTexture", void 0); C([ M() ], Vi.prototype, "_reflectionBlur", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Vi.prototype, "reflectionBlur", void 0); C([ en() ], Vi.prototype, "_diffuseTexture", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Vi.prototype, "diffuseTexture", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Vi.prototype, "shadowLights", void 0); C([ M() ], Vi.prototype, "_shadowLevel", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Vi.prototype, "shadowLevel", void 0); C([ fo() ], Vi.prototype, "_sceneCenter", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Vi.prototype, "sceneCenter", void 0); C([ M() ], Vi.prototype, "_opacityFresnel", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Vi.prototype, "opacityFresnel", void 0); C([ M() ], Vi.prototype, "_reflectionFresnel", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Vi.prototype, "reflectionFresnel", void 0); C([ M() ], Vi.prototype, "_reflectionFalloffDistance", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Vi.prototype, "reflectionFalloffDistance", void 0); C([ M() ], Vi.prototype, "_reflectionAmount", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Vi.prototype, "reflectionAmount", void 0); C([ M() ], Vi.prototype, "_reflectionReflectance0", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Vi.prototype, "reflectionReflectance0", void 0); C([ M() ], Vi.prototype, "_reflectionReflectance90", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Vi.prototype, "reflectionReflectance90", void 0); C([ M() ], Vi.prototype, "_useRGBColor", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Vi.prototype, "useRGBColor", void 0); C([ M() ], Vi.prototype, "_enableNoise", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Vi.prototype, "enableNoise", void 0); C([ M() ], Vi.prototype, "_maxSimultaneousLights", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], Vi.prototype, "maxSimultaneousLights", void 0); C([ M() ], Vi.prototype, "_shadowOnly", void 0); C([ At("_markAllSubMeshesAsLightsDirty") ], Vi.prototype, "shadowOnly", void 0); C([ CN() ], Vi.prototype, "_imageProcessingConfiguration", void 0); C([ M(), At("_markAllSubMeshesAsMiscDirty") ], Vi.prototype, "enableGroundProjection", void 0); C([ M() ], Vi.prototype, "projectedGroundRadius", void 0); C([ M() ], Vi.prototype, "projectedGroundHeight", void 0); Ue("BABYLON.BackgroundMaterial", Vi); class gD { /** * Creates the default options for the helper. * @param scene The scene the environment helper belongs to. */ static _GetDefaultOptions(e) { return { createGround: !0, groundSize: 15, groundTexture: this._GroundTextureCDNUrl, groundColor: new Ne(0.2, 0.2, 0.3).toLinearSpace(e.getEngine().useExactSrgbConversions).scale(3), groundOpacity: 0.9, enableGroundShadow: !0, groundShadowLevel: 0.5, enableGroundMirror: !1, groundMirrorSizeRatio: 0.3, groundMirrorBlurKernel: 64, groundMirrorAmount: 1, groundMirrorFresnelWeight: 1, groundMirrorFallOffDistance: 0, groundMirrorTextureType: 0, groundYBias: 1e-5, createSkybox: !0, skyboxSize: 20, skyboxTexture: this._SkyboxTextureCDNUrl, skyboxColor: new Ne(0.2, 0.2, 0.3).toLinearSpace(e.getEngine().useExactSrgbConversions).scale(3), backgroundYRotation: 0, sizeAuto: !0, rootPosition: S.Zero(), setupImageProcessing: !0, environmentTexture: this._EnvironmentTextureCDNUrl, cameraExposure: 0.8, cameraContrast: 1.2, toneMappingEnabled: !0 }; } /** * Gets the root mesh created by the helper. */ get rootMesh() { return this._rootMesh; } /** * Gets the skybox created by the helper. */ get skybox() { return this._skybox; } /** * Gets the skybox texture created by the helper. */ get skyboxTexture() { return this._skyboxTexture; } /** * Gets the skybox material created by the helper. */ get skyboxMaterial() { return this._skyboxMaterial; } /** * Gets the ground mesh created by the helper. */ get ground() { return this._ground; } /** * Gets the ground texture created by the helper. */ get groundTexture() { return this._groundTexture; } /** * Gets the ground mirror created by the helper. */ get groundMirror() { return this._groundMirror; } /** * Gets the ground mirror render list to helps pushing the meshes * you wish in the ground reflection. */ get groundMirrorRenderList() { return this._groundMirror ? this._groundMirror.renderList : null; } /** * Gets the ground material created by the helper. */ get groundMaterial() { return this._groundMaterial; } /** * constructor * @param options Defines the options we want to customize the helper * @param scene The scene to add the material to */ constructor(e, t) { this._errorHandler = (r, n) => { this.onErrorObservable.notifyObservers({ message: r, exception: n }); }, this._options = Object.assign(Object.assign({}, gD._GetDefaultOptions(t)), e), this._scene = t, this.onErrorObservable = new Oe(), this._setupBackground(), this._setupImageProcessing(); } /** * Updates the background according to the new options * @param options */ updateOptions(e) { const t = Object.assign(Object.assign({}, this._options), e); this._ground && !t.createGround && (this._ground.dispose(), this._ground = null), this._groundMaterial && !t.createGround && (this._groundMaterial.dispose(), this._groundMaterial = null), this._groundTexture && this._options.groundTexture != t.groundTexture && (this._groundTexture.dispose(), this._groundTexture = null), this._skybox && !t.createSkybox && (this._skybox.dispose(), this._skybox = null), this._skyboxMaterial && !t.createSkybox && (this._skyboxMaterial.dispose(), this._skyboxMaterial = null), this._skyboxTexture && this._options.skyboxTexture != t.skyboxTexture && (this._skyboxTexture.dispose(), this._skyboxTexture = null), this._groundMirror && !t.enableGroundMirror && (this._groundMirror.dispose(), this._groundMirror = null), this._scene.environmentTexture && this._options.environmentTexture != t.environmentTexture && this._scene.environmentTexture.dispose(), this._options = t, this._setupBackground(), this._setupImageProcessing(); } /** * Sets the primary color of all the available elements. * @param color the main color to affect to the ground and the background */ setMainColor(e) { this.groundMaterial && (this.groundMaterial.primaryColor = e), this.skyboxMaterial && (this.skyboxMaterial.primaryColor = e), this.groundMirror && (this.groundMirror.clearColor = new xt(e.r, e.g, e.b, 1)); } /** * Setup the image processing according to the specified options. */ _setupImageProcessing() { this._options.setupImageProcessing && (this._scene.imageProcessingConfiguration.contrast = this._options.cameraContrast, this._scene.imageProcessingConfiguration.exposure = this._options.cameraExposure, this._scene.imageProcessingConfiguration.toneMappingEnabled = this._options.toneMappingEnabled, this._setupEnvironmentTexture()); } /** * Setup the environment texture according to the specified options. */ _setupEnvironmentTexture() { if (this._scene.environmentTexture) return; if (this._options.environmentTexture instanceof ls) { this._scene.environmentTexture = this._options.environmentTexture; return; } const e = v1.CreateFromPrefilteredData(this._options.environmentTexture, this._scene); this._scene.environmentTexture = e; } /** * Setup the background according to the specified options. */ _setupBackground() { this._rootMesh || (this._rootMesh = new Ee("BackgroundHelper", this._scene)), this._rootMesh.rotation.y = this._options.backgroundYRotation; const e = this._getSceneSize(); this._options.createGround && (this._setupGround(e), this._setupGroundMaterial(), this._setupGroundDiffuseTexture(), this._options.enableGroundMirror && this._setupGroundMirrorTexture(e), this._setupMirrorInGroundMaterial()), this._options.createSkybox && (this._setupSkybox(e), this._setupSkyboxMaterial(), this._setupSkyboxReflectionTexture()), this._rootMesh.position.x = e.rootPosition.x, this._rootMesh.position.z = e.rootPosition.z, this._rootMesh.position.y = e.rootPosition.y; } /** * Get the scene sizes according to the setup. */ _getSceneSize() { let e = this._options.groundSize, t = this._options.skyboxSize, r = this._options.rootPosition; if (!this._scene.meshes || this._scene.meshes.length === 1) return { groundSize: e, skyboxSize: t, rootPosition: r }; const n = this._scene.getWorldExtends((s) => s !== this._ground && s !== this._rootMesh && s !== this._skybox), i = n.max.subtract(n.min); if (this._options.sizeAuto) { this._scene.activeCamera instanceof ps && this._scene.activeCamera.upperRadiusLimit && (e = this._scene.activeCamera.upperRadiusLimit * 2, t = e); const s = i.length(); s > e && (e = s * 2, t = e), e *= 1.1, t *= 1.5, r = n.min.add(i.scale(0.5)), r.y = n.min.y - this._options.groundYBias; } return { groundSize: e, skyboxSize: t, rootPosition: r }; } /** * Setup the ground according to the specified options. * @param sceneSize */ _setupGround(e) { (!this._ground || this._ground.isDisposed()) && (this._ground = u4("BackgroundPlane", { size: e.groundSize }, this._scene), this._ground.rotation.x = Math.PI / 2, this._ground.parent = this._rootMesh, this._ground.onDisposeObservable.add(() => { this._ground = null; })), this._ground.receiveShadows = this._options.enableGroundShadow; } /** * Setup the ground material according to the specified options. */ _setupGroundMaterial() { this._groundMaterial || (this._groundMaterial = new Vi("BackgroundPlaneMaterial", this._scene)), this._groundMaterial.alpha = this._options.groundOpacity, this._groundMaterial.alphaMode = 8, this._groundMaterial.shadowLevel = this._options.groundShadowLevel, this._groundMaterial.primaryColor = this._options.groundColor, this._groundMaterial.useRGBColor = !1, this._groundMaterial.enableNoise = !0, this._ground && (this._ground.material = this._groundMaterial); } /** * Setup the ground diffuse texture according to the specified options. */ _setupGroundDiffuseTexture() { if (this._groundMaterial && !this._groundTexture) { if (this._options.groundTexture instanceof ls) { this._groundMaterial.diffuseTexture = this._options.groundTexture; return; } this._groundTexture = new We(this._options.groundTexture, this._scene, void 0, void 0, void 0, void 0, this._errorHandler), this._groundTexture.gammaSpace = !1, this._groundTexture.hasAlpha = !0, this._groundMaterial.diffuseTexture = this._groundTexture; } } /** * Setup the ground mirror texture according to the specified options. * @param sceneSize */ _setupGroundMirrorTexture(e) { const t = We.CLAMP_ADDRESSMODE; if (!this._groundMirror && (this._groundMirror = new uD("BackgroundPlaneMirrorTexture", { ratio: this._options.groundMirrorSizeRatio }, this._scene, !1, this._options.groundMirrorTextureType, We.BILINEAR_SAMPLINGMODE, !0), this._groundMirror.mirrorPlane = new BA(0, -1, 0, e.rootPosition.y), this._groundMirror.anisotropicFilteringLevel = 1, this._groundMirror.wrapU = t, this._groundMirror.wrapV = t, this._groundMirror.renderList)) for (let n = 0; n < this._scene.meshes.length; n++) { const i = this._scene.meshes[n]; i !== this._ground && i !== this._skybox && i !== this._rootMesh && this._groundMirror.renderList.push(i); } const r = this._options.groundColor.toGammaSpace(this._scene.getEngine().useExactSrgbConversions); this._groundMirror.clearColor = new xt(r.r, r.g, r.b, 1), this._groundMirror.adaptiveBlurKernel = this._options.groundMirrorBlurKernel; } /** * Setup the ground to receive the mirror texture. */ _setupMirrorInGroundMaterial() { this._groundMaterial && (this._groundMaterial.reflectionTexture = this._groundMirror, this._groundMaterial.reflectionFresnel = !0, this._groundMaterial.reflectionAmount = this._options.groundMirrorAmount, this._groundMaterial.reflectionStandardFresnelWeight = this._options.groundMirrorFresnelWeight, this._groundMaterial.reflectionFalloffDistance = this._options.groundMirrorFallOffDistance); } /** * Setup the skybox according to the specified options. * @param sceneSize */ _setupSkybox(e) { (!this._skybox || this._skybox.isDisposed()) && (this._skybox = k0("BackgroundSkybox", { size: e.skyboxSize, sideOrientation: Ee.BACKSIDE }, this._scene), this._skybox.onDisposeObservable.add(() => { this._skybox = null; })), this._skybox.parent = this._rootMesh; } /** * Setup the skybox material according to the specified options. */ _setupSkyboxMaterial() { this._skybox && (this._skyboxMaterial || (this._skyboxMaterial = new Vi("BackgroundSkyboxMaterial", this._scene)), this._skyboxMaterial.useRGBColor = !1, this._skyboxMaterial.primaryColor = this._options.skyboxColor, this._skyboxMaterial.enableNoise = !0, this._skybox.material = this._skyboxMaterial); } /** * Setup the skybox reflection texture according to the specified options. */ _setupSkyboxReflectionTexture() { if (this._skyboxMaterial && !this._skyboxTexture) { if (this._options.skyboxTexture instanceof ls) { this._skyboxMaterial.reflectionTexture = this._options.skyboxTexture; return; } this._skyboxTexture = new v1(this._options.skyboxTexture, this._scene, void 0, void 0, void 0, void 0, this._errorHandler), this._skyboxTexture.coordinatesMode = We.SKYBOX_MODE, this._skyboxTexture.gammaSpace = !1, this._skyboxMaterial.reflectionTexture = this._skyboxTexture; } } /** * Dispose all the elements created by the Helper. */ dispose() { this._groundMaterial && this._groundMaterial.dispose(!0, !0), this._skyboxMaterial && this._skyboxMaterial.dispose(!0, !0), this._rootMesh.dispose(!1); } } gD._GroundTextureCDNUrl = "https://assets.babylonjs.com/environments/backgroundGround.png"; gD._SkyboxTextureCDNUrl = "https://assets.babylonjs.com/environments/backgroundSkybox.dds"; gD._EnvironmentTextureCDNUrl = "https://assets.babylonjs.com/environments/environmentSpecular.env"; class Ou extends Hr { /** * Gets the texture being displayed on the sphere */ get texture() { return this._texture; } /** * Sets the texture being displayed on the sphere */ set texture(e) { this._texture !== e && (this._texture = e, this._useDirectMapping ? (this._texture.wrapU = We.CLAMP_ADDRESSMODE, this._texture.wrapV = We.CLAMP_ADDRESSMODE, this._material.diffuseTexture = this._texture) : (this._texture.coordinatesMode = We.FIXED_EQUIRECTANGULAR_MIRRORED_MODE, this._texture.wrapV = We.CLAMP_ADDRESSMODE, this._material.reflectionTexture = this._texture), this._changeTextureMode(this._textureMode)); } /** * Gets the mesh used for the dome. */ get mesh() { return this._mesh; } /** * The current fov(field of view) multiplier, 0.0 - 2.0. Defaults to 1.0. Lower values "zoom in" and higher values "zoom out". * Also see the options.resolution property. */ get fovMultiplier() { return this._material.fovMultiplier; } set fovMultiplier(e) { this._material.fovMultiplier = e; } /** * Gets or set the current texture mode for the texture. It can be: * * TextureDome.MODE_MONOSCOPIC : Define the texture source as a Monoscopic panoramic 360. * * TextureDome.MODE_TOPBOTTOM : Define the texture source as a Stereoscopic TopBottom/OverUnder panoramic 360. * * TextureDome.MODE_SIDEBYSIDE : Define the texture source as a Stereoscopic Side by Side panoramic 360. */ get textureMode() { return this._textureMode; } /** * Sets the current texture mode for the texture. It can be: * * TextureDome.MODE_MONOSCOPIC : Define the texture source as a Monoscopic panoramic 360. * * TextureDome.MODE_TOPBOTTOM : Define the texture source as a Stereoscopic TopBottom/OverUnder panoramic 360. * * TextureDome.MODE_SIDEBYSIDE : Define the texture source as a Stereoscopic Side by Side panoramic 360. */ set textureMode(e) { this._textureMode !== e && this._changeTextureMode(e); } /** * Is it a 180 degrees dome (half dome) or 360 texture (full dome) */ get halfDome() { return this._halfDome; } /** * Set the halfDome mode. If set, only the front (180 degrees) will be displayed and the back will be blacked out. */ set halfDome(e) { this._halfDome = e, this._halfDomeMask.setEnabled(e), this._changeTextureMode(this._textureMode); } /** * Set the cross-eye mode. If set, images that can be seen when crossing eyes will render correctly */ set crossEye(e) { this._crossEye = e, this._changeTextureMode(this._textureMode); } /** * Is it a cross-eye texture? */ get crossEye() { return this._crossEye; } /** * The background material of this dome. */ get material() { return this._material; } /** * Create an instance of this class and pass through the parameters to the relevant classes- Texture, StandardMaterial, and Mesh. * @param name Element's name, child elements will append suffixes for their own names. * @param textureUrlOrElement defines the url(s) or the (video) HTML element to use * @param options An object containing optional or exposed sub element properties * @param options.resolution * @param options.clickToPlay * @param options.autoPlay * @param options.loop * @param options.size * @param options.poster * @param options.faceForward * @param options.useDirectMapping * @param options.halfDomeMode * @param options.crossEyeMode * @param options.generateMipMaps * @param options.mesh * @param scene * @param onError */ constructor(e, t, r, n, i = null) { super(e, n), this.onError = i, this._halfDome = !1, this._crossEye = !1, this._useDirectMapping = !1, this._textureMode = Ou.MODE_MONOSCOPIC, this._onBeforeCameraRenderObserver = null, this.onLoadErrorObservable = new Oe(), this.onLoadObservable = new Oe(), n = this.getScene(), e = e || "textureDome", r.resolution = Math.abs(r.resolution) | 0 || 32, r.clickToPlay = !!r.clickToPlay, r.autoPlay = r.autoPlay === void 0 ? !0 : !!r.autoPlay, r.loop = r.loop === void 0 ? !0 : !!r.loop, r.size = Math.abs(r.size) || (n.activeCamera ? n.activeCamera.maxZ * 0.48 : 1e3), r.useDirectMapping === void 0 ? this._useDirectMapping = !0 : this._useDirectMapping = r.useDirectMapping, r.faceForward === void 0 && (r.faceForward = !0), this._setReady(!1), r.mesh ? this._mesh = r.mesh : this._mesh = UA(e + "_mesh", { segments: r.resolution, diameter: r.size, updatable: !1, sideOrientation: Ee.BACKSIDE }, n); const s = this._material = new Vi(e + "_material", n); s.useEquirectangularFOV = !0, s.fovMultiplier = 1, s.opacityFresnel = !1; const a = this._initTexture(t, n, r); if (this.texture = a, this._mesh.material = s, this._mesh.parent = this, this._halfDomeMask = UA("", { slice: 0.5, diameter: r.size * 0.98, segments: r.resolution * 2, sideOrientation: Ee.BACKSIDE }, n), this._halfDomeMask.rotate(bf.X, -Math.PI / 2), this._halfDomeMask.parent = this._mesh, this._halfDome = !!r.halfDomeMode, this._halfDomeMask.setEnabled(this._halfDome), this._crossEye = !!r.crossEyeMode, this._texture.anisotropicFilteringLevel = 1, this._texture.onLoadObservable.addOnce(() => { this._setReady(!0); }), r.faceForward && n.activeCamera) { const f = n.activeCamera, o = S.Forward(), d = S.TransformNormal(o, f.getViewMatrix()); d.normalize(), this.rotation.y = Math.acos(S.Dot(o, d)); } this._changeTextureMode(this._textureMode); } _changeTextureMode(e) { switch (this._scene.onBeforeCameraRenderObservable.remove(this._onBeforeCameraRenderObserver), this._textureMode = e, this._texture.uScale = 1, this._texture.vScale = 1, this._texture.uOffset = 0, this._texture.vOffset = 0, this._texture.vAng = 0, e) { case Ou.MODE_MONOSCOPIC: this._halfDome && (this._texture.uScale = 2, this._texture.uOffset = -1); break; case Ou.MODE_SIDEBYSIDE: { this._texture.uScale = this._halfDome ? 0.99999 : 0.5; const t = this._halfDome ? 0 : 0.5, r = this._halfDome ? -0.5 : 0; this._onBeforeCameraRenderObserver = this._scene.onBeforeCameraRenderObservable.add((n) => { let i = n.isRightCamera; this._crossEye && (i = !i), i ? this._texture.uOffset = t : this._texture.uOffset = r; }); break; } case Ou.MODE_TOPBOTTOM: this._texture.vScale = this._halfDome ? 0.99999 : 0.5, this._onBeforeCameraRenderObserver = this._scene.onBeforeCameraRenderObservable.add((t) => { let r = t.isRightCamera; this._crossEye && (r = !r), this._texture.vOffset = r ? 0.5 : 0; }); break; } } /** * Releases resources associated with this node. * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default) * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default) */ dispose(e, t = !1) { this._texture.dispose(), this._mesh.dispose(), this._material.dispose(), this._scene.onBeforeCameraRenderObservable.remove(this._onBeforeCameraRenderObserver), this.onLoadErrorObservable.clear(), this.onLoadObservable.clear(), super.dispose(e, t); } } Ou.MODE_MONOSCOPIC = 0; Ou.MODE_TOPBOTTOM = 1; Ou.MODE_SIDEBYSIDE = 2; class US extends Ou { /** * Gets or sets the texture being displayed on the sphere */ get photoTexture() { return this.texture; } /** * sets the texture being displayed on the sphere */ set photoTexture(e) { this.texture = e; } /** * Gets the current video mode for the video. It can be: * * TextureDome.MODE_MONOSCOPIC : Define the texture source as a Monoscopic panoramic 360. * * TextureDome.MODE_TOPBOTTOM : Define the texture source as a Stereoscopic TopBottom/OverUnder panoramic 360. * * TextureDome.MODE_SIDEBYSIDE : Define the texture source as a Stereoscopic Side by Side panoramic 360. */ get imageMode() { return this.textureMode; } /** * Sets the current video mode for the video. It can be: * * TextureDome.MODE_MONOSCOPIC : Define the texture source as a Monoscopic panoramic 360. * * TextureDome.MODE_TOPBOTTOM : Define the texture source as a Stereoscopic TopBottom/OverUnder panoramic 360. * * TextureDome.MODE_SIDEBYSIDE : Define the texture source as a Stereoscopic Side by Side panoramic 360. */ set imageMode(e) { this.textureMode = e; } _initTexture(e, t, r) { return new We(e, t, !r.generateMipMaps, !this._useDirectMapping, void 0, () => { this.onLoadObservable.notifyObservers(); }, (n, i) => { this.onLoadErrorObservable.notifyObservers(n || "Unknown error occured"), this.onError && this.onError(n, i); }); } } US.MODE_MONOSCOPIC = Ou.MODE_MONOSCOPIC; US.MODE_TOPBOTTOM = Ou.MODE_TOPBOTTOM; US.MODE_SIDEBYSIDE = Ou.MODE_SIDEBYSIDE; const TPe = ""; let qPe = 0; const nV = (A) => { if (!A.environmentBRDFTexture) { const e = A.useDelayedTextureLoading; A.useDelayedTextureLoading = !1; const t = A._blockEntityCollection; A._blockEntityCollection = !1; const r = We.CreateFromBase64String(TPe, "EnvironmentBRDFTexture" + qPe++, A, !0, !1, We.BILINEAR_SAMPLINGMODE); A._blockEntityCollection = t; const n = A.getEngine().getLoadedTexturesCache(), i = n.indexOf(r.getInternalTexture()); i !== -1 && n.splice(i, 1), r.isRGBD = !0, r.wrapU = We.CLAMP_ADDRESSMODE, r.wrapV = We.CLAMP_ADDRESSMODE, A.environmentBRDFTexture = r, A.useDelayedTextureLoading = e, $C.ExpandRGBDTexture(r); const s = A.getEngine().onContextRestoredObservable.add(() => { r.isRGBD = !0; const a = () => { r.isReady() ? $C.ExpandRGBDTexture(r) : ye.SetImmediate(a); }; a(); }); A.onDisposeObservable.add(() => { A.getEngine().onContextRestoredObservable.remove(s); }); } return A.environmentBRDFTexture; }, bPe = { /** * Gets a default environment BRDF for MS-BRDF Height Correlated BRDF * @param scene defines the hosting scene * @returns the environment BRDF texture */ // eslint-disable-next-line @typescript-eslint/naming-convention GetEnvironmentBRDFTexture: nV }; class xPe extends na { constructor() { super(...arguments), this.BRDF_V_HEIGHT_CORRELATED = !1, this.MS_BRDF_ENERGY_CONSERVATION = !1, this.SPHERICAL_HARMONICS = !1, this.SPECULAR_GLOSSINESS_ENERGY_CONSERVATION = !1; } } class fd extends Gl { /** @internal */ _markAllSubMeshesAsMiscDirty() { this._internalMarkAllSubMeshesAsMiscDirty(); } constructor(e, t = !0) { super(e, "PBRBRDF", 90, new xPe(), t), this._useEnergyConservation = fd.DEFAULT_USE_ENERGY_CONSERVATION, this.useEnergyConservation = fd.DEFAULT_USE_ENERGY_CONSERVATION, this._useSmithVisibilityHeightCorrelated = fd.DEFAULT_USE_SMITH_VISIBILITY_HEIGHT_CORRELATED, this.useSmithVisibilityHeightCorrelated = fd.DEFAULT_USE_SMITH_VISIBILITY_HEIGHT_CORRELATED, this._useSphericalHarmonics = fd.DEFAULT_USE_SPHERICAL_HARMONICS, this.useSphericalHarmonics = fd.DEFAULT_USE_SPHERICAL_HARMONICS, this._useSpecularGlossinessInputEnergyConservation = fd.DEFAULT_USE_SPECULAR_GLOSSINESS_INPUT_ENERGY_CONSERVATION, this.useSpecularGlossinessInputEnergyConservation = fd.DEFAULT_USE_SPECULAR_GLOSSINESS_INPUT_ENERGY_CONSERVATION, this._internalMarkAllSubMeshesAsMiscDirty = e._dirtyCallbacks[16], this._enable(!0); } prepareDefines(e) { e.BRDF_V_HEIGHT_CORRELATED = this._useSmithVisibilityHeightCorrelated, e.MS_BRDF_ENERGY_CONSERVATION = this._useEnergyConservation && this._useSmithVisibilityHeightCorrelated, e.SPHERICAL_HARMONICS = this._useSphericalHarmonics, e.SPECULAR_GLOSSINESS_ENERGY_CONSERVATION = this._useSpecularGlossinessInputEnergyConservation; } getClassName() { return "PBRBRDFConfiguration"; } } fd.DEFAULT_USE_ENERGY_CONSERVATION = !0; fd.DEFAULT_USE_SMITH_VISIBILITY_HEIGHT_CORRELATED = !0; fd.DEFAULT_USE_SPHERICAL_HARMONICS = !0; fd.DEFAULT_USE_SPECULAR_GLOSSINESS_INPUT_ENERGY_CONSERVATION = !0; C([ M(), At("_markAllSubMeshesAsMiscDirty") ], fd.prototype, "useEnergyConservation", void 0); C([ M(), At("_markAllSubMeshesAsMiscDirty") ], fd.prototype, "useSmithVisibilityHeightCorrelated", void 0); C([ M(), At("_markAllSubMeshesAsMiscDirty") ], fd.prototype, "useSphericalHarmonics", void 0); C([ M(), At("_markAllSubMeshesAsMiscDirty") ], fd.prototype, "useSpecularGlossinessInputEnergyConservation", void 0); const DPe = "pbrFragmentDeclaration", jPe = `uniform vec4 vEyePosition;uniform vec3 vReflectionColor;uniform vec4 vAlbedoColor;uniform vec4 vLightingIntensity;uniform vec4 vReflectivityColor;uniform vec4 vMetallicReflectanceFactors;uniform vec3 vEmissiveColor;uniform float visibility;uniform vec3 vAmbientColor; #ifdef ALBEDO uniform vec2 vAlbedoInfos; #endif #ifdef AMBIENT uniform vec4 vAmbientInfos; #endif #ifdef BUMP uniform vec3 vBumpInfos;uniform vec2 vTangentSpaceParams; #endif #ifdef OPACITY uniform vec2 vOpacityInfos; #endif #ifdef EMISSIVE uniform vec2 vEmissiveInfos; #endif #ifdef LIGHTMAP uniform vec2 vLightmapInfos; #endif #ifdef REFLECTIVITY uniform vec3 vReflectivityInfos; #endif #ifdef MICROSURFACEMAP uniform vec2 vMicroSurfaceSamplerInfos; #endif #if defined(REFLECTIONMAP_SPHERICAL) || defined(REFLECTIONMAP_PROJECTION) || defined(SS_REFRACTION) || defined(PREPASS) uniform mat4 view; #endif #ifdef REFLECTION uniform vec2 vReflectionInfos; #ifdef REALTIME_FILTERING uniform vec2 vReflectionFilteringInfo; #endif uniform mat4 reflectionMatrix;uniform vec3 vReflectionMicrosurfaceInfos; #if defined(USE_LOCAL_REFLECTIONMAP_CUBIC) && defined(REFLECTIONMAP_CUBIC) uniform vec3 vReflectionPosition;uniform vec3 vReflectionSize; #endif #endif #if defined(SS_REFRACTION) && defined(SS_USE_LOCAL_REFRACTIONMAP_CUBIC) uniform vec3 vRefractionPosition;uniform vec3 vRefractionSize; #endif #ifdef CLEARCOAT uniform vec2 vClearCoatParams;uniform vec4 vClearCoatRefractionParams; #if defined(CLEARCOAT_TEXTURE) || defined(CLEARCOAT_TEXTURE_ROUGHNESS) uniform vec4 vClearCoatInfos; #endif #ifdef CLEARCOAT_TEXTURE uniform mat4 clearCoatMatrix; #endif #ifdef CLEARCOAT_TEXTURE_ROUGHNESS uniform mat4 clearCoatRoughnessMatrix; #endif #ifdef CLEARCOAT_BUMP uniform vec2 vClearCoatBumpInfos;uniform vec2 vClearCoatTangentSpaceParams;uniform mat4 clearCoatBumpMatrix; #endif #ifdef CLEARCOAT_TINT uniform vec4 vClearCoatTintParams;uniform float clearCoatColorAtDistance; #ifdef CLEARCOAT_TINT_TEXTURE uniform vec2 vClearCoatTintInfos;uniform mat4 clearCoatTintMatrix; #endif #endif #endif #ifdef IRIDESCENCE uniform vec4 vIridescenceParams; #if defined(IRIDESCENCE_TEXTURE) || defined(IRIDESCENCE_THICKNESS_TEXTURE) uniform vec4 vIridescenceInfos; #endif #ifdef IRIDESCENCE_TEXTURE uniform mat4 iridescenceMatrix; #endif #ifdef IRIDESCENCE_THICKNESS_TEXTURE uniform mat4 iridescenceThicknessMatrix; #endif #endif #ifdef ANISOTROPIC uniform vec3 vAnisotropy; #ifdef ANISOTROPIC_TEXTURE uniform vec2 vAnisotropyInfos;uniform mat4 anisotropyMatrix; #endif #endif #ifdef SHEEN uniform vec4 vSheenColor; #ifdef SHEEN_ROUGHNESS uniform float vSheenRoughness; #endif #if defined(SHEEN_TEXTURE) || defined(SHEEN_TEXTURE_ROUGHNESS) uniform vec4 vSheenInfos; #endif #ifdef SHEEN_TEXTURE uniform mat4 sheenMatrix; #endif #ifdef SHEEN_TEXTURE_ROUGHNESS uniform mat4 sheenRoughnessMatrix; #endif #endif #ifdef SUBSURFACE #ifdef SS_REFRACTION uniform vec4 vRefractionMicrosurfaceInfos;uniform vec4 vRefractionInfos;uniform mat4 refractionMatrix; #ifdef REALTIME_FILTERING uniform vec2 vRefractionFilteringInfo; #endif #ifdef SS_DISPERSION uniform float dispersion; #endif #endif #ifdef SS_THICKNESSANDMASK_TEXTURE uniform vec2 vThicknessInfos;uniform mat4 thicknessMatrix; #endif #ifdef SS_REFRACTIONINTENSITY_TEXTURE uniform vec2 vRefractionIntensityInfos;uniform mat4 refractionIntensityMatrix; #endif #ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE uniform vec2 vTranslucencyIntensityInfos;uniform mat4 translucencyIntensityMatrix; #endif uniform vec2 vThicknessParam;uniform vec3 vDiffusionDistance;uniform vec4 vTintColor;uniform vec3 vSubSurfaceIntensity; #endif #ifdef PREPASS #ifdef SS_SCATTERING uniform float scatteringDiffusionProfile; #endif #endif #if DEBUGMODE>0 uniform vec2 vDebugMode; #endif #ifdef DETAIL uniform vec4 vDetailInfos; #endif #include #ifdef USESPHERICALFROMREFLECTIONMAP #ifdef SPHERICAL_HARMONICS uniform vec3 vSphericalL00;uniform vec3 vSphericalL1_1;uniform vec3 vSphericalL10;uniform vec3 vSphericalL11;uniform vec3 vSphericalL2_2;uniform vec3 vSphericalL2_1;uniform vec3 vSphericalL20;uniform vec3 vSphericalL21;uniform vec3 vSphericalL22; #else uniform vec3 vSphericalX;uniform vec3 vSphericalY;uniform vec3 vSphericalZ;uniform vec3 vSphericalXX_ZZ;uniform vec3 vSphericalYY_ZZ;uniform vec3 vSphericalZZ;uniform vec3 vSphericalXY;uniform vec3 vSphericalYZ;uniform vec3 vSphericalZX; #endif #endif #define ADDITIONAL_FRAGMENT_DECLARATION `; Le.IncludesShadersStore[DPe] = jPe; const wPe = "pbrUboDeclaration", mPe = `layout(std140,column_major) uniform;uniform Material {vec2 vAlbedoInfos;vec4 vAmbientInfos;vec2 vOpacityInfos;vec2 vEmissiveInfos;vec2 vLightmapInfos;vec3 vReflectivityInfos;vec2 vMicroSurfaceSamplerInfos;vec2 vReflectionInfos;vec2 vReflectionFilteringInfo;vec3 vReflectionPosition;vec3 vReflectionSize;vec3 vBumpInfos;mat4 albedoMatrix;mat4 ambientMatrix;mat4 opacityMatrix;mat4 emissiveMatrix;mat4 lightmapMatrix;mat4 reflectivityMatrix;mat4 microSurfaceSamplerMatrix;mat4 bumpMatrix;vec2 vTangentSpaceParams;mat4 reflectionMatrix;vec3 vReflectionColor;vec4 vAlbedoColor;vec4 vLightingIntensity;vec3 vReflectionMicrosurfaceInfos;float pointSize;vec4 vReflectivityColor;vec3 vEmissiveColor;vec3 vAmbientColor;vec2 vDebugMode;vec4 vMetallicReflectanceFactors;vec2 vMetallicReflectanceInfos;mat4 metallicReflectanceMatrix;vec2 vReflectanceInfos;mat4 reflectanceMatrix;vec3 vSphericalL00;vec3 vSphericalL1_1;vec3 vSphericalL10;vec3 vSphericalL11;vec3 vSphericalL2_2;vec3 vSphericalL2_1;vec3 vSphericalL20;vec3 vSphericalL21;vec3 vSphericalL22;vec3 vSphericalX;vec3 vSphericalY;vec3 vSphericalZ;vec3 vSphericalXX_ZZ;vec3 vSphericalYY_ZZ;vec3 vSphericalZZ;vec3 vSphericalXY;vec3 vSphericalYZ;vec3 vSphericalZX; #define ADDITIONAL_UBO_DECLARATION }; #include #include `; Le.IncludesShadersStore[wPe] = mPe; const BPe = "pbrFragmentExtraDeclaration", WPe = `varying vec3 vPositionW; #if DEBUGMODE>0 varying vec4 vClipSpacePosition; #endif #include[1..7] #ifdef NORMAL varying vec3 vNormalW; #if defined(USESPHERICALFROMREFLECTIONMAP) && defined(USESPHERICALINVERTEX) varying vec3 vEnvironmentIrradiance; #endif #endif #if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES) varying vec4 vColor; #endif `; Le.IncludesShadersStore[BPe] = WPe; const SPe = "samplerFragmentAlternateDeclaration", UPe = `#ifdef _DEFINENAME_ #if _DEFINENAME_DIRECTUV==1 #define v_VARYINGNAME_UV vMainUV1 #elif _DEFINENAME_DIRECTUV==2 #define v_VARYINGNAME_UV vMainUV2 #elif _DEFINENAME_DIRECTUV==3 #define v_VARYINGNAME_UV vMainUV3 #elif _DEFINENAME_DIRECTUV==4 #define v_VARYINGNAME_UV vMainUV4 #elif _DEFINENAME_DIRECTUV==5 #define v_VARYINGNAME_UV vMainUV5 #elif _DEFINENAME_DIRECTUV==6 #define v_VARYINGNAME_UV vMainUV6 #else varying vec2 v_VARYINGNAME_UV; #endif #endif `; Le.IncludesShadersStore[SPe] = UPe; const IPe = "pbrFragmentSamplersDeclaration", RPe = `#include(_DEFINENAME_,ALBEDO,_VARYINGNAME_,Albedo,_SAMPLERNAME_,albedo) #include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient,_SAMPLERNAME_,ambient) #include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity,_SAMPLERNAME_,opacity) #include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive,_SAMPLERNAME_,emissive) #include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap,_SAMPLERNAME_,lightmap) #include(_DEFINENAME_,REFLECTIVITY,_VARYINGNAME_,Reflectivity,_SAMPLERNAME_,reflectivity) #include(_DEFINENAME_,MICROSURFACEMAP,_VARYINGNAME_,MicroSurfaceSampler,_SAMPLERNAME_,microSurface) #include(_DEFINENAME_,METALLIC_REFLECTANCE,_VARYINGNAME_,MetallicReflectance,_SAMPLERNAME_,metallicReflectance) #include(_DEFINENAME_,REFLECTANCE,_VARYINGNAME_,Reflectance,_SAMPLERNAME_,reflectance) #include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal,_SAMPLERNAME_,decal) #ifdef CLEARCOAT #include(_DEFINENAME_,CLEARCOAT_TEXTURE,_VARYINGNAME_,ClearCoat,_SAMPLERNAME_,clearCoat) #include(_DEFINENAME_,CLEARCOAT_TEXTURE_ROUGHNESS,_VARYINGNAME_,ClearCoatRoughness) #if defined(CLEARCOAT_TEXTURE_ROUGHNESS) && !defined(CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL) uniform sampler2D clearCoatRoughnessSampler; #endif #include(_DEFINENAME_,CLEARCOAT_BUMP,_VARYINGNAME_,ClearCoatBump,_SAMPLERNAME_,clearCoatBump) #include(_DEFINENAME_,CLEARCOAT_TINT_TEXTURE,_VARYINGNAME_,ClearCoatTint,_SAMPLERNAME_,clearCoatTint) #endif #ifdef IRIDESCENCE #include(_DEFINENAME_,IRIDESCENCE_TEXTURE,_VARYINGNAME_,Iridescence,_SAMPLERNAME_,iridescence) #include(_DEFINENAME_,IRIDESCENCE_THICKNESS_TEXTURE,_VARYINGNAME_,IridescenceThickness,_SAMPLERNAME_,iridescenceThickness) #endif #ifdef SHEEN #include(_DEFINENAME_,SHEEN_TEXTURE,_VARYINGNAME_,Sheen,_SAMPLERNAME_,sheen) #include(_DEFINENAME_,SHEEN_TEXTURE_ROUGHNESS,_VARYINGNAME_,SheenRoughness) #if defined(SHEEN_ROUGHNESS) && defined(SHEEN_TEXTURE_ROUGHNESS) && !defined(SHEEN_TEXTURE_ROUGHNESS_IDENTICAL) uniform sampler2D sheenRoughnessSampler; #endif #endif #ifdef ANISOTROPIC #include(_DEFINENAME_,ANISOTROPIC_TEXTURE,_VARYINGNAME_,Anisotropy,_SAMPLERNAME_,anisotropy) #endif #ifdef REFLECTION #ifdef REFLECTIONMAP_3D #define sampleReflection(s,c) textureCube(s,c) uniform samplerCube reflectionSampler; #ifdef LODBASEDMICROSFURACE #define sampleReflectionLod(s,c,l) textureCubeLodEXT(s,c,l) #else uniform samplerCube reflectionSamplerLow;uniform samplerCube reflectionSamplerHigh; #endif #ifdef USEIRRADIANCEMAP uniform samplerCube irradianceSampler; #endif #else #define sampleReflection(s,c) texture2D(s,c) uniform sampler2D reflectionSampler; #ifdef LODBASEDMICROSFURACE #define sampleReflectionLod(s,c,l) texture2DLodEXT(s,c,l) #else uniform sampler2D reflectionSamplerLow;uniform sampler2D reflectionSamplerHigh; #endif #ifdef USEIRRADIANCEMAP uniform sampler2D irradianceSampler; #endif #endif #ifdef REFLECTIONMAP_SKYBOX varying vec3 vPositionUVW; #else #if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED) varying vec3 vDirectionW; #endif #endif #endif #ifdef ENVIRONMENTBRDF uniform sampler2D environmentBrdfSampler; #endif #ifdef SUBSURFACE #ifdef SS_REFRACTION #ifdef SS_REFRACTIONMAP_3D #define sampleRefraction(s,c) textureCube(s,c) uniform samplerCube refractionSampler; #ifdef LODBASEDMICROSFURACE #define sampleRefractionLod(s,c,l) textureCubeLodEXT(s,c,l) #else uniform samplerCube refractionSamplerLow;uniform samplerCube refractionSamplerHigh; #endif #else #define sampleRefraction(s,c) texture2D(s,c) uniform sampler2D refractionSampler; #ifdef LODBASEDMICROSFURACE #define sampleRefractionLod(s,c,l) texture2DLodEXT(s,c,l) #else uniform sampler2D refractionSamplerLow;uniform sampler2D refractionSamplerHigh; #endif #endif #endif #include(_DEFINENAME_,SS_THICKNESSANDMASK_TEXTURE,_VARYINGNAME_,Thickness,_SAMPLERNAME_,thickness) #include(_DEFINENAME_,SS_REFRACTIONINTENSITY_TEXTURE,_VARYINGNAME_,RefractionIntensity,_SAMPLERNAME_,refractionIntensity) #include(_DEFINENAME_,SS_TRANSLUCENCYINTENSITY_TEXTURE,_VARYINGNAME_,TranslucencyIntensity,_SAMPLERNAME_,translucencyIntensity) #endif `; Le.IncludesShadersStore[IPe] = RPe; const VPe = "subSurfaceScatteringFunctions", CPe = `bool testLightingForSSS(float diffusionProfile) {return diffusionProfile<1.;}`; Le.IncludesShadersStore[VPe] = CPe; const OPe = "importanceSampling", yPe = `vec3 hemisphereCosSample(vec2 u) {float phi=2.*PI*u.x;float cosTheta2=1.-u.y;float cosTheta=sqrt(cosTheta2);float sinTheta=sqrt(1.-cosTheta2);return vec3(sinTheta*cos(phi),sinTheta*sin(phi),cosTheta);} vec3 hemisphereImportanceSampleDggx(vec2 u,float a) {float phi=2.*PI*u.x;float cosTheta2=(1.-u.y)/(1.+(a+1.)*((a-1.)*u.y));float cosTheta=sqrt(cosTheta2);float sinTheta=sqrt(1.-cosTheta2);return vec3(sinTheta*cos(phi),sinTheta*sin(phi),cosTheta);} vec3 hemisphereImportanceSampleDCharlie(vec2 u,float a) { float phi=2.*PI*u.x;float sinTheta=pow(u.y,a/(2.*a+1.));float cosTheta=sqrt(1.-sinTheta*sinTheta);return vec3(sinTheta*cos(phi),sinTheta*sin(phi),cosTheta);}`; Le.IncludesShadersStore[OPe] = yPe; const kPe = "pbrHelperFunctions", EPe = `#define MINIMUMVARIANCE 0.0005 float convertRoughnessToAverageSlope(float roughness) {return square(roughness)+MINIMUMVARIANCE;} float fresnelGrazingReflectance(float reflectance0) {float reflectance90=saturate(reflectance0*25.0);return reflectance90;} vec2 getAARoughnessFactors(vec3 normalVector) { #ifdef SPECULARAA vec3 nDfdx=dFdx(normalVector.xyz);vec3 nDfdy=dFdy(normalVector.xyz);float slopeSquare=max(dot(nDfdx,nDfdx),dot(nDfdy,nDfdy));float geometricRoughnessFactor=pow(saturate(slopeSquare),0.333);float geometricAlphaGFactor=sqrt(slopeSquare);geometricAlphaGFactor*=0.75;return vec2(geometricRoughnessFactor,geometricAlphaGFactor); #else return vec2(0.); #endif } #ifdef ANISOTROPIC #ifdef ANISOTROPIC_LEGACY vec2 getAnisotropicRoughness(float alphaG,float anisotropy) {float alphaT=max(alphaG*(1.0+anisotropy),MINIMUMVARIANCE);float alphaB=max(alphaG*(1.0-anisotropy),MINIMUMVARIANCE);return vec2(alphaT,alphaB);} vec3 getAnisotropicBentNormals(const vec3 T,const vec3 B,const vec3 N,const vec3 V,float anisotropy,float roughness) {vec3 anisotropicFrameDirection=anisotropy>=0.0 ? B : T;vec3 anisotropicFrameTangent=cross(normalize(anisotropicFrameDirection),V);vec3 anisotropicFrameNormal=cross(anisotropicFrameTangent,anisotropicFrameDirection);vec3 anisotropicNormal=normalize(mix(N,anisotropicFrameNormal,abs(anisotropy)));return anisotropicNormal;} #else vec2 getAnisotropicRoughness(float alphaG,float anisotropy) {float alphaT=max(mix(alphaG,1.0,anisotropy*anisotropy),MINIMUMVARIANCE);float alphaB=max(alphaG,MINIMUMVARIANCE);return vec2(alphaT,alphaB);} vec3 getAnisotropicBentNormals(const vec3 T,const vec3 B,const vec3 N,const vec3 V,float anisotropy,float roughness) {vec3 bentNormal=cross(B,V);bentNormal=normalize(cross(bentNormal,B));float a=square(square(1.0-anisotropy*(1.0-roughness)));bentNormal=normalize(mix(bentNormal,N,a));return bentNormal;} #endif #endif #if defined(CLEARCOAT) || defined(SS_REFRACTION) vec3 cocaLambert(vec3 alpha,float distance) {return exp(-alpha*distance);} vec3 cocaLambert(float NdotVRefract,float NdotLRefract,vec3 alpha,float thickness) {return cocaLambert(alpha,(thickness*((NdotLRefract+NdotVRefract)/(NdotLRefract*NdotVRefract))));} vec3 computeColorAtDistanceInMedia(vec3 color,float distance) {return -log(color)/distance;} vec3 computeClearCoatAbsorption(float NdotVRefract,float NdotLRefract,vec3 clearCoatColor,float clearCoatThickness,float clearCoatIntensity) {vec3 clearCoatAbsorption=mix(vec3(1.0), cocaLambert(NdotVRefract,NdotLRefract,clearCoatColor,clearCoatThickness), clearCoatIntensity);return clearCoatAbsorption;} #endif #ifdef MICROSURFACEAUTOMATIC float computeDefaultMicroSurface(float microSurface,vec3 reflectivityColor) {const float kReflectivityNoAlphaWorkflow_SmoothnessMax=0.95;float reflectivityLuminance=getLuminance(reflectivityColor);float reflectivityLuma=sqrt(reflectivityLuminance);microSurface=reflectivityLuma*kReflectivityNoAlphaWorkflow_SmoothnessMax;return microSurface;} #endif `; Le.IncludesShadersStore[kPe] = EPe; const FPe = "harmonicsFunctions", NPe = `#ifdef USESPHERICALFROMREFLECTIONMAP #ifdef SPHERICAL_HARMONICS vec3 computeEnvironmentIrradiance(vec3 normal) {return vSphericalL00 + vSphericalL1_1*(normal.y) + vSphericalL10*(normal.z) + vSphericalL11*(normal.x) + vSphericalL2_2*(normal.y*normal.x) + vSphericalL2_1*(normal.y*normal.z) + vSphericalL20*((3.0*normal.z*normal.z)-1.0) + vSphericalL21*(normal.z*normal.x) + vSphericalL22*(normal.x*normal.x-(normal.y*normal.y));} #else vec3 computeEnvironmentIrradiance(vec3 normal) {float Nx=normal.x;float Ny=normal.y;float Nz=normal.z;vec3 C1=vSphericalZZ.rgb;vec3 Cx=vSphericalX.rgb;vec3 Cy=vSphericalY.rgb;vec3 Cz=vSphericalZ.rgb;vec3 Cxx_zz=vSphericalXX_ZZ.rgb;vec3 Cyy_zz=vSphericalYY_ZZ.rgb;vec3 Cxy=vSphericalXY.rgb;vec3 Cyz=vSphericalYZ.rgb;vec3 Czx=vSphericalZX.rgb;vec3 a1=Cyy_zz*Ny+Cy;vec3 a2=Cyz*Nz+a1;vec3 b1=Czx*Nz+Cx;vec3 b2=Cxy*Ny+b1;vec3 b3=Cxx_zz*Nx+b2;vec3 t1=Cz *Nz+C1;vec3 t2=a2 *Ny+t1;vec3 t3=b3 *Nx+t2;return t3;} #endif #endif `; Le.IncludesShadersStore[FPe] = NPe; const QPe = "pbrDirectLightingSetupFunctions", YPe = `struct preLightingInfo {vec3 lightOffset;float lightDistanceSquared;float lightDistance;float attenuation;vec3 L;vec3 H;float NdotV;float NdotLUnclamped;float NdotL;float VdotH;float roughness; #ifdef IRIDESCENCE float iridescenceIntensity; #endif };preLightingInfo computePointAndSpotPreLightingInfo(vec4 lightData,vec3 V,vec3 N) {preLightingInfo result;result.lightOffset=lightData.xyz-vPositionW;result.lightDistanceSquared=dot(result.lightOffset,result.lightOffset);result.lightDistance=sqrt(result.lightDistanceSquared);result.L=normalize(result.lightOffset);result.H=normalize(V+result.L);result.VdotH=saturate(dot(V,result.H));result.NdotLUnclamped=dot(N,result.L);result.NdotL=saturateEps(result.NdotLUnclamped);return result;} preLightingInfo computeDirectionalPreLightingInfo(vec4 lightData,vec3 V,vec3 N) {preLightingInfo result;result.lightDistance=length(-lightData.xyz);result.L=normalize(-lightData.xyz);result.H=normalize(V+result.L);result.VdotH=saturate(dot(V,result.H));result.NdotLUnclamped=dot(N,result.L);result.NdotL=saturateEps(result.NdotLUnclamped);return result;} preLightingInfo computeHemisphericPreLightingInfo(vec4 lightData,vec3 V,vec3 N) {preLightingInfo result;result.NdotL=dot(N,lightData.xyz)*0.5+0.5;result.NdotL=saturateEps(result.NdotL);result.NdotLUnclamped=result.NdotL; #ifdef SPECULARTERM result.L=normalize(lightData.xyz);result.H=normalize(V+result.L);result.VdotH=saturate(dot(V,result.H)); #endif return result;}`; Le.IncludesShadersStore[QPe] = YPe; const MPe = "pbrDirectLightingFalloffFunctions", LPe = `float computeDistanceLightFalloff_Standard(vec3 lightOffset,float range) {return max(0.,1.0-length(lightOffset)/range);} float computeDistanceLightFalloff_Physical(float lightDistanceSquared) {return 1.0/maxEps(lightDistanceSquared);} float computeDistanceLightFalloff_GLTF(float lightDistanceSquared,float inverseSquaredRange) {float lightDistanceFalloff=1.0/maxEps(lightDistanceSquared);float factor=lightDistanceSquared*inverseSquaredRange;float attenuation=saturate(1.0-factor*factor);attenuation*=attenuation;lightDistanceFalloff*=attenuation;return lightDistanceFalloff;} float computeDistanceLightFalloff(vec3 lightOffset,float lightDistanceSquared,float range,float inverseSquaredRange) { #ifdef USEPHYSICALLIGHTFALLOFF return computeDistanceLightFalloff_Physical(lightDistanceSquared); #elif defined(USEGLTFLIGHTFALLOFF) return computeDistanceLightFalloff_GLTF(lightDistanceSquared,inverseSquaredRange); #else return computeDistanceLightFalloff_Standard(lightOffset,range); #endif } float computeDirectionalLightFalloff_Standard(vec3 lightDirection,vec3 directionToLightCenterW,float cosHalfAngle,float exponent) {float falloff=0.0;float cosAngle=maxEps(dot(-lightDirection,directionToLightCenterW));if (cosAngle>=cosHalfAngle) {falloff=max(0.,pow(cosAngle,exponent));} return falloff;} float computeDirectionalLightFalloff_Physical(vec3 lightDirection,vec3 directionToLightCenterW,float cosHalfAngle) {const float kMinusLog2ConeAngleIntensityRatio=6.64385618977; float concentrationKappa=kMinusLog2ConeAngleIntensityRatio/(1.0-cosHalfAngle);vec4 lightDirectionSpreadSG=vec4(-lightDirection*concentrationKappa,-concentrationKappa);float falloff=exp2(dot(vec4(directionToLightCenterW,1.0),lightDirectionSpreadSG));return falloff;} float computeDirectionalLightFalloff_GLTF(vec3 lightDirection,vec3 directionToLightCenterW,float lightAngleScale,float lightAngleOffset) {float cd=dot(-lightDirection,directionToLightCenterW);float falloff=saturate(cd*lightAngleScale+lightAngleOffset);falloff*=falloff;return falloff;} float computeDirectionalLightFalloff(vec3 lightDirection,vec3 directionToLightCenterW,float cosHalfAngle,float exponent,float lightAngleScale,float lightAngleOffset) { #ifdef USEPHYSICALLIGHTFALLOFF return computeDirectionalLightFalloff_Physical(lightDirection,directionToLightCenterW,cosHalfAngle); #elif defined(USEGLTFLIGHTFALLOFF) return computeDirectionalLightFalloff_GLTF(lightDirection,directionToLightCenterW,lightAngleScale,lightAngleOffset); #else return computeDirectionalLightFalloff_Standard(lightDirection,directionToLightCenterW,cosHalfAngle,exponent); #endif }`; Le.IncludesShadersStore[MPe] = LPe; const KPe = "pbrBRDFFunctions", JPe = `#define FRESNEL_MAXIMUM_ON_ROUGH 0.25 #ifdef MS_BRDF_ENERGY_CONSERVATION vec3 getEnergyConservationFactor(const vec3 specularEnvironmentR0,const vec3 environmentBrdf) {return 1.0+specularEnvironmentR0*(1.0/environmentBrdf.y-1.0);} #endif #ifdef ENVIRONMENTBRDF vec3 getBRDFLookup(float NdotV,float perceptualRoughness) {vec2 UV=vec2(NdotV,perceptualRoughness);vec4 brdfLookup=texture2D(environmentBrdfSampler,UV); #ifdef ENVIRONMENTBRDF_RGBD brdfLookup.rgb=fromRGBD(brdfLookup.rgba); #endif return brdfLookup.rgb;} vec3 getReflectanceFromBRDFLookup(const vec3 specularEnvironmentR0,const vec3 specularEnvironmentR90,const vec3 environmentBrdf) { #ifdef BRDF_V_HEIGHT_CORRELATED vec3 reflectance=(specularEnvironmentR90-specularEnvironmentR0)*environmentBrdf.x+specularEnvironmentR0*environmentBrdf.y; #else vec3 reflectance=specularEnvironmentR0*environmentBrdf.x+specularEnvironmentR90*environmentBrdf.y; #endif return reflectance;} vec3 getReflectanceFromBRDFLookup(const vec3 specularEnvironmentR0,const vec3 environmentBrdf) { #ifdef BRDF_V_HEIGHT_CORRELATED vec3 reflectance=mix(environmentBrdf.xxx,environmentBrdf.yyy,specularEnvironmentR0); #else vec3 reflectance=specularEnvironmentR0*environmentBrdf.x+environmentBrdf.y; #endif return reflectance;} #endif /* NOT USED #if defined(SHEEN) && defined(SHEEN_SOFTER) float getBRDFLookupCharlieSheen(float NdotV,float perceptualRoughness) {float c=1.0-NdotV;float c3=c*c*c;return 0.65584461*c3+1.0/(4.16526551+exp(-7.97291361*perceptualRoughness+6.33516894));} #endif */ #if !defined(ENVIRONMENTBRDF) || defined(REFLECTIONMAP_SKYBOX) || defined(ALPHAFRESNEL) vec3 getReflectanceFromAnalyticalBRDFLookup_Jones(float VdotN,vec3 reflectance0,vec3 reflectance90,float smoothness) {float weight=mix(FRESNEL_MAXIMUM_ON_ROUGH,1.0,smoothness);return reflectance0+weight*(reflectance90-reflectance0)*pow5(saturate(1.0-VdotN));} #endif #if defined(SHEEN) && defined(ENVIRONMENTBRDF) /** * The sheen BRDF not containing F can be easily stored in the blue channel of the BRDF texture. * The blue channel contains DCharlie*VAshikhmin*NdotL as a lokkup table */ vec3 getSheenReflectanceFromBRDFLookup(const vec3 reflectance0,const vec3 environmentBrdf) {vec3 sheenEnvironmentReflectance=reflectance0*environmentBrdf.b;return sheenEnvironmentReflectance;} #endif vec3 fresnelSchlickGGX(float VdotH,vec3 reflectance0,vec3 reflectance90) {return reflectance0+(reflectance90-reflectance0)*pow5(1.0-VdotH);} float fresnelSchlickGGX(float VdotH,float reflectance0,float reflectance90) {return reflectance0+(reflectance90-reflectance0)*pow5(1.0-VdotH);} #ifdef CLEARCOAT vec3 getR0RemappedForClearCoat(vec3 f0) { #ifdef CLEARCOAT_DEFAULTIOR #ifdef MOBILE return saturate(f0*(f0*0.526868+0.529324)-0.0482256); #else return saturate(f0*(f0*(0.941892-0.263008*f0)+0.346479)-0.0285998); #endif #else vec3 s=sqrt(f0);vec3 t=(vClearCoatRefractionParams.z+vClearCoatRefractionParams.w*s)/(vClearCoatRefractionParams.w+vClearCoatRefractionParams.z*s);return square(t); #endif } #endif #ifdef IRIDESCENCE const mat3 XYZ_TO_REC709=mat3( 3.2404542,-0.9692660, 0.0556434, -1.5371385, 1.8760108,-0.2040259, -0.4985314, 0.0415560, 1.0572252 );vec3 getIORTfromAirToSurfaceR0(vec3 f0) {vec3 sqrtF0=sqrt(f0);return (1.+sqrtF0)/(1.-sqrtF0);} vec3 getR0fromIORs(vec3 iorT,float iorI) {return square((iorT-vec3(iorI))/(iorT+vec3(iorI)));} float getR0fromIORs(float iorT,float iorI) {return square((iorT-iorI)/(iorT+iorI));} vec3 evalSensitivity(float opd,vec3 shift) {float phase=2.0*PI*opd*1.0e-9;const vec3 val=vec3(5.4856e-13,4.4201e-13,5.2481e-13);const vec3 pos=vec3(1.6810e+06,1.7953e+06,2.2084e+06);const vec3 var=vec3(4.3278e+09,9.3046e+09,6.6121e+09);vec3 xyz=val*sqrt(2.0*PI*var)*cos(pos*phase+shift)*exp(-square(phase)*var);xyz.x+=9.7470e-14*sqrt(2.0*PI*4.5282e+09)*cos(2.2399e+06*phase+shift[0])*exp(-4.5282e+09*square(phase));xyz/=1.0685e-7;vec3 srgb=XYZ_TO_REC709*xyz;return srgb;} vec3 evalIridescence(float outsideIOR,float eta2,float cosTheta1,float thinFilmThickness,vec3 baseF0) {vec3 I=vec3(1.0);float iridescenceIOR=mix(outsideIOR,eta2,smoothstep(0.0,0.03,thinFilmThickness));float sinTheta2Sq=square(outsideIOR/iridescenceIOR)*(1.0-square(cosTheta1));float cosTheta2Sq=1.0-sinTheta2Sq;if (cosTheta2Sq<0.0) {return I;} float cosTheta2=sqrt(cosTheta2Sq);float R0=getR0fromIORs(iridescenceIOR,outsideIOR);float R12=fresnelSchlickGGX(cosTheta1,R0,1.);float R21=R12;float T121=1.0-R12;float phi12=0.0;if (iridescenceIOR0 #if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE) float radicalInverse_VdC(uint bits) {bits=(bits<<16u) | (bits>>16u);bits=((bits & 0x55555555u)<<1u) | ((bits & 0xAAAAAAAAu)>>1u);bits=((bits & 0x33333333u)<<2u) | ((bits & 0xCCCCCCCCu)>>2u);bits=((bits & 0x0F0F0F0Fu)<<4u) | ((bits & 0xF0F0F0F0u)>>4u);bits=((bits & 0x00FF00FFu)<<8u) | ((bits & 0xFF00FF00u)>>8u);return float(bits)*2.3283064365386963e-10; } vec2 hammersley(uint i,uint N) {return vec2(float(i)/float(N),radicalInverse_VdC(i));} #else float vanDerCorpus(int n,int base) {float invBase=1.0/float(base);float denom =1.0;float result =0.0;for(int i=0; i<32; ++i) {if(n>0) {denom =mod(float(n),2.0);result+=denom*invBase;invBase=invBase/2.0;n =int(float(n)/2.0);}} return result;} vec2 hammersley(int i,int N) {return vec2(float(i)/float(N),vanDerCorpus(i,2));} #endif float log4(float x) {return log2(x)/2.;} const float NUM_SAMPLES_FLOAT=float(NUM_SAMPLES);const float NUM_SAMPLES_FLOAT_INVERSED=1./NUM_SAMPLES_FLOAT;const float K=4.; #define inline vec3 irradiance(samplerCube inputTexture,vec3 inputN,vec2 filteringInfo) {vec3 n=normalize(inputN);vec3 result=vec3(0.0);vec3 tangent=abs(n.z)<0.999 ? vec3(0.,0.,1.) : vec3(1.,0.,0.);tangent=normalize(cross(tangent,n));vec3 bitangent=cross(n,tangent);mat3 tbn=mat3(tangent,bitangent,n);float maxLevel=filteringInfo.y;float dim0=filteringInfo.x;float omegaP=(4.*PI)/(6.*dim0*dim0); #if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE) for(uint i=0u; i0.) {float pdf_inversed=PI/NoL;float omegaS=NUM_SAMPLES_FLOAT_INVERSED*pdf_inversed;float l=log4(omegaS)-log4(omegaP)+log4(K);float mipLevel=clamp(l,0.0,maxLevel);vec3 c=textureCubeLodEXT(inputTexture,tbn*Ls,mipLevel).rgb; #ifdef GAMMA_INPUT c=toLinearSpace(c); #endif result+=c;}} result=result*NUM_SAMPLES_FLOAT_INVERSED;return result;} #define inline vec3 radiance(float alphaG,samplerCube inputTexture,vec3 inputN,vec2 filteringInfo) {vec3 n=normalize(inputN);vec3 c=textureCube(inputTexture,n).rgb; if (alphaG==0.) { #ifdef GAMMA_INPUT c=toLinearSpace(c); #endif return c;} else {vec3 result=vec3(0.);vec3 tangent=abs(n.z)<0.999 ? vec3(0.,0.,1.) : vec3(1.,0.,0.);tangent=normalize(cross(tangent,n));vec3 bitangent=cross(n,tangent);mat3 tbn=mat3(tangent,bitangent,n);float maxLevel=filteringInfo.y;float dim0=filteringInfo.x;float omegaP=(4.*PI)/(6.*dim0*dim0);float weight=0.; #if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE) for(uint i=0u; i0.) {float pdf_inversed=4./normalDistributionFunction_TrowbridgeReitzGGX(NoH,alphaG);float omegaS=NUM_SAMPLES_FLOAT_INVERSED*pdf_inversed;float l=log4(omegaS)-log4(omegaP)+log4(K);float mipLevel=clamp(float(l),0.0,maxLevel);weight+=NoL;vec3 c=textureCubeLodEXT(inputTexture,tbn*L,mipLevel).rgb; #ifdef GAMMA_INPUT c=toLinearSpace(c); #endif result+=c*NoL;}} result=result/weight;return result;}} #endif #endif `; Le.IncludesShadersStore[zPe] = GPe; const ZPe = "pbrDirectLightingFunctions", _Pe = `#define CLEARCOATREFLECTANCE90 1.0 struct lightingInfo {vec3 diffuse; #ifdef SPECULARTERM vec3 specular; #endif #ifdef CLEARCOAT vec4 clearCoat; #endif #ifdef SHEEN vec3 sheen; #endif };float adjustRoughnessFromLightProperties(float roughness,float lightRadius,float lightDistance) { #if defined(USEPHYSICALLIGHTFALLOFF) || defined(USEGLTFLIGHTFALLOFF) float lightRoughness=lightRadius/lightDistance;float totalRoughness=saturate(lightRoughness+roughness);return totalRoughness; #else return roughness; #endif } vec3 computeHemisphericDiffuseLighting(preLightingInfo info,vec3 lightColor,vec3 groundColor) {return mix(groundColor,lightColor,info.NdotL);} vec3 computeDiffuseLighting(preLightingInfo info,vec3 lightColor) {float diffuseTerm=diffuseBRDF_Burley(info.NdotL,info.NdotV,info.VdotH,info.roughness);return diffuseTerm*info.attenuation*info.NdotL*lightColor;} #define inline vec3 computeProjectionTextureDiffuseLighting(sampler2D projectionLightSampler,mat4 textureProjectionMatrix){vec4 strq=textureProjectionMatrix*vec4(vPositionW,1.0);strq/=strq.w;vec3 textureColor=texture2D(projectionLightSampler,strq.xy).rgb;return toLinearSpace(textureColor);} #ifdef SS_TRANSLUCENCY vec3 computeDiffuseAndTransmittedLighting(preLightingInfo info,vec3 lightColor,vec3 transmittance) {float NdotL=absEps(info.NdotLUnclamped);float wrapNdotL=computeWrappedDiffuseNdotL(NdotL,0.02);float trAdapt=step(0.,info.NdotLUnclamped);vec3 transmittanceNdotL=mix(transmittance*wrapNdotL,vec3(wrapNdotL),trAdapt);float diffuseTerm=diffuseBRDF_Burley(NdotL,info.NdotV,info.VdotH,info.roughness);return diffuseTerm*transmittanceNdotL*info.attenuation*lightColor;} #endif #ifdef SPECULARTERM vec3 computeSpecularLighting(preLightingInfo info,vec3 N,vec3 reflectance0,vec3 reflectance90,float geometricRoughnessFactor,vec3 lightColor) {float NdotH=saturateEps(dot(N,info.H));float roughness=max(info.roughness,geometricRoughnessFactor);float alphaG=convertRoughnessToAverageSlope(roughness);vec3 fresnel=fresnelSchlickGGX(info.VdotH,reflectance0,reflectance90); #ifdef IRIDESCENCE fresnel=mix(fresnel,reflectance0,info.iridescenceIntensity); #endif float distribution=normalDistributionFunction_TrowbridgeReitzGGX(NdotH,alphaG); #ifdef BRDF_V_HEIGHT_CORRELATED float smithVisibility=smithVisibility_GGXCorrelated(info.NdotL,info.NdotV,alphaG); #else float smithVisibility=smithVisibility_TrowbridgeReitzGGXFast(info.NdotL,info.NdotV,alphaG); #endif vec3 specTerm=fresnel*distribution*smithVisibility;return specTerm*info.attenuation*info.NdotL*lightColor;} #endif #ifdef ANISOTROPIC vec3 computeAnisotropicSpecularLighting(preLightingInfo info,vec3 V,vec3 N,vec3 T,vec3 B,float anisotropy,vec3 reflectance0,vec3 reflectance90,float geometricRoughnessFactor,vec3 lightColor) {float NdotH=saturateEps(dot(N,info.H));float TdotH=dot(T,info.H);float BdotH=dot(B,info.H);float TdotV=dot(T,V);float BdotV=dot(B,V);float TdotL=dot(T,info.L);float BdotL=dot(B,info.L);float alphaG=convertRoughnessToAverageSlope(info.roughness);vec2 alphaTB=getAnisotropicRoughness(alphaG,anisotropy);alphaTB=max(alphaTB,square(geometricRoughnessFactor));vec3 fresnel=fresnelSchlickGGX(info.VdotH,reflectance0,reflectance90); #ifdef IRIDESCENCE fresnel=mix(fresnel,reflectance0,info.iridescenceIntensity); #endif float distribution=normalDistributionFunction_BurleyGGX_Anisotropic(NdotH,TdotH,BdotH,alphaTB);float smithVisibility=smithVisibility_GGXCorrelated_Anisotropic(info.NdotL,info.NdotV,TdotV,BdotV,TdotL,BdotL,alphaTB);vec3 specTerm=fresnel*distribution*smithVisibility;return specTerm*info.attenuation*info.NdotL*lightColor;} #endif #ifdef CLEARCOAT vec4 computeClearCoatLighting(preLightingInfo info,vec3 Ncc,float geometricRoughnessFactor,float clearCoatIntensity,vec3 lightColor) {float NccdotL=saturateEps(dot(Ncc,info.L));float NccdotH=saturateEps(dot(Ncc,info.H));float clearCoatRoughness=max(info.roughness,geometricRoughnessFactor);float alphaG=convertRoughnessToAverageSlope(clearCoatRoughness);float fresnel=fresnelSchlickGGX(info.VdotH,vClearCoatRefractionParams.x,CLEARCOATREFLECTANCE90);fresnel*=clearCoatIntensity;float distribution=normalDistributionFunction_TrowbridgeReitzGGX(NccdotH,alphaG);float kelemenVisibility=visibility_Kelemen(info.VdotH);float clearCoatTerm=fresnel*distribution*kelemenVisibility;return vec4( clearCoatTerm*info.attenuation*NccdotL*lightColor, 1.0-fresnel );} vec3 computeClearCoatLightingAbsorption(float NdotVRefract,vec3 L,vec3 Ncc,vec3 clearCoatColor,float clearCoatThickness,float clearCoatIntensity) {vec3 LRefract=-refract(L,Ncc,vClearCoatRefractionParams.y);float NdotLRefract=saturateEps(dot(Ncc,LRefract));vec3 absorption=computeClearCoatAbsorption(NdotVRefract,NdotLRefract,clearCoatColor,clearCoatThickness,clearCoatIntensity);return absorption;} #endif #ifdef SHEEN vec3 computeSheenLighting(preLightingInfo info,vec3 N,vec3 reflectance0,vec3 reflectance90,float geometricRoughnessFactor,vec3 lightColor) {float NdotH=saturateEps(dot(N,info.H));float roughness=max(info.roughness,geometricRoughnessFactor);float alphaG=convertRoughnessToAverageSlope(roughness);float fresnel=1.;float distribution=normalDistributionFunction_CharlieSheen(NdotH,alphaG);/*#ifdef SHEEN_SOFTER float visibility=visibility_CharlieSheen(info.NdotL,info.NdotV,alphaG); #else */ float visibility=visibility_Ashikhmin(info.NdotL,info.NdotV);/* #endif */ float sheenTerm=fresnel*distribution*visibility;return sheenTerm*info.attenuation*info.NdotL*lightColor;} #endif `; Le.IncludesShadersStore[ZPe] = _Pe; const $Pe = "pbrIBLFunctions", ece = `#if defined(REFLECTION) || defined(SS_REFRACTION) float getLodFromAlphaG(float cubeMapDimensionPixels,float microsurfaceAverageSlope) {float microsurfaceAverageSlopeTexels=cubeMapDimensionPixels*microsurfaceAverageSlope;float lod=log2(microsurfaceAverageSlopeTexels);return lod;} float getLinearLodFromRoughness(float cubeMapDimensionPixels,float roughness) {float lod=log2(cubeMapDimensionPixels)*roughness;return lod;} #endif #if defined(ENVIRONMENTBRDF) && defined(RADIANCEOCCLUSION) float environmentRadianceOcclusion(float ambientOcclusion,float NdotVUnclamped) {float temp=NdotVUnclamped+ambientOcclusion;return saturate(square(temp)-1.0+ambientOcclusion);} #endif #if defined(ENVIRONMENTBRDF) && defined(HORIZONOCCLUSION) float environmentHorizonOcclusion(vec3 view,vec3 normal,vec3 geometricNormal) {vec3 reflection=reflect(view,normal);float temp=saturate(1.0+1.1*dot(reflection,geometricNormal));return square(temp);} #endif #if defined(LODINREFLECTIONALPHA) || defined(SS_LODINREFRACTIONALPHA) #define UNPACK_LOD(x) (1.0-x)*255.0 float getLodFromAlphaG(float cubeMapDimensionPixels,float alphaG,float NdotV) {float microsurfaceAverageSlope=alphaG;microsurfaceAverageSlope*=sqrt(abs(NdotV));return getLodFromAlphaG(cubeMapDimensionPixels,microsurfaceAverageSlope);} #endif `; Le.IncludesShadersStore[$Pe] = ece; const tce = "pbrBlockAlbedoOpacity", rce = `struct albedoOpacityOutParams {vec3 surfaceAlbedo;float alpha;}; #define pbr_inline void albedoOpacityBlock( in vec4 vAlbedoColor, #ifdef ALBEDO in vec4 albedoTexture, in vec2 albedoInfos, #endif #ifdef OPACITY in vec4 opacityMap, in vec2 vOpacityInfos, #endif #ifdef DETAIL in vec4 detailColor, in vec4 vDetailInfos, #endif #ifdef DECAL in vec4 decalColor, in vec4 vDecalInfos, #endif out albedoOpacityOutParams outParams ) {vec3 surfaceAlbedo=vAlbedoColor.rgb;float alpha=vAlbedoColor.a; #ifdef ALBEDO #if defined(ALPHAFROMALBEDO) || defined(ALPHATEST) alpha*=albedoTexture.a; #endif #ifdef GAMMAALBEDO surfaceAlbedo*=toLinearSpace(albedoTexture.rgb); #else surfaceAlbedo*=albedoTexture.rgb; #endif surfaceAlbedo*=albedoInfos.y; #endif #ifndef DECAL_AFTER_DETAIL #include #endif #if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES) surfaceAlbedo*=vColor.rgb; #endif #ifdef DETAIL float detailAlbedo=2.0*mix(0.5,detailColor.r,vDetailInfos.y);surfaceAlbedo.rgb=surfaceAlbedo.rgb*detailAlbedo*detailAlbedo; #endif #ifdef DECAL_AFTER_DETAIL #include #endif #define CUSTOM_FRAGMENT_UPDATE_ALBEDO #ifdef OPACITY #ifdef OPACITYRGB alpha=getLuminance(opacityMap.rgb); #else alpha*=opacityMap.a; #endif alpha*=vOpacityInfos.y; #endif #if defined(VERTEXALPHA) || defined(INSTANCESCOLOR) && defined(INSTANCES) alpha*=vColor.a; #endif #if !defined(SS_LINKREFRACTIONTOTRANSPARENCY) && !defined(ALPHAFRESNEL) #ifdef ALPHATEST #if DEBUGMODE != 88 if (alpha0 #ifdef METALLICWORKFLOW vec2 metallicRoughness; #ifdef REFLECTIVITY vec4 surfaceMetallicColorMap; #endif #ifndef FROSTBITE_REFLECTANCE vec3 metallicF0; #endif #else #ifdef REFLECTIVITY vec4 surfaceReflectivityColorMap; #endif #endif #endif }; #define pbr_inline void reflectivityBlock( in vec4 vReflectivityColor, #ifdef METALLICWORKFLOW in vec3 surfaceAlbedo, in vec4 metallicReflectanceFactors, #endif #ifdef REFLECTIVITY in vec3 reflectivityInfos, in vec4 surfaceMetallicOrReflectivityColorMap, #endif #if defined(METALLICWORKFLOW) && defined(REFLECTIVITY) && defined(AOSTOREINMETALMAPRED) in vec3 ambientOcclusionColorIn, #endif #ifdef MICROSURFACEMAP in vec4 microSurfaceTexel, #endif #ifdef DETAIL in vec4 detailColor, in vec4 vDetailInfos, #endif out reflectivityOutParams outParams ) {float microSurface=vReflectivityColor.a;vec3 surfaceReflectivityColor=vReflectivityColor.rgb; #ifdef METALLICWORKFLOW vec2 metallicRoughness=surfaceReflectivityColor.rg; #ifdef REFLECTIVITY #if DEBUGMODE>0 outParams.surfaceMetallicColorMap=surfaceMetallicOrReflectivityColorMap; #endif #ifdef AOSTOREINMETALMAPRED vec3 aoStoreInMetalMap=vec3(surfaceMetallicOrReflectivityColorMap.r,surfaceMetallicOrReflectivityColorMap.r,surfaceMetallicOrReflectivityColorMap.r);outParams.ambientOcclusionColor=mix(ambientOcclusionColorIn,aoStoreInMetalMap,reflectivityInfos.z); #endif #ifdef METALLNESSSTOREINMETALMAPBLUE metallicRoughness.r*=surfaceMetallicOrReflectivityColorMap.b; #else metallicRoughness.r*=surfaceMetallicOrReflectivityColorMap.r; #endif #ifdef ROUGHNESSSTOREINMETALMAPALPHA metallicRoughness.g*=surfaceMetallicOrReflectivityColorMap.a; #else #ifdef ROUGHNESSSTOREINMETALMAPGREEN metallicRoughness.g*=surfaceMetallicOrReflectivityColorMap.g; #endif #endif #endif #ifdef DETAIL float detailRoughness=mix(0.5,detailColor.b,vDetailInfos.w);float loLerp=mix(0.,metallicRoughness.g,detailRoughness*2.);float hiLerp=mix(metallicRoughness.g,1.,(detailRoughness-0.5)*2.);metallicRoughness.g=mix(loLerp,hiLerp,step(detailRoughness,0.5)); #endif #ifdef MICROSURFACEMAP metallicRoughness.g*=microSurfaceTexel.r; #endif #if DEBUGMODE>0 outParams.metallicRoughness=metallicRoughness; #endif #define CUSTOM_FRAGMENT_UPDATE_METALLICROUGHNESS microSurface=1.0-metallicRoughness.g;vec3 baseColor=surfaceAlbedo; #ifdef FROSTBITE_REFLECTANCE outParams.surfaceAlbedo=baseColor.rgb*(1.0-metallicRoughness.r);surfaceReflectivityColor=mix(0.16*reflectance*reflectance,baseColor,metallicRoughness.r); #else vec3 metallicF0=metallicReflectanceFactors.rgb; #if DEBUGMODE>0 outParams.metallicF0=metallicF0; #endif outParams.surfaceAlbedo=mix(baseColor.rgb*(1.0-metallicF0),vec3(0.,0.,0.),metallicRoughness.r);surfaceReflectivityColor=mix(metallicF0,baseColor,metallicRoughness.r); #endif #else #ifdef REFLECTIVITY surfaceReflectivityColor*=surfaceMetallicOrReflectivityColorMap.rgb; #if DEBUGMODE>0 outParams.surfaceReflectivityColorMap=surfaceMetallicOrReflectivityColorMap; #endif #ifdef MICROSURFACEFROMREFLECTIVITYMAP microSurface*=surfaceMetallicOrReflectivityColorMap.a;microSurface*=reflectivityInfos.z; #else #ifdef MICROSURFACEAUTOMATIC microSurface*=computeDefaultMicroSurface(microSurface,surfaceReflectivityColor); #endif #ifdef MICROSURFACEMAP microSurface*=microSurfaceTexel.r; #endif #define CUSTOM_FRAGMENT_UPDATE_MICROSURFACE #endif #endif #endif microSurface=saturate(microSurface);float roughness=1.-microSurface;outParams.microSurface=microSurface;outParams.roughness=roughness;outParams.surfaceReflectivityColor=surfaceReflectivityColor;} `; Le.IncludesShadersStore[nce] = ice; const sce = "pbrBlockAmbientOcclusion", ace = `struct ambientOcclusionOutParams {vec3 ambientOcclusionColor; #if DEBUGMODE>0 && defined(AMBIENT) vec3 ambientOcclusionColorMap; #endif }; #define pbr_inline void ambientOcclusionBlock( #ifdef AMBIENT in vec3 ambientOcclusionColorMap_, in vec4 vAmbientInfos, #endif out ambientOcclusionOutParams outParams ) {vec3 ambientOcclusionColor=vec3(1.,1.,1.); #ifdef AMBIENT vec3 ambientOcclusionColorMap=ambientOcclusionColorMap_*vAmbientInfos.y; #ifdef AMBIENTINGRAYSCALE ambientOcclusionColorMap=vec3(ambientOcclusionColorMap.r,ambientOcclusionColorMap.r,ambientOcclusionColorMap.r); #endif ambientOcclusionColor=mix(ambientOcclusionColor,ambientOcclusionColorMap,vAmbientInfos.z); #if DEBUGMODE>0 outParams.ambientOcclusionColorMap=ambientOcclusionColorMap; #endif #endif outParams.ambientOcclusionColor=ambientOcclusionColor;} `; Le.IncludesShadersStore[sce] = ace; const oce = "pbrBlockAlphaFresnel", fce = `#ifdef ALPHAFRESNEL #if defined(ALPHATEST) || defined(ALPHABLEND) struct alphaFresnelOutParams {float alpha;}; #define pbr_inline void alphaFresnelBlock( in vec3 normalW, in vec3 viewDirectionW, in float alpha, in float microSurface, out alphaFresnelOutParams outParams ) {float opacityPerceptual=alpha; #ifdef LINEARALPHAFRESNEL float opacity0=opacityPerceptual; #else float opacity0=opacityPerceptual*opacityPerceptual; #endif float opacity90=fresnelGrazingReflectance(opacity0);vec3 normalForward=faceforward(normalW,-viewDirectionW,normalW);outParams.alpha=getReflectanceFromAnalyticalBRDFLookup_Jones(saturate(dot(viewDirectionW,normalForward)),vec3(opacity0),vec3(opacity90),sqrt(microSurface)).x; #ifdef ALPHATEST if (outParams.alpha0 && defined(ANISOTROPIC_TEXTURE) vec3 anisotropyMapData; #endif }; #define pbr_inline void anisotropicBlock( in vec3 vAnisotropy, in float roughness, #ifdef ANISOTROPIC_TEXTURE in vec3 anisotropyMapData, #endif in mat3 TBN, in vec3 normalW, in vec3 viewDirectionW, out anisotropicOutParams outParams ) {float anisotropy=vAnisotropy.b;vec3 anisotropyDirection=vec3(vAnisotropy.xy,0.); #ifdef ANISOTROPIC_TEXTURE anisotropy*=anisotropyMapData.b; #if DEBUGMODE>0 outParams.anisotropyMapData=anisotropyMapData; #endif anisotropyMapData.rg=anisotropyMapData.rg*2.0-1.0; #ifdef ANISOTROPIC_LEGACY anisotropyDirection.rg*=anisotropyMapData.rg; #else anisotropyDirection.xy=mat2(anisotropyDirection.x,anisotropyDirection.y,-anisotropyDirection.y,anisotropyDirection.x)*normalize(anisotropyMapData.rg); #endif #endif mat3 anisoTBN=mat3(normalize(TBN[0]),normalize(TBN[1]),normalize(TBN[2]));vec3 anisotropicTangent=normalize(anisoTBN*anisotropyDirection);vec3 anisotropicBitangent=normalize(cross(anisoTBN[2],anisotropicTangent));outParams.anisotropy=anisotropy;outParams.anisotropicTangent=anisotropicTangent;outParams.anisotropicBitangent=anisotropicBitangent;outParams.anisotropicNormal=getAnisotropicBentNormals(anisotropicTangent,anisotropicBitangent,normalW,viewDirectionW,anisotropy,roughness);} #endif `; Le.IncludesShadersStore[Ace] = dce; const vce = "pbrBlockReflection", uce = `#ifdef REFLECTION struct reflectionOutParams {vec4 environmentRadiance;vec3 environmentIrradiance; #ifdef REFLECTIONMAP_3D vec3 reflectionCoords; #else vec2 reflectionCoords; #endif #ifdef SS_TRANSLUCENCY #ifdef USESPHERICALFROMREFLECTIONMAP #if !defined(NORMAL) || !defined(USESPHERICALINVERTEX) vec3 irradianceVector; #endif #endif #endif }; #define pbr_inline void createReflectionCoords( in vec3 vPositionW, in vec3 normalW, #ifdef ANISOTROPIC in anisotropicOutParams anisotropicOut, #endif #ifdef REFLECTIONMAP_3D out vec3 reflectionCoords #else out vec2 reflectionCoords #endif ) { #ifdef ANISOTROPIC vec3 reflectionVector=computeReflectionCoords(vec4(vPositionW,1.0),anisotropicOut.anisotropicNormal); #else vec3 reflectionVector=computeReflectionCoords(vec4(vPositionW,1.0),normalW); #endif #ifdef REFLECTIONMAP_OPPOSITEZ reflectionVector.z*=-1.0; #endif #ifdef REFLECTIONMAP_3D reflectionCoords=reflectionVector; #else reflectionCoords=reflectionVector.xy; #ifdef REFLECTIONMAP_PROJECTION reflectionCoords/=reflectionVector.z; #endif reflectionCoords.y=1.0-reflectionCoords.y; #endif } #define pbr_inline #define inline void sampleReflectionTexture( in float alphaG, in vec3 vReflectionMicrosurfaceInfos, in vec2 vReflectionInfos, in vec3 vReflectionColor, #if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX) in float NdotVUnclamped, #endif #ifdef LINEARSPECULARREFLECTION in float roughness, #endif #ifdef REFLECTIONMAP_3D in samplerCube reflectionSampler, const vec3 reflectionCoords, #else in sampler2D reflectionSampler, const vec2 reflectionCoords, #endif #ifndef LODBASEDMICROSFURACE #ifdef REFLECTIONMAP_3D in samplerCube reflectionSamplerLow, in samplerCube reflectionSamplerHigh, #else in sampler2D reflectionSamplerLow, in sampler2D reflectionSamplerHigh, #endif #endif #ifdef REALTIME_FILTERING in vec2 vReflectionFilteringInfo, #endif out vec4 environmentRadiance ) { #if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX) float reflectionLOD=getLodFromAlphaG(vReflectionMicrosurfaceInfos.x,alphaG,NdotVUnclamped); #elif defined(LINEARSPECULARREFLECTION) float reflectionLOD=getLinearLodFromRoughness(vReflectionMicrosurfaceInfos.x,roughness); #else float reflectionLOD=getLodFromAlphaG(vReflectionMicrosurfaceInfos.x,alphaG); #endif #ifdef LODBASEDMICROSFURACE reflectionLOD=reflectionLOD*vReflectionMicrosurfaceInfos.y+vReflectionMicrosurfaceInfos.z; #ifdef LODINREFLECTIONALPHA float automaticReflectionLOD=UNPACK_LOD(sampleReflection(reflectionSampler,reflectionCoords).a);float requestedReflectionLOD=max(automaticReflectionLOD,reflectionLOD); #else float requestedReflectionLOD=reflectionLOD; #endif #ifdef REALTIME_FILTERING environmentRadiance=vec4(radiance(alphaG,reflectionSampler,reflectionCoords,vReflectionFilteringInfo),1.0); #else environmentRadiance=sampleReflectionLod(reflectionSampler,reflectionCoords,reflectionLOD); #endif #else float lodReflectionNormalized=saturate(reflectionLOD/log2(vReflectionMicrosurfaceInfos.x));float lodReflectionNormalizedDoubled=lodReflectionNormalized*2.0;vec4 environmentMid=sampleReflection(reflectionSampler,reflectionCoords);if (lodReflectionNormalizedDoubled<1.0){environmentRadiance=mix( sampleReflection(reflectionSamplerHigh,reflectionCoords), environmentMid, lodReflectionNormalizedDoubled );} else {environmentRadiance=mix( environmentMid, sampleReflection(reflectionSamplerLow,reflectionCoords), lodReflectionNormalizedDoubled-1.0 );} #endif #ifdef RGBDREFLECTION environmentRadiance.rgb=fromRGBD(environmentRadiance); #endif #ifdef GAMMAREFLECTION environmentRadiance.rgb=toLinearSpace(environmentRadiance.rgb); #endif environmentRadiance.rgb*=vReflectionInfos.x;environmentRadiance.rgb*=vReflectionColor.rgb;} #define pbr_inline #define inline void reflectionBlock( in vec3 vPositionW, in vec3 normalW, in float alphaG, in vec3 vReflectionMicrosurfaceInfos, in vec2 vReflectionInfos, in vec3 vReflectionColor, #ifdef ANISOTROPIC in anisotropicOutParams anisotropicOut, #endif #if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX) in float NdotVUnclamped, #endif #ifdef LINEARSPECULARREFLECTION in float roughness, #endif #ifdef REFLECTIONMAP_3D in samplerCube reflectionSampler, #else in sampler2D reflectionSampler, #endif #if defined(NORMAL) && defined(USESPHERICALINVERTEX) in vec3 vEnvironmentIrradiance, #endif #ifdef USESPHERICALFROMREFLECTIONMAP #if !defined(NORMAL) || !defined(USESPHERICALINVERTEX) in mat4 reflectionMatrix, #endif #endif #ifdef USEIRRADIANCEMAP #ifdef REFLECTIONMAP_3D in samplerCube irradianceSampler, #else in sampler2D irradianceSampler, #endif #endif #ifndef LODBASEDMICROSFURACE #ifdef REFLECTIONMAP_3D in samplerCube reflectionSamplerLow, in samplerCube reflectionSamplerHigh, #else in sampler2D reflectionSamplerLow, in sampler2D reflectionSamplerHigh, #endif #endif #ifdef REALTIME_FILTERING in vec2 vReflectionFilteringInfo, #endif out reflectionOutParams outParams ) {vec4 environmentRadiance=vec4(0.,0.,0.,0.); #ifdef REFLECTIONMAP_3D vec3 reflectionCoords=vec3(0.); #else vec2 reflectionCoords=vec2(0.); #endif createReflectionCoords( vPositionW, normalW, #ifdef ANISOTROPIC anisotropicOut, #endif reflectionCoords );sampleReflectionTexture( alphaG, vReflectionMicrosurfaceInfos, vReflectionInfos, vReflectionColor, #if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX) NdotVUnclamped, #endif #ifdef LINEARSPECULARREFLECTION roughness, #endif #ifdef REFLECTIONMAP_3D reflectionSampler, reflectionCoords, #else reflectionSampler, reflectionCoords, #endif #ifndef LODBASEDMICROSFURACE reflectionSamplerLow, reflectionSamplerHigh, #endif #ifdef REALTIME_FILTERING vReflectionFilteringInfo, #endif environmentRadiance );vec3 environmentIrradiance=vec3(0.,0.,0.); #ifdef USESPHERICALFROMREFLECTIONMAP #if defined(NORMAL) && defined(USESPHERICALINVERTEX) environmentIrradiance=vEnvironmentIrradiance; #else #ifdef ANISOTROPIC vec3 irradianceVector=vec3(reflectionMatrix*vec4(anisotropicOut.anisotropicNormal,0)).xyz; #else vec3 irradianceVector=vec3(reflectionMatrix*vec4(normalW,0)).xyz; #endif #ifdef REFLECTIONMAP_OPPOSITEZ irradianceVector.z*=-1.0; #endif #ifdef INVERTCUBICMAP irradianceVector.y*=-1.0; #endif #if defined(REALTIME_FILTERING) environmentIrradiance=irradiance(reflectionSampler,irradianceVector,vReflectionFilteringInfo); #else environmentIrradiance=computeEnvironmentIrradiance(irradianceVector); #endif #ifdef SS_TRANSLUCENCY outParams.irradianceVector=irradianceVector; #endif #endif #elif defined(USEIRRADIANCEMAP) vec4 environmentIrradiance4=sampleReflection(irradianceSampler,reflectionCoords);environmentIrradiance=environmentIrradiance4.rgb; #ifdef RGBDREFLECTION environmentIrradiance.rgb=fromRGBD(environmentIrradiance4); #endif #ifdef GAMMAREFLECTION environmentIrradiance.rgb=toLinearSpace(environmentIrradiance.rgb); #endif #endif environmentIrradiance*=vReflectionColor.rgb;outParams.environmentRadiance=environmentRadiance;outParams.environmentIrradiance=environmentIrradiance;outParams.reflectionCoords=reflectionCoords;} #endif `; Le.IncludesShadersStore[vce] = uce; const lce = "pbrBlockSheen", Pce = `#ifdef SHEEN struct sheenOutParams {float sheenIntensity;vec3 sheenColor;float sheenRoughness; #ifdef SHEEN_LINKWITHALBEDO vec3 surfaceAlbedo; #endif #if defined(ENVIRONMENTBRDF) && defined(SHEEN_ALBEDOSCALING) float sheenAlbedoScaling; #endif #if defined(REFLECTION) && defined(ENVIRONMENTBRDF) vec3 finalSheenRadianceScaled; #endif #if DEBUGMODE>0 #ifdef SHEEN_TEXTURE vec4 sheenMapData; #endif #if defined(REFLECTION) && defined(ENVIRONMENTBRDF) vec3 sheenEnvironmentReflectance; #endif #endif }; #define pbr_inline #define inline void sheenBlock( in vec4 vSheenColor, #ifdef SHEEN_ROUGHNESS in float vSheenRoughness, #if defined(SHEEN_TEXTURE_ROUGHNESS) && !defined(SHEEN_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE) in vec4 sheenMapRoughnessData, #endif #endif in float roughness, #ifdef SHEEN_TEXTURE in vec4 sheenMapData, in float sheenMapLevel, #endif in float reflectance, #ifdef SHEEN_LINKWITHALBEDO in vec3 baseColor, in vec3 surfaceAlbedo, #endif #ifdef ENVIRONMENTBRDF in float NdotV, in vec3 environmentBrdf, #endif #if defined(REFLECTION) && defined(ENVIRONMENTBRDF) in vec2 AARoughnessFactors, in vec3 vReflectionMicrosurfaceInfos, in vec2 vReflectionInfos, in vec3 vReflectionColor, in vec4 vLightingIntensity, #ifdef REFLECTIONMAP_3D in samplerCube reflectionSampler, in vec3 reflectionCoords, #else in sampler2D reflectionSampler, in vec2 reflectionCoords, #endif in float NdotVUnclamped, #ifndef LODBASEDMICROSFURACE #ifdef REFLECTIONMAP_3D in samplerCube reflectionSamplerLow, in samplerCube reflectionSamplerHigh, #else in sampler2D reflectionSamplerLow, in sampler2D reflectionSamplerHigh, #endif #endif #ifdef REALTIME_FILTERING in vec2 vReflectionFilteringInfo, #endif #if !defined(REFLECTIONMAP_SKYBOX) && defined(RADIANCEOCCLUSION) in float seo, #endif #if !defined(REFLECTIONMAP_SKYBOX) && defined(HORIZONOCCLUSION) && defined(BUMP) && defined(REFLECTIONMAP_3D) in float eho, #endif #endif out sheenOutParams outParams ) {float sheenIntensity=vSheenColor.a; #ifdef SHEEN_TEXTURE #if DEBUGMODE>0 outParams.sheenMapData=sheenMapData; #endif #endif #ifdef SHEEN_LINKWITHALBEDO float sheenFactor=pow5(1.0-sheenIntensity);vec3 sheenColor=baseColor.rgb*(1.0-sheenFactor);float sheenRoughness=sheenIntensity;outParams.surfaceAlbedo=surfaceAlbedo*sheenFactor; #ifdef SHEEN_TEXTURE sheenIntensity*=sheenMapData.a; #endif #else vec3 sheenColor=vSheenColor.rgb; #ifdef SHEEN_TEXTURE #ifdef SHEEN_GAMMATEXTURE sheenColor.rgb*=toLinearSpace(sheenMapData.rgb); #else sheenColor.rgb*=sheenMapData.rgb; #endif sheenColor.rgb*=sheenMapLevel; #endif #ifdef SHEEN_ROUGHNESS float sheenRoughness=vSheenRoughness; #ifdef SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE #if defined(SHEEN_TEXTURE) sheenRoughness*=sheenMapData.a; #endif #elif defined(SHEEN_TEXTURE_ROUGHNESS) #ifdef SHEEN_TEXTURE_ROUGHNESS_IDENTICAL sheenRoughness*=sheenMapData.a; #else sheenRoughness*=sheenMapRoughnessData.a; #endif #endif #else float sheenRoughness=roughness; #ifdef SHEEN_TEXTURE sheenIntensity*=sheenMapData.a; #endif #endif #if !defined(SHEEN_ALBEDOSCALING) sheenIntensity*=(1.-reflectance); #endif sheenColor*=sheenIntensity; #endif #ifdef ENVIRONMENTBRDF /*#ifdef SHEEN_SOFTER vec3 environmentSheenBrdf=vec3(0.,0.,getBRDFLookupCharlieSheen(NdotV,sheenRoughness)); #else*/ #ifdef SHEEN_ROUGHNESS vec3 environmentSheenBrdf=getBRDFLookup(NdotV,sheenRoughness); #else vec3 environmentSheenBrdf=environmentBrdf; #endif /*#endif*/ #endif #if defined(REFLECTION) && defined(ENVIRONMENTBRDF) float sheenAlphaG=convertRoughnessToAverageSlope(sheenRoughness); #ifdef SPECULARAA sheenAlphaG+=AARoughnessFactors.y; #endif vec4 environmentSheenRadiance=vec4(0.,0.,0.,0.);sampleReflectionTexture( sheenAlphaG, vReflectionMicrosurfaceInfos, vReflectionInfos, vReflectionColor, #if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX) NdotVUnclamped, #endif #ifdef LINEARSPECULARREFLECTION sheenRoughness, #endif reflectionSampler, reflectionCoords, #ifndef LODBASEDMICROSFURACE reflectionSamplerLow, reflectionSamplerHigh, #endif #ifdef REALTIME_FILTERING vReflectionFilteringInfo, #endif environmentSheenRadiance );vec3 sheenEnvironmentReflectance=getSheenReflectanceFromBRDFLookup(sheenColor,environmentSheenBrdf); #if !defined(REFLECTIONMAP_SKYBOX) && defined(RADIANCEOCCLUSION) sheenEnvironmentReflectance*=seo; #endif #if !defined(REFLECTIONMAP_SKYBOX) && defined(HORIZONOCCLUSION) && defined(BUMP) && defined(REFLECTIONMAP_3D) sheenEnvironmentReflectance*=eho; #endif #if DEBUGMODE>0 outParams.sheenEnvironmentReflectance=sheenEnvironmentReflectance; #endif outParams.finalSheenRadianceScaled= environmentSheenRadiance.rgb * sheenEnvironmentReflectance * vLightingIntensity.z; #endif #if defined(ENVIRONMENTBRDF) && defined(SHEEN_ALBEDOSCALING) outParams.sheenAlbedoScaling=1.0-sheenIntensity*max(max(sheenColor.r,sheenColor.g),sheenColor.b)*environmentSheenBrdf.b; #endif outParams.sheenIntensity=sheenIntensity;outParams.sheenColor=sheenColor;outParams.sheenRoughness=sheenRoughness;} #endif `; Le.IncludesShadersStore[lce] = Pce; const cce = "pbrBlockClearcoat", pce = `struct clearcoatOutParams {vec3 specularEnvironmentR0;float conservationFactor;vec3 clearCoatNormalW;vec2 clearCoatAARoughnessFactors;float clearCoatIntensity;float clearCoatRoughness; #ifdef REFLECTION vec3 finalClearCoatRadianceScaled; #endif #ifdef CLEARCOAT_TINT vec3 absorption;float clearCoatNdotVRefract;vec3 clearCoatColor;float clearCoatThickness; #endif #if defined(ENVIRONMENTBRDF) && defined(MS_BRDF_ENERGY_CONSERVATION) vec3 energyConservationFactorClearCoat; #endif #if DEBUGMODE>0 #ifdef CLEARCOAT_BUMP mat3 TBNClearCoat; #endif #ifdef CLEARCOAT_TEXTURE vec2 clearCoatMapData; #endif #if defined(CLEARCOAT_TINT) && defined(CLEARCOAT_TINT_TEXTURE) vec4 clearCoatTintMapData; #endif #ifdef REFLECTION vec4 environmentClearCoatRadiance;vec3 clearCoatEnvironmentReflectance; #endif float clearCoatNdotV; #endif }; #ifdef CLEARCOAT #define pbr_inline #define inline void clearcoatBlock( in vec3 vPositionW, in vec3 geometricNormalW, in vec3 viewDirectionW, in vec2 vClearCoatParams, #if defined(CLEARCOAT_TEXTURE_ROUGHNESS) && !defined(CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE) in vec4 clearCoatMapRoughnessData, #endif in vec3 specularEnvironmentR0, #ifdef CLEARCOAT_TEXTURE in vec2 clearCoatMapData, #endif #ifdef CLEARCOAT_TINT in vec4 vClearCoatTintParams, in float clearCoatColorAtDistance, in vec4 vClearCoatRefractionParams, #ifdef CLEARCOAT_TINT_TEXTURE in vec4 clearCoatTintMapData, #endif #endif #ifdef CLEARCOAT_BUMP in vec2 vClearCoatBumpInfos, in vec4 clearCoatBumpMapData, in vec2 vClearCoatBumpUV, #if defined(TANGENT) && defined(NORMAL) in mat3 vTBN, #else in vec2 vClearCoatTangentSpaceParams, #endif #ifdef OBJECTSPACE_NORMALMAP in mat4 normalMatrix, #endif #endif #if defined(FORCENORMALFORWARD) && defined(NORMAL) in vec3 faceNormal, #endif #ifdef REFLECTION in vec3 vReflectionMicrosurfaceInfos, in vec2 vReflectionInfos, in vec3 vReflectionColor, in vec4 vLightingIntensity, #ifdef REFLECTIONMAP_3D in samplerCube reflectionSampler, #else in sampler2D reflectionSampler, #endif #ifndef LODBASEDMICROSFURACE #ifdef REFLECTIONMAP_3D in samplerCube reflectionSamplerLow, in samplerCube reflectionSamplerHigh, #else in sampler2D reflectionSamplerLow, in sampler2D reflectionSamplerHigh, #endif #endif #ifdef REALTIME_FILTERING in vec2 vReflectionFilteringInfo, #endif #endif #if defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX) #ifdef RADIANCEOCCLUSION in float ambientMonochrome, #endif #endif #if defined(CLEARCOAT_BUMP) || defined(TWOSIDEDLIGHTING) in float frontFacingMultiplier, #endif out clearcoatOutParams outParams ) {float clearCoatIntensity=vClearCoatParams.x;float clearCoatRoughness=vClearCoatParams.y; #ifdef CLEARCOAT_TEXTURE clearCoatIntensity*=clearCoatMapData.x; #ifdef CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE clearCoatRoughness*=clearCoatMapData.y; #endif #if DEBUGMODE>0 outParams.clearCoatMapData=clearCoatMapData; #endif #endif #if defined(CLEARCOAT_TEXTURE_ROUGHNESS) && !defined(CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE) #ifdef CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL clearCoatRoughness*=clearCoatMapData.y; #else clearCoatRoughness*=clearCoatMapRoughnessData.y; #endif #endif outParams.clearCoatIntensity=clearCoatIntensity;outParams.clearCoatRoughness=clearCoatRoughness; #ifdef CLEARCOAT_TINT vec3 clearCoatColor=vClearCoatTintParams.rgb;float clearCoatThickness=vClearCoatTintParams.a; #ifdef CLEARCOAT_TINT_TEXTURE #ifdef CLEARCOAT_TINT_GAMMATEXTURE clearCoatColor*=toLinearSpace(clearCoatTintMapData.rgb); #else clearCoatColor*=clearCoatTintMapData.rgb; #endif clearCoatThickness*=clearCoatTintMapData.a; #if DEBUGMODE>0 outParams.clearCoatTintMapData=clearCoatTintMapData; #endif #endif outParams.clearCoatColor=computeColorAtDistanceInMedia(clearCoatColor,clearCoatColorAtDistance);outParams.clearCoatThickness=clearCoatThickness; #endif #ifdef CLEARCOAT_REMAP_F0 vec3 specularEnvironmentR0Updated=getR0RemappedForClearCoat(specularEnvironmentR0); #else vec3 specularEnvironmentR0Updated=specularEnvironmentR0; #endif outParams.specularEnvironmentR0=mix(specularEnvironmentR0,specularEnvironmentR0Updated,clearCoatIntensity);vec3 clearCoatNormalW=geometricNormalW; #ifdef CLEARCOAT_BUMP #ifdef NORMALXYSCALE float clearCoatNormalScale=1.0; #else float clearCoatNormalScale=vClearCoatBumpInfos.y; #endif #if defined(TANGENT) && defined(NORMAL) mat3 TBNClearCoat=vTBN; #else vec2 TBNClearCoatUV=vClearCoatBumpUV*frontFacingMultiplier;mat3 TBNClearCoat=cotangent_frame(clearCoatNormalW*clearCoatNormalScale,vPositionW,TBNClearCoatUV,vClearCoatTangentSpaceParams); #endif #if DEBUGMODE>0 outParams.TBNClearCoat=TBNClearCoat; #endif #ifdef OBJECTSPACE_NORMALMAP clearCoatNormalW=normalize(clearCoatBumpMapData.xyz *2.0-1.0);clearCoatNormalW=normalize(mat3(normalMatrix)*clearCoatNormalW); #else clearCoatNormalW=perturbNormal(TBNClearCoat,clearCoatBumpMapData.xyz,vClearCoatBumpInfos.y); #endif #endif #if defined(FORCENORMALFORWARD) && defined(NORMAL) clearCoatNormalW*=sign(dot(clearCoatNormalW,faceNormal)); #endif #if defined(TWOSIDEDLIGHTING) && defined(NORMAL) clearCoatNormalW=clearCoatNormalW*frontFacingMultiplier; #endif outParams.clearCoatNormalW=clearCoatNormalW;outParams.clearCoatAARoughnessFactors=getAARoughnessFactors(clearCoatNormalW.xyz);float clearCoatNdotVUnclamped=dot(clearCoatNormalW,viewDirectionW);float clearCoatNdotV=absEps(clearCoatNdotVUnclamped); #if DEBUGMODE>0 outParams.clearCoatNdotV=clearCoatNdotV; #endif #ifdef CLEARCOAT_TINT vec3 clearCoatVRefract=refract(-viewDirectionW,clearCoatNormalW,vClearCoatRefractionParams.y);outParams.clearCoatNdotVRefract=absEps(dot(clearCoatNormalW,clearCoatVRefract)); #endif #if defined(ENVIRONMENTBRDF) && (!defined(REFLECTIONMAP_SKYBOX) || defined(MS_BRDF_ENERGY_CONSERVATION)) vec3 environmentClearCoatBrdf=getBRDFLookup(clearCoatNdotV,clearCoatRoughness); #endif #if defined(REFLECTION) float clearCoatAlphaG=convertRoughnessToAverageSlope(clearCoatRoughness); #ifdef SPECULARAA clearCoatAlphaG+=outParams.clearCoatAARoughnessFactors.y; #endif vec4 environmentClearCoatRadiance=vec4(0.,0.,0.,0.);vec3 clearCoatReflectionVector=computeReflectionCoords(vec4(vPositionW,1.0),clearCoatNormalW); #ifdef REFLECTIONMAP_OPPOSITEZ clearCoatReflectionVector.z*=-1.0; #endif #ifdef REFLECTIONMAP_3D vec3 clearCoatReflectionCoords=clearCoatReflectionVector; #else vec2 clearCoatReflectionCoords=clearCoatReflectionVector.xy; #ifdef REFLECTIONMAP_PROJECTION clearCoatReflectionCoords/=clearCoatReflectionVector.z; #endif clearCoatReflectionCoords.y=1.0-clearCoatReflectionCoords.y; #endif sampleReflectionTexture( clearCoatAlphaG, vReflectionMicrosurfaceInfos, vReflectionInfos, vReflectionColor, #if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX) clearCoatNdotVUnclamped, #endif #ifdef LINEARSPECULARREFLECTION clearCoatRoughness, #endif reflectionSampler, clearCoatReflectionCoords, #ifndef LODBASEDMICROSFURACE reflectionSamplerLow, reflectionSamplerHigh, #endif #ifdef REALTIME_FILTERING vReflectionFilteringInfo, #endif environmentClearCoatRadiance ); #if DEBUGMODE>0 outParams.environmentClearCoatRadiance=environmentClearCoatRadiance; #endif #if defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX) vec3 clearCoatEnvironmentReflectance=getReflectanceFromBRDFLookup(vec3(vClearCoatRefractionParams.x),environmentClearCoatBrdf); #ifdef HORIZONOCCLUSION #ifdef BUMP #ifdef REFLECTIONMAP_3D float clearCoatEho=environmentHorizonOcclusion(-viewDirectionW,clearCoatNormalW,geometricNormalW);clearCoatEnvironmentReflectance*=clearCoatEho; #endif #endif #endif #else vec3 clearCoatEnvironmentReflectance=getReflectanceFromAnalyticalBRDFLookup_Jones(clearCoatNdotV,vec3(1.),vec3(1.),sqrt(1.-clearCoatRoughness)); #endif clearCoatEnvironmentReflectance*=clearCoatIntensity; #if DEBUGMODE>0 outParams.clearCoatEnvironmentReflectance=clearCoatEnvironmentReflectance; #endif outParams.finalClearCoatRadianceScaled= environmentClearCoatRadiance.rgb * clearCoatEnvironmentReflectance * vLightingIntensity.z; #endif #if defined(CLEARCOAT_TINT) outParams.absorption=computeClearCoatAbsorption(outParams.clearCoatNdotVRefract,outParams.clearCoatNdotVRefract,outParams.clearCoatColor,clearCoatThickness,clearCoatIntensity); #endif float fresnelIBLClearCoat=fresnelSchlickGGX(clearCoatNdotV,vClearCoatRefractionParams.x,CLEARCOATREFLECTANCE90);fresnelIBLClearCoat*=clearCoatIntensity;outParams.conservationFactor=(1.-fresnelIBLClearCoat); #if defined(ENVIRONMENTBRDF) && defined(MS_BRDF_ENERGY_CONSERVATION) outParams.energyConservationFactorClearCoat=getEnergyConservationFactor(outParams.specularEnvironmentR0,environmentClearCoatBrdf); #endif } #endif `; Le.IncludesShadersStore[cce] = pce; const hce = "pbrBlockIridescence", Hce = `struct iridescenceOutParams {float iridescenceIntensity;float iridescenceIOR;float iridescenceThickness;vec3 specularEnvironmentR0;}; #ifdef IRIDESCENCE #define pbr_inline #define inline void iridescenceBlock( in vec4 vIridescenceParams, in float viewAngle, in vec3 specularEnvironmentR0, #ifdef IRIDESCENCE_TEXTURE in vec2 iridescenceMapData, #endif #ifdef IRIDESCENCE_THICKNESS_TEXTURE in vec2 iridescenceThicknessMapData, #endif #ifdef CLEARCOAT in float NdotVUnclamped, #ifdef CLEARCOAT_TEXTURE in vec2 clearCoatMapData, #endif #endif out iridescenceOutParams outParams ) {float iridescenceIntensity=vIridescenceParams.x;float iridescenceIOR=vIridescenceParams.y;float iridescenceThicknessMin=vIridescenceParams.z;float iridescenceThicknessMax=vIridescenceParams.w;float iridescenceThicknessWeight=1.; #ifdef IRIDESCENCE_TEXTURE iridescenceIntensity*=iridescenceMapData.x; #ifdef IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE iridescenceThicknessWeight=iridescenceMapData.g; #endif #endif #if defined(IRIDESCENCE_THICKNESS_TEXTURE) iridescenceThicknessWeight=iridescenceThicknessMapData.g; #endif float iridescenceThickness=mix(iridescenceThicknessMin,iridescenceThicknessMax,iridescenceThicknessWeight);float topIor=1.; #ifdef CLEARCOAT float clearCoatIntensity=vClearCoatParams.x; #ifdef CLEARCOAT_TEXTURE clearCoatIntensity*=clearCoatMapData.x; #endif topIor=mix(1.0,vClearCoatRefractionParams.w-1.,clearCoatIntensity);viewAngle=sqrt(1.0+square(1.0/topIor)*(square(NdotVUnclamped)-1.0)); #endif vec3 iridescenceFresnel=evalIridescence(topIor,iridescenceIOR,viewAngle,iridescenceThickness,specularEnvironmentR0);outParams.specularEnvironmentR0=mix(specularEnvironmentR0,iridescenceFresnel,iridescenceIntensity);outParams.iridescenceIntensity=iridescenceIntensity;outParams.iridescenceThickness=iridescenceThickness;outParams.iridescenceIOR=iridescenceIOR;} #endif `; Le.IncludesShadersStore[hce] = Hce; const gce = "pbrBlockSubSurface", Xce = `struct subSurfaceOutParams {vec3 specularEnvironmentReflectance; #ifdef SS_REFRACTION vec3 finalRefraction;vec3 surfaceAlbedo; #ifdef SS_LINKREFRACTIONTOTRANSPARENCY float alpha; #endif #ifdef REFLECTION float refractionFactorForIrradiance; #endif #endif #ifdef SS_TRANSLUCENCY vec3 transmittance;float translucencyIntensity; #ifdef REFLECTION vec3 refractionIrradiance; #endif #endif #if DEBUGMODE>0 #ifdef SS_THICKNESSANDMASK_TEXTURE vec4 thicknessMap; #endif #ifdef SS_REFRACTION vec4 environmentRefraction;vec3 refractionTransmittance; #endif #endif }; #ifdef SUBSURFACE #define pbr_inline #define inline #ifdef SS_REFRACTION vec4 sampleEnvironmentRefraction( in float ior ,in float thickness ,in float refractionLOD ,in vec3 normalW ,in vec3 vPositionW ,in vec3 viewDirectionW ,in mat4 view ,in vec4 vRefractionInfos ,in mat4 refractionMatrix ,in vec4 vRefractionMicrosurfaceInfos ,in float alphaG #ifdef SS_REFRACTIONMAP_3D ,in samplerCube refractionSampler #ifndef LODBASEDMICROSFURACE ,in samplerCube refractionSamplerLow ,in samplerCube refractionSamplerHigh #endif #else ,in sampler2D refractionSampler #ifndef LODBASEDMICROSFURACE ,in sampler2D refractionSamplerLow ,in sampler2D refractionSamplerHigh #endif #endif #ifdef ANISOTROPIC ,in anisotropicOutParams anisotropicOut #endif #ifdef REALTIME_FILTERING ,in vec2 vRefractionFilteringInfo #endif #ifdef SS_USE_LOCAL_REFRACTIONMAP_CUBIC ,in vec3 refractionPosition ,in vec3 refractionSize #endif ) {vec4 environmentRefraction=vec4(0.,0.,0.,0.); #ifdef ANISOTROPIC vec3 refractionVector=refract(-viewDirectionW,anisotropicOut.anisotropicNormal,ior); #else vec3 refractionVector=refract(-viewDirectionW,normalW,ior); #endif #ifdef SS_REFRACTIONMAP_OPPOSITEZ refractionVector.z*=-1.0; #endif #ifdef SS_REFRACTIONMAP_3D #ifdef SS_USE_LOCAL_REFRACTIONMAP_CUBIC refractionVector=parallaxCorrectNormal(vPositionW,refractionVector,refractionSize,refractionPosition); #endif refractionVector.y=refractionVector.y*vRefractionInfos.w;vec3 refractionCoords=refractionVector;refractionCoords=vec3(refractionMatrix*vec4(refractionCoords,0)); #else #ifdef SS_USE_THICKNESS_AS_DEPTH vec3 vRefractionUVW=vec3(refractionMatrix*(view*vec4(vPositionW+refractionVector*thickness,1.0))); #else vec3 vRefractionUVW=vec3(refractionMatrix*(view*vec4(vPositionW+refractionVector*vRefractionInfos.z,1.0))); #endif vec2 refractionCoords=vRefractionUVW.xy/vRefractionUVW.z;refractionCoords.y=1.0-refractionCoords.y; #endif #ifdef LODBASEDMICROSFURACE refractionLOD=refractionLOD*vRefractionMicrosurfaceInfos.y+vRefractionMicrosurfaceInfos.z; #ifdef SS_LODINREFRACTIONALPHA float automaticRefractionLOD=UNPACK_LOD(sampleRefraction(refractionSampler,refractionCoords).a);float requestedRefractionLOD=max(automaticRefractionLOD,refractionLOD); #else float requestedRefractionLOD=refractionLOD; #endif #if defined(REALTIME_FILTERING) && defined(SS_REFRACTIONMAP_3D) environmentRefraction=vec4(radiance(alphaG,refractionSampler,refractionCoords,vRefractionFilteringInfo),1.0); #else environmentRefraction=sampleRefractionLod(refractionSampler,refractionCoords,requestedRefractionLOD); #endif #else float lodRefractionNormalized=saturate(refractionLOD/log2(vRefractionMicrosurfaceInfos.x));float lodRefractionNormalizedDoubled=lodRefractionNormalized*2.0;vec4 environmentRefractionMid=sampleRefraction(refractionSampler,refractionCoords);if (lodRefractionNormalizedDoubled<1.0){environmentRefraction=mix( sampleRefraction(refractionSamplerHigh,refractionCoords), environmentRefractionMid, lodRefractionNormalizedDoubled );} else {environmentRefraction=mix( environmentRefractionMid, sampleRefraction(refractionSamplerLow,refractionCoords), lodRefractionNormalizedDoubled-1.0 );} #endif #ifdef SS_RGBDREFRACTION environmentRefraction.rgb=fromRGBD(environmentRefraction); #endif #ifdef SS_GAMMAREFRACTION environmentRefraction.rgb=toLinearSpace(environmentRefraction.rgb); #endif return environmentRefraction;} #endif void subSurfaceBlock( in vec3 vSubSurfaceIntensity, in vec2 vThicknessParam, in vec4 vTintColor, in vec3 normalW, in vec3 specularEnvironmentReflectance, #ifdef SS_THICKNESSANDMASK_TEXTURE in vec4 thicknessMap, #endif #ifdef SS_REFRACTIONINTENSITY_TEXTURE in vec4 refractionIntensityMap, #endif #ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE in vec4 translucencyIntensityMap, #endif #ifdef REFLECTION #ifdef SS_TRANSLUCENCY in mat4 reflectionMatrix, #ifdef USESPHERICALFROMREFLECTIONMAP #if !defined(NORMAL) || !defined(USESPHERICALINVERTEX) in vec3 irradianceVector_, #endif #if defined(REALTIME_FILTERING) in samplerCube reflectionSampler, in vec2 vReflectionFilteringInfo, #endif #endif #ifdef USEIRRADIANCEMAP #ifdef REFLECTIONMAP_3D in samplerCube irradianceSampler, #else in sampler2D irradianceSampler, #endif #endif #endif #endif #if defined(SS_REFRACTION) || defined(SS_TRANSLUCENCY) in vec3 surfaceAlbedo, #endif #ifdef SS_REFRACTION in vec3 vPositionW, in vec3 viewDirectionW, in mat4 view, in vec4 vRefractionInfos, in mat4 refractionMatrix, in vec4 vRefractionMicrosurfaceInfos, in vec4 vLightingIntensity, #ifdef SS_LINKREFRACTIONTOTRANSPARENCY in float alpha, #endif #ifdef SS_LODINREFRACTIONALPHA in float NdotVUnclamped, #endif #ifdef SS_LINEARSPECULARREFRACTION in float roughness, #endif in float alphaG, #ifdef SS_REFRACTIONMAP_3D in samplerCube refractionSampler, #ifndef LODBASEDMICROSFURACE in samplerCube refractionSamplerLow, in samplerCube refractionSamplerHigh, #endif #else in sampler2D refractionSampler, #ifndef LODBASEDMICROSFURACE in sampler2D refractionSamplerLow, in sampler2D refractionSamplerHigh, #endif #endif #ifdef ANISOTROPIC in anisotropicOutParams anisotropicOut, #endif #ifdef REALTIME_FILTERING in vec2 vRefractionFilteringInfo, #endif #ifdef SS_USE_LOCAL_REFRACTIONMAP_CUBIC in vec3 refractionPosition, in vec3 refractionSize, #endif #ifdef SS_DISPERSION in float dispersion, #endif #endif #ifdef SS_TRANSLUCENCY in vec3 vDiffusionDistance, #endif out subSurfaceOutParams outParams ) {outParams.specularEnvironmentReflectance=specularEnvironmentReflectance; #ifdef SS_REFRACTION float refractionIntensity=vSubSurfaceIntensity.x; #ifdef SS_LINKREFRACTIONTOTRANSPARENCY refractionIntensity*=(1.0-alpha);outParams.alpha=1.0; #endif #endif #ifdef SS_TRANSLUCENCY float translucencyIntensity=vSubSurfaceIntensity.y; #endif #ifdef SS_THICKNESSANDMASK_TEXTURE #if defined(SS_USE_GLTF_TEXTURES) float thickness=thicknessMap.g*vThicknessParam.y+vThicknessParam.x; #else float thickness=thicknessMap.r*vThicknessParam.y+vThicknessParam.x; #endif #if DEBUGMODE>0 outParams.thicknessMap=thicknessMap; #endif #ifdef SS_MASK_FROM_THICKNESS_TEXTURE #if defined(SS_REFRACTION) && defined(SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE) #if defined(SS_USE_GLTF_TEXTURES) refractionIntensity*=thicknessMap.r; #else refractionIntensity*=thicknessMap.g; #endif #endif #if defined(SS_TRANSLUCENCY) && defined(SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE) translucencyIntensity*=thicknessMap.b; #endif #endif #else float thickness=vThicknessParam.y; #endif #ifdef SS_REFRACTIONINTENSITY_TEXTURE #ifdef SS_USE_GLTF_TEXTURES refractionIntensity*=refractionIntensityMap.r; #else refractionIntensity*=refractionIntensityMap.g; #endif #endif #ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE translucencyIntensity*=translucencyIntensityMap.b; #endif #ifdef SS_TRANSLUCENCY thickness=maxEps(thickness);vec3 transmittance=transmittanceBRDF_Burley(vTintColor.rgb,vDiffusionDistance,thickness);transmittance*=translucencyIntensity;outParams.transmittance=transmittance;outParams.translucencyIntensity=translucencyIntensity; #endif #ifdef SS_REFRACTION vec4 environmentRefraction=vec4(0.,0.,0.,0.); #ifdef SS_HAS_THICKNESS float ior=vRefractionInfos.y; #else float ior=vRefractionMicrosurfaceInfos.w; #endif #ifdef SS_LODINREFRACTIONALPHA float refractionAlphaG=alphaG;refractionAlphaG=mix(alphaG,0.0,clamp(ior*3.0-2.0,0.0,1.0));float refractionLOD=getLodFromAlphaG(vRefractionMicrosurfaceInfos.x,refractionAlphaG,NdotVUnclamped); #elif defined(SS_LINEARSPECULARREFRACTION) float refractionRoughness=alphaG;refractionRoughness=mix(alphaG,0.0,clamp(ior*3.0-2.0,0.0,1.0));float refractionLOD=getLinearLodFromRoughness(vRefractionMicrosurfaceInfos.x,refractionRoughness); #else float refractionAlphaG=alphaG;refractionAlphaG=mix(alphaG,0.0,clamp(ior*3.0-2.0,0.0,1.0));float refractionLOD=getLodFromAlphaG(vRefractionMicrosurfaceInfos.x,refractionAlphaG); #endif #ifdef SS_DISPERSION float realIOR=1.0/ior;float iorDispersionSpread=0.04*dispersion*(realIOR-1.0);vec3 iors=vec3(1.0/(realIOR-iorDispersionSpread),ior,1.0/(realIOR+iorDispersionSpread));for (int i=0; i<3; i++) {ior=iors[i]; #endif vec4 envSample=sampleEnvironmentRefraction(ior,thickness,refractionLOD,normalW,vPositionW,viewDirectionW,view,vRefractionInfos,refractionMatrix,vRefractionMicrosurfaceInfos,alphaG #ifdef SS_REFRACTIONMAP_3D ,refractionSampler #ifndef LODBASEDMICROSFURACE ,refractionSamplerLow ,refractionSamplerHigh #endif #else ,refractionSampler #ifndef LODBASEDMICROSFURACE ,refractionSamplerLow ,refractionSamplerHigh #endif #endif #ifdef ANISOTROPIC ,anisotropicOut #endif #ifdef REALTIME_FILTERING ,vRefractionFilteringInfo #endif #ifdef SS_USE_LOCAL_REFRACTIONMAP_CUBIC ,refractionPosition ,refractionSize #endif ); #ifdef SS_DISPERSION environmentRefraction[i]=envSample[i];} #else environmentRefraction=envSample; #endif environmentRefraction.rgb*=vRefractionInfos.x; #endif #ifdef SS_REFRACTION vec3 refractionTransmittance=vec3(refractionIntensity); #ifdef SS_THICKNESSANDMASK_TEXTURE vec3 volumeAlbedo=computeColorAtDistanceInMedia(vTintColor.rgb,vTintColor.w);refractionTransmittance*=cocaLambert(volumeAlbedo,thickness); #elif defined(SS_LINKREFRACTIONTOTRANSPARENCY) float maxChannel=max(max(surfaceAlbedo.r,surfaceAlbedo.g),surfaceAlbedo.b);vec3 volumeAlbedo=saturate(maxChannel*surfaceAlbedo);environmentRefraction.rgb*=volumeAlbedo; #else vec3 volumeAlbedo=computeColorAtDistanceInMedia(vTintColor.rgb,vTintColor.w);refractionTransmittance*=cocaLambert(volumeAlbedo,vThicknessParam.y); #endif #ifdef SS_ALBEDOFORREFRACTIONTINT environmentRefraction.rgb*=surfaceAlbedo.rgb; #endif outParams.surfaceAlbedo=surfaceAlbedo*(1.-refractionIntensity); #ifdef REFLECTION outParams.refractionFactorForIrradiance=(1.-refractionIntensity); #endif #ifdef UNUSED_MULTIPLEBOUNCES vec3 bounceSpecularEnvironmentReflectance=(2.0*specularEnvironmentReflectance)/(1.0+specularEnvironmentReflectance);outParams.specularEnvironmentReflectance=mix(bounceSpecularEnvironmentReflectance,specularEnvironmentReflectance,refractionIntensity); #endif refractionTransmittance*=1.0-outParams.specularEnvironmentReflectance; #if DEBUGMODE>0 outParams.refractionTransmittance=refractionTransmittance; #endif outParams.finalRefraction=environmentRefraction.rgb*refractionTransmittance*vLightingIntensity.z; #if DEBUGMODE>0 outParams.environmentRefraction=environmentRefraction; #endif #endif #if defined(REFLECTION) && defined(SS_TRANSLUCENCY) #if defined(NORMAL) && defined(USESPHERICALINVERTEX) || !defined(USESPHERICALFROMREFLECTIONMAP) vec3 irradianceVector=vec3(reflectionMatrix*vec4(normalW,0)).xyz; #ifdef REFLECTIONMAP_OPPOSITEZ irradianceVector.z*=-1.0; #endif #ifdef INVERTCUBICMAP irradianceVector.y*=-1.0; #endif #else vec3 irradianceVector=irradianceVector_; #endif #if defined(USESPHERICALFROMREFLECTIONMAP) #if defined(REALTIME_FILTERING) vec3 refractionIrradiance=irradiance(reflectionSampler,-irradianceVector,vReflectionFilteringInfo); #else vec3 refractionIrradiance=computeEnvironmentIrradiance(-irradianceVector); #endif #elif defined(USEIRRADIANCEMAP) #ifdef REFLECTIONMAP_3D vec3 irradianceCoords=irradianceVector; #else vec2 irradianceCoords=irradianceVector.xy; #ifdef REFLECTIONMAP_PROJECTION irradianceCoords/=irradianceVector.z; #endif irradianceCoords.y=1.0-irradianceCoords.y; #endif vec4 refractionIrradiance=sampleReflection(irradianceSampler,-irradianceCoords); #ifdef RGBDREFLECTION refractionIrradiance.rgb=fromRGBD(refractionIrradiance); #endif #ifdef GAMMAREFLECTION refractionIrradiance.rgb=toLinearSpace(refractionIrradiance.rgb); #endif #else vec4 refractionIrradiance=vec4(0.); #endif refractionIrradiance.rgb*=transmittance; #ifdef SS_ALBEDOFORTRANSLUCENCYTINT refractionIrradiance.rgb*=surfaceAlbedo.rgb; #endif outParams.refractionIrradiance=refractionIrradiance.rgb; #endif } #endif `; Le.IncludesShadersStore[gce] = Xce; const Tce = "pbrBlockNormalGeometric", qce = `vec3 viewDirectionW=normalize(vEyePosition.xyz-vPositionW); #ifdef NORMAL vec3 normalW=normalize(vNormalW); #else vec3 normalW=normalize(cross(dFdx(vPositionW),dFdy(vPositionW)))*vEyePosition.w; #endif vec3 geometricNormalW=normalW; #if defined(TWOSIDEDLIGHTING) && defined(NORMAL) geometricNormalW=gl_FrontFacing ? geometricNormalW : -geometricNormalW; #endif `; Le.IncludesShadersStore[Tce] = qce; const bce = "pbrBlockNormalFinal", xce = `#if defined(FORCENORMALFORWARD) && defined(NORMAL) vec3 faceNormal=normalize(cross(dFdx(vPositionW),dFdy(vPositionW)))*vEyePosition.w; #if defined(TWOSIDEDLIGHTING) faceNormal=gl_FrontFacing ? faceNormal : -faceNormal; #endif normalW*=sign(dot(normalW,faceNormal)); #endif #if defined(TWOSIDEDLIGHTING) && defined(NORMAL) normalW=gl_FrontFacing ? normalW : -normalW; #endif `; Le.IncludesShadersStore[bce] = xce; const Dce = "pbrBlockLightmapInit", jce = `#ifdef LIGHTMAP vec4 lightmapColor=texture2D(lightmapSampler,vLightmapUV+uvOffset); #ifdef RGBDLIGHTMAP lightmapColor.rgb=fromRGBD(lightmapColor); #endif #ifdef GAMMALIGHTMAP lightmapColor.rgb=toLinearSpace(lightmapColor.rgb); #endif lightmapColor.rgb*=vLightmapInfos.y; #endif `; Le.IncludesShadersStore[Dce] = jce; const wce = "pbrBlockGeometryInfo", mce = `float NdotVUnclamped=dot(normalW,viewDirectionW);float NdotV=absEps(NdotVUnclamped);float alphaG=convertRoughnessToAverageSlope(roughness);vec2 AARoughnessFactors=getAARoughnessFactors(normalW.xyz); #ifdef SPECULARAA alphaG+=AARoughnessFactors.y; #endif #if defined(ENVIRONMENTBRDF) vec3 environmentBrdf=getBRDFLookup(NdotV,roughness); #endif #if defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX) #ifdef RADIANCEOCCLUSION #ifdef AMBIENTINGRAYSCALE float ambientMonochrome=aoOut.ambientOcclusionColor.r; #else float ambientMonochrome=getLuminance(aoOut.ambientOcclusionColor); #endif float seo=environmentRadianceOcclusion(ambientMonochrome,NdotVUnclamped); #endif #ifdef HORIZONOCCLUSION #ifdef BUMP #ifdef REFLECTIONMAP_3D float eho=environmentHorizonOcclusion(-viewDirectionW,normalW,geometricNormalW); #endif #endif #endif #endif `; Le.IncludesShadersStore[wce] = mce; const Bce = "pbrBlockReflectance0", Wce = `float reflectance=max(max(reflectivityOut.surfaceReflectivityColor.r,reflectivityOut.surfaceReflectivityColor.g),reflectivityOut.surfaceReflectivityColor.b);vec3 specularEnvironmentR0=reflectivityOut.surfaceReflectivityColor.rgb; #ifdef METALLICWORKFLOW vec3 specularEnvironmentR90=vec3(metallicReflectanceFactors.a); #else vec3 specularEnvironmentR90=vec3(1.0,1.0,1.0); #endif #ifdef ALPHAFRESNEL float reflectance90=fresnelGrazingReflectance(reflectance);specularEnvironmentR90=specularEnvironmentR90*reflectance90; #endif `; Le.IncludesShadersStore[Bce] = Wce; const Sce = "pbrBlockReflectance", Uce = `#if defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX) vec3 specularEnvironmentReflectance=getReflectanceFromBRDFLookup(clearcoatOut.specularEnvironmentR0,specularEnvironmentR90,environmentBrdf); #ifdef RADIANCEOCCLUSION specularEnvironmentReflectance*=seo; #endif #ifdef HORIZONOCCLUSION #ifdef BUMP #ifdef REFLECTIONMAP_3D specularEnvironmentReflectance*=eho; #endif #endif #endif #else vec3 specularEnvironmentReflectance=getReflectanceFromAnalyticalBRDFLookup_Jones(NdotV,clearcoatOut.specularEnvironmentR0,specularEnvironmentR90,sqrt(microSurface)); #endif #ifdef CLEARCOAT specularEnvironmentReflectance*=clearcoatOut.conservationFactor; #if defined(CLEARCOAT_TINT) specularEnvironmentReflectance*=clearcoatOut.absorption; #endif #endif `; Le.IncludesShadersStore[Sce] = Uce; const Ice = "pbrBlockDirectLighting", Rce = `vec3 diffuseBase=vec3(0.,0.,0.); #ifdef SPECULARTERM vec3 specularBase=vec3(0.,0.,0.); #endif #ifdef CLEARCOAT vec3 clearCoatBase=vec3(0.,0.,0.); #endif #ifdef SHEEN vec3 sheenBase=vec3(0.,0.,0.); #endif preLightingInfo preInfo;lightingInfo info;float shadow=1.; float aggShadow=0.;float numLights=0.; #if defined(CLEARCOAT) && defined(CLEARCOAT_TINT) vec3 absorption=vec3(0.); #endif `; Le.IncludesShadersStore[Ice] = Rce; const Vce = "pbrBlockFinalLitComponents", Cce = `aggShadow=aggShadow/numLights; #if defined(ENVIRONMENTBRDF) #ifdef MS_BRDF_ENERGY_CONSERVATION vec3 energyConservationFactor=getEnergyConservationFactor(clearcoatOut.specularEnvironmentR0,environmentBrdf); #endif #endif #ifndef METALLICWORKFLOW #ifdef SPECULAR_GLOSSINESS_ENERGY_CONSERVATION surfaceAlbedo.rgb=(1.-reflectance)*surfaceAlbedo.rgb; #endif #endif #if defined(SHEEN) && defined(SHEEN_ALBEDOSCALING) && defined(ENVIRONMENTBRDF) surfaceAlbedo.rgb=sheenOut.sheenAlbedoScaling*surfaceAlbedo.rgb; #endif #ifdef REFLECTION vec3 finalIrradiance=reflectionOut.environmentIrradiance; #if defined(CLEARCOAT) finalIrradiance*=clearcoatOut.conservationFactor; #if defined(CLEARCOAT_TINT) finalIrradiance*=clearcoatOut.absorption; #endif #endif #if defined(SS_REFRACTION) finalIrradiance*=subSurfaceOut.refractionFactorForIrradiance; #endif #if defined(SS_TRANSLUCENCY) finalIrradiance*=(1.0-subSurfaceOut.translucencyIntensity);finalIrradiance+=subSurfaceOut.refractionIrradiance; #endif finalIrradiance*=surfaceAlbedo.rgb;finalIrradiance*=vLightingIntensity.z;finalIrradiance*=aoOut.ambientOcclusionColor; #endif #ifdef SPECULARTERM vec3 finalSpecular=specularBase;finalSpecular=max(finalSpecular,0.0);vec3 finalSpecularScaled=finalSpecular*vLightingIntensity.x*vLightingIntensity.w; #if defined(ENVIRONMENTBRDF) && defined(MS_BRDF_ENERGY_CONSERVATION) finalSpecularScaled*=energyConservationFactor; #endif #if defined(SHEEN) && defined(ENVIRONMENTBRDF) && defined(SHEEN_ALBEDOSCALING) finalSpecularScaled*=sheenOut.sheenAlbedoScaling; #endif #endif #ifdef REFLECTION vec3 finalRadiance=reflectionOut.environmentRadiance.rgb;finalRadiance*=subSurfaceOut.specularEnvironmentReflectance;vec3 finalRadianceScaled=finalRadiance*vLightingIntensity.z; #if defined(ENVIRONMENTBRDF) && defined(MS_BRDF_ENERGY_CONSERVATION) finalRadianceScaled*=energyConservationFactor; #endif #if defined(SHEEN) && defined(ENVIRONMENTBRDF) && defined(SHEEN_ALBEDOSCALING) finalRadianceScaled*=sheenOut.sheenAlbedoScaling; #endif #endif #ifdef SHEEN vec3 finalSheen=sheenBase*sheenOut.sheenColor;finalSheen=max(finalSheen,0.0);vec3 finalSheenScaled=finalSheen*vLightingIntensity.x*vLightingIntensity.w; #if defined(CLEARCOAT) && defined(REFLECTION) && defined(ENVIRONMENTBRDF) sheenOut.finalSheenRadianceScaled*=clearcoatOut.conservationFactor; #if defined(CLEARCOAT_TINT) sheenOut.finalSheenRadianceScaled*=clearcoatOut.absorption; #endif #endif #endif #ifdef CLEARCOAT vec3 finalClearCoat=clearCoatBase;finalClearCoat=max(finalClearCoat,0.0);vec3 finalClearCoatScaled=finalClearCoat*vLightingIntensity.x*vLightingIntensity.w; #if defined(ENVIRONMENTBRDF) && defined(MS_BRDF_ENERGY_CONSERVATION) finalClearCoatScaled*=clearcoatOut.energyConservationFactorClearCoat; #endif #ifdef SS_REFRACTION subSurfaceOut.finalRefraction*=clearcoatOut.conservationFactor; #ifdef CLEARCOAT_TINT subSurfaceOut.finalRefraction*=clearcoatOut.absorption; #endif #endif #endif #ifdef ALPHABLEND float luminanceOverAlpha=0.0; #if defined(REFLECTION) && defined(RADIANCEOVERALPHA) luminanceOverAlpha+=getLuminance(finalRadianceScaled); #if defined(CLEARCOAT) luminanceOverAlpha+=getLuminance(clearcoatOut.finalClearCoatRadianceScaled); #endif #endif #if defined(SPECULARTERM) && defined(SPECULAROVERALPHA) luminanceOverAlpha+=getLuminance(finalSpecularScaled); #endif #if defined(CLEARCOAT) && defined(CLEARCOATOVERALPHA) luminanceOverAlpha+=getLuminance(finalClearCoatScaled); #endif #if defined(RADIANCEOVERALPHA) || defined(SPECULAROVERALPHA) || defined(CLEARCOATOVERALPHA) alpha=saturate(alpha+luminanceOverAlpha*luminanceOverAlpha); #endif #endif `; Le.IncludesShadersStore[Vce] = Cce; const Oce = "pbrBlockFinalUnlitComponents", yce = `vec3 finalDiffuse=diffuseBase;finalDiffuse*=surfaceAlbedo.rgb;finalDiffuse=max(finalDiffuse,0.0);finalDiffuse*=vLightingIntensity.x;vec3 finalAmbient=vAmbientColor;finalAmbient*=surfaceAlbedo.rgb;vec3 finalEmissive=vEmissiveColor; #ifdef EMISSIVE vec3 emissiveColorTex=texture2D(emissiveSampler,vEmissiveUV+uvOffset).rgb; #ifdef GAMMAEMISSIVE finalEmissive*=toLinearSpace(emissiveColorTex.rgb); #else finalEmissive*=emissiveColorTex.rgb; #endif finalEmissive*= vEmissiveInfos.y; #endif finalEmissive*=vLightingIntensity.y; #ifdef AMBIENT vec3 ambientOcclusionForDirectDiffuse=mix(vec3(1.),aoOut.ambientOcclusionColor,vAmbientInfos.w); #else vec3 ambientOcclusionForDirectDiffuse=aoOut.ambientOcclusionColor; #endif finalAmbient*=aoOut.ambientOcclusionColor;finalDiffuse*=ambientOcclusionForDirectDiffuse; `; Le.IncludesShadersStore[Oce] = yce; const kce = "pbrBlockFinalColorComposition", Ece = `vec4 finalColor=vec4( #ifndef UNLIT #ifdef REFLECTION finalIrradiance + #endif #ifdef SPECULARTERM finalSpecularScaled + #endif #ifdef SHEEN finalSheenScaled + #endif #ifdef CLEARCOAT finalClearCoatScaled + #endif #ifdef REFLECTION finalRadianceScaled + #if defined(SHEEN) && defined(ENVIRONMENTBRDF) sheenOut.finalSheenRadianceScaled + #endif #ifdef CLEARCOAT clearcoatOut.finalClearCoatRadianceScaled + #endif #endif #ifdef SS_REFRACTION subSurfaceOut.finalRefraction + #endif #endif finalAmbient + finalDiffuse, alpha); #ifdef LIGHTMAP #ifndef LIGHTMAPEXCLUDED #ifdef USELIGHTMAPASSHADOWMAP finalColor.rgb*=lightmapColor.rgb; #else finalColor.rgb+=lightmapColor.rgb; #endif #endif #endif finalColor.rgb+=finalEmissive; #define CUSTOM_FRAGMENT_BEFORE_FOG finalColor=max(finalColor,0.0); `; Le.IncludesShadersStore[kce] = Ece; const Fce = "pbrBlockImageProcessing", Nce = `#if defined(IMAGEPROCESSINGPOSTPROCESS) || defined(SS_SCATTERING) #if !defined(SKIPFINALCOLORCLAMP) finalColor.rgb=clamp(finalColor.rgb,0.,30.0); #endif #else finalColor=applyImageProcessing(finalColor); #endif finalColor.a*=visibility; #ifdef PREMULTIPLYALPHA finalColor.rgb*=finalColor.a; #endif `; Le.IncludesShadersStore[Fce] = Nce; const Qce = "pbrDebug", Yce = `#if DEBUGMODE>0 if (vClipSpacePosition.x/vClipSpacePosition.w>=vDebugMode.x) { #if DEBUGMODE==1 gl_FragColor.rgb=vPositionW.rgb; #define DEBUGMODE_NORMALIZE #elif DEBUGMODE==2 && defined(NORMAL) gl_FragColor.rgb=vNormalW.rgb; #define DEBUGMODE_NORMALIZE #elif DEBUGMODE==3 && defined(BUMP) || DEBUGMODE==3 && defined(PARALLAX) || DEBUGMODE==3 && defined(ANISOTROPIC) gl_FragColor.rgb=TBN[0]; #define DEBUGMODE_NORMALIZE #elif DEBUGMODE==4 && defined(BUMP) || DEBUGMODE==4 && defined(PARALLAX) || DEBUGMODE==4 && defined(ANISOTROPIC) gl_FragColor.rgb=TBN[1]; #define DEBUGMODE_NORMALIZE #elif DEBUGMODE==5 gl_FragColor.rgb=normalW; #define DEBUGMODE_NORMALIZE #elif DEBUGMODE==6 && defined(MAINUV1) gl_FragColor.rgb=vec3(vMainUV1,0.0); #elif DEBUGMODE==7 && defined(MAINUV2) gl_FragColor.rgb=vec3(vMainUV2,0.0); #elif DEBUGMODE==8 && defined(CLEARCOAT) && defined(CLEARCOAT_BUMP) gl_FragColor.rgb=clearcoatOut.TBNClearCoat[0]; #define DEBUGMODE_NORMALIZE #elif DEBUGMODE==9 && defined(CLEARCOAT) && defined(CLEARCOAT_BUMP) gl_FragColor.rgb=clearcoatOut.TBNClearCoat[1]; #define DEBUGMODE_NORMALIZE #elif DEBUGMODE==10 && defined(CLEARCOAT) gl_FragColor.rgb=clearcoatOut.clearCoatNormalW; #define DEBUGMODE_NORMALIZE #elif DEBUGMODE==11 && defined(ANISOTROPIC) gl_FragColor.rgb=anisotropicOut.anisotropicNormal; #define DEBUGMODE_NORMALIZE #elif DEBUGMODE==12 && defined(ANISOTROPIC) gl_FragColor.rgb=anisotropicOut.anisotropicTangent; #define DEBUGMODE_NORMALIZE #elif DEBUGMODE==13 && defined(ANISOTROPIC) gl_FragColor.rgb=anisotropicOut.anisotropicBitangent; #define DEBUGMODE_NORMALIZE #elif DEBUGMODE==20 && defined(ALBEDO) gl_FragColor.rgb=albedoTexture.rgb; #ifndef GAMMAALBEDO #define DEBUGMODE_GAMMA #endif #elif DEBUGMODE==21 && defined(AMBIENT) gl_FragColor.rgb=aoOut.ambientOcclusionColorMap.rgb; #elif DEBUGMODE==22 && defined(OPACITY) gl_FragColor.rgb=opacityMap.rgb; #elif DEBUGMODE==23 && defined(EMISSIVE) gl_FragColor.rgb=emissiveColorTex.rgb; #ifndef GAMMAEMISSIVE #define DEBUGMODE_GAMMA #endif #elif DEBUGMODE==24 && defined(LIGHTMAP) gl_FragColor.rgb=lightmapColor.rgb; #ifndef GAMMALIGHTMAP #define DEBUGMODE_GAMMA #endif #elif DEBUGMODE==25 && defined(REFLECTIVITY) && defined(METALLICWORKFLOW) gl_FragColor.rgb=reflectivityOut.surfaceMetallicColorMap.rgb; #elif DEBUGMODE==26 && defined(REFLECTIVITY) && !defined(METALLICWORKFLOW) gl_FragColor.rgb=reflectivityOut.surfaceReflectivityColorMap.rgb; #define DEBUGMODE_GAMMA #elif DEBUGMODE==27 && defined(CLEARCOAT) && defined(CLEARCOAT_TEXTURE) gl_FragColor.rgb=vec3(clearcoatOut.clearCoatMapData.rg,0.0); #elif DEBUGMODE==28 && defined(CLEARCOAT) && defined(CLEARCOAT_TINT) && defined(CLEARCOAT_TINT_TEXTURE) gl_FragColor.rgb=clearcoatOut.clearCoatTintMapData.rgb; #elif DEBUGMODE==29 && defined(SHEEN) && defined(SHEEN_TEXTURE) gl_FragColor.rgb=sheenOut.sheenMapData.rgb; #elif DEBUGMODE==30 && defined(ANISOTROPIC) && defined(ANISOTROPIC_TEXTURE) gl_FragColor.rgb=anisotropicOut.anisotropyMapData.rgb; #elif DEBUGMODE==31 && defined(SUBSURFACE) && defined(SS_THICKNESSANDMASK_TEXTURE) gl_FragColor.rgb=subSurfaceOut.thicknessMap.rgb; #elif DEBUGMODE==32 && defined(BUMP) gl_FragColor.rgb=texture2D(bumpSampler,vBumpUV).rgb; #elif DEBUGMODE==40 && defined(SS_REFRACTION) gl_FragColor.rgb=subSurfaceOut.environmentRefraction.rgb; #define DEBUGMODE_GAMMA #elif DEBUGMODE==41 && defined(REFLECTION) gl_FragColor.rgb=reflectionOut.environmentRadiance.rgb; #ifndef GAMMAREFLECTION #define DEBUGMODE_GAMMA #endif #elif DEBUGMODE==42 && defined(CLEARCOAT) && defined(REFLECTION) gl_FragColor.rgb=clearcoatOut.environmentClearCoatRadiance.rgb; #define DEBUGMODE_GAMMA #elif DEBUGMODE==50 gl_FragColor.rgb=diffuseBase.rgb; #define DEBUGMODE_GAMMA #elif DEBUGMODE==51 && defined(SPECULARTERM) gl_FragColor.rgb=specularBase.rgb; #define DEBUGMODE_GAMMA #elif DEBUGMODE==52 && defined(CLEARCOAT) gl_FragColor.rgb=clearCoatBase.rgb; #define DEBUGMODE_GAMMA #elif DEBUGMODE==53 && defined(SHEEN) gl_FragColor.rgb=sheenBase.rgb; #define DEBUGMODE_GAMMA #elif DEBUGMODE==54 && defined(REFLECTION) gl_FragColor.rgb=reflectionOut.environmentIrradiance.rgb; #ifndef GAMMAREFLECTION #define DEBUGMODE_GAMMA #endif #elif DEBUGMODE==60 gl_FragColor.rgb=surfaceAlbedo.rgb; #define DEBUGMODE_GAMMA #elif DEBUGMODE==61 gl_FragColor.rgb=clearcoatOut.specularEnvironmentR0; #define DEBUGMODE_GAMMA #elif DEBUGMODE==62 && defined(METALLICWORKFLOW) gl_FragColor.rgb=vec3(reflectivityOut.metallicRoughness.r); #elif DEBUGMODE==71 && defined(METALLICWORKFLOW) gl_FragColor.rgb=reflectivityOut.metallicF0; #elif DEBUGMODE==63 gl_FragColor.rgb=vec3(roughness); #elif DEBUGMODE==64 gl_FragColor.rgb=vec3(alphaG); #elif DEBUGMODE==65 gl_FragColor.rgb=vec3(NdotV); #elif DEBUGMODE==66 && defined(CLEARCOAT) && defined(CLEARCOAT_TINT) gl_FragColor.rgb=clearcoatOut.clearCoatColor.rgb; #define DEBUGMODE_GAMMA #elif DEBUGMODE==67 && defined(CLEARCOAT) gl_FragColor.rgb=vec3(clearcoatOut.clearCoatRoughness); #elif DEBUGMODE==68 && defined(CLEARCOAT) gl_FragColor.rgb=vec3(clearcoatOut.clearCoatNdotV); #elif DEBUGMODE==69 && defined(SUBSURFACE) && defined(SS_TRANSLUCENCY) gl_FragColor.rgb=subSurfaceOut.transmittance; #elif DEBUGMODE==70 && defined(SUBSURFACE) && defined(SS_REFRACTION) gl_FragColor.rgb=subSurfaceOut.refractionTransmittance; #elif DEBUGMODE==72 gl_FragColor.rgb=vec3(microSurface); #elif DEBUGMODE==73 gl_FragColor.rgb=vAlbedoColor.rgb; #define DEBUGMODE_GAMMA #elif DEBUGMODE==74 && !defined(METALLICWORKFLOW) gl_FragColor.rgb=vReflectivityColor.rgb; #define DEBUGMODE_GAMMA #elif DEBUGMODE==75 gl_FragColor.rgb=vEmissiveColor.rgb; #define DEBUGMODE_GAMMA #elif DEBUGMODE==80 && defined(RADIANCEOCCLUSION) gl_FragColor.rgb=vec3(seo); #elif DEBUGMODE==81 && defined(HORIZONOCCLUSION) gl_FragColor.rgb=vec3(eho); #elif DEBUGMODE==82 && defined(MS_BRDF_ENERGY_CONSERVATION) gl_FragColor.rgb=vec3(energyConservationFactor); #elif DEBUGMODE==83 && defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX) gl_FragColor.rgb=specularEnvironmentReflectance; #define DEBUGMODE_GAMMA #elif DEBUGMODE==84 && defined(CLEARCOAT) && defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX) gl_FragColor.rgb=clearcoatOut.clearCoatEnvironmentReflectance; #define DEBUGMODE_GAMMA #elif DEBUGMODE==85 && defined(SHEEN) && defined(REFLECTION) gl_FragColor.rgb=sheenOut.sheenEnvironmentReflectance; #define DEBUGMODE_GAMMA #elif DEBUGMODE==86 && defined(ALPHABLEND) gl_FragColor.rgb=vec3(luminanceOverAlpha); #elif DEBUGMODE==87 gl_FragColor.rgb=vec3(alpha); #elif DEBUGMODE==88 && defined(ALBEDO) gl_FragColor.rgb=vec3(albedoTexture.a); #else float stripeWidth=30.;float stripePos=floor((gl_FragCoord.x+gl_FragCoord.y)/stripeWidth);float whichColor=mod(stripePos,2.);vec3 color1=vec3(.6,.2,.2);vec3 color2=vec3(.3,.1,.1);gl_FragColor.rgb=mix(color1,color2,whichColor); #endif gl_FragColor.rgb*=vDebugMode.y; #ifdef DEBUGMODE_NORMALIZE gl_FragColor.rgb=normalize(gl_FragColor.rgb)*0.5+0.5; #endif #ifdef DEBUGMODE_GAMMA gl_FragColor.rgb=toGammaSpace(gl_FragColor.rgb); #endif gl_FragColor.a=1.0; #ifdef PREPASS gl_FragData[0]=toLinearSpace(gl_FragColor); gl_FragData[1]=vec4(0.,0.,0.,0.); #endif #ifdef DEBUGMODE_FORCERETURN return; #endif } #endif `; Le.IncludesShadersStore[Qce] = Yce; const Mce = "pbrPixelShader", Lce = `#if defined(BUMP) || !defined(NORMAL) || defined(FORCENORMALFORWARD) || defined(SPECULARAA) || defined(CLEARCOAT_BUMP) || defined(ANISOTROPIC) #extension GL_OES_standard_derivatives : enable #endif #ifdef LODBASEDMICROSFURACE #extension GL_EXT_shader_texture_lod : enable #endif #define CUSTOM_FRAGMENT_BEGIN #ifdef LOGARITHMICDEPTH #extension GL_EXT_frag_depth : enable #endif #include[SCENE_MRT_COUNT] precision highp float; #include #ifndef FROMLINEARSPACE #define FROMLINEARSPACE #endif #include<__decl__pbrFragment> #include #include<__decl__lightFragment>[0..maxSimultaneousLights] #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef REFLECTION #include #endif #define CUSTOM_FRAGMENT_DEFINITIONS #include #include #include #include #include #include #include #include #include #include void main(void) { #define CUSTOM_FRAGMENT_MAIN_BEGIN #include #include #include #include albedoOpacityOutParams albedoOpacityOut; #ifdef ALBEDO vec4 albedoTexture=texture2D(albedoSampler,vAlbedoUV+uvOffset); #endif #ifdef OPACITY vec4 opacityMap=texture2D(opacitySampler,vOpacityUV+uvOffset); #endif #ifdef DECAL vec4 decalColor=texture2D(decalSampler,vDecalUV+uvOffset); #endif albedoOpacityBlock( vAlbedoColor, #ifdef ALBEDO albedoTexture, vAlbedoInfos, #endif #ifdef OPACITY opacityMap, vOpacityInfos, #endif #ifdef DETAIL detailColor, vDetailInfos, #endif #ifdef DECAL decalColor, vDecalInfos, #endif albedoOpacityOut );vec3 surfaceAlbedo=albedoOpacityOut.surfaceAlbedo;float alpha=albedoOpacityOut.alpha; #define CUSTOM_FRAGMENT_UPDATE_ALPHA #include #define CUSTOM_FRAGMENT_BEFORE_LIGHTS ambientOcclusionOutParams aoOut; #ifdef AMBIENT vec3 ambientOcclusionColorMap=texture2D(ambientSampler,vAmbientUV+uvOffset).rgb; #endif ambientOcclusionBlock( #ifdef AMBIENT ambientOcclusionColorMap, vAmbientInfos, #endif aoOut ); #include #ifdef UNLIT vec3 diffuseBase=vec3(1.,1.,1.); #else vec3 baseColor=surfaceAlbedo;reflectivityOutParams reflectivityOut; #if defined(REFLECTIVITY) vec4 surfaceMetallicOrReflectivityColorMap=texture2D(reflectivitySampler,vReflectivityUV+uvOffset);vec4 baseReflectivity=surfaceMetallicOrReflectivityColorMap; #ifndef METALLICWORKFLOW #ifdef REFLECTIVITY_GAMMA surfaceMetallicOrReflectivityColorMap=toLinearSpace(surfaceMetallicOrReflectivityColorMap); #endif surfaceMetallicOrReflectivityColorMap.rgb*=vReflectivityInfos.y; #endif #endif #if defined(MICROSURFACEMAP) vec4 microSurfaceTexel=texture2D(microSurfaceSampler,vMicroSurfaceSamplerUV+uvOffset)*vMicroSurfaceSamplerInfos.y; #endif #ifdef METALLICWORKFLOW vec4 metallicReflectanceFactors=vMetallicReflectanceFactors; #ifdef REFLECTANCE vec4 reflectanceFactorsMap=texture2D(reflectanceSampler,vReflectanceUV+uvOffset); #ifdef REFLECTANCE_GAMMA reflectanceFactorsMap=toLinearSpace(reflectanceFactorsMap); #endif metallicReflectanceFactors.rgb*=reflectanceFactorsMap.rgb; #endif #ifdef METALLIC_REFLECTANCE vec4 metallicReflectanceFactorsMap=texture2D(metallicReflectanceSampler,vMetallicReflectanceUV+uvOffset); #ifdef METALLIC_REFLECTANCE_GAMMA metallicReflectanceFactorsMap=toLinearSpace(metallicReflectanceFactorsMap); #endif #ifndef METALLIC_REFLECTANCE_USE_ALPHA_ONLY metallicReflectanceFactors.rgb*=metallicReflectanceFactorsMap.rgb; #endif metallicReflectanceFactors*=metallicReflectanceFactorsMap.a; #endif #endif reflectivityBlock( vReflectivityColor, #ifdef METALLICWORKFLOW surfaceAlbedo, metallicReflectanceFactors, #endif #ifdef REFLECTIVITY vReflectivityInfos, surfaceMetallicOrReflectivityColorMap, #endif #if defined(METALLICWORKFLOW) && defined(REFLECTIVITY) && defined(AOSTOREINMETALMAPRED) aoOut.ambientOcclusionColor, #endif #ifdef MICROSURFACEMAP microSurfaceTexel, #endif #ifdef DETAIL detailColor, vDetailInfos, #endif reflectivityOut );float microSurface=reflectivityOut.microSurface;float roughness=reflectivityOut.roughness; #ifdef METALLICWORKFLOW surfaceAlbedo=reflectivityOut.surfaceAlbedo; #endif #if defined(METALLICWORKFLOW) && defined(REFLECTIVITY) && defined(AOSTOREINMETALMAPRED) aoOut.ambientOcclusionColor=reflectivityOut.ambientOcclusionColor; #endif #ifdef ALPHAFRESNEL #if defined(ALPHATEST) || defined(ALPHABLEND) alphaFresnelOutParams alphaFresnelOut;alphaFresnelBlock( normalW, viewDirectionW, alpha, microSurface, alphaFresnelOut );alpha=alphaFresnelOut.alpha; #endif #endif #include #ifdef ANISOTROPIC anisotropicOutParams anisotropicOut; #ifdef ANISOTROPIC_TEXTURE vec3 anisotropyMapData=texture2D(anisotropySampler,vAnisotropyUV+uvOffset).rgb*vAnisotropyInfos.y; #endif anisotropicBlock( vAnisotropy, roughness, #ifdef ANISOTROPIC_TEXTURE anisotropyMapData, #endif TBN, normalW, viewDirectionW, anisotropicOut ); #endif #ifdef REFLECTION reflectionOutParams reflectionOut; #ifndef USE_CUSTOM_REFLECTION reflectionBlock( vPositionW, normalW, alphaG, vReflectionMicrosurfaceInfos, vReflectionInfos, vReflectionColor, #ifdef ANISOTROPIC anisotropicOut, #endif #if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX) NdotVUnclamped, #endif #ifdef LINEARSPECULARREFLECTION roughness, #endif reflectionSampler, #if defined(NORMAL) && defined(USESPHERICALINVERTEX) vEnvironmentIrradiance, #endif #ifdef USESPHERICALFROMREFLECTIONMAP #if !defined(NORMAL) || !defined(USESPHERICALINVERTEX) reflectionMatrix, #endif #endif #ifdef USEIRRADIANCEMAP irradianceSampler, #endif #ifndef LODBASEDMICROSFURACE reflectionSamplerLow, reflectionSamplerHigh, #endif #ifdef REALTIME_FILTERING vReflectionFilteringInfo, #endif reflectionOut ); #else #define CUSTOM_REFLECTION #endif #endif #include #ifdef SHEEN sheenOutParams sheenOut; #ifdef SHEEN_TEXTURE vec4 sheenMapData=texture2D(sheenSampler,vSheenUV+uvOffset); #endif #if defined(SHEEN_ROUGHNESS) && defined(SHEEN_TEXTURE_ROUGHNESS) && !defined(SHEEN_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE) vec4 sheenMapRoughnessData=texture2D(sheenRoughnessSampler,vSheenRoughnessUV+uvOffset)*vSheenInfos.w; #endif sheenBlock( vSheenColor, #ifdef SHEEN_ROUGHNESS vSheenRoughness, #if defined(SHEEN_TEXTURE_ROUGHNESS) && !defined(SHEEN_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE) sheenMapRoughnessData, #endif #endif roughness, #ifdef SHEEN_TEXTURE sheenMapData, vSheenInfos.y, #endif reflectance, #ifdef SHEEN_LINKWITHALBEDO baseColor, surfaceAlbedo, #endif #ifdef ENVIRONMENTBRDF NdotV, environmentBrdf, #endif #if defined(REFLECTION) && defined(ENVIRONMENTBRDF) AARoughnessFactors, vReflectionMicrosurfaceInfos, vReflectionInfos, vReflectionColor, vLightingIntensity, reflectionSampler, reflectionOut.reflectionCoords, NdotVUnclamped, #ifndef LODBASEDMICROSFURACE reflectionSamplerLow, reflectionSamplerHigh, #endif #ifdef REALTIME_FILTERING vReflectionFilteringInfo, #endif #if !defined(REFLECTIONMAP_SKYBOX) && defined(RADIANCEOCCLUSION) seo, #endif #if !defined(REFLECTIONMAP_SKYBOX) && defined(HORIZONOCCLUSION) && defined(BUMP) && defined(REFLECTIONMAP_3D) eho, #endif #endif sheenOut ); #ifdef SHEEN_LINKWITHALBEDO surfaceAlbedo=sheenOut.surfaceAlbedo; #endif #endif #ifdef CLEARCOAT #ifdef CLEARCOAT_TEXTURE vec2 clearCoatMapData=texture2D(clearCoatSampler,vClearCoatUV+uvOffset).rg*vClearCoatInfos.y; #endif #endif #ifdef IRIDESCENCE iridescenceOutParams iridescenceOut; #ifdef IRIDESCENCE_TEXTURE vec2 iridescenceMapData=texture2D(iridescenceSampler,vIridescenceUV+uvOffset).rg*vIridescenceInfos.y; #endif #ifdef IRIDESCENCE_THICKNESS_TEXTURE vec2 iridescenceThicknessMapData=texture2D(iridescenceThicknessSampler,vIridescenceThicknessUV+uvOffset).rg*vIridescenceInfos.w; #endif iridescenceBlock( vIridescenceParams, NdotV, specularEnvironmentR0, #ifdef IRIDESCENCE_TEXTURE iridescenceMapData, #endif #ifdef IRIDESCENCE_THICKNESS_TEXTURE iridescenceThicknessMapData, #endif #ifdef CLEARCOAT NdotVUnclamped, #ifdef CLEARCOAT_TEXTURE clearCoatMapData, #endif #endif iridescenceOut );float iridescenceIntensity=iridescenceOut.iridescenceIntensity;specularEnvironmentR0=iridescenceOut.specularEnvironmentR0; #endif clearcoatOutParams clearcoatOut; #ifdef CLEARCOAT #if defined(CLEARCOAT_TEXTURE_ROUGHNESS) && !defined(CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE) vec4 clearCoatMapRoughnessData=texture2D(clearCoatRoughnessSampler,vClearCoatRoughnessUV+uvOffset)*vClearCoatInfos.w; #endif #if defined(CLEARCOAT_TINT) && defined(CLEARCOAT_TINT_TEXTURE) vec4 clearCoatTintMapData=texture2D(clearCoatTintSampler,vClearCoatTintUV+uvOffset); #endif #ifdef CLEARCOAT_BUMP vec4 clearCoatBumpMapData=texture2D(clearCoatBumpSampler,vClearCoatBumpUV+uvOffset); #endif clearcoatBlock( vPositionW, geometricNormalW, viewDirectionW, vClearCoatParams, #if defined(CLEARCOAT_TEXTURE_ROUGHNESS) && !defined(CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE) clearCoatMapRoughnessData, #endif specularEnvironmentR0, #ifdef CLEARCOAT_TEXTURE clearCoatMapData, #endif #ifdef CLEARCOAT_TINT vClearCoatTintParams, clearCoatColorAtDistance, vClearCoatRefractionParams, #ifdef CLEARCOAT_TINT_TEXTURE clearCoatTintMapData, #endif #endif #ifdef CLEARCOAT_BUMP vClearCoatBumpInfos, clearCoatBumpMapData, vClearCoatBumpUV, #if defined(TANGENT) && defined(NORMAL) vTBN, #else vClearCoatTangentSpaceParams, #endif #ifdef OBJECTSPACE_NORMALMAP normalMatrix, #endif #endif #if defined(FORCENORMALFORWARD) && defined(NORMAL) faceNormal, #endif #ifdef REFLECTION vReflectionMicrosurfaceInfos, vReflectionInfos, vReflectionColor, vLightingIntensity, reflectionSampler, #ifndef LODBASEDMICROSFURACE reflectionSamplerLow, reflectionSamplerHigh, #endif #ifdef REALTIME_FILTERING vReflectionFilteringInfo, #endif #endif #if defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX) #ifdef RADIANCEOCCLUSION ambientMonochrome, #endif #endif #if defined(CLEARCOAT_BUMP) || defined(TWOSIDEDLIGHTING) (gl_FrontFacing ? 1. : -1.), #endif clearcoatOut ); #else clearcoatOut.specularEnvironmentR0=specularEnvironmentR0; #endif #include subSurfaceOutParams subSurfaceOut; #ifdef SUBSURFACE #ifdef SS_THICKNESSANDMASK_TEXTURE vec4 thicknessMap=texture2D(thicknessSampler,vThicknessUV+uvOffset); #endif #ifdef SS_REFRACTIONINTENSITY_TEXTURE vec4 refractionIntensityMap=texture2D(refractionIntensitySampler,vRefractionIntensityUV+uvOffset); #endif #ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE vec4 translucencyIntensityMap=texture2D(translucencyIntensitySampler,vTranslucencyIntensityUV+uvOffset); #endif subSurfaceBlock( vSubSurfaceIntensity, vThicknessParam, vTintColor, normalW, specularEnvironmentReflectance, #ifdef SS_THICKNESSANDMASK_TEXTURE thicknessMap, #endif #ifdef SS_REFRACTIONINTENSITY_TEXTURE refractionIntensityMap, #endif #ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE translucencyIntensityMap, #endif #ifdef REFLECTION #ifdef SS_TRANSLUCENCY reflectionMatrix, #ifdef USESPHERICALFROMREFLECTIONMAP #if !defined(NORMAL) || !defined(USESPHERICALINVERTEX) reflectionOut.irradianceVector, #endif #if defined(REALTIME_FILTERING) reflectionSampler, vReflectionFilteringInfo, #endif #endif #ifdef USEIRRADIANCEMAP irradianceSampler, #endif #endif #endif #if defined(SS_REFRACTION) || defined(SS_TRANSLUCENCY) surfaceAlbedo, #endif #ifdef SS_REFRACTION vPositionW, viewDirectionW, view, vRefractionInfos, refractionMatrix, vRefractionMicrosurfaceInfos, vLightingIntensity, #ifdef SS_LINKREFRACTIONTOTRANSPARENCY alpha, #endif #ifdef SS_LODINREFRACTIONALPHA NdotVUnclamped, #endif #ifdef SS_LINEARSPECULARREFRACTION roughness, #endif alphaG, refractionSampler, #ifndef LODBASEDMICROSFURACE refractionSamplerLow, refractionSamplerHigh, #endif #ifdef ANISOTROPIC anisotropicOut, #endif #ifdef REALTIME_FILTERING vRefractionFilteringInfo, #endif #ifdef SS_USE_LOCAL_REFRACTIONMAP_CUBIC vRefractionPosition, vRefractionSize, #endif #ifdef SS_DISPERSION dispersion, #endif #endif #ifdef SS_TRANSLUCENCY vDiffusionDistance, #endif subSurfaceOut ); #ifdef SS_REFRACTION surfaceAlbedo=subSurfaceOut.surfaceAlbedo; #ifdef SS_LINKREFRACTIONTOTRANSPARENCY alpha=subSurfaceOut.alpha; #endif #endif #else subSurfaceOut.specularEnvironmentReflectance=specularEnvironmentReflectance; #endif #include #include[0..maxSimultaneousLights] #include #endif #include #define CUSTOM_FRAGMENT_BEFORE_FINALCOLORCOMPOSITION #include #include #include(color,finalColor) #include #define CUSTOM_FRAGMENT_BEFORE_FRAGCOLOR #ifdef PREPASS float writeGeometryInfo=finalColor.a>0.4 ? 1.0 : 0.0; #ifdef PREPASS_POSITION gl_FragData[PREPASS_POSITION_INDEX]=vec4(vPositionW,writeGeometryInfo); #endif #ifdef PREPASS_VELOCITY vec2 a=(vCurrentPosition.xy/vCurrentPosition.w)*0.5+0.5;vec2 b=(vPreviousPosition.xy/vPreviousPosition.w)*0.5+0.5;vec2 velocity=abs(a-b);velocity=vec2(pow(velocity.x,1.0/3.0),pow(velocity.y,1.0/3.0))*sign(a-b)*0.5+0.5;gl_FragData[PREPASS_VELOCITY_INDEX]=vec4(velocity,0.0,writeGeometryInfo); #endif #ifdef PREPASS_ALBEDO_SQRT vec3 sqAlbedo=sqrt(surfaceAlbedo); #endif #ifdef PREPASS_IRRADIANCE vec3 irradiance=finalDiffuse; #ifndef UNLIT #ifdef REFLECTION irradiance+=finalIrradiance; #endif #endif #ifdef SS_SCATTERING gl_FragData[0]=vec4(finalColor.rgb-irradiance,finalColor.a); irradiance/=sqAlbedo; #else gl_FragData[0]=finalColor; float scatteringDiffusionProfile=255.; #endif gl_FragData[PREPASS_IRRADIANCE_INDEX]=vec4(clamp(irradiance,vec3(0.),vec3(1.)),writeGeometryInfo*scatteringDiffusionProfile/255.); #else gl_FragData[0]=vec4(finalColor.rgb,finalColor.a); #endif #ifdef PREPASS_DEPTH gl_FragData[PREPASS_DEPTH_INDEX]=vec4(vViewPos.z,0.0,0.0,writeGeometryInfo); #endif #ifdef PREPASS_NORMAL #ifdef PREPASS_NORMAL_WORLDSPACE gl_FragData[PREPASS_NORMAL_INDEX]=vec4(normalW,writeGeometryInfo); #else gl_FragData[PREPASS_NORMAL_INDEX]=vec4(normalize((view*vec4(normalW,0.0)).rgb),writeGeometryInfo); #endif #endif #ifdef PREPASS_ALBEDO_SQRT gl_FragData[PREPASS_ALBEDO_SQRT_INDEX]=vec4(sqAlbedo,writeGeometryInfo); #endif #ifdef PREPASS_REFLECTIVITY #ifndef UNLIT gl_FragData[PREPASS_REFLECTIVITY_INDEX]=vec4(specularEnvironmentR0,microSurface)*writeGeometryInfo; #else gl_FragData[PREPASS_REFLECTIVITY_INDEX]=vec4( 0.0,0.0,0.0,1.0 )*writeGeometryInfo; #endif #endif #endif #if !defined(PREPASS) || defined(WEBGL2) gl_FragColor=finalColor; #endif #include #if ORDER_INDEPENDENT_TRANSPARENCY if (fragDepth==nearestDepth) {frontColor.rgb+=finalColor.rgb*finalColor.a*alphaMultiplier;frontColor.a=1.0-alphaMultiplier*(1.0-finalColor.a);} else {backColor+=finalColor;} #endif #include #define CUSTOM_FRAGMENT_MAIN_END } `; Le.ShadersStore[Mce] = Lce; const Kce = "pbrVertexDeclaration", Jce = `uniform mat4 view;uniform mat4 viewProjection; #ifdef ALBEDO uniform mat4 albedoMatrix;uniform vec2 vAlbedoInfos; #endif #ifdef AMBIENT uniform mat4 ambientMatrix;uniform vec4 vAmbientInfos; #endif #ifdef OPACITY uniform mat4 opacityMatrix;uniform vec2 vOpacityInfos; #endif #ifdef EMISSIVE uniform vec2 vEmissiveInfos;uniform mat4 emissiveMatrix; #endif #ifdef LIGHTMAP uniform vec2 vLightmapInfos;uniform mat4 lightmapMatrix; #endif #ifdef REFLECTIVITY uniform vec3 vReflectivityInfos;uniform mat4 reflectivityMatrix; #endif #ifdef METALLIC_REFLECTANCE uniform vec2 vMetallicReflectanceInfos;uniform mat4 metallicReflectanceMatrix; #endif #ifdef REFLECTANCE uniform vec2 vReflectanceInfos;uniform mat4 reflectanceMatrix; #endif #ifdef MICROSURFACEMAP uniform vec2 vMicroSurfaceSamplerInfos;uniform mat4 microSurfaceSamplerMatrix; #endif #ifdef BUMP uniform vec3 vBumpInfos;uniform mat4 bumpMatrix; #endif #ifdef POINTSIZE uniform float pointSize; #endif #ifdef REFLECTION uniform vec2 vReflectionInfos;uniform mat4 reflectionMatrix; #endif #ifdef CLEARCOAT #if defined(CLEARCOAT_TEXTURE) || defined(CLEARCOAT_TEXTURE_ROUGHNESS) uniform vec4 vClearCoatInfos; #endif #ifdef CLEARCOAT_TEXTURE uniform mat4 clearCoatMatrix; #endif #ifdef CLEARCOAT_TEXTURE_ROUGHNESS uniform mat4 clearCoatRoughnessMatrix; #endif #ifdef CLEARCOAT_BUMP uniform vec2 vClearCoatBumpInfos;uniform mat4 clearCoatBumpMatrix; #endif #ifdef CLEARCOAT_TINT_TEXTURE uniform vec2 vClearCoatTintInfos;uniform mat4 clearCoatTintMatrix; #endif #endif #ifdef IRIDESCENCE #if defined(IRIDESCENCE_TEXTURE) || defined(IRIDESCENCE_THICKNESS_TEXTURE) uniform vec4 vIridescenceInfos; #endif #ifdef IRIDESCENCE_TEXTURE uniform mat4 iridescenceMatrix; #endif #ifdef IRIDESCENCE_THICKNESS_TEXTURE uniform mat4 iridescenceThicknessMatrix; #endif #endif #ifdef ANISOTROPIC #ifdef ANISOTROPIC_TEXTURE uniform vec2 vAnisotropyInfos;uniform mat4 anisotropyMatrix; #endif #endif #ifdef SHEEN #if defined(SHEEN_TEXTURE) || defined(SHEEN_TEXTURE_ROUGHNESS) uniform vec4 vSheenInfos; #endif #ifdef SHEEN_TEXTURE uniform mat4 sheenMatrix; #endif #ifdef SHEEN_TEXTURE_ROUGHNESS uniform mat4 sheenRoughnessMatrix; #endif #endif #ifdef SUBSURFACE #ifdef SS_REFRACTION uniform vec4 vRefractionInfos;uniform mat4 refractionMatrix; #endif #ifdef SS_THICKNESSANDMASK_TEXTURE uniform vec2 vThicknessInfos;uniform mat4 thicknessMatrix; #endif #ifdef SS_REFRACTIONINTENSITY_TEXTURE uniform vec2 vRefractionIntensityInfos;uniform mat4 refractionIntensityMatrix; #endif #ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE uniform vec2 vTranslucencyIntensityInfos;uniform mat4 translucencyIntensityMatrix; #endif #endif #ifdef NORMAL #if defined(USESPHERICALFROMREFLECTIONMAP) && defined(USESPHERICALINVERTEX) #ifdef USESPHERICALFROMREFLECTIONMAP #ifdef SPHERICAL_HARMONICS uniform vec3 vSphericalL00;uniform vec3 vSphericalL1_1;uniform vec3 vSphericalL10;uniform vec3 vSphericalL11;uniform vec3 vSphericalL2_2;uniform vec3 vSphericalL2_1;uniform vec3 vSphericalL20;uniform vec3 vSphericalL21;uniform vec3 vSphericalL22; #else uniform vec3 vSphericalX;uniform vec3 vSphericalY;uniform vec3 vSphericalZ;uniform vec3 vSphericalXX_ZZ;uniform vec3 vSphericalYY_ZZ;uniform vec3 vSphericalZZ;uniform vec3 vSphericalXY;uniform vec3 vSphericalYZ;uniform vec3 vSphericalZX; #endif #endif #endif #endif #ifdef DETAIL uniform vec4 vDetailInfos;uniform mat4 detailMatrix; #endif #include #define ADDITIONAL_VERTEX_DECLARATION `; Le.IncludesShadersStore[Kce] = Jce; const zce = "pbrVertexShader", Gce = `precision highp float; #include<__decl__pbrVertex> #define CUSTOM_VERTEX_BEGIN attribute vec3 position; #ifdef NORMAL attribute vec3 normal; #endif #ifdef TANGENT attribute vec4 tangent; #endif #ifdef UV1 attribute vec2 uv; #endif #include[2..7] #include[1..7] #ifdef VERTEXCOLOR attribute vec4 color; #endif #include #include #include #include #include #include(_DEFINENAME_,ALBEDO,_VARYINGNAME_,Albedo) #include(_DEFINENAME_,DETAIL,_VARYINGNAME_,Detail) #include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient) #include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity) #include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive) #include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap) #include(_DEFINENAME_,REFLECTIVITY,_VARYINGNAME_,Reflectivity) #include(_DEFINENAME_,MICROSURFACEMAP,_VARYINGNAME_,MicroSurfaceSampler) #include(_DEFINENAME_,METALLIC_REFLECTANCE,_VARYINGNAME_,MetallicReflectance) #include(_DEFINENAME_,REFLECTANCE,_VARYINGNAME_,Reflectance) #include(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump) #include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal) #ifdef CLEARCOAT #include(_DEFINENAME_,CLEARCOAT_TEXTURE,_VARYINGNAME_,ClearCoat) #include(_DEFINENAME_,CLEARCOAT_TEXTURE_ROUGHNESS,_VARYINGNAME_,ClearCoatRoughness) #include(_DEFINENAME_,CLEARCOAT_BUMP,_VARYINGNAME_,ClearCoatBump) #include(_DEFINENAME_,CLEARCOAT_TINT_TEXTURE,_VARYINGNAME_,ClearCoatTint) #endif #ifdef IRIDESCENCE #include(_DEFINENAME_,IRIDESCENCE_TEXTURE,_VARYINGNAME_,Iridescence) #include(_DEFINENAME_,IRIDESCENCE_THICKNESS_TEXTURE,_VARYINGNAME_,IridescenceThickness) #endif #ifdef SHEEN #include(_DEFINENAME_,SHEEN_TEXTURE,_VARYINGNAME_,Sheen) #include(_DEFINENAME_,SHEEN_TEXTURE_ROUGHNESS,_VARYINGNAME_,SheenRoughness) #endif #ifdef ANISOTROPIC #include(_DEFINENAME_,ANISOTROPIC_TEXTURE,_VARYINGNAME_,Anisotropy) #endif #ifdef SUBSURFACE #include(_DEFINENAME_,SS_THICKNESSANDMASK_TEXTURE,_VARYINGNAME_,Thickness) #include(_DEFINENAME_,SS_REFRACTIONINTENSITY_TEXTURE,_VARYINGNAME_,RefractionIntensity) #include(_DEFINENAME_,SS_TRANSLUCENCYINTENSITY_TEXTURE,_VARYINGNAME_,TranslucencyIntensity) #endif varying vec3 vPositionW; #if DEBUGMODE>0 varying vec4 vClipSpacePosition; #endif #ifdef NORMAL varying vec3 vNormalW; #if defined(USESPHERICALFROMREFLECTIONMAP) && defined(USESPHERICALINVERTEX) varying vec3 vEnvironmentIrradiance; #include #endif #endif #if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES) varying vec4 vColor; #endif #include #include #include #include<__decl__lightVxFragment>[0..maxSimultaneousLights] #include #include[0..maxSimultaneousMorphTargets] #ifdef REFLECTIONMAP_SKYBOX varying vec3 vPositionUVW; #endif #if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED) varying vec3 vDirectionW; #endif #include #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN vec3 positionUpdated=position; #ifdef NORMAL vec3 normalUpdated=normal; #endif #ifdef TANGENT vec4 tangentUpdated=tangent; #endif #ifdef UV1 vec2 uvUpdated=uv; #endif #include #include[0..maxSimultaneousMorphTargets] #ifdef REFLECTIONMAP_SKYBOX vPositionUVW=positionUpdated; #endif #define CUSTOM_VERTEX_UPDATE_POSITION #define CUSTOM_VERTEX_UPDATE_NORMAL #include #if defined(PREPASS) && defined(PREPASS_VELOCITY) && !defined(BONES_VELOCITY_ENABLED) vCurrentPosition=viewProjection*finalWorld*vec4(positionUpdated,1.0);vPreviousPosition=previousViewProjection*finalPreviousWorld*vec4(positionUpdated,1.0); #endif #include #include vec4 worldPos=finalWorld*vec4(positionUpdated,1.0);vPositionW=vec3(worldPos); #include #ifdef NORMAL mat3 normalWorld=mat3(finalWorld); #if defined(INSTANCES) && defined(THIN_INSTANCES) vNormalW=normalUpdated/vec3(dot(normalWorld[0],normalWorld[0]),dot(normalWorld[1],normalWorld[1]),dot(normalWorld[2],normalWorld[2]));vNormalW=normalize(normalWorld*vNormalW); #else #ifdef NONUNIFORMSCALING normalWorld=transposeMat3(inverseMat3(normalWorld)); #endif vNormalW=normalize(normalWorld*normalUpdated); #endif #if defined(USESPHERICALFROMREFLECTIONMAP) && defined(USESPHERICALINVERTEX) vec3 reflectionVector=vec3(reflectionMatrix*vec4(vNormalW,0)).xyz; #ifdef REFLECTIONMAP_OPPOSITEZ reflectionVector.z*=-1.0; #endif vEnvironmentIrradiance=computeEnvironmentIrradiance(reflectionVector); #endif #endif #define CUSTOM_VERTEX_UPDATE_WORLDPOS #ifdef MULTIVIEW if (gl_ViewID_OVR==0u) {gl_Position=viewProjection*worldPos;} else {gl_Position=viewProjectionR*worldPos;} #else gl_Position=viewProjection*worldPos; #endif #if DEBUGMODE>0 vClipSpacePosition=gl_Position; #endif #if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED) vDirectionW=normalize(vec3(finalWorld*vec4(positionUpdated,0.0))); #endif #ifndef UV1 vec2 uvUpdated=vec2(0.,0.); #endif #ifdef MAINUV1 vMainUV1=uvUpdated; #endif #include[2..7] #include(_DEFINENAME_,ALBEDO,_VARYINGNAME_,Albedo,_MATRIXNAME_,albedo,_INFONAME_,AlbedoInfos.x) #include(_DEFINENAME_,DETAIL,_VARYINGNAME_,Detail,_MATRIXNAME_,detail,_INFONAME_,DetailInfos.x) #include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient,_MATRIXNAME_,ambient,_INFONAME_,AmbientInfos.x) #include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity,_MATRIXNAME_,opacity,_INFONAME_,OpacityInfos.x) #include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive,_MATRIXNAME_,emissive,_INFONAME_,EmissiveInfos.x) #include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap,_MATRIXNAME_,lightmap,_INFONAME_,LightmapInfos.x) #include(_DEFINENAME_,REFLECTIVITY,_VARYINGNAME_,Reflectivity,_MATRIXNAME_,reflectivity,_INFONAME_,ReflectivityInfos.x) #include(_DEFINENAME_,MICROSURFACEMAP,_VARYINGNAME_,MicroSurfaceSampler,_MATRIXNAME_,microSurfaceSampler,_INFONAME_,MicroSurfaceSamplerInfos.x) #include(_DEFINENAME_,METALLIC_REFLECTANCE,_VARYINGNAME_,MetallicReflectance,_MATRIXNAME_,metallicReflectance,_INFONAME_,MetallicReflectanceInfos.x) #include(_DEFINENAME_,REFLECTANCE,_VARYINGNAME_,Reflectance,_MATRIXNAME_,reflectance,_INFONAME_,ReflectanceInfos.x) #include(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump,_MATRIXNAME_,bump,_INFONAME_,BumpInfos.x) #include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal,_MATRIXNAME_,decal,_INFONAME_,DecalInfos.x) #ifdef CLEARCOAT #include(_DEFINENAME_,CLEARCOAT_TEXTURE,_VARYINGNAME_,ClearCoat,_MATRIXNAME_,clearCoat,_INFONAME_,ClearCoatInfos.x) #include(_DEFINENAME_,CLEARCOAT_TEXTURE_ROUGHNESS,_VARYINGNAME_,ClearCoatRoughness,_MATRIXNAME_,clearCoatRoughness,_INFONAME_,ClearCoatInfos.z) #include(_DEFINENAME_,CLEARCOAT_BUMP,_VARYINGNAME_,ClearCoatBump,_MATRIXNAME_,clearCoatBump,_INFONAME_,ClearCoatBumpInfos.x) #include(_DEFINENAME_,CLEARCOAT_TINT_TEXTURE,_VARYINGNAME_,ClearCoatTint,_MATRIXNAME_,clearCoatTint,_INFONAME_,ClearCoatTintInfos.x) #endif #ifdef IRIDESCENCE #include(_DEFINENAME_,IRIDESCENCE_TEXTURE,_VARYINGNAME_,Iridescence,_MATRIXNAME_,iridescence,_INFONAME_,IridescenceInfos.x) #include(_DEFINENAME_,IRIDESCENCE_THICKNESS_TEXTURE,_VARYINGNAME_,IridescenceThickness,_MATRIXNAME_,iridescenceThickness,_INFONAME_,IridescenceInfos.z) #endif #ifdef SHEEN #include(_DEFINENAME_,SHEEN_TEXTURE,_VARYINGNAME_,Sheen,_MATRIXNAME_,sheen,_INFONAME_,SheenInfos.x) #include(_DEFINENAME_,SHEEN_TEXTURE_ROUGHNESS,_VARYINGNAME_,SheenRoughness,_MATRIXNAME_,sheen,_INFONAME_,SheenInfos.z) #endif #ifdef ANISOTROPIC #include(_DEFINENAME_,ANISOTROPIC_TEXTURE,_VARYINGNAME_,Anisotropy,_MATRIXNAME_,anisotropy,_INFONAME_,AnisotropyInfos.x) #endif #ifdef SUBSURFACE #include(_DEFINENAME_,SS_THICKNESSANDMASK_TEXTURE,_VARYINGNAME_,Thickness,_MATRIXNAME_,thickness,_INFONAME_,ThicknessInfos.x) #include(_DEFINENAME_,SS_REFRACTIONINTENSITY_TEXTURE,_VARYINGNAME_,RefractionIntensity,_MATRIXNAME_,refractionIntensity,_INFONAME_,RefractionIntensityInfos.x) #include(_DEFINENAME_,SS_TRANSLUCENCYINTENSITY_TEXTURE,_VARYINGNAME_,TranslucencyIntensity,_MATRIXNAME_,translucencyIntensity,_INFONAME_,TranslucencyIntensityInfos.x) #endif #include #include #include #include[0..maxSimultaneousLights] #include #if defined(POINTSIZE) && !defined(WEBGPU) gl_PointSize=pointSize; #endif #include #define CUSTOM_VERTEX_MAIN_END }`; Le.ShadersStore[zce] = Gce; class Dte extends na { constructor() { super(...arguments), this.CLEARCOAT = !1, this.CLEARCOAT_DEFAULTIOR = !1, this.CLEARCOAT_TEXTURE = !1, this.CLEARCOAT_TEXTURE_ROUGHNESS = !1, this.CLEARCOAT_TEXTUREDIRECTUV = 0, this.CLEARCOAT_TEXTURE_ROUGHNESSDIRECTUV = 0, this.CLEARCOAT_BUMP = !1, this.CLEARCOAT_BUMPDIRECTUV = 0, this.CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE = !1, this.CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL = !1, this.CLEARCOAT_REMAP_F0 = !1, this.CLEARCOAT_TINT = !1, this.CLEARCOAT_TINT_TEXTURE = !1, this.CLEARCOAT_TINT_TEXTUREDIRECTUV = 0, this.CLEARCOAT_TINT_GAMMATEXTURE = !1; } } class h9 extends Gl { /** @internal */ _markAllSubMeshesAsTexturesDirty() { this._enable(this._isEnabled), this._internalMarkAllSubMeshesAsTexturesDirty(); } constructor(e, t = !0) { super(e, "PBRClearCoat", 100, new Dte(), t), this._isEnabled = !1, this.isEnabled = !1, this.intensity = 1, this.roughness = 0, this._indexOfRefraction = h9._DefaultIndexOfRefraction, this.indexOfRefraction = h9._DefaultIndexOfRefraction, this._texture = null, this.texture = null, this._useRoughnessFromMainTexture = !0, this.useRoughnessFromMainTexture = !0, this._textureRoughness = null, this.textureRoughness = null, this._remapF0OnInterfaceChange = !0, this.remapF0OnInterfaceChange = !0, this._bumpTexture = null, this.bumpTexture = null, this._isTintEnabled = !1, this.isTintEnabled = !1, this.tintColor = Ne.White(), this.tintColorAtDistance = 1, this.tintThickness = 1, this._tintTexture = null, this.tintTexture = null, this._internalMarkAllSubMeshesAsTexturesDirty = e._dirtyCallbacks[1]; } isReadyForSubMesh(e, t, r) { if (!this._isEnabled) return !0; const n = this._material._disableBumpMap; return !(e._areTexturesDirty && t.texturesEnabled && (this._texture && Dt.ClearCoatTextureEnabled && !this._texture.isReadyOrNotBlocking() || this._textureRoughness && Dt.ClearCoatTextureEnabled && !this._textureRoughness.isReadyOrNotBlocking() || r.getCaps().standardDerivatives && this._bumpTexture && Dt.ClearCoatBumpTextureEnabled && !n && !this._bumpTexture.isReady() || this._isTintEnabled && this._tintTexture && Dt.ClearCoatTintTextureEnabled && !this._tintTexture.isReadyOrNotBlocking())); } prepareDefinesBeforeAttributes(e, t) { var r; this._isEnabled ? (e.CLEARCOAT = !0, e.CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE = this._useRoughnessFromMainTexture, e.CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL = this._texture !== null && this._texture._texture === ((r = this._textureRoughness) === null || r === void 0 ? void 0 : r._texture) && this._texture.checkTransformsAreIdentical(this._textureRoughness), e.CLEARCOAT_REMAP_F0 = this._remapF0OnInterfaceChange, e._areTexturesDirty && t.texturesEnabled && (this._texture && Dt.ClearCoatTextureEnabled ? Ye.PrepareDefinesForMergedUV(this._texture, e, "CLEARCOAT_TEXTURE") : e.CLEARCOAT_TEXTURE = !1, this._textureRoughness && Dt.ClearCoatTextureEnabled ? Ye.PrepareDefinesForMergedUV(this._textureRoughness, e, "CLEARCOAT_TEXTURE_ROUGHNESS") : e.CLEARCOAT_TEXTURE_ROUGHNESS = !1, this._bumpTexture && Dt.ClearCoatBumpTextureEnabled ? Ye.PrepareDefinesForMergedUV(this._bumpTexture, e, "CLEARCOAT_BUMP") : e.CLEARCOAT_BUMP = !1, e.CLEARCOAT_DEFAULTIOR = this._indexOfRefraction === h9._DefaultIndexOfRefraction, this._isTintEnabled ? (e.CLEARCOAT_TINT = !0, this._tintTexture && Dt.ClearCoatTintTextureEnabled ? (Ye.PrepareDefinesForMergedUV(this._tintTexture, e, "CLEARCOAT_TINT_TEXTURE"), e.CLEARCOAT_TINT_GAMMATEXTURE = this._tintTexture.gammaSpace) : e.CLEARCOAT_TINT_TEXTURE = !1) : (e.CLEARCOAT_TINT = !1, e.CLEARCOAT_TINT_TEXTURE = !1))) : (e.CLEARCOAT = !1, e.CLEARCOAT_TEXTURE = !1, e.CLEARCOAT_TEXTURE_ROUGHNESS = !1, e.CLEARCOAT_BUMP = !1, e.CLEARCOAT_TINT = !1, e.CLEARCOAT_TINT_TEXTURE = !1, e.CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE = !1, e.CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL = !1, e.CLEARCOAT_DEFAULTIOR = !1, e.CLEARCOAT_TEXTUREDIRECTUV = 0, e.CLEARCOAT_TEXTURE_ROUGHNESSDIRECTUV = 0, e.CLEARCOAT_BUMPDIRECTUV = 0, e.CLEARCOAT_REMAP_F0 = !1, e.CLEARCOAT_TINT_TEXTUREDIRECTUV = 0, e.CLEARCOAT_TINT_GAMMATEXTURE = !1); } bindForSubMesh(e, t, r, n) { var i, s, a, f, o, d, v, u; if (!this._isEnabled) return; const l = n.materialDefines, P = this._material.isFrozen, p = this._material._disableBumpMap, c = this._material._invertNormalMapX, H = this._material._invertNormalMapY, T = l.CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL; if (!e.useUbo || !P || !e.isSync) { T && Dt.ClearCoatTextureEnabled ? (e.updateFloat4("vClearCoatInfos", this._texture.coordinatesIndex, this._texture.level, -1, -1), Ye.BindTextureMatrix(this._texture, e, "clearCoat")) : (this._texture || this._textureRoughness) && Dt.ClearCoatTextureEnabled && (e.updateFloat4("vClearCoatInfos", (s = (i = this._texture) === null || i === void 0 ? void 0 : i.coordinatesIndex) !== null && s !== void 0 ? s : 0, (f = (a = this._texture) === null || a === void 0 ? void 0 : a.level) !== null && f !== void 0 ? f : 0, (d = (o = this._textureRoughness) === null || o === void 0 ? void 0 : o.coordinatesIndex) !== null && d !== void 0 ? d : 0, (u = (v = this._textureRoughness) === null || v === void 0 ? void 0 : v.level) !== null && u !== void 0 ? u : 0), this._texture && Ye.BindTextureMatrix(this._texture, e, "clearCoat"), this._textureRoughness && !T && !l.CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE && Ye.BindTextureMatrix(this._textureRoughness, e, "clearCoatRoughness")), this._bumpTexture && r.getCaps().standardDerivatives && Dt.ClearCoatTextureEnabled && !p && (e.updateFloat2("vClearCoatBumpInfos", this._bumpTexture.coordinatesIndex, this._bumpTexture.level), Ye.BindTextureMatrix(this._bumpTexture, e, "clearCoatBump"), t._mirroredCameraPosition ? e.updateFloat2("vClearCoatTangentSpaceParams", c ? 1 : -1, H ? 1 : -1) : e.updateFloat2("vClearCoatTangentSpaceParams", c ? -1 : 1, H ? -1 : 1)), this._tintTexture && Dt.ClearCoatTintTextureEnabled && (e.updateFloat2("vClearCoatTintInfos", this._tintTexture.coordinatesIndex, this._tintTexture.level), Ye.BindTextureMatrix(this._tintTexture, e, "clearCoatTint")), e.updateFloat2("vClearCoatParams", this.intensity, this.roughness); const q = 1 - this._indexOfRefraction, b = 1 + this._indexOfRefraction, j = Math.pow(-q / b, 2), w = 1 / this._indexOfRefraction; e.updateFloat4("vClearCoatRefractionParams", j, w, q, b), this._isTintEnabled && (e.updateFloat4("vClearCoatTintParams", this.tintColor.r, this.tintColor.g, this.tintColor.b, Math.max(1e-5, this.tintThickness)), e.updateFloat("clearCoatColorAtDistance", Math.max(1e-5, this.tintColorAtDistance))); } t.texturesEnabled && (this._texture && Dt.ClearCoatTextureEnabled && e.setTexture("clearCoatSampler", this._texture), this._textureRoughness && !T && !l.CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE && Dt.ClearCoatTextureEnabled && e.setTexture("clearCoatRoughnessSampler", this._textureRoughness), this._bumpTexture && r.getCaps().standardDerivatives && Dt.ClearCoatBumpTextureEnabled && !p && e.setTexture("clearCoatBumpSampler", this._bumpTexture), this._isTintEnabled && this._tintTexture && Dt.ClearCoatTintTextureEnabled && e.setTexture("clearCoatTintSampler", this._tintTexture)); } hasTexture(e) { return this._texture === e || this._textureRoughness === e || this._bumpTexture === e || this._tintTexture === e; } getActiveTextures(e) { this._texture && e.push(this._texture), this._textureRoughness && e.push(this._textureRoughness), this._bumpTexture && e.push(this._bumpTexture), this._tintTexture && e.push(this._tintTexture); } getAnimatables(e) { this._texture && this._texture.animations && this._texture.animations.length > 0 && e.push(this._texture), this._textureRoughness && this._textureRoughness.animations && this._textureRoughness.animations.length > 0 && e.push(this._textureRoughness), this._bumpTexture && this._bumpTexture.animations && this._bumpTexture.animations.length > 0 && e.push(this._bumpTexture), this._tintTexture && this._tintTexture.animations && this._tintTexture.animations.length > 0 && e.push(this._tintTexture); } dispose(e) { var t, r, n, i; e && ((t = this._texture) === null || t === void 0 || t.dispose(), (r = this._textureRoughness) === null || r === void 0 || r.dispose(), (n = this._bumpTexture) === null || n === void 0 || n.dispose(), (i = this._tintTexture) === null || i === void 0 || i.dispose()); } getClassName() { return "PBRClearCoatConfiguration"; } addFallbacks(e, t, r) { return e.CLEARCOAT_BUMP && t.addFallback(r++, "CLEARCOAT_BUMP"), e.CLEARCOAT_TINT && t.addFallback(r++, "CLEARCOAT_TINT"), e.CLEARCOAT && t.addFallback(r++, "CLEARCOAT"), r; } getSamplers(e) { e.push("clearCoatSampler", "clearCoatRoughnessSampler", "clearCoatBumpSampler", "clearCoatTintSampler"); } getUniforms() { return { ubo: [ { name: "vClearCoatParams", size: 2, type: "vec2" }, { name: "vClearCoatRefractionParams", size: 4, type: "vec4" }, { name: "vClearCoatInfos", size: 4, type: "vec4" }, { name: "clearCoatMatrix", size: 16, type: "mat4" }, { name: "clearCoatRoughnessMatrix", size: 16, type: "mat4" }, { name: "vClearCoatBumpInfos", size: 2, type: "vec2" }, { name: "vClearCoatTangentSpaceParams", size: 2, type: "vec2" }, { name: "clearCoatBumpMatrix", size: 16, type: "mat4" }, { name: "vClearCoatTintParams", size: 4, type: "vec4" }, { name: "clearCoatColorAtDistance", size: 1, type: "float" }, { name: "vClearCoatTintInfos", size: 2, type: "vec2" }, { name: "clearCoatTintMatrix", size: 16, type: "mat4" } ] }; } } h9._DefaultIndexOfRefraction = 1.5; C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], h9.prototype, "isEnabled", void 0); C([ M() ], h9.prototype, "intensity", void 0); C([ M() ], h9.prototype, "roughness", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], h9.prototype, "indexOfRefraction", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], h9.prototype, "texture", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], h9.prototype, "useRoughnessFromMainTexture", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], h9.prototype, "textureRoughness", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], h9.prototype, "remapF0OnInterfaceChange", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], h9.prototype, "bumpTexture", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], h9.prototype, "isTintEnabled", void 0); C([ Oi() ], h9.prototype, "tintColor", void 0); C([ M() ], h9.prototype, "tintColorAtDistance", void 0); C([ M() ], h9.prototype, "tintThickness", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], h9.prototype, "tintTexture", void 0); class jte extends na { constructor() { super(...arguments), this.IRIDESCENCE = !1, this.IRIDESCENCE_TEXTURE = !1, this.IRIDESCENCE_TEXTUREDIRECTUV = 0, this.IRIDESCENCE_THICKNESS_TEXTURE = !1, this.IRIDESCENCE_THICKNESS_TEXTUREDIRECTUV = 0, this.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE = !1; } } class dd extends Gl { /** @internal */ _markAllSubMeshesAsTexturesDirty() { this._enable(this._isEnabled), this._internalMarkAllSubMeshesAsTexturesDirty(); } constructor(e, t = !0) { super(e, "PBRIridescence", 110, new jte(), t), this._isEnabled = !1, this.isEnabled = !1, this.intensity = 1, this.minimumThickness = dd._DefaultMinimumThickness, this.maximumThickness = dd._DefaultMaximumThickness, this.indexOfRefraction = dd._DefaultIndexOfRefraction, this._texture = null, this.texture = null, this._thicknessTexture = null, this.thicknessTexture = null, this._internalMarkAllSubMeshesAsTexturesDirty = e._dirtyCallbacks[1]; } isReadyForSubMesh(e, t) { return this._isEnabled ? !(e._areTexturesDirty && t.texturesEnabled && (this._texture && Dt.IridescenceTextureEnabled && !this._texture.isReadyOrNotBlocking() || this._thicknessTexture && Dt.IridescenceTextureEnabled && !this._thicknessTexture.isReadyOrNotBlocking())) : !0; } prepareDefinesBeforeAttributes(e, t) { var r; this._isEnabled ? (e.IRIDESCENCE = !0, e.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE = this._texture !== null && this._texture._texture === ((r = this._thicknessTexture) === null || r === void 0 ? void 0 : r._texture) && this._texture.checkTransformsAreIdentical(this._thicknessTexture), e._areTexturesDirty && t.texturesEnabled && (this._texture && Dt.IridescenceTextureEnabled ? Ye.PrepareDefinesForMergedUV(this._texture, e, "IRIDESCENCE_TEXTURE") : e.IRIDESCENCE_TEXTURE = !1, !e.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE && this._thicknessTexture && Dt.IridescenceTextureEnabled ? Ye.PrepareDefinesForMergedUV(this._thicknessTexture, e, "IRIDESCENCE_THICKNESS_TEXTURE") : e.IRIDESCENCE_THICKNESS_TEXTURE = !1)) : (e.IRIDESCENCE = !1, e.IRIDESCENCE_TEXTURE = !1, e.IRIDESCENCE_THICKNESS_TEXTURE = !1, e.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE = !1, e.IRIDESCENCE_TEXTUREDIRECTUV = 0, e.IRIDESCENCE_THICKNESS_TEXTUREDIRECTUV = 0); } bindForSubMesh(e, t, r, n) { var i, s, a, f, o, d, v, u; if (!this._isEnabled) return; const l = n.materialDefines, P = this._material.isFrozen, p = l.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE; (!e.useUbo || !P || !e.isSync) && (p && Dt.IridescenceTextureEnabled ? (e.updateFloat4("vIridescenceInfos", this._texture.coordinatesIndex, this._texture.level, -1, -1), Ye.BindTextureMatrix(this._texture, e, "iridescence")) : (this._texture || this._thicknessTexture) && Dt.IridescenceTextureEnabled && (e.updateFloat4("vIridescenceInfos", (s = (i = this._texture) === null || i === void 0 ? void 0 : i.coordinatesIndex) !== null && s !== void 0 ? s : 0, (f = (a = this._texture) === null || a === void 0 ? void 0 : a.level) !== null && f !== void 0 ? f : 0, (d = (o = this._thicknessTexture) === null || o === void 0 ? void 0 : o.coordinatesIndex) !== null && d !== void 0 ? d : 0, (u = (v = this._thicknessTexture) === null || v === void 0 ? void 0 : v.level) !== null && u !== void 0 ? u : 0), this._texture && Ye.BindTextureMatrix(this._texture, e, "iridescence"), this._thicknessTexture && !p && !l.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE && Ye.BindTextureMatrix(this._thicknessTexture, e, "iridescenceThickness")), e.updateFloat4("vIridescenceParams", this.intensity, this.indexOfRefraction, this.minimumThickness, this.maximumThickness)), t.texturesEnabled && (this._texture && Dt.IridescenceTextureEnabled && e.setTexture("iridescenceSampler", this._texture), this._thicknessTexture && !p && !l.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE && Dt.IridescenceTextureEnabled && e.setTexture("iridescenceThicknessSampler", this._thicknessTexture)); } hasTexture(e) { return this._texture === e || this._thicknessTexture === e; } getActiveTextures(e) { this._texture && e.push(this._texture), this._thicknessTexture && e.push(this._thicknessTexture); } getAnimatables(e) { this._texture && this._texture.animations && this._texture.animations.length > 0 && e.push(this._texture), this._thicknessTexture && this._thicknessTexture.animations && this._thicknessTexture.animations.length > 0 && e.push(this._thicknessTexture); } dispose(e) { var t, r; e && ((t = this._texture) === null || t === void 0 || t.dispose(), (r = this._thicknessTexture) === null || r === void 0 || r.dispose()); } getClassName() { return "PBRIridescenceConfiguration"; } addFallbacks(e, t, r) { return e.IRIDESCENCE && t.addFallback(r++, "IRIDESCENCE"), r; } getSamplers(e) { e.push("iridescenceSampler", "iridescenceThicknessSampler"); } getUniforms() { return { ubo: [ { name: "vIridescenceParams", size: 4, type: "vec4" }, { name: "vIridescenceInfos", size: 4, type: "vec4" }, { name: "iridescenceMatrix", size: 16, type: "mat4" }, { name: "iridescenceThicknessMatrix", size: 16, type: "mat4" } ] }; } } dd._DefaultMinimumThickness = 100; dd._DefaultMaximumThickness = 400; dd._DefaultIndexOfRefraction = 1.3; C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], dd.prototype, "isEnabled", void 0); C([ M() ], dd.prototype, "intensity", void 0); C([ M() ], dd.prototype, "minimumThickness", void 0); C([ M() ], dd.prototype, "maximumThickness", void 0); C([ M() ], dd.prototype, "indexOfRefraction", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], dd.prototype, "texture", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], dd.prototype, "thicknessTexture", void 0); class wte extends na { constructor() { super(...arguments), this.ANISOTROPIC = !1, this.ANISOTROPIC_TEXTURE = !1, this.ANISOTROPIC_TEXTUREDIRECTUV = 0, this.ANISOTROPIC_LEGACY = !1, this.MAINUV1 = !1; } } class DD extends Gl { /** * Sets the anisotropy direction as an angle. */ set angle(e) { this.direction.x = Math.cos(e), this.direction.y = Math.sin(e); } /** * Gets the anisotropy angle value in radians. * @returns the anisotropy angle value in radians. */ get angle() { return Math.atan2(this.direction.y, this.direction.x); } /** @internal */ _markAllSubMeshesAsTexturesDirty() { this._enable(this._isEnabled), this._internalMarkAllSubMeshesAsTexturesDirty(); } /** @internal */ _markAllSubMeshesAsMiscDirty() { this._enable(this._isEnabled), this._internalMarkAllSubMeshesAsMiscDirty(); } constructor(e, t = !0) { super(e, "PBRAnisotropic", 110, new wte(), t), this._isEnabled = !1, this.isEnabled = !1, this.intensity = 1, this.direction = new at(1, 0), this._texture = null, this.texture = null, this._legacy = !1, this.legacy = !1, this._internalMarkAllSubMeshesAsTexturesDirty = e._dirtyCallbacks[1], this._internalMarkAllSubMeshesAsMiscDirty = e._dirtyCallbacks[16]; } isReadyForSubMesh(e, t) { return this._isEnabled ? !(e._areTexturesDirty && t.texturesEnabled && this._texture && Dt.AnisotropicTextureEnabled && !this._texture.isReadyOrNotBlocking()) : !0; } prepareDefinesBeforeAttributes(e, t, r) { this._isEnabled ? (e.ANISOTROPIC = this._isEnabled, this._isEnabled && !r.isVerticesDataPresent(J.TangentKind) && (e._needUVs = !0, e.MAINUV1 = !0), e._areTexturesDirty && t.texturesEnabled && (this._texture && Dt.AnisotropicTextureEnabled ? Ye.PrepareDefinesForMergedUV(this._texture, e, "ANISOTROPIC_TEXTURE") : e.ANISOTROPIC_TEXTURE = !1), e._areMiscDirty && (e.ANISOTROPIC_LEGACY = this._legacy)) : (e.ANISOTROPIC = !1, e.ANISOTROPIC_TEXTURE = !1, e.ANISOTROPIC_TEXTUREDIRECTUV = 0, e.ANISOTROPIC_LEGACY = !1); } bindForSubMesh(e, t) { if (!this._isEnabled) return; const r = this._material.isFrozen; (!e.useUbo || !r || !e.isSync) && (this._texture && Dt.AnisotropicTextureEnabled && (e.updateFloat2("vAnisotropyInfos", this._texture.coordinatesIndex, this._texture.level), Ye.BindTextureMatrix(this._texture, e, "anisotropy")), e.updateFloat3("vAnisotropy", this.direction.x, this.direction.y, this.intensity)), t.texturesEnabled && this._texture && Dt.AnisotropicTextureEnabled && e.setTexture("anisotropySampler", this._texture); } hasTexture(e) { return this._texture === e; } getActiveTextures(e) { this._texture && e.push(this._texture); } getAnimatables(e) { this._texture && this._texture.animations && this._texture.animations.length > 0 && e.push(this._texture); } dispose(e) { e && this._texture && this._texture.dispose(); } getClassName() { return "PBRAnisotropicConfiguration"; } addFallbacks(e, t, r) { return e.ANISOTROPIC && t.addFallback(r++, "ANISOTROPIC"), r; } getSamplers(e) { e.push("anisotropySampler"); } getUniforms() { return { ubo: [ { name: "vAnisotropy", size: 3, type: "vec3" }, { name: "vAnisotropyInfos", size: 2, type: "vec2" }, { name: "anisotropyMatrix", size: 16, type: "mat4" } ] }; } /** * Parses a anisotropy Configuration from a serialized object. * @param source - Serialized object. * @param scene Defines the scene we are parsing for * @param rootUrl Defines the rootUrl to load from */ parse(e, t, r) { super.parse(e, t, r), e.legacy === void 0 && (this.legacy = !0); } } C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], DD.prototype, "isEnabled", void 0); C([ M() ], DD.prototype, "intensity", void 0); C([ HR() ], DD.prototype, "direction", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], DD.prototype, "texture", void 0); C([ M(), At("_markAllSubMeshesAsMiscDirty") ], DD.prototype, "legacy", void 0); class mte extends na { constructor() { super(...arguments), this.SHEEN = !1, this.SHEEN_TEXTURE = !1, this.SHEEN_GAMMATEXTURE = !1, this.SHEEN_TEXTURE_ROUGHNESS = !1, this.SHEEN_TEXTUREDIRECTUV = 0, this.SHEEN_TEXTURE_ROUGHNESSDIRECTUV = 0, this.SHEEN_LINKWITHALBEDO = !1, this.SHEEN_ROUGHNESS = !1, this.SHEEN_ALBEDOSCALING = !1, this.SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE = !1, this.SHEEN_TEXTURE_ROUGHNESS_IDENTICAL = !1; } } class K0 extends Gl { /** @internal */ _markAllSubMeshesAsTexturesDirty() { this._enable(this._isEnabled), this._internalMarkAllSubMeshesAsTexturesDirty(); } constructor(e, t = !0) { super(e, "Sheen", 120, new mte(), t), this._isEnabled = !1, this.isEnabled = !1, this._linkSheenWithAlbedo = !1, this.linkSheenWithAlbedo = !1, this.intensity = 1, this.color = Ne.White(), this._texture = null, this.texture = null, this._useRoughnessFromMainTexture = !0, this.useRoughnessFromMainTexture = !0, this._roughness = null, this.roughness = null, this._textureRoughness = null, this.textureRoughness = null, this._albedoScaling = !1, this.albedoScaling = !1, this._internalMarkAllSubMeshesAsTexturesDirty = e._dirtyCallbacks[1]; } isReadyForSubMesh(e, t) { return this._isEnabled ? !(e._areTexturesDirty && t.texturesEnabled && (this._texture && Dt.SheenTextureEnabled && !this._texture.isReadyOrNotBlocking() || this._textureRoughness && Dt.SheenTextureEnabled && !this._textureRoughness.isReadyOrNotBlocking())) : !0; } prepareDefinesBeforeAttributes(e, t) { var r; this._isEnabled ? (e.SHEEN = !0, e.SHEEN_LINKWITHALBEDO = this._linkSheenWithAlbedo, e.SHEEN_ROUGHNESS = this._roughness !== null, e.SHEEN_ALBEDOSCALING = this._albedoScaling, e.SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE = this._useRoughnessFromMainTexture, e.SHEEN_TEXTURE_ROUGHNESS_IDENTICAL = this._texture !== null && this._texture._texture === ((r = this._textureRoughness) === null || r === void 0 ? void 0 : r._texture) && this._texture.checkTransformsAreIdentical(this._textureRoughness), e._areTexturesDirty && t.texturesEnabled && (this._texture && Dt.SheenTextureEnabled ? (Ye.PrepareDefinesForMergedUV(this._texture, e, "SHEEN_TEXTURE"), e.SHEEN_GAMMATEXTURE = this._texture.gammaSpace) : e.SHEEN_TEXTURE = !1, this._textureRoughness && Dt.SheenTextureEnabled ? Ye.PrepareDefinesForMergedUV(this._textureRoughness, e, "SHEEN_TEXTURE_ROUGHNESS") : e.SHEEN_TEXTURE_ROUGHNESS = !1)) : (e.SHEEN = !1, e.SHEEN_TEXTURE = !1, e.SHEEN_TEXTURE_ROUGHNESS = !1, e.SHEEN_LINKWITHALBEDO = !1, e.SHEEN_ROUGHNESS = !1, e.SHEEN_ALBEDOSCALING = !1, e.SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE = !1, e.SHEEN_TEXTURE_ROUGHNESS_IDENTICAL = !1, e.SHEEN_GAMMATEXTURE = !1, e.SHEEN_TEXTUREDIRECTUV = 0, e.SHEEN_TEXTURE_ROUGHNESSDIRECTUV = 0); } bindForSubMesh(e, t, r, n) { var i, s, a, f, o, d, v, u; if (!this._isEnabled) return; const l = n.materialDefines, P = this._material.isFrozen, p = l.SHEEN_TEXTURE_ROUGHNESS_IDENTICAL; (!e.useUbo || !P || !e.isSync) && (p && Dt.SheenTextureEnabled ? (e.updateFloat4("vSheenInfos", this._texture.coordinatesIndex, this._texture.level, -1, -1), Ye.BindTextureMatrix(this._texture, e, "sheen")) : (this._texture || this._textureRoughness) && Dt.SheenTextureEnabled && (e.updateFloat4("vSheenInfos", (s = (i = this._texture) === null || i === void 0 ? void 0 : i.coordinatesIndex) !== null && s !== void 0 ? s : 0, (f = (a = this._texture) === null || a === void 0 ? void 0 : a.level) !== null && f !== void 0 ? f : 0, (d = (o = this._textureRoughness) === null || o === void 0 ? void 0 : o.coordinatesIndex) !== null && d !== void 0 ? d : 0, (u = (v = this._textureRoughness) === null || v === void 0 ? void 0 : v.level) !== null && u !== void 0 ? u : 0), this._texture && Ye.BindTextureMatrix(this._texture, e, "sheen"), this._textureRoughness && !p && !l.SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE && Ye.BindTextureMatrix(this._textureRoughness, e, "sheenRoughness")), e.updateFloat4("vSheenColor", this.color.r, this.color.g, this.color.b, this.intensity), this._roughness !== null && e.updateFloat("vSheenRoughness", this._roughness)), t.texturesEnabled && (this._texture && Dt.SheenTextureEnabled && e.setTexture("sheenSampler", this._texture), this._textureRoughness && !p && !l.SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE && Dt.SheenTextureEnabled && e.setTexture("sheenRoughnessSampler", this._textureRoughness)); } hasTexture(e) { return this._texture === e || this._textureRoughness === e; } getActiveTextures(e) { this._texture && e.push(this._texture), this._textureRoughness && e.push(this._textureRoughness); } getAnimatables(e) { this._texture && this._texture.animations && this._texture.animations.length > 0 && e.push(this._texture), this._textureRoughness && this._textureRoughness.animations && this._textureRoughness.animations.length > 0 && e.push(this._textureRoughness); } dispose(e) { var t, r; e && ((t = this._texture) === null || t === void 0 || t.dispose(), (r = this._textureRoughness) === null || r === void 0 || r.dispose()); } getClassName() { return "PBRSheenConfiguration"; } addFallbacks(e, t, r) { return e.SHEEN && t.addFallback(r++, "SHEEN"), r; } getSamplers(e) { e.push("sheenSampler", "sheenRoughnessSampler"); } getUniforms() { return { ubo: [ { name: "vSheenColor", size: 4, type: "vec4" }, { name: "vSheenRoughness", size: 1, type: "float" }, { name: "vSheenInfos", size: 4, type: "vec4" }, { name: "sheenMatrix", size: 16, type: "mat4" }, { name: "sheenRoughnessMatrix", size: 16, type: "mat4" } ] }; } } C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], K0.prototype, "isEnabled", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], K0.prototype, "linkSheenWithAlbedo", void 0); C([ M() ], K0.prototype, "intensity", void 0); C([ Oi() ], K0.prototype, "color", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], K0.prototype, "texture", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], K0.prototype, "useRoughnessFromMainTexture", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], K0.prototype, "roughness", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], K0.prototype, "textureRoughness", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], K0.prototype, "albedoScaling", void 0); class Bte extends na { constructor() { super(...arguments), this.SUBSURFACE = !1, this.SS_REFRACTION = !1, this.SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE = !1, this.SS_TRANSLUCENCY = !1, this.SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE = !1, this.SS_SCATTERING = !1, this.SS_DISPERSION = !1, this.SS_THICKNESSANDMASK_TEXTURE = !1, this.SS_THICKNESSANDMASK_TEXTUREDIRECTUV = 0, this.SS_HAS_THICKNESS = !1, this.SS_REFRACTIONINTENSITY_TEXTURE = !1, this.SS_REFRACTIONINTENSITY_TEXTUREDIRECTUV = 0, this.SS_TRANSLUCENCYINTENSITY_TEXTURE = !1, this.SS_TRANSLUCENCYINTENSITY_TEXTUREDIRECTUV = 0, this.SS_REFRACTIONMAP_3D = !1, this.SS_REFRACTIONMAP_OPPOSITEZ = !1, this.SS_LODINREFRACTIONALPHA = !1, this.SS_GAMMAREFRACTION = !1, this.SS_RGBDREFRACTION = !1, this.SS_LINEARSPECULARREFRACTION = !1, this.SS_LINKREFRACTIONTOTRANSPARENCY = !1, this.SS_ALBEDOFORREFRACTIONTINT = !1, this.SS_ALBEDOFORTRANSLUCENCYTINT = !1, this.SS_USE_LOCAL_REFRACTIONMAP_CUBIC = !1, this.SS_USE_THICKNESS_AS_DEPTH = !1, this.SS_MASK_FROM_THICKNESS_TEXTURE = !1, this.SS_USE_GLTF_TEXTURES = !1; } } class _o extends Gl { /** * Diffusion profile for subsurface scattering. * Useful for better scattering in the skins or foliages. */ get scatteringDiffusionProfile() { return this._scene.subSurfaceConfiguration ? this._scene.subSurfaceConfiguration.ssDiffusionProfileColors[this._scatteringDiffusionProfileIndex] : null; } set scatteringDiffusionProfile(e) { this._scene.enableSubSurfaceForPrePass() && e && (this._scatteringDiffusionProfileIndex = this._scene.subSurfaceConfiguration.addDiffusionProfile(e)); } /** * Index of refraction of the material's volume. * https://en.wikipedia.org/wiki/List_of_refractive_indices * * This ONLY impacts refraction. If not provided or given a non-valid value, * the volume will use the same IOR as the surface. */ get volumeIndexOfRefraction() { return this._volumeIndexOfRefraction >= 1 ? this._volumeIndexOfRefraction : this._indexOfRefraction; } set volumeIndexOfRefraction(e) { e >= 1 ? this._volumeIndexOfRefraction = e : this._volumeIndexOfRefraction = -1; } /** @internal */ _markAllSubMeshesAsTexturesDirty() { this._enable(this._isRefractionEnabled || this._isTranslucencyEnabled || this._isScatteringEnabled), this._internalMarkAllSubMeshesAsTexturesDirty(); } /** @internal */ _markScenePrePassDirty() { this._internalMarkAllSubMeshesAsTexturesDirty(), this._internalMarkScenePrePassDirty(); } constructor(e, t = !0) { super(e, "PBRSubSurface", 130, new Bte(), t), this._isRefractionEnabled = !1, this.isRefractionEnabled = !1, this._isTranslucencyEnabled = !1, this.isTranslucencyEnabled = !1, this._isDispersionEnabled = !1, this.isDispersionEnabled = !1, this._isScatteringEnabled = !1, this.isScatteringEnabled = !1, this._scatteringDiffusionProfileIndex = 0, this.refractionIntensity = 1, this.translucencyIntensity = 1, this.useAlbedoToTintRefraction = !1, this.useAlbedoToTintTranslucency = !1, this._thicknessTexture = null, this.thicknessTexture = null, this._refractionTexture = null, this.refractionTexture = null, this._indexOfRefraction = 1.5, this.indexOfRefraction = 1.5, this._volumeIndexOfRefraction = -1, this._invertRefractionY = !1, this.invertRefractionY = !1, this._linkRefractionWithTransparency = !1, this.linkRefractionWithTransparency = !1, this.minimumThickness = 0, this.maximumThickness = 1, this.useThicknessAsDepth = !1, this.tintColor = Ne.White(), this.tintColorAtDistance = 1, this.dispersion = 0, this.diffusionDistance = Ne.White(), this._useMaskFromThicknessTexture = !1, this.useMaskFromThicknessTexture = !1, this._refractionIntensityTexture = null, this.refractionIntensityTexture = null, this._translucencyIntensityTexture = null, this.translucencyIntensityTexture = null, this._useGltfStyleTextures = !1, this.useGltfStyleTextures = !1, this._scene = e.getScene(), this.registerForExtraEvents = !0, this._internalMarkAllSubMeshesAsTexturesDirty = e._dirtyCallbacks[1], this._internalMarkScenePrePassDirty = e._dirtyCallbacks[32]; } isReadyForSubMesh(e, t) { if (!this._isRefractionEnabled && !this._isTranslucencyEnabled && !this._isScatteringEnabled) return !0; if (e._areTexturesDirty && t.texturesEnabled) { if (this._thicknessTexture && Dt.ThicknessTextureEnabled && !this._thicknessTexture.isReadyOrNotBlocking()) return !1; const r = this._getRefractionTexture(t); if (r && Dt.RefractionTextureEnabled && !r.isReadyOrNotBlocking()) return !1; } return !0; } prepareDefinesBeforeAttributes(e, t) { if (!this._isRefractionEnabled && !this._isTranslucencyEnabled && !this._isScatteringEnabled) { e.SUBSURFACE = !1, e.SS_DISPERSION = !1, e.SS_TRANSLUCENCY = !1, e.SS_SCATTERING = !1, e.SS_REFRACTION = !1, e.SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE = !1, e.SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE = !1, e.SS_THICKNESSANDMASK_TEXTURE = !1, e.SS_THICKNESSANDMASK_TEXTUREDIRECTUV = 0, e.SS_HAS_THICKNESS = !1, e.SS_REFRACTIONINTENSITY_TEXTURE = !1, e.SS_REFRACTIONINTENSITY_TEXTUREDIRECTUV = 0, e.SS_TRANSLUCENCYINTENSITY_TEXTURE = !1, e.SS_TRANSLUCENCYINTENSITY_TEXTUREDIRECTUV = 0, e.SS_REFRACTIONMAP_3D = !1, e.SS_REFRACTIONMAP_OPPOSITEZ = !1, e.SS_LODINREFRACTIONALPHA = !1, e.SS_GAMMAREFRACTION = !1, e.SS_RGBDREFRACTION = !1, e.SS_LINEARSPECULARREFRACTION = !1, e.SS_LINKREFRACTIONTOTRANSPARENCY = !1, e.SS_ALBEDOFORREFRACTIONTINT = !1, e.SS_ALBEDOFORTRANSLUCENCYTINT = !1, e.SS_USE_LOCAL_REFRACTIONMAP_CUBIC = !1, e.SS_USE_THICKNESS_AS_DEPTH = !1, e.SS_MASK_FROM_THICKNESS_TEXTURE = !1, e.SS_USE_GLTF_TEXTURES = !1; return; } if (e._areTexturesDirty) { e.SUBSURFACE = !0, e.SS_DISPERSION = this._isDispersionEnabled, e.SS_TRANSLUCENCY = this._isTranslucencyEnabled, e.SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE = !1, e.SS_SCATTERING = this._isScatteringEnabled, e.SS_THICKNESSANDMASK_TEXTURE = !1, e.SS_REFRACTIONINTENSITY_TEXTURE = !1, e.SS_TRANSLUCENCYINTENSITY_TEXTURE = !1, e.SS_HAS_THICKNESS = !1, e.SS_MASK_FROM_THICKNESS_TEXTURE = !1, e.SS_USE_GLTF_TEXTURES = !1, e.SS_REFRACTION = !1, e.SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE = !1, e.SS_REFRACTIONMAP_3D = !1, e.SS_GAMMAREFRACTION = !1, e.SS_RGBDREFRACTION = !1, e.SS_LINEARSPECULARREFRACTION = !1, e.SS_REFRACTIONMAP_OPPOSITEZ = !1, e.SS_LODINREFRACTIONALPHA = !1, e.SS_LINKREFRACTIONTOTRANSPARENCY = !1, e.SS_ALBEDOFORREFRACTIONTINT = !1, e.SS_ALBEDOFORTRANSLUCENCYTINT = !1, e.SS_USE_LOCAL_REFRACTIONMAP_CUBIC = !1, e.SS_USE_THICKNESS_AS_DEPTH = !1; const r = !!this._thicknessTexture && !!this._refractionIntensityTexture && this._refractionIntensityTexture.checkTransformsAreIdentical(this._thicknessTexture) && this._refractionIntensityTexture._texture === this._thicknessTexture._texture, n = !!this._thicknessTexture && !!this._translucencyIntensityTexture && this._translucencyIntensityTexture.checkTransformsAreIdentical(this._thicknessTexture) && this._translucencyIntensityTexture._texture === this._thicknessTexture._texture, i = (r || !this._refractionIntensityTexture) && (n || !this._translucencyIntensityTexture); if (e._areTexturesDirty && t.texturesEnabled && (this._thicknessTexture && Dt.ThicknessTextureEnabled && Ye.PrepareDefinesForMergedUV(this._thicknessTexture, e, "SS_THICKNESSANDMASK_TEXTURE"), this._refractionIntensityTexture && Dt.RefractionIntensityTextureEnabled && !i && Ye.PrepareDefinesForMergedUV(this._refractionIntensityTexture, e, "SS_REFRACTIONINTENSITY_TEXTURE"), this._translucencyIntensityTexture && Dt.TranslucencyIntensityTextureEnabled && !i && Ye.PrepareDefinesForMergedUV(this._translucencyIntensityTexture, e, "SS_TRANSLUCENCYINTENSITY_TEXTURE")), e.SS_HAS_THICKNESS = this.maximumThickness - this.minimumThickness !== 0, e.SS_MASK_FROM_THICKNESS_TEXTURE = (this._useMaskFromThicknessTexture || !!this._refractionIntensityTexture || !!this._translucencyIntensityTexture) && i, e.SS_USE_GLTF_TEXTURES = this._useGltfStyleTextures, e.SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE = (this._useMaskFromThicknessTexture || !!this._refractionIntensityTexture) && i, e.SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE = (this._useMaskFromThicknessTexture || !!this._translucencyIntensityTexture) && i, this._isRefractionEnabled && t.texturesEnabled) { const s = this._getRefractionTexture(t); s && Dt.RefractionTextureEnabled && (e.SS_REFRACTION = !0, e.SS_REFRACTIONMAP_3D = s.isCube, e.SS_GAMMAREFRACTION = s.gammaSpace, e.SS_RGBDREFRACTION = s.isRGBD, e.SS_LINEARSPECULARREFRACTION = s.linearSpecularLOD, e.SS_REFRACTIONMAP_OPPOSITEZ = this._scene.useRightHandedSystem && s.isCube ? !s.invertZ : s.invertZ, e.SS_LODINREFRACTIONALPHA = s.lodLevelInAlpha, e.SS_LINKREFRACTIONTOTRANSPARENCY = this._linkRefractionWithTransparency, e.SS_ALBEDOFORREFRACTIONTINT = this.useAlbedoToTintRefraction, e.SS_USE_LOCAL_REFRACTIONMAP_CUBIC = s.isCube && s.boundingBoxSize, e.SS_USE_THICKNESS_AS_DEPTH = this.useThicknessAsDepth); } this._isTranslucencyEnabled && (e.SS_ALBEDOFORTRANSLUCENCYTINT = this.useAlbedoToTintTranslucency); } } /** * Binds the material data (this function is called even if mustRebind() returns false) * @param uniformBuffer defines the Uniform buffer to fill in. * @param scene defines the scene the material belongs to. * @param engine defines the engine the material belongs to. * @param subMesh the submesh to bind data for */ hardBindForSubMesh(e, t, r, n) { if (!this._isRefractionEnabled && !this._isTranslucencyEnabled && !this._isScatteringEnabled) return; n.getRenderingMesh().getWorldMatrix().decompose(ue.Vector3[0]); const i = Math.max(Math.abs(ue.Vector3[0].x), Math.abs(ue.Vector3[0].y), Math.abs(ue.Vector3[0].z)); e.updateFloat2("vThicknessParam", this.minimumThickness * i, (this.maximumThickness - this.minimumThickness) * i); } bindForSubMesh(e, t, r, n) { if (!this._isRefractionEnabled && !this._isTranslucencyEnabled && !this._isScatteringEnabled) return; const i = n.materialDefines, s = this._material.isFrozen, a = this._material.realTimeFiltering, f = i.LODBASEDMICROSFURACE, o = this._getRefractionTexture(t); if (!e.useUbo || !s || !e.isSync) { if (this._thicknessTexture && Dt.ThicknessTextureEnabled && (e.updateFloat2("vThicknessInfos", this._thicknessTexture.coordinatesIndex, this._thicknessTexture.level), Ye.BindTextureMatrix(this._thicknessTexture, e, "thickness")), this._refractionIntensityTexture && Dt.RefractionIntensityTextureEnabled && i.SS_REFRACTIONINTENSITY_TEXTURE && (e.updateFloat2("vRefractionIntensityInfos", this._refractionIntensityTexture.coordinatesIndex, this._refractionIntensityTexture.level), Ye.BindTextureMatrix(this._refractionIntensityTexture, e, "refractionIntensity")), this._translucencyIntensityTexture && Dt.TranslucencyIntensityTextureEnabled && i.SS_TRANSLUCENCYINTENSITY_TEXTURE && (e.updateFloat2("vTranslucencyIntensityInfos", this._translucencyIntensityTexture.coordinatesIndex, this._translucencyIntensityTexture.level), Ye.BindTextureMatrix(this._translucencyIntensityTexture, e, "translucencyIntensity")), o && Dt.RefractionTextureEnabled) { e.updateMatrix("refractionMatrix", o.getRefractionTextureMatrix()); let d = 1; o.isCube || o.depth && (d = o.depth); const v = o.getSize().width, u = this.volumeIndexOfRefraction; if (e.updateFloat4("vRefractionInfos", o.level, 1 / u, d, this._invertRefractionY ? -1 : 1), e.updateFloat4("vRefractionMicrosurfaceInfos", v, o.lodGenerationScale, o.lodGenerationOffset, 1 / this.indexOfRefraction), a && e.updateFloat2("vRefractionFilteringInfo", v, Xt.Log2(v)), o.boundingBoxSize) { const l = o; e.updateVector3("vRefractionPosition", l.boundingBoxPosition), e.updateVector3("vRefractionSize", l.boundingBoxSize); } } this._isScatteringEnabled && e.updateFloat("scatteringDiffusionProfile", this._scatteringDiffusionProfileIndex), e.updateColor3("vDiffusionDistance", this.diffusionDistance), e.updateFloat4("vTintColor", this.tintColor.r, this.tintColor.g, this.tintColor.b, Math.max(1e-5, this.tintColorAtDistance)), e.updateFloat3("vSubSurfaceIntensity", this.refractionIntensity, this.translucencyIntensity, 0), e.updateFloat("dispersion", this.dispersion); } t.texturesEnabled && (this._thicknessTexture && Dt.ThicknessTextureEnabled && e.setTexture("thicknessSampler", this._thicknessTexture), this._refractionIntensityTexture && Dt.RefractionIntensityTextureEnabled && i.SS_REFRACTIONINTENSITY_TEXTURE && e.setTexture("refractionIntensitySampler", this._refractionIntensityTexture), this._translucencyIntensityTexture && Dt.TranslucencyIntensityTextureEnabled && i.SS_TRANSLUCENCYINTENSITY_TEXTURE && e.setTexture("translucencyIntensitySampler", this._translucencyIntensityTexture), o && Dt.RefractionTextureEnabled && (f ? e.setTexture("refractionSampler", o) : (e.setTexture("refractionSampler", o._lodTextureMid || o), e.setTexture("refractionSamplerLow", o._lodTextureLow || o), e.setTexture("refractionSamplerHigh", o._lodTextureHigh || o)))); } /** * Returns the texture used for refraction or null if none is used. * @param scene defines the scene the material belongs to. * @returns - Refraction texture if present. If no refraction texture and refraction * is linked with transparency, returns environment texture. Otherwise, returns null. */ _getRefractionTexture(e) { return this._refractionTexture ? this._refractionTexture : this._isRefractionEnabled ? e.environmentTexture : null; } /** * Returns true if alpha blending should be disabled. */ get disableAlphaBlending() { return this._isRefractionEnabled && this._linkRefractionWithTransparency; } /** * Fills the list of render target textures. * @param renderTargets the list of render targets to update */ fillRenderTargetTextures(e) { Dt.RefractionTextureEnabled && this._refractionTexture && this._refractionTexture.isRenderTarget && e.push(this._refractionTexture); } hasTexture(e) { return this._thicknessTexture === e || this._refractionTexture === e; } hasRenderTargetTextures() { return !!(Dt.RefractionTextureEnabled && this._refractionTexture && this._refractionTexture.isRenderTarget); } getActiveTextures(e) { this._thicknessTexture && e.push(this._thicknessTexture), this._refractionTexture && e.push(this._refractionTexture); } getAnimatables(e) { this._thicknessTexture && this._thicknessTexture.animations && this._thicknessTexture.animations.length > 0 && e.push(this._thicknessTexture), this._refractionTexture && this._refractionTexture.animations && this._refractionTexture.animations.length > 0 && e.push(this._refractionTexture); } dispose(e) { e && (this._thicknessTexture && this._thicknessTexture.dispose(), this._refractionTexture && this._refractionTexture.dispose()); } getClassName() { return "PBRSubSurfaceConfiguration"; } addFallbacks(e, t, r) { return e.SS_SCATTERING && t.addFallback(r++, "SS_SCATTERING"), e.SS_TRANSLUCENCY && t.addFallback(r++, "SS_TRANSLUCENCY"), r; } getSamplers(e) { e.push("thicknessSampler", "refractionIntensitySampler", "translucencyIntensitySampler", "refractionSampler", "refractionSamplerLow", "refractionSamplerHigh"); } getUniforms() { return { ubo: [ { name: "vRefractionMicrosurfaceInfos", size: 4, type: "vec4" }, { name: "vRefractionFilteringInfo", size: 2, type: "vec2" }, { name: "vTranslucencyIntensityInfos", size: 2, type: "vec2" }, { name: "vRefractionInfos", size: 4, type: "vec4" }, { name: "refractionMatrix", size: 16, type: "mat4" }, { name: "vThicknessInfos", size: 2, type: "vec2" }, { name: "vRefractionIntensityInfos", size: 2, type: "vec2" }, { name: "thicknessMatrix", size: 16, type: "mat4" }, { name: "refractionIntensityMatrix", size: 16, type: "mat4" }, { name: "translucencyIntensityMatrix", size: 16, type: "mat4" }, { name: "vThicknessParam", size: 2, type: "vec2" }, { name: "vDiffusionDistance", size: 3, type: "vec3" }, { name: "vTintColor", size: 4, type: "vec4" }, { name: "vSubSurfaceIntensity", size: 3, type: "vec3" }, { name: "vRefractionPosition", size: 3, type: "vec3" }, { name: "vRefractionSize", size: 3, type: "vec3" }, { name: "scatteringDiffusionProfile", size: 1, type: "float" }, { name: "dispersion", size: 1, type: "float" } ] }; } } C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], _o.prototype, "isRefractionEnabled", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], _o.prototype, "isTranslucencyEnabled", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], _o.prototype, "isDispersionEnabled", void 0); C([ M(), At("_markScenePrePassDirty") ], _o.prototype, "isScatteringEnabled", void 0); C([ M() ], _o.prototype, "_scatteringDiffusionProfileIndex", void 0); C([ M() ], _o.prototype, "refractionIntensity", void 0); C([ M() ], _o.prototype, "translucencyIntensity", void 0); C([ M() ], _o.prototype, "useAlbedoToTintRefraction", void 0); C([ M() ], _o.prototype, "useAlbedoToTintTranslucency", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], _o.prototype, "thicknessTexture", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], _o.prototype, "refractionTexture", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], _o.prototype, "indexOfRefraction", void 0); C([ M() ], _o.prototype, "_volumeIndexOfRefraction", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], _o.prototype, "volumeIndexOfRefraction", null); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], _o.prototype, "invertRefractionY", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], _o.prototype, "linkRefractionWithTransparency", void 0); C([ M() ], _o.prototype, "minimumThickness", void 0); C([ M() ], _o.prototype, "maximumThickness", void 0); C([ M() ], _o.prototype, "useThicknessAsDepth", void 0); C([ Oi() ], _o.prototype, "tintColor", void 0); C([ M() ], _o.prototype, "tintColorAtDistance", void 0); C([ M() ], _o.prototype, "dispersion", void 0); C([ Oi() ], _o.prototype, "diffusionDistance", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], _o.prototype, "useMaskFromThicknessTexture", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], _o.prototype, "refractionIntensityTexture", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], _o.prototype, "translucencyIntensityTexture", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], _o.prototype, "useGltfStyleTextures", void 0); const UW = { effect: null, subMesh: null }; class cF extends na { /** * Initializes the PBR Material defines. * @param externalProperties The external properties */ constructor(e) { super(e), this.PBR = !0, this.NUM_SAMPLES = "0", this.REALTIME_FILTERING = !1, this.MAINUV1 = !1, this.MAINUV2 = !1, this.MAINUV3 = !1, this.MAINUV4 = !1, this.MAINUV5 = !1, this.MAINUV6 = !1, this.UV1 = !1, this.UV2 = !1, this.UV3 = !1, this.UV4 = !1, this.UV5 = !1, this.UV6 = !1, this.ALBEDO = !1, this.GAMMAALBEDO = !1, this.ALBEDODIRECTUV = 0, this.VERTEXCOLOR = !1, this.BAKED_VERTEX_ANIMATION_TEXTURE = !1, this.AMBIENT = !1, this.AMBIENTDIRECTUV = 0, this.AMBIENTINGRAYSCALE = !1, this.OPACITY = !1, this.VERTEXALPHA = !1, this.OPACITYDIRECTUV = 0, this.OPACITYRGB = !1, this.ALPHATEST = !1, this.DEPTHPREPASS = !1, this.ALPHABLEND = !1, this.ALPHAFROMALBEDO = !1, this.ALPHATESTVALUE = "0.5", this.SPECULAROVERALPHA = !1, this.RADIANCEOVERALPHA = !1, this.ALPHAFRESNEL = !1, this.LINEARALPHAFRESNEL = !1, this.PREMULTIPLYALPHA = !1, this.EMISSIVE = !1, this.EMISSIVEDIRECTUV = 0, this.GAMMAEMISSIVE = !1, this.REFLECTIVITY = !1, this.REFLECTIVITY_GAMMA = !1, this.REFLECTIVITYDIRECTUV = 0, this.SPECULARTERM = !1, this.MICROSURFACEFROMREFLECTIVITYMAP = !1, this.MICROSURFACEAUTOMATIC = !1, this.LODBASEDMICROSFURACE = !1, this.MICROSURFACEMAP = !1, this.MICROSURFACEMAPDIRECTUV = 0, this.METALLICWORKFLOW = !1, this.ROUGHNESSSTOREINMETALMAPALPHA = !1, this.ROUGHNESSSTOREINMETALMAPGREEN = !1, this.METALLNESSSTOREINMETALMAPBLUE = !1, this.AOSTOREINMETALMAPRED = !1, this.METALLIC_REFLECTANCE = !1, this.METALLIC_REFLECTANCE_GAMMA = !1, this.METALLIC_REFLECTANCEDIRECTUV = 0, this.METALLIC_REFLECTANCE_USE_ALPHA_ONLY = !1, this.REFLECTANCE = !1, this.REFLECTANCE_GAMMA = !1, this.REFLECTANCEDIRECTUV = 0, this.ENVIRONMENTBRDF = !1, this.ENVIRONMENTBRDF_RGBD = !1, this.NORMAL = !1, this.TANGENT = !1, this.BUMP = !1, this.BUMPDIRECTUV = 0, this.OBJECTSPACE_NORMALMAP = !1, this.PARALLAX = !1, this.PARALLAX_RHS = !1, this.PARALLAXOCCLUSION = !1, this.NORMALXYSCALE = !0, this.LIGHTMAP = !1, this.LIGHTMAPDIRECTUV = 0, this.USELIGHTMAPASSHADOWMAP = !1, this.GAMMALIGHTMAP = !1, this.RGBDLIGHTMAP = !1, this.REFLECTION = !1, this.REFLECTIONMAP_3D = !1, this.REFLECTIONMAP_SPHERICAL = !1, this.REFLECTIONMAP_PLANAR = !1, this.REFLECTIONMAP_CUBIC = !1, this.USE_LOCAL_REFLECTIONMAP_CUBIC = !1, this.REFLECTIONMAP_PROJECTION = !1, this.REFLECTIONMAP_SKYBOX = !1, this.REFLECTIONMAP_EXPLICIT = !1, this.REFLECTIONMAP_EQUIRECTANGULAR = !1, this.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = !1, this.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = !1, this.INVERTCUBICMAP = !1, this.USESPHERICALFROMREFLECTIONMAP = !1, this.USEIRRADIANCEMAP = !1, this.USESPHERICALINVERTEX = !1, this.REFLECTIONMAP_OPPOSITEZ = !1, this.LODINREFLECTIONALPHA = !1, this.GAMMAREFLECTION = !1, this.RGBDREFLECTION = !1, this.LINEARSPECULARREFLECTION = !1, this.RADIANCEOCCLUSION = !1, this.HORIZONOCCLUSION = !1, this.INSTANCES = !1, this.THIN_INSTANCES = !1, this.INSTANCESCOLOR = !1, this.PREPASS = !1, this.PREPASS_IRRADIANCE = !1, this.PREPASS_IRRADIANCE_INDEX = -1, this.PREPASS_ALBEDO_SQRT = !1, this.PREPASS_ALBEDO_SQRT_INDEX = -1, this.PREPASS_DEPTH = !1, this.PREPASS_DEPTH_INDEX = -1, this.PREPASS_NORMAL = !1, this.PREPASS_NORMAL_INDEX = -1, this.PREPASS_NORMAL_WORLDSPACE = !1, this.PREPASS_POSITION = !1, this.PREPASS_POSITION_INDEX = -1, this.PREPASS_VELOCITY = !1, this.PREPASS_VELOCITY_INDEX = -1, this.PREPASS_REFLECTIVITY = !1, this.PREPASS_REFLECTIVITY_INDEX = -1, this.SCENE_MRT_COUNT = 0, this.NUM_BONE_INFLUENCERS = 0, this.BonesPerMesh = 0, this.BONETEXTURE = !1, this.BONES_VELOCITY_ENABLED = !1, this.NONUNIFORMSCALING = !1, this.MORPHTARGETS = !1, this.MORPHTARGETS_NORMAL = !1, this.MORPHTARGETS_TANGENT = !1, this.MORPHTARGETS_UV = !1, this.NUM_MORPH_INFLUENCERS = 0, this.MORPHTARGETS_TEXTURE = !1, this.IMAGEPROCESSING = !1, this.VIGNETTE = !1, this.VIGNETTEBLENDMODEMULTIPLY = !1, this.VIGNETTEBLENDMODEOPAQUE = !1, this.TONEMAPPING = !1, this.TONEMAPPING_ACES = !1, this.CONTRAST = !1, this.COLORCURVES = !1, this.COLORGRADING = !1, this.COLORGRADING3D = !1, this.SAMPLER3DGREENDEPTH = !1, this.SAMPLER3DBGRMAP = !1, this.DITHER = !1, this.IMAGEPROCESSINGPOSTPROCESS = !1, this.SKIPFINALCOLORCLAMP = !1, this.EXPOSURE = !1, this.MULTIVIEW = !1, this.ORDER_INDEPENDENT_TRANSPARENCY = !1, this.ORDER_INDEPENDENT_TRANSPARENCY_16BITS = !1, this.USEPHYSICALLIGHTFALLOFF = !1, this.USEGLTFLIGHTFALLOFF = !1, this.TWOSIDEDLIGHTING = !1, this.SHADOWFLOAT = !1, this.CLIPPLANE = !1, this.CLIPPLANE2 = !1, this.CLIPPLANE3 = !1, this.CLIPPLANE4 = !1, this.CLIPPLANE5 = !1, this.CLIPPLANE6 = !1, this.POINTSIZE = !1, this.FOG = !1, this.LOGARITHMICDEPTH = !1, this.CAMERA_ORTHOGRAPHIC = !1, this.CAMERA_PERSPECTIVE = !1, this.FORCENORMALFORWARD = !1, this.SPECULARAA = !1, this.UNLIT = !1, this.DECAL_AFTER_DETAIL = !1, this.DEBUGMODE = 0, this.rebuild(); } /** * Resets the PBR Material defines. */ reset() { super.reset(), this.ALPHATESTVALUE = "0.5", this.PBR = !0, this.NORMALXYSCALE = !0; } } class fs extends P1 { /** * Enables realtime filtering on the texture. */ get realTimeFiltering() { return this._realTimeFiltering; } set realTimeFiltering(e) { this._realTimeFiltering = e, this.markAsDirty(1); } /** * Quality switch for realtime filtering */ get realTimeFilteringQuality() { return this._realTimeFilteringQuality; } set realTimeFilteringQuality(e) { this._realTimeFilteringQuality = e, this.markAsDirty(1); } /** * Can this material render to several textures at once */ get canRenderToMRT() { return !0; } /** * Attaches a new image processing configuration to the PBR Material. * @param configuration */ _attachImageProcessingConfiguration(e) { e !== this._imageProcessingConfiguration && (this._imageProcessingConfiguration && this._imageProcessingObserver && this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver), e ? this._imageProcessingConfiguration = e : this._imageProcessingConfiguration = this.getScene().imageProcessingConfiguration, this._imageProcessingConfiguration && (this._imageProcessingObserver = this._imageProcessingConfiguration.onUpdateParameters.add(() => { this._markAllSubMeshesAsImageProcessingDirty(); }))); } /** * Instantiates a new PBRMaterial instance. * * @param name The material name * @param scene The scene the material will be use in. */ constructor(e, t) { super(e, t), this._directIntensity = 1, this._emissiveIntensity = 1, this._environmentIntensity = 1, this._specularIntensity = 1, this._lightingInfos = new Ir(this._directIntensity, this._emissiveIntensity, this._environmentIntensity, this._specularIntensity), this._disableBumpMap = !1, this._albedoTexture = null, this._ambientTexture = null, this._ambientTextureStrength = 1, this._ambientTextureImpactOnAnalyticalLights = fs.DEFAULT_AO_ON_ANALYTICAL_LIGHTS, this._opacityTexture = null, this._reflectionTexture = null, this._emissiveTexture = null, this._reflectivityTexture = null, this._metallicTexture = null, this._metallic = null, this._roughness = null, this._metallicF0Factor = 1, this._metallicReflectanceColor = Ne.White(), this._useOnlyMetallicFromMetallicReflectanceTexture = !1, this._metallicReflectanceTexture = null, this._reflectanceTexture = null, this._microSurfaceTexture = null, this._bumpTexture = null, this._lightmapTexture = null, this._ambientColor = new Ne(0, 0, 0), this._albedoColor = new Ne(1, 1, 1), this._reflectivityColor = new Ne(1, 1, 1), this._reflectionColor = new Ne(1, 1, 1), this._emissiveColor = new Ne(0, 0, 0), this._microSurface = 0.9, this._useLightmapAsShadowmap = !1, this._useHorizonOcclusion = !0, this._useRadianceOcclusion = !0, this._useAlphaFromAlbedoTexture = !1, this._useSpecularOverAlpha = !0, this._useMicroSurfaceFromReflectivityMapAlpha = !1, this._useRoughnessFromMetallicTextureAlpha = !0, this._useRoughnessFromMetallicTextureGreen = !1, this._useMetallnessFromMetallicTextureBlue = !1, this._useAmbientOcclusionFromMetallicTextureRed = !1, this._useAmbientInGrayScale = !1, this._useAutoMicroSurfaceFromReflectivityMap = !1, this._lightFalloff = fs.LIGHTFALLOFF_PHYSICAL, this._useRadianceOverAlpha = !0, this._useObjectSpaceNormalMap = !1, this._useParallax = !1, this._useParallaxOcclusion = !1, this._parallaxScaleBias = 0.05, this._disableLighting = !1, this._maxSimultaneousLights = 4, this._invertNormalMapX = !1, this._invertNormalMapY = !1, this._twoSidedLighting = !1, this._alphaCutOff = 0.4, this._forceAlphaTest = !1, this._useAlphaFresnel = !1, this._useLinearAlphaFresnel = !1, this._environmentBRDFTexture = null, this._forceIrradianceInFragment = !1, this._realTimeFiltering = !1, this._realTimeFilteringQuality = 8, this._forceNormalForward = !1, this._enableSpecularAntiAliasing = !1, this._imageProcessingObserver = null, this._renderTargets = new qf(16), this._globalAmbientColor = new Ne(0, 0, 0), this._unlit = !1, this._applyDecalMapAfterDetailMap = !1, this._debugMode = 0, this.debugMode = 0, this.debugLimit = -1, this.debugFactor = 1, this._cacheHasRenderTargetTextures = !1, this.brdf = new fd(this), this.clearCoat = new h9(this), this.iridescence = new dd(this), this.anisotropy = new DD(this), this.sheen = new K0(this), this.subSurface = new _o(this), this.detailMap = new l4(this), this._attachImageProcessingConfiguration(null), this.getRenderTargetTextures = () => (this._renderTargets.reset(), Dt.ReflectionTextureEnabled && this._reflectionTexture && this._reflectionTexture.isRenderTarget && this._renderTargets.push(this._reflectionTexture), this._eventInfo.renderTargets = this._renderTargets, this._callbackPluginEventFillRenderTargetTextures(this._eventInfo), this._renderTargets), this._environmentBRDFTexture = nV(this.getScene()), this.prePassConfiguration = new GC(); } /** * Gets a boolean indicating that current material needs to register RTT */ get hasRenderTargetTextures() { return Dt.ReflectionTextureEnabled && this._reflectionTexture && this._reflectionTexture.isRenderTarget ? !0 : this._cacheHasRenderTargetTextures; } /** * Can this material render to prepass */ get isPrePassCapable() { return !this.disableDepthWrite; } /** * Gets the name of the material class. */ getClassName() { return "PBRBaseMaterial"; } /** * Returns true if alpha blending should be disabled. */ get _disableAlphaBlending() { var e; return this._transparencyMode === fs.PBRMATERIAL_OPAQUE || this._transparencyMode === fs.PBRMATERIAL_ALPHATEST || ((e = this.subSurface) === null || e === void 0 ? void 0 : e.disableAlphaBlending); } /** * Specifies whether or not this material should be rendered in alpha blend mode. */ needAlphaBlending() { return this._disableAlphaBlending ? !1 : this.alpha < 1 || this._opacityTexture != null || this._shouldUseAlphaFromAlbedoTexture(); } /** * Specifies whether or not this material should be rendered in alpha test mode. */ needAlphaTesting() { var e; return this._forceAlphaTest ? !0 : !((e = this.subSurface) === null || e === void 0) && e.disableAlphaBlending ? !1 : this._hasAlphaChannel() && (this._transparencyMode == null || this._transparencyMode === fs.PBRMATERIAL_ALPHATEST); } /** * Specifies whether or not the alpha value of the albedo texture should be used for alpha blending. */ _shouldUseAlphaFromAlbedoTexture() { return this._albedoTexture != null && this._albedoTexture.hasAlpha && this._useAlphaFromAlbedoTexture && this._transparencyMode !== fs.PBRMATERIAL_OPAQUE; } /** * Specifies whether or not there is a usable alpha channel for transparency. */ _hasAlphaChannel() { return this._albedoTexture != null && this._albedoTexture.hasAlpha || this._opacityTexture != null; } /** * Gets the texture used for the alpha test. */ getAlphaTestTexture() { return this._albedoTexture; } /** * Specifies that the submesh is ready to be used. * @param mesh - BJS mesh. * @param subMesh - A submesh of the BJS mesh. Used to check if it is ready. * @param useInstances - Specifies that instances should be used. * @returns - boolean indicating that the submesh is ready or not. */ isReadyForSubMesh(e, t, r) { var n; if (this._uniformBufferLayoutBuilt || this.buildUniformLayout(), t.effect && this.isFrozen && t.effect._wasPreviouslyReady && t.effect._wasPreviouslyUsingInstances === r) return !0; t.materialDefines || (this._callbackPluginEventGeneric(x6.GetDefineNames, this._eventInfo), t.materialDefines = new cF(this._eventInfo.defineNames)); const i = t.materialDefines; if (this._isReadyForSubMesh(t)) return !0; const s = this.getScene(), a = s.getEngine(); if (i._areTexturesDirty && (this._eventInfo.hasRenderTargetTextures = !1, this._callbackPluginEventHasRenderTargetTextures(this._eventInfo), this._cacheHasRenderTargetTextures = this._eventInfo.hasRenderTargetTextures, s.texturesEnabled)) { if (this._albedoTexture && Dt.DiffuseTextureEnabled && !this._albedoTexture.isReadyOrNotBlocking() || this._ambientTexture && Dt.AmbientTextureEnabled && !this._ambientTexture.isReadyOrNotBlocking() || this._opacityTexture && Dt.OpacityTextureEnabled && !this._opacityTexture.isReadyOrNotBlocking()) return !1; const u = this._getReflectionTexture(); if (u && Dt.ReflectionTextureEnabled) { if (!u.isReadyOrNotBlocking()) return !1; if (u.irradianceTexture) { if (!u.irradianceTexture.isReadyOrNotBlocking()) return !1; } else if (!u.sphericalPolynomial && (!((n = u.getInternalTexture()) === null || n === void 0) && n._sphericalPolynomialPromise)) return !1; } if (this._lightmapTexture && Dt.LightmapTextureEnabled && !this._lightmapTexture.isReadyOrNotBlocking() || this._emissiveTexture && Dt.EmissiveTextureEnabled && !this._emissiveTexture.isReadyOrNotBlocking()) return !1; if (Dt.SpecularTextureEnabled) { if (this._metallicTexture) { if (!this._metallicTexture.isReadyOrNotBlocking()) return !1; } else if (this._reflectivityTexture && !this._reflectivityTexture.isReadyOrNotBlocking()) return !1; if (this._metallicReflectanceTexture && !this._metallicReflectanceTexture.isReadyOrNotBlocking() || this._reflectanceTexture && !this._reflectanceTexture.isReadyOrNotBlocking() || this._microSurfaceTexture && !this._microSurfaceTexture.isReadyOrNotBlocking()) return !1; } if (a.getCaps().standardDerivatives && this._bumpTexture && Dt.BumpTextureEnabled && !this._disableBumpMap && !this._bumpTexture.isReady() || this._environmentBRDFTexture && Dt.ReflectionTextureEnabled && !this._environmentBRDFTexture.isReady()) return !1; } if (this._eventInfo.isReadyForSubMesh = !0, this._eventInfo.defines = i, this._eventInfo.subMesh = t, this._callbackPluginEventIsReadyForSubMesh(this._eventInfo), !this._eventInfo.isReadyForSubMesh || i._areImageProcessingDirty && this._imageProcessingConfiguration && !this._imageProcessingConfiguration.isReady()) return !1; !a.getCaps().standardDerivatives && !e.isVerticesDataPresent(J.NormalKind) && (e.createNormals(!0), Se.Warn("PBRMaterial: Normals have been created for the mesh: " + e.name)); const f = t.effect, o = i._areLightsDisposed; let d = this._prepareEffect(e, i, this.onCompiled, this.onError, r, null, t.getRenderingMesh().hasThinInstances), v = !1; if (d) if (this._onEffectCreatedObservable && (UW.effect = d, UW.subMesh = t, this._onEffectCreatedObservable.notifyObservers(UW)), this.allowShaderHotSwapping && f && !d.isReady()) { if (d = f, i.markAsUnprocessed(), v = this.isFrozen, o) return i._areLightsDisposed = !0, !1; } else s.resetCachedMaterial(), t.setEffect(d, i, this._materialContext); return !t.effect || !t.effect.isReady() ? !1 : (i._renderId = s.getRenderId(), t.effect._wasPreviouslyReady = !v, t.effect._wasPreviouslyUsingInstances = !!r, this._checkScenePerformancePriority(), !0); } /** * Specifies if the material uses metallic roughness workflow. * @returns boolean specifying if the material uses metallic roughness workflow. */ isMetallicWorkflow() { return !!(this._metallic != null || this._roughness != null || this._metallicTexture); } _prepareEffect(e, t, r = null, n = null, i = null, s = null, a) { if (this._prepareDefines(e, t, i, s, a), !t.isDirty) return null; t.markAsProcessed(); const o = this.getScene().getEngine(), d = new c1(); let v = 0; t.USESPHERICALINVERTEX && d.addFallback(v++, "USESPHERICALINVERTEX"), t.FOG && d.addFallback(v, "FOG"), t.SPECULARAA && d.addFallback(v, "SPECULARAA"), t.POINTSIZE && d.addFallback(v, "POINTSIZE"), t.LOGARITHMICDEPTH && d.addFallback(v, "LOGARITHMICDEPTH"), t.PARALLAX && d.addFallback(v, "PARALLAX"), t.PARALLAX_RHS && d.addFallback(v, "PARALLAX_RHS"), t.PARALLAXOCCLUSION && d.addFallback(v++, "PARALLAXOCCLUSION"), t.ENVIRONMENTBRDF && d.addFallback(v++, "ENVIRONMENTBRDF"), t.TANGENT && d.addFallback(v++, "TANGENT"), t.BUMP && d.addFallback(v++, "BUMP"), v = Ye.HandleFallbacksForShadows(t, d, this._maxSimultaneousLights, v++), t.SPECULARTERM && d.addFallback(v++, "SPECULARTERM"), t.USESPHERICALFROMREFLECTIONMAP && d.addFallback(v++, "USESPHERICALFROMREFLECTIONMAP"), t.USEIRRADIANCEMAP && d.addFallback(v++, "USEIRRADIANCEMAP"), t.LIGHTMAP && d.addFallback(v++, "LIGHTMAP"), t.NORMAL && d.addFallback(v++, "NORMAL"), t.AMBIENT && d.addFallback(v++, "AMBIENT"), t.EMISSIVE && d.addFallback(v++, "EMISSIVE"), t.VERTEXCOLOR && d.addFallback(v++, "VERTEXCOLOR"), t.MORPHTARGETS && d.addFallback(v++, "MORPHTARGETS"), t.MULTIVIEW && d.addFallback(0, "MULTIVIEW"); const u = [J.PositionKind]; t.NORMAL && u.push(J.NormalKind), t.TANGENT && u.push(J.TangentKind); for (let j = 1; j <= 6; ++j) t["UV" + j] && u.push(`uv${j === 1 ? "" : j}`); t.VERTEXCOLOR && u.push(J.ColorKind), t.INSTANCESCOLOR && u.push(J.ColorInstanceKind), Ye.PrepareAttributesForBones(u, e, t, d), Ye.PrepareAttributesForInstances(u, t), Ye.PrepareAttributesForMorphTargets(u, e, t), Ye.PrepareAttributesForBakedVertexAnimation(u, e, t); let l = "pbr"; const P = [ "world", "view", "viewProjection", "vEyePosition", "vLightsType", "vAmbientColor", "vAlbedoColor", "vReflectivityColor", "vMetallicReflectanceFactors", "vEmissiveColor", "visibility", "vReflectionColor", "vFogInfos", "vFogColor", "pointSize", "vAlbedoInfos", "vAmbientInfos", "vOpacityInfos", "vReflectionInfos", "vReflectionPosition", "vReflectionSize", "vEmissiveInfos", "vReflectivityInfos", "vReflectionFilteringInfo", "vMetallicReflectanceInfos", "vReflectanceInfos", "vMicroSurfaceSamplerInfos", "vBumpInfos", "vLightmapInfos", "mBones", "albedoMatrix", "ambientMatrix", "opacityMatrix", "reflectionMatrix", "emissiveMatrix", "reflectivityMatrix", "normalMatrix", "microSurfaceSamplerMatrix", "bumpMatrix", "lightmapMatrix", "metallicReflectanceMatrix", "reflectanceMatrix", "vLightingIntensity", "logarithmicDepthConstant", "vSphericalX", "vSphericalY", "vSphericalZ", "vSphericalXX_ZZ", "vSphericalYY_ZZ", "vSphericalZZ", "vSphericalXY", "vSphericalYZ", "vSphericalZX", "vSphericalL00", "vSphericalL1_1", "vSphericalL10", "vSphericalL11", "vSphericalL2_2", "vSphericalL2_1", "vSphericalL20", "vSphericalL21", "vSphericalL22", "vReflectionMicrosurfaceInfos", "vTangentSpaceParams", "boneTextureWidth", "vDebugMode", "morphTargetTextureInfo", "morphTargetTextureIndices" ], p = [ "albedoSampler", "reflectivitySampler", "ambientSampler", "emissiveSampler", "bumpSampler", "lightmapSampler", "opacitySampler", "reflectionSampler", "reflectionSamplerLow", "reflectionSamplerHigh", "irradianceSampler", "microSurfaceSampler", "environmentBrdfSampler", "boneSampler", "metallicReflectanceSampler", "reflectanceSampler", "morphTargets", "oitDepthSampler", "oitFrontColorSampler" ], c = ["Material", "Scene", "Mesh"], H = { maxSimultaneousLights: this._maxSimultaneousLights, maxSimultaneousMorphTargets: t.NUM_MORPH_INFLUENCERS }; this._eventInfo.fallbacks = d, this._eventInfo.fallbackRank = v, this._eventInfo.defines = t, this._eventInfo.uniforms = P, this._eventInfo.attributes = u, this._eventInfo.samplers = p, this._eventInfo.uniformBuffersNames = c, this._eventInfo.customCode = void 0, this._eventInfo.mesh = e, this._eventInfo.indexParameters = H, this._callbackPluginEventGeneric(x6.PrepareEffect, this._eventInfo), GC.AddUniforms(P), Mf(P), Ui && (Ui.PrepareUniforms(P, t), Ui.PrepareSamplers(p, t)), Ye.PrepareUniformsAndSamplersList({ uniformsNames: P, uniformBuffersNames: c, samplers: p, defines: t, maxSimultaneousLights: this._maxSimultaneousLights }); const T = {}; this.customShaderNameResolve && (l = this.customShaderNameResolve(l, P, c, p, t, u, T)); const q = t.toString(), b = o.createEffect(l, { attributes: u, uniformsNames: P, uniformBuffersNames: c, samplers: p, defines: q, fallbacks: d, onCompiled: r, onError: n, indexParameters: H, processFinalCode: T.processFinalCode, processCodeAfterIncludes: this._eventInfo.customCode, multiTarget: t.PREPASS }, o); return this._eventInfo.customCode = void 0, b; } _prepareDefines(e, t, r = null, n = null, i = !1) { var s; const a = this.getScene(), f = a.getEngine(); Ye.PrepareDefinesForLights(a, e, t, !0, this._maxSimultaneousLights, this._disableLighting), t._needNormals = !0, Ye.PrepareDefinesForMultiview(a, t); const o = this.needAlphaBlendingForMesh(e) && this.getScene().useOrderIndependentTransparency; if (Ye.PrepareDefinesForPrePass(a, t, this.canRenderToMRT && !o), Ye.PrepareDefinesForOIT(a, t, o), t.METALLICWORKFLOW = this.isMetallicWorkflow(), t._areTexturesDirty) { t._needUVs = !1; for (let d = 1; d <= 6; ++d) t["MAINUV" + d] = !1; if (a.texturesEnabled) { t.ALBEDODIRECTUV = 0, t.AMBIENTDIRECTUV = 0, t.OPACITYDIRECTUV = 0, t.EMISSIVEDIRECTUV = 0, t.REFLECTIVITYDIRECTUV = 0, t.MICROSURFACEMAPDIRECTUV = 0, t.METALLIC_REFLECTANCEDIRECTUV = 0, t.REFLECTANCEDIRECTUV = 0, t.BUMPDIRECTUV = 0, t.LIGHTMAPDIRECTUV = 0, f.getCaps().textureLOD && (t.LODBASEDMICROSFURACE = !0), this._albedoTexture && Dt.DiffuseTextureEnabled ? (Ye.PrepareDefinesForMergedUV(this._albedoTexture, t, "ALBEDO"), t.GAMMAALBEDO = this._albedoTexture.gammaSpace) : t.ALBEDO = !1, this._ambientTexture && Dt.AmbientTextureEnabled ? (Ye.PrepareDefinesForMergedUV(this._ambientTexture, t, "AMBIENT"), t.AMBIENTINGRAYSCALE = this._useAmbientInGrayScale) : t.AMBIENT = !1, this._opacityTexture && Dt.OpacityTextureEnabled ? (Ye.PrepareDefinesForMergedUV(this._opacityTexture, t, "OPACITY"), t.OPACITYRGB = this._opacityTexture.getAlphaFromRGB) : t.OPACITY = !1; const d = this._getReflectionTexture(); if (d && Dt.ReflectionTextureEnabled) { switch (t.REFLECTION = !0, t.GAMMAREFLECTION = d.gammaSpace, t.RGBDREFLECTION = d.isRGBD, t.LODINREFLECTIONALPHA = d.lodLevelInAlpha, t.LINEARSPECULARREFLECTION = d.linearSpecularLOD, this.realTimeFiltering && this.realTimeFilteringQuality > 0 ? (t.NUM_SAMPLES = "" + this.realTimeFilteringQuality, f._features.needTypeSuffixInShaderConstants && (t.NUM_SAMPLES = t.NUM_SAMPLES + "u"), t.REALTIME_FILTERING = !0) : t.REALTIME_FILTERING = !1, t.INVERTCUBICMAP = d.coordinatesMode === We.INVCUBIC_MODE, t.REFLECTIONMAP_3D = d.isCube, t.REFLECTIONMAP_OPPOSITEZ = t.REFLECTIONMAP_3D && this.getScene().useRightHandedSystem ? !d.invertZ : d.invertZ, t.REFLECTIONMAP_CUBIC = !1, t.REFLECTIONMAP_EXPLICIT = !1, t.REFLECTIONMAP_PLANAR = !1, t.REFLECTIONMAP_PROJECTION = !1, t.REFLECTIONMAP_SKYBOX = !1, t.REFLECTIONMAP_SPHERICAL = !1, t.REFLECTIONMAP_EQUIRECTANGULAR = !1, t.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = !1, t.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = !1, d.coordinatesMode) { case We.EXPLICIT_MODE: t.REFLECTIONMAP_EXPLICIT = !0; break; case We.PLANAR_MODE: t.REFLECTIONMAP_PLANAR = !0; break; case We.PROJECTION_MODE: t.REFLECTIONMAP_PROJECTION = !0; break; case We.SKYBOX_MODE: t.REFLECTIONMAP_SKYBOX = !0; break; case We.SPHERICAL_MODE: t.REFLECTIONMAP_SPHERICAL = !0; break; case We.EQUIRECTANGULAR_MODE: t.REFLECTIONMAP_EQUIRECTANGULAR = !0; break; case We.FIXED_EQUIRECTANGULAR_MODE: t.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = !0; break; case We.FIXED_EQUIRECTANGULAR_MIRRORED_MODE: t.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = !0; break; case We.CUBIC_MODE: case We.INVCUBIC_MODE: default: t.REFLECTIONMAP_CUBIC = !0, t.USE_LOCAL_REFLECTIONMAP_CUBIC = !!d.boundingBoxSize; break; } d.coordinatesMode !== We.SKYBOX_MODE && (d.irradianceTexture ? (t.USEIRRADIANCEMAP = !0, t.USESPHERICALFROMREFLECTIONMAP = !1) : d.isCube && (t.USESPHERICALFROMREFLECTIONMAP = !0, t.USEIRRADIANCEMAP = !1, this._forceIrradianceInFragment || this.realTimeFiltering || this._twoSidedLighting || f.getCaps().maxVaryingVectors <= 8 ? t.USESPHERICALINVERTEX = !1 : t.USESPHERICALINVERTEX = !0)); } else t.REFLECTION = !1, t.REFLECTIONMAP_3D = !1, t.REFLECTIONMAP_SPHERICAL = !1, t.REFLECTIONMAP_PLANAR = !1, t.REFLECTIONMAP_CUBIC = !1, t.USE_LOCAL_REFLECTIONMAP_CUBIC = !1, t.REFLECTIONMAP_PROJECTION = !1, t.REFLECTIONMAP_SKYBOX = !1, t.REFLECTIONMAP_EXPLICIT = !1, t.REFLECTIONMAP_EQUIRECTANGULAR = !1, t.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = !1, t.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = !1, t.INVERTCUBICMAP = !1, t.USESPHERICALFROMREFLECTIONMAP = !1, t.USEIRRADIANCEMAP = !1, t.USESPHERICALINVERTEX = !1, t.REFLECTIONMAP_OPPOSITEZ = !1, t.LODINREFLECTIONALPHA = !1, t.GAMMAREFLECTION = !1, t.RGBDREFLECTION = !1, t.LINEARSPECULARREFLECTION = !1; if (this._lightmapTexture && Dt.LightmapTextureEnabled ? (Ye.PrepareDefinesForMergedUV(this._lightmapTexture, t, "LIGHTMAP"), t.USELIGHTMAPASSHADOWMAP = this._useLightmapAsShadowmap, t.GAMMALIGHTMAP = this._lightmapTexture.gammaSpace, t.RGBDLIGHTMAP = this._lightmapTexture.isRGBD) : t.LIGHTMAP = !1, this._emissiveTexture && Dt.EmissiveTextureEnabled ? (Ye.PrepareDefinesForMergedUV(this._emissiveTexture, t, "EMISSIVE"), t.GAMMAEMISSIVE = this._emissiveTexture.gammaSpace) : t.EMISSIVE = !1, Dt.SpecularTextureEnabled) { if (this._metallicTexture ? (Ye.PrepareDefinesForMergedUV(this._metallicTexture, t, "REFLECTIVITY"), t.ROUGHNESSSTOREINMETALMAPALPHA = this._useRoughnessFromMetallicTextureAlpha, t.ROUGHNESSSTOREINMETALMAPGREEN = !this._useRoughnessFromMetallicTextureAlpha && this._useRoughnessFromMetallicTextureGreen, t.METALLNESSSTOREINMETALMAPBLUE = this._useMetallnessFromMetallicTextureBlue, t.AOSTOREINMETALMAPRED = this._useAmbientOcclusionFromMetallicTextureRed, t.REFLECTIVITY_GAMMA = !1) : this._reflectivityTexture ? (Ye.PrepareDefinesForMergedUV(this._reflectivityTexture, t, "REFLECTIVITY"), t.MICROSURFACEFROMREFLECTIVITYMAP = this._useMicroSurfaceFromReflectivityMapAlpha, t.MICROSURFACEAUTOMATIC = this._useAutoMicroSurfaceFromReflectivityMap, t.REFLECTIVITY_GAMMA = this._reflectivityTexture.gammaSpace) : t.REFLECTIVITY = !1, this._metallicReflectanceTexture || this._reflectanceTexture) { const v = this._metallicReflectanceTexture !== null && this._metallicReflectanceTexture._texture === ((s = this._reflectanceTexture) === null || s === void 0 ? void 0 : s._texture) && this._metallicReflectanceTexture.checkTransformsAreIdentical(this._reflectanceTexture); t.METALLIC_REFLECTANCE_USE_ALPHA_ONLY = this._useOnlyMetallicFromMetallicReflectanceTexture && !v, this._metallicReflectanceTexture ? (Ye.PrepareDefinesForMergedUV(this._metallicReflectanceTexture, t, "METALLIC_REFLECTANCE"), t.METALLIC_REFLECTANCE_GAMMA = this._metallicReflectanceTexture.gammaSpace) : t.METALLIC_REFLECTANCE = !1, this._reflectanceTexture && !v && (!this._metallicReflectanceTexture || this._metallicReflectanceTexture && this._useOnlyMetallicFromMetallicReflectanceTexture) ? (Ye.PrepareDefinesForMergedUV(this._reflectanceTexture, t, "REFLECTANCE"), t.REFLECTANCE_GAMMA = this._reflectanceTexture.gammaSpace) : t.REFLECTANCE = !1; } else t.METALLIC_REFLECTANCE = !1, t.REFLECTANCE = !1; this._microSurfaceTexture ? Ye.PrepareDefinesForMergedUV(this._microSurfaceTexture, t, "MICROSURFACEMAP") : t.MICROSURFACEMAP = !1; } else t.REFLECTIVITY = !1, t.MICROSURFACEMAP = !1; f.getCaps().standardDerivatives && this._bumpTexture && Dt.BumpTextureEnabled && !this._disableBumpMap ? (Ye.PrepareDefinesForMergedUV(this._bumpTexture, t, "BUMP"), this._useParallax && this._albedoTexture && Dt.DiffuseTextureEnabled ? (t.PARALLAX = !0, t.PARALLAX_RHS = a.useRightHandedSystem, t.PARALLAXOCCLUSION = !!this._useParallaxOcclusion) : t.PARALLAX = !1, t.OBJECTSPACE_NORMALMAP = this._useObjectSpaceNormalMap) : (t.BUMP = !1, t.PARALLAX = !1, t.PARALLAX_RHS = !1, t.PARALLAXOCCLUSION = !1, t.OBJECTSPACE_NORMALMAP = !1), this._environmentBRDFTexture && Dt.ReflectionTextureEnabled ? (t.ENVIRONMENTBRDF = !0, t.ENVIRONMENTBRDF_RGBD = this._environmentBRDFTexture.isRGBD) : (t.ENVIRONMENTBRDF = !1, t.ENVIRONMENTBRDF_RGBD = !1), this._shouldUseAlphaFromAlbedoTexture() ? t.ALPHAFROMALBEDO = !0 : t.ALPHAFROMALBEDO = !1; } t.SPECULAROVERALPHA = this._useSpecularOverAlpha, this._lightFalloff === fs.LIGHTFALLOFF_STANDARD ? (t.USEPHYSICALLIGHTFALLOFF = !1, t.USEGLTFLIGHTFALLOFF = !1) : this._lightFalloff === fs.LIGHTFALLOFF_GLTF ? (t.USEPHYSICALLIGHTFALLOFF = !1, t.USEGLTFLIGHTFALLOFF = !0) : (t.USEPHYSICALLIGHTFALLOFF = !0, t.USEGLTFLIGHTFALLOFF = !1), t.RADIANCEOVERALPHA = this._useRadianceOverAlpha, !this.backFaceCulling && this._twoSidedLighting ? t.TWOSIDEDLIGHTING = !0 : t.TWOSIDEDLIGHTING = !1, t.SPECULARAA = f.getCaps().standardDerivatives && this._enableSpecularAntiAliasing; } (t._areTexturesDirty || t._areMiscDirty) && (t.ALPHATESTVALUE = `${this._alphaCutOff}${this._alphaCutOff % 1 === 0 ? "." : ""}`, t.PREMULTIPLYALPHA = this.alphaMode === 7 || this.alphaMode === 8, t.ALPHABLEND = this.needAlphaBlendingForMesh(e), t.ALPHAFRESNEL = this._useAlphaFresnel || this._useLinearAlphaFresnel, t.LINEARALPHAFRESNEL = this._useLinearAlphaFresnel), t._areImageProcessingDirty && this._imageProcessingConfiguration && this._imageProcessingConfiguration.prepareDefines(t), t.FORCENORMALFORWARD = this._forceNormalForward, t.RADIANCEOCCLUSION = this._useRadianceOcclusion, t.HORIZONOCCLUSION = this._useHorizonOcclusion, t._areMiscDirty && (Ye.PrepareDefinesForMisc(e, a, this._useLogarithmicDepth, this.pointsCloud, this.fogEnabled, this._shouldTurnAlphaTestOn(e) || this._forceAlphaTest, t, this._applyDecalMapAfterDetailMap), t.UNLIT = this._unlit || (this.pointsCloud || this.wireframe) && !e.isVerticesDataPresent(J.NormalKind), t.DEBUGMODE = this._debugMode), Ye.PrepareDefinesForFrameBoundValues(a, f, this, t, !!r, n, i), this._eventInfo.defines = t, this._eventInfo.mesh = e, this._callbackPluginEventPrepareDefinesBeforeAttributes(this._eventInfo), Ye.PrepareDefinesForAttributes(e, t, !0, !0, !0, this._transparencyMode !== fs.PBRMATERIAL_OPAQUE), this._callbackPluginEventPrepareDefines(this._eventInfo); } /** * Force shader compilation * @param mesh * @param onCompiled * @param options */ forceCompilation(e, t, r) { const n = Object.assign({ clipPlane: !1, useInstances: !1 }, r); this._uniformBufferLayoutBuilt || this.buildUniformLayout(), this._callbackPluginEventGeneric(x6.GetDefineNames, this._eventInfo); const i = new cF(this._eventInfo.defineNames), s = this._prepareEffect(e, i, void 0, void 0, n.useInstances, n.clipPlane, e.hasThinInstances); this._onEffectCreatedObservable && (UW.effect = s, UW.subMesh = null, this._onEffectCreatedObservable.notifyObservers(UW)), s.isReady() ? t && t(this) : s.onCompileObservable.add(() => { t && t(this); }); } /** * Initializes the uniform buffer layout for the shader. */ buildUniformLayout() { const e = this._uniformBuffer; e.addUniform("vAlbedoInfos", 2), e.addUniform("vAmbientInfos", 4), e.addUniform("vOpacityInfos", 2), e.addUniform("vEmissiveInfos", 2), e.addUniform("vLightmapInfos", 2), e.addUniform("vReflectivityInfos", 3), e.addUniform("vMicroSurfaceSamplerInfos", 2), e.addUniform("vReflectionInfos", 2), e.addUniform("vReflectionFilteringInfo", 2), e.addUniform("vReflectionPosition", 3), e.addUniform("vReflectionSize", 3), e.addUniform("vBumpInfos", 3), e.addUniform("albedoMatrix", 16), e.addUniform("ambientMatrix", 16), e.addUniform("opacityMatrix", 16), e.addUniform("emissiveMatrix", 16), e.addUniform("lightmapMatrix", 16), e.addUniform("reflectivityMatrix", 16), e.addUniform("microSurfaceSamplerMatrix", 16), e.addUniform("bumpMatrix", 16), e.addUniform("vTangentSpaceParams", 2), e.addUniform("reflectionMatrix", 16), e.addUniform("vReflectionColor", 3), e.addUniform("vAlbedoColor", 4), e.addUniform("vLightingIntensity", 4), e.addUniform("vReflectionMicrosurfaceInfos", 3), e.addUniform("pointSize", 1), e.addUniform("vReflectivityColor", 4), e.addUniform("vEmissiveColor", 3), e.addUniform("vAmbientColor", 3), e.addUniform("vDebugMode", 2), e.addUniform("vMetallicReflectanceFactors", 4), e.addUniform("vMetallicReflectanceInfos", 2), e.addUniform("metallicReflectanceMatrix", 16), e.addUniform("vReflectanceInfos", 2), e.addUniform("reflectanceMatrix", 16), e.addUniform("vSphericalL00", 3), e.addUniform("vSphericalL1_1", 3), e.addUniform("vSphericalL10", 3), e.addUniform("vSphericalL11", 3), e.addUniform("vSphericalL2_2", 3), e.addUniform("vSphericalL2_1", 3), e.addUniform("vSphericalL20", 3), e.addUniform("vSphericalL21", 3), e.addUniform("vSphericalL22", 3), e.addUniform("vSphericalX", 3), e.addUniform("vSphericalY", 3), e.addUniform("vSphericalZ", 3), e.addUniform("vSphericalXX_ZZ", 3), e.addUniform("vSphericalYY_ZZ", 3), e.addUniform("vSphericalZZ", 3), e.addUniform("vSphericalXY", 3), e.addUniform("vSphericalYZ", 3), e.addUniform("vSphericalZX", 3), super.buildUniformLayout(); } /** * Binds the submesh data. * @param world - The world matrix. * @param mesh - The BJS mesh. * @param subMesh - A submesh of the BJS mesh. */ bindForSubMesh(e, t, r) { var n, i, s, a; const f = this.getScene(), o = r.materialDefines; if (!o) return; const d = r.effect; if (!d) return; this._activeEffect = d, t.getMeshUniformBuffer().bindToEffect(d, "Mesh"), t.transferToEffect(e); const v = f.getEngine(); this._uniformBuffer.bindToEffect(d, "Material"), this.prePassConfiguration.bindForSubMesh(this._activeEffect, f, t, e, this.isFrozen), this._eventInfo.subMesh = r, this._callbackPluginEventHardBindForSubMesh(this._eventInfo), o.OBJECTSPACE_NORMALMAP && (e.toNormalMatrix(this._normalMatrix), this.bindOnlyNormalMatrix(this._normalMatrix)); const u = d._forceRebindOnNextCall || this._mustRebind(f, d, t.visibility); Ye.BindBonesParameters(t, this._activeEffect, this.prePassConfiguration); let l = null; const P = this._uniformBuffer; if (u) { if (this.bindViewProjection(d), l = this._getReflectionTexture(), !P.useUbo || !this.isFrozen || !P.isSync || d._forceRebindOnNextCall) { if (f.texturesEnabled) { if (this._albedoTexture && Dt.DiffuseTextureEnabled && (P.updateFloat2("vAlbedoInfos", this._albedoTexture.coordinatesIndex, this._albedoTexture.level), Ye.BindTextureMatrix(this._albedoTexture, P, "albedo")), this._ambientTexture && Dt.AmbientTextureEnabled && (P.updateFloat4("vAmbientInfos", this._ambientTexture.coordinatesIndex, this._ambientTexture.level, this._ambientTextureStrength, this._ambientTextureImpactOnAnalyticalLights), Ye.BindTextureMatrix(this._ambientTexture, P, "ambient")), this._opacityTexture && Dt.OpacityTextureEnabled && (P.updateFloat2("vOpacityInfos", this._opacityTexture.coordinatesIndex, this._opacityTexture.level), Ye.BindTextureMatrix(this._opacityTexture, P, "opacity")), l && Dt.ReflectionTextureEnabled) { if (P.updateMatrix("reflectionMatrix", l.getReflectionTextureMatrix()), P.updateFloat2("vReflectionInfos", l.level, 0), l.boundingBoxSize) { const p = l; P.updateVector3("vReflectionPosition", p.boundingBoxPosition), P.updateVector3("vReflectionSize", p.boundingBoxSize); } if (this.realTimeFiltering) { const p = l.getSize().width; P.updateFloat2("vReflectionFilteringInfo", p, Xt.Log2(p)); } if (!o.USEIRRADIANCEMAP) { const p = l.sphericalPolynomial; if (o.USESPHERICALFROMREFLECTIONMAP && p) if (o.SPHERICAL_HARMONICS) { const c = p.preScaledHarmonics; P.updateVector3("vSphericalL00", c.l00), P.updateVector3("vSphericalL1_1", c.l1_1), P.updateVector3("vSphericalL10", c.l10), P.updateVector3("vSphericalL11", c.l11), P.updateVector3("vSphericalL2_2", c.l2_2), P.updateVector3("vSphericalL2_1", c.l2_1), P.updateVector3("vSphericalL20", c.l20), P.updateVector3("vSphericalL21", c.l21), P.updateVector3("vSphericalL22", c.l22); } else P.updateFloat3("vSphericalX", p.x.x, p.x.y, p.x.z), P.updateFloat3("vSphericalY", p.y.x, p.y.y, p.y.z), P.updateFloat3("vSphericalZ", p.z.x, p.z.y, p.z.z), P.updateFloat3("vSphericalXX_ZZ", p.xx.x - p.zz.x, p.xx.y - p.zz.y, p.xx.z - p.zz.z), P.updateFloat3("vSphericalYY_ZZ", p.yy.x - p.zz.x, p.yy.y - p.zz.y, p.yy.z - p.zz.z), P.updateFloat3("vSphericalZZ", p.zz.x, p.zz.y, p.zz.z), P.updateFloat3("vSphericalXY", p.xy.x, p.xy.y, p.xy.z), P.updateFloat3("vSphericalYZ", p.yz.x, p.yz.y, p.yz.z), P.updateFloat3("vSphericalZX", p.zx.x, p.zx.y, p.zx.z); } P.updateFloat3("vReflectionMicrosurfaceInfos", l.getSize().width, l.lodGenerationScale, l.lodGenerationOffset); } this._emissiveTexture && Dt.EmissiveTextureEnabled && (P.updateFloat2("vEmissiveInfos", this._emissiveTexture.coordinatesIndex, this._emissiveTexture.level), Ye.BindTextureMatrix(this._emissiveTexture, P, "emissive")), this._lightmapTexture && Dt.LightmapTextureEnabled && (P.updateFloat2("vLightmapInfos", this._lightmapTexture.coordinatesIndex, this._lightmapTexture.level), Ye.BindTextureMatrix(this._lightmapTexture, P, "lightmap")), Dt.SpecularTextureEnabled && (this._metallicTexture ? (P.updateFloat3("vReflectivityInfos", this._metallicTexture.coordinatesIndex, this._metallicTexture.level, this._ambientTextureStrength), Ye.BindTextureMatrix(this._metallicTexture, P, "reflectivity")) : this._reflectivityTexture && (P.updateFloat3("vReflectivityInfos", this._reflectivityTexture.coordinatesIndex, this._reflectivityTexture.level, 1), Ye.BindTextureMatrix(this._reflectivityTexture, P, "reflectivity")), this._metallicReflectanceTexture && (P.updateFloat2("vMetallicReflectanceInfos", this._metallicReflectanceTexture.coordinatesIndex, this._metallicReflectanceTexture.level), Ye.BindTextureMatrix(this._metallicReflectanceTexture, P, "metallicReflectance")), this._reflectanceTexture && o.REFLECTANCE && (P.updateFloat2("vReflectanceInfos", this._reflectanceTexture.coordinatesIndex, this._reflectanceTexture.level), Ye.BindTextureMatrix(this._reflectanceTexture, P, "reflectance")), this._microSurfaceTexture && (P.updateFloat2("vMicroSurfaceSamplerInfos", this._microSurfaceTexture.coordinatesIndex, this._microSurfaceTexture.level), Ye.BindTextureMatrix(this._microSurfaceTexture, P, "microSurfaceSampler"))), this._bumpTexture && v.getCaps().standardDerivatives && Dt.BumpTextureEnabled && !this._disableBumpMap && (P.updateFloat3("vBumpInfos", this._bumpTexture.coordinatesIndex, this._bumpTexture.level, this._parallaxScaleBias), Ye.BindTextureMatrix(this._bumpTexture, P, "bump"), f._mirroredCameraPosition ? P.updateFloat2("vTangentSpaceParams", this._invertNormalMapX ? 1 : -1, this._invertNormalMapY ? 1 : -1) : P.updateFloat2("vTangentSpaceParams", this._invertNormalMapX ? -1 : 1, this._invertNormalMapY ? -1 : 1)); } if (this.pointsCloud && P.updateFloat("pointSize", this.pointSize), o.METALLICWORKFLOW) { Hs.Color3[0].r = this._metallic === void 0 || this._metallic === null ? 1 : this._metallic, Hs.Color3[0].g = this._roughness === void 0 || this._roughness === null ? 1 : this._roughness, P.updateColor4("vReflectivityColor", Hs.Color3[0], 1); const p = (i = (n = this.subSurface) === null || n === void 0 ? void 0 : n._indexOfRefraction) !== null && i !== void 0 ? i : 1.5, c = 1, H = Math.pow((p - c) / (p + c), 2); this._metallicReflectanceColor.scaleToRef(H * this._metallicF0Factor, Hs.Color3[0]); const T = this._metallicF0Factor; P.updateColor4("vMetallicReflectanceFactors", Hs.Color3[0], T); } else P.updateColor4("vReflectivityColor", this._reflectivityColor, this._microSurface); P.updateColor3("vEmissiveColor", Dt.EmissiveTextureEnabled ? this._emissiveColor : Ne.BlackReadOnly), P.updateColor3("vReflectionColor", this._reflectionColor), !o.SS_REFRACTION && (!((s = this.subSurface) === null || s === void 0) && s._linkRefractionWithTransparency) ? P.updateColor4("vAlbedoColor", this._albedoColor, 1) : P.updateColor4("vAlbedoColor", this._albedoColor, this.alpha), this._lightingInfos.x = this._directIntensity, this._lightingInfos.y = this._emissiveIntensity, this._lightingInfos.z = this._environmentIntensity * f.environmentIntensity, this._lightingInfos.w = this._specularIntensity, P.updateVector4("vLightingIntensity", this._lightingInfos), f.ambientColor.multiplyToRef(this._ambientColor, this._globalAmbientColor), P.updateColor3("vAmbientColor", this._globalAmbientColor), P.updateFloat2("vDebugMode", this.debugLimit, this.debugFactor); } f.texturesEnabled && (this._albedoTexture && Dt.DiffuseTextureEnabled && P.setTexture("albedoSampler", this._albedoTexture), this._ambientTexture && Dt.AmbientTextureEnabled && P.setTexture("ambientSampler", this._ambientTexture), this._opacityTexture && Dt.OpacityTextureEnabled && P.setTexture("opacitySampler", this._opacityTexture), l && Dt.ReflectionTextureEnabled && (o.LODBASEDMICROSFURACE ? P.setTexture("reflectionSampler", l) : (P.setTexture("reflectionSampler", l._lodTextureMid || l), P.setTexture("reflectionSamplerLow", l._lodTextureLow || l), P.setTexture("reflectionSamplerHigh", l._lodTextureHigh || l)), o.USEIRRADIANCEMAP && P.setTexture("irradianceSampler", l.irradianceTexture)), o.ENVIRONMENTBRDF && P.setTexture("environmentBrdfSampler", this._environmentBRDFTexture), this._emissiveTexture && Dt.EmissiveTextureEnabled && P.setTexture("emissiveSampler", this._emissiveTexture), this._lightmapTexture && Dt.LightmapTextureEnabled && P.setTexture("lightmapSampler", this._lightmapTexture), Dt.SpecularTextureEnabled && (this._metallicTexture ? P.setTexture("reflectivitySampler", this._metallicTexture) : this._reflectivityTexture && P.setTexture("reflectivitySampler", this._reflectivityTexture), this._metallicReflectanceTexture && P.setTexture("metallicReflectanceSampler", this._metallicReflectanceTexture), this._reflectanceTexture && o.REFLECTANCE && P.setTexture("reflectanceSampler", this._reflectanceTexture), this._microSurfaceTexture && P.setTexture("microSurfaceSampler", this._microSurfaceTexture)), this._bumpTexture && v.getCaps().standardDerivatives && Dt.BumpTextureEnabled && !this._disableBumpMap && P.setTexture("bumpSampler", this._bumpTexture)), this.getScene().useOrderIndependentTransparency && this.needAlphaBlendingForMesh(t) && this.getScene().depthPeelingRenderer.bind(d), this._eventInfo.subMesh = r, this._callbackPluginEventBindForSubMesh(this._eventInfo), Df(this._activeEffect, this, f), this.bindEyePosition(d); } else f.getEngine()._features.needToAlwaysBindUniformBuffers && (this._needToBindSceneUbo = !0); (u || !this.isFrozen) && (f.lightsEnabled && !this._disableLighting && Ye.BindLights(f, t, this._activeEffect, o, this._maxSimultaneousLights), (f.fogEnabled && t.applyFog && f.fogMode !== sr.FOGMODE_NONE || l || this.subSurface.refractionTexture || t.receiveShadows || o.PREPASS) && this.bindView(d), Ye.BindFogParameters(f, t, this._activeEffect, !0), o.NUM_MORPH_INFLUENCERS && Ye.BindMorphTargetParameters(t, this._activeEffect), o.BAKED_VERTEX_ANIMATION_TEXTURE && ((a = t.bakedVertexAnimationManager) === null || a === void 0 || a.bind(d, o.INSTANCES)), this._imageProcessingConfiguration.bind(this._activeEffect), Ye.BindLogDepth(o, this._activeEffect, f)), this._afterBind(t, this._activeEffect), P.update(); } /** * Returns the animatable textures. * If material have animatable metallic texture, then reflectivity texture will not be returned, even if it has animations. * @returns - Array of animatable textures. */ getAnimatables() { const e = super.getAnimatables(); return this._albedoTexture && this._albedoTexture.animations && this._albedoTexture.animations.length > 0 && e.push(this._albedoTexture), this._ambientTexture && this._ambientTexture.animations && this._ambientTexture.animations.length > 0 && e.push(this._ambientTexture), this._opacityTexture && this._opacityTexture.animations && this._opacityTexture.animations.length > 0 && e.push(this._opacityTexture), this._reflectionTexture && this._reflectionTexture.animations && this._reflectionTexture.animations.length > 0 && e.push(this._reflectionTexture), this._emissiveTexture && this._emissiveTexture.animations && this._emissiveTexture.animations.length > 0 && e.push(this._emissiveTexture), this._metallicTexture && this._metallicTexture.animations && this._metallicTexture.animations.length > 0 ? e.push(this._metallicTexture) : this._reflectivityTexture && this._reflectivityTexture.animations && this._reflectivityTexture.animations.length > 0 && e.push(this._reflectivityTexture), this._bumpTexture && this._bumpTexture.animations && this._bumpTexture.animations.length > 0 && e.push(this._bumpTexture), this._lightmapTexture && this._lightmapTexture.animations && this._lightmapTexture.animations.length > 0 && e.push(this._lightmapTexture), this._metallicReflectanceTexture && this._metallicReflectanceTexture.animations && this._metallicReflectanceTexture.animations.length > 0 && e.push(this._metallicReflectanceTexture), this._reflectanceTexture && this._reflectanceTexture.animations && this._reflectanceTexture.animations.length > 0 && e.push(this._reflectanceTexture), this._microSurfaceTexture && this._microSurfaceTexture.animations && this._microSurfaceTexture.animations.length > 0 && e.push(this._microSurfaceTexture), e; } /** * Returns the texture used for reflections. * @returns - Reflection texture if present. Otherwise, returns the environment texture. */ _getReflectionTexture() { return this._reflectionTexture ? this._reflectionTexture : this.getScene().environmentTexture; } /** * Returns an array of the actively used textures. * @returns - Array of BaseTextures */ getActiveTextures() { const e = super.getActiveTextures(); return this._albedoTexture && e.push(this._albedoTexture), this._ambientTexture && e.push(this._ambientTexture), this._opacityTexture && e.push(this._opacityTexture), this._reflectionTexture && e.push(this._reflectionTexture), this._emissiveTexture && e.push(this._emissiveTexture), this._reflectivityTexture && e.push(this._reflectivityTexture), this._metallicTexture && e.push(this._metallicTexture), this._metallicReflectanceTexture && e.push(this._metallicReflectanceTexture), this._reflectanceTexture && e.push(this._reflectanceTexture), this._microSurfaceTexture && e.push(this._microSurfaceTexture), this._bumpTexture && e.push(this._bumpTexture), this._lightmapTexture && e.push(this._lightmapTexture), e; } /** * Checks to see if a texture is used in the material. * @param texture - Base texture to use. * @returns - Boolean specifying if a texture is used in the material. */ hasTexture(e) { return !!(super.hasTexture(e) || this._albedoTexture === e || this._ambientTexture === e || this._opacityTexture === e || this._reflectionTexture === e || this._emissiveTexture === e || this._reflectivityTexture === e || this._metallicTexture === e || this._metallicReflectanceTexture === e || this._reflectanceTexture === e || this._microSurfaceTexture === e || this._bumpTexture === e || this._lightmapTexture === e); } /** * Sets the required values to the prepass renderer. * It can't be sets when subsurface scattering of this material is disabled. * When scene have ability to enable subsurface prepass effect, it will enable. */ setPrePassRenderer() { var e; if (!(!((e = this.subSurface) === null || e === void 0) && e.isScatteringEnabled)) return !1; const t = this.getScene().enableSubSurfaceForPrePass(); return t && (t.enabled = !0), !0; } /** * Disposes the resources of the material. * @param forceDisposeEffect - Forces the disposal of effects. * @param forceDisposeTextures - Forces the disposal of all textures. */ dispose(e, t) { var r, n, i, s, a, f, o, d, v, u, l, P; t && (this._environmentBRDFTexture && this.getScene().environmentBRDFTexture !== this._environmentBRDFTexture && this._environmentBRDFTexture.dispose(), (r = this._albedoTexture) === null || r === void 0 || r.dispose(), (n = this._ambientTexture) === null || n === void 0 || n.dispose(), (i = this._opacityTexture) === null || i === void 0 || i.dispose(), (s = this._reflectionTexture) === null || s === void 0 || s.dispose(), (a = this._emissiveTexture) === null || a === void 0 || a.dispose(), (f = this._metallicTexture) === null || f === void 0 || f.dispose(), (o = this._reflectivityTexture) === null || o === void 0 || o.dispose(), (d = this._bumpTexture) === null || d === void 0 || d.dispose(), (v = this._lightmapTexture) === null || v === void 0 || v.dispose(), (u = this._metallicReflectanceTexture) === null || u === void 0 || u.dispose(), (l = this._reflectanceTexture) === null || l === void 0 || l.dispose(), (P = this._microSurfaceTexture) === null || P === void 0 || P.dispose()), this._renderTargets.dispose(), this._imageProcessingConfiguration && this._imageProcessingObserver && this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver), super.dispose(e, t); } } fs.PBRMATERIAL_OPAQUE = gt.MATERIAL_OPAQUE; fs.PBRMATERIAL_ALPHATEST = gt.MATERIAL_ALPHATEST; fs.PBRMATERIAL_ALPHABLEND = gt.MATERIAL_ALPHABLEND; fs.PBRMATERIAL_ALPHATESTANDBLEND = gt.MATERIAL_ALPHATESTANDBLEND; fs.DEFAULT_AO_ON_ANALYTICAL_LIGHTS = 0; fs.LIGHTFALLOFF_PHYSICAL = 0; fs.LIGHTFALLOFF_GLTF = 1; fs.LIGHTFALLOFF_STANDARD = 2; C([ CN() ], fs.prototype, "_imageProcessingConfiguration", void 0); C([ At("_markAllSubMeshesAsMiscDirty") ], fs.prototype, "debugMode", void 0); class mr extends fs { /** * Stores the refracted light information in a texture. */ get refractionTexture() { return this.subSurface.refractionTexture; } set refractionTexture(e) { this.subSurface.refractionTexture = e, e ? this.subSurface.isRefractionEnabled = !0 : this.subSurface.linkRefractionWithTransparency || (this.subSurface.isRefractionEnabled = !1); } /** * Index of refraction of the material base layer. * https://en.wikipedia.org/wiki/List_of_refractive_indices * * This does not only impact refraction but also the Base F0 of Dielectric Materials. * * From dielectric fresnel rules: F0 = square((iorT - iorI) / (iorT + iorI)) */ get indexOfRefraction() { return this.subSurface.indexOfRefraction; } set indexOfRefraction(e) { this.subSurface.indexOfRefraction = e; } /** * Controls if refraction needs to be inverted on Y. This could be useful for procedural texture. */ get invertRefractionY() { return this.subSurface.invertRefractionY; } set invertRefractionY(e) { this.subSurface.invertRefractionY = e; } /** * This parameters will make the material used its opacity to control how much it is refracting against not. * Materials half opaque for instance using refraction could benefit from this control. */ get linkRefractionWithTransparency() { return this.subSurface.linkRefractionWithTransparency; } set linkRefractionWithTransparency(e) { this.subSurface.linkRefractionWithTransparency = e, e && (this.subSurface.isRefractionEnabled = !0); } /** * BJS is using an hardcoded light falloff based on a manually sets up range. * In PBR, one way to represents the falloff is to use the inverse squared root algorithm. * This parameter can help you switch back to the BJS mode in order to create scenes using both materials. */ get usePhysicalLightFalloff() { return this._lightFalloff === fs.LIGHTFALLOFF_PHYSICAL; } /** * BJS is using an hardcoded light falloff based on a manually sets up range. * In PBR, one way to represents the falloff is to use the inverse squared root algorithm. * This parameter can help you switch back to the BJS mode in order to create scenes using both materials. */ set usePhysicalLightFalloff(e) { e !== this.usePhysicalLightFalloff && (this._markAllSubMeshesAsTexturesDirty(), e ? this._lightFalloff = fs.LIGHTFALLOFF_PHYSICAL : this._lightFalloff = fs.LIGHTFALLOFF_STANDARD); } /** * In order to support the falloff compatibility with gltf, a special mode has been added * to reproduce the gltf light falloff. */ get useGLTFLightFalloff() { return this._lightFalloff === fs.LIGHTFALLOFF_GLTF; } /** * In order to support the falloff compatibility with gltf, a special mode has been added * to reproduce the gltf light falloff. */ set useGLTFLightFalloff(e) { e !== this.useGLTFLightFalloff && (this._markAllSubMeshesAsTexturesDirty(), e ? this._lightFalloff = fs.LIGHTFALLOFF_GLTF : this._lightFalloff = fs.LIGHTFALLOFF_STANDARD); } /** * Gets the image processing configuration used either in this material. */ get imageProcessingConfiguration() { return this._imageProcessingConfiguration; } /** * Sets the Default image processing configuration used either in the this material. * * If sets to null, the scene one is in use. */ set imageProcessingConfiguration(e) { this._attachImageProcessingConfiguration(e), this._markAllSubMeshesAsTexturesDirty(); } /** * Gets whether the color curves effect is enabled. */ get cameraColorCurvesEnabled() { return this.imageProcessingConfiguration.colorCurvesEnabled; } /** * Sets whether the color curves effect is enabled. */ set cameraColorCurvesEnabled(e) { this.imageProcessingConfiguration.colorCurvesEnabled = e; } /** * Gets whether the color grading effect is enabled. */ get cameraColorGradingEnabled() { return this.imageProcessingConfiguration.colorGradingEnabled; } /** * Gets whether the color grading effect is enabled. */ set cameraColorGradingEnabled(e) { this.imageProcessingConfiguration.colorGradingEnabled = e; } /** * Gets whether tonemapping is enabled or not. */ get cameraToneMappingEnabled() { return this._imageProcessingConfiguration.toneMappingEnabled; } /** * Sets whether tonemapping is enabled or not */ set cameraToneMappingEnabled(e) { this._imageProcessingConfiguration.toneMappingEnabled = e; } /** * The camera exposure used on this material. * This property is here and not in the camera to allow controlling exposure without full screen post process. * This corresponds to a photographic exposure. */ get cameraExposure() { return this._imageProcessingConfiguration.exposure; } /** * The camera exposure used on this material. * This property is here and not in the camera to allow controlling exposure without full screen post process. * This corresponds to a photographic exposure. */ set cameraExposure(e) { this._imageProcessingConfiguration.exposure = e; } /** * Gets The camera contrast used on this material. */ get cameraContrast() { return this._imageProcessingConfiguration.contrast; } /** * Sets The camera contrast used on this material. */ set cameraContrast(e) { this._imageProcessingConfiguration.contrast = e; } /** * Gets the Color Grading 2D Lookup Texture. */ get cameraColorGradingTexture() { return this._imageProcessingConfiguration.colorGradingTexture; } /** * Sets the Color Grading 2D Lookup Texture. */ set cameraColorGradingTexture(e) { this._imageProcessingConfiguration.colorGradingTexture = e; } /** * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT). * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects. * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image; * corresponding to low luminance, medium luminance, and high luminance areas respectively. */ get cameraColorCurves() { return this._imageProcessingConfiguration.colorCurves; } /** * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT). * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects. * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image; * corresponding to low luminance, medium luminance, and high luminance areas respectively. */ set cameraColorCurves(e) { this._imageProcessingConfiguration.colorCurves = e; } /** * Instantiates a new PBRMaterial instance. * * @param name The material name * @param scene The scene the material will be use in. */ constructor(e, t) { super(e, t), this.directIntensity = 1, this.emissiveIntensity = 1, this.environmentIntensity = 1, this.specularIntensity = 1, this.disableBumpMap = !1, this.ambientTextureStrength = 1, this.ambientTextureImpactOnAnalyticalLights = mr.DEFAULT_AO_ON_ANALYTICAL_LIGHTS, this.metallicF0Factor = 1, this.metallicReflectanceColor = Ne.White(), this.useOnlyMetallicFromMetallicReflectanceTexture = !1, this.ambientColor = new Ne(0, 0, 0), this.albedoColor = new Ne(1, 1, 1), this.reflectivityColor = new Ne(1, 1, 1), this.reflectionColor = new Ne(1, 1, 1), this.emissiveColor = new Ne(0, 0, 0), this.microSurface = 1, this.useLightmapAsShadowmap = !1, this.useAlphaFromAlbedoTexture = !1, this.forceAlphaTest = !1, this.alphaCutOff = 0.4, this.useSpecularOverAlpha = !0, this.useMicroSurfaceFromReflectivityMapAlpha = !1, this.useRoughnessFromMetallicTextureAlpha = !0, this.useRoughnessFromMetallicTextureGreen = !1, this.useMetallnessFromMetallicTextureBlue = !1, this.useAmbientOcclusionFromMetallicTextureRed = !1, this.useAmbientInGrayScale = !1, this.useAutoMicroSurfaceFromReflectivityMap = !1, this.useRadianceOverAlpha = !0, this.useObjectSpaceNormalMap = !1, this.useParallax = !1, this.useParallaxOcclusion = !1, this.parallaxScaleBias = 0.05, this.disableLighting = !1, this.forceIrradianceInFragment = !1, this.maxSimultaneousLights = 4, this.invertNormalMapX = !1, this.invertNormalMapY = !1, this.twoSidedLighting = !1, this.useAlphaFresnel = !1, this.useLinearAlphaFresnel = !1, this.environmentBRDFTexture = null, this.forceNormalForward = !1, this.enableSpecularAntiAliasing = !1, this.useHorizonOcclusion = !0, this.useRadianceOcclusion = !0, this.unlit = !1, this.applyDecalMapAfterDetailMap = !1, this._environmentBRDFTexture = nV(this.getScene()); } /** * Returns the name of this material class. */ getClassName() { return "PBRMaterial"; } /** * Makes a duplicate of the current material. * @param name - name to use for the new material. * @param cloneTexturesOnlyOnce - if a texture is used in more than one channel (e.g diffuse and opacity), only clone it once and reuse it on the other channels. Default false. * @param rootUrl defines the root URL to use to load textures */ clone(e, t = !0, r = "") { const n = jt.Clone(() => new mr(e, this.getScene()), this, { cloneTexturesOnlyOnce: t }); return n.id = e, n.name = e, this.stencil.copyTo(n.stencil), this._clonePlugins(n, r), n; } /** * Serializes this PBR Material. * @returns - An object with the serialized material. */ serialize() { const e = super.serialize(); return e.customType = "BABYLON.PBRMaterial", e; } // Statics /** * Parses a PBR Material from a serialized object. * @param source - Serialized object. * @param scene - BJS scene instance. * @param rootUrl - url for the scene object * @returns - PBRMaterial */ static Parse(e, t, r) { const n = jt.Parse(() => new mr(e.name, t), e, t, r); return e.stencil && n.stencil.parse(e.stencil, t, r), gt._parsePlugins(e, n, t, r), e.clearCoat && n.clearCoat.parse(e.clearCoat, t, r), e.anisotropy && n.anisotropy.parse(e.anisotropy, t, r), e.brdf && n.brdf.parse(e.brdf, t, r), e.sheen && n.sheen.parse(e.sheen, t, r), e.subSurface && n.subSurface.parse(e.subSurface, t, r), e.iridescence && n.iridescence.parse(e.iridescence, t, r), n; } } mr.PBRMATERIAL_OPAQUE = fs.PBRMATERIAL_OPAQUE; mr.PBRMATERIAL_ALPHATEST = fs.PBRMATERIAL_ALPHATEST; mr.PBRMATERIAL_ALPHABLEND = fs.PBRMATERIAL_ALPHABLEND; mr.PBRMATERIAL_ALPHATESTANDBLEND = fs.PBRMATERIAL_ALPHATESTANDBLEND; mr.DEFAULT_AO_ON_ANALYTICAL_LIGHTS = fs.DEFAULT_AO_ON_ANALYTICAL_LIGHTS; C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "directIntensity", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "emissiveIntensity", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "environmentIntensity", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "specularIntensity", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "disableBumpMap", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "albedoTexture", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "ambientTexture", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "ambientTextureStrength", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "ambientTextureImpactOnAnalyticalLights", void 0); C([ en(), At("_markAllSubMeshesAsTexturesAndMiscDirty") ], mr.prototype, "opacityTexture", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "reflectionTexture", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "emissiveTexture", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "reflectivityTexture", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "metallicTexture", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "metallic", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "roughness", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "metallicF0Factor", void 0); C([ Oi(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "metallicReflectanceColor", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useOnlyMetallicFromMetallicReflectanceTexture", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "metallicReflectanceTexture", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "reflectanceTexture", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "microSurfaceTexture", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "bumpTexture", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty", null) ], mr.prototype, "lightmapTexture", void 0); C([ Oi("ambient"), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "ambientColor", void 0); C([ Oi("albedo"), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "albedoColor", void 0); C([ Oi("reflectivity"), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "reflectivityColor", void 0); C([ Oi("reflection"), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "reflectionColor", void 0); C([ Oi("emissive"), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "emissiveColor", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "microSurface", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useLightmapAsShadowmap", void 0); C([ M(), At("_markAllSubMeshesAsTexturesAndMiscDirty") ], mr.prototype, "useAlphaFromAlbedoTexture", void 0); C([ M(), At("_markAllSubMeshesAsTexturesAndMiscDirty") ], mr.prototype, "forceAlphaTest", void 0); C([ M(), At("_markAllSubMeshesAsTexturesAndMiscDirty") ], mr.prototype, "alphaCutOff", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useSpecularOverAlpha", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useMicroSurfaceFromReflectivityMapAlpha", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useRoughnessFromMetallicTextureAlpha", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useRoughnessFromMetallicTextureGreen", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useMetallnessFromMetallicTextureBlue", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useAmbientOcclusionFromMetallicTextureRed", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useAmbientInGrayScale", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useAutoMicroSurfaceFromReflectivityMap", void 0); C([ M() ], mr.prototype, "usePhysicalLightFalloff", null); C([ M() ], mr.prototype, "useGLTFLightFalloff", null); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useRadianceOverAlpha", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useObjectSpaceNormalMap", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useParallax", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useParallaxOcclusion", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "parallaxScaleBias", void 0); C([ M(), At("_markAllSubMeshesAsLightsDirty") ], mr.prototype, "disableLighting", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "forceIrradianceInFragment", void 0); C([ M(), At("_markAllSubMeshesAsLightsDirty") ], mr.prototype, "maxSimultaneousLights", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "invertNormalMapX", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "invertNormalMapY", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "twoSidedLighting", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useAlphaFresnel", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useLinearAlphaFresnel", void 0); C([ At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "environmentBRDFTexture", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "forceNormalForward", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "enableSpecularAntiAliasing", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useHorizonOcclusion", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], mr.prototype, "useRadianceOcclusion", void 0); C([ M(), At("_markAllSubMeshesAsMiscDirty") ], mr.prototype, "unlit", void 0); C([ M(), At("_markAllSubMeshesAsMiscDirty") ], mr.prototype, "applyDecalMapAfterDetailMap", void 0); Ue("BABYLON.PBRMaterial", mr); const Zce = 542327876, wG = 131072, mG = 512, BG = 4, WG = 64, SG = 131072; function fy(A) { return A.charCodeAt(0) + (A.charCodeAt(1) << 8) + (A.charCodeAt(2) << 16) + (A.charCodeAt(3) << 24); } function _ce(A) { return String.fromCharCode(A & 255, A >> 8 & 255, A >> 16 & 255, A >> 24 & 255); } const UG = fy("DXT1"), IG = fy("DXT3"), RG = fy("DXT5"), wE = fy("DX10"), VG = 113, CG = 116, OG = 2, yG = 10, $ce = 88, mE = 31, e3e = 0, t3e = 1, kG = 2, EG = 3, BE = 4, FG = 7, WE = 20, NG = 21, r3e = 22, n3e = 23, i3e = 24, s3e = 25, a3e = 26, o3e = 28, f3e = 32; class Aa { /** * Gets DDS information from an array buffer * @param data defines the array buffer view to read data from * @returns the DDS information */ static GetDDSInfo(e) { const t = new Int32Array(e.buffer, e.byteOffset, mE), r = new Int32Array(e.buffer, e.byteOffset, mE + 4); let n = 1; t[kG] & wG && (n = Math.max(1, t[FG])); const i = t[NG], s = i === wE ? r[f3e] : 0; let a = 0; switch (i) { case VG: a = 2; break; case CG: a = 1; break; case wE: if (s === yG) { a = 2; break; } if (s === OG) { a = 1; break; } } return { width: t[BE], height: t[EG], mipmapCount: n, isFourCC: (t[WE] & BG) === BG, isRGB: (t[WE] & WG) === WG, isLuminance: (t[WE] & SG) === SG, isCube: (t[o3e] & mG) === mG, isCompressed: i === UG || i === IG || i === RG, dxgiFormat: s, textureType: a }; } static _GetHalfFloatAsFloatRGBAArrayBuffer(e, t, r, n, i, s) { const a = new Float32Array(n), f = new Uint16Array(i, r); let o = 0; for (let d = 0; d < t; d++) for (let v = 0; v < e; v++) { const u = (v + d * e) * 4; a[o] = QH(f[u]), a[o + 1] = QH(f[u + 1]), a[o + 2] = QH(f[u + 2]), Aa.StoreLODInAlphaChannel ? a[o + 3] = s : a[o + 3] = QH(f[u + 3]), o += 4; } return a; } static _GetHalfFloatRGBAArrayBuffer(e, t, r, n, i, s) { if (Aa.StoreLODInAlphaChannel) { const a = new Uint16Array(n), f = new Uint16Array(i, r); let o = 0; for (let d = 0; d < t; d++) for (let v = 0; v < e; v++) { const u = (v + d * e) * 4; a[o] = f[u], a[o + 1] = f[u + 1], a[o + 2] = f[u + 2], a[o + 3] = LH(s), o += 4; } return a; } return new Uint16Array(i, r, n); } static _GetFloatRGBAArrayBuffer(e, t, r, n, i, s) { if (Aa.StoreLODInAlphaChannel) { const a = new Float32Array(n), f = new Float32Array(i, r); let o = 0; for (let d = 0; d < t; d++) for (let v = 0; v < e; v++) { const u = (v + d * e) * 4; a[o] = f[u], a[o + 1] = f[u + 1], a[o + 2] = f[u + 2], a[o + 3] = s, o += 4; } return a; } return new Float32Array(i, r, n); } static _GetFloatAsHalfFloatRGBAArrayBuffer(e, t, r, n, i, s) { const a = new Uint16Array(n), f = new Float32Array(i, r); let o = 0; for (let d = 0; d < t; d++) for (let v = 0; v < e; v++) a[o] = LH(f[o]), a[o + 1] = LH(f[o + 1]), a[o + 2] = LH(f[o + 2]), Aa.StoreLODInAlphaChannel ? a[o + 3] = LH(s) : a[o + 3] = LH(f[o + 3]), o += 4; return a; } static _GetFloatAsUIntRGBAArrayBuffer(e, t, r, n, i, s) { const a = new Uint8Array(n), f = new Float32Array(i, r); let o = 0; for (let d = 0; d < t; d++) for (let v = 0; v < e; v++) { const u = (v + d * e) * 4; a[o] = Xt.Clamp(f[u]) * 255, a[o + 1] = Xt.Clamp(f[u + 1]) * 255, a[o + 2] = Xt.Clamp(f[u + 2]) * 255, Aa.StoreLODInAlphaChannel ? a[o + 3] = s : a[o + 3] = Xt.Clamp(f[u + 3]) * 255, o += 4; } return a; } static _GetHalfFloatAsUIntRGBAArrayBuffer(e, t, r, n, i, s) { const a = new Uint8Array(n), f = new Uint16Array(i, r); let o = 0; for (let d = 0; d < t; d++) for (let v = 0; v < e; v++) { const u = (v + d * e) * 4; a[o] = Xt.Clamp(QH(f[u])) * 255, a[o + 1] = Xt.Clamp(QH(f[u + 1])) * 255, a[o + 2] = Xt.Clamp(QH(f[u + 2])) * 255, Aa.StoreLODInAlphaChannel ? a[o + 3] = s : a[o + 3] = Xt.Clamp(QH(f[u + 3])) * 255, o += 4; } return a; } static _GetRGBAArrayBuffer(e, t, r, n, i, s, a, f, o) { const d = new Uint8Array(n), v = new Uint8Array(i, r); let u = 0; for (let l = 0; l < t; l++) for (let P = 0; P < e; P++) { const p = (P + l * e) * 4; d[u] = v[p + s], d[u + 1] = v[p + a], d[u + 2] = v[p + f], d[u + 3] = v[p + o], u += 4; } return d; } static _ExtractLongWordOrder(e) { return e === 0 || e === 255 || e === -16777216 ? 0 : 1 + Aa._ExtractLongWordOrder(e >> 8); } static _GetRGBArrayBuffer(e, t, r, n, i, s, a, f) { const o = new Uint8Array(n), d = new Uint8Array(i, r); let v = 0; for (let u = 0; u < t; u++) for (let l = 0; l < e; l++) { const P = (l + u * e) * 3; o[v] = d[P + s], o[v + 1] = d[P + a], o[v + 2] = d[P + f], v += 3; } return o; } static _GetLuminanceArrayBuffer(e, t, r, n, i) { const s = new Uint8Array(n), a = new Uint8Array(i, r); let f = 0; for (let o = 0; o < t; o++) for (let d = 0; d < e; d++) { const v = d + o * e; s[f] = a[v], f++; } return s; } /** * Uploads DDS Levels to a Babylon Texture * @internal */ static UploadDDSLevels(e, t, r, n, i, s, a = -1, f, o = !0) { let d = null; n.sphericalPolynomial && (d = []); const v = !!e.getCaps().s3tc; t.generateMipMaps = i; const u = new Int32Array(r.buffer, r.byteOffset, mE); let l, P, p, c = 0, H, T, q, b, j = 0, w = 1; if (u[e3e] !== Zce) { Se.Error("Invalid magic number in DDS header"); return; } if (!n.isFourCC && !n.isRGB && !n.isLuminance) { Se.Error("Unsupported format, must contain a FourCC, RGB or LUMINANCE code"); return; } if (n.isCompressed && !v) { Se.Error("Compressed textures are not supported on this platform."); return; } let m = u[r3e]; H = u[t3e] + 4; let I = !1; if (n.isFourCC) switch (l = u[NG], l) { case UG: w = 8, j = 33777; break; case IG: w = 16, j = 33778; break; case RG: w = 16, j = 33779; break; case VG: I = !0, m = 64; break; case CG: I = !0, m = 128; break; case wE: { H += 5 * 4; let ee = !1; switch (n.dxgiFormat) { case yG: I = !0, m = 64, ee = !0; break; case OG: I = !0, m = 128, ee = !0; break; case $ce: n.isRGB = !0, n.isFourCC = !1, m = 32, ee = !0; break; } if (ee) break; } default: console.error("Unsupported FourCC code:", _ce(l)); return; } const N = Aa._ExtractLongWordOrder(u[n3e]), k = Aa._ExtractLongWordOrder(u[i3e]), R = Aa._ExtractLongWordOrder(u[s3e]), y = Aa._ExtractLongWordOrder(u[a3e]); I && (j = e._getRGBABufferInternalSizedFormat(n.textureType)), q = 1, u[kG] & wG && i !== !1 && (q = Math.max(1, u[FG])); const O = f || 0, Y = e.getCaps(); for (let ee = O; ee < s; ee++) { for (P = u[BE], p = u[EG], b = 0; b < q; ++b) { if (a === -1 || a === b) { const Z = a === -1 ? b : 0; if (!n.isCompressed && n.isFourCC) { t.format = 5, c = P * p * 4; let te = null; if (e._badOS || e._badDesktopOS || !Y.textureHalfFloat && !Y.textureFloat) m === 128 ? (te = Aa._GetFloatAsUIntRGBAArrayBuffer(P, p, r.byteOffset + H, c, r.buffer, Z), d && Z == 0 && d.push(Aa._GetFloatRGBAArrayBuffer(P, p, r.byteOffset + H, c, r.buffer, Z))) : m === 64 && (te = Aa._GetHalfFloatAsUIntRGBAArrayBuffer(P, p, r.byteOffset + H, c, r.buffer, Z), d && Z == 0 && d.push(Aa._GetHalfFloatAsFloatRGBAArrayBuffer(P, p, r.byteOffset + H, c, r.buffer, Z))), t.type = 0; else { const fe = Y.textureFloat && (o && Y.textureFloatLinearFiltering || !o), _ = Y.textureHalfFloat && (o && Y.textureHalfFloatLinearFiltering || !o), G = (m === 128 || m === 64 && !_) && fe ? 1 : (m === 64 || m === 128 && !fe) && _ ? 2 : 0; let L, $ = null; switch (m) { case 128: { switch (G) { case 1: L = Aa._GetFloatRGBAArrayBuffer, $ = null; break; case 2: L = Aa._GetFloatAsHalfFloatRGBAArrayBuffer, $ = Aa._GetFloatRGBAArrayBuffer; break; case 0: L = Aa._GetFloatAsUIntRGBAArrayBuffer, $ = Aa._GetFloatRGBAArrayBuffer; break; } break; } default: { switch (G) { case 1: L = Aa._GetHalfFloatAsFloatRGBAArrayBuffer, $ = null; break; case 2: L = Aa._GetHalfFloatRGBAArrayBuffer, $ = Aa._GetHalfFloatAsFloatRGBAArrayBuffer; break; case 0: L = Aa._GetHalfFloatAsUIntRGBAArrayBuffer, $ = Aa._GetHalfFloatAsFloatRGBAArrayBuffer; break; } break; } } t.type = G, te = L(P, p, r.byteOffset + H, c, r.buffer, Z), d && Z == 0 && d.push($ ? $(P, p, r.byteOffset + H, c, r.buffer, Z) : te); } te && e._uploadDataToTextureDirectly(t, te, ee, Z); } else if (n.isRGB) t.type = 0, m === 24 ? (t.format = 4, c = P * p * 3, T = Aa._GetRGBArrayBuffer(P, p, r.byteOffset + H, c, r.buffer, N, k, R), e._uploadDataToTextureDirectly(t, T, ee, Z)) : (t.format = 5, c = P * p * 4, T = Aa._GetRGBAArrayBuffer(P, p, r.byteOffset + H, c, r.buffer, N, k, R, y), e._uploadDataToTextureDirectly(t, T, ee, Z)); else if (n.isLuminance) { const te = e._getUnpackAlignement(), fe = P; c = Math.floor((P + te - 1) / te) * te * (p - 1) + fe, T = Aa._GetLuminanceArrayBuffer(P, p, r.byteOffset + H, c, r.buffer), t.format = 1, t.type = 0, e._uploadDataToTextureDirectly(t, T, ee, Z); } else c = Math.max(4, P) / 4 * Math.max(4, p) / 4 * w, T = new Uint8Array(r.buffer, r.byteOffset + H, c), t.type = 0, e._uploadCompressedDataToTextureDirectly(t, j, P, p, T, ee, Z); } H += m ? P * p * (m / 8) : c, P *= 0.5, p *= 0.5, P = Math.max(1, P), p = Math.max(1, p); } if (f !== void 0) break; } d && d.length > 0 ? n.sphericalPolynomial = Sm.ConvertCubeMapToSphericalPolynomial({ size: u[BE], right: d[0], left: d[1], up: d[2], down: d[3], front: d[4], back: d[5], format: 5, type: 1, gammaSpace: !1 }) : n.sphericalPolynomial = void 0; } } Aa.StoreLODInAlphaChannel = !1; hr.prototype.createPrefilteredCubeTexture = function(A, e, t, r, n = null, i = null, s, a = null, f = !0) { const o = (d) => { if (!d) { n && n(null); return; } const v = d.texture; if (f ? d.info.sphericalPolynomial && (v._sphericalPolynomial = d.info.sphericalPolynomial) : v._sphericalPolynomial = new i4(), v._source = ri.CubePrefiltered, this.getCaps().textureLOD) { n && n(v); return; } const u = 3, l = this._gl, P = d.width; if (!P) return; const p = []; for (let c = 0; c < u; c++) { const T = 1 - c / (u - 1), q = r, b = Xt.Log2(P) * t + r, j = q + (b - q) * T, w = Math.round(Math.min(Math.max(j, 0), b)), m = new As(this, ri.Temp); if (m.type = v.type, m.format = v.format, m.width = Math.pow(2, Math.max(Xt.Log2(P) - w, 0)), m.height = m.width, m.isCube = !0, m._cachedWrapU = 0, m._cachedWrapV = 0, this._bindTextureDirectly(l.TEXTURE_CUBE_MAP, m, !0), m.samplingMode = 2, l.texParameteri(l.TEXTURE_CUBE_MAP, l.TEXTURE_MAG_FILTER, l.LINEAR), l.texParameteri(l.TEXTURE_CUBE_MAP, l.TEXTURE_MIN_FILTER, l.LINEAR), l.texParameteri(l.TEXTURE_CUBE_MAP, l.TEXTURE_WRAP_S, l.CLAMP_TO_EDGE), l.texParameteri(l.TEXTURE_CUBE_MAP, l.TEXTURE_WRAP_T, l.CLAMP_TO_EDGE), d.isDDS) { const N = d.info, k = d.data; this._unpackFlipY(N.isCompressed), Aa.UploadDDSLevels(this, m, k, N, !0, 6, w); } else Se.Warn("DDS is the only prefiltered cube map supported so far."); this._bindTextureDirectly(l.TEXTURE_CUBE_MAP, null); const I = new ls(e); I._isCube = !0, I._texture = m, m.isReady = !0, p.push(I); } v._lodTextureHigh = p[2], v._lodTextureMid = p[1], v._lodTextureLow = p[0], n && n(v); }; return this.createCubeTexture(A, e, null, !1, o, i, s, a, f, t, r); }; class Wte { constructor() { this.supportCascades = !0; } /** * This returns if the loader support the current file information. * @param extension defines the file extension of the file being loaded * @returns true if the loader can load the specified file */ canLoad(e) { return e.endsWith(".dds"); } /** * Uploads the cube texture data to the WebGL texture. It has already been bound. * @param imgs contains the cube maps * @param texture defines the BabylonJS internal texture * @param createPolynomials will be true if polynomials have been requested * @param onLoad defines the callback to trigger once the texture is ready */ loadCubeData(e, t, r, n) { const i = t.getEngine(); let s, a = !1, f = 1e3; if (Array.isArray(e)) for (let o = 0; o < e.length; o++) { const d = e[o]; s = Aa.GetDDSInfo(d), t.width = s.width, t.height = s.height, a = (s.isRGB || s.isLuminance || s.mipmapCount > 1) && t.generateMipMaps, i._unpackFlipY(s.isCompressed), Aa.UploadDDSLevels(i, t, d, s, a, 6, -1, o), !s.isFourCC && s.mipmapCount === 1 ? i.generateMipMapsForCubemap(t) : f = s.mipmapCount - 1; } else { const o = e; s = Aa.GetDDSInfo(o), t.width = s.width, t.height = s.height, r && (s.sphericalPolynomial = new i4()), a = (s.isRGB || s.isLuminance || s.mipmapCount > 1) && t.generateMipMaps, i._unpackFlipY(s.isCompressed), Aa.UploadDDSLevels(i, t, o, s, a, 6), !s.isFourCC && s.mipmapCount === 1 ? i.generateMipMapsForCubemap(t, !1) : f = s.mipmapCount - 1; } i._setCubeMapTextureParams(t, a, f), t.isReady = !0, t.onLoadedObservable.notifyObservers(t), t.onLoadedObservable.clear(), n && n({ isDDS: !0, width: t.width, info: s, data: e, texture: t }); } /** * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback. * @param data contains the texture data * @param texture defines the BabylonJS internal texture * @param callback defines the method to call once ready to upload */ loadData(e, t, r) { const n = Aa.GetDDSInfo(e), i = (n.isRGB || n.isLuminance || n.mipmapCount > 1) && t.generateMipMaps && n.width >> n.mipmapCount - 1 === 1; r(n.width, n.height, i, n.isFourCC, () => { Aa.UploadDDSLevels(t.getEngine(), t, e, n, i, 1); }); } } Ge._TextureLoaders.push(new Wte()); class Ste { constructor() { this.supportCascades = !1; } /** * This returns if the loader support the current file information. * @param extension defines the file extension of the file being loaded * @returns true if the loader can load the specified file */ canLoad(e) { return e.endsWith(".env"); } /** * Uploads the cube texture data to the WebGL texture. It has already been bound. * @param data contains the texture data * @param texture defines the BabylonJS internal texture * @param createPolynomials will be true if polynomials have been requested * @param onLoad defines the callback to trigger once the texture is ready * @param onError defines the callback to trigger in case of error */ loadCubeData(e, t, r, n, i) { if (Array.isArray(e)) return; const s = ry(e); if (s) { t.width = s.width, t.height = s.width; try { iy(t, s), EQ(t, e, s).then(() => { t.isReady = !0, t.onLoadedObservable.notifyObservers(t), t.onLoadedObservable.clear(), n && n(); }, (a) => { i == null || i("Can not upload environment levels", a); }); } catch (a) { i == null || i("Can not upload environment file", a); } } else i && i("Can not parse the environment file", null); } /** * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback. */ loadData() { throw ".env not supported in 2d."; } } Ge._TextureLoaders.push(new Ste()); class yv { /** * Creates a new KhronosTextureContainer * @param data contents of the KTX container file * @param facesExpected should be either 1 or 6, based whether a cube texture or or */ constructor(e, t) { if (this.data = e, this.isInvalid = !1, !yv.IsValid(e)) { this.isInvalid = !0, Se.Error("texture missing KTX identifier"); return; } const r = Uint32Array.BYTES_PER_ELEMENT, n = new DataView(this.data.buffer, this.data.byteOffset + 12, 13 * r), s = n.getUint32(0, !0) === 67305985; if (this.glType = n.getUint32(1 * r, s), this.glTypeSize = n.getUint32(2 * r, s), this.glFormat = n.getUint32(3 * r, s), this.glInternalFormat = n.getUint32(4 * r, s), this.glBaseInternalFormat = n.getUint32(5 * r, s), this.pixelWidth = n.getUint32(6 * r, s), this.pixelHeight = n.getUint32(7 * r, s), this.pixelDepth = n.getUint32(8 * r, s), this.numberOfArrayElements = n.getUint32(9 * r, s), this.numberOfFaces = n.getUint32(10 * r, s), this.numberOfMipmapLevels = n.getUint32(11 * r, s), this.bytesOfKeyValueData = n.getUint32(12 * r, s), this.glType !== 0) { Se.Error("only compressed formats currently supported"), this.isInvalid = !0; return; } else this.numberOfMipmapLevels = Math.max(1, this.numberOfMipmapLevels); if (this.pixelHeight === 0 || this.pixelDepth !== 0) { Se.Error("only 2D textures currently supported"), this.isInvalid = !0; return; } if (this.numberOfArrayElements !== 0) { Se.Error("texture arrays not currently supported"), this.isInvalid = !0; return; } if (this.numberOfFaces !== t) { Se.Error("number of faces expected" + t + ", but found " + this.numberOfFaces), this.isInvalid = !0; return; } this.loadType = yv.COMPRESSED_2D; } /** * Uploads KTX content to a Babylon Texture. * It is assumed that the texture has already been created & is currently bound * @internal */ uploadLevels(e, t) { switch (this.loadType) { case yv.COMPRESSED_2D: this._upload2DCompressedLevels(e, t); break; case yv.TEX_2D: case yv.COMPRESSED_3D: case yv.TEX_3D: } } _upload2DCompressedLevels(e, t) { let r = yv.HEADER_LEN + this.bytesOfKeyValueData, n = this.pixelWidth, i = this.pixelHeight; const s = t ? this.numberOfMipmapLevels : 1; for (let a = 0; a < s; a++) { const f = new Int32Array(this.data.buffer, this.data.byteOffset + r, 1)[0]; r += 4; for (let o = 0; o < this.numberOfFaces; o++) { const d = new Uint8Array(this.data.buffer, this.data.byteOffset + r, f); e.getEngine()._uploadCompressedDataToTextureDirectly(e, e.format, n, i, d, o, a), r += f, r += 3 - (f + 3) % 4; } n = Math.max(1, n * 0.5), i = Math.max(1, i * 0.5); } } /** * Checks if the given data starts with a KTX file identifier. * @param data the data to check * @returns true if the data is a KTX file or false otherwise */ static IsValid(e) { if (e.byteLength >= 12) { const t = new Uint8Array(e.buffer, e.byteOffset, 12); if (t[0] === 171 && t[1] === 75 && t[2] === 84 && t[3] === 88 && t[4] === 32 && t[5] === 49 && t[6] === 49 && t[7] === 187 && t[8] === 13 && t[9] === 10 && t[10] === 26 && t[11] === 10) return !0; } return !1; } } yv.HEADER_LEN = 12 + 13 * 4; yv.COMPRESSED_2D = 0; yv.COMPRESSED_3D = 1; yv.TEX_2D = 2; yv.TEX_3D = 3; class Ute { /** * Constructor * @param workers Array of workers to use for actions */ constructor(e) { this._pendingActions = new Array(), this._workerInfos = e.map((t) => ({ workerPromise: Promise.resolve(t), idle: !0 })); } /** * Terminates all workers and clears any pending actions. */ dispose() { for (const e of this._workerInfos) e.workerPromise.then((t) => { t.terminate(); }); this._workerInfos.length = 0, this._pendingActions.length = 0; } /** * Pushes an action to the worker pool. If all the workers are active, the action will be * pended until a worker has completed its action. * @param action The action to perform. Call onComplete when the action is complete. */ push(e) { this._executeOnIdleWorker(e) || this._pendingActions.push(e); } _executeOnIdleWorker(e) { for (const t of this._workerInfos) if (t.idle) return this._execute(t, e), !0; return !1; } _execute(e, t) { e.idle = !1, e.workerPromise.then((r) => { t(r, () => { const n = this._pendingActions.shift(); n ? this._execute(e, n) : e.idle = !0; }); }); } } class lU extends Ute { constructor(e, t, r = lU.DefaultOptions) { super([]), this._maxWorkers = e, this._createWorkerAsync = t, this._options = r; } push(e) { if (!this._executeOnIdleWorker(e)) if (this._workerInfos.length < this._maxWorkers) { const t = { workerPromise: this._createWorkerAsync(), idle: !1 }; this._workerInfos.push(t), this._execute(t, e); } else this._pendingActions.push(e); } _execute(e, t) { e.timeoutId && (clearTimeout(e.timeoutId), delete e.timeoutId), super._execute(e, (r, n) => { t(r, () => { n(), e.idle && (e.timeoutId = setTimeout(() => { e.workerPromise.then((s) => { s.terminate(); }); const i = this._workerInfos.indexOf(e); i !== -1 && this._workerInfos.splice(i, 1); }, this._options.idleTimeElapsedBeforeRelease)); }); }); } } lU.DefaultOptions = { idleTimeElapsedBeforeRelease: 1e3 }; var pF; (function(A) { A[A.ETC1S = 0] = "ETC1S", A[A.UASTC4x4 = 1] = "UASTC4x4"; })(pF || (pF = {})); var nS; (function(A) { A[A.ASTC_4X4_RGBA = 0] = "ASTC_4X4_RGBA", A[A.BC7_RGBA = 1] = "BC7_RGBA", A[A.BC3_RGBA = 2] = "BC3_RGBA", A[A.BC1_RGB = 3] = "BC1_RGB", A[A.PVRTC1_4_RGBA = 4] = "PVRTC1_4_RGBA", A[A.PVRTC1_4_RGB = 5] = "PVRTC1_4_RGB", A[A.ETC2_RGBA = 6] = "ETC2_RGBA", A[A.ETC1_RGB = 7] = "ETC1_RGB", A[A.RGBA32 = 8] = "RGBA32", A[A.R8 = 9] = "R8", A[A.RG8 = 10] = "RG8"; })(nS || (nS = {})); var nO; (function(A) { A[A.COMPRESSED_RGBA_BPTC_UNORM_EXT = 36492] = "COMPRESSED_RGBA_BPTC_UNORM_EXT", A[A.COMPRESSED_RGBA_ASTC_4X4_KHR = 37808] = "COMPRESSED_RGBA_ASTC_4X4_KHR", A[A.COMPRESSED_RGB_S3TC_DXT1_EXT = 33776] = "COMPRESSED_RGB_S3TC_DXT1_EXT", A[A.COMPRESSED_RGBA_S3TC_DXT5_EXT = 33779] = "COMPRESSED_RGBA_S3TC_DXT5_EXT", A[A.COMPRESSED_RGBA_PVRTC_4BPPV1_IMG = 35842] = "COMPRESSED_RGBA_PVRTC_4BPPV1_IMG", A[A.COMPRESSED_RGB_PVRTC_4BPPV1_IMG = 35840] = "COMPRESSED_RGB_PVRTC_4BPPV1_IMG", A[A.COMPRESSED_RGBA8_ETC2_EAC = 37496] = "COMPRESSED_RGBA8_ETC2_EAC", A[A.COMPRESSED_RGB8_ETC2 = 37492] = "COMPRESSED_RGB8_ETC2", A[A.COMPRESSED_RGB_ETC1_WEBGL = 36196] = "COMPRESSED_RGB_ETC1_WEBGL", A[A.RGBA8Format = 32856] = "RGBA8Format", A[A.R8Format = 33321] = "R8Format", A[A.RG8Format = 33323] = "RG8Format"; })(nO || (nO = {})); function hF(A) { A.wasmUASTCToASTC && (KTX2DECODER.LiteTranscoder_UASTC_ASTC.WasmModuleURL = A.wasmUASTCToASTC), A.wasmUASTCToBC7 && (KTX2DECODER.LiteTranscoder_UASTC_BC7.WasmModuleURL = A.wasmUASTCToBC7), A.wasmUASTCToRGBA_UNORM && (KTX2DECODER.LiteTranscoder_UASTC_RGBA_UNORM.WasmModuleURL = A.wasmUASTCToRGBA_UNORM), A.wasmUASTCToRGBA_SRGB && (KTX2DECODER.LiteTranscoder_UASTC_RGBA_SRGB.WasmModuleURL = A.wasmUASTCToRGBA_SRGB), A.wasmUASTCToR8_UNORM && (KTX2DECODER.LiteTranscoder_UASTC_R8_UNORM.WasmModuleURL = A.wasmUASTCToR8_UNORM), A.wasmUASTCToRG8_UNORM && (KTX2DECODER.LiteTranscoder_UASTC_RG8_UNORM.WasmModuleURL = A.wasmUASTCToRG8_UNORM), A.jsMSCTranscoder && (KTX2DECODER.MSCTranscoder.JSModuleURL = A.jsMSCTranscoder), A.wasmMSCTranscoder && (KTX2DECODER.MSCTranscoder.WasmModuleURL = A.wasmMSCTranscoder), A.wasmZSTDDecoder && (KTX2DECODER.ZSTDDecoder.WasmModuleURL = A.wasmZSTDDecoder); } class Ite { constructor() { this._isDirty = !0, this._useRGBAIfOnlyBC1BC3AvailableWhenUASTC = !0, this._ktx2DecoderOptions = {}; } /** * Gets the dirty flag */ get isDirty() { return this._isDirty; } /** * force a (uncompressed) RGBA transcoded format if transcoding a UASTC source format and ASTC + BC7 are not available as a compressed transcoded format */ get useRGBAIfASTCBC7NotAvailableWhenUASTC() { return this._useRGBAIfASTCBC7NotAvailableWhenUASTC; } set useRGBAIfASTCBC7NotAvailableWhenUASTC(e) { this._useRGBAIfASTCBC7NotAvailableWhenUASTC !== e && (this._useRGBAIfASTCBC7NotAvailableWhenUASTC = e, this._isDirty = !0); } /** * force a (uncompressed) RGBA transcoded format if transcoding a UASTC source format and only BC1 or BC3 are available as a compressed transcoded format. * This property is true by default to favor speed over memory, because currently transcoding from UASTC to BC1/3 is slow because the transcoder transcodes * to uncompressed and then recompresses the texture */ get useRGBAIfOnlyBC1BC3AvailableWhenUASTC() { return this._useRGBAIfOnlyBC1BC3AvailableWhenUASTC; } set useRGBAIfOnlyBC1BC3AvailableWhenUASTC(e) { this._useRGBAIfOnlyBC1BC3AvailableWhenUASTC !== e && (this._useRGBAIfOnlyBC1BC3AvailableWhenUASTC = e, this._isDirty = !0); } /** * force to always use (uncompressed) RGBA for transcoded format */ get forceRGBA() { return this._forceRGBA; } set forceRGBA(e) { this._forceRGBA !== e && (this._forceRGBA = e, this._isDirty = !0); } /** * force to always use (uncompressed) R8 for transcoded format */ get forceR8() { return this._forceR8; } set forceR8(e) { this._forceR8 !== e && (this._forceR8 = e, this._isDirty = !0); } /** * force to always use (uncompressed) RG8 for transcoded format */ get forceRG8() { return this._forceRG8; } set forceRG8(e) { this._forceRG8 !== e && (this._forceRG8 = e, this._isDirty = !0); } /** * list of transcoders to bypass when looking for a suitable transcoder. The available transcoders are: * UniversalTranscoder_UASTC_ASTC * UniversalTranscoder_UASTC_BC7 * UniversalTranscoder_UASTC_RGBA_UNORM * UniversalTranscoder_UASTC_RGBA_SRGB * UniversalTranscoder_UASTC_R8_UNORM * UniversalTranscoder_UASTC_RG8_UNORM * MSCTranscoder */ get bypassTranscoders() { return this._bypassTranscoders; } set bypassTranscoders(e) { this._bypassTranscoders !== e && (this._bypassTranscoders = e, this._isDirty = !0); } /** @internal */ _getKTX2DecoderOptions() { if (!this._isDirty) return this._ktx2DecoderOptions; this._isDirty = !1; const e = { useRGBAIfASTCBC7NotAvailableWhenUASTC: this._useRGBAIfASTCBC7NotAvailableWhenUASTC, forceRGBA: this._forceRGBA, forceR8: this._forceR8, forceRG8: this._forceRG8, bypassTranscoders: this._bypassTranscoders }; return this.useRGBAIfOnlyBC1BC3AvailableWhenUASTC && (e.transcodeFormatDecisionTree = { UASTC: { transcodeFormat: [nS.BC1_RGB, nS.BC3_RGBA], yes: { transcodeFormat: nS.RGBA32, engineFormat: nO.RGBA8Format, roundToMultiple4: !1 } } }), this._ktx2DecoderOptions = e, e; } } class H6 { static GetDefaultNumWorkers() { return typeof navigator != "object" || !navigator.hardwareConcurrency ? 1 : Math.min(Math.floor(navigator.hardwareConcurrency * 0.5), 4); } static _Initialize(e) { if (H6._WorkerPoolPromise || H6._DecoderModulePromise) return; const t = { jsDecoderModule: ye.GetBabylonScriptURL(this.URLConfig.jsDecoderModule, !0), wasmUASTCToASTC: ye.GetBabylonScriptURL(this.URLConfig.wasmUASTCToASTC, !0), wasmUASTCToBC7: ye.GetBabylonScriptURL(this.URLConfig.wasmUASTCToBC7, !0), wasmUASTCToRGBA_UNORM: ye.GetBabylonScriptURL(this.URLConfig.wasmUASTCToRGBA_UNORM, !0), wasmUASTCToRGBA_SRGB: ye.GetBabylonScriptURL(this.URLConfig.wasmUASTCToRGBA_SRGB, !0), wasmUASTCToR8_UNORM: ye.GetBabylonScriptURL(this.URLConfig.wasmUASTCToR8_UNORM, !0), wasmUASTCToRG8_UNORM: ye.GetBabylonScriptURL(this.URLConfig.wasmUASTCToRG8_UNORM, !0), jsMSCTranscoder: ye.GetBabylonScriptURL(this.URLConfig.jsMSCTranscoder, !0), wasmMSCTranscoder: ye.GetBabylonScriptURL(this.URLConfig.wasmMSCTranscoder, !0), wasmZSTDDecoder: ye.GetBabylonScriptURL(this.URLConfig.wasmZSTDDecoder, !0) }; e && typeof Worker == "function" && typeof URL < "u" ? H6._WorkerPoolPromise = new Promise((r) => { const n = `${hF}(${A3e})()`, i = URL.createObjectURL(new Blob([n], { type: "application/javascript" })); r(new lU(e, () => new Promise((s, a) => { const f = new Worker(i), o = (v) => { f.removeEventListener("error", o), f.removeEventListener("message", d), a(v); }, d = (v) => { v.data.action === "init" && (f.removeEventListener("error", o), f.removeEventListener("message", d), s(f)); }; f.addEventListener("error", o), f.addEventListener("message", d), f.postMessage({ action: "init", urls: t }); }))); }) : typeof KTX2DECODER > "u" ? H6._DecoderModulePromise = ye.LoadBabylonScriptAsync(t.jsDecoderModule).then(() => (KTX2DECODER.MSCTranscoder.UseFromWorkerThread = !1, KTX2DECODER.WASMMemoryManager.LoadBinariesFromCurrentThread = !0, hF(t), new KTX2DECODER.KTX2Decoder())) : (KTX2DECODER.MSCTranscoder.UseFromWorkerThread = !1, KTX2DECODER.WASMMemoryManager.LoadBinariesFromCurrentThread = !0, H6._DecoderModulePromise = Promise.resolve(new KTX2DECODER.KTX2Decoder())); } /** * Constructor * @param engine The engine to use * @param numWorkers The number of workers for async operations. Specify `0` to disable web workers and run synchronously in the current context. */ constructor(e, t = H6.DefaultNumWorkers) { this._engine = e, H6._Initialize(t); } /** * @internal */ uploadAsync(e, t, r) { const n = this._engine.getCaps(), i = { astc: !!n.astc, bptc: !!n.bptc, s3tc: !!n.s3tc, pvrtc: !!n.pvrtc, etc2: !!n.etc2, etc1: !!n.etc1 }; if (H6._WorkerPoolPromise) return H6._WorkerPoolPromise.then((s) => new Promise((a, f) => { s.push((o, d) => { const v = (P) => { o.removeEventListener("error", v), o.removeEventListener("message", u), f(P), d(); }, u = (P) => { if (P.data.action === "decoded") { if (o.removeEventListener("error", v), o.removeEventListener("message", u), !P.data.success) f({ message: P.data.msg }); else try { this._createTexture(P.data.decodedData, t, r), a(); } catch (p) { f({ message: p }); } d(); } }; o.addEventListener("error", v), o.addEventListener("message", u), o.postMessage({ action: "setDefaultDecoderOptions", options: H6.DefaultDecoderOptions._getKTX2DecoderOptions() }); const l = new Uint8Array(e.byteLength); l.set(new Uint8Array(e.buffer, e.byteOffset, e.byteLength)), o.postMessage({ action: "decode", data: l, caps: i, options: r }, [l.buffer]); }); })); if (H6._DecoderModulePromise) return H6._DecoderModulePromise.then((s) => (H6.DefaultDecoderOptions.isDirty && (KTX2DECODER.KTX2Decoder.DefaultDecoderOptions = H6.DefaultDecoderOptions._getKTX2DecoderOptions()), new Promise((a, f) => { s.decode(e, n).then((o) => { this._createTexture(o, t), a(); }).catch((o) => { f({ message: o }); }); }))); throw new Error("KTX2 decoder module is not available"); } _createTexture(e, t, r) { this._engine._bindTextureDirectly(3553, t), r && (r.transcodedFormat = e.transcodedFormat, r.isInGammaSpace = e.isInGammaSpace, r.hasAlpha = e.hasAlpha, r.transcoderName = e.transcoderName); let i = !0; switch (e.transcodedFormat) { case 32856: t.type = 0, t.format = 5; break; case 33321: t.type = 0, t.format = 6; break; case 33323: t.type = 0, t.format = 7; break; default: t.format = e.transcodedFormat, i = !1; break; } if (t._gammaSpace = e.isInGammaSpace, t.generateMipMaps = e.mipmaps.length > 1, e.errors) throw new Error("KTX2 container - could not transcode the data. " + e.errors); for (let s = 0; s < e.mipmaps.length; ++s) { const a = e.mipmaps[s]; if (!a || !a.data) throw new Error("KTX2 container - could not transcode one of the image"); i ? (t.width = a.width, t.height = a.height, this._engine._uploadDataToTextureDirectly(t, a.data, 0, s, void 0, !0)) : this._engine._uploadCompressedDataToTextureDirectly(t, e.transcodedFormat, a.width, a.height, a.data, 0, s); } t._extension = ".ktx2", t.width = e.mipmaps[0].width, t.height = e.mipmaps[0].height, t.isReady = !0, this._engine._bindTextureDirectly(3553, null); } /** * Checks if the given data starts with a KTX2 file identifier. * @param data the data to check * @returns true if the data is a KTX2 file or false otherwise */ static IsValid(e) { if (e.byteLength >= 12) { const t = new Uint8Array(e.buffer, e.byteOffset, 12); if (t[0] === 171 && t[1] === 75 && t[2] === 84 && t[3] === 88 && t[4] === 32 && t[5] === 50 && t[6] === 48 && t[7] === 187 && t[8] === 13 && t[9] === 10 && t[10] === 26 && t[11] === 10) return !0; } return !1; } } H6.URLConfig = { jsDecoderModule: "https://cdn.babylonjs.com/babylon.ktx2Decoder.js", wasmUASTCToASTC: null, wasmUASTCToBC7: null, wasmUASTCToRGBA_UNORM: null, wasmUASTCToRGBA_SRGB: null, wasmUASTCToR8_UNORM: null, wasmUASTCToRG8_UNORM: null, jsMSCTranscoder: null, wasmMSCTranscoder: null, wasmZSTDDecoder: null }; H6.DefaultNumWorkers = H6.GetDefaultNumWorkers(); H6.DefaultDecoderOptions = new Ite(); function A3e() { let A; onmessage = (e) => { if (e.data) switch (e.data.action) { case "init": { const t = e.data.urls; importScripts(t.jsDecoderModule), hF(t), A = new KTX2DECODER.KTX2Decoder(), postMessage({ action: "init" }); break; } case "setDefaultDecoderOptions": { KTX2DECODER.KTX2Decoder.DefaultDecoderOptions = e.data.options; break; } case "decode": A.decode(e.data.data, e.data.caps, e.data.options).then((t) => { const r = []; for (let n = 0; n < t.mipmaps.length; ++n) { const i = t.mipmaps[n]; i && i.data && r.push(i.data.buffer); } postMessage({ action: "decoded", success: !0, decodedData: t }, r); }).catch((t) => { postMessage({ action: "decoded", success: !1, msg: t }); }); break; } }; } function d3e(A) { switch (A) { case 35916: return 33776; case 35918: return 33778; case 35919: return 33779; case 37493: return 37492; case 37497: return 37496; case 37495: return 37494; case 37840: return 37808; case 36493: return 36492; } return null; } class Rte { constructor() { this.supportCascades = !1; } /** * This returns if the loader support the current file information. * @param extension defines the file extension of the file being loaded * @param mimeType defines the optional mime type of the file being loaded * @returns true if the loader can load the specified file */ canLoad(e, t) { return e.endsWith(".ktx") || e.endsWith(".ktx2") || t === "image/ktx" || t === "image/ktx2"; } /** * Uploads the cube texture data to the WebGL texture. It has already been bound. * @param data contains the texture data * @param texture defines the BabylonJS internal texture * @param createPolynomials will be true if polynomials have been requested * @param onLoad defines the callback to trigger once the texture is ready */ loadCubeData(e, t, r, n) { if (Array.isArray(e)) return; t._invertVScale = !t.invertY; const i = t.getEngine(), s = new yv(e, 6), a = s.numberOfMipmapLevels > 1 && t.generateMipMaps; i._unpackFlipY(!0), s.uploadLevels(t, t.generateMipMaps), t.width = s.pixelWidth, t.height = s.pixelHeight, i._setCubeMapTextureParams(t, a, s.numberOfMipmapLevels - 1), t.isReady = !0, t.onLoadedObservable.notifyObservers(t), t.onLoadedObservable.clear(), n && n(); } /** * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback. * @param data contains the texture data * @param texture defines the BabylonJS internal texture * @param callback defines the method to call once ready to upload * @param options */ loadData(e, t, r, n) { if (yv.IsValid(e)) { t._invertVScale = !t.invertY; const i = new yv(e, 1), s = d3e(i.glInternalFormat); s ? (t.format = s, t._useSRGBBuffer = t.getEngine()._getUseSRGBBuffer(!0, t.generateMipMaps), t._gammaSpace = !0) : t.format = i.glInternalFormat, r(i.pixelWidth, i.pixelHeight, t.generateMipMaps, !0, () => { i.uploadLevels(t, t.generateMipMaps); }, i.isInvalid); } else H6.IsValid(e) ? new H6(t.getEngine()).uploadAsync(e, t, n).then(() => { r(t.width, t.height, t.generateMipMaps, !0, () => { }, !1); }, (s) => { Se.Warn(`Failed to load KTX2 texture data: ${s.message}`), r(0, 0, !1, !1, () => { }, !0); }) : (Se.Error("texture missing KTX identifier"), r(0, 0, !1, !1, () => { }, !0)); } } Ge._TextureLoaders.unshift(new Rte()); class IS extends SA { /** * Creates a new webXRCamera, this should only be set at the camera after it has been updated by the xrSessionManager * @param name the name of the camera * @param scene the scene to add the camera to * @param _xrSessionManager a constructed xr session manager */ constructor(e, t, r) { super(e, S.Zero(), t), this._xrSessionManager = r, this._firstFrame = !1, this._referenceQuaternion = Ze.Identity(), this._referencedPosition = new S(), this._trackingState = Jw.NOT_TRACKING, this.onXRCameraInitializedObservable = new Oe(), this.onBeforeCameraTeleport = new Oe(), this.onAfterCameraTeleport = new Oe(), this.onTrackingStateChanged = new Oe(), this.compensateOnFirstFrame = !0, this._rotate180 = new Ze(0, 1, 0, 0), this.minZ = 0.1, this.rotationQuaternion = new Ze(), this.cameraRigMode = Tr.RIG_MODE_CUSTOM, this.updateUpVectorFromRotation = !0, this._updateNumberOfRigCameras(1), this.freezeProjectionMatrix(), this._deferOnly = !0, this._xrSessionManager.onXRSessionInit.add(() => { this._referencedPosition.copyFromFloats(0, 0, 0), this._referenceQuaternion.copyFromFloats(0, 0, 0, 1), this._firstFrame = this.compensateOnFirstFrame; }), this._xrSessionManager.onXRFrameObservable.add(() => { this._firstFrame && this._updateFromXRSession(), this.onXRCameraInitializedObservable.hasObservers() && (this.onXRCameraInitializedObservable.notifyObservers(this), this.onXRCameraInitializedObservable.clear()), this._deferredUpdated && (this.position.copyFrom(this._deferredPositionUpdate), this.rotationQuaternion.copyFrom(this._deferredRotationQuaternionUpdate)), this._updateReferenceSpace(), this._updateFromXRSession(); }, void 0, !0); } /** * Get the current XR tracking state of the camera */ get trackingState() { return this._trackingState; } _setTrackingState(e) { this._trackingState !== e && (this._trackingState = e, this.onTrackingStateChanged.notifyObservers(e)); } /** * Return the user's height, unrelated to the current ground. * This will be the y position of this camera, when ground level is 0. */ get realWorldHeight() { const e = this._xrSessionManager.currentFrame && this._xrSessionManager.currentFrame.getViewerPose(this._xrSessionManager.baseReferenceSpace); return e && e.transform ? e.transform.position.y : 0; } /** @internal */ _updateForDualEyeDebugging() { this._updateNumberOfRigCameras(2), this.rigCameras[0].viewport = new WA(0, 0, 0.5, 1), this.rigCameras[0].outputRenderTarget = null, this.rigCameras[1].viewport = new WA(0.5, 0, 0.5, 1), this.rigCameras[1].outputRenderTarget = null; } /** * Sets this camera's transformation based on a non-vr camera * @param otherCamera the non-vr camera to copy the transformation from * @param resetToBaseReferenceSpace should XR reset to the base reference space */ setTransformationFromNonVRCamera(e = this.getScene().activeCamera, t = !0) { if (!e || e === this) return; e.computeWorldMatrix().decompose(void 0, this.rotationQuaternion, this.position), this.position.y = 0, Ze.FromEulerAnglesToRef(0, this.rotationQuaternion.toEulerAngles().y, 0, this.rotationQuaternion), this._firstFrame = !0, t && this._xrSessionManager.resetReferenceSpace(); } /** * Gets the current instance class name ("WebXRCamera"). * @returns the class name */ getClassName() { return "WebXRCamera"; } /** * Set the target for the camera to look at. * Note that this only rotates around the Y axis, as opposed to the default behavior of other cameras * @param target the target to set the camera to look at */ setTarget(e) { const t = ue.Vector3[1]; e.subtractToRef(this.position, t), t.y = 0, t.normalize(); const r = Math.atan2(t.x, t.z); this.rotationQuaternion.toEulerAnglesToRef(t), Ze.FromEulerAnglesToRef(t.x, r, t.z, this.rotationQuaternion); } dispose() { super.dispose(), this._lastXRViewerPose = void 0; } _updateFromXRSession() { const e = this._xrSessionManager.currentFrame && this._xrSessionManager.currentFrame.getViewerPose(this._xrSessionManager.referenceSpace); if (this._lastXRViewerPose = e || void 0, !e) { this._setTrackingState(Jw.NOT_TRACKING); return; } const t = e.emulatedPosition ? Jw.TRACKING_LOST : Jw.TRACKING; if (this._setTrackingState(t), this.minZ !== this._cache.minZ || this.maxZ !== this._cache.maxZ) { const r = { // if maxZ is 0 it should be "Infinity", but it doesn't work with the WebXR API. Setting to a large number. depthFar: this.maxZ || 1e4, depthNear: this.minZ }; this._xrSessionManager.updateRenderState(r), this._cache.minZ = this.minZ, this._cache.maxZ = this.maxZ; } if (e.transform) { const r = e.transform.orientation; if (e.transform.orientation.x === void 0) return; const n = e.transform.position; this._referencedPosition.set(n.x, n.y, n.z), this._referenceQuaternion.set(r.x, r.y, r.z, r.w), this._scene.useRightHandedSystem || (this._referencedPosition.z *= -1, this._referenceQuaternion.z *= -1, this._referenceQuaternion.w *= -1), this._firstFrame ? (this._firstFrame = !1, this.position.y += this._referencedPosition.y, this._referenceQuaternion.copyFromFloats(0, 0, 0, 1)) : (this.rotationQuaternion.copyFrom(this._referenceQuaternion), this.position.copyFrom(this._referencedPosition)); } this.rigCameras.length !== e.views.length && this._updateNumberOfRigCameras(e.views.length), e.views.forEach((r, n) => { var i; const s = this.rigCameras[n]; !s.isLeftCamera && !s.isRightCamera && (r.eye === "right" ? s._isRightCamera = !0 : r.eye === "left" && (s._isLeftCamera = !0)); const a = r.transform.position, f = r.transform.orientation; s.parent = this.parent, s.position.set(a.x, a.y, a.z), s.rotationQuaternion.set(f.x, f.y, f.z, f.w), this._scene.useRightHandedSystem ? s.rotationQuaternion.multiplyInPlace(this._rotate180) : (s.position.z *= -1, s.rotationQuaternion.z *= -1, s.rotationQuaternion.w *= -1), he.FromFloat32ArrayToRefScaled(r.projectionMatrix, 0, 1, s._projectionMatrix), this._scene.useRightHandedSystem || s._projectionMatrix.toggleProjectionMatrixHandInPlace(), n === 0 && this._projectionMatrix.copyFrom(s._projectionMatrix); const o = this._xrSessionManager.getRenderTargetTextureForView(r); this._renderingMultiview = ((i = o == null ? void 0 : o._texture) === null || i === void 0 ? void 0 : i.isMultiview) || !1, this._renderingMultiview ? n == 0 && (this._xrSessionManager.trySetViewportForView(this.viewport, r), this.outputRenderTarget = o) : (this._xrSessionManager.trySetViewportForView(s.viewport, r), s.outputRenderTarget = o || this._xrSessionManager.getRenderTargetTextureForView(r)), s.layerMask = this.layerMask; }); } _updateNumberOfRigCameras(e = 1) { for (; this.rigCameras.length < e; ) { const t = new b1("XR-RigCamera: " + this.rigCameras.length, S.Zero(), this.getScene()); t.minZ = 0.1, t.rotationQuaternion = new Ze(), t.updateUpVectorFromRotation = !0, t.isRigCamera = !0, t.rigParent = this, t.freezeProjectionMatrix(), this.rigCameras.push(t); } for (; this.rigCameras.length > e; ) { const t = this.rigCameras.pop(); t && t.dispose(); } } _updateReferenceSpace() { if (!this.position.equals(this._referencedPosition) || !this.rotationQuaternion.equals(this._referenceQuaternion)) { const e = ue.Matrix[0], t = ue.Matrix[1], r = ue.Matrix[2]; he.ComposeToRef(IS._ScaleReadOnly, this._referenceQuaternion, this._referencedPosition, e), he.ComposeToRef(IS._ScaleReadOnly, this.rotationQuaternion, this.position, t), e.invert().multiplyToRef(t, r), r.invert(), this._scene.useRightHandedSystem || r.toggleModelMatrixHandInPlace(), r.decompose(void 0, this._referenceQuaternion, this._referencedPosition); const n = new XRRigidTransform({ x: this._referencedPosition.x, y: this._referencedPosition.y, z: this._referencedPosition.z }, { x: this._referenceQuaternion.x, y: this._referenceQuaternion.y, z: this._referenceQuaternion.z, w: this._referenceQuaternion.w }); this._xrSessionManager.referenceSpace = this._xrSessionManager.referenceSpace.getOffsetReferenceSpace(n); } } } IS._ScaleReadOnly = S.One(); class Ay { /** * Creates a WebXRExperienceHelper * @param _scene The scene the helper should be created in */ constructor(e) { this._scene = e, this._nonVRCamera = null, this._attachedToElement = !1, this._spectatorCamera = null, this._originalSceneAutoClear = !0, this._supported = !1, this._spectatorMode = !1, this._lastTimestamp = 0, this.onInitialXRPoseSetObservable = new Oe(), this.onStateChangedObservable = new Oe(), this.state = d9.NOT_IN_XR, this.sessionManager = new MR(e), this.camera = new IS("webxr", e, this.sessionManager), this.featuresManager = new So(this.sessionManager), e.onDisposeObservable.addOnce(() => { this.dispose(); }); } /** * Creates the experience helper * @param scene the scene to attach the experience helper to * @returns a promise for the experience helper */ static CreateAsync(e) { const t = new Ay(e); return t.sessionManager.initializeAsync().then(() => (t._supported = !0, t)).catch((r) => { throw t._setState(d9.NOT_IN_XR), t.dispose(), r; }); } /** * Disposes of the experience helper */ dispose() { var e; this.exitXRAsync(), this.camera.dispose(), this.onStateChangedObservable.clear(), this.onInitialXRPoseSetObservable.clear(), this.sessionManager.dispose(), (e = this._spectatorCamera) === null || e === void 0 || e.dispose(), this._nonVRCamera && (this._scene.activeCamera = this._nonVRCamera); } /** * Enters XR mode (This must be done within a user interaction in most browsers eg. button click) * @param sessionMode options for the XR session * @param referenceSpaceType frame of reference of the XR session * @param renderTarget the output canvas that will be used to enter XR mode * @param sessionCreationOptions optional XRSessionInit object to init the session with * @returns promise that resolves after xr mode has entered */ async enterXRAsync(e, t, r = this.sessionManager.getWebXRRenderTarget(), n = {}) { var i, s, a; if (!this._supported) throw "WebXR not supported in this browser or environment"; this._setState(d9.ENTERING_XR), t !== "viewer" && t !== "local" && (n.optionalFeatures = n.optionalFeatures || [], n.optionalFeatures.push(t)), n = await this.featuresManager._extendXRSessionInitObject(n), e === "immersive-ar" && t !== "unbounded" && Se.Warn("We recommend using 'unbounded' reference space type when using 'immersive-ar' session mode"); try { await this.sessionManager.initializeSessionAsync(e, n), await this.sessionManager.setReferenceSpaceTypeAsync(t); const f = await r.initializeXRLayerAsync(this.sessionManager.session), o = { // if maxZ is 0 it should be "Infinity", but it doesn't work with the WebXR API. Setting to a large number. depthFar: this.camera.maxZ || 1e4, depthNear: this.camera.minZ }; return this.featuresManager.getEnabledFeature(Gi.LAYERS) || (o.baseLayer = f), this.sessionManager.updateRenderState(o), this.sessionManager.runXRRenderLoop(), this._originalSceneAutoClear = this._scene.autoClear, this._nonVRCamera = this._scene.activeCamera, this._attachedToElement = !!(!((s = (i = this._nonVRCamera) === null || i === void 0 ? void 0 : i.inputs) === null || s === void 0) && s.attachedToElement), (a = this._nonVRCamera) === null || a === void 0 || a.detachControl(), this._scene.activeCamera = this.camera, e !== "immersive-ar" ? this._nonXRToXRCamera() : (this._scene.autoClear = !1, this.camera.compensateOnFirstFrame = !1, this.camera.position.set(0, 0, 0), this.camera.rotationQuaternion.set(0, 0, 0, 1), this.onInitialXRPoseSetObservable.notifyObservers(this.camera)), this.sessionManager.onXRSessionEnded.addOnce(() => { this.state !== d9.EXITING_XR && this._setState(d9.EXITING_XR), this.camera.rigCameras.forEach((d) => { d.outputRenderTarget = null; }), this._scene.autoClear = this._originalSceneAutoClear, this._scene.activeCamera = this._nonVRCamera, this._attachedToElement && this._nonVRCamera && this._nonVRCamera.attachControl(!!this._nonVRCamera.inputs.noPreventDefault), e !== "immersive-ar" && this.camera.compensateOnFirstFrame && (this._nonVRCamera.setPosition ? this._nonVRCamera.setPosition(this.camera.position) : this._nonVRCamera.position.copyFrom(this.camera.position)), this._setState(d9.NOT_IN_XR); }), this.sessionManager.onXRFrameObservable.addOnce(() => { this._setState(d9.IN_XR); }), this.sessionManager; } catch (f) { throw console.log(f), console.log(f.message), this._setState(d9.NOT_IN_XR), f; } } /** * Exits XR mode and returns the scene to its original state * @returns promise that resolves after xr mode has exited */ exitXRAsync() { return this.state !== d9.IN_XR ? Promise.resolve() : (this._setState(d9.EXITING_XR), this.sessionManager.exitXRAsync()); } /** * Enable spectator mode for desktop VR experiences. * When spectator mode is enabled a camera will be attached to the desktop canvas and will * display the first rig camera's view on the desktop canvas. * Please note that this will degrade performance, as it requires another camera render. * It is also not recommended to enable this in devices like the quest, as it brings no benefit there. * @param options giving WebXRSpectatorModeOption for specutator camera to setup when the spectator mode is enabled. */ enableSpectatorMode(e) { this._spectatorMode || (this._spectatorMode = !0, this._switchSpectatorMode(e)); } /** * Disable spectator mode for desktop VR experiences. */ disableSpecatatorMode() { this._spectatorMode && (this._spectatorMode = !1, this._switchSpectatorMode()); } _switchSpectatorMode(e) { const r = 1 / (e != null && e.fps ? e.fps : 1e3) * 1e3, n = e != null && e.preferredCameraIndex ? e == null ? void 0 : e.preferredCameraIndex : 0, i = () => { this._spectatorCamera && this.sessionManager.currentTimestamp - this._lastTimestamp >= r && (this._lastTimestamp = this.sessionManager.currentTimestamp, this._spectatorCamera.position.copyFrom(this.camera.rigCameras[n].globalPosition), this._spectatorCamera.rotationQuaternion.copyFrom(this.camera.rigCameras[n].absoluteRotation)); }; if (this._spectatorMode) { if (n >= this.camera.rigCameras.length) throw new Error("the preferred camera index is beyond the length of rig camera array."); const s = () => { this.state === d9.IN_XR ? (this._spectatorCamera = new bD("webxr-spectator", S.Zero(), this._scene), this._spectatorCamera.rotationQuaternion = new Ze(), this._scene.activeCameras = [this.camera, this._spectatorCamera], this.sessionManager.onXRFrameObservable.add(i), this._scene.onAfterRenderCameraObservable.add((a) => { a === this.camera && (this._scene.getEngine().framebufferDimensionsObject = null); })) : this.state === d9.EXITING_XR && (this.sessionManager.onXRFrameObservable.removeCallback(i), this._scene.activeCameras = null); }; this.onStateChangedObservable.add(s), s(); } else this.sessionManager.onXRFrameObservable.removeCallback(i), this._scene.activeCameras = [this.camera]; } _nonXRToXRCamera() { this.camera.setTransformationFromNonVRCamera(this._nonVRCamera), this.onInitialXRPoseSetObservable.notifyObservers(this.camera); } _setState(e) { this.state !== e && (this.state = e, this.onStateChangedObservable.notifyObservers(this.state)); } } class Jl { /** * Creates a new component for a motion controller. * It is created by the motion controller itself * * @param id the id of this component * @param type the type of the component * @param _buttonIndex index in the buttons array of the gamepad * @param _axesIndices indices of the values in the axes array of the gamepad */ constructor(e, t, r = -1, n = []) { this.id = e, this.type = t, this._buttonIndex = r, this._axesIndices = n, this._axes = { x: 0, y: 0 }, this._changes = {}, this._currentValue = 0, this._hasChanges = !1, this._pressed = !1, this._touched = !1, this.onAxisValueChangedObservable = new Oe(), this.onButtonStateChangedObservable = new Oe(); } /** * The current axes data. If this component has no axes it will still return an object { x: 0, y: 0 } */ get axes() { return this._axes; } /** * Get the changes. Elements will be populated only if they changed with their previous and current value */ get changes() { return this._changes; } /** * Return whether or not the component changed the last frame */ get hasChanges() { return this._hasChanges; } /** * is the button currently pressed */ get pressed() { return this._pressed; } /** * is the button currently touched */ get touched() { return this._touched; } /** * Get the current value of this component */ get value() { return this._currentValue; } /** * Dispose this component */ dispose() { this.onAxisValueChangedObservable.clear(), this.onButtonStateChangedObservable.clear(); } /** * Are there axes correlating to this component * @returns true is axes data is available */ isAxes() { return this._axesIndices.length !== 0; } /** * Is this component a button (hence - pressable) * @returns true if can be pressed */ isButton() { return this._buttonIndex !== -1; } /** * update this component using the gamepad object it is in. Called on every frame * @param nativeController the native gamepad controller object */ update(e) { let t = !1, r = !1; if (this._hasChanges = !1, this._changes = {}, this.isButton()) { const n = e.buttons[this._buttonIndex]; if (!n) return; this._currentValue !== n.value && (this.changes.value = { current: n.value, previous: this._currentValue }, t = !0, this._currentValue = n.value), this._touched !== n.touched && (this.changes.touched = { current: n.touched, previous: this._touched }, t = !0, this._touched = n.touched), this._pressed !== n.pressed && (this.changes.pressed = { current: n.pressed, previous: this._pressed }, t = !0, this._pressed = n.pressed); } this.isAxes() && (this._axes.x !== e.axes[this._axesIndices[0]] && (this.changes.axes = { current: { x: e.axes[this._axesIndices[0]], y: this._axes.y }, previous: { x: this._axes.x, y: this._axes.y } }, this._axes.x = e.axes[this._axesIndices[0]], r = !0), this._axes.y !== e.axes[this._axesIndices[1]] && (this.changes.axes ? this.changes.axes.current.y = e.axes[this._axesIndices[1]] : this.changes.axes = { current: { x: this._axes.x, y: e.axes[this._axesIndices[1]] }, previous: { x: this._axes.x, y: this._axes.y } }, this._axes.y = e.axes[this._axesIndices[1]], r = !0)), t && (this._hasChanges = !0, this.onButtonStateChangedObservable.notifyObservers(this)), r && (this._hasChanges = !0, this.onAxisValueChangedObservable.notifyObservers(this._axes)); } } Jl.BUTTON_TYPE = "button"; Jl.SQUEEZE_TYPE = "squeeze"; Jl.THUMBSTICK_TYPE = "thumbstick"; Jl.TOUCHPAD_TYPE = "touchpad"; Jl.TRIGGER_TYPE = "trigger"; class Um { /** * constructs a new abstract motion controller * @param scene the scene to which the model of the controller will be added * @param layout The profile layout to load * @param gamepadObject The gamepad object correlating to this controller * @param handedness handedness (left/right/none) of this controller * @param _doNotLoadControllerMesh set this flag to ignore the mesh loading * @param _controllerCache a cache holding controller models already loaded in this session */ constructor(e, t, r, n, i = !1, s) { this.scene = e, this.layout = t, this.gamepadObject = r, this.handedness = n, this._doNotLoadControllerMesh = i, this._controllerCache = s, this._initComponent = (a) => { if (!a) return; const f = this.layout.components[a], o = f.type, d = f.gamepadIndices.button, v = []; f.gamepadIndices.xAxis !== void 0 && f.gamepadIndices.yAxis !== void 0 && v.push(f.gamepadIndices.xAxis, f.gamepadIndices.yAxis), this.components[a] = new Jl(a, o, d, v); }, this._modelReady = !1, this.components = {}, this.disableAnimation = !1, this.onModelLoadedObservable = new Oe(), t.components && Object.keys(t.components).forEach(this._initComponent); } /** * Dispose this controller, the model mesh and all its components */ dispose() { this.getComponentIds().forEach((e) => this.getComponent(e).dispose()), this.rootMesh && (this.rootMesh.getChildren(void 0, !0).forEach((e) => { e.setEnabled(!1); }), this.rootMesh.dispose(!!this._controllerCache, !this._controllerCache)); } /** * Returns all components of specific type * @param type the type to search for * @returns an array of components with this type */ getAllComponentsOfType(e) { return this.getComponentIds().map((t) => this.components[t]).filter((t) => t.type === e); } /** * get a component based an its component id as defined in layout.components * @param id the id of the component * @returns the component correlates to the id or undefined if not found */ getComponent(e) { return this.components[e]; } /** * Get the list of components available in this motion controller * @returns an array of strings correlating to available components */ getComponentIds() { return Object.keys(this.components); } /** * Get the first component of specific type * @param type type of component to find * @returns a controller component or null if not found */ getComponentOfType(e) { return this.getAllComponentsOfType(e)[0] || null; } /** * Get the main (Select) component of this controller as defined in the layout * @returns the main component of this controller */ getMainComponent() { return this.getComponent(this.layout.selectComponentId); } /** * Loads the model correlating to this controller * When the mesh is loaded, the onModelLoadedObservable will be triggered * @returns A promise fulfilled with the result of the model loading */ async loadModel() { const e = !this._getModelLoadingConstraints(); let t = this._getGenericFilenameAndPath(); return e ? Se.Warn("Falling back to generic models") : t = this._getFilenameAndPath(), new Promise((r, n) => { const i = (s) => { e ? this._getGenericParentMesh(s) : this._setRootMesh(s), this._processLoadedModel(s), this._modelReady = !0, this.onModelLoadedObservable.notifyObservers(this), r(!0); }; if (this._controllerCache) { const s = this._controllerCache.filter((a) => a.filename === t.filename && a.path === t.path); if (s[0]) { s[0].meshes.forEach((a) => a.setEnabled(!0)), i(s[0].meshes); return; } } Hn.ImportMesh("", t.path, t.filename, this.scene, (s) => { this._controllerCache && this._controllerCache.push(Object.assign(Object.assign({}, t), { meshes: s })), i(s); }, null, (s, a) => { Se.Log(a), Se.Warn(`Failed to retrieve controller model of type ${this.profileId} from the remote server: ${t.path}${t.filename}`), n(a); }); }); } /** * Update this model using the current XRFrame * @param xrFrame the current xr frame to use and update the model */ updateFromXRFrame(e) { this.getComponentIds().forEach((t) => this.getComponent(t).update(this.gamepadObject)), this.updateModel(e); } /** * Backwards compatibility due to a deeply-integrated typo */ get handness() { return this.handedness; } /** * Pulse (vibrate) this controller * If the controller does not support pulses, this function will fail silently and return Promise directly after called * Consecutive calls to this function will cancel the last pulse call * * @param value the strength of the pulse in 0.0...1.0 range * @param duration Duration of the pulse in milliseconds * @param hapticActuatorIndex optional index of actuator (will usually be 0) * @returns a promise that will send true when the pulse has ended and false if the device doesn't support pulse or an error accrued */ pulse(e, t, r = 0) { return this.gamepadObject.hapticActuators && this.gamepadObject.hapticActuators[r] ? this.gamepadObject.hapticActuators[r].pulse(e, t) : Promise.resolve(!1); } // Look through all children recursively. This will return null if no mesh exists with the given name. _getChildByName(e, t) { return e.getChildren((r) => r.name === t, !1)[0]; } // Look through only immediate children. This will return null if no mesh exists with the given name. _getImmediateChildByName(e, t) { return e.getChildren((r) => r.name == t, !0)[0]; } /** * Moves the axis on the controller mesh based on its current state * @param axisMap * @param axisValue the value of the axis which determines the meshes new position * @internal */ _lerpTransform(e, t, r) { if (!e.minMesh || !e.maxMesh || !e.valueMesh || !e.minMesh.rotationQuaternion || !e.maxMesh.rotationQuaternion || !e.valueMesh.rotationQuaternion) return; const n = r ? t * 0.5 + 0.5 : t; Ze.SlerpToRef(e.minMesh.rotationQuaternion, e.maxMesh.rotationQuaternion, n, e.valueMesh.rotationQuaternion), S.LerpToRef(e.minMesh.position, e.maxMesh.position, n, e.valueMesh.position); } /** * Update the model itself with the current frame data * @param xrFrame the frame to use for updating the model mesh */ // eslint-disable-next-line @typescript-eslint/naming-convention updateModel(e) { this._modelReady && this._updateModel(e); } _getGenericFilenameAndPath() { return { filename: "generic.babylon", path: "https://controllers.babylonjs.com/generic/" }; } _getGenericParentMesh(e) { this.rootMesh = new Ee(this.profileId + " " + this.handedness, this.scene), e.forEach((t) => { t.parent || (t.isPickable = !1, t.setParent(this.rootMesh)); }), this.rootMesh.rotationQuaternion = Ze.FromEulerAngles(0, Math.PI, 0); } } class RS extends Um { constructor(e, t, r) { super(e, v3e[r], t, r), this.profileId = RS.ProfileId; } _getFilenameAndPath() { return { filename: "generic.babylon", path: "https://controllers.babylonjs.com/generic/" }; } _getModelLoadingConstraints() { return !0; } // eslint-disable-next-line @typescript-eslint/no-unused-vars _processLoadedModel(e) { } _setRootMesh(e) { this.rootMesh = new Ee(this.profileId + " " + this.handedness, this.scene), e.forEach((t) => { t.isPickable = !1, t.parent || t.setParent(this.rootMesh); }), this.rootMesh.rotationQuaternion = Ze.FromEulerAngles(0, Math.PI, 0); } _updateModel() { } } RS.ProfileId = "generic-trigger"; const v3e = { left: { selectComponentId: "xr-standard-trigger", components: { // eslint-disable-next-line @typescript-eslint/naming-convention "xr-standard-trigger": { type: "trigger", gamepadIndices: { button: 0 }, rootNodeName: "xr_standard_trigger", visualResponses: {} } }, gamepadMapping: "xr-standard", rootNodeName: "generic-trigger-left", assetPath: "left.glb" }, right: { selectComponentId: "xr-standard-trigger", components: { // eslint-disable-next-line @typescript-eslint/naming-convention "xr-standard-trigger": { type: "trigger", gamepadIndices: { button: 0 }, rootNodeName: "xr_standard_trigger", visualResponses: {} } }, gamepadMapping: "xr-standard", rootNodeName: "generic-trigger-right", assetPath: "right.glb" }, none: { selectComponentId: "xr-standard-trigger", components: { // eslint-disable-next-line @typescript-eslint/naming-convention "xr-standard-trigger": { type: "trigger", gamepadIndices: { button: 0 }, rootNodeName: "xr_standard_trigger", visualResponses: {} } }, gamepadMapping: "xr-standard", rootNodeName: "generic-trigger-none", assetPath: "none.glb" } }; class Vte extends Um { constructor(e, t, r, n, i) { super(e, r.layouts[t.handedness || "none"], t.gamepad, t.handedness, void 0, i), this._repositoryUrl = n, this.controllerCache = i, this._buttonMeshMapping = {}, this._touchDots = {}, this.profileId = r.profileId; } dispose() { super.dispose(), this.controllerCache || Object.keys(this._touchDots).forEach((e) => { this._touchDots[e].dispose(); }); } _getFilenameAndPath() { return { filename: this.layout.assetPath, path: `${this._repositoryUrl}/profiles/${this.profileId}/` }; } _getModelLoadingConstraints() { const e = Hn.IsPluginForExtensionAvailable(".glb"); return e || Se.Warn("glTF / glb loader was not registered, using generic controller instead"), e; } _processLoadedModel(e) { this.getComponentIds().forEach((t) => { const r = this.layout.components[t]; this._buttonMeshMapping[t] = { mainMesh: this._getChildByName(this.rootMesh, r.rootNodeName), states: {} }, Object.keys(r.visualResponses).forEach((n) => { const i = r.visualResponses[n]; if (i.valueNodeProperty === "transform") this._buttonMeshMapping[t].states[n] = { valueMesh: this._getChildByName(this.rootMesh, i.valueNodeName), minMesh: this._getChildByName(this.rootMesh, i.minNodeName), maxMesh: this._getChildByName(this.rootMesh, i.maxNodeName) }; else { const s = r.type === Jl.TOUCHPAD_TYPE && r.touchPointNodeName ? r.touchPointNodeName : i.valueNodeName; if (this._buttonMeshMapping[t].states[n] = { valueMesh: this._getChildByName(this.rootMesh, s) }, r.type === Jl.TOUCHPAD_TYPE && !this._touchDots[n]) { const a = UA(n + "dot", { diameter: 15e-4, segments: 8 }, this.scene); a.material = new Wt(n + "mat", this.scene), a.material.diffuseColor = Ne.Red(), a.parent = this._buttonMeshMapping[t].states[n].valueMesh || null, a.isVisible = !1, this._touchDots[n] = a; } } }); }); } _setRootMesh(e) { this.rootMesh = new Ee(this.profileId + "-" + this.handedness, this.scene), this.rootMesh.isPickable = !1; let t; for (let r = 0; r < e.length; r++) { const n = e[r]; n.isPickable = !1, n.parent || (t = n); } t && t.setParent(this.rootMesh), this.scene.useRightHandedSystem || this.rootMesh.rotate(bf.Y, Math.PI, ai.WORLD); } _updateModel(e) { this.disableAnimation || this.getComponentIds().forEach((t) => { const r = this.getComponent(t); if (!r.hasChanges) return; const n = this._buttonMeshMapping[t], i = this.layout.components[t]; Object.keys(i.visualResponses).forEach((s) => { const a = i.visualResponses[s]; let f = r.value; if (a.componentProperty === "xAxis" ? f = r.axes.x : a.componentProperty === "yAxis" && (f = r.axes.y), a.valueNodeProperty === "transform") this._lerpTransform(n.states[s], f, a.componentProperty !== "button"); else { const o = n.states[s].valueMesh; o && (o.isVisible = r.touched || r.pressed), this._touchDots[s] && (this._touchDots[s].isVisible = r.touched || r.pressed); } }); }); } } const SE = []; class E9 { /** * Clear the cache used for profile loading and reload when requested again */ static ClearProfilesCache() { this._ProfilesList = null, this._ProfileLoadingPromises = {}; } /** * Register the default fallbacks. * This function is called automatically when this file is imported. */ static DefaultFallbacks() { this.RegisterFallbacksForProfileId("google-daydream", ["generic-touchpad"]), this.RegisterFallbacksForProfileId("htc-vive-focus", ["generic-trigger-touchpad"]), this.RegisterFallbacksForProfileId("htc-vive", ["generic-trigger-squeeze-touchpad"]), this.RegisterFallbacksForProfileId("magicleap-one", ["generic-trigger-squeeze-touchpad"]), this.RegisterFallbacksForProfileId("windows-mixed-reality", ["generic-trigger-squeeze-touchpad-thumbstick"]), this.RegisterFallbacksForProfileId("microsoft-mixed-reality", ["windows-mixed-reality", "generic-trigger-squeeze-touchpad-thumbstick"]), this.RegisterFallbacksForProfileId("oculus-go", ["generic-trigger-touchpad"]), this.RegisterFallbacksForProfileId("oculus-touch-v2", ["oculus-touch", "generic-trigger-squeeze-thumbstick"]), this.RegisterFallbacksForProfileId("oculus-touch", ["generic-trigger-squeeze-thumbstick"]), this.RegisterFallbacksForProfileId("samsung-gearvr", ["windows-mixed-reality", "generic-trigger-squeeze-touchpad-thumbstick"]), this.RegisterFallbacksForProfileId("samsung-odyssey", ["generic-touchpad"]), this.RegisterFallbacksForProfileId("valve-index", ["generic-trigger-squeeze-touchpad-thumbstick"]), this.RegisterFallbacksForProfileId("generic-hand-select", ["generic-trigger"]); } /** * Find a fallback profile if the profile was not found. There are a few predefined generic profiles. * @param profileId the profile to which a fallback needs to be found * @returns an array with corresponding fallback profiles */ static FindFallbackWithProfileId(e) { const t = this._Fallbacks[e] || []; return t.unshift(e), t; } /** * When acquiring a new xrInput object (usually by the WebXRInput class), match it with the correct profile. * The order of search: * * 1) Iterate the profiles array of the xr input and try finding a corresponding motion controller * 2) (If not found) search in the gamepad id and try using it (legacy versions only) * 3) search for registered fallbacks (should be redundant, nonetheless it makes sense to check) * 4) return the generic trigger controller if none were found * * @param xrInput the xrInput to which a new controller is initialized * @param scene the scene to which the model will be added * @param forceProfile force a certain profile for this controller * @returns A promise that fulfils with the motion controller class for this profile id or the generic standard class if none was found */ static GetMotionControllerWithXRInput(e, t, r) { const n = []; if (r && n.push(r), n.push(...e.profiles || []), n.length && !n[0] && n.pop(), e.gamepad && e.gamepad.id) switch (e.gamepad.id) { case (e.gamepad.id.match(/oculus touch/gi) ? e.gamepad.id : void 0): n.push("oculus-touch-v2"); break; } const i = n.indexOf("windows-mixed-reality"); if (i !== -1 && n.splice(i, 0, "microsoft-mixed-reality"), n.length || n.push("generic-trigger"), this.UseOnlineRepository) { const s = this.PrioritizeOnlineRepository ? this._LoadProfileFromRepository : this._LoadProfilesFromAvailableControllers, a = this.PrioritizeOnlineRepository ? this._LoadProfilesFromAvailableControllers : this._LoadProfileFromRepository; return s.call(this, n, e, t).catch(() => a.call(this, n, e, t)); } else return this._LoadProfilesFromAvailableControllers(n, e, t); } /** * Register a new controller based on its profile. This function will be called by the controller classes themselves. * * If you are missing a profile, make sure it is imported in your source, otherwise it will not register. * * @param type the profile type to register * @param constructFunction the function to be called when loading this profile */ static RegisterController(e, t) { this._AvailableControllers[e] = t; } /** * Register a fallback to a specific profile. * @param profileId the profileId that will receive the fallbacks * @param fallbacks A list of fallback profiles */ static RegisterFallbacksForProfileId(e, t) { this._Fallbacks[e] ? this._Fallbacks[e].push(...t) : this._Fallbacks[e] = t; } /** * Will update the list of profiles available in the repository * @returns a promise that resolves to a map of profiles available online */ static UpdateProfilesList() { return this._ProfilesList = ye.LoadFileAsync(this.BaseRepositoryUrl + "/profiles/profilesList.json", !1).then((e) => JSON.parse(e.toString())), this._ProfilesList; } /** * Clear the controller's cache (usually happens at the end of a session) */ static ClearControllerCache() { SE.forEach((e) => { e.meshes.forEach((t) => { t.dispose(!1, !0); }); }), SE.length = 0; } static _LoadProfileFromRepository(e, t, r) { return Promise.resolve().then(() => this._ProfilesList ? this._ProfilesList : this.UpdateProfilesList()).then((n) => { for (let i = 0; i < e.length; ++i) if (e[i] && n[e[i]]) return e[i]; throw new Error(`neither controller ${e[0]} nor all fallbacks were found in the repository,`); }).then((n) => (this._ProfileLoadingPromises[n] || (this._ProfileLoadingPromises[n] = ye.LoadFileAsync(`${this.BaseRepositoryUrl}/profiles/${n}/profile.json`, !1).then((i) => JSON.parse(i))), this._ProfileLoadingPromises[n])).then((n) => new Vte(r, t, n, this.BaseRepositoryUrl, this.DisableControllerCache ? void 0 : SE)); } static _LoadProfilesFromAvailableControllers(e, t, r) { for (let n = 0; n < e.length; ++n) { if (!e[n]) continue; const i = this.FindFallbackWithProfileId(e[n]); for (let s = 0; s < i.length; ++s) { const a = this._AvailableControllers[i[s]]; if (a) return Promise.resolve(a(t, r)); } } throw new Error("no controller requested was found in the available controllers list"); } } E9._AvailableControllers = {}; E9._Fallbacks = {}; E9._ProfileLoadingPromises = {}; E9.BaseRepositoryUrl = "https://immersive-web.github.io/webxr-input-profiles/packages/viewer/dist"; E9.PrioritizeOnlineRepository = !0; E9.UseOnlineRepository = !0; E9.DisableControllerCache = !0; E9.RegisterController(RS.ProfileId, (A, e) => new RS(e, A.gamepad, A.handedness)); E9.DefaultFallbacks(); let u3e = 0; class Cte { /** * Creates the input source object * @see https://doc.babylonjs.com/features/featuresDeepDive/webXR/webXRInputControllerSupport * @param _scene the scene which the controller should be associated to * @param inputSource the underlying input source for the controller * @param _options options for this controller creation */ constructor(e, t, r = {}) { this._scene = e, this.inputSource = t, this._options = r, this._tmpVector = new S(), this._disposed = !1, this.onDisposeObservable = new Oe(), this.onMeshLoadedObservable = new Oe(), this.onMotionControllerInitObservable = new Oe(), this._uniqueId = `controller-${u3e++}-${t.targetRayMode}-${t.handedness}`, this.pointer = new jn(`${this._uniqueId}-pointer`, e), this.pointer.rotationQuaternion = new Ze(), this.inputSource.gripSpace && (this.grip = new jn(`${this._uniqueId}-grip`, this._scene), this.grip.rotationQuaternion = new Ze()), this._tmpVector.set(0, 0, this._scene.useRightHandedSystem ? -1 : 1), this.inputSource.gamepad && this.inputSource.targetRayMode === "tracked-pointer" && E9.GetMotionControllerWithXRInput(t, e, this._options.forceControllerProfile).then((n) => { this.motionController = n, this.onMotionControllerInitObservable.notifyObservers(n), !this._options.doNotLoadControllerMesh && !this.motionController._doNotLoadControllerMesh && this.motionController.loadModel().then((i) => { var s; i && this.motionController && this.motionController.rootMesh && (this._options.renderingGroupId && (this.motionController.rootMesh.renderingGroupId = this._options.renderingGroupId, this.motionController.rootMesh.getChildMeshes(!1).forEach((a) => a.renderingGroupId = this._options.renderingGroupId)), this.onMeshLoadedObservable.notifyObservers(this.motionController.rootMesh), this.motionController.rootMesh.parent = this.grip || this.pointer, this.motionController.disableAnimation = !!this._options.disableMotionControllerAnimation), this._disposed && ((s = this.motionController) === null || s === void 0 || s.dispose()); }); }, () => { ye.Warn("Could not find a matching motion controller for the registered input source"); }); } /** * Get this controllers unique id */ get uniqueId() { return this._uniqueId; } /** * Disposes of the object */ dispose() { this.grip && this.grip.dispose(!0), this.motionController && this.motionController.dispose(), this.pointer.dispose(!0), this.onMotionControllerInitObservable.clear(), this.onMeshLoadedObservable.clear(), this.onDisposeObservable.notifyObservers(this), this.onDisposeObservable.clear(), this._disposed = !0; } /** * Gets a world space ray coming from the pointer or grip * @param result the resulting ray * @param gripIfAvailable use the grip mesh instead of the pointer, if available */ getWorldPointerRayToRef(e, t = !1) { const r = t && this.grip ? this.grip : this.pointer; S.TransformNormalToRef(this._tmpVector, r.getWorldMatrix(), e.direction), e.direction.normalize(), e.origin.copyFrom(r.absolutePosition), e.length = 1e3; } /** * Updates the controller pose based on the given XRFrame * @param xrFrame xr frame to update the pose with * @param referenceSpace reference space to use * @param xrCamera the xr camera, used for parenting */ updateFromXRFrame(e, t, r) { const n = e.getPose(this.inputSource.targetRaySpace, t); if (this._lastXRPose = n, n) { const i = n.transform.position; this.pointer.position.set(i.x, i.y, i.z); const s = n.transform.orientation; this.pointer.rotationQuaternion.set(s.x, s.y, s.z, s.w), this._scene.useRightHandedSystem || (this.pointer.position.z *= -1, this.pointer.rotationQuaternion.z *= -1, this.pointer.rotationQuaternion.w *= -1), this.pointer.parent = r.parent; } if (this.inputSource.gripSpace && this.grip) { const i = e.getPose(this.inputSource.gripSpace, t); if (i) { const s = i.transform.position, a = i.transform.orientation; this.grip.position.set(s.x, s.y, s.z), this.grip.rotationQuaternion.set(a.x, a.y, a.z, a.w), this._scene.useRightHandedSystem || (this.grip.position.z *= -1, this.grip.rotationQuaternion.z *= -1, this.grip.rotationQuaternion.w *= -1); } this.grip.parent = r.parent; } this.motionController && this.motionController.updateFromXRFrame(e); } } class Ote { /** * Initializes the WebXRInput * @param xrSessionManager the xr session manager for this session * @param xrCamera the WebXR camera for this session. Mainly used for teleportation * @param _options = initialization options for this xr input */ constructor(e, t, r = {}) { if (this.xrSessionManager = e, this.xrCamera = t, this._options = r, this.controllers = [], this.onControllerAddedObservable = new Oe(), this.onControllerRemovedObservable = new Oe(), this._onInputSourcesChange = (n) => { this._addAndRemoveControllers(n.added, n.removed); }, this._sessionEndedObserver = this.xrSessionManager.onXRSessionEnded.add(() => { this._addAndRemoveControllers([], this.controllers.map((n) => n.inputSource)); }), this._sessionInitObserver = this.xrSessionManager.onXRSessionInit.add((n) => { n.addEventListener("inputsourceschange", this._onInputSourcesChange); }), this._frameObserver = this.xrSessionManager.onXRFrameObservable.add((n) => { this.controllers.forEach((i) => { i.updateFromXRFrame(n, this.xrSessionManager.referenceSpace, this.xrCamera); }); }), this._options.customControllersRepositoryURL && (E9.BaseRepositoryUrl = this._options.customControllersRepositoryURL), E9.UseOnlineRepository = !this._options.disableOnlineControllerRepository, E9.UseOnlineRepository) try { E9.UpdateProfilesList().catch(() => { E9.UseOnlineRepository = !1; }); } catch { E9.UseOnlineRepository = !1; } } _addAndRemoveControllers(e, t) { const r = this.controllers.map((s) => s.inputSource); for (const s of e) if (r.indexOf(s) === -1) { const a = new Cte(this.xrSessionManager.scene, s, Object.assign(Object.assign({}, this._options.controllerOptions || {}), { forceControllerProfile: this._options.forceInputProfile, doNotLoadControllerMesh: this._options.doNotLoadControllerMeshes, disableMotionControllerAnimation: this._options.disableControllerAnimation })); this.controllers.push(a), this.onControllerAddedObservable.notifyObservers(a); } const n = [], i = []; this.controllers.forEach((s) => { t.indexOf(s.inputSource) === -1 ? n.push(s) : i.push(s); }), this.controllers = n, i.forEach((s) => { this.onControllerRemovedObservable.notifyObservers(s), s.dispose(); }); } /** * Disposes of the object */ dispose() { this.controllers.forEach((e) => { e.dispose(); }), this.xrSessionManager.onXRFrameObservable.remove(this._frameObserver), this.xrSessionManager.onXRSessionInit.remove(this._sessionInitObserver), this.xrSessionManager.onXRSessionEnded.remove(this._sessionEndedObserver), this.onControllerAddedObservable.clear(), this.onControllerRemovedObservable.clear(), E9.ClearControllerCache(); } } class ig extends L9 { /** * constructs a new background remover module * @param _xrSessionManager the session manager for this module * @param _options read-only options to be used in this module */ constructor(e, t) { super(e), this._options = t, this._attachController = (r) => { if (this._controllers[r.uniqueId]) return; const { laserPointer: n, selectionMesh: i } = this._generateNewMeshPair(r.pointer); switch (this._controllers[r.uniqueId] = { xrController: r, laserPointer: n, selectionMesh: i, meshUnderPointer: null, pick: null, tmpRay: new Hi(new S(), new S()), disabledByNearInteraction: !1, id: ig._IdCounter++ }, this._attachedController ? !this._options.enablePointerSelectionOnAllControllers && this._options.preferredHandedness && r.inputSource.handedness === this._options.preferredHandedness && (this._attachedController = r.uniqueId) : this._options.enablePointerSelectionOnAllControllers || (this._attachedController = r.uniqueId), r.inputSource.targetRayMode) { case "tracked-pointer": return this._attachTrackedPointerRayMode(r); case "gaze": return this._attachGazeMode(r); case "screen": return this._attachScreenRayMode(r); } }, this._controllers = {}, this._tmpVectorForPickCompare = new S(), this.disablePointerLighting = !0, this.disableSelectionMeshLighting = !0, this.displayLaserPointer = !0, this.displaySelectionMesh = !0, this.laserPointerPickedColor = new Ne(0.9, 0.9, 0.9), this.laserPointerDefaultColor = new Ne(0.7, 0.7, 0.7), this.selectionMeshDefaultColor = new Ne(0.8, 0.8, 0.8), this.selectionMeshPickedColor = new Ne(0.3, 0.3, 1), this._identityMatrix = he.Identity(), this._screenCoordinatesRef = S.Zero(), this._viewportRef = new WA(0, 0, 0, 0), this._scene = this._xrSessionManager.scene; } /** * attach this feature * Will usually be called by the features manager * * @returns true if successful. */ attach() { if (!super.attach()) return !1; if (this._options.xrInput.controllers.forEach(this._attachController), this._addNewAttachObserver(this._options.xrInput.onControllerAddedObservable, this._attachController), this._addNewAttachObserver(this._options.xrInput.onControllerRemovedObservable, (e) => { this._detachController(e.uniqueId); }), this._scene.constantlyUpdateMeshUnderPointer = !0, this._options.gazeCamera) { const e = this._options.gazeCamera, { laserPointer: t, selectionMesh: r } = this._generateNewMeshPair(e); this._controllers.camera = { webXRCamera: e, laserPointer: t, selectionMesh: r, meshUnderPointer: null, pick: null, tmpRay: new Hi(new S(), new S()), disabledByNearInteraction: !1, id: ig._IdCounter++ }, this._attachGazeMode(); } return !0; } /** * detach this feature. * Will usually be called by the features manager * * @returns true if successful. */ detach() { return super.detach() ? (Object.keys(this._controllers).forEach((e) => { this._detachController(e); }), !0) : !1; } /** * Will get the mesh under a specific pointer. * `scene.meshUnderPointer` will only return one mesh - either left or right. * @param controllerId the controllerId to check * @returns The mesh under pointer or null if no mesh is under the pointer */ getMeshUnderPointer(e) { return this._controllers[e] ? this._controllers[e].meshUnderPointer : null; } /** * Get the xr controller that correlates to the pointer id in the pointer event * * @param id the pointer id to search for * @returns the controller that correlates to this id or null if not found */ getXRControllerByPointerId(e) { const t = Object.keys(this._controllers); for (let r = 0; r < t.length; ++r) if (this._controllers[t[r]].id === e) return this._controllers[t[r]].xrController || null; return null; } /** * @internal */ _getPointerSelectionDisabledByPointerId(e) { const t = Object.keys(this._controllers); for (let r = 0; r < t.length; ++r) if (this._controllers[t[r]].id === e) return this._controllers[t[r]].disabledByNearInteraction; return !0; } /** * @internal */ _setPointerSelectionDisabledByPointerId(e, t) { const r = Object.keys(this._controllers); for (let n = 0; n < r.length; ++n) if (this._controllers[r[n]].id === e) { this._controllers[r[n]].disabledByNearInteraction = t; return; } } _onXRFrame(e) { Object.keys(this._controllers).forEach((t) => { const r = this._controllers[t]; if (!this._options.enablePointerSelectionOnAllControllers && t !== this._attachedController || r.disabledByNearInteraction) { r.selectionMesh.isVisible = !1, r.laserPointer.isVisible = !1, r.pick = null; return; } r.laserPointer.isVisible = this.displayLaserPointer; let n; if (r.xrController) n = r.xrController.pointer.position, r.xrController.getWorldPointerRayToRef(r.tmpRay); else if (r.webXRCamera) n = r.webXRCamera.position, r.webXRCamera.getForwardRayToRef(r.tmpRay); else return; if (this._options.maxPointerDistance && (r.tmpRay.length = this._options.maxPointerDistance), !this._options.disableScenePointerVectorUpdate && n) { const f = this._xrSessionManager.scene, o = this._options.xrInput.xrCamera; o && (o.viewport.toGlobalToRef(f.getEngine().getRenderWidth(), f.getEngine().getRenderHeight(), this._viewportRef), S.ProjectToRef(n, this._identityMatrix, f.getTransformMatrix(), this._viewportRef, this._screenCoordinatesRef), typeof this._screenCoordinatesRef.x == "number" && typeof this._screenCoordinatesRef.y == "number" && !isNaN(this._screenCoordinatesRef.x) && !isNaN(this._screenCoordinatesRef.y) && (f.pointerX = this._screenCoordinatesRef.x, f.pointerY = this._screenCoordinatesRef.y, r.screenCoordinates = { x: this._screenCoordinatesRef.x, y: this._screenCoordinatesRef.y })); } let i = null; this._utilityLayerScene && (i = this._utilityLayerScene.pickWithRay(r.tmpRay, this._utilityLayerScene.pointerMovePredicate || this.raySelectionPredicate)); const s = this._scene.pickWithRay(r.tmpRay, this._scene.pointerMovePredicate || this.raySelectionPredicate); !i || !i.hit ? r.pick = s : !s || !s.hit || i.distance < s.distance ? r.pick = i : r.pick = s, r.pick && r.xrController && (r.pick.aimTransform = r.xrController.pointer, r.pick.gripTransform = r.xrController.grip || null); const a = r.pick; if (a && a.pickedPoint && a.hit) { this._updatePointerDistance(r.laserPointer, a.distance), r.selectionMesh.position.copyFrom(a.pickedPoint), r.selectionMesh.scaling.x = Math.sqrt(a.distance), r.selectionMesh.scaling.y = Math.sqrt(a.distance), r.selectionMesh.scaling.z = Math.sqrt(a.distance); const f = this._convertNormalToDirectionOfRay(a.getNormal(!0), r.tmpRay), o = 1e-3; if (r.selectionMesh.position.copyFrom(a.pickedPoint), f) { const d = S.Cross(bf.Y, f), v = S.Cross(f, d); S.RotationFromAxisToRef(v, f, d, r.selectionMesh.rotation), r.selectionMesh.position.addInPlace(f.scale(o)); } r.selectionMesh.isVisible = this.displaySelectionMesh, r.meshUnderPointer = a.pickedMesh; } else r.selectionMesh.isVisible = !1, this._updatePointerDistance(r.laserPointer, 1), r.meshUnderPointer = null; }); } get _utilityLayerScene() { return this._options.customUtilityLayerScene || Ds.DefaultUtilityLayer.utilityLayerScene; } _attachGazeMode(e) { const t = this._controllers[e && e.uniqueId || "camera"], r = this._options.timeToSelect || 3e3, n = this._options.useUtilityLayer ? this._utilityLayerScene : this._scene; let i = new F9(); const s = Ag("selection", { diameter: 35e-4 * 15, thickness: 25e-4 * 6, tessellation: 20 }, n); s.isVisible = !1, s.isPickable = !1, s.parent = t.selectionMesh; let a = 0, f = !1; const o = { pointerId: t.id, pointerType: "xr" }; t.onFrameObserver = this._xrSessionManager.onXRFrameObservable.add(() => { if (t.pick) { if (this._augmentPointerInit(o, t.id, t.screenCoordinates), t.laserPointer.material.alpha = 0, s.isVisible = !1, t.pick.hit) if (this._pickingMoved(i, t.pick)) f && (this._options.disablePointerUpOnTouchOut || this._scene.simulatePointerUp(t.pick, o)), f = !1, a = 0; else if (a > r / 10 && (s.isVisible = !0), a += this._scene.getEngine().getDeltaTime(), a >= r) this._scene.simulatePointerDown(t.pick, o), f = !0, this._options.disablePointerUpOnTouchOut && this._scene.simulatePointerUp(t.pick, o), s.isVisible = !1; else { const d = 1 - a / r; s.scaling.set(d, d, d); } else f = !1, a = 0; this._scene.simulatePointerMove(t.pick, o), i = t.pick; } }), this._options.renderingGroupId !== void 0 && (s.renderingGroupId = this._options.renderingGroupId), e && e.onDisposeObservable.addOnce(() => { t.pick && !this._options.disablePointerUpOnTouchOut && f && (this._scene.simulatePointerUp(t.pick, o), t.finalPointerUpTriggered = !0), s.dispose(); }); } _attachScreenRayMode(e) { const t = this._controllers[e.uniqueId]; let r = !1; const n = { pointerId: t.id, pointerType: "xr" }; t.onFrameObserver = this._xrSessionManager.onXRFrameObservable.add(() => { this._augmentPointerInit(n, t.id, t.screenCoordinates), !(!t.pick || this._options.disablePointerUpOnTouchOut && r) && (r ? this._scene.simulatePointerMove(t.pick, n) : (this._scene.simulatePointerDown(t.pick, n), t.pointerDownTriggered = !0, r = !0, this._options.disablePointerUpOnTouchOut && this._scene.simulatePointerUp(t.pick, n))); }), e.onDisposeObservable.addOnce(() => { this._augmentPointerInit(n, t.id, t.screenCoordinates), this._xrSessionManager.runInXRFrame(() => { t.pick && !t.finalPointerUpTriggered && r && !this._options.disablePointerUpOnTouchOut && (this._scene.simulatePointerUp(t.pick, n), t.finalPointerUpTriggered = !0); }); }); } _attachTrackedPointerRayMode(e) { const t = this._controllers[e.uniqueId]; if (this._options.forceGazeMode) return this._attachGazeMode(e); const r = { pointerId: t.id, pointerType: "xr" }; if (t.onFrameObserver = this._xrSessionManager.onXRFrameObservable.add(() => { t.laserPointer.material.disableLighting = this.disablePointerLighting, t.selectionMesh.material.disableLighting = this.disableSelectionMeshLighting, t.pick && (this._augmentPointerInit(r, t.id, t.screenCoordinates), this._scene.simulatePointerMove(t.pick, r)); }), e.inputSource.gamepad) { const n = (i) => { this._options.overrideButtonId && (t.selectionComponent = i.getComponent(this._options.overrideButtonId)), t.selectionComponent || (t.selectionComponent = i.getMainComponent()), t.onButtonChangedObserver = t.selectionComponent.onButtonStateChangedObservable.add((s) => { if (s.changes.pressed) { const a = s.changes.pressed.current; t.pick ? (this._options.enablePointerSelectionOnAllControllers || e.uniqueId === this._attachedController) && (this._augmentPointerInit(r, t.id, t.screenCoordinates), a ? (this._scene.simulatePointerDown(t.pick, r), t.pointerDownTriggered = !0, t.selectionMesh.material.emissiveColor = this.selectionMeshPickedColor, t.laserPointer.material.emissiveColor = this.laserPointerPickedColor) : (this._scene.simulatePointerUp(t.pick, r), t.selectionMesh.material.emissiveColor = this.selectionMeshDefaultColor, t.laserPointer.material.emissiveColor = this.laserPointerDefaultColor)) : a && !this._options.enablePointerSelectionOnAllControllers && !this._options.disableSwitchOnClick && (this._attachedController = e.uniqueId); } }); }; e.motionController ? n(e.motionController) : e.onMotionControllerInitObservable.add(n); } else { const n = (s) => { this._augmentPointerInit(r, t.id, t.screenCoordinates), t.xrController && s.inputSource === t.xrController.inputSource && t.pick && (this._scene.simulatePointerDown(t.pick, r), t.pointerDownTriggered = !0, t.selectionMesh.material.emissiveColor = this.selectionMeshPickedColor, t.laserPointer.material.emissiveColor = this.laserPointerPickedColor); }, i = (s) => { this._augmentPointerInit(r, t.id, t.screenCoordinates), t.xrController && s.inputSource === t.xrController.inputSource && t.pick && (this._scene.simulatePointerUp(t.pick, r), t.selectionMesh.material.emissiveColor = this.selectionMeshDefaultColor, t.laserPointer.material.emissiveColor = this.laserPointerDefaultColor); }; t.eventListeners = { selectend: i, selectstart: n }, this._xrSessionManager.session.addEventListener("selectstart", n), this._xrSessionManager.session.addEventListener("selectend", i); } } _convertNormalToDirectionOfRay(e, t) { return e && Math.acos(S.Dot(e, t.direction)) < Math.PI / 2 && e.scaleInPlace(-1), e; } _detachController(e) { const t = this._controllers[e]; if (t) { if (t.selectionComponent && t.onButtonChangedObserver && t.selectionComponent.onButtonStateChangedObservable.remove(t.onButtonChangedObserver), t.onFrameObserver && this._xrSessionManager.onXRFrameObservable.remove(t.onFrameObserver), t.eventListeners && Object.keys(t.eventListeners).forEach((r) => { const n = t.eventListeners && t.eventListeners[r]; n && this._xrSessionManager.session.removeEventListener(r, n); }), !t.finalPointerUpTriggered && t.pointerDownTriggered) { const r = { pointerId: t.id, pointerType: "xr" }; this._xrSessionManager.runInXRFrame(() => { this._augmentPointerInit(r, t.id, t.screenCoordinates), this._scene.simulatePointerUp(t.pick || new F9(), r), t.finalPointerUpTriggered = !0; }); } this._xrSessionManager.scene.onBeforeRenderObservable.addOnce(() => { try { if (t.selectionMesh.dispose(), t.laserPointer.dispose(), delete this._controllers[e], this._attachedController === e) { const r = Object.keys(this._controllers); r.length ? this._attachedController = r[0] : this._attachedController = ""; } } catch { ye.Warn("controller already detached."); } }); } } _generateNewMeshPair(e) { const t = this._options.useUtilityLayer ? this._options.customUtilityLayerScene || Ds.DefaultUtilityLayer.utilityLayerScene : this._scene, r = this._options.customLasterPointerMeshGenerator ? this._options.customLasterPointerMeshGenerator() : Ld("laserPointer", { height: 1, diameterTop: 2e-4, diameterBottom: 4e-3, tessellation: 20, subdivisions: 1 }, t); r.parent = e; const n = new Wt("laserPointerMat", t); n.emissiveColor = this.laserPointerDefaultColor, n.alpha = 0.7, r.material = n, r.rotation.x = Math.PI / 2, this._updatePointerDistance(r, 1), r.isPickable = !1, r.isVisible = !1; const i = this._options.customSelectionMeshGenerator ? this._options.customSelectionMeshGenerator() : Ag("gazeTracker", { diameter: 35e-4 * 3, thickness: 25e-4 * 3, tessellation: 20 }, t); i.bakeCurrentTransformIntoVertices(), i.isPickable = !1, i.isVisible = !1; const s = new Wt("targetMat", t); return s.specularColor = Ne.Black(), s.emissiveColor = this.selectionMeshDefaultColor, s.backFaceCulling = !1, i.material = s, this._options.renderingGroupId !== void 0 && (r.renderingGroupId = this._options.renderingGroupId, i.renderingGroupId = this._options.renderingGroupId), { laserPointer: r, selectionMesh: i }; } _pickingMoved(e, t) { var r; if (!e.hit || !t.hit || !e.pickedMesh || !e.pickedPoint || !t.pickedMesh || !t.pickedPoint || e.pickedMesh !== t.pickedMesh) return !0; (r = e.pickedPoint) === null || r === void 0 || r.subtractToRef(t.pickedPoint, this._tmpVectorForPickCompare), this._tmpVectorForPickCompare.set(Math.abs(this._tmpVectorForPickCompare.x), Math.abs(this._tmpVectorForPickCompare.y), Math.abs(this._tmpVectorForPickCompare.z)); const n = (this._options.gazeModePointerMovedFactor || 1) * 0.01 * t.distance; return this._tmpVectorForPickCompare.length() > n; } _updatePointerDistance(e, t = 100) { e.scaling.y = t, this._scene.useRightHandedSystem && (t *= -1), e.position.z = t / 2 + 0.05; } _augmentPointerInit(e, t, r) { e.pointerId = t, e.pointerType = "xr", r && (e.screenX = r.x, e.screenY = r.y); } /** @internal */ get lasterPointerDefaultColor() { return this.laserPointerDefaultColor; } } ig._IdCounter = 200; ig.Name = Gi.POINTER_SELECTION; ig.Version = 1; So.AddWebXRFeature(ig.Name, (A, e) => () => new ig(A, e), ig.Version, !0); rA.prototype._projectOnTrianglesToRef = function(A, e, t, r, n, i) { const s = ue.Vector3[0], a = ue.Vector3[1]; let f = 1 / 0; for (let o = this.indexStart; o < this.indexStart + this.indexCount - (3 - r); o += r) { const d = t[o], v = t[o + 1], u = t[o + 2]; if (n && u === 4294967295) { o += 2; continue; } const l = e[d], P = e[v], p = e[u]; if (!l || !P || !p) continue; const c = S.ProjectOnTriangleToRef(A, l, P, p, a); c < f && (s.copyFrom(a), f = c); } return i.copyFrom(s), f; }; rA.prototype._projectOnUnIndexedTrianglesToRef = function(A, e, t, r) { const n = ue.Vector3[0], i = ue.Vector3[1]; let s = 1 / 0; for (let a = this.verticesStart; a < this.verticesStart + this.verticesCount; a += 3) { const f = e[a], o = e[a + 1], d = e[a + 2], v = S.ProjectOnTriangleToRef(A, f, o, d, i); v < s && (n.copyFrom(i), s = v); } return r.copyFrom(n), s; }; rA.prototype.projectToRef = function(A, e, t, r) { const n = this.getMaterial(); if (!n) return -1; let i = 3, s = !1; switch (n.fillMode) { case 3: case 5: case 6: case 8: return -1; case 7: i = 1, s = !0; break; } return n.fillMode === 4 ? -1 : !t.length && this._mesh._unIndexed ? this._projectOnUnIndexedTrianglesToRef(A, e, t, r) : this._projectOnTrianglesToRef(A, e, t, i, s, r); }; var vc; (function(A) { A[A.DEHYDRATED = 0] = "DEHYDRATED", A[A.HOVER = 1] = "HOVER", A[A.TOUCH = 2] = "TOUCH"; })(vc || (vc = {})); var zx; (function(A) { A[A.DISABLED = 0] = "DISABLED", A[A.CENTERED_ON_CONTROLLER = 1] = "CENTERED_ON_CONTROLLER", A[A.CENTERED_IN_FRONT = 2] = "CENTERED_IN_FRONT"; })(zx || (zx = {})); class sg extends L9 { /** * constructs a new background remover module * @param _xrSessionManager the session manager for this module * @param _options read-only options to be used in this module */ constructor(e, t) { super(e), this._options = t, this._tmpRay = new Hi(new S(), new S()), this._attachController = (r) => { if (this._controllers[r.uniqueId]) return; const { touchCollisionMesh: n, touchCollisionMeshFunction: i, hydrateCollisionMeshFunction: s } = this._generateNewTouchPointMesh(), a = this._generateVisualCue(); switch (this._controllers[r.uniqueId] = { xrController: r, meshUnderPointer: null, nearInteractionTargetMesh: null, pick: null, stalePick: null, touchCollisionMesh: n, touchCollisionMeshFunction: i, hydrateCollisionMeshFunction: s, currentAnimationState: vc.DEHYDRATED, grabRay: new Hi(new S(), new S()), hoverInteraction: !1, nearInteraction: !1, grabInteraction: !1, id: sg._IdCounter++, pickedPointVisualCue: a }, this._attachedController ? !this._options.enableNearInteractionOnAllControllers && this._options.preferredHandedness && r.inputSource.handedness === this._options.preferredHandedness && (this._attachedController = r.uniqueId) : this._options.enableNearInteractionOnAllControllers || (this._attachedController = r.uniqueId), r.inputSource.targetRayMode) { case "tracked-pointer": return this._attachNearInteractionMode(r); case "gaze": return null; case "screen": return null; } }, this._controllers = {}, this._farInteractionFeature = null, this.selectionMeshDefaultColor = new Ne(0.8, 0.8, 0.8), this.selectionMeshPickedColor = new Ne(0.3, 0.3, 1), this._hoverRadius = 0.1, this._pickRadius = 0.02, this._controllerPickRadius = 0.03, this._nearGrabLengthScale = 5, this._scene = this._xrSessionManager.scene, this._options.nearInteractionControllerMode === void 0 && (this._options.nearInteractionControllerMode = zx.CENTERED_IN_FRONT), this._options.farInteractionFeature && (this._farInteractionFeature = this._options.farInteractionFeature); } /** * Attach this feature * Will usually be called by the features manager * * @returns true if successful. */ attach() { return super.attach() ? (this._options.xrInput.controllers.forEach(this._attachController), this._addNewAttachObserver(this._options.xrInput.onControllerAddedObservable, this._attachController), this._addNewAttachObserver(this._options.xrInput.onControllerRemovedObservable, (e) => { this._detachController(e.uniqueId); }), this._scene.constantlyUpdateMeshUnderPointer = !0, !0) : !1; } /** * Detach this feature. * Will usually be called by the features manager * * @returns true if successful. */ detach() { return super.detach() ? (Object.keys(this._controllers).forEach((e) => { this._detachController(e); }), !0) : !1; } /** * Will get the mesh under a specific pointer. * `scene.meshUnderPointer` will only return one mesh - either left or right. * @param controllerId the controllerId to check * @returns The mesh under pointer or null if no mesh is under the pointer */ getMeshUnderPointer(e) { return this._controllers[e] ? this._controllers[e].meshUnderPointer : null; } /** * Get the xr controller that correlates to the pointer id in the pointer event * * @param id the pointer id to search for * @returns the controller that correlates to this id or null if not found */ getXRControllerByPointerId(e) { const t = Object.keys(this._controllers); for (let r = 0; r < t.length; ++r) if (this._controllers[t[r]].id === e) return this._controllers[t[r]].xrController || null; return null; } /** * This function sets webXRControllerPointerSelection feature that will be disabled when * the hover range is reached for a mesh and will be reattached when not in hover range. * This is used to remove the selection rays when moving. * @param farInteractionFeature the feature to disable when finger is in hover range for a mesh */ setFarInteractionFeature(e) { this._farInteractionFeature = e; } /** * Filter used for near interaction pick and hover * @param mesh */ _nearPickPredicate(e) { return e.isEnabled() && e.isVisible && e.isPickable && e.isNearPickable; } /** * Filter used for near interaction grab * @param mesh */ _nearGrabPredicate(e) { return e.isEnabled() && e.isVisible && e.isPickable && e.isNearGrabbable; } /** * Filter used for any near interaction * @param mesh */ _nearInteractionPredicate(e) { return e.isEnabled() && e.isVisible && e.isPickable && (e.isNearPickable || e.isNearGrabbable); } _controllerAvailablePredicate(e, t) { let r = e; for (; r; ) { if (r.reservedDataStore && r.reservedDataStore.nearInteraction && r.reservedDataStore.nearInteraction.excludedControllerId === t) return !1; r = r.parent; } return !0; } _handleTransitionAnimation(e, t) { var r; if (!(e.currentAnimationState === t || this._options.nearInteractionControllerMode !== zx.CENTERED_IN_FRONT || !((r = e.xrController) === null || r === void 0) && r.inputSource.hand)) { if (t > e.currentAnimationState) switch (e.currentAnimationState) { case vc.DEHYDRATED: if (e.hydrateCollisionMeshFunction(!0), t === vc.HOVER) break; case vc.HOVER: if (e.touchCollisionMeshFunction(!0), t === vc.TOUCH) break; } else switch (e.currentAnimationState) { case vc.TOUCH: if (e.touchCollisionMeshFunction(!1), t === vc.HOVER) break; case vc.HOVER: if (e.hydrateCollisionMeshFunction(!1), t === vc.DEHYDRATED) break; } e.currentAnimationState = t; } } _processTouchPoint(e, t, r) { var n; const i = this._controllers[e]; i.grabRay.origin.copyFrom(t), r.toEulerAnglesToRef(ue.Vector3[0]), i.grabRay.direction.copyFrom(ue.Vector3[0]), this._options.nearInteractionControllerMode === zx.CENTERED_IN_FRONT && !(!((n = i.xrController) === null || n === void 0) && n.inputSource.hand) && (i.xrController.getWorldPointerRayToRef(this._tmpRay), i.grabRay.origin.addInPlace(this._tmpRay.direction.scale(0.05))), i.grabRay.length = this._nearGrabLengthScale * this._hoverRadius, i.touchCollisionMesh.position.copyFrom(i.grabRay.origin); } _onXRFrame(e) { Object.keys(this._controllers).forEach((t) => { var r; const n = this._controllers[t], i = (r = n.xrController) === null || r === void 0 ? void 0 : r.inputSource.hand; if (!this._options.enableNearInteractionOnAllControllers && t !== this._attachedController || !n.xrController || !i && (!this._options.nearInteractionControllerMode || !n.xrController.inputSource.gamepad)) { n.pick = null; return; } if (n.hoverInteraction = !1, n.nearInteraction = !1, n.xrController) { if (i) { const o = i.get("index-finger-tip"); if (o) { const d = e.getJointPose(o, this._xrSessionManager.referenceSpace); if (d && d.transform) { const v = this._scene.useRightHandedSystem ? 1 : -1; ue.Vector3[0].set(d.transform.position.x, d.transform.position.y, d.transform.position.z * v), ue.Quaternion[0].set(d.transform.orientation.x, d.transform.orientation.y, d.transform.orientation.z * v, d.transform.orientation.w * v), this._processTouchPoint(t, ue.Vector3[0], ue.Quaternion[0]); } } } else if (n.xrController.inputSource.gamepad && this._options.nearInteractionControllerMode !== zx.DISABLED) { let o = n.xrController.pointer; n.xrController.grip && this._options.nearInteractionControllerMode === zx.CENTERED_ON_CONTROLLER && (o = n.xrController.grip), this._processTouchPoint(t, o.position, o.rotationQuaternion); } } else return; const s = (o, d) => { let v = null; return !d || !d.hit ? v = o : !o || !o.hit || d.distance < o.distance ? v = d : v = o, v; }, a = (o) => { let d = new F9(), v = !1; const u = o && o.pickedPoint && o.hit; return o != null && o.pickedPoint && (v = o.pickedPoint.x === 0 && o.pickedPoint.y === 0 && o.pickedPoint.z === 0), u && !v && (d = o), d; }; if (!n.grabInteraction) { let o = null, d = null; this._options.useUtilityLayer && this._utilityLayerScene && (d = this._pickWithSphere(n, this._hoverRadius, this._utilityLayerScene, (l) => this._nearInteractionPredicate(l))); const v = this._pickWithSphere(n, this._hoverRadius, this._scene, (l) => this._nearInteractionPredicate(l)), u = s(v, d); if (u && u.hit && (o = a(u), o.hit && (n.hoverInteraction = !0)), n.hoverInteraction) { let l = null; const P = i ? this._pickRadius : this._controllerPickRadius; this._options.useUtilityLayer && this._utilityLayerScene && (l = this._pickWithSphere(n, P, this._utilityLayerScene, (T) => this._nearPickPredicate(T))); const p = this._pickWithSphere(n, P, this._scene, (T) => this._nearPickPredicate(T)), c = s(p, l), H = a(c); H.hit && (o = H, n.nearInteraction = !0); } n.stalePick = n.pick, n.pick = o, n.pick && n.pick.pickedPoint && n.pick.hit ? (n.meshUnderPointer = n.pick.pickedMesh, n.pickedPointVisualCue.position.copyFrom(n.pick.pickedPoint), n.pickedPointVisualCue.isVisible = !0, this._farInteractionFeature && this._farInteractionFeature.attached && this._farInteractionFeature._setPointerSelectionDisabledByPointerId(n.id, !0)) : (n.meshUnderPointer = null, n.pickedPointVisualCue.isVisible = !1, this._farInteractionFeature && this._farInteractionFeature.attached && this._farInteractionFeature._setPointerSelectionDisabledByPointerId(n.id, !1)); } let f = vc.DEHYDRATED; n.grabInteraction || n.nearInteraction ? f = vc.TOUCH : n.hoverInteraction && (f = vc.HOVER), this._handleTransitionAnimation(n, f); }); } get _utilityLayerScene() { return this._options.customUtilityLayerScene || Ds.DefaultUtilityLayer.utilityLayerScene; } _generateVisualCue() { const e = this._options.useUtilityLayer ? this._options.customUtilityLayerScene || Ds.DefaultUtilityLayer.utilityLayerScene : this._scene, t = UA("nearInteraction", { diameter: 35e-4 * 3 }, e); t.bakeCurrentTransformIntoVertices(), t.isPickable = !1, t.isVisible = !1, t.rotationQuaternion = Ze.Identity(); const r = new Wt("targetMat", e); return r.specularColor = Ne.Black(), r.emissiveColor = this.selectionMeshDefaultColor, r.backFaceCulling = !1, t.material = r, t; } _isControllerReadyForNearInteraction(e) { return this._farInteractionFeature ? this._farInteractionFeature._getPointerSelectionDisabledByPointerId(e) : !0; } _attachNearInteractionMode(e) { const t = this._controllers[e.uniqueId], r = { pointerId: t.id, pointerType: "xr-near" }; t.onFrameObserver = this._xrSessionManager.onXRFrameObservable.add(() => { !this._options.enableNearInteractionOnAllControllers && e.uniqueId !== this._attachedController || !t.xrController || !t.xrController.inputSource.hand && (!this._options.nearInteractionControllerMode || !t.xrController.inputSource.gamepad) || (t.pick && (t.pick.ray = t.grabRay), t.pick && this._isControllerReadyForNearInteraction(t.id) && this._scene.simulatePointerMove(t.pick, r), t.nearInteraction && t.pick && t.pick.hit ? t.nearInteractionTargetMesh || (this._scene.simulatePointerDown(t.pick, r), t.nearInteractionTargetMesh = t.meshUnderPointer) : t.nearInteractionTargetMesh && t.stalePick && (this._scene.simulatePointerUp(t.stalePick, r), t.nearInteractionTargetMesh = null)); }); const n = (i) => { this._options.enableNearInteractionOnAllControllers || e.uniqueId === this._attachedController && this._isControllerReadyForNearInteraction(t.id) ? (t.pick && (t.pick.ray = t.grabRay), i && t.pick && t.meshUnderPointer && this._nearGrabPredicate(t.meshUnderPointer) ? (t.grabInteraction = !0, t.pickedPointVisualCue.isVisible = !1, this._scene.simulatePointerDown(t.pick, r)) : !i && t.pick && t.grabInteraction && (this._scene.simulatePointerUp(t.pick, r), t.grabInteraction = !1, t.pickedPointVisualCue.isVisible = !0)) : i && !this._options.enableNearInteractionOnAllControllers && !this._options.disableSwitchOnClick && (this._attachedController = e.uniqueId); }; if (e.inputSource.gamepad) { const i = (s) => { t.squeezeComponent = s.getComponent("grasp"), t.squeezeComponent ? t.onSqueezeButtonChangedObserver = t.squeezeComponent.onButtonStateChangedObservable.add((a) => { if (a.changes.pressed) { const f = a.changes.pressed.current; n(f); } }) : (t.selectionComponent = s.getMainComponent(), t.onButtonChangedObserver = t.selectionComponent.onButtonStateChangedObservable.add((a) => { if (a.changes.pressed) { const f = a.changes.pressed.current; n(f); } })); }; e.motionController ? i(e.motionController) : e.onMotionControllerInitObservable.add(i); } else { const i = (a) => { t.xrController && a.inputSource === t.xrController.inputSource && t.pick && this._isControllerReadyForNearInteraction(t.id) && t.meshUnderPointer && this._nearGrabPredicate(t.meshUnderPointer) && (t.grabInteraction = !0, t.pickedPointVisualCue.isVisible = !1, this._scene.simulatePointerDown(t.pick, r)); }, s = (a) => { t.xrController && a.inputSource === t.xrController.inputSource && t.pick && this._isControllerReadyForNearInteraction(t.id) && (this._scene.simulatePointerUp(t.pick, r), t.grabInteraction = !1, t.pickedPointVisualCue.isVisible = !0); }; t.eventListeners = { selectend: s, selectstart: i }, this._xrSessionManager.session.addEventListener("selectstart", i), this._xrSessionManager.session.addEventListener("selectend", s); } } _detachController(e) { const t = this._controllers[e]; if (t && (t.squeezeComponent && t.onSqueezeButtonChangedObserver && t.squeezeComponent.onButtonStateChangedObservable.remove(t.onSqueezeButtonChangedObserver), t.selectionComponent && t.onButtonChangedObserver && t.selectionComponent.onButtonStateChangedObservable.remove(t.onButtonChangedObserver), t.onFrameObserver && this._xrSessionManager.onXRFrameObservable.remove(t.onFrameObserver), t.eventListeners && Object.keys(t.eventListeners).forEach((r) => { const n = t.eventListeners && t.eventListeners[r]; n && this._xrSessionManager.session.removeEventListener(r, n); }), t.touchCollisionMesh.dispose(), t.pickedPointVisualCue.dispose(), this._xrSessionManager.runInXRFrame(() => { const r = { pointerId: t.id, pointerType: "xr-near" }; this._scene.simulatePointerUp(new F9(), r); }), delete this._controllers[e], this._attachedController === e)) { const r = Object.keys(this._controllers); r.length ? this._attachedController = r[0] : this._attachedController = ""; } } _generateNewTouchPointMesh() { const e = this._options.useUtilityLayer ? this._options.customUtilityLayerScene || Ds.DefaultUtilityLayer.utilityLayerScene : this._scene, t = UA("PickSphere", { diameter: 1 }, e); t.isVisible = !1, this._options.motionControllerOrbMaterial ? t.material = this._options.motionControllerOrbMaterial : ja.ParseFromSnippetAsync("8RUNKL#3", e).then((m) => { t.material = m; }); const r = new _N(); r.setEasingMode(u1.EASINGMODE_EASEINOUT); const n = new S(this._controllerPickRadius, this._controllerPickRadius, this._controllerPickRadius), i = this._controllerPickRadius * (4 / 3), s = new S(i, i, i), a = this._controllerPickRadius * (7 / 6), f = new S(a, a, a), o = this._controllerPickRadius * (4 / 5), d = new S(o, o, o), v = this._controllerPickRadius * (3 / 2), u = new S(v, v, v), l = [ { frame: 0, value: n }, { frame: 10, value: u }, { frame: 18, value: s } ], P = [ { frame: 0, value: s }, { frame: 10, value: d }, { frame: 18, value: n } ], p = [ { frame: 0, value: S.ZeroReadOnly }, { frame: 12, value: f }, { frame: 15, value: n } ], c = [ { frame: 0, value: n }, { frame: 10, value: S.ZeroReadOnly }, { frame: 15, value: S.ZeroReadOnly } ], H = new st("touch", "scaling", 60, st.ANIMATIONTYPE_VECTOR3, st.ANIMATIONLOOPMODE_CONSTANT), T = new st("release", "scaling", 60, st.ANIMATIONTYPE_VECTOR3, st.ANIMATIONLOOPMODE_CONSTANT), q = new st("hydrate", "scaling", 60, st.ANIMATIONTYPE_VECTOR3, st.ANIMATIONLOOPMODE_CONSTANT), b = new st("dehydrate", "scaling", 60, st.ANIMATIONTYPE_VECTOR3, st.ANIMATIONLOOPMODE_CONSTANT); return H.setEasingFunction(r), T.setEasingFunction(r), q.setEasingFunction(r), b.setEasingFunction(r), H.setKeys(l), T.setKeys(P), q.setKeys(p), b.setKeys(c), { touchCollisionMesh: t, touchCollisionMeshFunction: (m) => { const I = m ? H : T; e.beginDirectAnimation(t, [I], 0, 18, !1, 1); }, hydrateCollisionMeshFunction: (m) => { const I = m ? q : b; m && (t.isVisible = !0), e.beginDirectAnimation(t, [I], 0, 15, !1, 1, () => { m || (t.isVisible = !1); }); } }; } _pickWithSphere(e, t, r, n) { const i = new F9(); if (i.distance = 1 / 0, e.touchCollisionMesh && e.xrController) { const s = e.touchCollisionMesh.position, a = rg.CreateFromCenterAndRadius(s, t); for (let f = 0; f < r.meshes.length; f++) { const o = r.meshes[f]; if (!n(o) || !this._controllerAvailablePredicate(o, e.xrController.uniqueId)) continue; const d = sg.PickMeshWithSphere(o, a); d && d.hit && d.distance < i.distance && (i.hit = d.hit, i.pickedMesh = o, i.pickedPoint = d.pickedPoint, i.aimTransform = e.xrController.pointer, i.gripTransform = e.xrController.grip || null, i.originMesh = e.touchCollisionMesh, i.distance = d.distance); } } return i; } /** * Picks a mesh with a sphere * @param mesh the mesh to pick * @param sphere picking sphere in world coordinates * @param skipBoundingInfo a boolean indicating if we should skip the bounding info check * @returns the picking info */ static PickMeshWithSphere(e, t, r = !1) { const n = e.subMeshes, i = new F9(), s = e.getBoundingInfo(); if (!e._generatePointsArray() || !e.subMeshes || !s || !r && !rg.Intersects(s.boundingSphere, t)) return i; const a = ue.Vector3[0], f = ue.Vector3[1]; let o = 1 / 0, d, v, u; const l = ue.Vector3[2], P = ue.Matrix[0]; P.copyFrom(e.getWorldMatrix()), P.invert(), S.TransformCoordinatesToRef(t.center, P, l); for (let p = 0; p < n.length; p++) n[p].projectToRef(l, e._positions, e.getIndices(), f), S.TransformCoordinatesToRef(f, e.getWorldMatrix(), f), d = S.Distance(f, t.center), u = S.Distance(f, e.getAbsolutePosition()), v = S.Distance(t.center, e.getAbsolutePosition()), v !== -1 && u !== -1 && u > v && (d = 0, f.copyFrom(t.center)), d !== -1 && d < o && (o = d, a.copyFrom(f)); return o < t.radius && (i.hit = !0, i.distance = o, i.pickedMesh = e, i.pickedPoint = a.clone()), i; } } sg._IdCounter = 200; sg.Name = Gi.NEAR_INTERACTION; sg.Version = 1; So.AddWebXRFeature(sg.Name, (A, e) => () => new sg(A, e), sg.Version, !0); class yte { /** * Creates a WebXREnterExitUIButton * @param element button element * @param sessionMode XR initialization session mode * @param referenceSpaceType the type of reference space to be used */ constructor(e, t, r) { this.element = e, this.sessionMode = t, this.referenceSpaceType = r; } /** * Extendable function which can be used to update the button's visuals when the state changes * @param activeButton the current active button in the UI */ // eslint-disable-next-line @typescript-eslint/no-unused-vars update(e) { } } class l3e { } class dy { /** * Construct a new EnterExit UI class * * @param _scene babylon scene object to use * @param options (read-only) version of the options passed to this UI */ constructor(e, t) { if (this._scene = e, this.options = t, this._activeButton = null, this._buttons = [], this.activeButtonChangedObservable = new Oe(), this._onSessionGranted = (n) => { this._helper && this._enterXRWithButtonIndex(0); }, this.overlay = document.createElement("div"), this.overlay.classList.add("xr-button-overlay"), !t.ignoreSessionGrantedEvent && navigator.xr && navigator.xr.addEventListener("sessiongranted", this._onSessionGranted), typeof window < "u" && window.location && window.location.protocol === "http:" && window.location.hostname !== "localhost") throw ye.Warn("WebXR can only be served over HTTPS"), new Error("WebXR can only be served over HTTPS"); if (t.customButtons) this._buttons = t.customButtons; else { this.overlay.style.cssText = "z-index:11;position: absolute; right: 20px;bottom: 50px;"; const n = t.sessionMode || "immersive-vr", i = t.referenceSpaceType || "local-floor"; let a = ".babylonVRicon { color: #868686; border-color: #868686; border-style: solid; margin-left: 10px; height: 50px; width: 80px; background-color: rgba(51,51,51,0.7); background-image: url(" + (typeof SVGSVGElement > "u" ? "https://cdn.babylonjs.com/Assets/vrButton.png" : "data:image/svg+xml;charset=UTF-8,%3Csvg%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%20width%3D%222048%22%20height%3D%221152%22%20viewBox%3D%220%200%202048%201152%22%20version%3D%221.1%22%3E%3Cpath%20transform%3D%22rotate%28180%201024%2C576.0000000000001%29%22%20d%3D%22m1109%2C896q17%2C0%2030%2C-12t13%2C-30t-12.5%2C-30.5t-30.5%2C-12.5l-170%2C0q-18%2C0%20-30.5%2C12.5t-12.5%2C30.5t13%2C30t30%2C12l170%2C0zm-85%2C256q59%2C0%20132.5%2C-1.5t154.5%2C-5.5t164.5%2C-11.5t163%2C-20t150%2C-30t124.5%2C-41.5q23%2C-11%2042%2C-24t38%2C-30q27%2C-25%2041%2C-61.5t14%2C-72.5l0%2C-257q0%2C-123%20-47%2C-232t-128%2C-190t-190%2C-128t-232%2C-47l-81%2C0q-37%2C0%20-68.5%2C14t-60.5%2C34.5t-55.5%2C45t-53%2C45t-53%2C34.5t-55.5%2C14t-55.5%2C-14t-53%2C-34.5t-53%2C-45t-55.5%2C-45t-60.5%2C-34.5t-68.5%2C-14l-81%2C0q-123%2C0%20-232%2C47t-190%2C128t-128%2C190t-47%2C232l0%2C257q0%2C68%2038%2C115t97%2C73q54%2C24%20124.5%2C41.5t150%2C30t163%2C20t164.5%2C11.5t154.5%2C5.5t132.5%2C1.5zm939%2C-298q0%2C39%20-24.5%2C67t-58.5%2C42q-54%2C23%20-122%2C39.5t-143.5%2C28t-155.5%2C19t-157%2C11t-148.5%2C5t-129.5%2C1.5q-59%2C0%20-130%2C-1.5t-148%2C-5t-157%2C-11t-155.5%2C-19t-143.5%2C-28t-122%2C-39.5q-34%2C-14%20-58.5%2C-42t-24.5%2C-67l0%2C-257q0%2C-106%2040.5%2C-199t110%2C-162.5t162.5%2C-109.5t199%2C-40l81%2C0q27%2C0%2052%2C14t50%2C34.5t51%2C44.5t55.5%2C44.5t63.5%2C34.5t74%2C14t74%2C-14t63.5%2C-34.5t55.5%2C-44.5t51%2C-44.5t50%2C-34.5t52%2C-14l14%2C0q37%2C0%2070%2C0.5t64.5%2C4.5t63.5%2C12t68%2C23q71%2C30%20128.5%2C78.5t98.5%2C110t63.5%2C133.5t22.5%2C149l0%2C257z%22%20fill%3D%22white%22%20/%3E%3C/svg%3E%0A") + "); background-size: 80%; background-repeat:no-repeat; background-position: center; border: none; outline: none; transition: transform 0.125s ease-out } .babylonVRicon:hover { transform: scale(1.05) } .babylonVRicon:active {background-color: rgba(51,51,51,1) } .babylonVRicon:focus {background-color: rgba(51,51,51,1) }"; a += '.babylonVRicon.vrdisplaypresenting { background-image: none;} .vrdisplaypresenting::after { content: "EXIT"} .xr-error::after { content: "ERROR"}'; const f = document.createElement("style"); f.appendChild(document.createTextNode(a)), document.getElementsByTagName("head")[0].appendChild(f); const o = document.createElement("button"); o.className = "babylonVRicon", o.title = `${n} - ${i}`, this._buttons.push(new yte(o, n, i)), this._buttons[this._buttons.length - 1].update = function(d) { this.element.style.display = d === null || d === this ? "" : "none", o.className = "babylonVRicon" + (d === this ? " vrdisplaypresenting" : ""); }, this._updateButtons(null); } const r = e.getEngine().getInputElement(); r && r.parentNode && (r.parentNode.appendChild(this.overlay), e.onDisposeObservable.addOnce(() => { this.dispose(); })); } /** * Set the helper to be used with this UI component. * The UI is bound to an experience helper. If not provided the UI can still be used but the events should be registered by the developer. * * @param helper the experience helper to attach * @param renderTarget an optional render target (in case it is created outside of the helper scope) * @returns a promise that resolves when the ui is ready */ async setHelperAsync(e, t) { this._helper = e, this._renderTarget = t; const r = this._buttons.map((i) => e.sessionManager.isSessionSupportedAsync(i.sessionMode)); e.onStateChangedObservable.add((i) => { i == d9.NOT_IN_XR && this._updateButtons(null); }), (await Promise.all(r)).forEach((i, s) => { i ? (this.overlay.appendChild(this._buttons[s].element), this._buttons[s].element.onclick = this._enterXRWithButtonIndex.bind(this, s)) : ye.Warn(`Session mode "${this._buttons[s].sessionMode}" not supported in browser`); }); } /** * Creates UI to allow the user to enter/exit XR mode * @param scene the scene to add the ui to * @param helper the xr experience helper to enter/exit xr with * @param options options to configure the UI * @returns the created ui */ static async CreateAsync(e, t, r) { const n = new dy(e, r); return await n.setHelperAsync(t, r.renderTarget || void 0), n; } async _enterXRWithButtonIndex(e = 0) { if (this._helper.state == d9.IN_XR) await this._helper.exitXRAsync(), this._updateButtons(null); else if (this._helper.state == d9.NOT_IN_XR) try { await this._helper.enterXRAsync(this._buttons[e].sessionMode, this._buttons[e].referenceSpaceType, this._renderTarget, { optionalFeatures: this.options.optionalFeatures, requiredFeatures: this.options.requiredFeatures }), this._updateButtons(this._buttons[e]); } catch (t) { this._updateButtons(null); const r = this._buttons[e].element, n = r.title; r.title = "Error entering XR session : " + n, r.classList.add("xr-error"), this.options.onError && this.options.onError(t); } } /** * Disposes of the XR UI component */ dispose() { const e = this._scene.getEngine().getInputElement(); e && e.parentNode && e.parentNode.contains(this.overlay) && e.parentNode.removeChild(this.overlay), this.activeButtonChangedObservable.clear(), navigator.xr.removeEventListener("sessiongranted", this._onSessionGranted); } _updateButtons(e) { this._activeButton = e, this._buttons.forEach((t) => { t.update(this._activeButton); }), this.activeButtonChangedObservable.notifyObservers(this._activeButton); } } var Gx; (function(A) { A[A.INIT = 0] = "INIT", A[A.STARTED = 1] = "STARTED", A[A.ENDED = 2] = "ENDED"; })(Gx || (Gx = {})); function HF(A) { var e; let t = 0; const r = Date.now(); A.observableParameters = (e = A.observableParameters) !== null && e !== void 0 ? e : {}; const n = A.contextObservable.add((i) => { const s = Date.now(); t = s - r; const a = { startTime: r, currentTime: s, deltaTime: t, completeRate: t / A.timeout, payload: i }; A.onTick && A.onTick(a), A.breakCondition && A.breakCondition() && (A.contextObservable.remove(n), A.onAborted && A.onAborted(a)), t >= A.timeout && (A.contextObservable.remove(n), A.onEnded && A.onEnded(a)); }, A.observableParameters.mask, A.observableParameters.insertFirst, A.observableParameters.scope); return n; } class kte { /** * Will construct a new advanced timer based on the options provided. Timer will not start until start() is called. * @param options construction options for this advanced timer */ constructor(e) { var t, r; this.onEachCountObservable = new Oe(), this.onTimerAbortedObservable = new Oe(), this.onTimerEndedObservable = new Oe(), this.onStateChangedObservable = new Oe(), this._observer = null, this._breakOnNextTick = !1, this._tick = (n) => { const i = Date.now(); this._timer = i - this._startTime; const s = { startTime: this._startTime, currentTime: i, deltaTime: this._timer, completeRate: this._timer / this._timeToEnd, payload: n }, a = this._breakOnNextTick || this._breakCondition(s); a || this._timer >= this._timeToEnd ? this._stop(s, a) : this.onEachCountObservable.notifyObservers(s); }, this._setState(Gx.INIT), this._contextObservable = e.contextObservable, this._observableParameters = (t = e.observableParameters) !== null && t !== void 0 ? t : {}, this._breakCondition = (r = e.breakCondition) !== null && r !== void 0 ? r : () => !1, this._timeToEnd = e.timeout, e.onEnded && this.onTimerEndedObservable.add(e.onEnded), e.onTick && this.onEachCountObservable.add(e.onTick), e.onAborted && this.onTimerAbortedObservable.add(e.onAborted); } /** * set a breaking condition for this timer. Default is to never break during count * @param predicate the new break condition. Returns true to break, false otherwise */ set breakCondition(e) { this._breakCondition = e; } /** * Reset ALL associated observables in this advanced timer */ clearObservables() { this.onEachCountObservable.clear(), this.onTimerAbortedObservable.clear(), this.onTimerEndedObservable.clear(), this.onStateChangedObservable.clear(); } /** * Will start a new iteration of this timer. Only one instance of this timer can run at a time. * * @param timeToEnd how much time to measure until timer ended */ start(e = this._timeToEnd) { if (this._state === Gx.STARTED) throw new Error("Timer already started. Please stop it before starting again"); this._timeToEnd = e, this._startTime = Date.now(), this._timer = 0, this._observer = this._contextObservable.add(this._tick, this._observableParameters.mask, this._observableParameters.insertFirst, this._observableParameters.scope), this._setState(Gx.STARTED); } /** * Will force a stop on the next tick. */ stop() { this._state === Gx.STARTED && (this._breakOnNextTick = !0); } /** * Dispose this timer, clearing all resources */ dispose() { this._observer && this._contextObservable.remove(this._observer), this.clearObservables(); } _setState(e) { this._state = e, this.onStateChangedObservable.notifyObservers(this._state); } _stop(e, t = !1) { this._contextObservable.remove(this._observer), this._setState(Gx.ENDED), t ? this.onTimerAbortedObservable.notifyObservers(e) : this.onTimerEndedObservable.notifyObservers(e); } } class nm extends L9 { /** * Is rotation enabled when moving forward? * Disabling this feature will prevent the user from deciding the direction when teleporting */ get rotationEnabled() { return this._rotationEnabled; } /** * Sets whether rotation is enabled or not * @param enabled is rotation enabled when teleportation is shown */ set rotationEnabled(e) { if (this._rotationEnabled = e, this._options.teleportationTargetMesh) { const t = this._options.teleportationTargetMesh.getChildMeshes(!1, (r) => r.name === "rotationCone"); t[0] && t[0].setEnabled(e); } } /** * Exposes the currently set teleportation target mesh. */ get teleportationTargetMesh() { return this._options.teleportationTargetMesh || null; } /** * constructs a new teleportation system * @param _xrSessionManager an instance of WebXRSessionManager * @param _options configuration object for this feature */ constructor(e, t) { super(e), this._options = t, this._controllers = {}, this._snappedToPoint = !1, this._cachedColor4White = new xt(1, 1, 1, 1), this._tmpRay = new Hi(new S(), new S()), this._tmpVector = new S(), this._tmpQuaternion = new Ze(), this.skipNextTeleportation = !1, this.backwardsMovementEnabled = !0, this.backwardsTeleportationDistance = 0.7, this.parabolicCheckRadius = 5, this.parabolicRayEnabled = !0, this.straightRayEnabled = !0, this.rotationAngle = Math.PI / 8, this.onTargetMeshPositionUpdatedObservable = new Oe(), this.teleportationEnabled = !0, this._rotationEnabled = !0, this._attachController = (r) => { if (this._controllers[r.uniqueId] || this._options.forceHandedness && r.inputSource.handedness !== this._options.forceHandedness) return; this._controllers[r.uniqueId] = { xrController: r, teleportationState: { forward: !1, backwards: !1, rotating: !1, currentRotation: 0, baseRotation: 0, blocked: !1 } }; const n = this._controllers[r.uniqueId]; if (n.xrController.inputSource.targetRayMode === "tracked-pointer" && n.xrController.inputSource.gamepad) { const i = () => { if (r.motionController) { const s = r.motionController.getComponentOfType(Jl.THUMBSTICK_TYPE) || r.motionController.getComponentOfType(Jl.TOUCHPAD_TYPE); if (!s || this._options.useMainComponentOnly) { const a = r.motionController.getMainComponent(); if (!a) return; n.teleportationComponent = a, n.onButtonChangedObserver = a.onButtonStateChangedObservable.add(() => { if (this.teleportationEnabled && a.changes.pressed) if (a.changes.pressed.current) { n.teleportationState.forward = !0, this._currentTeleportationControllerId = n.xrController.uniqueId, n.teleportationState.baseRotation = this._options.xrInput.xrCamera.rotationQuaternion.toEulerAngles().y, n.teleportationState.currentRotation = 0; const f = this._options.timeToTeleport || 3e3; HF({ timeout: f, contextObservable: this._xrSessionManager.onXRFrameObservable, breakCondition: () => !a.pressed, onEnded: () => { this._currentTeleportationControllerId === n.xrController.uniqueId && n.teleportationState.forward && this._teleportForward(r.uniqueId); } }); } else n.teleportationState.forward = !1, this._currentTeleportationControllerId = ""; }); } else n.teleportationComponent = s, n.onAxisChangedObserver = s.onAxisValueChangedObservable.add((a) => { if (a.y <= 0.7 && n.teleportationState.backwards && (n.teleportationState.backwards = !1), a.y > 0.7 && !n.teleportationState.forward && this.backwardsMovementEnabled && !this.snapPointsOnly && !n.teleportationState.backwards) { n.teleportationState.backwards = !0, this._tmpQuaternion.copyFrom(this._options.xrInput.xrCamera.rotationQuaternion), this._tmpQuaternion.toEulerAnglesToRef(this._tmpVector), this._tmpVector.x = 0, this._tmpVector.z = 0, Ze.FromEulerVectorToRef(this._tmpVector, this._tmpQuaternion), this._tmpVector.set(0, 0, this.backwardsTeleportationDistance * (this._xrSessionManager.scene.useRightHandedSystem ? 1 : -1)), this._tmpVector.rotateByQuaternionToRef(this._tmpQuaternion, this._tmpVector), this._tmpVector.addInPlace(this._options.xrInput.xrCamera.position), this._tmpRay.origin.copyFrom(this._tmpVector), this._tmpRay.length = this._options.xrInput.xrCamera.realWorldHeight + 0.1, this._tmpRay.direction.set(0, -1, 0); const f = this._xrSessionManager.scene.pickWithRay(this._tmpRay, (o) => this._floorMeshes.indexOf(o) !== -1); f && f.pickedPoint && (this._options.xrInput.xrCamera.position.x = f.pickedPoint.x, this._options.xrInput.xrCamera.position.z = f.pickedPoint.z); } if (a.y < -0.7 && !this._currentTeleportationControllerId && !n.teleportationState.rotating && this.teleportationEnabled && (n.teleportationState.forward = !0, this._currentTeleportationControllerId = n.xrController.uniqueId, n.teleportationState.baseRotation = this._options.xrInput.xrCamera.rotationQuaternion.toEulerAngles().y), a.x) { if (n.teleportationState.forward) this._currentTeleportationControllerId === n.xrController.uniqueId && (this.rotationEnabled ? setTimeout(() => { n.teleportationState.currentRotation = Math.atan2(a.x, a.y * (this._xrSessionManager.scene.useRightHandedSystem ? 1 : -1)); }) : n.teleportationState.currentRotation = 0); else if (!n.teleportationState.rotating && Math.abs(a.x) > 0.7) { n.teleportationState.rotating = !0; const f = this.rotationAngle * (a.x > 0 ? 1 : -1) * (this._xrSessionManager.scene.useRightHandedSystem ? -1 : 1); Ze.FromEulerAngles(0, f, 0).multiplyToRef(this._options.xrInput.xrCamera.rotationQuaternion, this._options.xrInput.xrCamera.rotationQuaternion); } } else n.teleportationState.rotating = !1; a.x === 0 && a.y === 0 && (n.teleportationState.blocked && (n.teleportationState.blocked = !1, this._setTargetMeshVisibility(!1)), n.teleportationState.forward && this._teleportForward(r.uniqueId)); }); } }; r.motionController ? i() : r.onMotionControllerInitObservable.addOnce(() => { i(); }); } else this._xrSessionManager.scene.onPointerObservable.add((i) => { if (i.type === ir.POINTERDOWN) { n.teleportationState.forward = !0, this._currentTeleportationControllerId = n.xrController.uniqueId, n.teleportationState.baseRotation = this._options.xrInput.xrCamera.rotationQuaternion.toEulerAngles().y, n.teleportationState.currentRotation = 0; const s = this._options.timeToTeleport || 3e3; HF({ timeout: s, contextObservable: this._xrSessionManager.onXRFrameObservable, onEnded: () => { this._currentTeleportationControllerId === n.xrController.uniqueId && n.teleportationState.forward && this._teleportForward(r.uniqueId); } }); } else i.type === ir.POINTERUP && (n.teleportationState.forward = !1, this._currentTeleportationControllerId = ""); }); }, this._options.teleportationTargetMesh || this._createDefaultTargetMesh(), this._floorMeshes = this._options.floorMeshes || [], this._snapToPositions = this._options.snapPositions || [], this._blockedRayColor = this._options.blockedRayColor || new xt(1, 0, 0, 0.75), this._setTargetMeshVisibility(!1); } /** * Get the snapPointsOnly flag */ get snapPointsOnly() { return !!this._options.snapPointsOnly; } /** * Sets the snapPointsOnly flag * @param snapToPoints should teleportation be exclusively to snap points */ set snapPointsOnly(e) { this._options.snapPointsOnly = e; } /** * Add a new mesh to the floor meshes array * @param mesh the mesh to use as floor mesh */ addFloorMesh(e) { this._floorMeshes.push(e); } /** * Add a mesh to the list of meshes blocking the teleportation ray * @param mesh The mesh to add to the teleportation-blocking meshes */ addBlockerMesh(e) { this._options.pickBlockerMeshes = this._options.pickBlockerMeshes || [], this._options.pickBlockerMeshes.push(e); } /** * Add a new snap-to point to fix teleportation to this position * @param newSnapPoint The new Snap-To point */ addSnapPoint(e) { this._snapToPositions.push(e); } attach() { return super.attach() ? (this._currentTeleportationControllerId = "", this._options.xrInput.controllers.forEach(this._attachController), this._addNewAttachObserver(this._options.xrInput.onControllerAddedObservable, this._attachController), this._addNewAttachObserver(this._options.xrInput.onControllerRemovedObservable, (e) => { this._detachController(e.uniqueId); }), !0) : !1; } detach() { return super.detach() ? (Object.keys(this._controllers).forEach((e) => { this._detachController(e); }), this._setTargetMeshVisibility(!1), this._currentTeleportationControllerId = "", this._controllers = {}, !0) : !1; } dispose() { super.dispose(), this._options.teleportationTargetMesh && this._options.teleportationTargetMesh.dispose(!1, !0); } /** * Remove a mesh from the floor meshes array * @param mesh the mesh to remove */ removeFloorMesh(e) { const t = this._floorMeshes.indexOf(e); t !== -1 && this._floorMeshes.splice(t, 1); } /** * Remove a mesh from the blocker meshes array * @param mesh the mesh to remove */ removeBlockerMesh(e) { this._options.pickBlockerMeshes = this._options.pickBlockerMeshes || []; const t = this._options.pickBlockerMeshes.indexOf(e); t !== -1 && this._options.pickBlockerMeshes.splice(t, 1); } /** * Remove a mesh from the floor meshes array using its name * @param name the mesh name to remove */ removeFloorMeshByName(e) { const t = this._xrSessionManager.scene.getMeshByName(e); t && this.removeFloorMesh(t); } /** * This function will iterate through the array, searching for this point or equal to it. It will then remove it from the snap-to array * @param snapPointToRemove the point (or a clone of it) to be removed from the array * @returns was the point found and removed or not */ removeSnapPoint(e) { let t = this._snapToPositions.indexOf(e); if (t === -1) { for (let r = 0; r < this._snapToPositions.length; ++r) if (this._snapToPositions[r].equals(e)) { t = r; break; } } return t !== -1 ? (this._snapToPositions.splice(t, 1), !0) : !1; } /** * This function sets a selection feature that will be disabled when * the forward ray is shown and will be reattached when hidden. * This is used to remove the selection rays when moving. * @param selectionFeature the feature to disable when forward movement is enabled */ setSelectionFeature(e) { this._selectionFeature = e; } _onXRFrame(e) { const t = this._xrSessionManager.currentFrame, r = this._xrSessionManager.scene; if (!this.attach || !t) return; const n = this._options.teleportationTargetMesh; if (this._currentTeleportationControllerId) { if (!n) return; n.rotationQuaternion = n.rotationQuaternion || new Ze(); const i = this._controllers[this._currentTeleportationControllerId]; if (i && i.teleportationState.forward) { Ze.RotationYawPitchRollToRef(i.teleportationState.currentRotation + i.teleportationState.baseRotation, 0, 0, n.rotationQuaternion); let s = !1; if (i.xrController.getWorldPointerRayToRef(this._tmpRay), this.straightRayEnabled) { const a = r.pickWithRay(this._tmpRay, (f) => { if (this._options.pickBlockerMeshes && this._options.pickBlockerMeshes.indexOf(f) !== -1) return !0; const o = this._floorMeshes.indexOf(f); return o === -1 ? !1 : this._floorMeshes[o].absolutePosition.y < this._options.xrInput.xrCamera.globalPosition.y; }); if (a && a.pickedMesh && this._options.pickBlockerMeshes && this._options.pickBlockerMeshes.indexOf(a.pickedMesh) !== -1) { i.teleportationState.blocked = !0, this._setTargetMeshVisibility(!1), this._showParabolicPath(a); return; } else a && a.pickedPoint && (i.teleportationState.blocked = !1, s = !0, this._setTargetMeshPosition(a), this._setTargetMeshVisibility(!0), this._showParabolicPath(a)); } if (this.parabolicRayEnabled && !s) { const a = i.xrController.pointer.rotationQuaternion.toEulerAngles().x, f = 1 + (Math.PI / 2 - Math.abs(a)), o = this.parabolicCheckRadius * f; this._tmpRay.origin.addToRef(this._tmpRay.direction.scale(o * 2), this._tmpVector), this._tmpVector.y = this._tmpRay.origin.y, this._tmpRay.origin.addInPlace(this._tmpRay.direction.scale(o)), this._tmpVector.subtractToRef(this._tmpRay.origin, this._tmpRay.direction), this._tmpRay.direction.normalize(); const d = r.pickWithRay(this._tmpRay, (v) => this._options.pickBlockerMeshes && this._options.pickBlockerMeshes.indexOf(v) !== -1 ? !0 : this._floorMeshes.indexOf(v) !== -1); if (d && d.pickedMesh && this._options.pickBlockerMeshes && this._options.pickBlockerMeshes.indexOf(d.pickedMesh) !== -1) { i.teleportationState.blocked = !0, this._setTargetMeshVisibility(!1), this._showParabolicPath(d); return; } else d && d.pickedPoint && (i.teleportationState.blocked = !1, s = !0, this._setTargetMeshPosition(d), this._setTargetMeshVisibility(!0), this._showParabolicPath(d)); } this._setTargetMeshVisibility(s); } else this._setTargetMeshVisibility(!1); } else this._disposeBezierCurve(), this._setTargetMeshVisibility(!1); } _createDefaultTargetMesh() { this._options.defaultTargetMeshOptions = this._options.defaultTargetMeshOptions || {}; const e = this._options.useUtilityLayer ? this._options.customUtilityLayerScene || Ds.DefaultUtilityLayer.utilityLayerScene : this._xrSessionManager.scene, t = Wm("teleportationTarget", { width: 2, height: 2, subdivisions: 2 }, e); if (t.isPickable = !1, this._options.defaultTargetMeshOptions.teleportationCircleMaterial) t.material = this._options.defaultTargetMeshOptions.teleportationCircleMaterial; else { const s = new Xp("teleportationPlaneDynamicTexture", 512, e, !0); s.hasAlpha = !0; const a = s.getContext(), f = 512 / 2, o = 512 / 2, d = 200; a.beginPath(), a.arc(f, o, d, 0, 2 * Math.PI, !1), a.fillStyle = this._options.defaultTargetMeshOptions.teleportationFillColor || "#444444", a.fill(), a.lineWidth = 10, a.strokeStyle = this._options.defaultTargetMeshOptions.teleportationBorderColor || "#FFFFFF", a.stroke(), a.closePath(), s.update(); const v = new Wt("teleportationPlaneMaterial", e); v.diffuseTexture = s, t.material = v; } const r = Ag("torusTeleportation", { diameter: 0.75, thickness: 0.1, tessellation: 20 }, e); if (r.isPickable = !1, r.parent = t, !this._options.defaultTargetMeshOptions.disableAnimation) { const i = new st("animationInnerCircle", "position.y", 30, st.ANIMATIONTYPE_FLOAT, st.ANIMATIONLOOPMODE_CYCLE), s = []; s.push({ frame: 0, value: 0 }), s.push({ frame: 30, value: 0.4 }), s.push({ frame: 60, value: 0 }), i.setKeys(s); const a = new $N(); a.setEasingMode(u1.EASINGMODE_EASEINOUT), i.setEasingFunction(a), r.animations = [], r.animations.push(i), e.beginAnimation(r, 0, 60, !0); } const n = Ld("rotationCone", { diameterTop: 0, tessellation: 4 }, e); if (n.isPickable = !1, n.scaling.set(0.5, 0.12, 0.2), n.rotate(bf.X, Math.PI / 2), n.position.z = 0.6, n.parent = r, this._options.defaultTargetMeshOptions.torusArrowMaterial) r.material = this._options.defaultTargetMeshOptions.torusArrowMaterial, n.material = this._options.defaultTargetMeshOptions.torusArrowMaterial; else { const i = new Wt("torusConsMat", e); i.disableLighting = !!this._options.defaultTargetMeshOptions.disableLighting, i.disableLighting ? i.emissiveColor = new Ne(0.3, 0.3, 1) : i.diffuseColor = new Ne(0.3, 0.3, 1), i.alpha = 0.9, r.material = i, n.material = i, this._teleportationRingMaterial = i; } this._options.renderingGroupId !== void 0 && (t.renderingGroupId = this._options.renderingGroupId, r.renderingGroupId = this._options.renderingGroupId, n.renderingGroupId = this._options.renderingGroupId), this._options.teleportationTargetMesh = t, this._setTargetMeshVisibility(!1); } _detachController(e) { const t = this._controllers[e]; t && (t.teleportationComponent && (t.onAxisChangedObserver && t.teleportationComponent.onAxisValueChangedObservable.remove(t.onAxisChangedObserver), t.onButtonChangedObserver && t.teleportationComponent.onButtonStateChangedObservable.remove(t.onButtonChangedObserver)), delete this._controllers[e]); } _findClosestSnapPointWithRadius(e, t = this._options.snapToPositionRadius || 0.8) { let r = null, n = Number.MAX_VALUE; if (this._snapToPositions.length) { const i = t * t; this._snapToPositions.forEach((s) => { const a = S.DistanceSquared(s, e); a <= i && a < n && (n = a, r = s); }); } return r; } _setTargetMeshPosition(e) { const t = e.pickedPoint; if (!this._options.teleportationTargetMesh || !t) return; const r = this._findClosestSnapPointWithRadius(t); this._snappedToPoint = !!r, this.snapPointsOnly && !this._snappedToPoint && this._teleportationRingMaterial ? this._teleportationRingMaterial.diffuseColor.set(1, 0.3, 0.3) : this.snapPointsOnly && this._snappedToPoint && this._teleportationRingMaterial && this._teleportationRingMaterial.diffuseColor.set(0.3, 0.3, 1), this._options.teleportationTargetMesh.position.copyFrom(r || t), this._options.teleportationTargetMesh.position.y += 0.01, this.onTargetMeshPositionUpdatedObservable.notifyObservers(e); } _setTargetMeshVisibility(e, t) { this._options.teleportationTargetMesh && (this._options.teleportationTargetMesh.isVisible === e && !t || (this._options.teleportationTargetMesh.isVisible = e, this._options.teleportationTargetMesh.getChildren(void 0, !1).forEach((r) => { r.isVisible = e; }), e ? this._selectionFeature && this._selectionFeature.detach() : (this._quadraticBezierCurve && (this._quadraticBezierCurve.dispose(), this._quadraticBezierCurve = null), this._selectionFeature && this._selectionFeature.attach()))); } _disposeBezierCurve() { this._quadraticBezierCurve && (this._quadraticBezierCurve.dispose(), this._quadraticBezierCurve = null); } _showParabolicPath(e) { if (!e.pickedPoint || !this._currentTeleportationControllerId) return; const t = this._options.useUtilityLayer ? this._options.customUtilityLayerScene || Ds.DefaultUtilityLayer.utilityLayerScene : this._xrSessionManager.scene, r = this._controllers[this._currentTeleportationControllerId], n = j0.CreateQuadraticBezier(r.xrController.pointer.absolutePosition, e.ray.origin, e.pickedPoint, 25), i = r.teleportationState.blocked ? this._blockedRayColor : void 0, s = new Array(26).fill(i || this._cachedColor4White); this._options.generateRayPathMesh ? this._quadraticBezierCurve = this._options.generateRayPathMesh(n.getPoints(), e) : this._quadraticBezierCurve = ka("teleportation path line", { points: n.getPoints(), instance: this._quadraticBezierCurve, updatable: !0, colors: s }, t), this._quadraticBezierCurve.isPickable = !1, this._options.renderingGroupId !== void 0 && (this._quadraticBezierCurve.renderingGroupId = this._options.renderingGroupId); } _teleportForward(e) { const t = this._controllers[e]; if (!(!t || !t.teleportationState.forward || !this.teleportationEnabled) && (t.teleportationState.forward = !1, this._currentTeleportationControllerId = "", !(this.snapPointsOnly && !this._snappedToPoint))) { if (this.skipNextTeleportation) { this.skipNextTeleportation = !1; return; } if (this._options.teleportationTargetMesh && this._options.teleportationTargetMesh.isVisible) { const r = this._options.xrInput.xrCamera.realWorldHeight; this._options.xrInput.xrCamera.onBeforeCameraTeleport.notifyObservers(this._options.xrInput.xrCamera.position), this._options.xrInput.xrCamera.position.copyFrom(this._options.teleportationTargetMesh.position), this._options.xrInput.xrCamera.position.y += r, Ze.FromEulerAngles(0, t.teleportationState.currentRotation - (this._xrSessionManager.scene.useRightHandedSystem ? Math.PI : 0), 0).multiplyToRef(this._options.xrInput.xrCamera.rotationQuaternion, this._options.xrInput.xrCamera.rotationQuaternion), this._options.xrInput.xrCamera.onAfterCameraTeleport.notifyObservers(this._options.xrInput.xrCamera.position); } } } } nm.Name = Gi.TELEPORTATION; nm.Version = 1; So.AddWebXRFeature(nm.Name, (A, e) => () => new nm(A, e), nm.Version, !0); class P3e { } class vy { constructor() { } /** * Creates the default xr experience * @param scene scene * @param options options for basic configuration * @returns resulting WebXRDefaultExperience */ static CreateAsync(e, t = {}) { const r = new vy(); if (e.onDisposeObservable.addOnce(() => { r.dispose(); }), !t.disableDefaultUI) { const n = Object.assign({ renderTarget: r.renderTarget }, t.uiOptions || {}); t.optionalFeatures && (typeof t.optionalFeatures == "boolean" ? n.optionalFeatures = ["hit-test", "anchors", "plane-detection", "hand-tracking"] : n.optionalFeatures = t.optionalFeatures), r.enterExitUI = new dy(e, n); } return Ay.CreateAsync(e).then((n) => { if (r.baseExperience = n, t.ignoreNativeCameraTransformation && (r.baseExperience.camera.compensateOnFirstFrame = !1), r.input = new Ote(n.sessionManager, n.camera, Object.assign({ controllerOptions: { renderingGroupId: t.renderingGroupId } }, t.inputOptions || {})), !t.disablePointerSelection) { const i = Object.assign(Object.assign({}, t.pointerSelectionOptions), { xrInput: r.input, renderingGroupId: t.renderingGroupId }); r.pointerSelection = r.baseExperience.featuresManager.enableFeature(ig.Name, t.useStablePlugins ? "stable" : "latest", i), t.disableTeleportation || (r.teleportation = r.baseExperience.featuresManager.enableFeature(nm.Name, t.useStablePlugins ? "stable" : "latest", Object.assign({ floorMeshes: t.floorMeshes, xrInput: r.input, renderingGroupId: t.renderingGroupId }, t.teleportationOptions)), r.teleportation.setSelectionFeature(r.pointerSelection)); } if (t.disableNearInteraction || (r.nearInteraction = r.baseExperience.featuresManager.enableFeature(sg.Name, t.useStablePlugins ? "stable" : "latest", Object.assign({ xrInput: r.input, farInteractionFeature: r.pointerSelection, renderingGroupId: t.renderingGroupId, useUtilityLayer: !0, enableNearInteractionOnAllControllers: !0 }, t.nearInteractionOptions))), r.renderTarget = r.baseExperience.sessionManager.getWebXRRenderTarget(t.outputCanvasOptions), !t.disableDefaultUI) return r.enterExitUI.setHelperAsync(r.baseExperience, r.renderTarget); }).then(() => r).catch((n) => (Se.Error("Error initializing XR"), Se.Error(n), r)); } /** * Disposes of the experience helper */ dispose() { this.baseExperience && this.baseExperience.dispose(), this.input && this.input.dispose(), this.enterExitUI && this.enterExitUI.dispose(), this.renderTarget && this.renderTarget.dispose(); } } var c3e = !0; sr.prototype.createDefaultLight = function(A = !1) { if (A && this.lights) for (let e = 0; e < this.lights.length; e++) this.lights[e].dispose(); this.lights.length === 0 && new y0("default light", S.Up(), this); }; sr.prototype.createDefaultCamera = function(A = !1, e = !1, t = !1) { if (e && this.activeCamera && (this.activeCamera.dispose(), this.activeCamera = null), !this.activeCamera) { const r = this.getWorldExtends((f) => f.isVisible && f.isEnabled()), n = r.max.subtract(r.min), i = r.min.add(n.scale(0.5)); let s, a = n.length() * 1.5; if (isFinite(a) || (a = 1, i.copyFromFloats(0, 0, 0)), A) { const f = new ps("default camera", -(Math.PI / 2), Math.PI / 2, a, i, this); f.lowerRadiusLimit = a * 0.01, f.wheelPrecision = 100 / a, s = f; } else { const f = new SA("default camera", new S(i.x, i.y, -a), this); f.setTarget(i), s = f; } s.minZ = a * 0.01, s.maxZ = a * 1e3, s.speed = a * 0.2, this.activeCamera = s, t && s.attachControl(); } }; sr.prototype.createDefaultCameraOrLight = function(A = !1, e = !1, t = !1) { this.createDefaultLight(e), this.createDefaultCamera(A, e, t); }; sr.prototype.createDefaultSkybox = function(A, e = !1, t = 1e3, r = 0, n = !0) { if (!A) return Se.Warn("Can not create default skybox without environment texture."), null; n && A && (this.environmentTexture = A); const i = k0("hdrSkyBox", { size: t }, this); if (e) { const s = new mr("skyBox", this); s.backFaceCulling = !1, s.reflectionTexture = A.clone(), s.reflectionTexture && (s.reflectionTexture.coordinatesMode = We.SKYBOX_MODE), s.microSurface = 1 - r, s.disableLighting = !0, s.twoSidedLighting = !0, i.material = s; } else { const s = new Wt("skyBox", this); s.backFaceCulling = !1, s.reflectionTexture = A.clone(), s.reflectionTexture && (s.reflectionTexture.coordinatesMode = We.SKYBOX_MODE), s.disableLighting = !0, i.material = s; } return i.isPickable = !1, i.infiniteDistance = !0, i.ignoreCameraMaxZ = !0, i; }; sr.prototype.createDefaultEnvironment = function(A) { return gD ? new gD(A, this) : null; }; sr.prototype.createDefaultVRExperience = function(A = {}) { return new lm(this, A); }; sr.prototype.createDefaultXRExperienceAsync = function(A = {}) { return vy.CreateAsync(this, A).then((e) => e); }; function QG(A) { for (; A.firstChild; ) A.removeChild(A.firstChild); A.srcObject = null, A.src = "", A.removeAttribute("src"); } class xc extends We { /** * Event triggered when a dom action is required by the user to play the video. * This happens due to recent changes in browser policies preventing video to auto start. */ get onUserActionRequestedObservable() { return this._onUserActionRequestedObservable || (this._onUserActionRequestedObservable = new Oe()), this._onUserActionRequestedObservable; } _processError(e) { this._errorFound = !0, this._onError ? this._onError(e == null ? void 0 : e.message) : Se.Error(e == null ? void 0 : e.message); } _handlePlay() { this._errorFound = !1, this.video.play().catch((e) => { if ((e == null ? void 0 : e.name) === "NotAllowedError") { if (this._onUserActionRequestedObservable && this._onUserActionRequestedObservable.hasObservers()) { this._onUserActionRequestedObservable.notifyObservers(this); return; } else if (!this.video.muted) { Se.Warn("Unable to autoplay a video with sound. Trying again with muted turned true"), this.video.muted = !0, this._errorFound = !1, this.video.play().catch((t) => { this._processError(t); }); return; } } this._processError(e); }); } /** * Creates a video texture. * If you want to display a video in your scene, this is the special texture for that. * This special texture works similar to other textures, with the exception of a few parameters. * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/videoTexture * @param name optional name, will detect from video source, if not defined * @param src can be used to provide an url, array of urls or an already setup HTML video element. * @param scene is obviously the current scene. * @param generateMipMaps can be used to turn on mipmaps (Can be expensive for videoTextures because they are often updated). * @param invertY is false by default but can be used to invert video on Y axis * @param samplingMode controls the sampling method and is set to TRILINEAR_SAMPLINGMODE by default * @param settings allows finer control over video usage * @param onError defines a callback triggered when an error occurred during the loading session * @param format defines the texture format to use (Engine.TEXTUREFORMAT_RGBA by default) */ constructor(e, t, r, n = !1, i = !1, s = We.TRILINEAR_SAMPLINGMODE, a = {}, f, o = 5) { var d, v; super(null, r, !n, i), this._onUserActionRequestedObservable = null, this._stillImageCaptured = !1, this._displayingPosterTexture = !1, this._frameId = -1, this._currentSrc = null, this._errorFound = !1, this.isVideo = !0, this._resizeInternalTexture = () => { var l; this._texture != null && this._texture.dispose(), !this._getEngine().needPOTTextures || ye.IsExponentOfTwo(this.video.videoWidth) && ye.IsExponentOfTwo(this.video.videoHeight) ? (this.wrapU = We.WRAP_ADDRESSMODE, this.wrapV = We.WRAP_ADDRESSMODE) : (this.wrapU = We.CLAMP_ADDRESSMODE, this.wrapV = We.CLAMP_ADDRESSMODE, this._generateMipMaps = !1), this._texture = this._getEngine().createDynamicTexture(this.video.videoWidth, this.video.videoHeight, this._generateMipMaps, this.samplingMode), this._texture.format = (l = this._format) !== null && l !== void 0 ? l : 5, this._frameId = -1, this._updateInternalTexture(); }, this._createInternalTexture = () => { if (this._texture != null) if (this._displayingPosterTexture) this._displayingPosterTexture = !1; else return; if (this.video.addEventListener("resize", this._resizeInternalTexture), this._resizeInternalTexture(), !this.video.autoplay && !this._settings.poster && !this._settings.independentVideoSource) { const l = this.video.onplaying, P = this.video.muted; this.video.muted = !0, this.video.onplaying = () => { this.video.muted = P, this.video.onplaying = l, this._updateInternalTexture(), this._errorFound || this.video.pause(), this.onLoadObservable.hasObservers() && this.onLoadObservable.notifyObservers(this); }, this._handlePlay(); } else this._updateInternalTexture(), this.onLoadObservable.hasObservers() && this.onLoadObservable.notifyObservers(this); }, this._reset = () => { this._texture != null && (this._displayingPosterTexture || (this._texture.dispose(), this._texture = null)); }, this._updateInternalTexture = () => { if (this._texture == null || this.video.readyState < this.video.HAVE_CURRENT_DATA || this._displayingPosterTexture) return; const l = this.getScene().getFrameId(); this._frameId !== l && (this._frameId = l, this._getEngine().updateVideoTexture(this._texture, this._externalTexture ? this._externalTexture : this.video, this._invertY)); }, this._settings = Object.assign({ autoPlay: !0, loop: !0, autoUpdateTexture: !0 }, a), this._onError = f, this._generateMipMaps = n, this._initialSamplingMode = s, this.autoUpdateTexture = this._settings.autoUpdateTexture, this._currentSrc = t, this.name = e || this._getName(t), this.video = this._getVideo(t), this._externalTexture = (v = (d = this._engine) === null || d === void 0 ? void 0 : d.createExternalTexture(this.video)) !== null && v !== void 0 ? v : null, this._settings.independentVideoSource || (this._settings.poster && (this.video.poster = this._settings.poster), this._settings.autoPlay !== void 0 && (this.video.autoplay = this._settings.autoPlay), this._settings.loop !== void 0 && (this.video.loop = this._settings.loop), this._settings.muted !== void 0 && (this.video.muted = this._settings.muted), this.video.setAttribute("playsinline", ""), this.video.addEventListener("paused", this._updateInternalTexture), this.video.addEventListener("seeked", this._updateInternalTexture), this.video.addEventListener("emptied", this._reset), this._settings.autoPlay && this._handlePlay()), this._createInternalTextureOnEvent = this._settings.poster && !this._settings.autoPlay ? "play" : "canplay", this.video.addEventListener(this._createInternalTextureOnEvent, this._createInternalTexture), this._format = o; const u = this.video.readyState >= this.video.HAVE_CURRENT_DATA; this._settings.poster && (!this._settings.autoPlay || !u) ? (this._texture = this._getEngine().createTexture(this._settings.poster, !1, !this.invertY, r), this._displayingPosterTexture = !0) : u && this._createInternalTexture(); } /** * Get the current class name of the video texture useful for serialization or dynamic coding. * @returns "VideoTexture" */ getClassName() { return "VideoTexture"; } _getName(e) { return e instanceof HTMLVideoElement ? e.currentSrc : typeof e == "object" ? e.toString() : e; } _getVideo(e) { if (e.isNative) return e; if (e instanceof HTMLVideoElement) return ye.SetCorsBehavior(e.currentSrc, e), e; const t = document.createElement("video"); return typeof e == "string" ? (ye.SetCorsBehavior(e, t), t.src = e) : (ye.SetCorsBehavior(e[0], t), e.forEach((r) => { const n = document.createElement("source"); n.src = r, t.appendChild(n); })), this.onDisposeObservable.addOnce(() => { QG(t); }), t; } /** * @internal Internal method to initiate `update`. */ _rebuild() { this.update(); } /** * Update Texture in the `auto` mode. Does not do anything if `settings.autoUpdateTexture` is false. */ update() { this.autoUpdateTexture && this.updateTexture(!0); } /** * Update Texture in `manual` mode. Does not do anything if not visible or paused. * @param isVisible Visibility state, detected by user using `scene.getActiveMeshes()` or otherwise. */ updateTexture(e) { e && (this.video.paused && this._stillImageCaptured || (this._stillImageCaptured = !0, this._updateInternalTexture())); } /** * Get the underlying external texture (if supported by the current engine, else null) */ get externalTexture() { return this._externalTexture; } /** * Change video content. Changing video instance or setting multiple urls (as in constructor) is not supported. * @param url New url. */ updateURL(e) { this.video.src = e, this._currentSrc = e; } /** * Clones the texture. * @returns the cloned texture */ clone() { return new xc(this.name, this._currentSrc, this.getScene(), this._generateMipMaps, this.invertY, this.samplingMode, this._settings); } /** * Dispose the texture and release its associated resources. */ dispose() { var e; super.dispose(), this._currentSrc = null, this._onUserActionRequestedObservable && (this._onUserActionRequestedObservable.clear(), this._onUserActionRequestedObservable = null), this.video.removeEventListener(this._createInternalTextureOnEvent, this._createInternalTexture), this._settings.independentVideoSource || (this.video.removeEventListener("paused", this._updateInternalTexture), this.video.removeEventListener("seeked", this._updateInternalTexture), this.video.removeEventListener("emptied", this._reset), this.video.removeEventListener("resize", this._resizeInternalTexture), this.video.pause()), (e = this._externalTexture) === null || e === void 0 || e.dispose(); } /** * Creates a video texture straight from a stream. * @param scene Define the scene the texture should be created in * @param stream Define the stream the texture should be created from * @param constraints video constraints * @param invertY Defines if the video should be stored with invert Y set to true (true by default) * @returns The created video texture as a promise */ static CreateFromStreamAsync(e, t, r, n = !0) { const i = e.getEngine().createVideoElement(r); return e.getEngine()._badOS && (document.body.appendChild(i), i.style.transform = "scale(0.0001, 0.0001)", i.style.opacity = "0", i.style.position = "fixed", i.style.bottom = "0px", i.style.right = "0px"), i.setAttribute("autoplay", ""), i.setAttribute("muted", "true"), i.setAttribute("playsinline", ""), i.muted = !0, i.isNative || (i.mozSrcObject !== void 0 ? i.mozSrcObject = t : typeof i.srcObject == "object" ? i.srcObject = t : i.src = window.URL && window.URL.createObjectURL(t)), new Promise((s) => { const a = () => { const f = new xc("video", i, e, !0, n, void 0, void 0, void 0, 4); e.getEngine()._badOS && f.onDisposeObservable.addOnce(() => { i.remove(); }), f.onDisposeObservable.addOnce(() => { QG(i); }), s(f), i.removeEventListener("playing", a); }; i.addEventListener("playing", a), i.play(); }); } /** * Creates a video texture straight from your WebCam video feed. * @param scene Define the scene the texture should be created in * @param constraints Define the constraints to use to create the web cam feed from WebRTC * @param audioConstaints Define the audio constraints to use to create the web cam feed from WebRTC * @param invertY Defines if the video should be stored with invert Y set to true (true by default) * @returns The created video texture as a promise */ static async CreateFromWebCamAsync(e, t, r = !1, n = !0) { if (navigator.mediaDevices) { const i = await navigator.mediaDevices.getUserMedia({ video: t, audio: r }), s = await this.CreateFromStreamAsync(e, i, t, n); return s.onDisposeObservable.addOnce(() => { i.getTracks().forEach((a) => { a.stop(); }); }), s; } return Promise.reject("No support for userMedia on this device"); } /** * Creates a video texture straight from your WebCam video feed. * @param scene Defines the scene the texture should be created in * @param onReady Defines a callback to triggered once the texture will be ready * @param constraints Defines the constraints to use to create the web cam feed from WebRTC * @param audioConstaints Defines the audio constraints to use to create the web cam feed from WebRTC * @param invertY Defines if the video should be stored with invert Y set to true (true by default) */ static CreateFromWebCam(e, t, r, n = !1, i = !0) { this.CreateFromWebCamAsync(e, r, n, i).then(function(s) { t && t(s); }).catch(function(s) { Se.Error(s.name); }); } } C([ M("settings") ], xc.prototype, "_settings", void 0); C([ M("src") ], xc.prototype, "_currentSrc", void 0); C([ M() ], xc.prototype, "isVideo", void 0); We._CreateVideoTexture = (A, e, t, r = !1, n = !1, i = We.TRILINEAR_SAMPLINGMODE, s = {}, a, f = 5) => new xc(A, e, t, r, n, i, s, a, f); Ue("BABYLON.VideoTexture", xc); class uy extends Ou { /** * Get the video texture associated with this video dome */ get videoTexture() { return this._texture; } /** * Get the video mode of this dome */ get videoMode() { return this.textureMode; } /** * Set the video mode of this dome. * @see textureMode */ set videoMode(e) { this.textureMode = e; } _initTexture(e, t, r) { const n = { loop: r.loop, autoPlay: r.autoPlay, autoUpdateTexture: !0, poster: r.poster }, i = new xc((this.name || "videoDome") + "_texture", e, t, r.generateMipMaps, this._useDirectMapping, We.TRILINEAR_SAMPLINGMODE, n); return r.clickToPlay && (this._pointerObserver = t.onPointerObservable.add((s) => { var a; ((a = s.pickInfo) === null || a === void 0 ? void 0 : a.pickedMesh) === this.mesh && this._texture.video.play(); }, ir.POINTERDOWN)), this._textureObserver = i.onLoadObservable.add(() => { this.onLoadObservable.notifyObservers(); }), i; } /** * Releases resources associated with this node. * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default) * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default) */ dispose(e, t = !1) { this._texture.onLoadObservable.remove(this._textureObserver), this._scene.onPointerObservable.remove(this._pointerObserver), super.dispose(e, t); } } uy.MODE_MONOSCOPIC = Ou.MODE_MONOSCOPIC; uy.MODE_TOPBOTTOM = Ou.MODE_TOPBOTTOM; uy.MODE_SIDEBYSIDE = Ou.MODE_SIDEBYSIDE; class Ete { // Properties /** * Gets the perf counter used for GPU frame time */ get gpuFrameTimeCounter() { return this.engine.getGPUFrameTimeCounter(); } /** * Gets the GPU frame time capture status */ get captureGPUFrameTime() { return this._captureGPUFrameTime; } /** * Enable or disable the GPU frame time capture */ set captureGPUFrameTime(e) { e !== this._captureGPUFrameTime && (this._captureGPUFrameTime = e, this.engine.captureGPUFrameTime(e)); } /** * Gets the perf counter used for shader compilation time */ get shaderCompilationTimeCounter() { return this._shaderCompilationTime; } /** * Gets the shader compilation time capture status */ get captureShaderCompilationTime() { return this._captureShaderCompilationTime; } /** * Enable or disable the shader compilation time capture */ set captureShaderCompilationTime(e) { e !== this._captureShaderCompilationTime && (this._captureShaderCompilationTime = e, e ? (this._onBeforeShaderCompilationObserver = this.engine.onBeforeShaderCompilationObservable.add(() => { this._shaderCompilationTime.fetchNewFrame(), this._shaderCompilationTime.beginMonitoring(); }), this._onAfterShaderCompilationObserver = this.engine.onAfterShaderCompilationObservable.add(() => { this._shaderCompilationTime.endMonitoring(); })) : (this.engine.onBeforeShaderCompilationObservable.remove(this._onBeforeShaderCompilationObserver), this._onBeforeShaderCompilationObserver = null, this.engine.onAfterShaderCompilationObservable.remove(this._onAfterShaderCompilationObserver), this._onAfterShaderCompilationObserver = null)); } /** * Instantiates a new engine instrumentation. * This class can be used to get instrumentation data from a Babylon engine * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#engineinstrumentation * @param engine Defines the engine to instrument */ constructor(e) { this.engine = e, this._captureGPUFrameTime = !1, this._captureShaderCompilationTime = !1, this._shaderCompilationTime = new v9(), this._onBeginFrameObserver = null, this._onEndFrameObserver = null, this._onBeforeShaderCompilationObserver = null, this._onAfterShaderCompilationObserver = null; } /** * Dispose and release associated resources. */ dispose() { this.engine.onBeginFrameObservable.remove(this._onBeginFrameObserver), this._onBeginFrameObserver = null, this.engine.onEndFrameObservable.remove(this._onEndFrameObserver), this._onEndFrameObserver = null, this.engine.onBeforeShaderCompilationObservable.remove(this._onBeforeShaderCompilationObserver), this._onBeforeShaderCompilationObserver = null, this.engine.onAfterShaderCompilationObservable.remove(this._onAfterShaderCompilationObserver), this._onAfterShaderCompilationObserver = null, this.engine = null; } } class Fte { // Properties /** * Gets the perf counter used for active meshes evaluation time */ get activeMeshesEvaluationTimeCounter() { return this._activeMeshesEvaluationTime; } /** * Gets the active meshes evaluation time capture status */ get captureActiveMeshesEvaluationTime() { return this._captureActiveMeshesEvaluationTime; } /** * Enable or disable the active meshes evaluation time capture */ set captureActiveMeshesEvaluationTime(e) { e !== this._captureActiveMeshesEvaluationTime && (this._captureActiveMeshesEvaluationTime = e, e ? (this._onBeforeActiveMeshesEvaluationObserver = this.scene.onBeforeActiveMeshesEvaluationObservable.add(() => { ye.StartPerformanceCounter("Active meshes evaluation"), this._activeMeshesEvaluationTime.beginMonitoring(); }), this._onAfterActiveMeshesEvaluationObserver = this.scene.onAfterActiveMeshesEvaluationObservable.add(() => { ye.EndPerformanceCounter("Active meshes evaluation"), this._activeMeshesEvaluationTime.endMonitoring(!1); })) : (this.scene.onBeforeActiveMeshesEvaluationObservable.remove(this._onBeforeActiveMeshesEvaluationObserver), this._onBeforeActiveMeshesEvaluationObserver = null, this.scene.onAfterActiveMeshesEvaluationObservable.remove(this._onAfterActiveMeshesEvaluationObserver), this._onAfterActiveMeshesEvaluationObserver = null)); } /** * Gets the perf counter used for render targets render time */ get renderTargetsRenderTimeCounter() { return this._renderTargetsRenderTime; } /** * Gets the render targets render time capture status */ get captureRenderTargetsRenderTime() { return this._captureRenderTargetsRenderTime; } /** * Enable or disable the render targets render time capture */ set captureRenderTargetsRenderTime(e) { e !== this._captureRenderTargetsRenderTime && (this._captureRenderTargetsRenderTime = e, e ? (this._onBeforeRenderTargetsRenderObserver = this.scene.onBeforeRenderTargetsRenderObservable.add(() => { ye.StartPerformanceCounter("Render targets rendering"), this._renderTargetsRenderTime.beginMonitoring(); }), this._onAfterRenderTargetsRenderObserver = this.scene.onAfterRenderTargetsRenderObservable.add(() => { ye.EndPerformanceCounter("Render targets rendering"), this._renderTargetsRenderTime.endMonitoring(!1); })) : (this.scene.onBeforeRenderTargetsRenderObservable.remove(this._onBeforeRenderTargetsRenderObserver), this._onBeforeRenderTargetsRenderObserver = null, this.scene.onAfterRenderTargetsRenderObservable.remove(this._onAfterRenderTargetsRenderObserver), this._onAfterRenderTargetsRenderObserver = null)); } /** * Gets the perf counter used for particles render time */ get particlesRenderTimeCounter() { return this._particlesRenderTime; } /** * Gets the particles render time capture status */ get captureParticlesRenderTime() { return this._captureParticlesRenderTime; } /** * Enable or disable the particles render time capture */ set captureParticlesRenderTime(e) { e !== this._captureParticlesRenderTime && (this._captureParticlesRenderTime = e, e ? (this._onBeforeParticlesRenderingObserver = this.scene.onBeforeParticlesRenderingObservable.add(() => { ye.StartPerformanceCounter("Particles"), this._particlesRenderTime.beginMonitoring(); }), this._onAfterParticlesRenderingObserver = this.scene.onAfterParticlesRenderingObservable.add(() => { ye.EndPerformanceCounter("Particles"), this._particlesRenderTime.endMonitoring(!1); })) : (this.scene.onBeforeParticlesRenderingObservable.remove(this._onBeforeParticlesRenderingObserver), this._onBeforeParticlesRenderingObserver = null, this.scene.onAfterParticlesRenderingObservable.remove(this._onAfterParticlesRenderingObserver), this._onAfterParticlesRenderingObserver = null)); } /** * Gets the perf counter used for sprites render time */ get spritesRenderTimeCounter() { return this._spritesRenderTime; } /** * Gets the sprites render time capture status */ get captureSpritesRenderTime() { return this._captureSpritesRenderTime; } /** * Enable or disable the sprites render time capture */ set captureSpritesRenderTime(e) { e !== this._captureSpritesRenderTime && (this._captureSpritesRenderTime = e, this.scene.spriteManagers && (e ? (this._onBeforeSpritesRenderingObserver = this.scene.onBeforeSpritesRenderingObservable.add(() => { ye.StartPerformanceCounter("Sprites"), this._spritesRenderTime.beginMonitoring(); }), this._onAfterSpritesRenderingObserver = this.scene.onAfterSpritesRenderingObservable.add(() => { ye.EndPerformanceCounter("Sprites"), this._spritesRenderTime.endMonitoring(!1); })) : (this.scene.onBeforeSpritesRenderingObservable.remove(this._onBeforeSpritesRenderingObserver), this._onBeforeSpritesRenderingObserver = null, this.scene.onAfterSpritesRenderingObservable.remove(this._onAfterSpritesRenderingObserver), this._onAfterSpritesRenderingObserver = null))); } /** * Gets the perf counter used for physics time */ get physicsTimeCounter() { return this._physicsTime; } /** * Gets the physics time capture status */ get capturePhysicsTime() { return this._capturePhysicsTime; } /** * Enable or disable the physics time capture */ set capturePhysicsTime(e) { e !== this._capturePhysicsTime && this.scene.onBeforePhysicsObservable && (this._capturePhysicsTime = e, e ? (this._onBeforePhysicsObserver = this.scene.onBeforePhysicsObservable.add(() => { ye.StartPerformanceCounter("Physics"), this._physicsTime.beginMonitoring(); }), this._onAfterPhysicsObserver = this.scene.onAfterPhysicsObservable.add(() => { ye.EndPerformanceCounter("Physics"), this._physicsTime.endMonitoring(); })) : (this.scene.onBeforePhysicsObservable.remove(this._onBeforePhysicsObserver), this._onBeforePhysicsObserver = null, this.scene.onAfterPhysicsObservable.remove(this._onAfterPhysicsObserver), this._onAfterPhysicsObserver = null)); } /** * Gets the perf counter used for animations time */ get animationsTimeCounter() { return this._animationsTime; } /** * Gets the animations time capture status */ get captureAnimationsTime() { return this._captureAnimationsTime; } /** * Enable or disable the animations time capture */ set captureAnimationsTime(e) { e !== this._captureAnimationsTime && (this._captureAnimationsTime = e, e ? this._onAfterAnimationsObserver = this.scene.onAfterAnimationsObservable.add(() => { this._animationsTime.endMonitoring(); }) : (this.scene.onAfterAnimationsObservable.remove(this._onAfterAnimationsObserver), this._onAfterAnimationsObserver = null)); } /** * Gets the perf counter used for frame time capture */ get frameTimeCounter() { return this._frameTime; } /** * Gets the frame time capture status */ get captureFrameTime() { return this._captureFrameTime; } /** * Enable or disable the frame time capture */ set captureFrameTime(e) { this._captureFrameTime = e; } /** * Gets the perf counter used for inter-frames time capture */ get interFrameTimeCounter() { return this._interFrameTime; } /** * Gets the inter-frames time capture status */ get captureInterFrameTime() { return this._captureInterFrameTime; } /** * Enable or disable the inter-frames time capture */ set captureInterFrameTime(e) { this._captureInterFrameTime = e; } /** * Gets the perf counter used for render time capture */ get renderTimeCounter() { return this._renderTime; } /** * Gets the render time capture status */ get captureRenderTime() { return this._captureRenderTime; } /** * Enable or disable the render time capture */ set captureRenderTime(e) { e !== this._captureRenderTime && (this._captureRenderTime = e, e ? (this._onBeforeDrawPhaseObserver = this.scene.onBeforeDrawPhaseObservable.add(() => { this._renderTime.beginMonitoring(), ye.StartPerformanceCounter("Main render"); }), this._onAfterDrawPhaseObserver = this.scene.onAfterDrawPhaseObservable.add(() => { this._renderTime.endMonitoring(!1), ye.EndPerformanceCounter("Main render"); })) : (this.scene.onBeforeDrawPhaseObservable.remove(this._onBeforeDrawPhaseObserver), this._onBeforeDrawPhaseObserver = null, this.scene.onAfterDrawPhaseObservable.remove(this._onAfterDrawPhaseObserver), this._onAfterDrawPhaseObserver = null)); } /** * Gets the perf counter used for camera render time capture */ get cameraRenderTimeCounter() { return this._cameraRenderTime; } /** * Gets the camera render time capture status */ get captureCameraRenderTime() { return this._captureCameraRenderTime; } /** * Enable or disable the camera render time capture */ set captureCameraRenderTime(e) { e !== this._captureCameraRenderTime && (this._captureCameraRenderTime = e, e ? (this._onBeforeCameraRenderObserver = this.scene.onBeforeCameraRenderObservable.add((t) => { this._cameraRenderTime.beginMonitoring(), ye.StartPerformanceCounter(`Rendering camera ${t.name}`); }), this._onAfterCameraRenderObserver = this.scene.onAfterCameraRenderObservable.add((t) => { this._cameraRenderTime.endMonitoring(!1), ye.EndPerformanceCounter(`Rendering camera ${t.name}`); })) : (this.scene.onBeforeCameraRenderObservable.remove(this._onBeforeCameraRenderObserver), this._onBeforeCameraRenderObserver = null, this.scene.onAfterCameraRenderObservable.remove(this._onAfterCameraRenderObserver), this._onAfterCameraRenderObserver = null)); } /** * Gets the perf counter used for draw calls */ get drawCallsCounter() { return this.scene.getEngine()._drawCalls; } /** * Instantiates a new scene instrumentation. * This class can be used to get instrumentation data from a Babylon engine * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#sceneinstrumentation * @param scene Defines the scene to instrument */ constructor(e) { this.scene = e, this._captureActiveMeshesEvaluationTime = !1, this._activeMeshesEvaluationTime = new v9(), this._captureRenderTargetsRenderTime = !1, this._renderTargetsRenderTime = new v9(), this._captureFrameTime = !1, this._frameTime = new v9(), this._captureRenderTime = !1, this._renderTime = new v9(), this._captureInterFrameTime = !1, this._interFrameTime = new v9(), this._captureParticlesRenderTime = !1, this._particlesRenderTime = new v9(), this._captureSpritesRenderTime = !1, this._spritesRenderTime = new v9(), this._capturePhysicsTime = !1, this._physicsTime = new v9(), this._captureAnimationsTime = !1, this._animationsTime = new v9(), this._captureCameraRenderTime = !1, this._cameraRenderTime = new v9(), this._onBeforeActiveMeshesEvaluationObserver = null, this._onAfterActiveMeshesEvaluationObserver = null, this._onBeforeRenderTargetsRenderObserver = null, this._onAfterRenderTargetsRenderObserver = null, this._onAfterRenderObserver = null, this._onBeforeDrawPhaseObserver = null, this._onAfterDrawPhaseObserver = null, this._onBeforeAnimationsObserver = null, this._onBeforeParticlesRenderingObserver = null, this._onAfterParticlesRenderingObserver = null, this._onBeforeSpritesRenderingObserver = null, this._onAfterSpritesRenderingObserver = null, this._onBeforePhysicsObserver = null, this._onAfterPhysicsObserver = null, this._onAfterAnimationsObserver = null, this._onBeforeCameraRenderObserver = null, this._onAfterCameraRenderObserver = null, this._onBeforeAnimationsObserver = e.onBeforeAnimationsObservable.add(() => { this._captureActiveMeshesEvaluationTime && this._activeMeshesEvaluationTime.fetchNewFrame(), this._captureRenderTargetsRenderTime && this._renderTargetsRenderTime.fetchNewFrame(), this._captureFrameTime && (ye.StartPerformanceCounter("Scene rendering"), this._frameTime.beginMonitoring()), this._captureInterFrameTime && this._interFrameTime.endMonitoring(), this._captureParticlesRenderTime && this._particlesRenderTime.fetchNewFrame(), this._captureSpritesRenderTime && this._spritesRenderTime.fetchNewFrame(), this._captureAnimationsTime && this._animationsTime.beginMonitoring(), this._captureRenderTime && this._renderTime.fetchNewFrame(), this._captureCameraRenderTime && this._cameraRenderTime.fetchNewFrame(), this.scene.getEngine()._drawCalls.fetchNewFrame(); }), this._onAfterRenderObserver = e.onAfterRenderObservable.add(() => { this._captureFrameTime && (ye.EndPerformanceCounter("Scene rendering"), this._frameTime.endMonitoring()), this._captureRenderTime && this._renderTime.endMonitoring(!1), this._captureInterFrameTime && this._interFrameTime.beginMonitoring(), this._captureActiveMeshesEvaluationTime && this._activeMeshesEvaluationTime.endFrame(), this._captureRenderTargetsRenderTime && this._renderTargetsRenderTime.endFrame(), this._captureParticlesRenderTime && this._particlesRenderTime.endFrame(), this._captureSpritesRenderTime && this._spritesRenderTime.endFrame(), this._captureRenderTime && this._renderTime.endFrame(), this._captureCameraRenderTime && this._cameraRenderTime.endFrame(); }); } /** * Dispose and release associated resources. */ dispose() { this.scene.onAfterRenderObservable.remove(this._onAfterRenderObserver), this._onAfterRenderObserver = null, this.scene.onBeforeActiveMeshesEvaluationObservable.remove(this._onBeforeActiveMeshesEvaluationObserver), this._onBeforeActiveMeshesEvaluationObserver = null, this.scene.onAfterActiveMeshesEvaluationObservable.remove(this._onAfterActiveMeshesEvaluationObserver), this._onAfterActiveMeshesEvaluationObserver = null, this.scene.onBeforeRenderTargetsRenderObservable.remove(this._onBeforeRenderTargetsRenderObserver), this._onBeforeRenderTargetsRenderObserver = null, this.scene.onAfterRenderTargetsRenderObservable.remove(this._onAfterRenderTargetsRenderObserver), this._onAfterRenderTargetsRenderObserver = null, this.scene.onBeforeAnimationsObservable.remove(this._onBeforeAnimationsObserver), this._onBeforeAnimationsObserver = null, this.scene.onBeforeParticlesRenderingObservable.remove(this._onBeforeParticlesRenderingObserver), this._onBeforeParticlesRenderingObserver = null, this.scene.onAfterParticlesRenderingObservable.remove(this._onAfterParticlesRenderingObserver), this._onAfterParticlesRenderingObserver = null, this._onBeforeSpritesRenderingObserver && (this.scene.onBeforeSpritesRenderingObservable.remove(this._onBeforeSpritesRenderingObserver), this._onBeforeSpritesRenderingObserver = null), this._onAfterSpritesRenderingObserver && (this.scene.onAfterSpritesRenderingObservable.remove(this._onAfterSpritesRenderingObserver), this._onAfterSpritesRenderingObserver = null), this.scene.onBeforeDrawPhaseObservable.remove(this._onBeforeDrawPhaseObserver), this._onBeforeDrawPhaseObserver = null, this.scene.onAfterDrawPhaseObservable.remove(this._onAfterDrawPhaseObserver), this._onAfterDrawPhaseObserver = null, this._onBeforePhysicsObserver && (this.scene.onBeforePhysicsObservable.remove(this._onBeforePhysicsObserver), this._onBeforePhysicsObserver = null), this._onAfterPhysicsObserver && (this.scene.onAfterPhysicsObservable.remove(this._onAfterPhysicsObserver), this._onAfterPhysicsObserver = null), this.scene.onAfterAnimationsObservable.remove(this._onAfterAnimationsObserver), this._onAfterAnimationsObserver = null, this.scene.onBeforeCameraRenderObservable.remove(this._onBeforeCameraRenderObserver), this._onBeforeCameraRenderObserver = null, this.scene.onAfterCameraRenderObservable.remove(this._onAfterCameraRenderObserver), this._onAfterCameraRenderObserver = null, this.scene = null; } } const p3e = "glowMapGenerationPixelShader", h3e = `#if defined(DIFFUSE_ISLINEAR) || defined(EMISSIVE_ISLINEAR) #include #endif #ifdef DIFFUSE varying vec2 vUVDiffuse;uniform sampler2D diffuseSampler; #endif #ifdef OPACITY varying vec2 vUVOpacity;uniform sampler2D opacitySampler;uniform float opacityIntensity; #endif #ifdef EMISSIVE varying vec2 vUVEmissive;uniform sampler2D emissiveSampler; #endif #ifdef VERTEXALPHA varying vec4 vColor; #endif uniform vec4 glowColor;uniform float glowIntensity; #include #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #include vec4 finalColor=glowColor; #ifdef DIFFUSE vec4 albedoTexture=texture2D(diffuseSampler,vUVDiffuse); #ifdef DIFFUSE_ISLINEAR albedoTexture=toGammaSpace(albedoTexture); #endif #ifdef GLOW finalColor.a*=albedoTexture.a; #endif #ifdef HIGHLIGHT finalColor.a=albedoTexture.a; #endif #endif #ifdef OPACITY vec4 opacityMap=texture2D(opacitySampler,vUVOpacity); #ifdef OPACITYRGB finalColor.a*=getLuminance(opacityMap.rgb); #else finalColor.a*=opacityMap.a; #endif finalColor.a*=opacityIntensity; #endif #ifdef VERTEXALPHA finalColor.a*=vColor.a; #endif #ifdef ALPHATEST if (finalColor.a #include #include #include[0..maxSimultaneousMorphTargets] #include #include uniform mat4 viewProjection;varying vec4 vPosition; #ifdef UV1 attribute vec2 uv; #endif #ifdef UV2 attribute vec2 uv2; #endif #ifdef DIFFUSE varying vec2 vUVDiffuse;uniform mat4 diffuseMatrix; #endif #ifdef OPACITY varying vec2 vUVOpacity;uniform mat4 opacityMatrix; #endif #ifdef EMISSIVE varying vec2 vUVEmissive;uniform mat4 emissiveMatrix; #endif #ifdef VERTEXALPHA attribute vec4 color;varying vec4 vColor; #endif #define CUSTOM_VERTEX_DEFINITIONS void main(void) {vec3 positionUpdated=position; #ifdef UV1 vec2 uvUpdated=uv; #endif #include #include[0..maxSimultaneousMorphTargets] #include #include #include vec4 worldPos=finalWorld*vec4(positionUpdated,1.0); #ifdef CUBEMAP vPosition=worldPos;gl_Position=viewProjection*finalWorld*vec4(position,1.0); #else vPosition=viewProjection*worldPos;gl_Position=vPosition; #endif #ifdef DIFFUSE #ifdef DIFFUSEUV1 vUVDiffuse=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0)); #endif #ifdef DIFFUSEUV2 vUVDiffuse=vec2(diffuseMatrix*vec4(uv2,1.0,0.0)); #endif #endif #ifdef OPACITY #ifdef OPACITYUV1 vUVOpacity=vec2(opacityMatrix*vec4(uvUpdated,1.0,0.0)); #endif #ifdef OPACITYUV2 vUVOpacity=vec2(opacityMatrix*vec4(uv2,1.0,0.0)); #endif #endif #ifdef EMISSIVE #ifdef EMISSIVEUV1 vUVEmissive=vec2(emissiveMatrix*vec4(uvUpdated,1.0,0.0)); #endif #ifdef EMISSIVEUV2 vUVEmissive=vec2(emissiveMatrix*vec4(uv2,1.0,0.0)); #endif #endif #ifdef VERTEXALPHA vColor=color; #endif #include }`; Le.ShadersStore[H3e] = g3e; class mc { /** * Gets the camera attached to the layer. */ get camera() { return this._effectLayerOptions.camera; } /** * Gets the rendering group id the layer should render in. */ get renderingGroupId() { return this._effectLayerOptions.renderingGroupId; } set renderingGroupId(e) { this._effectLayerOptions.renderingGroupId = e; } /** * Gets the main texture where the effect is rendered */ get mainTexture() { return this._mainTexture; } /** * Sets a specific material to be used to render a mesh/a list of meshes in the layer * @param mesh mesh or array of meshes * @param material material to use by the layer when rendering the mesh(es). If undefined is passed, the specific material created by the layer will be used. */ setMaterialForRendering(e, t) { if (this._mainTexture.setMaterialForRendering(e, t), Array.isArray(e)) for (let r = 0; r < e.length; ++r) { const n = e[r]; t ? this._materialForRendering[n.uniqueId] = [n, t] : delete this._materialForRendering[n.uniqueId]; } else t ? this._materialForRendering[e.uniqueId] = [e, t] : delete this._materialForRendering[e.uniqueId]; } /** * Gets the intensity of the effect for a specific mesh. * @param mesh The mesh to get the effect intensity for * @returns The intensity of the effect for the mesh */ getEffectIntensity(e) { var t; return (t = this._effectIntensity[e.uniqueId]) !== null && t !== void 0 ? t : 1; } /** * Sets the intensity of the effect for a specific mesh. * @param mesh The mesh to set the effect intensity for * @param intensity The intensity of the effect for the mesh */ setEffectIntensity(e, t) { this._effectIntensity[e.uniqueId] = t; } /** * Instantiates a new effect Layer and references it in the scene. * @param name The name of the layer * @param scene The scene to use the layer in */ constructor(e, t) { this._vertexBuffers = {}, this._maxSize = 0, this._mainTextureDesiredSize = { width: 0, height: 0 }, this._shouldRender = !0, this._postProcesses = [], this._textures = [], this._emissiveTextureAndColor = { texture: null, color: new xt() }, this._effectIntensity = {}, this.neutralColor = new xt(), this.isEnabled = !0, this.disableBoundingBoxesFromEffectLayer = !1, this.onDisposeObservable = new Oe(), this.onBeforeRenderMainTextureObservable = new Oe(), this.onBeforeComposeObservable = new Oe(), this.onBeforeRenderMeshToEffect = new Oe(), this.onAfterRenderMeshToEffect = new Oe(), this.onAfterComposeObservable = new Oe(), this.onSizeChangedObservable = new Oe(), this._materialForRendering = {}, this.name = e, this._scene = t || gr.LastCreatedScene, mc._SceneComponentInitialization(this._scene), this._engine = this._scene.getEngine(), this._maxSize = this._engine.getCaps().maxTextureSize, this._scene.effectLayers.push(this), this._mergeDrawWrapper = [], this._generateIndexBuffer(), this._generateVertexBuffer(); } /** * Number of times _internalRender will be called. Some effect layers need to render the mesh several times, so they should override this method with the number of times the mesh should be rendered * @returns Number of times a mesh must be rendered in the layer */ _numInternalDraws() { return 1; } /** * Initializes the effect layer with the required options. * @param options Sets of none mandatory options to use with the layer (see IEffectLayerOptions for more information) */ _init(e) { this._effectLayerOptions = Object.assign({ mainTextureRatio: 0.5, alphaBlendingMode: 2, camera: null, renderingGroupId: -1, mainTextureType: 0, generateStencilBuffer: !1 }, e), this._setMainTextureSize(), this._createMainTexture(), this._createTextureAndPostProcesses(); } /** * Generates the index buffer of the full screen quad blending to the main canvas. */ _generateIndexBuffer() { const e = []; e.push(0), e.push(1), e.push(2), e.push(0), e.push(2), e.push(3), this._indexBuffer = this._engine.createIndexBuffer(e); } /** * Generates the vertex buffer of the full screen quad blending to the main canvas. */ _generateVertexBuffer() { const e = []; e.push(1, 1), e.push(-1, 1), e.push(-1, -1), e.push(1, -1); const t = new J(this._engine, e, J.PositionKind, !1, !1, 2); this._vertexBuffers[J.PositionKind] = t; } /** * Sets the main texture desired size which is the closest power of two * of the engine canvas size. */ _setMainTextureSize() { this._effectLayerOptions.mainTextureFixedSize ? (this._mainTextureDesiredSize.width = this._effectLayerOptions.mainTextureFixedSize, this._mainTextureDesiredSize.height = this._effectLayerOptions.mainTextureFixedSize) : (this._mainTextureDesiredSize.width = this._engine.getRenderWidth() * this._effectLayerOptions.mainTextureRatio, this._mainTextureDesiredSize.height = this._engine.getRenderHeight() * this._effectLayerOptions.mainTextureRatio, this._mainTextureDesiredSize.width = this._engine.needPOTTextures ? Ge.GetExponentOfTwo(this._mainTextureDesiredSize.width, this._maxSize) : this._mainTextureDesiredSize.width, this._mainTextureDesiredSize.height = this._engine.needPOTTextures ? Ge.GetExponentOfTwo(this._mainTextureDesiredSize.height, this._maxSize) : this._mainTextureDesiredSize.height), this._mainTextureDesiredSize.width = Math.floor(this._mainTextureDesiredSize.width), this._mainTextureDesiredSize.height = Math.floor(this._mainTextureDesiredSize.height); } /** * Creates the main texture for the effect layer. */ _createMainTexture() { this._mainTexture = new Ta("EffectLayerMainRTT", { width: this._mainTextureDesiredSize.width, height: this._mainTextureDesiredSize.height }, this._scene, !1, !0, this._effectLayerOptions.mainTextureType, !1, We.TRILINEAR_SAMPLINGMODE, !0, this._effectLayerOptions.generateStencilBuffer), this._mainTexture.activeCamera = this._effectLayerOptions.camera, this._mainTexture.wrapU = We.CLAMP_ADDRESSMODE, this._mainTexture.wrapV = We.CLAMP_ADDRESSMODE, this._mainTexture.anisotropicFilteringLevel = 1, this._mainTexture.updateSamplingMode(We.BILINEAR_SAMPLINGMODE), this._mainTexture.renderParticles = !1, this._mainTexture.renderList = null, this._mainTexture.ignoreCameraViewport = !0; for (const e in this._materialForRendering) { const [t, r] = this._materialForRendering[e]; this._mainTexture.setMaterialForRendering(t, r); } if (this._mainTexture.customIsReadyFunction = (e, t, r) => { if ((r || t === 0) && e.subMeshes) for (let n = 0; n < e.subMeshes.length; ++n) { const i = e.subMeshes[n], s = i.getMaterial(), a = i.getRenderingMesh(); if (!s) continue; const o = a._getInstancesRenderList(i._id, !!i.getReplacementMesh()).hardwareInstancedRendering[i._id] || a.hasThinInstances; if (this._setEmissiveTextureAndColor(a, i, s), !this._isReady(i, o, this._emissiveTextureAndColor.texture)) return !1; } return !0; }, this._mainTexture.customRenderFunction = (e, t, r, n) => { this.onBeforeRenderMainTextureObservable.notifyObservers(this); let i; const s = this._scene.getEngine(); if (n.length) { for (s.setColorWrite(!1), i = 0; i < n.length; i++) this._renderSubMesh(n.data[i]); s.setColorWrite(!0); } for (i = 0; i < e.length; i++) this._renderSubMesh(e.data[i]); for (i = 0; i < t.length; i++) this._renderSubMesh(t.data[i]); const a = s.getAlphaMode(); for (i = 0; i < r.length; i++) this._renderSubMesh(r.data[i], !0); s.setAlphaMode(a); }, this._mainTexture.onClearObservable.add((e) => { e.clear(this.neutralColor, !0, !0, !0); }), this._scene.getBoundingBoxRenderer) { const e = this._scene.getBoundingBoxRenderer().enabled; this._mainTexture.onBeforeBindObservable.add(() => { this._scene.getBoundingBoxRenderer().enabled = !this.disableBoundingBoxesFromEffectLayer && e; }), this._mainTexture.onAfterUnbindObservable.add(() => { this._scene.getBoundingBoxRenderer().enabled = e; }); } } /** * Adds specific effects defines. * @param defines The defines to add specifics to. */ // eslint-disable-next-line @typescript-eslint/no-unused-vars _addCustomEffectDefines(e) { } /** * Checks for the readiness of the element composing the layer. * @param subMesh the mesh to check for * @param useInstances specify whether or not to use instances to render the mesh * @param emissiveTexture the associated emissive texture used to generate the glow * @returns true if ready otherwise, false */ _isReady(e, t, r) { var n; const i = this._scene.getEngine(), s = e.getMesh(), a = (n = s._internalAbstractMeshDataInfo._materialForRenderPass) === null || n === void 0 ? void 0 : n[i.currentRenderPassId]; if (a) return a.isReadyForSubMesh(s, e, t); const f = e.getMaterial(); if (!f) return !1; if (this._useMeshMaterial(e.getRenderingMesh())) return f.isReadyForSubMesh(e.getMesh(), e, t); const o = [], d = [J.PositionKind]; let v = !1, u = !1; if (f) { const q = f.needAlphaTesting(), b = f.getAlphaTestTexture(), j = b && b.hasAlpha && (f.useAlphaFromDiffuseTexture || f._useAlphaFromAlbedoTexture); b && (q || j) && (o.push("#define DIFFUSE"), s.isVerticesDataPresent(J.UV2Kind) && b.coordinatesIndex === 1 ? (o.push("#define DIFFUSEUV2"), u = !0) : s.isVerticesDataPresent(J.UVKind) && (o.push("#define DIFFUSEUV1"), v = !0), q && (o.push("#define ALPHATEST"), o.push("#define ALPHATESTVALUE 0.4")), b.gammaSpace || o.push("#define DIFFUSE_ISLINEAR")); const w = f.opacityTexture; w && (o.push("#define OPACITY"), s.isVerticesDataPresent(J.UV2Kind) && w.coordinatesIndex === 1 ? (o.push("#define OPACITYUV2"), u = !0) : s.isVerticesDataPresent(J.UVKind) && (o.push("#define OPACITYUV1"), v = !0)); } r && (o.push("#define EMISSIVE"), s.isVerticesDataPresent(J.UV2Kind) && r.coordinatesIndex === 1 ? (o.push("#define EMISSIVEUV2"), u = !0) : s.isVerticesDataPresent(J.UVKind) && (o.push("#define EMISSIVEUV1"), v = !0), r.gammaSpace || o.push("#define EMISSIVE_ISLINEAR")), s.useVertexColors && s.isVerticesDataPresent(J.ColorKind) && s.hasVertexAlpha && f.transparencyMode !== gt.MATERIAL_OPAQUE && (d.push(J.ColorKind), o.push("#define VERTEXALPHA")), v && (d.push(J.UVKind), o.push("#define UV1")), u && (d.push(J.UV2Kind), o.push("#define UV2")); const l = new c1(); if (s.useBones && s.computeBonesUsingShaders) { d.push(J.MatricesIndicesKind), d.push(J.MatricesWeightsKind), s.numBoneInfluencers > 4 && (d.push(J.MatricesIndicesExtraKind), d.push(J.MatricesWeightsExtraKind)), o.push("#define NUM_BONE_INFLUENCERS " + s.numBoneInfluencers); const q = s.skeleton; q && q.isUsingTextureForMatrices ? o.push("#define BONETEXTURE") : o.push("#define BonesPerMesh " + (q ? q.bones.length + 1 : 0)), s.numBoneInfluencers > 0 && l.addCPUSkinningFallback(0, s); } else o.push("#define NUM_BONE_INFLUENCERS 0"); const P = s.morphTargetManager; let p = 0; P && P.numInfluencers > 0 && (o.push("#define MORPHTARGETS"), p = P.numInfluencers, o.push("#define NUM_MORPH_INFLUENCERS " + p), P.isUsingTextureForTargets && o.push("#define MORPHTARGETS_TEXTURE"), Ye.PrepareAttributesForMorphTargetsInfluencers(d, s, p)), t && (o.push("#define INSTANCES"), Ye.PushAttributesForInstances(d), e.getRenderingMesh().hasThinInstances && o.push("#define THIN_INSTANCES")), xq(f, this._scene, o), this._addCustomEffectDefines(o); const c = e._getDrawWrapper(void 0, !0), H = c.defines, T = o.join(` `); if (H !== T) { const q = [ "world", "mBones", "viewProjection", "glowColor", "morphTargetInfluences", "boneTextureWidth", "diffuseMatrix", "emissiveMatrix", "opacityMatrix", "opacityIntensity", "morphTargetTextureInfo", "morphTargetTextureIndices", "glowIntensity" ]; Mf(q), c.setEffect(this._engine.createEffect("glowMapGeneration", d, q, ["diffuseSampler", "emissiveSampler", "opacitySampler", "boneSampler", "morphTargets"], T, l, void 0, void 0, { maxSimultaneousMorphTargets: p }), T); } return c.effect.isReady(); } /** * Renders the glowing part of the scene by blending the blurred glowing meshes on top of the rendered scene. */ render() { for (let s = 0; s < this._postProcesses.length; s++) if (!this._postProcesses[s].isReady()) return; const e = this._scene.getEngine(), t = this._numInternalDraws(); let r = !0; for (let s = 0; s < t; ++s) { let a = this._mergeDrawWrapper[s]; a || (a = this._mergeDrawWrapper[s] = new zo(this._engine), a.setEffect(this._createMergeEffect())), r = r && a.effect.isReady(); } if (!r) return; this.onBeforeComposeObservable.notifyObservers(this); const n = e.getAlphaMode(); for (let s = 0; s < t; ++s) { const a = this._mergeDrawWrapper[s]; e.enableEffect(a), e.setState(!1), e.bindBuffers(this._vertexBuffers, this._indexBuffer, a.effect), e.setAlphaMode(this._effectLayerOptions.alphaBlendingMode), this._internalRender(a.effect, s); } e.setAlphaMode(n), this.onAfterComposeObservable.notifyObservers(this); const i = this._mainTexture.getSize(); this._setMainTextureSize(), (i.width !== this._mainTextureDesiredSize.width || i.height !== this._mainTextureDesiredSize.height) && this._mainTextureDesiredSize.width !== 0 && this._mainTextureDesiredSize.height !== 0 && (this.onSizeChangedObservable.notifyObservers(this), this._disposeTextureAndPostProcesses(), this._createMainTexture(), this._createTextureAndPostProcesses()); } /** * Determine if a given mesh will be used in the current effect. * @param mesh mesh to test * @returns true if the mesh will be used */ hasMesh(e) { return this.renderingGroupId === -1 || e.renderingGroupId === this.renderingGroupId; } /** * Returns true if the layer contains information to display, otherwise false. * @returns true if the glow layer should be rendered */ shouldRender() { return this.isEnabled && this._shouldRender; } /** * Returns true if the mesh should render, otherwise false. * @param mesh The mesh to render * @returns true if it should render otherwise false */ // eslint-disable-next-line @typescript-eslint/no-unused-vars _shouldRenderMesh(e) { return !0; } /** * Returns true if the mesh can be rendered, otherwise false. * @param mesh The mesh to render * @param material The material used on the mesh * @returns true if it can be rendered otherwise false */ _canRenderMesh(e, t) { return !t.needAlphaBlendingForMesh(e); } /** * Returns true if the mesh should render, otherwise false. * @returns true if it should render otherwise false */ _shouldRenderEmissiveTextureForMesh() { return !0; } /** * Renders the submesh passed in parameter to the generation map. * @param subMesh * @param enableAlphaMode */ _renderSubMesh(e, t = !1) { var r, n; if (!this.shouldRender()) return; const i = e.getMaterial(), s = e.getMesh(), a = e.getReplacementMesh(), f = e.getRenderingMesh(), o = e.getEffectiveMesh(), d = this._scene, v = d.getEngine(); if (o._internalAbstractMeshDataInfo._isActiveIntermediate = !1, !i || !this._canRenderMesh(f, i)) return; let u = (r = f.overrideMaterialSideOrientation) !== null && r !== void 0 ? r : i.sideOrientation; o._getWorldMatrixDeterminant() < 0 && (u = u === gt.ClockWiseSideOrientation ? gt.CounterClockWiseSideOrientation : gt.ClockWiseSideOrientation); const P = u === gt.ClockWiseSideOrientation; v.setState(i.backFaceCulling, i.zOffset, void 0, P, i.cullBackFaces, void 0, i.zOffsetUnits); const p = f._getInstancesRenderList(e._id, !!a); if (p.mustReturn || !this._shouldRenderMesh(f)) return; const c = p.hardwareInstancedRendering[e._id] || f.hasThinInstances; if (this._setEmissiveTextureAndColor(f, e, i), this.onBeforeRenderMeshToEffect.notifyObservers(s), this._useMeshMaterial(f)) f.render(e, t, a || void 0); else if (this._isReady(e, c, this._emissiveTextureAndColor.texture)) { const H = (n = o._internalAbstractMeshDataInfo._materialForRenderPass) === null || n === void 0 ? void 0 : n[v.currentRenderPassId]; let T = e._getDrawWrapper(); if (!T && H && (T = H._getDrawWrapper()), !T) return; const q = T.effect; if (v.enableEffect(T), c || f._bind(e, q, i.fillMode), H ? H.bindForSubMesh(o.getWorldMatrix(), o, e) : (q.setMatrix("viewProjection", d.getTransformMatrix()), q.setMatrix("world", o.getWorldMatrix()), q.setFloat4("glowColor", this._emissiveTextureAndColor.color.r, this._emissiveTextureAndColor.color.g, this._emissiveTextureAndColor.color.b, this._emissiveTextureAndColor.color.a)), !H) { const b = i.needAlphaTesting(), j = i.getAlphaTestTexture(), w = j && j.hasAlpha && (i.useAlphaFromDiffuseTexture || i._useAlphaFromAlbedoTexture); if (j && (b || w)) { q.setTexture("diffuseSampler", j); const I = j.getTextureMatrix(); I && q.setMatrix("diffuseMatrix", I); } const m = i.opacityTexture; if (m) { q.setTexture("opacitySampler", m), q.setFloat("opacityIntensity", m.level); const I = m.getTextureMatrix(); I && q.setMatrix("opacityMatrix", I); } if (this._emissiveTextureAndColor.texture && (q.setTexture("emissiveSampler", this._emissiveTextureAndColor.texture), q.setMatrix("emissiveMatrix", this._emissiveTextureAndColor.texture.getTextureMatrix())), f.useBones && f.computeBonesUsingShaders && f.skeleton) { const I = f.skeleton; if (I.isUsingTextureForMatrices) { const N = I.getTransformMatrixTexture(f); if (!N) return; q.setTexture("boneSampler", N), q.setFloat("boneTextureWidth", 4 * (I.bones.length + 1)); } else q.setMatrices("mBones", I.getTransformMatrices(f)); } Ye.BindMorphTargetParameters(f, q), f.morphTargetManager && f.morphTargetManager.isUsingTextureForTargets && f.morphTargetManager._bind(q), t && v.setAlphaMode(i.alphaMode), q.setFloat("glowIntensity", this.getEffectIntensity(f)), Df(q, i, d); } f._processRendering(o, e, q, i.fillMode, p, c, (b, j) => q.setMatrix("world", j)); } else this._mainTexture.resetRefreshCounter(); this.onAfterRenderMeshToEffect.notifyObservers(s); } /** * Defines whether the current material of the mesh should be use to render the effect. * @param mesh defines the current mesh to render */ // eslint-disable-next-line @typescript-eslint/no-unused-vars _useMeshMaterial(e) { return !1; } /** * Rebuild the required buffers. * @internal Internal use only. */ _rebuild() { const e = this._vertexBuffers[J.PositionKind]; e && e._rebuild(), this._generateIndexBuffer(); } /** * Dispose only the render target textures and post process. */ _disposeTextureAndPostProcesses() { this._mainTexture.dispose(); for (let e = 0; e < this._postProcesses.length; e++) this._postProcesses[e] && this._postProcesses[e].dispose(); this._postProcesses = []; for (let e = 0; e < this._textures.length; e++) this._textures[e] && this._textures[e].dispose(); this._textures = []; } /** * Dispose the highlight layer and free resources. */ dispose() { const e = this._vertexBuffers[J.PositionKind]; e && (e.dispose(), this._vertexBuffers[J.PositionKind] = null), this._indexBuffer && (this._scene.getEngine()._releaseBuffer(this._indexBuffer), this._indexBuffer = null); for (const r of this._mergeDrawWrapper) r.dispose(); this._mergeDrawWrapper = [], this._disposeTextureAndPostProcesses(); const t = this._scene.effectLayers.indexOf(this, 0); t > -1 && this._scene.effectLayers.splice(t, 1), this.onDisposeObservable.notifyObservers(this), this.onDisposeObservable.clear(), this.onBeforeRenderMainTextureObservable.clear(), this.onBeforeComposeObservable.clear(), this.onBeforeRenderMeshToEffect.clear(), this.onAfterRenderMeshToEffect.clear(), this.onAfterComposeObservable.clear(), this.onSizeChangedObservable.clear(); } /** * Gets the class name of the effect layer * @returns the string with the class name of the effect layer */ getClassName() { return "EffectLayer"; } /** * Creates an effect layer from parsed effect layer data * @param parsedEffectLayer defines effect layer data * @param scene defines the current scene * @param rootUrl defines the root URL containing the effect layer information * @returns a parsed effect Layer */ static Parse(e, t, r) { return ye.Instantiate(e.customType).Parse(e, t, r); } } mc._SceneComponentInitialization = (A) => { throw qn("EffectLayerSceneComponent"); }; C([ M() ], mc.prototype, "name", void 0); C([ rU() ], mc.prototype, "neutralColor", void 0); C([ M() ], mc.prototype, "isEnabled", void 0); C([ U$() ], mc.prototype, "camera", null); C([ M() ], mc.prototype, "renderingGroupId", null); C([ M() ], mc.prototype, "disableBoundingBoxesFromEffectLayer", void 0); J1.AddParser(Ot.NAME_EFFECTLAYER, (A, e, t, r) => { if (A.effectLayers) { t.effectLayers || (t.effectLayers = []); for (let n = 0; n < A.effectLayers.length; n++) { const i = mc.Parse(A.effectLayers[n], e, r); t.effectLayers.push(i); } } }); J1.prototype.removeEffectLayer = function(A) { const e = this.effectLayers.indexOf(A); return e !== -1 && this.effectLayers.splice(e, 1), e; }; J1.prototype.addEffectLayer = function(A) { this.effectLayers.push(A); }; class Nte { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_EFFECTLAYER, this._renderEffects = !1, this._needStencil = !1, this._previousStencilState = !1, this.scene = e || gr.LastCreatedScene, this.scene && (this._engine = this.scene.getEngine(), this.scene.effectLayers = []); } /** * Registers the component in a given scene */ register() { this.scene._isReadyForMeshStage.registerStep(Ot.STEP_ISREADYFORMESH_EFFECTLAYER, this, this._isReadyForMesh), this.scene._cameraDrawRenderTargetStage.registerStep(Ot.STEP_CAMERADRAWRENDERTARGET_EFFECTLAYER, this, this._renderMainTexture), this.scene._beforeCameraDrawStage.registerStep(Ot.STEP_BEFORECAMERADRAW_EFFECTLAYER, this, this._setStencil), this.scene._afterRenderingGroupDrawStage.registerStep(Ot.STEP_AFTERRENDERINGGROUPDRAW_EFFECTLAYER_DRAW, this, this._drawRenderingGroup), this.scene._afterCameraDrawStage.registerStep(Ot.STEP_AFTERCAMERADRAW_EFFECTLAYER, this, this._setStencilBack), this.scene._afterCameraDrawStage.registerStep(Ot.STEP_AFTERCAMERADRAW_EFFECTLAYER_DRAW, this, this._drawCamera); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { const e = this.scene.effectLayers; for (const t of e) t._rebuild(); } /** * Serializes the component data to the specified json object * @param serializationObject The object to serialize to */ serialize(e) { e.effectLayers = []; const t = this.scene.effectLayers; for (const r of t) r.serialize && e.effectLayers.push(r.serialize()); } /** * Adds all the elements from the container to the scene * @param container the container holding the elements */ addFromContainer(e) { e.effectLayers && e.effectLayers.forEach((t) => { this.scene.addEffectLayer(t); }); } /** * Removes all the elements in the container from the scene * @param container contains the elements to remove * @param dispose if the removed element should be disposed (default: false) */ removeFromContainer(e, t) { e.effectLayers && e.effectLayers.forEach((r) => { this.scene.removeEffectLayer(r), t && r.dispose(); }); } /** * Disposes the component and the associated resources. */ dispose() { const e = this.scene.effectLayers; for (; e.length; ) e[0].dispose(); } _isReadyForMesh(e, t) { const r = this._engine.currentRenderPassId, n = this.scene.effectLayers; for (const i of n) { if (!i.hasMesh(e)) continue; const s = i._mainTexture; this._engine.currentRenderPassId = s.renderPassId; for (const a of e.subMeshes) if (!i.isReady(a, t)) return this._engine.currentRenderPassId = r, !1; } return this._engine.currentRenderPassId = r, !0; } _renderMainTexture(e) { this._renderEffects = !1, this._needStencil = !1; let t = !1; const r = this.scene.effectLayers; if (r && r.length > 0) { this._previousStencilState = this._engine.getStencilBuffer(); for (const n of r) if (n.shouldRender() && (!n.camera || n.camera.cameraRigMode === Tr.RIG_MODE_NONE && e === n.camera || n.camera.cameraRigMode !== Tr.RIG_MODE_NONE && n.camera._rigCameras.indexOf(e) > -1)) { this._renderEffects = !0, this._needStencil = this._needStencil || n.needStencil(); const i = n._mainTexture; i._shouldRender() && (this.scene.incrementRenderId(), i.render(!1, !1), t = !0); } this.scene.incrementRenderId(); } return t; } _setStencil() { this._needStencil && this._engine.setStencilBuffer(!0); } _setStencilBack() { this._needStencil && this._engine.setStencilBuffer(this._previousStencilState); } _draw(e) { if (this._renderEffects) { this._engine.setDepthBuffer(!1); const t = this.scene.effectLayers; for (let r = 0; r < t.length; r++) { const n = t[r]; n.renderingGroupId === e && n.shouldRender() && n.render(); } this._engine.setDepthBuffer(!0); } } _drawCamera() { this._renderEffects && this._draw(-1); } _drawRenderingGroup(e) { !this.scene._isInIntermediateRendering() && this._renderEffects && this._draw(e); } } mc._SceneComponentInitialization = (A) => { let e = A._getComponent(Ot.NAME_EFFECTLAYER); e || (e = new Nte(A), A._addComponent(e)); }; const X3e = "glowMapMergePixelShader", T3e = `varying vec2 vUV;uniform sampler2D textureSampler; #ifdef EMISSIVE uniform sampler2D textureSampler2; #endif uniform float offset; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #define CUSTOM_FRAGMENT_MAIN_BEGIN vec4 baseColor=texture2D(textureSampler,vUV); #ifdef EMISSIVE baseColor+=texture2D(textureSampler2,vUV);baseColor*=offset; #else baseColor.a=abs(offset-baseColor.a); #ifdef STROKE float alpha=smoothstep(.0,.1,baseColor.a);baseColor.a=alpha;baseColor.rgb=baseColor.rgb*alpha; #endif #endif #if LDR baseColor=clamp(baseColor,0.,1.0); #endif gl_FragColor=baseColor; #define CUSTOM_FRAGMENT_MAIN_END }`; Le.ShadersStore[X3e] = T3e; const q3e = "glowMapMergeVertexShader", b3e = `attribute vec2 position;varying vec2 vUV;const vec2 madd=vec2(0.5,0.5); #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN vUV=position*madd+madd;gl_Position=vec4(position,0.0,1.0); #define CUSTOM_VERTEX_MAIN_END }`; Le.ShadersStore[q3e] = b3e; J1.prototype.getGlowLayerByName = function(A) { var e; for (let t = 0; t < ((e = this.effectLayers) === null || e === void 0 ? void 0 : e.length); t++) if (this.effectLayers[t].name === A && this.effectLayers[t].getEffectName() === zl.EffectName) return this.effectLayers[t]; return null; }; class zl extends mc { /** * Sets the kernel size of the blur. */ set blurKernelSize(e) { if (e === this._options.blurKernelSize) return; this._options.blurKernelSize = e; const t = this._getEffectiveBlurKernelSize(); this._horizontalBlurPostprocess1.kernel = t, this._verticalBlurPostprocess1.kernel = t, this._horizontalBlurPostprocess2.kernel = t, this._verticalBlurPostprocess2.kernel = t; } /** * Gets the kernel size of the blur. */ get blurKernelSize() { return this._options.blurKernelSize; } /** * Sets the glow intensity. */ set intensity(e) { this._intensity = e; } /** * Gets the glow intensity. */ get intensity() { return this._intensity; } /** * Instantiates a new glow Layer and references it to the scene. * @param name The name of the layer * @param scene The scene to use the layer in * @param options Sets of none mandatory options to use with the layer (see IGlowLayerOptions for more information) */ constructor(e, t, r) { super(e, t), this._intensity = 1, this._includedOnlyMeshes = [], this._excludedMeshes = [], this._meshesUsingTheirOwnMaterials = [], this.neutralColor = new xt(0, 0, 0, 1), this._options = Object.assign({ mainTextureRatio: zl.DefaultTextureRatio, blurKernelSize: 32, mainTextureFixedSize: void 0, camera: null, mainTextureSamples: 1, renderingGroupId: -1, ldrMerge: !1, alphaBlendingMode: 1, mainTextureType: 0, generateStencilBuffer: !1 }, r), this._init({ alphaBlendingMode: this._options.alphaBlendingMode, camera: this._options.camera, mainTextureFixedSize: this._options.mainTextureFixedSize, mainTextureRatio: this._options.mainTextureRatio, renderingGroupId: this._options.renderingGroupId, mainTextureType: this._options.mainTextureType, generateStencilBuffer: this._options.generateStencilBuffer }); } /** * Get the effect name of the layer. * @returns The effect name */ getEffectName() { return zl.EffectName; } /** * Create the merge effect. This is the shader use to blit the information back * to the main canvas at the end of the scene rendering. */ _createMergeEffect() { let e = `#define EMISSIVE `; return this._options.ldrMerge && (e += `#define LDR `), this._engine.createEffect("glowMapMerge", [J.PositionKind], ["offset"], ["textureSampler", "textureSampler2"], e); } /** * Creates the render target textures and post processes used in the glow layer. */ _createTextureAndPostProcesses() { let e = this._mainTextureDesiredSize.width, t = this._mainTextureDesiredSize.height; e = this._engine.needPOTTextures ? Ge.GetExponentOfTwo(e, this._maxSize) : e, t = this._engine.needPOTTextures ? Ge.GetExponentOfTwo(t, this._maxSize) : t; let r = 0; this._engine.getCaps().textureHalfFloatRender ? r = 2 : r = 0, this._blurTexture1 = new Ta("GlowLayerBlurRTT", { width: e, height: t }, this._scene, !1, !0, r), this._blurTexture1.wrapU = We.CLAMP_ADDRESSMODE, this._blurTexture1.wrapV = We.CLAMP_ADDRESSMODE, this._blurTexture1.updateSamplingMode(We.BILINEAR_SAMPLINGMODE), this._blurTexture1.renderParticles = !1, this._blurTexture1.ignoreCameraViewport = !0; const n = Math.floor(e / 2), i = Math.floor(t / 2); this._blurTexture2 = new Ta("GlowLayerBlurRTT2", { width: n, height: i }, this._scene, !1, !0, r), this._blurTexture2.wrapU = We.CLAMP_ADDRESSMODE, this._blurTexture2.wrapV = We.CLAMP_ADDRESSMODE, this._blurTexture2.updateSamplingMode(We.BILINEAR_SAMPLINGMODE), this._blurTexture2.renderParticles = !1, this._blurTexture2.ignoreCameraViewport = !0, this._textures = [this._blurTexture1, this._blurTexture2]; const s = this._getEffectiveBlurKernelSize(); this._horizontalBlurPostprocess1 = new c9("GlowLayerHBP1", new at(1, 0), s, { width: e, height: t }, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1, r), this._horizontalBlurPostprocess1.width = e, this._horizontalBlurPostprocess1.height = t, this._horizontalBlurPostprocess1.externalTextureSamplerBinding = !0, this._horizontalBlurPostprocess1.onApplyObservable.add((a) => { a.setTexture("textureSampler", this._mainTexture); }), this._verticalBlurPostprocess1 = new c9("GlowLayerVBP1", new at(0, 1), s, { width: e, height: t }, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1, r), this._horizontalBlurPostprocess2 = new c9("GlowLayerHBP2", new at(1, 0), s, { width: n, height: i }, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1, r), this._horizontalBlurPostprocess2.width = n, this._horizontalBlurPostprocess2.height = i, this._horizontalBlurPostprocess2.externalTextureSamplerBinding = !0, this._horizontalBlurPostprocess2.onApplyObservable.add((a) => { a.setTexture("textureSampler", this._blurTexture1); }), this._verticalBlurPostprocess2 = new c9("GlowLayerVBP2", new at(0, 1), s, { width: n, height: i }, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1, r), this._postProcesses = [this._horizontalBlurPostprocess1, this._verticalBlurPostprocess1, this._horizontalBlurPostprocess2, this._verticalBlurPostprocess2], this._postProcesses1 = [this._horizontalBlurPostprocess1, this._verticalBlurPostprocess1], this._postProcesses2 = [this._horizontalBlurPostprocess2, this._verticalBlurPostprocess2], this._mainTexture.samples = this._options.mainTextureSamples, this._mainTexture.onAfterUnbindObservable.add(() => { const a = this._blurTexture1.renderTarget; if (a) { this._scene.postProcessManager.directRender(this._postProcesses1, a, !0); const f = this._blurTexture2.renderTarget; f && this._scene.postProcessManager.directRender(this._postProcesses2, f, !0), this._engine.unBindFramebuffer(f ?? a, !0); } }), this._postProcesses.map((a) => { a.autoClear = !1; }); } /** * @returns The blur kernel size used by the glow. * Note: The value passed in the options is divided by 2 for back compatibility. */ _getEffectiveBlurKernelSize() { return this._options.blurKernelSize / 2; } /** * Checks for the readiness of the element composing the layer. * @param subMesh the mesh to check for * @param useInstances specify whether or not to use instances to render the mesh * @returns true if ready otherwise, false */ isReady(e, t) { const r = e.getMaterial(), n = e.getRenderingMesh(); if (!r || !n) return !1; const i = r.emissiveTexture; return super._isReady(e, t, i); } /** * Returns whether or not the layer needs stencil enabled during the mesh rendering. */ needStencil() { return !1; } /** * Returns true if the mesh can be rendered, otherwise false. * @param mesh The mesh to render * @param material The material used on the mesh * @returns true if it can be rendered otherwise false */ _canRenderMesh(e, t) { return !0; } /** * Implementation specific of rendering the generating effect on the main canvas. * @param effect The effect used to render through */ _internalRender(e) { e.setTexture("textureSampler", this._blurTexture1), e.setTexture("textureSampler2", this._blurTexture2), e.setFloat("offset", this._intensity); const t = this._engine, r = t.getStencilBuffer(); t.setStencilBuffer(!1), t.drawElementsType(gt.TriangleFillMode, 0, 6), t.setStencilBuffer(r); } /** * Sets the required values for both the emissive texture and and the main color. * @param mesh * @param subMesh * @param material */ _setEmissiveTextureAndColor(e, t, r) { var n; let i = 1; if (this.customEmissiveTextureSelector ? this._emissiveTextureAndColor.texture = this.customEmissiveTextureSelector(e, t, r) : r ? (this._emissiveTextureAndColor.texture = r.emissiveTexture, this._emissiveTextureAndColor.texture && (i = this._emissiveTextureAndColor.texture.level)) : this._emissiveTextureAndColor.texture = null, this.customEmissiveColorSelector) this.customEmissiveColorSelector(e, t, r, this._emissiveTextureAndColor.color); else if (r.emissiveColor) { const s = (n = r.emissiveIntensity) !== null && n !== void 0 ? n : 1; i *= s, this._emissiveTextureAndColor.color.set(r.emissiveColor.r * i, r.emissiveColor.g * i, r.emissiveColor.b * i, r.alpha); } else this._emissiveTextureAndColor.color.set(this.neutralColor.r, this.neutralColor.g, this.neutralColor.b, this.neutralColor.a); } /** * Returns true if the mesh should render, otherwise false. * @param mesh The mesh to render * @returns true if it should render otherwise false */ _shouldRenderMesh(e) { return this.hasMesh(e); } /** * Adds specific effects defines. * @param defines The defines to add specifics to. */ _addCustomEffectDefines(e) { e.push("#define GLOW"); } /** * Add a mesh in the exclusion list to prevent it to impact or being impacted by the glow layer. * @param mesh The mesh to exclude from the glow layer */ addExcludedMesh(e) { this._excludedMeshes.indexOf(e.uniqueId) === -1 && this._excludedMeshes.push(e.uniqueId); } /** * Remove a mesh from the exclusion list to let it impact or being impacted by the glow layer. * @param mesh The mesh to remove */ removeExcludedMesh(e) { const t = this._excludedMeshes.indexOf(e.uniqueId); t !== -1 && this._excludedMeshes.splice(t, 1); } /** * Add a mesh in the inclusion list to impact or being impacted by the glow layer. * @param mesh The mesh to include in the glow layer */ addIncludedOnlyMesh(e) { this._includedOnlyMeshes.indexOf(e.uniqueId) === -1 && this._includedOnlyMeshes.push(e.uniqueId); } /** * Remove a mesh from the Inclusion list to prevent it to impact or being impacted by the glow layer. * @param mesh The mesh to remove */ removeIncludedOnlyMesh(e) { const t = this._includedOnlyMeshes.indexOf(e.uniqueId); t !== -1 && this._includedOnlyMeshes.splice(t, 1); } /** * Determine if a given mesh will be used in the glow layer * @param mesh The mesh to test * @returns true if the mesh will be highlighted by the current glow layer */ hasMesh(e) { return super.hasMesh(e) ? this._includedOnlyMeshes.length ? this._includedOnlyMeshes.indexOf(e.uniqueId) !== -1 : this._excludedMeshes.length ? this._excludedMeshes.indexOf(e.uniqueId) === -1 : !0 : !1; } /** * Defines whether the current material of the mesh should be use to render the effect. * @param mesh defines the current mesh to render */ _useMeshMaterial(e) { return this._meshesUsingTheirOwnMaterials.length == 0 ? !1 : this._meshesUsingTheirOwnMaterials.indexOf(e.uniqueId) > -1; } /** * Add a mesh to be rendered through its own material and not with emissive only. * @param mesh The mesh for which we need to use its material */ referenceMeshToUseItsOwnMaterial(e) { e.resetDrawCache(this._mainTexture.renderPassId), this._meshesUsingTheirOwnMaterials.push(e.uniqueId), e.onDisposeObservable.add(() => { this._disposeMesh(e); }); } /** * Remove a mesh from being rendered through its own material and not with emissive only. * @param mesh The mesh for which we need to not use its material */ unReferenceMeshFromUsingItsOwnMaterial(e) { let t = this._meshesUsingTheirOwnMaterials.indexOf(e.uniqueId); for (; t >= 0; ) this._meshesUsingTheirOwnMaterials.splice(t, 1), t = this._meshesUsingTheirOwnMaterials.indexOf(e.uniqueId); e.resetDrawCache(this._mainTexture.renderPassId); } /** * Free any resources and references associated to a mesh. * Internal use * @param mesh The mesh to free. * @internal */ _disposeMesh(e) { this.removeIncludedOnlyMesh(e), this.removeExcludedMesh(e); } /** * Gets the class name of the effect layer * @returns the string with the class name of the effect layer */ getClassName() { return "GlowLayer"; } /** * Serializes this glow layer * @returns a serialized glow layer object */ serialize() { const e = jt.Serialize(this); e.customType = "BABYLON.GlowLayer"; let t; if (e.includedMeshes = [], this._includedOnlyMeshes.length) for (t = 0; t < this._includedOnlyMeshes.length; t++) { const r = this._scene.getMeshByUniqueId(this._includedOnlyMeshes[t]); r && e.includedMeshes.push(r.id); } if (e.excludedMeshes = [], this._excludedMeshes.length) for (t = 0; t < this._excludedMeshes.length; t++) { const r = this._scene.getMeshByUniqueId(this._excludedMeshes[t]); r && e.excludedMeshes.push(r.id); } return e; } /** * Creates a Glow Layer from parsed glow layer data * @param parsedGlowLayer defines glow layer data * @param scene defines the current scene * @param rootUrl defines the root URL containing the glow layer information * @returns a parsed Glow Layer */ static Parse(e, t, r) { const n = jt.Parse(() => new zl(e.name, t, e.options), e, t, r); let i; for (i = 0; i < e.excludedMeshes.length; i++) { const s = t.getMeshById(e.excludedMeshes[i]); s && n.addExcludedMesh(s); } for (i = 0; i < e.includedMeshes.length; i++) { const s = t.getMeshById(e.includedMeshes[i]); s && n.addIncludedOnlyMesh(s); } return n; } } zl.EffectName = "GlowLayer"; zl.DefaultBlurKernelSize = 32; zl.DefaultTextureRatio = 0.5; C([ M() ], zl.prototype, "blurKernelSize", null); C([ M() ], zl.prototype, "intensity", null); C([ M("options") ], zl.prototype, "_options", void 0); Ue("BABYLON.GlowLayer", zl); const x3e = "glowBlurPostProcessPixelShader", D3e = `varying vec2 vUV;uniform sampler2D textureSampler;uniform vec2 screenSize;uniform vec2 direction;uniform float blurWidth;float getLuminance(vec3 color) {return dot(color,vec3(0.2126,0.7152,0.0722));} #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {float weights[7];weights[0]=0.05;weights[1]=0.1;weights[2]=0.2;weights[3]=0.3;weights[4]=0.2;weights[5]=0.1;weights[6]=0.05;vec2 texelSize=vec2(1.0/screenSize.x,1.0/screenSize.y);vec2 texelStep=texelSize*direction*blurWidth;vec2 start=vUV-3.0*texelStep;vec4 baseColor=vec4(0.,0.,0.,0.);vec2 texelOffset=vec2(0.,0.);for (int i=0; i<7; i++) {vec4 texel=texture2D(textureSampler,start+texelOffset);baseColor.a+=texel.a*weights[i];float luminance=getLuminance(baseColor.rgb);float luminanceTexel=getLuminance(texel.rgb);float choice=step(luminanceTexel,luminance);baseColor.rgb=choice*baseColor.rgb+(1.0-choice)*texel.rgb;texelOffset+=texelStep;} gl_FragColor=baseColor;}`; Le.ShadersStore[x3e] = D3e; J1.prototype.getHighlightLayerByName = function(A) { var e; for (let t = 0; t < ((e = this.effectLayers) === null || e === void 0 ? void 0 : e.length); t++) if (this.effectLayers[t].name === A && this.effectLayers[t].getEffectName() === Ad.EffectName) return this.effectLayers[t]; return null; }; class YG extends kr { constructor(e, t, r, n, i, s = We.BILINEAR_SAMPLINGMODE, a, f) { super(e, "glowBlurPostProcess", ["screenSize", "direction", "blurWidth"], null, n, i, s, a, f), this.direction = t, this.kernel = r, this.onApplyObservable.add((o) => { o.setFloat2("screenSize", this.width, this.height), o.setVector2("direction", this.direction), o.setFloat("blurWidth", this.kernel); }); } } class Ad extends mc { /** * Specifies the horizontal size of the blur. */ set blurHorizontalSize(e) { this._horizontalBlurPostprocess.kernel = e, this._options.blurHorizontalSize = e; } /** * Specifies the vertical size of the blur. */ set blurVerticalSize(e) { this._verticalBlurPostprocess.kernel = e, this._options.blurVerticalSize = e; } /** * Gets the horizontal size of the blur. */ get blurHorizontalSize() { return this._horizontalBlurPostprocess.kernel; } /** * Gets the vertical size of the blur. */ get blurVerticalSize() { return this._verticalBlurPostprocess.kernel; } /** * Instantiates a new highlight Layer and references it to the scene.. * @param name The name of the layer * @param scene The scene to use the layer in * @param options Sets of none mandatory options to use with the layer (see IHighlightLayerOptions for more information) */ constructor(e, t, r) { super(e, t), this.name = e, this.innerGlow = !0, this.outerGlow = !0, this.onBeforeBlurObservable = new Oe(), this.onAfterBlurObservable = new Oe(), this._instanceGlowingMeshStencilReference = Ad.GlowingMeshStencilReference++, this._meshes = {}, this._excludedMeshes = {}, this.neutralColor = Ad.NeutralColor, this._engine.isStencilEnable || Se.Warn("Rendering the Highlight Layer requires the stencil to be active on the canvas. var engine = new Engine(canvas, antialias, { stencil: true }"), this._options = Object.assign({ mainTextureRatio: 0.5, blurTextureSizeRatio: 0.5, blurHorizontalSize: 1, blurVerticalSize: 1, alphaBlendingMode: 2, camera: null, renderingGroupId: -1, mainTextureType: 0 }, r), this._init({ alphaBlendingMode: this._options.alphaBlendingMode, camera: this._options.camera, mainTextureFixedSize: this._options.mainTextureFixedSize, mainTextureRatio: this._options.mainTextureRatio, renderingGroupId: this._options.renderingGroupId, mainTextureType: this._options.mainTextureType }), this._shouldRender = !1; } /** * Get the effect name of the layer. * @returns The effect name */ getEffectName() { return Ad.EffectName; } _numInternalDraws() { return 2; } /** * Create the merge effect. This is the shader use to blit the information back * to the main canvas at the end of the scene rendering. */ _createMergeEffect() { return this._engine.createEffect("glowMapMerge", [J.PositionKind], ["offset"], ["textureSampler"], this._options.isStroke ? `#define STROKE ` : void 0); } /** * Creates the render target textures and post processes used in the highlight layer. */ _createTextureAndPostProcesses() { let e = this._mainTextureDesiredSize.width * this._options.blurTextureSizeRatio, t = this._mainTextureDesiredSize.height * this._options.blurTextureSizeRatio; e = this._engine.needPOTTextures ? Ge.GetExponentOfTwo(e, this._maxSize) : e, t = this._engine.needPOTTextures ? Ge.GetExponentOfTwo(t, this._maxSize) : t; let r = 0; this._engine.getCaps().textureHalfFloatRender ? r = 2 : r = 0, this._blurTexture = new Ta("HighlightLayerBlurRTT", { width: e, height: t }, this._scene, !1, !0, r), this._blurTexture.wrapU = We.CLAMP_ADDRESSMODE, this._blurTexture.wrapV = We.CLAMP_ADDRESSMODE, this._blurTexture.anisotropicFilteringLevel = 16, this._blurTexture.updateSamplingMode(We.TRILINEAR_SAMPLINGMODE), this._blurTexture.renderParticles = !1, this._blurTexture.ignoreCameraViewport = !0, this._textures = [this._blurTexture], this._options.alphaBlendingMode === 2 ? (this._downSamplePostprocess = new lg("HighlightLayerPPP", this._options.blurTextureSizeRatio, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine()), this._downSamplePostprocess.externalTextureSamplerBinding = !0, this._downSamplePostprocess.onApplyObservable.add((n) => { n.setTexture("textureSampler", this._mainTexture); }), this._horizontalBlurPostprocess = new YG("HighlightLayerHBP", new at(1, 0), this._options.blurHorizontalSize, 1, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine()), this._horizontalBlurPostprocess.onApplyObservable.add((n) => { n.setFloat2("screenSize", e, t); }), this._verticalBlurPostprocess = new YG("HighlightLayerVBP", new at(0, 1), this._options.blurVerticalSize, 1, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine()), this._verticalBlurPostprocess.onApplyObservable.add((n) => { n.setFloat2("screenSize", e, t); }), this._postProcesses = [this._downSamplePostprocess, this._horizontalBlurPostprocess, this._verticalBlurPostprocess]) : (this._horizontalBlurPostprocess = new c9("HighlightLayerHBP", new at(1, 0), this._options.blurHorizontalSize / 2, { width: e, height: t }, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1, r), this._horizontalBlurPostprocess.width = e, this._horizontalBlurPostprocess.height = t, this._horizontalBlurPostprocess.externalTextureSamplerBinding = !0, this._horizontalBlurPostprocess.onApplyObservable.add((n) => { n.setTexture("textureSampler", this._mainTexture); }), this._verticalBlurPostprocess = new c9("HighlightLayerVBP", new at(0, 1), this._options.blurVerticalSize / 2, { width: e, height: t }, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1, r), this._postProcesses = [this._horizontalBlurPostprocess, this._verticalBlurPostprocess]), this._mainTexture.onAfterUnbindObservable.add(() => { this.onBeforeBlurObservable.notifyObservers(this); const n = this._blurTexture.renderTarget; n && (this._scene.postProcessManager.directRender(this._postProcesses, n, !0), this._engine.unBindFramebuffer(n, !0)), this.onAfterBlurObservable.notifyObservers(this); }), this._postProcesses.map((n) => { n.autoClear = !1; }); } /** * Returns whether or not the layer needs stencil enabled during the mesh rendering. */ needStencil() { return !0; } /** * Checks for the readiness of the element composing the layer. * @param subMesh the mesh to check for * @param useInstances specify whether or not to use instances to render the mesh * @returns true if ready otherwise, false */ isReady(e, t) { const r = e.getMaterial(), n = e.getRenderingMesh(); if (!r || !n || !this._meshes) return !1; let i = null; const s = this._meshes[n.uniqueId]; return s && s.glowEmissiveOnly && r && (i = r.emissiveTexture), super._isReady(e, t, i); } /** * Implementation specific of rendering the generating effect on the main canvas. * @param effect The effect used to render through * @param renderIndex */ _internalRender(e, t) { e.setTexture("textureSampler", this._blurTexture); const r = this._engine; r.cacheStencilState(), r.setStencilOperationPass(7681), r.setStencilOperationFail(7680), r.setStencilOperationDepthFail(7680), r.setStencilMask(0), r.setStencilBuffer(!0), r.setStencilFunctionReference(this._instanceGlowingMeshStencilReference), this.outerGlow && t === 0 && (e.setFloat("offset", 0), r.setStencilFunction(517), r.drawElementsType(gt.TriangleFillMode, 0, 6)), this.innerGlow && t === 1 && (e.setFloat("offset", 1), r.setStencilFunction(514), r.drawElementsType(gt.TriangleFillMode, 0, 6)), r.restoreStencilState(); } /** * Returns true if the layer contains information to display, otherwise false. */ shouldRender() { return super.shouldRender() ? !!this._meshes : !1; } /** * Returns true if the mesh should render, otherwise false. * @param mesh The mesh to render * @returns true if it should render otherwise false */ _shouldRenderMesh(e) { return !(this._excludedMeshes && this._excludedMeshes[e.uniqueId] || !super.hasMesh(e)); } /** * Returns true if the mesh can be rendered, otherwise false. * @param mesh The mesh to render * @param material The material used on the mesh * @returns true if it can be rendered otherwise false */ _canRenderMesh(e, t) { return !0; } /** * Adds specific effects defines. * @param defines The defines to add specifics to. */ _addCustomEffectDefines(e) { e.push("#define HIGHLIGHT"); } /** * Sets the required values for both the emissive texture and and the main color. * @param mesh * @param subMesh * @param material */ _setEmissiveTextureAndColor(e, t, r) { const n = this._meshes[e.uniqueId]; n ? this._emissiveTextureAndColor.color.set(n.color.r, n.color.g, n.color.b, 1) : this._emissiveTextureAndColor.color.set(this.neutralColor.r, this.neutralColor.g, this.neutralColor.b, this.neutralColor.a), n && n.glowEmissiveOnly && r ? (this._emissiveTextureAndColor.texture = r.emissiveTexture, this._emissiveTextureAndColor.color.set(1, 1, 1, 1)) : this._emissiveTextureAndColor.texture = null; } /** * Add a mesh in the exclusion list to prevent it to impact or being impacted by the highlight layer. * @param mesh The mesh to exclude from the highlight layer */ addExcludedMesh(e) { if (!this._excludedMeshes) return; if (!this._excludedMeshes[e.uniqueId]) { const r = { mesh: e, beforeBind: null, afterRender: null, stencilState: !1 }; r.beforeBind = e.onBeforeBindObservable.add((n) => { r.stencilState = n.getEngine().getStencilBuffer(), n.getEngine().setStencilBuffer(!1); }), r.afterRender = e.onAfterRenderObservable.add((n) => { n.getEngine().setStencilBuffer(r.stencilState); }), this._excludedMeshes[e.uniqueId] = r; } } /** * Remove a mesh from the exclusion list to let it impact or being impacted by the highlight layer. * @param mesh The mesh to highlight */ removeExcludedMesh(e) { if (!this._excludedMeshes) return; const t = this._excludedMeshes[e.uniqueId]; t && (t.beforeBind && e.onBeforeBindObservable.remove(t.beforeBind), t.afterRender && e.onAfterRenderObservable.remove(t.afterRender)), this._excludedMeshes[e.uniqueId] = null; } /** * Determine if a given mesh will be highlighted by the current HighlightLayer * @param mesh mesh to test * @returns true if the mesh will be highlighted by the current HighlightLayer */ hasMesh(e) { return !this._meshes || !super.hasMesh(e) ? !1 : this._meshes[e.uniqueId] !== void 0 && this._meshes[e.uniqueId] !== null; } /** * Add a mesh in the highlight layer in order to make it glow with the chosen color. * @param mesh The mesh to highlight * @param color The color of the highlight * @param glowEmissiveOnly Extract the glow from the emissive texture */ addMesh(e, t, r = !1) { if (!this._meshes) return; const n = this._meshes[e.uniqueId]; n ? n.color = t : (this._meshes[e.uniqueId] = { mesh: e, color: t, // Lambda required for capture due to Observable this context observerHighlight: e.onBeforeBindObservable.add((i) => { this.isEnabled && (this._excludedMeshes && this._excludedMeshes[i.uniqueId] ? this._defaultStencilReference(i) : i.getScene().getEngine().setStencilFunctionReference(this._instanceGlowingMeshStencilReference)); }), observerDefault: e.onAfterRenderObservable.add((i) => { this.isEnabled && this._defaultStencilReference(i); }), glowEmissiveOnly: r }, e.onDisposeObservable.add(() => { this._disposeMesh(e); })), this._shouldRender = !0; } /** * Remove a mesh from the highlight layer in order to make it stop glowing. * @param mesh The mesh to highlight */ removeMesh(e) { if (!this._meshes) return; const t = this._meshes[e.uniqueId]; t && (t.observerHighlight && e.onBeforeBindObservable.remove(t.observerHighlight), t.observerDefault && e.onAfterRenderObservable.remove(t.observerDefault), delete this._meshes[e.uniqueId]), this._shouldRender = !1; for (const r in this._meshes) if (this._meshes[r]) { this._shouldRender = !0; break; } } /** * Remove all the meshes currently referenced in the highlight layer */ removeAllMeshes() { if (this._meshes) { for (const e in this._meshes) if (Object.prototype.hasOwnProperty.call(this._meshes, e)) { const t = this._meshes[e]; t && this.removeMesh(t.mesh); } } } /** * Force the stencil to the normal expected value for none glowing parts * @param mesh */ _defaultStencilReference(e) { e.getScene().getEngine().setStencilFunctionReference(Ad.NormalMeshStencilReference); } /** * Free any resources and references associated to a mesh. * Internal use * @param mesh The mesh to free. * @internal */ _disposeMesh(e) { this.removeMesh(e), this.removeExcludedMesh(e); } /** * Dispose the highlight layer and free resources. */ dispose() { if (this._meshes) { for (const e in this._meshes) { const t = this._meshes[e]; t && t.mesh && (t.observerHighlight && t.mesh.onBeforeBindObservable.remove(t.observerHighlight), t.observerDefault && t.mesh.onAfterRenderObservable.remove(t.observerDefault)); } this._meshes = null; } if (this._excludedMeshes) { for (const e in this._excludedMeshes) { const t = this._excludedMeshes[e]; t && (t.beforeBind && t.mesh.onBeforeBindObservable.remove(t.beforeBind), t.afterRender && t.mesh.onAfterRenderObservable.remove(t.afterRender)); } this._excludedMeshes = null; } super.dispose(); } /** * Gets the class name of the effect layer * @returns the string with the class name of the effect layer */ getClassName() { return "HighlightLayer"; } /** * Serializes this Highlight layer * @returns a serialized Highlight layer object */ serialize() { const e = jt.Serialize(this); if (e.customType = "BABYLON.HighlightLayer", e.meshes = [], this._meshes) for (const t in this._meshes) { const r = this._meshes[t]; r && e.meshes.push({ glowEmissiveOnly: r.glowEmissiveOnly, color: r.color.asArray(), meshId: r.mesh.id }); } if (e.excludedMeshes = [], this._excludedMeshes) for (const t in this._excludedMeshes) { const r = this._excludedMeshes[t]; r && e.excludedMeshes.push(r.mesh.id); } return e; } /** * Creates a Highlight layer from parsed Highlight layer data * @param parsedHightlightLayer defines the Highlight layer data * @param scene defines the current scene * @param rootUrl defines the root URL containing the Highlight layer information * @returns a parsed Highlight layer */ static Parse(e, t, r) { const n = jt.Parse(() => new Ad(e.name, t, e.options), e, t, r); let i; for (i = 0; i < e.excludedMeshes.length; i++) { const s = t.getMeshById(e.excludedMeshes[i]); s && n.addExcludedMesh(s); } for (i = 0; i < e.meshes.length; i++) { const s = e.meshes[i], a = t.getMeshById(s.meshId); a && n.addMesh(a, Ne.FromArray(s.color), s.glowEmissiveOnly); } return n; } } Ad.EffectName = "HighlightLayer"; Ad.NeutralColor = new xt(0, 0, 0, 0); Ad.GlowingMeshStencilReference = 2; Ad.NormalMeshStencilReference = 1; C([ M() ], Ad.prototype, "innerGlow", void 0); C([ M() ], Ad.prototype, "outerGlow", void 0); C([ M() ], Ad.prototype, "blurHorizontalSize", null); C([ M() ], Ad.prototype, "blurVerticalSize", null); C([ M("options") ], Ad.prototype, "_options", void 0); Ue("BABYLON.HighlightLayer", Ad); class Qte { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_LAYER, this.scene = e || gr.LastCreatedScene, this.scene && (this._engine = this.scene.getEngine(), this.scene.layers = []); } /** * Registers the component in a given scene */ register() { this.scene._beforeCameraDrawStage.registerStep(Ot.STEP_BEFORECAMERADRAW_LAYER, this, this._drawCameraBackground), this.scene._afterCameraDrawStage.registerStep(Ot.STEP_AFTERCAMERADRAW_LAYER, this, this._drawCameraForegroundWithPostProcessing), this.scene._afterCameraPostProcessStage.registerStep(Ot.STEP_AFTERCAMERAPOSTPROCESS_LAYER, this, this._drawCameraForegroundWithoutPostProcessing), this.scene._beforeRenderTargetDrawStage.registerStep(Ot.STEP_BEFORERENDERTARGETDRAW_LAYER, this, this._drawRenderTargetBackground), this.scene._afterRenderTargetDrawStage.registerStep(Ot.STEP_AFTERRENDERTARGETDRAW_LAYER, this, this._drawRenderTargetForegroundWithPostProcessing), this.scene._afterRenderTargetPostProcessStage.registerStep(Ot.STEP_AFTERRENDERTARGETPOSTPROCESS_LAYER, this, this._drawRenderTargetForegroundWithoutPostProcessing); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { const e = this.scene.layers; for (const t of e) t._rebuild(); } /** * Disposes the component and the associated resources. */ dispose() { const e = this.scene.layers; for (; e.length; ) e[0].dispose(); } _draw(e) { const t = this.scene.layers; if (t.length) { this._engine.setDepthBuffer(!1); for (const r of t) e(r) && r.render(); this._engine.setDepthBuffer(!0); } } _drawCameraPredicate(e, t, r, n) { return !e.renderOnlyInRenderTargetTextures && e.isBackground === t && e.applyPostProcess === r && (e.layerMask & n) !== 0; } _drawCameraBackground(e) { this._draw((t) => this._drawCameraPredicate(t, !0, !0, e.layerMask)); } _drawCameraForegroundWithPostProcessing(e) { this._draw((t) => this._drawCameraPredicate(t, !1, !0, e.layerMask)); } _drawCameraForegroundWithoutPostProcessing(e) { this._draw((t) => this._drawCameraPredicate(t, !1, !1, e.layerMask)); } _drawRenderTargetPredicate(e, t, r, n, i) { return e.renderTargetTextures.length > 0 && e.isBackground === t && e.applyPostProcess === r && e.renderTargetTextures.indexOf(i) > -1 && (e.layerMask & n) !== 0; } _drawRenderTargetBackground(e) { this._draw((t) => this._drawRenderTargetPredicate(t, !0, !0, this.scene.activeCamera.layerMask, e)); } _drawRenderTargetForegroundWithPostProcessing(e) { this._draw((t) => this._drawRenderTargetPredicate(t, !1, !0, this.scene.activeCamera.layerMask, e)); } _drawRenderTargetForegroundWithoutPostProcessing(e) { this._draw((t) => this._drawRenderTargetPredicate(t, !1, !1, this.scene.activeCamera.layerMask, e)); } /** * Adds all the elements from the container to the scene * @param container the container holding the elements */ addFromContainer(e) { e.layers && e.layers.forEach((t) => { this.scene.layers.push(t); }); } /** * Removes all the elements in the container from the scene * @param container contains the elements to remove * @param dispose if the removed element should be disposed (default: false) */ removeFromContainer(e, t = !1) { e.layers && e.layers.forEach((r) => { const n = this.scene.layers.indexOf(r); n !== -1 && this.scene.layers.splice(n, 1), t && r.dispose(); }); } } const j3e = "layerPixelShader", w3e = `varying vec2 vUV;uniform sampler2D textureSampler;uniform vec4 color; #include #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #define CUSTOM_FRAGMENT_MAIN_BEGIN vec4 baseColor=texture2D(textureSampler,vUV); #ifdef LINEAR baseColor.rgb=toGammaSpace(baseColor.rgb); #endif #ifdef ALPHATEST if (baseColor.a<0.4) discard; #endif gl_FragColor=baseColor*color; #define CUSTOM_FRAGMENT_MAIN_END }`; Le.ShadersStore[j3e] = w3e; const m3e = "layerVertexShader", B3e = `attribute vec2 position;uniform vec2 scale;uniform vec2 offset;uniform mat4 textureMatrix;varying vec2 vUV;const vec2 madd=vec2(0.5,0.5); #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN vec2 shiftedPosition=position*scale+offset;vUV=vec2(textureMatrix*vec4(shiftedPosition*madd+madd,1.0,0.0));gl_Position=vec4(shiftedPosition,0.0,1.0); #define CUSTOM_VERTEX_MAIN_END }`; Le.ShadersStore[m3e] = B3e; class Yte { /** * Determines if the layer is drawn before (true) or after (false) post-processing. * If the layer is background, it is always before. */ set applyPostProcess(e) { this._applyPostProcess = e; } get applyPostProcess() { return this.isBackground || this._applyPostProcess; } /** * Back compatibility with callback before the onDisposeObservable existed. * The set callback will be triggered when the layer has been disposed. */ set onDispose(e) { this._onDisposeObserver && this.onDisposeObservable.remove(this._onDisposeObserver), this._onDisposeObserver = this.onDisposeObservable.add(e); } /** * Back compatibility with callback before the onBeforeRenderObservable existed. * The set callback will be triggered just before rendering the layer. */ set onBeforeRender(e) { this._onBeforeRenderObserver && this.onBeforeRenderObservable.remove(this._onBeforeRenderObserver), this._onBeforeRenderObserver = this.onBeforeRenderObservable.add(e); } /** * Back compatibility with callback before the onAfterRenderObservable existed. * The set callback will be triggered just after rendering the layer. */ set onAfterRender(e) { this._onAfterRenderObserver && this.onAfterRenderObservable.remove(this._onAfterRenderObserver), this._onAfterRenderObserver = this.onAfterRenderObservable.add(e); } /** * Instantiates a new layer. * This represents a full screen 2d layer. * This can be useful to display a picture in the background of your scene for instance. * @see https://www.babylonjs-playground.com/#08A2BS#1 * @param name Define the name of the layer in the scene * @param imgUrl Define the url of the texture to display in the layer * @param scene Define the scene the layer belongs to * @param isBackground Defines whether the layer is displayed in front or behind the scene * @param color Defines a color for the layer */ constructor(e, t, r, n, i) { this.name = e, this._applyPostProcess = !0, this.scale = new at(1, 1), this.offset = new at(0, 0), this.alphaBlendingMode = 2, this.layerMask = 268435455, this.renderTargetTextures = [], this.renderOnlyInRenderTargetTextures = !1, this.isEnabled = !0, this._vertexBuffers = {}, this.onDisposeObservable = new Oe(), this.onBeforeRenderObservable = new Oe(), this.onAfterRenderObservable = new Oe(), this.texture = t ? new We(t, r, !0) : null, this.isBackground = n === void 0 ? !0 : n, this.color = i === void 0 ? new xt(1, 1, 1, 1) : i, this._scene = r || gr.LastCreatedScene; let s = this._scene._getComponent(Ot.NAME_LAYER); s || (s = new Qte(this._scene), this._scene._addComponent(s)), this._scene.layers.push(this); const a = this._scene.getEngine(); this._drawWrapper = new zo(a); const f = []; f.push(1, 1), f.push(-1, 1), f.push(-1, -1), f.push(1, -1); const o = new J(a, f, J.PositionKind, !1, !1, 2); this._vertexBuffers[J.PositionKind] = o, this._createIndexBuffer(); } _createIndexBuffer() { const e = this._scene.getEngine(), t = []; t.push(0), t.push(1), t.push(2), t.push(0), t.push(2), t.push(3), this._indexBuffer = e.createIndexBuffer(t); } /** @internal */ _rebuild() { const e = this._vertexBuffers[J.PositionKind]; e && e._rebuild(), this._createIndexBuffer(); } /** * Checks if the layer is ready to be rendered * @returns true if the layer is ready. False otherwise. */ isReady() { var e; const t = this._scene.getEngine(); let r = ""; this.alphaTest && (r = "#define ALPHATEST"), this.texture && !this.texture.gammaSpace && (r += ` #define LINEAR`), this._previousDefines !== r && (this._previousDefines = r, this._drawWrapper.effect = t.createEffect("layer", [J.PositionKind], ["textureMatrix", "color", "scale", "offset"], ["textureSampler"], r)); const n = this._drawWrapper.effect; return (n == null ? void 0 : n.isReady()) && ((e = this.texture) === null || e === void 0 ? void 0 : e.isReady()); } /** * Renders the layer in the scene. */ render() { if (!this.isEnabled) return; const e = this._scene.getEngine(); if (!this.isReady()) return; const t = this._drawWrapper.effect; this.onBeforeRenderObservable.notifyObservers(this), e.enableEffect(this._drawWrapper), e.setState(!1), t.setTexture("textureSampler", this.texture), t.setMatrix("textureMatrix", this.texture.getTextureMatrix()), t.setFloat4("color", this.color.r, this.color.g, this.color.b, this.color.a), t.setVector2("offset", this.offset), t.setVector2("scale", this.scale), e.bindBuffers(this._vertexBuffers, this._indexBuffer, t), this.alphaTest ? e.drawElementsType(gt.TriangleFillMode, 0, 6) : (e.setAlphaMode(this.alphaBlendingMode), e.drawElementsType(gt.TriangleFillMode, 0, 6), e.setAlphaMode(0)), this.onAfterRenderObservable.notifyObservers(this); } /** * Disposes and releases the associated resources. */ dispose() { const e = this._vertexBuffers[J.PositionKind]; e && (e.dispose(), this._vertexBuffers[J.PositionKind] = null), this._indexBuffer && (this._scene.getEngine()._releaseBuffer(this._indexBuffer), this._indexBuffer = null), this.texture && (this.texture.dispose(), this.texture = null), this.renderTargetTextures = []; const t = this._scene.layers.indexOf(this); this._scene.layers.splice(t, 1), this.onDisposeObservable.notifyObservers(this), this.onDisposeObservable.clear(), this.onAfterRenderObservable.clear(), this.onBeforeRenderObservable.clear(); } } class ly { /** * Creates a new Lens Flare. * This represents one of the lens effect in a `lensFlareSystem`. * It controls one of the individual texture used in the effect. * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/lenseFlare * @param size Define the size of the lens flare (a floating value between 0 and 1) * @param position Define the position of the lens flare in the system. (a floating value between -1 and 1). A value of 0 is located on the emitter. A value greater than 0 is beyond the emitter and a value lesser than 0 is behind. * @param color Define the lens color * @param imgUrl Define the lens texture url * @param system Define the `lensFlareSystem` this flare is part of * @returns The newly created Lens Flare */ static AddFlare(e, t, r, n, i) { return new ly(e, t, r, n, i); } /** * Instantiates a new Lens Flare. * This represents one of the lens effect in a `lensFlareSystem`. * It controls one of the individual texture used in the effect. * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/lenseFlare * @param size Define the size of the lens flare in the system (a floating value between 0 and 1) * @param position Define the position of the lens flare in the system. (a floating value between -1 and 1). A value of 0 is located on the emitter. A value greater than 0 is beyond the emitter and a value lesser than 0 is behind. * @param color Define the lens color * @param imgUrl Define the lens texture url * @param system Define the `lensFlareSystem` this flare is part of */ constructor(e, t, r, n, i) { this.size = e, this.position = t, this.alphaMode = 6, this.color = r || new Ne(1, 1, 1), this.texture = n ? new We(n, i.getScene(), !0) : null, this._system = i; const s = i.scene.getEngine(); this._drawWrapper = new zo(s), this._drawWrapper.effect = s.createEffect("lensFlare", [J.PositionKind], ["color", "viewportMatrix"], ["textureSampler"], ""), i.lensFlares.push(this); } /** * Dispose and release the lens flare with its associated resources. */ dispose() { this.texture && this.texture.dispose(); const e = this._system.lensFlares.indexOf(this); this._system.lensFlares.splice(e, 1); } } const W3e = "lensFlarePixelShader", S3e = `varying vec2 vUV;uniform sampler2D textureSampler;uniform vec4 color; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #define CUSTOM_FRAGMENT_MAIN_BEGIN vec4 baseColor=texture2D(textureSampler,vUV);gl_FragColor=baseColor*color; #define CUSTOM_FRAGMENT_MAIN_END }`; Le.ShadersStore[W3e] = S3e; const U3e = "lensFlareVertexShader", I3e = `attribute vec2 position;uniform mat4 viewportMatrix;varying vec2 vUV;const vec2 madd=vec2(0.5,0.5); #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN vUV=position*madd+madd;gl_Position=viewportMatrix*vec4(position,0.0,1.0); #define CUSTOM_VERTEX_MAIN_END }`; Le.ShadersStore[U3e] = I3e; class gm { /** Gets the scene */ get scene() { return this._scene; } /** * Instantiates a lens flare system. * This represents a Lens Flare System or the shiny effect created by the light reflection on the camera lenses. * It is usually composed of several `lensFlare`. * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/lenseFlare * @param name Define the name of the lens flare system in the scene * @param emitter Define the source (the emitter) of the lens flares (it can be a camera, a light or a mesh). * @param scene Define the scene the lens flare system belongs to */ constructor(e, t, r) { this.name = e, this.lensFlares = [], this.borderLimit = 300, this.viewportBorder = 0, this.layerMask = 268435455, this._vertexBuffers = {}, this._isEnabled = !0, this._scene = r || gr.LastCreatedScene, gm._SceneComponentInitialization(this._scene), this._emitter = t, this.id = e, r.lensFlareSystems.push(this), this.meshesSelectionPredicate = (s) => r.activeCamera && s.material && s.isVisible && s.isEnabled() && s.isBlocker && (s.layerMask & r.activeCamera.layerMask) != 0; const n = r.getEngine(), i = []; i.push(1, 1), i.push(-1, 1), i.push(-1, -1), i.push(1, -1), this._vertexBuffers[J.PositionKind] = new J(n, i, J.PositionKind, !1, !1, 2), this._createIndexBuffer(); } _createIndexBuffer() { const e = []; e.push(0), e.push(1), e.push(2), e.push(0), e.push(2), e.push(3), this._indexBuffer = this._scene.getEngine().createIndexBuffer(e); } /** * Define if the lens flare system is enabled. */ get isEnabled() { return this._isEnabled; } set isEnabled(e) { this._isEnabled = e; } /** * Get the scene the effects belongs to. * @returns the scene holding the lens flare system */ getScene() { return this._scene; } /** * Get the emitter of the lens flare system. * It defines the source of the lens flares (it can be a camera, a light or a mesh). * @returns the emitter of the lens flare system */ getEmitter() { return this._emitter; } /** * Set the emitter of the lens flare system. * It defines the source of the lens flares (it can be a camera, a light or a mesh). * @param newEmitter Define the new emitter of the system */ setEmitter(e) { this._emitter = e; } /** * Get the lens flare system emitter position. * The emitter defines the source of the lens flares (it can be a camera, a light or a mesh). * @returns the position */ getEmitterPosition() { return this._emitter.getAbsolutePosition ? this._emitter.getAbsolutePosition() : this._emitter.position; } /** * @internal */ computeEffectivePosition(e) { let t = this.getEmitterPosition(); t = S.Project(t, he.Identity(), this._scene.getTransformMatrix(), e), this._positionX = t.x, this._positionY = t.y, t = S.TransformCoordinates(this.getEmitterPosition(), this._scene.getViewMatrix()), this.viewportBorder > 0 && (e.x -= this.viewportBorder, e.y -= this.viewportBorder, e.width += this.viewportBorder * 2, e.height += this.viewportBorder * 2, t.x += this.viewportBorder, t.y += this.viewportBorder, this._positionX += this.viewportBorder, this._positionY += this.viewportBorder); const r = this._scene.useRightHandedSystem; return t.z > 0 && !r || t.z < 0 && r ? (this._positionX > e.x && this._positionX < e.x + e.width && this._positionY > e.y && this._positionY < e.y + e.height, !0) : !1; } /** @internal */ _isVisible() { if (!this._isEnabled || !this._scene.activeCamera) return !1; const t = this.getEmitterPosition().subtract(this._scene.activeCamera.globalPosition), r = t.length(); t.normalize(); const n = new Hi(this._scene.activeCamera.globalPosition, t), i = this._scene.pickWithRay(n, this.meshesSelectionPredicate, !0); return !i || !i.hit || i.distance > r; } /** * @internal */ render() { if (!this._scene.activeCamera) return !1; const e = this._scene.getEngine(), r = this._scene.activeCamera.viewport.toGlobal(e.getRenderWidth(!0), e.getRenderHeight(!0)); if (!this.computeEffectivePosition(r) || !this._isVisible()) return !1; let n, i; this._positionX < this.borderLimit + r.x ? n = this.borderLimit + r.x - this._positionX : this._positionX > r.x + r.width - this.borderLimit ? n = this._positionX - r.x - r.width + this.borderLimit : n = 0, this._positionY < this.borderLimit + r.y ? i = this.borderLimit + r.y - this._positionY : this._positionY > r.y + r.height - this.borderLimit ? i = this._positionY - r.y - r.height + this.borderLimit : i = 0; let s = n > i ? n : i; s -= this.viewportBorder, s > this.borderLimit && (s = this.borderLimit); let a = 1 - Xt.Clamp(s / this.borderLimit, 0, 1); if (a < 0) return !1; a > 1 && (a = 1), this.viewportBorder > 0 && (r.x += this.viewportBorder, r.y += this.viewportBorder, r.width -= this.viewportBorder * 2, r.height -= this.viewportBorder * 2, this._positionX -= this.viewportBorder, this._positionY -= this.viewportBorder); const f = r.x + r.width / 2, o = r.y + r.height / 2, d = f - this._positionX, v = o - this._positionY; e.setState(!1), e.setDepthBuffer(!1); for (let u = 0; u < this.lensFlares.length; u++) { const l = this.lensFlares[u]; if (!l._drawWrapper.effect.isReady() || l.texture && !l.texture.isReady()) continue; e.enableEffect(l._drawWrapper), e.bindBuffers(this._vertexBuffers, this._indexBuffer, l._drawWrapper.effect), e.setAlphaMode(l.alphaMode); const P = f - d * l.position, p = o - v * l.position, c = l.size, H = l.size * e.getAspectRatio(this._scene.activeCamera, !0), T = 2 * (P / (r.width + r.x * 2)) - 1, q = 1 - 2 * (p / (r.height + r.y * 2)), b = he.FromValues(c / 2, 0, 0, 0, 0, H / 2, 0, 0, 0, 0, 1, 0, T, q, 0, 1); l._drawWrapper.effect.setMatrix("viewportMatrix", b), l._drawWrapper.effect.setTexture("textureSampler", l.texture), l._drawWrapper.effect.setFloat4("color", l.color.r * a, l.color.g * a, l.color.b * a, 1), e.drawElementsType(gt.TriangleFillMode, 0, 6); } return e.setDepthBuffer(!0), e.setAlphaMode(0), !0; } /** * Rebuilds the lens flare system */ rebuild() { var e; this._createIndexBuffer(); for (const t in this._vertexBuffers) (e = this._vertexBuffers[t]) === null || e === void 0 || e._rebuild(); } /** * Dispose and release the lens flare with its associated resources. */ dispose() { const e = this._vertexBuffers[J.PositionKind]; for (e && (e.dispose(), this._vertexBuffers[J.PositionKind] = null), this._indexBuffer && (this._scene.getEngine()._releaseBuffer(this._indexBuffer), this._indexBuffer = null); this.lensFlares.length; ) this.lensFlares[0].dispose(); const t = this._scene.lensFlareSystems.indexOf(this); this._scene.lensFlareSystems.splice(t, 1); } /** * Parse a lens flare system from a JSON representation * @param parsedLensFlareSystem Define the JSON to parse * @param scene Define the scene the parsed system should be instantiated in * @param rootUrl Define the rootUrl of the load sequence to easily find a load relative dependencies such as textures * @returns the parsed system */ static Parse(e, t, r) { const n = t.getLastEntryById(e.emitterId), i = e.name || "lensFlareSystem#" + e.emitterId, s = new gm(i, n, t); s.id = e.id || i, s.borderLimit = e.borderLimit; for (let a = 0; a < e.flares.length; a++) { const f = e.flares[a]; ly.AddFlare(f.size, f.position, Ne.FromArray(f.color), f.textureName ? r + f.textureName : "", s); } return s; } /** * Serialize the current Lens Flare System into a JSON representation. * @returns the serialized JSON */ serialize() { const e = {}; e.id = this.id, e.name = this.name, e.emitterId = this.getEmitter().id, e.borderLimit = this.borderLimit, e.flares = []; for (let t = 0; t < this.lensFlares.length; t++) { const r = this.lensFlares[t]; e.flares.push({ size: r.size, position: r.position, color: r.color.asArray(), textureName: ye.GetFilename(r.texture ? r.texture.name : "") }); } return e; } } gm._SceneComponentInitialization = (A) => { throw qn("LensFlareSystemSceneComponent"); }; J1.AddParser(Ot.NAME_LENSFLARESYSTEM, (A, e, t, r) => { if (A.lensFlareSystems !== void 0 && A.lensFlareSystems !== null) { t.lensFlareSystems || (t.lensFlareSystems = []); for (let n = 0, i = A.lensFlareSystems.length; n < i; n++) { const s = A.lensFlareSystems[n], a = gm.Parse(s, e, r); t.lensFlareSystems.push(a); } } }); J1.prototype.getLensFlareSystemByName = function(A) { for (let e = 0; e < this.lensFlareSystems.length; e++) if (this.lensFlareSystems[e].name === A) return this.lensFlareSystems[e]; return null; }; J1.prototype.getLensFlareSystemById = function(A) { for (let e = 0; e < this.lensFlareSystems.length; e++) if (this.lensFlareSystems[e].id === A) return this.lensFlareSystems[e]; return null; }; J1.prototype.getLensFlareSystemByID = function(A) { return this.getLensFlareSystemById(A); }; J1.prototype.removeLensFlareSystem = function(A) { const e = this.lensFlareSystems.indexOf(A); return e !== -1 && this.lensFlareSystems.splice(e, 1), e; }; J1.prototype.addLensFlareSystem = function(A) { this.lensFlareSystems.push(A); }; class Mte { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_LENSFLARESYSTEM, this.scene = e, e.lensFlareSystems = []; } /** * Registers the component in a given scene */ register() { this.scene._afterCameraDrawStage.registerStep(Ot.STEP_AFTERCAMERADRAW_LENSFLARESYSTEM, this, this._draw); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { for (let e = 0; e < this.scene.lensFlareSystems.length; e++) this.scene.lensFlareSystems[e].rebuild(); } /** * Adds all the elements from the container to the scene * @param container the container holding the elements */ addFromContainer(e) { e.lensFlareSystems && e.lensFlareSystems.forEach((t) => { this.scene.addLensFlareSystem(t); }); } /** * Removes all the elements in the container from the scene * @param container contains the elements to remove * @param dispose if the removed element should be disposed (default: false) */ removeFromContainer(e, t) { e.lensFlareSystems && e.lensFlareSystems.forEach((r) => { this.scene.removeLensFlareSystem(r), t && r.dispose(); }); } /** * Serializes the component data to the specified json object * @param serializationObject The object to serialize to */ serialize(e) { e.lensFlareSystems = []; const t = this.scene.lensFlareSystems; for (const r of t) e.lensFlareSystems.push(r.serialize()); } /** * Disposes the component and the associated resources. */ dispose() { const e = this.scene.lensFlareSystems; for (; e.length; ) e[0].dispose(); } _draw(e) { if (this.scene.lensFlaresEnabled) { const t = this.scene.lensFlareSystems; ye.StartPerformanceCounter("Lens flares", t.length > 0); for (const r of t) e.layerMask & r.layerMask && r.render(); ye.EndPerformanceCounter("Lens flares", t.length > 0); } } } gm._SceneComponentInitialization = (A) => { let e = A._getComponent(Ot.NAME_LENSFLARESYSTEM); e || (e = new Mte(A), A._addComponent(e)); }; const R3e = "bayerDitherFunctions", V3e = `float bayerDither2(vec2 _P) {return mod(2.0*_P.y+_P.x+1.0,4.0);} float bayerDither4(vec2 _P) {vec2 P1=mod(_P,2.0); vec2 P2=floor(0.5*mod(_P,4.0)); return 4.0*bayerDither2(P1)+bayerDither2(P2);} float bayerDither8(vec2 _P) {vec2 P1=mod(_P,2.0); vec2 P2=floor(0.5 *mod(_P,4.0)); vec2 P4=floor(0.25*mod(_P,8.0)); return 4.0*(4.0*bayerDither2(P1)+bayerDither2(P2))+bayerDither2(P4);} `; Le.IncludesShadersStore[R3e] = V3e; const C3e = "shadowMapFragmentExtraDeclaration", O3e = `#if SM_FLOAT==0 #include #endif #if SM_SOFTTRANSPARENTSHADOW==1 #include uniform float softTransparentShadowSM; #endif varying float vDepthMetricSM; #if SM_USEDISTANCE==1 uniform vec3 lightDataSM;varying vec3 vPositionWSM; #endif uniform vec3 biasAndScaleSM;uniform vec2 depthValuesSM; #if defined(SM_DEPTHCLAMP) && SM_DEPTHCLAMP==1 varying float zSM; #endif `; Le.IncludesShadersStore[C3e] = O3e; const y3e = "shadowMapFragment", k3e = `float depthSM=vDepthMetricSM; #if defined(SM_DEPTHCLAMP) && SM_DEPTHCLAMP==1 #if SM_USEDISTANCE==1 depthSM=(length(vPositionWSM-lightDataSM)+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x; #else #ifdef USE_REVERSE_DEPTHBUFFER depthSM=(-zSM+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x; #else depthSM=(zSM+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x; #endif #endif #ifdef USE_REVERSE_DEPTHBUFFER gl_FragDepth=clamp(1.0-depthSM,0.0,1.0); #else gl_FragDepth=clamp(depthSM,0.0,1.0); #endif #elif SM_USEDISTANCE==1 depthSM=(length(vPositionWSM-lightDataSM)+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x; #endif #if SM_ESM==1 depthSM=clamp(exp(-min(87.,biasAndScaleSM.z*depthSM)),0.,1.); #endif #if SM_FLOAT==1 gl_FragColor=vec4(depthSM,1.0,1.0,1.0); #else gl_FragColor=pack(depthSM); #endif return;`; Le.IncludesShadersStore[y3e] = k3e; const E3e = "shadowMapPixelShader", F3e = `#include #ifdef ALPHATEXTURE varying vec2 vUV;uniform sampler2D diffuseSampler; #endif #include #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #include #ifdef ALPHATEXTURE float alphaFromAlphaTexture=texture2D(diffuseSampler,vUV).a; #ifdef ALPHATESTVALUE if (alphaFromAlphaTexture=softTransparentShadowSM*alphaFromAlphaTexture) discard; #else if ((bayerDither8(floor(mod(gl_FragCoord.xy,8.0))))/64.0>=softTransparentShadowSM) discard; #endif #endif #include }`; Le.ShadersStore[E3e] = F3e; const N3e = "sceneVertexDeclaration", Q3e = `uniform mat4 viewProjection; #ifdef MULTIVIEW uniform mat4 viewProjectionR; #endif uniform mat4 view;uniform mat4 projection;uniform vec4 vEyePosition; `; Le.IncludesShadersStore[N3e] = Q3e; const Y3e = "meshVertexDeclaration", M3e = `uniform mat4 world;uniform float visibility; `; Le.IncludesShadersStore[Y3e] = M3e; const L3e = "shadowMapVertexDeclaration", K3e = `#include #include `; Le.IncludesShadersStore[L3e] = K3e; const J3e = "shadowMapUboDeclaration", z3e = `layout(std140,column_major) uniform; #include #include `; Le.IncludesShadersStore[J3e] = z3e; const G3e = "shadowMapVertexExtraDeclaration", Z3e = `#if SM_NORMALBIAS==1 uniform vec3 lightDataSM; #endif uniform vec3 biasAndScaleSM;uniform vec2 depthValuesSM;varying float vDepthMetricSM; #if SM_USEDISTANCE==1 varying vec3 vPositionWSM; #endif #if defined(SM_DEPTHCLAMP) && SM_DEPTHCLAMP==1 varying float zSM; #endif `; Le.IncludesShadersStore[G3e] = Z3e; const _3e = "shadowMapVertexNormalBias", $3e = `#if SM_NORMALBIAS==1 #if SM_DIRECTIONINLIGHTDATA==1 vec3 worldLightDirSM=normalize(-lightDataSM.xyz); #else vec3 directionToLightSM=lightDataSM.xyz-worldPos.xyz;vec3 worldLightDirSM=normalize(directionToLightSM); #endif float ndlSM=dot(vNormalW,worldLightDirSM);float sinNLSM=sqrt(1.0-ndlSM*ndlSM);float normalBiasSM=biasAndScaleSM.y*sinNLSM;worldPos.xyz-=vNormalW*normalBiasSM; #endif `; Le.IncludesShadersStore[_3e] = $3e; const epe = "shadowMapVertexMetric", tpe = `#if SM_USEDISTANCE==1 vPositionWSM=worldPos.xyz; #endif #if SM_DEPTHTEXTURE==1 #ifdef IS_NDC_HALF_ZRANGE #define BIASFACTOR 0.5 #else #define BIASFACTOR 1.0 #endif #ifdef USE_REVERSE_DEPTHBUFFER gl_Position.z-=biasAndScaleSM.x*gl_Position.w*BIASFACTOR; #else gl_Position.z+=biasAndScaleSM.x*gl_Position.w*BIASFACTOR; #endif #endif #if defined(SM_DEPTHCLAMP) && SM_DEPTHCLAMP==1 zSM=gl_Position.z;gl_Position.z=0.0; #elif SM_USEDISTANCE==0 #ifdef USE_REVERSE_DEPTHBUFFER vDepthMetricSM=(-gl_Position.z+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x; #else vDepthMetricSM=(gl_Position.z+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x; #endif #endif `; Le.IncludesShadersStore[epe] = tpe; const rpe = "shadowMapVertexShader", npe = `attribute vec3 position; #ifdef NORMAL attribute vec3 normal; #endif #include #include #include #include[0..maxSimultaneousMorphTargets] #ifdef INSTANCES attribute vec4 world0;attribute vec4 world1;attribute vec4 world2;attribute vec4 world3; #endif #include #include<__decl__shadowMapVertex> #ifdef ALPHATEXTURE varying vec2 vUV;uniform mat4 diffuseMatrix; #ifdef UV1 attribute vec2 uv; #endif #ifdef UV2 attribute vec2 uv2; #endif #endif #include #include #define CUSTOM_VERTEX_DEFINITIONS void main(void) {vec3 positionUpdated=position; #ifdef UV1 vec2 uvUpdated=uv; #endif #ifdef NORMAL vec3 normalUpdated=normal; #endif #include #include[0..maxSimultaneousMorphTargets] #include #include #include vec4 worldPos=finalWorld*vec4(positionUpdated,1.0); #ifdef NORMAL mat3 normWorldSM=mat3(finalWorld); #if defined(INSTANCES) && defined(THIN_INSTANCES) vec3 vNormalW=normalUpdated/vec3(dot(normWorldSM[0],normWorldSM[0]),dot(normWorldSM[1],normWorldSM[1]),dot(normWorldSM[2],normWorldSM[2]));vNormalW=normalize(normWorldSM*vNormalW); #else #ifdef NONUNIFORMSCALING normWorldSM=transposeMat3(inverseMat3(normWorldSM)); #endif vec3 vNormalW=normalize(normWorldSM*normalUpdated); #endif #endif #include gl_Position=viewProjection*worldPos; #include #ifdef ALPHATEXTURE #ifdef UV1 vUV=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0)); #endif #ifdef UV2 vUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0)); #endif #endif #include }`; Le.ShadersStore[rpe] = npe; const ipe = "depthBoxBlurPixelShader", spe = `varying vec2 vUV;uniform sampler2D textureSampler;uniform vec2 screenSize; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {vec4 colorDepth=vec4(0.0);for (int x=-OFFSET; x<=OFFSET; x++) for (int y=-OFFSET; y<=OFFSET; y++) colorDepth+=texture2D(textureSampler,vUV+vec2(x,y)/screenSize);gl_FragColor=(colorDepth/float((OFFSET*2+1)*(OFFSET*2+1)));}`; Le.ShadersStore[ipe] = spe; const ape = "shadowMapFragmentSoftTransparentShadow", ope = `#if SM_SOFTTRANSPARENTSHADOW==1 if ((bayerDither8(floor(mod(gl_FragCoord.xy,8.0))))/64.0>=softTransparentShadowSM*alpha) discard; #endif `; Le.IncludesShadersStore[ape] = ope; class ln { /** * Gets the bias: offset applied on the depth preventing acnea (in light direction). */ get bias() { return this._bias; } /** * Sets the bias: offset applied on the depth preventing acnea (in light direction). */ set bias(e) { this._bias = e; } /** * Gets the normalBias: offset applied on the depth preventing acnea (along side the normal direction and proportional to the light/normal angle). */ get normalBias() { return this._normalBias; } /** * Sets the normalBias: offset applied on the depth preventing acnea (along side the normal direction and proportional to the light/normal angle). */ set normalBias(e) { this._normalBias = e; } /** * Gets the blur box offset: offset applied during the blur pass. * Only useful if useKernelBlur = false */ get blurBoxOffset() { return this._blurBoxOffset; } /** * Sets the blur box offset: offset applied during the blur pass. * Only useful if useKernelBlur = false */ set blurBoxOffset(e) { this._blurBoxOffset !== e && (this._blurBoxOffset = e, this._disposeBlurPostProcesses()); } /** * Gets the blur scale: scale of the blurred texture compared to the main shadow map. * 2 means half of the size. */ get blurScale() { return this._blurScale; } /** * Sets the blur scale: scale of the blurred texture compared to the main shadow map. * 2 means half of the size. */ set blurScale(e) { this._blurScale !== e && (this._blurScale = e, this._disposeBlurPostProcesses()); } /** * Gets the blur kernel: kernel size of the blur pass. * Only useful if useKernelBlur = true */ get blurKernel() { return this._blurKernel; } /** * Sets the blur kernel: kernel size of the blur pass. * Only useful if useKernelBlur = true */ set blurKernel(e) { this._blurKernel !== e && (this._blurKernel = e, this._disposeBlurPostProcesses()); } /** * Gets whether the blur pass is a kernel blur (if true) or box blur. * Only useful in filtered mode (useBlurExponentialShadowMap...) */ get useKernelBlur() { return this._useKernelBlur; } /** * Sets whether the blur pass is a kernel blur (if true) or box blur. * Only useful in filtered mode (useBlurExponentialShadowMap...) */ set useKernelBlur(e) { this._useKernelBlur !== e && (this._useKernelBlur = e, this._disposeBlurPostProcesses()); } /** * Gets the depth scale used in ESM mode. */ get depthScale() { return this._depthScale !== void 0 ? this._depthScale : this._light.getDepthScale(); } /** * Sets the depth scale used in ESM mode. * This can override the scale stored on the light. */ set depthScale(e) { this._depthScale = e; } _validateFilter(e) { return e; } /** * Gets the current mode of the shadow generator (normal, PCF, ESM...). * The returned value is a number equal to one of the available mode defined in ShadowMap.FILTER_x like _FILTER_NONE */ get filter() { return this._filter; } /** * Sets the current mode of the shadow generator (normal, PCF, ESM...). * The returned value is a number equal to one of the available mode defined in ShadowMap.FILTER_x like _FILTER_NONE */ set filter(e) { if (e = this._validateFilter(e), this._light.needCube()) { if (e === ln.FILTER_BLUREXPONENTIALSHADOWMAP) { this.useExponentialShadowMap = !0; return; } else if (e === ln.FILTER_BLURCLOSEEXPONENTIALSHADOWMAP) { this.useCloseExponentialShadowMap = !0; return; } else if (e === ln.FILTER_PCF || e === ln.FILTER_PCSS) { this.usePoissonSampling = !0; return; } } if ((e === ln.FILTER_PCF || e === ln.FILTER_PCSS) && !this._scene.getEngine()._features.supportShadowSamplers) { this.usePoissonSampling = !0; return; } this._filter !== e && (this._filter = e, this._disposeBlurPostProcesses(), this._applyFilterValues(), this._light._markMeshesAsLightDirty()); } /** * Gets if the current filter is set to Poisson Sampling. */ get usePoissonSampling() { return this.filter === ln.FILTER_POISSONSAMPLING; } /** * Sets the current filter to Poisson Sampling. */ set usePoissonSampling(e) { const t = this._validateFilter(ln.FILTER_POISSONSAMPLING); !e && this.filter !== ln.FILTER_POISSONSAMPLING || (this.filter = e ? t : ln.FILTER_NONE); } /** * Gets if the current filter is set to ESM. */ get useExponentialShadowMap() { return this.filter === ln.FILTER_EXPONENTIALSHADOWMAP; } /** * Sets the current filter is to ESM. */ set useExponentialShadowMap(e) { const t = this._validateFilter(ln.FILTER_EXPONENTIALSHADOWMAP); !e && this.filter !== ln.FILTER_EXPONENTIALSHADOWMAP || (this.filter = e ? t : ln.FILTER_NONE); } /** * Gets if the current filter is set to filtered ESM. */ get useBlurExponentialShadowMap() { return this.filter === ln.FILTER_BLUREXPONENTIALSHADOWMAP; } /** * Gets if the current filter is set to filtered ESM. */ set useBlurExponentialShadowMap(e) { const t = this._validateFilter(ln.FILTER_BLUREXPONENTIALSHADOWMAP); !e && this.filter !== ln.FILTER_BLUREXPONENTIALSHADOWMAP || (this.filter = e ? t : ln.FILTER_NONE); } /** * Gets if the current filter is set to "close ESM" (using the inverse of the * exponential to prevent steep falloff artifacts). */ get useCloseExponentialShadowMap() { return this.filter === ln.FILTER_CLOSEEXPONENTIALSHADOWMAP; } /** * Sets the current filter to "close ESM" (using the inverse of the * exponential to prevent steep falloff artifacts). */ set useCloseExponentialShadowMap(e) { const t = this._validateFilter(ln.FILTER_CLOSEEXPONENTIALSHADOWMAP); !e && this.filter !== ln.FILTER_CLOSEEXPONENTIALSHADOWMAP || (this.filter = e ? t : ln.FILTER_NONE); } /** * Gets if the current filter is set to filtered "close ESM" (using the inverse of the * exponential to prevent steep falloff artifacts). */ get useBlurCloseExponentialShadowMap() { return this.filter === ln.FILTER_BLURCLOSEEXPONENTIALSHADOWMAP; } /** * Sets the current filter to filtered "close ESM" (using the inverse of the * exponential to prevent steep falloff artifacts). */ set useBlurCloseExponentialShadowMap(e) { const t = this._validateFilter(ln.FILTER_BLURCLOSEEXPONENTIALSHADOWMAP); !e && this.filter !== ln.FILTER_BLURCLOSEEXPONENTIALSHADOWMAP || (this.filter = e ? t : ln.FILTER_NONE); } /** * Gets if the current filter is set to "PCF" (percentage closer filtering). */ get usePercentageCloserFiltering() { return this.filter === ln.FILTER_PCF; } /** * Sets the current filter to "PCF" (percentage closer filtering). */ set usePercentageCloserFiltering(e) { const t = this._validateFilter(ln.FILTER_PCF); !e && this.filter !== ln.FILTER_PCF || (this.filter = e ? t : ln.FILTER_NONE); } /** * Gets the PCF or PCSS Quality. * Only valid if usePercentageCloserFiltering or usePercentageCloserFiltering is true. */ get filteringQuality() { return this._filteringQuality; } /** * Sets the PCF or PCSS Quality. * Only valid if usePercentageCloserFiltering or usePercentageCloserFiltering is true. */ set filteringQuality(e) { this._filteringQuality !== e && (this._filteringQuality = e, this._disposeBlurPostProcesses(), this._applyFilterValues(), this._light._markMeshesAsLightDirty()); } /** * Gets if the current filter is set to "PCSS" (contact hardening). */ get useContactHardeningShadow() { return this.filter === ln.FILTER_PCSS; } /** * Sets the current filter to "PCSS" (contact hardening). */ set useContactHardeningShadow(e) { const t = this._validateFilter(ln.FILTER_PCSS); !e && this.filter !== ln.FILTER_PCSS || (this.filter = e ? t : ln.FILTER_NONE); } /** * Gets the Light Size (in shadow map uv unit) used in PCSS to determine the blocker search area and the penumbra size. * Using a ratio helps keeping shape stability independently of the map size. * * It does not account for the light projection as it was having too much * instability during the light setup or during light position changes. * * Only valid if useContactHardeningShadow is true. */ get contactHardeningLightSizeUVRatio() { return this._contactHardeningLightSizeUVRatio; } /** * Sets the Light Size (in shadow map uv unit) used in PCSS to determine the blocker search area and the penumbra size. * Using a ratio helps keeping shape stability independently of the map size. * * It does not account for the light projection as it was having too much * instability during the light setup or during light position changes. * * Only valid if useContactHardeningShadow is true. */ set contactHardeningLightSizeUVRatio(e) { this._contactHardeningLightSizeUVRatio = e; } /** Gets or sets the actual darkness of a shadow */ get darkness() { return this._darkness; } set darkness(e) { this.setDarkness(e); } /** * Returns the darkness value (float). This can only decrease the actual darkness of a shadow. * 0 means strongest and 1 would means no shadow. * @returns the darkness. */ getDarkness() { return this._darkness; } /** * Sets the darkness value (float). This can only decrease the actual darkness of a shadow. * @param darkness The darkness value 0 means strongest and 1 would means no shadow. * @returns the shadow generator allowing fluent coding. */ setDarkness(e) { return e >= 1 ? this._darkness = 1 : e <= 0 ? this._darkness = 0 : this._darkness = e, this; } /** Gets or sets the ability to have transparent shadow */ get transparencyShadow() { return this._transparencyShadow; } set transparencyShadow(e) { this.setTransparencyShadow(e); } /** * Sets the ability to have transparent shadow (boolean). * @param transparent True if transparent else False * @returns the shadow generator allowing fluent coding */ setTransparencyShadow(e) { return this._transparencyShadow = e, this; } /** * Gets the main RTT containing the shadow map (usually storing depth from the light point of view). * @returns The render target texture if present otherwise, null */ getShadowMap() { return this._shadowMap; } /** * Gets the RTT used during rendering (can be a blurred version of the shadow map or the shadow map itself). * @returns The render target texture if the shadow map is present otherwise, null */ getShadowMapForRendering() { return this._shadowMap2 ? this._shadowMap2 : this._shadowMap; } /** * Gets the class name of that object * @returns "ShadowGenerator" */ getClassName() { return ln.CLASSNAME; } /** * Helper function to add a mesh and its descendants to the list of shadow casters. * @param mesh Mesh to add * @param includeDescendants boolean indicating if the descendants should be added. Default to true * @returns the Shadow Generator itself */ addShadowCaster(e, t = !0) { if (!this._shadowMap) return this; if (this._shadowMap.renderList || (this._shadowMap.renderList = []), this._shadowMap.renderList.indexOf(e) === -1 && this._shadowMap.renderList.push(e), t) for (const r of e.getChildMeshes()) this._shadowMap.renderList.indexOf(r) === -1 && this._shadowMap.renderList.push(r); return this; } /** * Helper function to remove a mesh and its descendants from the list of shadow casters * @param mesh Mesh to remove * @param includeDescendants boolean indicating if the descendants should be removed. Default to true * @returns the Shadow Generator itself */ removeShadowCaster(e, t = !0) { if (!this._shadowMap || !this._shadowMap.renderList) return this; const r = this._shadowMap.renderList.indexOf(e); if (r !== -1 && this._shadowMap.renderList.splice(r, 1), t) for (const n of e.getChildren()) this.removeShadowCaster(n); return this; } /** * Returns the associated light object. * @returns the light generating the shadow */ getLight() { return this._light; } _getCamera() { var e; return (e = this._camera) !== null && e !== void 0 ? e : this._scene.activeCamera; } /** * Gets or sets the size of the texture what stores the shadows */ get mapSize() { return this._mapSize; } set mapSize(e) { this._mapSize = e, this._light._markMeshesAsLightDirty(), this.recreateShadowMap(); } /** * Creates a ShadowGenerator object. * A ShadowGenerator is the required tool to use the shadows. * Each light casting shadows needs to use its own ShadowGenerator. * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/shadows * @param mapSize The size of the texture what stores the shadows. Example : 1024. * @param light The light object generating the shadows. * @param usefullFloatFirst By default the generator will try to use half float textures but if you need precision (for self shadowing for instance), you can use this option to enforce full float texture. * @param camera Camera associated with this shadow generator (default: null). If null, takes the scene active camera at the time we need to access it * @param useRedTextureType Forces the generator to use a Red instead of a RGBA type for the shadow map texture format (default: false) */ constructor(e, t, r, n, i) { this.onBeforeShadowMapRenderObservable = new Oe(), this.onAfterShadowMapRenderObservable = new Oe(), this.onBeforeShadowMapRenderMeshObservable = new Oe(), this.onAfterShadowMapRenderMeshObservable = new Oe(), this._bias = 5e-5, this._normalBias = 0, this._blurBoxOffset = 1, this._blurScale = 2, this._blurKernel = 1, this._useKernelBlur = !1, this._filter = ln.FILTER_NONE, this._filteringQuality = ln.QUALITY_HIGH, this._contactHardeningLightSizeUVRatio = 0.1, this._darkness = 0, this._transparencyShadow = !1, this.enableSoftTransparentShadow = !1, this.useOpacityTextureForTransparentShadow = !1, this.frustumEdgeFalloff = 0, this.forceBackFacesOnly = !1, this._lightDirection = S.Zero(), this._viewMatrix = he.Zero(), this._projectionMatrix = he.Zero(), this._transformMatrix = he.Zero(), this._cachedPosition = new S(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE), this._cachedDirection = new S(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE), this._currentFaceIndex = 0, this._currentFaceIndexCache = 0, this._defaultTextureMatrix = he.Identity(), this._mapSize = e, this._light = t, this._scene = t.getScene(), this._camera = n ?? null, this._useRedTextureType = !!i; let s = t._shadowGenerators; s || (s = t._shadowGenerators = /* @__PURE__ */ new Map()), s.set(this._camera, this), this.id = t.id, this._useUBO = this._scene.getEngine().supportsUniformBuffers, this._useUBO && (this._sceneUBOs = [], this._sceneUBOs.push(this._scene.createSceneUniformBuffer(`Scene for Shadow Generator (light "${this._light.name}")`))), ln._SceneComponentInitialization(this._scene); const a = this._scene.getEngine().getCaps(); r ? a.textureFloatRender && a.textureFloatLinearFiltering ? this._textureType = 1 : a.textureHalfFloatRender && a.textureHalfFloatLinearFiltering ? this._textureType = 2 : this._textureType = 0 : a.textureHalfFloatRender && a.textureHalfFloatLinearFiltering ? this._textureType = 2 : a.textureFloatRender && a.textureFloatLinearFiltering ? this._textureType = 1 : this._textureType = 0, this._initializeGenerator(), this._applyFilterValues(); } _initializeGenerator() { this._light._markMeshesAsLightDirty(), this._initializeShadowMap(); } _createTargetRenderTexture() { const e = this._scene.getEngine(); e._features.supportDepthStencilTexture ? (this._shadowMap = new Ta(this._light.name + "_shadowMap", this._mapSize, this._scene, !1, !0, this._textureType, this._light.needCube(), void 0, !1, !1, void 0, this._useRedTextureType ? 6 : 5), this._shadowMap.createDepthStencilTexture(e.useReverseDepthBuffer ? 516 : 513, !0)) : this._shadowMap = new Ta(this._light.name + "_shadowMap", this._mapSize, this._scene, !1, !0, this._textureType, this._light.needCube()), this._shadowMap.noPrePassRenderer = !0; } _initializeShadowMap() { if (this._createTargetRenderTexture(), this._shadowMap === null) return; this._shadowMap.wrapU = We.CLAMP_ADDRESSMODE, this._shadowMap.wrapV = We.CLAMP_ADDRESSMODE, this._shadowMap.anisotropicFilteringLevel = 1, this._shadowMap.updateSamplingMode(We.BILINEAR_SAMPLINGMODE), this._shadowMap.renderParticles = !1, this._shadowMap.ignoreCameraViewport = !0, this._storedUniqueId && (this._shadowMap.uniqueId = this._storedUniqueId), this._shadowMap.customRenderFunction = (n, i, s, a) => this._renderForShadowMap(n, i, s, a), this._shadowMap.customIsReadyFunction = () => !0; const e = this._scene.getEngine(); this._shadowMap.onBeforeBindObservable.add(() => { var n; this._currentSceneUBO = this._scene.getSceneUniformBuffer(), (n = e._debugPushGroup) === null || n === void 0 || n.call(e, `shadow map generation for pass id ${e.currentRenderPassId}`, 1); }), this._shadowMap.onBeforeRenderObservable.add((n) => { this._sceneUBOs && this._scene.setSceneUniformBuffer(this._sceneUBOs[0]), this._currentFaceIndex = n, this._filter === ln.FILTER_PCF && e.setColorWrite(!1), this.getTransformMatrix(), this._scene.setTransformMatrix(this._viewMatrix, this._projectionMatrix), this._useUBO && (this._scene.getSceneUniformBuffer().unbindEffect(), this._scene.finalizeSceneUbo()); }), this._shadowMap.onAfterUnbindObservable.add(() => { var n, i; if (this._sceneUBOs && this._scene.setSceneUniformBuffer(this._currentSceneUBO), this._scene.updateTransformMatrix(), this._filter === ln.FILTER_PCF && e.setColorWrite(!0), !this.useBlurExponentialShadowMap && !this.useBlurCloseExponentialShadowMap) { (n = e._debugPopGroup) === null || n === void 0 || n.call(e, 1); return; } const s = this.getShadowMapForRendering(); s && (this._scene.postProcessManager.directRender(this._blurPostProcesses, s.renderTarget, !0), e.unBindFramebuffer(s.renderTarget, !0), (i = e._debugPopGroup) === null || i === void 0 || i.call(e, 1)); }); const t = new xt(0, 0, 0, 0), r = new xt(1, 1, 1, 1); this._shadowMap.onClearObservable.add((n) => { this._filter === ln.FILTER_PCF ? n.clear(r, !1, !0, !1) : this.useExponentialShadowMap || this.useBlurExponentialShadowMap ? n.clear(t, !0, !0, !1) : n.clear(r, !0, !0, !1); }), this._shadowMap.onResizeObservable.add((n) => { this._storedUniqueId = this._shadowMap.uniqueId, this._mapSize = n.getRenderSize(), this._light._markMeshesAsLightDirty(), this.recreateShadowMap(); }); for (let n = $6.MIN_RENDERINGGROUPS; n < $6.MAX_RENDERINGGROUPS; n++) this._shadowMap.setRenderingAutoClearDepthStencil(n, !1); } _initializeBlurRTTAndPostProcesses() { const e = this._scene.getEngine(), t = this._mapSize / this.blurScale; (!this.useKernelBlur || this.blurScale !== 1) && (this._shadowMap2 = new Ta(this._light.name + "_shadowMap2", t, this._scene, !1, !0, this._textureType, void 0, void 0, !1), this._shadowMap2.wrapU = We.CLAMP_ADDRESSMODE, this._shadowMap2.wrapV = We.CLAMP_ADDRESSMODE, this._shadowMap2.updateSamplingMode(We.BILINEAR_SAMPLINGMODE)), this.useKernelBlur ? (this._kernelBlurXPostprocess = new c9(this._light.name + "KernelBlurX", new at(1, 0), this.blurKernel, 1, null, We.BILINEAR_SAMPLINGMODE, e, !1, this._textureType), this._kernelBlurXPostprocess.width = t, this._kernelBlurXPostprocess.height = t, this._kernelBlurXPostprocess.externalTextureSamplerBinding = !0, this._kernelBlurXPostprocess.onApplyObservable.add((r) => { r.setTexture("textureSampler", this._shadowMap); }), this._kernelBlurYPostprocess = new c9(this._light.name + "KernelBlurY", new at(0, 1), this.blurKernel, 1, null, We.BILINEAR_SAMPLINGMODE, e, !1, this._textureType), this._kernelBlurXPostprocess.autoClear = !1, this._kernelBlurYPostprocess.autoClear = !1, this._textureType === 0 && (this._kernelBlurXPostprocess.packedFloat = !0, this._kernelBlurYPostprocess.packedFloat = !0), this._blurPostProcesses = [this._kernelBlurXPostprocess, this._kernelBlurYPostprocess]) : (this._boxBlurPostprocess = new kr(this._light.name + "DepthBoxBlur", "depthBoxBlur", ["screenSize", "boxOffset"], [], 1, null, We.BILINEAR_SAMPLINGMODE, e, !1, "#define OFFSET " + this._blurBoxOffset, this._textureType), this._boxBlurPostprocess.externalTextureSamplerBinding = !0, this._boxBlurPostprocess.onApplyObservable.add((r) => { r.setFloat2("screenSize", t, t), r.setTexture("textureSampler", this._shadowMap); }), this._boxBlurPostprocess.autoClear = !1, this._blurPostProcesses = [this._boxBlurPostprocess]); } _renderForShadowMap(e, t, r, n) { let i; if (n.length) for (i = 0; i < n.length; i++) this._renderSubMeshForShadowMap(n.data[i]); for (i = 0; i < e.length; i++) this._renderSubMeshForShadowMap(e.data[i]); for (i = 0; i < t.length; i++) this._renderSubMeshForShadowMap(t.data[i]); if (this._transparencyShadow) for (i = 0; i < r.length; i++) this._renderSubMeshForShadowMap(r.data[i], !0); else for (i = 0; i < r.length; i++) r.data[i].getEffectiveMesh()._internalAbstractMeshDataInfo._isActiveIntermediate = !1; } // eslint-disable-next-line @typescript-eslint/no-unused-vars _bindCustomEffectForRenderSubMeshForShadowMap(e, t, r) { t.setMatrix("viewProjection", this.getTransformMatrix()); } _renderSubMeshForShadowMap(e, t = !1) { var r, n; const i = e.getRenderingMesh(), s = e.getEffectiveMesh(), a = this._scene, f = a.getEngine(), o = e.getMaterial(); if (s._internalAbstractMeshDataInfo._isActiveIntermediate = !1, !o || e.verticesCount === 0 || e._renderId === a.getRenderId()) return; const d = s._getWorldMatrixDeterminant() < 0; let v = (r = i.overrideMaterialSideOrientation) !== null && r !== void 0 ? r : o.sideOrientation; d && (v = v === 0 ? 1 : 0); const u = v === 0; f.setState(o.backFaceCulling, void 0, void 0, u, o.cullBackFaces); const l = i._getInstancesRenderList(e._id, !!e.getReplacementMesh()); if (l.mustReturn) return; const P = f.getCaps().instancedArrays && (l.visibleInstances[e._id] !== null && l.visibleInstances[e._id] !== void 0 || i.hasThinInstances); if (!(this.customAllowRendering && !this.customAllowRendering(e))) if (this.isReady(e, P, t)) { e._renderId = a.getRenderId(); const p = o.shadowDepthWrapper, c = (n = p == null ? void 0 : p.getEffect(e, this, f.currentRenderPassId)) !== null && n !== void 0 ? n : e._getDrawWrapper(), H = zo.GetEffect(c); f.enableEffect(c), P || i._bind(e, H, o.fillMode), this.getTransformMatrix(), H.setFloat3("biasAndScaleSM", this.bias, this.normalBias, this.depthScale), this.getLight().getTypeID() === ci.LIGHTTYPEID_DIRECTIONALLIGHT ? H.setVector3("lightDataSM", this._cachedDirection) : H.setVector3("lightDataSM", this._cachedPosition); const T = this._getCamera(); if (T && H.setFloat2("depthValuesSM", this.getLight().getDepthMinZ(T), this.getLight().getDepthMinZ(T) + this.getLight().getDepthMaxZ(T)), t && this.enableSoftTransparentShadow && H.setFloat("softTransparentShadowSM", s.visibility * o.alpha), p) e._setMainDrawWrapperOverride(c), p.standalone ? p.baseMaterial.bindForSubMesh(s.getWorldMatrix(), i, e) : o.bindForSubMesh(s.getWorldMatrix(), i, e), e._setMainDrawWrapperOverride(null); else { if (this._opacityTexture && (H.setTexture("diffuseSampler", this._opacityTexture), H.setMatrix("diffuseMatrix", this._opacityTexture.getTextureMatrix() || this._defaultTextureMatrix)), i.useBones && i.computeBonesUsingShaders && i.skeleton) { const b = i.skeleton; if (b.isUsingTextureForMatrices) { const j = b.getTransformMatrixTexture(i); if (!j) return; H.setTexture("boneSampler", j), H.setFloat("boneTextureWidth", 4 * (b.bones.length + 1)); } else H.setMatrices("mBones", b.getTransformMatrices(i)); } Ye.BindMorphTargetParameters(i, H), i.morphTargetManager && i.morphTargetManager.isUsingTextureForTargets && i.morphTargetManager._bind(H), Df(H, o, a); } !this._useUBO && !p && this._bindCustomEffectForRenderSubMeshForShadowMap(e, H, s), Ye.BindSceneUniformBuffer(H, this._scene.getSceneUniformBuffer()), this._scene.getSceneUniformBuffer().bindUniformBuffer(); const q = s.getWorldMatrix(); P && (s.getMeshUniformBuffer().bindToEffect(H, "Mesh"), s.transferToEffect(q)), this.forceBackFacesOnly && f.setState(!0, 0, !1, !0, o.cullBackFaces), this.onBeforeShadowMapRenderMeshObservable.notifyObservers(i), this.onBeforeShadowMapRenderObservable.notifyObservers(H), i._processRendering(s, e, H, o.fillMode, l, P, (b, j) => { s !== i && !b ? (i.getMeshUniformBuffer().bindToEffect(H, "Mesh"), i.transferToEffect(j)) : (s.getMeshUniformBuffer().bindToEffect(H, "Mesh"), s.transferToEffect(b ? j : q)); }), this.forceBackFacesOnly && f.setState(!0, 0, !1, !1, o.cullBackFaces), this.onAfterShadowMapRenderObservable.notifyObservers(H), this.onAfterShadowMapRenderMeshObservable.notifyObservers(i); } else this._shadowMap && this._shadowMap.resetRefreshCounter(); } _applyFilterValues() { this._shadowMap && (this.filter === ln.FILTER_NONE || this.filter === ln.FILTER_PCSS ? this._shadowMap.updateSamplingMode(We.NEAREST_SAMPLINGMODE) : this._shadowMap.updateSamplingMode(We.BILINEAR_SAMPLINGMODE)); } /** * Forces all the attached effect to compile to enable rendering only once ready vs. lazily compiling effects. * @param onCompiled Callback triggered at the and of the effects compilation * @param options Sets of optional options forcing the compilation with different modes */ forceCompilation(e, t) { const r = Object.assign({ useInstances: !1 }, t), n = this.getShadowMap(); if (!n) { e && e(this); return; } const i = n.renderList; if (!i) { e && e(this); return; } const s = []; for (const o of i) s.push(...o.subMeshes); if (s.length === 0) { e && e(this); return; } let a = 0; const f = () => { var o, d; if (!(!this._scene || !this._scene.getEngine())) { for (; this.isReady(s[a], r.useInstances, (d = (o = s[a].getMaterial()) === null || o === void 0 ? void 0 : o.needAlphaBlendingForMesh(s[a].getMesh())) !== null && d !== void 0 ? d : !1); ) if (a++, a >= s.length) { e && e(this); return; } setTimeout(f, 16); } }; f(); } /** * Forces all the attached effect to compile to enable rendering only once ready vs. lazily compiling effects. * @param options Sets of optional options forcing the compilation with different modes * @returns A promise that resolves when the compilation completes */ forceCompilationAsync(e) { return new Promise((t) => { this.forceCompilation(() => { t(); }, e); }); } // eslint-disable-next-line @typescript-eslint/no-unused-vars _isReadyCustomDefines(e, t, r) { } _prepareShadowDefines(e, t, r, n) { r.push("#define SM_LIGHTTYPE_" + this._light.getClassName().toUpperCase()), r.push("#define SM_FLOAT " + (this._textureType !== 0 ? "1" : "0")), r.push("#define SM_ESM " + (this.useExponentialShadowMap || this.useBlurExponentialShadowMap ? "1" : "0")), r.push("#define SM_DEPTHTEXTURE " + (this.usePercentageCloserFiltering || this.useContactHardeningShadow ? "1" : "0")); const i = e.getMesh(); return r.push("#define SM_NORMALBIAS " + (this.normalBias && i.isVerticesDataPresent(J.NormalKind) ? "1" : "0")), r.push("#define SM_DIRECTIONINLIGHTDATA " + (this.getLight().getTypeID() === ci.LIGHTTYPEID_DIRECTIONALLIGHT ? "1" : "0")), r.push("#define SM_USEDISTANCE " + (this._light.needCube() ? "1" : "0")), r.push("#define SM_SOFTTRANSPARENTSHADOW " + (this.enableSoftTransparentShadow && n ? "1" : "0")), this._isReadyCustomDefines(r, e, t), r; } /** * Determine whether the shadow generator is ready or not (mainly all effects and related post processes needs to be ready). * @param subMesh The submesh we want to render in the shadow map * @param useInstances Defines whether will draw in the map using instances * @param isTransparent Indicates that isReady is called for a transparent subMesh * @returns true if ready otherwise, false */ isReady(e, t, r) { var n; const i = e.getMaterial(), s = i == null ? void 0 : i.shadowDepthWrapper; if (this._opacityTexture = null, !i) return !1; const a = []; if (this._prepareShadowDefines(e, t, a, r), s) { if (!s.isReadyForSubMesh(e, a, this, t, this._scene.getEngine().currentRenderPassId)) return !1; } else { const f = e._getDrawWrapper(void 0, !0); let o = f.effect, d = f.defines; const v = [J.PositionKind], u = e.getMesh(); this.normalBias && u.isVerticesDataPresent(J.NormalKind) && (v.push(J.NormalKind), a.push("#define NORMAL"), u.nonUniformScaling && a.push("#define NONUNIFORMSCALING")); const l = i.needAlphaTesting(); if ((l || i.needAlphaBlending()) && (this.useOpacityTextureForTransparentShadow ? this._opacityTexture = i.opacityTexture : this._opacityTexture = i.getAlphaTestTexture(), this._opacityTexture)) { if (!this._opacityTexture.isReady()) return !1; const T = (n = i.alphaCutOff) !== null && n !== void 0 ? n : ln.DEFAULT_ALPHA_CUTOFF; a.push("#define ALPHATEXTURE"), l && a.push(`#define ALPHATESTVALUE ${T}${T % 1 === 0 ? "." : ""}`), u.isVerticesDataPresent(J.UVKind) && (v.push(J.UVKind), a.push("#define UV1")), u.isVerticesDataPresent(J.UV2Kind) && this._opacityTexture.coordinatesIndex === 1 && (v.push(J.UV2Kind), a.push("#define UV2")); } const P = new c1(); if (u.useBones && u.computeBonesUsingShaders && u.skeleton) { v.push(J.MatricesIndicesKind), v.push(J.MatricesWeightsKind), u.numBoneInfluencers > 4 && (v.push(J.MatricesIndicesExtraKind), v.push(J.MatricesWeightsExtraKind)); const T = u.skeleton; a.push("#define NUM_BONE_INFLUENCERS " + u.numBoneInfluencers), u.numBoneInfluencers > 0 && P.addCPUSkinningFallback(0, u), T.isUsingTextureForMatrices ? a.push("#define BONETEXTURE") : a.push("#define BonesPerMesh " + (T.bones.length + 1)); } else a.push("#define NUM_BONE_INFLUENCERS 0"); const p = u.morphTargetManager; let c = 0; if (p && p.numInfluencers > 0 && (a.push("#define MORPHTARGETS"), c = p.numInfluencers, a.push("#define NUM_MORPH_INFLUENCERS " + c), p.isUsingTextureForTargets && a.push("#define MORPHTARGETS_TEXTURE"), Ye.PrepareAttributesForMorphTargetsInfluencers(v, u, c)), xq(i, this._scene, a), t && (a.push("#define INSTANCES"), Ye.PushAttributesForInstances(v), e.getRenderingMesh().hasThinInstances && a.push("#define THIN_INSTANCES")), this.customShaderOptions && this.customShaderOptions.defines) for (const T of this.customShaderOptions.defines) a.indexOf(T) === -1 && a.push(T); const H = a.join(` `); if (d !== H) { d = H; let T = "shadowMap"; const q = [ "world", "mBones", "viewProjection", "diffuseMatrix", "lightDataSM", "depthValuesSM", "biasAndScaleSM", "morphTargetInfluences", "boneTextureWidth", "softTransparentShadowSM", "morphTargetTextureInfo", "morphTargetTextureIndices" ], b = ["diffuseSampler", "boneSampler", "morphTargets"], j = ["Scene", "Mesh"]; if (Mf(q), this.customShaderOptions) { if (T = this.customShaderOptions.shaderName, this.customShaderOptions.attributes) for (const m of this.customShaderOptions.attributes) v.indexOf(m) === -1 && v.push(m); if (this.customShaderOptions.uniforms) for (const m of this.customShaderOptions.uniforms) q.indexOf(m) === -1 && q.push(m); if (this.customShaderOptions.samplers) for (const m of this.customShaderOptions.samplers) b.indexOf(m) === -1 && b.push(m); } const w = this._scene.getEngine(); o = w.createEffect(T, { attributes: v, uniformsNames: q, uniformBuffersNames: j, samplers: b, defines: H, fallbacks: P, onCompiled: null, onError: null, indexParameters: { maxSimultaneousMorphTargets: c } }, w), f.setEffect(o, d); } if (!o.isReady()) return !1; } return (this.useBlurExponentialShadowMap || this.useBlurCloseExponentialShadowMap) && (!this._blurPostProcesses || !this._blurPostProcesses.length) && this._initializeBlurRTTAndPostProcesses(), !(this._kernelBlurXPostprocess && !this._kernelBlurXPostprocess.isReady() || this._kernelBlurYPostprocess && !this._kernelBlurYPostprocess.isReady() || this._boxBlurPostprocess && !this._boxBlurPostprocess.isReady()); } /** * Prepare all the defines in a material relying on a shadow map at the specified light index. * @param defines Defines of the material we want to update * @param lightIndex Index of the light in the enabled light list of the material */ prepareDefines(e, t) { const r = this._scene, n = this._light; !r.shadowsEnabled || !n.shadowEnabled || (e["SHADOW" + t] = !0, this.useContactHardeningShadow ? (e["SHADOWPCSS" + t] = !0, this._filteringQuality === ln.QUALITY_LOW ? e["SHADOWLOWQUALITY" + t] = !0 : this._filteringQuality === ln.QUALITY_MEDIUM && (e["SHADOWMEDIUMQUALITY" + t] = !0)) : this.usePercentageCloserFiltering ? (e["SHADOWPCF" + t] = !0, this._filteringQuality === ln.QUALITY_LOW ? e["SHADOWLOWQUALITY" + t] = !0 : this._filteringQuality === ln.QUALITY_MEDIUM && (e["SHADOWMEDIUMQUALITY" + t] = !0)) : this.usePoissonSampling ? e["SHADOWPOISSON" + t] = !0 : this.useExponentialShadowMap || this.useBlurExponentialShadowMap ? e["SHADOWESM" + t] = !0 : (this.useCloseExponentialShadowMap || this.useBlurCloseExponentialShadowMap) && (e["SHADOWCLOSEESM" + t] = !0), n.needCube() && (e["SHADOWCUBE" + t] = !0)); } /** * Binds the shadow related information inside of an effect (information like near, far, darkness... * defined in the generator but impacting the effect). * @param lightIndex Index of the light in the enabled light list of the material owning the effect * @param effect The effect we are binding the information for */ bindShadowLight(e, t) { const r = this._light; if (!this._scene.shadowsEnabled || !r.shadowEnabled) return; const i = this._getCamera(); if (!i) return; const s = this.getShadowMap(); s && (r.needCube() || t.setMatrix("lightMatrix" + e, this.getTransformMatrix()), this._filter === ln.FILTER_PCF ? (t.setDepthStencilTexture("shadowSampler" + e, this.getShadowMapForRendering()), r._uniformBuffer.updateFloat4("shadowsInfo", this.getDarkness(), s.getSize().width, 1 / s.getSize().width, this.frustumEdgeFalloff, e)) : this._filter === ln.FILTER_PCSS ? (t.setDepthStencilTexture("shadowSampler" + e, this.getShadowMapForRendering()), t.setTexture("depthSampler" + e, this.getShadowMapForRendering()), r._uniformBuffer.updateFloat4("shadowsInfo", this.getDarkness(), 1 / s.getSize().width, this._contactHardeningLightSizeUVRatio * s.getSize().width, this.frustumEdgeFalloff, e)) : (t.setTexture("shadowSampler" + e, this.getShadowMapForRendering()), r._uniformBuffer.updateFloat4("shadowsInfo", this.getDarkness(), this.blurScale / s.getSize().width, this.depthScale, this.frustumEdgeFalloff, e)), r._uniformBuffer.updateFloat2("depthValues", this.getLight().getDepthMinZ(i), this.getLight().getDepthMinZ(i) + this.getLight().getDepthMaxZ(i), e)); } /** * Gets the view matrix used to render the shadow map. */ get viewMatrix() { return this._viewMatrix; } /** * Gets the projection matrix used to render the shadow map. */ get projectionMatrix() { return this._projectionMatrix; } /** * Gets the transformation matrix used to project the meshes into the map from the light point of view. * (eq to shadow projection matrix * light transform matrix) * @returns The transform matrix used to create the shadow map */ getTransformMatrix() { const e = this._scene; if (this._currentRenderId === e.getRenderId() && this._currentFaceIndexCache === this._currentFaceIndex) return this._transformMatrix; this._currentRenderId = e.getRenderId(), this._currentFaceIndexCache = this._currentFaceIndex; let t = this._light.position; if (this._light.computeTransformedInformation() && (t = this._light.transformedPosition), S.NormalizeToRef(this._light.getShadowDirection(this._currentFaceIndex), this._lightDirection), Math.abs(S.Dot(this._lightDirection, S.Up())) === 1 && (this._lightDirection.z = 1e-13), this._light.needProjectionMatrixCompute() || !this._cachedPosition || !this._cachedDirection || !t.equals(this._cachedPosition) || !this._lightDirection.equals(this._cachedDirection)) { this._cachedPosition.copyFrom(t), this._cachedDirection.copyFrom(this._lightDirection), he.LookAtLHToRef(t, t.add(this._lightDirection), S.Up(), this._viewMatrix); const r = this.getShadowMap(); if (r) { const n = r.renderList; n && this._light.setShadowProjectionMatrix(this._projectionMatrix, this._viewMatrix, n); } this._viewMatrix.multiplyToRef(this._projectionMatrix, this._transformMatrix); } return this._transformMatrix; } /** * Recreates the shadow map dependencies like RTT and post processes. This can be used during the switch between * Cube and 2D textures for instance. */ recreateShadowMap() { const e = this._shadowMap; if (!e) return; const t = e.renderList; if (this._disposeRTTandPostProcesses(), this._initializeGenerator(), this.filter = this._filter, this._applyFilterValues(), t) { this._shadowMap.renderList || (this._shadowMap.renderList = []); for (const r of t) this._shadowMap.renderList.push(r); } else this._shadowMap.renderList = null; } _disposeBlurPostProcesses() { this._shadowMap2 && (this._shadowMap2.dispose(), this._shadowMap2 = null), this._boxBlurPostprocess && (this._boxBlurPostprocess.dispose(), this._boxBlurPostprocess = null), this._kernelBlurXPostprocess && (this._kernelBlurXPostprocess.dispose(), this._kernelBlurXPostprocess = null), this._kernelBlurYPostprocess && (this._kernelBlurYPostprocess.dispose(), this._kernelBlurYPostprocess = null), this._blurPostProcesses = []; } _disposeRTTandPostProcesses() { this._shadowMap && (this._shadowMap.dispose(), this._shadowMap = null), this._disposeBlurPostProcesses(); } _disposeSceneUBOs() { if (this._sceneUBOs) { for (const e of this._sceneUBOs) e.dispose(); this._sceneUBOs = []; } } /** * Disposes the ShadowGenerator. * Returns nothing. */ dispose() { if (this._disposeRTTandPostProcesses(), this._disposeSceneUBOs(), this._light) { if (this._light._shadowGenerators) { const e = this._light._shadowGenerators.entries(); for (let t = e.next(); t.done !== !0; t = e.next()) { const [r, n] = t.value; n === this && this._light._shadowGenerators.delete(r); } this._light._shadowGenerators.size === 0 && (this._light._shadowGenerators = null); } this._light._markMeshesAsLightDirty(); } this.onBeforeShadowMapRenderMeshObservable.clear(), this.onBeforeShadowMapRenderObservable.clear(), this.onAfterShadowMapRenderMeshObservable.clear(), this.onAfterShadowMapRenderObservable.clear(); } /** * Serializes the shadow generator setup to a json object. * @returns The serialized JSON object */ serialize() { var e; const t = {}, r = this.getShadowMap(); if (!r) return t; if (t.className = this.getClassName(), t.lightId = this._light.id, t.cameraId = (e = this._camera) === null || e === void 0 ? void 0 : e.id, t.id = this.id, t.mapSize = r.getRenderSize(), t.forceBackFacesOnly = this.forceBackFacesOnly, t.darkness = this.getDarkness(), t.transparencyShadow = this._transparencyShadow, t.frustumEdgeFalloff = this.frustumEdgeFalloff, t.bias = this.bias, t.normalBias = this.normalBias, t.usePercentageCloserFiltering = this.usePercentageCloserFiltering, t.useContactHardeningShadow = this.useContactHardeningShadow, t.contactHardeningLightSizeUVRatio = this.contactHardeningLightSizeUVRatio, t.filteringQuality = this.filteringQuality, t.useExponentialShadowMap = this.useExponentialShadowMap, t.useBlurExponentialShadowMap = this.useBlurExponentialShadowMap, t.useCloseExponentialShadowMap = this.useBlurExponentialShadowMap, t.useBlurCloseExponentialShadowMap = this.useBlurExponentialShadowMap, t.usePoissonSampling = this.usePoissonSampling, t.depthScale = this.depthScale, t.blurBoxOffset = this.blurBoxOffset, t.blurKernel = this.blurKernel, t.blurScale = this.blurScale, t.useKernelBlur = this.useKernelBlur, t.renderList = [], r.renderList) for (let n = 0; n < r.renderList.length; n++) { const i = r.renderList[n]; t.renderList.push(i.id); } return t; } /** * Parses a serialized ShadowGenerator and returns a new ShadowGenerator. * @param parsedShadowGenerator The JSON object to parse * @param scene The scene to create the shadow map for * @param constr A function that builds a shadow generator or undefined to create an instance of the default shadow generator * @returns The parsed shadow generator */ static Parse(e, t, r) { const n = t.getLightById(e.lightId), i = e.cameraId !== void 0 ? t.getCameraById(e.cameraId) : null, s = r ? r(e.mapSize, n, i) : new ln(e.mapSize, n, void 0, i), a = s.getShadowMap(); for (let f = 0; f < e.renderList.length; f++) t.getMeshesById(e.renderList[f]).forEach(function(d) { a && (a.renderList || (a.renderList = []), a.renderList.push(d)); }); return e.id !== void 0 && (s.id = e.id), s.forceBackFacesOnly = !!e.forceBackFacesOnly, e.darkness !== void 0 && s.setDarkness(e.darkness), e.transparencyShadow && s.setTransparencyShadow(!0), e.frustumEdgeFalloff !== void 0 && (s.frustumEdgeFalloff = e.frustumEdgeFalloff), e.bias !== void 0 && (s.bias = e.bias), e.normalBias !== void 0 && (s.normalBias = e.normalBias), e.usePercentageCloserFiltering ? s.usePercentageCloserFiltering = !0 : e.useContactHardeningShadow ? s.useContactHardeningShadow = !0 : e.usePoissonSampling ? s.usePoissonSampling = !0 : e.useExponentialShadowMap ? s.useExponentialShadowMap = !0 : e.useBlurExponentialShadowMap ? s.useBlurExponentialShadowMap = !0 : e.useCloseExponentialShadowMap ? s.useCloseExponentialShadowMap = !0 : e.useBlurCloseExponentialShadowMap ? s.useBlurCloseExponentialShadowMap = !0 : e.useVarianceShadowMap ? s.useExponentialShadowMap = !0 : e.useBlurVarianceShadowMap && (s.useBlurExponentialShadowMap = !0), e.contactHardeningLightSizeUVRatio !== void 0 && (s.contactHardeningLightSizeUVRatio = e.contactHardeningLightSizeUVRatio), e.filteringQuality !== void 0 && (s.filteringQuality = e.filteringQuality), e.depthScale && (s.depthScale = e.depthScale), e.blurScale && (s.blurScale = e.blurScale), e.blurBoxOffset && (s.blurBoxOffset = e.blurBoxOffset), e.useKernelBlur && (s.useKernelBlur = e.useKernelBlur), e.blurKernel && (s.blurKernel = e.blurKernel), s; } } ln.CLASSNAME = "ShadowGenerator"; ln.FILTER_NONE = 0; ln.FILTER_EXPONENTIALSHADOWMAP = 1; ln.FILTER_POISSONSAMPLING = 2; ln.FILTER_BLUREXPONENTIALSHADOWMAP = 3; ln.FILTER_CLOSEEXPONENTIALSHADOWMAP = 4; ln.FILTER_BLURCLOSEEXPONENTIALSHADOWMAP = 5; ln.FILTER_PCF = 6; ln.FILTER_PCSS = 7; ln.QUALITY_HIGH = 0; ln.QUALITY_MEDIUM = 1; ln.QUALITY_LOW = 2; ln.DEFAULT_ALPHA_CUTOFF = 0.5; ln._SceneComponentInitialization = (A) => { throw qn("ShadowGeneratorSceneComponent"); }; const fpe = "depthPixelShader", Ape = `#ifdef ALPHATEST varying vec2 vUV;uniform sampler2D diffuseSampler; #endif #include varying float vDepthMetric; #ifdef PACKED #include #endif #ifdef STORE_CAMERASPACE_Z varying vec4 vViewPos; #endif #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #include #ifdef ALPHATEST if (texture2D(diffuseSampler,vUV).a<0.4) discard; #endif #ifdef STORE_CAMERASPACE_Z #ifdef PACKED gl_FragColor=pack(vViewPos.z); #else gl_FragColor=vec4(vViewPos.z,0.0,0.0,1.0); #endif #else #ifdef NONLINEARDEPTH #ifdef PACKED gl_FragColor=pack(gl_FragCoord.z); #else gl_FragColor=vec4(gl_FragCoord.z,0.0,0.0,0.0); #endif #else #ifdef PACKED gl_FragColor=pack(vDepthMetric); #else gl_FragColor=vec4(vDepthMetric,0.0,0.0,1.0); #endif #endif #endif }`; Le.ShadersStore[fpe] = Ape; const dpe = "pointCloudVertexDeclaration", vpe = `#ifdef POINTSIZE uniform float pointSize; #endif `; Le.IncludesShadersStore[dpe] = vpe; const upe = "depthVertexShader", lpe = `attribute vec3 position; #include #include #include #include[0..maxSimultaneousMorphTargets] #include #include uniform mat4 viewProjection;uniform vec2 depthValues; #if defined(ALPHATEST) || defined(NEED_UV) varying vec2 vUV;uniform mat4 diffuseMatrix; #ifdef UV1 attribute vec2 uv; #endif #ifdef UV2 attribute vec2 uv2; #endif #endif #ifdef STORE_CAMERASPACE_Z uniform mat4 view;varying vec4 vViewPos; #endif #include varying float vDepthMetric; #define CUSTOM_VERTEX_DEFINITIONS void main(void) {vec3 positionUpdated=position; #ifdef UV1 vec2 uvUpdated=uv; #endif #include #include[0..maxSimultaneousMorphTargets] #include #include #include vec4 worldPos=finalWorld*vec4(positionUpdated,1.0); #include gl_Position=viewProjection*worldPos; #ifdef STORE_CAMERASPACE_Z vViewPos=view*worldPos; #else #ifdef USE_REVERSE_DEPTHBUFFER vDepthMetric=((-gl_Position.z+depthValues.x)/(depthValues.y)); #else vDepthMetric=((gl_Position.z+depthValues.x)/(depthValues.y)); #endif #endif #if defined(ALPHATEST) || defined(BASIC_RENDER) #ifdef UV1 vUV=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0)); #endif #ifdef UV2 vUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0)); #endif #endif #include } `; Le.ShadersStore[upe] = lpe; class jD { /** * Sets a specific material to be used to render a mesh/a list of meshes by the depth renderer * @param mesh mesh or array of meshes * @param material material to use by the depth render when rendering the mesh(es). If undefined is passed, the specific material created by the depth renderer will be used. */ setMaterialForRendering(e, t) { this._depthMap.setMaterialForRendering(e, t); } /** * Instantiates a depth renderer * @param scene The scene the renderer belongs to * @param type The texture type of the depth map (default: Engine.TEXTURETYPE_FLOAT) * @param camera The camera to be used to render the depth map (default: scene's active camera) * @param storeNonLinearDepth Defines whether the depth is stored linearly like in Babylon Shadows or directly like glFragCoord.z * @param samplingMode The sampling mode to be used with the render target (Linear, Nearest...) (default: TRILINEAR_SAMPLINGMODE) * @param storeCameraSpaceZ Defines whether the depth stored is the Z coordinate in camera space. If true, storeNonLinearDepth has no effect. (Default: false) * @param name Name of the render target (default: DepthRenderer) */ constructor(e, t = 1, r = null, n = !1, i = We.TRILINEAR_SAMPLINGMODE, s = !1, a) { this.enabled = !0, this.forceDepthWriteTransparentMeshes = !1, this.useOnlyInActiveCamera = !1, this.reverseCulling = !1, this._scene = e, this._storeNonLinearDepth = n, this._storeCameraSpaceZ = s, this.isPacked = t === 0, this.isPacked ? this.clearColor = new xt(1, 1, 1, 1) : this.clearColor = new xt(s ? 1e8 : 1, 0, 0, 1), jD._SceneComponentInitialization(this._scene); const f = e.getEngine(); this._camera = r, i !== We.NEAREST_SAMPLINGMODE && (t === 1 && !f._caps.textureFloatLinearFiltering && (i = We.NEAREST_SAMPLINGMODE), t === 2 && !f._caps.textureHalfFloatLinearFiltering && (i = We.NEAREST_SAMPLINGMODE)); const o = this.isPacked || !f._features.supportExtendedTextureFormats ? 5 : 6; this._depthMap = new Ta(a ?? "DepthRenderer", { width: f.getRenderWidth(), height: f.getRenderHeight() }, this._scene, !1, !0, t, !1, i, void 0, void 0, void 0, o), this._depthMap.wrapU = We.CLAMP_ADDRESSMODE, this._depthMap.wrapV = We.CLAMP_ADDRESSMODE, this._depthMap.refreshRate = 1, this._depthMap.renderParticles = !1, this._depthMap.renderList = null, this._depthMap.noPrePassRenderer = !0, this._depthMap.activeCamera = this._camera, this._depthMap.ignoreCameraViewport = !0, this._depthMap.useCameraPostProcesses = !1, this._depthMap.onClearObservable.add((v) => { v.clear(this.clearColor, !0, !0, !0); }), this._depthMap.onBeforeBindObservable.add(() => { var v; (v = f._debugPushGroup) === null || v === void 0 || v.call(f, "depth renderer", 1); }), this._depthMap.onAfterUnbindObservable.add(() => { var v; (v = f._debugPopGroup) === null || v === void 0 || v.call(f, 1); }), this._depthMap.customIsReadyFunction = (v, u, l) => { if ((l || u === 0) && v.subMeshes) for (let P = 0; P < v.subMeshes.length; ++P) { const p = v.subMeshes[P], c = p.getRenderingMesh(), H = c._getInstancesRenderList(p._id, !!p.getReplacementMesh()), T = f.getCaps().instancedArrays && (H.visibleInstances[p._id] !== null && H.visibleInstances[p._id] !== void 0 || c.hasThinInstances); if (!this.isReady(p, T)) return !1; } return !0; }; const d = (v) => { var u, l; const P = v.getRenderingMesh(), p = v.getEffectiveMesh(), c = this._scene, H = c.getEngine(), T = v.getMaterial(); if (p._internalAbstractMeshDataInfo._isActiveIntermediate = !1, !T || p.infiniteDistance || T.disableDepthWrite || v.verticesCount === 0 || v._renderId === c.getRenderId()) return; const q = p._getWorldMatrixDeterminant() < 0; let b = (u = P.overrideMaterialSideOrientation) !== null && u !== void 0 ? u : T.sideOrientation; q && (b = b === 0 ? 1 : 0); const j = b === 0; H.setState(T.backFaceCulling, 0, !1, j, this.reverseCulling ? !T.cullBackFaces : T.cullBackFaces); const w = P._getInstancesRenderList(v._id, !!v.getReplacementMesh()); if (w.mustReturn) return; const m = H.getCaps().instancedArrays && (w.visibleInstances[v._id] !== null && w.visibleInstances[v._id] !== void 0 || P.hasThinInstances), I = this._camera || c.activeCamera; if (this.isReady(v, m) && I) { v._renderId = c.getRenderId(); const N = (l = p._internalAbstractMeshDataInfo._materialForRenderPass) === null || l === void 0 ? void 0 : l[H.currentRenderPassId]; let k = v._getDrawWrapper(); !k && N && (k = N._getDrawWrapper()); const R = I.mode === Tr.ORTHOGRAPHIC_CAMERA; if (!k) return; const y = k.effect; H.enableEffect(k), m || P._bind(v, y, T.fillMode), N ? N.bindForSubMesh(p.getWorldMatrix(), p, v) : (y.setMatrix("viewProjection", c.getTransformMatrix()), y.setMatrix("world", p.getWorldMatrix()), this._storeCameraSpaceZ && y.setMatrix("view", c.getViewMatrix())); let O, Y; if (R ? (O = !H.useReverseDepthBuffer && H.isNDCHalfZRange ? 0 : 1, Y = H.useReverseDepthBuffer && H.isNDCHalfZRange ? 0 : 1) : (O = H.useReverseDepthBuffer && H.isNDCHalfZRange ? I.minZ : H.isNDCHalfZRange ? 0 : I.minZ, Y = H.useReverseDepthBuffer && H.isNDCHalfZRange ? 0 : I.maxZ), y.setFloat2("depthValues", O, O + Y), !N) { if (T.needAlphaTesting()) { const ee = T.getAlphaTestTexture(); ee && (y.setTexture("diffuseSampler", ee), y.setMatrix("diffuseMatrix", ee.getTextureMatrix())); } if (P.useBones && P.computeBonesUsingShaders && P.skeleton) { const ee = P.skeleton; if (ee.isUsingTextureForMatrices) { const Z = ee.getTransformMatrixTexture(P); if (!Z) return; y.setTexture("boneSampler", Z), y.setFloat("boneTextureWidth", 4 * (ee.bones.length + 1)); } else y.setMatrices("mBones", ee.getTransformMatrices(P)); } Df(y, T, c), Ye.BindMorphTargetParameters(P, y), P.morphTargetManager && P.morphTargetManager.isUsingTextureForTargets && P.morphTargetManager._bind(y), T.pointsCloud && y.setFloat("pointSize", T.pointSize); } P._processRendering(p, v, y, T.fillMode, w, m, (ee, Z) => y.setMatrix("world", Z)); } }; this._depthMap.customRenderFunction = (v, u, l, P) => { let p; if (P.length) for (p = 0; p < P.length; p++) d(P.data[p]); for (p = 0; p < v.length; p++) d(v.data[p]); for (p = 0; p < u.length; p++) d(u.data[p]); if (this.forceDepthWriteTransparentMeshes) for (p = 0; p < l.length; p++) d(l.data[p]); else for (p = 0; p < l.length; p++) l.data[p].getEffectiveMesh()._internalAbstractMeshDataInfo._isActiveIntermediate = !1; }; } /** * Creates the depth rendering effect and checks if the effect is ready. * @param subMesh The submesh to be used to render the depth map of * @param useInstances If multiple world instances should be used * @returns if the depth renderer is ready to render the depth map */ isReady(e, t) { var r; const n = this._scene.getEngine(), i = e.getMesh(), s = i.getScene(), a = (r = i._internalAbstractMeshDataInfo._materialForRenderPass) === null || r === void 0 ? void 0 : r[n.currentRenderPassId]; if (a) return a.isReadyForSubMesh(i, e, t); const f = e.getMaterial(); if (!f || f.disableDepthWrite) return !1; const o = [], d = [J.PositionKind]; if (f && f.needAlphaTesting() && f.getAlphaTestTexture() && (o.push("#define ALPHATEST"), i.isVerticesDataPresent(J.UVKind) && (d.push(J.UVKind), o.push("#define UV1")), i.isVerticesDataPresent(J.UV2Kind) && (d.push(J.UV2Kind), o.push("#define UV2"))), i.useBones && i.computeBonesUsingShaders) { d.push(J.MatricesIndicesKind), d.push(J.MatricesWeightsKind), i.numBoneInfluencers > 4 && (d.push(J.MatricesIndicesExtraKind), d.push(J.MatricesWeightsExtraKind)), o.push("#define NUM_BONE_INFLUENCERS " + i.numBoneInfluencers), o.push("#define BonesPerMesh " + (i.skeleton ? i.skeleton.bones.length + 1 : 0)); const c = e.getRenderingMesh().skeleton; c != null && c.isUsingTextureForMatrices && o.push("#define BONETEXTURE"); } else o.push("#define NUM_BONE_INFLUENCERS 0"); const v = i.morphTargetManager; let u = 0; v && v.numInfluencers > 0 && (u = v.numInfluencers, o.push("#define MORPHTARGETS"), o.push("#define NUM_MORPH_INFLUENCERS " + u), v.isUsingTextureForTargets && o.push("#define MORPHTARGETS_TEXTURE"), Ye.PrepareAttributesForMorphTargetsInfluencers(d, i, u)), f.pointsCloud && o.push("#define POINTSIZE"), t && (o.push("#define INSTANCES"), Ye.PushAttributesForInstances(d), e.getRenderingMesh().hasThinInstances && o.push("#define THIN_INSTANCES")), this._storeNonLinearDepth && o.push("#define NONLINEARDEPTH"), this._storeCameraSpaceZ && o.push("#define STORE_CAMERASPACE_Z"), this.isPacked && o.push("#define PACKED"), xq(f, s, o); const l = e._getDrawWrapper(void 0, !0), P = l.defines, p = o.join(` `); if (P !== p) { const c = [ "world", "mBones", "boneTextureWidth", "pointSize", "viewProjection", "view", "diffuseMatrix", "depthValues", "morphTargetInfluences", "morphTargetTextureInfo", "morphTargetTextureIndices" ]; Mf(c), l.setEffect(n.createEffect("depth", d, c, ["diffuseSampler", "morphTargets", "boneSampler"], p, void 0, void 0, void 0, { maxSimultaneousMorphTargets: u }), p); } return l.effect.isReady(); } /** * Gets the texture which the depth map will be written to. * @returns The depth map texture */ getDepthMap() { return this._depthMap; } /** * Disposes of the depth renderer. */ dispose() { const e = []; for (const t in this._scene._depthRenderer) this._scene._depthRenderer[t] === this && e.push(t); if (e.length > 0) { this._depthMap.dispose(); for (const t of e) delete this._scene._depthRenderer[t]; } } } jD._SceneComponentInitialization = (A) => { throw qn("DepthRendererSceneComponent"); }; const Ppe = "minmaxReduxPixelShader", cpe = `varying vec2 vUV;uniform sampler2D textureSampler; #if defined(INITIAL) uniform sampler2D sourceTexture;uniform vec2 texSize;void main(void) {ivec2 coord=ivec2(vUV*(texSize-1.0));float f1=texelFetch(sourceTexture,coord,0).r;float f2=texelFetch(sourceTexture,coord+ivec2(1,0),0).r;float f3=texelFetch(sourceTexture,coord+ivec2(1,1),0).r;float f4=texelFetch(sourceTexture,coord+ivec2(0,1),0).r;float minz=min(min(min(f1,f2),f3),f4); #ifdef DEPTH_REDUX float maxz=max(max(max(sign(1.0-f1)*f1,sign(1.0-f2)*f2),sign(1.0-f3)*f3),sign(1.0-f4)*f4); #else float maxz=max(max(max(f1,f2),f3),f4); #endif glFragColor=vec4(minz,maxz,0.,0.);} #elif defined(MAIN) uniform vec2 texSize;void main(void) {ivec2 coord=ivec2(vUV*(texSize-1.0));vec2 f1=texelFetch(textureSampler,coord,0).rg;vec2 f2=texelFetch(textureSampler,coord+ivec2(1,0),0).rg;vec2 f3=texelFetch(textureSampler,coord+ivec2(1,1),0).rg;vec2 f4=texelFetch(textureSampler,coord+ivec2(0,1),0).rg;float minz=min(min(min(f1.x,f2.x),f3.x),f4.x);float maxz=max(max(max(f1.y,f2.y),f3.y),f4.y);glFragColor=vec4(minz,maxz,0.,0.);} #elif defined(ONEBEFORELAST) uniform ivec2 texSize;void main(void) {ivec2 coord=ivec2(vUV*vec2(texSize-1));vec2 f1=texelFetch(textureSampler,coord % texSize,0).rg;vec2 f2=texelFetch(textureSampler,(coord+ivec2(1,0)) % texSize,0).rg;vec2 f3=texelFetch(textureSampler,(coord+ivec2(1,1)) % texSize,0).rg;vec2 f4=texelFetch(textureSampler,(coord+ivec2(0,1)) % texSize,0).rg;float minz=min(f1.x,f2.x);float maxz=max(f1.y,f2.y);glFragColor=vec4(minz,maxz,0.,0.);} #elif defined(LAST) void main(void) {glFragColor=vec4(0.);if (true) { discard;}} #endif `; Le.ShadersStore[Ppe] = cpe; class Lte { /** * Creates a min/max reducer * @param camera The camera to use for the post processes */ constructor(e) { this.onAfterReductionPerformed = new Oe(), this._forceFullscreenViewport = !0, this._activated = !1, this._camera = e, this._postProcessManager = new yI(e.getScene()), this._onContextRestoredObserver = e.getEngine().onContextRestoredObservable.add(() => { this._postProcessManager._rebuild(); }); } /** * Gets the texture used to read the values from. */ get sourceTexture() { return this._sourceTexture; } /** * Sets the source texture to read the values from. * One must indicate if the texture is a depth texture or not through the depthRedux parameter * because in such textures '1' value must not be taken into account to compute the maximum * as this value is used to clear the texture. * Note that the computation is not activated by calling this function, you must call activate() for that! * @param sourceTexture The texture to read the values from. The values should be in the red channel. * @param depthRedux Indicates if the texture is a depth texture or not * @param type The type of the textures created for the reduction (defaults to TEXTURETYPE_HALF_FLOAT) * @param forceFullscreenViewport Forces the post processes used for the reduction to be applied without taking into account viewport (defaults to true) */ setSourceTexture(e, t, r = 2, n = !0) { if (e === this._sourceTexture) return; this.dispose(!1), this._sourceTexture = e, this._reductionSteps = [], this._forceFullscreenViewport = n; const i = this._camera.getScene(), s = new kr( "Initial reduction phase", "minmaxRedux", // shader ["texSize"], ["sourceTexture"], // textures 1, // options null, // camera 1, // sampling i.getEngine(), // engine !1, // reusable "#define INITIAL" + (t ? ` #define DEPTH_REDUX` : ""), // defines r, void 0, void 0, void 0, 7 ); s.autoClear = !1, s.forceFullscreenViewport = n; let a = this._sourceTexture.getRenderWidth(), f = this._sourceTexture.getRenderHeight(); s.onApply = /* @__PURE__ */ ((d, v) => (u) => { u.setTexture("sourceTexture", this._sourceTexture), u.setFloat2("texSize", d, v); })(a, f), this._reductionSteps.push(s); let o = 1; for (; a > 1 || f > 1; ) { a = Math.max(Math.round(a / 2), 1), f = Math.max(Math.round(f / 2), 1); const d = new kr( "Reduction phase " + o, "minmaxRedux", // shader ["texSize"], null, { width: a, height: f }, // options null, // camera 1, // sampling i.getEngine(), // engine !1, // reusable "#define " + (a == 1 && f == 1 ? "LAST" : a == 1 || f == 1 ? "ONEBEFORELAST" : "MAIN"), // defines r, void 0, void 0, void 0, 7 ); if (d.autoClear = !1, d.forceFullscreenViewport = n, d.onApply = /* @__PURE__ */ ((v, u) => (l) => { v == 1 || u == 1 ? l.setInt2("texSize", v, u) : l.setFloat2("texSize", v, u); })(a, f), this._reductionSteps.push(d), o++, a == 1 && f == 1) { const v = (u, l, P) => { const p = new Float32Array(4 * u * l), c = { min: 0, max: 0 }; return () => { i.getEngine()._readTexturePixels(P.inputTexture.texture, u, l, -1, 0, p, !1), c.min = p[0], c.max = p[1], this.onAfterReductionPerformed.notifyObservers(c); }; }; d.onAfterRenderObservable.add(v(a, f, d)); } } } /** * Defines the refresh rate of the computation. * Use 0 to compute just once, 1 to compute on every frame, 2 to compute every two frames and so on... */ get refreshRate() { return this._sourceTexture ? this._sourceTexture.refreshRate : -1; } set refreshRate(e) { this._sourceTexture && (this._sourceTexture.refreshRate = e); } /** * Gets the activation status of the reducer */ get activated() { return this._activated; } /** * Activates the reduction computation. * When activated, the observers registered in onAfterReductionPerformed are * called after the computation is performed */ activate() { this._onAfterUnbindObserver || !this._sourceTexture || (this._onAfterUnbindObserver = this._sourceTexture.onAfterUnbindObservable.add(() => { var e, t; const r = this._camera.getScene().getEngine(); (e = r._debugPushGroup) === null || e === void 0 || e.call(r, "min max reduction", 1), this._reductionSteps[0].activate(this._camera), this._postProcessManager.directRender(this._reductionSteps, this._reductionSteps[0].inputTexture, this._forceFullscreenViewport), r.unBindFramebuffer(this._reductionSteps[0].inputTexture, !1), (t = r._debugPopGroup) === null || t === void 0 || t.call(r, 1); }), this._activated = !0); } /** * Deactivates the reduction computation. */ deactivate() { !this._onAfterUnbindObserver || !this._sourceTexture || (this._sourceTexture.onAfterUnbindObservable.remove(this._onAfterUnbindObserver), this._onAfterUnbindObserver = null, this._activated = !1); } /** * Disposes the min/max reducer * @param disposeAll true to dispose all the resources. You should always call this function with true as the parameter (or without any parameter as it is the default one). This flag is meant to be used internally. */ dispose(e = !0) { if (e && (this.onAfterReductionPerformed.clear(), this._onContextRestoredObserver && (this._camera.getEngine().onContextRestoredObservable.remove(this._onContextRestoredObserver), this._onContextRestoredObserver = null)), this.deactivate(), this._reductionSteps) { for (let t = 0; t < this._reductionSteps.length; ++t) this._reductionSteps[t].dispose(); this._reductionSteps = null; } this._postProcessManager && e && this._postProcessManager.dispose(), this._sourceTexture = null; } } class Kte extends Lte { /** * Gets the depth renderer used for the computation. * Note that the result is null if you provide your own renderer when calling setDepthRenderer. */ get depthRenderer() { return this._depthRenderer; } /** * Creates a depth reducer * @param camera The camera used to render the depth texture */ constructor(e) { super(e); } /** * Sets the depth renderer to use to generate the depth map * @param depthRenderer The depth renderer to use. If not provided, a new one will be created automatically * @param type The texture type of the depth map (default: TEXTURETYPE_HALF_FLOAT) * @param forceFullscreenViewport Forces the post processes used for the reduction to be applied without taking into account viewport (defaults to true) */ setDepthRenderer(e = null, t = 2, r = !0) { const n = this._camera.getScene(); this._depthRenderer && (delete n._depthRenderer[this._depthRendererId], this._depthRenderer.dispose(), this._depthRenderer = null), e === null && (n._depthRenderer || (n._depthRenderer = {}), e = this._depthRenderer = new jD(n, t, this._camera, !1, 1), e.enabled = !1, this._depthRendererId = "minmax" + this._camera.id, n._depthRenderer[this._depthRendererId] = e), super.setSourceTexture(e.getDepthMap(), !0, t, r); } /** * @internal */ setSourceTexture(e, t, r = 2, n = !0) { super.setSourceTexture(e, t, r, n); } /** * Activates the reduction computation. * When activated, the observers registered in onAfterReductionPerformed are * called after the computation is performed */ activate() { this._depthRenderer && (this._depthRenderer.enabled = !0), super.activate(); } /** * Deactivates the reduction computation. */ deactivate() { super.deactivate(), this._depthRenderer && (this._depthRenderer.enabled = !1); } /** * Disposes the depth reducer * @param disposeAll true to dispose all the resources. You should always call this function with true as the parameter (or without any parameter as it is the default one). This flag is meant to be used internally. */ dispose(e = !0) { if (super.dispose(e), this._depthRenderer && e) { const t = this._depthRenderer.getDepthMap().getScene(); t && delete t._depthRenderer[this._depthRendererId], this._depthRenderer.dispose(), this._depthRenderer = null; } } } const MG = S.Up(), ppe = S.Zero(), R9 = new S(), IW = new S(), HC = new he(); class X6 extends ln { _validateFilter(e) { return e === ln.FILTER_NONE || e === ln.FILTER_PCF || e === ln.FILTER_PCSS ? e : (console.error('Unsupported filter "' + e + '"!'), ln.FILTER_NONE); } /** * Gets or set the number of cascades used by the CSM. */ get numCascades() { return this._numCascades; } set numCascades(e) { e = Math.min(Math.max(e, X6.MIN_CASCADES_COUNT), X6.MAX_CASCADES_COUNT), e !== this._numCascades && (this._numCascades = e, this.recreateShadowMap(), this._recreateSceneUBOs()); } /** * Enables or disables the shadow casters bounding info computation. * If your shadow casters don't move, you can disable this feature. * If it is enabled, the bounding box computation is done every frame. */ get freezeShadowCastersBoundingInfo() { return this._freezeShadowCastersBoundingInfo; } set freezeShadowCastersBoundingInfo(e) { this._freezeShadowCastersBoundingInfoObservable && e && (this._scene.onBeforeRenderObservable.remove(this._freezeShadowCastersBoundingInfoObservable), this._freezeShadowCastersBoundingInfoObservable = null), !this._freezeShadowCastersBoundingInfoObservable && !e && (this._freezeShadowCastersBoundingInfoObservable = this._scene.onBeforeRenderObservable.add(() => this._computeShadowCastersBoundingInfo())), this._freezeShadowCastersBoundingInfo = e, e && this._computeShadowCastersBoundingInfo(); } _computeShadowCastersBoundingInfo() { if (this._scbiMin.copyFromFloats(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE), this._scbiMax.copyFromFloats(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE), this._shadowMap && this._shadowMap.renderList) { const e = this._shadowMap.renderList; for (let r = 0; r < e.length; r++) { const n = e[r]; if (!n) continue; const i = n.getBoundingInfo(), s = i.boundingBox; this._scbiMin.minimizeInPlace(s.minimumWorld), this._scbiMax.maximizeInPlace(s.maximumWorld); } const t = this._scene.meshes; for (let r = 0; r < t.length; r++) { const n = t[r]; if (!n || !n.isVisible || !n.isEnabled || !n.receiveShadows) continue; const i = n.getBoundingInfo(), s = i.boundingBox; this._scbiMin.minimizeInPlace(s.minimumWorld), this._scbiMax.maximizeInPlace(s.maximumWorld); } } this._shadowCastersBoundingInfo.reConstruct(this._scbiMin, this._scbiMax); } /** * Gets or sets the shadow casters bounding info. * If you provide your own shadow casters bounding info, first enable freezeShadowCastersBoundingInfo * so that the system won't overwrite the bounds you provide */ get shadowCastersBoundingInfo() { return this._shadowCastersBoundingInfo; } set shadowCastersBoundingInfo(e) { this._shadowCastersBoundingInfo = e; } /** * Sets the minimal and maximal distances to use when computing the cascade breaks. * * The values of min / max are typically the depth zmin and zmax values of your scene, for a given frame. * If you don't know these values, simply leave them to their defaults and don't call this function. * @param min minimal distance for the breaks (default to 0.) * @param max maximal distance for the breaks (default to 1.) */ setMinMaxDistance(e, t) { this._minDistance === e && this._maxDistance === t || (e > t && (e = 0, t = 1), e < 0 && (e = 0), t > 1 && (t = 1), this._minDistance = e, this._maxDistance = t, this._breaksAreDirty = !0); } /** Gets the minimal distance used in the cascade break computation */ get minDistance() { return this._minDistance; } /** Gets the maximal distance used in the cascade break computation */ get maxDistance() { return this._maxDistance; } /** * Gets the class name of that object * @returns "CascadedShadowGenerator" */ getClassName() { return X6.CLASSNAME; } /** * Gets a cascade minimum extents * @param cascadeIndex index of the cascade * @returns the minimum cascade extents */ getCascadeMinExtents(e) { return e >= 0 && e < this._numCascades ? this._cascadeMinExtents[e] : null; } /** * Gets a cascade maximum extents * @param cascadeIndex index of the cascade * @returns the maximum cascade extents */ getCascadeMaxExtents(e) { return e >= 0 && e < this._numCascades ? this._cascadeMaxExtents[e] : null; } /** * Gets the shadow max z distance. It's the limit beyond which shadows are not displayed. * It defaults to camera.maxZ */ get shadowMaxZ() { return this._getCamera() ? this._shadowMaxZ : 0; } /** * Sets the shadow max z distance. */ set shadowMaxZ(e) { const t = this._getCamera(); if (!t) { this._shadowMaxZ = e; return; } this._shadowMaxZ === e || e < t.minZ || e > t.maxZ && t.maxZ !== 0 || (this._shadowMaxZ = e, this._light._markMeshesAsLightDirty(), this._breaksAreDirty = !0); } /** * Gets or sets the debug flag. * When enabled, the cascades are materialized by different colors on the screen. */ get debug() { return this._debug; } set debug(e) { this._debug = e, this._light._markMeshesAsLightDirty(); } /** * Gets or sets the depth clamping value. * * When enabled, it improves the shadow quality because the near z plane of the light frustum don't need to be adjusted * to account for the shadow casters far away. * * Note that this property is incompatible with PCSS filtering, so it won't be used in that case. */ get depthClamp() { return this._depthClamp; } set depthClamp(e) { this._depthClamp = e; } /** * Gets or sets the percentage of blending between two cascades (value between 0. and 1.). * It defaults to 0.1 (10% blending). */ get cascadeBlendPercentage() { return this._cascadeBlendPercentage; } set cascadeBlendPercentage(e) { this._cascadeBlendPercentage = e, this._light._markMeshesAsLightDirty(); } /** * Gets or set the lambda parameter. * This parameter is used to split the camera frustum and create the cascades. * It's a value between 0. and 1.: If 0, the split is a uniform split of the frustum, if 1 it is a logarithmic split. * For all values in-between, it's a linear combination of the uniform and logarithm split algorithm. */ get lambda() { return this._lambda; } set lambda(e) { const t = Math.min(Math.max(e, 0), 1); this._lambda != t && (this._lambda = t, this._breaksAreDirty = !0); } /** * Gets the view matrix corresponding to a given cascade * @param cascadeNum cascade to retrieve the view matrix from * @returns the cascade view matrix */ getCascadeViewMatrix(e) { return e >= 0 && e < this._numCascades ? this._viewMatrices[e] : null; } /** * Gets the projection matrix corresponding to a given cascade * @param cascadeNum cascade to retrieve the projection matrix from * @returns the cascade projection matrix */ getCascadeProjectionMatrix(e) { return e >= 0 && e < this._numCascades ? this._projectionMatrices[e] : null; } /** * Gets the transformation matrix corresponding to a given cascade * @param cascadeNum cascade to retrieve the transformation matrix from * @returns the cascade transformation matrix */ getCascadeTransformMatrix(e) { return e >= 0 && e < this._numCascades ? this._transformMatrices[e] : null; } /** * Sets the depth renderer to use when autoCalcDepthBounds is enabled. * * Note that if no depth renderer is set, a new one will be automatically created internally when necessary. * * You should call this function if you already have a depth renderer enabled in your scene, to avoid * doing multiple depth rendering each frame. If you provide your own depth renderer, make sure it stores linear depth! * @param depthRenderer The depth renderer to use when autoCalcDepthBounds is enabled. If you pass null or don't call this function at all, a depth renderer will be automatically created */ setDepthRenderer(e) { this._depthRenderer = e, this._depthReducer && this._depthReducer.setDepthRenderer(this._depthRenderer); } /** * Gets or sets the autoCalcDepthBounds property. * * When enabled, a depth rendering pass is first performed (with an internally created depth renderer or with the one * you provide by calling setDepthRenderer). Then, a min/max reducing is applied on the depth map to compute the * minimal and maximal depth of the map and those values are used as inputs for the setMinMaxDistance() function. * It can greatly enhance the shadow quality, at the expense of more GPU works. * When using this option, you should increase the value of the lambda parameter, and even set it to 1 for best results. */ get autoCalcDepthBounds() { return this._autoCalcDepthBounds; } set autoCalcDepthBounds(e) { const t = this._getCamera(); if (t) { if (this._autoCalcDepthBounds = e, !e) { this._depthReducer && this._depthReducer.deactivate(), this.setMinMaxDistance(0, 1); return; } this._depthReducer || (this._depthReducer = new Kte(t), this._depthReducer.onAfterReductionPerformed.add((r) => { let n = r.min, i = r.max; n >= i && (n = 0, i = 1), (n != this._minDistance || i != this._maxDistance) && this.setMinMaxDistance(n, i); }), this._depthReducer.setDepthRenderer(this._depthRenderer)), this._depthReducer.activate(); } } /** * Defines the refresh rate of the min/max computation used when autoCalcDepthBounds is set to true * Use 0 to compute just once, 1 to compute on every frame, 2 to compute every two frames and so on... * Note that if you provided your own depth renderer through a call to setDepthRenderer, you are responsible * for setting the refresh rate on the renderer yourself! */ get autoCalcDepthBoundsRefreshRate() { var e, t, r; return (r = (t = (e = this._depthReducer) === null || e === void 0 ? void 0 : e.depthRenderer) === null || t === void 0 ? void 0 : t.getDepthMap().refreshRate) !== null && r !== void 0 ? r : -1; } set autoCalcDepthBoundsRefreshRate(e) { var t; !((t = this._depthReducer) === null || t === void 0) && t.depthRenderer && (this._depthReducer.depthRenderer.getDepthMap().refreshRate = e); } /** * Create the cascade breaks according to the lambda, shadowMaxZ and min/max distance properties, as well as the camera near and far planes. * This function is automatically called when updating lambda, shadowMaxZ and min/max distances, however you should call it yourself if * you change the camera near/far planes! */ splitFrustum() { this._breaksAreDirty = !0; } _splitFrustum() { const e = this._getCamera(); if (!e) return; const t = e.minZ, r = e.maxZ || this._shadowMaxZ, n = r - t, i = this._minDistance, s = this._shadowMaxZ < r && this._shadowMaxZ >= t ? Math.min((this._shadowMaxZ - t) / (r - t), this._maxDistance) : this._maxDistance, a = t + i * n, f = t + s * n, o = f - a, d = f / a; for (let v = 0; v < this._cascades.length; ++v) { const u = (v + 1) / this._numCascades, l = a * d ** u, P = a + o * u, p = this._lambda * (l - P) + P; this._cascades[v].prevBreakDistance = v === 0 ? i : this._cascades[v - 1].breakDistance, this._cascades[v].breakDistance = (p - t) / n, this._viewSpaceFrustumsZ[v] = p, this._frustumLengths[v] = (this._cascades[v].breakDistance - this._cascades[v].prevBreakDistance) * n; } this._breaksAreDirty = !1; } _computeMatrices() { const e = this._scene; if (!this._getCamera()) return; S.NormalizeToRef(this._light.getShadowDirection(0), this._lightDirection), Math.abs(S.Dot(this._lightDirection, S.Up())) === 1 && (this._lightDirection.z = 1e-13), this._cachedDirection.copyFrom(this._lightDirection); const r = e.getEngine().useReverseDepthBuffer; for (let n = 0; n < this._numCascades; ++n) { this._computeFrustumInWorldSpace(n), this._computeCascadeFrustum(n), this._cascadeMaxExtents[n].subtractToRef(this._cascadeMinExtents[n], R9), this._frustumCenter[n].addToRef(this._lightDirection.scale(this._cascadeMinExtents[n].z), this._shadowCameraPos[n]), he.LookAtLHToRef(this._shadowCameraPos[n], this._frustumCenter[n], MG, this._viewMatrices[n]); let i = 0, s = R9.z; const a = this._shadowCastersBoundingInfo; a.update(this._viewMatrices[n]), s = Math.min(s, a.boundingBox.maximumWorld.z), !this._depthClamp || this.filter === ln.FILTER_PCSS ? i = Math.min(i, a.boundingBox.minimumWorld.z) : i = Math.max(i, a.boundingBox.minimumWorld.z), he.OrthoOffCenterLHToRef(this._cascadeMinExtents[n].x, this._cascadeMaxExtents[n].x, this._cascadeMinExtents[n].y, this._cascadeMaxExtents[n].y, r ? s : i, r ? i : s, this._projectionMatrices[n], e.getEngine().isNDCHalfZRange), this._cascadeMinExtents[n].z = i, this._cascadeMaxExtents[n].z = s, this._viewMatrices[n].multiplyToRef(this._projectionMatrices[n], this._transformMatrices[n]), S.TransformCoordinatesToRef(ppe, this._transformMatrices[n], R9), R9.scaleInPlace(this._mapSize / 2), IW.copyFromFloats(Math.round(R9.x), Math.round(R9.y), Math.round(R9.z)), IW.subtractInPlace(R9).scaleInPlace(2 / this._mapSize), he.TranslationToRef(IW.x, IW.y, 0, HC), this._projectionMatrices[n].multiplyToRef(HC, this._projectionMatrices[n]), this._viewMatrices[n].multiplyToRef(this._projectionMatrices[n], this._transformMatrices[n]), this._transformMatrices[n].copyToArray(this._transformMatricesAsArray, n * 16); } } // Get the 8 points of the view frustum in world space _computeFrustumInWorldSpace(e) { const t = this._getCamera(); if (!t) return; const r = this._cascades[e].prevBreakDistance, n = this._cascades[e].breakDistance, i = this._scene.getEngine().isNDCHalfZRange; t.getViewMatrix(); const s = t.maxZ === 0, a = t.maxZ; s && (t.maxZ = this._shadowMaxZ, t.getProjectionMatrix(!0)); const f = he.Invert(t.getTransformationMatrix()); s && (t.maxZ = a, t.getProjectionMatrix(!0)); const o = this._scene.getEngine().useReverseDepthBuffer ? 4 : 0; for (let d = 0; d < X6._FrustumCornersNDCSpace.length; ++d) R9.copyFrom(X6._FrustumCornersNDCSpace[(d + o) % X6._FrustumCornersNDCSpace.length]), i && R9.z === -1 && (R9.z = 0), S.TransformCoordinatesToRef(R9, f, this._frustumCornersWorldSpace[e][d]); for (let d = 0; d < X6._FrustumCornersNDCSpace.length / 2; ++d) R9.copyFrom(this._frustumCornersWorldSpace[e][d + 4]).subtractInPlace(this._frustumCornersWorldSpace[e][d]), IW.copyFrom(R9).scaleInPlace(r), R9.scaleInPlace(n), R9.addInPlace(this._frustumCornersWorldSpace[e][d]), this._frustumCornersWorldSpace[e][d + 4].copyFrom(R9), this._frustumCornersWorldSpace[e][d].addInPlace(IW); } _computeCascadeFrustum(e) { if (this._cascadeMinExtents[e].copyFromFloats(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE), this._cascadeMaxExtents[e].copyFromFloats(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE), this._frustumCenter[e].copyFromFloats(0, 0, 0), !!this._getCamera()) { for (let r = 0; r < this._frustumCornersWorldSpace[e].length; ++r) this._frustumCenter[e].addInPlace(this._frustumCornersWorldSpace[e][r]); if (this._frustumCenter[e].scaleInPlace(1 / this._frustumCornersWorldSpace[e].length), this.stabilizeCascades) { let r = 0; for (let n = 0; n < this._frustumCornersWorldSpace[e].length; ++n) { const i = this._frustumCornersWorldSpace[e][n].subtractToRef(this._frustumCenter[e], R9).length(); r = Math.max(r, i); } r = Math.ceil(r * 16) / 16, this._cascadeMaxExtents[e].copyFromFloats(r, r, r), this._cascadeMinExtents[e].copyFromFloats(-r, -r, -r); } else { const r = this._frustumCenter[e]; this._frustumCenter[e].addToRef(this._lightDirection, R9), he.LookAtLHToRef(r, R9, MG, HC); for (let n = 0; n < this._frustumCornersWorldSpace[e].length; ++n) S.TransformCoordinatesToRef(this._frustumCornersWorldSpace[e][n], HC, R9), this._cascadeMinExtents[e].minimizeInPlace(R9), this._cascadeMaxExtents[e].maximizeInPlace(R9); } } } _recreateSceneUBOs() { if (this._disposeSceneUBOs(), this._sceneUBOs) for (let e = 0; e < this._numCascades; ++e) this._sceneUBOs.push(this._scene.createSceneUniformBuffer(`Scene for CSM Shadow Generator (light "${this._light.name}" cascade #${e})`)); } /** * Support test. */ static get IsSupported() { const e = gr.LastCreatedEngine; return e ? e._features.supportCSM : !1; } /** * Creates a Cascaded Shadow Generator object. * A ShadowGenerator is the required tool to use the shadows. * Each directional light casting shadows needs to use its own ShadowGenerator. * Documentation : https://doc.babylonjs.com/babylon101/cascadedShadows * @param mapSize The size of the texture what stores the shadows. Example : 1024. * @param light The directional light object generating the shadows. * @param usefulFloatFirst By default the generator will try to use half float textures but if you need precision (for self shadowing for instance), you can use this option to enforce full float texture. * @param camera Camera associated with this shadow generator (default: null). If null, takes the scene active camera at the time we need to access it * @param useRedTextureType Forces the generator to use a Red instead of a RGBA type for the shadow map texture format (default: true) */ constructor(e, t, r, n, i = !0) { if (!X6.IsSupported) { Se.Error("CascadedShadowMap is not supported by the current engine."); return; } super(e, t, r, n, i), this.usePercentageCloserFiltering = !0; } _initializeGenerator() { var e, t, r, n, i, s, a, f, o, d, v, u, l, P, p, c, H, T, q, b; this.penumbraDarkness = (e = this.penumbraDarkness) !== null && e !== void 0 ? e : 1, this._numCascades = (t = this._numCascades) !== null && t !== void 0 ? t : X6.DEFAULT_CASCADES_COUNT, this.stabilizeCascades = (r = this.stabilizeCascades) !== null && r !== void 0 ? r : !1, this._freezeShadowCastersBoundingInfoObservable = (n = this._freezeShadowCastersBoundingInfoObservable) !== null && n !== void 0 ? n : null, this.freezeShadowCastersBoundingInfo = (i = this.freezeShadowCastersBoundingInfo) !== null && i !== void 0 ? i : !1, this._scbiMin = (s = this._scbiMin) !== null && s !== void 0 ? s : new S(0, 0, 0), this._scbiMax = (a = this._scbiMax) !== null && a !== void 0 ? a : new S(0, 0, 0), this._shadowCastersBoundingInfo = (f = this._shadowCastersBoundingInfo) !== null && f !== void 0 ? f : new Md(new S(0, 0, 0), new S(0, 0, 0)), this._breaksAreDirty = (o = this._breaksAreDirty) !== null && o !== void 0 ? o : !0, this._minDistance = (d = this._minDistance) !== null && d !== void 0 ? d : 0, this._maxDistance = (v = this._maxDistance) !== null && v !== void 0 ? v : 1, this._currentLayer = (u = this._currentLayer) !== null && u !== void 0 ? u : 0, this._shadowMaxZ = (p = (l = this._shadowMaxZ) !== null && l !== void 0 ? l : (P = this._getCamera()) === null || P === void 0 ? void 0 : P.maxZ) !== null && p !== void 0 ? p : 1e4, this._debug = (c = this._debug) !== null && c !== void 0 ? c : !1, this._depthClamp = (H = this._depthClamp) !== null && H !== void 0 ? H : !0, this._cascadeBlendPercentage = (T = this._cascadeBlendPercentage) !== null && T !== void 0 ? T : 0.1, this._lambda = (q = this._lambda) !== null && q !== void 0 ? q : 0.5, this._autoCalcDepthBounds = (b = this._autoCalcDepthBounds) !== null && b !== void 0 ? b : !1, this._recreateSceneUBOs(), super._initializeGenerator(); } _createTargetRenderTexture() { const e = this._scene.getEngine(), t = { width: this._mapSize, height: this._mapSize, layers: this.numCascades }; this._shadowMap = new Ta(this._light.name + "_CSMShadowMap", t, this._scene, !1, !0, this._textureType, !1, void 0, !1, !1, void 0, this._useRedTextureType ? 6 : 5), this._shadowMap.createDepthStencilTexture(e.useReverseDepthBuffer ? 516 : 513, !0), this._shadowMap.noPrePassRenderer = !0; } _initializeShadowMap() { if (super._initializeShadowMap(), this._shadowMap === null) return; this._transformMatricesAsArray = new Float32Array(this._numCascades * 16), this._viewSpaceFrustumsZ = new Array(this._numCascades), this._frustumLengths = new Array(this._numCascades), this._lightSizeUVCorrection = new Array(this._numCascades * 2), this._depthCorrection = new Array(this._numCascades), this._cascades = [], this._viewMatrices = [], this._projectionMatrices = [], this._transformMatrices = [], this._cascadeMinExtents = [], this._cascadeMaxExtents = [], this._frustumCenter = [], this._shadowCameraPos = [], this._frustumCornersWorldSpace = []; for (let t = 0; t < this._numCascades; ++t) { this._cascades[t] = { prevBreakDistance: 0, breakDistance: 0 }, this._viewMatrices[t] = he.Zero(), this._projectionMatrices[t] = he.Zero(), this._transformMatrices[t] = he.Zero(), this._cascadeMinExtents[t] = new S(), this._cascadeMaxExtents[t] = new S(), this._frustumCenter[t] = new S(), this._shadowCameraPos[t] = new S(), this._frustumCornersWorldSpace[t] = new Array(X6._FrustumCornersNDCSpace.length); for (let r = 0; r < X6._FrustumCornersNDCSpace.length; ++r) this._frustumCornersWorldSpace[t][r] = new S(); } const e = this._scene.getEngine(); this._shadowMap.onBeforeBindObservable.clear(), this._shadowMap.onBeforeRenderObservable.clear(), this._shadowMap.onBeforeRenderObservable.add((t) => { this._sceneUBOs && this._scene.setSceneUniformBuffer(this._sceneUBOs[t]), this._currentLayer = t, this._filter === ln.FILTER_PCF && e.setColorWrite(!1), this._scene.setTransformMatrix(this.getCascadeViewMatrix(t), this.getCascadeProjectionMatrix(t)), this._useUBO && (this._scene.getSceneUniformBuffer().unbindEffect(), this._scene.finalizeSceneUbo()); }), this._shadowMap.onBeforeBindObservable.add(() => { var t; this._currentSceneUBO = this._scene.getSceneUniformBuffer(), (t = e._debugPushGroup) === null || t === void 0 || t.call(e, `cascaded shadow map generation for pass id ${e.currentRenderPassId}`, 1), this._breaksAreDirty && this._splitFrustum(), this._computeMatrices(); }), this._splitFrustum(); } _bindCustomEffectForRenderSubMeshForShadowMap(e, t) { t.setMatrix("viewProjection", this.getCascadeTransformMatrix(this._currentLayer)); } _isReadyCustomDefines(e) { e.push("#define SM_DEPTHCLAMP " + (this._depthClamp && this._filter !== ln.FILTER_PCSS ? "1" : "0")); } /** * Prepare all the defines in a material relying on a shadow map at the specified light index. * @param defines Defines of the material we want to update * @param lightIndex Index of the light in the enabled light list of the material */ prepareDefines(e, t) { super.prepareDefines(e, t); const r = this._scene, n = this._light; if (!r.shadowsEnabled || !n.shadowEnabled) return; e["SHADOWCSM" + t] = !0, e["SHADOWCSMDEBUG" + t] = this.debug, e["SHADOWCSMNUM_CASCADES" + t] = this.numCascades, e["SHADOWCSM_RIGHTHANDED" + t] = r.useRightHandedSystem; const i = this._getCamera(); i && this._shadowMaxZ <= (i.maxZ || this._shadowMaxZ) && (e["SHADOWCSMUSESHADOWMAXZ" + t] = !0), this.cascadeBlendPercentage === 0 && (e["SHADOWCSMNOBLEND" + t] = !0); } /** * Binds the shadow related information inside of an effect (information like near, far, darkness... * defined in the generator but impacting the effect). * @param lightIndex Index of the light in the enabled light list of the material owning the effect * @param effect The effect we are binfing the information for */ bindShadowLight(e, t) { const r = this._light; if (!this._scene.shadowsEnabled || !r.shadowEnabled) return; const i = this._getCamera(); if (!i) return; const s = this.getShadowMap(); if (!s) return; const a = s.getSize().width; if (t.setMatrices("lightMatrix" + e, this._transformMatricesAsArray), t.setArray("viewFrustumZ" + e, this._viewSpaceFrustumsZ), t.setFloat("cascadeBlendFactor" + e, this.cascadeBlendPercentage === 0 ? 1e4 : 1 / this.cascadeBlendPercentage), t.setArray("frustumLengths" + e, this._frustumLengths), this._filter === ln.FILTER_PCF) t.setDepthStencilTexture("shadowSampler" + e, s), r._uniformBuffer.updateFloat4("shadowsInfo", this.getDarkness(), a, 1 / a, this.frustumEdgeFalloff, e); else if (this._filter === ln.FILTER_PCSS) { for (let f = 0; f < this._numCascades; ++f) this._lightSizeUVCorrection[f * 2 + 0] = f === 0 ? 1 : (this._cascadeMaxExtents[0].x - this._cascadeMinExtents[0].x) / (this._cascadeMaxExtents[f].x - this._cascadeMinExtents[f].x), this._lightSizeUVCorrection[f * 2 + 1] = f === 0 ? 1 : (this._cascadeMaxExtents[0].y - this._cascadeMinExtents[0].y) / (this._cascadeMaxExtents[f].y - this._cascadeMinExtents[f].y), this._depthCorrection[f] = f === 0 ? 1 : (this._cascadeMaxExtents[f].z - this._cascadeMinExtents[f].z) / (this._cascadeMaxExtents[0].z - this._cascadeMinExtents[0].z); t.setDepthStencilTexture("shadowSampler" + e, s), t.setTexture("depthSampler" + e, s), t.setArray2("lightSizeUVCorrection" + e, this._lightSizeUVCorrection), t.setArray("depthCorrection" + e, this._depthCorrection), t.setFloat("penumbraDarkness" + e, this.penumbraDarkness), r._uniformBuffer.updateFloat4("shadowsInfo", this.getDarkness(), 1 / a, this._contactHardeningLightSizeUVRatio * a, this.frustumEdgeFalloff, e); } else t.setTexture("shadowSampler" + e, s), r._uniformBuffer.updateFloat4("shadowsInfo", this.getDarkness(), a, 1 / a, this.frustumEdgeFalloff, e); r._uniformBuffer.updateFloat2("depthValues", this.getLight().getDepthMinZ(i), this.getLight().getDepthMinZ(i) + this.getLight().getDepthMaxZ(i), e); } /** * Gets the transformation matrix of the first cascade used to project the meshes into the map from the light point of view. * (eq to view projection * shadow projection matrices) * @returns The transform matrix used to create the shadow map */ getTransformMatrix() { return this.getCascadeTransformMatrix(0); } /** * Disposes the ShadowGenerator. * Returns nothing. */ dispose() { super.dispose(), this._freezeShadowCastersBoundingInfoObservable && (this._scene.onBeforeRenderObservable.remove(this._freezeShadowCastersBoundingInfoObservable), this._freezeShadowCastersBoundingInfoObservable = null), this._depthReducer && (this._depthReducer.dispose(), this._depthReducer = null); } /** * Serializes the shadow generator setup to a json object. * @returns The serialized JSON object */ serialize() { const e = super.serialize(), t = this.getShadowMap(); if (!t) return e; if (e.numCascades = this._numCascades, e.debug = this._debug, e.stabilizeCascades = this.stabilizeCascades, e.lambda = this._lambda, e.cascadeBlendPercentage = this.cascadeBlendPercentage, e.depthClamp = this._depthClamp, e.autoCalcDepthBounds = this.autoCalcDepthBounds, e.shadowMaxZ = this._shadowMaxZ, e.penumbraDarkness = this.penumbraDarkness, e.freezeShadowCastersBoundingInfo = this._freezeShadowCastersBoundingInfo, e.minDistance = this.minDistance, e.maxDistance = this.maxDistance, e.renderList = [], t.renderList) for (let r = 0; r < t.renderList.length; r++) { const n = t.renderList[r]; e.renderList.push(n.id); } return e; } /** * Parses a serialized ShadowGenerator and returns a new ShadowGenerator. * @param parsedShadowGenerator The JSON object to parse * @param scene The scene to create the shadow map for * @returns The parsed shadow generator */ static Parse(e, t) { const r = ln.Parse(e, t, (n, i, s) => new X6(n, i, void 0, s)); return e.numCascades !== void 0 && (r.numCascades = e.numCascades), e.debug !== void 0 && (r.debug = e.debug), e.stabilizeCascades !== void 0 && (r.stabilizeCascades = e.stabilizeCascades), e.lambda !== void 0 && (r.lambda = e.lambda), e.cascadeBlendPercentage !== void 0 && (r.cascadeBlendPercentage = e.cascadeBlendPercentage), e.depthClamp !== void 0 && (r.depthClamp = e.depthClamp), e.autoCalcDepthBounds !== void 0 && (r.autoCalcDepthBounds = e.autoCalcDepthBounds), e.shadowMaxZ !== void 0 && (r.shadowMaxZ = e.shadowMaxZ), e.penumbraDarkness !== void 0 && (r.penumbraDarkness = e.penumbraDarkness), e.freezeShadowCastersBoundingInfo !== void 0 && (r.freezeShadowCastersBoundingInfo = e.freezeShadowCastersBoundingInfo), e.minDistance !== void 0 && e.maxDistance !== void 0 && r.setMinMaxDistance(e.minDistance, e.maxDistance), r; } } X6._FrustumCornersNDCSpace = [ new S(-1, 1, -1), new S(1, 1, -1), new S(1, -1, -1), new S(-1, -1, -1), new S(-1, 1, 1), new S(1, 1, 1), new S(1, -1, 1), new S(-1, -1, 1) ]; X6.CLASSNAME = "CascadedShadowGenerator"; X6.DEFAULT_CASCADES_COUNT = 4; X6.MIN_CASCADES_COUNT = 2; X6.MAX_CASCADES_COUNT = 4; X6._SceneComponentInitialization = (A) => { throw qn("ShadowGeneratorSceneComponent"); }; J1.AddParser(Ot.NAME_SHADOWGENERATOR, (A, e) => { if (A.shadowGenerators !== void 0 && A.shadowGenerators !== null) for (let t = 0, r = A.shadowGenerators.length; t < r; t++) { const n = A.shadowGenerators[t]; n.className === X6.CLASSNAME ? X6.Parse(n, e) : ln.Parse(n, e); } }); class Jte { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_SHADOWGENERATOR, this.scene = e; } /** * Registers the component in a given scene */ register() { this.scene._gatherRenderTargetsStage.registerStep(Ot.STEP_GATHERRENDERTARGETS_SHADOWGENERATOR, this, this._gatherRenderTargets); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { } /** * Serializes the component data to the specified json object * @param serializationObject The object to serialize to */ serialize(e) { e.shadowGenerators = []; const t = this.scene.lights; for (const r of t) { const n = r.getShadowGenerators(); if (n) { const i = n.values(); for (let s = i.next(); s.done !== !0; s = i.next()) { const a = s.value; e.shadowGenerators.push(a.serialize()); } } } } /** * Adds all the elements from the container to the scene * @param container the container holding the elements */ // eslint-disable-next-line @typescript-eslint/no-unused-vars addFromContainer(e) { } /** * Removes all the elements in the container from the scene * @param container contains the elements to remove * @param dispose if the removed element should be disposed (default: false) */ // eslint-disable-next-line @typescript-eslint/no-unused-vars removeFromContainer(e, t) { } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ dispose() { } _gatherRenderTargets(e) { const t = this.scene; if (this.scene.shadowsEnabled) for (let r = 0; r < t.lights.length; r++) { const n = t.lights[r], i = n.getShadowGenerators(); if (n.isEnabled() && n.shadowEnabled && i) { const s = i.values(); for (let a = s.next(); a.done !== !0; a = s.next()) { const o = a.value.getShadowMap(); t.textures.indexOf(o) !== -1 && e.push(o); } } } } } ln._SceneComponentInitialization = (A) => { let e = A._getComponent(Ot.NAME_SHADOWGENERATOR); e || (e = new Jte(A), A._addComponent(e)); }; Cs.AddNodeConstructor("Light_Type_0", (A, e) => () => new ag(A, S.Zero(), e)); class ag extends xD { /** * Getter: In case of direction provided, the shadow will not use a cube texture but simulate a spot shadow as a fallback * This specifies what angle the shadow will use to be created. * * It default to 90 degrees to work nicely with the cube texture generation for point lights shadow maps. */ get shadowAngle() { return this._shadowAngle; } /** * Setter: In case of direction provided, the shadow will not use a cube texture but simulate a spot shadow as a fallback * This specifies what angle the shadow will use to be created. * * It default to 90 degrees to work nicely with the cube texture generation for point lights shadow maps. */ set shadowAngle(e) { this._shadowAngle = e, this.forceProjectionMatrixCompute(); } /** * Gets the direction if it has been set. * In case of direction provided, the shadow will not use a cube texture but simulate a spot shadow as a fallback */ get direction() { return this._direction; } /** * In case of direction provided, the shadow will not use a cube texture but simulate a spot shadow as a fallback */ set direction(e) { const t = this.needCube(); if (this._direction = e, this.needCube() !== t && this._shadowGenerators) { const r = this._shadowGenerators.values(); for (let n = r.next(); n.done !== !0; n = r.next()) n.value.recreateShadowMap(); } } /** * Creates a PointLight object from the passed name and position (Vector3) and adds it in the scene. * A PointLight emits the light in every direction. * It can cast shadows. * If the scene camera is already defined and you want to set your PointLight at the camera position, just set it : * ```javascript * var pointLight = new PointLight("pl", camera.position, scene); * ``` * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction * @param name The light friendly name * @param position The position of the point light in the scene * @param scene The scene the lights belongs to */ constructor(e, t, r) { super(e, r), this._shadowAngle = Math.PI / 2, this.position = t; } /** * Returns the string "PointLight" * @returns the class name */ getClassName() { return "PointLight"; } /** * Returns the integer 0. * @returns The light Type id as a constant defines in Light.LIGHTTYPEID_x */ getTypeID() { return ci.LIGHTTYPEID_POINTLIGHT; } /** * Specifies whether or not the shadowmap should be a cube texture. * @returns true if the shadowmap needs to be a cube texture. */ needCube() { return !this.direction; } /** * Returns a new Vector3 aligned with the PointLight cube system according to the passed cube face index (integer). * @param faceIndex The index of the face we are computed the direction to generate shadow * @returns The set direction in 2d mode otherwise the direction to the cubemap face if needCube() is true */ getShadowDirection(e) { if (this.direction) return super.getShadowDirection(e); switch (e) { case 0: return new S(1, 0, 0); case 1: return new S(-1, 0, 0); case 2: return new S(0, -1, 0); case 3: return new S(0, 1, 0); case 4: return new S(0, 0, 1); case 5: return new S(0, 0, -1); } return S.Zero(); } /** * Sets the passed matrix "matrix" as a left-handed perspective projection matrix with the following settings : * - fov = PI / 2 * - aspect ratio : 1.0 * - z-near and far equal to the active camera minZ and maxZ. * Returns the PointLight. * @param matrix * @param viewMatrix * @param renderList */ // eslint-disable-next-line @typescript-eslint/no-unused-vars _setDefaultShadowProjectionMatrix(e, t, r) { const n = this.getScene().activeCamera; if (!n) return; const i = this.shadowMinZ !== void 0 ? this.shadowMinZ : n.minZ, s = this.shadowMaxZ !== void 0 ? this.shadowMaxZ : n.maxZ, a = this.getScene().getEngine().useReverseDepthBuffer; he.PerspectiveFovLHToRef(this.shadowAngle, 1, a ? s : i, a ? i : s, e, !0, this._scene.getEngine().isNDCHalfZRange, void 0, a); } _buildUniformLayout() { this._uniformBuffer.addUniform("vLightData", 4), this._uniformBuffer.addUniform("vLightDiffuse", 4), this._uniformBuffer.addUniform("vLightSpecular", 4), this._uniformBuffer.addUniform("vLightFalloff", 4), this._uniformBuffer.addUniform("shadowsInfo", 3), this._uniformBuffer.addUniform("depthValues", 2), this._uniformBuffer.create(); } /** * Sets the passed Effect "effect" with the PointLight transformed position (or position, if none) and passed name (string). * @param effect The effect to update * @param lightIndex The index of the light in the effect to update * @returns The point light */ transferToEffect(e, t) { return this.computeTransformedInformation() ? this._uniformBuffer.updateFloat4("vLightData", this.transformedPosition.x, this.transformedPosition.y, this.transformedPosition.z, 0, t) : this._uniformBuffer.updateFloat4("vLightData", this.position.x, this.position.y, this.position.z, 0, t), this._uniformBuffer.updateFloat4("vLightFalloff", this.range, this._inverseSquaredRange, 0, 0, t), this; } transferToNodeMaterialEffect(e, t) { return this.computeTransformedInformation() ? e.setFloat3(t, this.transformedPosition.x, this.transformedPosition.y, this.transformedPosition.z) : e.setFloat3(t, this.position.x, this.position.y, this.position.z), this; } /** * Prepares the list of defines specific to the light type. * @param defines the list of defines * @param lightIndex defines the index of the light for the effect */ prepareLightSpecificDefines(e, t) { e["POINTLIGHT" + t] = !0; } } C([ M() ], ag.prototype, "shadowAngle", null); class oq { /** * Creates a new default loading screen * @param _renderingCanvas defines the canvas used to render the scene * @param _loadingText defines the default text to display * @param _loadingDivBackgroundColor defines the default background color */ constructor(e, t = "", r = "black") { this._renderingCanvas = e, this._loadingText = t, this._loadingDivBackgroundColor = r, this._resizeLoadingUI = () => { const n = this._renderingCanvas.getBoundingClientRect(), i = window.getComputedStyle(this._renderingCanvas).position; this._loadingDiv && (this._loadingDiv.style.position = i === "fixed" ? "fixed" : "absolute", this._loadingDiv.style.left = n.left + "px", this._loadingDiv.style.top = n.top + "px", this._loadingDiv.style.width = n.width + "px", this._loadingDiv.style.height = n.height + "px"); }; } /** * Function called to display the loading screen */ displayLoadingUI() { if (this._loadingDiv) return; this._loadingDiv = document.createElement("div"), this._loadingDiv.id = "babylonjsLoadingDiv", this._loadingDiv.style.opacity = "0", this._loadingDiv.style.transition = "opacity 1.5s ease", this._loadingDiv.style.pointerEvents = "none", this._loadingDiv.style.display = "grid", this._loadingDiv.style.gridTemplateRows = "100%", this._loadingDiv.style.gridTemplateColumns = "100%", this._loadingDiv.style.justifyItems = "center", this._loadingDiv.style.alignItems = "center", this._loadingTextDiv = document.createElement("div"), this._loadingTextDiv.style.position = "absolute", this._loadingTextDiv.style.left = "0", this._loadingTextDiv.style.top = "50%", this._loadingTextDiv.style.marginTop = "80px", this._loadingTextDiv.style.width = "100%", this._loadingTextDiv.style.height = "20px", this._loadingTextDiv.style.fontFamily = "Arial", this._loadingTextDiv.style.fontSize = "14px", this._loadingTextDiv.style.color = "white", this._loadingTextDiv.style.textAlign = "center", this._loadingTextDiv.style.zIndex = "1", this._loadingTextDiv.innerHTML = "Loading", this._loadingDiv.appendChild(this._loadingTextDiv), this._loadingTextDiv.innerHTML = this._loadingText, this._style = document.createElement("style"), this._style.type = "text/css"; const e = `@-webkit-keyframes spin1 { 0% { -webkit-transform: rotate(0deg);} 100% { -webkit-transform: rotate(360deg);} } @keyframes spin1 { 0% { transform: rotate(0deg);} 100% { transform: rotate(360deg);} }`; this._style.innerHTML = e, document.getElementsByTagName("head")[0].appendChild(this._style); const t = !!window.SVGSVGElement, r = new Image(); oq.DefaultLogoUrl ? r.src = oq.DefaultLogoUrl : r.src = t ? "" : "https://cdn.babylonjs.com/Assets/babylonLogo.png", r.style.width = "150px", r.style.gridColumn = "1", r.style.gridRow = "1", r.style.top = "50%", r.style.left = "50%", r.style.transform = "translate(-50%, -50%)", r.style.position = "absolute"; const n = document.createElement("div"); n.style.width = "300px", n.style.gridColumn = "1", n.style.gridRow = "1", n.style.top = "50%", n.style.left = "50%", n.style.transform = "translate(-50%, -50%)", n.style.position = "absolute"; const i = new Image(); if (oq.DefaultSpinnerUrl ? i.src = oq.DefaultSpinnerUrl : i.src = t ? "" : "https://cdn.babylonjs.com/Assets/loadingIcon.png", i.style.animation = "spin1 0.75s infinite linear", i.style.webkitAnimation = "spin1 0.75s infinite linear", i.style.transformOrigin = "50% 50%", i.style.webkitTransformOrigin = "50% 50%", !t) { const s = { w: 16, h: 18.5 }, a = { w: 30, h: 30 }; r.style.width = `${s.w}vh`, r.style.height = `${s.h}vh`, r.style.left = `calc(50% - ${s.w / 2}vh)`, r.style.top = `calc(50% - ${s.h / 2}vh)`, i.style.width = `${a.w}vh`, i.style.height = `${a.h}vh`, i.style.left = `calc(50% - ${a.w / 2}vh)`, i.style.top = `calc(50% - ${a.h / 2}vh)`; } n.appendChild(i), this._loadingDiv.appendChild(r), this._loadingDiv.appendChild(n), this._resizeLoadingUI(), window.addEventListener("resize", this._resizeLoadingUI), this._loadingDiv.style.backgroundColor = this._loadingDivBackgroundColor, document.body.appendChild(this._loadingDiv), this._loadingDiv.style.opacity = "1"; } /** * Function called to hide the loading screen */ hideLoadingUI() { if (!this._loadingDiv) return; const e = () => { this._loadingTextDiv && (this._loadingTextDiv.remove(), this._loadingTextDiv = null), this._loadingDiv && (this._loadingDiv.remove(), this._loadingDiv = null), this._style && (this._style.remove(), this._style = null), window.removeEventListener("resize", this._resizeLoadingUI); }; this._loadingDiv.style.opacity = "0", this._loadingDiv.addEventListener("transitionend", e); } /** * Gets or sets the text to display while loading */ set loadingUIText(e) { this._loadingText = e, this._loadingTextDiv && (this._loadingTextDiv.innerHTML = this._loadingText); } get loadingUIText() { return this._loadingText; } /** * Gets or sets the color to use for the background */ get loadingUIBackgroundColor() { return this._loadingDivBackgroundColor; } set loadingUIBackgroundColor(e) { this._loadingDivBackgroundColor = e, this._loadingDiv && (this._loadingDiv.style.backgroundColor = this._loadingDivBackgroundColor); } } oq.DefaultLogoUrl = ""; oq.DefaultSpinnerUrl = ""; Ge.DefaultLoadingScreenFactory = (A) => new oq(A); class wq { /** * Converts a panorama stored in RGB right to left up to down format into a cubemap (6 faces). * * @param float32Array The source data. * @param inputWidth The width of the input panorama. * @param inputHeight The height of the input panorama. * @param size The willing size of the generated cubemap (each faces will be size * size pixels) * @returns The cubemap data */ static ConvertPanoramaToCubemap(e, t, r, n, i = !1) { if (!e) throw "ConvertPanoramaToCubemap: input cannot be null"; if (e.length != t * r * 3) throw "ConvertPanoramaToCubemap: input size is wrong"; const s = this.CreateCubemapTexture(n, this.FACE_FRONT, e, t, r, i), a = this.CreateCubemapTexture(n, this.FACE_BACK, e, t, r, i), f = this.CreateCubemapTexture(n, this.FACE_LEFT, e, t, r, i), o = this.CreateCubemapTexture(n, this.FACE_RIGHT, e, t, r, i), d = this.CreateCubemapTexture(n, this.FACE_UP, e, t, r, i), v = this.CreateCubemapTexture(n, this.FACE_DOWN, e, t, r, i); return { front: s, back: a, left: f, right: o, up: d, down: v, size: n, type: 1, format: 4, gammaSpace: !1 }; } static CreateCubemapTexture(e, t, r, n, i, s = !1) { const a = new ArrayBuffer(e * e * 4 * 3), f = new Float32Array(a), o = s ? Math.max(1, Math.round(n / 4 / e)) : 1, d = 1 / o, v = d * d, u = t[1].subtract(t[0]).scale(d / e), l = t[3].subtract(t[2]).scale(d / e), P = 1 / e; let p = 0; for (let c = 0; c < e; c++) for (let H = 0; H < o; H++) { let T = t[0], q = t[2]; for (let b = 0; b < e; b++) for (let j = 0; j < o; j++) { const w = q.subtract(T).scale(p).add(T); w.normalize(); const m = this.CalcProjectionSpherical(w, r, n, i); f[c * e * 3 + b * 3 + 0] += m.r * v, f[c * e * 3 + b * 3 + 1] += m.g * v, f[c * e * 3 + b * 3 + 2] += m.b * v, T = T.add(u), q = q.add(l); } p += P * d; } return f; } static CalcProjectionSpherical(e, t, r, n) { let i = Math.atan2(e.z, e.x); const s = Math.acos(e.y); for (; i < -Math.PI; ) i += 2 * Math.PI; for (; i > Math.PI; ) i -= 2 * Math.PI; let a = i / Math.PI; const f = s / Math.PI; a = a * 0.5 + 0.5; let o = Math.round(a * r); o < 0 ? o = 0 : o >= r && (o = r - 1); let d = Math.round(f * n); d < 0 ? d = 0 : d >= n && (d = n - 1); const v = n - d - 1, u = t[v * r * 3 + o * 3 + 0], l = t[v * r * 3 + o * 3 + 1], P = t[v * r * 3 + o * 3 + 2]; return { r: u, g: l, b: P }; } } wq.FACE_LEFT = [new S(-1, -1, -1), new S(1, -1, -1), new S(-1, 1, -1), new S(1, 1, -1)]; wq.FACE_RIGHT = [new S(1, -1, 1), new S(-1, -1, 1), new S(1, 1, 1), new S(-1, 1, 1)]; wq.FACE_FRONT = [new S(1, -1, -1), new S(1, -1, 1), new S(1, 1, -1), new S(1, 1, 1)]; wq.FACE_BACK = [new S(-1, -1, 1), new S(-1, -1, -1), new S(-1, 1, 1), new S(-1, 1, -1)]; wq.FACE_DOWN = [new S(1, 1, -1), new S(1, 1, 1), new S(-1, 1, -1), new S(-1, 1, 1)]; wq.FACE_UP = [new S(-1, -1, -1), new S(-1, -1, 1), new S(1, -1, -1), new S(1, -1, 1)]; class iO { static _Ldexp(e, t) { return t > 1023 ? e * Math.pow(2, 1023) * Math.pow(2, t - 1023) : t < -1074 ? e * Math.pow(2, -1074) * Math.pow(2, t + 1074) : e * Math.pow(2, t); } static _Rgbe2float(e, t, r, n, i, s) { i > 0 ? (i = this._Ldexp(1, i - 136), e[s + 0] = t * i, e[s + 1] = r * i, e[s + 2] = n * i) : (e[s + 0] = 0, e[s + 1] = 0, e[s + 2] = 0); } static _ReadStringLine(e, t) { let r = "", n = ""; for (let i = t; i < e.length - t && (n = String.fromCharCode(e[i]), n != ` `); i++) r += n; return r; } /** * Reads header information from an RGBE texture stored in a native array. * More information on this format are available here: * https://en.wikipedia.org/wiki/RGBE_image_format * * @param uint8array The binary file stored in native array. * @returns The header information. */ // eslint-disable-next-line @typescript-eslint/naming-convention static RGBE_ReadHeader(e) { let t = 0, r = 0, n = this._ReadStringLine(e, 0); if (n[0] != "#" || n[1] != "?") throw "Bad HDR Format."; let i = !1, s = !1, a = 0; do a += n.length + 1, n = this._ReadStringLine(e, a), n == "FORMAT=32-bit_rle_rgbe" ? s = !0 : n.length == 0 && (i = !0); while (!i); if (!s) throw "HDR Bad header format, unsupported FORMAT"; a += n.length + 1, n = this._ReadStringLine(e, a); const o = /^-Y (.*) \+X (.*)$/g.exec(n); if (!o || o.length < 3) throw "HDR Bad header format, no size"; if (r = parseInt(o[2]), t = parseInt(o[1]), r < 8 || r > 32767) throw "HDR Bad header format, unsupported size"; return a += n.length + 1, { height: t, width: r, dataPosition: a }; } /** * Returns the cubemap information (each faces texture data) extracted from an RGBE texture. * This RGBE texture needs to store the information as a panorama. * * More information on this format are available here: * https://en.wikipedia.org/wiki/RGBE_image_format * * @param buffer The binary file stored in an array buffer. * @param size The expected size of the extracted cubemap. * @returns The Cube Map information. */ static GetCubeMapTextureData(e, t, r = !1) { const n = new Uint8Array(e), i = this.RGBE_ReadHeader(n), s = this.RGBE_ReadPixels(n, i); return wq.ConvertPanoramaToCubemap(s, i.width, i.height, t, r); } /** * Returns the pixels data extracted from an RGBE texture. * This pixels will be stored left to right up to down in the R G B order in one array. * * More information on this format are available here: * https://en.wikipedia.org/wiki/RGBE_image_format * * @param uint8array The binary file stored in an array buffer. * @param hdrInfo The header information of the file. * @returns The pixels data in RGB right to left up to down order. */ // eslint-disable-next-line @typescript-eslint/naming-convention static RGBE_ReadPixels(e, t) { return this._RGBEReadPixelsRLE(e, t); } static _RGBEReadPixelsRLE(e, t) { let r = t.height; const n = t.width; let i, s, a, f, o, d = t.dataPosition, v = 0, u = 0, l = 0; const P = new ArrayBuffer(n * 4), p = new Uint8Array(P), c = new ArrayBuffer(t.width * t.height * 4 * 3), H = new Float32Array(c); for (; r > 0; ) { if (i = e[d++], s = e[d++], a = e[d++], f = e[d++], i != 2 || s != 2 || a & 128 || t.width < 8 || t.width > 32767) return this._RGBEReadPixelsNOTRLE(e, t); if ((a << 8 | f) != n) throw "HDR Bad header format, wrong scan line width"; for (v = 0, l = 0; l < 4; l++) for (u = (l + 1) * n; v < u; ) if (i = e[d++], s = e[d++], i > 128) { if (o = i - 128, o == 0 || o > u - v) throw "HDR Bad Format, bad scanline data (run)"; for (; o-- > 0; ) p[v++] = s; } else { if (o = i, o == 0 || o > u - v) throw "HDR Bad Format, bad scanline data (non-run)"; if (p[v++] = s, --o > 0) for (let T = 0; T < o; T++) p[v++] = e[d++]; } for (l = 0; l < n; l++) i = p[l], s = p[l + n], a = p[l + 2 * n], f = p[l + 3 * n], this._Rgbe2float(H, i, s, a, f, (t.height - r) * n * 3 + l * 3); r--; } return H; } static _RGBEReadPixelsNOTRLE(e, t) { let r = t.height; const n = t.width; let i, s, a, f, o, d = t.dataPosition; const v = new ArrayBuffer(t.width * t.height * 4 * 3), u = new Float32Array(v); for (; r > 0; ) { for (o = 0; o < t.width; o++) i = e[d++], s = e[d++], a = e[d++], f = e[d++], this._Rgbe2float(u, i, s, a, f, (t.height - r) * n * 3 + o * 3); r--; } return u; } } const hpe = "hdrFilteringVertexShader", Hpe = `attribute vec2 position;varying vec3 direction;uniform vec3 up;uniform vec3 right;uniform vec3 front; #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN mat3 view=mat3(up,right,front);direction=view*vec3(position,1.0);gl_Position=vec4(position,0.0,1.0); #define CUSTOM_VERTEX_MAIN_END }`; Le.ShadersStore[hpe] = Hpe; const gpe = "hdrFilteringPixelShader", Xpe = `#include #include #include #include uniform float alphaG;uniform samplerCube inputTexture;uniform vec2 vFilteringInfo;uniform float hdrScale;varying vec3 direction;void main() {vec3 color=radiance(alphaG,inputTexture,direction,vFilteringInfo);gl_FragColor=vec4(color*hdrScale,1.0);}`; Le.ShadersStore[gpe] = Xpe; class _Q { /** * Instantiates HDR filter for reflection maps * * @param engine Thin engine * @param options Options */ constructor(e, t = {}) { this._lodGenerationOffset = 0, this._lodGenerationScale = 0.8, this.quality = 4096, this.hdrScale = 1, this._engine = e, this.hdrScale = t.hdrScale || this.hdrScale, this.quality = t.quality || this.quality; } _createRenderTarget(e) { let t = 0; this._engine.getCaps().textureHalfFloatRender ? t = 2 : this._engine.getCaps().textureFloatRender && (t = 1); const r = this._engine.createRenderTargetCubeTexture(e, { format: 5, type: t, createMipMaps: !0, generateMipMaps: !1, generateDepthBuffer: !1, generateStencilBuffer: !1, samplingMode: 1 }); return this._engine.updateTextureWrappingMode(r.texture, 0, 0, 0), this._engine.updateTextureSamplingMode(3, r.texture, !0), r; } _prefilterInternal(e) { const t = e.getSize().width, r = Xt.ILog2(t) + 1, n = this._effectWrapper.effect, i = this._createRenderTarget(t); this._effectRenderer.saveStates(), this._effectRenderer.setViewport(); const s = e.getInternalTexture(); s && this._engine.updateTextureSamplingMode(3, s, !0), this._effectRenderer.applyEffectWrapper(this._effectWrapper); const a = [ [new S(0, 0, -1), new S(0, -1, 0), new S(1, 0, 0)], [new S(0, 0, 1), new S(0, -1, 0), new S(-1, 0, 0)], [new S(1, 0, 0), new S(0, 0, 1), new S(0, 1, 0)], [new S(1, 0, 0), new S(0, 0, -1), new S(0, -1, 0)], [new S(1, 0, 0), new S(0, -1, 0), new S(0, 0, 1)], [new S(-1, 0, 0), new S(0, -1, 0), new S(0, 0, -1)] // NegativeZ ]; n.setFloat("hdrScale", this.hdrScale), n.setFloat2("vFilteringInfo", e.getSize().width, r), n.setTexture("inputTexture", e); for (let d = 0; d < 6; d++) { n.setVector3("up", a[d][0]), n.setVector3("right", a[d][1]), n.setVector3("front", a[d][2]); for (let v = 0; v < r; v++) { this._engine.bindFramebuffer(i, d, void 0, void 0, !0, v), this._effectRenderer.applyEffectWrapper(this._effectWrapper); let u = Math.pow(2, (v - this._lodGenerationOffset) / this._lodGenerationScale) / t; v === 0 && (u = 0), n.setFloat("alphaG", u), this._effectRenderer.draw(); } } this._effectRenderer.restoreStates(), this._engine.restoreDefaultFramebuffer(), this._engine._releaseTexture(e._texture); const f = i.texture.type, o = i.texture.format; return i._swapAndDie(e._texture), e._texture.type = f, e._texture.format = o, e.gammaSpace = !1, e.lodGenerationOffset = this._lodGenerationOffset, e.lodGenerationScale = this._lodGenerationScale, e._prefiltered = !0, e; } _createEffect(e, t) { const r = []; return e.gammaSpace && r.push("#define GAMMA_INPUT"), r.push("#define NUM_SAMPLES " + this.quality + "u"), new ng({ engine: this._engine, name: "hdrFiltering", vertexShader: "hdrFiltering", fragmentShader: "hdrFiltering", samplerNames: ["inputTexture"], uniformNames: ["vSampleDirections", "vWeights", "up", "right", "front", "vFilteringInfo", "hdrScale", "alphaG"], useShaderStore: !0, defines: r, onCompiled: t }); } /** * Get a value indicating if the filter is ready to be used * @param texture Texture to filter * @returns true if the filter is ready */ isReady(e) { return e.isReady() && this._effectWrapper.effect.isReady(); } /** * Prefilters a cube texture to have mipmap levels representing roughness values. * Prefiltering will be invoked at the end of next rendering pass. * This has to be done once the map is loaded, and has not been prefiltered by a third party software. * See http://blog.selfshadow.com/publications/s2013-shading-course/karis/s2013_pbs_epic_notes_v2.pdf for more information * @param texture Texture to filter * @param onFinished Callback when filtering is done * @returns Promise called when prefiltering is done */ prefilter(e, t = null) { return this._engine._features.allowTexturePrefiltering ? new Promise((r) => { this._effectRenderer = new fU(this._engine), this._effectWrapper = this._createEffect(e), this._effectWrapper.effect.executeWhenCompiled(() => { this._prefilterInternal(e), this._effectRenderer.dispose(), this._effectWrapper.dispose(), r(), t && t(); }); }) : (Se.Warn("HDR prefiltering is not available in WebGL 1., you can use real time filtering instead."), Promise.reject("HDR prefiltering is not available in WebGL 1., you can use real time filtering instead.")); } } class _2 extends ls { /** * Sets whether or not the texture is blocking during loading. */ set isBlocking(e) { this._isBlocking = e; } /** * Gets whether or not the texture is blocking during loading. */ get isBlocking() { return this._isBlocking; } /** * Sets texture matrix rotation angle around Y axis in radians. */ set rotationY(e) { this._rotationY = e, this.setReflectionTextureMatrix(he.RotationY(this._rotationY)); } /** * Gets texture matrix rotation angle around Y axis radians. */ get rotationY() { return this._rotationY; } /** * Gets or sets the size of the bounding box associated with the cube texture * When defined, the cubemap will switch to local mode * @see https://community.arm.com/graphics/b/blog/posts/reflections-based-on-local-cubemaps-in-unity * @example https://www.babylonjs-playground.com/#RNASML */ set boundingBoxSize(e) { if (this._boundingBoxSize && this._boundingBoxSize.equals(e)) return; this._boundingBoxSize = e; const t = this.getScene(); t && t.markAllMaterialsAsDirty(1); } get boundingBoxSize() { return this._boundingBoxSize; } /** * Instantiates an HDRTexture from the following parameters. * * @param url The location of the HDR raw data (Panorama stored in RGBE format) * @param sceneOrEngine The scene or engine the texture will be used in * @param size The cubemap desired size (the more it increases the longer the generation will be) * @param noMipmap Forces to not generate the mipmap if true * @param generateHarmonics Specifies whether you want to extract the polynomial harmonics during the generation process * @param gammaSpace Specifies if the texture will be use in gamma or linear space (the PBR material requires those texture in linear space, but the standard material would require them in Gamma space) * @param prefilterOnLoad Prefilters HDR texture to allow use of this texture as a PBR reflection texture. * @param onLoad * @param onError */ constructor(e, t, r, n = !1, i = !0, s = !1, a = !1, f = null, o = null, d = !1) { var v; super(t), this._generateHarmonics = !0, this._onError = null, this._isBlocking = !0, this._rotationY = 0, this.boundingBoxPosition = S.Zero(), this.onLoadObservable = new Oe(), e && (this._coordinatesMode = We.CUBIC_MODE, this.name = e, this.url = e, this.hasAlpha = !1, this.isCube = !0, this._textureMatrix = he.Identity(), this._prefilterOnLoad = a, this._onLoad = () => { this.onLoadObservable.notifyObservers(this), f && f(); }, this._onError = o, this.gammaSpace = s, this._noMipmap = n, this._size = r, this._supersample = d, this._generateHarmonics = i, this._texture = this._getFromCache(e, this._noMipmap, void 0, void 0, void 0, this.isCube), this._texture ? this._texture.isReady ? ye.SetImmediate(() => this._onLoad()) : this._texture.onLoadedObservable.add(this._onLoad) : !((v = this.getScene()) === null || v === void 0) && v.useDelayedTextureLoading ? this.delayLoadState = 4 : this._loadTexture()); } /** * Get the current class name of the texture useful for serialization or dynamic coding. * @returns "HDRCubeTexture" */ getClassName() { return "HDRCubeTexture"; } /** * Occurs when the file is raw .hdr file. */ _loadTexture() { const e = this._getEngine(), t = e.getCaps(); let r = 0; t.textureFloat && t.textureFloatLinearFiltering ? r = 1 : t.textureHalfFloat && t.textureHalfFloatLinearFiltering && (r = 2); const n = (i) => { this.lodGenerationOffset = 0, this.lodGenerationScale = 0.8; const s = iO.GetCubeMapTextureData(i, this._size, this._supersample); if (this._generateHarmonics) { const d = Sm.ConvertCubeMapToSphericalPolynomial(s); this.sphericalPolynomial = d; } const a = []; let f = null, o = null; for (let d = 0; d < 6; d++) { r === 2 ? o = new Uint16Array(this._size * this._size * 3) : r === 0 && (f = new Uint8Array(this._size * this._size * 3)); const v = s[_2._FacesMapping[d]]; if (this.gammaSpace || o || f) { for (let u = 0; u < this._size * this._size; u++) if (this.gammaSpace && (v[u * 3 + 0] = Math.pow(v[u * 3 + 0], JW), v[u * 3 + 1] = Math.pow(v[u * 3 + 1], JW), v[u * 3 + 2] = Math.pow(v[u * 3 + 2], JW)), o && (o[u * 3 + 0] = LH(v[u * 3 + 0]), o[u * 3 + 1] = LH(v[u * 3 + 1]), o[u * 3 + 2] = LH(v[u * 3 + 2])), f) { let l = Math.max(v[u * 3 + 0] * 255, 0), P = Math.max(v[u * 3 + 1] * 255, 0), p = Math.max(v[u * 3 + 2] * 255, 0); const c = Math.max(Math.max(l, P), p); if (c > 255) { const H = 255 / c; l *= H, P *= H, p *= H; } f[u * 3 + 0] = l, f[u * 3 + 1] = P, f[u * 3 + 2] = p; } } o ? a.push(o) : f ? a.push(f) : a.push(v); } return a; }; if (e._features.allowTexturePrefiltering && this._prefilterOnLoad) { const i = this._onLoad, s = new _Q(e); this._onLoad = () => { s.prefilter(this, i); }; } this._texture = e.createRawCubeTextureFromUrl(this.url, this.getScene(), this._size, 4, r, this._noMipmap, n, null, this._onLoad, this._onError); } clone() { const e = new _2(this.url, this.getScene() || this._getEngine(), this._size, this._noMipmap, this._generateHarmonics, this.gammaSpace); return e.level = this.level, e.wrapU = this.wrapU, e.wrapV = this.wrapV, e.coordinatesIndex = this.coordinatesIndex, e.coordinatesMode = this.coordinatesMode, e; } // Methods delayLoad() { this.delayLoadState === 4 && (this.delayLoadState = 1, this._texture = this._getFromCache(this.url, this._noMipmap), this._texture || this._loadTexture()); } /** * Get the texture reflection matrix used to rotate/transform the reflection. * @returns the reflection matrix */ getReflectionTextureMatrix() { return this._textureMatrix; } /** * Set the texture reflection matrix used to rotate/transform the reflection. * @param value Define the reflection matrix to set */ setReflectionTextureMatrix(e) { var t; this._textureMatrix = e, e.updateFlag !== this._textureMatrix.updateFlag && e.isIdentity() !== this._textureMatrix.isIdentity() && ((t = this.getScene()) === null || t === void 0 || t.markAllMaterialsAsDirty(1, (r) => r.getActiveTextures().indexOf(this) !== -1)); } /** * Dispose the texture and release its associated resources. */ dispose() { this.onLoadObservable.clear(), super.dispose(); } /** * Parses a JSON representation of an HDR Texture in order to create the texture * @param parsedTexture Define the JSON representation * @param scene Define the scene the texture should be created in * @param rootUrl Define the root url in case we need to load relative dependencies * @returns the newly created texture after parsing */ static Parse(e, t, r) { let n = null; return e.name && !e.isRenderTarget && (n = new _2(r + e.name, t, e.size, e.noMipmap, e.generateHarmonics, e.useInGammaSpace), n.name = e.name, n.hasAlpha = e.hasAlpha, n.level = e.level, n.coordinatesMode = e.coordinatesMode, n.isBlocking = e.isBlocking), n && (e.boundingBoxPosition && (n.boundingBoxPosition = S.FromArray(e.boundingBoxPosition)), e.boundingBoxSize && (n.boundingBoxSize = S.FromArray(e.boundingBoxSize)), e.rotationY && (n.rotationY = e.rotationY)), n; } serialize() { if (!this.name) return null; const e = {}; return e.name = this.name, e.hasAlpha = this.hasAlpha, e.isCube = !0, e.level = this.level, e.size = this._size, e.coordinatesMode = this.coordinatesMode, e.useInGammaSpace = this.gammaSpace, e.generateHarmonics = this._generateHarmonics, e.customType = "BABYLON.HDRCubeTexture", e.noMipmap = this._noMipmap, e.isBlocking = this._isBlocking, e.rotationY = this._rotationY, e; } } _2._FacesMapping = ["right", "left", "up", "down", "front", "back"]; Ue("BABYLON.HDRCubeTexture", _2); class lD { /** * Gets or sets the influence of this target (ie. its weight in the overall morphing) */ get influence() { return this._influence; } set influence(e) { if (this._influence === e) return; const t = this._influence; this._influence = e, this.onInfluenceChanged.hasObservers() && this.onInfluenceChanged.notifyObservers(t === 0 || e === 0); } /** * Gets or sets the animation properties override */ get animationPropertiesOverride() { return !this._animationPropertiesOverride && this._scene ? this._scene.animationPropertiesOverride : this._animationPropertiesOverride; } set animationPropertiesOverride(e) { this._animationPropertiesOverride = e; } /** * Creates a new MorphTarget * @param name defines the name of the target * @param influence defines the influence to use * @param scene defines the scene the morphtarget belongs to */ constructor(e, t = 0, r = null) { this.name = e, this.animations = [], this._positions = null, this._normals = null, this._tangents = null, this._uvs = null, this._uniqueId = 0, this.onInfluenceChanged = new Oe(), this._onDataLayoutChanged = new Oe(), this._animationPropertiesOverride = null, this._scene = r || gr.LastCreatedScene, this.influence = t, this._scene && (this._uniqueId = this._scene.getUniqueId()); } /** * Gets the unique ID of this manager */ get uniqueId() { return this._uniqueId; } /** * Gets a boolean defining if the target contains position data */ get hasPositions() { return !!this._positions; } /** * Gets a boolean defining if the target contains normal data */ get hasNormals() { return !!this._normals; } /** * Gets a boolean defining if the target contains tangent data */ get hasTangents() { return !!this._tangents; } /** * Gets a boolean defining if the target contains texture coordinates data */ get hasUVs() { return !!this._uvs; } /** * Affects position data to this target * @param data defines the position data to use */ setPositions(e) { const t = this.hasPositions; this._positions = e, t !== this.hasPositions && this._onDataLayoutChanged.notifyObservers(void 0); } /** * Gets the position data stored in this target * @returns a FloatArray containing the position data (or null if not present) */ getPositions() { return this._positions; } /** * Affects normal data to this target * @param data defines the normal data to use */ setNormals(e) { const t = this.hasNormals; this._normals = e, t !== this.hasNormals && this._onDataLayoutChanged.notifyObservers(void 0); } /** * Gets the normal data stored in this target * @returns a FloatArray containing the normal data (or null if not present) */ getNormals() { return this._normals; } /** * Affects tangent data to this target * @param data defines the tangent data to use */ setTangents(e) { const t = this.hasTangents; this._tangents = e, t !== this.hasTangents && this._onDataLayoutChanged.notifyObservers(void 0); } /** * Gets the tangent data stored in this target * @returns a FloatArray containing the tangent data (or null if not present) */ getTangents() { return this._tangents; } /** * Affects texture coordinates data to this target * @param data defines the texture coordinates data to use */ setUVs(e) { const t = this.hasUVs; this._uvs = e, t !== this.hasUVs && this._onDataLayoutChanged.notifyObservers(void 0); } /** * Gets the texture coordinates data stored in this target * @returns a FloatArray containing the texture coordinates data (or null if not present) */ getUVs() { return this._uvs; } /** * Clone the current target * @returns a new MorphTarget */ clone() { const e = jt.Clone(() => new lD(this.name, this.influence, this._scene), this); return e._positions = this._positions, e._normals = this._normals, e._tangents = this._tangents, e._uvs = this._uvs, e; } /** * Serializes the current target into a Serialization object * @returns the serialized object */ serialize() { const e = {}; return e.name = this.name, e.influence = this.influence, e.positions = Array.prototype.slice.call(this.getPositions()), this.id != null && (e.id = this.id), this.hasNormals && (e.normals = Array.prototype.slice.call(this.getNormals())), this.hasTangents && (e.tangents = Array.prototype.slice.call(this.getTangents())), this.hasUVs && (e.uvs = Array.prototype.slice.call(this.getUVs())), jt.AppendSerializedAnimations(this, e), e; } /** * Returns the string "MorphTarget" * @returns "MorphTarget" */ getClassName() { return "MorphTarget"; } // Statics /** * Creates a new target from serialized data * @param serializationObject defines the serialized data to use * @param scene defines the hosting scene * @returns a new MorphTarget */ static Parse(e, t) { const r = new lD(e.name, e.influence); if (r.setPositions(e.positions), e.id != null && (r.id = e.id), e.normals && r.setNormals(e.normals), e.tangents && r.setTangents(e.tangents), e.uvs && r.setUVs(e.uvs), e.animations) { for (let n = 0; n < e.animations.length; n++) { const i = e.animations[n], s = Jo("BABYLON.Animation"); s && r.animations.push(s.Parse(i)); } e.autoAnimate && t && t.beginAnimation(r, e.autoAnimateFrom, e.autoAnimateTo, e.autoAnimateLoop, e.autoAnimateSpeed || 1); } return r; } /** * Creates a MorphTarget from mesh data * @param mesh defines the source mesh * @param name defines the name to use for the new target * @param influence defines the influence to attach to the target * @returns a new MorphTarget */ static FromMesh(e, t, r) { t || (t = e.name); const n = new lD(t, r, e.getScene()); return n.setPositions(e.getVerticesData(J.PositionKind)), e.isVerticesDataPresent(J.NormalKind) && n.setNormals(e.getVerticesData(J.NormalKind)), e.isVerticesDataPresent(J.TangentKind) && n.setTangents(e.getVerticesData(J.TangentKind)), e.isVerticesDataPresent(J.UVKind) && n.setUVs(e.getVerticesData(J.UVKind)), n; } } C([ M() ], lD.prototype, "id", void 0); class Py extends We { /** * Gets the number of layers of the texture */ get depth() { return this._depth; } /** * Create a new RawTexture2DArray * @param data defines the data of the texture * @param width defines the width of the texture * @param height defines the height of the texture * @param depth defines the number of layers of the texture * @param format defines the texture format to use * @param scene defines the hosting scene * @param generateMipMaps defines a boolean indicating if mip levels should be generated (true by default) * @param invertY defines if texture must be stored with Y axis inverted * @param samplingMode defines the sampling mode to use (Texture.TRILINEAR_SAMPLINGMODE by default) * @param textureType defines the texture Type (Engine.TEXTURETYPE_UNSIGNED_INT, Engine.TEXTURETYPE_FLOAT...) * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg) */ constructor(e, t, r, n, i, s, a = !0, f = !1, o = We.TRILINEAR_SAMPLINGMODE, d = 0, v) { super(null, s, !a, f), this.format = i, this._texture = s.getEngine().createRawTexture2DArray(e, t, r, n, i, a, f, o, null, d, v), this._depth = n, this.is2DArray = !0; } /** * Update the texture with new data * @param data defines the data to store in the texture */ update(e) { this._texture && this._getEngine().updateRawTexture2DArray(this._texture, e, this._texture.format, this._texture.invertY, null, this._texture.type); } /** * Creates a RGBA texture from some data. * @param data Define the texture data * @param width Define the width of the texture * @param height Define the height of the texture * @param depth defines the number of layers of the texture * @param scene defines the scene the texture will belong to * @param generateMipMaps Define whether or not to create mip maps for the texture * @param invertY define if the data should be flipped on Y when uploaded to the GPU * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE) * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx) * @returns the RGBA texture */ static CreateRGBATexture(e, t, r, n, i, s = !0, a = !1, f = 3, o = 0) { return new Py(e, t, r, n, 5, i, s, a, f, o); } } class U0 { /** * Sets a boolean indicating that adding new target or updating an existing target will not update the underlying data buffers */ set areUpdatesFrozen(e) { e ? this._blockCounter++ : (this._blockCounter--, this._blockCounter <= 0 && (this._blockCounter = 0, this._syncActiveTargets(!0))); } get areUpdatesFrozen() { return this._blockCounter > 0; } /** * Creates a new MorphTargetManager * @param scene defines the current scene */ constructor(e = null) { if (this._targets = new Array(), this._targetInfluenceChangedObservers = new Array(), this._targetDataLayoutChangedObservers = new Array(), this._activeTargets = new qf(16), this._supportsNormals = !1, this._supportsTangents = !1, this._supportsUVs = !1, this._vertexCount = 0, this._textureVertexStride = 0, this._textureWidth = 0, this._textureHeight = 1, this._uniqueId = 0, this._tempInfluences = new Array(), this._canUseTextureForTargets = !1, this._blockCounter = 0, this._parentContainer = null, this.optimizeInfluencers = !0, this.enableNormalMorphing = !0, this.enableTangentMorphing = !0, this.enableUVMorphing = !0, this._useTextureToStoreTargets = !0, e || (e = gr.LastCreatedScene), this._scene = e, this._scene) { this._scene.addMorphTargetManager(this), this._uniqueId = this._scene.getUniqueId(); const t = this._scene.getEngine().getCaps(); this._canUseTextureForTargets = t.canUseGLVertexID && t.textureFloat && t.maxVertexTextureImageUnits > 0 && t.texture2DArrayMaxLayerCount > 1; } } /** * Gets the unique ID of this manager */ get uniqueId() { return this._uniqueId; } /** * Gets the number of vertices handled by this manager */ get vertexCount() { return this._vertexCount; } /** * Gets a boolean indicating if this manager supports morphing of normals */ get supportsNormals() { return this._supportsNormals && this.enableNormalMorphing; } /** * Gets a boolean indicating if this manager supports morphing of tangents */ get supportsTangents() { return this._supportsTangents && this.enableTangentMorphing; } /** * Gets a boolean indicating if this manager supports morphing of texture coordinates */ get supportsUVs() { return this._supportsUVs && this.enableUVMorphing; } /** * Gets the number of targets stored in this manager */ get numTargets() { return this._targets.length; } /** * Gets the number of influencers (ie. the number of targets with influences > 0) */ get numInfluencers() { return this._activeTargets.length; } /** * Gets the list of influences (one per target) */ get influences() { return this._influences; } /** * Gets or sets a boolean indicating that targets should be stored as a texture instead of using vertex attributes (default is true). * Please note that this option is not available if the hardware does not support it */ get useTextureToStoreTargets() { return this._useTextureToStoreTargets; } set useTextureToStoreTargets(e) { this._useTextureToStoreTargets = e; } /** * Gets a boolean indicating that the targets are stored into a texture (instead of as attributes) */ get isUsingTextureForTargets() { var e; return U0.EnableTextureStorage && this.useTextureToStoreTargets && this._canUseTextureForTargets && !(!((e = this._scene) === null || e === void 0) && e.getEngine().getCaps().disableMorphTargetTexture); } /** * Gets the active target at specified index. An active target is a target with an influence > 0 * @param index defines the index to check * @returns the requested target */ getActiveTarget(e) { return this._activeTargets.data[e]; } /** * Gets the target at specified index * @param index defines the index to check * @returns the requested target */ getTarget(e) { return this._targets[e]; } /** * Add a new target to this manager * @param target defines the target to add */ addTarget(e) { this._targets.push(e), this._targetInfluenceChangedObservers.push(e.onInfluenceChanged.add((t) => { this._syncActiveTargets(t); })), this._targetDataLayoutChangedObservers.push(e._onDataLayoutChanged.add(() => { this._syncActiveTargets(!0); })), this._syncActiveTargets(!0); } /** * Removes a target from the manager * @param target defines the target to remove */ removeTarget(e) { const t = this._targets.indexOf(e); t >= 0 && (this._targets.splice(t, 1), e.onInfluenceChanged.remove(this._targetInfluenceChangedObservers.splice(t, 1)[0]), e._onDataLayoutChanged.remove(this._targetDataLayoutChangedObservers.splice(t, 1)[0]), this._syncActiveTargets(!0)), this._scene && this._scene.stopAnimation(e); } /** * @internal */ _bind(e) { e.setFloat3("morphTargetTextureInfo", this._textureVertexStride, this._textureWidth, this._textureHeight), e.setFloatArray("morphTargetTextureIndices", this._morphTargetTextureIndices), e.setTexture("morphTargets", this._targetStoreTexture); } /** * Clone the current manager * @returns a new MorphTargetManager */ clone() { const e = new U0(this._scene); for (const t of this._targets) e.addTarget(t.clone()); return e.enableNormalMorphing = this.enableNormalMorphing, e.enableTangentMorphing = this.enableTangentMorphing, e.enableUVMorphing = this.enableUVMorphing, e; } /** * Serializes the current manager into a Serialization object * @returns the serialized object */ serialize() { const e = {}; e.id = this.uniqueId, e.targets = []; for (const t of this._targets) e.targets.push(t.serialize()); return e; } _syncActiveTargets(e) { if (this.areUpdatesFrozen) return; let t = 0; this._activeTargets.reset(), this._supportsNormals = !0, this._supportsTangents = !0, this._supportsUVs = !0, this._vertexCount = 0, this._scene && this._targets.length > this._scene.getEngine().getCaps().texture2DArrayMaxLayerCount && (this.useTextureToStoreTargets = !1), (!this._morphTargetTextureIndices || this._morphTargetTextureIndices.length !== this._targets.length) && (this._morphTargetTextureIndices = new Float32Array(this._targets.length)); let r = -1; for (const n of this._targets) { if (r++, n.influence === 0 && this.optimizeInfluencers) continue; if (this._activeTargets.length >= U0.MaxActiveMorphTargetsInVertexAttributeMode && !this.isUsingTextureForTargets) break; this._activeTargets.push(n), this._morphTargetTextureIndices[t] = r, this._tempInfluences[t++] = n.influence, this._supportsNormals = this._supportsNormals && n.hasNormals, this._supportsTangents = this._supportsTangents && n.hasTangents, this._supportsUVs = this._supportsUVs && n.hasUVs; const i = n.getPositions(); if (i) { const s = i.length / 3; if (this._vertexCount === 0) this._vertexCount = s; else if (this._vertexCount !== s) { Se.Error("Incompatible target. Targets must all have the same vertices count."); return; } } } this._morphTargetTextureIndices.length !== t && (this._morphTargetTextureIndices = this._morphTargetTextureIndices.slice(0, t)), (!this._influences || this._influences.length !== t) && (this._influences = new Float32Array(t)); for (let n = 0; n < t; n++) this._influences[n] = this._tempInfluences[n]; e && this.synchronize(); } /** * Synchronize the targets with all the meshes using this morph target manager */ synchronize() { if (!(!this._scene || this.areUpdatesFrozen)) { if (this.isUsingTextureForTargets && this._vertexCount) { this._textureVertexStride = 1, this._supportsNormals && this._textureVertexStride++, this._supportsTangents && this._textureVertexStride++, this._supportsUVs && this._textureVertexStride++, this._textureWidth = this._vertexCount * this._textureVertexStride, this._textureHeight = 1; const e = this._scene.getEngine().getCaps().maxTextureSize; this._textureWidth > e && (this._textureHeight = Math.ceil(this._textureWidth / e), this._textureWidth = e); let t = !0; if (this._targetStoreTexture) { const r = this._targetStoreTexture.getSize(); r.width === this._textureWidth && r.height === this._textureHeight && this._targetStoreTexture.depth === this._targets.length && (t = !1); } if (t) { this._targetStoreTexture && this._targetStoreTexture.dispose(); const r = this._targets.length, n = new Float32Array(r * this._textureWidth * this._textureHeight * 4); let i = 0; for (let s = 0; s < r; s++) { const a = this._targets[s], f = a.getPositions(), o = a.getNormals(), d = a.getUVs(), v = a.getTangents(); if (!f) { s === 0 && Se.Error("Invalid morph target. Target must have positions."); return; } i = s * this._textureWidth * this._textureHeight * 4; for (let u = 0; u < this._vertexCount; u++) n[i] = f[u * 3], n[i + 1] = f[u * 3 + 1], n[i + 2] = f[u * 3 + 2], i += 4, this._supportsNormals && o && (n[i] = o[u * 3], n[i + 1] = o[u * 3 + 1], n[i + 2] = o[u * 3 + 2], i += 4), this._supportsUVs && d && (n[i] = d[u * 2], n[i + 1] = d[u * 2 + 1], i += 4), this._supportsTangents && v && (n[i] = v[u * 3], n[i + 1] = v[u * 3 + 1], n[i + 2] = v[u * 3 + 2], i += 4); } this._targetStoreTexture = Py.CreateRGBATexture(n, this._textureWidth, this._textureHeight, r, this._scene, !1, !1, 1, 1); } } for (const e of this._scene.meshes) e.morphTargetManager === this && e._syncGeometryWithMorphTargetManager(); } } /** * Release all resources */ dispose() { if (this._targetStoreTexture && this._targetStoreTexture.dispose(), this._targetStoreTexture = null, this._scene) { if (this._scene.removeMorphTargetManager(this), this._parentContainer) { const e = this._parentContainer.morphTargetManagers.indexOf(this); e > -1 && this._parentContainer.morphTargetManagers.splice(e, 1), this._parentContainer = null; } for (const e of this._targets) this._scene.stopAnimation(e); } } // Statics /** * Creates a new MorphTargetManager from serialized data * @param serializationObject defines the serialized data * @param scene defines the hosting scene * @returns the new MorphTargetManager */ static Parse(e, t) { const r = new U0(t); r._uniqueId = e.id; for (const n of e.targets) r.addTarget(lD.Parse(n, t)); return r; } } U0.EnableTextureStorage = !0; U0.MaxActiveMorphTargetsInVertexAttributeMode = 8; class iV { constructor() { this._hasHit = !1, this._hitDistance = 0, this._hitNormalWorld = S.Zero(), this._hitPointWorld = S.Zero(), this._rayFromWorld = S.Zero(), this._rayToWorld = S.Zero(), this._triangleIndex = -1; } /** * Gets if there was a hit */ get hasHit() { return this._hasHit; } /** * Gets the distance from the hit */ get hitDistance() { return this._hitDistance; } /** * Gets the hit normal/direction in the world */ get hitNormalWorld() { return this._hitNormalWorld; } /** * Gets the hit point in the world */ get hitPointWorld() { return this._hitPointWorld; } /** * Gets the ray "start point" of the ray in the world */ get rayFromWorld() { return this._rayFromWorld; } /** * Gets the ray "end point" of the ray in the world */ get rayToWorld() { return this._rayToWorld; } /* * The index of the original triangle which was hit. Will be -1 if contact point is not on a mesh shape */ get triangleIndex() { return this._triangleIndex; } /** * Sets the hit data (normal & point in world space) * @param hitNormalWorld defines the normal in world space * @param hitPointWorld defines the point in world space */ setHitData(e, t, r) { this._hasHit = !0, this._hitNormalWorld.set(e.x, e.y, e.z), this._hitPointWorld.set(t.x, t.y, t.z), this._triangleIndex = r ?? -1; } /** * Sets the distance from the start point to the hit point * @param distance */ setHitDistance(e) { this._hitDistance = e; } /** * Calculates the distance manually */ calculateHitDistance() { this._hitDistance = S.Distance(this._rayFromWorld, this._hitPointWorld); } /** * Resets all the values to default * @param from The from point on world space * @param to The to point on world space */ reset(e = S.Zero(), t = S.Zero()) { this._rayFromWorld.copyFrom(e), this._rayToWorld.copyFrom(t), this._hasHit = !1, this._hitDistance = 0, this._hitNormalWorld.setAll(0), this._hitPointWorld.setAll(0), this._triangleIndex = -1, this.body = void 0, this.bodyIndex = void 0; } } let $Q = class zte { /** * * @returns version */ getPluginVersion() { return this._physicsPlugin.getPluginVersion(); } /** * Factory used to create the default physics plugin. * @returns The default physics plugin */ static DefaultPluginFactory() { throw qn("CannonJSPlugin"); } /** * Creates a new Physics Engine * @param gravity defines the gravity vector used by the simulation * @param _physicsPlugin defines the plugin to use (CannonJS by default) */ constructor(e, t = zte.DefaultPluginFactory()) { if (this._physicsPlugin = t, this._impostors = [], this._joints = [], this._subTimeStep = 0, this._uniqueIdCounter = 0, !this._physicsPlugin.isSupported()) throw new Error("Physics Engine " + this._physicsPlugin.name + " cannot be found. Please make sure it is included."); e = e || new S(0, -9.807, 0), this.setGravity(e), this.setTimeStep(); } /** * Sets the gravity vector used by the simulation * @param gravity defines the gravity vector to use */ setGravity(e) { this.gravity = e, this._physicsPlugin.setGravity(this.gravity); } /** * Set the time step of the physics engine. * Default is 1/60. * To slow it down, enter 1/600 for example. * To speed it up, 1/30 * @param newTimeStep defines the new timestep to apply to this world. */ setTimeStep(e = 1 / 60) { this._physicsPlugin.setTimeStep(e); } /** * Get the time step of the physics engine. * @returns the current time step */ getTimeStep() { return this._physicsPlugin.getTimeStep(); } /** * Set the sub time step of the physics engine. * Default is 0 meaning there is no sub steps * To increase physics resolution precision, set a small value (like 1 ms) * @param subTimeStep defines the new sub timestep used for physics resolution. */ setSubTimeStep(e = 0) { this._subTimeStep = e; } /** * Get the sub time step of the physics engine. * @returns the current sub time step */ getSubTimeStep() { return this._subTimeStep; } /** * Release all resources */ dispose() { this._impostors.forEach(function(e) { e.dispose(); }), this._physicsPlugin.dispose(); } /** * Gets the name of the current physics plugin * @returns the name of the plugin */ getPhysicsPluginName() { return this._physicsPlugin.name; } /** * Adding a new impostor for the impostor tracking. * This will be done by the impostor itself. * @param impostor the impostor to add */ addImpostor(e) { this._impostors.push(e), e.uniqueId = this._uniqueIdCounter++, e.parent || this._physicsPlugin.generatePhysicsBody(e); } /** * Remove an impostor from the engine. * This impostor and its mesh will not longer be updated by the physics engine. * @param impostor the impostor to remove */ removeImpostor(e) { const t = this._impostors.indexOf(e); t > -1 && this._impostors.splice(t, 1).length && this.getPhysicsPlugin().removePhysicsBody(e); } /** * Add a joint to the physics engine * @param mainImpostor defines the main impostor to which the joint is added. * @param connectedImpostor defines the impostor that is connected to the main impostor using this joint * @param joint defines the joint that will connect both impostors. */ addJoint(e, t, r) { const n = { mainImpostor: e, connectedImpostor: t, joint: r }; r.physicsPlugin = this._physicsPlugin, this._joints.push(n), this._physicsPlugin.generateJoint(n); } /** * Removes a joint from the simulation * @param mainImpostor defines the impostor used with the joint * @param connectedImpostor defines the other impostor connected to the main one by the joint * @param joint defines the joint to remove */ removeJoint(e, t, r) { const n = this._joints.filter(function(i) { return i.connectedImpostor === t && i.joint === r && i.mainImpostor === e; }); n.length && this._physicsPlugin.removeJoint(n[0]); } /** * Called by the scene. No need to call it. * @param delta defines the timespan between frames */ _step(e) { this._impostors.forEach((t) => { t.isBodyInitRequired() && this._physicsPlugin.generatePhysicsBody(t); }), e > 0.1 ? e = 0.1 : e <= 0 && (e = 1 / 60), this._physicsPlugin.executeStep(e, this._impostors); } /** * Gets the current plugin used to run the simulation * @returns current plugin */ getPhysicsPlugin() { return this._physicsPlugin; } /** * Gets the list of physic impostors * @returns an array of PhysicsImpostor */ getImpostors() { return this._impostors; } /** * Gets the impostor for a physics enabled object * @param object defines the object impersonated by the impostor * @returns the PhysicsImpostor or null if not found */ getImpostorForPhysicsObject(e) { for (let t = 0; t < this._impostors.length; ++t) if (this._impostors[t].object === e) return this._impostors[t]; return null; } /** * Gets the impostor for a physics body object * @param body defines physics body used by the impostor * @returns the PhysicsImpostor or null if not found */ getImpostorWithPhysicsBody(e) { for (let t = 0; t < this._impostors.length; ++t) if (this._impostors[t].physicsBody === e) return this._impostors[t]; return null; } /** * Does a raycast in the physics world * @param from when should the ray start? * @param to when should the ray end? * @returns PhysicsRaycastResult */ raycast(e, t) { return this._physicsPlugin.raycast(e, t); } /** * Does a raycast in the physics world * @param from when should the ray start? * @param to when should the ray end? * @param result resulting PhysicsRaycastResult */ raycastToRef(e, t, r) { return this._physicsPlugin.raycastToRef(e, t, r); } }; class sO { constructor(e = !0, t = 10, r = CANNON) { if (this._useDeltaForWorldStep = e, this.name = "CannonJSPlugin", this._physicsMaterials = new Array(), this._fixedTimeStep = 1 / 60, this._physicsBodiesToRemoveAfterStep = new Array(), this._firstFrame = !0, this._tmpQuaternion = new Ze(), this._minus90X = new Ze(-0.7071067811865475, 0, 0, 0.7071067811865475), this._plus90X = new Ze(0.7071067811865475, 0, 0, 0.7071067811865475), this._tmpPosition = S.Zero(), this._tmpDeltaPosition = S.Zero(), this._tmpUnityRotation = new Ze(), this.BJSCANNON = r, !this.isSupported()) { Se.Error("CannonJS is not available. Please make sure you included the js file."); return; } this._extendNamespace(), this.world = new this.BJSCANNON.World(), this.world.broadphase = new this.BJSCANNON.NaiveBroadphase(), this.world.solver.iterations = t, this._cannonRaycastResult = new this.BJSCANNON.RaycastResult(), this._raycastResult = new iV(); } /** * * @returns plugin version */ getPluginVersion() { return 1; } setGravity(e) { const t = e; this.world.gravity.set(t.x, t.y, t.z); } setTimeStep(e) { this._fixedTimeStep = e; } getTimeStep() { return this._fixedTimeStep; } executeStep(e, t) { if (this._firstFrame) { this._firstFrame = !1; for (const r of t) r.type == tn.HeightmapImpostor || r.type === tn.PlaneImpostor || r.beforeStep(); } this.world.step(this._useDeltaForWorldStep ? e : this._fixedTimeStep), this._removeMarkedPhysicsBodiesFromWorld(); } _removeMarkedPhysicsBodiesFromWorld() { this._physicsBodiesToRemoveAfterStep.length > 0 && (this._physicsBodiesToRemoveAfterStep.forEach((e) => { typeof this.world.removeBody == "function" ? this.world.removeBody(e) : this.world.remove(e); }), this._physicsBodiesToRemoveAfterStep.length = 0); } applyImpulse(e, t, r) { const n = new this.BJSCANNON.Vec3(r.x, r.y, r.z), i = new this.BJSCANNON.Vec3(t.x, t.y, t.z); e.physicsBody.applyImpulse(i, n); } applyForce(e, t, r) { const n = new this.BJSCANNON.Vec3(r.x, r.y, r.z), i = new this.BJSCANNON.Vec3(t.x, t.y, t.z); e.physicsBody.applyForce(i, n); } generatePhysicsBody(e) { if (this._removeMarkedPhysicsBodiesFromWorld(), e.parent) { e.physicsBody && (this.removePhysicsBody(e), e.forceUpdate()); return; } if (e.isBodyInitRequired()) { const t = this._createShape(e); if (!t) { Se.Warn("It was not possible to create a physics body for this object."); return; } const r = e.physicsBody; r && this.removePhysicsBody(e); const n = this._addMaterial("mat-" + e.uniqueId, e.getParam("friction"), e.getParam("restitution")), i = { mass: e.getParam("mass"), material: n }, s = e.getParam("nativeOptions"); for (const a in s) Object.prototype.hasOwnProperty.call(s, a) && (i[a] = s[a]); e.physicsBody = new this.BJSCANNON.Body(i), e.physicsBody.addEventListener("collide", e.onCollide), this.world.addEventListener("preStep", e.beforeStep), this.world.addEventListener("postStep", e.afterStep), e.physicsBody.addShape(t), typeof this.world.addBody == "function" ? this.world.addBody(e.physicsBody) : this.world.add(e.physicsBody), r && ["force", "torque", "velocity", "angularVelocity"].forEach(function(a) { const f = r[a]; e.physicsBody[a].set(f.x, f.y, f.z); }), this._processChildMeshes(e); } this._updatePhysicsBodyTransformation(e); } _processChildMeshes(e) { const t = e.object.getChildMeshes ? e.object.getChildMeshes(!0) : [], r = e.object.rotationQuaternion; if (r ? r.conjugateToRef(this._tmpQuaternion) : this._tmpQuaternion.set(0, 0, 0, 1), t.length) { const n = (i) => { if (!i.rotationQuaternion) return; const s = i.getPhysicsImpostor(); if (s && s.parent !== e && i.parent) { const f = i.getAbsolutePosition().subtract(i.parent.getAbsolutePosition()), o = i.rotationQuaternion.multiply(this._tmpQuaternion); s.physicsBody && (this.removePhysicsBody(s), s.physicsBody = null), s.parent = e, s.resetUpdateFlags(), e.physicsBody.addShape(this._createShape(s), new this.BJSCANNON.Vec3(f.x, f.y, f.z), new this.BJSCANNON.Quaternion(o.x, o.y, o.z, o.w)), e.physicsBody.mass += s.getParam("mass"); } i.getChildMeshes(!0).filter((a) => !!a.physicsImpostor).forEach(n); }; t.filter((i) => !!i.physicsImpostor).forEach(n); } } removePhysicsBody(e) { e.physicsBody.removeEventListener("collide", e.onCollide), this.world.removeEventListener("preStep", e.beforeStep), this.world.removeEventListener("postStep", e.afterStep), this._physicsBodiesToRemoveAfterStep.indexOf(e.physicsBody) === -1 && this._physicsBodiesToRemoveAfterStep.push(e.physicsBody); } generateJoint(e) { const t = e.mainImpostor.physicsBody, r = e.connectedImpostor.physicsBody; if (!t || !r) return; let n; const i = e.joint.jointData, s = { pivotA: i.mainPivot ? new this.BJSCANNON.Vec3().set(i.mainPivot.x, i.mainPivot.y, i.mainPivot.z) : null, pivotB: i.connectedPivot ? new this.BJSCANNON.Vec3().set(i.connectedPivot.x, i.connectedPivot.y, i.connectedPivot.z) : null, axisA: i.mainAxis ? new this.BJSCANNON.Vec3().set(i.mainAxis.x, i.mainAxis.y, i.mainAxis.z) : null, axisB: i.connectedAxis ? new this.BJSCANNON.Vec3().set(i.connectedAxis.x, i.connectedAxis.y, i.connectedAxis.z) : null, maxForce: i.nativeParams.maxForce, collideConnected: !!i.collision }; switch (e.joint.type) { case ta.HingeJoint: case ta.Hinge2Joint: n = new this.BJSCANNON.HingeConstraint(t, r, s); break; case ta.DistanceJoint: n = new this.BJSCANNON.DistanceConstraint(t, r, i.maxDistance || 2); break; case ta.SpringJoint: { const a = i; n = new this.BJSCANNON.Spring(t, r, { restLength: a.length, stiffness: a.stiffness, damping: a.damping, localAnchorA: s.pivotA, localAnchorB: s.pivotB }); break; } case ta.LockJoint: n = new this.BJSCANNON.LockConstraint(t, r, s); break; case ta.PointToPointJoint: case ta.BallAndSocketJoint: default: n = new this.BJSCANNON.PointToPointConstraint(t, s.pivotA, r, s.pivotB, s.maxForce); break; } n.collideConnected = !!i.collision, e.joint.physicsJoint = n, e.joint.type !== ta.SpringJoint ? this.world.addConstraint(n) : (e.joint.jointData.forceApplicationCallback = e.joint.jointData.forceApplicationCallback || function() { n.applyForce(); }, e.mainImpostor.registerAfterPhysicsStep(e.joint.jointData.forceApplicationCallback)); } removeJoint(e) { e.joint.type !== ta.SpringJoint ? this.world.removeConstraint(e.joint.physicsJoint) : e.mainImpostor.unregisterAfterPhysicsStep(e.joint.jointData.forceApplicationCallback); } _addMaterial(e, t, r) { let n, i; for (n = 0; n < this._physicsMaterials.length; n++) if (i = this._physicsMaterials[n], i.friction === t && i.restitution === r) return i; const s = new this.BJSCANNON.Material(e); return s.friction = t, s.restitution = r, this._physicsMaterials.push(s), s; } _checkWithEpsilon(e) { return e < Dn ? Dn : e; } _createShape(e) { const t = e.object; let r; const n = e.getObjectExtents(); switch (e.type) { case tn.SphereImpostor: { const i = n.x, s = n.y, a = n.z; r = new this.BJSCANNON.Sphere(Math.max(this._checkWithEpsilon(i), this._checkWithEpsilon(s), this._checkWithEpsilon(a)) / 2); break; } case tn.CylinderImpostor: { let i = e.getParam("nativeOptions"); i || (i = {}); const s = i.radiusTop !== void 0 ? i.radiusTop : this._checkWithEpsilon(n.x) / 2, a = i.radiusBottom !== void 0 ? i.radiusBottom : this._checkWithEpsilon(n.x) / 2, f = i.height !== void 0 ? i.height : this._checkWithEpsilon(n.y), o = i.numSegments !== void 0 ? i.numSegments : 16; r = new this.BJSCANNON.Cylinder(s, a, f, o); const d = new this.BJSCANNON.Quaternion(); d.setFromAxisAngle(new this.BJSCANNON.Vec3(1, 0, 0), -Math.PI / 2); const v = new this.BJSCANNON.Vec3(0, 0, 0); r.transformAllPoints(v, d); break; } case tn.BoxImpostor: { const i = n.scale(0.5); r = new this.BJSCANNON.Box(new this.BJSCANNON.Vec3(this._checkWithEpsilon(i.x), this._checkWithEpsilon(i.y), this._checkWithEpsilon(i.z))); break; } case tn.PlaneImpostor: Se.Warn("Attention, PlaneImposter might not behave as you expect. Consider using BoxImposter instead"), r = new this.BJSCANNON.Plane(); break; case tn.MeshImpostor: { const i = t.getVerticesData ? t.getVerticesData(J.PositionKind) : [], s = t.getIndices ? t.getIndices() : []; if (!i) { Se.Warn("Tried to create a MeshImpostor for an object without vertices. This will fail."); return; } const a = t.position.clone(), f = t.rotation && t.rotation.clone(), o = t.rotationQuaternion && t.rotationQuaternion.clone(); t.position.copyFromFloats(0, 0, 0), t.rotation && t.rotation.copyFromFloats(0, 0, 0), t.rotationQuaternion && t.rotationQuaternion.copyFrom(e.getParentsRotation()), t.rotationQuaternion && t.parent && t.rotationQuaternion.conjugateInPlace(); const d = t.computeWorldMatrix(!0), v = []; let u; for (u = 0; u < i.length; u += 3) S.TransformCoordinates(S.FromArray(i, u), d).toArray(v, u); Se.Warn("MeshImpostor only collides against spheres."), r = new this.BJSCANNON.Trimesh(v, s), t.position.copyFrom(a), f && t.rotation && t.rotation.copyFrom(f), o && t.rotationQuaternion && t.rotationQuaternion.copyFrom(o); break; } case tn.HeightmapImpostor: { const i = t.position.clone(), s = t.rotation && t.rotation.clone(), a = t.rotationQuaternion && t.rotationQuaternion.clone(); t.position.copyFromFloats(0, 0, 0), t.rotation && t.rotation.copyFromFloats(0, 0, 0), t.rotationQuaternion && t.rotationQuaternion.copyFrom(e.getParentsRotation()), t.rotationQuaternion && t.parent && t.rotationQuaternion.conjugateInPlace(), t.rotationQuaternion && t.rotationQuaternion.multiplyInPlace(this._minus90X), r = this._createHeightmap(t), t.position.copyFrom(i), s && t.rotation && t.rotation.copyFrom(s), a && t.rotationQuaternion && t.rotationQuaternion.copyFrom(a), t.computeWorldMatrix(!0); break; } case tn.ParticleImpostor: r = new this.BJSCANNON.Particle(); break; case tn.NoImpostor: r = new this.BJSCANNON.Box(new this.BJSCANNON.Vec3(0, 0, 0)); break; } return r; } _createHeightmap(e, t) { let r = e.getVerticesData(J.PositionKind); const n = e.computeWorldMatrix(!0), i = []; let s; for (s = 0; s < r.length; s += 3) S.TransformCoordinates(S.FromArray(r, s), n).toArray(i, s); r = i; const a = new Array(), f = t || ~~(Math.sqrt(r.length / 3) - 1), o = e.getBoundingInfo(), d = Math.min(o.boundingBox.extendSizeWorld.x, o.boundingBox.extendSizeWorld.y), v = o.boundingBox.extendSizeWorld.z, u = d * 2 / f; for (let P = 0; P < r.length; P = P + 3) { const p = Math.round(r[P + 0] / u + f / 2), c = Math.round((r[P + 1] / u - f / 2) * -1), H = -r[P + 2] + v; a[p] || (a[p] = []), a[p][c] || (a[p][c] = H), a[p][c] = Math.max(H, a[p][c]); } for (let P = 0; P <= f; ++P) { if (!a[P]) { let p = 1; for (; !a[(P + p) % f]; ) p++; a[P] = a[(P + p) % f].slice(); } for (let p = 0; p <= f; ++p) if (!a[P][p]) { let c = 1, H; for (; H === void 0; ) H = a[P][(p + c++) % f]; a[P][p] = H; } } const l = new this.BJSCANNON.Heightfield(a, { elementSize: u }); return l.minY = v, l; } _updatePhysicsBodyTransformation(e) { const t = e.object; if (t.computeWorldMatrix && t.computeWorldMatrix(!0), !t.getBoundingInfo()) return; const r = e.getObjectCenter(); this._tmpDeltaPosition.copyFrom(t.getAbsolutePivotPoint().subtract(r)), this._tmpDeltaPosition.divideInPlace(e.object.scaling), this._tmpPosition.copyFrom(r); let n = t.rotationQuaternion; if (n) { if ((e.type === tn.PlaneImpostor || e.type === tn.HeightmapImpostor) && (n = n.multiply(this._minus90X), e.setDeltaRotation(this._plus90X)), e.type === tn.HeightmapImpostor) { const i = t; let s = i.getBoundingInfo(); const a = i.rotationQuaternion; i.rotationQuaternion = this._tmpUnityRotation, i.computeWorldMatrix(!0); const f = r.clone(); let o = i.getPivotMatrix(); o ? o = o.clone() : o = he.Identity(); const d = he.Translation(s.boundingBox.extendSizeWorld.x, 0, -s.boundingBox.extendSizeWorld.z); i.setPreTransformMatrix(d), i.computeWorldMatrix(!0), s = i.getBoundingInfo(); const v = s.boundingBox.centerWorld.subtract(r).subtract(i.position).negate(); this._tmpPosition.copyFromFloats(v.x, v.y - s.boundingBox.extendSizeWorld.y, v.z), this._tmpDeltaPosition.copyFrom(s.boundingBox.centerWorld.subtract(f)), this._tmpDeltaPosition.y += s.boundingBox.extendSizeWorld.y, i.rotationQuaternion = a, i.setPreTransformMatrix(o), i.computeWorldMatrix(!0); } else e.type === tn.MeshImpostor && this._tmpDeltaPosition.copyFromFloats(0, 0, 0); e.setDeltaPosition(this._tmpDeltaPosition), e.physicsBody.position.set(this._tmpPosition.x, this._tmpPosition.y, this._tmpPosition.z), e.physicsBody.quaternion.set(n.x, n.y, n.z, n.w); } } setTransformationFromPhysicsBody(e) { if (e.object.position.set(e.physicsBody.position.x, e.physicsBody.position.y, e.physicsBody.position.z), e.object.rotationQuaternion) { const t = e.physicsBody.quaternion; e.object.rotationQuaternion.set(t.x, t.y, t.z, t.w); } } setPhysicsBodyTransformation(e, t, r) { e.physicsBody.position.set(t.x, t.y, t.z), e.physicsBody.quaternion.set(r.x, r.y, r.z, r.w); } isSupported() { return this.BJSCANNON !== void 0; } setLinearVelocity(e, t) { e.physicsBody.velocity.set(t.x, t.y, t.z); } setAngularVelocity(e, t) { e.physicsBody.angularVelocity.set(t.x, t.y, t.z); } getLinearVelocity(e) { const t = e.physicsBody.velocity; return t ? new S(t.x, t.y, t.z) : null; } getAngularVelocity(e) { const t = e.physicsBody.angularVelocity; return t ? new S(t.x, t.y, t.z) : null; } setBodyMass(e, t) { e.physicsBody.mass = t, e.physicsBody.updateMassProperties(); } getBodyMass(e) { return e.physicsBody.mass; } getBodyFriction(e) { return e.physicsBody.material.friction; } setBodyFriction(e, t) { e.physicsBody.material.friction = t; } getBodyRestitution(e) { return e.physicsBody.material.restitution; } setBodyRestitution(e, t) { e.physicsBody.material.restitution = t; } sleepBody(e) { e.physicsBody.sleep(); } wakeUpBody(e) { e.physicsBody.wakeUp(); } updateDistanceJoint(e, t) { e.physicsJoint.distance = t; } setMotor(e, t, r, n) { n || (e.physicsJoint.enableMotor(), e.physicsJoint.setMotorSpeed(t), r && this.setLimit(e, r)); } setLimit(e, t, r) { e.physicsJoint.motorEquation.maxForce = r, e.physicsJoint.motorEquation.minForce = t === void 0 ? -t : t; } syncMeshWithImpostor(e, t) { const r = t.physicsBody; e.position.x = r.position.x, e.position.y = r.position.y, e.position.z = r.position.z, e.rotationQuaternion && (e.rotationQuaternion.x = r.quaternion.x, e.rotationQuaternion.y = r.quaternion.y, e.rotationQuaternion.z = r.quaternion.z, e.rotationQuaternion.w = r.quaternion.w); } getRadius(e) { return e.physicsBody.shapes[0].boundingSphereRadius; } getBoxSizeToRef(e, t) { const r = e.physicsBody.shapes[0]; t.x = r.halfExtents.x * 2, t.y = r.halfExtents.y * 2, t.z = r.halfExtents.z * 2; } dispose() { } _extendNamespace() { const e = new this.BJSCANNON.Vec3(), t = this.BJSCANNON; this.BJSCANNON.World.prototype.step = function(r, n, i) { if (i = i || 10, n = n || 0, n === 0) this.internalStep(r), this.time += r; else { let s = Math.floor((this.time + n) / r) - Math.floor(this.time / r); s = Math.min(s, i) || 1; const a = performance.now(); for (let u = 0; u !== s && (this.internalStep(r), !(performance.now() - a > r * 1e3)); u++) ; this.time += n; const o = this.time % r / r, d = e, v = this.bodies; for (let u = 0; u !== v.length; u++) { const l = v[u]; l.type !== t.Body.STATIC && l.sleepState !== t.Body.SLEEPING ? (l.position.vsub(l.previousPosition, d), d.scale(o, d), l.position.vadd(d, l.interpolatedPosition)) : (l.interpolatedPosition.set(l.position.x, l.position.y, l.position.z), l.interpolatedQuaternion.set(l.quaternion.x, l.quaternion.y, l.quaternion.z, l.quaternion.w)); } } }; } /** * Does a raycast in the physics world * @param from when should the ray start? * @param to when should the ray end? * @returns PhysicsRaycastResult */ raycast(e, t) { return this._raycastResult.reset(e, t), this.raycastToRef(e, t, this._raycastResult), this._raycastResult; } /** * Does a raycast in the physics world * @param from when should the ray start? * @param to when should the ray end? * @param result resulting PhysicsRaycastResult */ raycastToRef(e, t, r) { this._cannonRaycastResult.reset(), this.world.raycastClosest(e, t, {}, this._cannonRaycastResult), r.reset(e, t), this._cannonRaycastResult.hasHit && (r.setHitData({ x: this._cannonRaycastResult.hitNormalWorld.x, y: this._cannonRaycastResult.hitNormalWorld.y, z: this._cannonRaycastResult.hitNormalWorld.z }, { x: this._cannonRaycastResult.hitPointWorld.x, y: this._cannonRaycastResult.hitPointWorld.y, z: this._cannonRaycastResult.hitPointWorld.z }), r.setHitDistance(this._cannonRaycastResult.distance)); } } $Q.DefaultPluginFactory = () => new sO(); class gF { constructor(e = !0, t, r = OIMO) { this._useDeltaForWorldStep = e, this.name = "OimoJSPlugin", this._fixedTimeStep = 1 / 60, this._tmpImpostorsArray = [], this._tmpPositionVector = S.Zero(), this.BJSOIMO = r, this.world = new this.BJSOIMO.World({ iterations: t }), this.world.clear(), this._raycastResult = new iV(); } /** * * @returns plugin version */ getPluginVersion() { return 1; } setGravity(e) { this.world.gravity.set(e.x, e.y, e.z); } setTimeStep(e) { this.world.timeStep = e; } getTimeStep() { return this.world.timeStep; } executeStep(e, t) { t.forEach(function(n) { n.beforeStep(); }), this.world.timeStep = this._useDeltaForWorldStep ? e : this._fixedTimeStep, this.world.step(), t.forEach((n) => { n.afterStep(), this._tmpImpostorsArray[n.uniqueId] = n; }); let r = this.world.contacts; for (; r !== null; ) { if (r.touching && !r.body1.sleeping && !r.body2.sleeping) { r = r.next; continue; } const n = this._tmpImpostorsArray[+r.body1.name], i = this._tmpImpostorsArray[+r.body2.name]; if (!n || !i) { r = r.next; continue; } n.onCollide({ body: i.physicsBody, point: null, distance: 0, impulse: 0, normal: null }), i.onCollide({ body: n.physicsBody, point: null, distance: 0, impulse: 0, normal: null }), r = r.next; } } applyImpulse(e, t, r) { const n = e.physicsBody.mass; e.physicsBody.applyImpulse(r.scale(this.world.invScale), t.scale(this.world.invScale * n)); } applyForce(e, t, r) { Se.Warn("Oimo doesn't support applying force. Using impulse instead."), this.applyImpulse(e, t, r); } generatePhysicsBody(e) { if (e.parent) { e.physicsBody && (this.removePhysicsBody(e), e.forceUpdate()); return; } if (e.isBodyInitRequired()) { const t = { name: e.uniqueId, //Oimo must have mass, also for static objects. config: [e.getParam("mass") || 1e-3, e.getParam("friction"), e.getParam("restitution")], size: [], type: [], pos: [], posShape: [], rot: [], rotShape: [], move: e.getParam("mass") !== 0, density: e.getParam("mass"), friction: e.getParam("friction"), restitution: e.getParam("restitution"), //Supporting older versions of Oimo world: this.world }, r = [e]; ((a) => { a.getChildMeshes && a.getChildMeshes().forEach(function(f) { f.physicsImpostor && r.push(f.physicsImpostor); }); })(e.object); const i = (a) => Math.max(a, Dn), s = new Ze(); r.forEach((a) => { if (!a.object.rotationQuaternion) return; const f = a.object.rotationQuaternion; s.copyFrom(f), a.object.rotationQuaternion.set(0, 0, 0, 1), a.object.computeWorldMatrix(!0); const o = s.toEulerAngles(), d = a.getObjectExtents(), v = 57.29577951308232; if (a === e) { const u = e.getObjectCenter(); e.object.getAbsolutePivotPoint().subtractToRef(u, this._tmpPositionVector), this._tmpPositionVector.divideInPlace(e.object.scaling), t.pos.push(u.x), t.pos.push(u.y), t.pos.push(u.z), t.posShape.push(0, 0, 0), t.rotShape.push(0, 0, 0); } else { const u = a.object.position.clone(); t.posShape.push(u.x), t.posShape.push(u.y), t.posShape.push(u.z), t.rotShape.push(o.x * v, o.y * v, o.z * v); } switch (a.object.rotationQuaternion.copyFrom(s), a.type) { case tn.ParticleImpostor: Se.Warn("No Particle support in OIMO.js. using SphereImpostor instead"); case tn.SphereImpostor: { const u = d.x, l = d.y, P = d.z, p = Math.max(i(u), i(l), i(P)) / 2; t.type.push("sphere"), t.size.push(p), t.size.push(p), t.size.push(p); break; } case tn.CylinderImpostor: { const u = i(d.x) / 2, l = i(d.y); t.type.push("cylinder"), t.size.push(u), t.size.push(l), t.size.push(l); break; } case tn.PlaneImpostor: case tn.BoxImpostor: default: { const u = i(d.x), l = i(d.y), P = i(d.z); t.type.push("box"), t.size.push(u), t.size.push(l), t.size.push(P); break; } } a.object.rotationQuaternion = f; }), e.physicsBody = this.world.add(t), e.physicsBody.resetQuaternion(s), e.physicsBody.updatePosition(0); } else this._tmpPositionVector.copyFromFloats(0, 0, 0); e.setDeltaPosition(this._tmpPositionVector); } removePhysicsBody(e) { this.world.removeRigidBody(e.physicsBody); } generateJoint(e) { const t = e.mainImpostor.physicsBody, r = e.connectedImpostor.physicsBody; if (!t || !r) return; const n = e.joint.jointData, i = n.nativeParams || {}; let s; const a = { body1: t, body2: r, axe1: i.axe1 || (n.mainAxis ? n.mainAxis.asArray() : null), axe2: i.axe2 || (n.connectedAxis ? n.connectedAxis.asArray() : null), pos1: i.pos1 || (n.mainPivot ? n.mainPivot.asArray() : null), pos2: i.pos2 || (n.connectedPivot ? n.connectedPivot.asArray() : null), min: i.min, max: i.max, collision: i.collision || n.collision, spring: i.spring, //supporting older version of Oimo world: this.world }; switch (e.joint.type) { case ta.BallAndSocketJoint: s = "jointBall"; break; case ta.SpringJoint: { Se.Warn("OIMO.js doesn't support Spring Constraint. Simulating using DistanceJoint instead"); const f = n; a.min = f.length || a.min, a.max = Math.max(a.min, a.max); } case ta.DistanceJoint: s = "jointDistance", a.max = n.maxDistance; break; case ta.PrismaticJoint: s = "jointPrisme"; break; case ta.SliderJoint: s = "jointSlide"; break; case ta.WheelJoint: s = "jointWheel"; break; case ta.HingeJoint: default: s = "jointHinge"; break; } a.type = s, e.joint.physicsJoint = this.world.add(a); } removeJoint(e) { try { this.world.removeJoint(e.joint.physicsJoint); } catch (t) { Se.Warn(t); } } isSupported() { return this.BJSOIMO !== void 0; } setTransformationFromPhysicsBody(e) { if (!e.physicsBody.sleeping) { if (e.physicsBody.shapes.next) { let t = e.physicsBody.shapes; for (; t.next; ) t = t.next; e.object.position.set(t.position.x, t.position.y, t.position.z); } else { const t = e.physicsBody.getPosition(); e.object.position.set(t.x, t.y, t.z); } if (e.object.rotationQuaternion) { const t = e.physicsBody.getQuaternion(); e.object.rotationQuaternion.set(t.x, t.y, t.z, t.w); } } } setPhysicsBodyTransformation(e, t, r) { const n = e.physicsBody; e.physicsBody.shapes.next || (n.position.set(t.x, t.y, t.z), n.orientation.set(r.x, r.y, r.z, r.w), n.syncShapes(), n.awake()); } /*private _getLastShape(body: any): any { var lastShape = body.shapes; while (lastShape.next) { lastShape = lastShape.next; } return lastShape; }*/ setLinearVelocity(e, t) { e.physicsBody.linearVelocity.set(t.x, t.y, t.z); } setAngularVelocity(e, t) { e.physicsBody.angularVelocity.set(t.x, t.y, t.z); } getLinearVelocity(e) { const t = e.physicsBody.linearVelocity; return t ? new S(t.x, t.y, t.z) : null; } getAngularVelocity(e) { const t = e.physicsBody.angularVelocity; return t ? new S(t.x, t.y, t.z) : null; } setBodyMass(e, t) { const r = t === 0; e.physicsBody.shapes.density = r ? 1 : t, e.physicsBody.setupMass(r ? 2 : 1); } getBodyMass(e) { return e.physicsBody.shapes.density; } getBodyFriction(e) { return e.physicsBody.shapes.friction; } setBodyFriction(e, t) { e.physicsBody.shapes.friction = t; } getBodyRestitution(e) { return e.physicsBody.shapes.restitution; } setBodyRestitution(e, t) { e.physicsBody.shapes.restitution = t; } sleepBody(e) { e.physicsBody.sleep(); } wakeUpBody(e) { e.physicsBody.awake(); } updateDistanceJoint(e, t, r) { e.physicsJoint.limitMotor.upperLimit = t, r !== void 0 && (e.physicsJoint.limitMotor.lowerLimit = r); } setMotor(e, t, r, n) { r !== void 0 ? Se.Warn("OimoJS plugin currently has unexpected behavior when using setMotor with force parameter") : r = 1e6, t *= -1; const i = n ? e.physicsJoint.rotationalLimitMotor2 : e.physicsJoint.rotationalLimitMotor1 || e.physicsJoint.rotationalLimitMotor || e.physicsJoint.limitMotor; i && i.setMotor(t, r); } setLimit(e, t, r, n) { const i = n ? e.physicsJoint.rotationalLimitMotor2 : e.physicsJoint.rotationalLimitMotor1 || e.physicsJoint.rotationalLimitMotor || e.physicsJoint.limitMotor; i && i.setLimit(t, r === void 0 ? -t : r); } syncMeshWithImpostor(e, t) { const r = t.physicsBody; e.position.x = r.position.x, e.position.y = r.position.y, e.position.z = r.position.z, e.rotationQuaternion && (e.rotationQuaternion.x = r.orientation.x, e.rotationQuaternion.y = r.orientation.y, e.rotationQuaternion.z = r.orientation.z, e.rotationQuaternion.w = r.orientation.w); } getRadius(e) { return e.physicsBody.shapes.radius; } getBoxSizeToRef(e, t) { const r = e.physicsBody.shapes; t.x = r.halfWidth * 2, t.y = r.halfHeight * 2, t.z = r.halfDepth * 2; } dispose() { this.world.clear(); } /** * Does a raycast in the physics world * @param from when should the ray start? * @param to when should the ray end? * @returns PhysicsRaycastResult */ raycast(e, t) { return Se.Warn("raycast is not currently supported by the Oimo physics plugin"), this._raycastResult.reset(e, t), this._raycastResult; } /** * Does a raycast in the physics world * @param from when should the ray start? * @param to when should the ray end? * @param result resulting PhysicsRaycastResult */ raycastToRef(e, t, r) { Se.Warn("raycast is not currently supported by the Oimo physics plugin"), r.reset(e, t); } } class eg { /** * Initializes the ammoJS plugin * @param _useDeltaForWorldStep if the time between frames should be used when calculating physics steps (Default: true) * @param ammoInjection can be used to inject your own ammo reference * @param overlappingPairCache can be used to specify your own overlapping pair cache */ constructor(e = !0, t = Ammo, r = null) { if (this._useDeltaForWorldStep = e, this.bjsAMMO = {}, this.name = "AmmoJSPlugin", this._timeStep = 1 / 60, this._fixedTimeStep = 1 / 60, this._maxSteps = 5, this._tmpQuaternion = new Ze(), this._tmpContactCallbackResult = !1, this._tmpContactPoint = new S(), this._tmpContactNormal = new S(), this._tmpVec3 = new S(), this._tmpMatrix = new he(), typeof t == "function") { Se.Error("AmmoJS is not ready. Please make sure you await Ammo() before using the plugin."); return; } else this.bjsAMMO = t; if (!this.isSupported()) { Se.Error("AmmoJS is not available. Please make sure you included the js file."); return; } this._collisionConfiguration = new this.bjsAMMO.btSoftBodyRigidBodyCollisionConfiguration(), this._dispatcher = new this.bjsAMMO.btCollisionDispatcher(this._collisionConfiguration), this._overlappingPairCache = r || new this.bjsAMMO.btDbvtBroadphase(), this._solver = new this.bjsAMMO.btSequentialImpulseConstraintSolver(), this._softBodySolver = new this.bjsAMMO.btDefaultSoftBodySolver(), this.world = new this.bjsAMMO.btSoftRigidDynamicsWorld(this._dispatcher, this._overlappingPairCache, this._solver, this._collisionConfiguration, this._softBodySolver), this._tmpAmmoConcreteContactResultCallback = new this.bjsAMMO.ConcreteContactResultCallback(), this._tmpAmmoConcreteContactResultCallback.addSingleResult = (n) => { n = this.bjsAMMO.wrapPointer(n, this.bjsAMMO.btManifoldPoint); const i = n.getPositionWorldOnA(), s = n.m_normalWorldOnB; this._tmpContactPoint.x = i.x(), this._tmpContactPoint.y = i.y(), this._tmpContactPoint.z = i.z(), this._tmpContactNormal.x = s.x(), this._tmpContactNormal.y = s.y(), this._tmpContactNormal.z = s.z(), this._tmpContactImpulse = n.getAppliedImpulse(), this._tmpContactDistance = n.getDistance(), this._tmpContactCallbackResult = !0; }, this._raycastResult = new iV(), this._tmpAmmoTransform = new this.bjsAMMO.btTransform(), this._tmpAmmoTransform.setIdentity(), this._tmpAmmoQuaternion = new this.bjsAMMO.btQuaternion(0, 0, 0, 1), this._tmpAmmoVectorA = new this.bjsAMMO.btVector3(0, 0, 0), this._tmpAmmoVectorB = new this.bjsAMMO.btVector3(0, 0, 0), this._tmpAmmoVectorC = new this.bjsAMMO.btVector3(0, 0, 0), this._tmpAmmoVectorD = new this.bjsAMMO.btVector3(0, 0, 0); } /** * * @returns plugin version */ getPluginVersion() { return 1; } /** * Sets the gravity of the physics world (m/(s^2)) * @param gravity Gravity to set */ setGravity(e) { this._tmpAmmoVectorA.setValue(e.x, e.y, e.z), this.world.setGravity(this._tmpAmmoVectorA), this.world.getWorldInfo().set_m_gravity(this._tmpAmmoVectorA); } /** * Amount of time to step forward on each frame (only used if useDeltaForWorldStep is false in the constructor) * @param timeStep timestep to use in seconds */ setTimeStep(e) { this._timeStep = e; } /** * Increment to step forward in the physics engine (If timeStep is set to 1/60 and fixedTimeStep is set to 1/120 the physics engine should run 2 steps per frame) (Default: 1/60) * @param fixedTimeStep fixedTimeStep to use in seconds */ setFixedTimeStep(e) { this._fixedTimeStep = e; } /** * Sets the maximum number of steps by the physics engine per frame (Default: 5) * @param maxSteps the maximum number of steps by the physics engine per frame */ setMaxSteps(e) { this._maxSteps = e; } /** * Gets the current timestep (only used if useDeltaForWorldStep is false in the constructor) * @returns the current timestep in seconds */ getTimeStep() { return this._timeStep; } // Ammo's contactTest and contactPairTest take a callback that runs synchronously, wrap them so that they are easier to consume _isImpostorInContact(e) { return this._tmpContactCallbackResult = !1, this.world.contactTest(e.physicsBody, this._tmpAmmoConcreteContactResultCallback), this._tmpContactCallbackResult; } // Ammo's collision events have some weird quirks // contactPairTest fires too many events as it fires events even when objects are close together but contactTest does not // so only fire event if both contactTest and contactPairTest have a hit _isImpostorPairInContact(e, t) { return this._tmpContactCallbackResult = !1, this.world.contactPairTest(e.physicsBody, t.physicsBody, this._tmpAmmoConcreteContactResultCallback), this._tmpContactCallbackResult; } // Ammo's behavior when maxSteps > 0 does not behave as described in docs // @see http://www.bulletphysics.org/mediawiki-1.5.8/index.php/Stepping_The_World // // When maxSteps is 0 do the entire simulation in one step // When maxSteps is > 0, run up to maxStep times, if on the last step the (remaining step - fixedTimeStep) is < fixedTimeStep, the remainder will be used for the step. (eg. if remainder is 1.001 and fixedTimeStep is 1 the last step will be 1.001, if instead it did 2 steps (1, 0.001) issues occuered when having a tiny step in ammo) // Note: To get deterministic physics, timeStep would always need to be divisible by fixedTimeStep _stepSimulation(e = 1 / 60, t = 10, r = 1 / 60) { if (t == 0) this.world.stepSimulation(e, 0); else for (; t > 0 && e > 0; ) e - r < r ? (this.world.stepSimulation(e, 0), e = 0) : (e -= r, this.world.stepSimulation(r, 0)), t--; } /** * Moves the physics simulation forward delta seconds and updates the given physics imposters * Prior to the step the imposters physics location is set to the position of the babylon meshes * After the step the babylon meshes are set to the position of the physics imposters * @param delta amount of time to step forward * @param impostors array of imposters to update before/after the step */ executeStep(e, t) { for (const r of t) r.soft || r.beforeStep(); this._stepSimulation(this._useDeltaForWorldStep ? e : this._timeStep, this._maxSteps, this._fixedTimeStep); for (const r of t) if (r.soft ? this._afterSoftStep(r) : r.afterStep(), r._onPhysicsCollideCallbacks.length > 0 && this._isImpostorInContact(r)) for (const n of r._onPhysicsCollideCallbacks) for (const i of n.otherImpostors) (r.physicsBody.isActive() || i.physicsBody.isActive()) && this._isImpostorPairInContact(r, i) && (r.onCollide({ body: i.physicsBody, point: this._tmpContactPoint, distance: this._tmpContactDistance, impulse: this._tmpContactImpulse, normal: this._tmpContactNormal }), i.onCollide({ body: r.physicsBody, point: this._tmpContactPoint, distance: this._tmpContactDistance, impulse: this._tmpContactImpulse, normal: this._tmpContactNormal })); } /** * Update babylon mesh to match physics world object * @param impostor imposter to match */ _afterSoftStep(e) { e.type === tn.RopeImpostor ? this._ropeStep(e) : this._softbodyOrClothStep(e); } /** * Update babylon mesh vertices vertices to match physics world softbody or cloth * @param impostor imposter to match */ _ropeStep(e) { const t = e.physicsBody.get_m_nodes(), r = t.size(); let n, i, s, a, f; const o = new Array(); for (let u = 0; u < r; u++) n = t.at(u), i = n.get_m_x(), s = i.x(), a = i.y(), f = i.z(), o.push(new S(s, a, f)); const d = e.object, v = e.getParam("shape"); e._isFromLine ? e.object = ka("lines", { points: o, instance: d }) : e.object = GR("ext", { shape: v, path: o, instance: d }); } /** * Update babylon mesh vertices vertices to match physics world softbody or cloth * @param impostor imposter to match */ _softbodyOrClothStep(e) { const t = e.type === tn.ClothImpostor ? 1 : -1, r = e.object; let n = r.getVerticesData(J.PositionKind); n || (n = []); let i = r.getVerticesData(J.NormalKind); i || (i = []); const s = n.length / 3, a = e.physicsBody.get_m_nodes(); let f, o, d, v, u, l, P, p; for (let H = 0; H < s; H++) { f = a.at(H), o = f.get_m_x(), d = o.x(), v = o.y(), u = o.z() * t; const T = f.get_m_n(); l = T.x(), P = T.y(), p = T.z() * t, n[3 * H] = d, n[3 * H + 1] = v, n[3 * H + 2] = u, i[3 * H] = l, i[3 * H + 1] = P, i[3 * H + 2] = p; } const c = new Ut(); c.positions = n, c.normals = i, c.uvs = r.getVerticesData(J.UVKind), c.colors = r.getVerticesData(J.ColorKind), r && r.getIndices && (c.indices = r.getIndices()), c.applyToMesh(r); } /** * Applies an impulse on the imposter * @param impostor imposter to apply impulse to * @param force amount of force to be applied to the imposter * @param contactPoint the location to apply the impulse on the imposter */ applyImpulse(e, t, r) { if (e.soft) Se.Warn("Cannot be applied to a soft body"); else { e.physicsBody.activate(); const n = this._tmpAmmoVectorA, i = this._tmpAmmoVectorB; e.object && e.object.getWorldMatrix && r.subtractInPlace(e.object.getWorldMatrix().getTranslation()), n.setValue(r.x, r.y, r.z), i.setValue(t.x, t.y, t.z), e.physicsBody.applyImpulse(i, n); } } /** * Applies a force on the imposter * @param impostor imposter to apply force * @param force amount of force to be applied to the imposter * @param contactPoint the location to apply the force on the imposter */ applyForce(e, t, r) { if (e.soft) Se.Warn("Cannot be applied to a soft body"); else { e.physicsBody.activate(); const n = this._tmpAmmoVectorA, i = this._tmpAmmoVectorB; if (e.object && e.object.getWorldMatrix) { const s = e.object.getWorldMatrix().getTranslation(); n.setValue(r.x - s.x, r.y - s.y, r.z - s.z); } else n.setValue(r.x, r.y, r.z); i.setValue(t.x, t.y, t.z), e.physicsBody.applyForce(i, n); } } /** * Creates a physics body using the plugin * @param impostor the imposter to create the physics body on */ generatePhysicsBody(e) { if (e._pluginData.toDispose = [], e.parent) { e.physicsBody && (this.removePhysicsBody(e), e.forceUpdate()); return; } if (e.isBodyInitRequired()) { const t = this._createShape(e), r = e.getParam("mass"); if (e._pluginData.mass = r, e.soft) t.get_m_cfg().set_collisions(17), t.get_m_cfg().set_kDP(e.getParam("damping")), this.bjsAMMO.castObject(t, this.bjsAMMO.btCollisionObject).getCollisionShape().setMargin(e.getParam("margin")), t.setActivationState(eg._DISABLE_DEACTIVATION_FLAG), this.world.addSoftBody(t, 1, -1), e.physicsBody = t, e._pluginData.toDispose.push(t), this.setBodyPressure(e, 0), e.type === tn.SoftbodyImpostor && this.setBodyPressure(e, e.getParam("pressure")), this.setBodyStiffness(e, e.getParam("stiffness")), this.setBodyVelocityIterations(e, e.getParam("velocityIterations")), this.setBodyPositionIterations(e, e.getParam("positionIterations")); else { const n = new this.bjsAMMO.btVector3(0, 0, 0), i = new this.bjsAMMO.btTransform(); e.object.computeWorldMatrix(!0), i.setIdentity(), r !== 0 && t.calculateLocalInertia(r, n), this._tmpAmmoVectorA.setValue(e.object.position.x, e.object.position.y, e.object.position.z), this._tmpAmmoQuaternion.setValue(e.object.rotationQuaternion.x, e.object.rotationQuaternion.y, e.object.rotationQuaternion.z, e.object.rotationQuaternion.w), i.setOrigin(this._tmpAmmoVectorA), i.setRotation(this._tmpAmmoQuaternion); const s = new this.bjsAMMO.btDefaultMotionState(i), a = new this.bjsAMMO.btRigidBodyConstructionInfo(r, s, t, n), f = new this.bjsAMMO.btRigidBody(a); if (r === 0 && (f.setCollisionFlags(f.getCollisionFlags() | eg._KINEMATIC_FLAG), f.setActivationState(eg._DISABLE_DEACTIVATION_FLAG)), e.type == tn.NoImpostor && !t.getChildShape && f.setCollisionFlags(f.getCollisionFlags() | eg._DISABLE_COLLISION_FLAG), e.type !== tn.MeshImpostor && e.type !== tn.NoImpostor) { const v = e.object.getBoundingInfo(); this._tmpVec3.copyFrom(e.object.getAbsolutePosition()), this._tmpVec3.subtractInPlace(v.boundingBox.centerWorld), this._tmpVec3.x /= e.object.scaling.x, this._tmpVec3.y /= e.object.scaling.y, this._tmpVec3.z /= e.object.scaling.z, e.setDeltaPosition(this._tmpVec3); } const o = e.getParam("group"), d = e.getParam("mask"); o && d ? this.world.addRigidBody(f, o, d) : this.world.addRigidBody(f), e.physicsBody = f, e._pluginData.toDispose = e._pluginData.toDispose.concat([f, a, s, i, n, t]); } this.setBodyRestitution(e, e.getParam("restitution")), this.setBodyFriction(e, e.getParam("friction")); } } /** * Removes the physics body from the imposter and disposes of the body's memory * @param impostor imposter to remove the physics body from */ removePhysicsBody(e) { this.world && (e.soft ? this.world.removeSoftBody(e.physicsBody) : this.world.removeRigidBody(e.physicsBody), e._pluginData && (e._pluginData.toDispose.forEach((t) => { this.bjsAMMO.destroy(t); }), e._pluginData.toDispose = [])); } /** * Generates a joint * @param impostorJoint the imposter joint to create the joint with */ generateJoint(e) { const t = e.mainImpostor.physicsBody, r = e.connectedImpostor.physicsBody; if (!t || !r) return; const n = e.joint.jointData; n.mainPivot || (n.mainPivot = new S(0, 0, 0)), n.connectedPivot || (n.connectedPivot = new S(0, 0, 0)); let i; switch (e.joint.type) { case ta.DistanceJoint: { const s = n.maxDistance; s && (n.mainPivot = new S(0, -s / 2, 0), n.connectedPivot = new S(0, s / 2, 0)), i = new this.bjsAMMO.btPoint2PointConstraint(t, r, new this.bjsAMMO.btVector3(n.mainPivot.x, n.mainPivot.y, n.mainPivot.z), new this.bjsAMMO.btVector3(n.connectedPivot.x, n.connectedPivot.y, n.connectedPivot.z)); break; } case ta.HingeJoint: { n.mainAxis || (n.mainAxis = new S(0, 0, 0)), n.connectedAxis || (n.connectedAxis = new S(0, 0, 0)); const s = new this.bjsAMMO.btVector3(n.mainAxis.x, n.mainAxis.y, n.mainAxis.z), a = new this.bjsAMMO.btVector3(n.connectedAxis.x, n.connectedAxis.y, n.connectedAxis.z); i = new this.bjsAMMO.btHingeConstraint(t, r, new this.bjsAMMO.btVector3(n.mainPivot.x, n.mainPivot.y, n.mainPivot.z), new this.bjsAMMO.btVector3(n.connectedPivot.x, n.connectedPivot.y, n.connectedPivot.z), s, a); break; } case ta.BallAndSocketJoint: i = new this.bjsAMMO.btPoint2PointConstraint(t, r, new this.bjsAMMO.btVector3(n.mainPivot.x, n.mainPivot.y, n.mainPivot.z), new this.bjsAMMO.btVector3(n.connectedPivot.x, n.connectedPivot.y, n.connectedPivot.z)); break; default: Se.Warn("JointType not currently supported by the Ammo plugin, falling back to PhysicsJoint.BallAndSocketJoint"), i = new this.bjsAMMO.btPoint2PointConstraint(t, r, new this.bjsAMMO.btVector3(n.mainPivot.x, n.mainPivot.y, n.mainPivot.z), new this.bjsAMMO.btVector3(n.connectedPivot.x, n.connectedPivot.y, n.connectedPivot.z)); break; } this.world.addConstraint(i, !e.joint.jointData.collision), e.joint.physicsJoint = i; } /** * Removes a joint * @param impostorJoint the imposter joint to remove the joint from */ removeJoint(e) { this.world && this.world.removeConstraint(e.joint.physicsJoint); } // adds all verticies (including child verticies) to the triangle mesh _addMeshVerts(e, t, r) { let n = 0; if (r && r.getIndices && r.getWorldMatrix && r.getChildMeshes) { let i = r.getIndices(); i || (i = []); let s = r.getVerticesData(J.PositionKind); s || (s = []); let a; if (t && t !== r) { let o; t.rotationQuaternion ? o = t.rotationQuaternion : t.rotation ? o = Ze.FromEulerAngles(t.rotation.x, t.rotation.y, t.rotation.z) : o = Ze.Identity(), he.Compose(S.One(), o, t.position).invertToRef(this._tmpMatrix), a = r.computeWorldMatrix(!1).multiply(this._tmpMatrix); } else he.ScalingToRef(r.scaling.x, r.scaling.y, r.scaling.z, this._tmpMatrix), a = this._tmpMatrix; const f = i.length / 3; for (let o = 0; o < f; o++) { const d = []; for (let v = 0; v < 3; v++) { let u = new S(s[i[o * 3 + v] * 3 + 0], s[i[o * 3 + v] * 3 + 1], s[i[o * 3 + v] * 3 + 2]); u = S.TransformCoordinates(u, a); let l; v == 0 ? l = this._tmpAmmoVectorA : v == 1 ? l = this._tmpAmmoVectorB : l = this._tmpAmmoVectorC, l.setValue(u.x, u.y, u.z), d.push(l); } e.addTriangle(d[0], d[1], d[2]), n++; } r.getChildMeshes().forEach((o) => { n += this._addMeshVerts(e, t, o); }); } return n; } /** * Initialise the soft body vertices to match its object's (mesh) vertices * Softbody vertices (nodes) are in world space and to match this * The object's position and rotation is set to zero and so its vertices are also then set in world space * @param impostor to create the softbody for */ _softVertexData(e) { const t = e.object; if (t && t.getIndices && t.getWorldMatrix && t.getChildMeshes) { t.getIndices(); let r = t.getVerticesData(J.PositionKind); r || (r = []); let n = t.getVerticesData(J.NormalKind); n || (n = []), t.computeWorldMatrix(!1); const i = [], s = []; for (let f = 0; f < r.length; f += 3) { let o = new S(r[f], r[f + 1], r[f + 2]), d = new S(n[f], n[f + 1], n[f + 2]); o = S.TransformCoordinates(o, t.getWorldMatrix()), d = S.TransformNormal(d, t.getWorldMatrix()), i.push(o.x, o.y, o.z), s.push(d.x, d.y, d.z); } const a = new Ut(); return a.positions = i, a.normals = s, a.uvs = t.getVerticesData(J.UVKind), a.colors = t.getVerticesData(J.ColorKind), t && t.getIndices && (a.indices = t.getIndices()), a.applyToMesh(t), t.position = S.Zero(), t.rotationQuaternion = null, t.rotation = S.Zero(), t.computeWorldMatrix(!0), a; } return Ut.ExtractFromMesh(t); } /** * Create an impostor's soft body * @param impostor to create the softbody for */ _createSoftbody(e) { const t = e.object; if (t && t.getIndices) { let r = t.getIndices(); r || (r = []); const n = this._softVertexData(e), i = n.positions, s = n.normals; if (i === null || s === null) return new this.bjsAMMO.btCompoundShape(); { const a = [], f = []; for (let P = 0; P < i.length; P += 3) { const p = new S(i[P], i[P + 1], i[P + 2]), c = new S(s[P], s[P + 1], s[P + 2]); a.push(p.x, p.y, -p.z), f.push(c.x, c.y, -c.z); } const o = new this.bjsAMMO.btSoftBodyHelpers().CreateFromTriMesh(this.world.getWorldInfo(), a, t.getIndices(), r.length / 3, !0), d = i.length / 3, v = o.get_m_nodes(); let u, l; for (let P = 0; P < d; P++) u = v.at(P), l = u.get_m_n(), l.setX(f[3 * P]), l.setY(f[3 * P + 1]), l.setZ(f[3 * P + 2]); return o; } } } /** * Create cloth for an impostor * @param impostor to create the softbody for */ _createCloth(e) { const t = e.object; if (t && t.getIndices) { t.getIndices(); const r = this._softVertexData(e), n = r.positions, i = r.normals; if (n === null || i === null) return new this.bjsAMMO.btCompoundShape(); { const s = n.length, a = Math.sqrt(s / 3); e.segments = a; const f = a - 1; return this._tmpAmmoVectorA.setValue(n[0], n[1], n[2]), this._tmpAmmoVectorB.setValue(n[3 * f], n[3 * f + 1], n[3 * f + 2]), this._tmpAmmoVectorD.setValue(n[s - 3], n[s - 2], n[s - 1]), this._tmpAmmoVectorC.setValue(n[s - 3 - 3 * f], n[s - 2 - 3 * f], n[s - 1 - 3 * f]), new this.bjsAMMO.btSoftBodyHelpers().CreatePatch(this.world.getWorldInfo(), this._tmpAmmoVectorA, this._tmpAmmoVectorB, this._tmpAmmoVectorC, this._tmpAmmoVectorD, a, a, e.getParam("fixedPoints"), !0); } } } /** * Create rope for an impostor * @param impostor to create the softbody for */ _createRope(e) { let t, r; const n = this._softVertexData(e), i = n.positions, s = n.normals; if (i === null || s === null) return new this.bjsAMMO.btCompoundShape(); n.applyToMesh(e.object, !0), e._isFromLine = !0; const a = s.map((u) => u * u), f = (u, l) => u + l; if (a.reduce(f) === 0) t = i.length, r = t / 3 - 1, this._tmpAmmoVectorA.setValue(i[0], i[1], i[2]), this._tmpAmmoVectorB.setValue(i[t - 3], i[t - 2], i[t - 1]); else { e._isFromLine = !1; const u = e.getParam("path"); if (e.getParam("shape") === null) return Se.Warn("No shape available for extruded mesh"), new this.bjsAMMO.btCompoundShape(); t = u.length, r = t - 1, this._tmpAmmoVectorA.setValue(u[0].x, u[0].y, u[0].z), this._tmpAmmoVectorB.setValue(u[t - 1].x, u[t - 1].y, u[t - 1].z); } e.segments = r; let d = e.getParam("fixedPoints"); d = d > 3 ? 3 : d; const v = new this.bjsAMMO.btSoftBodyHelpers().CreateRope(this.world.getWorldInfo(), this._tmpAmmoVectorA, this._tmpAmmoVectorB, r - 1, d); return v.get_m_cfg().set_collisions(17), v; } /** * Create a custom physics impostor shape using the plugin's onCreateCustomShape handler * @param impostor to create the custom physics shape for */ _createCustom(e) { let t = null; return this.onCreateCustomShape && (t = this.onCreateCustomShape(e)), t == null && (t = new this.bjsAMMO.btCompoundShape()), t; } // adds all verticies (including child verticies) to the convex hull shape _addHullVerts(e, t, r) { let n = 0; if (r && r.getIndices && r.getWorldMatrix && r.getChildMeshes) { let i = r.getIndices(); i || (i = []); let s = r.getVerticesData(J.PositionKind); s || (s = []), r.computeWorldMatrix(!1); const a = i.length / 3; for (let f = 0; f < a; f++) { const o = []; for (let d = 0; d < 3; d++) { let v = new S(s[i[f * 3 + d] * 3 + 0], s[i[f * 3 + d] * 3 + 1], s[i[f * 3 + d] * 3 + 2]); he.ScalingToRef(r.scaling.x, r.scaling.y, r.scaling.z, this._tmpMatrix), v = S.TransformCoordinates(v, this._tmpMatrix); let u; d == 0 ? u = this._tmpAmmoVectorA : d == 1 ? u = this._tmpAmmoVectorB : u = this._tmpAmmoVectorC, u.setValue(v.x, v.y, v.z), o.push(u); } e.addPoint(o[0], !0), e.addPoint(o[1], !0), e.addPoint(o[2], !0), n++; } r.getChildMeshes().forEach((f) => { n += this._addHullVerts(e, t, f); }); } return n; } _createShape(e, t = !1) { const r = e.object; let n; const i = e.getObjectExtents(); if (!t) { const s = e.object.getChildMeshes ? e.object.getChildMeshes(!0) : []; n = new this.bjsAMMO.btCompoundShape(); let a = 0; if (s.forEach((f) => { const o = f.getPhysicsImpostor(); if (o) { if (o.type == tn.MeshImpostor) throw "A child MeshImpostor is not supported. Only primitive impostors are supported as children (eg. box or sphere)"; const d = this._createShape(o), v = f.parent.getWorldMatrix().clone(), u = new S(); v.decompose(u), this._tmpAmmoTransform.getOrigin().setValue(f.position.x * u.x, f.position.y * u.y, f.position.z * u.z), this._tmpAmmoQuaternion.setValue(f.rotationQuaternion.x, f.rotationQuaternion.y, f.rotationQuaternion.z, f.rotationQuaternion.w), this._tmpAmmoTransform.setRotation(this._tmpAmmoQuaternion), n.addChildShape(this._tmpAmmoTransform, d), o.dispose(), a++; } }), a > 0) { if (e.type != tn.NoImpostor) { const f = this._createShape(e, !0); f && (this._tmpAmmoTransform.getOrigin().setValue(0, 0, 0), this._tmpAmmoQuaternion.setValue(0, 0, 0, 1), this._tmpAmmoTransform.setRotation(this._tmpAmmoQuaternion), n.addChildShape(this._tmpAmmoTransform, f)); } return n; } else this.bjsAMMO.destroy(n), n = null; } switch (e.type) { case tn.SphereImpostor: if (Xt.WithinEpsilon(i.x, i.y, 1e-4) && Xt.WithinEpsilon(i.x, i.z, 1e-4)) n = new this.bjsAMMO.btSphereShape(i.x / 2); else { const s = [new this.bjsAMMO.btVector3(0, 0, 0)], a = [1]; n = new this.bjsAMMO.btMultiSphereShape(s, a, 1), n.setLocalScaling(new this.bjsAMMO.btVector3(i.x / 2, i.y / 2, i.z / 2)); } break; case tn.CapsuleImpostor: { const s = i.x / 2; n = new this.bjsAMMO.btCapsuleShape(s, i.y - s * 2); } break; case tn.CylinderImpostor: this._tmpAmmoVectorA.setValue(i.x / 2, i.y / 2, i.z / 2), n = new this.bjsAMMO.btCylinderShape(this._tmpAmmoVectorA); break; case tn.PlaneImpostor: case tn.BoxImpostor: this._tmpAmmoVectorA.setValue(i.x / 2, i.y / 2, i.z / 2), n = new this.bjsAMMO.btBoxShape(this._tmpAmmoVectorA); break; case tn.MeshImpostor: if (e.getParam("mass") == 0) { if (this.onCreateCustomMeshImpostor) n = this.onCreateCustomMeshImpostor(e); else { const s = new this.bjsAMMO.btTriangleMesh(); e._pluginData.toDispose.push(s), this._addMeshVerts(s, r, r) == 0 ? n = new this.bjsAMMO.btCompoundShape() : n = new this.bjsAMMO.btBvhTriangleMeshShape(s); } break; } case tn.ConvexHullImpostor: { if (this.onCreateCustomConvexHullImpostor) n = this.onCreateCustomConvexHullImpostor(e); else { const s = new this.bjsAMMO.btConvexHullShape(); this._addHullVerts(s, r, r) == 0 ? (e._pluginData.toDispose.push(s), n = new this.bjsAMMO.btCompoundShape()) : n = s; } break; } case tn.NoImpostor: n = new this.bjsAMMO.btSphereShape(i.x / 2); break; case tn.CustomImpostor: n = this._createCustom(e); break; case tn.SoftbodyImpostor: n = this._createSoftbody(e); break; case tn.ClothImpostor: n = this._createCloth(e); break; case tn.RopeImpostor: n = this._createRope(e); break; default: Se.Warn("The impostor type is not currently supported by the ammo plugin."); break; } return n; } /** * Sets the mesh body position/rotation from the babylon impostor * @param impostor imposter containing the physics body and babylon object */ setTransformationFromPhysicsBody(e) { e.physicsBody.getMotionState().getWorldTransform(this._tmpAmmoTransform), e.object.position.set(this._tmpAmmoTransform.getOrigin().x(), this._tmpAmmoTransform.getOrigin().y(), this._tmpAmmoTransform.getOrigin().z()), e.object.rotationQuaternion ? e.object.rotationQuaternion.set(this._tmpAmmoTransform.getRotation().x(), this._tmpAmmoTransform.getRotation().y(), this._tmpAmmoTransform.getRotation().z(), this._tmpAmmoTransform.getRotation().w()) : e.object.rotation && (this._tmpQuaternion.set(this._tmpAmmoTransform.getRotation().x(), this._tmpAmmoTransform.getRotation().y(), this._tmpAmmoTransform.getRotation().z(), this._tmpAmmoTransform.getRotation().w()), this._tmpQuaternion.toEulerAnglesToRef(e.object.rotation)); } /** * Sets the babylon object's position/rotation from the physics body's position/rotation * @param impostor imposter containing the physics body and babylon object * @param newPosition new position * @param newRotation new rotation */ setPhysicsBodyTransformation(e, t, r) { const n = e.physicsBody.getWorldTransform(); if (Math.abs(n.getOrigin().x() - t.x) > Dn || Math.abs(n.getOrigin().y() - t.y) > Dn || Math.abs(n.getOrigin().z() - t.z) > Dn || Math.abs(n.getRotation().x() - r.x) > Dn || Math.abs(n.getRotation().y() - r.y) > Dn || Math.abs(n.getRotation().z() - r.z) > Dn || Math.abs(n.getRotation().w() - r.w) > Dn) if (this._tmpAmmoVectorA.setValue(t.x, t.y, t.z), n.setOrigin(this._tmpAmmoVectorA), this._tmpAmmoQuaternion.setValue(r.x, r.y, r.z, r.w), n.setRotation(this._tmpAmmoQuaternion), e.physicsBody.setWorldTransform(n), e.mass == 0) { const i = e.physicsBody.getMotionState(); i && i.setWorldTransform(n); } else e.physicsBody.activate(); } /** * If this plugin is supported * @returns true if its supported */ isSupported() { return this.bjsAMMO !== void 0; } /** * Sets the linear velocity of the physics body * @param impostor imposter to set the velocity on * @param velocity velocity to set */ setLinearVelocity(e, t) { this._tmpAmmoVectorA.setValue(t.x, t.y, t.z), e.soft ? e.physicsBody.linearVelocity(this._tmpAmmoVectorA) : e.physicsBody.setLinearVelocity(this._tmpAmmoVectorA); } /** * Sets the angular velocity of the physics body * @param impostor imposter to set the velocity on * @param velocity velocity to set */ setAngularVelocity(e, t) { this._tmpAmmoVectorA.setValue(t.x, t.y, t.z), e.soft ? e.physicsBody.angularVelocity(this._tmpAmmoVectorA) : e.physicsBody.setAngularVelocity(this._tmpAmmoVectorA); } /** * gets the linear velocity * @param impostor imposter to get linear velocity from * @returns linear velocity */ getLinearVelocity(e) { let t; if (e.soft ? t = e.physicsBody.linearVelocity() : t = e.physicsBody.getLinearVelocity(), !t) return null; const r = new S(t.x(), t.y(), t.z()); return this.bjsAMMO.destroy(t), r; } /** * gets the angular velocity * @param impostor imposter to get angular velocity from * @returns angular velocity */ getAngularVelocity(e) { let t; if (e.soft ? t = e.physicsBody.angularVelocity() : t = e.physicsBody.getAngularVelocity(), !t) return null; const r = new S(t.x(), t.y(), t.z()); return this.bjsAMMO.destroy(t), r; } /** * Sets the mass of physics body * @param impostor imposter to set the mass on * @param mass mass to set */ setBodyMass(e, t) { e.soft ? e.physicsBody.setTotalMass(t, !1) : e.physicsBody.setMassProps(t), e._pluginData.mass = t; } /** * Gets the mass of the physics body * @param impostor imposter to get the mass from * @returns mass */ getBodyMass(e) { return e._pluginData.mass || 0; } /** * Gets friction of the impostor * @param impostor impostor to get friction from * @returns friction value */ getBodyFriction(e) { return e._pluginData.friction || 0; } /** * Sets friction of the impostor * @param impostor impostor to set friction on * @param friction friction value */ setBodyFriction(e, t) { e.soft ? e.physicsBody.get_m_cfg().set_kDF(t) : e.physicsBody.setFriction(t), e._pluginData.friction = t; } /** * Gets restitution of the impostor * @param impostor impostor to get restitution from * @returns restitution value */ getBodyRestitution(e) { return e._pluginData.restitution || 0; } /** * Sets restitution of the impostor * @param impostor impostor to set resitution on * @param restitution resitution value */ setBodyRestitution(e, t) { e.physicsBody.setRestitution(t), e._pluginData.restitution = t; } /** * Gets pressure inside the impostor * @param impostor impostor to get pressure from * @returns pressure value */ getBodyPressure(e) { return e.soft ? e._pluginData.pressure || 0 : (Se.Warn("Pressure is not a property of a rigid body"), 0); } /** * Sets pressure inside a soft body impostor * Cloth and rope must remain 0 pressure * @param impostor impostor to set pressure on * @param pressure pressure value */ setBodyPressure(e, t) { e.soft ? e.type === tn.SoftbodyImpostor ? (e.physicsBody.get_m_cfg().set_kPR(t), e._pluginData.pressure = t) : (e.physicsBody.get_m_cfg().set_kPR(0), e._pluginData.pressure = 0) : Se.Warn("Pressure can only be applied to a softbody"); } /** * Gets stiffness of the impostor * @param impostor impostor to get stiffness from * @returns pressure value */ getBodyStiffness(e) { return e.soft ? e._pluginData.stiffness || 0 : (Se.Warn("Stiffness is not a property of a rigid body"), 0); } /** * Sets stiffness of the impostor * @param impostor impostor to set stiffness on * @param stiffness stiffness value from 0 to 1 */ setBodyStiffness(e, t) { e.soft ? (t = t < 0 ? 0 : t, t = t > 1 ? 1 : t, e.physicsBody.get_m_materials().at(0).set_m_kLST(t), e._pluginData.stiffness = t) : Se.Warn("Stiffness cannot be applied to a rigid body"); } /** * Gets velocityIterations of the impostor * @param impostor impostor to get velocity iterations from * @returns velocityIterations value */ getBodyVelocityIterations(e) { return e.soft ? e._pluginData.velocityIterations || 0 : (Se.Warn("Velocity iterations is not a property of a rigid body"), 0); } /** * Sets velocityIterations of the impostor * @param impostor impostor to set velocity iterations on * @param velocityIterations velocityIterations value */ setBodyVelocityIterations(e, t) { e.soft ? (t = t < 0 ? 0 : t, e.physicsBody.get_m_cfg().set_viterations(t), e._pluginData.velocityIterations = t) : Se.Warn("Velocity iterations cannot be applied to a rigid body"); } /** * Gets positionIterations of the impostor * @param impostor impostor to get position iterations from * @returns positionIterations value */ getBodyPositionIterations(e) { return e.soft ? e._pluginData.positionIterations || 0 : (Se.Warn("Position iterations is not a property of a rigid body"), 0); } /** * Sets positionIterations of the impostor * @param impostor impostor to set position on * @param positionIterations positionIterations value */ setBodyPositionIterations(e, t) { e.soft ? (t = t < 0 ? 0 : t, e.physicsBody.get_m_cfg().set_piterations(t), e._pluginData.positionIterations = t) : Se.Warn("Position iterations cannot be applied to a rigid body"); } /** * Append an anchor to a cloth object * @param impostor is the cloth impostor to add anchor to * @param otherImpostor is the rigid impostor to anchor to * @param width ratio across width from 0 to 1 * @param height ratio up height from 0 to 1 * @param influence the elasticity between cloth impostor and anchor from 0, very stretchy to 1, little stretch * @param noCollisionBetweenLinkedBodies when true collisions between soft impostor and anchor are ignored; default false */ appendAnchor(e, t, r, n, i = 1, s = !1) { const a = e.segments, f = Math.round((a - 1) * r), o = Math.round((a - 1) * n), d = a - 1 - o, v = f + a * d; e.physicsBody.appendAnchor(v, t.physicsBody, s, i); } /** * Append an hook to a rope object * @param impostor is the rope impostor to add hook to * @param otherImpostor is the rigid impostor to hook to * @param length ratio along the rope from 0 to 1 * @param influence the elasticity between soft impostor and anchor from 0, very stretchy to 1, little stretch * @param noCollisionBetweenLinkedBodies when true collisions between soft impostor and anchor are ignored; default false */ appendHook(e, t, r, n = 1, i = !1) { const s = Math.round(e.segments * r); e.physicsBody.appendAnchor(s, t.physicsBody, i, n); } /** * Sleeps the physics body and stops it from being active * @param impostor impostor to sleep */ sleepBody(e) { e.physicsBody.forceActivationState(0); } /** * Activates the physics body * @param impostor impostor to activate */ wakeUpBody(e) { e.physicsBody.activate(); } /** * Updates the distance parameters of the joint */ updateDistanceJoint() { Se.Warn("updateDistanceJoint is not currently supported by the Ammo physics plugin"); } /** * Sets a motor on the joint * @param joint joint to set motor on * @param speed speed of the motor * @param maxForce maximum force of the motor */ setMotor(e, t, r) { e.physicsJoint.enableAngularMotor(!0, t, r); } /** * Sets the motors limit */ setLimit() { Se.Warn("setLimit is not currently supported by the Ammo physics plugin"); } /** * Syncs the position and rotation of a mesh with the impostor * @param mesh mesh to sync * @param impostor impostor to update the mesh with */ syncMeshWithImpostor(e, t) { t.physicsBody.getMotionState().getWorldTransform(this._tmpAmmoTransform), e.position.x = this._tmpAmmoTransform.getOrigin().x(), e.position.y = this._tmpAmmoTransform.getOrigin().y(), e.position.z = this._tmpAmmoTransform.getOrigin().z(), e.rotationQuaternion && (e.rotationQuaternion.x = this._tmpAmmoTransform.getRotation().x(), e.rotationQuaternion.y = this._tmpAmmoTransform.getRotation().y(), e.rotationQuaternion.z = this._tmpAmmoTransform.getRotation().z(), e.rotationQuaternion.w = this._tmpAmmoTransform.getRotation().w()); } /** * Gets the radius of the impostor * @param impostor impostor to get radius from * @returns the radius */ getRadius(e) { return e.getObjectExtents().x / 2; } /** * Gets the box size of the impostor * @param impostor impostor to get box size from * @param result the resulting box size */ getBoxSizeToRef(e, t) { const r = e.getObjectExtents(); t.x = r.x, t.y = r.y, t.z = r.z; } /** * Disposes of the impostor */ dispose() { this.bjsAMMO.destroy(this.world), this.bjsAMMO.destroy(this._solver), this.bjsAMMO.destroy(this._overlappingPairCache), this.bjsAMMO.destroy(this._dispatcher), this.bjsAMMO.destroy(this._collisionConfiguration), this.bjsAMMO.destroy(this._tmpAmmoVectorA), this.bjsAMMO.destroy(this._tmpAmmoVectorB), this.bjsAMMO.destroy(this._tmpAmmoVectorC), this.bjsAMMO.destroy(this._tmpAmmoTransform), this.bjsAMMO.destroy(this._tmpAmmoQuaternion), this.bjsAMMO.destroy(this._tmpAmmoConcreteContactResultCallback), this.world = null; } /** * Does a raycast in the physics world * @param from where should the ray start? * @param to where should the ray end? * @returns PhysicsRaycastResult */ raycast(e, t) { return this.raycastToRef(e, t, this._raycastResult), this._raycastResult; } /** * Does a raycast in the physics world * @param from when should the ray start? * @param to when should the ray end? * @param result resulting PhysicsRaycastResult */ raycastToRef(e, t, r) { this._tmpAmmoVectorRCA = new this.bjsAMMO.btVector3(e.x, e.y, e.z), this._tmpAmmoVectorRCB = new this.bjsAMMO.btVector3(t.x, t.y, t.z); const n = new this.bjsAMMO.ClosestRayResultCallback(this._tmpAmmoVectorRCA, this._tmpAmmoVectorRCB); this.world.rayTest(this._tmpAmmoVectorRCA, this._tmpAmmoVectorRCB, n), r.reset(e, t), n.hasHit() && (r.setHitData({ x: n.get_m_hitNormalWorld().x(), y: n.get_m_hitNormalWorld().y(), z: n.get_m_hitNormalWorld().z() }, { x: n.get_m_hitPointWorld().x(), y: n.get_m_hitPointWorld().y(), z: n.get_m_hitPointWorld().z() }), r.calculateHitDistance()), this.bjsAMMO.destroy(n), this.bjsAMMO.destroy(this._tmpAmmoVectorRCA), this.bjsAMMO.destroy(this._tmpAmmoVectorRCB); } } eg._DISABLE_COLLISION_FLAG = 4; eg._KINEMATIC_FLAG = 2; eg._DISABLE_DEACTIVATION_FLAG = 4; J1.prototype.removeReflectionProbe = function(A) { if (!this.reflectionProbes) return -1; const e = this.reflectionProbes.indexOf(A); return e !== -1 && this.reflectionProbes.splice(e, 1), e; }; J1.prototype.addReflectionProbe = function(A) { this.reflectionProbes || (this.reflectionProbes = []), this.reflectionProbes.push(A); }; class Im { /** * Creates a new reflection probe * @param name defines the name of the probe * @param size defines the texture resolution (for each face) * @param scene defines the hosting scene * @param generateMipMaps defines if mip maps should be generated automatically (true by default) * @param useFloat defines if HDR data (float data) should be used to store colors (false by default) * @param linearSpace defines if the probe should be generated in linear space or not (false by default) */ constructor(e, t, r, n = !0, i = !1, s = !1) { if (this.name = e, this._viewMatrix = he.Identity(), this._target = S.Zero(), this._add = S.Zero(), this._invertYAxis = !1, this.position = S.Zero(), this.metadata = null, this._parentContainer = null, this._scene = r, r.getEngine().supportsUniformBuffers) { this._sceneUBOs = []; for (let d = 0; d < 6; ++d) this._sceneUBOs.push(r.createSceneUniformBuffer(`Scene for Reflection Probe (name "${e}") face #${d}`)); } this._scene.reflectionProbes || (this._scene.reflectionProbes = []), this._scene.reflectionProbes.push(this); let a = 0; if (i) { const d = this._scene.getEngine().getCaps(); d.textureHalfFloatRender ? a = 2 : d.textureFloatRender && (a = 1); } this._renderTargetTexture = new Ta(e, t, r, n, !0, a, !0), this._renderTargetTexture.gammaSpace = !s, this._renderTargetTexture.invertZ = r.useRightHandedSystem; const f = r.getEngine().useReverseDepthBuffer; this._renderTargetTexture.onBeforeRenderObservable.add((d) => { switch (this._sceneUBOs && (r.setSceneUniformBuffer(this._sceneUBOs[d]), r.getSceneUniformBuffer().unbindEffect()), d) { case 0: this._add.copyFromFloats(1, 0, 0); break; case 1: this._add.copyFromFloats(-1, 0, 0); break; case 2: this._add.copyFromFloats(0, this._invertYAxis ? 1 : -1, 0); break; case 3: this._add.copyFromFloats(0, this._invertYAxis ? -1 : 1, 0); break; case 4: this._add.copyFromFloats(0, 0, r.useRightHandedSystem ? -1 : 1); break; case 5: this._add.copyFromFloats(0, 0, r.useRightHandedSystem ? 1 : -1); break; } this._attachedMesh && this.position.copyFrom(this._attachedMesh.getAbsolutePosition()), this.position.addToRef(this._add, this._target); const v = r.useRightHandedSystem ? he.LookAtRHToRef : he.LookAtLHToRef, u = r.useRightHandedSystem ? he.PerspectiveFovRH : he.PerspectiveFovLH; v(this.position, this._target, S.Up(), this._viewMatrix), r.activeCamera && (this._projectionMatrix = u(Math.PI / 2, 1, f ? r.activeCamera.maxZ : r.activeCamera.minZ, f ? r.activeCamera.minZ : r.activeCamera.maxZ, this._scene.getEngine().isNDCHalfZRange), r.setTransformMatrix(this._viewMatrix, this._projectionMatrix), r.activeCamera.isRigCamera && !this._renderTargetTexture.activeCamera && (this._renderTargetTexture.activeCamera = r.activeCamera.rigParent || null)), r._forcedViewPosition = this.position; }); let o; this._renderTargetTexture.onBeforeBindObservable.add(() => { var d, v; this._currentSceneUBO = r.getSceneUniformBuffer(), (v = (d = r.getEngine())._debugPushGroup) === null || v === void 0 || v.call(d, `reflection probe generation for ${e}`, 1), o = this._scene.imageProcessingConfiguration.applyByPostProcess, s && (r.imageProcessingConfiguration.applyByPostProcess = !0); }), this._renderTargetTexture.onAfterUnbindObservable.add(() => { var d, v; r.imageProcessingConfiguration.applyByPostProcess = o, r._forcedViewPosition = null, this._sceneUBOs && r.setSceneUniformBuffer(this._currentSceneUBO), r.updateTransformMatrix(!0), (v = (d = r.getEngine())._debugPopGroup) === null || v === void 0 || v.call(d, 1); }); } /** Gets or sets the number of samples to use for multi-sampling (0 by default). Required WebGL2 */ get samples() { return this._renderTargetTexture.samples; } set samples(e) { this._renderTargetTexture.samples = e; } /** Gets or sets the refresh rate to use (on every frame by default) */ get refreshRate() { return this._renderTargetTexture.refreshRate; } set refreshRate(e) { this._renderTargetTexture.refreshRate = e; } /** * Gets the hosting scene * @returns a Scene */ getScene() { return this._scene; } /** Gets the internal CubeTexture used to render to */ get cubeTexture() { return this._renderTargetTexture; } /** Gets or sets the list of meshes to render */ get renderList() { return this._renderTargetTexture.renderList; } set renderList(e) { this._renderTargetTexture.renderList = e; } /** * Attach the probe to a specific mesh (Rendering will be done from attached mesh's position) * @param mesh defines the mesh to attach to */ attachToMesh(e) { this._attachedMesh = e; } /** * Specifies whether or not the stencil and depth buffer are cleared between two rendering groups * @param renderingGroupId The rendering group id corresponding to its index * @param autoClearDepthStencil Automatically clears depth and stencil between groups if true. */ setRenderingAutoClearDepthStencil(e, t) { this._renderTargetTexture.setRenderingAutoClearDepthStencil(e, t); } /** * Clean all associated resources */ dispose() { const e = this._scene.reflectionProbes.indexOf(this); if (e !== -1 && this._scene.reflectionProbes.splice(e, 1), this._parentContainer) { const t = this._parentContainer.reflectionProbes.indexOf(this); t > -1 && this._parentContainer.reflectionProbes.splice(t, 1), this._parentContainer = null; } if (this._renderTargetTexture && (this._renderTargetTexture.dispose(), this._renderTargetTexture = null), this._sceneUBOs) { for (const t of this._sceneUBOs) t.dispose(); this._sceneUBOs = []; } } /** * Converts the reflection probe information to a readable string for debug purpose. * @param fullDetails Supports for multiple levels of logging within scene loading * @returns the human readable reflection probe info */ toString(e) { let t = "Name: " + this.name; return e && (t += ", position: " + this.position.toString(), this._attachedMesh && (t += ", attached mesh: " + this._attachedMesh.name)), t; } /** * Get the class name of the refection probe. * @returns "ReflectionProbe" */ getClassName() { return "ReflectionProbe"; } /** * Serialize the reflection probe to a JSON representation we can easily use in the respective Parse function. * @returns The JSON representation of the texture */ serialize() { const e = jt.Serialize(this, this._renderTargetTexture.serialize()); return e.isReflectionProbe = !0, e.metadata = this.metadata, e; } /** * Parse the JSON representation of a reflection probe in order to recreate the reflection probe in the given scene. * @param parsedReflectionProbe Define the JSON representation of the reflection probe * @param scene Define the scene the parsed reflection probe should be instantiated in * @param rootUrl Define the root url of the parsing sequence in the case of relative dependencies * @returns The parsed reflection probe if successful */ static Parse(e, t, r) { let n = null; if (t.reflectionProbes) for (let i = 0; i < t.reflectionProbes.length; i++) { const s = t.reflectionProbes[i]; if (s.name === e.name) { n = s; break; } } return n = jt.Parse(() => n || new Im(e.name, e.renderTargetSize, t, e._generateMipMaps), e, t, r), n.cubeTexture._waitingRenderList = e.renderList, e._attachedMesh && n.attachToMesh(t.getMeshById(e._attachedMesh)), e.metadata && (n.metadata = e.metadata), n; } } C([ tU() ], Im.prototype, "_attachedMesh", void 0); C([ fo() ], Im.prototype, "position", void 0); var Tpe = !0; class SI { } SI.LoaderInjectedPhysicsEngine = void 0; let EW = {}, nD = {}; const LG = (A, e, t, r) => { if (!e.materials) return null; for (let n = 0, i = e.materials.length; n < i; n++) { const s = e.materials[n]; if (A(s)) return { parsedMaterial: s, material: gt.Parse(s, t, r) }; } return null; }, qpe = (A, e, t) => { for (const r in e) if (A.name === e[r]) return t.push(A.id), !0; return A.parentId !== void 0 && t.indexOf(A.parentId) !== -1 ? (t.push(A.id), !0) : !1; }, YW = (A, e) => A + " of " + (e ? e.file + " from " + e.name + " version: " + e.version + ", exporter version: " + e.exporter_version : "unknown"), Gte = (A, e) => { const t = e; if (e._waitingData.lods) { if (e._waitingData.lods.ids && e._waitingData.lods.ids.length > 0) { const r = e._waitingData.lods.ids, n = t.isEnabled(!1); if (e._waitingData.lods.distances) { const i = e._waitingData.lods.distances; if (i.length >= r.length) { const s = i.length > r.length ? i[i.length - 1] : 0; t.setEnabled(!1); for (let a = 0; a < r.length; a++) { const f = r[a], o = A.getMeshById(f); o != null && t.addLODLevel(i[a], o); } s > 0 && t.addLODLevel(s, null), n === !0 && t.setEnabled(!0); } else ye.Warn("Invalid level of detail distances for " + e.name); } } e._waitingData.lods = null; } }, gC = (A, e, t) => { if (typeof A != "number") { const n = t.getLastEntryById(A); return n && e !== void 0 && e !== null ? n.instances[parseInt(e)] : n; } const r = EW[A]; return r && e !== void 0 && e !== null ? r.instances[parseInt(e)] : r; }, aO = (A, e) => typeof A != "number" ? e.getLastMaterialById(A, !0) : nD[A], KG = (A, e, t, r, n = !1) => { const i = new xR(A); let s = "importScene has failed JSON parse"; try { var a = JSON.parse(e); s = ""; const f = Hn.loggingLevel === Hn.DETAILED_LOGGING; let o, d; if (a.environmentTexture !== void 0 && a.environmentTexture !== null) { const u = a.isPBR !== void 0 ? a.isPBR : !0; if (a.environmentTextureType && a.environmentTextureType === "BABYLON.HDRCubeTexture") { const l = a.environmentTextureSize ? a.environmentTextureSize : 128, P = new _2((a.environmentTexture.match(/https?:\/\//g) ? "" : t) + a.environmentTexture, A, l, !0, !u, void 0, a.environmentTexturePrefilterOnLoad); a.environmentTextureRotationY && (P.rotationY = a.environmentTextureRotationY), A.environmentTexture = P; } else if (typeof a.environmentTexture == "object") { const l = v1.Parse(a.environmentTexture, A, t); A.environmentTexture = l; } else if (a.environmentTexture.endsWith(".env")) { const l = new v1((a.environmentTexture.match(/https?:\/\//g) ? "" : t) + a.environmentTexture, A, a.environmentTextureForcedExtension); a.environmentTextureRotationY && (l.rotationY = a.environmentTextureRotationY), A.environmentTexture = l; } else { const l = v1.CreateFromPrefilteredData((a.environmentTexture.match(/https?:\/\//g) ? "" : t) + a.environmentTexture, A, a.environmentTextureForcedExtension); a.environmentTextureRotationY && (l.rotationY = a.environmentTextureRotationY), A.environmentTexture = l; } if (a.createDefaultSkybox === !0) { const l = A.activeCamera !== void 0 && A.activeCamera !== null ? (A.activeCamera.maxZ - A.activeCamera.minZ) / 2 : 1e3, P = a.skyboxBlurLevel || 0; A.createDefaultSkybox(A.environmentTexture, u, l, P); } i.environmentTexture = A.environmentTexture; } if (a.environmentIntensity !== void 0 && a.environmentIntensity !== null && (A.environmentIntensity = a.environmentIntensity), a.lights !== void 0 && a.lights !== null) for (o = 0, d = a.lights.length; o < d; o++) { const u = a.lights[o], l = ci.Parse(u, A); l && (EW[u.uniqueId] = l, i.lights.push(l), l._parentContainer = i, s += o === 0 ? ` Lights:` : "", s += ` ` + l.toString(f)); } if (a.reflectionProbes !== void 0 && a.reflectionProbes !== null) for (o = 0, d = a.reflectionProbes.length; o < d; o++) { const u = a.reflectionProbes[o], l = Im.Parse(u, A, t); l && (i.reflectionProbes.push(l), l._parentContainer = i, s += o === 0 ? ` Reflection Probes:` : "", s += ` ` + l.toString(f)); } if (a.animations !== void 0 && a.animations !== null) for (o = 0, d = a.animations.length; o < d; o++) { const u = a.animations[o], l = Jo("BABYLON.Animation"); if (l) { const P = l.Parse(u); A.animations.push(P), i.animations.push(P), s += o === 0 ? ` Animations:` : "", s += ` ` + P.toString(f); } } if (a.materials !== void 0 && a.materials !== null) for (o = 0, d = a.materials.length; o < d; o++) { const u = a.materials[o], l = gt.Parse(u, A, t); l && (nD[u.uniqueId || u.id] = l, i.materials.push(l), l._parentContainer = i, s += o === 0 ? ` Materials:` : "", s += ` ` + l.toString(f), l.getActiveTextures().forEach((p) => { i.textures.indexOf(p) == -1 && (i.textures.push(p), p._parentContainer = i); })); } if (a.multiMaterials !== void 0 && a.multiMaterials !== null) for (o = 0, d = a.multiMaterials.length; o < d; o++) { const u = a.multiMaterials[o], l = Dc.ParseMultiMaterial(u, A); nD[u.uniqueId || u.id] = l, i.multiMaterials.push(l), l._parentContainer = i, s += o === 0 ? ` MultiMaterials:` : "", s += ` ` + l.toString(f), l.getActiveTextures().forEach((p) => { i.textures.indexOf(p) == -1 && (i.textures.push(p), p._parentContainer = i); }); } if (a.morphTargetManagers !== void 0 && a.morphTargetManagers !== null) for (const u of a.morphTargetManagers) { const l = U0.Parse(u, A); i.morphTargetManagers.push(l), l._parentContainer = i; } if (a.skeletons !== void 0 && a.skeletons !== null) for (o = 0, d = a.skeletons.length; o < d; o++) { const u = a.skeletons[o], l = r4.Parse(u, A); i.skeletons.push(l), l._parentContainer = i, s += o === 0 ? ` Skeletons:` : "", s += ` ` + l.toString(f); } const v = a.geometries; if (v != null) { const u = new Array(), l = v.vertexData; if (l != null) for (o = 0, d = l.length; o < d; o++) { const P = l[o]; u.push(Tf.Parse(P, A, t)); } u.forEach((P) => { P && (i.geometries.push(P), P._parentContainer = i); }); } if (a.transformNodes !== void 0 && a.transformNodes !== null) for (o = 0, d = a.transformNodes.length; o < d; o++) { const u = a.transformNodes[o], l = Hr.Parse(u, A, t); EW[u.uniqueId] = l, i.transformNodes.push(l), l._parentContainer = i; } if (a.meshes !== void 0 && a.meshes !== null) for (o = 0, d = a.meshes.length; o < d; o++) { const u = a.meshes[o], l = Ee.Parse(u, A, t); if (EW[u.uniqueId] = l, i.meshes.push(l), l._parentContainer = i, l.hasInstances) for (const P of l.instances) i.meshes.push(P), P._parentContainer = i; s += o === 0 ? ` Meshes:` : "", s += ` ` + l.toString(f); } if (a.cameras !== void 0 && a.cameras !== null) for (o = 0, d = a.cameras.length; o < d; o++) { const u = a.cameras[o], l = Tr.Parse(u, A); EW[u.uniqueId] = l, i.cameras.push(l), l._parentContainer = i, s += o === 0 ? ` Cameras:` : "", s += ` ` + l.toString(f); } if (a.postProcesses !== void 0 && a.postProcesses !== null) for (o = 0, d = a.postProcesses.length; o < d; o++) { const u = a.postProcesses[o], l = kr.Parse(u, A, t); l && (i.postProcesses.push(l), l._parentContainer = i, s += o === 0 ? ` Postprocesses:` : "", s += ` ` + l.toString()); } if (a.animationGroups !== void 0 && a.animationGroups !== null) for (o = 0, d = a.animationGroups.length; o < d; o++) { const u = a.animationGroups[o], l = w0.Parse(u, A); i.animationGroups.push(l), l._parentContainer = i, s += o === 0 ? ` AnimationGroups:` : "", s += ` ` + l.toString(f); } for (o = 0, d = A.cameras.length; o < d; o++) { const u = A.cameras[o]; u._waitingParentId !== null && (u.parent = gC(u._waitingParentId, u._waitingParentInstanceIndex, A), u._waitingParentId = null, u._waitingParentInstanceIndex = null); } for (o = 0, d = A.lights.length; o < d; o++) { const u = A.lights[o]; u && u._waitingParentId !== null && (u.parent = gC(u._waitingParentId, u._waitingParentInstanceIndex, A), u._waitingParentId = null, u._waitingParentInstanceIndex = null); } for (o = 0, d = A.transformNodes.length; o < d; o++) { const u = A.transformNodes[o]; u._waitingParentId !== null && (u.parent = gC(u._waitingParentId, u._waitingParentInstanceIndex, A), u._waitingParentId = null, u._waitingParentInstanceIndex = null); } for (o = 0, d = A.meshes.length; o < d; o++) { const u = A.meshes[o]; u._waitingParentId !== null && (u.parent = gC(u._waitingParentId, u._waitingParentInstanceIndex, A), u._waitingParentId = null, u._waitingParentInstanceIndex = null), u._waitingData.lods && Gte(A, u); } for (A.multiMaterials.forEach((u) => { u._waitingSubMaterialsUniqueIds.forEach((l) => { u.subMaterials.push(aO(l, A)); }), u._waitingSubMaterialsUniqueIds = []; }), A.meshes.forEach((u) => { u._waitingMaterialId !== null && (u.material = aO(u._waitingMaterialId, A), u._waitingMaterialId = null); }), o = 0, d = A.skeletons.length; o < d; o++) { const u = A.skeletons[o]; u._hasWaitingData && (u.bones != null && u.bones.forEach((l) => { if (l._waitingTransformNodeId) { const P = A.getLastEntryById(l._waitingTransformNodeId); P && l.linkTransformNode(P), l._waitingTransformNodeId = null; } }), u._hasWaitingData = null); } for (o = 0, d = A.meshes.length; o < d; o++) { const u = A.meshes[o]; u._waitingData.freezeWorldMatrix ? (u.freezeWorldMatrix(), u._waitingData.freezeWorldMatrix = null) : u.computeWorldMatrix(!0); } for (o = 0, d = A.lights.length; o < d; o++) { const u = A.lights[o]; if (u._excludedMeshesIds.length > 0) { for (let l = 0; l < u._excludedMeshesIds.length; l++) { const P = A.getMeshById(u._excludedMeshesIds[l]); P && u.excludedMeshes.push(P); } u._excludedMeshesIds = []; } if (u._includedOnlyMeshesIds.length > 0) { for (let l = 0; l < u._includedOnlyMeshesIds.length; l++) { const P = A.getMeshById(u._includedOnlyMeshesIds[l]); P && u.includedOnlyMeshes.push(P); } u._includedOnlyMeshesIds = []; } } for (A.geometries.forEach((u) => { u._loadedUniqueId = ""; }), J1.Parse(a, A, i, t), o = 0, d = A.meshes.length; o < d; o++) { const u = A.meshes[o]; u._waitingData.actions && (Rs.Parse(u._waitingData.actions, u, A), u._waitingData.actions = null); } a.actions !== void 0 && a.actions !== null && Rs.Parse(a.actions, null, A); } catch (f) { const o = YW("loadAssets", a ? a.producer : "Unknown") + s; if (r) r(o, f); else throw Se.Log(o), f; } finally { EW = {}, nD = {}, n || i.removeAllFromScene(), s !== null && Hn.loggingLevel !== Hn.NO_LOGGING && Se.Log(YW("loadAssets", a ? a.producer : "Unknown") + (Hn.loggingLevel !== Hn.MINIMAL_LOGGING ? s : "")); } return i; }; Hn.RegisterPlugin({ name: "babylon.js", extensions: ".babylon", canDirectLoad: (A) => A.indexOf("babylon") !== -1, importMesh: (A, e, t, r, n, i, s, a) => { var f; let o = "importMesh has failed JSON parse"; try { var d = JSON.parse(t); o = ""; const v = Hn.loggingLevel === Hn.DETAILED_LOGGING; A ? Array.isArray(A) || (A = [A]) : A = null; const u = [], l = /* @__PURE__ */ new Map(), P = []; if (d.transformNodes !== void 0 && d.transformNodes !== null) for (let p = 0, c = d.transformNodes.length; p < c; p++) { const H = d.transformNodes[p], T = Hr.Parse(H, e, r); P.push(T), l.set(T._waitingParsedUniqueId, T), T._waitingParsedUniqueId = null; } if (d.meshes !== void 0 && d.meshes !== null) { const p = [], c = [], H = [], T = []; for (let b = 0, j = d.meshes.length; b < j; b++) { const w = d.meshes[b]; if (A === null || qpe(w, A, u)) { if (A !== null && delete A[A.indexOf(w.name)], w.geometryId !== void 0 && w.geometryId !== null && d.geometries !== void 0 && d.geometries !== null) { let I = !1; ["boxes", "spheres", "cylinders", "toruses", "grounds", "planes", "torusKnots", "vertexData"].forEach((N) => { I === !0 || !d.geometries[N] || !Array.isArray(d.geometries[N]) || d.geometries[N].forEach((k) => { if (k.id === w.geometryId) { switch (N) { case "vertexData": Tf.Parse(k, e, r); break; } I = !0; } }); }), I === !1 && Se.Warn("Geometry not found for mesh " + w.id); } if (w.materialUniqueId || w.materialId) { const I = w.materialUniqueId ? H : c; let N = I.indexOf(w.materialUniqueId || w.materialId) !== -1; if (N === !1 && d.multiMaterials !== void 0 && d.multiMaterials !== null) { const k = (R, y) => { I.push(R); const O = LG(y, d, e, r); O && O.material && (nD[O.parsedMaterial.uniqueId || O.parsedMaterial.id] = O.material, o += ` Material ` + O.material.toString(v)); }; for (let R = 0, y = d.multiMaterials.length; R < y; R++) { const O = d.multiMaterials[R]; if (w.materialUniqueId && O.uniqueId === w.materialUniqueId || O.id === w.materialId) { O.materialsUniqueIds ? O.materialsUniqueIds.forEach((ee) => k(ee, (Z) => Z.uniqueId === ee)) : O.materials.forEach((ee) => k(ee, (Z) => Z.id === ee)), I.push(O.uniqueId || O.id); const Y = Dc.ParseMultiMaterial(O, e); nD[O.uniqueId || O.id] = Y, Y && (N = !0, o += ` Multi-Material ` + Y.toString(v)); break; } } } if (N === !1) { I.push(w.materialUniqueId || w.materialId); const k = LG((R) => w.materialUniqueId && R.uniqueId === w.materialUniqueId || R.id === w.materialId, d, e, r); !k || !k.material ? Se.Warn("Material not found for mesh " + w.id) : (nD[k.parsedMaterial.uniqueId || k.parsedMaterial.id] = k.material, o += ` Material ` + k.material.toString(v)); } } if (w.skeletonId !== null && w.skeletonId !== void 0 && d.skeletonId !== -1 && d.skeletons !== void 0 && d.skeletons !== null && !(p.indexOf(w.skeletonId) > -1)) for (let N = 0, k = d.skeletons.length; N < k; N++) { const R = d.skeletons[N]; if (R.id === w.skeletonId) { const y = r4.Parse(R, e); s.push(y), p.push(R.id), o += ` Skeleton ` + y.toString(v); } } if (w.morphTargetManagerId > -1 && d.morphTargetManagers !== void 0 && d.morphTargetManagers !== null && !(T.indexOf(w.morphTargetManagerId) > -1)) for (let N = 0, k = d.morphTargetManagers.length; N < k; N++) { const R = d.morphTargetManagers[N]; if (R.id === w.morphTargetManagerId) { const y = U0.Parse(R, e); T.push(y.uniqueId), o += ` Morph target ` + y.toString(); } } const m = Ee.Parse(w, e, r); n.push(m), l.set(m._waitingParsedUniqueId, m), m._waitingParsedUniqueId = null, o += ` Mesh ` + m.toString(v); } } e.multiMaterials.forEach((b) => { b._waitingSubMaterialsUniqueIds.forEach((j) => { b.subMaterials.push(aO(j, e)); }), b._waitingSubMaterialsUniqueIds = []; }), e.meshes.forEach((b) => { b._waitingMaterialId !== null && (b.material = aO(b._waitingMaterialId, e), b._waitingMaterialId = null); }); for (let b = 0, j = e.transformNodes.length; b < j; b++) { const w = e.transformNodes[b]; if (w._waitingParentId !== null) { let m = l.get(parseInt(w._waitingParentId)) || null; m === null && (m = e.getLastEntryById(w._waitingParentId)); let I = m; w._waitingParentInstanceIndex && (I = m.instances[parseInt(w._waitingParentInstanceIndex)], w._waitingParentInstanceIndex = null), w.parent = I, w._waitingParentId = null; } } let q; for (let b = 0, j = e.meshes.length; b < j; b++) { if (q = e.meshes[b], q._waitingParentId) { let w = l.get(parseInt(q._waitingParentId)) || null; w === null && (w = e.getLastEntryById(q._waitingParentId)); let m = w; if (q._waitingParentInstanceIndex && (m = w.instances[parseInt(q._waitingParentInstanceIndex)], q._waitingParentInstanceIndex = null), q.parent = m, ((f = q.parent) === null || f === void 0 ? void 0 : f.getClassName()) === "TransformNode") { const I = P.indexOf(q.parent); I > -1 && P.splice(I, 1); } q._waitingParentId = null; } q._waitingData.lods && Gte(e, q); } for (const b of P) b.dispose(); for (let b = 0, j = e.skeletons.length; b < j; b++) { const w = e.skeletons[b]; w._hasWaitingData && (w.bones != null && w.bones.forEach((m) => { if (m._waitingTransformNodeId) { const I = e.getLastEntryById(m._waitingTransformNodeId); I && m.linkTransformNode(I), m._waitingTransformNodeId = null; } }), w._hasWaitingData = null); } for (let b = 0, j = e.meshes.length; b < j; b++) q = e.meshes[b], q._waitingData.freezeWorldMatrix ? (q.freezeWorldMatrix(), q._waitingData.freezeWorldMatrix = null) : q.computeWorldMatrix(!0); } if (d.particleSystems !== void 0 && d.particleSystems !== null) { const p = J1.GetIndividualParser(Ot.NAME_PARTICLESYSTEM); if (p) for (let c = 0, H = d.particleSystems.length; c < H; c++) { const T = d.particleSystems[c]; u.indexOf(T.emitterId) !== -1 && i.push(p(T, e, r)); } } return e.geometries.forEach((p) => { p._loadedUniqueId = ""; }), !0; } catch (v) { const u = YW("importMesh", d ? d.producer : "Unknown") + o; if (a) a(u, v); else throw Se.Log(u), v; } finally { o !== null && Hn.loggingLevel !== Hn.NO_LOGGING && Se.Log(YW("importMesh", d ? d.producer : "Unknown") + (Hn.loggingLevel !== Hn.MINIMAL_LOGGING ? o : "")), nD = {}; } return !1; }, load: (A, e, t, r) => { let n = "importScene has failed JSON parse"; try { var i = JSON.parse(e); if (n = "", i.useDelayedTextureLoading !== void 0 && i.useDelayedTextureLoading !== null && (A.useDelayedTextureLoading = i.useDelayedTextureLoading && !Hn.ForceFullSceneLoadingForIncremental), i.autoClear !== void 0 && i.autoClear !== null && (A.autoClear = i.autoClear), i.clearColor !== void 0 && i.clearColor !== null && (A.clearColor = xt.FromArray(i.clearColor)), i.ambientColor !== void 0 && i.ambientColor !== null && (A.ambientColor = Ne.FromArray(i.ambientColor)), i.gravity !== void 0 && i.gravity !== null && (A.gravity = S.FromArray(i.gravity)), i.useRightHandedSystem !== void 0 && (A.useRightHandedSystem = !!i.useRightHandedSystem), i.fogMode && i.fogMode !== 0) switch (A.fogMode = i.fogMode, A.fogColor = Ne.FromArray(i.fogColor), A.fogStart = i.fogStart, A.fogEnd = i.fogEnd, A.fogDensity = i.fogDensity, n += " Fog mode for scene: ", A.fogMode) { case 1: n += `exp `; break; case 2: n += `exp2 `; break; case 3: n += `linear `; break; } if (i.physicsEnabled) { let a; i.physicsEngine === "cannon" || i.physicsEngine === sO.name ? a = new sO(void 0, void 0, SI.LoaderInjectedPhysicsEngine) : i.physicsEngine === "oimo" || i.physicsEngine === gF.name ? a = new gF(void 0, SI.LoaderInjectedPhysicsEngine) : (i.physicsEngine === "ammo" || i.physicsEngine === eg.name) && (a = new eg(void 0, SI.LoaderInjectedPhysicsEngine, void 0)), n = " Physics engine " + (i.physicsEngine ? i.physicsEngine : "oimo") + ` enabled `; const f = i.physicsGravity ? S.FromArray(i.physicsGravity) : null; A.enablePhysics(f, a); } return i.metadata !== void 0 && i.metadata !== null && (A.metadata = i.metadata), i.collisionsEnabled !== void 0 && i.collisionsEnabled !== null && (A.collisionsEnabled = i.collisionsEnabled), KG(A, e, t, r, !0) ? (i.autoAnimate && A.beginAnimation(A, i.autoAnimateFrom, i.autoAnimateTo, i.autoAnimateLoop, i.autoAnimateSpeed || 1), i.activeCameraID !== void 0 && i.activeCameraID !== null && A.setActiveCameraById(i.activeCameraID), !0) : !1; } catch (s) { const a = YW("importScene", i ? i.producer : "Unknown") + n; if (r) r(a, s); else throw Se.Log(a), s; } finally { n !== null && Hn.loggingLevel !== Hn.NO_LOGGING && Se.Log(YW("importScene", i ? i.producer : "Unknown") + (Hn.loggingLevel !== Hn.MINIMAL_LOGGING ? n : "")); } return !1; }, loadAssetContainer: (A, e, t, r) => KG(A, e, t, r) }); class GI { /** * Define if the fresnel effect is enable or not. */ get isEnabled() { return this._isEnabled; } set isEnabled(e) { this._isEnabled !== e && (this._isEnabled = e, Ge.MarkAllMaterialsAsDirty(20)); } /** * Creates a new FresnelParameters object. * * @param options provide your own settings to optionally to override defaults */ constructor(e = {}) { this._isEnabled = !0, this.bias = e.bias === void 0 ? 0 : e.bias, this.power = e.power === void 0 ? 1 : e.power, this.leftColor = e.leftColor || Ne.White(), this.rightColor = e.rightColor || Ne.Black(), e.isEnabled === !1 && (this.isEnabled = !1); } /** * Clones the current fresnel and its values * @returns a clone fresnel configuration */ clone() { const e = new GI(); return sA.DeepCopy(this, e), e; } /** * Determines equality between FresnelParameters objects * @param otherFresnelParameters defines the second operand * @returns true if the power, bias, leftColor, rightColor and isEnabled values are equal to the given ones */ equals(e) { return e && this.bias === e.bias && this.power === e.power && this.leftColor.equals(e.leftColor) && this.rightColor.equals(e.rightColor) && this.isEnabled === e.isEnabled; } /** * Serializes the current fresnel parameters to a JSON representation. * @returns the JSON serialization */ serialize() { return { isEnabled: this.isEnabled, leftColor: this.leftColor.asArray(), rightColor: this.rightColor.asArray(), bias: this.bias, power: this.power }; } /** * Parse a JSON object and deserialize it to a new Fresnel parameter object. * @param parsedFresnelParameters Define the JSON representation * @returns the parsed parameters */ static Parse(e) { return new GI({ isEnabled: e.isEnabled, leftColor: Ne.FromArray(e.leftColor), rightColor: Ne.FromArray(e.rightColor), bias: e.bias, power: e.power || 1 }); } } jt._FresnelParametersParser = GI.Parse; class bpe extends Zo { constructor(e, t) { super(e, t, "color", { attributes: ["position"], uniforms: ["world", "viewProjection", "color"] }), this.disableColorWrite = !0, this.forceDepthWrite = !0, this.setColor4("color", new xt(0, 0, 0, 1)); } } class Yv extends fs { /** * Gets the current double sided mode. */ get doubleSided() { return this._twoSidedLighting; } /** * If sets to true and backfaceCulling is false, normals will be flipped on the backside. */ set doubleSided(e) { this._twoSidedLighting !== e && (this._twoSidedLighting = e, this.backFaceCulling = !e, this._markAllSubMeshesAsTexturesDirty()); } /** * Instantiates a new PBRMaterial instance. * * @param name The material name * @param scene The scene the material will be use in. */ constructor(e, t) { super(e, t), this.maxSimultaneousLights = 4, this.disableLighting = !1, this.invertNormalMapX = !1, this.invertNormalMapY = !1, this.emissiveColor = new Ne(0, 0, 0), this.occlusionStrength = 1, this.useLightmapAsShadowmap = !1, this._useAlphaFromAlbedoTexture = !0, this._useAmbientInGrayScale = !0; } getClassName() { return "PBRBaseSimpleMaterial"; } } C([ M(), At("_markAllSubMeshesAsLightsDirty") ], Yv.prototype, "maxSimultaneousLights", void 0); C([ M(), At("_markAllSubMeshesAsLightsDirty") ], Yv.prototype, "disableLighting", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty", "_reflectionTexture") ], Yv.prototype, "environmentTexture", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], Yv.prototype, "invertNormalMapX", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], Yv.prototype, "invertNormalMapY", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty", "_bumpTexture") ], Yv.prototype, "normalTexture", void 0); C([ Oi("emissive"), At("_markAllSubMeshesAsTexturesDirty") ], Yv.prototype, "emissiveColor", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty") ], Yv.prototype, "emissiveTexture", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty", "_ambientTextureStrength") ], Yv.prototype, "occlusionStrength", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty", "_ambientTexture") ], Yv.prototype, "occlusionTexture", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty", "_alphaCutOff") ], Yv.prototype, "alphaCutOff", void 0); C([ M() ], Yv.prototype, "doubleSided", null); C([ en(), At("_markAllSubMeshesAsTexturesDirty", null) ], Yv.prototype, "lightmapTexture", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], Yv.prototype, "useLightmapAsShadowmap", void 0); class s4 extends Yv { /** * Instantiates a new PBRMetalRoughnessMaterial instance. * * @param name The material name * @param scene The scene the material will be use in. */ constructor(e, t) { super(e, t), this._useRoughnessFromMetallicTextureAlpha = !1, this._useRoughnessFromMetallicTextureGreen = !0, this._useMetallnessFromMetallicTextureBlue = !0, this.metallic = 1, this.roughness = 1; } /** * Return the current class name of the material. */ getClassName() { return "PBRMetallicRoughnessMaterial"; } /** * Makes a duplicate of the current material. * @param name - name to use for the new material. */ clone(e) { const t = jt.Clone(() => new s4(e, this.getScene()), this); return t.id = e, t.name = e, this.clearCoat.copyTo(t.clearCoat), this.anisotropy.copyTo(t.anisotropy), this.brdf.copyTo(t.brdf), this.sheen.copyTo(t.sheen), this.subSurface.copyTo(t.subSurface), t; } /** * Serialize the material to a parsable JSON object. */ serialize() { const e = jt.Serialize(this); return e.customType = "BABYLON.PBRMetallicRoughnessMaterial", e.clearCoat = this.clearCoat.serialize(), e.anisotropy = this.anisotropy.serialize(), e.brdf = this.brdf.serialize(), e.sheen = this.sheen.serialize(), e.subSurface = this.subSurface.serialize(), e.iridescence = this.iridescence.serialize(), e; } /** * Parses a JSON object corresponding to the serialize function. * @param source * @param scene * @param rootUrl */ static Parse(e, t, r) { const n = jt.Parse(() => new s4(e.name, t), e, t, r); return e.clearCoat && n.clearCoat.parse(e.clearCoat, t, r), e.anisotropy && n.anisotropy.parse(e.anisotropy, t, r), e.brdf && n.brdf.parse(e.brdf, t, r), e.sheen && n.sheen.parse(e.sheen, t, r), e.subSurface && n.subSurface.parse(e.subSurface, t, r), e.iridescence && n.iridescence.parse(e.iridescence, t, r), n; } } C([ Oi(), At("_markAllSubMeshesAsTexturesDirty", "_albedoColor") ], s4.prototype, "baseColor", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty", "_albedoTexture") ], s4.prototype, "baseTexture", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], s4.prototype, "metallic", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], s4.prototype, "roughness", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty", "_metallicTexture") ], s4.prototype, "metallicRoughnessTexture", void 0); Ue("BABYLON.PBRMetallicRoughnessMaterial", s4); class a4 extends Yv { /** * Specifies if the reflectivity texture contains the glossiness information in its alpha channel. */ get useMicroSurfaceFromReflectivityMapAlpha() { return this._useMicroSurfaceFromReflectivityMapAlpha; } /** * Instantiates a new PBRSpecularGlossinessMaterial instance. * * @param name The material name * @param scene The scene the material will be use in. */ constructor(e, t) { super(e, t), this._useMicroSurfaceFromReflectivityMapAlpha = !0; } /** * Return the current class name of the material. */ getClassName() { return "PBRSpecularGlossinessMaterial"; } /** * Makes a duplicate of the current material. * @param name - name to use for the new material. */ clone(e) { const t = jt.Clone(() => new a4(e, this.getScene()), this); return t.id = e, t.name = e, this.clearCoat.copyTo(t.clearCoat), this.anisotropy.copyTo(t.anisotropy), this.brdf.copyTo(t.brdf), this.sheen.copyTo(t.sheen), this.subSurface.copyTo(t.subSurface), t; } /** * Serialize the material to a parsable JSON object. */ serialize() { const e = jt.Serialize(this); return e.customType = "BABYLON.PBRSpecularGlossinessMaterial", e.clearCoat = this.clearCoat.serialize(), e.anisotropy = this.anisotropy.serialize(), e.brdf = this.brdf.serialize(), e.sheen = this.sheen.serialize(), e.subSurface = this.subSurface.serialize(), e.iridescence = this.iridescence.serialize(), e; } /** * Parses a JSON object corresponding to the serialize function. * @param source * @param scene * @param rootUrl */ static Parse(e, t, r) { const n = jt.Parse(() => new a4(e.name, t), e, t, r); return e.clearCoat && n.clearCoat.parse(e.clearCoat, t, r), e.anisotropy && n.anisotropy.parse(e.anisotropy, t, r), e.brdf && n.brdf.parse(e.brdf, t, r), e.sheen && n.sheen.parse(e.sheen, t, r), e.subSurface && n.subSurface.parse(e.subSurface, t, r), e.iridescence && n.iridescence.parse(e.iridescence, t, r), n; } } C([ Oi("diffuse"), At("_markAllSubMeshesAsTexturesDirty", "_albedoColor") ], a4.prototype, "diffuseColor", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty", "_albedoTexture") ], a4.prototype, "diffuseTexture", void 0); C([ Oi("specular"), At("_markAllSubMeshesAsTexturesDirty", "_reflectivityColor") ], a4.prototype, "specularColor", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty", "_microSurface") ], a4.prototype, "glossiness", void 0); C([ en(), At("_markAllSubMeshesAsTexturesDirty", "_reflectivityTexture") ], a4.prototype, "specularGlossinessTexture", void 0); Ue("BABYLON.PBRSpecularGlossinessMaterial", a4); class PD extends ls { /** * Instantiates a ColorGradingTexture from the following parameters. * * @param url The location of the color grading data (currently only supporting 3dl) * @param sceneOrEngine The scene or engine the texture will be used in * @param onLoad defines a callback triggered when the texture has been loaded */ constructor(e, t, r = null) { if (super(t), !!e) if (this._textureMatrix = he.Identity(), this.name = e, this.url = e, this._onLoad = r, this._texture = this._getFromCache(e, !0), this._texture) this._triggerOnLoad(); else { const n = this.getScene(); n ? n.useDelayedTextureLoading ? this.delayLoadState = 4 : this._loadTexture() : this._loadTexture(); } } /** * Fires the onload event from the constructor if requested. */ _triggerOnLoad() { this._onLoad && this._onLoad(); } /** * Returns the texture matrix used in most of the material. * This is not used in color grading but keep for troubleshooting purpose (easily swap diffuse by colorgrading to look in). */ getTextureMatrix() { return this._textureMatrix; } /** * Occurs when the file being loaded is a .3dl LUT file. */ _load3dlTexture() { const e = this._getEngine(); let t; e._features.support3DTextures ? t = e.createRawTexture3D(null, 1, 1, 1, 5, !1, !1, 2, null, 0) : t = e.createRawTexture(null, 1, 1, 5, !1, !1, 2, null, 0), this._texture = t, this._texture.isReady = !1, this.isCube = !1, this.is3D = e._features.support3DTextures, this.wrapU = 0, this.wrapV = 0, this.wrapR = 0, this.anisotropicFilteringLevel = 1; const r = (i) => { if (typeof i != "string") return; let s = null, a = null, f; const o = i.split(` `); let d = 0, v = 0, u = 0, l = 0, P = 0; for (let p = 0; p < o.length; p++) { if (f = o[p], !PD._NoneEmptyLineRegex.test(f) || f.indexOf("#") === 0) continue; const c = f.split(" "); if (d === 0) { d = c.length, s = new Uint8Array(d * d * d * 4), a = new Float32Array(d * d * d * 4); continue; } if (d != 0) { const H = Math.max(parseInt(c[0]), 0), T = Math.max(parseInt(c[1]), 0), q = Math.max(parseInt(c[2]), 0); P = Math.max(H, P), P = Math.max(T, P), P = Math.max(q, P); const b = (v + l * d + u * d * d) * 4; a && (a[b + 0] = H, a[b + 1] = T, a[b + 2] = q), u++, u % d == 0 && (l++, u = 0, l % d == 0 && (v++, l = 0)); } } if (a && s) for (let p = 0; p < a.length; p++) if (p > 0 && (p + 1) % 4 === 0) s[p] = 255; else { const c = a[p]; s[p] = c / P * 255; } t.is3D ? (t.updateSize(d, d, d), e.updateRawTexture3D(t, s, 5, !1)) : (t.updateSize(d * d, d), e.updateRawTexture(t, s, 5, !1)), t.isReady = !0, this._triggerOnLoad(); }, n = this.getScene(); return n ? n._loadFile(this.url, r) : e._loadFile(this.url, r), this._texture; } /** * Starts the loading process of the texture. */ _loadTexture() { this.url && this.url.toLocaleLowerCase().indexOf(".3dl") == this.url.length - 4 && this._load3dlTexture(); } /** * Clones the color grading texture. */ clone() { const e = new PD(this.url, this.getScene() || this._getEngine()); return e.level = this.level, e; } /** * Called during delayed load for textures. */ delayLoad() { this.delayLoadState === 4 && (this.delayLoadState = 1, this._texture = this._getFromCache(this.url, !0), this._texture || this._loadTexture()); } /** * Parses a color grading texture serialized by Babylon. * @param parsedTexture The texture information being parsedTexture * @param scene The scene to load the texture in * @returns A color grading texture */ static Parse(e, t) { let r = null; return e.name && !e.isRenderTarget && (r = new PD(e.name, t), r.name = e.name, r.level = e.level), r; } /** * Serializes the LUT texture to json format. */ serialize() { if (!this.name) return null; const e = {}; return e.name = this.name, e.level = this.level, e.customType = "BABYLON.ColorGradingTexture", e; } } PD._NoneEmptyLineRegex = /\S+/; Ue("BABYLON.ColorGradingTexture", PD); class VS extends ls { /** * Instantiates an EquiRectangularCubeTexture from the following parameters. * @param url The location of the image * @param scene The scene the texture will be used in * @param size The cubemap desired size (the more it increases the longer the generation will be) * @param noMipmap Forces to not generate the mipmap if true * @param gammaSpace Specifies if the texture will be used in gamma or linear space * (the PBR material requires those textures in linear space, but the standard material would require them in Gamma space) * @param onLoad — defines a callback called when texture is loaded * @param onError — defines a callback called if there is an error */ constructor(e, t, r, n = !1, i = !0, s = null, a = null, f = !1) { if (super(t), this._onLoad = null, this._onError = null, !e) throw new Error("Image url is not set"); this._coordinatesMode = We.CUBIC_MODE, this.name = e, this.url = e, this._size = r, this._supersample = f, this._noMipmap = n, this.gammaSpace = i, this._onLoad = s, this._onError = a, this.hasAlpha = !1, this.isCube = !0, this._texture = this._getFromCache(e, this._noMipmap, void 0, void 0, void 0, this.isCube), this._texture ? s && (this._texture.isReady ? ye.SetImmediate(() => s()) : this._texture.onLoadedObservable.add(s)) : t.useDelayedTextureLoading ? this.delayLoadState = 4 : this._loadImage(() => this._loadTexture(), this._onError); } /** * Load the image data, by putting the image on a canvas and extracting its buffer. * @param loadTextureCallback * @param onError */ _loadImage(e, t) { const r = this.getScene(); if (!r) return; const n = r.getEngine().createRawCubeTexture(null, this._size, 4, r.getEngine().getCaps().textureFloat ? 1 : 7, this._noMipmap, !1, 3); n.generateMipMaps = !this._noMipmap, r.addPendingData(n), n.url = this.url, n.isReady = !1, r.getEngine()._internalTexturesCache.push(n), this._texture = n; const i = document.createElement("canvas"); nU(this.url, (s) => { this._width = s.width, this._height = s.height, i.width = this._width, i.height = this._height; const a = i.getContext("2d"); a.drawImage(s, 0, 0); const f = a.getImageData(0, 0, s.width, s.height); this._buffer = f.data.buffer, i.remove(), e(); }, (s, a) => { r.removePendingData(n), t && t(`${this.getClassName()} could not be loaded`, a); }, r ? r.offlineProvider : null); } /** * Convert the image buffer into a cubemap and create a CubeTexture. */ _loadTexture() { const e = this.getScene(), t = () => { const i = this._getFloat32ArrayFromArrayBuffer(this._buffer), s = wq.ConvertPanoramaToCubemap(i, this._width, this._height, this._size, this._supersample), a = []; for (let f = 0; f < 6; f++) { const o = s[VS._FacesMapping[f]]; a.push(o); } return a; }; if (!e) return; const r = t(), n = this._texture; e.getEngine().updateRawCubeTexture(n, r, n.format, n.type, n.invertY), n.isReady = !0, e.removePendingData(n), n.onLoadedObservable.notifyObservers(n), n.onLoadedObservable.clear(), this._onLoad && this._onLoad(); } /** * Convert the ArrayBuffer into a Float32Array and drop the transparency channel. * @param buffer The ArrayBuffer that should be converted. * @returns The buffer as Float32Array. */ _getFloat32ArrayFromArrayBuffer(e) { const t = new DataView(e), r = new Float32Array(e.byteLength * 3 / 4); let n = 0; for (let i = 0; i < e.byteLength; i++) (i + 1) % 4 !== 0 && (r[n++] = t.getUint8(i) / 255); return r; } /** * Get the current class name of the texture useful for serialization or dynamic coding. * @returns "EquiRectangularCubeTexture" */ getClassName() { return "EquiRectangularCubeTexture"; } /** * Create a clone of the current EquiRectangularCubeTexture and return it. * @returns A clone of the current EquiRectangularCubeTexture. */ clone() { const e = this.getScene(); if (!e) return this; const t = new VS(this.url, e, this._size, this._noMipmap, this.gammaSpace); return t.level = this.level, t.wrapU = this.wrapU, t.wrapV = this.wrapV, t.coordinatesIndex = this.coordinatesIndex, t.coordinatesMode = this.coordinatesMode, t; } } VS._FacesMapping = ["right", "left", "up", "down", "front", "back"]; class cy extends ls { /** * Instantiates a HtmlElementTexture from the following parameters. * * @param name Defines the name of the texture * @param element Defines the video or canvas the texture is filled with * @param options Defines the other none mandatory texture creation options */ constructor(e, t, r) { var n, i; super(r.scene || r.engine), this.onLoadObservable = new Oe(), !(!t || !r.engine && !r.scene) && (r = Object.assign(Object.assign({}, cy._DefaultOptions), r), this._generateMipMaps = r.generateMipMaps, this._samplingMode = r.samplingMode, this._textureMatrix = he.Identity(), this._format = r.format, this.name = e, this.element = t, this._isVideo = !!t.getVideoPlaybackQuality, this._externalTexture = this._isVideo && (i = (n = this._engine) === null || n === void 0 ? void 0 : n.createExternalTexture(t)) !== null && i !== void 0 ? i : null, this.anisotropicFilteringLevel = 1, this._createInternalTexture()); } _createInternalTexture() { let e = 0, t = 0; this._isVideo ? (e = this.element.videoWidth, t = this.element.videoHeight) : (e = this.element.width, t = this.element.height); const r = this._getEngine(); r && (this._texture = r.createDynamicTexture(e, t, this._generateMipMaps, this._samplingMode), this._texture.format = this._format), this.update(); } /** * Returns the texture matrix used in most of the material. */ getTextureMatrix() { return this._textureMatrix; } /** * Updates the content of the texture. * @param invertY Defines whether the texture should be inverted on Y (false by default on video and true on canvas) */ update(e = null) { const t = this._getEngine(); if (this._texture == null || t == null) return; const r = this.isReady(); if (this._isVideo) { const n = this.element; if (n.readyState < n.HAVE_CURRENT_DATA) return; t.updateVideoTexture(this._texture, this._externalTexture ? this._externalTexture : n, e === null ? !0 : e); } else { const n = this.element; t.updateDynamicTexture(this._texture, n, e === null ? !0 : e, !1, this._format); } !r && this.isReady() && this.onLoadObservable.notifyObservers(this); } /** * Dispose the texture and release its associated resources. */ dispose() { this.onLoadObservable.clear(), super.dispose(); } } cy._DefaultOptions = { generateMipMaps: !1, samplingMode: 2, format: 5, engine: null, scene: null }; const xpe = 1, Dpe = 2, jpe = 3, wpe = 9, mpe = 10, Bpe = 11, Wpe = 48, Spe = 4, Upe = 0, Ipe = 1, Rpe = 2, Vpe = 3; function sV(A) { let e = 0; return { id_length: A[e++], colormap_type: A[e++], image_type: A[e++], colormap_index: A[e++] | A[e++] << 8, colormap_length: A[e++] | A[e++] << 8, colormap_size: A[e++], origin: [A[e++] | A[e++] << 8, A[e++] | A[e++] << 8], width: A[e++] | A[e++] << 8, height: A[e++] | A[e++] << 8, pixel_size: A[e++], flags: A[e++] }; } function eY(A, e) { if (e.length < 19) { Se.Error("Unable to load TGA file - Not enough data to contain header"); return; } let t = 18; const r = sV(e); if (r.id_length + t > e.length) { Se.Error("Unable to load TGA file - Not enough data"); return; } t += r.id_length; let n = !1, i = !1, s = !1; switch (r.image_type) { case wpe: n = !0; case xpe: i = !0; break; case mpe: n = !0; case Dpe: break; case Bpe: n = !0; case jpe: s = !0; break; } let a; const f = r.pixel_size >> 3, o = r.width * r.height * f; let d; if (i && (d = e.subarray(t, t += r.colormap_length * (r.colormap_size >> 3))), n) { a = new Uint8Array(o); let b, j, w, m = 0; const I = new Uint8Array(f); for (; t < o && m < o; ) if (b = e[t++], j = (b & 127) + 1, b & 128) { for (w = 0; w < f; ++w) I[w] = e[t++]; for (w = 0; w < j; ++w) a.set(I, m + w * f); m += f * j; } else { for (j *= f, w = 0; w < j; ++w) a[m + w] = e[t++]; m += j; } } else a = e.subarray(t, t += i ? r.width * r.height : o); let v, u, l, P, p, c; switch ((r.flags & Wpe) >> Spe) { default: case Rpe: v = 0, l = 1, c = r.width, u = 0, P = 1, p = r.height; break; case Upe: v = 0, l = 1, c = r.width, u = r.height - 1, P = -1, p = -1; break; case Vpe: v = r.width - 1, l = -1, c = -1, u = 0, P = 1, p = r.height; break; case Ipe: v = r.width - 1, l = -1, c = -1, u = r.height - 1, P = -1, p = -1; break; } const H = "_getImageData" + (s ? "Grey" : "") + r.pixel_size + "bits", T = Zte[H](r, d, a, u, P, p, v, l, c); A.getEngine()._uploadDataToTextureDirectly(A, T); } function Cpe(A, e, t, r, n, i, s, a, f) { const o = t, d = e, v = A.width, u = A.height; let l, P = 0, p, c; const H = new Uint8Array(v * u * 4); for (c = r; c !== i; c += n) for (p = s; p !== f; p += a, P++) l = o[P], H[(p + v * c) * 4 + 3] = 255, H[(p + v * c) * 4 + 2] = d[l * 3 + 0], H[(p + v * c) * 4 + 1] = d[l * 3 + 1], H[(p + v * c) * 4 + 0] = d[l * 3 + 2]; return H; } function Ope(A, e, t, r, n, i, s, a, f) { const o = t, d = A.width, v = A.height; let u, l = 0, P, p; const c = new Uint8Array(d * v * 4); for (p = r; p !== i; p += n) for (P = s; P !== f; P += a, l += 2) { u = o[l + 0] + (o[l + 1] << 8); const H = ((u & 31744) >> 10) * 255 / 31 | 0, T = ((u & 992) >> 5) * 255 / 31 | 0, q = (u & 31) * 255 / 31 | 0; c[(P + d * p) * 4 + 0] = H, c[(P + d * p) * 4 + 1] = T, c[(P + d * p) * 4 + 2] = q, c[(P + d * p) * 4 + 3] = u & 32768 ? 0 : 255; } return c; } function ype(A, e, t, r, n, i, s, a, f) { const o = t, d = A.width, v = A.height; let u = 0, l, P; const p = new Uint8Array(d * v * 4); for (P = r; P !== i; P += n) for (l = s; l !== f; l += a, u += 3) p[(l + d * P) * 4 + 3] = 255, p[(l + d * P) * 4 + 2] = o[u + 0], p[(l + d * P) * 4 + 1] = o[u + 1], p[(l + d * P) * 4 + 0] = o[u + 2]; return p; } function kpe(A, e, t, r, n, i, s, a, f) { const o = t, d = A.width, v = A.height; let u = 0, l, P; const p = new Uint8Array(d * v * 4); for (P = r; P !== i; P += n) for (l = s; l !== f; l += a, u += 4) p[(l + d * P) * 4 + 2] = o[u + 0], p[(l + d * P) * 4 + 1] = o[u + 1], p[(l + d * P) * 4 + 0] = o[u + 2], p[(l + d * P) * 4 + 3] = o[u + 3]; return p; } function Epe(A, e, t, r, n, i, s, a, f) { const o = t, d = A.width, v = A.height; let u, l = 0, P, p; const c = new Uint8Array(d * v * 4); for (p = r; p !== i; p += n) for (P = s; P !== f; P += a, l++) u = o[l], c[(P + d * p) * 4 + 0] = u, c[(P + d * p) * 4 + 1] = u, c[(P + d * p) * 4 + 2] = u, c[(P + d * p) * 4 + 3] = 255; return c; } function Fpe(A, e, t, r, n, i, s, a, f) { const o = t, d = A.width, v = A.height; let u = 0, l, P; const p = new Uint8Array(d * v * 4); for (P = r; P !== i; P += n) for (l = s; l !== f; l += a, u += 2) p[(l + d * P) * 4 + 0] = o[u + 0], p[(l + d * P) * 4 + 1] = o[u + 0], p[(l + d * P) * 4 + 2] = o[u + 0], p[(l + d * P) * 4 + 3] = o[u + 1]; return p; } const Zte = { /** * Gets the header of a TGA file * @param data defines the TGA data * @returns the header */ GetTGAHeader: sV, /** * Uploads TGA content to a Babylon Texture * @internal */ UploadContent: eY, /** @internal */ _getImageData8bits: Cpe, /** @internal */ _getImageData16bits: Ope, /** @internal */ _getImageData24bits: ype, /** @internal */ _getImageData32bits: kpe, /** @internal */ _getImageDataGrey8bits: Epe, /** @internal */ _getImageDataGrey16bits: Fpe }; class _te { constructor() { this.supportCascades = !1; } /** * This returns if the loader support the current file information. * @param extension defines the file extension of the file being loaded * @returns true if the loader can load the specified file */ canLoad(e) { return e.endsWith(".tga"); } /** * Uploads the cube texture data to the WebGL texture. It has already been bound. */ loadCubeData() { throw ".env not supported in Cube."; } /** * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback. * @param data contains the texture data * @param texture defines the BabylonJS internal texture * @param callback defines the method to call once ready to upload */ loadData(e, t, r) { const n = new Uint8Array(e.buffer, e.byteOffset, e.byteLength), i = sV(n); r(i.width, i.height, t.generateMipMaps, !1, () => { eY(t, n); }); } } Ge._TextureLoaders.push(new _te()); class $te { constructor() { this.supportCascades = !1; } /** * This returns if the loader support the current file information. * @param extension defines the file extension of the file being loaded * @returns true if the loader can load the specified file */ canLoad(e) { return e.endsWith(".hdr"); } /** * Uploads the cube texture data to the WebGL texture. It has already been bound. */ loadCubeData() { throw ".env not supported in Cube."; } /** * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback. * @param data contains the texture data * @param texture defines the BabylonJS internal texture * @param callback defines the method to call once ready to upload */ loadData(e, t, r) { const n = new Uint8Array(e.buffer, e.byteOffset, e.byteLength), i = iO.RGBE_ReadHeader(n), s = iO.RGBE_ReadPixels(n, i), a = i.width * i.height, f = new Float32Array(a * 4); for (let o = 0; o < a; o += 1) f[o * 4] = s[o * 3], f[o * 4 + 1] = s[o * 3 + 1], f[o * 4 + 2] = s[o * 3 + 2], f[o * 4 + 3] = 1; r(i.width, i.height, t.generateMipMaps, !1, () => { const o = t.getEngine(); t.type = 1, t.format = 5, t._gammaSpace = !1, o._uploadDataToTextureDirectly(t, f); }); } } Ge._TextureLoaders.push(new $te()); class Npe { } var _8; (function(A) { A[A.cTFETC1 = 0] = "cTFETC1", A[A.cTFETC2 = 1] = "cTFETC2", A[A.cTFBC1 = 2] = "cTFBC1", A[A.cTFBC3 = 3] = "cTFBC3", A[A.cTFBC4 = 4] = "cTFBC4", A[A.cTFBC5 = 5] = "cTFBC5", A[A.cTFBC7 = 6] = "cTFBC7", A[A.cTFPVRTC1_4_RGB = 8] = "cTFPVRTC1_4_RGB", A[A.cTFPVRTC1_4_RGBA = 9] = "cTFPVRTC1_4_RGBA", A[A.cTFASTC_4x4 = 10] = "cTFASTC_4x4", A[A.cTFATC_RGB = 11] = "cTFATC_RGB", A[A.cTFATC_RGBA_INTERPOLATED_ALPHA = 12] = "cTFATC_RGBA_INTERPOLATED_ALPHA", A[A.cTFRGBA32 = 13] = "cTFRGBA32", A[A.cTFRGB565 = 14] = "cTFRGB565", A[A.cTFBGR565 = 15] = "cTFBGR565", A[A.cTFRGBA4444 = 16] = "cTFRGBA4444", A[A.cTFFXT1_RGB = 17] = "cTFFXT1_RGB", A[A.cTFPVRTC2_4_RGB = 18] = "cTFPVRTC2_4_RGB", A[A.cTFPVRTC2_4_RGBA = 19] = "cTFPVRTC2_4_RGBA", A[A.cTFETC2_EAC_R11 = 20] = "cTFETC2_EAC_R11", A[A.cTFETC2_EAC_RG11 = 21] = "cTFETC2_EAC_RG11"; })(_8 || (_8 = {})); const qq = { /** * URL to use when loading the basis transcoder */ JSModuleURL: `${ye._DefaultCdnUrl}/basisTranscoder/1/basis_transcoder.js`, /** * URL to use when loading the wasm module for the transcoder */ WasmModuleURL: `${ye._DefaultCdnUrl}/basisTranscoder/1/basis_transcoder.wasm` }, ere = (A, e) => { let t; switch (A) { case _8.cTFETC1: t = 36196; break; case _8.cTFBC1: t = 33776; break; case _8.cTFBC4: t = 33779; break; case _8.cTFASTC_4x4: t = 37808; break; case _8.cTFETC2: t = 37496; break; case _8.cTFBC7: t = 36492; break; } if (t === void 0) throw "The chosen Basis transcoder format is not currently supported"; return t; }; let UE = null, U2 = null, Qpe = 0; const Ype = !1, Mpe = () => (UE || (UE = new Promise((A, e) => { U2 ? A(U2) : ye.LoadFileAsync(ye.GetBabylonScriptURL(qq.WasmModuleURL)).then((t) => { if (typeof URL != "function") return e("Basis transcoder requires an environment with a URL constructor"); const r = URL.createObjectURL(new Blob([`(${Lpe})()`], { type: "application/javascript" })); U2 = new Worker(r); const n = (i) => { i.data.action === "init" ? (U2.removeEventListener("message", n), A(U2)) : i.data.action === "error" && e(i.data.error || "error initializing worker"); }; U2.addEventListener("message", n), U2.postMessage({ action: "init", url: ye.GetBabylonScriptURL(qq.JSModuleURL), wasmBinary: t }); }).catch(e); })), UE), oO = (A, e) => { const t = A instanceof ArrayBuffer ? new Uint8Array(A) : A; return new Promise((r, n) => { Mpe().then(() => { const i = Qpe++, s = (f) => { f.data.action === "transcode" && f.data.id === i && (U2.removeEventListener("message", s), f.data.success ? r(f.data) : n("Transcode is not supported on this device")); }; U2.addEventListener("message", s); const a = new Uint8Array(t.byteLength); a.set(new Uint8Array(t.buffer, t.byteOffset, t.byteLength)), U2.postMessage({ action: "transcode", id: i, imageData: a, config: e, ignoreSupportedFormats: Ype }, [ a.buffer ]); }, (i) => { n(i); }); }); }, XC = (A, e) => { var t, r; let n = (t = e._gl) === null || t === void 0 ? void 0 : t.TEXTURE_2D; A.isCube && (n = (r = e._gl) === null || r === void 0 ? void 0 : r.TEXTURE_CUBE_MAP), e._bindTextureDirectly(n, A, !0); }, fO = (A, e) => { const t = A.getEngine(); for (let r = 0; r < e.fileInfo.images.length; r++) { const n = e.fileInfo.images[r].levels[0]; if (A._invertVScale = A.invertY, e.format === -1 || e.format === _8.cTFRGB565) if (A.type = 10, A.format = 4, t._features.basisNeedsPOT && (Xt.Log2(n.width) % 1 !== 0 || Xt.Log2(n.height) % 1 !== 0)) { const i = new As(t, ri.Temp); A._invertVScale = A.invertY, i.type = 10, i.format = 4, i.width = n.width + 3 & -4, i.height = n.height + 3 & -4, XC(i, t), t._uploadDataToTextureDirectly(i, new Uint16Array(n.transcodedPixels.buffer), r, 0, 4, !0), t._rescaleTexture(i, A, t.scenes[0], t._getInternalFormat(4), () => { t._releaseTexture(i), XC(A, t); }); } else A._invertVScale = !A.invertY, A.width = n.width + 3 & -4, A.height = n.height + 3 & -4, A.samplingMode = 2, XC(A, t), t._uploadDataToTextureDirectly(A, new Uint16Array(n.transcodedPixels.buffer), r, 0, 4, !0); else { A.width = n.width, A.height = n.height, A.generateMipMaps = e.fileInfo.images[r].levels.length > 1; const i = py.GetInternalFormatFromBasisFormat(e.format, t); A.format = i, XC(A, t), e.fileInfo.images[r].levels.forEach((s, a) => { t._uploadCompressedDataToTextureDirectly(A, i, s.width, s.height, s.transcodedPixels, r, a); }), t._features.basisNeedsPOT && (Xt.Log2(A.width) % 1 !== 0 || Xt.Log2(A.height) % 1 !== 0) && (ye.Warn("Loaded .basis texture width and height are not a power of two. Texture wrapping will be set to Texture.CLAMP_ADDRESSMODE as other modes are not supported with non power of two dimensions in webGL 1."), A._cachedWrapU = We.CLAMP_ADDRESSMODE, A._cachedWrapV = We.CLAMP_ADDRESSMODE); } } }, py = { /** * URL to use when loading the basis transcoder */ JSModuleURL: qq.JSModuleURL, /** * URL to use when loading the wasm module for the transcoder */ WasmModuleURL: qq.WasmModuleURL, /** * Get the internal format to be passed to texImage2D corresponding to the .basis format value * @param basisFormat format chosen from GetSupportedTranscodeFormat * @returns internal format corresponding to the Basis format */ GetInternalFormatFromBasisFormat: ere, /** * Transcodes a loaded image file to compressed pixel data * @param data image data to transcode * @param config configuration options for the transcoding * @returns a promise resulting in the transcoded image */ TranscodeAsync: oO, /** * Loads a texture from the transcode result * @param texture texture load to * @param transcodeResult the result of transcoding the basis file to load from */ LoadTextureFromTranscodeResult: fO }; function Lpe() { const A = { cTFETC1: 0, cTFETC2: 1, cTFBC1: 2, cTFBC3: 3, cTFBC4: 4, cTFBC5: 5, cTFBC7: 6, cTFPVRTC1_4_RGB: 8, cTFPVRTC1_4_RGBA: 9, cTFASTC_4x4: 10, cTFATC_RGB: 11, cTFATC_RGBA_INTERPOLATED_ALPHA: 12, cTFRGBA32: 13, cTFRGB565: 14, cTFBGR565: 15, cTFRGBA4444: 16, cTFFXT1_RGB: 17, cTFPVRTC2_4_RGB: 18, cTFPVRTC2_4_RGBA: 19, cTFETC2_EAC_R11: 20, cTFETC2_EAC_RG11: 21 }; let e = null; onmessage = (s) => { if (s.data.action === "init") { if (!e) { try { importScripts(s.data.url); } catch (a) { postMessage({ action: "error", error: a }); } e = BASIS({ // Override wasm binary wasmBinary: s.data.wasmBinary }); } e !== null && e.then((a) => { BASIS = a, a.initializeBasis(), postMessage({ action: "init" }); }); } else if (s.data.action === "transcode") { const a = s.data.config, f = s.data.imageData, o = new BASIS.BasisFile(f), d = r(o); let v = s.data.ignoreSupportedFormats ? null : t(s.data.config, d), u = !1; v === null && (u = !0, v = d.hasAlpha ? A.cTFBC3 : A.cTFBC1); let l = !0; o.startTranscoding() || (l = !1); const P = []; for (let p = 0; p < d.images.length && l; p++) { const c = d.images[p]; if (a.loadSingleImage === void 0 || a.loadSingleImage === p) { let H = c.levels.length; a.loadMipmapLevels === !1 && (H = 1); for (let T = 0; T < H; T++) { const q = c.levels[T], b = n(o, p, T, v, u); if (!b) { l = !1; break; } q.transcodedPixels = b, P.push(q.transcodedPixels.buffer); } } } o.close(), o.delete(), u && (v = -1), l ? postMessage({ action: "transcode", success: l, id: s.data.id, fileInfo: d, format: v }, P) : postMessage({ action: "transcode", success: l, id: s.data.id }); } }; function t(s, a) { let f = null; return s.supportedCompressionFormats && (s.supportedCompressionFormats.astc ? f = A.cTFASTC_4x4 : s.supportedCompressionFormats.bc7 ? f = A.cTFBC7 : s.supportedCompressionFormats.s3tc ? f = a.hasAlpha ? A.cTFBC3 : A.cTFBC1 : s.supportedCompressionFormats.pvrtc ? f = a.hasAlpha ? A.cTFPVRTC1_4_RGBA : A.cTFPVRTC1_4_RGB : s.supportedCompressionFormats.etc2 ? f = A.cTFETC2 : s.supportedCompressionFormats.etc1 ? f = A.cTFETC1 : f = A.cTFRGB565), f; } function r(s) { const a = s.getHasAlpha(), f = s.getNumImages(), o = []; for (let v = 0; v < f; v++) { const u = { levels: [] }, l = s.getNumLevels(v); for (let P = 0; P < l; P++) { const p = { width: s.getImageWidth(v, P), height: s.getImageHeight(v, P) }; u.levels.push(p); } o.push(u); } return { hasAlpha: a, images: o }; } function n(s, a, f, o, d) { const v = s.getImageTranscodedSizeInBytes(a, f, o); let u = new Uint8Array(v); if (!s.transcodeImage(u, a, f, o, 1, 0)) return null; if (d) { const l = s.getImageWidth(a, f) + 3 & -4, P = s.getImageHeight(a, f) + 3 & -4; u = i(u, 0, l, P); } return u; } function i(s, a, f, o) { const d = new Uint16Array(4), v = new Uint16Array(f * o), u = f / 4, l = o / 4; for (let P = 0; P < l; P++) for (let p = 0; p < u; p++) { const c = a + 8 * (P * u + p); d[0] = s[c] | s[c + 1] << 8, d[1] = s[c + 2] | s[c + 3] << 8, d[2] = (2 * (d[0] & 31) + 1 * (d[1] & 31)) / 3 | (2 * (d[0] & 2016) + 1 * (d[1] & 2016)) / 3 & 2016 | (2 * (d[0] & 63488) + 1 * (d[1] & 63488)) / 3 & 63488, d[3] = (2 * (d[1] & 31) + 1 * (d[0] & 31)) / 3 | (2 * (d[1] & 2016) + 1 * (d[0] & 2016)) / 3 & 2016 | (2 * (d[1] & 63488) + 1 * (d[0] & 63488)) / 3 & 63488; for (let H = 0; H < 4; H++) { const T = s[c + 4 + H]; let q = (P * 4 + H) * f + p * 4; v[q++] = d[T & 3], v[q++] = d[T >> 2 & 3], v[q++] = d[T >> 4 & 3], v[q++] = d[T >> 6 & 3]; } } return v; } } Object.defineProperty(py, "JSModuleURL", { get: function() { return qq.JSModuleURL; }, set: function(A) { qq.JSModuleURL = A; } }); Object.defineProperty(py, "WasmModuleURL", { get: function() { return qq.WasmModuleURL; }, set: function(A) { qq.WasmModuleURL = A; } }); class tre { constructor() { this.supportCascades = !1; } /** * This returns if the loader support the current file information. * @param extension defines the file extension of the file being loaded * @returns true if the loader can load the specified file */ canLoad(e) { return e.endsWith(".basis"); } /** * Uploads the cube texture data to the WebGL texture. It has already been bound. * @param data contains the texture data * @param texture defines the BabylonJS internal texture * @param createPolynomials will be true if polynomials have been requested * @param onLoad defines the callback to trigger once the texture is ready * @param onError defines the callback to trigger in case of error */ loadCubeData(e, t, r, n, i) { if (Array.isArray(e)) return; const s = t.getEngine().getCaps(), a = { supportedCompressionFormats: { etc1: !!s.etc1, s3tc: !!s.s3tc, pvrtc: !!s.pvrtc, etc2: !!s.etc2, astc: !!s.astc, bc7: !!s.bptc } }; oO(e, a).then((f) => { const o = f.fileInfo.images[0].levels.length > 1 && t.generateMipMaps; fO(t, f), t.getEngine()._setCubeMapTextureParams(t, o), t.isReady = !0, t.onLoadedObservable.notifyObservers(t), t.onLoadedObservable.clear(), n && n(); }).catch((f) => { const o = "Failed to transcode Basis file, transcoding may not be supported on this device"; ye.Warn(o), t.isReady = !0, i && i(f); }); } /** * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback. * @param data contains the texture data * @param texture defines the BabylonJS internal texture * @param callback defines the method to call once ready to upload */ loadData(e, t, r) { const n = t.getEngine().getCaps(), i = { supportedCompressionFormats: { etc1: !!n.etc1, s3tc: !!n.s3tc, pvrtc: !!n.pvrtc, etc2: !!n.etc2, astc: !!n.astc, bc7: !!n.bptc } }; oO(e, i).then((s) => { const a = s.fileInfo.images[0].levels[0], f = s.fileInfo.images[0].levels.length > 1 && t.generateMipMaps; r(a.width, a.height, f, s.format !== -1, () => { fO(t, s); }); }).catch((s) => { ye.Warn("Failed to transcode Basis file, transcoding may not be supported on this device"), ye.Warn(`Failed to transcode Basis file: ${s}`), r(0, 0, !1, !1, () => { }, !0); }); } } Ge._TextureLoaders.push(new tre()); class Zx extends Ta { /** * Get if draw buffers (render textures) are currently supported by the used hardware and browser. */ get isSupported() { var e, t; return (t = (e = this._engine) === null || e === void 0 ? void 0 : e.getCaps().drawBuffersExtension) !== null && t !== void 0 ? t : !1; } /** * Get the list of textures generated by the multi render target. */ get textures() { return this._textures; } /** * Gets the number of textures in this MRT. This number can be different from `_textures.length` in case a depth texture is generated. */ get count() { return this._count; } /** * Get the depth texture generated by the multi render target if options.generateDepthTexture has been set */ get depthTexture() { return this._textures[this._textures.length - 1]; } /** * Set the wrapping mode on U of all the textures we are rendering to. * Can be any of the Texture. (CLAMP_ADDRESSMODE, MIRROR_ADDRESSMODE or WRAP_ADDRESSMODE) */ set wrapU(e) { if (this._textures) for (let t = 0; t < this._textures.length; t++) this._textures[t].wrapU = e; } /** * Set the wrapping mode on V of all the textures we are rendering to. * Can be any of the Texture. (CLAMP_ADDRESSMODE, MIRROR_ADDRESSMODE or WRAP_ADDRESSMODE) */ set wrapV(e) { if (this._textures) for (let t = 0; t < this._textures.length; t++) this._textures[t].wrapV = e; } /** * Instantiate a new multi render target texture. * A multi render target, like a render target provides the ability to render to a texture. * Unlike the render target, it can render to several draw buffers (render textures) in one draw. * This is specially interesting in deferred rendering or for any effects requiring more than * just one color from a single pass. * @param name Define the name of the texture * @param size Define the size of the buffers to render to * @param count Define the number of target we are rendering into * @param scene Define the scene the texture belongs to * @param options Define the options used to create the multi render target * @param textureNames Define the names to set to the textures (if count \> 0 - optional) */ constructor(e, t, r, n, i, s) { const a = i && i.generateMipMaps ? i.generateMipMaps : !1, f = i && i.generateDepthTexture ? i.generateDepthTexture : !1, o = i && i.depthTextureFormat ? i.depthTextureFormat : 15, d = !i || i.doNotChangeAspectRatio === void 0 ? !0 : i.doNotChangeAspectRatio, v = i && i.drawOnlyOnFirstAttachmentByDefault ? i.drawOnlyOnFirstAttachmentByDefault : !1; if (super(e, t, n, a, d, void 0, void 0, void 0, void 0, void 0, void 0, void 0, !0), !this.isSupported) { this.dispose(); return; } this._textureNames = s; const u = [], l = [], P = [], p = [], c = [], H = [], T = [], q = []; this._initTypes(r, u, l, P, p, c, H, T, q, i); const b = !i || i.generateDepthBuffer === void 0 ? !0 : i.generateDepthBuffer, j = !i || i.generateStencilBuffer === void 0 ? !1 : i.generateStencilBuffer; this._multiRenderTargetOptions = { samplingModes: l, generateMipMaps: a, generateDepthBuffer: b, generateStencilBuffer: j, generateDepthTexture: f, depthTextureFormat: o, types: u, textureCount: r, useSRGBBuffers: P, formats: p, targetTypes: c, faceIndex: H, layerIndex: T, layerCounts: q, labels: s }, this._count = r, this._drawOnlyOnFirstAttachmentByDefault = v, r > 0 && (this._createInternalTextures(), this._createTextures(s)); } _initTypes(e, t, r, n, i, s, a, f, o, d) { for (let v = 0; v < e; v++) d && d.types && d.types[v] !== void 0 ? t.push(d.types[v]) : t.push(d && d.defaultType ? d.defaultType : 0), d && d.samplingModes && d.samplingModes[v] !== void 0 ? r.push(d.samplingModes[v]) : r.push(We.BILINEAR_SAMPLINGMODE), d && d.useSRGBBuffers && d.useSRGBBuffers[v] !== void 0 ? n.push(d.useSRGBBuffers[v]) : n.push(!1), d && d.formats && d.formats[v] !== void 0 ? i.push(d.formats[v]) : i.push(5), d && d.targetTypes && d.targetTypes[v] !== void 0 ? s.push(d.targetTypes[v]) : s.push(3553), d && d.faceIndex && d.faceIndex[v] !== void 0 ? a.push(d.faceIndex[v]) : a.push(0), d && d.layerIndex && d.layerIndex[v] !== void 0 ? f.push(d.layerIndex[v]) : f.push(0), d && d.layerCounts && d.layerCounts[v] !== void 0 ? o.push(d.layerCounts[v]) : o.push(1); } _createInternaTextureIndexMapping() { const e = {}, t = []; if (!this._renderTarget) return t; const r = this._renderTarget.textures; for (let n = 0; n < r.length; n++) { const i = r[n]; if (!i) continue; const s = e[i.uniqueId]; s !== void 0 ? t[n] = s : e[i.uniqueId] = n; } return t; } /** * @internal */ _rebuild(e = !1, t) { if (this._count < 1) return; const r = this._createInternaTextureIndexMapping(); this.releaseInternalTextures(), this._createInternalTextures(), e && (this._releaseTextures(), this._createTextures(t)); const n = this._renderTarget.textures; for (let i = 0; i < n.length; i++) { const s = this._textures[i]; r[i] !== void 0 && this._renderTarget.setTexture(n[r[i]], i), s._texture = n[i], s._texture && (s._noMipmap = !s._texture.useMipMaps, s._useSRGBBuffer = s._texture._useSRGBBuffer); } this.samples !== 1 && this._renderTarget.setSamples(this.samples, !this._drawOnlyOnFirstAttachmentByDefault, !0); } _createInternalTextures() { this._renderTarget = this._getEngine().createMultipleRenderTarget(this._size, this._multiRenderTargetOptions, !this._drawOnlyOnFirstAttachmentByDefault), this._texture = this._renderTarget.texture; } _releaseTextures() { if (this._textures) for (let e = 0; e < this._textures.length; e++) this._textures[e]._texture = null, this._textures[e].dispose(); } _createTextures(e) { const t = this._renderTarget.textures; this._textures = []; for (let r = 0; r < t.length; r++) { const n = new We(null, this.getScene()); e != null && e[r] && (n.name = e[r]), n._texture = t[r], n._texture && (n._noMipmap = !n._texture.useMipMaps, n._useSRGBBuffer = n._texture._useSRGBBuffer), this._textures.push(n); } } /** * Replaces an internal texture within the MRT. Useful to share textures between MultiRenderTarget. * @param texture The new texture to set in the MRT * @param index The index of the texture to replace * @param disposePrevious Set to true if the previous internal texture should be disposed */ setInternalTexture(e, t, r = !0) { var n, i; if (this.renderTarget && (t === 0 && (this._texture = e), this.renderTarget.setTexture(e, t, r), this.textures[t] || (this.textures[t] = new We(null, this.getScene()), this.textures[t].name = (i = (n = this._textureNames) === null || n === void 0 ? void 0 : n[t]) !== null && i !== void 0 ? i : this.textures[t].name), this.textures[t]._texture = e, this.textures[t]._noMipmap = !e.useMipMaps, this.textures[t]._useSRGBBuffer = e._useSRGBBuffer, this._count = this.renderTarget.textures ? this.renderTarget.textures.length : 0, this._multiRenderTargetOptions.types && (this._multiRenderTargetOptions.types[t] = e.type), this._multiRenderTargetOptions.samplingModes && (this._multiRenderTargetOptions.samplingModes[t] = e.samplingMode), this._multiRenderTargetOptions.useSRGBBuffers && (this._multiRenderTargetOptions.useSRGBBuffers[t] = e._useSRGBBuffer), this._multiRenderTargetOptions.targetTypes && this._multiRenderTargetOptions.targetTypes[t] !== -1)) { let s = 0; e.is2DArray ? s = 35866 : e.isCube ? s = 34067 : e.is3D ? s = 32879 : s = 3553, this._multiRenderTargetOptions.targetTypes[t] = s; } } /** * Changes an attached texture's face index or layer. * @param index The index of the texture to modify the attachment of * @param layerIndex The layer index of the texture to be attached to the framebuffer * @param faceIndex The face index of the texture to be attached to the framebuffer */ setLayerAndFaceIndex(e, t = -1, r = -1) { !this.textures[e] || !this.renderTarget || (this._multiRenderTargetOptions.layerIndex && (this._multiRenderTargetOptions.layerIndex[e] = t), this._multiRenderTargetOptions.faceIndex && (this._multiRenderTargetOptions.faceIndex[e] = r), this.renderTarget.setLayerAndFaceIndex(e, t, r)); } /** * Changes every attached texture's face index or layer. * @param layerIndices The layer indices of the texture to be attached to the framebuffer * @param faceIndices The face indices of the texture to be attached to the framebuffer */ setLayerAndFaceIndices(e, t) { this.renderTarget && (this._multiRenderTargetOptions.layerIndex = e, this._multiRenderTargetOptions.faceIndex = t, this.renderTarget.setLayerAndFaceIndices(e, t)); } /** * Define the number of samples used if MSAA is enabled. */ get samples() { return this._samples; } set samples(e) { this._renderTarget ? this._samples = this._renderTarget.setSamples(e) : this._samples = e; } /** * Resize all the textures in the multi render target. * Be careful as it will recreate all the data in the new texture. * @param size Define the new size */ resize(e) { this._processSizeParameter(e, !1), this._rebuild(void 0, this._textureNames); } /** * Changes the number of render targets in this MRT * Be careful as it will recreate all the data in the new texture. * @param count new texture count * @param options Specifies texture types and sampling modes for new textures * @param textureNames Specifies the names of the textures (optional) */ updateCount(e, t, r) { this._multiRenderTargetOptions.textureCount = e, this._count = e; const n = [], i = [], s = [], a = [], f = [], o = [], d = [], v = []; this._textureNames = r, this._initTypes(e, n, i, s, a, f, o, d, v, t), this._multiRenderTargetOptions.types = n, this._multiRenderTargetOptions.samplingModes = i, this._multiRenderTargetOptions.useSRGBBuffers = s, this._multiRenderTargetOptions.formats = a, this._multiRenderTargetOptions.targetTypes = f, this._multiRenderTargetOptions.faceIndex = o, this._multiRenderTargetOptions.layerIndex = d, this._multiRenderTargetOptions.layerCounts = v, this._multiRenderTargetOptions.labels = r, this._rebuild(!0, r); } _unbindFrameBuffer(e, t) { this._renderTarget && e.unBindMultiColorAttachmentFramebuffer(this._renderTarget, this.isCube, () => { this.onAfterRenderObservable.notifyObservers(t); }); } /** * Dispose the render targets and their associated resources * @param doNotDisposeInternalTextures */ dispose(e = !1) { this._releaseTextures(), e ? this._texture = null : this.releaseInternalTextures(), super.dispose(); } /** * Release all the underlying texture used as draw buffers (render textures). */ releaseInternalTextures() { var e, t; const r = (e = this._renderTarget) === null || e === void 0 ? void 0 : e.textures; if (r) { for (let n = r.length - 1; n >= 0; n--) this._textures[n]._texture = null; (t = this._renderTarget) === null || t === void 0 || t.dispose(), this._renderTarget = null; } } } class XF { /** * Initializes a texture package frame. * @param id The numerical frame identifier * @param scale Scalar Vector2 for UV frame * @param offset Vector2 for the frame position in UV units. * @returns TexturePackerFrame */ constructor(e, t, r) { this.id = e, this.scale = t, this.offset = r; } } class I0 { /** * Initializes a texture package series from an array of meshes or a single mesh. * @param name The name of the package * @param meshes The target meshes to compose the package from * @param options The arguments that texture packer should follow while building. * @param scene The scene which the textures are scoped to. * @returns TexturePacker */ constructor(e, t, r, n) { var i, s, a, f, o, d, v, u, l, P, p, c, H; return this.name = e, this.meshes = t, this.scene = n, this.options = r, this.options.map = (i = this.options.map) !== null && i !== void 0 ? i : [ "ambientTexture", "bumpTexture", "diffuseTexture", "emissiveTexture", "lightmapTexture", "opacityTexture", "reflectionTexture", "refractionTexture", "specularTexture" ], this.options.uvsIn = (s = this.options.uvsIn) !== null && s !== void 0 ? s : J.UVKind, this.options.uvsOut = (a = this.options.uvsOut) !== null && a !== void 0 ? a : J.UVKind, this.options.layout = (f = this.options.layout) !== null && f !== void 0 ? f : I0.LAYOUT_STRIP, this.options.layout === I0.LAYOUT_COLNUM && (this.options.colnum = (o = this.options.colnum) !== null && o !== void 0 ? o : 8), this.options.updateInputMeshes = (d = this.options.updateInputMeshes) !== null && d !== void 0 ? d : !0, this.options.disposeSources = (v = this.options.disposeSources) !== null && v !== void 0 ? v : !0, this._expecting = 0, this.options.fillBlanks = (u = this.options.fillBlanks) !== null && u !== void 0 ? u : !0, this.options.fillBlanks === !0 && (this.options.customFillColor = (l = this.options.customFillColor) !== null && l !== void 0 ? l : "black"), this.options.frameSize = (P = this.options.frameSize) !== null && P !== void 0 ? P : 256, this.options.paddingRatio = (p = this.options.paddingRatio) !== null && p !== void 0 ? p : 0.0115, this._paddingValue = Math.ceil(this.options.frameSize * this.options.paddingRatio), this._paddingValue % 2 !== 0 && this._paddingValue++, this.options.paddingMode = (c = this.options.paddingMode) !== null && c !== void 0 ? c : I0.SUBUV_WRAP, this.options.paddingMode === I0.SUBUV_COLOR && (this.options.paddingColor = (H = this.options.paddingColor) !== null && H !== void 0 ? H : new xt(0, 0, 0, 1)), this.sets = {}, this.frames = [], this; } /** * Starts the package process * @param resolve The promises resolution function * @returns TexturePacker */ _createFrames(e) { const t = this._calculateSize(), r = new at(1, 1).divide(t); let n = 0; const i = this._expecting, s = this.meshes.length, a = Object.keys(this.sets); for (let u = 0; u < a.length; u++) { const l = a[u], P = new Xp( this.name + ".TexturePack." + l + "Set", { width: t.x, height: t.y }, this.scene, !0, //Generate Mips We.TRILINEAR_SAMPLINGMODE, Ge.TEXTUREFORMAT_RGBA ), p = P.getContext(); p.fillStyle = "rgba(0,0,0,0)", p.fillRect(0, 0, t.x, t.y), P.update(!1), this.sets[l] = P; } const f = this.options.frameSize || 256, o = this._paddingValue, d = f + 2 * o, v = () => { this._calculateMeshUVFrames(f, o, t, r, this.options.updateInputMeshes || !1); }; for (let u = 0; u < s; u++) { const P = this.meshes[u].material; for (let p = 0; p < a.length; p++) { const c = new Xp("temp", d, this.scene, !0), H = c.getContext(), T = this._getFrameOffset(u), q = () => { n++, c.update(!1); const j = H.getImageData(0, 0, d, d), w = this.sets[b]; if (w.getContext().putImageData(j, t.x * T.x, t.y * T.y), c.dispose(), w.update(!1), n == i) { v(), e(); return; } }, b = a[p] || "_blank"; if (!P || P[b] === null) H.fillStyle = "rgba(0,0,0,0)", this.options.fillBlanks && (H.fillStyle = this.options.customFillColor), H.fillRect(0, 0, d, d), q(); else { const j = P[b], w = new Image(); j instanceof Xp ? w.src = j.getContext().canvas.toDataURL("image/png") : w.src = j.url, ye.SetCorsBehavior(w.src, w), w.onload = () => { H.fillStyle = "rgba(0,0,0,0)", H.fillRect(0, 0, d, d), c.update(!1), H.setTransform(1, 0, 0, -1, 0, 0); const m = [0, 0, 1, 0, 1, 1, 0, 1, -1, 1, -1, 0, -2, 0, -1, 1, -1]; switch (this.options.paddingMode) { case 0: for (let I = 0; I < 9; I++) H.drawImage(w, 0, 0, w.width, w.height, o + f * m[I], o + f * m[I + 1] - d, f, f); break; case 1: for (let I = 0; I < o; I++) H.drawImage(w, 0, 0, w.width, w.height, I + f * m[0], o - d, f, f), H.drawImage(w, 0, 0, w.width, w.height, o * 2 - I, o - d, f, f), H.drawImage(w, 0, 0, w.width, w.height, o, I - d, f, f), H.drawImage(w, 0, 0, w.width, w.height, o, o * 2 - I - d, f, f); H.drawImage(w, 0, 0, w.width, w.height, o + f * m[0], o + f * m[1] - d, f, f); break; case 2: H.fillStyle = (this.options.paddingColor || Ne.Black()).toHexString(), H.fillRect(0, 0, d, -d), H.clearRect(o, o, f, f), H.drawImage(w, 0, 0, w.width, w.height, o + f * m[0], o + f * m[1] - d, f, f); break; } H.setTransform(1, 0, 0, 1, 0, 0), q(); }; } } } } /** * Calculates the Size of the Channel Sets * @returns Vector2 */ _calculateSize() { const e = this.meshes.length || 0, t = this.options.frameSize || 0, r = this._paddingValue || 0; switch (this.options.layout) { case 0: return new at(t * e + 2 * r * e, t + 2 * r); case 1: { const n = Math.max(2, Math.ceil(Math.sqrt(e))), i = t * n + 2 * r * n; return new at(i, i); } case 2: { const n = this.options.colnum || 1, i = Math.max(1, Math.ceil(e / n)); return new at(t * n + 2 * r * n, t * i + 2 * r * i); } } return at.Zero(); } /** * Calculates the UV data for the frames. * @param baseSize the base frameSize * @param padding the base frame padding * @param dtSize size of the Dynamic Texture for that channel * @param dtUnits is 1/dtSize * @param update flag to update the input meshes */ _calculateMeshUVFrames(e, t, r, n, i) { const s = this.meshes.length; for (let a = 0; a < s; a++) { const f = this.meshes[a], o = new at(e / r.x, e / r.y), d = n.clone().scale(t), u = this._getFrameOffset(a).add(d), l = new XF(a, o, u); this.frames.push(l), i && (this._updateMeshUV(f, a), this._updateTextureReferences(f)); } } /** * Calculates the frames Offset. * @param index of the frame * @returns Vector2 */ _getFrameOffset(e) { const t = this.meshes.length; let r, n, i; switch (this.options.layout) { case 0: return r = 1 / t, new at(e * r, 0); case 1: { const s = Math.max(2, Math.ceil(Math.sqrt(t))); return n = Math.floor(e / s), i = e - n * s, r = 1 / s, new at(i * r, n * r); } case 2: { const s = this.options.colnum || 1, a = Math.max(1, Math.ceil(t / s)); return i = Math.floor(e / a), n = e - i * a, r = new at(1 / s, 1 / a), new at(i * r.x, n * r.y); } } return at.Zero(); } /** * Updates a Mesh to the frame data * @param mesh that is the target * @param frameID or the frame index */ _updateMeshUV(e, t) { const r = this.frames[t], n = e.getVerticesData(this.options.uvsIn || J.UVKind), i = []; let s = 0; n.length && (s = n.length || 0); for (let a = 0; a < s; a += 2) i.push(n[a] * r.scale.x + r.offset.x, n[a + 1] * r.scale.y + r.offset.y); e.setVerticesData(this.options.uvsOut || J.UVKind, i); } /** * Updates a Meshes materials to use the texture packer channels * @param m is the mesh to target * @param force all channels on the packer to be set. */ _updateTextureReferences(e, t = !1) { const r = e.material, n = Object.keys(this.sets), i = (s) => { s.dispose && s.dispose(); }; for (let s = 0; s < n.length; s++) { const a = n[s]; if (t) r[a] !== null && i(r[a]), r[a] = this.sets[a]; else { if (!r) return; r[a] !== null && (i(r[a]), r[a] = this.sets[a]); } } } /** * Public method to set a Mesh to a frame * @param m that is the target * @param frameID or the frame index * @param updateMaterial trigger for if the Meshes attached Material be updated? */ setMeshToFrame(e, t, r = !1) { this._updateMeshUV(e, t), r && this._updateTextureReferences(e, !0); } /** * Starts the async promise to compile the texture packer. * @returns Promise */ processAsync() { return new Promise((e, t) => { try { if (this.meshes.length === 0) { e(); return; } let r = 0; const n = (i) => { if (r++, this.options.map) { for (let s = 0; s < this.options.map.length; s++) { const a = this.options.map[s]; i[a] !== null && (this.sets[this.options.map[s]] || (this.sets[this.options.map[s]] = !0), this._expecting++); } r === this.meshes.length && this._createFrames(e); } }; for (let i = 0; i < this.meshes.length; i++) { const s = this.meshes[i], a = s.material; if (!a) { if (r++, r === this.meshes.length) return this._createFrames(e); continue; } a.forceCompilationAsync(s).then(() => { n(a); }); } } catch (r) { return t(r); } }); } /** * Disposes all textures associated with this packer */ dispose() { const e = Object.keys(this.sets); for (let t = 0; t < e.length; t++) { const r = e[t]; this.sets[r].dispose(); } } /** * Starts the download process for all the channels converting them to base64 data and embedding it all in a JSON file. * @param imageType is the image type to use. * @param quality of the image if downloading as jpeg, Ranges from >0 to 1. */ download(e = "png", t = 1) { setTimeout(() => { const r = { name: this.name, sets: {}, options: {}, frames: [] }, n = Object.keys(this.sets), i = Object.keys(this.options); try { for (let f = 0; f < n.length; f++) { const o = n[f], d = this.sets[o]; r.sets[o] = d.getContext().canvas.toDataURL("image/" + e, t); } for (let f = 0; f < i.length; f++) { const o = i[f]; r.options[o] = this.options[o]; } for (let f = 0; f < this.frames.length; f++) { const o = this.frames[f]; r.frames.push(o.scale.x, o.scale.y, o.offset.x, o.offset.y); } } catch (f) { Se.Warn("Unable to download: " + f); return; } const s = "data:text/json;charset=utf-8," + encodeURIComponent(JSON.stringify(r, null, 4)), a = document.createElement("a"); a.setAttribute("href", s), a.setAttribute("download", this.name + "_texurePackage.json"), document.body.appendChild(a), a.click(), a.remove(); }, 0); } /** * Public method to load a texturePacker JSON file. * @param data of the JSON file in string format. */ updateFromJSON(e) { try { const t = JSON.parse(e); this.name = t.name; const r = Object.keys(t.options); for (let i = 0; i < r.length; i++) this.options[r[i]] = t.options[r[i]]; for (let i = 0; i < t.frames.length; i += 4) { const s = new XF(i / 4, new at(t.frames[i], t.frames[i + 1]), new at(t.frames[i + 2], t.frames[i + 3])); this.frames.push(s); } const n = Object.keys(t.sets); for (let i = 0; i < n.length; i++) { const s = new We(t.sets[n[i]], this.scene, !1, !1); this.sets[n[i]] = s; } } catch (t) { Se.Warn("Unable to update from JSON: " + t); } } } I0.LAYOUT_STRIP = 0; I0.LAYOUT_POWER2 = 1; I0.LAYOUT_COLNUM = 2; I0.SUBUV_WRAP = 0; I0.SUBUV_EXTEND = 1; I0.SUBUV_COLOR = 2; class rre extends Y0 { /** * Instantiates a new Custom Procedural Texture. * Procedural texturing is a way to programmatically create a texture. There are 2 types of procedural textures: code-only, and code that references some classic 2D images, sometimes called 'refMaps' or 'sampler' images. * Custom Procedural textures are the easiest way to create your own procedural in your application. * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/proceduralTextures#creating-custom-procedural-textures * @param name Define the name of the texture * @param texturePath Define the folder path containing all the custom texture related files (config, shaders...) * @param size Define the size of the texture to create * @param scene Define the scene the texture belongs to * @param fallbackTexture Define a fallback texture in case there were issues to create the custom texture * @param generateMipMaps Define if the texture should creates mip maps or not * @param skipJson Define a boolena indicating that there is no json config file to load */ constructor(e, t, r, n, i, s, a) { super(e, r, null, n, i, s), this._animate = !0, this._time = 0, this._texturePath = t, a ? this.setFragment(this._texturePath) : this._loadJson(t), this.refreshRate = 1; } _loadJson(e) { const t = () => { try { this.setFragment(this._texturePath); } catch { Se.Log("No json or ShaderStore or DOM element found for CustomProceduralTexture"); } }, r = e + "/config.json", n = new ho(); n.open("GET", r), n.addEventListener("load", () => { if (n.status === 200 || n.responseText && n.responseText.length > 0) try { this._config = JSON.parse(n.response), this.updateShaderUniforms(), this.updateTextures(), this.setFragment(this._texturePath + "/custom"), this._animate = this._config.animate, this.refreshRate = this._config.refreshrate; } catch { t(); } else t(); }, !1), n.addEventListener("error", () => { t(); }, !1); try { n.send(); } catch { Se.Error("CustomProceduralTexture: Error on XHR send request."); } } /** * Is the texture ready to be used ? (rendered at least once) * @returns true if ready, otherwise, false. */ isReady() { if (!super.isReady()) return !1; for (const e in this._textures) if (!this._textures[e].isReady()) return !1; return !0; } /** * Render the texture to its associated render target. * @param useCameraPostProcess Define if camera post process should be applied to the texture */ render(e) { const t = this.getScene(); this._animate && t && (this._time += t.getAnimationRatio() * 0.03, this.updateShaderUniforms()), super.render(e); } /** * Update the list of dependant textures samplers in the shader. */ updateTextures() { for (let e = 0; e < this._config.sampler2Ds.length; e++) this.setTexture(this._config.sampler2Ds[e].sample2Dname, new We(this._texturePath + "/" + this._config.sampler2Ds[e].textureRelativeUrl, this.getScene())); } /** * Update the uniform values of the procedural texture in the shader. */ updateShaderUniforms() { if (this._config) for (let e = 0; e < this._config.uniforms.length; e++) { const t = this._config.uniforms[e]; switch (t.type) { case "float": this.setFloat(t.name, t.value); break; case "color3": this.setColor3(t.name, new Ne(t.r, t.g, t.b)); break; case "color4": this.setColor4(t.name, new xt(t.r, t.g, t.b, t.a)); break; case "vector2": this.setVector2(t.name, new at(t.x, t.y)); break; case "vector3": this.setVector3(t.name, new S(t.x, t.y, t.z)); break; } } this.setFloat("time", this._time); } /** * Define if the texture animates or not. */ get animate() { return this._animate; } set animate(e) { this._animate = e; } } const Kpe = "noisePixelShader", Jpe = `uniform float brightness;uniform float persistence;uniform float timeScale;varying vec2 vUV;vec2 hash22(vec2 p) {p=p*mat2(127.1,311.7,269.5,183.3);p=-1.0+2.0*fract(sin(p)*43758.5453123);return sin(p*6.283+timeScale);} float interpolationNoise(vec2 p) {vec2 pi=floor(p);vec2 pf=p-pi;vec2 w=pf*pf*(3.-2.*pf);float f00=dot(hash22(pi+vec2(.0,.0)),pf-vec2(.0,.0));float f01=dot(hash22(pi+vec2(.0,1.)),pf-vec2(.0,1.));float f10=dot(hash22(pi+vec2(1.0,0.)),pf-vec2(1.0,0.));float f11=dot(hash22(pi+vec2(1.0,1.)),pf-vec2(1.0,1.));float xm1=mix(f00,f10,w.x);float xm2=mix(f01,f11,w.x);float ym=mix(xm1,xm2,w.y); return ym;} float perlinNoise2D(float x,float y) {float sum=0.0;float frequency=0.0;float amplitude=0.0;for(int i=0; i { }); } /** * Clones the raw cube texture. * @returns a new cube texture */ clone() { return jt.Clone(() => { const e = this.getScene(), t = this._texture, r = new hy(e, t._bufferViewArray, t.width, t.format, t.type, t.generateMipMaps, t.invertY, t.samplingMode, t._compression); return t.source === ri.CubeRawRGBD && r.updateRGBDAsync(t._bufferViewArrayArray, t._sphericalPolynomial, t._lodGenerationScale, t._lodGenerationOffset), r; }, this); } } class zpe extends We { /** * Create a new RawTexture3D * @param data defines the data of the texture * @param width defines the width of the texture * @param height defines the height of the texture * @param depth defines the depth of the texture * @param format defines the texture format to use * @param scene defines the hosting scene * @param generateMipMaps defines a boolean indicating if mip levels should be generated (true by default) * @param invertY defines if texture must be stored with Y axis inverted * @param samplingMode defines the sampling mode to use (Texture.TRILINEAR_SAMPLINGMODE by default) * @param textureType defines the texture Type (Engine.TEXTURETYPE_UNSIGNED_INT, Engine.TEXTURETYPE_FLOAT...) * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg) */ constructor(e, t, r, n, i, s, a = !0, f = !1, o = We.TRILINEAR_SAMPLINGMODE, d = 0, v) { super(null, s, !a, f), this.format = i, this._texture = s.getEngine().createRawTexture3D(e, t, r, n, i, a, f, o, null, d, v), this.is3D = !0; } /** * Update the texture with new data * @param data defines the data to store in the texture */ update(e) { this._texture && this._getEngine().updateRawTexture3D(this._texture, e, this._texture.format, this._texture.invertY, null, this._texture.type); } } class tY extends Ta { /** * Creates a refraction texture used by refraction channel of the standard material. * It is like a mirror but to see through a material. * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#refraction * @param name Define the texture name * @param size Define the size of the underlying texture * @param scene Define the scene the refraction belongs to * @param generateMipMaps Define if we need to generate mips level for the refraction */ constructor(e, t, r, n) { super(e, t, r, n, !0), this.refractionPlane = new BA(0, 1, 0, 1), this.depth = 2, this.onBeforeRenderObservable.add(() => { this.getScene().clipPlane = this.refractionPlane; }), this.onAfterRenderObservable.add(() => { this.getScene().clipPlane = null; }); } /** * Clone the refraction texture. * @returns the cloned texture */ clone() { const e = this.getScene(); if (!e) return this; const t = this.getSize(), r = new tY(this.name, t.width, e, this._generateMipMaps); return r.hasAlpha = this.hasAlpha, r.level = this.level, r.refractionPlane = this.refractionPlane.clone(), this.renderList && (r.renderList = this.renderList.slice(0)), r.depth = this.depth, r; } /** * Serialize the texture to a JSON representation you could use in Parse later on * @returns the serialized JSON representation */ serialize() { if (!this.name) return null; const e = super.serialize(); return e.mirrorPlane = this.refractionPlane.asArray(), e.depth = this.depth, e; } } class Gpe extends nq { /** * Gets the render target wrapper associated with this render target */ get renderTarget() { return this._renderTarget; } /** * Instantiates a new ThinRenderTargetTexture. * Tiny helper class to wrap a RenderTargetWrapper in a texture. * This can be used as an internal texture wrapper in ThinEngine to benefit from the cache and to hold on the associated RTT * @param engine Define the internalTexture to wrap * @param size Define the size of the RTT to create * @param options Define rendertarget options */ constructor(e, t, r) { super(null), this._renderTarget = null, this._engine = e, this._renderTargetOptions = r, this.resize(t); } /** * Resize the texture to a new desired size. * Be careful as it will recreate all the data in the new texture. * @param size Define the new size. It can be: * - a number for squared texture, * - an object containing { width: number, height: number } */ resize(e) { var t; (t = this._renderTarget) === null || t === void 0 || t.dispose(), this._renderTarget = null, this._texture = null, this._size = e, this._engine && (this._renderTarget = this._engine.createRenderTargetTexture(this._size, this._renderTargetOptions)), this._texture = this.renderTarget.texture; } /** * Get the underlying lower level texture from Babylon. * @returns the internal texture */ getInternalTexture() { return this._texture; } /** * Get the class name of the texture. * @returns "ThinRenderTargetTexture" */ getClassName() { return "ThinRenderTargetTexture"; } /** * Dispose the texture and release its associated resources. * @param disposeOnlyFramebuffers */ dispose(e = !1) { var t; (t = this._renderTarget) === null || t === void 0 || t.dispose(!0), this._renderTarget = null, e || super.dispose(); } } class Ko extends um { /** * Creates a new connection point * @param name defines the connection point name * @param ownerBlock defines the block hosting this connection point * @param direction defines the direction of the connection point * @param _blockType * @param _blockName */ constructor(e, t, r, n, i) { super(e, t, r), this._blockType = n, this._blockName = i, this.needDualDirectionValidation = !0; } /** * Gets a number indicating if the current point can be connected to another point * @param connectionPoint defines the other connection point * @returns a number defining the compatibility state */ checkCompatibilityState(e) { return e instanceof Ko && e._blockName === this._blockName ? pc.Compatible : pc.TypeIncompatible; } /** * Creates a block suitable to be used as an input for this input point. * If null is returned, a block based on the point type will be created. * @returns The returned string parameter is the name of the output point of NodeMaterialBlock (first parameter of the returned array) that can be connected to the input */ createCustomInputBlock() { return [new this._blockType(this._blockName), this.name]; } } class nre extends Mr { /** * Creates a new BonesBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Vertex), this.registerInput("matricesIndices", de.Vector4), this.registerInput("matricesWeights", de.Vector4), this.registerInput("matricesIndicesExtra", de.Vector4, !0), this.registerInput("matricesWeightsExtra", de.Vector4, !0), this.registerInput("world", de.Matrix), this.registerOutput("output", de.Matrix); } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("boneSampler"), e._excludeVariableName("boneTextureWidth"), e._excludeVariableName("mBones"), e._excludeVariableName("BonesPerMesh"); } /** * Gets the current class name * @returns the class name */ getClassName() { return "BonesBlock"; } /** * Gets the matrix indices input component */ get matricesIndices() { return this._inputs[0]; } /** * Gets the matrix weights input component */ get matricesWeights() { return this._inputs[1]; } /** * Gets the extra matrix indices input component */ get matricesIndicesExtra() { return this._inputs[2]; } /** * Gets the extra matrix weights input component */ get matricesWeightsExtra() { return this._inputs[3]; } /** * Gets the world input component */ get world() { return this._inputs[4]; } /** * Gets the output component */ get output() { return this._outputs[0]; } autoConfigure(e, t = () => !0) { if (!this.matricesIndices.isConnected) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "matricesIndices" && t(n)); r || (r = new gi("matricesIndices"), r.setAsAttribute("matricesIndices")), r.output.connectTo(this.matricesIndices); } if (!this.matricesWeights.isConnected) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "matricesWeights" && t(n)); r || (r = new gi("matricesWeights"), r.setAsAttribute("matricesWeights")), r.output.connectTo(this.matricesWeights); } if (!this.world.isConnected) { let r = e.getInputBlockByPredicate((n) => n.systemValue === Bi.World && t(n)); r || (r = new gi("world"), r.setAsSystemValue(Bi.World)), r.output.connectTo(this.world); } } provideFallbacks(e, t) { e && e.useBones && e.computeBonesUsingShaders && e.skeleton && t.addCPUSkinningFallback(0, e); } bind(e, t, r) { Ye.BindBonesParameters(r, e); } prepareDefines(e, t, r) { r._areAttributesDirty && Ye.PrepareDefinesForBones(e, r); } _buildBlock(e) { super._buildBlock(e), e.sharedData.blocksWithFallbacks.push(this), e.sharedData.forcedBindableBlocks.push(this), e.sharedData.blocksWithDefines.push(this), e.uniforms.push("boneTextureWidth"), e.uniforms.push("mBones"), e.samplers.push("boneSampler"); const t = `//${this.name}`; e._emitFunctionFromInclude("bonesDeclaration", t, { removeAttributes: !0, removeUniforms: !1, removeVaryings: !0, removeIfDef: !1 }); const r = e._getFreeVariableName("influence"); e.compilationString += e._emitCodeFromInclude("bonesVertex", t, { replaceStrings: [ { search: /finalWorld=finalWorld\*influence;/, replace: "" }, { search: /influence/gm, replace: r } ] }); const n = this._outputs[0], i = this.world; return e.compilationString += `#if NUM_BONE_INFLUENCERS>0 `, e.compilationString += this._declareOutput(n, e) + ` = ${i.associatedVariableName} * ${r}; `, e.compilationString += `#else `, e.compilationString += this._declareOutput(n, e) + ` = ${i.associatedVariableName}; `, e.compilationString += `#endif `, this; } } Ue("BABYLON.BonesBlock", nre); class ire extends Mr { /** * Creates a new InstancesBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Vertex), this.registerInput("world0", de.Vector4), this.registerInput("world1", de.Vector4), this.registerInput("world2", de.Vector4), this.registerInput("world3", de.Vector4), this.registerInput("world", de.Matrix, !0), this.registerOutput("output", de.Matrix), this.registerOutput("instanceID", de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "InstancesBlock"; } /** * Gets the first world row input component */ get world0() { return this._inputs[0]; } /** * Gets the second world row input component */ get world1() { return this._inputs[1]; } /** * Gets the third world row input component */ get world2() { return this._inputs[2]; } /** * Gets the forth world row input component */ get world3() { return this._inputs[3]; } /** * Gets the world input component */ get world() { return this._inputs[4]; } /** * Gets the output component */ get output() { return this._outputs[0]; } /** * Gets the instanceID component */ get instanceID() { return this._outputs[1]; } autoConfigure(e, t = () => !0) { if (!this.world0.connectedPoint) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "world0" && t(n)); r || (r = new gi("world0"), r.setAsAttribute("world0")), r.output.connectTo(this.world0); } if (!this.world1.connectedPoint) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "world1" && t(n)); r || (r = new gi("world1"), r.setAsAttribute("world1")), r.output.connectTo(this.world1); } if (!this.world2.connectedPoint) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "world2" && t(n)); r || (r = new gi("world2"), r.setAsAttribute("world2")), r.output.connectTo(this.world2); } if (!this.world3.connectedPoint) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "world3" && t(n)); r || (r = new gi("world3"), r.setAsAttribute("world3")), r.output.connectTo(this.world3); } if (!this.world.connectedPoint) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "world" && t(n)); r || (r = new gi("world"), r.setAsSystemValue(Bi.World)), r.output.connectTo(this.world); } this.world.define = "!INSTANCES || THIN_INSTANCES"; } prepareDefines(e, t, r, n = !1, i) { let s = !1; r.INSTANCES !== n && (r.setValue("INSTANCES", n), s = !0), i && r.THIN_INSTANCES !== !!(i != null && i.getRenderingMesh().hasThinInstances) && (r.setValue("THIN_INSTANCES", !!(i != null && i.getRenderingMesh().hasThinInstances)), s = !0), s && r.markAsUnprocessed(); } _buildBlock(e) { super._buildBlock(e); const t = e.sharedData.scene.getEngine(); e.sharedData.blocksWithDefines.push(this); const r = this._outputs[0], n = this._outputs[1], i = this.world0, s = this.world1, a = this.world2, f = this.world3; return e.compilationString += `#ifdef INSTANCES `, e.compilationString += this._declareOutput(r, e) + ` = mat4(${i.associatedVariableName}, ${s.associatedVariableName}, ${a.associatedVariableName}, ${f.associatedVariableName}); `, e.compilationString += `#ifdef THIN_INSTANCES `, e.compilationString += `${r.associatedVariableName} = ${this.world.associatedVariableName} * ${r.associatedVariableName}; `, e.compilationString += `#endif `, t._caps.canUseGLInstanceID ? e.compilationString += this._declareOutput(n, e) + ` = float(gl_InstanceID); ` : e.compilationString += this._declareOutput(n, e) + ` = 0.0; `, e.compilationString += `#else `, e.compilationString += this._declareOutput(r, e) + ` = ${this.world.associatedVariableName}; `, e.compilationString += this._declareOutput(n, e) + ` = 0.0; `, e.compilationString += `#endif `, this; } } Ue("BABYLON.InstancesBlock", ire); class rY extends Mr { /** * Create a new MorphTargetsBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Vertex), this.registerInput("position", de.Vector3), this.registerInput("normal", de.Vector3), this.registerInput("tangent", de.AutoDetect), this.tangent.addExcludedConnectionPointFromAllowedTypes(de.Color4 | de.Vector4 | de.Vector3), this.registerInput("uv", de.Vector2), this.registerOutput("positionOutput", de.Vector3), this.registerOutput("normalOutput", de.Vector3), this.registerOutput("tangentOutput", de.Vector4), this.registerOutput("uvOutput", de.Vector2); } /** * Gets the current class name * @returns the class name */ getClassName() { return "MorphTargetsBlock"; } /** * Gets the position input component */ get position() { return this._inputs[0]; } /** * Gets the normal input component */ get normal() { return this._inputs[1]; } /** * Gets the tangent input component */ get tangent() { return this._inputs[2]; } /** * Gets the tangent input component */ get uv() { return this._inputs[3]; } /** * Gets the position output component */ get positionOutput() { return this._outputs[0]; } /** * Gets the normal output component */ get normalOutput() { return this._outputs[1]; } /** * Gets the tangent output component */ get tangentOutput() { return this._outputs[2]; } /** * Gets the tangent output component */ get uvOutput() { return this._outputs[3]; } initialize(e) { e._excludeVariableName("morphTargetInfluences"); } autoConfigure(e, t = () => !0) { if (!this.position.isConnected) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "position" && t(n)); r || (r = new gi("position"), r.setAsAttribute()), r.output.connectTo(this.position); } if (!this.normal.isConnected) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "normal" && t(n)); r || (r = new gi("normal"), r.setAsAttribute("normal")), r.output.connectTo(this.normal); } if (!this.tangent.isConnected) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "tangent" && t(n)); r || (r = new gi("tangent"), r.setAsAttribute("tangent")), r.output.connectTo(this.tangent); } if (!this.uv.isConnected) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "uv" && t(n)); r || (r = new gi("uv"), r.setAsAttribute("uv")), r.output.connectTo(this.uv); } } prepareDefines(e, t, r) { if (e.morphTargetManager) { const n = e.morphTargetManager; n != null && n.isUsingTextureForTargets && n.numInfluencers !== r.NUM_MORPH_INFLUENCERS && r.markAsAttributesDirty(); } r._areAttributesDirty && Ye.PrepareDefinesForMorphTargets(e, r); } bind(e, t, r) { r && r.morphTargetManager && r.morphTargetManager.numInfluencers > 0 && (Ye.BindMorphTargetParameters(r, e), r.morphTargetManager.isUsingTextureForTargets && r.morphTargetManager._bind(e)); } replaceRepeatableContent(e, t, r, n) { const i = this.position, s = this.normal, a = this.tangent, f = this.uv, o = this.positionOutput, d = this.normalOutput, v = this.tangentOutput, u = this.uvOutput, l = e, P = n.NUM_MORPH_INFLUENCERS, p = r.morphTargetManager, c = p && p.supportsNormals && n.NORMAL, H = p && p.supportsTangents && n.TANGENT, T = p && p.supportsUVs && n.UV1; let q = ""; p != null && p.isUsingTextureForTargets && P > 0 && (q += `float vertexID; `); for (let b = 0; b < P; b++) q += `#ifdef MORPHTARGETS `, p != null && p.isUsingTextureForTargets ? (q += `vertexID = float(gl_VertexID) * morphTargetTextureInfo.x; `, q += `${o.associatedVariableName} += (readVector3FromRawSampler(${b}, vertexID) - ${i.associatedVariableName}) * morphTargetInfluences[${b}]; `, q += `vertexID += 1.0; `) : q += `${o.associatedVariableName} += (position${b} - ${i.associatedVariableName}) * morphTargetInfluences[${b}]; `, c && (q += `#ifdef MORPHTARGETS_NORMAL `, p != null && p.isUsingTextureForTargets ? (q += `${d.associatedVariableName} += (readVector3FromRawSampler(${b}, vertexID) - ${s.associatedVariableName}) * morphTargetInfluences[${b}]; `, q += `vertexID += 1.0; `) : q += `${d.associatedVariableName} += (normal${b} - ${s.associatedVariableName}) * morphTargetInfluences[${b}]; `, q += `#endif `), T && (q += `#ifdef MORPHTARGETS_UV `, p != null && p.isUsingTextureForTargets ? (q += `${u.associatedVariableName} += (readVector3FromRawSampler(${b}, vertexID).xy - ${f.associatedVariableName}) * morphTargetInfluences[${b}]; `, q += `vertexID += 1.0; `) : q += `${u.associatedVariableName}.xy += (uv_${b} - ${f.associatedVariableName}.xy) * morphTargetInfluences[${b}]; `, q += `#endif `), H && (q += `#ifdef MORPHTARGETS_TANGENT `, p != null && p.isUsingTextureForTargets ? q += `${v.associatedVariableName}.xyz += (readVector3FromRawSampler(${b}, vertexID) - ${a.associatedVariableName}.xyz) * morphTargetInfluences[${b}]; ` : q += `${v.associatedVariableName}.xyz += (tangent${b} - ${a.associatedVariableName}.xyz) * morphTargetInfluences[${b}]; `, a.type === de.Vector4 ? q += `${v.associatedVariableName}.w = ${a.associatedVariableName}.w; ` : q += `${v.associatedVariableName}.w = 1.; `, q += `#endif `), q += `#endif `; if (l.compilationString = l.compilationString.replace(this._repeatableContentAnchor, q), P > 0) for (let b = 0; b < P; b++) l.attributes.push(J.PositionKind + b), c && l.attributes.push(J.NormalKind + b), H && l.attributes.push(J.TangentKind + b), T && l.attributes.push(J.UVKind + "_" + b); } _buildBlock(e) { super._buildBlock(e), e.sharedData.blocksWithDefines.push(this), e.sharedData.bindableBlocks.push(this), e.sharedData.repeatableContentBlocks.push(this); const t = this.position, r = this.normal, n = this.tangent, i = this.uv, s = this.positionOutput, a = this.normalOutput, f = this.tangentOutput, o = this.uvOutput, d = `//${this.name}`; return e.uniforms.push("morphTargetInfluences"), e.uniforms.push("morphTargetTextureInfo"), e.uniforms.push("morphTargetTextureIndices"), e.samplers.push("morphTargets"), e._emitFunctionFromInclude("morphTargetsVertexGlobalDeclaration", d), e._emitFunctionFromInclude("morphTargetsVertexDeclaration", d, { repeatKey: "maxSimultaneousMorphTargets" }), e.compilationString += `${this._declareOutput(s, e)} = ${t.associatedVariableName}; `, e.compilationString += `#ifdef NORMAL `, e.compilationString += `${this._declareOutput(a, e)} = ${r.associatedVariableName}; `, e.compilationString += `#else `, e.compilationString += `${this._declareOutput(a, e)} = vec3(0., 0., 0.); `, e.compilationString += `#endif `, e.compilationString += `#ifdef TANGENT `, e.compilationString += `${this._declareOutput(f, e)} = ${n.associatedVariableName}; `, e.compilationString += `#else `, e.compilationString += `${this._declareOutput(f, e)} = vec4(0., 0., 0., 0.); `, e.compilationString += `#endif `, e.compilationString += `#ifdef UV1 `, e.compilationString += `${this._declareOutput(o, e)} = ${i.associatedVariableName}; `, e.compilationString += `#else `, e.compilationString += `${this._declareOutput(o, e)} = vec2(0., 0.); `, e.compilationString += `#endif `, this._repeatableContentAnchor = e._repeatableContentAnchor, e.compilationString += this._repeatableContentAnchor, this; } } Ue("BABYLON.MorphTargetsBlock", rY); class sre extends Mr { /** * Creates a new LightInformationBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Vertex), this.registerInput("worldPosition", de.Vector4, !1, Ve.Vertex), this.registerOutput("direction", de.Vector3), this.registerOutput("color", de.Color3), this.registerOutput("intensity", de.Float), this.registerOutput("shadowBias", de.Float), this.registerOutput("shadowNormalBias", de.Float), this.registerOutput("shadowDepthScale", de.Float), this.registerOutput("shadowDepthRange", de.Vector2); } /** * Gets the current class name * @returns the class name */ getClassName() { return "LightInformationBlock"; } /** * Gets the world position input component */ get worldPosition() { return this._inputs[0]; } /** * Gets the direction output component */ get direction() { return this._outputs[0]; } /** * Gets the direction output component */ get color() { return this._outputs[1]; } /** * Gets the direction output component */ get intensity() { return this._outputs[2]; } /** * Gets the shadow bias output component */ get shadowBias() { return this._outputs[3]; } /** * Gets the shadow normal bias output component */ get shadowNormalBias() { return this._outputs[4]; } /** * Gets the shadow depth scale component */ get shadowDepthScale() { return this._outputs[5]; } /** * Gets the shadow depth range component */ get shadowDepthRange() { return this._outputs[6]; } bind(e, t, r) { if (!r) return; this.light && this.light.isDisposed() && (this.light = null); let n = this.light; const i = t.getScene(); if (!n && i.lights.length && (n = this.light = i.lights[0], this._forcePrepareDefines = !0), !n || !n.isEnabled) { e.setFloat3(this._lightDataUniformName, 0, 0, 0), e.setFloat4(this._lightColorUniformName, 0, 0, 0, 0); return; } n.transferToNodeMaterialEffect(e, this._lightDataUniformName), e.setColor4(this._lightColorUniformName, n.diffuse, n.intensity); const s = n.getShadowGenerator(); if ((this.shadowBias.hasEndpoints || this.shadowNormalBias.hasEndpoints || this.shadowDepthScale.hasEndpoints) && (s ? e.setFloat3(this._lightShadowUniformName, s.bias, s.normalBias, s.depthScale) : e.setFloat3(this._lightShadowUniformName, 0, 0, 0)), this.shadowDepthRange) if (s && i.activeCamera) { const a = n; e.setFloat2(this._lightShadowExtraUniformName, a.getDepthMinZ(i.activeCamera), a.getDepthMinZ(i.activeCamera) + a.getDepthMaxZ(i.activeCamera)); } else e.setFloat2(this._lightShadowExtraUniformName, 0, 0); } prepareDefines(e, t, r) { if (!r._areLightsDirty && !this._forcePrepareDefines) return; this._forcePrepareDefines = !1; const n = this.light; r.setValue(this._lightTypeDefineName, !!(n && n instanceof ag), !0); } _buildBlock(e) { super._buildBlock(e), e.sharedData.bindableBlocks.push(this), e.sharedData.blocksWithDefines.push(this); const t = this.direction, r = this.color, n = this.intensity, i = this.shadowBias, s = this.shadowNormalBias, a = this.shadowDepthScale, f = this.shadowDepthRange; return this._lightDataUniformName = e._getFreeVariableName("lightData"), this._lightColorUniformName = e._getFreeVariableName("lightColor"), this._lightShadowUniformName = e._getFreeVariableName("shadowData"), this._lightShadowExtraUniformName = e._getFreeVariableName("shadowExtraData"), this._lightTypeDefineName = e._getFreeDefineName("LIGHTPOINTTYPE"), e._emitUniformFromString(this._lightDataUniformName, "vec3"), e._emitUniformFromString(this._lightColorUniformName, "vec4"), e.compilationString += `#ifdef ${this._lightTypeDefineName} `, e.compilationString += this._declareOutput(t, e) + ` = normalize(${this.worldPosition.associatedVariableName}.xyz - ${this._lightDataUniformName}); `, e.compilationString += `#else `, e.compilationString += this._declareOutput(t, e) + ` = ${this._lightDataUniformName}; `, e.compilationString += `#endif `, e.compilationString += this._declareOutput(r, e) + ` = ${this._lightColorUniformName}.rgb; `, e.compilationString += this._declareOutput(n, e) + ` = ${this._lightColorUniformName}.a; `, (i.hasEndpoints || s.hasEndpoints || a.hasEndpoints) && (e._emitUniformFromString(this._lightShadowUniformName, "vec3"), i.hasEndpoints && (e.compilationString += this._declareOutput(i, e) + ` = ${this._lightShadowUniformName}.x; `), s.hasEndpoints && (e.compilationString += this._declareOutput(s, e) + ` = ${this._lightShadowUniformName}.y; `), a.hasEndpoints && (e.compilationString += this._declareOutput(a, e) + ` = ${this._lightShadowUniformName}.z; `)), f.hasEndpoints && (e._emitUniformFromString(this._lightShadowExtraUniformName, "vec2"), e.compilationString += this._declareOutput(f, e) + ` = ${this._lightShadowUniformName}; `), this; } serialize() { const e = super.serialize(); return this.light && (e.lightId = this.light.id), e; } _deserialize(e, t, r) { super._deserialize(e, t, r), e.lightId && (this.light = t.getLightById(e.lightId)); } } Ue("BABYLON.LightInformationBlock", sre); class nY extends Mr { /** * Create a new ImageProcessingBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this.convertInputToLinearSpace = !0, this.registerInput("color", de.AutoDetect), this.registerOutput("output", de.Color4), this.registerOutput("rgb", de.Color3), this._inputs[0].addExcludedConnectionPointFromAllowedTypes(de.Color3 | de.Color4 | de.Vector3 | de.Vector4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ImageProcessingBlock"; } /** * Gets the color input component */ get color() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } /** * Gets the rgb component */ get rgb() { return this._outputs[1]; } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("exposureLinear"), e._excludeVariableName("contrast"), e._excludeVariableName("vInverseScreenSize"), e._excludeVariableName("vignetteSettings1"), e._excludeVariableName("vignetteSettings2"), e._excludeVariableName("vCameraColorCurveNegative"), e._excludeVariableName("vCameraColorCurveNeutral"), e._excludeVariableName("vCameraColorCurvePositive"), e._excludeVariableName("txColorTransform"), e._excludeVariableName("colorTransformSettings"), e._excludeVariableName("ditherIntensity"); } isReady(e, t, r) { return !(r._areImageProcessingDirty && t.imageProcessingConfiguration && !t.imageProcessingConfiguration.isReady()); } prepareDefines(e, t, r) { r._areImageProcessingDirty && t.imageProcessingConfiguration && t.imageProcessingConfiguration.prepareDefines(r); } bind(e, t, r) { r && t.imageProcessingConfiguration && t.imageProcessingConfiguration.bind(e); } _buildBlock(e) { var t; super._buildBlock(e), e.sharedData.blocksWithDefines.push(this), e.sharedData.blockingBlocks.push(this), e.sharedData.bindableBlocks.push(this), e.uniforms.push("exposureLinear"), e.uniforms.push("contrast"), e.uniforms.push("vInverseScreenSize"), e.uniforms.push("vignetteSettings1"), e.uniforms.push("vignetteSettings2"), e.uniforms.push("vCameraColorCurveNegative"), e.uniforms.push("vCameraColorCurveNeutral"), e.uniforms.push("vCameraColorCurvePositive"), e.uniforms.push("txColorTransform"), e.uniforms.push("colorTransformSettings"), e.uniforms.push("ditherIntensity"); const r = this.color, n = this._outputs[0], i = `//${this.name}`; return e._emitFunctionFromInclude("helperFunctions", i), e._emitFunctionFromInclude("imageProcessingDeclaration", i), e._emitFunctionFromInclude("imageProcessingFunctions", i), !((t = r.connectedPoint) === null || t === void 0) && t.isConnected && (r.connectedPoint.type === de.Color4 || r.connectedPoint.type === de.Vector4 ? e.compilationString += `${this._declareOutput(n, e)} = ${r.associatedVariableName}; ` : e.compilationString += `${this._declareOutput(n, e)} = vec4(${r.associatedVariableName}, 1.0); `, e.compilationString += `#ifdef IMAGEPROCESSINGPOSTPROCESS `, this.convertInputToLinearSpace && (e.compilationString += `${n.associatedVariableName}.rgb = toLinearSpace(${r.associatedVariableName}.rgb); `), e.compilationString += `#else `, e.compilationString += `#ifdef IMAGEPROCESSING `, this.convertInputToLinearSpace && (e.compilationString += `${n.associatedVariableName}.rgb = toLinearSpace(${r.associatedVariableName}.rgb); `), e.compilationString += `${n.associatedVariableName} = applyImageProcessing(${n.associatedVariableName}); `, e.compilationString += `#endif `, e.compilationString += `#endif `, this.rgb.hasEndpoints && (e.compilationString += this._declareOutput(this.rgb, e) + ` = ${this.output.associatedVariableName}.xyz; `)), this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return e += `${this._codeVariableName}.convertInputToLinearSpace = ${this.convertInputToLinearSpace}; `, e; } serialize() { const e = super.serialize(); return e.convertInputToLinearSpace = this.convertInputToLinearSpace, e; } _deserialize(e, t, r) { var n; super._deserialize(e, t, r), this.convertInputToLinearSpace = (n = e.convertInputToLinearSpace) !== null && n !== void 0 ? n : !0; } } C([ rn("Convert input to linear space", Gr.Boolean, "ADVANCED") ], nY.prototype, "convertInputToLinearSpace", void 0); Ue("BABYLON.ImageProcessingBlock", nY); class Rm extends Mr { /** * Create a new TBNBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment, !0), this.registerInput("normal", de.AutoDetect, !1), this.normal.addExcludedConnectionPointFromAllowedTypes(de.Color4 | de.Vector4 | de.Vector3), this.registerInput("tangent", de.Vector4, !1), this.registerInput("world", de.Matrix, !1), this.registerOutput("TBN", de.Object, Ve.Fragment, new Ko("TBN", this, ao.Output, Rm, "TBNBlock")), this.registerOutput("row0", de.Vector3, Ve.Fragment), this.registerOutput("row1", de.Vector3, Ve.Fragment), this.registerOutput("row2", de.Vector3, Ve.Fragment); } /** * Gets the current class name * @returns the class name */ getClassName() { return "TBNBlock"; } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("tbnNormal"), e._excludeVariableName("tbnTangent"), e._excludeVariableName("tbnBitangent"), e._excludeVariableName("TBN"); } /** * Gets the normal input component */ get normal() { return this._inputs[0]; } /** * Gets the tangent input component */ get tangent() { return this._inputs[1]; } /** * Gets the world matrix input component */ get world() { return this._inputs[2]; } /** * Gets the TBN output component */ // eslint-disable-next-line @typescript-eslint/naming-convention get TBN() { return this._outputs[0]; } /** * Gets the row0 of the output matrix */ get row0() { return this._outputs[1]; } /** * Gets the row1 of the output matrix */ get row1() { return this._outputs[2]; } /** * Gets the row2 of the output matrix */ get row2() { return this._outputs[3]; } get target() { return Ve.Fragment; } set target(e) { } autoConfigure(e, t = () => !0) { if (!this.world.isConnected) { let r = e.getInputBlockByPredicate((n) => n.isSystemValue && n.systemValue === Bi.World && t(n)); r || (r = new gi("world"), r.setAsSystemValue(Bi.World)), r.output.connectTo(this.world); } if (!this.normal.isConnected) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "normal" && t(n)); r || (r = new gi("normal"), r.setAsAttribute("normal")), r.output.connectTo(this.normal); } if (!this.tangent.isConnected) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "tangent" && n.type === de.Vector4 && t(n)); r || (r = new gi("tangent"), r.setAsAttribute("tangent")), r.output.connectTo(this.tangent); } } prepareDefines(e, t, r) { var n, i, s, a; const f = this.normal, o = this.tangent; let d = f.isConnected; !((n = f.connectInputBlock) === null || n === void 0) && n.isAttribute && !e.isVerticesDataPresent((i = f.connectInputBlock) === null || i === void 0 ? void 0 : i.name) && (d = !1); let v = o.isConnected; !((s = o.connectInputBlock) === null || s === void 0) && s.isAttribute && !e.isVerticesDataPresent((a = o.connectInputBlock) === null || a === void 0 ? void 0 : a.name) && (v = !1); const u = d && v; r.setValue("TBNBLOCK", u, !0); } _buildBlock(e) { super._buildBlock(e); const t = this.normal, r = this.tangent, n = this.world, i = this.TBN, s = this.row0, a = this.row1, f = this.row2; return e.target === Ve.Fragment && (e.compilationString += ` // ${this.name} vec3 tbnNormal = normalize(${t.associatedVariableName}).xyz; vec3 tbnTangent = normalize(${r.associatedVariableName}.xyz); vec3 tbnBitangent = cross(tbnNormal, tbnTangent) * ${r.associatedVariableName}.w; mat3 ${i.associatedVariableName} = mat3(${n.associatedVariableName}) * mat3(tbnTangent, tbnBitangent, tbnNormal); `, s.hasEndpoints && (e.compilationString += this._declareOutput(s, e) + ` = vec3(${i.associatedVariableName}[0][0], ${i.associatedVariableName}[0][1], ${i.associatedVariableName}[0][2]); `), a.hasEndpoints && (e.compilationString += this._declareOutput(a, e) + ` = vec3(${i.associatedVariableName}[1[0], ${i.associatedVariableName}[1][1], ${i.associatedVariableName}[1][2]); `), f.hasEndpoints && (e.compilationString += this._declareOutput(f, e) + ` = vec3(${i.associatedVariableName}[2][0], ${i.associatedVariableName}[2][1], ${i.associatedVariableName}[2][2]); `), e.sharedData.blocksWithDefines.push(this)), this; } } Ue("BABYLON.TBNBlock", Rm); class PU extends Mr { /** * Create a new PerturbNormalBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this._tangentSpaceParameterName = "", this._tangentCorrectionFactorName = "", this._worldMatrixName = "", this.invertX = !1, this.invertY = !1, this.useParallaxOcclusion = !1, this.useObjectSpaceNormalMap = !1, this._isUnique = !0, this.registerInput("worldPosition", de.Vector4, !1), this.registerInput("worldNormal", de.Vector4, !1), this.registerInput("worldTangent", de.Vector4, !0), this.registerInput("uv", de.Vector2, !1), this.registerInput("normalMapColor", de.Color3, !1), this.registerInput("strength", de.Float, !1), this.registerInput("viewDirection", de.Vector3, !0), this.registerInput("parallaxScale", de.Float, !0), this.registerInput("parallaxHeight", de.Float, !0), this.registerInput("TBN", de.Object, !0, Ve.VertexAndFragment, new Ko("TBN", this, ao.Input, Rm, "TBNBlock")), this.registerInput("world", de.Matrix, !0), this.registerOutput("output", de.Vector4), this.registerOutput("uvOffset", de.Vector2); } /** * Gets the current class name * @returns the class name */ getClassName() { return "PerturbNormalBlock"; } /** * Gets the world position input component */ get worldPosition() { return this._inputs[0]; } /** * Gets the world normal input component */ get worldNormal() { return this._inputs[1]; } /** * Gets the world tangent input component */ get worldTangent() { return this._inputs[2]; } /** * Gets the uv input component */ get uv() { return this._inputs[3]; } /** * Gets the normal map color input component */ get normalMapColor() { return this._inputs[4]; } /** * Gets the strength input component */ get strength() { return this._inputs[5]; } /** * Gets the view direction input component */ get viewDirection() { return this._inputs[6]; } /** * Gets the parallax scale input component */ get parallaxScale() { return this._inputs[7]; } /** * Gets the parallax height input component */ get parallaxHeight() { return this._inputs[8]; } /** * Gets the TBN input component */ // eslint-disable-next-line @typescript-eslint/naming-convention get TBN() { return this._inputs[9]; } /** * Gets the World input component */ get world() { return this._inputs[10]; } /** * Gets the output component */ get output() { return this._outputs[0]; } /** * Gets the uv offset output component */ get uvOffset() { return this._outputs[1]; } prepareDefines(e, t, r) { const n = this.normalMapColor.connectedPoint._ownerBlock.samplerName, i = this.viewDirection.isConnected && (this.useParallaxOcclusion && n || !this.useParallaxOcclusion && this.parallaxHeight.isConnected); r.setValue("BUMP", !0), r.setValue("PARALLAX", i, !0), r.setValue("PARALLAX_RHS", t.getScene().useRightHandedSystem, !0), r.setValue("PARALLAXOCCLUSION", this.useParallaxOcclusion, !0), r.setValue("OBJECTSPACE_NORMALMAP", this.useObjectSpaceNormalMap, !0); } bind(e, t, r) { t.getScene()._mirroredCameraPosition ? e.setFloat2(this._tangentSpaceParameterName, this.invertX ? 1 : -1, this.invertY ? 1 : -1) : e.setFloat2(this._tangentSpaceParameterName, this.invertX ? -1 : 1, this.invertY ? -1 : 1), r && (e.setFloat(this._tangentCorrectionFactorName, r.getWorldMatrix().determinant() < 0 ? -1 : 1), this.useObjectSpaceNormalMap && !this.world.isConnected && e.setMatrix(this._worldMatrixName, r.getWorldMatrix())); } autoConfigure(e, t = () => !0) { if (!this.uv.isConnected) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "uv" && t(n)); r || (r = new gi("uv"), r.setAsAttribute()), r.output.connectTo(this.uv); } if (!this.strength.isConnected) { const r = new gi("strength"); r.value = 1, r.output.connectTo(this.strength); } } _buildBlock(e) { super._buildBlock(e); const t = `//${this.name}`, r = this.uv, n = this.worldPosition, i = this.worldNormal, s = this.worldTangent; e.sharedData.blocksWithDefines.push(this), e.sharedData.bindableBlocks.push(this), this._tangentSpaceParameterName = e._getFreeDefineName("tangentSpaceParameter"), e._emitUniformFromString(this._tangentSpaceParameterName, "vec2"), this._tangentCorrectionFactorName = e._getFreeDefineName("tangentCorrectionFactor"), e._emitUniformFromString(this._tangentCorrectionFactorName, "float"), this._worldMatrixName = e._getFreeDefineName("perturbNormalWorldMatrix"), e._emitUniformFromString(this._worldMatrixName, "mat4"); let a = null; this.normalMapColor.connectedPoint && (a = this.normalMapColor.connectedPoint._ownerBlock.samplerName); const f = this.viewDirection.isConnected && (this.useParallaxOcclusion && a || !this.useParallaxOcclusion && this.parallaxHeight.isConnected), o = this.parallaxScale.isConnectedToInputBlock ? this.parallaxScale.connectInputBlock.isConstant ? e._emitFloat(this.parallaxScale.connectInputBlock.value) : this.parallaxScale.associatedVariableName : "0.05", d = this.strength.isConnectedToInputBlock && this.strength.connectInputBlock.isConstant ? ` #if !defined(NORMALXYSCALE) 1.0/ #endif ${e._emitFloat(this.strength.connectInputBlock.value)}` : ` #if !defined(NORMALXYSCALE) 1.0/ #endif ${this.strength.associatedVariableName}`; e._emitExtension("derivatives", "#extension GL_OES_standard_derivatives : enable"); const v = { search: /defined\(TANGENT\)/g, replace: s.isConnected ? "defined(TANGENT)" : "defined(IGNORE)" }, u = { search: /varying mat3 vTBN;/g, replace: "" }, l = { search: /uniform mat4 normalMatrix;/g, replace: "" }, P = this.TBN; P.isConnected ? e.compilationString += ` #ifdef TBNBLOCK mat3 vTBN = ${P.associatedVariableName}; #endif ` : s.isConnected && (e.compilationString += `vec3 tbnNormal = normalize(${i.associatedVariableName}.xyz); `, e.compilationString += `vec3 tbnTangent = normalize(${s.associatedVariableName}.xyz); `, e.compilationString += `vec3 tbnBitangent = cross(tbnNormal, tbnTangent) * ${this._tangentCorrectionFactorName}; `, e.compilationString += `mat3 vTBN = mat3(tbnTangent, tbnBitangent, tbnNormal); `), e._emitFunctionFromInclude("bumpFragmentMainFunctions", t, { replaceStrings: [v, u, l] }), e._emitFunctionFromInclude("bumpFragmentFunctions", t, { replaceStrings: [ { search: /#include\(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump,_SAMPLERNAME_,bump\)/g, replace: "" }, { search: /uniform sampler2D bumpSampler;/g, replace: "" }, { search: /vec2 parallaxOcclusion\(vec3 vViewDirCoT,vec3 vNormalCoT,vec2 texCoord,float parallaxScale\)/g, replace: `#define inline vec2 parallaxOcclusion(vec3 vViewDirCoT, vec3 vNormalCoT, vec2 texCoord, float parallaxScale, sampler2D bumpSampler)` }, { search: /vec2 parallaxOffset\(vec3 viewDir,float heightScale\)/g, replace: "vec2 parallaxOffset(vec3 viewDir, float heightScale, float height_)" }, { search: /texture2D\(bumpSampler,vBumpUV\)\.w/g, replace: "height_" } ] }); const p = !f || !a ? this.normalMapColor.associatedVariableName : `texture2D(${a}, ${r.associatedVariableName} + uvOffset).xyz`; return e.compilationString += this._declareOutput(this.output, e) + ` = vec4(0.); `, e.compilationString += e._emitCodeFromInclude("bumpFragment", t, { replaceStrings: [ { search: /texture2D\(bumpSampler,vBumpUV\)/g, replace: `${p}` }, { search: /#define CUSTOM_FRAGMENT_BUMP_FRAGMENT/g, replace: `mat4 normalMatrix = toNormalMatrix(${this.world.isConnected ? this.world.associatedVariableName : this._worldMatrixName});` }, { search: /perturbNormal\(TBN,texture2D\(bumpSampler,vBumpUV\+uvOffset\).xyz,vBumpInfos.y\)/g, replace: `perturbNormal(TBN, ${p}, vBumpInfos.y)` }, { search: /parallaxOcclusion\(invTBN\*-viewDirectionW,invTBN\*normalW,vBumpUV,vBumpInfos.z\)/g, replace: `parallaxOcclusion((invTBN * -viewDirectionW), (invTBN * normalW), vBumpUV, vBumpInfos.z, ${f && this.useParallaxOcclusion ? a : "bumpSampler"})` }, { search: /parallaxOffset\(invTBN\*viewDirectionW,vBumpInfos\.z\)/g, replace: `parallaxOffset(invTBN * viewDirectionW, vBumpInfos.z, ${f ? this.parallaxHeight.associatedVariableName : "0."})` }, { search: /vTangentSpaceParams/g, replace: this._tangentSpaceParameterName }, { search: /vBumpInfos.y/g, replace: d }, { search: /vBumpInfos.z/g, replace: o }, { search: /vBumpUV/g, replace: r.associatedVariableName }, { search: /vPositionW/g, replace: n.associatedVariableName + ".xyz" }, { search: /normalW=/g, replace: this.output.associatedVariableName + ".xyz = " }, { search: /mat3\(normalMatrix\)\*normalW/g, replace: "mat3(normalMatrix) * " + this.output.associatedVariableName + ".xyz" }, { search: /normalW/g, replace: i.associatedVariableName + ".xyz" }, { search: /viewDirectionW/g, replace: f ? this.viewDirection.associatedVariableName : "vec3(0.)" }, v ] }), this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode() + `${this._codeVariableName}.invertX = ${this.invertX}; `; return e += `${this._codeVariableName}.invertY = ${this.invertY}; `, e += `${this._codeVariableName}.useParallaxOcclusion = ${this.useParallaxOcclusion}; `, e += `${this._codeVariableName}.useObjectSpaceNormalMap = ${this.useObjectSpaceNormalMap}; `, e; } serialize() { const e = super.serialize(); return e.invertX = this.invertX, e.invertY = this.invertY, e.useParallaxOcclusion = this.useParallaxOcclusion, e.useObjectSpaceNormalMap = this.useObjectSpaceNormalMap, e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.invertX = e.invertX, this.invertY = e.invertY, this.useParallaxOcclusion = !!e.useParallaxOcclusion, this.useObjectSpaceNormalMap = !!e.useObjectSpaceNormalMap; } } C([ rn("Invert X axis", Gr.Boolean, "PROPERTIES", { notifiers: { update: !1 } }) ], PU.prototype, "invertX", void 0); C([ rn("Invert Y axis", Gr.Boolean, "PROPERTIES", { notifiers: { update: !1 } }) ], PU.prototype, "invertY", void 0); C([ rn("Use parallax occlusion", Gr.Boolean) ], PU.prototype, "useParallaxOcclusion", void 0); C([ rn("Object Space Mode", Gr.Boolean, "PROPERTIES", { notifiers: { update: !1 } }) ], PU.prototype, "useObjectSpaceNormalMap", void 0); Ue("BABYLON.PerturbNormalBlock", PU); class are extends Mr { /** * Create a new DiscardBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment, !0), this.registerInput("value", de.Float, !0), this.registerInput("cutoff", de.Float, !0); } /** * Gets the current class name * @returns the class name */ getClassName() { return "DiscardBlock"; } /** * Gets the color input component */ get value() { return this._inputs[0]; } /** * Gets the cutoff input component */ get cutoff() { return this._inputs[1]; } _buildBlock(e) { if (super._buildBlock(e), e.sharedData.hints.needAlphaTesting = !0, !(!this.cutoff.isConnected || !this.value.isConnected)) return e.compilationString += `if (${this.value.associatedVariableName} < ${this.cutoff.associatedVariableName}) discard; `, this; } } Ue("BABYLON.DiscardBlock", are); class ore extends Mr { /** * Creates a new FrontFacingBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this.registerOutput("output", de.Float, Ve.Fragment); } /** * Gets the current class name * @returns the class name */ getClassName() { return "FrontFacingBlock"; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { if (super._buildBlock(e), e.target === Ve.Vertex) throw "FrontFacingBlock must only be used in a fragment shader"; const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = gl_FrontFacing ? 1.0 : 0.0; `, this; } } Ue("BABYLON.FrontFacingBlock", ore); class fre extends Mr { /** * Create a new DerivativeBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this.registerInput("input", de.AutoDetect, !1), this.registerOutput("dx", de.BasedOnInput), this.registerOutput("dy", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._outputs[1]._typeConnectionSource = this._inputs[0]; } /** * Gets the current class name * @returns the class name */ getClassName() { return "DerivativeBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the derivative output on x */ get dx() { return this._outputs[0]; } /** * Gets the derivative output on y */ get dy() { return this._outputs[1]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0], r = this._outputs[1]; return e._emitExtension("derivatives", "#extension GL_OES_standard_derivatives : enable"), t.hasEndpoints && (e.compilationString += this._declareOutput(t, e) + ` = dFdx(${this.input.associatedVariableName}); `), r.hasEndpoints && (e.compilationString += this._declareOutput(r, e) + ` = dFdy(${this.input.associatedVariableName}); `), this; } } Ue("BABYLON.DerivativeBlock", fre); class Are extends Mr { /** * Creates a new FragCoordBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this.registerOutput("xy", de.Vector2, Ve.Fragment), this.registerOutput("xyz", de.Vector3, Ve.Fragment), this.registerOutput("xyzw", de.Vector4, Ve.Fragment), this.registerOutput("x", de.Float, Ve.Fragment), this.registerOutput("y", de.Float, Ve.Fragment), this.registerOutput("z", de.Float, Ve.Fragment), this.registerOutput("w", de.Float, Ve.Fragment); } /** * Gets the current class name * @returns the class name */ getClassName() { return "FragCoordBlock"; } /** * Gets the xy component */ get xy() { return this._outputs[0]; } /** * Gets the xyz component */ get xyz() { return this._outputs[1]; } /** * Gets the xyzw component */ get xyzw() { return this._outputs[2]; } /** * Gets the x component */ get x() { return this._outputs[3]; } /** * Gets the y component */ get y() { return this._outputs[4]; } /** * Gets the z component */ get z() { return this._outputs[5]; } /** * Gets the w component */ get output() { return this._outputs[6]; } // eslint-disable-next-line @typescript-eslint/naming-convention writeOutputs(e) { let t = ""; for (const r of this._outputs) r.hasEndpoints && (t += `${this._declareOutput(r, e)} = gl_FragCoord.${r.name}; `); return t; } _buildBlock(e) { if (super._buildBlock(e), e.target === Ve.Vertex) throw "FragCoordBlock must only be used in a fragment shader"; return e.compilationString += this.writeOutputs(e), this; } } Ue("BABYLON.FragCoordBlock", Are); class dre extends Mr { /** * Creates a new ScreenSizeBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this.registerOutput("xy", de.Vector2, Ve.Fragment), this.registerOutput("x", de.Float, Ve.Fragment), this.registerOutput("y", de.Float, Ve.Fragment); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ScreenSizeBlock"; } /** * Gets the xy component */ get xy() { return this._outputs[0]; } /** * Gets the x component */ get x() { return this._outputs[1]; } /** * Gets the y component */ get y() { return this._outputs[2]; } bind(e) { const t = this._scene.getEngine(); e.setFloat2(this._varName, t.getRenderWidth(), t.getRenderHeight()); } // eslint-disable-next-line @typescript-eslint/naming-convention writeOutputs(e, t) { let r = ""; for (const n of this._outputs) n.hasEndpoints && (r += `${this._declareOutput(n, e)} = ${t}.${n.name}; `); return r; } _buildBlock(e) { if (super._buildBlock(e), this._scene = e.sharedData.scene, e.target === Ve.Vertex) throw "ScreenSizeBlock must only be used in a fragment shader"; return e.sharedData.bindableBlocks.push(this), this._varName = e._getFreeVariableName("screenSize"), e._emitUniformFromString(this._varName, "vec2"), e.compilationString += this.writeOutputs(e, this._varName), this; } } Ue("BABYLON.ScreenSizeBlock", dre); class vre extends Mr { /** * Creates a new ScreenSpaceBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this.registerInput("vector", de.AutoDetect), this.registerInput("worldViewProjection", de.Matrix), this.registerOutput("output", de.Vector2), this.registerOutput("x", de.Float), this.registerOutput("y", de.Float), this.inputs[0].addExcludedConnectionPointFromAllowedTypes(de.Color3 | de.Vector3 | de.Vector4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ScreenSpaceBlock"; } /** * Gets the vector input */ get vector() { return this._inputs[0]; } /** * Gets the worldViewProjection transform input */ get worldViewProjection() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } /** * Gets the x output component */ get x() { return this._outputs[1]; } /** * Gets the y output component */ get y() { return this._outputs[2]; } autoConfigure(e, t = () => !0) { if (!this.worldViewProjection.isConnected) { let r = e.getInputBlockByPredicate((n) => n.systemValue === Bi.WorldViewProjection && t(n)); r || (r = new gi("worldViewProjection"), r.setAsSystemValue(Bi.WorldViewProjection)), r.output.connectTo(this.worldViewProjection); } } _buildBlock(e) { super._buildBlock(e); const t = this.vector, r = this.worldViewProjection; if (!t.connectedPoint) return; const n = r.associatedVariableName, i = e._getFreeVariableName("screenSpaceTemp"); switch (t.connectedPoint.type) { case de.Vector3: e.compilationString += `vec4 ${i} = ${n} * vec4(${t.associatedVariableName}, 1.0); `; break; case de.Vector4: e.compilationString += `vec4 ${i} = ${n} * ${t.associatedVariableName}; `; break; } return e.compilationString += `${i}.xy /= ${i}.w;`, e.compilationString += `${i}.xy = ${i}.xy * 0.5 + vec2(0.5, 0.5);`, this.output.hasEndpoints && (e.compilationString += this._declareOutput(this.output, e) + ` = ${i}.xy; `), this.x.hasEndpoints && (e.compilationString += this._declareOutput(this.x, e) + ` = ${i}.x; `), this.y.hasEndpoints && (e.compilationString += this._declareOutput(this.y, e) + ` = ${i}.y; `), this; } } Ue("BABYLON.ScreenSpaceBlock", vre); class ure extends Mr { /** * Creates a new TwirlBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this.registerInput("input", de.Vector2), this.registerInput("strength", de.Float), this.registerInput("center", de.Vector2), this.registerInput("offset", de.Vector2), this.registerOutput("output", de.Vector2), this.registerOutput("x", de.Float), this.registerOutput("y", de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "TwirlBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the strength component */ get strength() { return this._inputs[1]; } /** * Gets the center component */ get center() { return this._inputs[2]; } /** * Gets the offset component */ get offset() { return this._inputs[3]; } /** * Gets the output component */ get output() { return this._outputs[0]; } /** * Gets the x output component */ get x() { return this._outputs[1]; } /** * Gets the y output component */ get y() { return this._outputs[2]; } autoConfigure() { if (!this.center.isConnected) { const e = new gi("center"); e.value = new at(0.5, 0.5), e.output.connectTo(this.center); } if (!this.strength.isConnected) { const e = new gi("strength"); e.value = 1, e.output.connectTo(this.strength); } if (!this.offset.isConnected) { const e = new gi("offset"); e.value = new at(0, 0), e.output.connectTo(this.offset); } } _buildBlock(e) { super._buildBlock(e); const t = e._getFreeVariableName("delta"), r = e._getFreeVariableName("angle"), n = e._getFreeVariableName("x"), i = e._getFreeVariableName("y"), s = e._getFreeVariableName("result"); return e.compilationString += ` vec2 ${t} = ${this.input.associatedVariableName} - ${this.center.associatedVariableName}; float ${r} = ${this.strength.associatedVariableName} * length(${t}); float ${n} = cos(${r}) * ${t}.x - sin(${r}) * ${t}.y; float ${i} = sin(${r}) * ${t}.x + cos(${r}) * ${t}.y; vec2 ${s} = vec2(${n} + ${this.center.associatedVariableName}.x + ${this.offset.associatedVariableName}.x, ${i} + ${this.center.associatedVariableName}.y + ${this.offset.associatedVariableName}.y); `, this.output.hasEndpoints && (e.compilationString += this._declareOutput(this.output, e) + ` = ${s}; `), this.x.hasEndpoints && (e.compilationString += this._declareOutput(this.x, e) + ` = ${s}.x; `), this.y.hasEndpoints && (e.compilationString += this._declareOutput(this.y, e) + ` = ${s}.y; `), this; } } Ue("BABYLON.TwirlBlock", ure); class aV extends Mr { /** * Creates a new HeightToNormalBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this.generateInWorldSpace = !1, this.automaticNormalizationNormal = !0, this.automaticNormalizationTangent = !0, this.registerInput("input", de.Float), this.registerInput("worldPosition", de.Vector3), this.registerInput("worldNormal", de.Vector3), this.registerInput("worldTangent", de.AutoDetect, !0), this.registerOutput("output", de.Vector4), this.registerOutput("xyz", de.Vector3), this._inputs[3].addExcludedConnectionPointFromAllowedTypes(de.Color3 | de.Vector3 | de.Vector4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "HeightToNormalBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the position component */ get worldPosition() { return this._inputs[1]; } /** * Gets the normal component */ get worldNormal() { return this._inputs[2]; } /** * Gets the tangent component */ get worldTangent() { return this._inputs[3]; } /** * Gets the output component */ get output() { return this._outputs[0]; } /** * Gets the xyz component */ get xyz() { return this._outputs[1]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; !this.generateInWorldSpace && !this.worldTangent.isConnected && console.error(`You must connect the 'worldTangent' input of the ${this.name} block!`); const r = this.generateInWorldSpace ? "" : ` vec3 biTangent = cross(normal, tangent); mat3 TBN = mat3(tangent, biTangent, normal); `, n = this.generateInWorldSpace ? "" : ` result = TBN * result; result = result * vec3(0.5) + vec3(0.5); `, i = ` vec4 heightToNormal(in float height, in vec3 position, in vec3 tangent, in vec3 normal) { ${r} ${this.automaticNormalizationTangent ? "tangent = normalize(tangent);" : ""} ${this.automaticNormalizationNormal ? "normal = normalize(normal);" : ""} vec3 worlddX = dFdx(position); vec3 worlddY = dFdy(position); vec3 crossX = cross(normal, worlddX); vec3 crossY = cross(normal, worlddY); float d = abs(dot(crossY, worlddX)); vec3 inToNormal = vec3(((((height + dFdx(height)) - height) * crossY) + (((height + dFdy(height)) - height) * crossX)) * sign(d)); inToNormal.y *= -1.0; vec3 result = normalize((d * normal) - inToNormal); ${n} return vec4(result, 0.); }`; return e._emitExtension("derivatives", "#extension GL_OES_standard_derivatives : enable"), e._emitFunction("heightToNormal", i, "// heightToNormal"), e.compilationString += this._declareOutput(t, e) + ` = heightToNormal(${this.input.associatedVariableName}, ${this.worldPosition.associatedVariableName}, ${this.worldTangent.isConnected ? this.worldTangent.associatedVariableName : "vec3(0.)"}.xyz, ${this.worldNormal.associatedVariableName}); `, this.xyz.hasEndpoints && (e.compilationString += this._declareOutput(this.xyz, e) + ` = ${this.output.associatedVariableName}.xyz; `), this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return e += `${this._codeVariableName}.generateInWorldSpace = ${this.generateInWorldSpace}; `, e += `${this._codeVariableName}.automaticNormalizationNormal = ${this.automaticNormalizationNormal}; `, e += `${this._codeVariableName}.automaticNormalizationTangent = ${this.automaticNormalizationTangent}; `, e; } serialize() { const e = super.serialize(); return e.generateInWorldSpace = this.generateInWorldSpace, e.automaticNormalizationNormal = this.automaticNormalizationNormal, e.automaticNormalizationTangent = this.automaticNormalizationTangent, e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.generateInWorldSpace = e.generateInWorldSpace, this.automaticNormalizationNormal = e.automaticNormalizationNormal, this.automaticNormalizationTangent = e.automaticNormalizationTangent; } } C([ rn("Generate in world space instead of tangent space", Gr.Boolean, "PROPERTIES", { notifiers: { update: !0 } }) ], aV.prototype, "generateInWorldSpace", void 0); C([ rn("Force normalization for the worldNormal input", Gr.Boolean, "PROPERTIES", { notifiers: { update: !0 } }) ], aV.prototype, "automaticNormalizationNormal", void 0); C([ rn("Force normalization for the worldTangent input", Gr.Boolean, "PROPERTIES", { notifiers: { update: !0 } }) ], aV.prototype, "automaticNormalizationTangent", void 0); Ue("BABYLON.HeightToNormalBlock", aV); class lre extends Mr { /** * Create a new FragDepthBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment, !0), this.registerInput("depth", de.Float, !0), this.registerInput("worldPos", de.Vector4, !0), this.registerInput("viewProjection", de.Matrix, !0); } /** * Gets the current class name * @returns the class name */ getClassName() { return "FragDepthBlock"; } /** * Gets the depth input component */ get depth() { return this._inputs[0]; } /** * Gets the worldPos input component */ get worldPos() { return this._inputs[1]; } /** * Gets the viewProjection input component */ get viewProjection() { return this._inputs[2]; } _buildBlock(e) { return super._buildBlock(e), this.depth.isConnected ? e.compilationString += `gl_FragDepth = ${this.depth.associatedVariableName}; ` : this.worldPos.isConnected && this.viewProjection.isConnected ? e.compilationString += ` vec4 p = ${this.viewProjection.associatedVariableName} * ${this.worldPos.associatedVariableName}; float v = p.z / p.w; #ifndef IS_NDC_HALF_ZRANGE v = v * 0.5 + 0.5; #endif gl_FragDepth = v; ` : console.warn("FragDepthBlock: either the depth input or both the worldPos and viewProjection inputs must be connected!"), this; } } Ue("BABYLON.FragDepthBlock", lre); class Pre extends Mr { /** * Create a new ShadowMapBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this.registerInput("worldPosition", de.Vector4, !1), this.registerInput("viewProjection", de.Matrix, !1), this.registerInput("worldNormal", de.AutoDetect, !0), this.registerOutput("depth", de.Vector3), this.worldNormal.addExcludedConnectionPointFromAllowedTypes(de.Color3 | de.Vector3 | de.Vector4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ShadowMapBlock"; } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("vPositionWSM"), e._excludeVariableName("lightDataSM"), e._excludeVariableName("biasAndScaleSM"), e._excludeVariableName("depthValuesSM"), e._excludeVariableName("clipPos"), e._excludeVariableName("worldPos"), e._excludeVariableName("zSM"); } /** * Gets the world position input component */ get worldPosition() { return this._inputs[0]; } /** * Gets the view x projection input component */ get viewProjection() { return this._inputs[1]; } /** * Gets the world normal input component */ get worldNormal() { return this._inputs[2]; } /** * Gets the depth output component */ get depth() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = `//${this.name}`; return e._emitUniformFromString("biasAndScaleSM", "vec3"), e._emitUniformFromString("lightDataSM", "vec3"), e._emitUniformFromString("depthValuesSM", "vec2"), e._emitFunctionFromInclude("packingFunctions", t), e.compilationString += `vec4 worldPos = ${this.worldPosition.associatedVariableName}; `, e.compilationString += `vec3 vPositionWSM; `, e.compilationString += `float vDepthMetricSM = 0.0; `, e.compilationString += `float zSM; `, this.worldNormal.isConnected && (e.compilationString += `vec3 vNormalW = ${this.worldNormal.associatedVariableName}.xyz; `, e.compilationString += e._emitCodeFromInclude("shadowMapVertexNormalBias", t)), e.compilationString += `vec4 clipPos = ${this.viewProjection.associatedVariableName} * worldPos; `, e.compilationString += e._emitCodeFromInclude("shadowMapVertexMetric", t, { replaceStrings: [ { search: /gl_Position/g, replace: "clipPos" } ] }), e.compilationString += e._emitCodeFromInclude("shadowMapFragment", t, { replaceStrings: [ { search: /return;/g, replace: "" } ] }), e.compilationString += ` #if SM_DEPTHTEXTURE == 1 #ifdef IS_NDC_HALF_ZRANGE gl_FragDepth = (clipPos.z / clipPos.w); #else gl_FragDepth = (clipPos.z / clipPos.w) * 0.5 + 0.5; #endif #endif `, e.compilationString += `${this._declareOutput(this.depth, e)} = vec3(depthSM, 1., 1.); `, this; } } Ue("BABYLON.ShadowMapBlock", Pre); class cre extends Mr { /** * Create a new PrePassOutputBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment, !0), this.registerInput("viewDepth", de.Float, !0), this.registerInput("worldPosition", de.AutoDetect, !0), this.registerInput("viewNormal", de.AutoDetect, !0), this.registerInput("reflectivity", de.AutoDetect, !0), this.inputs[1].addExcludedConnectionPointFromAllowedTypes(de.Vector3 | de.Vector4), this.inputs[2].addExcludedConnectionPointFromAllowedTypes(de.Vector3 | de.Vector4), this.inputs[3].addExcludedConnectionPointFromAllowedTypes(de.Vector3 | de.Vector4 | de.Color3 | de.Color4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "PrePassOutputBlock"; } /** * Gets the view depth component */ get viewDepth() { return this._inputs[0]; } /** * Gets the world position component */ get worldPosition() { return this._inputs[1]; } /** * Gets the view normal component */ get viewNormal() { return this._inputs[2]; } /** * Gets the reflectivity component */ get reflectivity() { return this._inputs[3]; } _buildBlock(e) { super._buildBlock(e); const t = this.worldPosition, r = this.viewNormal, n = this.viewDepth, i = this.reflectivity; e.sharedData.blocksWithDefines.push(this); const s = `//${this.name}`; return e._emitFunctionFromInclude("helperFunctions", s), e.compilationString += `#if defined(PREPASS)\r `, e.compilationString += `#ifdef PREPASS_DEPTH\r `, n.connectedPoint ? e.compilationString += ` gl_FragData[PREPASS_DEPTH_INDEX] = vec4(${n.associatedVariableName}, 0.0, 0.0, 1.0);\r ` : e.compilationString += ` gl_FragData[PREPASS_DEPTH_INDEX] = vec4(0.0, 0.0, 0.0, 0.0);\r `, e.compilationString += `#endif\r `, e.compilationString += `#ifdef PREPASS_POSITION\r `, t.connectedPoint ? e.compilationString += ` gl_FragData[PREPASS_POSITION_INDEX] = vec4(${t.associatedVariableName}.rgb, ${t.connectedPoint.type === de.Vector4 ? t.associatedVariableName + ".a" : "1.0"});\r ` : e.compilationString += ` gl_FragData[PREPASS_POSITION_INDEX] = vec4(0.0, 0.0, 0.0, 0.0);\r `, e.compilationString += `#endif\r `, e.compilationString += `#ifdef PREPASS_NORMAL\r `, r.connectedPoint ? e.compilationString += ` gl_FragData[PREPASS_NORMAL_INDEX] = vec4(${r.associatedVariableName}.rgb, ${r.connectedPoint.type === de.Vector4 ? r.associatedVariableName + ".a" : "1.0"});\r ` : e.compilationString += ` gl_FragData[PREPASS_NORMAL_INDEX] = vec4(0.0, 0.0, 0.0, 0.0);\r `, e.compilationString += `#endif\r `, e.compilationString += `#ifdef PREPASS_REFLECTIVITY\r `, i.connectedPoint ? e.compilationString += ` gl_FragData[PREPASS_REFLECTIVITY_INDEX] = vec4(${i.associatedVariableName}.rgb, ${i.connectedPoint.type === de.Vector4 ? i.associatedVariableName + ".a" : "1.0"});\r ` : e.compilationString += ` gl_FragData[PREPASS_REFLECTIVITY_INDEX] = vec4(0.0, 0.0, 0.0, 1.0);\r `, e.compilationString += `#endif\r `, e.compilationString += `#endif\r `, this; } } Ue("BABYLON.PrePassOutputBlock", cre); class pre extends Mr { /** * Create a new FogBlock * @param name defines the block name */ constructor(e) { super(e, Ve.VertexAndFragment, !1), this.registerInput("worldPosition", de.Vector4, !1, Ve.Vertex), this.registerInput("view", de.Matrix, !1, Ve.Vertex), this.registerInput("input", de.AutoDetect, !1, Ve.Fragment), this.registerInput("fogColor", de.AutoDetect, !1, Ve.Fragment), this.registerOutput("output", de.Color3, Ve.Fragment), this.input.addExcludedConnectionPointFromAllowedTypes(de.Color3 | de.Vector3 | de.Color4), this.fogColor.addExcludedConnectionPointFromAllowedTypes(de.Color3 | de.Vector3 | de.Color4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "FogBlock"; } /** * Gets the world position input component */ get worldPosition() { return this._inputs[0]; } /** * Gets the view input component */ get view() { return this._inputs[1]; } /** * Gets the color input component */ get input() { return this._inputs[2]; } /** * Gets the fog color input component */ get fogColor() { return this._inputs[3]; } /** * Gets the output component */ get output() { return this._outputs[0]; } autoConfigure(e, t = () => !0) { if (!this.view.isConnected) { let r = e.getInputBlockByPredicate((n) => n.systemValue === Bi.View && t(n)); r || (r = new gi("view"), r.setAsSystemValue(Bi.View)), r.output.connectTo(this.view); } if (!this.fogColor.isConnected) { let r = e.getInputBlockByPredicate((n) => n.systemValue === Bi.FogColor && t(n)); r || (r = new gi("fogColor", void 0, de.Color3), r.setAsSystemValue(Bi.FogColor)), r.output.connectTo(this.fogColor); } } prepareDefines(e, t, r) { const n = e.getScene(); r.setValue("FOG", t.fogEnabled && Ye.GetFogState(e, n)); } bind(e, t, r) { if (!r) return; const n = r.getScene(); e.setFloat4(this._fogParameters, n.fogMode, n.fogStart, n.fogEnd, n.fogDensity); } _buildBlock(e) { if (super._buildBlock(e), e.target === Ve.Fragment) { e.sharedData.blocksWithDefines.push(this), e.sharedData.bindableBlocks.push(this), e._emitFunctionFromInclude("fogFragmentDeclaration", `//${this.name}`, { removeUniforms: !0, removeVaryings: !0, removeIfDef: !1, replaceStrings: [{ search: /float CalcFogFactor\(\)/, replace: "float CalcFogFactor(vec3 vFogDistance, vec4 vFogInfos)" }] }); const t = e._getFreeVariableName("fog"), r = this.input, n = this.fogColor; this._fogParameters = e._getFreeVariableName("fogParameters"); const i = this._outputs[0]; e._emitUniformFromString(this._fogParameters, "vec4"), e.compilationString += `#ifdef FOG `, e.compilationString += `float ${t} = CalcFogFactor(${this._fogDistanceName}, ${this._fogParameters}); `, e.compilationString += this._declareOutput(i, e) + ` = ${t} * ${r.associatedVariableName}.rgb + (1.0 - ${t}) * ${n.associatedVariableName}.rgb; `, e.compilationString += `#else ${this._declareOutput(i, e)} = ${r.associatedVariableName}.rgb; `, e.compilationString += `#endif `; } else { const t = this.worldPosition, r = this.view; this._fogDistanceName = e._getFreeVariableName("vFogDistance"), e._emitVaryingFromString(this._fogDistanceName, "vec3"), e.compilationString += `${this._fogDistanceName} = (${r.associatedVariableName} * ${t.associatedVariableName}).xyz; `; } return this; } } Ue("BABYLON.FogBlock", pre); class AO extends Mr { static _OnGenerateOnlyFragmentCodeChanged(e, t) { const r = e; return r.worldPosition.isConnected ? (r.generateOnlyFragmentCode = !r.generateOnlyFragmentCode, console.error("The worldPosition input must not be connected to be able to switch!"), !1) : (r._setTarget(), !0); } _setTarget() { this._setInitialTarget(this.generateOnlyFragmentCode ? Ve.Fragment : Ve.VertexAndFragment), this.getInputByName("worldPosition").target = this.generateOnlyFragmentCode ? Ve.Fragment : Ve.Vertex; } /** * Create a new LightBlock * @param name defines the block name */ constructor(e) { super(e, Ve.VertexAndFragment), this._lightId = 0, this.generateOnlyFragmentCode = !1, this._isUnique = !0, this.registerInput("worldPosition", de.Vector4, !1, Ve.Vertex), this.registerInput("worldNormal", de.Vector4, !1, Ve.Fragment), this.registerInput("cameraPosition", de.Vector3, !1, Ve.Fragment), this.registerInput("glossiness", de.Float, !0, Ve.Fragment), this.registerInput("glossPower", de.Float, !0, Ve.Fragment), this.registerInput("diffuseColor", de.Color3, !0, Ve.Fragment), this.registerInput("specularColor", de.Color3, !0, Ve.Fragment), this.registerInput("view", de.Matrix, !0), this.registerOutput("diffuseOutput", de.Color3, Ve.Fragment), this.registerOutput("specularOutput", de.Color3, Ve.Fragment), this.registerOutput("shadow", de.Float, Ve.Fragment); } /** * Gets the current class name * @returns the class name */ getClassName() { return "LightBlock"; } /** * Gets the world position input component */ get worldPosition() { return this._inputs[0]; } /** * Gets the world normal input component */ get worldNormal() { return this._inputs[1]; } /** * Gets the camera (or eye) position component */ get cameraPosition() { return this._inputs[2]; } /** * Gets the glossiness component */ get glossiness() { return this._inputs[3]; } /** * Gets the glossiness power component */ get glossPower() { return this._inputs[4]; } /** * Gets the diffuse color component */ get diffuseColor() { return this._inputs[5]; } /** * Gets the specular color component */ get specularColor() { return this._inputs[6]; } /** * Gets the view matrix component */ get view() { return this._inputs[7]; } /** * Gets the diffuse output component */ get diffuseOutput() { return this._outputs[0]; } /** * Gets the specular output component */ get specularOutput() { return this._outputs[1]; } /** * Gets the shadow output component */ get shadow() { return this._outputs[2]; } autoConfigure(e, t = () => !0) { if (!this.cameraPosition.isConnected) { let r = e.getInputBlockByPredicate((n) => n.systemValue === Bi.CameraPosition && t(n)); r || (r = new gi("cameraPosition"), r.setAsSystemValue(Bi.CameraPosition)), r.output.connectTo(this.cameraPosition); } } prepareDefines(e, t, r) { if (!r._areLightsDirty) return; const n = e.getScene(); if (!this.light) Ye.PrepareDefinesForLights(n, e, r, !0, t.maxSimultaneousLights); else { const i = { needNormals: !1, needRebuild: !1, lightmapMode: !1, shadowEnabled: !1, specularEnabled: !1 }; Ye.PrepareDefinesForLight(n, e, this.light, this._lightId, r, !0, i), i.needRebuild && r.rebuild(); } } updateUniformsAndSamples(e, t, r, n) { for (let i = 0; i < t.maxSimultaneousLights && r["LIGHT" + i]; i++) { const s = e.uniforms.indexOf("vLightData" + i) >= 0; Ye.PrepareUniformsAndSamplersForLight(i, e.uniforms, e.samplers, r["PROJECTEDLIGHTTEXTURE" + i], n, s); } } bind(e, t, r) { if (!r) return; const n = r.getScene(); this.light ? Ye.BindLight(this.light, this._lightId, n, e, !0) : Ye.BindLights(n, r, e, !0, t.maxSimultaneousLights); } _injectVertexCode(e) { const t = this.worldPosition, r = `//${this.name}`; this.light ? (this._lightId = (e.counters.lightCounter !== void 0 ? e.counters.lightCounter : -1) + 1, e.counters.lightCounter = this._lightId, e._emitFunctionFromInclude(e.supportUniformBuffers ? "lightVxUboDeclaration" : "lightVxFragmentDeclaration", r, { replaceStrings: [{ search: /{X}/g, replace: this._lightId.toString() }] }, this._lightId.toString())) : (e._emitFunctionFromInclude(e.supportUniformBuffers ? "lightVxUboDeclaration" : "lightVxFragmentDeclaration", r, { repeatKey: "maxSimultaneousLights" }), this._lightId = 0, e.sharedData.dynamicUniformBlocks.push(this)); const n = "v_" + t.associatedVariableName; e._emitVaryingFromString(n, "vec4") && (e.compilationString += `${n} = ${t.associatedVariableName}; `), this.light ? e.compilationString += e._emitCodeFromInclude("shadowsVertex", r, { replaceStrings: [ { search: /{X}/g, replace: this._lightId.toString() }, { search: /worldPos/g, replace: t.associatedVariableName } ] }) : (e.compilationString += `vec4 worldPos = ${t.associatedVariableName}; `, this.view.isConnected && (e.compilationString += `mat4 view = ${this.view.associatedVariableName}; `), e.compilationString += e._emitCodeFromInclude("shadowsVertex", r, { repeatKey: "maxSimultaneousLights" })); } _buildBlock(e) { if (super._buildBlock(e), e.target !== Ve.Fragment) { this._injectVertexCode(e); return; } this.generateOnlyFragmentCode && e.sharedData.dynamicUniformBlocks.push(this), e.sharedData.forcedBindableBlocks.push(this), e.sharedData.blocksWithDefines.push(this); const t = `//${this.name}`, r = this.worldPosition; let n = r.associatedVariableName; this.generateOnlyFragmentCode ? (n = e._getFreeVariableName("globalWorldPos"), e._emitFunction("light_globalworldpos", `vec3 ${n}; `, t), e.compilationString += `${n} = ${r.associatedVariableName}.xyz; `, e.compilationString += e._emitCodeFromInclude("shadowsVertex", t, { repeatKey: "maxSimultaneousLights", substitutionVars: this.generateOnlyFragmentCode ? `worldPos,${r.associatedVariableName}` : void 0 })) : n = "v_" + n + ".xyz", e._emitFunctionFromInclude("helperFunctions", t), e._emitFunctionFromInclude("lightsFragmentFunctions", t, { replaceStrings: [{ search: /vPositionW/g, replace: n }] }), e._emitFunctionFromInclude("shadowsFragmentFunctions", t, { replaceStrings: [{ search: /vPositionW/g, replace: n }] }), this.light ? e._emitFunctionFromInclude(e.supportUniformBuffers ? "lightUboDeclaration" : "lightFragmentDeclaration", t, { replaceStrings: [{ search: /{X}/g, replace: this._lightId.toString() }] }, this._lightId.toString()) : e._emitFunctionFromInclude(e.supportUniformBuffers ? "lightUboDeclaration" : "lightFragmentDeclaration", t, { repeatKey: "maxSimultaneousLights", substitutionVars: this.generateOnlyFragmentCode ? "varying," : void 0 }), this._lightId === 0 && (e._registerTempVariable("viewDirectionW") && (e.compilationString += `vec3 viewDirectionW = normalize(${this.cameraPosition.associatedVariableName} - ${n}); `), e.compilationString += `lightingInfo info; `, e.compilationString += `float shadow = 1.; `, e.compilationString += `float aggShadow = 0.; `, e.compilationString += `float numLights = 0.; `, e.compilationString += `float glossiness = ${this.glossiness.isConnected ? this.glossiness.associatedVariableName : "1.0"} * ${this.glossPower.isConnected ? this.glossPower.associatedVariableName : "1024.0"}; `, e.compilationString += `vec3 diffuseBase = vec3(0., 0., 0.); `, e.compilationString += `vec3 specularBase = vec3(0., 0., 0.); `, e.compilationString += `vec3 normalW = ${this.worldNormal.associatedVariableName}.xyz; `), this.light ? e.compilationString += e._emitCodeFromInclude("lightFragment", t, { replaceStrings: [ { search: /{X}/g, replace: this._lightId.toString() }, { search: /vPositionW/g, replace: n + ".xyz" } ] }) : e.compilationString += e._emitCodeFromInclude("lightFragment", t, { repeatKey: "maxSimultaneousLights", substitutionVars: `vPositionW,${n}.xyz` }), this._lightId === 0 && (e.compilationString += `aggShadow = aggShadow / numLights; `); const i = this.diffuseOutput, s = this.specularOutput; return e.compilationString += this._declareOutput(i, e) + ` = diffuseBase${this.diffuseColor.isConnected ? " * " + this.diffuseColor.associatedVariableName : ""}; `, s.hasEndpoints && (e.compilationString += this._declareOutput(s, e) + ` = specularBase${this.specularColor.isConnected ? " * " + this.specularColor.associatedVariableName : ""}; `), this.shadow.hasEndpoints && (e.compilationString += this._declareOutput(this.shadow, e) + ` = aggShadow; `), this; } serialize() { const e = super.serialize(); return e.generateOnlyFragmentCode = this.generateOnlyFragmentCode, this.light && (e.lightId = this.light.id), e; } _deserialize(e, t, r) { super._deserialize(e, t, r), e.lightId && (this.light = t.getLightById(e.lightId)), this.generateOnlyFragmentCode = e.generateOnlyFragmentCode, this._setTarget(); } } C([ rn("Generate only fragment code", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0, update: !0, onValidation: AO._OnGenerateOnlyFragmentCodeChanged } }) ], AO.prototype, "generateOnlyFragmentCode", void 0); Ue("BABYLON.LightBlock", AO); class E0 extends Mr { /** * Gets or sets the texture associated with the node */ get texture() { return this._texture; } set texture(e) { var t; if (this._texture === e) return; const r = (t = e == null ? void 0 : e.getScene()) !== null && t !== void 0 ? t : gr.LastCreatedScene; !e && r && r.markAllMaterialsAsDirty(1, (n) => n.hasTexture(this._texture)), this._texture = e, e && r && r.markAllMaterialsAsDirty(1, (n) => n.hasTexture(e)); } /** * Gets the sampler name associated with this image source */ get samplerName() { return this._samplerName; } /** * Creates a new ImageSourceBlock * @param name defines the block name */ constructor(e) { super(e, Ve.VertexAndFragment), this.registerOutput("source", de.Object, Ve.VertexAndFragment, new Ko("source", this, ao.Output, E0, "ImageSourceBlock")); } bind(e) { this.texture && e.setTexture(this._samplerName, this.texture); } isReady() { return !(this.texture && !this.texture.isReadyOrNotBlocking()); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ImageSourceBlock"; } /** * Gets the output component */ get source() { return this._outputs[0]; } _buildBlock(e) { return super._buildBlock(e), e.target === Ve.Vertex && (this._samplerName = e._getFreeVariableName(this.name + "Sampler"), e.sharedData.blockingBlocks.push(this), e.sharedData.textureBlocks.push(this), e.sharedData.bindableBlocks.push(this)), e._emit2DSampler(this._samplerName), this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return this.texture && (e += `${this._codeVariableName}.texture = new BABYLON.Texture("${this.texture.name}", null, ${this.texture.noMipmap}, ${this.texture.invertY}, ${this.texture.samplingMode}); `, e += `${this._codeVariableName}.texture.wrapU = ${this.texture.wrapU}; `, e += `${this._codeVariableName}.texture.wrapV = ${this.texture.wrapV}; `, e += `${this._codeVariableName}.texture.uAng = ${this.texture.uAng}; `, e += `${this._codeVariableName}.texture.vAng = ${this.texture.vAng}; `, e += `${this._codeVariableName}.texture.wAng = ${this.texture.wAng}; `, e += `${this._codeVariableName}.texture.uOffset = ${this.texture.uOffset}; `, e += `${this._codeVariableName}.texture.vOffset = ${this.texture.vOffset}; `, e += `${this._codeVariableName}.texture.uScale = ${this.texture.uScale}; `, e += `${this._codeVariableName}.texture.vScale = ${this.texture.vScale}; `, e += `${this._codeVariableName}.texture.coordinatesMode = ${this.texture.coordinatesMode}; `), e; } serialize() { const e = super.serialize(); return this.texture && !this.texture.isRenderTarget && this.texture.getClassName() !== "VideoTexture" && (e.texture = this.texture.serialize()), e; } _deserialize(e, t, r) { super._deserialize(e, t, r), e.texture && !ja.IgnoreTexturesAtLoadTime && e.texture.url !== void 0 && (r = e.texture.url.indexOf("data:") === 0 ? "" : r, this.texture = We.Parse(e.texture, t, r)); } } Ue("BABYLON.ImageSourceBlock", E0); class _I extends Mr { /** * Gets or sets the texture associated with the node */ get texture() { var e; return this.source.isConnected ? ((e = this.source.connectedPoint) === null || e === void 0 ? void 0 : e.ownerBlock).texture : this._texture; } set texture(e) { var t; if (this._texture === e) return; const r = (t = e == null ? void 0 : e.getScene()) !== null && t !== void 0 ? t : gr.LastCreatedScene; !e && r && r.markAllMaterialsAsDirty(1, (n) => n.hasTexture(this._texture)), this._texture = e, e && r && r.markAllMaterialsAsDirty(1, (n) => n.hasTexture(e)); } static _IsPrePassTextureBlock(e) { return (e == null ? void 0 : e.getClassName()) === "PrePassTextureBlock"; } get _isSourcePrePass() { return _I._IsPrePassTextureBlock(this._imageSource); } /** * Gets the sampler name associated with this texture */ get samplerName() { if (this._imageSource) { if (!_I._IsPrePassTextureBlock(this._imageSource)) return this._imageSource.samplerName; if (this.source.connectedPoint) return this._imageSource.getSamplerName(this.source.connectedPoint); } return this._samplerName; } /** * Gets a boolean indicating that this block is linked to an ImageSourceBlock */ get hasImageSource() { return this.source.isConnected; } /** * Gets or sets a boolean indicating if content needs to be converted to gamma space */ set convertToGammaSpace(e) { var t; if (e !== this._convertToGammaSpace && (this._convertToGammaSpace = e, this.texture)) { const r = (t = this.texture.getScene()) !== null && t !== void 0 ? t : gr.LastCreatedScene; r == null || r.markAllMaterialsAsDirty(1, (n) => n.hasTexture(this.texture)); } } get convertToGammaSpace() { return this._convertToGammaSpace; } /** * Gets or sets a boolean indicating if content needs to be converted to linear space */ set convertToLinearSpace(e) { var t; if (e !== this._convertToLinearSpace && (this._convertToLinearSpace = e, this.texture)) { const r = (t = this.texture.getScene()) !== null && t !== void 0 ? t : gr.LastCreatedScene; r == null || r.markAllMaterialsAsDirty(1, (n) => n.hasTexture(this.texture)); } } get convertToLinearSpace() { return this._convertToLinearSpace; } /** * Create a new TextureBlock * @param name defines the block name * @param fragmentOnly */ constructor(e, t = !1) { super(e, t ? Ve.Fragment : Ve.VertexAndFragment), this._convertToGammaSpace = !1, this._convertToLinearSpace = !1, this.disableLevelMultiplication = !1, this._fragmentOnly = t, this.registerInput("uv", de.AutoDetect, !1, Ve.VertexAndFragment), this.registerInput("source", de.Object, !0, Ve.VertexAndFragment, new Ko("source", this, ao.Input, E0, "ImageSourceBlock")), this.registerInput("layer", de.Float, !0), this.registerInput("lod", de.Float, !0), this.registerOutput("rgba", de.Color4, Ve.Neutral), this.registerOutput("rgb", de.Color3, Ve.Neutral), this.registerOutput("r", de.Float, Ve.Neutral), this.registerOutput("g", de.Float, Ve.Neutral), this.registerOutput("b", de.Float, Ve.Neutral), this.registerOutput("a", de.Float, Ve.Neutral), this.registerOutput("level", de.Float, Ve.Neutral), this._inputs[0].addExcludedConnectionPointFromAllowedTypes(de.Vector2 | de.Vector3 | de.Vector4), this._inputs[0]._prioritizeVertex = !t; } /** * Gets the current class name * @returns the class name */ getClassName() { return "TextureBlock"; } /** * Gets the uv input component */ get uv() { return this._inputs[0]; } /** * Gets the source input component */ get source() { return this._inputs[1]; } /** * Gets the layer input component */ get layer() { return this._inputs[2]; } /** * Gets the LOD input component */ get lod() { return this._inputs[3]; } /** * Gets the rgba output component */ get rgba() { return this._outputs[0]; } /** * Gets the rgb output component */ get rgb() { return this._outputs[1]; } /** * Gets the r output component */ get r() { return this._outputs[2]; } /** * Gets the g output component */ get g() { return this._outputs[3]; } /** * Gets the b output component */ get b() { return this._outputs[4]; } /** * Gets the a output component */ get a() { return this._outputs[5]; } /** * Gets the level output component */ get level() { return this._outputs[6]; } get target() { if (this._fragmentOnly) return Ve.Fragment; if (!this.uv.isConnected || this.uv.sourceBlock.isInput) return Ve.VertexAndFragment; let e = this.uv.connectedPoint; for (; e; ) { if (e.target === Ve.Fragment) return Ve.Fragment; if (e.target === Ve.Vertex) return Ve.VertexAndFragment; if (e.target === Ve.Neutral || e.target === Ve.VertexAndFragment) { const t = e.ownerBlock; if (t.target === Ve.Fragment) return Ve.Fragment; e = null; for (const r of t.inputs) if (r.connectedPoint) { e = r.connectedPoint; break; } } } return Ve.VertexAndFragment; } set target(e) { } autoConfigure(e, t = () => !0) { if (!this.uv.isConnected) if (e.mode === Vv.PostProcess) { const r = e.getBlockByPredicate((n) => n.name === "uv" && t(n)); r && r.connectTo(this); } else { const r = e.mode === Vv.Particle ? "particle_uv" : "uv"; let n = e.getInputBlockByPredicate((i) => i.isAttribute && i.name === r && t(i)); n || (n = new gi("uv"), n.setAsAttribute(r)), n.output.connectTo(this.uv); } } initializeDefines(e, t, r) { r._areTexturesDirty && this._mainUVDefineName !== void 0 && r.setValue(this._mainUVDefineName, !1, !0); } prepareDefines(e, t, r) { if (!r._areTexturesDirty) return; if (!this.texture || !this.texture.getTextureMatrix) { this._isMixed && (r.setValue(this._defineName, !1, !0), r.setValue(this._mainUVDefineName, !0, !0)); return; } const n = this.convertToGammaSpace && this.texture && !this.texture.gammaSpace, i = this.convertToLinearSpace && this.texture && this.texture.gammaSpace; r.setValue(this._linearDefineName, n, !0), r.setValue(this._gammaDefineName, i, !0), this._isMixed && (this.texture.getTextureMatrix().isIdentityAs3x2() ? (r.setValue(this._defineName, !1, !0), r.setValue(this._mainUVDefineName, !0, !0)) : (r.setValue(this._defineName, !0), r[this._mainUVDefineName] == null && r.setValue(this._mainUVDefineName, !1, !0))); } isReady() { return this._isSourcePrePass ? !0 : !(this.texture && !this.texture.isReadyOrNotBlocking()); } bind(e) { this._isSourcePrePass && e.setFloat(this._textureInfoName, 1), this.texture && (this._isMixed && (e.setFloat(this._textureInfoName, this.texture.level), e.setMatrix(this._textureTransformName, this.texture.getTextureMatrix())), this._imageSource || e.setTexture(this._samplerName, this.texture)); } get _isMixed() { return this.target !== Ve.Fragment; } _injectVertexCode(e) { const t = this.uv; if (this._defineName = e._getFreeDefineName("UVTRANSFORM"), this._mainUVDefineName = "VMAIN" + t.associatedVariableName.toUpperCase(), this._mainUVName = "vMain" + t.associatedVariableName, this._transformedUVName = e._getFreeVariableName("transformedUV"), this._textureTransformName = e._getFreeVariableName("textureTransform"), this._textureInfoName = e._getFreeVariableName("textureInfoName"), this.level.associatedVariableName = this._textureInfoName, e._emitVaryingFromString(this._transformedUVName, "vec2", this._defineName), e._emitVaryingFromString(this._mainUVName, "vec2", this._mainUVDefineName), e._emitUniformFromString(this._textureTransformName, "mat4", this._defineName), e.compilationString += `#ifdef ${this._defineName} `, e.compilationString += `${this._transformedUVName} = vec2(${this._textureTransformName} * vec4(${t.associatedVariableName}.xy, 1.0, 0.0)); `, e.compilationString += `#elif defined(${this._mainUVDefineName}) `, e.compilationString += `${this._mainUVName} = ${t.associatedVariableName}.xy; `, e.compilationString += `#endif `, !!this._outputs.some((r) => r.isConnectedInVertexShader)) { this._writeTextureRead(e, !0); for (const r of this._outputs) r.hasEndpoints && r.name !== "level" && this._writeOutput(e, r, r.name, !0); } } _getUVW(e) { var t, r, n; let i = e; if ((n = (r = (t = this._texture) === null || t === void 0 ? void 0 : t._texture) === null || r === void 0 ? void 0 : r.is2DArray) !== null && n !== void 0 ? n : !1) { const a = this.layer.isConnected ? this.layer.associatedVariableName : "0"; i = `vec3(${e}, ${a})`; } return i; } get _samplerFunc() { return this.lod.isConnected ? "texture2DLodEXT" : "texture2D"; } get _samplerLodSuffix() { return this.lod.isConnected ? `, ${this.lod.associatedVariableName}` : ""; } _generateTextureLookup(e) { const t = this.samplerName; e.compilationString += `#ifdef ${this._defineName} `, e.compilationString += `vec4 ${this._tempTextureRead} = ${this._samplerFunc}(${t}, ${this._getUVW(this._transformedUVName)}${this._samplerLodSuffix}); `, e.compilationString += `#elif defined(${this._mainUVDefineName}) `, e.compilationString += `vec4 ${this._tempTextureRead} = ${this._samplerFunc}(${t}, ${this._getUVW(this._mainUVName ? this._mainUVName : this.uv.associatedVariableName)}${this._samplerLodSuffix}); `, e.compilationString += `#endif `; } _writeTextureRead(e, t = !1) { const r = this.uv; if (t) { if (e.target === Ve.Fragment) return; this._generateTextureLookup(e); return; } if (this.uv.ownerBlock.target === Ve.Fragment) { e.compilationString += `vec4 ${this._tempTextureRead} = ${this._samplerFunc}(${this.samplerName}, ${this._getUVW(r.associatedVariableName)}${this._samplerLodSuffix}); `; return; } this._generateTextureLookup(e); } _generateConversionCode(e, t, r) { r !== "a" && ((!this.texture || !this.texture.gammaSpace) && (e.compilationString += `#ifdef ${this._linearDefineName} ${t.associatedVariableName} = toGammaSpace(${t.associatedVariableName}); #endif `), e.compilationString += `#ifdef ${this._gammaDefineName} ${t.associatedVariableName} = toLinearSpace(${t.associatedVariableName}); #endif `); } _writeOutput(e, t, r, n = !1) { if (n) { if (e.target === Ve.Fragment) return; e.compilationString += `${this._declareOutput(t, e)} = ${this._tempTextureRead}.${r}; `, this._generateConversionCode(e, t, r); return; } if (this.uv.ownerBlock.target === Ve.Fragment) { e.compilationString += `${this._declareOutput(t, e)} = ${this._tempTextureRead}.${r}; `, this._generateConversionCode(e, t, r); return; } let i = ""; this.disableLevelMultiplication || (i = ` * ${this._textureInfoName}`), e.compilationString += `${this._declareOutput(t, e)} = ${this._tempTextureRead}.${r}${i}; `, this._generateConversionCode(e, t, r); } _buildBlock(e) { var t, r, n, i; if (super._buildBlock(e), this.source.isConnected ? this._imageSource = this.source.connectedPoint.ownerBlock : this._imageSource = null, (e.target === Ve.Vertex || this._fragmentOnly || e.target === Ve.Fragment) && (this._tempTextureRead = e._getFreeVariableName("tempTextureRead"), this._linearDefineName = e._getFreeDefineName("ISLINEAR"), this._gammaDefineName = e._getFreeDefineName("ISGAMMA")), (!this._isMixed && e.target === Ve.Fragment || this._isMixed && e.target === Ve.Vertex) && (this._imageSource || (this._samplerName = e._getFreeVariableName(this.name + "Sampler"), !((r = (t = this._texture) === null || t === void 0 ? void 0 : t._texture) === null || r === void 0) && r.is2DArray ? e._emit2DArraySampler(this._samplerName) : e._emit2DSampler(this._samplerName)), e.sharedData.blockingBlocks.push(this), e.sharedData.textureBlocks.push(this), e.sharedData.blocksWithDefines.push(this), e.sharedData.bindableBlocks.push(this)), e.target !== Ve.Fragment) { this._injectVertexCode(e); return; } if (!this._outputs.some((a) => a.isConnectedInFragmentShader)) return; this._isMixed && !this._imageSource && (!((i = (n = this._texture) === null || n === void 0 ? void 0 : n._texture) === null || i === void 0) && i.is2DArray ? e._emit2DArraySampler(this._samplerName) : e._emit2DSampler(this._samplerName)); const s = `//${this.name}`; e._emitFunctionFromInclude("helperFunctions", s), this._isMixed && e._emitUniformFromString(this._textureInfoName, "float"), this._writeTextureRead(e); for (const a of this._outputs) a.hasEndpoints && a.name !== "level" && this._writeOutput(e, a, a.name); return this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return e += `${this._codeVariableName}.convertToGammaSpace = ${this.convertToGammaSpace}; `, e += `${this._codeVariableName}.convertToLinearSpace = ${this.convertToLinearSpace}; `, e += `${this._codeVariableName}.disableLevelMultiplication = ${this.disableLevelMultiplication}; `, this.texture && (e += `${this._codeVariableName}.texture = new BABYLON.Texture("${this.texture.name}", null, ${this.texture.noMipmap}, ${this.texture.invertY}, ${this.texture.samplingMode}); `, e += `${this._codeVariableName}.texture.wrapU = ${this.texture.wrapU}; `, e += `${this._codeVariableName}.texture.wrapV = ${this.texture.wrapV}; `, e += `${this._codeVariableName}.texture.uAng = ${this.texture.uAng}; `, e += `${this._codeVariableName}.texture.vAng = ${this.texture.vAng}; `, e += `${this._codeVariableName}.texture.wAng = ${this.texture.wAng}; `, e += `${this._codeVariableName}.texture.uOffset = ${this.texture.uOffset}; `, e += `${this._codeVariableName}.texture.vOffset = ${this.texture.vOffset}; `, e += `${this._codeVariableName}.texture.uScale = ${this.texture.uScale}; `, e += `${this._codeVariableName}.texture.vScale = ${this.texture.vScale}; `, e += `${this._codeVariableName}.texture.coordinatesMode = ${this.texture.coordinatesMode}; `), e; } serialize() { const e = super.serialize(); return e.convertToGammaSpace = this.convertToGammaSpace, e.convertToLinearSpace = this.convertToLinearSpace, e.fragmentOnly = this._fragmentOnly, e.disableLevelMultiplication = this.disableLevelMultiplication, !this.hasImageSource && this.texture && !this.texture.isRenderTarget && this.texture.getClassName() !== "VideoTexture" && (e.texture = this.texture.serialize()), e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.convertToGammaSpace = e.convertToGammaSpace, this.convertToLinearSpace = !!e.convertToLinearSpace, this._fragmentOnly = !!e.fragmentOnly, this.disableLevelMultiplication = !!e.disableLevelMultiplication, e.texture && !ja.IgnoreTexturesAtLoadTime && e.texture.url !== void 0 && (r = e.texture.url.indexOf("data:") === 0 ? "" : r, this.texture = We.Parse(e.texture, t, r)); } } Ue("BABYLON.TextureBlock", _I); class $I extends Mr { /** * Gets or sets the texture associated with the node */ get texture() { return this._texture; } set texture(e) { var t; if (this._texture === e) return; const r = (t = e == null ? void 0 : e.getScene()) !== null && t !== void 0 ? t : gr.LastCreatedScene; !e && r && r.markAllMaterialsAsDirty(1, (n) => n.hasTexture(this._texture)), this._texture = e, e && r && r.markAllMaterialsAsDirty(1, (n) => n.hasTexture(e)); } static _OnGenerateOnlyFragmentCodeChanged(e, t) { return e._onGenerateOnlyFragmentCodeChanged(); } _onGenerateOnlyFragmentCodeChanged() { return this._setTarget(), !0; } _setTarget() { this._setInitialTarget(this.generateOnlyFragmentCode ? Ve.Fragment : Ve.VertexAndFragment); } /** * Create a new ReflectionTextureBaseBlock * @param name defines the block name */ constructor(e) { super(e, Ve.VertexAndFragment), this.generateOnlyFragmentCode = !1; } /** * Gets the current class name * @returns the class name */ getClassName() { return "ReflectionTextureBaseBlock"; } _getTexture() { return this.texture; } autoConfigure(e, t = () => !0) { if (!this.position.isConnected) { let r = e.getInputBlockByPredicate((n) => n.isAttribute && n.name === "position" && t(n)); r || (r = new gi("position"), r.setAsAttribute()), r.output.connectTo(this.position); } if (!this.world.isConnected) { let r = e.getInputBlockByPredicate((n) => n.systemValue === Bi.World && t(n)); r || (r = new gi("world"), r.setAsSystemValue(Bi.World)), r.output.connectTo(this.world); } if (this.view && !this.view.isConnected) { let r = e.getInputBlockByPredicate((n) => n.systemValue === Bi.View && t(n)); r || (r = new gi("view"), r.setAsSystemValue(Bi.View)), r.output.connectTo(this.view); } } prepareDefines(e, t, r) { if (!r._areTexturesDirty) return; const n = this._getTexture(); !n || !n.getTextureMatrix || (r.setValue(this._define3DName, n.isCube, !0), r.setValue(this._defineLocalCubicName, !!n.boundingBoxSize, !0), r.setValue(this._defineExplicitName, n.coordinatesMode === 0, !0), r.setValue(this._defineSkyboxName, n.coordinatesMode === 5, !0), r.setValue(this._defineCubicName, n.coordinatesMode === 3 || n.coordinatesMode === 6, !0), r.setValue("INVERTCUBICMAP", n.coordinatesMode === 6, !0), r.setValue(this._defineSphericalName, n.coordinatesMode === 1, !0), r.setValue(this._definePlanarName, n.coordinatesMode === 2, !0), r.setValue(this._defineProjectionName, n.coordinatesMode === 4, !0), r.setValue(this._defineEquirectangularName, n.coordinatesMode === 7, !0), r.setValue(this._defineEquirectangularFixedName, n.coordinatesMode === 8, !0), r.setValue(this._defineMirroredEquirectangularFixedName, n.coordinatesMode === 9, !0)); } isReady() { const e = this._getTexture(); return !(e && !e.isReadyOrNotBlocking()); } bind(e, t, r) { const n = this._getTexture(); if (!(!r || !n) && (e.setMatrix(this._reflectionMatrixName, n.getReflectionTextureMatrix()), n.isCube ? e.setTexture(this._cubeSamplerName, n) : e.setTexture(this._2DSamplerName, n), n.boundingBoxSize)) { const i = n; e.setVector3(this._reflectionPositionName, i.boundingBoxPosition), e.setVector3(this._reflectionSizeName, i.boundingBoxSize); } } /** * Gets the code to inject in the vertex shader * @param state current state of the node material building * @returns the shader code */ handleVertexSide(e) { if (this.generateOnlyFragmentCode && e.target === Ve.Vertex) return ""; this._define3DName = e._getFreeDefineName("REFLECTIONMAP_3D"), this._defineCubicName = e._getFreeDefineName("REFLECTIONMAP_CUBIC"), this._defineSphericalName = e._getFreeDefineName("REFLECTIONMAP_SPHERICAL"), this._definePlanarName = e._getFreeDefineName("REFLECTIONMAP_PLANAR"), this._defineProjectionName = e._getFreeDefineName("REFLECTIONMAP_PROJECTION"), this._defineExplicitName = e._getFreeDefineName("REFLECTIONMAP_EXPLICIT"), this._defineEquirectangularName = e._getFreeDefineName("REFLECTIONMAP_EQUIRECTANGULAR"), this._defineLocalCubicName = e._getFreeDefineName("USE_LOCAL_REFLECTIONMAP_CUBIC"), this._defineMirroredEquirectangularFixedName = e._getFreeDefineName("REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED"), this._defineEquirectangularFixedName = e._getFreeDefineName("REFLECTIONMAP_EQUIRECTANGULAR_FIXED"), this._defineSkyboxName = e._getFreeDefineName("REFLECTIONMAP_SKYBOX"), this._defineOppositeZ = e._getFreeDefineName("REFLECTIONMAP_OPPOSITEZ"), this._reflectionMatrixName = e._getFreeVariableName("reflectionMatrix"), e._emitUniformFromString(this._reflectionMatrixName, "mat4"); let t = ""; this._worldPositionNameInFragmentOnlyMode = e._getFreeVariableName("worldPosition"); const r = this.generateOnlyFragmentCode ? this._worldPositionNameInFragmentOnlyMode : "v_" + this.worldPosition.associatedVariableName; return (this.generateOnlyFragmentCode || e._emitVaryingFromString(r, "vec4")) && (t += `${this.generateOnlyFragmentCode ? "vec4 " : ""}${r} = ${this.worldPosition.associatedVariableName}; `), this._positionUVWName = e._getFreeVariableName("positionUVW"), this._directionWName = e._getFreeVariableName("directionW"), (this.generateOnlyFragmentCode || e._emitVaryingFromString(this._positionUVWName, "vec3", this._defineSkyboxName)) && (t += `#ifdef ${this._defineSkyboxName} `, t += `${this.generateOnlyFragmentCode ? "vec3 " : ""}${this._positionUVWName} = ${this.position.associatedVariableName}.xyz; `, t += `#endif `), (this.generateOnlyFragmentCode || e._emitVaryingFromString(this._directionWName, "vec3", `defined(${this._defineEquirectangularFixedName}) || defined(${this._defineMirroredEquirectangularFixedName})`)) && (t += `#if defined(${this._defineEquirectangularFixedName}) || defined(${this._defineMirroredEquirectangularFixedName}) `, t += `${this.generateOnlyFragmentCode ? "vec3 " : ""}${this._directionWName} = normalize(vec3(${this.world.associatedVariableName} * vec4(${this.position.associatedVariableName}.xyz, 0.0))); `, t += `#endif `), t; } /** * Handles the inits for the fragment code path * @param state node material build state */ handleFragmentSideInits(e) { e.sharedData.blockingBlocks.push(this), e.sharedData.textureBlocks.push(this), this._cubeSamplerName = e._getFreeVariableName(this.name + "CubeSampler"), e.samplers.push(this._cubeSamplerName), this._2DSamplerName = e._getFreeVariableName(this.name + "2DSampler"), e.samplers.push(this._2DSamplerName), e._samplerDeclaration += `#ifdef ${this._define3DName} `, e._samplerDeclaration += `uniform samplerCube ${this._cubeSamplerName}; `, e._samplerDeclaration += `#else `, e._samplerDeclaration += `uniform sampler2D ${this._2DSamplerName}; `, e._samplerDeclaration += `#endif `, e.sharedData.blocksWithDefines.push(this), e.sharedData.bindableBlocks.push(this); const t = `//${this.name}`; e._emitFunctionFromInclude("helperFunctions", t), e._emitFunctionFromInclude("reflectionFunction", t, { replaceStrings: [{ search: /vec3 computeReflectionCoords/g, replace: "void DUMMYFUNC" }] }), this._reflectionColorName = e._getFreeVariableName("reflectionColor"), this._reflectionVectorName = e._getFreeVariableName("reflectionUVW"), this._reflectionCoordsName = e._getFreeVariableName("reflectionCoords"), this._reflectionPositionName = e._getFreeVariableName("vReflectionPosition"), e._emitUniformFromString(this._reflectionPositionName, "vec3"), this._reflectionSizeName = e._getFreeVariableName("vReflectionPosition"), e._emitUniformFromString(this._reflectionSizeName, "vec3"); } /** * Generates the reflection coords code for the fragment code path * @param worldNormalVarName name of the world normal variable * @param worldPos name of the world position variable. If not provided, will use the world position connected to this block * @param onlyReflectionVector if true, generates code only for the reflection vector computation, not for the reflection coordinates * @param doNotEmitInvertZ if true, does not emit the invertZ code * @returns the shader code */ handleFragmentSideCodeReflectionCoords(e, t, r = !1, n = !1) { t || (t = this.generateOnlyFragmentCode ? this._worldPositionNameInFragmentOnlyMode : `v_${this.worldPosition.associatedVariableName}`); const i = this._reflectionMatrixName, s = `normalize(${this._directionWName})`, a = `${this._positionUVWName}`, f = `${this.cameraPosition.associatedVariableName}`, o = `${this.view.associatedVariableName}`; e += ".xyz"; let d = ` #ifdef ${this._defineMirroredEquirectangularFixedName} vec3 ${this._reflectionVectorName} = computeMirroredFixedEquirectangularCoords(${t}, ${e}, ${s}); #endif #ifdef ${this._defineEquirectangularFixedName} vec3 ${this._reflectionVectorName} = computeFixedEquirectangularCoords(${t}, ${e}, ${s}); #endif #ifdef ${this._defineEquirectangularName} vec3 ${this._reflectionVectorName} = computeEquirectangularCoords(${t}, ${e}, ${f}.xyz, ${i}); #endif #ifdef ${this._defineSphericalName} vec3 ${this._reflectionVectorName} = computeSphericalCoords(${t}, ${e}, ${o}, ${i}); #endif #ifdef ${this._definePlanarName} vec3 ${this._reflectionVectorName} = computePlanarCoords(${t}, ${e}, ${f}.xyz, ${i}); #endif #ifdef ${this._defineCubicName} #ifdef ${this._defineLocalCubicName} vec3 ${this._reflectionVectorName} = computeCubicLocalCoords(${t}, ${e}, ${f}.xyz, ${i}, ${this._reflectionSizeName}, ${this._reflectionPositionName}); #else vec3 ${this._reflectionVectorName} = computeCubicCoords(${t}, ${e}, ${f}.xyz, ${i}); #endif #endif #ifdef ${this._defineProjectionName} vec3 ${this._reflectionVectorName} = computeProjectionCoords(${t}, ${o}, ${i}); #endif #ifdef ${this._defineSkyboxName} vec3 ${this._reflectionVectorName} = computeSkyBoxCoords(${a}, ${i}); #endif #ifdef ${this._defineExplicitName} vec3 ${this._reflectionVectorName} = vec3(0, 0, 0); #endif `; return n || (d += `#ifdef ${this._defineOppositeZ} ${this._reflectionVectorName}.z *= -1.0; #endif `), r || (d += ` #ifdef ${this._define3DName} vec3 ${this._reflectionCoordsName} = ${this._reflectionVectorName}; #else vec2 ${this._reflectionCoordsName} = ${this._reflectionVectorName}.xy; #ifdef ${this._defineProjectionName} ${this._reflectionCoordsName} /= ${this._reflectionVectorName}.z; #endif ${this._reflectionCoordsName}.y = 1.0 - ${this._reflectionCoordsName}.y; #endif `), d; } /** * Generates the reflection color code for the fragment code path * @param lodVarName name of the lod variable * @param swizzleLookupTexture swizzle to use for the final color variable * @returns the shader code */ handleFragmentSideCodeReflectionColor(e, t = ".rgb") { let n = `${"vec" + (t.length === 0 ? "4" : t.length - 1)} ${this._reflectionColorName}; #ifdef ${this._define3DName} `; return e ? n += `${this._reflectionColorName} = textureCubeLodEXT(${this._cubeSamplerName}, ${this._reflectionVectorName}, ${e})${t}; ` : n += `${this._reflectionColorName} = textureCube(${this._cubeSamplerName}, ${this._reflectionVectorName})${t}; `, n += ` #else `, e ? n += `${this._reflectionColorName} = texture2DLodEXT(${this._2DSamplerName}, ${this._reflectionCoordsName}, ${e})${t}; ` : n += `${this._reflectionColorName} = texture2D(${this._2DSamplerName}, ${this._reflectionCoordsName})${t}; `, n += `#endif `, n; } /** * Generates the code corresponding to the connected output points * @param state node material build state * @param varName name of the variable to output * @returns the shader code */ writeOutputs(e, t) { let r = ""; if (e.target === Ve.Fragment) for (const n of this._outputs) n.hasEndpoints && (r += `${this._declareOutput(n, e)} = ${t}.${n.name}; `); return r; } _buildBlock(e) { return super._buildBlock(e), this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); if (!this.texture) return e; if (this.texture.isCube) { const t = this.texture.forcedExtension; e += `${this._codeVariableName}.texture = new BABYLON.CubeTexture("${this.texture.name}", undefined, undefined, ${this.texture.noMipmap}, null, undefined, undefined, undefined, ${this.texture._prefiltered}, ${t ? '"' + t + '"' : "null"}); `; } else e += `${this._codeVariableName}.texture = new BABYLON.Texture("${this.texture.name}", null); `; return e += `${this._codeVariableName}.texture.coordinatesMode = ${this.texture.coordinatesMode}; `, e; } serialize() { const e = super.serialize(); return this.texture && !this.texture.isRenderTarget && (e.texture = this.texture.serialize()), e.generateOnlyFragmentCode = this.generateOnlyFragmentCode, e; } _deserialize(e, t, r) { super._deserialize(e, t, r), e.texture && !ja.IgnoreTexturesAtLoadTime && (r = e.texture.url.indexOf("data:") === 0 ? "" : r, e.texture.isCube ? this.texture = v1.Parse(e.texture, t, r) : this.texture = We.Parse(e.texture, t, r)), this.generateOnlyFragmentCode = e.generateOnlyFragmentCode, this._setTarget(); } } C([ rn("Generate only fragment code", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0, update: !0, onValidation: $I._OnGenerateOnlyFragmentCodeChanged } }) ], $I.prototype, "generateOnlyFragmentCode", void 0); Ue("BABYLON.ReflectionTextureBaseBlock", $I); class hre extends $I { _onGenerateOnlyFragmentCodeChanged() { return this.position.isConnected ? (this.generateOnlyFragmentCode = !this.generateOnlyFragmentCode, console.error("The position input must not be connected to be able to switch!"), !1) : this.worldPosition.isConnected ? (this.generateOnlyFragmentCode = !this.generateOnlyFragmentCode, console.error("The worldPosition input must not be connected to be able to switch!"), !1) : (this._setTarget(), !0); } _setTarget() { super._setTarget(), this.getInputByName("position").target = this.generateOnlyFragmentCode ? Ve.Fragment : Ve.Vertex, this.getInputByName("worldPosition").target = this.generateOnlyFragmentCode ? Ve.Fragment : Ve.Vertex; } /** * Create a new ReflectionTextureBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("position", de.AutoDetect, !1, Ve.Vertex), this.registerInput("worldPosition", de.Vector4, !1, Ve.Vertex), this.registerInput("worldNormal", de.Vector4, !1, Ve.Fragment), this.registerInput("world", de.Matrix, !1, Ve.Vertex), this.registerInput("cameraPosition", de.Vector3, !1, Ve.Fragment), this.registerInput("view", de.Matrix, !1, Ve.Fragment), this.registerOutput("rgb", de.Color3, Ve.Fragment), this.registerOutput("rgba", de.Color4, Ve.Fragment), this.registerOutput("r", de.Float, Ve.Fragment), this.registerOutput("g", de.Float, Ve.Fragment), this.registerOutput("b", de.Float, Ve.Fragment), this.registerOutput("a", de.Float, Ve.Fragment), this._inputs[0].addExcludedConnectionPointFromAllowedTypes(de.Color3 | de.Vector3 | de.Vector4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ReflectionTextureBlock"; } /** * Gets the world position input component */ get position() { return this._inputs[0]; } /** * Gets the world position input component */ get worldPosition() { return this._inputs[1]; } /** * Gets the world normal input component */ get worldNormal() { return this._inputs[2]; } /** * Gets the world input component */ get world() { return this._inputs[3]; } /** * Gets the camera (or eye) position component */ get cameraPosition() { return this._inputs[4]; } /** * Gets the view input component */ get view() { return this._inputs[5]; } /** * Gets the rgb output component */ get rgb() { return this._outputs[0]; } /** * Gets the rgba output component */ get rgba() { return this._outputs[1]; } /** * Gets the r output component */ get r() { return this._outputs[2]; } /** * Gets the g output component */ get g() { return this._outputs[3]; } /** * Gets the b output component */ get b() { return this._outputs[4]; } /** * Gets the a output component */ get a() { return this._outputs[5]; } autoConfigure(e, t = () => !0) { if (super.autoConfigure(e), !this.cameraPosition.isConnected) { let r = e.getInputBlockByPredicate((n) => n.systemValue === Bi.CameraPosition && t(n)); r || (r = new gi("cameraPosition"), r.setAsSystemValue(Bi.CameraPosition)), r.output.connectTo(this.cameraPosition); } } _buildBlock(e) { if (super._buildBlock(e), !this.texture) return e.compilationString += this.writeOutputs(e, "vec4(0.)"), this; if (e.target !== Ve.Fragment) return e.compilationString += this.handleVertexSide(e), this; this.generateOnlyFragmentCode && (e.compilationString += this.handleVertexSide(e)), this.handleFragmentSideInits(e); const t = e._getFreeVariableName("normalWUnit"); return e.compilationString += `vec4 ${t} = normalize(${this.worldNormal.associatedVariableName}); `, e.compilationString += this.handleFragmentSideCodeReflectionCoords(t), e.compilationString += this.handleFragmentSideCodeReflectionColor(void 0, ""), e.compilationString += this.writeOutputs(e, this._reflectionColorName), this; } } Ue("BABYLON.ReflectionTextureBlock", hre); class oV extends Mr { /** * Create a new SceneDepthBlock * @param name defines the block name */ constructor(e) { super(e, Ve.VertexAndFragment), this.useNonLinearDepth = !1, this.storeCameraSpaceZ = !1, this.force32itsFloat = !1, this._isUnique = !0, this.registerInput("uv", de.AutoDetect, !1, Ve.VertexAndFragment), this.registerOutput("depth", de.Float, Ve.Neutral), this._inputs[0].addExcludedConnectionPointFromAllowedTypes(de.Vector2 | de.Vector3 | de.Vector4), this._inputs[0]._prioritizeVertex = !1; } /** * Gets the current class name * @returns the class name */ getClassName() { return "SceneDepthBlock"; } /** * Gets the uv input component */ get uv() { return this._inputs[0]; } /** * Gets the depth output component */ get depth() { return this._outputs[0]; } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("textureSampler"); } get target() { return !this.uv.isConnected || this.uv.sourceBlock.isInput ? Ve.VertexAndFragment : Ve.Fragment; } _getTexture(e) { return e.enableDepthRenderer(void 0, this.useNonLinearDepth, this.force32itsFloat, void 0, this.storeCameraSpaceZ).getDepthMap(); } bind(e, t) { const r = this._getTexture(t.getScene()); e.setTexture(this._samplerName, r); } _injectVertexCode(e) { const t = this.uv; if (t.connectedPoint.ownerBlock.isInput && (t.connectedPoint.ownerBlock.isAttribute || e._emitUniformFromString(t.associatedVariableName, "vec" + (t.type === de.Vector3 ? "3" : t.type === de.Vector4 ? "4" : "2"))), this._mainUVName = "vMain" + t.associatedVariableName, e._emitVaryingFromString(this._mainUVName, "vec2"), e.compilationString += `${this._mainUVName} = ${t.associatedVariableName}.xy; `, !!this._outputs.some((r) => r.isConnectedInVertexShader)) { this._writeTextureRead(e, !0); for (const r of this._outputs) r.hasEndpoints && this._writeOutput(e, r, "r", !0); } } _writeTextureRead(e, t = !1) { const r = this.uv; if (t) { if (e.target === Ve.Fragment) return; e.compilationString += `vec4 ${this._tempTextureRead} = texture2D(${this._samplerName}, ${r.associatedVariableName}.xy); `; return; } if (this.uv.ownerBlock.target === Ve.Fragment) { e.compilationString += `vec4 ${this._tempTextureRead} = texture2D(${this._samplerName}, ${r.associatedVariableName}.xy); `; return; } e.compilationString += `vec4 ${this._tempTextureRead} = texture2D(${this._samplerName}, ${this._mainUVName}); `; } _writeOutput(e, t, r, n = !1) { if (n) { if (e.target === Ve.Fragment) return; e.compilationString += `${this._declareOutput(t, e)} = ${this._tempTextureRead}.${r}; `; return; } if (this.uv.ownerBlock.target === Ve.Fragment) { e.compilationString += `${this._declareOutput(t, e)} = ${this._tempTextureRead}.${r}; `; return; } e.compilationString += `${this._declareOutput(t, e)} = ${this._tempTextureRead}.${r}; `; } _buildBlock(e) { if (super._buildBlock(e), this._samplerName = e._getFreeVariableName(this.name + "Sampler"), this._tempTextureRead = e._getFreeVariableName("tempTextureRead"), e.sharedData.bindableBlocks.indexOf(this) < 0 && e.sharedData.bindableBlocks.push(this), e.target !== Ve.Fragment) { e._emit2DSampler(this._samplerName), this._injectVertexCode(e); return; } if (this._outputs.some((t) => t.isConnectedInFragmentShader)) { e._emit2DSampler(this._samplerName), this._writeTextureRead(e); for (const t of this._outputs) t.hasEndpoints && this._writeOutput(e, t, "r"); return this; } } serialize() { const e = super.serialize(); return e.useNonLinearDepth = this.useNonLinearDepth, e.storeCameraSpaceZ = this.storeCameraSpaceZ, e.force32itsFloat = this.force32itsFloat, e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.useNonLinearDepth = e.useNonLinearDepth, this.storeCameraSpaceZ = !!e.storeCameraSpaceZ, this.force32itsFloat = e.force32itsFloat; } } C([ rn("Use non linear depth", Gr.Boolean, "ADVANCED", { notifiers: { activatePreviewCommand: !0, callback: (A, e) => { const t = e; let r = !1; return t.useNonLinearDepth && (t.storeCameraSpaceZ = !1, r = !0), A && A.disableDepthRenderer(), r; } } }) ], oV.prototype, "useNonLinearDepth", void 0); C([ rn("Store Camera space Z", Gr.Boolean, "ADVANCED", { notifiers: { activatePreviewCommand: !0, callback: (A, e) => { const t = e; let r = !1; return t.storeCameraSpaceZ && (t.useNonLinearDepth = !1, r = !0), A && A.disableDepthRenderer(), r; } } }) ], oV.prototype, "storeCameraSpaceZ", void 0); C([ rn("Force 32 bits float", Gr.Boolean, "ADVANCED", { notifiers: { activatePreviewCommand: !0, callback: (A) => A == null ? void 0 : A.disableDepthRenderer() } }) ], oV.prototype, "force32itsFloat", void 0); Ue("BABYLON.SceneDepthBlock", oV); class Hre extends Mr { /** * Create a new ClipPlanesBlock * @param name defines the block name */ constructor(e) { super(e, Ve.VertexAndFragment, !0), this.registerInput("worldPosition", de.Vector4, !1); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ClipPlanesBlock"; } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("vClipPlane"), e._excludeVariableName("fClipDistance"), e._excludeVariableName("vClipPlane2"), e._excludeVariableName("fClipDistance2"), e._excludeVariableName("vClipPlane3"), e._excludeVariableName("fClipDistance3"), e._excludeVariableName("vClipPlane4"), e._excludeVariableName("fClipDistance4"), e._excludeVariableName("vClipPlane5"), e._excludeVariableName("fClipDistance5"), e._excludeVariableName("vClipPlane6"), e._excludeVariableName("fClipDistance6"); } /** * Gets the worldPosition input component */ get worldPosition() { return this._inputs[0]; } get target() { return Ve.VertexAndFragment; } set target(e) { } prepareDefines(e, t, r) { var n, i, s, a, f, o; const d = e.getScene(), v = !!((n = t.clipPlane) !== null && n !== void 0 ? n : d.clipPlane), u = !!((i = t.clipPlane2) !== null && i !== void 0 ? i : d.clipPlane2), l = !!((s = t.clipPlane3) !== null && s !== void 0 ? s : d.clipPlane3), P = !!((a = t.clipPlane4) !== null && a !== void 0 ? a : d.clipPlane4), p = !!((f = t.clipPlane5) !== null && f !== void 0 ? f : d.clipPlane5), c = !!((o = t.clipPlane6) !== null && o !== void 0 ? o : d.clipPlane6); r.setValue("CLIPPLANE", v, !0), r.setValue("CLIPPLANE2", u, !0), r.setValue("CLIPPLANE3", l, !0), r.setValue("CLIPPLANE4", P, !0), r.setValue("CLIPPLANE5", p, !0), r.setValue("CLIPPLANE6", c, !0); } bind(e, t, r) { if (!r) return; const n = r.getScene(); Df(e, t, n); } _buildBlock(e) { super._buildBlock(e); const t = `//${this.name}`; if (e.target !== Ve.Fragment) { const r = this.worldPosition; e._emitFunctionFromInclude("clipPlaneVertexDeclaration", t, { replaceStrings: [{ search: /uniform vec4 vClipPlane\d*;/g, replace: "" }] }), e.compilationString += e._emitCodeFromInclude("clipPlaneVertex", t, { replaceStrings: [{ search: /worldPos/g, replace: r.associatedVariableName }] }), e._emitUniformFromString("vClipPlane", "vec4"), e._emitUniformFromString("vClipPlane2", "vec4"), e._emitUniformFromString("vClipPlane3", "vec4"), e._emitUniformFromString("vClipPlane4", "vec4"), e._emitUniformFromString("vClipPlane5", "vec4"), e._emitUniformFromString("vClipPlane6", "vec4"); return; } return e.sharedData.bindableBlocks.push(this), e.sharedData.blocksWithDefines.push(this), e._emitFunctionFromInclude("clipPlaneFragmentDeclaration", t), e.compilationString += e._emitCodeFromInclude("clipPlaneFragment", t), this; } } Ue("BABYLON.ClipPlanesBlock", Hre); class gre extends Mr { /** * The texture associated with the node is the prepass texture */ get texture() { return null; } set texture(e) { } /** * Creates a new PrePassTextureBlock * @param name defines the block name * @param target defines the target of that block (VertexAndFragment by default) */ constructor(e, t = Ve.VertexAndFragment) { super(e, t, !1), this.registerOutput("position", de.Object, Ve.VertexAndFragment, new Ko("position", this, ao.Output, E0, "ImageSourceBlock")), this.registerOutput("depth", de.Object, Ve.VertexAndFragment, new Ko("depth", this, ao.Output, E0, "ImageSourceBlock")), this.registerOutput("normal", de.Object, Ve.VertexAndFragment, new Ko("normal", this, ao.Output, E0, "ImageSourceBlock")); } /** * Returns the sampler name associated with the node connection point * @param output * @returns */ getSamplerName(e) { return e === this._outputs[0] ? this._positionSamplerName : e === this._outputs[1] ? this._depthSamplerName : e === this._outputs[2] ? this._normalSamplerName : ""; } /** * Gets the position texture */ get position() { return this._outputs[0]; } /** * Gets the depth texture */ get depth() { return this._outputs[1]; } /** * Gets the normal texture */ get normal() { return this._outputs[2]; } /** * Gets the sampler name associated with this image source */ get positionSamplerName() { return this._positionSamplerName; } /** * Gets the sampler name associated with this image source */ get normalSamplerName() { return this._normalSamplerName; } /** * Gets the sampler name associated with this image source */ get depthSamplerName() { return this._depthSamplerName; } /** * Gets the current class name * @returns the class name */ getClassName() { return "PrePassTextureBlock"; } _buildBlock(e) { if (super._buildBlock(e), e.target !== Ve.Vertex) return this._positionSamplerName = "prepassPositionSampler", this._depthSamplerName = "prepassDepthSampler", this._normalSamplerName = "prepassNormalSampler", e.sharedData.variableNames.prepassPositionSampler = 0, e.sharedData.variableNames.prepassDepthSampler = 0, e.sharedData.variableNames.prepassNormalSampler = 0, e.sharedData.textureBlocks.push(this), e.sharedData.bindableBlocks.push(this), e._emit2DSampler(this._positionSamplerName), e._emit2DSampler(this._depthSamplerName), e._emit2DSampler(this._normalSamplerName), this; } bind(e, t) { const n = t.getScene().enablePrePassRenderer(); if (!n) return; const i = n.defaultRT; i.textures && (this.position.isConnected && e.setTexture(this._positionSamplerName, i.textures[n.getIndex(1)]), this.depth.isConnected && e.setTexture(this._depthSamplerName, i.textures[n.getIndex(5)]), this.normal.isConnected && e.setTexture(this._normalSamplerName, i.textures[n.getIndex(6)])); } } Ue("BABYLON.PrePassTextureBlock", gre); class Xre extends Mr { /** Gets the list of attached endpoints */ get endpoints() { return this._endpoints; } /** * Gets or sets the target of the block */ get target() { const e = this._inputs[0]; if (e.isConnected) { const t = e.connectedPoint.ownerBlock; if (t.target !== Ve.VertexAndFragment) return t.target; if (e.connectedPoint.target !== Ve.VertexAndFragment) return e.connectedPoint.target; } return this._target; } set target(e) { this._target & e || (this._target = e); } /** * Create a new NodeMaterialTeleportInBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this._endpoints = [], this.registerInput("input", de.AutoDetect); } /** * Gets the current class name * @returns the class name */ getClassName() { return "NodeMaterialTeleportInBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** Gets a boolean indicating that this connection will be used in the fragment shader */ isConnectedInFragmentShader() { return this.endpoints.some((e) => e.output.isConnectedInFragmentShader); } _dumpCode(e, t) { let r = super._dumpCode(e, t); for (const n of this.endpoints) t.indexOf(n) === -1 && (r += n._dumpCode(e, t)); return r; } /** * Checks if the current block is an ancestor of a given block * @param block defines the potential descendant block to check * @returns true if block is a descendant */ isAnAncestorOf(e) { for (const t of this.endpoints) if (t === e || t.isAnAncestorOf(e)) return !0; return !1; } /** * Add an enpoint to this block * @param endpoint define the endpoint to attach to */ attachToEndpoint(e) { e.detach(), this._endpoints.push(e), e._entryPoint = this, e._outputs[0]._typeConnectionSource = this._inputs[0], e._tempEntryPointUniqueId = null, e.name = "> " + this.name; } /** * Remove enpoint from this block * @param endpoint define the endpoint to remove */ detachFromEndpoint(e) { const t = this._endpoints.indexOf(e); t !== -1 && (this._endpoints.splice(t, 1), e._outputs[0]._typeConnectionSource = null, e._entryPoint = null); } /** * Release resources */ dispose() { super.dispose(); for (const e of this._endpoints) this.detachFromEndpoint(e); this._endpoints = []; } } Ue("BABYLON.NodeMaterialTeleportInBlock", Xre); class Tre extends Mr { /** * Create a new TeleportOutBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this._entryPoint = null, this._tempEntryPointUniqueId = null, this.registerOutput("output", de.BasedOnInput); } /** * Gets the entry point */ get entryPoint() { return this._entryPoint; } /** * Gets the current class name * @returns the class name */ getClassName() { return "NodeMaterialTeleportOutBlock"; } /** * Gets the output component */ get output() { return this._outputs[0]; } /** * Gets or sets the target of the block */ get target() { return this._entryPoint ? this._entryPoint.target : this._target; } set target(e) { this._target & e || (this._target = e); } /** Detach from entry point */ detach() { this._entryPoint && this._entryPoint.detachFromEndpoint(this); } _buildBlock(e) { super._buildBlock(e), this.entryPoint && (e.compilationString += this._declareOutput(this.output, e) + ` = ${this.entryPoint.input.associatedVariableName}; `); } /** * Clone the current block to a new identical block * @param scene defines the hosting scene * @param rootUrl defines the root URL to use to load textures and relative dependencies * @returns a copy of the current block */ clone(e, t = "") { const r = super.clone(e, t); return this.entryPoint && this.entryPoint.attachToEndpoint(r), r; } _customBuildStep(e, t) { this.entryPoint && this.entryPoint.build(e, t); } _dumpCode(e, t) { let r = ""; return this.entryPoint && t.indexOf(this.entryPoint) === -1 && (r += this.entryPoint._dumpCode(e, t)), r + super._dumpCode(e, t); } _dumpCodeForOutputConnections(e) { let t = super._dumpCodeForOutputConnections(e); return this.entryPoint && (t += this.entryPoint._dumpCodeForOutputConnections(e)), t; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return this.entryPoint && (e += `${this.entryPoint._codeVariableName}.attachToEndpoint(${this._codeVariableName}); `), e; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { var e, t; const r = super.serialize(); return r.entryPoint = (t = (e = this.entryPoint) === null || e === void 0 ? void 0 : e.uniqueId) !== null && t !== void 0 ? t : "", r; } _deserialize(e, t, r) { super._deserialize(e, t, r), this._tempEntryPointUniqueId = e.entryPoint; } } Ue("BABYLON.NodeMaterialTeleportOutBlock", Tre); class qre extends Mr { /** * Creates a new AddBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("left", de.AutoDetect), this.registerInput("right", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._linkConnectionTypes(0, 1), this._inputs[0].acceptedConnectionPointTypes.push(de.Float), this._inputs[1].acceptedConnectionPointTypes.push(de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "AddBlock"; } /** * Gets the left operand input component */ get left() { return this._inputs[0]; } /** * Gets the right operand input component */ get right() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = ${this.left.associatedVariableName} + ${this.right.associatedVariableName}; `, this; } } Ue("BABYLON.AddBlock", qre); class bre extends Mr { /** * Creates a new ScaleBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("input", de.AutoDetect), this.registerInput("factor", de.Float), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0]; } /** * Gets the current class name * @returns the class name */ getClassName() { return "ScaleBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the factor input component */ get factor() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = ${this.input.associatedVariableName} * ${this.factor.associatedVariableName}; `, this; } } Ue("BABYLON.ScaleBlock", bre); class Hy extends Mr { /** * Creates a new ClampBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.minimum = 0, this.maximum = 1, this.registerInput("value", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0]; } /** * Gets the current class name * @returns the class name */ getClassName() { return "ClampBlock"; } /** * Gets the value input component */ get value() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = clamp(${this.value.associatedVariableName}, ${this._writeFloat(this.minimum)}, ${this._writeFloat(this.maximum)}); `, this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode() + `${this._codeVariableName}.minimum = ${this.minimum}; `; return e += `${this._codeVariableName}.maximum = ${this.maximum}; `, e; } serialize() { const e = super.serialize(); return e.minimum = this.minimum, e.maximum = this.maximum, e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.minimum = e.minimum, this.maximum = e.maximum; } } C([ rn("Minimum", Gr.Float) ], Hy.prototype, "minimum", void 0); C([ rn("Maximum", Gr.Float) ], Hy.prototype, "maximum", void 0); Ue("BABYLON.ClampBlock", Hy); class xre extends Mr { /** * Creates a new CrossBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("left", de.AutoDetect), this.registerInput("right", de.AutoDetect), this.registerOutput("output", de.Vector3), this._linkConnectionTypes(0, 1), this._inputs[0].excludedConnectionPointTypes.push(de.Float), this._inputs[0].excludedConnectionPointTypes.push(de.Matrix), this._inputs[0].excludedConnectionPointTypes.push(de.Vector2), this._inputs[1].excludedConnectionPointTypes.push(de.Float), this._inputs[1].excludedConnectionPointTypes.push(de.Matrix), this._inputs[1].excludedConnectionPointTypes.push(de.Vector2); } /** * Gets the current class name * @returns the class name */ getClassName() { return "CrossBlock"; } /** * Gets the left operand input component */ get left() { return this._inputs[0]; } /** * Gets the right operand input component */ get right() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = cross(${this.left.associatedVariableName}.xyz, ${this.right.associatedVariableName}.xyz); `, this; } } Ue("BABYLON.CrossBlock", xre); class Dre extends Mr { /** * Gets or sets the options for this custom block */ get options() { return this._options; } set options(e) { this._deserializeOptions(e); } /** * Creates a new CustomBlock * @param name defines the block name */ constructor(e) { super(e); } /** * Gets the current class name * @returns the class name */ getClassName() { return "CustomBlock"; } _buildBlock(e) { super._buildBlock(e); let t = this._code, r = this._options.functionName; this._inputs.forEach((i) => { const s = new RegExp("\\{TYPE_" + i.name + "\\}", "gm"), a = e._getGLType(i.type); t = t.replace(s, a), r = r.replace(s, a); }), this._outputs.forEach((i) => { const s = new RegExp("\\{TYPE_" + i.name + "\\}", "gm"), a = e._getGLType(i.type); t = t.replace(s, a), r = r.replace(s, a); }), e._emitFunction(r, t, ""), this._outputs.forEach((i) => { e.compilationString += this._declareOutput(i, e) + `; `; }), e.compilationString += r + "("; let n = !1; return this._inputs.forEach((i, s) => { var a, f, o; s > 0 && (e.compilationString += ", "), this._inputSamplers && this._inputSamplers.indexOf(i.name) !== -1 ? e.compilationString += (o = (f = (a = i.connectedPoint) === null || a === void 0 ? void 0 : a.ownerBlock) === null || f === void 0 ? void 0 : f.samplerName) !== null && o !== void 0 ? o : i.associatedVariableName : e.compilationString += i.associatedVariableName, n = !0; }), this._outputs.forEach((i, s) => { (s > 0 || n) && (e.compilationString += ", "), e.compilationString += i.associatedVariableName; }), e.compilationString += `); `, this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return e += `${this._codeVariableName}.options = ${JSON.stringify(this._options)}; `, e; } serialize() { const e = super.serialize(); return e.options = this._options, e; } _deserialize(e, t, r) { this._deserializeOptions(e.options), super._deserialize(e, t, r); } _deserializeOptions(e) { var t, r, n; this._options = e, this._code = e.code.join(` `) + ` `, this.name = this.name || e.name, this.target = Ve[e.target], (t = e.inParameters) === null || t === void 0 || t.forEach((i, s) => { const a = de[i.type]; i.type === "sampler2D" || i.type === "samplerCube" ? (this._inputSamplers = this._inputSamplers || [], this._inputSamplers.push(i.name), this.registerInput(i.name, de.Object, !0, Ve.VertexAndFragment, new Ko(i.name, this, ao.Input, E0, "ImageSourceBlock"))) : this.registerInput(i.name, a), Object.defineProperty(this, i.name, { get: function() { return this._inputs[s]; }, enumerable: !0, configurable: !0 }); }), (r = e.outParameters) === null || r === void 0 || r.forEach((i, s) => { this.registerOutput(i.name, de[i.type]), Object.defineProperty(this, i.name, { get: function() { return this._outputs[s]; }, enumerable: !0, configurable: !0 }), i.type === "BasedOnInput" && (this._outputs[s]._typeConnectionSource = this._findInputByName(i.typeFromInput)[0]); }), (n = e.inLinkedConnectionTypes) === null || n === void 0 || n.forEach((i) => { this._linkConnectionTypes(this._findInputByName(i.input1)[1], this._findInputByName(i.input2)[1]); }); } _findInputByName(e) { if (!e) return null; for (let t = 0; t < this._inputs.length; t++) if (this._inputs[t].name === e) return [this._inputs[t], t]; return null; } } Ue("BABYLON.CustomBlock", Dre); class jre extends Mr { /** * Creates a new DotBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("left", de.AutoDetect), this.registerInput("right", de.AutoDetect), this.registerOutput("output", de.Float), this._linkConnectionTypes(0, 1), this._inputs[0].excludedConnectionPointTypes.push(de.Float), this._inputs[0].excludedConnectionPointTypes.push(de.Matrix), this._inputs[1].excludedConnectionPointTypes.push(de.Float), this._inputs[1].excludedConnectionPointTypes.push(de.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "DotBlock"; } /** * Gets the left operand input component */ get left() { return this._inputs[0]; } /** * Gets the right operand input component */ get right() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = dot(${this.left.associatedVariableName}, ${this.right.associatedVariableName}); `, this; } } Ue("BABYLON.DotBlock", jre); class wre extends Mr { /** * Creates a new NormalizeBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("input", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._inputs[0].excludedConnectionPointTypes.push(de.Float), this._inputs[0].excludedConnectionPointTypes.push(de.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "NormalizeBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0], r = this._inputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = normalize(${r.associatedVariableName}); `, this; } } Ue("BABYLON.NormalizeBlock", wre); class mre extends Mr { /** * Create a new ColorMergerBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.rSwizzle = "r", this.gSwizzle = "g", this.bSwizzle = "b", this.aSwizzle = "a", this.registerInput("rgb ", de.Color3, !0), this.registerInput("r", de.Float, !0), this.registerInput("g", de.Float, !0), this.registerInput("b", de.Float, !0), this.registerInput("a", de.Float, !0), this.registerOutput("rgba", de.Color4), this.registerOutput("rgb", de.Color3); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ColorMergerBlock"; } /** * Gets the rgb component (input) */ get rgbIn() { return this._inputs[0]; } /** * Gets the r component (input) */ get r() { return this._inputs[1]; } /** * Gets the g component (input) */ get g() { return this._inputs[2]; } /** * Gets the b component (input) */ get b() { return this._inputs[3]; } /** * Gets the a component (input) */ get a() { return this._inputs[4]; } /** * Gets the rgba component (output) */ get rgba() { return this._outputs[0]; } /** * Gets the rgb component (output) */ get rgbOut() { return this._outputs[1]; } /** * Gets the rgb component (output) * @deprecated Please use rgbOut instead. */ get rgb() { return this.rgbOut; } _inputRename(e) { return e === "rgb " ? "rgbIn" : e; } _buildSwizzle(e) { return "." + (this.rSwizzle + this.gSwizzle + this.bSwizzle + this.aSwizzle).substr(0, e); } _buildBlock(e) { super._buildBlock(e); const t = this.r, r = this.g, n = this.b, i = this.a, s = this.rgbIn, a = this._outputs[0], f = this._outputs[1]; return s.isConnected ? (a.hasEndpoints && (e.compilationString += this._declareOutput(a, e) + ` = vec4(${s.associatedVariableName}, ${i.isConnected ? this._writeVariable(i) : "0.0"})${this._buildSwizzle(4)}; `), f.hasEndpoints && (e.compilationString += this._declareOutput(f, e) + ` = ${s.associatedVariableName}${this._buildSwizzle(3)}; `)) : (a.hasEndpoints && (e.compilationString += this._declareOutput(a, e) + ` = vec4(${t.isConnected ? this._writeVariable(t) : "0.0"}, ${r.isConnected ? this._writeVariable(r) : "0.0"}, ${n.isConnected ? this._writeVariable(n) : "0.0"}, ${i.isConnected ? this._writeVariable(i) : "0.0"})${this._buildSwizzle(4)}; `), f.hasEndpoints && (e.compilationString += this._declareOutput(f, e) + ` = vec3(${t.isConnected ? this._writeVariable(t) : "0.0"}, ${r.isConnected ? this._writeVariable(r) : "0.0"}, ${n.isConnected ? this._writeVariable(n) : "0.0"})${this._buildSwizzle(3)}; `)), this; } serialize() { const e = super.serialize(); return e.rSwizzle = this.rSwizzle, e.gSwizzle = this.gSwizzle, e.bSwizzle = this.bSwizzle, e.aSwizzle = this.aSwizzle, e; } _deserialize(e, t, r) { var n, i, s, a; super._deserialize(e, t, r), this.rSwizzle = (n = e.rSwizzle) !== null && n !== void 0 ? n : "r", this.gSwizzle = (i = e.gSwizzle) !== null && i !== void 0 ? i : "g", this.bSwizzle = (s = e.bSwizzle) !== null && s !== void 0 ? s : "b", this.aSwizzle = (a = e.aSwizzle) !== null && a !== void 0 ? a : "a"; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return e += `${this._codeVariableName}.rSwizzle = "${this.rSwizzle}"; `, e += `${this._codeVariableName}.gSwizzle = "${this.gSwizzle}"; `, e += `${this._codeVariableName}.bSwizzle = "${this.bSwizzle}"; `, e += `${this._codeVariableName}.aSwizzle = "${this.aSwizzle}"; `, e; } } Ue("BABYLON.ColorMergerBlock", mre); class Bre extends Mr { /** * Create a new VectorSplitterBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("xyzw", de.Vector4, !0), this.registerInput("xyz ", de.Vector3, !0), this.registerInput("xy ", de.Vector2, !0), this.registerOutput("xyz", de.Vector3), this.registerOutput("xy", de.Vector2), this.registerOutput("zw", de.Vector2), this.registerOutput("x", de.Float), this.registerOutput("y", de.Float), this.registerOutput("z", de.Float), this.registerOutput("w", de.Float), this.inputsAreExclusive = !0; } /** * Gets the current class name * @returns the class name */ getClassName() { return "VectorSplitterBlock"; } /** * Gets the xyzw component (input) */ get xyzw() { return this._inputs[0]; } /** * Gets the xyz component (input) */ get xyzIn() { return this._inputs[1]; } /** * Gets the xy component (input) */ get xyIn() { return this._inputs[2]; } /** * Gets the xyz component (output) */ get xyzOut() { return this._outputs[0]; } /** * Gets the xy component (output) */ get xyOut() { return this._outputs[1]; } /** * Gets the zw component (output) */ get zw() { return this._outputs[2]; } /** * Gets the x component (output) */ get x() { return this._outputs[3]; } /** * Gets the y component (output) */ get y() { return this._outputs[4]; } /** * Gets the z component (output) */ get z() { return this._outputs[5]; } /** * Gets the w component (output) */ get w() { return this._outputs[6]; } _inputRename(e) { switch (e) { case "xy ": return "xyIn"; case "xyz ": return "xyzIn"; default: return e; } } _outputRename(e) { switch (e) { case "xy": return "xyOut"; case "xyz": return "xyzOut"; default: return e; } } _buildBlock(e) { super._buildBlock(e); const t = this.xyzw.isConnected ? this.xyzw : this.xyzIn.isConnected ? this.xyzIn : this.xyIn, r = this._outputs[0], n = this._outputs[1], i = this._outputs[2], s = this._outputs[3], a = this._outputs[4], f = this._outputs[5], o = this._outputs[6]; return r.hasEndpoints && (t === this.xyIn ? e.compilationString += this._declareOutput(r, e) + ` = vec3(${t.associatedVariableName}, 0.0); ` : e.compilationString += this._declareOutput(r, e) + ` = ${t.associatedVariableName}.xyz; `), i.hasEndpoints && this.xyzw.isConnected && (e.compilationString += this._declareOutput(i, e) + ` = ${this.xyzw.associatedVariableName}.zw; `), n.hasEndpoints && (e.compilationString += this._declareOutput(n, e) + ` = ${t.associatedVariableName}.xy; `), s.hasEndpoints && (e.compilationString += this._declareOutput(s, e) + ` = ${t.associatedVariableName}.x; `), a.hasEndpoints && (e.compilationString += this._declareOutput(a, e) + ` = ${t.associatedVariableName}.y; `), f.hasEndpoints && (e.compilationString += this._declareOutput(f, e) + ` = ${t.associatedVariableName}.z; `), o.hasEndpoints && (e.compilationString += this._declareOutput(o, e) + ` = ${t.associatedVariableName}.w; `), this; } } Ue("BABYLON.VectorSplitterBlock", Bre); class Wre extends Mr { /** * Creates a new LerpBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("left", de.AutoDetect), this.registerInput("right", de.AutoDetect), this.registerInput("gradient", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._linkConnectionTypes(0, 1), this._linkConnectionTypes(1, 2, !0), this._inputs[2].acceptedConnectionPointTypes.push(de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "LerpBlock"; } /** * Gets the left operand input component */ get left() { return this._inputs[0]; } /** * Gets the right operand input component */ get right() { return this._inputs[1]; } /** * Gets the gradient operand input component */ get gradient() { return this._inputs[2]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = mix(${this.left.associatedVariableName} , ${this.right.associatedVariableName}, ${this.gradient.associatedVariableName}); `, this; } } Ue("BABYLON.LerpBlock", Wre); class Sre extends Mr { /** * Creates a new DivideBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("left", de.AutoDetect), this.registerInput("right", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._linkConnectionTypes(0, 1), this._inputs[0].acceptedConnectionPointTypes.push(de.Float), this._inputs[1].acceptedConnectionPointTypes.push(de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "DivideBlock"; } /** * Gets the left operand input component */ get left() { return this._inputs[0]; } /** * Gets the right operand input component */ get right() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = ${this.left.associatedVariableName} / ${this.right.associatedVariableName}; `, this; } } Ue("BABYLON.DivideBlock", Sre); class Ure extends Mr { /** * Creates a new SubtractBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("left", de.AutoDetect), this.registerInput("right", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._linkConnectionTypes(0, 1), this._inputs[0].acceptedConnectionPointTypes.push(de.Float), this._inputs[1].acceptedConnectionPointTypes.push(de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "SubtractBlock"; } /** * Gets the left operand input component */ get left() { return this._inputs[0]; } /** * Gets the right operand input component */ get right() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = ${this.left.associatedVariableName} - ${this.right.associatedVariableName}; `, this; } } Ue("BABYLON.SubtractBlock", Ure); class Ire extends Mr { /** * Creates a new StepBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("value", de.Float), this.registerInput("edge", de.Float), this.registerOutput("output", de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "StepBlock"; } /** * Gets the value operand input component */ get value() { return this._inputs[0]; } /** * Gets the edge operand input component */ get edge() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = step(${this.edge.associatedVariableName}, ${this.value.associatedVariableName}); `, this; } } Ue("BABYLON.StepBlock", Ire); class iY extends Mr { /** * Creates a new OneMinusBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("input", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._outputs[0].excludedConnectionPointTypes.push(de.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "OneMinusBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = 1. - ${this.input.associatedVariableName}; `, this; } } Ue("BABYLON.OneMinusBlock", iY); Ue("BABYLON.OppositeBlock", iY); class sY extends Mr { /** * Creates a new ViewDirectionBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("worldPosition", de.Vector4), this.registerInput("cameraPosition", de.Vector3), this.registerOutput("output", de.Vector3); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ViewDirectionBlock"; } /** * Gets the world position component */ get worldPosition() { return this._inputs[0]; } /** * Gets the camera position component */ get cameraPosition() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } autoConfigure(e, t = () => !0) { if (!this.cameraPosition.isConnected) { let r = e.getInputBlockByPredicate((n) => n.systemValue === Bi.CameraPosition && t(n)); r || (r = new gi("cameraPosition"), r.setAsSystemValue(Bi.CameraPosition)), r.output.connectTo(this.cameraPosition); } } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = normalize(${this.cameraPosition.associatedVariableName} - ${this.worldPosition.associatedVariableName}.xyz); `, this; } } Ue("BABYLON.ViewDirectionBlock", sY); class Rre extends Mr { /** * Create a new FresnelBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("worldNormal", de.Vector4), this.registerInput("viewDirection", de.Vector3), this.registerInput("bias", de.Float), this.registerInput("power", de.Float), this.registerOutput("fresnel", de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "FresnelBlock"; } /** * Gets the world normal input component */ get worldNormal() { return this._inputs[0]; } /** * Gets the view direction input component */ get viewDirection() { return this._inputs[1]; } /** * Gets the bias input component */ get bias() { return this._inputs[2]; } /** * Gets the camera (or eye) position component */ get power() { return this._inputs[3]; } /** * Gets the fresnel output component */ get fresnel() { return this._outputs[0]; } autoConfigure(e) { if (!this.viewDirection.isConnected) { const t = new sY("View direction"); t.output.connectTo(this.viewDirection), t.autoConfigure(e); } if (!this.bias.isConnected) { const t = new gi("bias"); t.value = 0, t.output.connectTo(this.bias); } if (!this.power.isConnected) { const t = new gi("power"); t.value = 1, t.output.connectTo(this.power); } } _buildBlock(e) { super._buildBlock(e); const t = `//${this.name}`; return e._emitFunctionFromInclude("fresnelFunction", t, { removeIfDef: !0 }), e.compilationString += this._declareOutput(this.fresnel, e) + ` = computeFresnelTerm(${this.viewDirection.associatedVariableName}.xyz, ${this.worldNormal.associatedVariableName}.xyz, ${this.bias.associatedVariableName}, ${this.power.associatedVariableName}); `, this; } } Ue("BABYLON.FresnelBlock", Rre); class Vre extends Mr { /** * Creates a new MaxBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("left", de.AutoDetect), this.registerInput("right", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._linkConnectionTypes(0, 1); } /** * Gets the current class name * @returns the class name */ getClassName() { return "MaxBlock"; } /** * Gets the left operand input component */ get left() { return this._inputs[0]; } /** * Gets the right operand input component */ get right() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = max(${this.left.associatedVariableName}, ${this.right.associatedVariableName}); `, this; } } Ue("BABYLON.MaxBlock", Vre); class Cre extends Mr { /** * Creates a new MinBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("left", de.AutoDetect), this.registerInput("right", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._linkConnectionTypes(0, 1); } /** * Gets the current class name * @returns the class name */ getClassName() { return "MinBlock"; } /** * Gets the left operand input component */ get left() { return this._inputs[0]; } /** * Gets the right operand input component */ get right() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = min(${this.left.associatedVariableName}, ${this.right.associatedVariableName}); `, this; } } Ue("BABYLON.MinBlock", Cre); class Ore extends Mr { /** * Creates a new DistanceBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("left", de.AutoDetect), this.registerInput("right", de.AutoDetect), this.registerOutput("output", de.Float), this._linkConnectionTypes(0, 1), this._inputs[0].excludedConnectionPointTypes.push(de.Float), this._inputs[0].excludedConnectionPointTypes.push(de.Matrix), this._inputs[1].excludedConnectionPointTypes.push(de.Float), this._inputs[1].excludedConnectionPointTypes.push(de.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "DistanceBlock"; } /** * Gets the left operand input component */ get left() { return this._inputs[0]; } /** * Gets the right operand input component */ get right() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = length(${this.left.associatedVariableName} - ${this.right.associatedVariableName}); `, this; } } Ue("BABYLON.DistanceBlock", Ore); class yre extends Mr { /** * Creates a new LengthBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("value", de.AutoDetect), this.registerOutput("output", de.Float), this._inputs[0].excludedConnectionPointTypes.push(de.Float), this._inputs[0].excludedConnectionPointTypes.push(de.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "LengthBlock"; } /** * Gets the value input component */ get value() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = length(${this.value.associatedVariableName}); `, this; } } Ue("BABYLON.LengthBlock", yre); class kre extends Mr { /** * Creates a new NegateBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("value", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0]; } /** * Gets the current class name * @returns the class name */ getClassName() { return "NegateBlock"; } /** * Gets the value input component */ get value() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = -1.0 * ${this.value.associatedVariableName}; `, this; } } Ue("BABYLON.NegateBlock", kre); class Ere extends Mr { /** * Creates a new PowBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("value", de.AutoDetect), this.registerInput("power", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._linkConnectionTypes(0, 1); } /** * Gets the current class name * @returns the class name */ getClassName() { return "PowBlock"; } /** * Gets the value operand input component */ get value() { return this._inputs[0]; } /** * Gets the power operand input component */ get power() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = pow(${this.value.associatedVariableName}, ${this.power.associatedVariableName}); `, this; } } Ue("BABYLON.PowBlock", Ere); class Fre extends Mr { /** * Creates a new RandomNumberBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("seed", de.AutoDetect), this.registerOutput("output", de.Float), this._inputs[0].addExcludedConnectionPointFromAllowedTypes(de.Vector2 | de.Vector3 | de.Vector4 | de.Color3 | de.Color4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "RandomNumberBlock"; } /** * Gets the seed input component */ get seed() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0], r = `//${this.name}`; return e._emitFunctionFromInclude("helperFunctions", r), e.compilationString += this._declareOutput(t, e) + ` = getRand(${this.seed.associatedVariableName}.xy); `, this; } } Ue("BABYLON.RandomNumberBlock", Fre); class Nre extends Mr { /** * Creates a new ArcTan2Block * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("x", de.Float), this.registerInput("y", de.Float), this.registerOutput("output", de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ArcTan2Block"; } /** * Gets the x operand input component */ get x() { return this._inputs[0]; } /** * Gets the y operand input component */ get y() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = atan(${this.x.associatedVariableName}, ${this.y.associatedVariableName}); `, this; } } Ue("BABYLON.ArcTan2Block", Nre); class Qre extends Mr { /** * Creates a new SmoothStepBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("value", de.AutoDetect), this.registerInput("edge0", de.Float), this.registerInput("edge1", de.Float), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0]; } /** * Gets the current class name * @returns the class name */ getClassName() { return "SmoothStepBlock"; } /** * Gets the value operand input component */ get value() { return this._inputs[0]; } /** * Gets the first edge operand input component */ get edge0() { return this._inputs[1]; } /** * Gets the second edge operand input component */ get edge1() { return this._inputs[2]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = smoothstep(${this.edge0.associatedVariableName}, ${this.edge1.associatedVariableName}, ${this.value.associatedVariableName}); `, this; } } Ue("BABYLON.SmoothStepBlock", Qre); class Yre extends Mr { /** * Creates a new ReciprocalBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("input", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0]; } /** * Gets the current class name * @returns the class name */ getClassName() { return "ReciprocalBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return this.input.type === de.Matrix ? e.compilationString += this._declareOutput(t, e) + ` = inverse(${this.input.associatedVariableName}); ` : e.compilationString += this._declareOutput(t, e) + ` = 1. / ${this.input.associatedVariableName}; `, this; } } Ue("BABYLON.ReciprocalBlock", Yre); class Mre extends Mr { /** * Creates a new ReplaceColorBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("value", de.AutoDetect), this.registerInput("reference", de.AutoDetect), this.registerInput("distance", de.Float), this.registerInput("replacement", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._linkConnectionTypes(0, 1), this._linkConnectionTypes(0, 3), this._inputs[0].excludedConnectionPointTypes.push(de.Float), this._inputs[0].excludedConnectionPointTypes.push(de.Matrix), this._inputs[1].excludedConnectionPointTypes.push(de.Float), this._inputs[1].excludedConnectionPointTypes.push(de.Matrix), this._inputs[3].excludedConnectionPointTypes.push(de.Float), this._inputs[3].excludedConnectionPointTypes.push(de.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ReplaceColorBlock"; } /** * Gets the value input component */ get value() { return this._inputs[0]; } /** * Gets the reference input component */ get reference() { return this._inputs[1]; } /** * Gets the distance input component */ get distance() { return this._inputs[2]; } /** * Gets the replacement input component */ get replacement() { return this._inputs[3]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + `; `, e.compilationString += `if (length(${this.value.associatedVariableName} - ${this.reference.associatedVariableName}) < ${this.distance.associatedVariableName}) { `, e.compilationString += `${t.associatedVariableName} = ${this.replacement.associatedVariableName}; `, e.compilationString += `} else { `, e.compilationString += `${t.associatedVariableName} = ${this.value.associatedVariableName}; `, e.compilationString += `} `, this; } } Ue("BABYLON.ReplaceColorBlock", Mre); class Lre extends Mr { /** * Creates a new PosterizeBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("value", de.AutoDetect), this.registerInput("steps", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._linkConnectionTypes(0, 1), this._inputs[0].excludedConnectionPointTypes.push(de.Matrix), this._inputs[1].excludedConnectionPointTypes.push(de.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "PosterizeBlock"; } /** * Gets the value input component */ get value() { return this._inputs[0]; } /** * Gets the steps input component */ get steps() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = floor(${this.value.associatedVariableName} / (1.0 / ${this.steps.associatedVariableName})) * (1.0 / ${this.steps.associatedVariableName}); `, this; } } Ue("BABYLON.PosterizeBlock", Lre); var Gw; (function(A) { A[A.SawTooth = 0] = "SawTooth", A[A.Square = 1] = "Square", A[A.Triangle = 2] = "Triangle"; })(Gw || (Gw = {})); class Kre extends Mr { /** * Creates a new WaveBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.kind = Gw.SawTooth, this.registerInput("input", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._inputs[0].excludedConnectionPointTypes.push(de.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "WaveBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; switch (this.kind) { case Gw.SawTooth: { e.compilationString += this._declareOutput(t, e) + ` = ${this.input.associatedVariableName} - floor(0.5 + ${this.input.associatedVariableName}); `; break; } case Gw.Square: { e.compilationString += this._declareOutput(t, e) + ` = 1.0 - 2.0 * round(fract(${this.input.associatedVariableName})); `; break; } case Gw.Triangle: { e.compilationString += this._declareOutput(t, e) + ` = 2.0 * abs(2.0 * (${this.input.associatedVariableName} - floor(0.5 + ${this.input.associatedVariableName}))) - 1.0; `; break; } } return this; } serialize() { const e = super.serialize(); return e.kind = this.kind, e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.kind = e.kind; } } Ue("BABYLON.WaveBlock", Kre); class mC { /** * Gets value indicating which step this color is associated with (between 0 and 1) */ get step() { return this._step; } /** * Sets a value indicating which step this color is associated with (between 0 and 1) */ set step(e) { this._step = e; } /** * Gets the color associated with this step */ get color() { return this._color; } /** * Sets the color associated with this step */ set color(e) { this._color = e; } /** * Creates a new GradientBlockColorStep * @param step defines a value indicating which step this color is associated with (between 0 and 1) * @param color defines the color associated with this step */ constructor(e, t) { this.step = e, this.color = t; } } class Jre extends Mr { /** calls observable when the value is changed*/ colorStepsUpdated() { this.onValueChangedObservable.notifyObservers(this); } /** * Creates a new GradientBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.colorSteps = [new mC(0, Ne.Black()), new mC(1, Ne.White())], this.onValueChangedObservable = new Oe(), this.registerInput("gradient", de.AutoDetect), this.registerOutput("output", de.Color3), this._inputs[0].addExcludedConnectionPointFromAllowedTypes(de.Float | de.Vector2 | de.Vector3 | de.Vector4 | de.Color3 | de.Color4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "GradientBlock"; } /** * Gets the gradient input component */ get gradient() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _writeColorConstant(e) { const t = this.colorSteps[e]; return `vec3(${t.color.r}, ${t.color.g}, ${t.color.b})`; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; if (!this.colorSteps.length || !this.gradient.connectedPoint) { e.compilationString += this._declareOutput(t, e) + ` = vec3(0., 0., 0.); `; return; } const r = e._getFreeVariableName("gradientTempColor"), n = e._getFreeVariableName("gradientTempPosition"); e.compilationString += `vec3 ${r} = ${this._writeColorConstant(0)}; `, e.compilationString += `float ${n}; `; let i = this.gradient.associatedVariableName; this.gradient.connectedPoint.type !== de.Float && (i += ".x"); for (let s = 1; s < this.colorSteps.length; s++) { const a = this.colorSteps[s], f = this.colorSteps[s - 1]; e.compilationString += `${n} = clamp((${i} - ${e._emitFloat(f.step)}) / (${e._emitFloat(a.step)} - ${e._emitFloat(f.step)}), 0.0, 1.0) * step(${e._emitFloat(s)}, ${e._emitFloat(this.colorSteps.length - 1)}); `, e.compilationString += `${r} = mix(${r}, ${this._writeColorConstant(s)}, ${n}); `; } return e.compilationString += this._declareOutput(t, e) + ` = ${r}; `, this; } serialize() { const e = super.serialize(); e.colorSteps = []; for (const t of this.colorSteps) e.colorSteps.push({ step: t.step, color: { r: t.color.r, g: t.color.g, b: t.color.b } }); return e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.colorSteps.length = 0; for (const n of e.colorSteps) this.colorSteps.push(new mC(n.step, new Ne(n.color.r, n.color.g, n.color.b))); } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); e += `${this._codeVariableName}.colorSteps = []; `; for (const t of this.colorSteps) e += `${this._codeVariableName}.colorSteps.push(new BABYLON.GradientBlockColorStep(${t.step}, new BABYLON.Color3(${t.color.r}, ${t.color.g}, ${t.color.b}))); `; return e; } } Ue("BABYLON.GradientBlock", Jre); class zre extends Mr { /** * Creates a new NLerpBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("left", de.AutoDetect), this.registerInput("right", de.AutoDetect), this.registerInput("gradient", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._linkConnectionTypes(0, 1), this._linkConnectionTypes(1, 2, !0), this._inputs[2].acceptedConnectionPointTypes.push(de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "NLerpBlock"; } /** * Gets the left operand input component */ get left() { return this._inputs[0]; } /** * Gets the right operand input component */ get right() { return this._inputs[1]; } /** * Gets the gradient operand input component */ get gradient() { return this._inputs[2]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = normalize(mix(${this.left.associatedVariableName} , ${this.right.associatedVariableName}, ${this.gradient.associatedVariableName})); `, this; } } Ue("BABYLON.NLerpBlock", zre); class aY extends Mr { /** * Creates a new WorleyNoise3DBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.manhattanDistance = !1, this.registerInput("seed", de.Vector3), this.registerInput("jitter", de.Float), this.registerOutput("output", de.Vector2), this.registerOutput("x", de.Float), this.registerOutput("y", de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "WorleyNoise3DBlock"; } /** * Gets the seed input component */ get seed() { return this._inputs[0]; } /** * Gets the jitter input component */ get jitter() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } /** * Gets the x component */ get x() { return this._outputs[1]; } /** * Gets the y component */ get y() { return this._outputs[2]; } _buildBlock(e) { if (super._buildBlock(e), !this.seed.isConnected || !this.output.hasEndpoints && !this.x.hasEndpoints && !this.y.hasEndpoints) return; let t = `vec3 permute(vec3 x){ `; t += ` return mod((34.0 * x + 1.0) * x, 289.0); `, t += `} `, t += `vec3 dist(vec3 x, vec3 y, vec3 z, bool manhattanDistance){ `, t += ` return manhattanDistance ? abs(x) + abs(y) + abs(z) : (x * x + y * y + z * z); `, t += `} `, t += `vec2 worley(vec3 P, float jitter, bool manhattanDistance){ `, t += ` float K = 0.142857142857; // 1/7 `, t += ` float Ko = 0.428571428571; // 1/2-K/2 `, t += ` float K2 = 0.020408163265306; // 1/(7*7) `, t += ` float Kz = 0.166666666667; // 1/6 `, t += ` float Kzo = 0.416666666667; // 1/2-1/6*2 `, t += ` `, t += ` vec3 Pi = mod(floor(P), 289.0); `, t += ` vec3 Pf = fract(P) - 0.5; `, t += ` `, t += ` vec3 Pfx = Pf.x + vec3(1.0, 0.0, -1.0); `, t += ` vec3 Pfy = Pf.y + vec3(1.0, 0.0, -1.0); `, t += ` vec3 Pfz = Pf.z + vec3(1.0, 0.0, -1.0); `, t += ` `, t += ` vec3 p = permute(Pi.x + vec3(-1.0, 0.0, 1.0)); `, t += ` vec3 p1 = permute(p + Pi.y - 1.0); `, t += ` vec3 p2 = permute(p + Pi.y); `, t += ` vec3 p3 = permute(p + Pi.y + 1.0); `, t += ` `, t += ` vec3 p11 = permute(p1 + Pi.z - 1.0); `, t += ` vec3 p12 = permute(p1 + Pi.z); `, t += ` vec3 p13 = permute(p1 + Pi.z + 1.0); `, t += ` `, t += ` vec3 p21 = permute(p2 + Pi.z - 1.0); `, t += ` vec3 p22 = permute(p2 + Pi.z); `, t += ` vec3 p23 = permute(p2 + Pi.z + 1.0); `, t += ` `, t += ` vec3 p31 = permute(p3 + Pi.z - 1.0); `, t += ` vec3 p32 = permute(p3 + Pi.z); `, t += ` vec3 p33 = permute(p3 + Pi.z + 1.0); `, t += ` `, t += ` vec3 ox11 = fract(p11*K) - Ko; `, t += ` vec3 oy11 = mod(floor(p11*K), 7.0)*K - Ko; `, t += ` vec3 oz11 = floor(p11*K2)*Kz - Kzo; // p11 < 289 guaranteed `, t += ` `, t += ` vec3 ox12 = fract(p12*K) - Ko; `, t += ` vec3 oy12 = mod(floor(p12*K), 7.0)*K - Ko; `, t += ` vec3 oz12 = floor(p12*K2)*Kz - Kzo; `, t += ` `, t += ` vec3 ox13 = fract(p13*K) - Ko; `, t += ` vec3 oy13 = mod(floor(p13*K), 7.0)*K - Ko; `, t += ` vec3 oz13 = floor(p13*K2)*Kz - Kzo; `, t += ` `, t += ` vec3 ox21 = fract(p21*K) - Ko; `, t += ` vec3 oy21 = mod(floor(p21*K), 7.0)*K - Ko; `, t += ` vec3 oz21 = floor(p21*K2)*Kz - Kzo; `, t += ` `, t += ` vec3 ox22 = fract(p22*K) - Ko; `, t += ` vec3 oy22 = mod(floor(p22*K), 7.0)*K - Ko; `, t += ` vec3 oz22 = floor(p22*K2)*Kz - Kzo; `, t += ` `, t += ` vec3 ox23 = fract(p23*K) - Ko; `, t += ` vec3 oy23 = mod(floor(p23*K), 7.0)*K - Ko; `, t += ` vec3 oz23 = floor(p23*K2)*Kz - Kzo; `, t += ` `, t += ` vec3 ox31 = fract(p31*K) - Ko; `, t += ` vec3 oy31 = mod(floor(p31*K), 7.0)*K - Ko; `, t += ` vec3 oz31 = floor(p31*K2)*Kz - Kzo; `, t += ` `, t += ` vec3 ox32 = fract(p32*K) - Ko; `, t += ` vec3 oy32 = mod(floor(p32*K), 7.0)*K - Ko; `, t += ` vec3 oz32 = floor(p32*K2)*Kz - Kzo; `, t += ` `, t += ` vec3 ox33 = fract(p33*K) - Ko; `, t += ` vec3 oy33 = mod(floor(p33*K), 7.0)*K - Ko; `, t += ` vec3 oz33 = floor(p33*K2)*Kz - Kzo; `, t += ` `, t += ` vec3 dx11 = Pfx + jitter*ox11; `, t += ` vec3 dy11 = Pfy.x + jitter*oy11; `, t += ` vec3 dz11 = Pfz.x + jitter*oz11; `, t += ` `, t += ` vec3 dx12 = Pfx + jitter*ox12; `, t += ` vec3 dy12 = Pfy.x + jitter*oy12; `, t += ` vec3 dz12 = Pfz.y + jitter*oz12; `, t += ` `, t += ` vec3 dx13 = Pfx + jitter*ox13; `, t += ` vec3 dy13 = Pfy.x + jitter*oy13; `, t += ` vec3 dz13 = Pfz.z + jitter*oz13; `, t += ` `, t += ` vec3 dx21 = Pfx + jitter*ox21; `, t += ` vec3 dy21 = Pfy.y + jitter*oy21; `, t += ` vec3 dz21 = Pfz.x + jitter*oz21; `, t += ` `, t += ` vec3 dx22 = Pfx + jitter*ox22; `, t += ` vec3 dy22 = Pfy.y + jitter*oy22; `, t += ` vec3 dz22 = Pfz.y + jitter*oz22; `, t += ` `, t += ` vec3 dx23 = Pfx + jitter*ox23; `, t += ` vec3 dy23 = Pfy.y + jitter*oy23; `, t += ` vec3 dz23 = Pfz.z + jitter*oz23; `, t += ` `, t += ` vec3 dx31 = Pfx + jitter*ox31; `, t += ` vec3 dy31 = Pfy.z + jitter*oy31; `, t += ` vec3 dz31 = Pfz.x + jitter*oz31; `, t += ` `, t += ` vec3 dx32 = Pfx + jitter*ox32; `, t += ` vec3 dy32 = Pfy.z + jitter*oy32; `, t += ` vec3 dz32 = Pfz.y + jitter*oz32; `, t += ` `, t += ` vec3 dx33 = Pfx + jitter*ox33; `, t += ` vec3 dy33 = Pfy.z + jitter*oy33; `, t += ` vec3 dz33 = Pfz.z + jitter*oz33; `, t += ` `, t += ` vec3 d11 = dist(dx11, dy11, dz11, manhattanDistance); `, t += ` vec3 d12 =dist(dx12, dy12, dz12, manhattanDistance); `, t += ` vec3 d13 = dist(dx13, dy13, dz13, manhattanDistance); `, t += ` vec3 d21 = dist(dx21, dy21, dz21, manhattanDistance); `, t += ` vec3 d22 = dist(dx22, dy22, dz22, manhattanDistance); `, t += ` vec3 d23 = dist(dx23, dy23, dz23, manhattanDistance); `, t += ` vec3 d31 = dist(dx31, dy31, dz31, manhattanDistance); `, t += ` vec3 d32 = dist(dx32, dy32, dz32, manhattanDistance); `, t += ` vec3 d33 = dist(dx33, dy33, dz33, manhattanDistance); `, t += ` `, t += ` vec3 d1a = min(d11, d12); `, t += ` d12 = max(d11, d12); `, t += ` d11 = min(d1a, d13); // Smallest now not in d12 or d13 `, t += ` d13 = max(d1a, d13); `, t += ` d12 = min(d12, d13); // 2nd smallest now not in d13 `, t += ` vec3 d2a = min(d21, d22); `, t += ` d22 = max(d21, d22); `, t += ` d21 = min(d2a, d23); // Smallest now not in d22 or d23 `, t += ` d23 = max(d2a, d23); `, t += ` d22 = min(d22, d23); // 2nd smallest now not in d23 `, t += ` vec3 d3a = min(d31, d32); `, t += ` d32 = max(d31, d32); `, t += ` d31 = min(d3a, d33); // Smallest now not in d32 or d33 `, t += ` d33 = max(d3a, d33); `, t += ` d32 = min(d32, d33); // 2nd smallest now not in d33 `, t += ` vec3 da = min(d11, d21); `, t += ` d21 = max(d11, d21); `, t += ` d11 = min(da, d31); // Smallest now in d11 `, t += ` d31 = max(da, d31); // 2nd smallest now not in d31 `, t += ` d11.xy = (d11.x < d11.y) ? d11.xy : d11.yx; `, t += ` d11.xz = (d11.x < d11.z) ? d11.xz : d11.zx; // d11.x now smallest `, t += ` d12 = min(d12, d21); // 2nd smallest now not in d21 `, t += ` d12 = min(d12, d22); // nor in d22 `, t += ` d12 = min(d12, d31); // nor in d31 `, t += ` d12 = min(d12, d32); // nor in d32 `, t += ` d11.yz = min(d11.yz,d12.xy); // nor in d12.yz `, t += ` d11.y = min(d11.y,d12.z); // Only two more to go `, t += ` d11.y = min(d11.y,d11.z); // Done! (Phew!) `, t += ` return sqrt(d11.xy); // F1, F2 `, t += `} `, e._emitFunction("worley3D", t, "// Worley3D"); const r = e._getFreeVariableName("worleyTemp"); return e.compilationString += `vec2 ${r} = worley(${this.seed.associatedVariableName}, ${this.jitter.associatedVariableName}, ${this.manhattanDistance}); `, this.output.hasEndpoints && (e.compilationString += this._declareOutput(this.output, e) + ` = ${r}; `), this.x.hasEndpoints && (e.compilationString += this._declareOutput(this.x, e) + ` = ${r}.x; `), this.y.hasEndpoints && (e.compilationString += this._declareOutput(this.y, e) + ` = ${r}.y; `), this; } /** * Exposes the properties to the UI? */ _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.manhattanDistance = ${this.manhattanDistance}; `; } /** * Exposes the properties to the Serialize? */ serialize() { const e = super.serialize(); return e.manhattanDistance = this.manhattanDistance, e; } /** * Exposes the properties to the deserialize? * @param serializationObject * @param scene * @param rootUrl */ _deserialize(e, t, r) { super._deserialize(e, t, r), this.manhattanDistance = e.manhattanDistance; } } C([ rn("Use Manhattan Distance", Gr.Boolean, "PROPERTIES", { notifiers: { update: !1 } }) ], aY.prototype, "manhattanDistance", void 0); Ue("BABYLON.WorleyNoise3DBlock", aY); class Gre extends Mr { /** * Creates a new SimplexPerlin3DBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("seed", de.Vector3), this.registerOutput("output", de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "SimplexPerlin3DBlock"; } /** * Gets the seed operand input component */ get seed() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { if (super._buildBlock(e), !this.seed.isConnected || !this._outputs[0].hasEndpoints) return; let t = `const float SKEWFACTOR = 1.0/3.0; `; return t += `const float UNSKEWFACTOR = 1.0/6.0; `, t += `const float SIMPLEX_CORNER_POS = 0.5; `, t += `const float SIMPLEX_TETRAHADRON_HEIGHT = 0.70710678118654752440084436210485; `, t += `float SimplexPerlin3D( vec3 P ){ `, t += ` P.x = P == vec3(0., 0., 0.) ? 0.00001 : P.x; `, t += ` P *= SIMPLEX_TETRAHADRON_HEIGHT; `, t += " vec3 Pi = floor( P + dot( P, vec3( SKEWFACTOR) ) );", t += ` vec3 x0 = P - Pi + dot(Pi, vec3( UNSKEWFACTOR ) ); `, t += ` vec3 g = step(x0.yzx, x0.xyz); `, t += ` vec3 l = 1.0 - g; `, t += ` vec3 Pi_1 = min( g.xyz, l.zxy ); `, t += ` vec3 Pi_2 = max( g.xyz, l.zxy ); `, t += ` vec3 x1 = x0 - Pi_1 + UNSKEWFACTOR; `, t += ` vec3 x2 = x0 - Pi_2 + SKEWFACTOR; `, t += ` vec3 x3 = x0 - SIMPLEX_CORNER_POS; `, t += ` vec4 v1234_x = vec4( x0.x, x1.x, x2.x, x3.x ); `, t += ` vec4 v1234_y = vec4( x0.y, x1.y, x2.y, x3.y ); `, t += ` vec4 v1234_z = vec4( x0.z, x1.z, x2.z, x3.z ); `, t += ` Pi.xyz = Pi.xyz - floor(Pi.xyz * ( 1.0 / 69.0 )) * 69.0; `, t += ` vec3 Pi_inc1 = step( Pi, vec3( 69.0 - 1.5 ) ) * ( Pi + 1.0 ); `, t += ` vec4 Pt = vec4( Pi.xy, Pi_inc1.xy ) + vec2( 50.0, 161.0 ).xyxy; `, t += ` Pt *= Pt; `, t += ` vec4 V1xy_V2xy = mix( Pt.xyxy, Pt.zwzw, vec4( Pi_1.xy, Pi_2.xy ) ); `, t += ` Pt = vec4( Pt.x, V1xy_V2xy.xz, Pt.z ) * vec4( Pt.y, V1xy_V2xy.yw, Pt.w ); `, t += ` const vec3 SOMELARGEFLOATS = vec3( 635.298681, 682.357502, 668.926525 ); `, t += ` const vec3 ZINC = vec3( 48.500388, 65.294118, 63.934599 ); `, t += ` vec3 lowz_mods = vec3( 1.0 / ( SOMELARGEFLOATS.xyz + Pi.zzz * ZINC.xyz ) ); `, t += ` vec3 highz_mods = vec3( 1.0 / ( SOMELARGEFLOATS.xyz + Pi_inc1.zzz * ZINC.xyz ) ); `, t += ` Pi_1 = ( Pi_1.z < 0.5 ) ? lowz_mods : highz_mods; `, t += ` Pi_2 = ( Pi_2.z < 0.5 ) ? lowz_mods : highz_mods; `, t += ` vec4 hash_0 = fract( Pt * vec4( lowz_mods.x, Pi_1.x, Pi_2.x, highz_mods.x ) ) - 0.49999; `, t += ` vec4 hash_1 = fract( Pt * vec4( lowz_mods.y, Pi_1.y, Pi_2.y, highz_mods.y ) ) - 0.49999; `, t += ` vec4 hash_2 = fract( Pt * vec4( lowz_mods.z, Pi_1.z, Pi_2.z, highz_mods.z ) ) - 0.49999; `, t += ` vec4 grad_results = inversesqrt( hash_0 * hash_0 + hash_1 * hash_1 + hash_2 * hash_2 ) * ( hash_0 * v1234_x + hash_1 * v1234_y + hash_2 * v1234_z ); `, t += ` const float FINAL_NORMALIZATION = 37.837227241611314102871574478976; `, t += ` vec4 kernel_weights = v1234_x * v1234_x + v1234_y * v1234_y + v1234_z * v1234_z; `, t += ` kernel_weights = max(0.5 - kernel_weights, 0.0); `, t += ` kernel_weights = kernel_weights*kernel_weights*kernel_weights; `, t += ` return dot( kernel_weights, grad_results ) * FINAL_NORMALIZATION; `, t += `} `, e._emitFunction("SimplexPerlin3D", t, "// SimplexPerlin3D"), e.compilationString += this._declareOutput(this._outputs[0], e) + ` = SimplexPerlin3D(${this.seed.associatedVariableName}); `, this; } } Ue("BABYLON.SimplexPerlin3DBlock", Gre); class Zre extends Mr { /** * Creates a new NormalBlendBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("normalMap0", de.AutoDetect), this.registerInput("normalMap1", de.AutoDetect), this.registerOutput("output", de.Vector3), this._inputs[0].addExcludedConnectionPointFromAllowedTypes(de.Color3 | de.Color4 | de.Vector3 | de.Vector4), this._inputs[1].addExcludedConnectionPointFromAllowedTypes(de.Color3 | de.Color4 | de.Vector3 | de.Vector4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "NormalBlendBlock"; } /** * Gets the first input component */ get normalMap0() { return this._inputs[0]; } /** * Gets the second input component */ get normalMap1() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0], r = this._inputs[0], n = this._inputs[1], i = e._getFreeVariableName("stepR"), s = e._getFreeVariableName("stepG"); return e.compilationString += `float ${i} = step(0.5, ${r.associatedVariableName}.r); `, e.compilationString += `float ${s} = step(0.5, ${r.associatedVariableName}.g); `, e.compilationString += this._declareOutput(t, e) + `; `, e.compilationString += `${t.associatedVariableName}.r = (1.0 - ${i}) * ${r.associatedVariableName}.r * ${n.associatedVariableName}.r * 2.0 + ${i} * (1.0 - (1.0 - ${r.associatedVariableName}.r) * (1.0 - ${n.associatedVariableName}.r) * 2.0); `, e.compilationString += `${t.associatedVariableName}.g = (1.0 - ${s}) * ${r.associatedVariableName}.g * ${n.associatedVariableName}.g * 2.0 + ${s} * (1.0 - (1.0 - ${r.associatedVariableName}.g) * (1.0 - ${n.associatedVariableName}.g) * 2.0); `, e.compilationString += `${t.associatedVariableName}.b = ${r.associatedVariableName}.b * ${n.associatedVariableName}.b; `, this; } } Ue("BABYLON.NormalBlendBlock", Zre); class _re extends Mr { /** * Creates a new Rotate2dBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("input", de.Vector2), this.registerInput("angle", de.Float), this.registerOutput("output", de.Vector2); } /** * Gets the current class name * @returns the class name */ getClassName() { return "Rotate2dBlock"; } /** * Gets the input vector */ get input() { return this._inputs[0]; } /** * Gets the input angle */ get angle() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } autoConfigure() { if (!this.angle.isConnected) { const e = new gi("angle"); e.value = 0, e.output.connectTo(this.angle); } } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0], r = this.angle, n = this.input; return e.compilationString += this._declareOutput(t, e) + ` = vec2(cos(${r.associatedVariableName}) * ${n.associatedVariableName}.x - sin(${r.associatedVariableName}) * ${n.associatedVariableName}.y, sin(${r.associatedVariableName}) * ${n.associatedVariableName}.x + cos(${r.associatedVariableName}) * ${n.associatedVariableName}.y); `, this; } } Ue("BABYLON.Rotate2dBlock", _re); class $re extends Mr { /** * Creates a new ReflectBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("incident", de.AutoDetect), this.registerInput("normal", de.AutoDetect), this.registerOutput("output", de.Vector3), this._inputs[0].addExcludedConnectionPointFromAllowedTypes(de.Vector3 | de.Vector4 | de.Color3 | de.Color4), this._inputs[1].addExcludedConnectionPointFromAllowedTypes(de.Vector3 | de.Vector4 | de.Color3 | de.Color4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ReflectBlock"; } /** * Gets the incident component */ get incident() { return this._inputs[0]; } /** * Gets the normal component */ get normal() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = reflect(${this.incident.associatedVariableName}.xyz, ${this.normal.associatedVariableName}.xyz); `, this; } } Ue("BABYLON.ReflectBlock", $re); class ene extends Mr { /** * Creates a new RefractBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("incident", de.AutoDetect), this.registerInput("normal", de.AutoDetect), this.registerInput("ior", de.Float), this.registerOutput("output", de.Vector3), this._inputs[0].addExcludedConnectionPointFromAllowedTypes(de.Vector3 | de.Vector4 | de.Color3 | de.Color4), this._inputs[1].addExcludedConnectionPointFromAllowedTypes(de.Vector3 | de.Vector4 | de.Color3 | de.Color4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "RefractBlock"; } /** * Gets the incident component */ get incident() { return this._inputs[0]; } /** * Gets the normal component */ get normal() { return this._inputs[1]; } /** * Gets the index of refraction component */ get ior() { return this._inputs[2]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = refract(${this.incident.associatedVariableName}.xyz, ${this.normal.associatedVariableName}.xyz, ${this.ior.associatedVariableName}); `, this; } } Ue("BABYLON.RefractBlock", ene); class tne extends Mr { /** * Creates a new DesaturateBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("color", de.Color3), this.registerInput("level", de.Float), this.registerOutput("output", de.Color3); } /** * Gets the current class name * @returns the class name */ getClassName() { return "DesaturateBlock"; } /** * Gets the color operand input component */ get color() { return this._inputs[0]; } /** * Gets the level operand input component */ get level() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0], n = this.color.associatedVariableName, i = e._getFreeVariableName("colorMin"), s = e._getFreeVariableName("colorMax"), a = e._getFreeVariableName("colorMerge"); return e.compilationString += `float ${i} = min(min(${n}.x, ${n}.y), ${n}.z); `, e.compilationString += `float ${s} = max(max(${n}.x, ${n}.y), ${n}.z); `, e.compilationString += `float ${a} = 0.5 * (${i} + ${s}); `, e.compilationString += this._declareOutput(t, e) + ` = mix(${n}, vec3(${a}, ${a}, ${a}), ${this.level.associatedVariableName}); `, this; } } Ue("BABYLON.DesaturateBlock", tne); class Vm extends Mr { /** * Create a new SheenBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this.albedoScaling = !1, this.linkSheenWithAlbedo = !1, this._isUnique = !0, this.registerInput("intensity", de.Float, !0, Ve.Fragment), this.registerInput("color", de.Color3, !0, Ve.Fragment), this.registerInput("roughness", de.Float, !0, Ve.Fragment), this.registerOutput("sheen", de.Object, Ve.Fragment, new Ko("sheen", this, ao.Output, Vm, "SheenBlock")); } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("sheenOut"), e._excludeVariableName("sheenMapData"), e._excludeVariableName("vSheenColor"), e._excludeVariableName("vSheenRoughness"); } /** * Gets the current class name * @returns the class name */ getClassName() { return "SheenBlock"; } /** * Gets the intensity input component */ get intensity() { return this._inputs[0]; } /** * Gets the color input component */ get color() { return this._inputs[1]; } /** * Gets the roughness input component */ get roughness() { return this._inputs[2]; } /** * Gets the sheen object output component */ get sheen() { return this._outputs[0]; } prepareDefines(e, t, r) { super.prepareDefines(e, t, r), r.setValue("SHEEN", !0), r.setValue("SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE", !0, !0), r.setValue("SHEEN_LINKWITHALBEDO", this.linkSheenWithAlbedo, !0), r.setValue("SHEEN_ROUGHNESS", this.roughness.isConnected, !0), r.setValue("SHEEN_ALBEDOSCALING", this.albedoScaling, !0); } /** * Gets the main code of the block (fragment side) * @param reflectionBlock instance of a ReflectionBlock null if the code must be generated without an active reflection module * @returns the shader code */ getCode(e) { let t = ""; const r = this.color.isConnected ? this.color.associatedVariableName : "vec3(1.)", n = this.intensity.isConnected ? this.intensity.associatedVariableName : "1.", i = this.roughness.isConnected ? this.roughness.associatedVariableName : "0."; return t = `#ifdef SHEEN sheenOutParams sheenOut; vec4 vSheenColor = vec4(${r}, ${n}); sheenBlock( vSheenColor, #ifdef SHEEN_ROUGHNESS ${i}, #endif roughness, #ifdef SHEEN_TEXTURE vec4(0.), 1.0, #endif reflectance, #ifdef SHEEN_LINKWITHALBEDO baseColor, surfaceAlbedo, #endif #ifdef ENVIRONMENTBRDF NdotV, environmentBrdf, #endif #if defined(REFLECTION) && defined(ENVIRONMENTBRDF) AARoughnessFactors, ${e == null ? void 0 : e._vReflectionMicrosurfaceInfosName}, ${e == null ? void 0 : e._vReflectionInfosName}, ${e == null ? void 0 : e.reflectionColor}, vLightingIntensity, #ifdef ${e == null ? void 0 : e._define3DName} ${e == null ? void 0 : e._cubeSamplerName}, #else ${e == null ? void 0 : e._2DSamplerName}, #endif reflectionOut.reflectionCoords, NdotVUnclamped, #ifndef LODBASEDMICROSFURACE #ifdef ${e == null ? void 0 : e._define3DName} ${e == null ? void 0 : e._cubeSamplerName}, ${e == null ? void 0 : e._cubeSamplerName}, #else ${e == null ? void 0 : e._2DSamplerName}, ${e == null ? void 0 : e._2DSamplerName}, #endif #endif #if !defined(${e == null ? void 0 : e._defineSkyboxName}) && defined(RADIANCEOCCLUSION) seo, #endif #if !defined(${e == null ? void 0 : e._defineSkyboxName}) && defined(HORIZONOCCLUSION) && defined(BUMP) && defined(${e == null ? void 0 : e._define3DName}) eho, #endif #endif sheenOut ); #ifdef SHEEN_LINKWITHALBEDO surfaceAlbedo = sheenOut.surfaceAlbedo; #endif #endif `, t; } _buildBlock(e) { return e.target === Ve.Fragment && e.sharedData.blocksWithDefines.push(this), this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return e += `${this._codeVariableName}.albedoScaling = ${this.albedoScaling}; `, e += `${this._codeVariableName}.linkSheenWithAlbedo = ${this.linkSheenWithAlbedo}; `, e; } serialize() { const e = super.serialize(); return e.albedoScaling = this.albedoScaling, e.linkSheenWithAlbedo = this.linkSheenWithAlbedo, e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.albedoScaling = e.albedoScaling, this.linkSheenWithAlbedo = e.linkSheenWithAlbedo; } } C([ rn("Albedo scaling", Gr.Boolean, "PROPERTIES", { notifiers: { update: !0 } }) ], Vm.prototype, "albedoScaling", void 0); C([ rn("Link sheen with albedo", Gr.Boolean, "PROPERTIES", { notifiers: { update: !0 } }) ], Vm.prototype, "linkSheenWithAlbedo", void 0); Ue("BABYLON.SheenBlock", Vm); class fV extends Mr { /** * Create a new AnisotropyBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this._tangentCorrectionFactorName = "", this._isUnique = !0, this.registerInput("intensity", de.Float, !0, Ve.Fragment), this.registerInput("direction", de.Vector2, !0, Ve.Fragment), this.registerInput("uv", de.Vector2, !0), this.registerInput("worldTangent", de.Vector4, !0), this.registerInput("TBN", de.Object, !0, Ve.VertexAndFragment, new Ko("TBN", this, ao.Input, Rm, "TBNBlock")), this.registerInput("roughness", de.Float, !0, Ve.Fragment), this.registerOutput("anisotropy", de.Object, Ve.Fragment, new Ko("anisotropy", this, ao.Output, fV, "AnisotropyBlock")); } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("anisotropicOut"), e._excludeVariableName("TBN"); } /** * Gets the current class name * @returns the class name */ getClassName() { return "AnisotropyBlock"; } /** * Gets the intensity input component */ get intensity() { return this._inputs[0]; } /** * Gets the direction input component */ get direction() { return this._inputs[1]; } /** * Gets the uv input component */ get uv() { return this._inputs[2]; } /** * Gets the worldTangent input component */ get worldTangent() { return this._inputs[3]; } /** * Gets the TBN input component */ // eslint-disable-next-line @typescript-eslint/naming-convention get TBN() { return this._inputs[4]; } /** * Gets the roughness input component */ get roughness() { return this._inputs[5]; } /** * Gets the anisotropy object output component */ get anisotropy() { return this._outputs[0]; } _generateTBNSpace(e) { let t = ""; const r = `//${this.name}`, n = this.uv, i = this.worldPositionConnectionPoint, s = this.worldNormalConnectionPoint, a = this.worldTangent; n.isConnected || console.error("You must connect the 'uv' input of the Anisotropy block!"), e._emitExtension("derivatives", "#extension GL_OES_standard_derivatives : enable"); const f = { search: /defined\(TANGENT\)/g, replace: a.isConnected ? "defined(TANGENT)" : "defined(IGNORE)" }, o = this.TBN; return o.isConnected ? e.compilationString += ` #ifdef TBNBLOCK mat3 vTBN = ${o.associatedVariableName}; #endif ` : a.isConnected && (t += `vec3 tbnNormal = normalize(${s.associatedVariableName}.xyz); `, t += `vec3 tbnTangent = normalize(${a.associatedVariableName}.xyz); `, t += `vec3 tbnBitangent = cross(tbnNormal, tbnTangent) * ${this._tangentCorrectionFactorName}; `, t += `mat3 vTBN = mat3(tbnTangent, tbnBitangent, tbnNormal); `), t += ` #if defined(${a.isConnected ? "TANGENT" : "IGNORE"}) && defined(NORMAL) mat3 TBN = vTBN; #else mat3 TBN = cotangent_frame(${s.associatedVariableName + ".xyz"}, ${"v_" + i.associatedVariableName + ".xyz"}, ${n.isConnected ? n.associatedVariableName : "vec2(0.)"}, vec2(1., 1.)); #endif `, e._emitFunctionFromInclude("bumpFragmentMainFunctions", r, { replaceStrings: [f] }), t; } /** * Gets the main code of the block (fragment side) * @param state current state of the node material building * @param generateTBNSpace if true, the code needed to create the TBN coordinate space is generated * @returns the shader code */ getCode(e, t = !1) { let r = ""; t && (r += this._generateTBNSpace(e)); const n = this.intensity.isConnected ? this.intensity.associatedVariableName : "1.0", i = this.direction.isConnected ? this.direction.associatedVariableName : "vec2(1., 0.)", s = this.roughness.isConnected ? this.roughness.associatedVariableName : "0."; return r += `anisotropicOutParams anisotropicOut; anisotropicBlock( vec3(${i}, ${n}), ${s}, #ifdef ANISOTROPIC_TEXTURE vec3(0.), #endif TBN, normalW, viewDirectionW, anisotropicOut ); `, r; } prepareDefines(e, t, r) { super.prepareDefines(e, t, r), r.setValue("ANISOTROPIC", !0), r.setValue("ANISOTROPIC_TEXTURE", !1, !0), r.setValue("ANISOTROPIC_LEGACY", !this.roughness.isConnected); } bind(e, t, r) { super.bind(e, t, r), r && e.setFloat(this._tangentCorrectionFactorName, r.getWorldMatrix().determinant() < 0 ? -1 : 1); } _buildBlock(e) { return e.target === Ve.Fragment && (e.sharedData.blocksWithDefines.push(this), e.sharedData.bindableBlocks.push(this), this._tangentCorrectionFactorName = e._getFreeDefineName("tangentCorrectionFactor"), e._emitUniformFromString(this._tangentCorrectionFactorName, "float")), this; } } Ue("BABYLON.AnisotropyBlock", fV); class Cm extends $I { _onGenerateOnlyFragmentCodeChanged() { return this.position.isConnected ? (this.generateOnlyFragmentCode = !this.generateOnlyFragmentCode, console.error("The position input must not be connected to be able to switch!"), !1) : (this._setTarget(), !0); } _setTarget() { super._setTarget(), this.getInputByName("position").target = this.generateOnlyFragmentCode ? Ve.Fragment : Ve.Vertex, this.generateOnlyFragmentCode && (this.forceIrradianceInFragment = !0); } /** * Create a new ReflectionBlock * @param name defines the block name */ constructor(e) { super(e), this.useSphericalHarmonics = !0, this.forceIrradianceInFragment = !1, this._isUnique = !0, this.registerInput("position", de.AutoDetect, !1, Ve.Vertex), this.registerInput("world", de.Matrix, !1, Ve.Vertex), this.registerInput("color", de.Color3, !0, Ve.Fragment), this.registerOutput("reflection", de.Object, Ve.Fragment, new Ko("reflection", this, ao.Output, Cm, "ReflectionBlock")), this.position.addExcludedConnectionPointFromAllowedTypes(de.Color3 | de.Vector3 | de.Vector4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ReflectionBlock"; } /** * Gets the position input component */ get position() { return this._inputs[0]; } /** * Gets the world position input component */ get worldPosition() { return this.worldPositionConnectionPoint; } /** * Gets the world normal input component */ get worldNormal() { return this.worldNormalConnectionPoint; } /** * Gets the world input component */ get world() { return this._inputs[1]; } /** * Gets the camera (or eye) position component */ get cameraPosition() { return this.cameraPositionConnectionPoint; } /** * Gets the view input component */ get view() { return this.viewConnectionPoint; } /** * Gets the color input component */ get color() { return this._inputs[2]; } /** * Gets the reflection object output component */ get reflection() { return this._outputs[0]; } /** * Returns true if the block has a texture (either its own texture or the environment texture from the scene, if set) */ get hasTexture() { return !!this._getTexture(); } /** * Gets the reflection color (either the name of the variable if the color input is connected, else a default value) */ get reflectionColor() { return this.color.isConnected ? this.color.associatedVariableName : "vec3(1., 1., 1.)"; } _getTexture() { return this.texture ? this.texture : this._scene.environmentTexture; } prepareDefines(e, t, r) { super.prepareDefines(e, t, r); const n = this._getTexture(), i = n && n.getTextureMatrix; r.setValue("REFLECTION", i, !0), i && (r.setValue(this._defineLODReflectionAlpha, n.lodLevelInAlpha, !0), r.setValue(this._defineLinearSpecularReflection, n.linearSpecularLOD, !0), r.setValue(this._defineOppositeZ, this._scene.useRightHandedSystem ? !n.invertZ : n.invertZ, !0), r.setValue("SPHERICAL_HARMONICS", this.useSphericalHarmonics, !0), r.setValue("GAMMAREFLECTION", n.gammaSpace, !0), r.setValue("RGBDREFLECTION", n.isRGBD, !0), n && n.coordinatesMode !== We.SKYBOX_MODE && n.isCube && (r.setValue("USESPHERICALFROMREFLECTIONMAP", !0), r.setValue("USEIRRADIANCEMAP", !1), this.forceIrradianceInFragment || this._scene.getEngine().getCaps().maxVaryingVectors <= 8 ? r.setValue("USESPHERICALINVERTEX", !1) : r.setValue("USESPHERICALINVERTEX", !0))); } bind(e, t, r, n) { super.bind(e, t, r); const i = this._getTexture(); if (!i || !n) return; i.isCube ? e.setTexture(this._cubeSamplerName, i) : e.setTexture(this._2DSamplerName, i); const s = i.getSize().width; e.setFloat3(this._vReflectionMicrosurfaceInfosName, s, i.lodGenerationScale, i.lodGenerationOffset), e.setFloat2(this._vReflectionFilteringInfoName, s, Xt.Log2(s)); const a = n.materialDefines, f = i.sphericalPolynomial; if (a.USESPHERICALFROMREFLECTIONMAP && f) if (a.SPHERICAL_HARMONICS) { const o = f.preScaledHarmonics; e.setVector3("vSphericalL00", o.l00), e.setVector3("vSphericalL1_1", o.l1_1), e.setVector3("vSphericalL10", o.l10), e.setVector3("vSphericalL11", o.l11), e.setVector3("vSphericalL2_2", o.l2_2), e.setVector3("vSphericalL2_1", o.l2_1), e.setVector3("vSphericalL20", o.l20), e.setVector3("vSphericalL21", o.l21), e.setVector3("vSphericalL22", o.l22); } else e.setFloat3("vSphericalX", f.x.x, f.x.y, f.x.z), e.setFloat3("vSphericalY", f.y.x, f.y.y, f.y.z), e.setFloat3("vSphericalZ", f.z.x, f.z.y, f.z.z), e.setFloat3("vSphericalXX_ZZ", f.xx.x - f.zz.x, f.xx.y - f.zz.y, f.xx.z - f.zz.z), e.setFloat3("vSphericalYY_ZZ", f.yy.x - f.zz.x, f.yy.y - f.zz.y, f.yy.z - f.zz.z), e.setFloat3("vSphericalZZ", f.zz.x, f.zz.y, f.zz.z), e.setFloat3("vSphericalXY", f.xy.x, f.xy.y, f.xy.z), e.setFloat3("vSphericalYZ", f.yz.x, f.yz.y, f.yz.z), e.setFloat3("vSphericalZX", f.zx.x, f.zx.y, f.zx.z); } /** * Gets the code to inject in the vertex shader * @param state current state of the node material building * @returns the shader code */ handleVertexSide(e) { let t = super.handleVertexSide(e); e._emitFunctionFromInclude("harmonicsFunctions", `//${this.name}`, { replaceStrings: [ { search: /uniform vec3 vSphericalL00;[\s\S]*?uniform vec3 vSphericalL22;/g, replace: "" }, { search: /uniform vec3 vSphericalX;[\s\S]*?uniform vec3 vSphericalZX;/g, replace: "" } ] }); const r = e._getFreeVariableName("reflectionVector"); return this._vEnvironmentIrradianceName = e._getFreeVariableName("vEnvironmentIrradiance"), e._emitVaryingFromString(this._vEnvironmentIrradianceName, "vec3", "defined(USESPHERICALFROMREFLECTIONMAP) && defined(USESPHERICALINVERTEX)"), e._emitUniformFromString("vSphericalL00", "vec3", "SPHERICAL_HARMONICS"), e._emitUniformFromString("vSphericalL1_1", "vec3", "SPHERICAL_HARMONICS"), e._emitUniformFromString("vSphericalL10", "vec3", "SPHERICAL_HARMONICS"), e._emitUniformFromString("vSphericalL11", "vec3", "SPHERICAL_HARMONICS"), e._emitUniformFromString("vSphericalL2_2", "vec3", "SPHERICAL_HARMONICS"), e._emitUniformFromString("vSphericalL2_1", "vec3", "SPHERICAL_HARMONICS"), e._emitUniformFromString("vSphericalL20", "vec3", "SPHERICAL_HARMONICS"), e._emitUniformFromString("vSphericalL21", "vec3", "SPHERICAL_HARMONICS"), e._emitUniformFromString("vSphericalL22", "vec3", "SPHERICAL_HARMONICS"), e._emitUniformFromString("vSphericalX", "vec3", "SPHERICAL_HARMONICS", !0), e._emitUniformFromString("vSphericalY", "vec3", "SPHERICAL_HARMONICS", !0), e._emitUniformFromString("vSphericalZ", "vec3", "SPHERICAL_HARMONICS", !0), e._emitUniformFromString("vSphericalXX_ZZ", "vec3", "SPHERICAL_HARMONICS", !0), e._emitUniformFromString("vSphericalYY_ZZ", "vec3", "SPHERICAL_HARMONICS", !0), e._emitUniformFromString("vSphericalZZ", "vec3", "SPHERICAL_HARMONICS", !0), e._emitUniformFromString("vSphericalXY", "vec3", "SPHERICAL_HARMONICS", !0), e._emitUniformFromString("vSphericalYZ", "vec3", "SPHERICAL_HARMONICS", !0), e._emitUniformFromString("vSphericalZX", "vec3", "SPHERICAL_HARMONICS", !0), t += `#if defined(USESPHERICALFROMREFLECTIONMAP) && defined(USESPHERICALINVERTEX) vec3 ${r} = vec3(${this._reflectionMatrixName} * vec4(normalize(${this.worldNormal.associatedVariableName}).xyz, 0)).xyz; #ifdef ${this._defineOppositeZ} ${r}.z *= -1.0; #endif ${this._vEnvironmentIrradianceName} = computeEnvironmentIrradiance(${r}); #endif `, t; } /** * Gets the main code of the block (fragment side) * @param state current state of the node material building * @param normalVarName name of the existing variable corresponding to the normal * @returns the shader code */ getCode(e, t) { let r = ""; this.handleFragmentSideInits(e), e._emitFunctionFromInclude("harmonicsFunctions", `//${this.name}`, { replaceStrings: [ { search: /uniform vec3 vSphericalL00;[\s\S]*?uniform vec3 vSphericalL22;/g, replace: "" }, { search: /uniform vec3 vSphericalX;[\s\S]*?uniform vec3 vSphericalZX;/g, replace: "" } ] }), e._emitFunction("sampleReflection", ` #ifdef ${this._define3DName} #define sampleReflection(s, c) textureCube(s, c) #else #define sampleReflection(s, c) texture2D(s, c) #endif `, `//${this.name}`), e._emitFunction("sampleReflectionLod", ` #ifdef ${this._define3DName} #define sampleReflectionLod(s, c, l) textureCubeLodEXT(s, c, l) #else #define sampleReflectionLod(s, c, l) texture2DLodEXT(s, c, l) #endif `, `//${this.name}`); const n = ` vec3 computeReflectionCoordsPBR(vec4 worldPos, vec3 worldNormal) { ${this.handleFragmentSideCodeReflectionCoords("worldNormal", "worldPos", !0, !0)} return ${this._reflectionVectorName}; } `; return e._emitFunction("computeReflectionCoordsPBR", n, `//${this.name}`), this._vReflectionMicrosurfaceInfosName = e._getFreeVariableName("vReflectionMicrosurfaceInfos"), e._emitUniformFromString(this._vReflectionMicrosurfaceInfosName, "vec3"), this._vReflectionInfosName = e._getFreeVariableName("vReflectionInfos"), this._vReflectionFilteringInfoName = e._getFreeVariableName("vReflectionFilteringInfo"), e._emitUniformFromString(this._vReflectionFilteringInfoName, "vec2"), r += `#ifdef REFLECTION vec2 ${this._vReflectionInfosName} = vec2(1., 0.); reflectionOutParams reflectionOut; reflectionBlock( ${this.generateOnlyFragmentCode ? this._worldPositionNameInFragmentOnlyMode : "v_" + this.worldPosition.associatedVariableName}.xyz, ${t}, alphaG, ${this._vReflectionMicrosurfaceInfosName}, ${this._vReflectionInfosName}, ${this.reflectionColor}, #ifdef ANISOTROPIC anisotropicOut, #endif #if defined(${this._defineLODReflectionAlpha}) && !defined(${this._defineSkyboxName}) NdotVUnclamped, #endif #ifdef ${this._defineLinearSpecularReflection} roughness, #endif #ifdef ${this._define3DName} ${this._cubeSamplerName}, #else ${this._2DSamplerName}, #endif #if defined(NORMAL) && defined(USESPHERICALINVERTEX) ${this._vEnvironmentIrradianceName}, #endif #ifdef USESPHERICALFROMREFLECTIONMAP #if !defined(NORMAL) || !defined(USESPHERICALINVERTEX) ${this._reflectionMatrixName}, #endif #endif #ifdef USEIRRADIANCEMAP irradianceSampler, // ** not handled ** #endif #ifndef LODBASEDMICROSFURACE #ifdef ${this._define3DName} ${this._cubeSamplerName}, ${this._cubeSamplerName}, #else ${this._2DSamplerName}, ${this._2DSamplerName}, #endif #endif #ifdef REALTIME_FILTERING ${this._vReflectionFilteringInfoName}, #endif reflectionOut ); #endif `, r; } _buildBlock(e) { return this._scene = e.sharedData.scene, e.target !== Ve.Fragment && (this._defineLODReflectionAlpha = e._getFreeDefineName("LODINREFLECTIONALPHA"), this._defineLinearSpecularReflection = e._getFreeDefineName("LINEARSPECULARREFLECTION")), this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return this.texture && (e += `${this._codeVariableName}.texture.gammaSpace = ${this.texture.gammaSpace}; `), e += `${this._codeVariableName}.useSphericalHarmonics = ${this.useSphericalHarmonics}; `, e += `${this._codeVariableName}.forceIrradianceInFragment = ${this.forceIrradianceInFragment}; `, e; } serialize() { var e, t; const r = super.serialize(); return r.useSphericalHarmonics = this.useSphericalHarmonics, r.forceIrradianceInFragment = this.forceIrradianceInFragment, r.gammaSpace = (t = (e = this.texture) === null || e === void 0 ? void 0 : e.gammaSpace) !== null && t !== void 0 ? t : !0, r; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.useSphericalHarmonics = e.useSphericalHarmonics, this.forceIrradianceInFragment = e.forceIrradianceInFragment, this.texture && (this.texture.gammaSpace = e.gammaSpace); } } C([ rn("Spherical Harmonics", Gr.Boolean, "ADVANCED", { notifiers: { update: !0 } }) ], Cm.prototype, "useSphericalHarmonics", void 0); C([ rn("Force irradiance in fragment", Gr.Boolean, "ADVANCED", { notifiers: { update: !0 } }) ], Cm.prototype, "forceIrradianceInFragment", void 0); Ue("BABYLON.ReflectionBlock", Cm); class Xm extends Mr { /** * Create a new ClearCoatBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this._tangentCorrectionFactorName = "", this.remapF0OnInterfaceChange = !0, this._isUnique = !0, this.registerInput("intensity", de.Float, !1, Ve.Fragment), this.registerInput("roughness", de.Float, !0, Ve.Fragment), this.registerInput("indexOfRefraction", de.Float, !0, Ve.Fragment), this.registerInput("normalMapColor", de.Color3, !0, Ve.Fragment), this.registerInput("uv", de.Vector2, !0, Ve.Fragment), this.registerInput("tintColor", de.Color3, !0, Ve.Fragment), this.registerInput("tintAtDistance", de.Float, !0, Ve.Fragment), this.registerInput("tintThickness", de.Float, !0, Ve.Fragment), this.registerInput("worldTangent", de.Vector4, !0), this.registerInput("worldNormal", de.AutoDetect, !0), this.worldNormal.addExcludedConnectionPointFromAllowedTypes(de.Color4 | de.Vector4 | de.Vector3), this.registerInput("TBN", de.Object, !0, Ve.VertexAndFragment, new Ko("TBN", this, ao.Input, Rm, "TBNBlock")), this.registerOutput("clearcoat", de.Object, Ve.Fragment, new Ko("clearcoat", this, ao.Output, Xm, "ClearCoatBlock")); } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("clearcoatOut"), e._excludeVariableName("vClearCoatParams"), e._excludeVariableName("vClearCoatTintParams"), e._excludeVariableName("vClearCoatRefractionParams"), e._excludeVariableName("vClearCoatTangentSpaceParams"), e._excludeVariableName("vGeometricNormaClearCoatW"); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ClearCoatBlock"; } /** * Gets the intensity input component */ get intensity() { return this._inputs[0]; } /** * Gets the roughness input component */ get roughness() { return this._inputs[1]; } /** * Gets the ior input component */ get indexOfRefraction() { return this._inputs[2]; } /** * Gets the bump texture input component */ get normalMapColor() { return this._inputs[3]; } /** * Gets the uv input component */ get uv() { return this._inputs[4]; } /** * Gets the tint color input component */ get tintColor() { return this._inputs[5]; } /** * Gets the tint "at distance" input component */ get tintAtDistance() { return this._inputs[6]; } /** * Gets the tint thickness input component */ get tintThickness() { return this._inputs[7]; } /** * Gets the world tangent input component */ get worldTangent() { return this._inputs[8]; } /** * Gets the world normal input component */ get worldNormal() { return this._inputs[9]; } /** * Gets the TBN input component */ // eslint-disable-next-line @typescript-eslint/naming-convention get TBN() { return this._inputs[10]; } /** * Gets the clear coat object output component */ get clearcoat() { return this._outputs[0]; } autoConfigure() { if (!this.intensity.isConnected) { const e = new gi("ClearCoat intensity", Ve.Fragment, de.Float); e.value = 1, e.output.connectTo(this.intensity); } } prepareDefines(e, t, r) { super.prepareDefines(e, t, r), r.setValue("CLEARCOAT", !0), r.setValue("CLEARCOAT_TEXTURE", !1, !0), r.setValue("CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE", !0, !0), r.setValue("CLEARCOAT_TINT", this.tintColor.isConnected || this.tintThickness.isConnected || this.tintAtDistance.isConnected, !0), r.setValue("CLEARCOAT_BUMP", this.normalMapColor.isConnected, !0), r.setValue("CLEARCOAT_DEFAULTIOR", this.indexOfRefraction.isConnected ? this.indexOfRefraction.connectInputBlock.value === h9._DefaultIndexOfRefraction : !0, !0), r.setValue("CLEARCOAT_REMAP_F0", this.remapF0OnInterfaceChange, !0); } bind(e, t, r) { var n, i; super.bind(e, t, r); const s = (i = (n = this.indexOfRefraction.connectInputBlock) === null || n === void 0 ? void 0 : n.value) !== null && i !== void 0 ? i : h9._DefaultIndexOfRefraction, a = 1 - s, f = 1 + s, o = Math.pow(-a / f, 2), d = 1 / s; e.setFloat4("vClearCoatRefractionParams", o, d, a, f); const v = this.clearcoat.hasEndpoints ? this.clearcoat.endpoints[0].ownerBlock : null, u = v != null && v.perturbedNormal.isConnected ? v.perturbedNormal.connectedPoint.ownerBlock : null; this._scene._mirroredCameraPosition ? e.setFloat2("vClearCoatTangentSpaceParams", u != null && u.invertX ? 1 : -1, u != null && u.invertY ? 1 : -1) : e.setFloat2("vClearCoatTangentSpaceParams", u != null && u.invertX ? -1 : 1, u != null && u.invertY ? -1 : 1), r && e.setFloat(this._tangentCorrectionFactorName, r.getWorldMatrix().determinant() < 0 ? -1 : 1); } _generateTBNSpace(e, t, r) { let n = ""; const i = `//${this.name}`, s = this.worldTangent; e._emitExtension("derivatives", "#extension GL_OES_standard_derivatives : enable"); const a = { search: /defined\(TANGENT\)/g, replace: s.isConnected ? "defined(TANGENT)" : "defined(IGNORE)" }, f = this.TBN; return f.isConnected ? e.compilationString += ` #ifdef TBNBLOCK mat3 vTBN = ${f.associatedVariableName}; #endif ` : s.isConnected && (n += `vec3 tbnNormal = normalize(${r}.xyz); `, n += `vec3 tbnTangent = normalize(${s.associatedVariableName}.xyz); `, n += `vec3 tbnBitangent = cross(tbnNormal, tbnTangent) * ${this._tangentCorrectionFactorName}; `, n += `mat3 vTBN = mat3(tbnTangent, tbnBitangent, tbnNormal); `), e._emitFunctionFromInclude("bumpFragmentMainFunctions", i, { replaceStrings: [a] }), n; } /** * Gets the main code of the block (fragment side) * @param state current state of the node material building * @param ccBlock instance of a ClearCoatBlock or null if the code must be generated without an active clear coat module * @param reflectionBlock instance of a ReflectionBlock null if the code must be generated without an active reflection module * @param worldPosVarName name of the variable holding the world position * @param generateTBNSpace if true, the code needed to create the TBN coordinate space is generated * @param vTBNAvailable indicate that the vTBN variable is already existing because it has already been generated by another block (PerturbNormal or Anisotropy) * @param worldNormalVarName name of the variable holding the world normal * @returns the shader code */ static GetCode(e, t, r, n, i, s, a) { let f = ""; const o = t != null && t.intensity.isConnected ? t.intensity.associatedVariableName : "1.", d = t != null && t.roughness.isConnected ? t.roughness.associatedVariableName : "0.", v = t != null && t.normalMapColor.isConnected ? t.normalMapColor.associatedVariableName : "vec3(0.)", u = t != null && t.uv.isConnected ? t.uv.associatedVariableName : "vec2(0.)", l = t != null && t.tintColor.isConnected ? t.tintColor.associatedVariableName : "vec3(1.)", P = t != null && t.tintThickness.isConnected ? t.tintThickness.associatedVariableName : "1.", p = t != null && t.tintAtDistance.isConnected ? t.tintAtDistance.associatedVariableName : "1.", c = "vec4(0.)"; if (t) { e._emitUniformFromString("vClearCoatRefractionParams", "vec4"), e._emitUniformFromString("vClearCoatTangentSpaceParams", "vec2"); const H = t.worldNormal; f += `vec3 vGeometricNormaClearCoatW = ${H.isConnected ? "normalize(" + H.associatedVariableName + ".xyz)" : "geometricNormalW"}; `; } else f += `vec3 vGeometricNormaClearCoatW = geometricNormalW; `; return i && t && (f += t._generateTBNSpace(e, n, a), s = t.worldTangent.isConnected), f += `clearcoatOutParams clearcoatOut; #ifdef CLEARCOAT vec2 vClearCoatParams = vec2(${o}, ${d}); vec4 vClearCoatTintParams = vec4(${l}, ${P}); clearcoatBlock( ${n}.xyz, vGeometricNormaClearCoatW, viewDirectionW, vClearCoatParams, specularEnvironmentR0, #ifdef CLEARCOAT_TEXTURE vec2(0.), #endif #ifdef CLEARCOAT_TINT vClearCoatTintParams, ${p}, vClearCoatRefractionParams, #ifdef CLEARCOAT_TINT_TEXTURE ${c}, #endif #endif #ifdef CLEARCOAT_BUMP vec2(0., 1.), vec4(${v}, 0.), ${u}, #if defined(${s ? "TANGENT" : "IGNORE"}) && defined(NORMAL) vTBN, #else vClearCoatTangentSpaceParams, #endif #ifdef OBJECTSPACE_NORMALMAP normalMatrix, #endif #endif #if defined(FORCENORMALFORWARD) && defined(NORMAL) faceNormal, #endif #ifdef REFLECTION ${r == null ? void 0 : r._vReflectionMicrosurfaceInfosName}, ${r == null ? void 0 : r._vReflectionInfosName}, ${r == null ? void 0 : r.reflectionColor}, vLightingIntensity, #ifdef ${r == null ? void 0 : r._define3DName} ${r == null ? void 0 : r._cubeSamplerName}, #else ${r == null ? void 0 : r._2DSamplerName}, #endif #ifndef LODBASEDMICROSFURACE #ifdef ${r == null ? void 0 : r._define3DName} ${r == null ? void 0 : r._cubeSamplerName}, ${r == null ? void 0 : r._cubeSamplerName}, #else ${r == null ? void 0 : r._2DSamplerName}, ${r == null ? void 0 : r._2DSamplerName}, #endif #endif #endif #if defined(ENVIRONMENTBRDF) && !defined(${r == null ? void 0 : r._defineSkyboxName}) #ifdef RADIANCEOCCLUSION ambientMonochrome, #endif #endif #if defined(CLEARCOAT_BUMP) || defined(TWOSIDEDLIGHTING) (gl_FrontFacing ? 1. : -1.), #endif clearcoatOut ); #else clearcoatOut.specularEnvironmentR0 = specularEnvironmentR0; #endif `, f; } _buildBlock(e) { return this._scene = e.sharedData.scene, e.target === Ve.Fragment && (e.sharedData.bindableBlocks.push(this), e.sharedData.blocksWithDefines.push(this), this._tangentCorrectionFactorName = e._getFreeDefineName("tangentCorrectionFactor"), e._emitUniformFromString(this._tangentCorrectionFactorName, "float")), this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return e += `${this._codeVariableName}.remapF0OnInterfaceChange = ${this.remapF0OnInterfaceChange}; `, e; } serialize() { const e = super.serialize(); return e.remapF0OnInterfaceChange = this.remapF0OnInterfaceChange, e; } _deserialize(e, t, r) { var n; super._deserialize(e, t, r), this.remapF0OnInterfaceChange = (n = e.remapF0OnInterfaceChange) !== null && n !== void 0 ? n : !0; } } C([ rn("Remap F0 on interface change", Gr.Boolean, "ADVANCED") ], Xm.prototype, "remapF0OnInterfaceChange", void 0); Ue("BABYLON.ClearCoatBlock", Xm); class eR extends Mr { /** * Create a new IridescenceBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this._isUnique = !0, this.registerInput("intensity", de.Float, !0, Ve.Fragment), this.registerInput("indexOfRefraction", de.Float, !0, Ve.Fragment), this.registerInput("thickness", de.Float, !0, Ve.Fragment), this.registerOutput("iridescence", de.Object, Ve.Fragment, new Ko("iridescence", this, ao.Output, eR, "IridescenceBlock")); } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("iridescenceOut"), e._excludeVariableName("vIridescenceParams"); } /** * Gets the current class name * @returns the class name */ getClassName() { return "IridescenceBlock"; } /** * Gets the intensity input component */ get intensity() { return this._inputs[0]; } /** * Gets the indexOfRefraction input component */ get indexOfRefraction() { return this._inputs[1]; } /** * Gets the thickness input component */ get thickness() { return this._inputs[2]; } /** * Gets the iridescence object output component */ get iridescence() { return this._outputs[0]; } autoConfigure() { if (!this.intensity.isConnected) { const e = new gi("Iridescence intensity", Ve.Fragment, de.Float); e.value = 1, e.output.connectTo(this.intensity); const t = new gi("Iridescence ior", Ve.Fragment, de.Float); t.value = 1.3, t.output.connectTo(this.indexOfRefraction); const r = new gi("Iridescence thickness", Ve.Fragment, de.Float); r.value = 400, r.output.connectTo(this.thickness); } } prepareDefines(e, t, r) { super.prepareDefines(e, t, r), r.setValue("IRIDESCENCE", !0, !0), r.setValue("IRIDESCENCE_TEXTURE", !1, !0), r.setValue("IRIDESCENCE_THICKNESS_TEXTURE", !1, !0); } /** * Gets the main code of the block (fragment side) * @param iridescenceBlock instance of a IridescenceBlock or null if the code must be generated without an active iridescence module * @returns the shader code */ static GetCode(e) { let t = ""; const r = e != null && e.intensity.isConnected ? e.intensity.associatedVariableName : "1.", n = e != null && e.indexOfRefraction.isConnected ? e.indexOfRefraction.associatedVariableName : dd._DefaultIndexOfRefraction, i = e != null && e.thickness.isConnected ? e.thickness.associatedVariableName : dd._DefaultMaximumThickness; return t += `iridescenceOutParams iridescenceOut; #ifdef IRIDESCENCE iridescenceBlock( vec4(${r}, ${n}, 1., ${i}), NdotV, specularEnvironmentR0, #ifdef CLEARCOAT NdotVUnclamped, #endif iridescenceOut ); float iridescenceIntensity = iridescenceOut.iridescenceIntensity; specularEnvironmentR0 = iridescenceOut.specularEnvironmentR0; #endif `, t; } _buildBlock(e) { return e.target === Ve.Fragment && (e.sharedData.bindableBlocks.push(this), e.sharedData.blocksWithDefines.push(this)), this; } serialize() { return super.serialize(); } _deserialize(e, t, r) { super._deserialize(e, t, r); } } Ue("BABYLON.IridescenceBlock", eR); class wD extends Mr { /** * Create a new RefractionBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this.linkRefractionWithTransparency = !1, this.invertRefractionY = !1, this.useThicknessAsDepth = !1, this._isUnique = !0, this.registerInput("intensity", de.Float, !1, Ve.Fragment), this.registerInput("tintAtDistance", de.Float, !0, Ve.Fragment), this.registerInput("volumeIndexOfRefraction", de.Float, !0, Ve.Fragment), this.registerOutput("refraction", de.Object, Ve.Fragment, new Ko("refraction", this, ao.Output, wD, "RefractionBlock")); } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("vRefractionPosition"), e._excludeVariableName("vRefractionSize"); } /** * Gets the current class name * @returns the class name */ getClassName() { return "RefractionBlock"; } /** * Gets the intensity input component */ get intensity() { return this._inputs[0]; } /** * Gets the tint at distance input component */ get tintAtDistance() { return this._inputs[1]; } /** * Gets the volume index of refraction input component */ get volumeIndexOfRefraction() { return this._inputs[2]; } /** * Gets the view input component */ get view() { return this.viewConnectionPoint; } /** * Gets the refraction object output component */ get refraction() { return this._outputs[0]; } /** * Returns true if the block has a texture */ get hasTexture() { return !!this._getTexture(); } _getTexture() { return this.texture ? this.texture : this._scene.environmentTexture; } autoConfigure(e, t = () => !0) { if (!this.intensity.isConnected) { const r = new gi("Refraction intensity", Ve.Fragment, de.Float); r.value = 1, r.output.connectTo(this.intensity); } if (this.view && !this.view.isConnected) { let r = e.getInputBlockByPredicate((n) => n.systemValue === Bi.View && t(n)); r || (r = new gi("view"), r.setAsSystemValue(Bi.View)), r.output.connectTo(this.view); } } prepareDefines(e, t, r) { super.prepareDefines(e, t, r); const n = this._getTexture(), i = n && n.getTextureMatrix; r.setValue("SS_REFRACTION", i, !0), i && (r.setValue(this._define3DName, n.isCube, !0), r.setValue(this._defineLODRefractionAlpha, n.lodLevelInAlpha, !0), r.setValue(this._defineLinearSpecularRefraction, n.linearSpecularLOD, !0), r.setValue(this._defineOppositeZ, this._scene.useRightHandedSystem && n.isCube ? !n.invertZ : n.invertZ, !0), r.setValue("SS_LINKREFRACTIONTOTRANSPARENCY", this.linkRefractionWithTransparency, !0), r.setValue("SS_GAMMAREFRACTION", n.gammaSpace, !0), r.setValue("SS_RGBDREFRACTION", n.isRGBD, !0), r.setValue("SS_USE_LOCAL_REFRACTIONMAP_CUBIC", !!n.boundingBoxSize, !0), r.setValue("SS_USE_THICKNESS_AS_DEPTH", this.useThicknessAsDepth, !0)); } isReady() { const e = this._getTexture(); return !(e && !e.isReadyOrNotBlocking()); } bind(e, t, r) { var n, i, s, a; super.bind(e, t, r); const f = this._getTexture(); if (!f) return; f.isCube ? e.setTexture(this._cubeSamplerName, f) : e.setTexture(this._2DSamplerName, f), e.setMatrix(this._refractionMatrixName, f.getRefractionTextureMatrix()); let o = 1; f.isCube || f.depth && (o = f.depth); const d = (a = (i = (n = this.volumeIndexOfRefraction.connectInputBlock) === null || n === void 0 ? void 0 : n.value) !== null && i !== void 0 ? i : (s = this.indexOfRefractionConnectionPoint.connectInputBlock) === null || s === void 0 ? void 0 : s.value) !== null && a !== void 0 ? a : 1.5; e.setFloat4(this._vRefractionInfosName, f.level, 1 / d, o, this.invertRefractionY ? -1 : 1), e.setFloat4(this._vRefractionMicrosurfaceInfosName, f.getSize().width, f.lodGenerationScale, f.lodGenerationOffset, 1 / d); const v = f.getSize().width; if (e.setFloat2(this._vRefractionFilteringInfoName, v, Xt.Log2(v)), f.boundingBoxSize) { const u = f; e.setVector3("vRefractionPosition", u.boundingBoxPosition), e.setVector3("vRefractionSize", u.boundingBoxSize); } } /** * Gets the main code of the block (fragment side) * @param state current state of the node material building * @returns the shader code */ getCode(e) { const t = ""; return e.sharedData.blockingBlocks.push(this), e.sharedData.textureBlocks.push(this), this._cubeSamplerName = e._getFreeVariableName(this.name + "CubeSampler"), e.samplers.push(this._cubeSamplerName), this._2DSamplerName = e._getFreeVariableName(this.name + "2DSampler"), e.samplers.push(this._2DSamplerName), this._define3DName = e._getFreeDefineName("SS_REFRACTIONMAP_3D"), e._samplerDeclaration += `#ifdef ${this._define3DName} `, e._samplerDeclaration += `uniform samplerCube ${this._cubeSamplerName}; `, e._samplerDeclaration += `#else `, e._samplerDeclaration += `uniform sampler2D ${this._2DSamplerName}; `, e._samplerDeclaration += `#endif `, e.sharedData.blocksWithDefines.push(this), e.sharedData.bindableBlocks.push(this), this._defineLODRefractionAlpha = e._getFreeDefineName("SS_LODINREFRACTIONALPHA"), this._defineLinearSpecularRefraction = e._getFreeDefineName("SS_LINEARSPECULARREFRACTION"), this._defineOppositeZ = e._getFreeDefineName("SS_REFRACTIONMAP_OPPOSITEZ"), this._refractionMatrixName = e._getFreeVariableName("refractionMatrix"), e._emitUniformFromString(this._refractionMatrixName, "mat4"), e._emitFunction("sampleRefraction", ` #ifdef ${this._define3DName} #define sampleRefraction(s, c) textureCube(s, c) #else #define sampleRefraction(s, c) texture2D(s, c) #endif `, `//${this.name}`), e._emitFunction("sampleRefractionLod", ` #ifdef ${this._define3DName} #define sampleRefractionLod(s, c, l) textureCubeLodEXT(s, c, l) #else #define sampleRefractionLod(s, c, l) texture2DLodEXT(s, c, l) #endif `, `//${this.name}`), this._vRefractionMicrosurfaceInfosName = e._getFreeVariableName("vRefractionMicrosurfaceInfos"), e._emitUniformFromString(this._vRefractionMicrosurfaceInfosName, "vec4"), this._vRefractionInfosName = e._getFreeVariableName("vRefractionInfos"), e._emitUniformFromString(this._vRefractionInfosName, "vec4"), this._vRefractionFilteringInfoName = e._getFreeVariableName("vRefractionFilteringInfo"), e._emitUniformFromString(this._vRefractionFilteringInfoName, "vec2"), e._emitUniformFromString("vRefractionPosition", "vec3"), e._emitUniformFromString("vRefractionSize", "vec3"), t; } _buildBlock(e) { return this._scene = e.sharedData.scene, this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return this.texture && (this.texture.isCube ? e = `${this._codeVariableName}.texture = new BABYLON.CubeTexture("${this.texture.name}"); ` : e = `${this._codeVariableName}.texture = new BABYLON.Texture("${this.texture.name}"); `, e += `${this._codeVariableName}.texture.coordinatesMode = ${this.texture.coordinatesMode}; `), e += `${this._codeVariableName}.linkRefractionWithTransparency = ${this.linkRefractionWithTransparency}; `, e += `${this._codeVariableName}.invertRefractionY = ${this.invertRefractionY}; `, e += `${this._codeVariableName}.useThicknessAsDepth = ${this.useThicknessAsDepth}; `, e; } serialize() { const e = super.serialize(); return this.texture && !this.texture.isRenderTarget && (e.texture = this.texture.serialize()), e.linkRefractionWithTransparency = this.linkRefractionWithTransparency, e.invertRefractionY = this.invertRefractionY, e.useThicknessAsDepth = this.useThicknessAsDepth, e; } _deserialize(e, t, r) { super._deserialize(e, t, r), e.texture && (r = e.texture.url.indexOf("data:") === 0 ? "" : r, e.texture.isCube ? this.texture = v1.Parse(e.texture, t, r) : this.texture = We.Parse(e.texture, t, r)), this.linkRefractionWithTransparency = e.linkRefractionWithTransparency, this.invertRefractionY = e.invertRefractionY, this.useThicknessAsDepth = !!e.useThicknessAsDepth; } } C([ rn("Link refraction to transparency", Gr.Boolean, "ADVANCED", { notifiers: { update: !0 } }) ], wD.prototype, "linkRefractionWithTransparency", void 0); C([ rn("Invert refraction Y", Gr.Boolean, "ADVANCED", { notifiers: { update: !0 } }) ], wD.prototype, "invertRefractionY", void 0); C([ rn("Use thickness as depth", Gr.Boolean, "ADVANCED", { notifiers: { update: !0 } }) ], wD.prototype, "useThicknessAsDepth", void 0); Ue("BABYLON.RefractionBlock", wD); class CS extends Mr { /** * Create a new SubSurfaceBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Fragment), this._isUnique = !0, this.registerInput("thickness", de.Float, !1, Ve.Fragment), this.registerInput("tintColor", de.Color3, !0, Ve.Fragment), this.registerInput("translucencyIntensity", de.Float, !0, Ve.Fragment), this.registerInput("translucencyDiffusionDist", de.Color3, !0, Ve.Fragment), this.registerInput("refraction", de.Object, !0, Ve.Fragment, new Ko("refraction", this, ao.Input, wD, "RefractionBlock")), this.registerInput("dispersion", de.Float, !0, Ve.Fragment), this.registerOutput("subsurface", de.Object, Ve.Fragment, new Ko("subsurface", this, ao.Output, CS, "SubSurfaceBlock")); } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("subSurfaceOut"), e._excludeVariableName("vThicknessParam"), e._excludeVariableName("vTintColor"), e._excludeVariableName("vSubSurfaceIntensity"), e._excludeVariableName("dispersion"); } /** * Gets the current class name * @returns the class name */ getClassName() { return "SubSurfaceBlock"; } /** * Gets the thickness component */ get thickness() { return this._inputs[0]; } /** * Gets the tint color input component */ get tintColor() { return this._inputs[1]; } /** * Gets the translucency intensity input component */ get translucencyIntensity() { return this._inputs[2]; } /** * Gets the translucency diffusion distance input component */ get translucencyDiffusionDist() { return this._inputs[3]; } /** * Gets the refraction object parameters */ get refraction() { return this._inputs[4]; } /** * Gets the dispersion input component */ get dispersion() { return this._inputs[5]; } /** * Gets the sub surface object output component */ get subsurface() { return this._outputs[0]; } autoConfigure() { if (!this.thickness.isConnected) { const e = new gi("SubSurface thickness", Ve.Fragment, de.Float); e.value = 0, e.output.connectTo(this.thickness); } } prepareDefines(e, t, r) { super.prepareDefines(e, t, r); const n = this.translucencyDiffusionDist.isConnected || this.translucencyIntensity.isConnected; r.setValue("SUBSURFACE", n || this.refraction.isConnected, !0), r.setValue("SS_TRANSLUCENCY", n, !0), r.setValue("SS_THICKNESSANDMASK_TEXTURE", !1, !0), r.setValue("SS_REFRACTIONINTENSITY_TEXTURE", !1, !0), r.setValue("SS_TRANSLUCENCYINTENSITY_TEXTURE", !1, !0), r.setValue("SS_MASK_FROM_THICKNESS_TEXTURE", !1, !0), r.setValue("SS_USE_GLTF_TEXTURES", !1, !0), r.setValue("SS_DISPERSION", this.dispersion.isConnected, !0); } /** * Gets the main code of the block (fragment side) * @param state current state of the node material building * @param ssBlock instance of a SubSurfaceBlock or null if the code must be generated without an active sub surface module * @param reflectionBlock instance of a ReflectionBlock null if the code must be generated without an active reflection module * @param worldPosVarName name of the variable holding the world position * @returns the shader code */ static GetCode(e, t, r, n) { var i, s, a, f, o, d, v, u, l, P, p, c, H, T, q, b; let j = ""; const w = t != null && t.thickness.isConnected ? t.thickness.associatedVariableName : "0.", m = t != null && t.tintColor.isConnected ? t.tintColor.associatedVariableName : "vec3(1.)", I = t != null && t.translucencyIntensity.isConnected ? t == null ? void 0 : t.translucencyIntensity.associatedVariableName : "1.", N = t != null && t.translucencyDiffusionDist.isConnected ? t == null ? void 0 : t.translucencyDiffusionDist.associatedVariableName : "vec3(1.)", k = t != null && t.refraction.isConnected ? (i = t == null ? void 0 : t.refraction.connectedPoint) === null || i === void 0 ? void 0 : i.ownerBlock : null, R = k != null && k.tintAtDistance.isConnected ? k.tintAtDistance.associatedVariableName : "1.", y = k != null && k.intensity.isConnected ? k.intensity.associatedVariableName : "1.", O = k != null && k.view.isConnected ? k.view.associatedVariableName : "", Y = t != null && t.dispersion.isConnected ? t == null ? void 0 : t.dispersion.associatedVariableName : "0.0"; return j += (s = k == null ? void 0 : k.getCode(e)) !== null && s !== void 0 ? s : "", j += `subSurfaceOutParams subSurfaceOut; #ifdef SUBSURFACE vec2 vThicknessParam = vec2(0., ${w}); vec4 vTintColor = vec4(${m}, ${R}); vec3 vSubSurfaceIntensity = vec3(${y}, ${I}, 0.); float dispersion = ${Y}; subSurfaceBlock( vSubSurfaceIntensity, vThicknessParam, vTintColor, normalW, specularEnvironmentReflectance, #ifdef SS_THICKNESSANDMASK_TEXTURE vec4(0.), #endif #ifdef REFLECTION #ifdef SS_TRANSLUCENCY ${r == null ? void 0 : r._reflectionMatrixName}, #ifdef USESPHERICALFROMREFLECTIONMAP #if !defined(NORMAL) || !defined(USESPHERICALINVERTEX) reflectionOut.irradianceVector, #endif #if defined(REALTIME_FILTERING) ${r == null ? void 0 : r._cubeSamplerName}, ${r == null ? void 0 : r._vReflectionFilteringInfoName}, #endif #endif #ifdef USEIRRADIANCEMAP irradianceSampler, #endif #endif #endif #if defined(SS_REFRACTION) || defined(SS_TRANSLUCENCY) surfaceAlbedo, #endif #ifdef SS_REFRACTION ${n}.xyz, viewDirectionW, ${O}, ${(a = k == null ? void 0 : k._vRefractionInfosName) !== null && a !== void 0 ? a : ""}, ${(f = k == null ? void 0 : k._refractionMatrixName) !== null && f !== void 0 ? f : ""}, ${(o = k == null ? void 0 : k._vRefractionMicrosurfaceInfosName) !== null && o !== void 0 ? o : ""}, vLightingIntensity, #ifdef SS_LINKREFRACTIONTOTRANSPARENCY alpha, #endif #ifdef ${(d = k == null ? void 0 : k._defineLODRefractionAlpha) !== null && d !== void 0 ? d : "IGNORE"} NdotVUnclamped, #endif #ifdef ${(v = k == null ? void 0 : k._defineLinearSpecularRefraction) !== null && v !== void 0 ? v : "IGNORE"} roughness, #endif alphaG, #ifdef ${(u = k == null ? void 0 : k._define3DName) !== null && u !== void 0 ? u : "IGNORE"} ${(l = k == null ? void 0 : k._cubeSamplerName) !== null && l !== void 0 ? l : ""}, #else ${(P = k == null ? void 0 : k._2DSamplerName) !== null && P !== void 0 ? P : ""}, #endif #ifndef LODBASEDMICROSFURACE #ifdef ${(p = k == null ? void 0 : k._define3DName) !== null && p !== void 0 ? p : "IGNORE"} ${(c = k == null ? void 0 : k._cubeSamplerName) !== null && c !== void 0 ? c : ""}, ${(H = k == null ? void 0 : k._cubeSamplerName) !== null && H !== void 0 ? H : ""}, #else ${(T = k == null ? void 0 : k._2DSamplerName) !== null && T !== void 0 ? T : ""}, ${(q = k == null ? void 0 : k._2DSamplerName) !== null && q !== void 0 ? q : ""}, #endif #endif #ifdef ANISOTROPIC anisotropicOut, #endif #ifdef REALTIME_FILTERING ${(b = k == null ? void 0 : k._vRefractionFilteringInfoName) !== null && b !== void 0 ? b : ""}, #endif #ifdef SS_USE_LOCAL_REFRACTIONMAP_CUBIC vRefractionPosition, vRefractionSize, #endif #ifdef SS_DISPERSION dispersion, #endif #endif #ifdef SS_TRANSLUCENCY ${N}, #endif subSurfaceOut ); #ifdef SS_REFRACTION surfaceAlbedo = subSurfaceOut.surfaceAlbedo; #ifdef SS_LINKREFRACTIONTOTRANSPARENCY alpha = subSurfaceOut.alpha; #endif #endif #else subSurfaceOut.specularEnvironmentReflectance = specularEnvironmentReflectance; #endif `, j; } _buildBlock(e) { return e.target === Ve.Fragment && e.sharedData.blocksWithDefines.push(this), this; } } Ue("BABYLON.SubSurfaceBlock", CS); const Zpe = { ambientClr: ["finalAmbient", ""], diffuseDir: ["finalDiffuse", ""], specularDir: ["finalSpecularScaled", "!defined(UNLIT) && defined(SPECULARTERM)"], clearcoatDir: ["finalClearCoatScaled", "!defined(UNLIT) && defined(CLEARCOAT)"], sheenDir: ["finalSheenScaled", "!defined(UNLIT) && defined(SHEEN)"], diffuseInd: ["finalIrradiance", "!defined(UNLIT) && defined(REFLECTION)"], specularInd: ["finalRadianceScaled", "!defined(UNLIT) && defined(REFLECTION)"], clearcoatInd: ["clearcoatOut.finalClearCoatRadianceScaled", "!defined(UNLIT) && defined(REFLECTION) && defined(CLEARCOAT)"], sheenInd: ["sheenOut.finalSheenRadianceScaled", "!defined(UNLIT) && defined(REFLECTION) && defined(SHEEN) && defined(ENVIRONMENTBRDF)"], refraction: ["subSurfaceOut.finalRefraction", "!defined(UNLIT) && defined(SS_REFRACTION)"], lighting: ["finalColor.rgb", ""], shadow: ["aggShadow", ""], alpha: ["alpha", ""] }; class g9 extends Mr { static _OnGenerateOnlyFragmentCodeChanged(e, t) { const r = e; return r.worldPosition.isConnected ? (r.generateOnlyFragmentCode = !r.generateOnlyFragmentCode, console.error("The worldPosition input must not be connected to be able to switch!"), !1) : (r._setTarget(), !0); } _setTarget() { this._setInitialTarget(this.generateOnlyFragmentCode ? Ve.Fragment : Ve.VertexAndFragment), this.getInputByName("worldPosition").target = this.generateOnlyFragmentCode ? Ve.Fragment : Ve.Vertex; } /** * Create a new ReflectionBlock * @param name defines the block name */ constructor(e) { super(e, Ve.VertexAndFragment), this._environmentBRDFTexture = null, this._metallicReflectanceColor = Ne.White(), this._metallicF0Factor = 1, this.directIntensity = 1, this.environmentIntensity = 1, this.specularIntensity = 1, this.lightFalloff = 0, this.useAlphaTest = !1, this.alphaTestCutoff = 0.5, this.useAlphaBlending = !1, this.useRadianceOverAlpha = !0, this.useSpecularOverAlpha = !0, this.enableSpecularAntiAliasing = !1, this.realTimeFiltering = !1, this.realTimeFilteringQuality = 8, this.useEnergyConservation = !0, this.useRadianceOcclusion = !0, this.useHorizonOcclusion = !0, this.unlit = !1, this.forceNormalForward = !1, this.generateOnlyFragmentCode = !1, this.debugMode = 0, this.debugLimit = 0, this.debugFactor = 1, this._isUnique = !0, this.registerInput("worldPosition", de.Vector4, !1, Ve.Vertex), this.registerInput("worldNormal", de.Vector4, !1, Ve.Fragment), this.registerInput("view", de.Matrix, !1), this.registerInput("cameraPosition", de.Vector3, !1, Ve.Fragment), this.registerInput("perturbedNormal", de.Vector4, !0, Ve.Fragment), this.registerInput("baseColor", de.Color3, !0, Ve.Fragment), this.registerInput("metallic", de.Float, !1, Ve.Fragment), this.registerInput("roughness", de.Float, !1, Ve.Fragment), this.registerInput("ambientOcc", de.Float, !0, Ve.Fragment), this.registerInput("opacity", de.Float, !0, Ve.Fragment), this.registerInput("indexOfRefraction", de.Float, !0, Ve.Fragment), this.registerInput("ambientColor", de.Color3, !0, Ve.Fragment), this.registerInput("reflection", de.Object, !0, Ve.Fragment, new Ko("reflection", this, ao.Input, Cm, "ReflectionBlock")), this.registerInput("clearcoat", de.Object, !0, Ve.Fragment, new Ko("clearcoat", this, ao.Input, Xm, "ClearCoatBlock")), this.registerInput("sheen", de.Object, !0, Ve.Fragment, new Ko("sheen", this, ao.Input, Vm, "SheenBlock")), this.registerInput("subsurface", de.Object, !0, Ve.Fragment, new Ko("subsurface", this, ao.Input, CS, "SubSurfaceBlock")), this.registerInput("anisotropy", de.Object, !0, Ve.Fragment, new Ko("anisotropy", this, ao.Input, fV, "AnisotropyBlock")), this.registerInput("iridescence", de.Object, !0, Ve.Fragment, new Ko("iridescence", this, ao.Input, eR, "IridescenceBlock")), this.registerOutput("ambientClr", de.Color3, Ve.Fragment), this.registerOutput("diffuseDir", de.Color3, Ve.Fragment), this.registerOutput("specularDir", de.Color3, Ve.Fragment), this.registerOutput("clearcoatDir", de.Color3, Ve.Fragment), this.registerOutput("sheenDir", de.Color3, Ve.Fragment), this.registerOutput("diffuseInd", de.Color3, Ve.Fragment), this.registerOutput("specularInd", de.Color3, Ve.Fragment), this.registerOutput("clearcoatInd", de.Color3, Ve.Fragment), this.registerOutput("sheenInd", de.Color3, Ve.Fragment), this.registerOutput("refraction", de.Color3, Ve.Fragment), this.registerOutput("lighting", de.Color3, Ve.Fragment), this.registerOutput("shadow", de.Float, Ve.Fragment), this.registerOutput("alpha", de.Float, Ve.Fragment); } /** * Initialize the block and prepare the context for build * @param state defines the state that will be used for the build */ initialize(e) { e._excludeVariableName("vLightingIntensity"), e._excludeVariableName("geometricNormalW"), e._excludeVariableName("normalW"), e._excludeVariableName("faceNormal"), e._excludeVariableName("albedoOpacityOut"), e._excludeVariableName("surfaceAlbedo"), e._excludeVariableName("alpha"), e._excludeVariableName("aoOut"), e._excludeVariableName("baseColor"), e._excludeVariableName("reflectivityOut"), e._excludeVariableName("microSurface"), e._excludeVariableName("roughness"), e._excludeVariableName("NdotVUnclamped"), e._excludeVariableName("NdotV"), e._excludeVariableName("alphaG"), e._excludeVariableName("AARoughnessFactors"), e._excludeVariableName("environmentBrdf"), e._excludeVariableName("ambientMonochrome"), e._excludeVariableName("seo"), e._excludeVariableName("eho"), e._excludeVariableName("environmentRadiance"), e._excludeVariableName("irradianceVector"), e._excludeVariableName("environmentIrradiance"), e._excludeVariableName("diffuseBase"), e._excludeVariableName("specularBase"), e._excludeVariableName("preInfo"), e._excludeVariableName("info"), e._excludeVariableName("shadow"), e._excludeVariableName("finalDiffuse"), e._excludeVariableName("finalAmbient"), e._excludeVariableName("ambientOcclusionForDirectDiffuse"), e._excludeVariableName("finalColor"), e._excludeVariableName("vClipSpacePosition"), e._excludeVariableName("vDebugMode"); } /** * Gets the current class name * @returns the class name */ getClassName() { return "PBRMetallicRoughnessBlock"; } /** * Gets the world position input component */ get worldPosition() { return this._inputs[0]; } /** * Gets the world normal input component */ get worldNormal() { return this._inputs[1]; } /** * Gets the view matrix parameter */ get view() { return this._inputs[2]; } /** * Gets the camera position input component */ get cameraPosition() { return this._inputs[3]; } /** * Gets the perturbed normal input component */ get perturbedNormal() { return this._inputs[4]; } /** * Gets the base color input component */ get baseColor() { return this._inputs[5]; } /** * Gets the metallic input component */ get metallic() { return this._inputs[6]; } /** * Gets the roughness input component */ get roughness() { return this._inputs[7]; } /** * Gets the ambient occlusion input component */ get ambientOcc() { return this._inputs[8]; } /** * Gets the opacity input component */ get opacity() { return this._inputs[9]; } /** * Gets the index of refraction input component */ get indexOfRefraction() { return this._inputs[10]; } /** * Gets the ambient color input component */ get ambientColor() { return this._inputs[11]; } /** * Gets the reflection object parameters */ get reflection() { return this._inputs[12]; } /** * Gets the clear coat object parameters */ get clearcoat() { return this._inputs[13]; } /** * Gets the sheen object parameters */ get sheen() { return this._inputs[14]; } /** * Gets the sub surface object parameters */ get subsurface() { return this._inputs[15]; } /** * Gets the anisotropy object parameters */ get anisotropy() { return this._inputs[16]; } /** * Gets the iridescence object parameters */ get iridescence() { return this._inputs[17]; } /** * Gets the ambient output component */ get ambientClr() { return this._outputs[0]; } /** * Gets the diffuse output component */ get diffuseDir() { return this._outputs[1]; } /** * Gets the specular output component */ get specularDir() { return this._outputs[2]; } /** * Gets the clear coat output component */ get clearcoatDir() { return this._outputs[3]; } /** * Gets the sheen output component */ get sheenDir() { return this._outputs[4]; } /** * Gets the indirect diffuse output component */ get diffuseInd() { return this._outputs[5]; } /** * Gets the indirect specular output component */ get specularInd() { return this._outputs[6]; } /** * Gets the indirect clear coat output component */ get clearcoatInd() { return this._outputs[7]; } /** * Gets the indirect sheen output component */ get sheenInd() { return this._outputs[8]; } /** * Gets the refraction output component */ get refraction() { return this._outputs[9]; } /** * Gets the global lighting output component */ get lighting() { return this._outputs[10]; } /** * Gets the shadow output component */ get shadow() { return this._outputs[11]; } /** * Gets the alpha output component */ get alpha() { return this._outputs[12]; } autoConfigure(e, t = () => !0) { if (!this.cameraPosition.isConnected) { let r = e.getInputBlockByPredicate((n) => n.systemValue === Bi.CameraPosition && t(n)); r || (r = new gi("cameraPosition"), r.setAsSystemValue(Bi.CameraPosition)), r.output.connectTo(this.cameraPosition); } if (!this.view.isConnected) { let r = e.getInputBlockByPredicate((n) => n.systemValue === Bi.View && t(n)); r || (r = new gi("view"), r.setAsSystemValue(Bi.View)), r.output.connectTo(this.view); } } prepareDefines(e, t, r) { r.setValue("PBR", !0), r.setValue("METALLICWORKFLOW", !0), r.setValue("DEBUGMODE", this.debugMode, !0), r.setValue("DEBUGMODE_FORCERETURN", !0), r.setValue("NORMALXYSCALE", !0), r.setValue("BUMP", this.perturbedNormal.isConnected, !0), r.setValue("LODBASEDMICROSFURACE", this._scene.getEngine().getCaps().textureLOD), r.setValue("ALBEDO", !1, !0), r.setValue("OPACITY", this.opacity.isConnected, !0), r.setValue("AMBIENT", !0, !0), r.setValue("AMBIENTINGRAYSCALE", !1, !0), r.setValue("REFLECTIVITY", !1, !0), r.setValue("AOSTOREINMETALMAPRED", !1, !0), r.setValue("METALLNESSSTOREINMETALMAPBLUE", !1, !0), r.setValue("ROUGHNESSSTOREINMETALMAPALPHA", !1, !0), r.setValue("ROUGHNESSSTOREINMETALMAPGREEN", !1, !0), this.lightFalloff === fs.LIGHTFALLOFF_STANDARD ? (r.setValue("USEPHYSICALLIGHTFALLOFF", !1), r.setValue("USEGLTFLIGHTFALLOFF", !1)) : this.lightFalloff === fs.LIGHTFALLOFF_GLTF ? (r.setValue("USEPHYSICALLIGHTFALLOFF", !1), r.setValue("USEGLTFLIGHTFALLOFF", !0)) : (r.setValue("USEPHYSICALLIGHTFALLOFF", !0), r.setValue("USEGLTFLIGHTFALLOFF", !1)); const n = this.alphaTestCutoff.toString(); r.setValue("ALPHABLEND", this.useAlphaBlending, !0), r.setValue("ALPHAFROMALBEDO", !1, !0), r.setValue("ALPHATEST", this.useAlphaTest, !0), r.setValue("ALPHATESTVALUE", n.indexOf(".") < 0 ? n + "." : n, !0), r.setValue("OPACITYRGB", !1, !0), r.setValue("RADIANCEOVERALPHA", this.useRadianceOverAlpha, !0), r.setValue("SPECULAROVERALPHA", this.useSpecularOverAlpha, !0), r.setValue("SPECULARAA", this._scene.getEngine().getCaps().standardDerivatives && this.enableSpecularAntiAliasing, !0), r.setValue("REALTIME_FILTERING", this.realTimeFiltering, !0); const i = e.getScene(); if (i.getEngine()._features.needTypeSuffixInShaderConstants ? r.setValue("NUM_SAMPLES", this.realTimeFilteringQuality + "u", !0) : r.setValue("NUM_SAMPLES", "" + this.realTimeFilteringQuality, !0), r.setValue("BRDF_V_HEIGHT_CORRELATED", !0), r.setValue("MS_BRDF_ENERGY_CONSERVATION", this.useEnergyConservation, !0), r.setValue("RADIANCEOCCLUSION", this.useRadianceOcclusion, !0), r.setValue("HORIZONOCCLUSION", this.useHorizonOcclusion, !0), r.setValue("UNLIT", this.unlit, !0), r.setValue("FORCENORMALFORWARD", this.forceNormalForward, !0), this._environmentBRDFTexture && Dt.ReflectionTextureEnabled ? (r.setValue("ENVIRONMENTBRDF", !0), r.setValue("ENVIRONMENTBRDF_RGBD", this._environmentBRDFTexture.isRGBD, !0)) : (r.setValue("ENVIRONMENTBRDF", !1), r.setValue("ENVIRONMENTBRDF_RGBD", !1)), r._areImageProcessingDirty && t.imageProcessingConfiguration && t.imageProcessingConfiguration.prepareDefines(r), !!r._areLightsDirty) if (!this.light) Ye.PrepareDefinesForLights(i, e, r, !0, t.maxSimultaneousLights), r._needNormals = !0, Ye.PrepareDefinesForMultiview(i, r); else { const a = { needNormals: !1, needRebuild: !1, lightmapMode: !1, shadowEnabled: !1, specularEnabled: !1 }; Ye.PrepareDefinesForLight(i, e, this.light, this._lightId, r, !0, a), a.needRebuild && r.rebuild(); } } updateUniformsAndSamples(e, t, r, n) { for (let i = 0; i < t.maxSimultaneousLights && r["LIGHT" + i]; i++) { const s = e.uniforms.indexOf("vLightData" + i) >= 0; Ye.PrepareUniformsAndSamplersForLight(i, e.uniforms, e.samplers, r["PROJECTEDLIGHTTEXTURE" + i], n, s); } } isReady(e, t, r) { return !(this._environmentBRDFTexture && !this._environmentBRDFTexture.isReady() || r._areImageProcessingDirty && t.imageProcessingConfiguration && !t.imageProcessingConfiguration.isReady()); } bind(e, t, r) { var n, i; if (!r) return; const s = r.getScene(); this.light ? Ye.BindLight(this.light, this._lightId, s, e, !0) : Ye.BindLights(s, r, e, !0, t.maxSimultaneousLights), e.setTexture(this._environmentBrdfSamplerName, this._environmentBRDFTexture), e.setFloat2("vDebugMode", this.debugLimit, this.debugFactor); const a = this._scene.ambientColor; a && e.setColor3("ambientFromScene", a); const f = s.useRightHandedSystem === (s._mirroredCameraPosition != null); e.setFloat(this._invertNormalName, f ? -1 : 1), e.setFloat4("vLightingIntensity", this.directIntensity, 1, this.environmentIntensity * this._scene.environmentIntensity, this.specularIntensity); const o = 1, d = (i = (n = this.indexOfRefraction.connectInputBlock) === null || n === void 0 ? void 0 : n.value) !== null && i !== void 0 ? i : 1.5, v = Math.pow((d - o) / (d + o), 2); this._metallicReflectanceColor.scaleToRef(v * this._metallicF0Factor, Hs.Color3[0]); const u = this._metallicF0Factor; e.setColor4(this._vMetallicReflectanceFactorsName, Hs.Color3[0], u), t.imageProcessingConfiguration && t.imageProcessingConfiguration.bind(e); } _injectVertexCode(e) { var t, r; const n = this.worldPosition, i = `//${this.name}`; this.light ? (this._lightId = (e.counters.lightCounter !== void 0 ? e.counters.lightCounter : -1) + 1, e.counters.lightCounter = this._lightId, e._emitFunctionFromInclude(e.supportUniformBuffers ? "lightVxUboDeclaration" : "lightVxFragmentDeclaration", i, { replaceStrings: [{ search: /{X}/g, replace: this._lightId.toString() }] }, this._lightId.toString())) : (e._emitFunctionFromInclude(e.supportUniformBuffers ? "lightVxUboDeclaration" : "lightVxFragmentDeclaration", i, { repeatKey: "maxSimultaneousLights" }), this._lightId = 0, e.sharedData.dynamicUniformBlocks.push(this)); const s = "v_" + n.associatedVariableName; e._emitVaryingFromString(s, "vec4") && (e.compilationString += `${s} = ${n.associatedVariableName}; `); const a = this.reflection.isConnected ? (t = this.reflection.connectedPoint) === null || t === void 0 ? void 0 : t.ownerBlock : null; a && (a.viewConnectionPoint = this.view), e.compilationString += (r = a == null ? void 0 : a.handleVertexSide(e)) !== null && r !== void 0 ? r : "", e._emitVaryingFromString("vClipSpacePosition", "vec4", "defined(IGNORE) || DEBUGMODE > 0") && (e._injectAtEnd += `#if DEBUGMODE > 0 `, e._injectAtEnd += `vClipSpacePosition = gl_Position; `, e._injectAtEnd += `#endif `), this.light ? e.compilationString += e._emitCodeFromInclude("shadowsVertex", i, { replaceStrings: [ { search: /{X}/g, replace: this._lightId.toString() }, { search: /worldPos/g, replace: n.associatedVariableName } ] }) : (e.compilationString += `vec4 worldPos = ${n.associatedVariableName}; `, this.view.isConnected && (e.compilationString += `mat4 view = ${this.view.associatedVariableName}; `), e.compilationString += e._emitCodeFromInclude("shadowsVertex", i, { repeatKey: "maxSimultaneousLights" })); } _getAlbedoOpacityCode() { let e = `albedoOpacityOutParams albedoOpacityOut; `; const t = this.baseColor.isConnected ? this.baseColor.associatedVariableName : "vec3(1.)", r = this.opacity.isConnected ? this.opacity.associatedVariableName : "1."; return e += `albedoOpacityBlock( vec4(${t}, 1.), #ifdef ALBEDO vec4(1.), vec2(1., 1.), #endif #ifdef OPACITY vec4(${r}), vec2(1., 1.), #endif albedoOpacityOut ); vec3 surfaceAlbedo = albedoOpacityOut.surfaceAlbedo; float alpha = albedoOpacityOut.alpha; `, e; } _getAmbientOcclusionCode() { let e = `ambientOcclusionOutParams aoOut; `; const t = this.ambientOcc.isConnected ? this.ambientOcc.associatedVariableName : "1."; return e += `ambientOcclusionBlock( #ifdef AMBIENT vec3(${t}), vec4(0., 1.0, 1.0, 0.), #endif aoOut ); `, e; } _getReflectivityCode(e) { let t = `reflectivityOutParams reflectivityOut; `; const r = "1."; return this._vMetallicReflectanceFactorsName = e._getFreeVariableName("vMetallicReflectanceFactors"), e._emitUniformFromString(this._vMetallicReflectanceFactorsName, "vec4"), t += `vec3 baseColor = surfaceAlbedo; reflectivityBlock( vec4(${this.metallic.associatedVariableName}, ${this.roughness.associatedVariableName}, 0., 0.), #ifdef METALLICWORKFLOW surfaceAlbedo, ${this._vMetallicReflectanceFactorsName}, #endif #ifdef REFLECTIVITY vec3(0., 0., ${r}), vec4(1.), #endif #if defined(METALLICWORKFLOW) && defined(REFLECTIVITY) && defined(AOSTOREINMETALMAPRED) aoOut.ambientOcclusionColor, #endif #ifdef MICROSURFACEMAP microSurfaceTexel, <== not handled! #endif reflectivityOut ); float microSurface = reflectivityOut.microSurface; float roughness = reflectivityOut.roughness; #ifdef METALLICWORKFLOW surfaceAlbedo = reflectivityOut.surfaceAlbedo; #endif #if defined(METALLICWORKFLOW) && defined(REFLECTIVITY) && defined(AOSTOREINMETALMAPRED) aoOut.ambientOcclusionColor = reflectivityOut.ambientOcclusionColor; #endif `, t; } _buildBlock(e) { var t, r, n, i, s, a, f, o, d, v, u, l, P, p, c, H, T, q, b, j, w, m, I, N, k, R, y, O, Y, ee, Z, te, fe, _, G, L, $, ae, Pe, ge, me; super._buildBlock(e), this._scene = e.sharedData.scene, this._environmentBRDFTexture || (this._environmentBRDFTexture = nV(this._scene)); const Xe = this.reflection.isConnected ? (t = this.reflection.connectedPoint) === null || t === void 0 ? void 0 : t.ownerBlock : null; if (Xe && (Xe.worldPositionConnectionPoint = this.worldPosition, Xe.cameraPositionConnectionPoint = this.cameraPosition, Xe.worldNormalConnectionPoint = this.worldNormal, Xe.viewConnectionPoint = this.view), e.target !== Ve.Fragment) return this._injectVertexCode(e), this; e.sharedData.forcedBindableBlocks.push(this), e.sharedData.blocksWithDefines.push(this), e.sharedData.blockingBlocks.push(this), this.generateOnlyFragmentCode && e.sharedData.dynamicUniformBlocks.push(this); const De = `//${this.name}`, ne = this.perturbedNormal; let re = this.worldPosition.associatedVariableName; this.generateOnlyFragmentCode ? (re = e._getFreeVariableName("globalWorldPos"), e._emitFunction("pbr_globalworldpos", `vec3 ${re}; `, De), e.compilationString += `${re} = ${this.worldPosition.associatedVariableName}.xyz; `, e.compilationString += e._emitCodeFromInclude("shadowsVertex", De, { repeatKey: "maxSimultaneousLights", substitutionVars: this.generateOnlyFragmentCode ? `worldPos,${this.worldPosition.associatedVariableName}` : void 0 }), e.compilationString += `#if DEBUGMODE > 0 `, e.compilationString += `vec4 vClipSpacePosition = vec4((vec2(gl_FragCoord.xy) / vec2(1.0)) * 2.0 - 1.0, 0.0, 1.0); `, e.compilationString += `#endif `) : re = "v_" + re, this._environmentBrdfSamplerName = e._getFreeVariableName("environmentBrdfSampler"), e._emit2DSampler(this._environmentBrdfSamplerName), e.sharedData.hints.needAlphaBlending = e.sharedData.hints.needAlphaBlending || this.useAlphaBlending, e.sharedData.hints.needAlphaTesting = e.sharedData.hints.needAlphaTesting || this.useAlphaTest, e._emitExtension("lod", "#extension GL_EXT_shader_texture_lod : enable", "defined(LODBASEDMICROSFURACE)"), e._emitExtension("derivatives", "#extension GL_OES_standard_derivatives : enable"), e._emitUniformFromString("vDebugMode", "vec2", "defined(IGNORE) || DEBUGMODE > 0"), e._emitUniformFromString("ambientFromScene", "vec3"), e.uniforms.push("exposureLinear"), e.uniforms.push("contrast"), e.uniforms.push("vInverseScreenSize"), e.uniforms.push("vignetteSettings1"), e.uniforms.push("vignetteSettings2"), e.uniforms.push("vCameraColorCurveNegative"), e.uniforms.push("vCameraColorCurveNeutral"), e.uniforms.push("vCameraColorCurvePositive"), e.uniforms.push("txColorTransform"), e.uniforms.push("colorTransformSettings"), e.uniforms.push("ditherIntensity"), this.light ? e._emitFunctionFromInclude(e.supportUniformBuffers ? "lightUboDeclaration" : "lightFragmentDeclaration", De, { replaceStrings: [{ search: /{X}/g, replace: this._lightId.toString() }] }, this._lightId.toString()) : e._emitFunctionFromInclude(e.supportUniformBuffers ? "lightUboDeclaration" : "lightFragmentDeclaration", De, { repeatKey: "maxSimultaneousLights", substitutionVars: this.generateOnlyFragmentCode ? "varying," : void 0 }), e._emitFunctionFromInclude("helperFunctions", De), e._emitFunctionFromInclude("importanceSampling", De), e._emitFunctionFromInclude("pbrHelperFunctions", De), e._emitFunctionFromInclude("imageProcessingDeclaration", De), e._emitFunctionFromInclude("imageProcessingFunctions", De), e._emitFunctionFromInclude("shadowsFragmentFunctions", De), e._emitFunctionFromInclude("pbrDirectLightingSetupFunctions", De, { replaceStrings: [{ search: /vPositionW/g, replace: re + ".xyz" }] }), e._emitFunctionFromInclude("pbrDirectLightingFalloffFunctions", De), e._emitFunctionFromInclude("pbrBRDFFunctions", De, { replaceStrings: [{ search: /REFLECTIONMAP_SKYBOX/g, replace: (r = Xe == null ? void 0 : Xe._defineSkyboxName) !== null && r !== void 0 ? r : "REFLECTIONMAP_SKYBOX" }] }), e._emitFunctionFromInclude("hdrFilteringFunctions", De), e._emitFunctionFromInclude("pbrDirectLightingFunctions", De, { replaceStrings: [{ search: /vPositionW/g, replace: re + ".xyz" }] }), e._emitFunctionFromInclude("pbrIBLFunctions", De), e._emitFunctionFromInclude("pbrBlockAlbedoOpacity", De), e._emitFunctionFromInclude("pbrBlockReflectivity", De), e._emitFunctionFromInclude("pbrBlockAmbientOcclusion", De), e._emitFunctionFromInclude("pbrBlockAlphaFresnel", De), e._emitFunctionFromInclude("pbrBlockAnisotropic", De), e._emitUniformFromString("vLightingIntensity", "vec4"), Xe != null && Xe.generateOnlyFragmentCode && (e.compilationString += Xe.handleVertexSide(e)), this._vNormalWName = e._getFreeVariableName("vNormalW"), e.compilationString += `vec4 ${this._vNormalWName} = normalize(${this.worldNormal.associatedVariableName}); `, e._registerTempVariable("viewDirectionW") && (e.compilationString += `vec3 viewDirectionW = normalize(${this.cameraPosition.associatedVariableName} - ${re}.xyz); `), e.compilationString += `vec3 geometricNormalW = ${this._vNormalWName}.xyz; `, e.compilationString += `vec3 normalW = ${ne.isConnected ? "normalize(" + ne.associatedVariableName + ".xyz)" : "geometricNormalW"}; `, this._invertNormalName = e._getFreeVariableName("invertNormal"), e._emitUniformFromString(this._invertNormalName, "float"), e.compilationString += e._emitCodeFromInclude("pbrBlockNormalFinal", De, { replaceStrings: [ { search: /vPositionW/g, replace: re + ".xyz" }, { search: /vEyePosition.w/g, replace: this._invertNormalName } ] }), e.compilationString += this._getAlbedoOpacityCode(), e.compilationString += e._emitCodeFromInclude("depthPrePass", De), e.compilationString += this._getAmbientOcclusionCode(), e.compilationString += e._emitCodeFromInclude("pbrBlockLightmapInit", De), e.compilationString += `#ifdef UNLIT vec3 diffuseBase = vec3(1., 1., 1.); #else `, e.compilationString += this._getReflectivityCode(e), e.compilationString += e._emitCodeFromInclude("pbrBlockGeometryInfo", De, { replaceStrings: [ { search: /REFLECTIONMAP_SKYBOX/g, replace: (n = Xe == null ? void 0 : Xe._defineSkyboxName) !== null && n !== void 0 ? n : "REFLECTIONMAP_SKYBOX" }, { search: /REFLECTIONMAP_3D/g, replace: (i = Xe == null ? void 0 : Xe._define3DName) !== null && i !== void 0 ? i : "REFLECTIONMAP_3D" } ] }); const ve = this.anisotropy.isConnected ? (s = this.anisotropy.connectedPoint) === null || s === void 0 ? void 0 : s.ownerBlock : null; ve && (ve.worldPositionConnectionPoint = this.worldPosition, ve.worldNormalConnectionPoint = this.worldNormal, e.compilationString += ve.getCode(e, !this.perturbedNormal.isConnected)), Xe && Xe.hasTexture && (e.compilationString += Xe.getCode(e, ve ? "anisotropicOut.anisotropicNormal" : "normalW")), e._emitFunctionFromInclude("pbrBlockReflection", De, { replaceStrings: [ { search: /computeReflectionCoords/g, replace: "computeReflectionCoordsPBR" }, { search: /REFLECTIONMAP_3D/g, replace: (a = Xe == null ? void 0 : Xe._define3DName) !== null && a !== void 0 ? a : "REFLECTIONMAP_3D" }, { search: /REFLECTIONMAP_OPPOSITEZ/g, replace: (f = Xe == null ? void 0 : Xe._defineOppositeZ) !== null && f !== void 0 ? f : "REFLECTIONMAP_OPPOSITEZ" }, { search: /REFLECTIONMAP_PROJECTION/g, replace: (o = Xe == null ? void 0 : Xe._defineProjectionName) !== null && o !== void 0 ? o : "REFLECTIONMAP_PROJECTION" }, { search: /REFLECTIONMAP_SKYBOX/g, replace: (d = Xe == null ? void 0 : Xe._defineSkyboxName) !== null && d !== void 0 ? d : "REFLECTIONMAP_SKYBOX" }, { search: /LODINREFLECTIONALPHA/g, replace: (v = Xe == null ? void 0 : Xe._defineLODReflectionAlpha) !== null && v !== void 0 ? v : "LODINREFLECTIONALPHA" }, { search: /LINEARSPECULARREFLECTION/g, replace: (u = Xe == null ? void 0 : Xe._defineLinearSpecularReflection) !== null && u !== void 0 ? u : "LINEARSPECULARREFLECTION" }, { search: /vReflectionFilteringInfo/g, replace: (l = Xe == null ? void 0 : Xe._vReflectionFilteringInfoName) !== null && l !== void 0 ? l : "vReflectionFilteringInfo" } ] }), e.compilationString += e._emitCodeFromInclude("pbrBlockReflectance0", De, { replaceStrings: [{ search: /metallicReflectanceFactors/g, replace: this._vMetallicReflectanceFactorsName }] }); const qe = this.sheen.isConnected ? (P = this.sheen.connectedPoint) === null || P === void 0 ? void 0 : P.ownerBlock : null; qe && (e.compilationString += qe.getCode(Xe)), e._emitFunctionFromInclude("pbrBlockSheen", De, { replaceStrings: [ { search: /REFLECTIONMAP_3D/g, replace: (p = Xe == null ? void 0 : Xe._define3DName) !== null && p !== void 0 ? p : "REFLECTIONMAP_3D" }, { search: /REFLECTIONMAP_SKYBOX/g, replace: (c = Xe == null ? void 0 : Xe._defineSkyboxName) !== null && c !== void 0 ? c : "REFLECTIONMAP_SKYBOX" }, { search: /LODINREFLECTIONALPHA/g, replace: (H = Xe == null ? void 0 : Xe._defineLODReflectionAlpha) !== null && H !== void 0 ? H : "LODINREFLECTIONALPHA" }, { search: /LINEARSPECULARREFLECTION/g, replace: (T = Xe == null ? void 0 : Xe._defineLinearSpecularReflection) !== null && T !== void 0 ? T : "LINEARSPECULARREFLECTION" } ] }); const ke = this.iridescence.isConnected ? (q = this.iridescence.connectedPoint) === null || q === void 0 ? void 0 : q.ownerBlock : null; e.compilationString += eR.GetCode(ke), e._emitFunctionFromInclude("pbrBlockIridescence", De, { replaceStrings: [] }); const be = this.clearcoat.isConnected ? (b = this.clearcoat.connectedPoint) === null || b === void 0 ? void 0 : b.ownerBlock : null, Fe = !this.perturbedNormal.isConnected && !this.anisotropy.isConnected, Ke = this.perturbedNormal.isConnected && ((w = ((j = this.perturbedNormal.connectedPoint) === null || j === void 0 ? void 0 : j.ownerBlock).worldTangent) === null || w === void 0 ? void 0 : w.isConnected), nt = this.anisotropy.isConnected && ((m = this.anisotropy.connectedPoint) === null || m === void 0 ? void 0 : m.ownerBlock).worldTangent.isConnected; let ut = Ke || !this.perturbedNormal.isConnected && nt; e.compilationString += Xm.GetCode(e, be, Xe, re, Fe, ut, this.worldNormal.associatedVariableName), Fe && (ut = (I = be == null ? void 0 : be.worldTangent.isConnected) !== null && I !== void 0 ? I : !1), e._emitFunctionFromInclude("pbrBlockClearcoat", De, { replaceStrings: [ { search: /computeReflectionCoords/g, replace: "computeReflectionCoordsPBR" }, { search: /REFLECTIONMAP_3D/g, replace: (N = Xe == null ? void 0 : Xe._define3DName) !== null && N !== void 0 ? N : "REFLECTIONMAP_3D" }, { search: /REFLECTIONMAP_OPPOSITEZ/g, replace: (k = Xe == null ? void 0 : Xe._defineOppositeZ) !== null && k !== void 0 ? k : "REFLECTIONMAP_OPPOSITEZ" }, { search: /REFLECTIONMAP_PROJECTION/g, replace: (R = Xe == null ? void 0 : Xe._defineProjectionName) !== null && R !== void 0 ? R : "REFLECTIONMAP_PROJECTION" }, { search: /REFLECTIONMAP_SKYBOX/g, replace: (y = Xe == null ? void 0 : Xe._defineSkyboxName) !== null && y !== void 0 ? y : "REFLECTIONMAP_SKYBOX" }, { search: /LODINREFLECTIONALPHA/g, replace: (O = Xe == null ? void 0 : Xe._defineLODReflectionAlpha) !== null && O !== void 0 ? O : "LODINREFLECTIONALPHA" }, { search: /LINEARSPECULARREFLECTION/g, replace: (Y = Xe == null ? void 0 : Xe._defineLinearSpecularReflection) !== null && Y !== void 0 ? Y : "LINEARSPECULARREFLECTION" }, { search: /defined\(TANGENT\)/g, replace: ut ? "defined(TANGENT)" : "defined(IGNORE)" } ] }), e.compilationString += e._emitCodeFromInclude("pbrBlockReflectance", De, { replaceStrings: [ { search: /REFLECTIONMAP_SKYBOX/g, replace: (ee = Xe == null ? void 0 : Xe._defineSkyboxName) !== null && ee !== void 0 ? ee : "REFLECTIONMAP_SKYBOX" }, { search: /REFLECTIONMAP_3D/g, replace: (Z = Xe == null ? void 0 : Xe._define3DName) !== null && Z !== void 0 ? Z : "REFLECTIONMAP_3D" } ] }); const bt = this.subsurface.isConnected ? (te = this.subsurface.connectedPoint) === null || te === void 0 ? void 0 : te.ownerBlock : null, wt = this.subsurface.isConnected ? (_ = ((fe = this.subsurface.connectedPoint) === null || fe === void 0 ? void 0 : fe.ownerBlock).refraction.connectedPoint) === null || _ === void 0 ? void 0 : _.ownerBlock : null; wt && (wt.viewConnectionPoint = this.view, wt.indexOfRefractionConnectionPoint = this.indexOfRefraction), e.compilationString += CS.GetCode(e, bt, Xe, re), e._emitFunctionFromInclude("pbrBlockSubSurface", De, { replaceStrings: [ { search: /REFLECTIONMAP_3D/g, replace: (G = Xe == null ? void 0 : Xe._define3DName) !== null && G !== void 0 ? G : "REFLECTIONMAP_3D" }, { search: /REFLECTIONMAP_OPPOSITEZ/g, replace: (L = Xe == null ? void 0 : Xe._defineOppositeZ) !== null && L !== void 0 ? L : "REFLECTIONMAP_OPPOSITEZ" }, { search: /REFLECTIONMAP_PROJECTION/g, replace: ($ = Xe == null ? void 0 : Xe._defineProjectionName) !== null && $ !== void 0 ? $ : "REFLECTIONMAP_PROJECTION" }, { search: /SS_REFRACTIONMAP_3D/g, replace: (ae = wt == null ? void 0 : wt._define3DName) !== null && ae !== void 0 ? ae : "SS_REFRACTIONMAP_3D" }, { search: /SS_LODINREFRACTIONALPHA/g, replace: (Pe = wt == null ? void 0 : wt._defineLODRefractionAlpha) !== null && Pe !== void 0 ? Pe : "SS_LODINREFRACTIONALPHA" }, { search: /SS_LINEARSPECULARREFRACTION/g, replace: (ge = wt == null ? void 0 : wt._defineLinearSpecularRefraction) !== null && ge !== void 0 ? ge : "SS_LINEARSPECULARREFRACTION" }, { search: /SS_REFRACTIONMAP_OPPOSITEZ/g, replace: (me = wt == null ? void 0 : wt._defineOppositeZ) !== null && me !== void 0 ? me : "SS_REFRACTIONMAP_OPPOSITEZ" } ] }), e.compilationString += e._emitCodeFromInclude("pbrBlockDirectLighting", De), this.light ? e.compilationString += e._emitCodeFromInclude("lightFragment", De, { replaceStrings: [ { search: /{X}/g, replace: this._lightId.toString() }, { search: /vPositionW/g, replace: re + ".xyz" } ] }) : e.compilationString += e._emitCodeFromInclude("lightFragment", De, { repeatKey: "maxSimultaneousLights", substitutionVars: `vPositionW,${re}.xyz` }), e.compilationString += e._emitCodeFromInclude("pbrBlockFinalLitComponents", De), e.compilationString += `#endif `; const Tt = this.ambientColor.isConnected ? this.ambientColor.associatedVariableName : "vec3(0., 0., 0.)"; let lr = fs.DEFAULT_AO_ON_ANALYTICAL_LIGHTS.toString(); lr.indexOf(".") === -1 && (lr += "."), e.compilationString += e._emitCodeFromInclude("pbrBlockFinalUnlitComponents", De, { replaceStrings: [ { search: /vec3 finalEmissive[\s\S]*?finalEmissive\*=vLightingIntensity\.y;/g, replace: "" }, { search: /vAmbientColor/g, replace: Tt + " * ambientFromScene" }, { search: /vAmbientInfos\.w/g, replace: lr } ] }), e.compilationString += e._emitCodeFromInclude("pbrBlockFinalColorComposition", De, { replaceStrings: [{ search: /finalEmissive/g, replace: "vec3(0.)" }] }), e.compilationString += e._emitCodeFromInclude("pbrBlockImageProcessing", De, { replaceStrings: [{ search: /visibility/g, replace: "1." }] }), e.compilationString += e._emitCodeFromInclude("pbrDebug", De, { replaceStrings: [ { search: /vNormalW/g, replace: this._vNormalWName }, { search: /vPositionW/g, replace: re }, { search: /albedoTexture\.rgb;/g, replace: `vec3(1.); gl_FragColor.rgb = toGammaSpace(gl_FragColor.rgb); ` } ] }); for (const Qt of this._outputs) if (Qt.hasEndpoints) { const tr = Zpe[Qt.name]; if (tr) { const [br, Xn] = tr; Xn && (e.compilationString += `#if ${Xn} `), e.compilationString += `${this._declareOutput(Qt, e)} = ${br}; `, Xn && (e.compilationString += `#else `, e.compilationString += `${this._declareOutput(Qt, e)} = vec3(0.); `, e.compilationString += `#endif `); } else console.error(`There's no remapping for the ${Qt.name} end point! No code generated`); } return this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return e += `${this._codeVariableName}.lightFalloff = ${this.lightFalloff}; `, e += `${this._codeVariableName}.useAlphaTest = ${this.useAlphaTest}; `, e += `${this._codeVariableName}.alphaTestCutoff = ${this.alphaTestCutoff}; `, e += `${this._codeVariableName}.useAlphaBlending = ${this.useAlphaBlending}; `, e += `${this._codeVariableName}.useRadianceOverAlpha = ${this.useRadianceOverAlpha}; `, e += `${this._codeVariableName}.useSpecularOverAlpha = ${this.useSpecularOverAlpha}; `, e += `${this._codeVariableName}.enableSpecularAntiAliasing = ${this.enableSpecularAntiAliasing}; `, e += `${this._codeVariableName}.realTimeFiltering = ${this.realTimeFiltering}; `, e += `${this._codeVariableName}.realTimeFilteringQuality = ${this.realTimeFilteringQuality}; `, e += `${this._codeVariableName}.useEnergyConservation = ${this.useEnergyConservation}; `, e += `${this._codeVariableName}.useRadianceOcclusion = ${this.useRadianceOcclusion}; `, e += `${this._codeVariableName}.useHorizonOcclusion = ${this.useHorizonOcclusion}; `, e += `${this._codeVariableName}.unlit = ${this.unlit}; `, e += `${this._codeVariableName}.forceNormalForward = ${this.forceNormalForward}; `, e += `${this._codeVariableName}.debugMode = ${this.debugMode}; `, e += `${this._codeVariableName}.debugLimit = ${this.debugLimit}; `, e += `${this._codeVariableName}.debugFactor = ${this.debugFactor}; `, e; } serialize() { const e = super.serialize(); return this.light && (e.lightId = this.light.id), e.lightFalloff = this.lightFalloff, e.useAlphaTest = this.useAlphaTest, e.alphaTestCutoff = this.alphaTestCutoff, e.useAlphaBlending = this.useAlphaBlending, e.useRadianceOverAlpha = this.useRadianceOverAlpha, e.useSpecularOverAlpha = this.useSpecularOverAlpha, e.enableSpecularAntiAliasing = this.enableSpecularAntiAliasing, e.realTimeFiltering = this.realTimeFiltering, e.realTimeFilteringQuality = this.realTimeFilteringQuality, e.useEnergyConservation = this.useEnergyConservation, e.useRadianceOcclusion = this.useRadianceOcclusion, e.useHorizonOcclusion = this.useHorizonOcclusion, e.unlit = this.unlit, e.forceNormalForward = this.forceNormalForward, e.debugMode = this.debugMode, e.debugLimit = this.debugLimit, e.debugFactor = this.debugFactor, e.generateOnlyFragmentCode = this.generateOnlyFragmentCode, e; } _deserialize(e, t, r) { var n, i; super._deserialize(e, t, r), e.lightId && (this.light = t.getLightById(e.lightId)), this.lightFalloff = (n = e.lightFalloff) !== null && n !== void 0 ? n : 0, this.useAlphaTest = e.useAlphaTest, this.alphaTestCutoff = e.alphaTestCutoff, this.useAlphaBlending = e.useAlphaBlending, this.useRadianceOverAlpha = e.useRadianceOverAlpha, this.useSpecularOverAlpha = e.useSpecularOverAlpha, this.enableSpecularAntiAliasing = e.enableSpecularAntiAliasing, this.realTimeFiltering = !!e.realTimeFiltering, this.realTimeFilteringQuality = (i = e.realTimeFilteringQuality) !== null && i !== void 0 ? i : 8, this.useEnergyConservation = e.useEnergyConservation, this.useRadianceOcclusion = e.useRadianceOcclusion, this.useHorizonOcclusion = e.useHorizonOcclusion, this.unlit = e.unlit, this.forceNormalForward = !!e.forceNormalForward, this.debugMode = e.debugMode, this.debugLimit = e.debugLimit, this.debugFactor = e.debugFactor, this.generateOnlyFragmentCode = !!e.generateOnlyFragmentCode, this._setTarget(); } } C([ rn("Direct lights", Gr.Float, "INTENSITY", { min: 0, max: 1, notifiers: { update: !0 } }) ], g9.prototype, "directIntensity", void 0); C([ rn("Environment lights", Gr.Float, "INTENSITY", { min: 0, max: 1, notifiers: { update: !0 } }) ], g9.prototype, "environmentIntensity", void 0); C([ rn("Specular highlights", Gr.Float, "INTENSITY", { min: 0, max: 1, notifiers: { update: !0 } }) ], g9.prototype, "specularIntensity", void 0); C([ rn("Light falloff", Gr.List, "LIGHTING & COLORS", { notifiers: { update: !0 }, options: [ { label: "Physical", value: fs.LIGHTFALLOFF_PHYSICAL }, { label: "GLTF", value: fs.LIGHTFALLOFF_GLTF }, { label: "Standard", value: fs.LIGHTFALLOFF_STANDARD } ] }) ], g9.prototype, "lightFalloff", void 0); C([ rn("Alpha Testing", Gr.Boolean, "OPACITY") ], g9.prototype, "useAlphaTest", void 0); C([ rn("Alpha CutOff", Gr.Float, "OPACITY", { min: 0, max: 1, notifiers: { update: !0 } }) ], g9.prototype, "alphaTestCutoff", void 0); C([ rn("Alpha blending", Gr.Boolean, "OPACITY") ], g9.prototype, "useAlphaBlending", void 0); C([ rn("Radiance over alpha", Gr.Boolean, "RENDERING", { notifiers: { update: !0 } }) ], g9.prototype, "useRadianceOverAlpha", void 0); C([ rn("Specular over alpha", Gr.Boolean, "RENDERING", { notifiers: { update: !0 } }) ], g9.prototype, "useSpecularOverAlpha", void 0); C([ rn("Specular anti-aliasing", Gr.Boolean, "RENDERING", { notifiers: { update: !0 } }) ], g9.prototype, "enableSpecularAntiAliasing", void 0); C([ rn("Realtime filtering", Gr.Boolean, "RENDERING", { notifiers: { update: !0 } }) ], g9.prototype, "realTimeFiltering", void 0); C([ rn("Realtime filtering quality", Gr.List, "RENDERING", { notifiers: { update: !0 }, options: [ { label: "Low", value: 8 }, { label: "Medium", value: 16 }, { label: "High", value: 64 } ] }) ], g9.prototype, "realTimeFilteringQuality", void 0); C([ rn("Energy Conservation", Gr.Boolean, "ADVANCED", { notifiers: { update: !0 } }) ], g9.prototype, "useEnergyConservation", void 0); C([ rn("Radiance occlusion", Gr.Boolean, "ADVANCED", { notifiers: { update: !0 } }) ], g9.prototype, "useRadianceOcclusion", void 0); C([ rn("Horizon occlusion", Gr.Boolean, "ADVANCED", { notifiers: { update: !0 } }) ], g9.prototype, "useHorizonOcclusion", void 0); C([ rn("Unlit", Gr.Boolean, "ADVANCED", { notifiers: { update: !0 } }) ], g9.prototype, "unlit", void 0); C([ rn("Force normal forward", Gr.Boolean, "ADVANCED", { notifiers: { update: !0 } }) ], g9.prototype, "forceNormalForward", void 0); C([ rn("Generate only fragment code", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0, update: !0, onValidation: g9._OnGenerateOnlyFragmentCodeChanged } }) ], g9.prototype, "generateOnlyFragmentCode", void 0); C([ rn("Debug mode", Gr.List, "DEBUG", { notifiers: { update: !0 }, options: [ { label: "None", value: 0 }, // Geometry { label: "Normalized position", value: 1 }, { label: "Normals", value: 2 }, { label: "Tangents", value: 3 }, { label: "Bitangents", value: 4 }, { label: "Bump Normals", value: 5 }, //{ label: "UV1", value: 6 }, //{ label: "UV2", value: 7 }, { label: "ClearCoat Normals", value: 8 }, { label: "ClearCoat Tangents", value: 9 }, { label: "ClearCoat Bitangents", value: 10 }, { label: "Anisotropic Normals", value: 11 }, { label: "Anisotropic Tangents", value: 12 }, { label: "Anisotropic Bitangents", value: 13 }, // Maps //{ label: "Emissive Map", value: 23 }, //{ label: "Light Map", value: 24 }, // Env { label: "Env Refraction", value: 40 }, { label: "Env Reflection", value: 41 }, { label: "Env Clear Coat", value: 42 }, // Lighting { label: "Direct Diffuse", value: 50 }, { label: "Direct Specular", value: 51 }, { label: "Direct Clear Coat", value: 52 }, { label: "Direct Sheen", value: 53 }, { label: "Env Irradiance", value: 54 }, // Lighting Params { label: "Surface Albedo", value: 60 }, { label: "Reflectance 0", value: 61 }, { label: "Metallic", value: 62 }, { label: "Metallic F0", value: 71 }, { label: "Roughness", value: 63 }, { label: "AlphaG", value: 64 }, { label: "NdotV", value: 65 }, { label: "ClearCoat Color", value: 66 }, { label: "ClearCoat Roughness", value: 67 }, { label: "ClearCoat NdotV", value: 68 }, { label: "Transmittance", value: 69 }, { label: "Refraction Transmittance", value: 70 }, // Misc { label: "SEO", value: 80 }, { label: "EHO", value: 81 }, { label: "Energy Factor", value: 82 }, { label: "Specular Reflectance", value: 83 }, { label: "Clear Coat Reflectance", value: 84 }, { label: "Sheen Reflectance", value: 85 }, { label: "Luminance Over Alpha", value: 86 }, { label: "Alpha", value: 87 } ] }) ], g9.prototype, "debugMode", void 0); C([ rn("Split position", Gr.Float, "DEBUG", { min: -1, max: 1, notifiers: { update: !0 } }) ], g9.prototype, "debugLimit", void 0); C([ rn("Output factor", Gr.Float, "DEBUG", { min: 0, max: 5, notifiers: { update: !0 } }) ], g9.prototype, "debugFactor", void 0); Ue("BABYLON.PBRMetallicRoughnessBlock", g9); class rne extends Mr { /** * Creates a new ModBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("left", de.AutoDetect), this.registerInput("right", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._linkConnectionTypes(0, 1), this._inputs[1].acceptedConnectionPointTypes.push(de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ModBlock"; } /** * Gets the left operand input component */ get left() { return this._inputs[0]; } /** * Gets the right operand input component */ get right() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = mod(${this.left.associatedVariableName}, ${this.right.associatedVariableName}); `, this; } } Ue("BABYLON.ModBlock", rne); class nne extends Mr { /** * Creates a new MatrixBuilder * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("row0", de.Vector4), this.registerInput("row1", de.Vector4), this.registerInput("row2", de.Vector4), this.registerInput("row3", de.Vector4), this.registerOutput("output", de.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "MatrixBuilder"; } /** * Gets the row0 vector */ get row0() { return this._inputs[0]; } /** * Gets the row1 vector */ get row1() { return this._inputs[1]; } /** * Gets the row2 vector */ get row2() { return this._inputs[2]; } /** * Gets the row3 vector */ get row3() { return this._inputs[3]; } /** * Gets the output component */ get output() { return this._outputs[0]; } autoConfigure() { if (!this.row0.isConnected) { const e = new gi("row0"); e.value = new Ir(1, 0, 0, 0), e.output.connectTo(this.row0); } if (!this.row1.isConnected) { const e = new gi("row1"); e.value = new Ir(0, 1, 0, 0), e.output.connectTo(this.row1); } if (!this.row2.isConnected) { const e = new gi("row2"); e.value = new Ir(0, 0, 1, 0), e.output.connectTo(this.row2); } if (!this.row3.isConnected) { const e = new gi("row3"); e.value = new Ir(0, 0, 0, 1), e.output.connectTo(this.row3); } } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0], r = this.row0, n = this.row1, i = this.row2, s = this.row3; return e.compilationString += this._declareOutput(t, e) + ` = mat4(${r.associatedVariableName}, ${n.associatedVariableName}, ${i.associatedVariableName}, ${s.associatedVariableName}); `, this; } } Ue("BABYLON.MatrixBuilder", nne); var lc; (function(A) { A[A.Equal = 0] = "Equal", A[A.NotEqual = 1] = "NotEqual", A[A.LessThan = 2] = "LessThan", A[A.GreaterThan = 3] = "GreaterThan", A[A.LessOrEqual = 4] = "LessOrEqual", A[A.GreaterOrEqual = 5] = "GreaterOrEqual", A[A.Xor = 6] = "Xor", A[A.Or = 7] = "Or", A[A.And = 8] = "And"; })(lc || (lc = {})); class ine extends Mr { /** * Creates a new ConditionalBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.condition = lc.LessThan, this.registerInput("a", de.Float), this.registerInput("b", de.Float), this.registerInput("true", de.AutoDetect, !0), this.registerInput("false", de.AutoDetect, !0), this.registerOutput("output", de.BasedOnInput), this._linkConnectionTypes(2, 3), this._outputs[0]._typeConnectionSource = this._inputs[2], this._outputs[0]._defaultConnectionPointType = de.Float; } /** * Gets the current class name * @returns the class name */ getClassName() { return "ConditionalBlock"; } /** * Gets the first operand component */ get a() { return this._inputs[0]; } /** * Gets the second operand component */ get b() { return this._inputs[1]; } /** * Gets the value to return if condition is true */ get true() { return this._inputs[2]; } /** * Gets the value to return if condition is false */ get false() { return this._inputs[3]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0], r = this.true.isConnected ? this.true.associatedVariableName : "1.0", n = this.false.isConnected ? this.false.associatedVariableName : "0.0"; switch (this.condition) { case lc.Equal: { e.compilationString += this._declareOutput(t, e) + ` = ${this.a.associatedVariableName} == ${this.b.associatedVariableName} ? ${r} : ${n}; `; break; } case lc.NotEqual: { e.compilationString += this._declareOutput(t, e) + ` = ${this.a.associatedVariableName} != ${this.b.associatedVariableName} ? ${r} : ${n}; `; break; } case lc.LessThan: { e.compilationString += this._declareOutput(t, e) + ` = ${this.a.associatedVariableName} < ${this.b.associatedVariableName} ? ${r} : ${n}; `; break; } case lc.LessOrEqual: { e.compilationString += this._declareOutput(t, e) + ` = ${this.a.associatedVariableName} <= ${this.b.associatedVariableName} ? ${r} : ${n}; `; break; } case lc.GreaterThan: { e.compilationString += this._declareOutput(t, e) + ` = ${this.a.associatedVariableName} > ${this.b.associatedVariableName} ? ${r} : ${n}; `; break; } case lc.GreaterOrEqual: { e.compilationString += this._declareOutput(t, e) + ` = ${this.a.associatedVariableName} >= ${this.b.associatedVariableName} ? ${r} : ${n}; `; break; } case lc.Xor: { e.compilationString += this._declareOutput(t, e) + ` = (mod(${this.a.associatedVariableName} + ${this.b.associatedVariableName}, 2.0) > 0.0) ? ${r} : ${n}; `; break; } case lc.Or: { e.compilationString += this._declareOutput(t, e) + ` = (min(${this.a.associatedVariableName} + ${this.b.associatedVariableName}, 1.0) > 0.0) ? ${r} : ${n}; `; break; } case lc.And: { e.compilationString += this._declareOutput(t, e) + ` = (${this.a.associatedVariableName} * ${this.b.associatedVariableName} > 0.0) ? ${r} : ${n}; `; break; } } return this; } serialize() { const e = super.serialize(); return e.condition = this.condition, e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.condition = e.condition; } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.condition = BABYLON.ConditionalBlockConditions.${lc[this.condition]}; `; } } Ue("BABYLON.ConditionalBlock", ine); class oY extends Mr { /** * Creates a new CloudBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.octaves = 6, this.registerInput("seed", de.AutoDetect), this.registerInput("chaos", de.AutoDetect, !0), this.registerInput("offsetX", de.Float, !0), this.registerInput("offsetY", de.Float, !0), this.registerInput("offsetZ", de.Float, !0), this.registerOutput("output", de.Float), this._inputs[0].acceptedConnectionPointTypes.push(de.Vector2), this._inputs[0].acceptedConnectionPointTypes.push(de.Vector3), this._linkConnectionTypes(0, 1); } /** * Gets the current class name * @returns the class name */ getClassName() { return "CloudBlock"; } /** * Gets the seed input component */ get seed() { return this._inputs[0]; } /** * Gets the chaos input component */ get chaos() { return this._inputs[1]; } /** * Gets the offset X input component */ get offsetX() { return this._inputs[2]; } /** * Gets the offset Y input component */ get offsetY() { return this._inputs[3]; } /** * Gets the offset Z input component */ get offsetZ() { return this._inputs[4]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { var t, r; if (super._buildBlock(e), !this.seed.isConnected || !this._outputs[0].hasEndpoints) return; const n = ` float cloudRandom(in float p) { p = fract(p * 0.011); p *= p + 7.5; p *= p + p; return fract(p); } // Based on Morgan McGuire @morgan3d // https://www.shadertoy.com/view/4dS3Wd float cloudNoise(in vec2 x, in vec2 chaos) { vec2 step = chaos * vec2(75., 120.) + vec2(75., 120.); vec2 i = floor(x); vec2 f = fract(x); float n = dot(i, step); vec2 u = f * f * (3.0 - 2.0 * f); return mix( mix(cloudRandom(n + dot(step, vec2(0, 0))), cloudRandom(n + dot(step, vec2(1, 0))), u.x), mix(cloudRandom(n + dot(step, vec2(0, 1))), cloudRandom(n + dot(step, vec2(1, 1))), u.x), u.y ); } float cloudNoise(in vec3 x, in vec3 chaos) { vec3 step = chaos * vec3(60., 120., 75.) + vec3(60., 120., 75.); vec3 i = floor(x); vec3 f = fract(x); float n = dot(i, step); vec3 u = f * f * (3.0 - 2.0 * f); return mix(mix(mix( cloudRandom(n + dot(step, vec3(0, 0, 0))), cloudRandom(n + dot(step, vec3(1, 0, 0))), u.x), mix( cloudRandom(n + dot(step, vec3(0, 1, 0))), cloudRandom(n + dot(step, vec3(1, 1, 0))), u.x), u.y), mix(mix( cloudRandom(n + dot(step, vec3(0, 0, 1))), cloudRandom(n + dot(step, vec3(1, 0, 1))), u.x), mix( cloudRandom(n + dot(step, vec3(0, 1, 1))), cloudRandom(n + dot(step, vec3(1, 1, 1))), u.x), u.y), u.z); }`, i = ` float fbm(in vec2 st, in vec2 chaos) { // Initial values float value = 0.0; float amplitude = .5; float frequency = 0.; // Loop of octaves for (int i = 0; i < OCTAVES; i++) { value += amplitude * cloudNoise(st, chaos); st *= 2.0; amplitude *= 0.5; } return value; } float fbm(in vec3 x, in vec3 chaos) { // Initial values float value = 0.0; float amplitude = 0.5; for (int i = 0; i < OCTAVES; ++i) { value += amplitude * cloudNoise(x, chaos); x = x * 2.0; amplitude *= 0.5; } return value; }`, s = `fbm${this.octaves}`; e._emitFunction("CloudBlockCode", n, "// CloudBlockCode"), e._emitFunction("CloudBlockCodeFBM" + this.octaves, i.replace(/fbm/gi, s).replace(/OCTAVES/gi, (this.octaves | 0).toString()), "// CloudBlockCode FBM"); const a = e._getFreeVariableName("st"), f = ((t = this.seed.connectedPoint) === null || t === void 0 ? void 0 : t.type) === de.Vector2 ? "vec2" : "vec3"; e.compilationString += `${f} ${a} = ${this.seed.associatedVariableName}; `, this.offsetX.isConnected && (e.compilationString += `${a}.x += 0.1 * ${this.offsetX.associatedVariableName}; `), this.offsetY.isConnected && (e.compilationString += `${a}.y += 0.1 * ${this.offsetY.associatedVariableName}; `), this.offsetZ.isConnected && f === "vec3" && (e.compilationString += `${a}.z += 0.1 * ${this.offsetZ.associatedVariableName}; `); let o = ""; return this.chaos.isConnected ? o = this.chaos.associatedVariableName : o = ((r = this.seed.connectedPoint) === null || r === void 0 ? void 0 : r.type) === de.Vector2 ? "vec2(0., 0.)" : "vec3(0., 0., 0.)", e.compilationString += this._declareOutput(this._outputs[0], e) + ` = ${s}(${a}, ${o}); `, this; } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.octaves = ${this.octaves}; `; } serialize() { const e = super.serialize(); return e.octaves = this.octaves, e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.octaves = e.octaves; } } C([ rn("Octaves", Gr.Int) ], oY.prototype, "octaves", void 0); Ue("BABYLON.CloudBlock", oY); class sne extends Mr { /** * Creates a new VoronoiNoiseBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("seed", de.Vector2), this.registerInput("offset", de.Float), this.registerInput("density", de.Float), this.registerOutput("output", de.Float), this.registerOutput("cells", de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "VoronoiNoiseBlock"; } /** * Gets the seed input component */ get seed() { return this._inputs[0]; } /** * Gets the offset input component */ get offset() { return this._inputs[1]; } /** * Gets the density input component */ get density() { return this._inputs[2]; } /** * Gets the output component */ get output() { return this._outputs[0]; } /** * Gets the output component */ get cells() { return this._outputs[1]; } _buildBlock(e) { if (super._buildBlock(e), !this.seed.isConnected) return; let t = `vec2 voronoiRandom(vec2 seed, float offset){ mat2 m = mat2(15.27, 47.63, 99.41, 89.98); vec2 uv = fract(sin(m * seed) * 46839.32); return vec2(sin(uv.y * offset) * 0.5 + 0.5, cos(uv.x * offset) * 0.5 + 0.5); } `; e._emitFunction("voronoiRandom", t, "// Voronoi random generator"), t = `void voronoi(vec2 seed, float offset, float density, out float outValue, out float cells){ vec2 g = floor(seed * density); vec2 f = fract(seed * density); float t = 8.0; vec3 res = vec3(8.0, 0.0, 0.0); for(int y=-1; y<=1; y++) { for(int x=-1; x<=1; x++) { vec2 lattice = vec2(x,y); vec2 randomOffset = voronoiRandom(lattice + g, offset); float d = distance(lattice + randomOffset, f); if(d < res.x) { res = vec3(d, randomOffset.x, randomOffset.y); outValue = res.x; cells = res.y; } } } } `, e._emitFunction("voronoi", t, "// Voronoi"); const r = e._getFreeVariableName("tempOutput"), n = e._getFreeVariableName("tempCells"); return e.compilationString += `float ${r} = 0.0; `, e.compilationString += `float ${n} = 0.0; `, e.compilationString += `voronoi(${this.seed.associatedVariableName}, ${this.offset.associatedVariableName}, ${this.density.associatedVariableName}, ${r}, ${n}); `, this.output.hasEndpoints && (e.compilationString += this._declareOutput(this.output, e) + ` = ${r}; `), this.cells.hasEndpoints && (e.compilationString += this._declareOutput(this.cells, e) + ` = ${n}; `), this; } } Ue("BABYLON.VoronoiNoiseBlock", sne); class ane extends Mr { /** * Creates a new ElbowBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("input", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0]; } /** * Gets the current class name * @returns the class name */ getClassName() { return "ElbowBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } /** * Gets or sets the target of the block */ get target() { const e = this._inputs[0]; if (e.isConnected) { const t = e.connectedPoint.ownerBlock; if (t.target !== Ve.VertexAndFragment) return t.target; if (e.connectedPoint.target !== Ve.VertexAndFragment) return e.connectedPoint.target; } return this._target; } set target(e) { this._target & e || (this._target = e); } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0], r = this._inputs[0]; return e.compilationString += this._declareOutput(t, e) + ` = ${r.associatedVariableName}; `, this; } } Ue("BABYLON.ElbowBlock", ane); class gy extends Mr { /** * Gets or sets the texture associated with the node */ get texture() { var e; return this.source.isConnected ? ((e = this.source.connectedPoint) === null || e === void 0 ? void 0 : e.ownerBlock).texture : this._texture; } set texture(e) { var t; if (this._texture === e) return; const r = (t = e == null ? void 0 : e.getScene()) !== null && t !== void 0 ? t : gr.LastCreatedScene; !e && r && r.markAllMaterialsAsDirty(1, (n) => n.hasTexture(this._texture)), this._texture = e, e && r && r.markAllMaterialsAsDirty(1, (n) => n.hasTexture(e)); } /** * Gets the textureY associated with the node */ get textureY() { var e; return this.sourceY.isConnected ? ((e = this.sourceY.connectedPoint) === null || e === void 0 ? void 0 : e.ownerBlock).texture : null; } /** * Gets the textureZ associated with the node */ get textureZ() { var e, t; return !((e = this.sourceZ) === null || e === void 0) && e.isConnected ? ((t = this.sourceY.connectedPoint) === null || t === void 0 ? void 0 : t.ownerBlock).texture : null; } _getImageSourceBlock(e) { return e != null && e.isConnected ? e.connectedPoint.ownerBlock : null; } /** * Gets the sampler name associated with this texture */ get samplerName() { const e = this._getImageSourceBlock(this.source); return e ? e.samplerName : this._samplerName; } /** * Gets the samplerY name associated with this texture */ get samplerYName() { var e, t; return (t = (e = this._getImageSourceBlock(this.sourceY)) === null || e === void 0 ? void 0 : e.samplerName) !== null && t !== void 0 ? t : null; } /** * Gets the samplerZ name associated with this texture */ get samplerZName() { var e, t; return (t = (e = this._getImageSourceBlock(this.sourceZ)) === null || e === void 0 ? void 0 : e.samplerName) !== null && t !== void 0 ? t : null; } /** * Gets a boolean indicating that this block is linked to an ImageSourceBlock */ get hasImageSource() { return this.source.isConnected; } /** * Gets or sets a boolean indicating if content needs to be converted to gamma space */ set convertToGammaSpace(e) { var t; if (e !== this._convertToGammaSpace && (this._convertToGammaSpace = e, this.texture)) { const r = (t = this.texture.getScene()) !== null && t !== void 0 ? t : gr.LastCreatedScene; r == null || r.markAllMaterialsAsDirty(1, (n) => n.hasTexture(this.texture)); } } get convertToGammaSpace() { return this._convertToGammaSpace; } /** * Gets or sets a boolean indicating if content needs to be converted to linear space */ set convertToLinearSpace(e) { var t; if (e !== this._convertToLinearSpace && (this._convertToLinearSpace = e, this.texture)) { const r = (t = this.texture.getScene()) !== null && t !== void 0 ? t : gr.LastCreatedScene; r == null || r.markAllMaterialsAsDirty(1, (n) => n.hasTexture(this.texture)); } } get convertToLinearSpace() { return this._convertToLinearSpace; } /** * Create a new TriPlanarBlock * @param name defines the block name */ constructor(e, t = !1) { super(e, Ve.Neutral), this.projectAsCube = !1, this._convertToGammaSpace = !1, this._convertToLinearSpace = !1, this.disableLevelMultiplication = !1, this.registerInput("position", de.AutoDetect, !1), this.registerInput("normal", de.AutoDetect, !1), this.registerInput("sharpness", de.Float, !0), this.registerInput("source", de.Object, !0, Ve.VertexAndFragment, new Ko("source", this, ao.Input, E0, "ImageSourceBlock")), this.registerInput("sourceY", de.Object, !0, Ve.VertexAndFragment, new Ko("sourceY", this, ao.Input, E0, "ImageSourceBlock")), t || this.registerInput("sourceZ", de.Object, !0, Ve.VertexAndFragment, new Ko("sourceZ", this, ao.Input, E0, "ImageSourceBlock")), this.registerOutput("rgba", de.Color4, Ve.Neutral), this.registerOutput("rgb", de.Color3, Ve.Neutral), this.registerOutput("r", de.Float, Ve.Neutral), this.registerOutput("g", de.Float, Ve.Neutral), this.registerOutput("b", de.Float, Ve.Neutral), this.registerOutput("a", de.Float, Ve.Neutral), this.registerOutput("level", de.Float, Ve.Neutral), this._inputs[0].addExcludedConnectionPointFromAllowedTypes(de.Color3 | de.Vector3 | de.Vector4), this._inputs[1].addExcludedConnectionPointFromAllowedTypes(de.Color3 | de.Vector3 | de.Vector4); } /** * Gets the current class name * @returns the class name */ getClassName() { return "TriPlanarBlock"; } /** * Gets the position input component */ get position() { return this._inputs[0]; } /** * Gets the normal input component */ get normal() { return this._inputs[1]; } /** * Gets the sharpness input component */ get sharpness() { return this._inputs[2]; } /** * Gets the source input component */ get source() { return this._inputs[3]; } /** * Gets the sourceY input component */ get sourceY() { return this._inputs[4]; } /** * Gets the sourceZ input component */ get sourceZ() { return this._inputs[5]; } /** * Gets the rgba output component */ get rgba() { return this._outputs[0]; } /** * Gets the rgb output component */ get rgb() { return this._outputs[1]; } /** * Gets the r output component */ get r() { return this._outputs[2]; } /** * Gets the g output component */ get g() { return this._outputs[3]; } /** * Gets the b output component */ get b() { return this._outputs[4]; } /** * Gets the a output component */ get a() { return this._outputs[5]; } /** * Gets the level output component */ get level() { return this._outputs[6]; } prepareDefines(e, t, r) { if (!r._areTexturesDirty) return; const n = this.convertToGammaSpace && this.texture && !this.texture.gammaSpace, i = this.convertToLinearSpace && this.texture && this.texture.gammaSpace; r.setValue(this._linearDefineName, n, !0), r.setValue(this._gammaDefineName, i, !0); } isReady() { return !(this.texture && !this.texture.isReadyOrNotBlocking()); } bind(e) { this.texture && (e.setFloat(this._textureInfoName, this.texture.level), this._imageSource || e.setTexture(this._samplerName, this.texture)); } _generateTextureLookup(e) { var t, r; const n = this.samplerName, i = (t = this.samplerYName) !== null && t !== void 0 ? t : n, s = (r = this.samplerZName) !== null && r !== void 0 ? r : n, a = this.sharpness.isConnected ? this.sharpness.associatedVariableName : "1.0", f = e._getFreeVariableName("x"), o = e._getFreeVariableName("y"), d = e._getFreeVariableName("z"), v = e._getFreeVariableName("w"), u = e._getFreeVariableName("n"), l = e._getFreeVariableName("uvx"), P = e._getFreeVariableName("uvy"), p = e._getFreeVariableName("uvz"); e.compilationString += ` vec3 ${u} = ${this.normal.associatedVariableName}.xyz; vec2 ${l} = ${this.position.associatedVariableName}.yz; vec2 ${P} = ${this.position.associatedVariableName}.zx; vec2 ${p} = ${this.position.associatedVariableName}.xy; `, this.projectAsCube && (e.compilationString += ` ${l}.xy = ${l}.yx; if (${u}.x >= 0.0) { ${l}.x = -${l}.x; } if (${u}.y < 0.0) { ${P}.y = -${P}.y; } if (${u}.z < 0.0) { ${p}.x = -${p}.x; } `), e.compilationString += ` vec4 ${f} = texture2D(${n}, ${l}); vec4 ${o} = texture2D(${i}, ${P}); vec4 ${d} = texture2D(${s}, ${p}); // blend weights vec3 ${v} = pow(abs(${u}), vec3(${a})); // blend and return vec4 ${this._tempTextureRead} = (${f}*${v}.x + ${o}*${v}.y + ${d}*${v}.z) / (${v}.x + ${v}.y + ${v}.z); `; } _generateConversionCode(e, t, r) { r !== "a" && ((!this.texture || !this.texture.gammaSpace) && (e.compilationString += `#ifdef ${this._linearDefineName} ${t.associatedVariableName} = toGammaSpace(${t.associatedVariableName}); #endif `), e.compilationString += `#ifdef ${this._gammaDefineName} ${t.associatedVariableName} = toLinearSpace(${t.associatedVariableName}); #endif `); } _writeOutput(e, t, r) { let n = ""; this.disableLevelMultiplication || (n = ` * ${this._textureInfoName}`), e.compilationString += `${this._declareOutput(t, e)} = ${this._tempTextureRead}.${r}${n}; `, this._generateConversionCode(e, t, r); } _buildBlock(e) { super._buildBlock(e), this.source.isConnected ? this._imageSource = this.source.connectedPoint.ownerBlock : this._imageSource = null, this._textureInfoName = e._getFreeVariableName("textureInfoName"), this.level.associatedVariableName = this._textureInfoName, this._tempTextureRead = e._getFreeVariableName("tempTextureRead"), this._linearDefineName = e._getFreeDefineName("ISLINEAR"), this._gammaDefineName = e._getFreeDefineName("ISGAMMA"), this._imageSource || (this._samplerName = e._getFreeVariableName(this.name + "Sampler"), e._emit2DSampler(this._samplerName)), e.sharedData.blockingBlocks.push(this), e.sharedData.textureBlocks.push(this), e.sharedData.blocksWithDefines.push(this), e.sharedData.bindableBlocks.push(this); const t = `//${this.name}`; e._emitFunctionFromInclude("helperFunctions", t), e._emitUniformFromString(this._textureInfoName, "float"), this._generateTextureLookup(e); for (const r of this._outputs) r.hasEndpoints && r.name !== "level" && this._writeOutput(e, r, r.name); return this; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return e += `${this._codeVariableName}.convertToGammaSpace = ${this.convertToGammaSpace}; `, e += `${this._codeVariableName}.convertToLinearSpace = ${this.convertToLinearSpace}; `, e += `${this._codeVariableName}.disableLevelMultiplication = ${this.disableLevelMultiplication}; `, e += `${this._codeVariableName}.projectAsCube = ${this.projectAsCube}; `, this.texture && (e += `${this._codeVariableName}.texture = new BABYLON.Texture("${this.texture.name}", null, ${this.texture.noMipmap}, ${this.texture.invertY}, ${this.texture.samplingMode}); `, e += `${this._codeVariableName}.texture.wrapU = ${this.texture.wrapU}; `, e += `${this._codeVariableName}.texture.wrapV = ${this.texture.wrapV}; `, e += `${this._codeVariableName}.texture.uAng = ${this.texture.uAng}; `, e += `${this._codeVariableName}.texture.vAng = ${this.texture.vAng}; `, e += `${this._codeVariableName}.texture.wAng = ${this.texture.wAng}; `, e += `${this._codeVariableName}.texture.uOffset = ${this.texture.uOffset}; `, e += `${this._codeVariableName}.texture.vOffset = ${this.texture.vOffset}; `, e += `${this._codeVariableName}.texture.uScale = ${this.texture.uScale}; `, e += `${this._codeVariableName}.texture.vScale = ${this.texture.vScale}; `, e += `${this._codeVariableName}.texture.coordinatesMode = ${this.texture.coordinatesMode}; `), e; } serialize() { const e = super.serialize(); return e.convertToGammaSpace = this.convertToGammaSpace, e.convertToLinearSpace = this.convertToLinearSpace, e.disableLevelMultiplication = this.disableLevelMultiplication, e.projectAsCube = this.projectAsCube, !this.hasImageSource && this.texture && !this.texture.isRenderTarget && this.texture.getClassName() !== "VideoTexture" && (e.texture = this.texture.serialize()), e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.convertToGammaSpace = e.convertToGammaSpace, this.convertToLinearSpace = !!e.convertToLinearSpace, this.disableLevelMultiplication = !!e.disableLevelMultiplication, this.projectAsCube = !!e.projectAsCube, e.texture && !ja.IgnoreTexturesAtLoadTime && e.texture.url !== void 0 && (r = e.texture.url.indexOf("data:") === 0 ? "" : r, this.texture = We.Parse(e.texture, t, r)); } } C([ rn("Project as cube", Gr.Boolean, "ADVANCED", { notifiers: { update: !0 } }) ], gy.prototype, "projectAsCube", void 0); Ue("BABYLON.TriPlanarBlock", gy); class one extends gy { /** * Create a new BiPlanarBlock * @param name defines the block name */ constructor(e) { super(e, !0); } /** * Gets the current class name * @returns the class name */ getClassName() { return "BiPlanarBlock"; } _generateTextureLookup(e) { var t; const r = this.samplerName, n = (t = this.samplerYName) !== null && t !== void 0 ? t : this.samplerName, i = this.sharpness.isConnected ? this.sharpness.associatedVariableName : "1.0", s = e._getFreeVariableName("dpdx"), a = e._getFreeVariableName("dpdy"), f = e._getFreeVariableName("n"), o = e._getFreeVariableName("ma"), d = e._getFreeVariableName("mi"), v = e._getFreeVariableName("me"), u = e._getFreeVariableName("x"), l = e._getFreeVariableName("y"), P = e._getFreeVariableName("y"); e.compilationString += ` // grab coord derivatives for texturing vec3 ${s} = dFdx(${this.position.associatedVariableName}.xyz); vec3 ${a} = dFdy(${this.position.associatedVariableName}.xyz); vec3 ${f} = abs(${this.normal.associatedVariableName}.xyz); // determine major axis (in x; yz are following axis) ivec3 ${o} = (${f}.x>${f}.y && ${f}.x>${f}.z) ? ivec3(0,1,2) : (${f}.y>${f}.z) ? ivec3(1,2,0) : ivec3(2,0,1) ; // determine minor axis (in x; yz are following axis) ivec3 ${d} = (${f}.x<${f}.y && ${f}.x<${f}.z) ? ivec3(0,1,2) : (${f}.y<${f}.z) ? ivec3(1,2,0) : ivec3(2,0,1) ; // determine median axis (in x; yz are following axis) ivec3 ${v} = ivec3(3) - ${d} - ${o}; // project+fetch vec4 ${u} = textureGrad( ${r}, vec2( ${this.position.associatedVariableName}[${o}.y], ${this.position.associatedVariableName}[${o}.z]), vec2(${s}[${o}.y],${s}[${o}.z]), vec2(${a}[${o}.y],${a}[${o}.z]) ); vec4 ${l} = textureGrad( ${n}, vec2( ${this.position.associatedVariableName}[${v}.y], ${this.position.associatedVariableName}[${v}.z]), vec2(${s}[${v}.y],${s}[${v}.z]), vec2(${a}[${v}.y],${a}[${v}.z]) ); // blend factors vec2 ${P} = vec2(${f}[${o}.x],${f}[${v}.x]); // make local support ${P} = clamp( (${P}-0.5773)/(1.0-0.5773), 0.0, 1.0 ); // shape transition ${P} = pow( ${P}, vec2(${i}/8.0) ); // blend and return vec4 ${this._tempTextureRead} = (${u}*${P}.x + ${l}*${P}.y) / (${P}.x + ${P}.y); `; } } Ue("BABYLON.BiPlanarBlock", one); class fne extends Mr { /** * Creates a new MatrixDeterminantBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("input", de.Matrix), this.registerOutput("output", de.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "MatrixDeterminantBlock"; } /** * Gets the input matrix */ get input() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this.output, r = this.input; return e.compilationString += this._declareOutput(t, e) + `${t.associatedVariableName} = determinant(${r.associatedVariableName}); `, this; } } Ue("BABYLON.MatrixDeterminantBlock", fne); class Ane extends Mr { /** * Creates a new MatrixTransposeBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.registerInput("input", de.Matrix), this.registerOutput("output", de.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "MatrixTransposeBlock"; } /** * Gets the input matrix */ get input() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this.output, r = this.input; return e.compilationString += this._declareOutput(t, e) + `${t.associatedVariableName} = transpose(${r.associatedVariableName}); `, this; } } Ue("BABYLON.MatrixTransposeBlock", Ane); var xa; (function(A) { A[A.None = 0] = "None", A[A.Normal = 1] = "Normal", A[A.Tangent = 2] = "Tangent", A[A.VertexColor = 3] = "VertexColor", A[A.UV1 = 4] = "UV1", A[A.UV2 = 5] = "UV2", A[A.UV3 = 6] = "UV3", A[A.UV4 = 7] = "UV4", A[A.UV5 = 8] = "UV5", A[A.UV6 = 9] = "UV6"; })(xa || (xa = {})); class fY extends Mr { /** * Creates a new MeshAttributeExistsBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.attributeType = xa.None, this.registerInput("input", de.AutoDetect), this.registerInput("fallback", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._linkConnectionTypes(0, 1), this._inputs[0].onConnectionObservable.add((t) => { var r; if (this.attributeType) return; const n = t.ownerBlock; if (n instanceof gi && n.isAttribute) switch (n.name) { case "color": this.attributeType = xa.VertexColor; break; case "normal": this.attributeType = xa.Normal; break; case "tangent": this.attributeType = xa.Tangent; break; case "uv": this.attributeType = xa.UV1; break; case "uv2": this.attributeType = xa.UV2; break; case "uv3": this.attributeType = xa.UV3; break; case "uv4": this.attributeType = xa.UV4; break; case "uv5": this.attributeType = xa.UV5; break; case "uv6": this.attributeType = xa.UV6; break; } else if (n instanceof rY) switch ((r = this.input.connectedPoint) === null || r === void 0 ? void 0 : r.name) { case "normalOutput": this.attributeType = xa.Normal; break; case "tangentOutput": this.attributeType = xa.Tangent; break; case "uvOutput": this.attributeType = xa.UV1; break; } }); } /** * Gets the current class name * @returns the class name */ getClassName() { return "MeshAttributeExistsBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the fallback component when speciefied attribute doesn't exist */ get fallback() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); let t = null; switch (this.attributeType) { case xa.VertexColor: t = "VERTEXCOLOR_NME"; break; case xa.Normal: t = "NORMAL"; break; case xa.Tangent: t = "TANGENT"; break; case xa.UV1: t = "UV1"; break; case xa.UV2: t = "UV2"; break; case xa.UV3: t = "UV3"; break; case xa.UV4: t = "UV4"; break; case xa.UV5: t = "UV5"; break; case xa.UV6: t = "UV6"; break; } const r = this._declareOutput(this.output, e); return t && (e.compilationString += `#ifdef ${t} `), e.compilationString += `${r} = ${this.input.associatedVariableName}; `, t && (e.compilationString += `#else `, e.compilationString += `${r} = ${this.fallback.associatedVariableName}; `, e.compilationString += `#endif `), this; } serialize() { const e = super.serialize(); return e.attributeType = this.attributeType, e; } _deserialize(e, t, r) { var n; super._deserialize(e, t, r), this.attributeType = (n = e.attributeType) !== null && n !== void 0 ? n : xa.None; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return e += `${this._codeVariableName}.attributeType = ${this.attributeType}; `, e; } } C([ rn("Attribute lookup", Gr.List, void 0, { notifiers: { update: !0 }, options: [ { label: "(None)", value: xa.None }, { label: "Normal", value: xa.Normal }, { label: "Tangent", value: xa.Tangent }, { label: "Vertex Color", value: xa.VertexColor }, { label: "UV1", value: xa.UV1 }, { label: "UV2", value: xa.UV2 }, { label: "UV3", value: xa.UV3 }, { label: "UV4", value: xa.UV4 }, { label: "UV5", value: xa.UV5 }, { label: "UV6", value: xa.UV6 } ] }) ], fY.prototype, "attributeType", void 0); Ue("BABYLON.MeshAttributeExistsBlock", fY); var no; (function(A) { A[A.EaseInSine = 0] = "EaseInSine", A[A.EaseOutSine = 1] = "EaseOutSine", A[A.EaseInOutSine = 2] = "EaseInOutSine", A[A.EaseInQuad = 3] = "EaseInQuad", A[A.EaseOutQuad = 4] = "EaseOutQuad", A[A.EaseInOutQuad = 5] = "EaseInOutQuad", A[A.EaseInCubic = 6] = "EaseInCubic", A[A.EaseOutCubic = 7] = "EaseOutCubic", A[A.EaseInOutCubic = 8] = "EaseInOutCubic", A[A.EaseInQuart = 9] = "EaseInQuart", A[A.EaseOutQuart = 10] = "EaseOutQuart", A[A.EaseInOutQuart = 11] = "EaseInOutQuart", A[A.EaseInQuint = 12] = "EaseInQuint", A[A.EaseOutQuint = 13] = "EaseOutQuint", A[A.EaseInOutQuint = 14] = "EaseInOutQuint", A[A.EaseInExpo = 15] = "EaseInExpo", A[A.EaseOutExpo = 16] = "EaseOutExpo", A[A.EaseInOutExpo = 17] = "EaseInOutExpo", A[A.EaseInCirc = 18] = "EaseInCirc", A[A.EaseOutCirc = 19] = "EaseOutCirc", A[A.EaseInOutCirc = 20] = "EaseInOutCirc", A[A.EaseInBack = 21] = "EaseInBack", A[A.EaseOutBack = 22] = "EaseOutBack", A[A.EaseInOutBack = 23] = "EaseInOutBack", A[A.EaseInElastic = 24] = "EaseInElastic", A[A.EaseOutElastic = 25] = "EaseOutElastic", A[A.EaseInOutElastic = 26] = "EaseInOutElastic"; })(no || (no = {})); class dne extends Mr { /** * Creates a new CurveBlock * @param name defines the block name */ constructor(e) { super(e, Ve.Neutral), this.type = no.EaseInOutSine, this.registerInput("input", de.AutoDetect), this.registerOutput("output", de.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._inputs[0].excludedConnectionPointTypes.push(de.Matrix), this._inputs[0].excludedConnectionPointTypes.push(de.Object), this._inputs[0].excludedConnectionPointTypes.push(de.Int); } /** * Gets the current class name * @returns the class name */ getClassName() { return "CurveBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _duplicateEntry(e, t) { return `ret.${t} = ${e.replace(/VAL/g, "v." + t)}`; } _duplicateEntryDirect(e) { return `return ${e.replace(/VAL/g, "v")}`; } _duplicateVector(e, t) { if (t === "float") return this._duplicateEntryDirect(e); const r = parseInt(t.replace("vec", "")); let n = ` vec${r} ret = vec${r}(0.0); `; for (let i = 1; i <= r; i++) n += this._duplicateEntry(e, i === 1 ? "x" : i === 2 ? "y" : i === 3 ? "z" : "w") + `; `; return n += `return ret; `, n; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0]; let r = "", n = "", i = ""; switch (this.input.type) { case de.Float: i = "float"; break; case de.Vector2: i = "vec2"; break; case de.Vector3: case de.Color3: i = "vec3"; break; case de.Vector4: case de.Color4: i = "vec4"; break; } switch (n = no[this.type] + "_" + i, this.type) { case no.EaseInSine: r = "return 1.0 - cos((v * 3.1415) / 2.0)"; break; case no.EaseOutSine: r = "return sin((v * 3.1415) / 2.0)"; break; case no.EaseInOutSine: r = "return -(cos(v * 3.1415) - 1.0) / 2.0"; break; case no.EaseInQuad: r = "return v * v"; break; case no.EaseOutQuad: r = "return (1.0 - v) * (1.0 - v)"; break; case no.EaseInOutQuad: { const s = "VAL < 0.5 ? 2.0 * VAL * VAL : 1.0 - pow(-2.0 * VAL + 2.0, 2.0) / 2.0"; r = this._duplicateVector(s, i); break; } case no.EaseInCubic: r = "return v * v * v"; break; case no.EaseOutCubic: { const s = "1.0 - pow(1.0 - VAL, 3.0)"; r = this._duplicateVector(s, i); break; } case no.EaseInOutCubic: { const s = "VAL < 0.5 ? 4.0 * VAL * VAL * VAL : 1.0 - pow(-2.0 * VAL + 2.0, 3.0) / 2.0"; r = this._duplicateVector(s, i); break; } case no.EaseInQuart: r = "return v * v * v * v"; break; case no.EaseOutQuart: { const s = "1.0 - pow(1.0 - VAL, 4.0)"; r = this._duplicateVector(s, i); break; } case no.EaseInOutQuart: { const s = "VAL < 0.5 ? 8.0 * VAL * VAL * VAL * VAL : 1.0 - pow(-2.0 * VAL + 2.0, 4.0) / 2.0"; r = this._duplicateVector(s, i); break; } case no.EaseInQuint: r = "return v * v * v * v * v"; break; case no.EaseOutQuint: { const s = "1.0 - pow(1.0 - VAL, 5.0)"; r = this._duplicateVector(s, i); break; } case no.EaseInOutQuint: { const s = "VAL < 0.5 ? 16.0 * VAL * VAL * VAL * VAL * VAL : 1.0 - pow(-2.0 * VAL + 2.0, 5.0) / 2.0"; r = this._duplicateVector(s, i); break; } case no.EaseInExpo: { const s = "VAL == 0.0 ? 0.0 : pow(2.0, 10.0 * VAL - 10.0)"; r = this._duplicateVector(s, i); break; } case no.EaseOutExpo: { const s = "VAL == 1.0 ? 1.0 : 1.0 - pow(2.0, -10.0 * VAL)"; r = this._duplicateVector(s, i); break; } case no.EaseInOutExpo: { const s = "VAL == 0.0 ? 0.0 : VAL == 1.0 ? 1.0 : VAL < 0.5 ? pow(2.0, 20.0 * VAL - 10.0) / 2.0 : (2.0 - pow(2.0, -20.0 * VAL + 10.0)) / 2.0"; r = this._duplicateVector(s, i); break; } case no.EaseInCirc: { const s = "1.0 - sqrt(1.0 - pow(VAL, 2.0))"; r = this._duplicateVector(s, i); break; } case no.EaseOutCirc: { const s = "sqrt(1.0 - pow(VAL - 1.0, 2.0))"; r = this._duplicateVector(s, i); break; } case no.EaseInOutCirc: { const s = "VAL < 0.5 ? (1.0 - sqrt(1.0 - pow(2.0 * VAL, 2.0))) / 2.0 : (sqrt(1.0 - pow(-2.0 * VAL + 2.0, 2.0)) + 1.0) / 2.0"; r = this._duplicateVector(s, i); break; } case no.EaseInBack: { r = "return 2.70158 * v * v * v - 1.70158 * v * v"; break; } case no.EaseOutBack: { const s = "2.70158 * pow(VAL - 1.0, 3.0) + 1.70158 * pow(VAL - 1.0, 2.0)"; r = this._duplicateVector(s, i); break; } case no.EaseInOutBack: { const s = "VAL < 0.5 ? (pow(2.0 * VAL, 2.0) * ((3.5949095) * 2.0 * VAL - 2.5949095)) / 2.0 : (pow(2.0 * VAL - 2.0, 2.0) * (3.5949095 * (VAL * 2.0 - 2.0) + 3.5949095) + 2.0) / 2.0"; r = this._duplicateVector(s, i); break; } case no.EaseInElastic: { const s = "VAL == 0.0 ? 0.0 : VAL == 1.0 ? 1.0 : -pow(2.0, 10.0 * VAL - 10.0) * sin((VAL * 10.0 - 10.75) * ((2.0 * 3.1415) / 3.0))"; r = this._duplicateVector(s, i); break; } case no.EaseOutElastic: { const s = "VAL == 0.0 ? 0.0 : VAL == 1.0 ? 1.0 : pow(2.0, -10.0 * VAL) * sin((VAL * 10.0 - 0.75) * ((2.0 * 3.1415) / 3.0)) + 1.0"; r = this._duplicateVector(s, i); break; } case no.EaseInOutElastic: { const s = "VAL == 0.0 ? 0.0 : VAL == 1.0 ? 1.0 : VAL < 0.5 ? -(pow(2.0, 20.0 * VAL - 10.0) * sin((20.0 * VAL - 11.125) * ((2.0 * 3.1415) / 4.5))) / 2.0 : (pow(2.0, -20.0 * VAL + 10.0) * sin((20.0 * VAL - 11.125) * ((2.0 * 3.1415) / 4.5))) / 2.0 + 1.0"; r = this._duplicateVector(s, i); break; } } return e._emitFunction(n, `${i} ${n}(${i} v) {${r};} `, ""), e.compilationString += this._declareOutput(t, e) + ` = ${n}(${this.input.associatedVariableName}); `, this; } serialize() { const e = super.serialize(); return e.curveType = this.type, e; } _deserialize(e, t, r) { super._deserialize(e, t, r), this.type = e.curveType; } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.type = BABYLON.CurveBlockTypes.${no[this.type]}; `; } } Ue("BABYLON.CurveBlock", dne); class _pe { /** * Function used to optimize a NodeMaterial graph * @param _vertexOutputNodes defines the list of output nodes for the vertex shader * @param _fragmentOutputNodes defines the list of output nodes for the fragment shader */ optimize(e, t) { } } class $pe { constructor() { this.mm = /* @__PURE__ */ new Map(); } get(e, t) { const r = this.mm.get(e); if (r !== void 0) return r.get(t); } set(e, t, r) { let n = this.mm.get(e); n === void 0 && this.mm.set(e, n = /* @__PURE__ */ new Map()), n.set(t, r); } } class ehe { /** Gets the standalone status of the wrapper */ get standalone() { var e, t; return (t = (e = this._options) === null || e === void 0 ? void 0 : e.standalone) !== null && t !== void 0 ? t : !1; } /** Gets the base material the wrapper is built upon */ get baseMaterial() { return this._baseMaterial; } /** Gets the doNotInjectCode status of the wrapper */ get doNotInjectCode() { var e, t; return (t = (e = this._options) === null || e === void 0 ? void 0 : e.doNotInjectCode) !== null && t !== void 0 ? t : !1; } /** * Instantiate a new shadow depth wrapper. * It works by injecting some specific code in the vertex/fragment shaders of the base material and is used by a shadow generator to * generate the shadow depth map. For more information, please refer to the documentation: * https://doc.babylonjs.com/features/featuresDeepDive/lights/shadows * @param baseMaterial Material to wrap * @param scene Define the scene the material belongs to * @param options Options used to create the wrapper */ constructor(e, t, r) { this._baseMaterial = e, this._scene = t ?? gr.LastCreatedScene, this._options = r, this._subMeshToEffect = /* @__PURE__ */ new Map(), this._subMeshToDepthWrapper = new $pe(), this._meshes = /* @__PURE__ */ new Map(), this._onEffectCreatedObserver = this._baseMaterial.onEffectCreatedObservable.add((n) => { var i, s; const a = (i = n.subMesh) === null || i === void 0 ? void 0 : i.getMesh(); a && !this._meshes.has(a) && this._meshes.set(a, a.onDisposeObservable.add((f) => { const o = this._subMeshToEffect.keys(); for (let d = o.next(); d.done !== !0; d = o.next()) { const v = d.value; (v == null ? void 0 : v.getMesh()) === f && (this._subMeshToEffect.delete(v), this._deleteDepthWrapperEffect(v)); } })), ((s = this._subMeshToEffect.get(n.subMesh)) === null || s === void 0 ? void 0 : s[0]) !== n.effect && (this._subMeshToEffect.set(n.subMesh, [n.effect, this._scene.getEngine().currentRenderPassId]), this._deleteDepthWrapperEffect(n.subMesh)); }); } _deleteDepthWrapperEffect(e) { const t = this._subMeshToDepthWrapper.mm.get(e); t && (t.forEach((r) => { var n; (n = r.mainDrawWrapper.effect) === null || n === void 0 || n.dispose(); }), this._subMeshToDepthWrapper.mm.delete(e)); } /** * Gets the effect to use to generate the depth map * @param subMesh subMesh to get the effect for * @param shadowGenerator shadow generator to get the effect for * @param passIdForDrawWrapper Id of the pass for which the effect from the draw wrapper must be retrieved from * @returns the effect to use to generate the depth map for the subMesh + shadow generator specified */ getEffect(e, t, r) { var n; const i = (n = this._subMeshToDepthWrapper.mm.get(e)) === null || n === void 0 ? void 0 : n.get(t); if (!i) return null; let s = i.drawWrapper[r]; return s || (s = i.drawWrapper[r] = new zo(this._scene.getEngine()), s.setEffect(i.mainDrawWrapper.effect, i.mainDrawWrapper.defines)), s; } /** * Specifies that the submesh is ready to be used for depth rendering * @param subMesh submesh to check * @param defines the list of defines to take into account when checking the effect * @param shadowGenerator combined with subMesh, it defines the effect to check * @param useInstances specifies that instances should be used * @param passIdForDrawWrapper Id of the pass for which the draw wrapper should be created * @returns a boolean indicating that the submesh is ready or not */ isReadyForSubMesh(e, t, r, n, i) { var s, a; return this.standalone && !this._baseMaterial.isReadyForSubMesh(e.getMesh(), e, n) ? !1 : (a = (s = this._makeEffect(e, t, r, i)) === null || s === void 0 ? void 0 : s.isReady()) !== null && a !== void 0 ? a : !1; } /** * Disposes the resources */ dispose() { this._baseMaterial.onEffectCreatedObservable.remove(this._onEffectCreatedObserver), this._onEffectCreatedObserver = null; const e = this._meshes.entries(); for (let t = e.next(); t.done !== !0; t = e.next()) { const [r, n] = t.value; r.onDisposeObservable.remove(n); } } _makeEffect(e, t, r, n) { var i, s, a; const f = this._scene.getEngine(), o = this._subMeshToEffect.get(e); if (!o) return null; const [d, v] = o; let u = this._subMeshToDepthWrapper.get(e, r); if (!u) { const H = new zo(f); H.defines = (s = (i = e._getDrawWrapper(v)) === null || i === void 0 ? void 0 : i.defines) !== null && s !== void 0 ? s : null, u = { drawWrapper: [], mainDrawWrapper: H, depthDefines: "", token: v4() }, u.drawWrapper[n] = H, this._subMeshToDepthWrapper.set(e, r, u); } const l = t.join(` `); if (u.mainDrawWrapper.effect && l === u.depthDefines) return u.mainDrawWrapper.effect; u.depthDefines = l; const P = d.getUniformNames().slice(); let p = d.vertexSourceCodeBeforeMigration, c = d.fragmentSourceCodeBeforeMigration; if (!this.doNotInjectCode) { const H = this._options && this._options.remappedVariables ? `#include(${this._options.remappedVariables.join(",")})` : An.IncludesShadersStore.shadowMapVertexNormalBias, T = this._options && this._options.remappedVariables ? `#include(${this._options.remappedVariables.join(",")})` : An.IncludesShadersStore.shadowMapVertexMetric, q = this._options && this._options.remappedVariables ? `#include(${this._options.remappedVariables.join(",")})` : An.IncludesShadersStore.shadowMapFragmentSoftTransparentShadow, b = An.IncludesShadersStore.shadowMapFragment; p = p.replace(/void\s+?main/g, An.IncludesShadersStore.shadowMapVertexExtraDeclaration + ` void main`), p = p.replace(/#define SHADOWDEPTH_NORMALBIAS|#define CUSTOM_VERTEX_UPDATE_WORLDPOS/g, H), p.indexOf("#define SHADOWDEPTH_METRIC") !== -1 ? p = p.replace(/#define SHADOWDEPTH_METRIC/g, T) : p = p.replace(/}\s*$/g, T + ` }`), p = p.replace(/#define SHADER_NAME.*?\n|out vec4 glFragColor;\n/g, ""); const j = c.indexOf("#define SHADOWDEPTH_SOFTTRANSPARENTSHADOW") >= 0 || c.indexOf("#define CUSTOM_FRAGMENT_BEFORE_FOG") >= 0, w = c.indexOf("#define SHADOWDEPTH_FRAGMENT") !== -1; let m = ""; j ? c = c.replace(/#define SHADOWDEPTH_SOFTTRANSPARENTSHADOW|#define CUSTOM_FRAGMENT_BEFORE_FOG/g, q) : m = q + ` `, c = c.replace(/void\s+?main/g, An.IncludesShadersStore.shadowMapFragmentExtraDeclaration + ` void main`), w ? c = c.replace(/#define SHADOWDEPTH_FRAGMENT/g, b) : m += b + ` `, m && (c = c.replace(/}\s*$/g, m + "}")), P.push("biasAndScaleSM", "depthValuesSM", "lightDataSM", "softTransparentShadowSM"); } u.mainDrawWrapper.effect = f.createEffect({ vertexSource: p, fragmentSource: c, vertexToken: u.token, fragmentToken: u.token }, { attributes: d.getAttributesNames(), uniformsNames: P, uniformBuffersNames: d.getUniformBuffersNames(), samplers: d.getSamplers(), defines: l + ` ` + d.defines.replace("#define SHADOWS", "").replace(/#define SHADOW\d/g, ""), indexParameters: d.getIndexParameters() }, f); for (let H = 0; H < u.drawWrapper.length; ++H) H !== n && ((a = u.drawWrapper[H]) === null || a === void 0 || a.setEffect(u.mainDrawWrapper.effect, u.mainDrawWrapper.defines)); return u.mainDrawWrapper.effect; } } class vne extends na { constructor() { super(...arguments), this.DECAL = !1, this.DECALDIRECTUV = 0, this.DECAL_SMOOTHALPHA = !1, this.GAMMADECAL = !1; } } class cU extends Gl { /** @internal */ _markAllSubMeshesAsTexturesDirty() { this._enable(this._isEnabled), this._internalMarkAllSubMeshesAsTexturesDirty(); } /** * Creates a new DecalMapConfiguration * @param material The material to attach the decal map plugin to * @param addToPluginList If the plugin should be added to the material plugin list */ constructor(e, t = !0) { super(e, "DecalMap", 150, new vne(), t), this._isEnabled = !1, this.isEnabled = !1, this._smoothAlpha = !1, this.smoothAlpha = !1, this.registerForExtraEvents = !0, this._internalMarkAllSubMeshesAsTexturesDirty = e._dirtyCallbacks[1]; } isReadyForSubMesh(e, t, r, n) { const i = n.getMesh().decalMap; return !this._isEnabled || !(i != null && i.texture) || !Dt.DecalMapEnabled || !t.texturesEnabled ? !0 : i.isReady(); } prepareDefines(e, t, r) { const n = r.decalMap; !this._isEnabled || !(n != null && n.texture) || !Dt.DecalMapEnabled || !t.texturesEnabled ? (e.DECAL && e.markAsTexturesDirty(), e.DECAL = !1) : ((!e.DECAL || e.GAMMADECAL !== n.texture.gammaSpace) && e.markAsTexturesDirty(), e.DECAL = !0, e.GAMMADECAL = n.texture.gammaSpace, e.DECAL_SMOOTHALPHA = this._smoothAlpha, Ye.PrepareDefinesForMergedUV(n.texture, e, "DECAL")); } /** * Note that we override hardBindForSubMesh and not bindForSubMesh because the material can be shared by multiple meshes, * in which case mustRebind could return false even though the decal map is different for each mesh: that's because the decal map * is not part of the material but hosted by the decalMap of the mesh instead. */ hardBindForSubMesh(e, t, r, n) { const i = n.getMesh().decalMap; if (!this._isEnabled || !(i != null && i.texture) || !Dt.DecalMapEnabled || !t.texturesEnabled) return; const s = this._material.isFrozen, a = i.texture; (!e.useUbo || !s || !e.isSync) && (e.updateFloat4("vDecalInfos", a.coordinatesIndex, 0, 0, 0), Ye.BindTextureMatrix(a, e, "decal")), e.setTexture("decalSampler", a); } getClassName() { return "DecalMapConfiguration"; } getSamplers(e) { e.push("decalSampler"); } getUniforms() { return { ubo: [ { name: "vDecalInfos", size: 4, type: "vec4" }, { name: "decalMatrix", size: 16, type: "mat4" } ] }; } } C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], cU.prototype, "isEnabled", void 0); C([ M(), At("_markAllSubMeshesAsTexturesDirty") ], cU.prototype, "smoothAlpha", void 0); Ue("BABYLON.DecalMapConfiguration", cU); function the(A) { return A instanceof fs ? new DD(A) : null; } function rhe(A) { return A instanceof fs ? new fd(A) : null; } function nhe(A) { return A instanceof fs ? new h9(A) : null; } function ihe(A) { return A instanceof fs ? new dd(A) : null; } function she(A) { return A instanceof fs ? new K0(A) : null; } function ahe(A) { return A instanceof fs ? new _o(A) : null; } function ohe(A) { return A instanceof fs || A instanceof Wt ? new l4(A) : null; } var OS; (function(A) { A[A.MATERIAL_TYPE_STANDARD = 0] = "MATERIAL_TYPE_STANDARD", A[A.MATERIAL_TYPE_PBR = 1] = "MATERIAL_TYPE_PBR", A[A.MATERIAL_TYPE_SIMPLE = 2] = "MATERIAL_TYPE_SIMPLE"; })(OS || (OS = {})); var Tc; (function(A) { A[A.COLOR_MODE_SET = 0] = "COLOR_MODE_SET", A[A.COLOR_MODE_ADD = 1] = "COLOR_MODE_ADD", A[A.COLOR_MODE_MULTIPLY = 2] = "COLOR_MODE_MULTIPLY"; })(Tc || (Tc = {})); var K2; (function(A) { A[A.COLOR_DISTRIBUTION_TYPE_SEGMENT = 0] = "COLOR_DISTRIBUTION_TYPE_SEGMENT", A[A.COLOR_DISTRIBUTION_TYPE_LINE = 1] = "COLOR_DISTRIBUTION_TYPE_LINE"; })(K2 || (K2 = {})); class D6 { } D6.DEFAULT_COLOR = Ne.White(); D6.DEFAULT_WIDTH_ATTENUATED = 1; D6.DEFAULT_WIDTH = 0.1; class Qs { /** * Converts GreasedLinePoints to number[][] * @param points GreasedLinePoints * @returns number[][] with x, y, z coordinates of the points, like [[x, y, z, x, y, z, ...], [x, y, z, ...]] */ static ConvertPoints(e) { if (e.length && Array.isArray(e) && typeof e[0] == "number") return [e]; if (e.length && Array.isArray(e[0]) && typeof e[0][0] == "number") return e; if (e.length && !Array.isArray(e[0]) && e[0] instanceof S) { const t = []; for (let r = 0; r < e.length; r++) { const n = e[r]; t.push(n.x, n.y, n.z); } return [t]; } else if (e.length > 0 && Array.isArray(e[0]) && e[0].length > 0 && e[0][0] instanceof S) { const t = []; return e.forEach((n) => { t.push(n.flatMap((i) => [i.x, i.y, i.z])); }), t; } else { if (e instanceof Float32Array) return [Array.from(e)]; if (e.length && e[0] instanceof Float32Array) { const t = []; return e.forEach((r) => { t.push(Array.from(r)); }), t; } } return []; } /** * Omit zero length lines predicate for the MeshesToLines function * @param p1 point1 position of the face * @param p2 point2 position of the face * @param p3 point3 position of the face * @returns original points or null if any edge length is zero */ static OmitZeroLengthPredicate(e, t, r) { const n = []; return t.subtract(e).lengthSquared() > 0 && n.push([e, t]), r.subtract(t).lengthSquared() > 0 && n.push([t, r]), e.subtract(r).lengthSquared() > 0 && n.push([r, e]), n.length === 0 ? null : n; } /** * Omit duplicate lines predicate for the MeshesToLines function * @param p1 point1 position of the face * @param p2 point2 position of the face * @param p3 point3 position of the face * @returns original points or null if any edge length is zero */ static OmitDuplicatesPredicate(e, t, r, n) { const i = []; return Qs._SearchInPoints(e, t, n) || i.push([e, t]), Qs._SearchInPoints(t, r, n) || i.push([t, r]), Qs._SearchInPoints(r, e, n) || i.push([r, e]), i.length === 0 ? null : i; } static _SearchInPoints(e, t, r) { var n, i, s; for (const a of r) for (let f = 0; f < a.length; f++) if (!((n = a[f]) === null || n === void 0) && n.equals(e) && (!((i = a[f + 1]) === null || i === void 0) && i.equals(t) || !((s = a[f - 1]) === null || s === void 0) && s.equals(t))) return !0; return !1; } /** * Gets mesh triangles as line positions * @param meshes array of meshes * @param predicate predicate function which decides whether to include the mesh triangle/face in the ouput * @returns array of arrays of points */ static MeshesToLines(e, t) { const r = []; return e.forEach((n, i) => { const s = n.getVerticesData(J.PositionKind), a = n.getIndices(); if (s && a) for (let f = 0, o = 0; f < a.length; f++) { const d = a[o++] * 3, v = a[o++] * 3, u = a[o++] * 3, l = new S(s[d], s[d + 1], s[d + 2]), P = new S(s[v], s[v + 1], s[v + 2]), p = new S(s[u], s[u + 1], s[u + 2]); if (t) { const c = t(l, P, p, r, f, d, n, i, s, a); if (c) for (const H of c) r.push(H); } else r.push([l, P], [P, p], [p, l]); } }), r; } /** * Converts number coordinates to Vector3s * @param points number array of x, y, z, x, y z, ... coordinates * @returns Vector3 array */ static ToVector3Array(e) { if (Array.isArray(e[0])) { const n = [], i = e; for (const s of i) { const a = []; for (let f = 0; f < s.length; f += 3) a.push(new S(s[f], s[f + 1], s[f + 2])); n.push(a); } return n; } const t = e, r = []; for (let n = 0; n < t.length; n += 3) r.push(new S(t[n], t[n + 1], t[n + 2])); return r; } /** * Gets a number array from a Vector3 array. * You can you for example to convert your Vector3[] offsets to the required number[] for the offsets option. * @param points Vector3 array * @returns an array of x, y, z coordinates as numbers [x, y, z, x, y, z, x, y, z, ....] */ static ToNumberArray(e) { return e.flatMap((t) => [t.x, t.y, t.z]); } /** * Calculates the sum of points of every line and the number of points in each line. * This function is useful when you are drawing multiple lines in one mesh and you want * to know the counts. For example for creating an offsets table. * @param points point array * @returns points count info */ static GetPointsCountInfo(e) { const t = new Array(e.length); let r = 0; for (let n = e.length; n--; ) t[n] = e[n].length / 3, r += t[n]; return { total: r, counts: t }; } /** * Gets the length of the line counting all it's segments length * @param data array of line points * @returns length of the line */ static GetLineLength(e) { if (e.length === 0) return 0; let t; typeof e[0] == "number" ? t = Qs.ToVector3Array(e) : t = e; const r = ue.Vector3[0]; let n = 0; for (let i = 0; i < t.length - 1; i++) { const s = t[i], a = t[i + 1]; n += a.subtractToRef(s, r).length(); } return n; } /** * Divides a segment into smaller segments. * A segment is a part of the line between it's two points. * @param point1 first point of the line * @param point2 second point of the line * @param segmentCount number of segments we want to have in the divided line * @returns */ static SegmentizeSegmentByCount(e, t, r) { const n = [], i = t.subtract(e), s = ue.Vector3[0]; s.setAll(r); const a = ue.Vector3[1]; i.divideToRef(s, a); let f = e.clone(); n.push(f); for (let o = 0; o < r; o++) f = f.clone(), n.push(f.addInPlace(a)); return n; } /** * Divides a line into segments. * A segment is a part of the line between it's two points. * @param what line points * @param segmentLength length of each segment of the resulting line (distance between two line points) * @returns line point */ static SegmentizeLineBySegmentLength(e, t) { const r = e[0] instanceof S ? Qs.GetLineSegments(e) : typeof e[0] == "number" ? Qs.GetLineSegments(Qs.ToVector3Array(e)) : e, n = []; return r.forEach((i) => { i.length > t ? Qs.SegmentizeSegmentByCount(i.point1, i.point2, Math.ceil(i.length / t)).forEach((a) => { n.push(a); }) : (n.push(i.point1), n.push(i.point2)); }), n; } /** * Divides a line into segments. * A segment is a part of the line between it's two points. * @param what line points * @param segmentCount number of segments * @returns line point */ static SegmentizeLineBySegmentCount(e, t) { const r = typeof e[0] == "number" ? Qs.ToVector3Array(e) : e, n = Qs.GetLineLength(r) / t; return Qs.SegmentizeLineBySegmentLength(r, n); } /** * Gets line segments. * A segment is a part of the line between it's two points. * @param points line points * @returns segments information of the line segment including starting point, ending point and the distance between them */ static GetLineSegments(e) { const t = []; for (let r = 0; r < e.length - 1; r++) { const n = e[r], i = e[r + 1], s = i.subtract(n).length(); t.push({ point1: n, point2: i, length: s }); } return t; } /** * Gets the minimum and the maximum length of a line segment in the line. * A segment is a part of the line between it's two points. * @param points line points * @returns */ static GetMinMaxSegmentLength(e) { const r = Qs.GetLineSegments(e).sort((n) => n.length); return { min: r[0].length, max: r[r.length - 1].length }; } /** * Finds the last visible position in world space of the line according to the visibility parameter * @param lineSegments segments of the line * @param lineLength total length of the line * @param visbility normalized value of visibility * @returns world space coordinate of the last visible piece of the line */ static GetPositionOnLineByVisibility(e, t, r, n = !1) { const i = t * r; let s = 0, a = 0; const f = e.length; for (let d = 0; d < f; d++) { if (i <= s + e[d].length) { a = d; break; } s += e[d].length; } const o = (i - s) / e[a].length; return e[a].point2.subtractToRef(e[a].point1, ue.Vector3[0]), ue.Vector3[1] = ue.Vector3[0].multiplyByFloats(o, o, o), n || ue.Vector3[1].addInPlace(e[a].point1), ue.Vector3[1].clone(); } /** * Creates lines in a shape of circle/arc. * A segment is a part of the line between it's two points. * @param radiusX radiusX of the circle * @param segments number of segments in the circle * @param z z coordinate of the points. Defaults to 0. * @param radiusY radiusY of the circle - you can draw an oval if using different values * @param segmentAngle angle offset of the segments. Defaults to Math.PI * 2 / segments. Change this value to draw a part of the circle. * @returns line points */ static GetCircleLinePoints(e, t, r = 0, n = e, i = Math.PI * 2 / t) { const s = []; for (let a = 0; a <= t; a++) s.push(new S(Math.cos(a * i) * e, Math.sin(a * i) * n, r)); return s; } /** * Gets line points in a shape of a bezier curve * @param p0 bezier point0 * @param p1 bezier point1 * @param p2 bezier point2 * @param segments number of segments in the curve * @returns */ static GetBezierLinePoints(e, t, r, n) { return j0.CreateQuadraticBezier(e, t, r, n).getPoints().flatMap((i) => [i.x, i.y, i.z]); } /** * * @param position position of the arrow cap (mainly you want to create a triangle, set widthUp and widthDown to the same value and omit widthStartUp and widthStartDown) * @param direction direction which the arrow points to * @param length length (size) of the arrow cap itself * @param widthUp the arrow width above the line * @param widthDown the arrow width belove the line * @param widthStartUp the arrow width at the start of the arrow above the line. In most scenarios this is 0. * @param widthStartDown the arrow width at the start of the arrow below the line. In most scenarios this is 0. * @returns */ static GetArrowCap(e, t, r, n, i, s = 0, a = 0) { return { points: [e.clone(), e.add(t.multiplyByFloats(r, r, r))], widths: [n, i, s, a] }; } /** * Gets 3D positions of points from a text and font * @param text Text * @param size Size of the font * @param resolution Resolution of the font * @param fontData defines the font data (can be generated with http://gero3.github.io/facetype.js/) * @param z z coordinate * @param includeInner include the inner parts of the font in the result. Default true. If false, only the outlines will be returned. * @returns number[][] of 3D positions */ static GetPointsFromText(e, t, r, n, i = 0, s = !0) { const a = [], f = OQ(e, t, r, n); for (const o of f) { for (const d of o.paths) { const v = [], u = d.getPoints(); for (const l of u) v.push(l.x, l.y, i); a.push(v); } if (s) for (const d of o.holes) { const v = [], u = d.getPoints(); for (const l of u) v.push(l.x, l.y, i); a.push(v); } } return a; } /** * Converts an array of Color3 to Uint8Array * @param colors Arrray of Color3 * @returns Uin8Array of colors [r, g, b, a, r, g, b, a, ...] */ static Color3toRGBAUint8(e) { const t = new Uint8Array(e.length * 4); for (let r = 0, n = 0; r < e.length; r++) t[n++] = e[r].r * 255, t[n++] = e[r].g * 255, t[n++] = e[r].b * 255, t[n++] = 255; return t; } /** * Creates a RawTexture from an RGBA color array and sets it on the plugin material instance. * @param name name of the texture * @param colors Uint8Array of colors */ static CreateColorsTexture(e, t, r, n) { const i = Qs.Color3toRGBAUint8(t), s = new Bo(i, t.length, 1, Ge.TEXTUREFORMAT_RGBA, n, !1, !0, r); return s.name = e, s; } /** * A minimum size texture for the colors sampler2D when there is no colors texture defined yet. * For fast switching using the useColors property without the need to use defines. * @param scene Scene * @returns empty colors texture */ static PrepareEmptyColorsTexture(e) { if (!D6.EmptyColorsTexture) { const t = new Uint8Array(4); D6.EmptyColorsTexture = new Bo(t, 1, 1, Ge.TEXTUREFORMAT_RGBA, e, !1, !1, Bo.NEAREST_NEAREST), D6.EmptyColorsTexture.name = "grlEmptyColorsTexture"; } return D6.EmptyColorsTexture; } /** * Diposes the shared empty colors texture */ static DisposeEmptyColorsTexture() { var e; (e = D6.EmptyColorsTexture) === null || e === void 0 || e.dispose(), D6.EmptyColorsTexture = null; } /** * Converts boolean to number. * @param bool * @returns 1 if true, 0 if false. */ static BooleanToNumber(e) { return e ? 1 : 0; } } class une extends na { constructor() { super(...arguments), this.GREASED_LINE_HAS_COLOR = !1, this.GREASED_LINE_SIZE_ATTENUATION = !1, this.GREASED_LINE_COLOR_DISTRIBUTION_TYPE_LINE = !1, this.GREASED_LINE_RIGHT_HANDED_COORDINATE_SYSTEM = !1, this.GREASED_LINE_CAMERA_FACING = !0; } } class o4 extends Gl { /** * Creates a new instance of the GreasedLinePluginMaterial * @param material base material for the plugin * @param scene the scene * @param options plugin options */ constructor(e, t, r) { var n, i, s, a, f, o, d, v, u, l, P, p, c, H, T, q, b; r = r || { color: D6.DEFAULT_COLOR }; const j = new une(); j.GREASED_LINE_HAS_COLOR = !!r.color && !r.useColors, j.GREASED_LINE_SIZE_ATTENUATION = (n = r.sizeAttenuation) !== null && n !== void 0 ? n : !1, j.GREASED_LINE_COLOR_DISTRIBUTION_TYPE_LINE = r.colorDistributionType === K2.COLOR_DISTRIBUTION_TYPE_LINE, j.GREASED_LINE_RIGHT_HANDED_COORDINATE_SYSTEM = (t ?? e.getScene()).useRightHandedSystem, j.GREASED_LINE_CAMERA_FACING = (i = r.cameraFacing) !== null && i !== void 0 ? i : !0, super(e, o4.GREASED_LINE_MATERIAL_NAME, 200, j), this.colorsTexture = null, this._scene = t ?? e.getScene(), this._engine = this._scene.getEngine(), this._cameraFacing = (s = r.cameraFacing) !== null && s !== void 0 ? s : !0, this.visibility = (a = r.visibility) !== null && a !== void 0 ? a : 1, this.useDash = (f = r.useDash) !== null && f !== void 0 ? f : !1, this.dashRatio = (o = r.dashRatio) !== null && o !== void 0 ? o : 0.5, this.dashOffset = (d = r.dashOffset) !== null && d !== void 0 ? d : 0, this.width = r.width ? r.width : r.sizeAttenuation ? D6.DEFAULT_WIDTH_ATTENUATED : D6.DEFAULT_WIDTH, this._sizeAttenuation = (v = r.sizeAttenuation) !== null && v !== void 0 ? v : !1, this.colorMode = (u = r.colorMode) !== null && u !== void 0 ? u : Tc.COLOR_MODE_SET, this._color = (l = r.color) !== null && l !== void 0 ? l : null, this.useColors = (P = r.useColors) !== null && P !== void 0 ? P : !1, this._colorsDistributionType = (p = r.colorDistributionType) !== null && p !== void 0 ? p : K2.COLOR_DISTRIBUTION_TYPE_SEGMENT, this.colorsSampling = (c = r.colorsSampling) !== null && c !== void 0 ? c : Bo.NEAREST_NEAREST, this._colors = (H = r.colors) !== null && H !== void 0 ? H : null, this.dashCount = (T = r.dashCount) !== null && T !== void 0 ? T : 1, this.resolution = (q = r.resolution) !== null && q !== void 0 ? q : new at(this._engine.getRenderWidth(), this._engine.getRenderHeight()), r.colorsTexture ? this.colorsTexture = r.colorsTexture : this._colors ? this.colorsTexture = Qs.CreateColorsTexture(`${e.name}-colors-texture`, this._colors, this.colorsSampling, this._scene) : (this._color = (b = this._color) !== null && b !== void 0 ? b : D6.DEFAULT_COLOR, Qs.PrepareEmptyColorsTexture(this._scene)), this._engine.onDisposeObservable.add(() => { Qs.DisposeEmptyColorsTexture(); }), this._enable(!0); } /** * Get the shader attributes * @param attributes array which will be filled with the attributes */ getAttributes(e) { e.push("grl_offsets"), e.push("grl_widths"), e.push("grl_colorPointers"), e.push("grl_counters"), this._cameraFacing ? (e.push("grl_previousAndSide"), e.push("grl_nextAndCounters")) : e.push("grl_slopes"); } /** * Get the shader samplers * @param samplers */ getSamplers(e) { e.push("grl_colors"); } /** * Get the shader textures * @param activeTextures */ getActiveTextures(e) { this.colorsTexture && e.push(this.colorsTexture); } /** * Get the shader uniforms * @returns uniforms */ getUniforms() { const e = [ { name: "grl_singleColor", size: 3, type: "vec3" }, { name: "grl_dashOptions", size: 4, type: "vec4" }, { name: "grl_colorMode_visibility_colorsWidth_useColors", size: 4, type: "vec4" } ]; return this._cameraFacing && e.push({ name: "grl_projection", size: 16, type: "mat4" }, { name: "grl_aspect_resolution_lineWidth", size: 4, type: "vec4" }), { ubo: e, vertex: this._cameraFacing ? ` uniform vec4 grl_aspect_resolution_lineWidth; uniform mat4 grl_projection; ` : "", fragment: ` uniform vec4 grl_dashOptions; uniform vec4 grl_colorMode_visibility_colorsWidth_useColors; uniform vec3 grl_singleColor; ` }; } // only getter, it doesn't make sense to use this plugin on a mesh other than GreasedLineMesh // and it doesn't make sense to disable it on the mesh get isEnabled() { return !0; } /** * Bind the uniform buffer * @param uniformBuffer */ bindForSubMesh(e) { var t; if (this._cameraFacing) { const i = this._scene.activeCamera; if (i) { const a = i.getProjectionMatrix(); e.updateMatrix("grl_projection", a); } else throw Error("GreasedLinePluginMaterial requires an active camera."); const s = ue.Vector4[0]; s.x = this._aspect, s.y = this._resolution.x, s.z = this._resolution.y, s.w = this.width, e.updateVector4("grl_aspect_resolution_lineWidth", s); } const r = ue.Vector4[0]; r.x = Qs.BooleanToNumber(this.useDash), r.y = this._dashArray, r.z = this.dashOffset, r.w = this.dashRatio, e.updateVector4("grl_dashOptions", r); const n = ue.Vector4[1]; n.x = this.colorMode, n.y = this.visibility, n.z = this.colorsTexture ? this.colorsTexture.getSize().width : 0, n.w = Qs.BooleanToNumber(this.useColors), e.updateVector4("grl_colorMode_visibility_colorsWidth_useColors", n), this._color && e.updateColor3("grl_singleColor", this._color), e.setTexture("grl_colors", (t = this.colorsTexture) !== null && t !== void 0 ? t : D6.EmptyColorsTexture); } /** * Prepare the defines * @param defines * @param _scene * @param _mesh */ prepareDefines(e, t, r) { e.GREASED_LINE_HAS_COLOR = !!this.color && !this.useColors, e.GREASED_LINE_SIZE_ATTENUATION = this._sizeAttenuation, e.GREASED_LINE_COLOR_DISTRIBUTION_TYPE_LINE = this._colorsDistributionType === K2.COLOR_DISTRIBUTION_TYPE_LINE, e.GREASED_LINE_RIGHT_HANDED_COORDINATE_SYSTEM = t.useRightHandedSystem, e.GREASED_LINE_CAMERA_FACING = this._cameraFacing; } /** * Get the class name * @returns class name */ getClassName() { return o4.GREASED_LINE_MATERIAL_NAME; } /** * Get shader code * @param shaderType vertex/fragment * @returns shader code */ getCustomCode(e) { if (e === "vertex") { const t = { // eslint-disable-next-line @typescript-eslint/naming-convention CUSTOM_VERTEX_DEFINITIONS: ` attribute float grl_widths; attribute vec3 grl_offsets; attribute float grl_colorPointers; varying float grlCounters; varying float grlColorPointer; #ifdef GREASED_LINE_CAMERA_FACING attribute vec4 grl_previousAndSide; attribute vec4 grl_nextAndCounters; vec2 grlFix( vec4 i, float aspect ) { vec2 res = i.xy / i.w; res.x *= aspect; return res; } #else attribute vec3 grl_slopes; attribute float grl_counters; #endif `, // eslint-disable-next-line @typescript-eslint/naming-convention CUSTOM_VERTEX_UPDATE_POSITION: ` #ifdef GREASED_LINE_CAMERA_FACING vec3 grlPositionOffset = grl_offsets; positionUpdated += grlPositionOffset; #else positionUpdated = (positionUpdated + grl_offsets) + (grl_slopes * grl_widths); #endif `, // eslint-disable-next-line @typescript-eslint/naming-convention CUSTOM_VERTEX_MAIN_END: ` grlColorPointer = grl_colorPointers; #ifdef GREASED_LINE_CAMERA_FACING float grlAspect = grl_aspect_resolution_lineWidth.x; float grlBaseWidth = grl_aspect_resolution_lineWidth.w; vec3 grlPrevious = grl_previousAndSide.xyz; float grlSide = grl_previousAndSide.w; vec3 grlNext = grl_nextAndCounters.xyz; grlCounters = grl_nextAndCounters.w; mat4 grlMatrix = viewProjection * finalWorld; vec4 grlFinalPosition = grlMatrix * vec4( positionUpdated , 1.0 ); vec4 grlPrevPos = grlMatrix * vec4( grlPrevious + grlPositionOffset, 1.0 ); vec4 grlNextPos = grlMatrix * vec4( grlNext + grlPositionOffset, 1.0 ); vec2 grlCurrentP = grlFix( grlFinalPosition, grlAspect ); vec2 grlPrevP = grlFix( grlPrevPos, grlAspect ); vec2 grlNextP = grlFix( grlNextPos, grlAspect ); float grlWidth = grlBaseWidth * grl_widths; vec2 grlDir; if( grlNextP == grlCurrentP ) grlDir = normalize( grlCurrentP - grlPrevP ); else if( grlPrevP == grlCurrentP ) grlDir = normalize( grlNextP - grlCurrentP ); else { vec2 grlDir1 = normalize( grlCurrentP - grlPrevP ); vec2 grlDir2 = normalize( grlNextP - grlCurrentP ); grlDir = normalize( grlDir1 + grlDir2 ); } vec4 grlNormal = vec4( -grlDir.y, grlDir.x, 0., 1. ); #ifdef GREASED_LINE_RIGHT_HANDED_COORDINATE_SYSTEM grlNormal.xy *= -.5 * grlWidth; #else grlNormal.xy *= .5 * grlWidth; #endif grlNormal *= grl_projection; #ifdef GREASED_LINE_SIZE_ATTENUATION grlNormal.xy *= grlFinalPosition.w; grlNormal.xy /= ( vec4( grl_aspect_resolution_lineWidth.yz, 0., 1. ) * grl_projection ).xy; #endif grlFinalPosition.xy += grlNormal.xy * grlSide; gl_Position = grlFinalPosition; vPositionW = vec3(grlFinalPosition); #else grlCounters = grl_counters; #endif ` }; return this._cameraFacing && (t["!gl_Position\\=viewProjection\\*worldPos;"] = "//"), t; } return e === "fragment" ? { // eslint-disable-next-line @typescript-eslint/naming-convention CUSTOM_FRAGMENT_DEFINITIONS: ` varying float grlCounters; varying float grlColorPointer; uniform sampler2D grl_colors; `, // eslint-disable-next-line @typescript-eslint/naming-convention CUSTOM_FRAGMENT_MAIN_END: ` float grlColorMode = grl_colorMode_visibility_colorsWidth_useColors.x; float grlVisibility = grl_colorMode_visibility_colorsWidth_useColors.y; float grlColorsWidth = grl_colorMode_visibility_colorsWidth_useColors.z; float grlUseColors = grl_colorMode_visibility_colorsWidth_useColors.w; float grlUseDash = grl_dashOptions.x; float grlDashArray = grl_dashOptions.y; float grlDashOffset = grl_dashOptions.z; float grlDashRatio = grl_dashOptions.w; gl_FragColor.a *= step(grlCounters, grlVisibility); if( gl_FragColor.a == 0. ) discard; if(grlUseDash == 1.){ gl_FragColor.a *= ceil(mod(grlCounters + grlDashOffset, grlDashArray) - (grlDashArray * grlDashRatio)); if (gl_FragColor.a == 0.) discard; } #ifdef GREASED_LINE_HAS_COLOR if (grlColorMode == ${Tc.COLOR_MODE_SET}.) { gl_FragColor.rgb = grl_singleColor; } else if (grlColorMode == ${Tc.COLOR_MODE_ADD}.) { gl_FragColor.rgb += grl_singleColor; } else if (grlColorMode == ${Tc.COLOR_MODE_MULTIPLY}.) { gl_FragColor.rgb *= grl_singleColor; } #else if (grlUseColors == 1.) { #ifdef GREASED_LINE_COLOR_DISTRIBUTION_TYPE_LINE vec4 grlColor = texture2D(grl_colors, vec2(grlCounters, 0.), 0.); #else vec4 grlColor = texture2D(grl_colors, vec2(grlColorPointer/grlColorsWidth, 0.), 0.); #endif if (grlColorMode == ${Tc.COLOR_MODE_SET}.) { gl_FragColor = grlColor; } else if (grlColorMode == ${Tc.COLOR_MODE_ADD}.) { gl_FragColor += grlColor; } else if (grlColorMode == ${Tc.COLOR_MODE_MULTIPLY}.) { gl_FragColor *= grlColor; } } #endif ` } : null; } /** * Disposes the plugin material. */ dispose() { var e; (e = this.colorsTexture) === null || e === void 0 || e.dispose(), super.dispose(); } /** * Returns the colors used to colorize the line */ get colors() { return this._colors; } /** * Sets the colors used to colorize the line */ set colors(e) { this.setColors(e); } /** * Creates or updates the colors texture * @param colors color table RGBA * @param lazy if lazy, the colors are not updated * @param forceNewTexture force creation of a new texture * @returns */ setColors(e, t = !1, r = !1) { var n, i, s, a; const f = (i = (n = this._colors) === null || n === void 0 ? void 0 : n.length) !== null && i !== void 0 ? i : 0; if (this._colors = e, e === null || e.length === 0) { (s = this.colorsTexture) === null || s === void 0 || s.dispose(); return; } if (!(t && !r)) if (this.colorsTexture && f === e.length && !r) { const o = Qs.Color3toRGBAUint8(e); this.colorsTexture.update(o); } else (a = this.colorsTexture) === null || a === void 0 || a.dispose(), this.colorsTexture = Qs.CreateColorsTexture(`${this._material.name}-colors-texture`, e, this.colorsSampling, this._scene); } /** * Updates the material. Use when material created in lazy mode. */ updateLazy() { this._colors && this.setColors(this._colors, !1, !0); } /** * Gets the number of dashes in the line */ get dashCount() { return this._dashCount; } /** * Sets the number of dashes in the line * @param value dash */ set dashCount(e) { this._dashCount = e, this._dashArray = 1 / e; } /** * If set to true the line will be rendered always with the same width regardless how far it is located from the camera. * Not supported for non camera facing lines. */ get sizeAttenuation() { return this._sizeAttenuation; } /** * Turn on/off size attenuation of the width option and widths array. * Not supported for non camera facing lines. * @param value If set to true the line will be rendered always with the same width regardless how far it is located from the camera. */ set sizeAttenuation(e) { this._sizeAttenuation = e, this.markAllDefinesAsDirty(); } /** * Gets the color of the line */ get color() { return this._color; } /** * Sets the color of the line * @param value Color3 or null to clear the color. You need to clear the color if you use colors and useColors = true */ set color(e) { this.setColor(e); } /** * Sets the color of the line. If set the whole line will be mixed with this color according to the colorMode option. * @param value color */ setColor(e, t = !1) { this._color === null && e !== null || this._color !== null && e === null ? (this._color = e, !t && this.markAllDefinesAsDirty()) : this._color = e; } /** * Gets the color distributiopn type */ get colorsDistributionType() { return this._colorsDistributionType; } /** * Sets the color distribution type * @see GreasedLineMeshColorDistributionType * @param value color distribution type */ set colorsDistributionType(e) { this._colorsDistributionType = e, this.markAllDefinesAsDirty(); } /** * Gets the resolution */ get resolution() { return this._resolution; } /** * Sets the resolution * @param value resolution of the screen for GreasedLine */ set resolution(e) { this._aspect = e.x / e.y, this._resolution = e; } /** * Serializes this plugin material * @returns serializationObjec */ serialize() { const e = super.serialize(), t = { colorDistributionType: this._colorsDistributionType, colorsSampling: this.colorsSampling, colorMode: this.colorMode, dashCount: this._dashCount, dashOffset: this.dashOffset, dashRatio: this.dashRatio, resolution: this._resolution, sizeAttenuation: this._sizeAttenuation, useColors: this.useColors, useDash: this.useDash, visibility: this.visibility, width: this.width }; return this._colors && (t.colors = this._colors), this._color && (t.color = this._color), e.greasedLineMaterialOptions = t, e; } /** * Parses a serialized objects * @param source serialized object * @param scene scene * @param rootUrl root url for textures */ parse(e, t, r) { var n; super.parse(e, t, r); const i = e.greasedLineMaterialOptions; (n = this.colorsTexture) === null || n === void 0 || n.dispose(), i.color && this.setColor(i.color, !0), i.colorDistributionType && (this.colorsDistributionType = i.colorDistributionType), i.colors && (this.colors = i.colors), i.colorsSampling && (this.colorsSampling = i.colorsSampling), i.colorMode && (this.colorMode = i.colorMode), i.useColors && (this.useColors = i.useColors), i.visibility && (this.visibility = i.visibility), i.useDash && (this.useDash = i.useDash), i.dashCount && (this.dashCount = i.dashCount), i.dashRatio && (this.dashRatio = i.dashRatio), i.dashOffset && (this.dashOffset = i.dashOffset), i.width && (this.width = i.width), i.sizeAttenuation && (this.sizeAttenuation = i.sizeAttenuation), i.resolution && (this.resolution = i.resolution), this.colors ? this.colorsTexture = Qs.CreateColorsTexture(`${this._material.name}-colors-texture`, this.colors, this.colorsSampling, t) : Qs.PrepareEmptyColorsTexture(t), this.markAllDefinesAsDirty(); } /** * Makes a duplicate of the current configuration into another one. * @param plugin define the config where to copy the info */ copyTo(e) { var t; const r = e; (t = r.colorsTexture) === null || t === void 0 || t.dispose(), this._colors && (r.colorsTexture = Qs.CreateColorsTexture(`${r._material.name}-colors-texture`, this._colors, r.colorsSampling, this._scene)), r.setColor(this.color, !0), r.colorsDistributionType = this.colorsDistributionType, r.colorsSampling = this.colorsSampling, r.colorMode = this.colorMode, r.useColors = this.useColors, r.visibility = this.visibility, r.useDash = this.useDash, r.dashCount = this.dashCount, r.dashRatio = this.dashRatio, r.dashOffset = this.dashOffset, r.width = this.width, r.sizeAttenuation = this.sizeAttenuation, r.resolution = this.resolution, r.markAllDefinesAsDirty(); } } o4.GREASED_LINE_MATERIAL_NAME = "GreasedLinePluginMaterial"; Ue(`BABYLON.${o4.GREASED_LINE_MATERIAL_NAME}`, o4); const fhe = "greasedLinePixelShader", Ahe = `precision highp float;uniform sampler2D grlColors;uniform float grlUseColors;uniform float grlUseDash;uniform float grlDashArray;uniform float grlDashOffset;uniform float grlDashRatio;uniform float grlVisibility;uniform float grlColorsWidth;uniform vec2 grl_colorModeAndColorDistributionType;uniform vec3 grlColor;varying float grlCounters;varying float grlColorPointer;void main() {float grlColorMode=grl_colorModeAndColorDistributionType.x;float grlColorDistributionType=grl_colorModeAndColorDistributionType.y;gl_FragColor=vec4(grlColor,1.);gl_FragColor.a=step(grlCounters,grlVisibility);if (gl_FragColor.a==0.) discard;if( grlUseDash==1. ){gl_FragColor.a=ceil(mod(grlCounters+grlDashOffset,grlDashArray)-(grlDashArray*grlDashRatio));if (gl_FragColor.a==0.) discard;} if (grlUseColors==1.) {vec4 textureColor;if (grlColorDistributionType==COLOR_DISTRIBUTION_TYPE_LINE) { textureColor=texture2D(grlColors,vec2(grlCounters,0.),0.);} else {textureColor=texture2D(grlColors,vec2(grlColorPointer/grlColorsWidth,0.),0.);} if (grlColorMode==COLOR_MODE_SET) {gl_FragColor=textureColor;} else if (grlColorMode==COLOR_MODE_ADD) {gl_FragColor+=textureColor;} else if (grlColorMode==COLOR_MODE_MULTIPLY) {gl_FragColor*=textureColor;}}} `; Le.ShadersStore[fhe] = Ahe; const dhe = "greasedLineVertexShader", vhe = `precision highp float; #include attribute float grl_widths;attribute vec3 grl_offsets;attribute float grl_colorPointers;attribute vec3 position;uniform mat4 viewProjection;uniform mat4 projection;varying float grlCounters;varying float grlColorPointer; #ifdef GREASED_LINE_CAMERA_FACING attribute vec4 grl_nextAndCounters;attribute vec4 grl_previousAndSide;uniform vec2 grlResolution;uniform float grlAspect;uniform float grlWidth;uniform float grlSizeAttenuation;vec2 grlFix( vec4 i,float aspect ) {vec2 res=i.xy/i.w;res.x*=aspect;return res;} #else attribute vec3 grl_slopes;attribute float grl_counters; #endif void main() { #include grlColorPointer=grl_colorPointers; #ifdef GREASED_LINE_CAMERA_FACING float grlBaseWidth=grlWidth;vec3 grlPrevious=grl_previousAndSide.xyz;float grlSide=grl_previousAndSide.w;vec3 grlNext=grl_nextAndCounters.xyz;grlCounters=grl_nextAndCounters.w;mat4 grlMatrix=viewProjection*finalWorld ;vec3 grlPositionOffset=grl_offsets;vec4 grlFinalPosition=grlMatrix*vec4( position+grlPositionOffset ,1.0 );vec4 grlPrevPos=grlMatrix*vec4( grlPrevious+grlPositionOffset,1.0 );vec4 grlNextPos=grlMatrix*vec4( grlNext+grlPositionOffset,1.0 );vec2 grlCurrentP=grlFix( grlFinalPosition,grlAspect );vec2 grlPrevP=grlFix( grlPrevPos,grlAspect );vec2 grlNextP=grlFix( grlNextPos,grlAspect );float grlWidth=grlBaseWidth*grl_widths;vec2 grlDir;if( grlNextP==grlCurrentP ) grlDir=normalize( grlCurrentP-grlPrevP );else if( grlPrevP==grlCurrentP ) grlDir=normalize( grlNextP-grlCurrentP );else {vec2 grlDir1=normalize( grlCurrentP-grlPrevP );vec2 grlDir2=normalize( grlNextP-grlCurrentP );grlDir=normalize( grlDir1+grlDir2 );} vec4 grlNormal=vec4( -grlDir.y,grlDir.x,0.,1. ); #ifdef GREASED_LINE_RIGHT_HANDED_COORDINATE_SYSTEM grlNormal.xy*=-.5*grlWidth; #else grlNormal.xy*=.5*grlWidth; #endif grlNormal*=projection;if (grlSizeAttenuation==1.) {grlNormal.xy*=grlFinalPosition.w;grlNormal.xy/=( vec4( grlResolution,0.,1. )*projection ).xy;} grlFinalPosition.xy+=grlNormal.xy*grlSide;gl_Position=grlFinalPosition; #else grlCounters=grl_counters;vec4 grlFinalPosition=worldViewProjection*vec4( (position+grl_offsets)+grl_slopes*grl_widths ,1.0 ) ;gl_Position=grlFinalPosition; #endif } `; Le.ShadersStore[dhe] = vhe; class AY extends Zo { /** * GreasedLineSimple material constructor * @param name material name * @param scene the scene * @param options material options */ constructor(e, t, r) { var n, i, s, a, f, o, d, v, u, l, P, p, c, H; const T = [ `COLOR_DISTRIBUTION_TYPE_LINE ${K2.COLOR_DISTRIBUTION_TYPE_LINE}.`, `COLOR_DISTRIBUTION_TYPE_SEGMENT ${K2.COLOR_DISTRIBUTION_TYPE_SEGMENT}.`, `COLOR_MODE_SET ${Tc.COLOR_MODE_SET}.`, `COLOR_MODE_ADD ${Tc.COLOR_MODE_ADD}.`, `COLOR_MODE_MULTIPLY ${Tc.COLOR_MODE_MULTIPLY}.` ], q = ["position", "grl_widths", "grl_offsets", "grl_colorPointers"]; t.useRightHandedSystem && T.push("GREASED_LINE_RIGHT_HANDED_COORDINATE_SYSTEM"), r.cameraFacing ? (T.push("GREASED_LINE_CAMERA_FACING"), q.push("grl_previousAndSide", "grl_nextAndCounters")) : (q.push("grl_slopes"), q.push("grl_counters")), super(e, t, { vertex: "greasedLine", fragment: "greasedLine" }, { attributes: q, uniforms: [ "world", "viewProjection", "view", "projection", "grlColorsWidth", "grlUseColors", "grlWidth", "grlColor", "grl_colorModeAndColorDistributionType", "grlResolution", "grlAspect", "grlAizeAttenuation", "grlDashArray", "grlDashOffset", "grlDashRatio", "grlUseDash", "grlVisibility" ], samplers: ["grlColors"], defines: T }), this._color = Ne.White(), this._colorsDistributionType = K2.COLOR_DISTRIBUTION_TYPE_SEGMENT, this._colorsTexture = null, r = r || { color: D6.DEFAULT_COLOR }; const b = t.getEngine(); this.visibility = (n = r.visibility) !== null && n !== void 0 ? n : 1, this.useDash = (i = r.useDash) !== null && i !== void 0 ? i : !1, this.dashRatio = (s = r.dashRatio) !== null && s !== void 0 ? s : 0.5, this.dashOffset = (a = r.dashOffset) !== null && a !== void 0 ? a : 0, this.dashCount = (f = r.dashCount) !== null && f !== void 0 ? f : 1, this.width = r.width ? r.width : r.sizeAttenuation && r.cameraFacing ? D6.DEFAULT_WIDTH_ATTENUATED : D6.DEFAULT_WIDTH, this.sizeAttenuation = (o = r.sizeAttenuation) !== null && o !== void 0 ? o : !1, this.color = (d = r.color) !== null && d !== void 0 ? d : Ne.White(), this.useColors = (v = r.useColors) !== null && v !== void 0 ? v : !1, this.colorsDistributionType = (u = r.colorDistributionType) !== null && u !== void 0 ? u : K2.COLOR_DISTRIBUTION_TYPE_SEGMENT, this.colorsSampling = (l = r.colorsSampling) !== null && l !== void 0 ? l : Bo.NEAREST_NEAREST, this.colorMode = (P = r.colorMode) !== null && P !== void 0 ? P : Tc.COLOR_MODE_SET, this._colors = (p = r.colors) !== null && p !== void 0 ? p : null, this._cameraFacing = (c = r.cameraFacing) !== null && c !== void 0 ? c : !0, this.resolution = (H = r.resolution) !== null && H !== void 0 ? H : new at(b.getRenderWidth(), b.getRenderHeight()), r.colorsTexture ? this.colorsTexture = r.colorsTexture : this.colorsTexture = Qs.PrepareEmptyColorsTexture(t), this._colors && this.setColors(this._colors), b.onDisposeObservable.add(() => { Qs.DisposeEmptyColorsTexture(); }); } /** * Disposes the plugin material. */ dispose() { var e; (e = this._colorsTexture) === null || e === void 0 || e.dispose(), super.dispose(); } _setColorModeAndColorDistributionType() { this.setVector2("grl_colorModeAndColorDistributionType", new at(this._colorMode, this._colorsDistributionType)); } /** * Updates the material. Use when material created in lazy mode. */ updateLazy() { this._colors && this.setColors(this._colors, !1, !0); } /** * Returns the colors used to colorize the line */ get colors() { return this._colors; } /** * Sets the colors used to colorize the line */ set colors(e) { this.setColors(e); } /** * Creates or updates the colors texture * @param colors color table RGBA * @param lazy if lazy, the colors are not updated * @param forceNewTexture force creation of a new texture * @returns */ setColors(e, t = !1, r = !1) { var n, i, s, a; const f = (i = (n = this._colors) === null || n === void 0 ? void 0 : n.length) !== null && i !== void 0 ? i : 0; if (this._colors = e, e === null || e.length === 0) { (s = this._colorsTexture) === null || s === void 0 || s.dispose(); return; } if (!(t && !r)) if (this._colorsTexture && f === e.length && !r) { const o = Qs.Color3toRGBAUint8(e); this._colorsTexture.update(o); } else (a = this._colorsTexture) === null || a === void 0 || a.dispose(), this.colorsTexture = Qs.CreateColorsTexture(`${this.name}-colors-texture`, e, this.colorsSampling, this.getScene()); } /** * Gets the colors texture */ get colorsTexture() { var e; return (e = this._colorsTexture) !== null && e !== void 0 ? e : null; } /** * Sets the colorsTexture */ set colorsTexture(e) { this._colorsTexture = e, this.setFloat("grlColorsWidth", this._colorsTexture.getSize().width), this.setTexture("grlColors", this._colorsTexture); } /** * Line base width. At each point the line width is calculated by widths[pointIndex] * width */ get width() { return this._width; } /** * Line base width. At each point the line width is calculated by widths[pointIndex] * width */ set width(e) { this._width = e, this.setFloat("grlWidth", e); } /** * Whether to use the colors option to colorize the line */ get useColors() { return this._useColors; } set useColors(e) { this._useColors = e, this.setFloat("grlUseColors", Qs.BooleanToNumber(e)); } /** * The type of sampling of the colors texture. The values are the same when using with textures. */ get colorsSampling() { return this._colorsSampling; } /** * The type of sampling of the colors texture. The values are the same when using with textures. */ set colorsSampling(e) { this._colorsSampling = e; } /** * Normalized value of how much of the line will be visible * 0 - 0% of the line will be visible * 1 - 100% of the line will be visible */ get visibility() { return this._visibility; } set visibility(e) { this._visibility = e, this.setFloat("grlVisibility", e); } /** * Turns on/off dash mode */ get useDash() { return this._useDash; } /** * Turns on/off dash mode */ set useDash(e) { this._useDash = e, this.setFloat("grlUseDash", Qs.BooleanToNumber(e)); } /** * Gets the dash offset */ get dashOffset() { return this._dashOffset; } /** * Sets the dash offset */ set dashOffset(e) { this._dashOffset = e, this.setFloat("grlDashOffset", e); } /** * Length of the dash. 0 to 1. 0.5 means half empty, half drawn. */ get dashRatio() { return this._dashRatio; } /** * Length of the dash. 0 to 1. 0.5 means half empty, half drawn. */ set dashRatio(e) { this._dashRatio = e, this.setFloat("grlDashRatio", e); } /** * Gets the number of dashes in the line */ get dashCount() { return this._dashCount; } /** * Sets the number of dashes in the line * @param value dash */ set dashCount(e) { this._dashCount = e, this._dashArray = 1 / e, this.setFloat("grlDashArray", this._dashArray); } /** * False means 1 unit in width = 1 unit on scene, true means 1 unit in width is reduced on the screen to make better looking lines */ get sizeAttenuation() { return this._sizeAttenuation; } /** * Turn on/off attenuation of the width option and widths array. * @param value false means 1 unit in width = 1 unit on scene, true means 1 unit in width is reduced on the screen to make better looking lines */ set sizeAttenuation(e) { this._sizeAttenuation = e, this.setFloat("grlSizeAttenuation", Qs.BooleanToNumber(e)); } /** * Gets the color of the line */ get color() { return this.color; } /** * Sets the color of the line * @param value Color3 */ set color(e) { this.setColor(e); } /** * Sets the color of the line. If set the whole line will be mixed with this color according to the colorMode option. * The simple material always needs a color to be set. If you set it to null it will set the color to the default color (GreasedLineSimpleMaterial.DEFAULT_COLOR). * @param value color */ setColor(e) { e = e ?? D6.DEFAULT_COLOR, this._color = e, this.setColor3("grlColor", e); } /** * Gets the color distributiopn type */ get colorsDistributionType() { return this._colorsDistributionType; } /** * Sets the color distribution type * @see GreasedLineMeshColorDistributionType * @param value color distribution type */ set colorsDistributionType(e) { this._colorsDistributionType = e, this._setColorModeAndColorDistributionType(); } /** * Gets the mixing mode of the color and colors paramaters. Default value is GreasedLineMeshColorMode.SET. * MATERIAL_TYPE_SIMPLE mixes the color and colors of the greased line material. * @see GreasedLineMeshColorMode */ get colorMode() { return this._colorMode; } /** * Sets the mixing mode of the color and colors paramaters. Default value is GreasedLineMeshColorMode.SET. * MATERIAL_TYPE_SIMPLE mixes the color and colors of the greased line material. * @see GreasedLineMeshColorMode */ set colorMode(e) { this._colorMode = e, this._setColorModeAndColorDistributionType(); } /** * Gets the resolution */ get resolution() { return this._resolution; } /** * Sets the resolution * @param value resolution of the screen for GreasedLine */ set resolution(e) { this._resolution = e, this.setVector2("grlResolution", e), this.setFloat("grlAspect", e.x / e.y); } /** * Serializes this plugin material * @returns serializationObjec */ serialize() { const e = super.serialize(), t = { colorDistributionType: this._colorsDistributionType, colorsSampling: this._colorsSampling, colorMode: this._colorMode, color: this._color, dashCount: this._dashCount, dashOffset: this._dashOffset, dashRatio: this._dashRatio, resolution: this._resolution, sizeAttenuation: this._sizeAttenuation, useColors: this._useColors, useDash: this._useDash, visibility: this._visibility, width: this._width, cameraFacing: this._cameraFacing }; return this._colors && (t.colors = this._colors), e.greasedLineMaterialOptions = t, e; } /** * Parses a serialized objects * @param source serialized object * @param scene scene * @param _rootUrl root url for textures */ parse(e, t, r) { var n, i; const s = e.greasedLineMaterialOptions; (n = this._colorsTexture) === null || n === void 0 || n.dispose(), s.color && (this.color = s.color), s.colorDistributionType && (this.colorsDistributionType = s.colorDistributionType), s.colorsSampling && (this.colorsSampling = s.colorsSampling), s.colorMode && (this.colorMode = s.colorMode), s.useColors && (this.useColors = s.useColors), s.visibility && (this.visibility = s.visibility), s.useDash && (this.useDash = s.useDash), s.dashCount && (this.dashCount = s.dashCount), s.dashRatio && (this.dashRatio = s.dashRatio), s.dashOffset && (this.dashOffset = s.dashOffset), s.width && (this.width = s.width), s.sizeAttenuation && (this.sizeAttenuation = s.sizeAttenuation), s.resolution && (this.resolution = s.resolution), s.colors ? this.colorsTexture = Qs.CreateColorsTexture(`${this.name}-colors-texture`, s.colors, this.colorsSampling, this.getScene()) : this.colorsTexture = Qs.PrepareEmptyColorsTexture(t), this._cameraFacing = (i = s.cameraFacing) !== null && i !== void 0 ? i : !0, this.setDefine("GREASED_LINE_CAMERA_FACING", this._cameraFacing); } } const uhe = `#if defined(DBG_ENABLED) attribute float dbg_initialPass; varying vec3 dbg_vBarycentric; flat varying vec3 dbg_vVertexWorldPos; flat varying float dbg_vPass; #endif`, lhe = `#if defined(DBG_ENABLED) float dbg_vertexIndex = mod(float(gl_VertexID), 3.); if (dbg_vertexIndex == 0.0) { dbg_vBarycentric = vec3(1.,0.,0.); } else if (dbg_vertexIndex == 1.0) { dbg_vBarycentric = vec3(0.,1.,0.); } else { dbg_vBarycentric = vec3(0.,0.,1.); } dbg_vVertexWorldPos = vPositionW; dbg_vPass = dbg_initialPass; #endif`, Phe = `#if defined(DBG_ENABLED) uniform vec3 dbg_shadedDiffuseColor; uniform vec4 dbg_shadedSpecularColorPower; uniform vec3 dbg_thicknessRadiusScale; #if DBG_MODE == 2 || DBG_MODE == 3 uniform vec3 dbg_vertexColor; #endif #if DBG_MODE == 1 uniform vec3 dbg_wireframeTrianglesColor; #elif DBG_MODE == 3 uniform vec3 dbg_wireframeVerticesColor; #elif DBG_MODE == 4 || DBG_MODE == 5 uniform vec3 dbg_uvPrimaryColor; uniform vec3 dbg_uvSecondaryColor; #elif DBG_MODE == 7 uniform vec3 dbg_materialColor; #endif #endif`, che = `#if defined(DBG_ENABLED) varying vec3 dbg_vBarycentric; flat varying vec3 dbg_vVertexWorldPos; flat varying float dbg_vPass; #if !defined(DBG_MULTIPLY) vec3 dbg_applyShading(vec3 color) { vec3 N = vNormalW.xyz; vec3 L = normalize(vEyePosition.xyz - vPositionW.xyz); vec3 H = normalize(L + L); float LdotN = clamp(dot(L,N), 0., 1.); float HdotN = clamp(dot(H,N), 0., 1.); float specTerm = pow(HdotN, dbg_shadedSpecularColorPower.w); color *= (LdotN / PI); color += dbg_shadedSpecularColorPower.rgb * (specTerm / PI); return color; } #endif #if DBG_MODE == 1 || DBG_MODE == 3 float dbg_edgeFactor() { vec3 d = fwidth(dbg_vBarycentric); vec3 a3 = smoothstep(vec3(0.), d * dbg_thicknessRadiusScale.x, dbg_vBarycentric); return min(min(a3.x, a3.y), a3.z); } #endif #if DBG_MODE == 2 || DBG_MODE == 3 float dbg_cornerFactor() { vec3 worldPos = vPositionW; float dist = length(worldPos - dbg_vVertexWorldPos); float camDist = length(worldPos - vEyePosition.xyz); float d = sqrt(camDist) * .001; return smoothstep((dbg_thicknessRadiusScale.y * d), ((dbg_thicknessRadiusScale.y * 1.01) * d), dist); } #endif #if (DBG_MODE == 4 && defined(UV1)) || (DBG_MODE == 5 && defined(UV2)) float dbg_checkerboardFactor(vec2 uv) { vec2 f = fract(uv * dbg_thicknessRadiusScale.z); f -= .5; return (f.x * f.y) > 0. ? 1. : 0.; } #endif #endif`, phe = `#if defined(DBG_ENABLED) vec3 dbg_color = vec3(1.); #if DBG_MODE == 1 dbg_color = mix(dbg_wireframeTrianglesColor, vec3(1.), dbg_edgeFactor()); #elif DBG_MODE == 2 || DBG_MODE == 3 float dbg_cornerFactor = dbg_cornerFactor(); if (dbg_vPass == 0. && dbg_cornerFactor == 1.) discard; dbg_color = mix(dbg_vertexColor, vec3(1.), dbg_cornerFactor); #if DBG_MODE == 3 dbg_color *= mix(dbg_wireframeVerticesColor, vec3(1.), dbg_edgeFactor()); #endif #elif DBG_MODE == 4 && defined(UV1) dbg_color = mix(dbg_uvPrimaryColor, dbg_uvSecondaryColor, dbg_checkerboardFactor(vMainUV1)); #elif DBG_MODE == 5 && defined(UV2) dbg_color = mix(dbg_uvPrimaryColor, dbg_uvSecondaryColor, dbg_checkerboardFactor(vMainUV2)); #elif DBG_MODE == 6 && defined(VERTEXCOLOR) dbg_color = vColor.rgb; #elif DBG_MODE == 7 dbg_color = dbg_materialColor; #endif #if defined(DBG_MULTIPLY) gl_FragColor *= vec4(dbg_color, 1.); #else #if DBG_MODE != 6 gl_FragColor = vec4(dbg_applyShading(dbg_shadedDiffuseColor) * dbg_color, 1.); #else gl_FragColor = vec4(dbg_color, 1.); #endif #endif #endif`, lne = [ new Ne(0.98, 0.26, 0.38), new Ne(0.47, 0.75, 0.3), new Ne(0, 0.26, 0.77), new Ne(0.97, 0.6, 0.76), new Ne(0.19, 0.63, 0.78), new Ne(0.98, 0.8, 0.6), new Ne(0.65, 0.43, 0.15), new Ne(0.15, 0.47, 0.22), new Ne(0.67, 0.71, 0.86), new Ne(0.09, 0.46, 0.56), new Ne(0.8, 0.98, 0.02), new Ne(0.39, 0.29, 0.13), new Ne(0.53, 0.63, 0.06), new Ne(0.95, 0.96, 0.41), new Ne(1, 0.72, 0.94), new Ne(0.63, 0.08, 0.31), new Ne(0.66, 0.96, 0.95), new Ne(0.22, 0.14, 0.19), new Ne(0.14, 0.65, 0.59), new Ne(0.93, 1, 0.68), new Ne(0.93, 0.14, 0.44), new Ne(0.47, 0.86, 0.67), new Ne(0.85, 0.07, 0.78), new Ne(0.53, 0.64, 0.98), new Ne(0.43, 0.37, 0.56), new Ne(0.71, 0.65, 0.25), new Ne(0.66, 0.19, 0.01), new Ne(0.94, 0.53, 0.12), new Ne(0.41, 0.44, 0.44), new Ne(0.24, 0.71, 0.96), new Ne(0.57, 0.28, 0.56), new Ne(0.44, 0.98, 0.42) ]; var im; (function(A) { A[A.NONE = 0] = "NONE", A[A.TRIANGLES = 1] = "TRIANGLES", A[A.VERTICES = 2] = "VERTICES", A[A.TRIANGLES_VERTICES = 3] = "TRIANGLES_VERTICES", A[A.UV0 = 4] = "UV0", A[A.UV1 = 5] = "UV1", A[A.VERTEXCOLORS = 6] = "VERTEXCOLORS", A[A.MATERIALIDS = 7] = "MATERIALIDS"; })(im || (im = {})); class hhe extends na { constructor() { super(...arguments), this.DBG_MODE = im.NONE, this.DBG_MULTIPLY = !0, this.DBG_ENABLED = !0; } } class Q9 extends Gl { /** @internal */ _markAllDefinesAsDirty() { this._enable(this._isEnabled), this.markAllDefinesAsDirty(); } /** * Creates a new MeshDebugPluginMaterial * @param material Material to attach the mesh debug plugin to * @param options Options for the mesh debug plugin */ constructor(e, t = {}) { var r, n, i, s, a, f, o, d, v, u, l, P, p; const c = new hhe(); c.DBG_MODE = (r = t.mode) !== null && r !== void 0 ? r : c.DBG_MODE, c.DBG_MULTIPLY = (n = t.multiply) !== null && n !== void 0 ? n : c.DBG_MULTIPLY, super(e, "MeshDebug", 200, c, !0, !0), this._mode = c.DBG_MODE, this._multiply = c.DBG_MULTIPLY, this.shadedDiffuseColor = (i = t.shadedDiffuseColor) !== null && i !== void 0 ? i : new Ne(1, 1, 1), this.shadedSpecularColor = (s = t.shadedSpecularColor) !== null && s !== void 0 ? s : new Ne(0.8, 0.8, 0.8), this.shadedSpecularPower = (a = t.shadedSpecularPower) !== null && a !== void 0 ? a : 10, this.wireframeThickness = (f = t.wireframeThickness) !== null && f !== void 0 ? f : 0.7, this.wireframeTrianglesColor = (o = t.wireframeTrianglesColor) !== null && o !== void 0 ? o : new Ne(0, 0, 0), this.wireframeVerticesColor = (d = t.wireframeVerticesColor) !== null && d !== void 0 ? d : new Ne(0.8, 0.8, 0.8), this.vertexColor = (v = t.vertexColor) !== null && v !== void 0 ? v : new Ne(0, 0, 0), this.vertexRadius = (u = t.vertexRadius) !== null && u !== void 0 ? u : 1.2, this.uvScale = (l = t.uvScale) !== null && l !== void 0 ? l : 20, this.uvPrimaryColor = (P = t.uvPrimaryColor) !== null && P !== void 0 ? P : new Ne(1, 1, 1), this.uvSecondaryColor = (p = t.uvSecondaryColor) !== null && p !== void 0 ? p : new Ne(0.5, 0.5, 0.5), this._materialColor = Q9.MaterialColors[Q9._PluginCount++ % Q9.MaterialColors.length], this.isEnabled = !0; } /** * Get the class name * @returns Class name */ getClassName() { return "MeshDebugPluginMaterial"; } /** * Gets whether the mesh debug plugin is enabled in the material. */ get isEnabled() { return this._isEnabled; } /** * Sets whether the mesh debug plugin is enabled in the material. * @param value enabled */ set isEnabled(e) { if (this._isEnabled !== e) { if (!this._material.getScene().getEngine().isWebGPU && this._material.getScene().getEngine().webGLVersion == 1) { Se.Error("MeshDebugPluginMaterial is not supported on WebGL 1.0."), this._isEnabled = !1; return; } this._isEnabled = e, this._markAllDefinesAsDirty(); } } /** * Prepare the defines * @param defines Mesh debug defines * @param scene Scene * @param mesh Mesh associated with material */ prepareDefines(e, t, r) { (this._mode == im.VERTICES || this._mode == im.TRIANGLES || this._mode == im.TRIANGLES_VERTICES) && !r.isVerticesDataPresent("dbg_initialPass") && Se.Warn("For best results with TRIANGLES, TRIANGLES_VERTICES, or VERTICES modes, please use MeshDebugPluginMaterial.PrepareMeshForTrianglesAndVerticesMode() on mesh.", 1), e.DBG_MODE = this._mode, e.DBG_MULTIPLY = this._multiply, e.DBG_ENABLED = this._isEnabled; } /** * Get the shader attributes * @param attributes Array of attributes */ getAttributes(e) { e.push("dbg_initialPass"); } /** * Get the shader uniforms * @returns Uniforms */ getUniforms() { return { ubo: [ { name: "dbg_shadedDiffuseColor", size: 3, type: "vec3" }, { name: "dbg_shadedSpecularColorPower", size: 4, type: "vec4" }, { name: "dbg_thicknessRadiusScale", size: 3, type: "vec3" }, { name: "dbg_wireframeTrianglesColor", size: 3, type: "vec3" }, { name: "dbg_wireframeVerticesColor", size: 3, type: "vec3" }, { name: "dbg_vertexColor", size: 3, type: "vec3" }, { name: "dbg_uvPrimaryColor", size: 3, type: "vec3" }, { name: "dbg_uvSecondaryColor", size: 3, type: "vec3" }, { name: "dbg_materialColor", size: 3, type: "vec3" } ], fragment: Phe }; } /** * Bind the uniform buffer * @param uniformBuffer Uniform buffer */ bindForSubMesh(e) { this._isEnabled && (e.updateFloat3("dbg_shadedDiffuseColor", this.shadedDiffuseColor.r, this.shadedDiffuseColor.g, this.shadedDiffuseColor.b), e.updateFloat4("dbg_shadedSpecularColorPower", this.shadedSpecularColor.r, this.shadedSpecularColor.g, this.shadedSpecularColor.b, this.shadedSpecularPower), e.updateFloat3("dbg_thicknessRadiusScale", this.wireframeThickness, this.vertexRadius, this.uvScale), e.updateColor3("dbg_wireframeTrianglesColor", this.wireframeTrianglesColor), e.updateColor3("dbg_wireframeVerticesColor", this.wireframeVerticesColor), e.updateColor3("dbg_vertexColor", this.vertexColor), e.updateColor3("dbg_uvPrimaryColor", this.uvPrimaryColor), e.updateColor3("dbg_uvSecondaryColor", this.uvSecondaryColor), e.updateColor3("dbg_materialColor", this._materialColor)); } /** * Get shader code * @param shaderType "vertex" or "fragment" * @returns Shader code */ getCustomCode(e) { return e === "vertex" ? { CUSTOM_VERTEX_DEFINITIONS: uhe, CUSTOM_VERTEX_MAIN_END: lhe } : { CUSTOM_FRAGMENT_DEFINITIONS: che, CUSTOM_FRAGMENT_MAIN_END: phe }; } /** * Resets static variables of the plugin to their original state */ static Reset() { this._PluginCount = 0, this.MaterialColors = lne; } /** * Renders triangles in a mesh 3 times by tripling the indices in the index buffer. * Used to prepare a mesh to be rendered in `TRIANGLES`, `VERTICES`, or `TRIANGLES_VERTICES` modes. * NOTE: This is a destructive operation. The mesh's index buffer and vertex buffers are modified, and a new vertex buffer is allocated. * If you'd like the ability to revert these changes, toggle the optional `returnRollback` flag. * @param mesh the mesh to target * @param returnRollback whether or not to return a function that reverts mesh to its initial state. Default: false. * @returns a rollback function if `returnRollback` is true, otherwise an empty function. */ static PrepareMeshForTrianglesAndVerticesMode(e, t = !1) { let r = () => { }; if (e.getTotalIndices() == 0) return r; if (t) { const d = e.getVerticesDataKinds(), v = e.getIndices(), u = {}; for (const l of d) u[l] = e.getVerticesData(l); r = function() { e.setIndices(v); for (const l of d) { const P = e.getVertexBuffer(l).getStrideSize(); e.setVerticesData(l, u[l], void 0, P); } e.removeVerticesData("dbg_initialPass"); }; } let n = Array.from(e.getIndices()); const i = []; for (let d = 0; d < n.length; d += 3) i.push(n[d + 1], n[d + 2], n[d + 0]); e.setIndices(n.concat(i)), e.convertToUnIndexedMesh(), e.isUnIndexed = !1, n = Array.from(e.getIndices()); const s = []; for (let d = n.length / 2; d < n.length; d += 3) s.push(n[d + 1], n[d + 2], n[d + 0]); e.setIndices(n.concat(s)); const a = e.getTotalVertices(), f = a / 2, o = new Array(a).fill(1, 0, f).fill(0, f, a); return e.setVerticesData("dbg_initialPass", o, !1, 1), r; } } Q9._PluginCount = 0; Q9.MaterialColors = lne; C([ Oi() ], Q9.prototype, "_materialColor", void 0); C([ M() ], Q9.prototype, "_isEnabled", void 0); C([ M(), At("_markAllDefinesAsDirty") ], Q9.prototype, "mode", void 0); C([ M(), At("_markAllDefinesAsDirty") ], Q9.prototype, "multiply", void 0); C([ Oi() ], Q9.prototype, "shadedDiffuseColor", void 0); C([ Oi() ], Q9.prototype, "shadedSpecularColor", void 0); C([ M() ], Q9.prototype, "shadedSpecularPower", void 0); C([ M() ], Q9.prototype, "wireframeThickness", void 0); C([ Oi() ], Q9.prototype, "wireframeTrianglesColor", void 0); C([ Oi() ], Q9.prototype, "wireframeVerticesColor", void 0); C([ Oi() ], Q9.prototype, "vertexColor", void 0); C([ M() ], Q9.prototype, "vertexRadius", void 0); C([ M() ], Q9.prototype, "uvScale", void 0); C([ Oi() ], Q9.prototype, "uvPrimaryColor", void 0); C([ Oi() ], Q9.prototype, "uvSecondaryColor", void 0); Ue("BABYLON.MeshDebugPluginMaterial", Q9); Object.defineProperty(Wt.prototype, "decalMap", { get: function() { if (!this._decalMap) { if (this._uniformBufferLayoutBuilt) return null; this._decalMap = new cU(this); } return this._decalMap; }, enumerable: !0, configurable: !0 }); Object.defineProperty(fs.prototype, "decalMap", { get: function() { if (!this._decalMap) { if (this._uniformBufferLayoutBuilt) return null; this._decalMap = new cU(this); } return this._decalMap; }, enumerable: !0, configurable: !0 }); Object.defineProperty(jn.prototype, "decalMap", { get: function() { return this._decalMap; }, set: function(A) { this._decalMap = A; }, enumerable: !0, configurable: !0 }); class g0 { /** * Creates a new Polar object * @param radius the radius of the vector * @param theta the angle of the vector */ constructor(e, t) { this.radius = e, this.theta = t; } /** * Gets the class name * @returns the string "Polar" */ getClassName() { return "Polar"; } /** * Converts the current polar to a string * @returns the current polar as a string */ toString() { return JSON.stringify(this); } /** * Converts the current polar to an array * @reutrns the current polar as an array */ asArray() { return [this.radius, this.theta]; } /** * Adds the current Polar and the given Polar and stores the result * @param polar the polar to add * @param ref the polar to store the result in * @returns the updated ref */ addToRef(e, t) { return t.radius = this.radius + e.radius, t.theta = this.theta + e.theta, t; } /** * Adds the current Polar and the given Polar * @param polar the polar to add * @returns the sum polar */ add(e) { const t = new g0(0, 0); return this.addToRef(e, t), t; } /** * Adds the given polar to the current polar * @param polar the polar to add * @returns the current polar */ addInPlace(e) { return this.addToRef(e, this), this; } /** * Adds the provided values to the current polar * @param radius the amount to add to the radius * @param theta the amount to add to the theta * @returns the current polar */ addInPlaceFromFloats(e, t) { return this.radius += e, this.theta += t, this; } /** * Subtracts the given Polar from the current Polar and stores the result * @param polar the polar to subtract * @param ref the polar to store the result in * @returns the updated ref */ subtractToRef(e, t) { return t.radius = this.radius - e.radius, t.theta = this.theta - e.theta, t; } /** * Subtracts the given Polar from the current Polar * @param polar the polar to subtract * @returns the difference polar */ subtract(e) { const t = new g0(0, 0); return this.subtractToRef(e, t), t; } /** * Subtracts the given Polar from the current Polar * @param polar the polar to subtract * @returns the current polar */ subtractInPlace(e) { return this.subtractToRef(e, this), this; } /** * Subtracts the given floats from the current polar * @param radius the amount to subtract from the radius * @param theta the amount to subtract from the theta * @param ref the polar to store the result in * @returns the updated ref */ subtractFromFloatsToRef(e, t, r) { return r.radius = this.radius - e, r.theta = this.theta - t, r; } /** * Subtracts the given floats from the current polar * @param radius the amount to subtract from the radius * @param theta the amount to subtract from the theta * @returns the difference polar */ subtractFromFloats(e, t) { const r = new g0(0, 0); return this.subtractFromFloatsToRef(e, t, r), r; } /** * Multiplies the given Polar with the current Polar and stores the result * @param polar the polar to multiply * @param ref the polar to store the result in * @returns the updated ref */ multiplyToRef(e, t) { return t.radius = this.radius * e.radius, t.theta = this.theta * e.theta, t; } /** * Multiplies the given Polar with the current Polar * @param polar the polar to multiply * @returns the product polar */ multiply(e) { const t = new g0(0, 0); return this.multiplyToRef(e, t), t; } /** * Multiplies the given Polar with the current Polar * @param polar the polar to multiply * @returns the current polar */ multiplyInPlace(e) { return this.multiplyToRef(e, this), this; } /** * Divides the current Polar by the given Polar and stores the result * @param polar the polar to divide * @param ref the polar to store the result in * @returns the updated ref */ divideToRef(e, t) { return t.radius = this.radius / e.radius, t.theta = this.theta / e.theta, t; } /** * Divides the current Polar by the given Polar * @param polar the polar to divide * @returns the quotient polar */ divide(e) { const t = new g0(0, 0); return this.divideToRef(e, t), t; } /** * Divides the current Polar by the given Polar * @param polar the polar to divide * @returns the current polar */ divideInPlace(e) { return this.divideToRef(e, this), this; } /** * Clones the current polar * @returns a clone of the current polar */ clone() { return new g0(this.radius, this.theta); } /** * Copies the source polar into the current polar * @param source the polar to copy from * @returns the current polar */ copyFrom(e) { return this.radius = e.radius, this.theta = e.theta, this; } /** * Copies the given values into the current polar * @param radius the radius to use * @param theta the theta to use * @returns the current polar */ copyFromFloats(e, t) { return this.radius = e, this.theta = t, this; } /** * Scales the current polar and stores the result * @param scale defines the multiplication factor * @param ref where to store the result * @returns the updated ref */ scaleToRef(e, t) { return t.radius = this.radius * e, t.theta = this.theta * e, t; } /** * Scales the current polar and returns a new polar with the scaled coordinates * @param scale defines the multiplication factor * @returns the scaled polar */ scale(e) { const t = new g0(0, 0); return this.scaleToRef(e, t), t; } /** * Scales the current polar * @param scale defines the multiplication factor * @returns the current polar */ scaleInPlace(e) { return this.scaleToRef(e, this), this; } /** * Sets the values of the current polar * @param radius the new radius * @param theta the new theta * @returns the current polar */ set(e, t) { return this.radius = e, this.theta = t, this; } /** * Sets the values of the current polar * @param value the new values * @returns the current polar */ setAll(e) { return this.set(e, e), this; } /** * Gets the rectangular coordinates of the current Polar * @param ref the reference to assign the result * @returns the updated reference */ toVector2ToRef(e) { const t = this.radius * Math.cos(this.theta), r = this.radius * Math.sin(this.theta); return e.set(t, r), e; } /** * Gets the rectangular coordinates of the current Polar * @returns the rectangular coordinates */ toVector2() { const e = new at(0, 0); return this.toVector2ToRef(e); } /** * Converts a given Vector2 to its polar coordinates * @param v the Vector2 to convert * @param ref the reference to assign the result * @returns the updated reference */ static FromVector2ToRef(e, t) { const r = Math.sign(e.y) * Math.acos(e.x / e.length()); return t.radius = e.length(), t.theta = r, t; } /** * Converts a given Vector2 to its polar coordinates * @param v the Vector2 to convert * @returns a Polar */ static FromVector2(e) { const t = new g0(0, 0); return g0.FromVector2ToRef(e, t), t; } /** * Converts an array of floats to a polar * @param array the array to convert * @returns the converted polar */ static FromArray(e) { return new g0(e[0], e[1]); } } class X0 { /** * @param radius spherical radius * @param theta angle from positive y axis to radial line from 0 to PI (vertical) * @param phi angle from positive x axis measured anticlockwise from -PI to PI (horizontal) */ constructor(e, t, r) { this.radius = e, this.theta = t, this.phi = r; } /** * Gets the class name * @returns the string "Spherical" */ getClassName() { return "Spherical"; } /** * Converts the current spherical to a string * @returns the current spherical as a string */ toString() { return JSON.stringify(this); } /** * Converts the current spherical to an array * @reutrns the current spherical as an array */ asArray() { return [this.radius, this.theta, this.phi]; } /** * Adds the current Spherical and the given Spherical and stores the result * @param spherical the spherical to add * @param ref the spherical to store the result in * @returns the updated ref */ addToRef(e, t) { return t.radius = this.radius + e.radius, t.theta = this.theta + e.theta, t.phi = this.phi + e.phi, t; } /** * Adds the current Spherical and the given Spherical * @param spherical the spherical to add * @returns the sum spherical */ add(e) { const t = new X0(0, 0, 0); return this.addToRef(e, t), t; } /** * Adds the given spherical to the current spherical * @param spherical the spherical to add * @returns the current spherical */ addInPlace(e) { return this.addToRef(e, this), this; } /** * Adds the provided values to the current spherical * @param radius the amount to add to the radius * @param theta the amount to add to the theta * @param phi the amount to add to the phi * @returns the current spherical */ addInPlaceFromFloats(e, t, r) { return this.radius += e, this.theta += t, this.phi += r, this; } /** * Subtracts the given Spherical from the current Spherical and stores the result * @param spherical the spherical to subtract * @param ref the spherical to store the result in * @returns the updated ref */ subtractToRef(e, t) { return t.radius = this.radius - e.radius, t.theta = this.theta - e.theta, t.phi = this.phi - e.phi, t; } /** * Subtracts the given Spherical from the current Spherical * @param spherical the spherical to subtract * @returns the difference spherical */ subtract(e) { const t = new X0(0, 0, 0); return this.subtractToRef(e, t), t; } /** * Subtracts the given Spherical from the current Spherical * @param spherical the spherical to subtract * @returns the current spherical */ subtractInPlace(e) { return this.subtractToRef(e, this), this; } /** * Subtracts the given floats from the current spherical * @param radius the amount to subtract from the radius * @param theta the amount to subtract from the theta * @param phi the amount to subtract from the phi * @param ref the spherical to store the result in * @returns the updated ref */ subtractFromFloatsToRef(e, t, r, n) { return n.radius = this.radius - e, n.theta = this.theta - t, n.phi = this.phi - r, n; } /** * Subtracts the given floats from the current spherical * @param radius the amount to subtract from the radius * @param theta the amount to subtract from the theta * @param phi the amount to subtract from the phi * @returns the difference spherical */ subtractFromFloats(e, t, r) { const n = new X0(0, 0, 0); return this.subtractFromFloatsToRef(e, t, r, n), n; } /** * Multiplies the given Spherical with the current Spherical and stores the result * @param spherical the spherical to multiply * @param ref the spherical to store the result in * @returns the updated ref */ multiplyToRef(e, t) { return t.radius = this.radius * e.radius, t.theta = this.theta * e.theta, t.phi = this.phi * e.phi, t; } /** * Multiplies the given Spherical with the current Spherical * @param spherical the spherical to multiply * @returns the product spherical */ multiply(e) { const t = new X0(0, 0, 0); return this.multiplyToRef(e, t), t; } /** * Multiplies the given Spherical with the current Spherical * @param spherical the spherical to multiply * @returns the current spherical */ multiplyInPlace(e) { return this.multiplyToRef(e, this), this; } /** * Divides the current Spherical by the given Spherical and stores the result * @param spherical the spherical to divide * @param ref the spherical to store the result in * @returns the updated ref */ divideToRef(e, t) { return t.radius = this.radius / e.radius, t.theta = this.theta / e.theta, t.phi = this.phi / e.phi, t; } /** * Divides the current Spherical by the given Spherical * @param spherical the spherical to divide * @returns the quotient spherical */ divide(e) { const t = new X0(0, 0, 0); return this.divideToRef(e, t), t; } /** * Divides the current Spherical by the given Spherical * @param spherical the spherical to divide * @returns the current spherical */ divideInPlace(e) { return this.divideToRef(e, this), this; } /** * Clones the current spherical * @returns a clone of the current spherical */ clone() { return new X0(this.radius, this.theta, this.phi); } /** * Copies the source spherical into the current spherical * @param source the spherical to copy from * @returns the current spherical */ copyFrom(e) { return this.radius = e.radius, this.theta = e.theta, this.phi = e.phi, this; } /** * Copies the given values into the current spherical * @param radius the radius to use * @param theta the theta to use * @param phi the phi to use * @returns the current spherical */ copyFromFloats(e, t, r) { return this.radius = e, this.theta = t, this.phi = r, this; } /** * Scales the current spherical and stores the result * @param scale defines the multiplication factor * @param ref where to store the result * @returns the updated ref */ scaleToRef(e, t) { return t.radius = this.radius * e, t.theta = this.theta * e, t.phi = this.phi * e, t; } /** * Scales the current spherical and returns a new spherical with the scaled coordinates * @param scale defines the multiplication factor * @returns the scaled spherical */ scale(e) { const t = new X0(0, 0, 0); return this.scaleToRef(e, t), t; } /** * Scales the current spherical * @param scale defines the multiplication factor * @returns the current spherical */ scaleInPlace(e) { return this.scaleToRef(e, this), this; } /** * Sets the values of the current spherical * @param radius the new radius * @param theta the new theta * @param phi the new phi * @returns the current spherical */ set(e, t, r) { return this.radius = e, this.theta = t, this.phi = r, this; } /** * Sets the values of the current spherical * @param value the new values * @returns the current spherical */ setAll(e) { return this.set(e, e, e), this; } /** * Assigns the rectangular coordinates of the current Spherical to a Vector3 * @param ref the Vector3 to update * @returns the updated Vector3 */ toVector3ToRef(e) { const t = this.radius * Math.sin(this.theta) * Math.cos(this.phi), r = this.radius * Math.cos(this.theta), n = this.radius * Math.sin(this.theta) * Math.sin(this.phi); return e.set(t, r, n), e; } /** * Gets a Vector3 from the current spherical coordinates * @returns the (x, y,z) form of the current Spherical */ toVector3() { const e = new S(0, 0, 0); return this.toVector3ToRef(e); } /** * Assigns the spherical coordinates from a Vector3 * @param vector the vector to convert * @param ref the Spherical to update * @returns the updated ref */ static FromVector3ToRef(e, t) { return t.radius = e.length(), t.theta = Math.acos(e.y / t.radius), t.phi = Math.atan2(e.z, e.x), t; } /** * Gets a Spherical from a Vector3 * @param vector defines the vector in (x, y, z) coordinate space * @returns a new Spherical */ static FromVector3(e) { const t = new X0(0, 0, 0); return X0.FromVector3ToRef(e, t), t; } /** * Converts an array of floats to a spherical * @param array the array to convert * @returns the converted spherical */ static FromArray(e) { return new X0(e[0], e[1], e[2]); } } function Hhe(A) { return new Promise((e) => { DracoDecoderModule({ wasmBinary: A }).then((t) => { e({ module: t }); }); }); } function TF(A, e, t, r, n) { let i = null, s = null, a = null; try { i = new A.Decoder(), s = new A.DecoderBuffer(), s.Init(e, e.byteLength); let f; const o = i.GetEncodedGeometryType(s); switch (o) { case A.TRIANGULAR_MESH: { const u = new A.Mesh(); if (f = i.DecodeBufferToMesh(s, u), !f.ok() || u.ptr === 0) throw new Error(f.error_msg()); const P = u.num_faces() * 3, p = P * 4, c = A._malloc(p); try { i.GetTrianglesUInt32Array(u, p, c); const H = new Uint32Array(P); H.set(new Uint32Array(A.HEAPF32.buffer, c, P)), r(H); } finally { A._free(c); } a = u; break; } case A.POINT_CLOUD: { const u = new A.PointCloud(); if (f = i.DecodeBufferToPointCloud(s, u), !f.ok() || !u.ptr) throw new Error(f.error_msg()); a = u; break; } default: throw new Error(`Invalid geometry type ${o}`); } const d = a.num_points(), v = (u, l, P, p) => { const c = p.data_type(), H = p.num_components(), T = p.normalized(), q = p.byte_stride(), b = p.byte_offset(), w = { [A.DT_FLOAT32]: { typedArrayConstructor: Float32Array, heap: A.HEAPF32 }, [A.DT_INT8]: { typedArrayConstructor: Int8Array, heap: A.HEAP8 }, [A.DT_INT16]: { typedArrayConstructor: Int16Array, heap: A.HEAP16 }, [A.DT_INT32]: { typedArrayConstructor: Int32Array, heap: A.HEAP32 }, [A.DT_UINT8]: { typedArrayConstructor: Uint8Array, heap: A.HEAPU8 }, [A.DT_UINT16]: { typedArrayConstructor: Uint16Array, heap: A.HEAPU16 }, [A.DT_UINT32]: { typedArrayConstructor: Uint32Array, heap: A.HEAPU32 } }[c]; if (!w) throw new Error(`Invalid data type ${c}`); const m = d * H, I = m * w.typedArrayConstructor.BYTES_PER_ELEMENT, N = A._malloc(I); try { u.GetAttributeDataArrayForAllPoints(l, p, c, I, N); const k = new w.typedArrayConstructor(w.heap.buffer, N, m); n(P, k.slice(), H, b, q, T); } finally { A._free(N); } }; if (t) for (const u in t) { const l = t[u], P = i.GetAttributeByUniqueId(a, l); v(i, a, u, P); } else { const u = { position: A.POSITION, normal: A.NORMAL, color: A.COLOR, uv: A.TEX_COORD }; for (const l in u) { const P = i.GetAttributeId(a, u[l]); if (P !== -1) { const p = i.GetAttribute(a, P); v(i, a, l, p); } } } return d; } finally { a && A.destroy(a), s && A.destroy(s), i && A.destroy(i); } } function ghe() { let A; onmessage = (e) => { const t = e.data; switch (t.id) { case "init": { const r = t.decoder; r.url && (importScripts(r.url), A = DracoDecoderModule({ wasmBinary: r.wasmBinary })), postMessage({ id: "initDone" }); break; } case "decodeMesh": { if (!A) throw new Error("Draco decoder module is not available"); A.then((r) => { const n = TF(r, t.dataView, t.attributes, (i) => { postMessage({ id: "indices", data: i }, [i.buffer]); }, (i, s, a, f, o, d) => { postMessage({ id: "attribute", kind: i, data: s, size: a, byteOffset: f, byteStride: o, normalized: d }, [s.buffer]); }); postMessage({ id: "decodeMeshDone", totalVertices: n }); }); break; } } }; } class Yd { /** * Returns true if the decoder configuration is available. */ static get DecoderAvailable() { const e = Yd.Configuration.decoder; return !!(e.wasmUrl && e.wasmBinaryUrl && typeof WebAssembly == "object" || e.fallbackUrl); } static GetDefaultNumWorkers() { return typeof navigator != "object" || !navigator.hardwareConcurrency ? 1 : Math.min(Math.floor(navigator.hardwareConcurrency * 0.5), 4); } /** * Default instance for the draco compression object. */ static get Default() { return Yd._Default || (Yd._Default = new Yd()), Yd._Default; } /** * Constructor * @param numWorkers The number of workers for async operations. Specify `0` to disable web workers and run synchronously in the current context. */ constructor(e = Yd.DefaultNumWorkers) { const t = Yd.Configuration.decoder, r = t.wasmUrl && t.wasmBinaryUrl && typeof WebAssembly == "object" ? { url: ye.GetBabylonScriptURL(t.wasmUrl, !0), wasmBinaryPromise: ye.LoadFileAsync(ye.GetBabylonScriptURL(t.wasmBinaryUrl, !0)) } : { url: ye.GetBabylonScriptURL(t.fallbackUrl), wasmBinaryPromise: Promise.resolve(void 0) }; e && typeof Worker == "function" && typeof URL == "function" ? this._workerPoolPromise = r.wasmBinaryPromise.then((n) => { const i = `${TF}(${ghe})()`, s = URL.createObjectURL(new Blob([i], { type: "application/javascript" })); return new lU(e, () => new Promise((a, f) => { const o = new Worker(s), d = (u) => { o.removeEventListener("error", d), o.removeEventListener("message", v), f(u); }, v = (u) => { u.data.id === "initDone" && (o.removeEventListener("error", d), o.removeEventListener("message", v), a(o)); }; o.addEventListener("error", d), o.addEventListener("message", v), o.postMessage({ id: "init", decoder: { url: r.url, wasmBinary: n } }); })); }) : this._decoderModulePromise = r.wasmBinaryPromise.then((n) => { if (!r.url) throw new Error("Draco decoder module is not available"); return ye.LoadBabylonScriptAsync(r.url).then(() => Hhe(n)); }); } /** * Stop all async operations and release resources. */ dispose() { this._workerPoolPromise && this._workerPoolPromise.then((e) => { e.dispose(); }), delete this._workerPoolPromise, delete this._decoderModulePromise; } /** * Returns a promise that resolves when ready. Call this manually to ensure draco compression is ready before use. * @returns a promise that resolves when ready */ whenReadyAsync() { return this._workerPoolPromise ? this._workerPoolPromise.then(() => { }) : this._decoderModulePromise ? this._decoderModulePromise.then(() => { }) : Promise.resolve(); } _decodeMeshAsync(e, t, r) { const n = e instanceof ArrayBuffer ? new Int8Array(e) : new Int8Array(e.buffer, e.byteOffset, e.byteLength), i = (s, a) => r && r[s] !== void 0 ? (a !== r[s] && Se.Warn(`Normalized flag from Draco data (${a}) does not match normalized flag from glTF accessor (${r[s]}). Using flag from glTF accessor.`), r[s]) : a; if (this._workerPoolPromise) return this._workerPoolPromise.then((s) => new Promise((a, f) => { s.push((o, d) => { let v = null; const u = [], l = (c) => { o.removeEventListener("error", l), o.removeEventListener("message", P), f(c), d(); }, P = (c) => { const H = c.data; switch (H.id) { case "decodeMeshDone": { o.removeEventListener("error", l), o.removeEventListener("message", P), a({ indices: v, attributes: u, totalVertices: H.totalVertices }), d(); break; } case "indices": { v = H.data; break; } case "attribute": { u.push({ kind: H.kind, data: H.data, size: H.size, byteOffset: H.byteOffset, byteStride: H.byteStride, normalized: i(H.kind, H.normalized) }); break; } } }; o.addEventListener("error", l), o.addEventListener("message", P); const p = n.slice(); o.postMessage({ id: "decodeMesh", dataView: p, attributes: t }, [p.buffer]); }); })); if (this._decoderModulePromise) return this._decoderModulePromise.then((s) => { let a = null; const f = [], o = TF(s.module, n, t, (d) => { a = d; }, (d, v, u, l, P, p) => { f.push({ kind: d, data: v, size: u, byteOffset: l, byteStride: P, normalized: p }); }); return { indices: a, attributes: f, totalVertices: o }; }); throw new Error("Draco decoder module is not available"); } /** * Decode Draco compressed mesh data to Babylon geometry. * @param name The name to use when creating the geometry * @param scene The scene to use when creating the geometry * @param data The ArrayBuffer or ArrayBufferView for the Draco compression data * @param attributes A map of attributes from vertex buffer kinds to Draco unique ids * @returns A promise that resolves with the decoded geometry */ decodeMeshToGeometryAsync(e, t, r, n) { return this._decodeMeshAsync(r, n).then((i) => { const s = new Tf(e, t); i.indices && s.setIndices(i.indices); for (const a of i.attributes) s.setVerticesBuffer(new J(t.getEngine(), a.data, a.kind, !1, void 0, a.byteStride, void 0, a.byteOffset, a.size, void 0, a.normalized, !0), i.totalVertices); return s; }); } /** @internal */ _decodeMeshToGeometryForGltfAsync(e, t, r, n, i) { return this._decodeMeshAsync(r, n, i).then((s) => { const a = new Tf(e, t); s.indices && a.setIndices(s.indices); for (const f of s.attributes) a.setVerticesBuffer(new J(t.getEngine(), f.data, f.kind, !1, void 0, f.byteStride, void 0, f.byteOffset, f.size, void 0, f.normalized, !0), s.totalVertices); return a; }); } /** * Decode Draco compressed mesh data to Babylon vertex data. * @param data The ArrayBuffer or ArrayBufferView for the Draco compression data * @param attributes A map of attributes from vertex buffer kinds to Draco unique ids * @returns A promise that resolves with the decoded vertex data * @deprecated Use {@link decodeMeshToGeometryAsync} for better performance in some cases */ decodeMeshAsync(e, t) { return this._decodeMeshAsync(e, t).then((r) => { const n = new Ut(); r.indices && (n.indices = r.indices); for (const i of r.attributes) { const s = J.GetFloatData(i.data, i.size, J.GetDataType(i.data), i.byteOffset, i.byteStride, i.normalized, r.totalVertices); n.set(s, i.kind); } return n; }); } } Yd.Configuration = { decoder: { wasmUrl: `${ye._DefaultCdnUrl}/draco_wasm_wrapper_gltf.js`, wasmBinaryUrl: `${ye._DefaultCdnUrl}/draco_decoder_gltf.wasm`, fallbackUrl: `${ye._DefaultCdnUrl}/draco_decoder_gltf.js` } }; Yd.DefaultNumWorkers = Yd.GetDefaultNumWorkers(); Yd._Default = null; class Q2 { /** * Default instance for the meshoptimizer object. */ static get Default() { return Q2._Default || (Q2._Default = new Q2()), Q2._Default; } /** * Constructor */ constructor() { const e = Q2.Configuration.decoder; this._decoderModulePromise = ye.LoadBabylonScriptAsync(e.url).then(() => MeshoptDecoder.ready); } /** * Stop all async operations and release resources. */ dispose() { delete this._decoderModulePromise; } /** * Decode meshopt data. * @see https://github.com/zeux/meshoptimizer/tree/master/js#decoder * @param source The input data. * @param count The number of elements. * @param stride The stride in bytes. * @param mode The compression mode. * @param filter The compression filter. * @returns a Promise that resolves to the decoded data */ decodeGltfBufferAsync(e, t, r, n, i) { return this._decoderModulePromise.then(() => { const s = new Uint8Array(t * r); return MeshoptDecoder.decodeGltfBuffer(s, t, r, e, n, i), s; }); } } Q2.Configuration = { decoder: { url: `${ye._DefaultCdnUrl}/meshopt_decoder.js` } }; Q2._Default = null; let TC = 0; class tR { /** * Initializes the vertex * @param pos The position of the vertex * @param normal The normal of the vertex * @param uv The texture coordinate of the vertex * @param vertColor The RGBA color of the vertex */ constructor(e, t, r, n) { this.pos = e, this.normal = t, this.uv = r, this.vertColor = n; } /** * Make a clone, or deep copy, of the vertex * @returns A new Vertex */ clone() { var e, t; return new tR(this.pos.clone(), this.normal.clone(), (e = this.uv) === null || e === void 0 ? void 0 : e.clone(), (t = this.vertColor) === null || t === void 0 ? void 0 : t.clone()); } /** * Invert all orientation-specific data (e.g. vertex normal). Called when the * orientation of a polygon is flipped. */ flip() { this.normal = this.normal.scale(-1); } /** * Create a new vertex between this vertex and `other` by linearly * interpolating all properties using a parameter of `t`. Subclasses should * override this to interpolate additional properties. * @param other the vertex to interpolate against * @param t The factor used to linearly interpolate between the vertices */ interpolate(e, t) { return new tR(S.Lerp(this.pos, e.pos, t), S.Lerp(this.normal, e.normal, t), this.uv && e.uv ? at.Lerp(this.uv, e.uv, t) : void 0, this.vertColor && e.vertColor ? xt.Lerp(this.vertColor, e.vertColor, t) : void 0); } } class Zw { /** * Initializes the plane * @param normal The normal for the plane * @param w */ constructor(e, t) { this.normal = e, this.w = t; } /** * Construct a plane from three points * @param a Point a * @param b Point b * @param c Point c */ static FromPoints(e, t, r) { const n = r.subtract(e), i = t.subtract(e); if (n.lengthSquared() === 0 || i.lengthSquared() === 0) return null; const s = S.Normalize(S.Cross(n, i)); return new Zw(s, S.Dot(s, e)); } /** * Clone, or make a deep copy of the plane * @returns a new Plane */ clone() { return new Zw(this.normal.clone(), this.w); } /** * Flip the face of the plane */ flip() { this.normal.scaleInPlace(-1), this.w = -this.w; } /** * Split `polygon` by this plane if needed, then put the polygon or polygon * fragments in the appropriate lists. Coplanar polygons go into either `* coplanarFront` or `coplanarBack` depending on their orientation with * respect to this plane. Polygons in front or in back of this plane go into * either `front` or `back` * @param polygon The polygon to be split * @param coplanarFront Will contain polygons coplanar with the plane that are oriented to the front of the plane * @param coplanarBack Will contain polygons coplanar with the plane that are oriented to the back of the plane * @param front Will contain the polygons in front of the plane * @param back Will contain the polygons begind the plane */ splitPolygon(e, t, r, n, i) { let d = 0; const v = []; let u, l; for (u = 0; u < e.vertices.length; u++) { l = S.Dot(this.normal, e.vertices[u].pos) - this.w; const P = l < -Zw.EPSILON ? 2 : l > Zw.EPSILON ? 1 : 0; d |= P, v.push(P); } switch (d) { case 0: (S.Dot(this.normal, e.plane.normal) > 0 ? t : r).push(e); break; case 1: n.push(e); break; case 2: i.push(e); break; case 3: { const P = [], p = []; for (u = 0; u < e.vertices.length; u++) { const H = (u + 1) % e.vertices.length, T = v[u], q = v[H], b = e.vertices[u], j = e.vertices[H]; if (T !== 2 && P.push(b), T !== 1 && p.push(T !== 2 ? b.clone() : b), (T | q) === 3) { l = (this.w - S.Dot(this.normal, b.pos)) / S.Dot(this.normal, j.pos.subtract(b.pos)); const w = b.interpolate(j, l); P.push(w), p.push(w.clone()); } } let c; P.length >= 3 && (c = new yS(P, e.shared), c.plane && n.push(c)), p.length >= 3 && (c = new yS(p, e.shared), c.plane && i.push(c)); break; } } } } Zw.EPSILON = 1e-5; class yS { /** * Initializes the polygon * @param vertices The vertices of the polygon * @param shared The properties shared across all polygons */ constructor(e, t) { this.vertices = e, this.shared = t, this.plane = Zw.FromPoints(e[0].pos, e[1].pos, e[2].pos); } /** * Clones, or makes a deep copy, or the polygon */ clone() { const e = this.vertices.map((t) => t.clone()); return new yS(e, this.shared); } /** * Flips the faces of the polygon */ flip() { this.vertices.reverse().map((e) => { e.flip(); }), this.plane.flip(); } } let p0 = class BC { /** * Initializes the node * @param polygons A collection of polygons held in the node */ constructor(e) { this._plane = null, this._front = null, this._back = null, this._polygons = new Array(), e && this.build(e); } /** * Clones, or makes a deep copy, of the node * @returns The cloned node */ clone() { const e = new BC(); return e._plane = this._plane && this._plane.clone(), e._front = this._front && this._front.clone(), e._back = this._back && this._back.clone(), e._polygons = this._polygons.map((t) => t.clone()), e; } /** * Convert solid space to empty space and empty space to solid space */ invert() { for (let t = 0; t < this._polygons.length; t++) this._polygons[t].flip(); this._plane && this._plane.flip(), this._front && this._front.invert(), this._back && this._back.invert(); const e = this._front; this._front = this._back, this._back = e; } /** * Recursively remove all polygons in `polygons` that are inside this BSP * tree. * @param polygons Polygons to remove from the BSP * @returns Polygons clipped from the BSP */ clipPolygons(e) { if (!this._plane) return e.slice(); let t = [], r = []; for (let n = 0; n < e.length; n++) this._plane.splitPolygon(e[n], t, r, t, r); return this._front && (t = this._front.clipPolygons(t)), this._back ? r = this._back.clipPolygons(r) : r = [], t.concat(r); } /** * Remove all polygons in this BSP tree that are inside the other BSP tree * `bsp`. * @param bsp BSP containing polygons to remove from this BSP */ clipTo(e) { this._polygons = e.clipPolygons(this._polygons), this._front && this._front.clipTo(e), this._back && this._back.clipTo(e); } /** * Return a list of all polygons in this BSP tree * @returns List of all polygons in this BSP tree */ allPolygons() { let e = this._polygons.slice(); return this._front && (e = e.concat(this._front.allPolygons())), this._back && (e = e.concat(this._back.allPolygons())), e; } /** * Build a BSP tree out of `polygons`. When called on an existing tree, the * new polygons are filtered down to the bottom of the tree and become new * nodes there. Each set of polygons is partitioned using the first polygon * (no heuristic is used to pick a good split) * @param polygons Polygons used to construct the BSP tree */ build(e) { if (!e.length) return; this._plane || (this._plane = e[0].plane.clone()); const t = [], r = []; for (let n = 0; n < e.length; n++) this._plane.splitPolygon(e[n], this._polygons, this._polygons, t, r); t.length && (this._front || (this._front = new BC()), this._front.build(t)), r.length && (this._back || (this._back = new BC()), this._back.build(r)); } }; class YH { constructor() { this._polygons = new Array(); } /** * Convert a VertexData to CSG * @param mesh defines the VertexData to convert to CSG * @returns the new CSG */ static FromVertexData(e) { let t, r, n; const i = [], s = e.indices, a = e.positions, f = e.normals, o = e.uvs, d = e.colors; if (!s || !a) throw "BABYLON.CSG: VertexData must at least contain positions and indices"; for (let u = 0; u < s.length; u += 3) { n = []; for (let l = 0; l < 3; l++) { const P = u + l, p = s[P], c = f ? S.FromArray(f, p * 3) : S.Zero(), H = o ? at.FromArray(o, p * 2) : void 0, T = d ? xt.FromArray(d, p * 4) : void 0, q = S.FromArray(a, p * 3); t = new tR(q, c, H, T), n.push(t); } r = new yS(n, { subMeshId: 0, meshId: TC, materialIndex: 0 }), r.plane && i.push(r); } const v = YH._FromPolygons(i); return v.matrix = he.Identity(), v.position = S.Zero(), v.rotation = S.Zero(), v.scaling = S.One(), v.rotationQuaternion = Ze.Identity(), TC++, v; } /** * Convert the Mesh to CSG * @param mesh The Mesh to convert to CSG * @param absolute If true, the final (local) matrix transformation is set to the identity and not to that of `mesh`. It can help when dealing with right-handed meshes (default: false) * @returns A new CSG from the Mesh */ static FromMesh(e, t = !1) { let r, n, i, s, a, f, o; const d = []; let v, u, l, P = null, p, c = !1; if (e instanceof Ee) e.computeWorldMatrix(!0), v = e.getWorldMatrix(), u = e.position.clone(), l = e.rotation.clone(), e.rotationQuaternion && (P = e.rotationQuaternion.clone()), p = e.scaling.clone(), e.material && t && (c = e.material.sideOrientation === 0); else throw "BABYLON.CSG: Wrong Mesh type, must be BABYLON.Mesh"; const H = e.getIndices(), T = e.getVerticesData(J.PositionKind), q = e.getVerticesData(J.NormalKind), b = e.getVerticesData(J.UVKind), j = e.getVerticesData(J.ColorKind), w = e.subMeshes; for (let I = 0, N = w.length; I < N; I++) for (let k = w[I].indexStart, R = w[I].indexCount + w[I].indexStart; k < R; k += 3) { o = []; for (let y = 0; y < 3; y++) { const O = y === 0 ? k + y : c ? k + 3 - y : k + y, Y = new S(q[H[O] * 3], q[H[O] * 3 + 1], q[H[O] * 3 + 2]); b && (i = new at(b[H[O] * 2], b[H[O] * 2 + 1])), j && (a = new xt(j[H[O] * 4], j[H[O] * 4 + 1], j[H[O] * 4 + 2], j[H[O] * 4 + 3])); const ee = new S(T[H[O] * 3], T[H[O] * 3 + 1], T[H[O] * 3 + 2]); s = S.TransformCoordinates(ee, v), n = S.TransformNormal(Y, v), r = new tR(s, n, i, a), o.push(r); } f = new yS(o, { subMeshId: I, meshId: TC, materialIndex: w[I].materialIndex }), f.plane && d.push(f); } const m = YH._FromPolygons(d); return m.matrix = t ? he.Identity() : v, m.position = t ? S.Zero() : u, m.rotation = t ? S.Zero() : l, m.scaling = t ? S.One() : p, m.rotationQuaternion = t && P ? Ze.Identity() : P, TC++, m; } /** * Construct a CSG solid from a list of `CSG.Polygon` instances. * @param polygons Polygons used to construct a CSG solid */ static _FromPolygons(e) { const t = new YH(); return t._polygons = e, t; } /** * Clones, or makes a deep copy, of the CSG * @returns A new CSG */ clone() { const e = new YH(); return e._polygons = this._polygons.map((t) => t.clone()), e.copyTransformAttributes(this), e; } /** * Unions this CSG with another CSG * @param csg The CSG to union against this CSG * @returns The unioned CSG */ union(e) { const t = new p0(this.clone()._polygons), r = new p0(e.clone()._polygons); return t.clipTo(r), r.clipTo(t), r.invert(), r.clipTo(t), r.invert(), t.build(r.allPolygons()), YH._FromPolygons(t.allPolygons()).copyTransformAttributes(this); } /** * Unions this CSG with another CSG in place * @param csg The CSG to union against this CSG */ unionInPlace(e) { const t = new p0(this._polygons), r = new p0(e._polygons); t.clipTo(r), r.clipTo(t), r.invert(), r.clipTo(t), r.invert(), t.build(r.allPolygons()), this._polygons = t.allPolygons(); } /** * Subtracts this CSG with another CSG * @param csg The CSG to subtract against this CSG * @returns A new CSG */ subtract(e) { const t = new p0(this.clone()._polygons), r = new p0(e.clone()._polygons); return t.invert(), t.clipTo(r), r.clipTo(t), r.invert(), r.clipTo(t), r.invert(), t.build(r.allPolygons()), t.invert(), YH._FromPolygons(t.allPolygons()).copyTransformAttributes(this); } /** * Subtracts this CSG with another CSG in place * @param csg The CSG to subtract against this CSG */ subtractInPlace(e) { const t = new p0(this._polygons), r = new p0(e._polygons); t.invert(), t.clipTo(r), r.clipTo(t), r.invert(), r.clipTo(t), r.invert(), t.build(r.allPolygons()), t.invert(), this._polygons = t.allPolygons(); } /** * Intersect this CSG with another CSG * @param csg The CSG to intersect against this CSG * @returns A new CSG */ intersect(e) { const t = new p0(this.clone()._polygons), r = new p0(e.clone()._polygons); return t.invert(), r.clipTo(t), r.invert(), t.clipTo(r), r.clipTo(t), t.build(r.allPolygons()), t.invert(), YH._FromPolygons(t.allPolygons()).copyTransformAttributes(this); } /** * Intersects this CSG with another CSG in place * @param csg The CSG to intersect against this CSG */ intersectInPlace(e) { const t = new p0(this._polygons), r = new p0(e._polygons); t.invert(), r.clipTo(t), r.invert(), t.clipTo(r), r.clipTo(t), t.build(r.allPolygons()), t.invert(), this._polygons = t.allPolygons(); } /** * Return a new CSG solid with solid and empty space switched. This solid is * not modified. * @returns A new CSG solid with solid and empty space switched */ inverse() { const e = this.clone(); return e.inverseInPlace(), e; } /** * Inverses the CSG in place */ inverseInPlace() { this._polygons.map((e) => { e.flip(); }); } /** * This is used to keep meshes transformations so they can be restored * when we build back a Babylon Mesh * NB : All CSG operations are performed in world coordinates * @param csg The CSG to copy the transform attributes from * @returns This CSG */ copyTransformAttributes(e) { return this.matrix = e.matrix, this.position = e.position, this.rotation = e.rotation, this.scaling = e.scaling, this.rotationQuaternion = e.rotationQuaternion, this; } /** * Build vertex data from CSG * Coordinates here are in world space * @returns the final vertex data */ toVertexData(e = null, t = null) { const r = this.matrix.clone(); r.invert(); const n = this._polygons, i = [], s = [], a = []; let f = null, o = null; const d = S.Zero(), v = S.Zero(), u = at.Zero(), l = new xt(0, 0, 0, 0), P = [0, 0, 0], p = {}; let c; for (let T = 0, q = n.length; T < q; T++) { const b = n[T]; e && e(b); for (let j = 2, w = b.vertices.length; j < w; j++) { P[0] = 0, P[1] = j - 1, P[2] = j; for (let m = 0; m < 3; m++) { d.copyFrom(b.vertices[P[m]].pos), v.copyFrom(b.vertices[P[m]].normal), b.vertices[P[m]].uv && (f || (f = []), u.copyFrom(b.vertices[P[m]].uv)), b.vertices[P[m]].vertColor && (o || (o = []), l.copyFrom(b.vertices[P[m]].vertColor)); const I = S.TransformCoordinates(d, r), N = S.TransformNormal(v, r); c = p[I.x + "," + I.y + "," + I.z]; let k = !1; f && !(f[c * 2] === u.x || f[c * 2 + 1] === u.y) && (k = !0); let R = !1; o && !(o[c * 4] === l.r || o[c * 4 + 1] === l.g || o[c * 4 + 2] === l.b || o[c * 4 + 3] === l.a) && (R = !0), (!(typeof c < "u" && a[c * 3] === N.x && a[c * 3 + 1] === N.y && a[c * 3 + 2] === N.z) || k || R) && (i.push(I.x, I.y, I.z), f && f.push(u.x, u.y), a.push(v.x, v.y, v.z), o && o.push(l.r, l.g, l.b, l.a), c = p[I.x + "," + I.y + "," + I.z] = i.length / 3 - 1), s.push(c), t && t(); } } } const H = new Ut(); return H.positions = i, H.normals = a, f && (H.uvs = f), o && (H.colors = o), H.indices = s, H; } /** * Build Raw mesh from CSG * Coordinates here are in world space * @param name The name of the mesh geometry * @param scene The Scene * @param keepSubMeshes Specifies if the submeshes should be kept * @returns A new Mesh */ buildMeshGeometry(e, t, r) { const n = new Ee(e, t), i = this._polygons; let s = 0; const a = {}; let f; if (r && i.sort((d, v) => d.shared.meshId === v.shared.meshId ? d.shared.subMeshId - v.shared.subMeshId : d.shared.meshId - v.shared.meshId), this.toVertexData((d) => { a[d.shared.meshId] || (a[d.shared.meshId] = {}), a[d.shared.meshId][d.shared.subMeshId] || (a[d.shared.meshId][d.shared.subMeshId] = { indexStart: 1 / 0, indexEnd: -1 / 0, materialIndex: d.shared.materialIndex }), f = a[d.shared.meshId][d.shared.subMeshId]; }, () => { f.indexStart = Math.min(s, f.indexStart), f.indexEnd = Math.max(s, f.indexEnd), s++; }).applyToMesh(n), r) { let d = 0, v; n.subMeshes = []; for (const u in a) { v = -1; for (const l in a[u]) f = a[u][l], rA.CreateFromIndices(f.materialIndex + d, f.indexStart, f.indexEnd - f.indexStart + 1, n), v = Math.max(f.materialIndex, v); d += ++v; } } return n; } /** * Build Mesh from CSG taking material and transforms into account * @param name The name of the Mesh * @param material The material of the Mesh * @param scene The Scene * @param keepSubMeshes Specifies if submeshes should be kept * @returns The new Mesh */ toMesh(e, t = null, r, n) { const i = this.buildMeshGeometry(e, r, n); return i.material = t, i.position.copyFrom(this.position), i.rotation.copyFrom(this.rotation), this.rotationQuaternion && (i.rotationQuaternion = this.rotationQuaternion.clone()), i.scaling.copyFrom(this.scaling), i.computeWorldMatrix(!0), i; } } const Xhe = "meshUVSpaceRendererVertexShader", The = `precision highp float;attribute vec3 position;attribute vec3 normal;attribute vec2 uv;uniform mat4 projMatrix;varying vec2 vDecalTC; #include #include #include #include[0..maxSimultaneousMorphTargets] #include void main(void) {vec3 positionUpdated=position;vec3 normalUpdated=normal; #include #include[0..maxSimultaneousMorphTargets] #include #include #include vec4 worldPos=finalWorld*vec4(positionUpdated,1.0);mat3 normWorldSM=mat3(finalWorld);vec3 vNormalW; #if defined(INSTANCES) && defined(THIN_INSTANCES) vNormalW=normalUpdated/vec3(dot(normWorldSM[0],normWorldSM[0]),dot(normWorldSM[1],normWorldSM[1]),dot(normWorldSM[2],normWorldSM[2]));vNormalW=normalize(normWorldSM*vNormalW); #else #ifdef NONUNIFORMSCALING normWorldSM=transposeMat3(inverseMat3(normWorldSM)); #endif vNormalW=normalize(normWorldSM*normalUpdated); #endif vec3 normalView=normalize((projMatrix*vec4(vNormalW,0.0)).xyz);vec3 decalTC=(projMatrix*worldPos).xyz;vDecalTC=decalTC.xy;gl_Position=vec4(uv*2.0-1.0,normalView.z>0.0 ? 2. : decalTC.z,1.0);} `; Le.ShadersStore[Xhe] = The; const qhe = "meshUVSpaceRendererPixelShader", bhe = `precision highp float;varying vec2 vDecalTC;uniform sampler2D textureSampler;void main(void) {if (vDecalTC.x<0. || vDecalTC.x>1. || vDecalTC.y<0. || vDecalTC.y>1.) {discard;} gl_FragColor=texture2D(textureSampler,vDecalTC);} `; Le.ShadersStore[qhe] = bhe; class Qw { static _GetShader(e) { if (!e._meshUVSpaceRendererShader) { const t = new Zo("meshUVSpaceRendererShader", e, { vertex: "meshUVSpaceRenderer", fragment: "meshUVSpaceRenderer" }, { attributes: ["position", "normal", "uv"], uniforms: ["world", "projMatrix"], samplers: ["textureSampler"], needAlphaBlending: !0 }); t.backFaceCulling = !1, t.alphaMode = 2, e.onDisposeObservable.add(() => { var r; (r = e._meshUVSpaceRendererShader) === null || r === void 0 || r.dispose(), e._meshUVSpaceRendererShader = null; }), e._meshUVSpaceRendererShader = t; } return e._meshUVSpaceRendererShader; } static _IsRenderTargetTexture(e) { return e.renderList !== void 0; } /** * Creates a new MeshUVSpaceRenderer * @param mesh The mesh used for the source UV space * @param scene The scene the mesh belongs to * @param options The options to use when creating the texture */ constructor(e, t, r) { this._textureCreatedInternally = !1, this.clearColor = new xt(0, 0, 0, 0), this._mesh = e, this._scene = t, this._options = Object.assign({ width: 1024, height: 1024, textureType: 0, generateMipMaps: !0, optimizeUVAllocation: !0 }, r); } /** * Checks if the texture is ready to be used * @returns true if the texture is ready to be used */ isReady() { return this.texture || this._createDiffuseRTT(), Qw._IsRenderTargetTexture(this.texture) ? this.texture.isReadyForRendering() : this.texture.isReady(); } /** * Projects and renders a texture in the mesh UV space * @param texture The texture * @param position The position of the center of projection (world space coordinates) * @param normal The direction of the projection (world space coordinates) * @param size The size of the projection * @param angle The rotation angle around the direction of the projection */ renderTexture(e, t, r, n, i = 0) { if (this.texture || this._createDiffuseRTT(), Qw._IsRenderTargetTexture(this.texture)) { const s = this._createProjectionMatrix(t, r, n, i), a = Qw._GetShader(this._scene); a.setTexture("textureSampler", e), a.setMatrix("projMatrix", s), this.texture.render(); } } /** * Clears the texture map */ clear() { if (Qw._IsRenderTargetTexture(this.texture) && this.texture.renderTarget) { const e = this._scene.getEngine(); e.bindFramebuffer(this.texture.renderTarget), e.clear(this.clearColor, !0, !0, !0), e.unBindFramebuffer(this.texture.renderTarget); } } /** * Disposes of the ressources */ dispose() { this._textureCreatedInternally && (this.texture.dispose(), this._textureCreatedInternally = !1); } _createDiffuseRTT() { this._textureCreatedInternally = !0; const e = this._createRenderTargetTexture(this._options.width, this._options.height); e.setMaterialForRendering(this._mesh, Qw._GetShader(this._scene)), this.texture = e; } _createRenderTargetTexture(e, t) { const r = new Ta(this._mesh.name + "_uvspaceTexture", { width: e, height: t }, this._scene, this._options.generateMipMaps, !0, this._options.textureType, !1, this._options.generateMipMaps ? 3 : 2, !1, !1, !1, 5); return r.renderParticles = !1, r.optimizeUVAllocation = !!this._options.optimizeUVAllocation, r.onClearObservable.addOnce(() => { this._scene.getEngine().clear(this.clearColor, !0, !0, !0), r.onClearObservable.add(() => { }); }), r.renderList = [this._mesh], r; } _createProjectionMatrix(e, t, r, n = 0) { const i = -Math.atan2(t.z, t.x) - Math.PI / 2, s = Math.sqrt(t.x * t.x + t.z * t.z), a = Math.atan2(t.y, s), f = e.add(t.scale(r.z * 0.5)), o = he.RotationYawPitchRoll(i, a, n).multiply(he.Translation(f.x, f.y, f.z)), d = he.Invert(o), v = he.FromArray([2 / r.x, 0, 0, 0, 0, 2 / r.y, 0, 0, 0, 0, 1 / r.z, 0, 0, 0, 0, 1]), u = he.FromArray([0.5, 0, 0, 0, 0, 0.5, 0, 0, 0, 0, 1, 0, 0.5, 0.5, 0, 1]); return d.multiply(v).multiply(u); } } Ee._TrailMeshParser = (A, e) => rR.Parse(A, e); class rR extends Ee { /** * Creates a new TrailMesh. * @param name The value used by scene.getMeshByName() to do a lookup. * @param generator The mesh or transform node to generate a trail. * @param scene The scene to add this mesh to. * @param diameter Diameter of trailing mesh. Default is 1. * @param length Length of trailing mesh. Default is 60. * @param autoStart Automatically start trailing mesh. Default true. */ constructor(e, t, r, n = 1, i = 60, s = !0) { super(e, r), this._sectionPolygonPointsCount = 4, this._running = !1, this._autoStart = s, this._generator = t, this.diameter = n, this._length = i, this._sectionVectors = [], this._sectionNormalVectors = []; for (let a = 0; a < this._sectionPolygonPointsCount; a++) this._sectionVectors[a] = S.Zero(), this._sectionNormalVectors[a] = S.Zero(); this._createMesh(); } /** * "TrailMesh" * @returns "TrailMesh" */ getClassName() { return "TrailMesh"; } _createMesh() { const e = new Ut(), t = [], r = [], n = []; let i = S.Zero(); this._generator instanceof jn && this._generator.hasBoundingInfo ? i = this._generator.getBoundingInfo().boundingBox.centerWorld : i = this._generator.position; const s = 2 * Math.PI / this._sectionPolygonPointsCount; for (let a = 0; a < this._sectionPolygonPointsCount; a++) t.push(i.x + Math.cos(a * s) * this.diameter, i.y + Math.sin(a * s) * this.diameter, i.z); for (let a = 1; a <= this._length; a++) { for (let o = 0; o < this._sectionPolygonPointsCount; o++) t.push(i.x + Math.cos(o * s) * this.diameter, i.y + Math.sin(o * s) * this.diameter, i.z); const f = t.length / 3 - 2 * this._sectionPolygonPointsCount; for (let o = 0; o < this._sectionPolygonPointsCount - 1; o++) n.push(f + o, f + o + this._sectionPolygonPointsCount, f + o + this._sectionPolygonPointsCount + 1), n.push(f + o, f + o + this._sectionPolygonPointsCount + 1, f + o + 1); n.push(f + this._sectionPolygonPointsCount - 1, f + this._sectionPolygonPointsCount - 1 + this._sectionPolygonPointsCount, f + this._sectionPolygonPointsCount), n.push(f + this._sectionPolygonPointsCount - 1, f + this._sectionPolygonPointsCount, f); } Ut.ComputeNormals(t, n, r), e.positions = t, e.normals = r, e.indices = n, e.applyToMesh(this, !0), this._autoStart && this.start(); } /** * Start trailing mesh. */ start() { this._running || (this._running = !0, this._beforeRenderObserver = this.getScene().onBeforeRenderObservable.add(() => { this.update(); })); } /** * Stop trailing mesh. */ stop() { this._beforeRenderObserver && this._running && (this._running = !1, this.getScene().onBeforeRenderObservable.remove(this._beforeRenderObserver)); } /** * Update trailing mesh geometry. */ update() { const e = this.getVerticesData(J.PositionKind), t = this.getVerticesData(J.NormalKind), r = this._generator.getWorldMatrix(); if (e && t) { for (let s = 3 * this._sectionPolygonPointsCount; s < e.length; s++) e[s - 3 * this._sectionPolygonPointsCount] = e[s] - t[s] / this._length * this.diameter; for (let s = 3 * this._sectionPolygonPointsCount; s < t.length; s++) t[s - 3 * this._sectionPolygonPointsCount] = t[s]; const n = e.length - 3 * this._sectionPolygonPointsCount, i = 2 * Math.PI / this._sectionPolygonPointsCount; for (let s = 0; s < this._sectionPolygonPointsCount; s++) this._sectionVectors[s].copyFromFloats(Math.cos(s * i) * this.diameter, Math.sin(s * i) * this.diameter, 0), this._sectionNormalVectors[s].copyFromFloats(Math.cos(s * i), Math.sin(s * i), 0), S.TransformCoordinatesToRef(this._sectionVectors[s], r, this._sectionVectors[s]), S.TransformNormalToRef(this._sectionNormalVectors[s], r, this._sectionNormalVectors[s]); for (let s = 0; s < this._sectionPolygonPointsCount; s++) e[n + 3 * s] = this._sectionVectors[s].x, e[n + 3 * s + 1] = this._sectionVectors[s].y, e[n + 3 * s + 2] = this._sectionVectors[s].z, t[n + 3 * s] = this._sectionNormalVectors[s].x, t[n + 3 * s + 1] = this._sectionNormalVectors[s].y, t[n + 3 * s + 2] = this._sectionNormalVectors[s].z; this.updateVerticesData(J.PositionKind, e, !0, !1), this.updateVerticesData(J.NormalKind, t, !0, !1); } } /** * Returns a new TrailMesh object. * @param name is a string, the name given to the new mesh * @param newGenerator use new generator object for cloned trail mesh * @returns a new mesh */ clone(e = "", t) { return new rR(e, t === void 0 ? this._generator : t, this.getScene(), this.diameter, this._length, this._autoStart); } /** * Serializes this trail mesh * @param serializationObject object to write serialization to */ serialize(e) { super.serialize(e), e.generatorId = this._generator.id; } /** * Parses a serialized trail mesh * @param parsedMesh the serialized mesh * @param scene the scene to create the trail mesh in * @returns the created trail mesh */ static Parse(e, t) { var r, n; const i = (r = t.getLastMeshById(e.generatorId)) !== null && r !== void 0 ? r : t.getLastTransformNodeById(e.generatorId); if (!i) throw new Error("TrailMesh: generator not found with ID " + e.generatorId); return new rR(e.name, i, t, (n = e.diameter) !== null && n !== void 0 ? n : e._diameter, e._length, e._autoStart); } } class xhe { /** * Creates a SimplificationSettings * @param quality expected quality * @param distance distance when this optimized version should be used * @param optimizeMesh already optimized mesh */ constructor(e, t, r) { this.quality = e, this.distance = t, this.optimizeMesh = r; } } class Pne { /** * Creates a new queue */ constructor() { this.running = !1, this._simplificationArray = []; } /** * Adds a new simplification task * @param task defines a task to add */ addTask(e) { this._simplificationArray.push(e); } /** * Execute next task */ executeNext() { const e = this._simplificationArray.pop(); e ? (this.running = !0, this.runSimplification(e)) : this.running = !1; } /** * Execute a simplification task * @param task defines the task to run */ runSimplification(e) { if (e.parallelProcessing) e.settings.forEach((t) => { this._getSimplifier(e).simplify(t, (n) => { t.distance !== void 0 && e.mesh.addLODLevel(t.distance, n), n.isVisible = !0, t.quality === e.settings[e.settings.length - 1].quality && e.successCallback && e.successCallback(), this.executeNext(); }); }); else { const t = this._getSimplifier(e), r = (n, i) => { t.simplify(n, (s) => { n.distance !== void 0 && e.mesh.addLODLevel(n.distance, s), s.isVisible = !0, i(); }); }; up.Run(e.settings.length, (n) => { r(e.settings[n.index], () => { n.executeNext(); }); }, () => { e.successCallback && e.successCallback(), this.executeNext(); }); } } _getSimplifier(e) { switch (e.simplificationType) { case nR.QUADRATIC: default: return new cne(e.mesh); } } } var nR; (function(A) { A[A.QUADRATIC = 0] = "QUADRATIC"; })(nR || (nR = {})); class Dhe { constructor(e) { this._vertices = e, this.error = new Array(4), this.deleted = !1, this.isDirty = !1, this.deletePending = !1, this.borderFactor = 0; } } class jhe { constructor(e, t) { this.position = e, this.id = t, this.isBorder = !0, this.q = new iS(), this.triangleCount = 0, this.triangleStart = 0, this.originalOffsets = []; } updatePosition(e) { this.position.copyFrom(e); } } class iS { constructor(e) { this.data = new Array(10); for (let t = 0; t < 10; ++t) e && e[t] ? this.data[t] = e[t] : this.data[t] = 0; } det(e, t, r, n, i, s, a, f, o) { return this.data[e] * this.data[i] * this.data[o] + this.data[r] * this.data[n] * this.data[f] + this.data[t] * this.data[s] * this.data[a] - this.data[r] * this.data[i] * this.data[a] - this.data[e] * this.data[s] * this.data[f] - this.data[t] * this.data[n] * this.data[o]; } addInPlace(e) { for (let t = 0; t < 10; ++t) this.data[t] += e.data[t]; } addArrayInPlace(e) { for (let t = 0; t < 10; ++t) this.data[t] += e[t]; } add(e) { const t = new iS(); for (let r = 0; r < 10; ++r) t.data[r] = this.data[r] + e.data[r]; return t; } static FromData(e, t, r, n) { return new iS(iS.DataFromNumbers(e, t, r, n)); } //returning an array to avoid garbage collection static DataFromNumbers(e, t, r, n) { return [e * e, e * t, e * r, e * n, t * t, t * r, t * n, r * r, r * n, n * n]; } } class whe { constructor(e, t) { this.vertexId = e, this.triangleId = t; } } class cne { /** * Creates a new QuadraticErrorSimplification * @param _mesh defines the target mesh */ constructor(e) { this._mesh = e, this.syncIterations = 5e3, this.aggressiveness = 7, this.decimationIterations = 100, this.boundingBoxEpsilon = Dn; } /** * Simplification of a given mesh according to the given settings. * Since this requires computation, it is assumed that the function runs async. * @param settings The settings of the simplification, including quality and distance * @param successCallback A callback that will be called after the mesh was simplified. */ simplify(e, t) { this._initDecimatedMesh(), up.Run(this._mesh.subMeshes.length, (r) => { this._initWithMesh(r.index, () => { this._runDecimation(e, r.index, () => { r.executeNext(); }); }, e.optimizeMesh); }, () => { setTimeout(() => { t(this._reconstructedMesh); }, 0); }); } _runDecimation(e, t, r) { const n = ~~(this._triangles.length * e.quality); let i = 0; const s = this._triangles.length, a = (f, o) => { setTimeout(() => { f % 5 === 0 && this._updateMesh(f === 0); for (let u = 0; u < this._triangles.length; ++u) this._triangles[u].isDirty = !1; const d = 1e-9 * Math.pow(f + 3, this.aggressiveness), v = (u) => { const l = ~~((this._triangles.length / 2 + u) % this._triangles.length), P = this._triangles[l]; if (P && !(P.error[3] > d || P.deleted || P.isDirty)) { for (let p = 0; p < 3; ++p) if (P.error[p] < d) { const c = [], H = [], T = P._vertices[p], q = P._vertices[(p + 1) % 3]; if (T.isBorder || q.isBorder) continue; const b = S.Zero(); this._calculateError(T, q, b); const j = []; if (this._isFlipped(T, q, b, c, j) || this._isFlipped(q, T, b, H, j) || c.indexOf(!0) < 0 || H.indexOf(!0) < 0) continue; const w = []; if (j.forEach((N) => { w.indexOf(N) === -1 && (N.deletePending = !0, w.push(N)); }), w.length % 2 !== 0) continue; T.q = q.q.add(T.q), T.updatePosition(b); const m = this._references.length; i = this._updateTriangles(T, T, c, i), i = this._updateTriangles(T, q, H, i); const I = this._references.length - m; if (I <= T.triangleCount) { if (I) for (let N = 0; N < I; N++) this._references[T.triangleStart + N] = this._references[m + N]; } else T.triangleStart = m; T.triangleCount = I; break; } } }; up.SyncAsyncForLoop(this._triangles.length, this.syncIterations, v, o, () => s - i <= n); }, 0); }; up.Run(this.decimationIterations, (f) => { s - i <= n ? f.breakLoop() : a(f.index, () => { f.executeNext(); }); }, () => { setTimeout(() => { this._reconstructMesh(t), r(); }, 0); }); } _initWithMesh(e, t, r) { this._vertices = [], this._triangles = []; const n = this._mesh.getVerticesData(J.PositionKind), i = this._mesh.getIndices(), s = this._mesh.subMeshes[e], a = (v) => { if (r) { for (let u = 0; u < this._vertices.length; ++u) if (this._vertices[u].position.equalsWithEpsilon(v, 1e-4)) return this._vertices[u]; } return null; }, f = [], o = (v) => { if (!n) return; const u = v + s.verticesStart, l = S.FromArray(n, u * 3), P = a(l) || new jhe(l, this._vertices.length); P.originalOffsets.push(u), P.id === this._vertices.length && this._vertices.push(P), f.push(P.id); }, d = s.verticesCount; up.SyncAsyncForLoop(d, this.syncIterations / 4 >> 0, o, () => { const v = (u) => { if (!i) return; const P = (s.indexStart / 3 + u) * 3, p = i[P + 0], c = i[P + 1], H = i[P + 2], T = this._vertices[f[p - s.verticesStart]], q = this._vertices[f[c - s.verticesStart]], b = this._vertices[f[H - s.verticesStart]], j = new Dhe([T, q, b]); j.originalOffset = P, this._triangles.push(j); }; up.SyncAsyncForLoop(s.indexCount / 3, this.syncIterations, v, () => { this._init(t); }); }); } _init(e) { const t = (r) => { const n = this._triangles[r]; n.normal = S.Cross(n._vertices[1].position.subtract(n._vertices[0].position), n._vertices[2].position.subtract(n._vertices[0].position)).normalize(); for (let i = 0; i < 3; i++) n._vertices[i].q.addArrayInPlace(iS.DataFromNumbers(n.normal.x, n.normal.y, n.normal.z, -S.Dot(n.normal, n._vertices[0].position))); }; up.SyncAsyncForLoop(this._triangles.length, this.syncIterations, t, () => { const r = (n) => { const i = this._triangles[n]; for (let s = 0; s < 3; ++s) i.error[s] = this._calculateError(i._vertices[s], i._vertices[(s + 1) % 3]); i.error[3] = Math.min(i.error[0], i.error[1], i.error[2]); }; up.SyncAsyncForLoop(this._triangles.length, this.syncIterations, r, () => { e(); }); }); } _reconstructMesh(e) { const t = []; let r; for (r = 0; r < this._vertices.length; ++r) this._vertices[r].triangleCount = 0; let n, i; for (r = 0; r < this._triangles.length; ++r) if (!this._triangles[r].deleted) { for (n = this._triangles[r], i = 0; i < 3; ++i) n._vertices[i].triangleCount = 1; t.push(n); } const s = this._reconstructedMesh.getVerticesData(J.PositionKind) || [], a = this._reconstructedMesh.getVerticesData(J.NormalKind) || [], f = this._reconstructedMesh.getVerticesData(J.UVKind) || [], o = this._reconstructedMesh.getVerticesData(J.ColorKind) || [], d = this._mesh.getVerticesData(J.NormalKind), v = this._mesh.getVerticesData(J.UVKind), u = this._mesh.getVerticesData(J.ColorKind); let l = 0; for (r = 0; r < this._vertices.length; ++r) { const b = this._vertices[r]; b.id = l, b.triangleCount && b.originalOffsets.forEach((j) => { s.push(b.position.x), s.push(b.position.y), s.push(b.position.z), d && d.length && (a.push(d[j * 3]), a.push(d[j * 3 + 1]), a.push(d[j * 3 + 2])), v && v.length && (f.push(v[j * 2]), f.push(v[j * 2 + 1])), u && u.length && (o.push(u[j * 4]), o.push(u[j * 4 + 1]), o.push(u[j * 4 + 2]), o.push(u[j * 4 + 3])), ++l; }); } const P = this._reconstructedMesh.getTotalIndices(), p = this._reconstructedMesh.getTotalVertices(), c = this._reconstructedMesh.subMeshes; this._reconstructedMesh.subMeshes = []; const H = this._reconstructedMesh.getIndices(), T = this._mesh.getIndices(); for (r = 0; r < t.length; ++r) n = t[r], [0, 1, 2].forEach((b) => { const j = T[n.originalOffset + b]; let w = n._vertices[b].originalOffsets.indexOf(j); w < 0 && (w = 0), H.push(n._vertices[b].id + w + p); }); this._reconstructedMesh.setIndices(H), this._reconstructedMesh.setVerticesData(J.PositionKind, s), a.length > 0 && this._reconstructedMesh.setVerticesData(J.NormalKind, a), f.length > 0 && this._reconstructedMesh.setVerticesData(J.UVKind, f), o.length > 0 && this._reconstructedMesh.setVerticesData(J.ColorKind, o); const q = this._mesh.subMeshes[e]; e > 0 && (this._reconstructedMesh.subMeshes = [], c.forEach((b) => { rA.AddToMesh( b.materialIndex, b.verticesStart, b.verticesCount, /* 0, newPositionData.length/3, */ b.indexStart, b.indexCount, b.getMesh() ); }), rA.AddToMesh( q.materialIndex, p, l, /* 0, newPositionData.length / 3, */ P, t.length * 3, this._reconstructedMesh )); } _initDecimatedMesh() { this._reconstructedMesh = new Ee(this._mesh.name + "Decimated", this._mesh.getScene()), this._reconstructedMesh.material = this._mesh.material, this._reconstructedMesh.parent = this._mesh.parent, this._reconstructedMesh.isVisible = !1, this._reconstructedMesh.renderingGroupId = this._mesh.renderingGroupId; } _isFlipped(e, t, r, n, i) { for (let s = 0; s < e.triangleCount; ++s) { const a = this._triangles[this._references[e.triangleStart + s].triangleId]; if (a.deleted) continue; const f = this._references[e.triangleStart + s].vertexId, o = a._vertices[(f + 1) % 3], d = a._vertices[(f + 2) % 3]; if (o === t || d === t) { n[s] = !0, i.push(a); continue; } let v = o.position.subtract(r); v = v.normalize(); let u = d.position.subtract(r); if (u = u.normalize(), Math.abs(S.Dot(v, u)) > 0.999) return !0; const l = S.Cross(v, u).normalize(); if (n[s] = !1, S.Dot(l, a.normal) < 0.2) return !0; } return !1; } _updateTriangles(e, t, r, n) { let i = n; for (let s = 0; s < t.triangleCount; ++s) { const a = this._references[t.triangleStart + s], f = this._triangles[a.triangleId]; if (!f.deleted) { if (r[s] && f.deletePending) { f.deleted = !0, i++; continue; } f._vertices[a.vertexId] = e, f.isDirty = !0, f.error[0] = this._calculateError(f._vertices[0], f._vertices[1]) + f.borderFactor / 2, f.error[1] = this._calculateError(f._vertices[1], f._vertices[2]) + f.borderFactor / 2, f.error[2] = this._calculateError(f._vertices[2], f._vertices[0]) + f.borderFactor / 2, f.error[3] = Math.min(f.error[0], f.error[1], f.error[2]), this._references.push(a); } } return i; } _identifyBorder() { for (let e = 0; e < this._vertices.length; ++e) { const t = [], r = [], n = this._vertices[e]; let i; for (i = 0; i < n.triangleCount; ++i) { const s = this._triangles[this._references[n.triangleStart + i].triangleId]; for (let a = 0; a < 3; a++) { let f = 0; const o = s._vertices[a]; for (; f < t.length && r[f] !== o.id; ) ++f; f === t.length ? (t.push(1), r.push(o.id)) : t[f]++; } } for (i = 0; i < t.length; ++i) t[i] === 1 ? this._vertices[r[i]].isBorder = !0 : this._vertices[r[i]].isBorder = !1; } } _updateMesh(e = !1) { let t; if (!e) { const f = []; for (t = 0; t < this._triangles.length; ++t) this._triangles[t].deleted || f.push(this._triangles[t]); this._triangles = f; } for (t = 0; t < this._vertices.length; ++t) this._vertices[t].triangleCount = 0, this._vertices[t].triangleStart = 0; let r, n, i; for (t = 0; t < this._triangles.length; ++t) for (r = this._triangles[t], n = 0; n < 3; ++n) i = r._vertices[n], i.triangleCount++; let s = 0; for (t = 0; t < this._vertices.length; ++t) this._vertices[t].triangleStart = s, s += this._vertices[t].triangleCount, this._vertices[t].triangleCount = 0; const a = new Array(this._triangles.length * 3); for (t = 0; t < this._triangles.length; ++t) for (r = this._triangles[t], n = 0; n < 3; ++n) i = r._vertices[n], a[i.triangleStart + i.triangleCount] = new whe(n, t), i.triangleCount++; this._references = a, e && this._identifyBorder(); } _vertexError(e, t) { const r = t.x, n = t.y, i = t.z; return e.data[0] * r * r + 2 * e.data[1] * r * n + 2 * e.data[2] * r * i + 2 * e.data[3] * r + e.data[4] * n * n + 2 * e.data[5] * n * i + 2 * e.data[6] * n + e.data[7] * i * i + 2 * e.data[8] * i + e.data[9]; } _calculateError(e, t, r) { const n = e.q.add(t.q), i = e.isBorder && t.isBorder; let s = 0; const a = n.det(0, 1, 2, 1, 4, 5, 2, 5, 7); if (a !== 0 && !i) r || (r = S.Zero()), r.x = -1 / a * n.det(1, 2, 3, 4, 5, 6, 5, 7, 8), r.y = 1 / a * n.det(0, 2, 3, 1, 5, 6, 2, 7, 8), r.z = -1 / a * n.det(0, 1, 3, 1, 4, 6, 2, 5, 8), s = this._vertexError(n, r); else { const f = e.position.add(t.position).divide(new S(2, 2, 2)), o = this._vertexError(n, e.position), d = this._vertexError(n, t.position), v = this._vertexError(n, f); s = Math.min(o, d, v), s === o ? r && r.copyFrom(e.position) : s === d ? r && r.copyFrom(t.position) : r && r.copyFrom(f); } return s; } } Object.defineProperty(sr.prototype, "simplificationQueue", { get: function() { if (!this._simplificationQueue) { this._simplificationQueue = new Pne(); let A = this._getComponent(Ot.NAME_SIMPLIFICATIONQUEUE); A || (A = new pne(this), this._addComponent(A)); } return this._simplificationQueue; }, set: function(A) { this._simplificationQueue = A; }, enumerable: !0, configurable: !0 }); Ee.prototype.simplify = function(A, e = !0, t = nR.QUADRATIC, r) { return this.getScene().simplificationQueue.addTask({ settings: A, parallelProcessing: e, mesh: this, simplificationType: t, successCallback: r }), this; }; class pne { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_SIMPLIFICATIONQUEUE, this.scene = e; } /** * Registers the component in a given scene */ register() { this.scene._beforeCameraUpdateStage.registerStep(Ot.STEP_BEFORECAMERAUPDATE_SIMPLIFICATIONQUEUE, this, this._beforeCameraUpdate); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { } /** * Disposes the component and the associated resources */ dispose() { } _beforeCameraUpdate() { this.scene._simplificationQueue && !this.scene._simplificationQueue.running && this.scene._simplificationQueue.executeNext(); } } var m0; (function(A) { A[A.POINTS_MODE_POINTS = 0] = "POINTS_MODE_POINTS", A[A.POINTS_MODE_PATHS = 1] = "POINTS_MODE_PATHS"; })(m0 || (m0 = {})); var kS; (function(A) { A[A.FACES_MODE_SINGLE_SIDED = 0] = "FACES_MODE_SINGLE_SIDED", A[A.FACES_MODE_SINGLE_SIDED_NO_BACKFACE_CULLING = 1] = "FACES_MODE_SINGLE_SIDED_NO_BACKFACE_CULLING", A[A.FACES_MODE_DOUBLE_SIDED = 2] = "FACES_MODE_DOUBLE_SIDED"; })(kS || (kS = {})); var cD; (function(A) { A[A.AUTO_DIRECTIONS_FROM_FIRST_SEGMENT = 0] = "AUTO_DIRECTIONS_FROM_FIRST_SEGMENT", A[A.AUTO_DIRECTIONS_FROM_ALL_SEGMENTS = 1] = "AUTO_DIRECTIONS_FROM_ALL_SEGMENTS", A[A.AUTO_DIRECTIONS_ENHANCED = 2] = "AUTO_DIRECTIONS_ENHANCED", A[A.AUTO_DIRECTIONS_NONE = 99] = "AUTO_DIRECTIONS_NONE"; })(cD || (cD = {})); class dY extends Ee { constructor(e, t, r) { var n, i, s, a; super(e, t, null, null, !1, !1), this.name = e, this._options = r, this._lazy = !1, this._updatable = !1, this._engine = t.getEngine(), this._lazy = (n = r.lazy) !== null && n !== void 0 ? n : !1, this._updatable = (i = r.updatable) !== null && i !== void 0 ? i : !1, this._vertexPositions = [], this._indices = [], this._uvs = [], this._points = [], this._colorPointers = (s = r.colorPointers) !== null && s !== void 0 ? s : [], this._widths = (a = r.widths) !== null && a !== void 0 ? a : new Array(r.points.length).fill(1); } /** * "GreasedLineMesh" * @returns "GreasedLineMesh" */ getClassName() { return "GreasedLineMesh"; } _updateWidthsWithValue(e) { let t = 0; for (const n of this._points) t += n.length; const r = t / 3 * 2 - this._widths.length; for (let n = 0; n < r; n++) this._widths.push(e); } /** * Updated a lazy line. Rerenders the line and updates boundinfo as well. */ updateLazy() { var e, t; this._setPoints(this._points), this._options.colorPointers || this._updateColorPointers(), this._createVertexBuffers((e = this._options.ribbonOptions) === null || e === void 0 ? void 0 : e.smoothShading), this.refreshBoundingInfo(), (t = this.greasedLineMaterial) === null || t === void 0 || t.updateLazy(); } /** * Adds new points to the line. It doesn't rerenders the line if in lazy mode. * @param points points table */ addPoints(e, t) { for (const r of e) this._points.push(r); this._lazy || this.setPoints(this._points, t); } /** * Dispose the line and it's resources */ dispose() { super.dispose(); } /** * * @returns true if the mesh was created in lazy mode */ isLazy() { return this._lazy; } /** * Return the points offsets */ get offsets() { return this._offsets; } /** * Sets point offests * @param offsets offset table [x,y,z, x,y,z, ....] */ set offsets(e) { this._offsets = e, this._offsetsBuffer ? this._offsetsBuffer.update(e) : this._createOffsetsBuffer(e); } /** * Gets widths at each line point like [widthLower, widthUpper, widthLower, widthUpper, ...] */ get widths() { return this._widths; } /** * Sets widths at each line point * @param widths width table [widthLower, widthUpper, widthLower, widthUpper ...] */ set widths(e) { this._widths = e, this._lazy || this._widthsBuffer && this._widthsBuffer.update(e); } /** * Gets the color pointer. Each vertex need a color pointer. These color pointers points to the colors in the color table @see colors */ get colorPointers() { return this._colorPointers; } /** * Sets the color pointer * @param colorPointers array of color pointer in the colors array. One pointer for every vertex is needed. */ set colorPointers(e) { this._colorPointers = e, this._lazy || this._colorPointersBuffer && this._colorPointersBuffer.update(e); } /** * Gets the pluginMaterial associated with line */ get greasedLineMaterial() { var e, t; if (this.material && this.material instanceof AY) return this.material; const r = (t = (e = this.material) === null || e === void 0 ? void 0 : e.pluginManager) === null || t === void 0 ? void 0 : t.getPlugin(o4.GREASED_LINE_MATERIAL_NAME); if (r) return r; } /** * Return copy the points. */ get points() { const e = []; return sA.DeepCopy(this._points, e), e; } /** * Sets line points and rerenders the line. * @param points points table */ setPoints(e, t) { this._points = e, this._updateWidths(), t != null && t.colorPointers || this._updateColorPointers(), this._setPoints(e, t); } _initGreasedLine() { this._vertexPositions = [], this._indices = [], this._uvs = []; } _createLineOptions() { return { points: this._points, colorPointers: this._colorPointers, lazy: this._lazy, updatable: this._updatable, uvs: this._uvs, widths: this._widths, ribbonOptions: this._options.ribbonOptions }; } /** * Serializes this GreasedLineMesh * @param serializationObject object to write serialization to */ serialize(e) { super.serialize(e), e.type = this.getClassName(), e.lineOptions = this._createLineOptions(); } _createVertexBuffers(e = !1) { const t = new Ut(); return t.positions = this._vertexPositions, t.indices = this._indices, t.uvs = this._uvs, e && (t.normals = [], Ut.ComputeNormals(this._vertexPositions, this._indices, t.normals)), t.applyToMesh(this, this._options.updatable), t; } _createOffsetsBuffer(e) { const t = this._scene.getEngine(), r = new P9(t, e, this._updatable, 3); this.setVerticesBuffer(r.createVertexBuffer("grl_offsets", 0, 3)), this._offsetsBuffer = r; } } Ee._GreasedLineMeshParser = (A, e) => T1.Parse(A, e); class T1 extends dY { /** * GreasedLineMesh * @param name name of the mesh * @param scene the scene * @param _options mesh options */ constructor(e, t, r) { super(e, t, r), this.name = e, this.intersectionThreshold = 0.1, this._previousAndSide = [], this._nextAndCounters = [], r.points && this.addPoints(Qs.ConvertPoints(r.points)); } /** * "GreasedLineMesh" * @returns "GreasedLineMesh" */ getClassName() { return "GreasedLineMesh"; } _updateColorPointers() { if (this._options.colorPointers) return; let e = 0; this._colorPointers = [], this._points.forEach((t) => { for (let r = 0; r < t.length; r += 3) this._colorPointers.push(e), this._colorPointers.push(e++); }); } _updateWidths() { super._updateWidthsWithValue(0); } _setPoints(e) { this._points = e, this._options.points = e, this._initGreasedLine(); let t = 0; e.forEach((r) => { var n; const i = [], s = [], a = [], f = Qs.GetLineLength(r); for (let l = 0, P = 0; P < r.length; l++, P += 3) { const p = r.slice(0, P + 3), H = Qs.GetLineLength(p) / f; if (s.push(r[P], r[P + 1], r[P + 2]), s.push(r[P], r[P + 1], r[P + 2]), i.push(H), i.push(H), P < r.length - 3) { const T = l * 2 + t; a.push(T, T + 1, T + 2), a.push(T + 2, T + 1, T + 3); } } t += r.length / 3 * 2; const o = [], d = [], v = []; let u = []; this._preprocess(s, o, d, v, u); for (const l of s) this._vertexPositions.push(l); for (const l of a) this._indices.push(l); for (let l = 0; l < v.length; l++) this._previousAndSide.push(o[l * 3], o[l * 3 + 1], o[l * 3 + 2], v[l]), this._nextAndCounters.push(d[l * 3], d[l * 3 + 1], d[l * 3 + 2], i[l]); u = (n = this._options.uvs) !== null && n !== void 0 ? n : u; for (const l of u) this._uvs.push(l); }), this._lazy || (this._options.colorPointers || this._updateColorPointers(), this._createVertexBuffers(), this.refreshBoundingInfo()); } /** * Clones the GreasedLineMesh. * @param name new line name * @param newParent new parent node * @returns cloned line */ clone(e = `${this.name}-cloned`, t) { const r = this._createLineOptions(), n = {}; sA.DeepCopy(r, n, ["instance"], void 0, !0); const i = new T1(e, this._scene, n); return t && (i.parent = t), i.material = this.material, i; } /** * Serializes this GreasedLineMesh * @param serializationObject object to write serialization to */ serialize(e) { super.serialize(e), e.type = this.getClassName(), e.lineOptions = this._createLineOptions(); } /** * Parses a serialized GreasedLineMesh * @param parsedMesh the serialized GreasedLineMesh * @param scene the scene to create the GreasedLineMesh in * @returns the created GreasedLineMesh */ static Parse(e, t) { const r = e.lineOptions, n = e.name; return new T1(n, t, r); } _initGreasedLine() { super._initGreasedLine(), this._previousAndSide = [], this._nextAndCounters = []; } /** * Checks whether a ray is intersecting this GreasedLineMesh * @param ray ray to check the intersection of this mesh with * @param fastCheck not supported * @param trianglePredicate not supported * @param onlyBoundingInfo defines a boolean indicating if picking should only happen using bounding info (false by default) * @param worldToUse not supported * @param skipBoundingInfo a boolean indicating if we should skip the bounding info check * @returns the picking info */ intersects(e, t, r, n = !1, i, s = !1) { const a = new F9(), f = this.findAllIntersections(e, t, r, n, i, s, !0); if ((f == null ? void 0 : f.length) === 1) { const o = f[0]; a.hit = !0, a.distance = o.distance, a.ray = e, a.pickedMesh = this, a.pickedPoint = o.point; } return a; } /** * Gets all intersections of a ray and the line * @param ray Ray to check the intersection of this mesh with * @param _fastCheck not supported * @param _trianglePredicate not supported * @param onlyBoundingInfo defines a boolean indicating if picking should only happen using bounding info (false by default) * @param _worldToUse not supported * @param skipBoundingInfo a boolean indicating if we should skip the bounding info check * @param firstOnly If true, the first and only intersection is immediatelly returned if found * @returns intersection(s) */ findAllIntersections(e, t, r, n = !1, i, s = !1, a = !1) { var f, o; if (n && !s && e.intersectsSphere(this._boundingSphere, this.intersectionThreshold) === !1) return; const d = this.getIndices(), v = this.getVerticesData(J.PositionKind), u = this._widths, l = (o = (f = this.greasedLineMaterial) === null || f === void 0 ? void 0 : f.width) !== null && o !== void 0 ? o : 1, P = []; if (d && v && u) { let p = 0, c = 0; for (p = 0, c = d.length - 1; p < c; p += 3) { const H = d[p], T = d[p + 1]; T1._V_START.fromArray(v, H * 3), T1._V_END.fromArray(v, T * 3), this._offsets && (T1._V_OFFSET_START.fromArray(this._offsets, H * 3), T1._V_OFFSET_END.fromArray(this._offsets, T * 3), T1._V_START.addInPlace(T1._V_OFFSET_START), T1._V_END.addInPlace(T1._V_OFFSET_END)); const q = Math.floor(p / 3), b = u[q] !== void 0 ? u[q] : 1, j = this.intersectionThreshold * (l * b) / 2, w = e.intersectionSegment(T1._V_START, T1._V_END, j); if (w !== -1 && (P.push({ distance: w, point: e.direction.normalize().multiplyByFloats(w, w, w).add(e.origin) }), a)) return P; } p = c; } return P; } get _boundingSphere() { return this.getBoundingInfo().boundingSphere; } static _CompareV3(e, t, r) { const n = e * 6, i = t * 6; return r[n] === r[i] && r[n + 1] === r[i + 1] && r[n + 2] === r[i + 2]; } static _CopyV3(e, t) { const r = e * 6; return [t[r], t[r + 1], t[r + 2]]; } _preprocess(e, t, r, n, i) { const s = e.length / 6; let a = []; T1._CompareV3(0, s - 1, e) ? a = T1._CopyV3(s - 2, e) : a = T1._CopyV3(0, e), t.push(a[0], a[1], a[2]), t.push(a[0], a[1], a[2]); for (let f = 0; f < s; f++) n.push(1), n.push(-1), this._options.uvs || (i.push(f / (s - 1), 0), i.push(f / (s - 1), 1)), f < s - 1 && (a = T1._CopyV3(f, e), t.push(a[0], a[1], a[2]), t.push(a[0], a[1], a[2])), f > 0 && (a = T1._CopyV3(f, e), r.push(a[0], a[1], a[2]), r.push(a[0], a[1], a[2])); return T1._CompareV3(s - 1, 0, e) ? a = T1._CopyV3(1, e) : a = T1._CopyV3(s - 1, e), r.push(a[0], a[1], a[2]), r.push(a[0], a[1], a[2]), { previous: t, next: r, uvs: i, side: n }; } _createVertexBuffers() { const e = super._createVertexBuffers(), t = this._scene.getEngine(), r = new P9(t, this._previousAndSide, !1, 4); this.setVerticesBuffer(r.createVertexBuffer("grl_previousAndSide", 0, 4)); const n = new P9(t, this._nextAndCounters, !1, 4); this.setVerticesBuffer(n.createVertexBuffer("grl_nextAndCounters", 0, 4)); const i = new P9(t, this._widths, this._updatable, 1); this.setVerticesBuffer(i.createVertexBuffer("grl_widths", 0, 1)), this._widthsBuffer = i; const s = new P9(t, this._colorPointers, this._updatable, 1); return this.setVerticesBuffer(s.createVertexBuffer("grl_colorPointers", 0, 1)), this._colorPointersBuffer = s, e; } } T1._V_START = new S(); T1._V_END = new S(); T1._V_OFFSET_START = new S(); T1._V_OFFSET_END = new S(); Ee._GreasedLineRibbonMeshParser = (A, e) => L1.Parse(A, e); class L1 extends dY { /** * GreasedLineRibbonMesh * @param name name of the mesh * @param scene the scene * @param _options mesh options * @param _pathOptions used internaly when parsing a serialized GreasedLineRibbonMesh */ constructor(e, t, r, n) { var i; if (super(e, t, r), this.name = e, !r.ribbonOptions) throw "'GreasedLineMeshOptions.ribbonOptions' is not set."; this._paths = [], this._counters = [], this._slopes = [], this._widths = (i = r.widths) !== null && i !== void 0 ? i : [], this._ribbonWidths = [], this._pathsOptions = n ?? [], r.points && this.addPoints(Qs.ConvertPoints(r.points), r, !!n); } /** * Adds new points to the line. It doesn't rerenders the line if in lazy mode. * @param points points table */ addPoints(e, t, r = !1) { if (!t.ribbonOptions) throw "addPoints() on GreasedLineRibbonMesh instance requires 'GreasedLineMeshOptions.ribbonOptions'."; r || this._pathsOptions.push({ options: t, pathCount: e.length }), super.addPoints(e, t); } /** * "GreasedLineRibbonMesh" * @returns "GreasedLineRibbonMesh" */ getClassName() { return "GreasedLineRibbonMesh"; } /** * Return true if the line was created from two edge paths or one points path. * In this case the line is always flat. */ get isFlatLine() { return this._paths.length < 3; } /** * Returns the slopes of the line at each point relative to the center of the line */ get slopes() { return this._slopes; } /** * Set the slopes of the line at each point relative to the center of the line */ set slopes(e) { this._slopes = e; } _updateColorPointers() { if (this._options.colorPointers) return; let e = 0; this._colorPointers = []; for (let t = 0; t < this._pathsOptions.length; t++) { const { options: r, pathCount: n } = this._pathsOptions[t], i = this._points[t]; if (r.ribbonOptions.pointsMode === m0.POINTS_MODE_POINTS) for (let s = 0; s < n; s++) for (let a = 0; a < i.length; a += 3) this._colorPointers.push(e), this._colorPointers.push(e++); else for (let s = 0; s < i.length; s += 3) { for (let a = 0; a < n; a++) this._colorPointers.push(e); e++; } } } _updateWidths() { super._updateWidthsWithValue(1); } _setPoints(e, t) { var r, n; if (!this._options.ribbonOptions) throw "No 'GreasedLineMeshOptions.ribbonOptions' provided."; this._points = e, this._options.points = e, this._initGreasedLine(); let i = 0, s; for (let a = 0, f = 0; a < this._pathsOptions.length; a++) { const { options: o, pathCount: d } = this._pathsOptions[a], v = e.slice(f, f + d); if (f += d, ((r = o.ribbonOptions) === null || r === void 0 ? void 0 : r.pointsMode) === m0.POINTS_MODE_PATHS) i = this._preprocess(Qs.ToVector3Array(v), i, o); else { if (((n = o.ribbonOptions) === null || n === void 0 ? void 0 : n.directionsAutoMode) === cD.AUTO_DIRECTIONS_NONE) { if (!o.ribbonOptions.directions) throw "In GreasedLineRibbonAutoDirectionMode.AUTO_DIRECTIONS_NONE 'GreasedLineMeshOptions.ribbonOptions.directions' must be defined."; s = L1._GetDirectionPlanesFromDirectionsOption(v.length, o.ribbonOptions.directions); } v.forEach((u, l) => { const P = L1._ConvertToRibbonPath(u, o.ribbonOptions, this._scene.useRightHandedSystem, s && s[l]); i = this._preprocess(P, i, o); }); } } this._lazy || (this._createVertexBuffers(), this.refreshBoundingInfo()); } static _GetDirectionPlanesFromDirectionsOption(e, t) { return Array.isArray(t) ? t : new Array(e).fill(t); } static _CreateRibbonVertexData(e, t) { var r, n, i; const s = e.length; if (s < 2) throw "Minimum of two paths are required to create a GreasedLineRibbonMesh."; const a = [], f = [], o = e[0]; for (let l = 0; l < o.length; l++) for (let P = 0; P < e.length; P++) { const p = e[P][l]; a.push(p.x, p.y, p.z); } const d = [1, 0, s], v = (n = ((r = t.ribbonOptions) === null || r === void 0 ? void 0 : r.facesMode) === kS.FACES_MODE_DOUBLE_SIDED) !== null && n !== void 0 ? n : !1, u = ((i = t.ribbonOptions) === null || i === void 0 ? void 0 : i.pointsMode) === m0.POINTS_MODE_PATHS && t.ribbonOptions.closePath; if (s > 2) for (let l = 0; l < o.length - 1; l++) { d[0] = 1 + s * l, d[1] = s * l, d[2] = (l + 1) * s; for (let P = 0; P < (s - 1) * 2; P++) P % 2 !== 0 && (d[2] += 1), P % 2 === 0 && P > 0 && (d[0] += 1, d[1] += 1), f.push(d[1] + (P % 2 !== 0 ? s : 0), d[0], d[2]), v && f.push(d[0], d[1] + (P % 2 !== 0 ? s : 0), d[2]); } else for (let l = 0; l < a.length / 3 - 3; l += 2) f.push(l, l + 1, l + 2), f.push(l + 2, l + 1, l + 3), v && (f.push(l + 1, l, l + 2), f.push(l + 1, l + 2, l + 3)); if (u) { let l = s * (o.length - 1); for (let P = 0; P < s - 1; P++) f.push(l, P + 1, P), f.push(l + 1, P + 1, l), v && (f.push(P, P + 1, l), f.push(l, P + 1, l + 1)), l++; } return { positions: a, indices: f }; } _preprocess(e, t, r) { var n, i, s, a; this._paths = e; const f = L1._CreateRibbonVertexData(e, r), o = f.positions; if (!this._options.widths) throw "No 'GreasedLineMeshOptions.widths' table is specified."; for (const P of o) this._vertexPositions.push(P); let d = e; if (((n = r.ribbonOptions) === null || n === void 0 ? void 0 : n.pointsMode) === m0.POINTS_MODE_PATHS && r.ribbonOptions.closePath) { d = []; for (let P = 0; P < e.length; P++) { const p = e[P].slice(); p.push(e[P][0].clone()), d.push(p); } } this._calculateSegmentLengths(d); const v = d.length, u = new Array(v).fill(0); for (let P = 0; P < d[0].length; P++) { let p = 0; for (let c = 0; c < v; c++) { const H = u[c] + this._vSegmentLengths[c][P] / this._vTotalLengths[c]; this._counters.push(H), this._uvs.push(H, p), u[c] = H, p += this._uSegmentLengths[P][c] / this._uTotalLengths[P]; } } for (let P = 0, p = 0; P < d[0].length; P++) { const c = this._uSegmentLengths[P][0] / 2, H = this._uSegmentLengths[P][v - 1] / 2; this._ribbonWidths.push((((i = this._widths[p++]) !== null && i !== void 0 ? i : 1) - 1) * c); for (let T = 0; T < v - 2; T++) this._ribbonWidths.push(0); this._ribbonWidths.push((((s = this._widths[p++]) !== null && s !== void 0 ? s : 1) - 1) * H); } const l = ((a = r.ribbonOptions) === null || a === void 0 ? void 0 : a.pointsMode) === m0.POINTS_MODE_PATHS ? new Array(d[0].length * d.length * 6).fill(0) : L1._CalculateSlopes(d); for (const P of l) this._slopes.push(P); if (f.indices) for (let P = 0; P < f.indices.length; P++) this._indices.push(f.indices[P] + t); return t += o.length / 3, t; } static _ConvertToRibbonPath(e, t, r, n) { if (t.pointsMode === m0.POINTS_MODE_POINTS && !t.width) throw "'GreasedLineMeshOptions.ribbonOptiosn.width' must be specified in GreasedLineRibbonPointsMode.POINTS_MODE_POINTS."; const i = [], s = []; if (t.pointsMode === m0.POINTS_MODE_POINTS) { const a = t.width / 2, f = Qs.ToVector3Array(e); let o = null, d = null; t.directionsAutoMode === cD.AUTO_DIRECTIONS_FROM_FIRST_SEGMENT && (n = L1._GetDirectionFromPoints(f[0], f[1], null)); for (let v = 0; v < f.length - (n ? 0 : 1); v++) { const u = f[v], l = f[v + 1]; if (n) o = n; else if (t.directionsAutoMode === cD.AUTO_DIRECTIONS_FROM_ALL_SEGMENTS) o = L1._GetDirectionFromPoints(u, l, o); else { const P = l.subtract(u); P.applyRotationQuaternionInPlace(P.x > P.y && P.x > P.z ? r ? L1._RightHandedForwardReadOnlyQuaternion : L1._LeftHandedForwardReadOnlyQuaternion : L1._LeftReadOnlyQuaternion), o = P.normalize(); } d = o.multiplyByFloats(a, a, a), i.push(u.add(d)), s.push(u.subtract(d)); } n || (i.push(f[f.length - 1].add(d)), s.push(f[f.length - 1].subtract(d))); } return [i, s]; } static _GetDirectionFromPoints(e, t, r) { return e.x === t.x && (!r || (r == null ? void 0 : r.x) === 1) ? L1.DIRECTION_YZ : e.y === t.y ? L1.DIRECTION_XZ : e.z === t.z ? L1.DIRECTION_XY : L1.DIRECTION_XZ; } /** * Clones the GreasedLineRibbonMesh. * @param name new line name * @param newParent new parent node * @returns cloned line */ clone(e = `${this.name}-cloned`, t) { const r = this._createLineOptions(), n = {}, i = []; sA.DeepCopy(this._pathsOptions, i, void 0, void 0, !0), sA.DeepCopy(r, n, ["instance"], void 0, !0); const s = new L1(e, this._scene, n, i); return t && (s.parent = t), s.material = this.material, s; } /** * Serializes this GreasedLineRibbonMesh * @param serializationObject object to write serialization to */ serialize(e) { super.serialize(e), e.type = this.getClassName(), e.lineOptions = this._createLineOptions(), e.pathsOptions = this._pathsOptions; } /** * Parses a serialized GreasedLineRibbonMesh * @param parsedMesh the serialized GreasedLineRibbonMesh * @param scene the scene to create the GreasedLineRibbonMesh in * @returns the created GreasedLineRibbonMesh */ static Parse(e, t) { const r = e.lineOptions, n = e.name, i = e.pathOptions; return new L1(n, t, r, i); } _initGreasedLine() { super._initGreasedLine(), this._paths = [], this._counters = [], this._slopes = [], this._ribbonWidths = []; } _calculateSegmentLengths(e) { const t = e.length; this._vSegmentLengths = new Array(t), this._vTotalLengths = new Array(t); let r = 0; for (let s = 0; s < t; s++) { const a = e[s]; this._vSegmentLengths[s] = [0], r = 0; for (let f = 0; f < a.length - 1; f++) { const o = Math.abs(a[f].subtract(a[f + 1]).lengthSquared()); r += o, this._vSegmentLengths[s].push(o); } this._vTotalLengths[s] = r; } const n = e[0].length; this._uSegmentLengths = new Array(n).fill([]), this._uTotalLengths = new Array(n).fill([]); const i = new S(); for (let s = 0; s < n; s++) { r = 0; for (let a = 1; a < t; a++) { e[a][s].subtractToRef(e[a - 1][s], i); const f = i.length(); r += f, this._uSegmentLengths[s].push(f); } this._uTotalLengths[s] = r; } } static _CalculateSlopes(e) { const t = e[0], r = e.length === 2 ? e[1] : e[e.length - 1], n = [], i = new S(); for (let s = 0; s < t.length; s++) for (let a = 0; a < e.length; a++) a === 0 || a === e.length - 1 ? (t[s].subtract(r[s]).normalizeToRef(i), n.push(i.x, i.y, i.z), n.push(-i.x, -i.y, -i.z)) : n.push(0, 0, 0, 0, 0, 0); return n; } _createVertexBuffers() { var e, t; this._uvs = (e = this._options.uvs) !== null && e !== void 0 ? e : this._uvs; const r = super._createVertexBuffers((t = this._options.ribbonOptions) === null || t === void 0 ? void 0 : t.smoothShading), n = new P9(this._engine, this._counters, this._updatable, 1); this.setVerticesBuffer(n.createVertexBuffer("grl_counters", 0, 1)); const i = new P9(this._engine, this._colorPointers, this._updatable, 1); this.setVerticesBuffer(i.createVertexBuffer("grl_colorPointers", 0, 1)); const s = new P9(this._engine, this._slopes, this._updatable, 3); this.setVerticesBuffer(s.createVertexBuffer("grl_slopes", 0, 3)); const a = new P9(this._engine, this._ribbonWidths, this._updatable, 1); return this.setVerticesBuffer(a.createVertexBuffer("grl_widths", 0, 1)), this._widthsBuffer = a, r; } } L1.DEFAULT_WIDTH = 0.1; L1._RightHandedForwardReadOnlyQuaternion = Ze.RotationAxis(S.RightHandedForwardReadOnly, Math.PI / 2); L1._LeftHandedForwardReadOnlyQuaternion = Ze.RotationAxis(S.LeftHandedForwardReadOnly, Math.PI / 2); L1._LeftReadOnlyQuaternion = Ze.RotationAxis(S.LeftReadOnly, Math.PI / 2); L1.DIRECTION_XY = S.LeftHandedForwardReadOnly; L1.DIRECTION_XZ = S.UpReadOnly; L1.DIRECTION_YZ = S.LeftReadOnly; var E2; (function(A) { A[A.COLOR_DISTRIBUTION_NONE = 0] = "COLOR_DISTRIBUTION_NONE", A[A.COLOR_DISTRIBUTION_REPEAT = 1] = "COLOR_DISTRIBUTION_REPEAT", A[A.COLOR_DISTRIBUTION_EVEN = 2] = "COLOR_DISTRIBUTION_EVEN", A[A.COLOR_DISTRIBUTION_START = 3] = "COLOR_DISTRIBUTION_START", A[A.COLOR_DISTRIBUTION_END = 4] = "COLOR_DISTRIBUTION_END", A[A.COLOR_DISTRIBUTION_START_END = 5] = "COLOR_DISTRIBUTION_START_END"; })(E2 || (E2 = {})); var rq; (function(A) { A[A.WIDTH_DISTRIBUTION_NONE = 0] = "WIDTH_DISTRIBUTION_NONE", A[A.WIDTH_DISTRIBUTION_REPEAT = 1] = "WIDTH_DISTRIBUTION_REPEAT", A[A.WIDTH_DISTRIBUTION_EVEN = 2] = "WIDTH_DISTRIBUTION_EVEN", A[A.WIDTH_DISTRIBUTION_START = 3] = "WIDTH_DISTRIBUTION_START", A[A.WIDTH_DISTRIBUTION_END = 4] = "WIDTH_DISTRIBUTION_END", A[A.WIDTH_DISTRIBUTION_START_END = 5] = "WIDTH_DISTRIBUTION_START_END"; })(rq || (rq = {})); function hne(A, e, t) { t = t ?? gr.LastCreatedScene; let r; switch (e.materialType) { case OS.MATERIAL_TYPE_PBR: r = new mr(A, t), new o4(r, t, e); break; case OS.MATERIAL_TYPE_SIMPLE: r = new AY(A, t, e); break; default: r = new Wt(A, t), new o4(r, t, e); break; } return r; } function mhe(A, e, t, r) { var n, i, s, a, f, o, d, v, u, l, P, p; r = r ?? gr.LastCreatedScene; let c; const H = Qs.ConvertPoints(e.points); e.widthDistribution = (n = e.widthDistribution) !== null && n !== void 0 ? n : rq.WIDTH_DISTRIBUTION_START, e.ribbonOptions && (e.ribbonOptions.facesMode = (i = e.ribbonOptions.facesMode) !== null && i !== void 0 ? i : kS.FACES_MODE_SINGLE_SIDED_NO_BACKFACE_CULLING, e.ribbonOptions.pointsMode = (s = e.ribbonOptions.pointsMode) !== null && s !== void 0 ? s : m0.POINTS_MODE_POINTS, e.ribbonOptions.directionsAutoMode = (a = e.ribbonOptions.directionsAutoMode) !== null && a !== void 0 ? a : e.ribbonOptions.directions ? cD.AUTO_DIRECTIONS_NONE : cD.AUTO_DIRECTIONS_FROM_FIRST_SEGMENT), t = t ?? { color: D6.DEFAULT_COLOR }, t.createAndAssignMaterial = (f = t.createAndAssignMaterial) !== null && f !== void 0 ? f : !0, t.colorDistribution = (o = t == null ? void 0 : t.colorDistribution) !== null && o !== void 0 ? o : E2.COLOR_DISTRIBUTION_START, t.materialType = (d = t.materialType) !== null && d !== void 0 ? d : OS.MATERIAL_TYPE_STANDARD; let T = 0; Array.isArray(H[0]) && H.forEach((w) => { T += w.length / 3; }); const q = Hne(T, (v = e.widths) !== null && v !== void 0 ? v : [], e.widthDistribution), b = t != null && t.colors ? gne(T, t.colors, t.colorDistribution, (u = t.color) !== null && u !== void 0 ? u : D6.DEFAULT_COLOR) : void 0, j = { points: H, updatable: e.updatable, widths: q, lazy: e.lazy, ribbonOptions: e.ribbonOptions, uvs: e.uvs, colorPointers: e.colorPointers }; if (j.ribbonOptions && j.ribbonOptions.pointsMode === m0.POINTS_MODE_POINTS && (j.ribbonOptions.width = (P = (l = t.width) !== null && l !== void 0 ? l : j.ribbonOptions.width) !== null && P !== void 0 ? P : D6.DEFAULT_WIDTH), e.instance) if (c = e.instance, c instanceof L1) c.addPoints(H, j); else { const w = c.widths; if (w) { const m = w.slice(); for (const I of q) m.push(I); c.widths = m; } else c.widths = q; c.addPoints(H); } else if (c = j.ribbonOptions ? new L1(A, r, j) : new T1(A, r, j), t) { const w = { materialType: t.materialType, dashCount: t.dashCount, dashOffset: t.dashOffset, dashRatio: t.dashRatio, resolution: t.resolution, sizeAttenuation: t.sizeAttenuation, useColors: t.useColors, useDash: t.useDash, visibility: t.visibility, width: t.width, color: t.color, colorMode: t.colorMode, colorsSampling: t.colorsSampling, colorDistributionType: t.colorDistributionType, colors: b, cameraFacing: !e.ribbonOptions, colorsTexture: t.colorsTexture }; if (t.createAndAssignMaterial) { const m = hne(A, w, r); c.material = m, ((p = e.ribbonOptions) === null || p === void 0 ? void 0 : p.facesMode) === kS.FACES_MODE_SINGLE_SIDED_NO_BACKFACE_CULLING && (m.backFaceCulling = !1); } } if (b && e.instance && e.instance.greasedLineMaterial) { const w = e.instance.greasedLineMaterial.colors; if (w) { const m = w.concat(b); e.instance.greasedLineMaterial.setColors(m, c.isLazy()); } } return c; } function Hne(A, e, t, r = 1, n = 1) { const i = A - e.length / 2, s = []; if (i < 0) return e.slice(0, A * 2); if (i > 0) { if (e.length % 2 != 0 && e.push(r), t === rq.WIDTH_DISTRIBUTION_START_END) { const a = Math.floor(e.length / 2); for (let d = 0, v = 0; d < a - 1; d++) s.push(e[v++]), s.push(e[v++]); const f = e[a / 2], o = e[a / 2 + 1]; for (let d = 0; d < i; d++) s.push(o), s.push(f); for (let d = a; d < e.length; d += 2) s.push(e[d]), s.push(e[d + 1]); } else if (t === rq.WIDTH_DISTRIBUTION_START) { for (let a = 0; a < e.length; a += 2) s.push(e[a]), s.push(e[a + 1]); for (let a = 0; a < i; a++) s.push(r), s.push(n); } else if (t === rq.WIDTH_DISTRIBUTION_END) { for (let a = 0; a < i; a++) s.push(r), s.push(n); for (let a = 0; a < e.length; a += 2) s.push(e[a]), s.push(e[a + 1]); } else if (t === rq.WIDTH_DISTRIBUTION_REPEAT) { let a = 0; for (let f = 0; f < A; f++) s.push(e[a++]), s.push(e[a++]), a === e.length && (a = 0); } else if (t === rq.WIDTH_DISTRIBUTION_EVEN) { let a = 0; const f = e.length / ((A - 1) * 2); for (let o = 0; o < A; o++) { const d = Math.floor(a); s.push(e[d]), s.push(e[d + 1]), a += f; } } } else for (let a = 0; a < e.length; a++) s.push(e[a]); return s; } function gne(A, e, t, r) { A = Math.max(e.length, A); const n = A - e.length; if (n < 0) return e.slice(0, A); const i = []; if (n > 0) { if (t === E2.COLOR_DISTRIBUTION_START_END) { const s = Math.floor(e.length / 2); for (let a = 0; a < s; a++) i.push(e[a]); for (let a = 0; a < n - 1; a++) i.push(r); for (let a = s; a < e.length; a++) i.push(e[a]); } else if (t === E2.COLOR_DISTRIBUTION_START) { for (let s = 0; s < e.length; s++) i.push(e[s]); for (let s = 0; s < n; s++) i.push(r); } else if (t === E2.COLOR_DISTRIBUTION_END) { for (let s = 0; s < n - 1; s++) i.push(r); for (let s = 0; s < e.length; s++) i.push(e[s]); } else if (t === E2.COLOR_DISTRIBUTION_REPEAT) { let s = 0; for (let a = 0; a < A; a++) i.push(e[s]), s++, s === e.length && (s = 0); } else if (t === E2.COLOR_DISTRIBUTION_EVEN) { let s = 0; const a = e.length / (A - 1); for (let f = 0; f < A - 1; f++) { const o = Math.floor(s); i.push(e[o]), s += a; } } else if (t === E2.COLOR_DISTRIBUTION_NONE) for (let s = 0; s < e.length; s++) i.push(e[s]); } else for (let s = 0; s < A; s++) i.push(e[s]); return i; } Ee.prototype.thinInstanceAdd = function(A, e = !0) { if (!this.getScene().getEngine().getCaps().instancedArrays) return Se.Error("Thin Instances are not supported on this device as Instanced Array extension not supported"), -1; this._thinInstanceUpdateBufferSize("matrix", Array.isArray(A) ? A.length : 1); const t = this._thinInstanceDataStorage.instancesCount; if (Array.isArray(A)) for (let r = 0; r < A.length; ++r) this.thinInstanceSetMatrixAt(this._thinInstanceDataStorage.instancesCount++, A[r], r === A.length - 1 && e); else this.thinInstanceSetMatrixAt(this._thinInstanceDataStorage.instancesCount++, A, e); return t; }; Ee.prototype.thinInstanceAddSelf = function(A = !0) { return this.thinInstanceAdd(he.IdentityReadOnly, A); }; Ee.prototype.thinInstanceRegisterAttribute = function(A, e) { A === J.ColorKind && (A = J.ColorInstanceKind), this.removeVerticesData(A), this._thinInstanceInitializeUserStorage(), this._userThinInstanceBuffersStorage.strides[A] = e, this._userThinInstanceBuffersStorage.sizes[A] = e * Math.max(32, this._thinInstanceDataStorage.instancesCount), this._userThinInstanceBuffersStorage.data[A] = new Float32Array(this._userThinInstanceBuffersStorage.sizes[A]), this._userThinInstanceBuffersStorage.vertexBuffers[A] = new J(this.getEngine(), this._userThinInstanceBuffersStorage.data[A], A, !0, !1, e, !0), this.setVerticesBuffer(this._userThinInstanceBuffersStorage.vertexBuffers[A]); }; Ee.prototype.thinInstanceSetMatrixAt = function(A, e, t = !0) { if (!this._thinInstanceDataStorage.matrixData || A >= this._thinInstanceDataStorage.instancesCount) return !1; const r = this._thinInstanceDataStorage.matrixData; return e.copyToArray(r, A * 16), this._thinInstanceDataStorage.worldMatrices && (this._thinInstanceDataStorage.worldMatrices[A] = e), t && (this.thinInstanceBufferUpdated("matrix"), this.doNotSyncBoundingInfo || this.thinInstanceRefreshBoundingInfo(!1)), !0; }; Ee.prototype.thinInstanceSetAttributeAt = function(A, e, t, r = !0) { return A === J.ColorKind && (A = J.ColorInstanceKind), !this._userThinInstanceBuffersStorage || !this._userThinInstanceBuffersStorage.data[A] || e >= this._thinInstanceDataStorage.instancesCount ? !1 : (this._thinInstanceUpdateBufferSize(A, 0), this._userThinInstanceBuffersStorage.data[A].set(t, e * this._userThinInstanceBuffersStorage.strides[A]), r && this.thinInstanceBufferUpdated(A), !0); }; Object.defineProperty(Ee.prototype, "thinInstanceCount", { get: function() { return this._thinInstanceDataStorage.instancesCount; }, set: function(A) { var e, t; const r = (e = this._thinInstanceDataStorage.matrixData) !== null && e !== void 0 ? e : (t = this.source) === null || t === void 0 ? void 0 : t._thinInstanceDataStorage.matrixData, n = r ? r.length / 16 : 0; A <= n && (this._thinInstanceDataStorage.instancesCount = A); }, enumerable: !0, configurable: !0 }); Ee.prototype._thinInstanceCreateMatrixBuffer = function(A, e, t = !1) { A === J.ColorKind && (A = J.ColorInstanceKind); const r = new P9(this.getEngine(), e, !t, 16, !1, !0); for (let n = 0; n < 4; n++) this.setVerticesBuffer(r.createVertexBuffer(A + n, n * 4, 4)); return r; }; Ee.prototype.thinInstanceSetBuffer = function(A, e, t = 0, r = !1) { var n, i, s; t = t || 16, A === "matrix" ? ((n = this._thinInstanceDataStorage.matrixBuffer) === null || n === void 0 || n.dispose(), this._thinInstanceDataStorage.matrixBuffer = null, this._thinInstanceDataStorage.matrixBufferSize = e ? e.length : 32 * t, this._thinInstanceDataStorage.matrixData = e, this._thinInstanceDataStorage.worldMatrices = null, e !== null ? (this._thinInstanceDataStorage.instancesCount = e.length / t, this._thinInstanceDataStorage.matrixBuffer = this._thinInstanceCreateMatrixBuffer("world", e, r), this.doNotSyncBoundingInfo || this.thinInstanceRefreshBoundingInfo(!1)) : (this._thinInstanceDataStorage.instancesCount = 0, this.doNotSyncBoundingInfo || this.refreshBoundingInfo())) : A === "previousMatrix" ? ((i = this._thinInstanceDataStorage.previousMatrixBuffer) === null || i === void 0 || i.dispose(), this._thinInstanceDataStorage.previousMatrixBuffer = null, this._thinInstanceDataStorage.previousMatrixData = e, e !== null && (this._thinInstanceDataStorage.previousMatrixBuffer = this._thinInstanceCreateMatrixBuffer("previousWorld", e, r))) : (A === J.ColorKind && (A = J.ColorInstanceKind), e === null ? !((s = this._userThinInstanceBuffersStorage) === null || s === void 0) && s.data[A] && (this.removeVerticesData(A), delete this._userThinInstanceBuffersStorage.data[A], delete this._userThinInstanceBuffersStorage.strides[A], delete this._userThinInstanceBuffersStorage.sizes[A], delete this._userThinInstanceBuffersStorage.vertexBuffers[A]) : (this._thinInstanceInitializeUserStorage(), this._userThinInstanceBuffersStorage.data[A] = e, this._userThinInstanceBuffersStorage.strides[A] = t, this._userThinInstanceBuffersStorage.sizes[A] = e.length, this._userThinInstanceBuffersStorage.vertexBuffers[A] = new J(this.getEngine(), e, A, !r, !1, t, !0), this.setVerticesBuffer(this._userThinInstanceBuffersStorage.vertexBuffers[A]))); }; Ee.prototype.thinInstanceBufferUpdated = function(A) { var e, t, r; A === "matrix" ? (e = this._thinInstanceDataStorage.matrixBuffer) === null || e === void 0 || e.updateDirectly(this._thinInstanceDataStorage.matrixData, 0, this._thinInstanceDataStorage.instancesCount) : A === "previousMatrix" ? (t = this._thinInstanceDataStorage.previousMatrixBuffer) === null || t === void 0 || t.updateDirectly(this._thinInstanceDataStorage.previousMatrixData, 0, this._thinInstanceDataStorage.instancesCount) : (A === J.ColorKind && (A = J.ColorInstanceKind), !((r = this._userThinInstanceBuffersStorage) === null || r === void 0) && r.vertexBuffers[A] && this._userThinInstanceBuffersStorage.vertexBuffers[A].updateDirectly(this._userThinInstanceBuffersStorage.data[A], 0)); }; Ee.prototype.thinInstancePartialBufferUpdate = function(A, e, t) { var r; A === "matrix" ? this._thinInstanceDataStorage.matrixBuffer && this._thinInstanceDataStorage.matrixBuffer.updateDirectly(e, t) : (A === J.ColorKind && (A = J.ColorInstanceKind), !((r = this._userThinInstanceBuffersStorage) === null || r === void 0) && r.vertexBuffers[A] && this._userThinInstanceBuffersStorage.vertexBuffers[A].updateDirectly(e, t)); }; Ee.prototype.thinInstanceGetWorldMatrices = function() { if (!this._thinInstanceDataStorage.matrixData || !this._thinInstanceDataStorage.matrixBuffer) return []; const A = this._thinInstanceDataStorage.matrixData; if (!this._thinInstanceDataStorage.worldMatrices) { this._thinInstanceDataStorage.worldMatrices = []; for (let e = 0; e < this._thinInstanceDataStorage.instancesCount; ++e) this._thinInstanceDataStorage.worldMatrices[e] = he.FromArray(A, e * 16); } return this._thinInstanceDataStorage.worldMatrices; }; Ee.prototype.thinInstanceRefreshBoundingInfo = function(A = !1, e = !1, t = !1) { if (!this._thinInstanceDataStorage.matrixData || !this._thinInstanceDataStorage.matrixBuffer) return; const r = this._thinInstanceDataStorage.boundingVectors; if (A || !this.rawBoundingInfo) { r.length = 0, this.refreshBoundingInfo(e, t); const s = this.getBoundingInfo(); this.rawBoundingInfo = new Md(s.minimum, s.maximum); } const n = this.getBoundingInfo(), i = this._thinInstanceDataStorage.matrixData; if (r.length === 0) for (let s = 0; s < n.boundingBox.vectors.length; ++s) r.push(n.boundingBox.vectors[s].clone()); ue.Vector3[0].setAll(Number.POSITIVE_INFINITY), ue.Vector3[1].setAll(Number.NEGATIVE_INFINITY); for (let s = 0; s < this._thinInstanceDataStorage.instancesCount; ++s) { he.FromArrayToRef(i, s * 16, ue.Matrix[0]); for (let a = 0; a < r.length; ++a) S.TransformCoordinatesToRef(r[a], ue.Matrix[0], ue.Vector3[2]), ue.Vector3[0].minimizeInPlace(ue.Vector3[2]), ue.Vector3[1].maximizeInPlace(ue.Vector3[2]); } n.reConstruct(ue.Vector3[0], ue.Vector3[1]), this._updateBoundingInfo(); }; Ee.prototype._thinInstanceUpdateBufferSize = function(A, e = 1) { var t, r, n; A === J.ColorKind && (A = J.ColorInstanceKind); const i = A === "matrix"; if (!i && (!this._userThinInstanceBuffersStorage || !this._userThinInstanceBuffersStorage.strides[A])) return; const s = i ? 16 : this._userThinInstanceBuffersStorage.strides[A], a = i ? this._thinInstanceDataStorage.matrixBufferSize : this._userThinInstanceBuffersStorage.sizes[A]; let f = i ? this._thinInstanceDataStorage.matrixData : this._userThinInstanceBuffersStorage.data[A]; const o = (this._thinInstanceDataStorage.instancesCount + e) * s; let d = a; for (; d < o; ) d *= 2; if (!f || a != d) { if (!f) f = new Float32Array(d); else { const v = new Float32Array(d); v.set(f, 0), f = v; } i ? ((t = this._thinInstanceDataStorage.matrixBuffer) === null || t === void 0 || t.dispose(), this._thinInstanceDataStorage.matrixBuffer = this._thinInstanceCreateMatrixBuffer("world", f, !1), this._thinInstanceDataStorage.matrixData = f, this._thinInstanceDataStorage.matrixBufferSize = d, this._scene.needsPreviousWorldMatrices && !this._thinInstanceDataStorage.previousMatrixData && ((r = this._thinInstanceDataStorage.previousMatrixBuffer) === null || r === void 0 || r.dispose(), this._thinInstanceDataStorage.previousMatrixBuffer = this._thinInstanceCreateMatrixBuffer("previousWorld", f, !1))) : ((n = this._userThinInstanceBuffersStorage.vertexBuffers[A]) === null || n === void 0 || n.dispose(), this._userThinInstanceBuffersStorage.data[A] = f, this._userThinInstanceBuffersStorage.sizes[A] = d, this._userThinInstanceBuffersStorage.vertexBuffers[A] = new J(this.getEngine(), f, A, !0, !1, s, !0), this.setVerticesBuffer(this._userThinInstanceBuffersStorage.vertexBuffers[A])); } }; Ee.prototype._thinInstanceInitializeUserStorage = function() { this._userThinInstanceBuffersStorage || (this._userThinInstanceBuffersStorage = { data: {}, sizes: {}, vertexBuffers: {}, strides: {} }); }; Ee.prototype._disposeThinInstanceSpecificData = function() { var A; !((A = this._thinInstanceDataStorage) === null || A === void 0) && A.matrixBuffer && (this._thinInstanceDataStorage.matrixBuffer.dispose(), this._thinInstanceDataStorage.matrixBuffer = null); }; var Me; (function(A) { A[A.Int = 1] = "Int", A[A.Float = 2] = "Float", A[A.Vector2 = 4] = "Vector2", A[A.Vector3 = 8] = "Vector3", A[A.Vector4 = 16] = "Vector4", A[A.Matrix = 32] = "Matrix", A[A.Geometry = 64] = "Geometry", A[A.Texture = 128] = "Texture", A[A.AutoDetect = 1024] = "AutoDetect", A[A.BasedOnInput = 2048] = "BasedOnInput", A[A.Undefined = 4096] = "Undefined", A[A.All = 4095] = "All"; })(Me || (Me = {})); var $8; (function(A) { A[A.Compatible = 0] = "Compatible", A[A.TypeIncompatible = 1] = "TypeIncompatible", A[A.HierarchyIssue = 2] = "HierarchyIssue"; })($8 || ($8 = {})); var ES; (function(A) { A[A.Input = 0] = "Input", A[A.Output = 1] = "Output"; })(ES || (ES = {})); class qF { /** Gets the direction of the point */ get direction() { return this._direction; } /** * Gets or sets the connection point type (default is float) */ get type() { if (this._type === Me.AutoDetect) { if (this._ownerBlock.isInput) return this._ownerBlock.type; if (this._connectedPoint) return this._connectedPoint.type; if (this._linkedConnectionSource && this._linkedConnectionSource.isConnected) return this._linkedConnectionSource.type; } if (this._type === Me.BasedOnInput) { if (this._typeConnectionSource) return !this._typeConnectionSource.isConnected && this._defaultConnectionPointType ? this._defaultConnectionPointType : this._typeConnectionSource.type; if (this._defaultConnectionPointType) return this._defaultConnectionPointType; } return this._type; } set type(e) { this._type = e; } /** * Gets a boolean indicating that the current point is connected to another NodeMaterialBlock */ get isConnected() { return this.connectedPoint !== null || this.hasEndpoints; } /** Get the other side of the connection (if any) */ get connectedPoint() { return this._connectedPoint; } /** Get the block that owns this connection point */ get ownerBlock() { return this._ownerBlock; } /** Get the block connected on the other side of this connection (if any) */ get sourceBlock() { return this._connectedPoint ? this._connectedPoint.ownerBlock : null; } /** Get the block connected on the endpoints of this connection (if any) */ get connectedBlocks() { return this._endpoints.length === 0 ? [] : this._endpoints.map((e) => e.ownerBlock); } /** Gets the list of connected endpoints */ get endpoints() { return this._endpoints; } /** Gets a boolean indicating if that output point is connected to at least one input */ get hasEndpoints() { return this._endpoints && this._endpoints.length > 0; } /** Get the inner type (ie AutoDetect for instance instead of the inferred one) */ get innerType() { return this._linkedConnectionSource && this._linkedConnectionSource.isConnected ? this.type : this._type; } /** @internal */ _resetCounters() { this._callCount = 0, this._executionCount = 0; } /** * Gets the number of times this point was called */ get callCount() { return this._callCount; } /** * Gets the number of times this point was executed */ get executionCount() { return this._executionCount; } /** * Gets the value represented by this connection point * @param state current evaluation state * @returns the connected value or the value if nothing is connected */ getConnectedValue(e) { var t; return this.isConnected ? !((t = this._connectedPoint) === null || t === void 0) && t._storedFunction ? (this._connectedPoint._callCount++, this._connectedPoint._executionCount++, this._connectedPoint._storedFunction(e)) : (this._connectedPoint._callCount++, this._connectedPoint._executionCount = 1, this._connectedPoint._storedValue) : (this._callCount++, this._executionCount = 1, this.value); } /** * Creates a new connection point * @param name defines the connection point name * @param ownerBlock defines the block hosting this connection point * @param direction defines the direction of the connection point */ constructor(e, t, r) { this._connectedPoint = null, this._storedValue = null, this._storedFunction = null, this._acceptedConnectionPointType = null, this._endpoints = new Array(), this._type = Me.Geometry, this._linkedConnectionSource = null, this._typeConnectionSource = null, this._defaultConnectionPointType = null, this.acceptedConnectionPointTypes = [], this.excludedConnectionPointTypes = [], this.onConnectionObservable = new Oe(), this.isExposedOnFrame = !1, this.exposedPortPosition = -1, this.defaultValue = null, this.value = null, this.valueMin = null, this.valueMax = null, this._callCount = 0, this._executionCount = 0, this._ownerBlock = t, this.name = e, this._direction = r; } /** * Gets the current class name e.g. "NodeMaterialConnectionPoint" * @returns the class name */ getClassName() { return "NodeGeometryConnectionPoint"; } /** * Gets a boolean indicating if the current point can be connected to another point * @param connectionPoint defines the other connection point * @returns a boolean */ canConnectTo(e) { return this.checkCompatibilityState(e) === $8.Compatible; } /** * Gets a number indicating if the current point can be connected to another point * @param connectionPoint defines the other connection point * @returns a number defining the compatibility state */ checkCompatibilityState(e) { const t = this._ownerBlock, r = e.ownerBlock; if (this.type !== e.type && e.innerType !== Me.AutoDetect) return e.acceptedConnectionPointTypes && e.acceptedConnectionPointTypes.indexOf(this.type) !== -1 ? $8.Compatible : $8.TypeIncompatible; if (e.excludedConnectionPointTypes && e.excludedConnectionPointTypes.indexOf(this.type) !== -1) return $8.TypeIncompatible; let n = r, i = t; return this.direction === ES.Input && (n = t, i = r), n.isAnAncestorOf(i) ? $8.HierarchyIssue : $8.Compatible; } /** * Connect this point to another connection point * @param connectionPoint defines the other connection point * @param ignoreConstraints defines if the system will ignore connection type constraints (default is false) * @returns the current connection point */ connectTo(e, t = !1) { if (!t && !this.canConnectTo(e)) throw "Cannot connect these two connectors."; return this._endpoints.push(e), e._connectedPoint = this, this.onConnectionObservable.notifyObservers(e), e.onConnectionObservable.notifyObservers(this), this; } /** * Disconnect this point from one of his endpoint * @param endpoint defines the other connection point * @returns the current connection point */ disconnectFrom(e) { const t = this._endpoints.indexOf(e); return t === -1 ? this : (this._endpoints.splice(t, 1), e._connectedPoint = null, this); } /** * Fill the list of excluded connection point types with all types other than those passed in the parameter * @param mask Types (ORed values of NodeMaterialBlockConnectionPointTypes) that are allowed, and thus will not be pushed to the excluded list */ addExcludedConnectionPointFromAllowedTypes(e) { let t = 1; for (; t < Me.All; ) e & t || this.excludedConnectionPointTypes.push(t), t = t << 1; } /** * Serializes this point in a JSON representation * @param isInput defines if the connection point is an input (default is true) * @returns the serialized point object */ serialize(e = !0) { const t = {}; return t.name = this.name, t.displayName = this.displayName, this.value !== void 0 && this.value !== null && (this.value.asArray ? (t.valueType = "BABYLON." + this.value.getClassName(), t.value = this.value.asArray()) : (t.valueType = "number", t.value = this.value)), e && this.connectedPoint && (t.inputName = this.name, t.targetBlockId = this.connectedPoint.ownerBlock.uniqueId, t.targetConnectionName = this.connectedPoint.name), t; } /** * Release resources */ dispose() { this.onConnectionObservable.clear(); } } class yi { /** * Gets the time spent to build this block (in ms) */ get buildExecutionTime() { return this._buildExecutionTime; } /** * Gets the list of input points */ get inputs() { return this._inputs; } /** Gets the list of output points */ get outputs() { return this._outputs; } /** * Gets or set the name of the block */ get name() { return this._name; } set name(e) { this._name = e; } /** * Gets a boolean indicating if this block is an input */ get isInput() { return this._isInput; } /** * Gets a boolean indicating if this block is a teleport out */ get isTeleportOut() { return this._isTeleportOut; } /** * Gets a boolean indicating if this block is a teleport in */ get isTeleportIn() { return this._isTeleportIn; } /** * Gets a boolean indicating if this block is a debug block */ get isDebug() { return this._isDebug; } /** * Gets a boolean indicating that this block can only be used once per NodeGeometry */ get isUnique() { return this._isUnique; } /** * Gets the current class name e.g. "NodeGeometryBlock" * @returns the class name */ getClassName() { return "NodeGeometryBlock"; } _inputRename(e) { return e; } _outputRename(e) { return e; } /** * Checks if the current block is an ancestor of a given block * @param block defines the potential descendant block to check * @returns true if block is a descendant */ isAnAncestorOf(e) { for (const t of this._outputs) if (t.hasEndpoints) { for (const r of t.endpoints) if (r.ownerBlock === e || r.ownerBlock.isAnAncestorOf(e)) return !0; } return !1; } /** * Checks if the current block is an ancestor of a given type * @param type defines the potential type to check * @returns true if block is a descendant */ isAnAncestorOfType(e) { if (this.getClassName() === e) return !0; for (const t of this._outputs) if (t.hasEndpoints) { for (const r of t.endpoints) if (r.ownerBlock.isAnAncestorOfType(e)) return !0; } return !1; } /** * Get the first descendant using a predicate * @param predicate defines the predicate to check * @returns descendant or null if none found */ getDescendantOfPredicate(e) { if (e(this)) return this; for (const t of this._outputs) if (t.hasEndpoints) for (const r of t.endpoints) { const n = r.ownerBlock.getDescendantOfPredicate(e); if (n) return n; } return null; } /** * Creates a new NodeGeometryBlock * @param name defines the block name */ constructor(e) { this._name = "", this._isInput = !1, this._isTeleportOut = !1, this._isTeleportIn = !1, this._isDebug = !1, this._isUnique = !1, this._buildExecutionTime = 0, this.onBuildObservable = new Oe(), this._inputs = new Array(), this._outputs = new Array(), this._codeVariableName = "", this.visibleOnFrame = !1, this._name = e, this.uniqueId = bR.UniqueId; } /** * Register a new input. Must be called inside a block constructor * @param name defines the connection point name * @param type defines the connection point type * @param isOptional defines a boolean indicating that this input can be omitted * @param value value to return if there is no connection * @param valueMin min value accepted for value * @param valueMax max value accepted for value * @returns the current block */ registerInput(e, t, r = !1, n, i, s) { const a = new qF(e, this, ES.Input); return a.type = t, a.isOptional = r, a.defaultValue = n, a.value = n, a.valueMin = i, a.valueMax = s, this._inputs.push(a), this; } /** * Register a new output. Must be called inside a block constructor * @param name defines the connection point name * @param type defines the connection point type * @param point an already created connection point. If not provided, create a new one * @returns the current block */ registerOutput(e, t, r) { return r = r ?? new qF(e, this, ES.Output), r.type = t, this._outputs.push(r), this; } // eslint-disable-next-line @typescript-eslint/no-unused-vars _buildBlock(e) { } // eslint-disable-next-line @typescript-eslint/no-unused-vars _customBuildStep(e) { } /** * Build the current node and generate the vertex data * @param state defines the current generation state * @returns true if already built */ build(e) { if (this._buildId === e.buildId) return !0; if (this._outputs.length > 0) { if (!this._outputs.some((r) => r.hasEndpoints) && !this.isDebug) return !1; this.outputs.forEach((r) => r._resetCounters()); } this._buildId = e.buildId; for (const r of this._inputs) { if (!r.connectedPoint) { r.isOptional || e.notConnectedNonOptionalInputs.push(r); continue; } const n = r.connectedPoint.ownerBlock; n && n !== this && n.build(e); } this._customBuildStep(e), e.verbose && console.log(`Building ${this.name} [${this.getClassName()}]`); const t = Yi.Now; this._buildBlock(e), this._buildExecutionTime = Yi.Now - t; for (const r of this._outputs) for (const n of r.endpoints) { const i = n.ownerBlock; i && i.build(e); } return this.onBuildObservable.notifyObservers(this), !1; } _linkConnectionTypes(e, t, r = !1) { r ? this._inputs[t]._acceptedConnectionPointType = this._inputs[e] : this._inputs[e]._linkedConnectionSource = this._inputs[t], this._inputs[t]._linkedConnectionSource = this._inputs[e]; } /** * Initialize the block and prepare the context for build */ initialize() { } /** * Lets the block try to connect some inputs automatically */ autoConfigure() { } /** * Find an input by its name * @param name defines the name of the input to look for * @returns the input or null if not found */ getInputByName(e) { const t = this._inputs.filter((r) => r.name === e); return t.length ? t[0] : null; } /** * Find an output by its name * @param name defines the name of the output to look for * @returns the output or null if not found */ getOutputByName(e) { const t = this._outputs.filter((r) => r.name === e); return t.length ? t[0] : null; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = {}; e.customType = "BABYLON." + this.getClassName(), e.id = this.uniqueId, e.name = this.name, e.inputs = [], e.outputs = []; for (const t of this.inputs) e.inputs.push(t.serialize()); for (const t of this.outputs) e.outputs.push(t.serialize(!1)); return e; } /** * @internal */ _deserialize(e) { this._name = e.name, this.comments = e.comments, this.visibleOnFrame = !!e.visibleOnFrame, this._deserializePortDisplayNamesAndExposedOnFrame(e); } _deserializePortDisplayNamesAndExposedOnFrame(e) { const t = e.inputs, r = e.outputs; t && t.forEach((n) => { const i = this.inputs.find((s) => s.name === n.name); if (i && (n.displayName && (i.displayName = n.displayName), n.isExposedOnFrame && (i.isExposedOnFrame = n.isExposedOnFrame, i.exposedPortPosition = n.exposedPortPosition), n.value !== void 0 && n.value !== null)) if (n.valueType === "number") i.value = n.value; else { const s = Jo(n.valueType); s && (i.value = s.FromArray(n.value)); } }), r && r.forEach((n, i) => { n.displayName && (this.outputs[i].displayName = n.displayName), n.isExposedOnFrame && (this.outputs[i].isExposedOnFrame = n.isExposedOnFrame, this.outputs[i].exposedPortPosition = n.exposedPortPosition); }); } _dumpPropertiesCode() { return `${this._codeVariableName}.visibleOnFrame = ${this.visibleOnFrame}; `; } /** * @internal */ _dumpCodeForOutputConnections(e) { let t = ""; if (e.indexOf(this) !== -1) return t; e.push(this); for (const r of this.inputs) { if (!r.isConnected) continue; const n = r.connectedPoint, i = n.ownerBlock; t += i._dumpCodeForOutputConnections(e), t += `${i._codeVariableName}.${i._outputRename(n.name)}.connectTo(${this._codeVariableName}.${this._inputRename(r.name)}); `; } return t; } /** * @internal */ _dumpCode(e, t) { t.push(this); const r = this.name.replace(/[^A-Za-z_]+/g, ""); if (this._codeVariableName = r || `${this.getClassName()}_${this.uniqueId}`, e.indexOf(this._codeVariableName) !== -1) { let i = 0; do i++, this._codeVariableName = r + i; while (e.indexOf(this._codeVariableName) !== -1); } e.push(this._codeVariableName); let n = ` // ${this.getClassName()} `; this.comments && (n += `// ${this.comments} `), n += `var ${this._codeVariableName} = new BABYLON.${this.getClassName()}("${this.name}"); `, n += this._dumpPropertiesCode(); for (const i of this.inputs) { if (!i.isConnected) continue; const a = i.connectedPoint.ownerBlock; t.indexOf(a) === -1 && (n += a._dumpCode(e, t)); } for (const i of this.outputs) if (i.hasEndpoints) for (const s of i.endpoints) { const a = s.ownerBlock; a && t.indexOf(a) === -1 && (n += a._dumpCode(e, t)); } return n; } /** * Clone the current block to a new identical block * @returns a copy of the current block */ clone() { const e = this.serialize(), t = Jo(e.customType); if (t) { const r = new t(); return r._deserialize(e), r; } return null; } /** * Release resources */ dispose() { for (const e of this.inputs) e.dispose(); for (const e of this.outputs) e.dispose(); this.onBuildObservable.clear(); } } C([ M("comment") ], yi.prototype, "comments", void 0); class vY extends yi { /** * Gets the current vertex data if the graph was successfully built */ get currentVertexData() { return this._vertexData; } /** * Create a new GeometryOutputBlock * @param name defines the block name */ constructor(e) { super(e), this._vertexData = null, this._isUnique = !0, this.registerInput("geometry", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "GeometryOutputBlock"; } /** * Gets the geometry input component */ get geometry() { return this._inputs[0]; } _buildBlock(e) { e.vertexData = this.geometry.getConnectedValue(e), this._vertexData = e.vertexData; } } Ue("BABYLON.GeometryOutputBlock", vY); var Ws; (function(A) { A[A.None = 0] = "None", A[A.Positions = 1] = "Positions", A[A.Normals = 2] = "Normals", A[A.Tangents = 3] = "Tangents", A[A.UV = 4] = "UV", A[A.UV2 = 5] = "UV2", A[A.UV3 = 6] = "UV3", A[A.UV4 = 7] = "UV4", A[A.UV5 = 8] = "UV5", A[A.UV6 = 9] = "UV6", A[A.Colors = 10] = "Colors", A[A.VertexID = 11] = "VertexID", A[A.FaceID = 12] = "FaceID", A[A.GeometryID = 13] = "GeometryID", A[A.CollectionID = 14] = "CollectionID", A[A.LoopID = 15] = "LoopID", A[A.InstanceID = 16] = "InstanceID"; })(Ws || (Ws = {})); class Xne { constructor() { this._rotationMatrix = new he(), this._scalingMatrix = new he(), this._positionMatrix = new he(), this._scalingRotationMatrix = new he(), this._transformMatrix = new he(), this._tempVector3 = new S(), this.notConnectedNonOptionalInputs = [], this.noContextualData = [], this.vertexData = null, this._geometryContext = null, this._executionContext = null, this._instancingContext = null, this._geometryContextStack = [], this._executionContextStack = [], this._instancingContextStack = []; } /** Gets or sets the geometry context */ get geometryContext() { return this._geometryContext; } /** Gets or sets the execution context */ get executionContext() { return this._executionContext; } /** Gets or sets the instancing context */ get instancingContext() { return this._instancingContext; } /** * Push the new active geometry context * @param geometryContext defines the geometry context */ pushGeometryContext(e) { this._geometryContext = e, this._geometryContextStack.push(this._geometryContext); } /** * Push the new active execution context * @param executionContext defines the execution context */ pushExecutionContext(e) { this._executionContext = e, this._executionContextStack.push(this._executionContext); } /** * Push the new active instancing context * @param instancingContext defines the instancing context */ pushInstancingContext(e) { this._instancingContext = e, this._instancingContextStack.push(this._instancingContext); } /** * Remove current geometry context and restore the previous one */ restoreGeometryContext() { this._geometryContextStack.pop(), this._geometryContext = this._geometryContextStack.length > 0 ? this._geometryContextStack[this._geometryContextStack.length - 1] : null; } /** * Remove current execution context and restore the previous one */ restoreExecutionContext() { this._executionContextStack.pop(), this._executionContext = this._executionContextStack.length > 0 ? this._executionContextStack[this._executionContextStack.length - 1] : null; } /** * Remove current isntancing context and restore the previous one */ restoreInstancingContext() { this._instancingContextStack.pop(), this._instancingContext = this._instancingContextStack.length > 0 ? this._instancingContextStack[this._instancingContextStack.length - 1] : null; } /** * Gets the value associated with a contextual source * @param source Source of the contextual value * @param skipWarning Do not store the warning for reporting if true * @returns the value associated with the source */ getContextualValue(e, t = !1) { if (!this.executionContext) return t || this.noContextualData.push(e), null; const r = this.executionContext.getExecutionIndex(); switch (e) { case Ws.Positions: return this.executionContext.getOverridePositionsContextualValue ? this.executionContext.getOverridePositionsContextualValue() : !this.geometryContext || !this.geometryContext.positions ? S.Zero() : S.FromArray(this.geometryContext.positions, r * 3); case Ws.Normals: return this.executionContext.getOverrideNormalsContextualValue ? this.executionContext.getOverrideNormalsContextualValue() : !this.geometryContext || !this.geometryContext.normals ? S.Zero() : S.FromArray(this.geometryContext.normals, r * 3); case Ws.Colors: return !this.geometryContext || !this.geometryContext.colors ? Ir.Zero() : Ir.FromArray(this.geometryContext.colors, r * 4); case Ws.Tangents: return !this.geometryContext || !this.geometryContext.tangents ? Ir.Zero() : Ir.FromArray(this.geometryContext.tangents, r * 4); case Ws.UV: return this.executionContext.getOverrideUVs1ContextualValue ? this.executionContext.getOverrideUVs1ContextualValue() : !this.geometryContext || !this.geometryContext.uvs ? at.Zero() : at.FromArray(this.geometryContext.uvs, r * 2); case Ws.UV2: return !this.geometryContext || !this.geometryContext.uvs2 ? at.Zero() : at.FromArray(this.geometryContext.uvs2, r * 2); case Ws.UV3: return !this.geometryContext || !this.geometryContext.uvs3 ? at.Zero() : at.FromArray(this.geometryContext.uvs3, r * 2); case Ws.UV4: return !this.geometryContext || !this.geometryContext.uvs4 ? at.Zero() : at.FromArray(this.geometryContext.uvs4, r * 2); case Ws.UV5: return !this.geometryContext || !this.geometryContext.uvs5 ? at.Zero() : at.FromArray(this.geometryContext.uvs5, r * 2); case Ws.UV6: return !this.geometryContext || !this.geometryContext.uvs6 ? at.Zero() : at.FromArray(this.geometryContext.uvs6, r * 2); case Ws.VertexID: return r; case Ws.FaceID: return this.executionContext.getExecutionFaceIndex(); case Ws.LoopID: return this.executionContext.getExecutionLoopIndex(); case Ws.InstanceID: return this.instancingContext ? this.instancingContext.getInstanceIndex() : 0; case Ws.GeometryID: return this.geometryContext ? this.geometryContext.uniqueId : 0; case Ws.CollectionID: return !this.geometryContext || !this.geometryContext.metadata ? 0 : this.geometryContext.metadata.collectionId || 0; } return null; } /** * Adapt a value to a target type * @param source defines the value to adapt * @param targetType defines the target type * @returns the adapted value */ adapt(e, t) { const r = e.getConnectedValue(this) || 0; if (e.type === t) return r; switch (t) { case Me.Vector2: return new at(r, r); case Me.Vector3: return new S(r, r, r); case Me.Vector4: return new Ir(r, r, r, r); } return null; } /** * Adapt an input value to a target type * @param source defines the value to adapt * @param targetType defines the target type * @param defaultValue defines the default value to use if not connected * @returns the adapted value */ adaptInput(e, t, r) { var n; if (!e.isConnected) return e.value || r; const i = e.getConnectedValue(this); if (((n = e._connectedPoint) === null || n === void 0 ? void 0 : n.type) === t) return i; switch (t) { case Me.Vector2: return new at(i, i); case Me.Vector3: return new S(i, i, i); case Me.Vector4: return new Ir(i, i, i, i); } return null; } /** * Emits console errors and exceptions if there is a failing check */ emitErrors() { let e = ""; for (const t of this.notConnectedNonOptionalInputs) e += `input ${t.name} from block ${t.ownerBlock.name}[${t.ownerBlock.getClassName()}] is not connected and is not optional. `; for (const t of this.noContextualData) e += `Contextual input ${Ws[t]} has no context to pull data from (must be connected to a setXXX block or a instantiateXXX block). `; if (e) throw `Build of NodeGeometry failed: ` + e; } /** @internal */ _instantiate(e, t, r, n, i) { he.ScalingToRef(n.x, n.y, n.z, this._scalingMatrix), he.RotationYawPitchRollToRef(r.y, r.x, r.z, this._rotationMatrix), he.TranslationToRef(t.x, t.y, t.z, this._positionMatrix), this._scalingMatrix.multiplyToRef(this._rotationMatrix, this._scalingRotationMatrix), this._scalingRotationMatrix.multiplyToRef(this._positionMatrix, this._transformMatrix); for (let s = 0; s < e.positions.length; s += 3) this._tempVector3.fromArray(e.positions, s), S.TransformCoordinatesToRef(this._tempVector3, this._transformMatrix, this._tempVector3), this._tempVector3.toArray(e.positions, s), e.normals && (this._tempVector3.fromArray(e.normals, s), S.TransformNormalToRef(this._tempVector3, this._scalingRotationMatrix, this._tempVector3), this._tempVector3.toArray(e.normals, s)); i.push(e); } /** @internal */ _instantiateWithMatrix(e, t, r) { for (let n = 0; n < e.positions.length; n += 3) this._tempVector3.fromArray(e.positions, n), S.TransformCoordinatesToRef(this._tempVector3, t, this._tempVector3), this._tempVector3.toArray(e.positions, n), e.normals && (this._tempVector3.fromArray(e.normals, n), S.TransformNormalToRef(this._tempVector3, t, this._tempVector3), this._tempVector3.toArray(e.normals, n)); r.push(e); } /** @internal */ _instantiateWithPositionAndMatrix(e, t, r, n) { he.TranslationToRef(t.x, t.y, t.z, this._positionMatrix), r.multiplyToRef(this._positionMatrix, this._transformMatrix); for (let i = 0; i < e.positions.length; i += 3) this._tempVector3.fromArray(e.positions, i), S.TransformCoordinatesToRef(this._tempVector3, this._transformMatrix, this._tempVector3), this._tempVector3.toArray(e.positions, i), e.normals && (this._tempVector3.fromArray(e.normals, i), S.TransformNormalToRef(this._tempVector3, this._transformMatrix, this._tempVector3), this._tempVector3.toArray(e.normals, i)); n.push(e); } } class xf extends yi { /** * Gets or sets the connection point type (default is float) */ get type() { if (this._type === Me.AutoDetect && this.value != null) { if (!isNaN(this.value)) return this._type = Me.Float, this._type; switch (this.value.getClassName()) { case "Vector2": return this._type = Me.Vector2, this._type; case "Vector3": return this._type = Me.Vector3, this._type; case "Vector4": return this._type = Me.Vector4, this._type; case "Matrix": return this._type = Me.Matrix, this._type; } } return this._type; } /** * Gets a boolean indicating that the current connection point is a contextual value */ get isContextual() { return this._contextualSource !== Ws.None; } /** * Gets or sets the current contextual value */ get contextualValue() { return this._contextualSource; } set contextualValue(e) { switch (this._contextualSource = e, e) { case Ws.Positions: case Ws.Normals: this._type = Me.Vector3; break; case Ws.Colors: case Ws.Tangents: this._type = Me.Vector4; break; case Ws.UV: case Ws.UV2: case Ws.UV3: case Ws.UV4: case Ws.UV5: case Ws.UV6: this._type = Me.Vector2; break; case Ws.VertexID: case Ws.GeometryID: case Ws.CollectionID: case Ws.FaceID: case Ws.LoopID: case Ws.InstanceID: this._type = Me.Int; break; } this.output && (this.output.type = this._type); } /** * Creates a new InputBlock * @param name defines the block name * @param type defines the type of the input (can be set to NodeGeometryBlockConnectionPointTypes.AutoDetect) */ constructor(e, t = Me.AutoDetect) { super(e), this._type = Me.Undefined, this._contextualSource = Ws.None, this.min = 0, this.max = 0, this.groupInInspector = "", this.onValueChangedObservable = new Oe(), this._type = t, this._isInput = !0, this.setDefaultValue(), this.registerOutput("output", t); } /** * Gets or sets the value of that point. * Please note that this value will be ignored if valueCallback is defined */ get value() { return this._storedValue; } set value(e) { this.type === Me.Float && this.min !== this.max && (e = Math.max(this.min, e), e = Math.min(this.max, e)), this._storedValue = e, this.onValueChangedObservable.notifyObservers(this); } /** * Gets or sets a callback used to get the value of that point. * Please note that setting this value will force the connection point to ignore the value property */ get valueCallback() { return this._valueCallback; } set valueCallback(e) { this._valueCallback = e; } /** * Gets the current class name * @returns the class name */ getClassName() { return "GeometryInputBlock"; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } /** * Set the input block to its default value (based on its type) */ setDefaultValue() { switch (this.contextualValue = Ws.None, this.type) { case Me.Int: case Me.Float: this.value = 0; break; case Me.Vector2: this.value = at.Zero(); break; case Me.Vector3: this.value = S.Zero(); break; case Me.Vector4: this.value = Ir.Zero(); break; case Me.Matrix: this.value = he.Identity(); break; } } _buildBlock(e) { super._buildBlock(e), this.isContextual ? (this.output._storedValue = null, this.output._storedFunction = (t) => t.getContextualValue(this._contextualSource)) : (this.output._storedFunction = null, this.output._storedValue = this.value); } dispose() { this.onValueChangedObservable.clear(), super.dispose(); } _dumpPropertiesCode() { const e = this._codeVariableName; if (this.isContextual) return super._dumpPropertiesCode() + `${e}.contextualValue = BABYLON.NodeGeometryContextualSources.${Ws[this._contextualSource]}; `; const t = []; let r = ""; switch (this.type) { case Me.Float: case Me.Int: r = `${this.value}`; break; case Me.Vector2: r = `new BABYLON.Vector2(${this.value.x}, ${this.value.y})`; break; case Me.Vector3: r = `new BABYLON.Vector3(${this.value.x}, ${this.value.y}, ${this.value.z})`; break; case Me.Vector4: r = `new BABYLON.Vector4(${this.value.x}, ${this.value.y}, ${this.value.z}, ${this.value.w})`; break; } return t.push(`${e}.value = ${r}`), (this.type === Me.Float || this.type === Me.Int) && t.push(`${e}.min = ${this.min}`, `${e}.max = ${this.max}`), t.push(""), super._dumpPropertiesCode() + t.join(`; `); } serialize() { const e = super.serialize(); return e.type = this.type, e.contextualValue = this.contextualValue, e.min = this.min, e.max = this.max, e.groupInInspector = this.groupInInspector, this._storedValue !== null && !this.isContextual && (this._storedValue.asArray ? (e.valueType = "BABYLON." + this._storedValue.getClassName(), e.value = this._storedValue.asArray()) : (e.valueType = "number", e.value = this._storedValue)), e; } _deserialize(e) { if (super._deserialize(e), this._type = e.type, this.contextualValue = e.contextualValue, this.min = e.min || 0, this.max = e.max || 0, this.groupInInspector = e.groupInInspector || "", !!e.valueType) if (e.valueType === "number") this._storedValue = e.value; else { const t = Jo(e.valueType); t && (this._storedValue = t.FromArray(e.value)); } } } Ue("BABYLON.GeometryInputBlock", xf); class Xy extends yi { /** * Create a new BoxBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !1, this.registerInput("size", Me.Float, !0, 1), this.registerInput("width", Me.Float, !0, 0), this.registerInput("height", Me.Float, !0, 0), this.registerInput("depth", Me.Float, !0, 0), this.registerInput("subdivisions", Me.Int, !0, 1), this.registerInput("subdivisionsX", Me.Int, !0, 0), this.registerInput("subdivisionsY", Me.Int, !0, 0), this.registerInput("subdivisionsZ", Me.Int, !0, 0), this.registerOutput("geometry", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "BoxBlock"; } /** * Gets the size input component */ get size() { return this._inputs[0]; } /** * Gets the width input component */ get width() { return this._inputs[1]; } /** * Gets the height input component */ get height() { return this._inputs[2]; } /** * Gets the depth input component */ get depth() { return this._inputs[3]; } /** * Gets the subdivisions input component */ get subdivisions() { return this._inputs[4]; } /** * Gets the subdivisionsX input component */ get subdivisionsX() { return this._inputs[5]; } /** * Gets the subdivisionsY input component */ get subdivisionsY() { return this._inputs[6]; } /** * Gets the subdivisionsZ input component */ get subdivisionsZ() { return this._inputs[7]; } /** * Gets the geometry output component */ get geometry() { return this._outputs[0]; } autoConfigure() { if (!this.size.isConnected) { if (!this.width.isConnected && !this.height.isConnected && !this.depth.isConnected) { const e = new xf("Size"); e.value = 1, e.output.connectTo(this.size); return; } if (!this.width.isConnected) { const e = new xf("Width"); e.value = 1, e.output.connectTo(this.width); } if (!this.height.isConnected) { const e = new xf("Height"); e.value = 1, e.output.connectTo(this.height); } if (!this.depth.isConnected) { const e = new xf("Depth"); e.value = 1, e.output.connectTo(this.depth); } } } _buildBlock(e) { const t = {}, r = (n) => { t.size = this.size.getConnectedValue(n), t.width = this.width.getConnectedValue(n), t.height = this.height.getConnectedValue(n), t.depth = this.depth.getConnectedValue(n); const i = this.subdivisions.getConnectedValue(n), s = this.subdivisionsX.getConnectedValue(n), a = this.subdivisionsY.getConnectedValue(n), f = this.subdivisionsZ.getConnectedValue(n); return i && (t.segments = i), s && (t.widthSegments = s), a && (t.heightSegments = a), f && (t.depthSegments = f), $ee(t); }; if (this.evaluateContext) this.geometry._storedFunction = r; else { const n = r(e); this.geometry._storedFunction = () => (this.geometry._executionCount = 1, n.clone()); } } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), this.evaluateContext = e.evaluateContext; } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], Xy.prototype, "evaluateContext", void 0); Ue("BABYLON.BoxBlock", Xy); class Ql { /** Get the inspector from bundle or global */ _getGlobalNodeGeometryEditor() { if (typeof NODEGEOMETRYEDITOR < "u") return NODEGEOMETRYEDITOR; if (typeof BABYLON < "u" && typeof BABYLON.NodeGeometryEditor < "u") return BABYLON; } /** * Gets the time spent to build this block (in ms) */ get buildExecutionTime() { return this._buildExecutionTime; } /** * Creates a new geometry * @param name defines the name of the geometry */ constructor(e) { this._buildId = Ql._BuildIdGenerator++, this._buildWasSuccessful = !1, this._vertexData = null, this._buildExecutionTime = 0, this.BJSNODEGEOMETRYEDITOR = this._getGlobalNodeGeometryEditor(), this.editorData = null, this.attachedBlocks = [], this.onBuildObservable = new Oe(), this.outputBlock = null, this.name = e; } /** * Gets the current class name of the geometry e.g. "NodeGeometry" * @returns the class name */ getClassName() { return "NodeGeometry"; } /** * Get a block by its name * @param name defines the name of the block to retrieve * @returns the required block or null if not found */ getBlockByName(e) { let t = null; for (const r of this.attachedBlocks) if (r.name === e) if (!t) t = r; else return ye.Warn("More than one block was found with the name `" + e + "`"), t; return t; } /** * Get a block using a predicate * @param predicate defines the predicate used to find the good candidate * @returns the required block or null if not found */ getBlockByPredicate(e) { for (const t of this.attachedBlocks) if (e(t)) return t; return null; } /** * Gets the list of input blocks attached to this material * @returns an array of InputBlocks */ getInputBlocks() { const e = []; for (const t of this.attachedBlocks) t.isInput && e.push(t); return e; } /** * Launch the node geometry editor * @param config Define the configuration of the editor * @returns a promise fulfilled when the node editor is visible */ edit(e) { return new Promise((t) => { if (this.BJSNODEGEOMETRYEDITOR = this.BJSNODEGEOMETRYEDITOR || this._getGlobalNodeGeometryEditor(), typeof this.BJSNODEGEOMETRYEDITOR > "u") { const r = e && e.editorURL ? e.editorURL : Ql.EditorURL; ye.LoadBabylonScript(r, () => { this.BJSNODEGEOMETRYEDITOR = this.BJSNODEGEOMETRYEDITOR || this._getGlobalNodeGeometryEditor(), this._createNodeEditor(e == null ? void 0 : e.nodeGeometryEditorConfig), t(); }); } else this._createNodeEditor(e == null ? void 0 : e.nodeGeometryEditorConfig), t(); }); } /** Creates the node editor window. */ _createNodeEditor(e) { const t = Object.assign({ nodeGeometry: this }, e); this.BJSNODEGEOMETRYEDITOR.NodeGeometryEditor.Show(t); } /** * Build the final geometry * @param verbose defines if the build should log activity * @param updateBuildId defines if the internal build Id should be updated (default is true) * @param autoConfigure defines if the autoConfigure method should be called when initializing blocks (default is false) */ build(e = !1, t = !0, r = !1) { if (this._buildWasSuccessful = !1, !this.outputBlock) throw "You must define the outputBlock property before building the geometry"; const n = Yi.Now; this._initializeBlock(this.outputBlock, r); const i = new Xne(); i.buildId = this._buildId, i.verbose = e, this.outputBlock.build(i), t && (this._buildId = Ql._BuildIdGenerator++), this._buildExecutionTime = Yi.Now - n, i.emitErrors(), this._buildWasSuccessful = !0, this._vertexData = i.vertexData, this.onBuildObservable.notifyObservers(this); } /** * Creates a mesh from the geometry blocks * @param name defines the name of the mesh * @param scene The scene the mesh is scoped to * @returns The new mesh */ createMesh(e, t = null) { if (this._buildWasSuccessful || this.build(), !this._vertexData) return null; const r = new Ee(e, t); return this._vertexData.applyToMesh(r), r._internalMetadata = r._internalMetadata || {}, r._internalMetadata.nodeGeometry = this, r; } /** * Creates a mesh from the geometry blocks * @param mesh the mesh to update * @returns True if successfully updated */ updateMesh(e) { return this._buildWasSuccessful || this.build(), this._vertexData ? (this._vertexData.applyToMesh(e), e._internalMetadata = e._internalMetadata || {}, e._internalMetadata.nodeGeometry = this, e) : !1; } _initializeBlock(e, t = !0) { e.initialize(), t && e.autoConfigure(), e._preparationId = this._buildId, this.attachedBlocks.indexOf(e) === -1 && this.attachedBlocks.push(e); for (const r of e.inputs) { const n = r.connectedPoint; if (n) { const i = n.ownerBlock; i !== e && this._initializeBlock(i, t); } } } /** * Clear the current geometry */ clear() { this.outputBlock = null, this.attachedBlocks.length = 0; } /** * Remove a block from the current geometry * @param block defines the block to remove */ removeBlock(e) { const t = this.attachedBlocks.indexOf(e); t > -1 && this.attachedBlocks.splice(t, 1), e === this.outputBlock && (this.outputBlock = null); } /** * Clear the current graph and load a new one from a serialization object * @param source defines the JSON representation of the geometry * @param merge defines whether or not the source must be merged or replace the current content */ parseSerializedObject(e, t = !1) { t || this.clear(); const r = {}; for (const n of e.blocks) { const i = Jo(n.customType); if (i) { const s = new i(); s._deserialize(n), r[n.id] = s, this.attachedBlocks.push(s); } } for (const n of this.attachedBlocks) if (n.isTeleportOut) { const i = n, s = i._tempEntryPointUniqueId; if (s) { const a = r[s]; a && a.attachToEndpoint(i); } } for (let n = 0; n < e.blocks.length; n++) { const i = e.blocks[n], s = r[i.id]; s && (s.inputs.length && i.inputs.some((a) => a.targetConnectionName) && !t || this._restoreConnections(s, e, r)); } if (e.outputNodeId && (this.outputBlock = r[e.outputNodeId]), e.locations || e.editorData && e.editorData.locations) { const n = e.locations || e.editorData.locations; for (const s of n) r[s.blockId] && (s.blockId = r[s.blockId].uniqueId); t && this.editorData && this.editorData.locations && n.concat(this.editorData.locations), e.locations ? this.editorData = { locations: n } : (this.editorData = e.editorData, this.editorData.locations = n); const i = []; for (const s in r) i[s] = r[s].uniqueId; this.editorData.map = i; } this.comment = e.comment; } _restoreConnections(e, t, r) { for (const n of e.outputs) for (const i of t.blocks) { const s = r[i.id]; if (s) { for (const a of i.inputs) if (r[a.targetBlockId] === e && a.targetConnectionName === n.name) { const f = s.getInputByName(a.inputName); if (!f || f.isConnected) continue; n.connectTo(f, !0), this._restoreConnections(s, t, r); continue; } } } } /** * Generate a string containing the code declaration required to create an equivalent of this geometry * @returns a string */ generateCode() { let e = []; const t = [], r = ["const", "var", "let"]; this.outputBlock && this._gatherBlocks(this.outputBlock, t); let n = `let nodeGeometry = new BABYLON.NodeGeometry("${this.name || "node geometry"}"); `; for (const i of t) i.isInput && e.indexOf(i) === -1 && (n += i._dumpCode(r, e)); return this.outputBlock && (e = [], n += `// Connections `, n += this.outputBlock._dumpCodeForOutputConnections(e), n += `// Output nodes `, n += `nodeGeometry.outputBlock = ${this.outputBlock._codeVariableName}; `, n += `nodeGeometry.build(); `), n; } _gatherBlocks(e, t) { if (t.indexOf(e) === -1) { t.push(e); for (const r of e.inputs) { const n = r.connectedPoint; if (n) { const i = n.ownerBlock; i !== e && this._gatherBlocks(i, t); } } if (e.isTeleportOut) { const r = e; r.entryPoint && this._gatherBlocks(r.entryPoint, t); } } } /** * Clear the current geometry and set it to a default state */ setToDefault() { this.clear(), this.editorData = null; const e = new Xy("Box"); e.autoConfigure(); const t = new vY("Geometry Output"); e.geometry.connectTo(t.geometry), this.outputBlock = t; } /** * Makes a duplicate of the current geometry. * @param name defines the name to use for the new geometry */ clone(e) { const t = this.serialize(), r = jt.Clone(() => new Ql(e), this); return r.name = e, r.parseSerializedObject(t), r._buildId = this._buildId, r.build(!1), r; } /** * Serializes this geometry in a JSON representation * @param selectedBlocks defines the list of blocks to save (if null the whole geometry will be saved) * @returns the serialized geometry object */ serialize(e) { const t = e ? {} : jt.Serialize(this); t.editorData = JSON.parse(JSON.stringify(this.editorData)); let r = []; e ? r = e : (t.customType = "BABYLON.NodeGeometry", this.outputBlock && (t.outputNodeId = this.outputBlock.uniqueId)), t.blocks = []; for (const n of r) t.blocks.push(n.serialize()); if (!e) for (const n of this.attachedBlocks) r.indexOf(n) === -1 && t.blocks.push(n.serialize()); return t; } /** * Disposes the ressources */ dispose() { for (const e of this.attachedBlocks) e.dispose(); this.attachedBlocks.length = 0, this.onBuildObservable.clear(); } /** * Creates a new node geometry set to default basic configuration * @param name defines the name of the geometry * @returns a new NodeGeometry */ static CreateDefault(e) { const t = new Ql(e); return t.setToDefault(), t.build(), t; } /** * Creates a node geometry from parsed geometry data * @param source defines the JSON representation of the geometry * @returns a new node geometry */ static Parse(e) { const t = jt.Parse(() => new Ql(e.name), e, null); return t.parseSerializedObject(e), t.build(), t; } /** * Creates a node geometry from a snippet saved by the node geometry editor * @param snippetId defines the snippet to load * @param nodeGeometry defines a node geometry to update (instead of creating a new one) * @param skipBuild defines whether to build the node geometry * @returns a promise that will resolve to the new node geometry */ static ParseFromSnippetAsync(e, t, r = !1) { return e === "_BLANK" ? Promise.resolve(Ql.CreateDefault("blank")) : new Promise((n, i) => { const s = new ho(); s.addEventListener("readystatechange", () => { if (s.readyState == 4) if (s.status == 200) { const a = JSON.parse(JSON.parse(s.responseText).jsonPayload), f = JSON.parse(a.nodeGeometry); t || (t = jt.Parse(() => new Ql(e), f, null)), t.parseSerializedObject(f), t.snippetId = e; try { r || t.build(), n(t); } catch (o) { i(o); } } else i("Unable to load the snippet " + e); }), s.open("GET", this.SnippetUrl + "/" + e.replace(/#/g, "/")), s.send(); }); } } Ql._BuildIdGenerator = 0; Ql.EditorURL = `${ye._DefaultCdnUrl}/v${Ge.Version}/nodeGeometryEditor/babylon.nodeGeometryEditor.js`; Ql.SnippetUrl = "https://snippet.babylonjs.com"; C([ M() ], Ql.prototype, "name", void 0); C([ M("comment") ], Ql.prototype, "comment", void 0); class Ty extends yi { /** * Creates a new GeometryOptimizeBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !0, this.epsilon = Dn, this.registerInput("geometry", Me.Geometry), this.registerOutput("output", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "GeometryOptimizeBlock"; } /** * Gets the geometry component */ get geometry() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { const t = (r) => { if (!this.geometry.isConnected) return null; const n = this.geometry.getConnectedValue(r), i = [], s = {}; for (let f = 0; f < n.positions.length; f += 3) { const o = n.positions[f], d = n.positions[f + 1], v = n.positions[f + 2]; let u = !1; for (let l = 0; l < i.length; l += 3) if (Xt.WithinEpsilon(o, i[l], this.epsilon) && Xt.WithinEpsilon(d, i[l + 1], this.epsilon) && Xt.WithinEpsilon(v, i[l + 2], this.epsilon)) { s[f / 3] = l / 3, u = !0; continue; } u || (s[f / 3] = i.length / 3, i.push(o, d, v)); } const a = new Ut(); return a.positions = i, a.indices = n.indices.map((f) => s[f]), a; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } _dumpPropertiesCode() { let e = super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; return e += `${this._codeVariableName}.epsilon = ${this.epsilon}; `, e; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e.epsilon = this.epsilon, e; } _deserialize(e) { super._deserialize(e), this.evaluateContext = e.evaluateContext, this.epsilon = e.epsilon; } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], Ty.prototype, "evaluateContext", void 0); C([ rn("Epsilon", Gr.Float, "ADVANCED", { notifiers: { rebuild: !0 } }) ], Ty.prototype, "epsilon", void 0); Ue("BABYLON.GeometryOptimizeBlock", Ty); class uY extends yi { /** * Create a new PlaneBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !1, this.registerInput("size", Me.Float, !0, 1), this.registerInput("width", Me.Float, !0, 0), this.registerInput("height", Me.Float, !0, 0), this.registerOutput("geometry", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "PlaneBlock"; } /** * Gets the size input component */ get size() { return this._inputs[0]; } /** * Gets the width input component */ get width() { return this._inputs[1]; } /** * Gets the height input component */ get height() { return this._inputs[2]; } /** * Gets the geometry output component */ get geometry() { return this._outputs[0]; } autoConfigure() { if (!this.size.isConnected) { if (!this.width.isConnected && !this.height.isConnected) { const e = new xf("Size"); e.value = 1, e.output.connectTo(this.size); return; } if (!this.width.isConnected) { const e = new xf("Width"); e.value = 1, e.output.connectTo(this.width); } if (!this.height.isConnected) { const e = new xf("Height"); e.value = 1, e.output.connectTo(this.height); } } } _buildBlock(e) { const t = {}, r = (n) => (t.size = this.size.getConnectedValue(n), t.width = this.width.getConnectedValue(n), t.height = this.height.getConnectedValue(n), UO(t)); if (this.evaluateContext) this.geometry._storedFunction = r; else { const n = r(e); this.geometry._storedFunction = () => (this.geometry._executionCount = 1, n.clone()); } } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), this.evaluateContext = e.evaluateContext; } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], uY.prototype, "evaluateContext", void 0); Ue("BABYLON.PlaneBlock", uY); class lY extends yi { /** * Gets or sets the mesh to use to get vertex data */ get mesh() { return this._mesh; } set mesh(e) { this._mesh = e; } /** * Create a new MeshBlock * @param name defines the block name */ constructor(e) { super(e), this._cachedVertexData = null, this.reverseWindingOrder = !1, this.serializedCachedData = !1, this.registerOutput("geometry", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "MeshBlock"; } /** * Gets a boolean indicating if the block is using cached data */ get isUsingCachedData() { return !this.mesh && !!this._cachedVertexData; } /** * Gets the geometry output component */ get geometry() { return this._outputs[0]; } /** * Remove stored data */ cleanData() { this._mesh = null, this._cachedVertexData = null; } _buildBlock() { if (!this._mesh) { this._cachedVertexData ? this.geometry._storedValue = this._cachedVertexData.clone() : this.geometry._storedValue = null; return; } const e = Ut.ExtractFromMesh(this._mesh, !1, !0); if (this._cachedVertexData = null, this.reverseWindingOrder && e.indices) for (let t = 0; t < e.indices.length; t += 3) { const r = e.indices[t]; e.indices[t] = e.indices[t + 2], e.indices[t + 2] = r; } this.geometry._storedFunction = () => e.clone(); } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.serializedCachedData = this.serializedCachedData, this.serializedCachedData && (this._mesh ? e.cachedVertexData = Ut.ExtractFromMesh(this._mesh, !1, !0).serialize() : this._cachedVertexData && (e.cachedVertexData = this._cachedVertexData.serialize())), e.reverseWindingOrder = this.reverseWindingOrder, e; } _deserialize(e) { super._deserialize(e), e.cachedVertexData && (this._cachedVertexData = Ut.Parse(e.cachedVertexData)), this.serializedCachedData = !!e.serializedCachedData, this.reverseWindingOrder = e.reverseWindingOrder; } } C([ rn("Serialize cached data", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], lY.prototype, "serializedCachedData", void 0); Ue("BABYLON.MeshBlock", lY); class PY extends yi { /** * Create a new IcoSphereBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !1, this.registerInput("radius", Me.Float, !0, 1), this.registerInput("radiusX", Me.Float, !0, 0), this.registerInput("radiusY", Me.Float, !0, 0), this.registerInput("radiusZ", Me.Float, !0, 0), this.registerInput("subdivisions", Me.Int, !0, 4), this.registerOutput("geometry", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "IcoSphereBlock"; } /** * Gets the radius input component */ get radius() { return this._inputs[0]; } /** * Gets the radiusX input component */ get radiusX() { return this._inputs[1]; } /** * Gets the radiusY input component */ get radiusY() { return this._inputs[2]; } /** * Gets the radiusZ input component */ get radiusZ() { return this._inputs[3]; } /** * Gets the subdivisions input component */ get subdivisions() { return this._inputs[4]; } /** * Gets the geometry output component */ get geometry() { return this._outputs[0]; } autoConfigure() { if (!this.radius.isConnected) { const e = new xf("Radius"); e.value = 0.2, e.output.connectTo(this.radius); } } _buildBlock(e) { const t = {}, r = (n) => (t.radius = this.radius.getConnectedValue(n), t.subdivisions = this.subdivisions.getConnectedValue(n), t.radiusX = this.radiusX.getConnectedValue(n), t.radiusY = this.radiusY.getConnectedValue(n), t.radiusZ = this.radiusZ.getConnectedValue(n), RO(t)); if (this.evaluateContext) this.geometry._storedFunction = r; else { const n = r(e); this.geometry._storedFunction = () => (this.geometry._executionCount = 1, n.clone()); } } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), this.evaluateContext = e.evaluateContext; } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], PY.prototype, "evaluateContext", void 0); Ue("BABYLON.IcoSphereBlock", PY); class cY extends yi { /** * Create a new SphereBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !1, this.registerInput("segments", Me.Int, !0, 32), this.registerInput("diameter", Me.Float, !0, 1), this.registerInput("diameterX", Me.Float, !0, 0), this.registerInput("diameterY", Me.Float, !0, 0), this.registerInput("diameterZ", Me.Float, !0, 0), this.registerInput("arc", Me.Float, !0, 1), this.registerInput("slice", Me.Float, !0, 1), this.registerOutput("geometry", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "SphereBlock"; } /** * Gets the segments input component */ get segments() { return this._inputs[0]; } /** * Gets the diameter input component */ get diameter() { return this._inputs[1]; } /** * Gets the diameterX input component */ get diameterX() { return this._inputs[2]; } /** * Gets the diameterY input component */ get diameterY() { return this._inputs[3]; } /** * Gets the diameterZ input component */ get diameterZ() { return this._inputs[4]; } /** * Gets the arc input component */ get arc() { return this._inputs[5]; } /** * Gets the slice input component */ get slice() { return this._inputs[6]; } /** * Gets the geometry output component */ get geometry() { return this._outputs[0]; } autoConfigure() { if (!this.diameter.isConnected) { const e = new xf("Diameter"); e.value = 1, e.output.connectTo(this.diameter); } } _buildBlock(e) { const t = {}, r = (n) => (t.segments = this.segments.getConnectedValue(n), t.diameter = this.diameter.getConnectedValue(n), t.diameterX = this.diameterX.getConnectedValue(n), t.diameterY = this.diameterY.getConnectedValue(n), t.diameterZ = this.diameterZ.getConnectedValue(n), t.arc = this.arc.getConnectedValue(n), t.slice = this.slice.getConnectedValue(n), KO(t)); if (this.evaluateContext) this.geometry._storedFunction = r; else { const n = r(e); this.geometry._storedFunction = () => (this.geometry._executionCount = 1, n.clone()); } } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), this.evaluateContext = e.evaluateContext; } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], cY.prototype, "evaluateContext", void 0); Ue("BABYLON.SphereBlock", cY); class pY extends yi { /** * Create a new GridBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !1, this.registerInput("width", Me.Float, !0, 1), this.registerInput("height", Me.Float, !0, 1), this.registerInput("subdivisions", Me.Int, !0, 1), this.registerInput("subdivisionsX", Me.Int, !0, 0), this.registerInput("subdivisionsY", Me.Int, !0, 0), this.registerOutput("geometry", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "GridBlock"; } /** * Gets the width input component */ get width() { return this._inputs[0]; } /** * Gets the height input component */ get height() { return this._inputs[1]; } /** * Gets the subdivisions input component */ get subdivisions() { return this._inputs[2]; } /** * Gets the subdivisionsX input component */ get subdivisionsX() { return this._inputs[3]; } /** * Gets the subdivisionsY input component */ get subdivisionsY() { return this._inputs[4]; } /** * Gets the geometry output component */ get geometry() { return this._outputs[0]; } autoConfigure() { if (!this.width.isConnected) { const e = new xf("Width"); e.value = 1, e.output.connectTo(this.width); } if (!this.height.isConnected) { const e = new xf("Height"); e.value = 1, e.output.connectTo(this.height); } } _buildBlock(e) { const t = {}, r = (n) => (t.width = this.width.getConnectedValue(n), t.height = this.height.getConnectedValue(n), t.subdivisions = this.subdivisions.getConnectedValue(n), t.subdivisionsX = this.subdivisionsX.getConnectedValue(n), t.subdivisionsY = this.subdivisionsY.getConnectedValue(n), y2(t)); if (this.evaluateContext) this.geometry._storedFunction = r; else { const n = r(e); this.geometry._storedFunction = () => (this.geometry._executionCount = 1, n.clone()); } } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), this.evaluateContext = e.evaluateContext; } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], pY.prototype, "evaluateContext", void 0); Ue("BABYLON.GridBlock", pY); class hY extends yi { /** * Create a new TorusBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !1, this.registerInput("diameter", Me.Float, !0, 1), this.registerInput("thickness", Me.Float, !0, 0.5), this.registerInput("tessellation", Me.Int, !0, 16), this.registerOutput("geometry", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "TorusBlock"; } /** * Gets the diameter input component */ get diameter() { return this._inputs[0]; } /** * Gets the thickness input component */ get thickness() { return this._inputs[1]; } /** * Gets the tessellation input component */ get tessellation() { return this._inputs[2]; } /** * Gets the geometry output component */ get geometry() { return this._outputs[0]; } autoConfigure() { if (!this.diameter.isConnected) { const e = new xf("Diameter"); e.value = 1, e.output.connectTo(this.diameter); } } _buildBlock(e) { const t = {}, r = (n) => (t.thickness = this.thickness.getConnectedValue(n), t.diameter = this.diameter.getConnectedValue(n), t.tessellation = this.tessellation.getConnectedValue(n), QO(t)); if (this.evaluateContext) this.geometry._storedFunction = r; else { const n = r(e); this.geometry._storedFunction = () => (this.geometry._executionCount = 1, n.clone()); } } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), this.evaluateContext = e.evaluateContext; } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], hY.prototype, "evaluateContext", void 0); Ue("BABYLON.TorusBlock", hY); class HY extends yi { /** * Create a new SphereBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !1, this.registerInput("height", Me.Float, !0, 25), this.registerInput("diameter", Me.Float, !0, 1), this.registerInput("diameterTop", Me.Float, !0, -1), this.registerInput("diameterBottom", Me.Float, !0, -1), this.registerInput("subdivisions", Me.Int, !0, 1), this.registerInput("tessellation", Me.Int, !0, 24), this.registerInput("arc", Me.Float, !0, 1), this.registerOutput("geometry", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "CylinderBlock"; } /** * Gets the height input component */ get height() { return this._inputs[0]; } /** * Gets the diameter input component */ get diameter() { return this._inputs[1]; } /** * Gets the diameterTop input component */ get diameterTop() { return this._inputs[2]; } /** * Gets the diameterBottom input component */ get diameterBottom() { return this._inputs[3]; } /** * Gets the subdivisions input component */ get subdivisions() { return this._inputs[4]; } /** * Gets the tessellation input component */ get tessellation() { return this._inputs[5]; } /** * Gets the arc input component */ get arc() { return this._inputs[6]; } /** * Gets the geometry output component */ get geometry() { return this._outputs[0]; } autoConfigure() { if (!this.diameter.isConnected) { const e = new xf("Diameter"); e.value = 1, e.output.connectTo(this.diameter); } if (!this.height.isConnected) { const e = new xf("Height"); e.value = 1, e.output.connectTo(this.height); } } _buildBlock(e) { const t = {}, r = (n) => (t.height = this.height.getConnectedValue(n), t.diameter = this.diameter.getConnectedValue(n), t.diameterTop = this.diameterTop.getConnectedValue(n), t.diameterBottom = this.diameterBottom.getConnectedValue(n), t.diameterTop === -1 && (t.diameterTop = t.diameter), t.diameterBottom === -1 && (t.diameterBottom = t.diameter), t.tessellation = this.tessellation.getConnectedValue(n), t.subdivisions = this.subdivisions.getConnectedValue(n), t.arc = this.arc.getConnectedValue(n), MO(t)); if (this.evaluateContext) this.geometry._storedFunction = r; else { const n = r(e); this.geometry._storedFunction = () => (this.geometry._executionCount = 1, n.clone()); } } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), this.evaluateContext = e.evaluateContext; } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], HY.prototype, "evaluateContext", void 0); Ue("BABYLON.CylinderBlock", HY); class gY extends yi { /** * Create a new CapsuleBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !1, this.registerInput("height", Me.Float, !0, 1), this.registerInput("radius", Me.Float, !0, 0.25), this.registerInput("tessellation", Me.Int, !0, 16), this.registerInput("subdivisions", Me.Int, !0, 2), this.registerOutput("geometry", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "CapsuleBlock"; } /** * Gets the height input component */ get height() { return this._inputs[0]; } /** * Gets the radius input component */ get radius() { return this._inputs[1]; } /** * Gets the tessellation input component */ get tessellation() { return this._inputs[2]; } /** * Gets the subdivisions input component */ get subdivisions() { return this._inputs[3]; } /** * Gets the geometry output component */ get geometry() { return this._outputs[0]; } autoConfigure() { if (!this.height.isConnected) { const e = new xf("Height"); e.value = 1, e.output.connectTo(this.height); } if (!this.radius.isConnected) { const e = new xf("Radius"); e.value = 0.2, e.output.connectTo(this.radius); } } _buildBlock(e) { const t = {}, r = (n) => (t.height = this.height.getConnectedValue(n), t.radius = this.radius.getConnectedValue(n), t.tessellation = this.tessellation.getConnectedValue(n), t.subdivisions = this.subdivisions.getConnectedValue(n), JO(t)); if (this.evaluateContext) this.geometry._storedFunction = r; else { const n = r(e); this.geometry._storedFunction = () => (this.geometry._executionCount = 1, n.clone()); } } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), this.evaluateContext = e.evaluateContext; } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], gY.prototype, "evaluateContext", void 0); Ue("BABYLON.CapsuleBlock", gY); class XY extends yi { /** * Create a new DiscBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !1, this.registerInput("radius", Me.Float, !0, 0.5), this.registerInput("tessellation", Me.Int, !0, 64), this.registerInput("arc", Me.Float, !0, 1), this.registerOutput("geometry", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "DiscBlock"; } /** * Gets the radius input component */ get radius() { return this._inputs[0]; } /** * Gets the tessellation input component */ get tessellation() { return this._inputs[1]; } /** * Gets the arc input component */ get arc() { return this._inputs[2]; } /** * Gets the geometry output component */ get geometry() { return this._outputs[0]; } autoConfigure() { if (!this.radius.isConnected) { const e = new xf("Radius"); e.value = 0.2, e.output.connectTo(this.radius); } } _buildBlock(e) { const t = {}, r = (n) => (t.radius = this.radius.getConnectedValue(n), t.tessellation = this.tessellation.getConnectedValue(n), t.arc = this.arc.getConnectedValue(n), zO(t)); if (this.evaluateContext) this.geometry._storedFunction = r; else { const n = r(e); this.geometry._storedFunction = () => (this.geometry._executionCount = 1, n.clone()); } } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), this.evaluateContext = e.evaluateContext; } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], XY.prototype, "evaluateContext", void 0); Ue("BABYLON.DiscBlock", XY); class Tne extends yi { /** * Create a new NullBlock * @param name defines the block name */ constructor(e) { super(e), this.registerOutput("geometry", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "NullBlock"; } /** * Gets the geometry output component */ get geometry() { return this._outputs[0]; } _buildBlock() { this.geometry._storedValue = null; } } Ue("BABYLON.NullBlock", Tne); class TY extends yi { /** * Create a new SetPositionsBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !0, this.registerInput("geometry", Me.Geometry), this.registerInput("positions", Me.Vector3), this.registerOutput("output", Me.Geometry); } /** * Gets the current index in the current flow * @returns the current index */ getExecutionIndex() { return this._currentIndex; } /** * Gets the current loop index in the current flow * @returns the current loop index */ getExecutionLoopIndex() { return this._currentIndex; } /** * Gets the current face index in the current flow * @returns the current face index */ getExecutionFaceIndex() { return 0; } /** * Gets the current class name * @returns the class name */ getClassName() { return "SetPositionsBlock"; } /** * Gets the geometry input component */ get geometry() { return this._inputs[0]; } /** * Gets the positions input component */ get positions() { return this._inputs[1]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } _buildBlock(e) { const t = (r) => { if (r.pushExecutionContext(this), this._vertexData = this.geometry.getConnectedValue(r), this._vertexData && (this._vertexData = this._vertexData.clone()), r.pushGeometryContext(this._vertexData), !this._vertexData || !this._vertexData.positions || !this.positions.isConnected) { r.restoreGeometryContext(), r.restoreExecutionContext(), this.output._storedValue = null; return; } const n = this._vertexData.positions.length / 3; for (this._currentIndex = 0; this._currentIndex < n; this._currentIndex++) { const i = this.positions.getConnectedValue(r); i && i.toArray(this._vertexData.positions, this._currentIndex * 3); } return r.restoreGeometryContext(), r.restoreExecutionContext(), this._vertexData; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), e.evaluateContext !== void 0 && (this.evaluateContext = e.evaluateContext); } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], TY.prototype, "evaluateContext", void 0); Ue("BABYLON.SetPositionsBlock", TY); class qY extends yi { /** * Create a new SetNormalsBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !0, this.registerInput("geometry", Me.Geometry), this.registerInput("normals", Me.Vector3), this.registerOutput("output", Me.Geometry); } /** * Gets the current index in the current flow * @returns the current index */ getExecutionIndex() { return this._currentIndex; } /** * Gets the current loop index in the current flow * @returns the current loop index */ getExecutionLoopIndex() { return this._currentIndex; } /** * Gets the current face index in the current flow * @returns the current face index */ getExecutionFaceIndex() { return 0; } /** * Gets the current class name * @returns the class name */ getClassName() { return "SetNormalsBlock"; } /** * Gets the geometry input component */ get geometry() { return this._inputs[0]; } /** * Gets the normals input component */ get normals() { return this._inputs[1]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } _buildBlock(e) { const t = (r) => { if (r.pushExecutionContext(this), this._vertexData = this.geometry.getConnectedValue(r), this._vertexData && (this._vertexData = this._vertexData.clone()), r.pushGeometryContext(this._vertexData), !this._vertexData || !this._vertexData.positions) { r.restoreGeometryContext(), r.restoreExecutionContext(), this.output._storedValue = null; return; } if (!this.normals.isConnected) { r.restoreGeometryContext(), r.restoreExecutionContext(), this.output._storedValue = this._vertexData; return; } this._vertexData.normals || (this._vertexData.normals = []); const n = this._vertexData.positions.length / 3; for (this._currentIndex = 0; this._currentIndex < n; this._currentIndex++) { const i = this.normals.getConnectedValue(r); i && i.toArray(this._vertexData.normals, this._currentIndex * 3); } return r.restoreGeometryContext(), r.restoreExecutionContext(), this._vertexData; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), e.evaluateContext !== void 0 && (this.evaluateContext = e.evaluateContext); } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], qY.prototype, "evaluateContext", void 0); Ue("BABYLON.SetNormalsBlock", qY); class qy extends yi { /** * Create a new SetUVsBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !0, this.textureCoordinateIndex = 0, this.registerInput("geometry", Me.Geometry), this.registerInput("uvs", Me.Vector2), this.registerOutput("output", Me.Geometry); } /** * Gets the current index in the current flow * @returns the current index */ getExecutionIndex() { return this._currentIndex; } /** * Gets the current loop index in the current flow * @returns the current loop index */ getExecutionLoopIndex() { return this._currentIndex; } /** * Gets the current face index in the current flow * @returns the current face index */ getExecutionFaceIndex() { return 0; } /** * Gets the current class name * @returns the class name */ getClassName() { return "SetUVsBlock"; } /** * Gets the geometry input component */ get geometry() { return this._inputs[0]; } /** * Gets the uvs input component */ get uvs() { return this._inputs[1]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } _buildBlock(e) { const t = (r) => { if (r.pushExecutionContext(this), this._vertexData = this.geometry.getConnectedValue(r), this._vertexData && (this._vertexData = this._vertexData.clone()), r.pushGeometryContext(this._vertexData), !this._vertexData || !this._vertexData.positions) { r.restoreGeometryContext(), r.restoreExecutionContext(), this.output._storedValue = null; return; } if (!this.uvs.isConnected) { r.restoreGeometryContext(), r.restoreExecutionContext(), this.output._storedValue = this._vertexData; return; } const n = [], i = this._vertexData.positions.length / 3; for (this._currentIndex = 0; this._currentIndex < i; this._currentIndex++) { const s = this.uvs.getConnectedValue(r); s && s.toArray(n, this._currentIndex * 2); } switch (this.textureCoordinateIndex) { case 0: this._vertexData.uvs = n; break; case 1: this._vertexData.uvs2 = n; break; case 2: this._vertexData.uvs3 = n; break; case 3: this._vertexData.uvs4 = n; break; case 4: this._vertexData.uvs5 = n; break; case 5: this._vertexData.uvs6 = n; break; } return r.restoreGeometryContext(), r.restoreExecutionContext(), this._vertexData; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } _dumpPropertiesCode() { let e = super._dumpPropertiesCode() + `${this._codeVariableName}.textureCoordinateIndex}; `; return e += `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `, e; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e.textureCoordinateIndex = this.textureCoordinateIndex, e; } _deserialize(e) { super._deserialize(e), this.textureCoordinateIndex = e.textureCoordinateIndex, e.evaluateContext !== void 0 && (this.evaluateContext = e.evaluateContext); } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], qy.prototype, "evaluateContext", void 0); C([ rn("Texture coordinates index", Gr.List, "ADVANCED", { notifiers: { update: !0 }, options: [ { label: "UV1", value: 0 }, { label: "UV2", value: 1 }, { label: "UV3", value: 2 }, { label: "UV4", value: 3 }, { label: "UV5", value: 4 }, { label: "UV6", value: 5 } ] }) ], qy.prototype, "textureCoordinateIndex", void 0); Ue("BABYLON.SetUVsBlock", qy); class bY extends yi { /** * Create a new SetColorsBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !0, this.registerInput("geometry", Me.Geometry), this.registerInput("colors", Me.Vector4), this.registerOutput("output", Me.Geometry); } /** * Gets the current index in the current flow * @returns the current index */ getExecutionIndex() { return this._currentIndex; } /** * Gets the current loop index in the current flow * @returns the current loop index */ getExecutionLoopIndex() { return this._currentIndex; } /** * Gets the current face index in the current flow * @returns the current face index */ getExecutionFaceIndex() { return 0; } /** * Gets the current class name * @returns the class name */ getClassName() { return "SetColorsBlock"; } /** * Gets the geometry input component */ get geometry() { return this._inputs[0]; } /** * Gets the colors input component */ get colors() { return this._inputs[1]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } _buildBlock(e) { const t = (r) => { if (r.pushExecutionContext(this), this._vertexData = this.geometry.getConnectedValue(r), this._vertexData && (this._vertexData = this._vertexData.clone()), r.pushGeometryContext(this._vertexData), !this._vertexData || !this._vertexData.positions) { r.restoreGeometryContext(), r.restoreExecutionContext(), this.output._storedValue = null; return; } if (!this.colors.isConnected) { r.restoreGeometryContext(), r.restoreExecutionContext(), this.output._storedValue = this._vertexData; return; } this._vertexData.colors || (this._vertexData.colors = []); const n = this._vertexData.positions.length / 3; for (this._currentIndex = 0; this._currentIndex < n; this._currentIndex++) { const i = this.colors.getConnectedValue(r); i && i.toArray(this._vertexData.colors, this._currentIndex * 4); } return r.restoreGeometryContext(), r.restoreExecutionContext(), this._vertexData; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), e.evaluateContext !== void 0 && (this.evaluateContext = e.evaluateContext); } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], bY.prototype, "evaluateContext", void 0); Ue("BABYLON.SetColorsBlock", bY); class xY extends yi { /** * Create a new SetTangentsBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !0, this.registerInput("geometry", Me.Geometry), this.registerInput("tangents", Me.Vector4), this.registerOutput("output", Me.Geometry); } /** * Gets the current index in the current flow * @returns the current index */ getExecutionIndex() { return this._currentIndex; } /** * Gets the current loop index in the current flow * @returns the current loop index */ getExecutionLoopIndex() { return this._currentIndex; } /** * Gets the current face index in the current flow * @returns the current face index */ getExecutionFaceIndex() { return 0; } /** * Gets the current class name * @returns the class name */ getClassName() { return "SetTangentsBlock"; } /** * Gets the geometry input component */ get geometry() { return this._inputs[0]; } /** * Gets the tangents input component */ get tangents() { return this._inputs[1]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } _buildBlock(e) { const t = (r) => { if (r.pushExecutionContext(this), this._vertexData = this.geometry.getConnectedValue(r), this._vertexData && (this._vertexData = this._vertexData.clone()), r.pushGeometryContext(this._vertexData), !this._vertexData || !this._vertexData.positions) { r.restoreGeometryContext(), r.restoreExecutionContext(), this.output._storedValue = null; return; } if (!this.tangents.isConnected) { r.restoreGeometryContext(), r.restoreExecutionContext(), this.output._storedValue = this._vertexData; return; } this._vertexData.tangents || (this._vertexData.tangents = []); const n = this._vertexData.positions.length / 3; for (this._currentIndex = 0; this._currentIndex < n; this._currentIndex++) { const i = this.tangents.getConnectedValue(r); i && i.toArray(this._vertexData.tangents, this._currentIndex * 4); } return r.restoreGeometryContext(), r.restoreExecutionContext(), this._vertexData; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), e.evaluateContext !== void 0 && (this.evaluateContext = e.evaluateContext); } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], xY.prototype, "evaluateContext", void 0); Ue("BABYLON.SetTangentsBlock", xY); var Ov; (function(A) { A[A.Add = 0] = "Add", A[A.Subtract = 1] = "Subtract", A[A.Multiply = 2] = "Multiply", A[A.Divide = 3] = "Divide", A[A.Max = 4] = "Max", A[A.Min = 5] = "Min"; })(Ov || (Ov = {})); class DY extends yi { /** * Create a new MathBlock * @param name defines the block name */ constructor(e) { super(e), this.operation = Ov.Add, this.registerInput("left", Me.AutoDetect), this.registerInput("right", Me.AutoDetect), this.registerOutput("output", Me.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._inputs[0].excludedConnectionPointTypes.push(Me.Matrix), this._inputs[0].excludedConnectionPointTypes.push(Me.Geometry), this._inputs[0].excludedConnectionPointTypes.push(Me.Texture), this._inputs[1].excludedConnectionPointTypes.push(Me.Matrix), this._inputs[1].excludedConnectionPointTypes.push(Me.Geometry), this._inputs[1].excludedConnectionPointTypes.push(Me.Texture), this._inputs[1].acceptedConnectionPointTypes.push(Me.Float), this._linkConnectionTypes(0, 1); } /** * Gets the current class name * @returns the class name */ getClassName() { return "MathBlock"; } /** * Gets the left input component */ get left() { return this._inputs[0]; } /** * Gets the right input component */ get right() { return this._inputs[1]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } _buildBlock() { let e; const t = this.left, r = this.right; if (!t.isConnected || !r.isConnected) { this.output._storedFunction = null, this.output._storedValue = null; return; } const n = t.type === Me.Float || t.type === Me.Int; switch (this.operation) { case Ov.Add: { n ? e = (i) => t.getConnectedValue(i) + r.getConnectedValue(i) : e = (i) => t.getConnectedValue(i).add(i.adapt(r, t.type)); break; } case Ov.Subtract: { n ? e = (i) => t.getConnectedValue(i) - r.getConnectedValue(i) : e = (i) => t.getConnectedValue(i).subtract(i.adapt(r, t.type)); break; } case Ov.Multiply: { n ? e = (i) => t.getConnectedValue(i) * r.getConnectedValue(i) : e = (i) => t.getConnectedValue(i).multiply(i.adapt(r, t.type)); break; } case Ov.Divide: { n ? e = (i) => t.getConnectedValue(i) / r.getConnectedValue(i) : e = (i) => t.getConnectedValue(i).divide(i.adapt(r, t.type)); break; } case Ov.Min: { if (n) e = (i) => Math.min(t.getConnectedValue(i), r.getConnectedValue(i)); else switch (t.type) { case Me.Vector2: { e = (i) => at.Minimize(t.getConnectedValue(i), i.adapt(r, t.type)); break; } case Me.Vector3: { e = (i) => S.Minimize(t.getConnectedValue(i), i.adapt(r, t.type)); break; } case Me.Vector4: { e = (i) => Ir.Minimize(t.getConnectedValue(i), i.adapt(r, t.type)); break; } } break; } case Ov.Max: if (n) e = (i) => Math.max(t.getConnectedValue(i), r.getConnectedValue(i)); else { switch (t.type) { case Me.Vector2: { e = (i) => at.Maximize(t.getConnectedValue(i), i.adapt(r, t.type)); break; } case Me.Vector3: { e = (i) => S.Maximize(t.getConnectedValue(i), i.adapt(r, t.type)); break; } case Me.Vector4: { e = (i) => Ir.Maximize(t.getConnectedValue(i), i.adapt(r, t.type)); break; } } break; } } this.output._storedFunction = (i) => t.type === Me.Int ? e(i) | 0 : e(i); } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.operation = BABYLON.MathBlockOperations.${Ov[this.operation]}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.operation = this.operation, e; } _deserialize(e) { super._deserialize(e), this.operation = e.operation; } } C([ rn("Operation", Gr.List, "ADVANCED", { notifiers: { rebuild: !0 }, options: [ { label: "Add", value: Ov.Add }, { label: "Subtract", value: Ov.Subtract }, { label: "Multiply", value: Ov.Multiply }, { label: "Divide", value: Ov.Divide }, { label: "Max", value: Ov.Max }, { label: "Min", value: Ov.Min } ] }) ], DY.prototype, "operation", void 0); Ue("BABYLON.MathBlock", DY); class qne extends yi { /** * Create a new MapRangeBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("value", Me.AutoDetect), this.registerInput("fromMin", Me.Float, !0, 0), this.registerInput("fromMax", Me.Float, !0, 1), this.registerInput("toMin", Me.Float, !0, 0), this.registerInput("toMax", Me.Float, !0, 1), this.registerOutput("output", Me.BasedOnInput), this._inputs[0].excludedConnectionPointTypes.push(Me.Vector2), this._inputs[0].excludedConnectionPointTypes.push(Me.Vector3), this._inputs[0].excludedConnectionPointTypes.push(Me.Vector4), this._inputs[0].excludedConnectionPointTypes.push(Me.Matrix), this._inputs[0].excludedConnectionPointTypes.push(Me.Geometry), this._inputs[0].excludedConnectionPointTypes.push(Me.Texture), this._outputs[0]._typeConnectionSource = this._inputs[0]; } /** * Gets the current class name * @returns the class name */ getClassName() { return "MapRangeBlock"; } /** * Gets the value input component */ get value() { return this._inputs[0]; } /** * Gets the fromMin input component */ get fromMin() { return this._inputs[1]; } /** * Gets the fromMax input component */ get fromMax() { return this._inputs[2]; } /** * Gets the toMin input component */ get toMin() { return this._inputs[3]; } /** * Gets the toMax input component */ get toMax() { return this._inputs[4]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock() { if (!this.value.isConnected) { this.output._storedFunction = null, this.output._storedValue = null; return; } this.output._storedFunction = (e) => { const t = this.value.getConnectedValue(e), r = this.fromMin.getConnectedValue(e), n = this.fromMax.getConnectedValue(e), i = this.toMin.getConnectedValue(e), s = this.toMax.getConnectedValue(e), a = (t - r) / (n - r) * (s - i) + i; return this.output.type === Me.Int ? Math.floor(a) : a; }; } } Ue("BABYLON.MapRangeBlock", qne); var y9; (function(A) { A[A.Equal = 0] = "Equal", A[A.NotEqual = 1] = "NotEqual", A[A.LessThan = 2] = "LessThan", A[A.GreaterThan = 3] = "GreaterThan", A[A.LessOrEqual = 4] = "LessOrEqual", A[A.GreaterOrEqual = 5] = "GreaterOrEqual", A[A.Xor = 6] = "Xor", A[A.Or = 7] = "Or", A[A.And = 8] = "And"; })(y9 || (y9 = {})); class jY extends yi { /** * Create a new ConditionBlock * @param name defines the block name */ constructor(e) { super(e), this.test = y9.Equal, this.registerInput("left", Me.Float), this.registerInput("right", Me.Float, !0, 0), this.registerInput("ifTrue", Me.AutoDetect, !0, 1), this.registerInput("ifFalse", Me.AutoDetect, !0, 0), this.registerOutput("output", Me.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[2], this._outputs[0]._defaultConnectionPointType = Me.Float, this._inputs[0].acceptedConnectionPointTypes.push(Me.Int), this._inputs[1].acceptedConnectionPointTypes.push(Me.Int), this._linkConnectionTypes(2, 3); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ConditionBlock"; } /** * Gets the left input component */ get left() { return this._inputs[0]; } /** * Gets the right input component */ get right() { return this._inputs[1]; } /** * Gets the ifTrue input component */ get ifTrue() { return this._inputs[2]; } /** * Gets the ifFalse input component */ get ifFalse() { return this._inputs[3]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock() { if (!this.left.isConnected) { this.output._storedFunction = null, this.output._storedValue = null; return; } const e = (t) => { const r = this.left.getConnectedValue(t), n = this.right.getConnectedValue(t); let i = !1; switch (this.test) { case y9.Equal: i = Xt.WithinEpsilon(r, n, Dn); break; case y9.NotEqual: i = r !== n; break; case y9.LessThan: i = r < n; break; case y9.GreaterThan: i = r > n; break; case y9.LessOrEqual: i = r <= n; break; case y9.GreaterOrEqual: i = r >= n; break; case y9.Xor: i = !!r && !n || !r && !!n; break; case y9.Or: i = !!r || !!n; break; case y9.And: i = !!r && !!n; break; } return i; }; this.output._storedFunction = (t) => e(t) ? this.ifTrue.getConnectedValue(t) : this.ifFalse.getConnectedValue(t); } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.test = BABYLON.ConditionBlockTests.${y9[this.test]}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.test = this.test, e; } _deserialize(e) { super._deserialize(e), this.test = e.test; } } C([ rn("Test", Gr.List, "ADVANCED", { notifiers: { rebuild: !0 }, options: [ { label: "Equal", value: y9.Equal }, { label: "NotEqual", value: y9.NotEqual }, { label: "LessThan", value: y9.LessThan }, { label: "GreaterThan", value: y9.GreaterThan }, { label: "LessOrEqual", value: y9.LessOrEqual }, { label: "GreaterOrEqual", value: y9.GreaterOrEqual }, { label: "Xor", value: y9.Xor }, { label: "Or", value: y9.Or }, { label: "And", value: y9.And } ] }) ], jY.prototype, "test", void 0); Ue("BABYLON.ConditionBlock", jY); var B0; (function(A) { A[A.None = 0] = "None", A[A.LoopID = 1] = "LoopID", A[A.InstanceID = 2] = "InstanceID"; })(B0 || (B0 = {})); class wY extends yi { /** * Create a new RandomBlock * @param name defines the block name */ constructor(e) { super(e), this._currentLockId = -1, this.lockMode = B0.None, this.registerInput("min", Me.AutoDetect), this.registerInput("max", Me.AutoDetect), this.registerOutput("output", Me.BasedOnInput), this._inputs[0].excludedConnectionPointTypes.push(Me.Matrix), this._inputs[0].excludedConnectionPointTypes.push(Me.Geometry), this._inputs[0].excludedConnectionPointTypes.push(Me.Texture), this._inputs[1].excludedConnectionPointTypes.push(Me.Matrix), this._inputs[1].excludedConnectionPointTypes.push(Me.Geometry), this._inputs[1].excludedConnectionPointTypes.push(Me.Texture), this._outputs[0]._typeConnectionSource = this._inputs[0], this._linkConnectionTypes(0, 1); } /** * Gets the current class name * @returns the class name */ getClassName() { return "RandomBlock"; } /** * Gets the min input component */ get min() { return this._inputs[0]; } /** * Gets the max input component */ get max() { return this._inputs[1]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } autoConfigure() { if (!this.min.isConnected) { const e = new xf("Min"); e.value = 0, e.output.connectTo(this.min); } if (!this.max.isConnected) { const e = new xf("Max"); e.value = 1, e.output.connectTo(this.max); } } _buildBlock() { let e = null; switch (this._currentLockId = -1, this.min.type) { case Me.Int: case Me.Float: { e = (t) => { const r = this.min.getConnectedValue(t) || 0, n = this.max.getConnectedValue(t) || 0; return r + Math.random() * (n - r); }; break; } case Me.Vector2: { e = (t) => { const r = this.min.getConnectedValue(t) || at.Zero(), n = this.max.getConnectedValue(t) || at.Zero(); return new at(r.x + Math.random() * (n.x - r.x), r.y + Math.random() * (n.y - r.y)); }; break; } case Me.Vector3: { e = (t) => { const r = this.min.getConnectedValue(t) || S.Zero(), n = this.max.getConnectedValue(t) || S.Zero(); return new S(r.x + Math.random() * (n.x - r.x), r.y + Math.random() * (n.y - r.y), r.z + Math.random() * (n.z - r.z)); }; break; } case Me.Vector4: { e = (t) => { const r = this.min.getConnectedValue(t) || Ir.Zero(), n = this.max.getConnectedValue(t) || Ir.Zero(); return new Ir(r.x + Math.random() * (n.x - r.x), r.y + Math.random() * (n.y - r.y), r.z + Math.random() * (n.z - r.z), r.w + Math.random() * (n.w - r.w)); }; break; } } this.lockMode === B0.None || !e ? this.output._storedFunction = e : this.output._storedFunction = (t) => { let r = 0; switch (this.lockMode) { case B0.InstanceID: r = t.getContextualValue(Ws.InstanceID, !0) || 0; break; case B0.LoopID: r = t.getContextualValue(Ws.LoopID, !0) || 0; break; } return (this._currentLockId !== r || this.lockMode === B0.None) && (this._currentLockId = r, this.output._storedValue = e(t)), this.output._storedValue; }; } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.lockMode = BABYLON.RandomBlockLocks.${B0[this.lockMode]}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.lockMode = this.lockMode, e; } _deserialize(e) { super._deserialize(e), this.lockMode = e.lockMode; } } C([ rn("LockMode", Gr.List, "ADVANCED", { notifiers: { rebuild: !0 }, options: [ { label: "None", value: B0.None }, { label: "LoopID", value: B0.LoopID }, { label: "InstanceID", value: B0.InstanceID } ] }) ], wY.prototype, "lockMode", void 0); Ue("BABYLON.RandomBlock", wY); class bne extends yi { /** * Create a new NoiseBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("offset", Me.Vector3, !0, S.Zero()), this.registerInput("scale", Me.Float, !0, 1), this.registerInput("octaves", Me.Float, !0, 2, 0, 16), this.registerInput("roughness", Me.Float, !0, 0.5, 0, 1), this.registerOutput("output", Me.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "NoiseBlock"; } /** * Gets the offset input component */ get offset() { return this._inputs[0]; } /** * Gets the scale input component */ get scale() { return this._inputs[1]; } /** * Gets the octaves input component */ get octaves() { return this._inputs[2]; } /** * Gets the roughtness input component */ get roughness() { return this._inputs[3]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } _negateIf(e, t) { return t !== 0 ? -e : e; } _noiseGrad(e, t, r, n) { const i = e & 15, s = i < 8 ? t : r, a = i === 12 || i == 14 ? t : n, f = i < 4 ? r : a; return this._negateIf(s, i & s) + this._negateIf(f, i & 2); } _fade(e) { return e * e * e * (e * (e * 6 - 15) + 10); } _hashBitRotate(e, t) { return e << t | e >> 32 - t; } _hash(e, t, r) { let n, i, s; return n = i = s = 3735928584, s += r, i += t, n += e, s ^= i, s -= this._hashBitRotate(i, 14), n ^= s, n -= this._hashBitRotate(s, 11), i ^= n, i -= this._hashBitRotate(n, 25), s ^= i, s -= this._hashBitRotate(i, 16), n ^= s, n -= this._hashBitRotate(s, 4), i ^= n, i -= this._hashBitRotate(n, 14), s ^= i, s -= this._hashBitRotate(i, 24), s; } _mix(e, t, r, n, i, s, a, f, o, d, v) { const u = 1 - o, l = 1 - d; return (1 - v) * (l * (e * u + t * o) + d * (r * u + n * o)) + v * (l * (i * u + s * o) + d * (a * u + f * o)); } _perlinNoise(e) { const t = (e.x | 0) - (e.x < 0 ? 1 : 0), r = (e.y | 0) - (e.y < 0 ? 1 : 0), n = (e.z | 0) - (e.z < 0 ? 1 : 0), i = e.x - t, s = e.y - r, a = e.z - n, f = this._fade(i), o = this._fade(s), d = this._fade(a); return this._mix(this._noiseGrad(this._hash(t, r, n), i, s, a), this._noiseGrad(this._hash(t + 1, r, n), i - 1, s, a), this._noiseGrad(this._hash(t, r + 1, n), i, s - 1, a), this._noiseGrad(this._hash(t + 1, r + 1, n), i - 1, s - 1, a), this._noiseGrad(this._hash(t, r, n + 1), i, s, a - 1), this._noiseGrad(this._hash(t + 1, r, n + 1), i - 1, s, a - 1), this._noiseGrad(this._hash(t, r + 1, n + 1), i, s - 1, a - 1), this._noiseGrad(this._hash(t + 1, r + 1, n + 1), i - 1, s - 1, a - 1), f, o, d); } _perlinSigned(e) { return this._perlinNoise(e) * 0.982; } _perlin(e) { return this._perlinSigned(e) / 2 + 0.5; } /** * Gets a perlin noise value * @param octaves * @param roughness * @param position * @returns a value between 0 and 1 * @see Based on https://github.com/blender/blender/blob/main/source/blender/blenlib/intern/noise.cc#L533 */ noise(e, t, r, n, i) { const s = new S(r.x * i + n.x, r.y * i + n.y, r.z * i + n.z); let a = 1, f = 1, o = 0, d = 0; e = Xt.Clamp(e, 0, 15); const v = e | 0; for (let p = 0; p <= v; p++) { const c = this._perlin(s.scale(a)); d += c * f, o += f, f *= Xt.Clamp(t, 0, 1), a *= 2; } const u = e - Math.floor(e); if (u == 0) return d / o; const l = this._perlin(s.scale(a)); let P = d + l * f; return d /= o, P /= o + f, (1 - u) * d + u * P; } _buildBlock() { this.output._storedFunction = (e) => { const t = e.getContextualValue(Ws.Positions), r = this.octaves.getConnectedValue(e), n = this.roughness.getConnectedValue(e), i = this.offset.getConnectedValue(e), s = this.scale.getConnectedValue(e); return this.noise(r, n, t, i, s); }; } } Ue("BABYLON.NoiseBlock", bne); class mY extends yi { /** * Create a new MergeGeometryBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !1, this.registerInput("geometry0", Me.Geometry), this.registerInput("geometry1", Me.Geometry, !0), this.registerInput("geometry2", Me.Geometry, !0), this.registerInput("geometry3", Me.Geometry, !0), this.registerInput("geometry4", Me.Geometry, !0), this.registerOutput("output", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "MergeGeometryBlock"; } /** * Gets the geometry0 input component */ get geometry0() { return this._inputs[0]; } /** * Gets the geometry1 input component */ get geometry1() { return this._inputs[1]; } /** * Gets the geometry2 input component */ get geometry2() { return this._inputs[2]; } /** * Gets the geometry3 input component */ get geometry3() { return this._inputs[3]; } /** * Gets the geometry4 input component */ get geometry4() { return this._inputs[4]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } _buildBlock(e) { const t = (r) => { let n = this.geometry0.getConnectedValue(r); const i = []; if (n) n = n.clone(); else return null; if (this.geometry1.isConnected) { const s = this.geometry1.getConnectedValue(r); s && i.push(s); } if (this.geometry2.isConnected) { const s = this.geometry2.getConnectedValue(r); s && i.push(s); } if (this.geometry3.isConnected) { const s = this.geometry3.getConnectedValue(r); s && i.push(s); } if (this.geometry4.isConnected) { const s = this.geometry4.getConnectedValue(r); s && i.push(s); } return i.length && n && (n = n.merge(i, !0, !1, !0, !0)), n; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), this.evaluateContext = e.evaluateContext; } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], mY.prototype, "evaluateContext", void 0); Ue("BABYLON.MergeGeometryBlock", mY); class BY extends yi { /** * Create a new GeometryCollectionBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !0, this.registerInput("geometry0", Me.Geometry, !0), this.registerInput("geometry1", Me.Geometry, !0), this.registerInput("geometry2", Me.Geometry, !0), this.registerInput("geometry3", Me.Geometry, !0), this.registerInput("geometry4", Me.Geometry, !0), this.registerInput("geometry5", Me.Geometry, !0), this.registerInput("geometry6", Me.Geometry, !0), this.registerInput("geometry7", Me.Geometry, !0), this.registerInput("geometry8", Me.Geometry, !0), this.registerInput("geometry9", Me.Geometry, !0), this.registerOutput("output", Me.Geometry), this._outputs[0]._typeConnectionSource = this._inputs[0], this._linkConnectionTypes(0, 1); } /** * Gets the current class name * @returns the class name */ getClassName() { return "GeometryCollectionBlock"; } /** * Gets the geometry0 input component */ get geometry0() { return this._inputs[0]; } /** * Gets the geometry1 input component */ get geometry1() { return this._inputs[1]; } /** * Gets the geometry2 input component */ get geometry2() { return this._inputs[2]; } /** * Gets the geometry3 input component */ get geometry3() { return this._inputs[3]; } /** * Gets the geometry4 input component */ get geometry4() { return this._inputs[4]; } /** * Gets the geometry5 input component */ get geometry5() { return this._inputs[5]; } /** * Gets the geometry6 input component */ get geometry6() { return this._inputs[6]; } /** * Gets the geometry7 input component */ get geometry7() { return this._inputs[7]; } /** * Gets the geometry8 input component */ get geometry8() { return this._inputs[8]; } /** * Gets the geometry9 input component */ get geometry9() { return this._inputs[9]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } _storeGeometry(e, t, r, n) { if (e.isConnected) { const i = e.getConnectedValue(t); if (!i) return; i.metadata = i.metadata || {}, i.metadata.collectionId = r, n.push(i); } } _buildBlock(e) { const t = (r) => { const n = []; return this._storeGeometry(this.geometry0, r, 0, n), this._storeGeometry(this.geometry1, r, 1, n), this._storeGeometry(this.geometry2, r, 2, n), this._storeGeometry(this.geometry3, r, 3, n), this._storeGeometry(this.geometry4, r, 4, n), this._storeGeometry(this.geometry5, r, 5, n), this._storeGeometry(this.geometry6, r, 6, n), this._storeGeometry(this.geometry7, r, 7, n), this._storeGeometry(this.geometry8, r, 8, n), this._storeGeometry(this.geometry9, r, 9, n), n.length ? n[Math.round(Math.random() * (n.length - 1))] : null; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), this.evaluateContext = e.evaluateContext; } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], BY.prototype, "evaluateContext", void 0); Ue("BABYLON.GeometryCollectionBlock", BY); class xne extends yi { /** * Creates a new GeometryElbowBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("input", Me.AutoDetect), this.registerOutput("output", Me.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0]; } /** * Gets the time spent to build this block (in ms) */ get buildExecutionTime() { return 0; } /** * Gets the current class name * @returns the class name */ getClassName() { return "GeometryElbowBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); const t = this._outputs[0], r = this._inputs[0]; t._storedFunction = (n) => r.getConnectedValue(n); } } Ue("BABYLON.GeometryElbowBlock", xne); class Dne extends yi { /** * Creates a new ComputeNormalsBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("geometry", Me.Geometry), this.registerOutput("output", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ComputeNormalsBlock"; } /** * Gets the geometry component */ get geometry() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock() { this.output._storedFunction = (e) => { if (!this.geometry.isConnected) return null; const t = this.geometry.getConnectedValue(e); return t.normals || (t.normals = []), Ut.ComputeNormals(t.positions, t.indices, t.normals), t; }; } } Ue("BABYLON.ComputeNormalsBlock", Dne); class jne extends yi { /** * Create a new VectorConverterBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("xyzw ", Me.Vector4, !0), this.registerInput("xyz ", Me.Vector3, !0), this.registerInput("xy ", Me.Vector2, !0), this.registerInput("zw ", Me.Vector2, !0), this.registerInput("x ", Me.Float, !0), this.registerInput("y ", Me.Float, !0), this.registerInput("z ", Me.Float, !0), this.registerInput("w ", Me.Float, !0), this.registerOutput("xyzw", Me.Vector4), this.registerOutput("xyz", Me.Vector3), this.registerOutput("xy", Me.Vector2), this.registerOutput("zw", Me.Vector2), this.registerOutput("x", Me.Float), this.registerOutput("y", Me.Float), this.registerOutput("z", Me.Float), this.registerOutput("w", Me.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "VectorConverterBlock"; } /** * Gets the xyzw component (input) */ get xyzwIn() { return this._inputs[0]; } /** * Gets the xyz component (input) */ get xyzIn() { return this._inputs[1]; } /** * Gets the xy component (input) */ get xyIn() { return this._inputs[2]; } /** * Gets the zw component (input) */ get zwIn() { return this._inputs[3]; } /** * Gets the x component (input) */ get xIn() { return this._inputs[4]; } /** * Gets the y component (input) */ get yIn() { return this._inputs[5]; } /** * Gets the z component (input) */ get zIn() { return this._inputs[6]; } /** * Gets the w component (input) */ get wIn() { return this._inputs[7]; } /** * Gets the xyzw component (output) */ get xyzwOut() { return this._outputs[0]; } /** * Gets the xyz component (output) */ get xyzOut() { return this._outputs[1]; } /** * Gets the xy component (output) */ get xyOut() { return this._outputs[2]; } /** * Gets the zw component (output) */ get zwOut() { return this._outputs[3]; } /** * Gets the x component (output) */ get xOut() { return this._outputs[4]; } /** * Gets the y component (output) */ get yOut() { return this._outputs[5]; } /** * Gets the z component (output) */ get zOut() { return this._outputs[6]; } /** * Gets the w component (output) */ get wOut() { return this._outputs[7]; } _inputRename(e) { return e === "xyzw " ? "xyzwIn" : e === "xyz " ? "xyzIn" : e === "xy " ? "xyIn" : e === "zw " ? "zwIn" : e === "x " ? "xIn" : e === "y " ? "yIn" : e === "z " ? "zIn" : e === "w " ? "wIn" : e; } _outputRename(e) { switch (e) { case "x": return "xOut"; case "y": return "yOut"; case "z": return "zOut"; case "w": return "wOut"; case "xy": return "xyOut"; case "zw": return "zwOut"; case "xyz": return "xyzOut"; case "xyzw": return "xyzwOut"; default: return e; } } _buildBlock(e) { super._buildBlock(e); const t = this.xIn, r = this.yIn, n = this.zIn, i = this.wIn, s = this.xyIn, a = this.zwIn, f = this.xyzIn, o = this.xyzwIn, d = this.xyzwOut, v = this.xyzOut, u = this.xyOut, l = this.zwOut, P = this.xOut, p = this.yOut, c = this.zOut, H = this.wOut, T = (q) => { if (o.isConnected) return o.getConnectedValue(q); let b = 0, j = 0, w = 0, m = 0; if (t.isConnected && (b = t.getConnectedValue(q)), r.isConnected && (j = r.getConnectedValue(q)), n.isConnected && (w = n.getConnectedValue(q)), i.isConnected && (m = i.getConnectedValue(q)), s.isConnected) { const I = s.getConnectedValue(q); I && (b = I.x, j = I.y); } if (a.isConnected) { const I = a.getConnectedValue(q); I && (w = I.x, m = I.y); } if (f.isConnected) { const I = f.getConnectedValue(q); I && (b = I.x, j = I.y, w = I.z); } return new Ir(b, j, w, m); }; d._storedFunction = (q) => T(q), v._storedFunction = (q) => { const b = T(q); return new S(b.x, b.y, b.z); }, u._storedFunction = (q) => { const b = T(q); return new at(b.x, b.y); }, l._storedFunction = (q) => { const b = T(q); return new at(b.z, b.w); }, P._storedFunction = (q) => T(q).x, p._storedFunction = (q) => T(q).y, c._storedFunction = (q) => T(q).z, H._storedFunction = (q) => T(q).w; } } Ue("BABYLON.VectorConverterBlock", jne); class wne extends yi { /** * Creates a new NormalizeVectorBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("input", Me.AutoDetect), this.registerOutput("output", Me.BasedOnInput), this._inputs[0].excludedConnectionPointTypes.push(Me.Float), this._inputs[0].excludedConnectionPointTypes.push(Me.Matrix), this._inputs[0].excludedConnectionPointTypes.push(Me.Geometry), this._inputs[0].excludedConnectionPointTypes.push(Me.Texture), this._outputs[0]._typeConnectionSource = this._inputs[0]; } /** * Gets the current class name * @returns the class name */ getClassName() { return "NormalizeVectorBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { if (super._buildBlock(e), this.output._storedFunction = null, !this.input.isConnected) { this.output._storedValue = null; return; } this.output._storedFunction = (t) => this.input.getConnectedValue(t).normalize(); } } Ue("BABYLON.NormalizeVectorBlock", wne); class WY extends yi { /** * Create a new SetMaterialIDBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !0, this.registerInput("geometry", Me.Geometry), this.registerInput("id", Me.Int, !0, 0), this.registerOutput("output", Me.Geometry), this.id.acceptedConnectionPointTypes.push(Me.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "SetMaterialIDBlock"; } /** * Gets the geometry input component */ get geometry() { return this._inputs[0]; } /** * Gets the id input component */ get id() { return this._inputs[1]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } _buildBlock(e) { if (!this.geometry.isConnected) { this.output._storedFunction = null, this.output._storedValue = null; return; } const t = (r) => { const n = this.geometry.getConnectedValue(r); if (!n || !n.indices || !n.positions) return n; const i = new wI(); return i.materialIndex = this.id.getConnectedValue(r) | 0, i.indexStart = 0, i.indexCount = n.indices.length, i.verticesStart = 0, i.verticesCount = n.positions.length / 3, n.materialInfos = [i], n; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), e.evaluateContext !== void 0 && (this.evaluateContext = e.evaluateContext); } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], WY.prototype, "evaluateContext", void 0); Ue("BABYLON.SetMaterialIDBlock", WY); var Bs; (function(A) { A[A.Cos = 0] = "Cos", A[A.Sin = 1] = "Sin", A[A.Abs = 2] = "Abs", A[A.Exp = 3] = "Exp", A[A.Round = 4] = "Round", A[A.Floor = 5] = "Floor", A[A.Ceiling = 6] = "Ceiling", A[A.Sqrt = 7] = "Sqrt", A[A.Log = 8] = "Log", A[A.Tan = 9] = "Tan", A[A.ArcTan = 10] = "ArcTan", A[A.ArcCos = 11] = "ArcCos", A[A.ArcSin = 12] = "ArcSin", A[A.Sign = 13] = "Sign", A[A.Negate = 14] = "Negate", A[A.OneMinus = 15] = "OneMinus", A[A.Reciprocal = 16] = "Reciprocal", A[A.ToDegrees = 17] = "ToDegrees", A[A.ToRadians = 18] = "ToRadians"; })(Bs || (Bs = {})); class SY extends yi { /** * Creates a new GeometryTrigonometryBlock * @param name defines the block name */ constructor(e) { super(e), this.operation = Bs.Cos, this.registerInput("input", Me.AutoDetect), this.registerOutput("output", Me.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._inputs[0].excludedConnectionPointTypes.push(Me.Matrix), this._inputs[0].excludedConnectionPointTypes.push(Me.Geometry), this._inputs[0].excludedConnectionPointTypes.push(Me.Texture); } /** * Gets the current class name * @returns the class name */ getClassName() { return "GeometryTrigonometryBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e); let t = null; switch (this.operation) { case Bs.Cos: { t = (r) => Math.cos(r); break; } case Bs.Sin: { t = (r) => Math.sin(r); break; } case Bs.Abs: { t = (r) => Math.abs(r); break; } case Bs.Exp: { t = (r) => Math.exp(r); break; } case Bs.Round: { t = (r) => Math.round(r); break; } case Bs.Floor: { t = (r) => Math.floor(r); break; } case Bs.Ceiling: { t = (r) => Math.ceil(r); break; } case Bs.Sqrt: { t = (r) => Math.sqrt(r); break; } case Bs.Log: { t = (r) => Math.log(r); break; } case Bs.Tan: { t = (r) => Math.tan(r); break; } case Bs.ArcTan: { t = (r) => Math.atan(r); break; } case Bs.ArcCos: { t = (r) => Math.acos(r); break; } case Bs.ArcSin: { t = (r) => Math.asin(r); break; } case Bs.Sign: { t = (r) => Math.sign(r); break; } case Bs.Negate: { t = (r) => -r; break; } case Bs.OneMinus: { t = (r) => 1 - r; break; } case Bs.Reciprocal: { t = (r) => 1 / r; break; } case Bs.ToRadians: { t = (r) => r * Math.PI / 180; break; } case Bs.ToDegrees: { t = (r) => r * 180 / Math.PI; break; } } if (!t) { this.input._storedFunction = null, this.input._storedValue = null; return; } switch (this.input.type) { case Me.Int: case Me.Float: { this.output._storedFunction = (r) => { const n = this.input.getConnectedValue(r); return t(n); }; break; } case Me.Vector2: { this.output._storedFunction = (r) => { const n = this.input.getConnectedValue(r); return new at(t(n.x), t(n.y)); }; break; } case Me.Vector3: { this.output._storedFunction = (r) => { const n = this.input.getConnectedValue(r); return new S(t(n.x), t(n.y), t(n.z)); }; break; } case Me.Vector4: { this.output._storedFunction = (r) => { const n = this.input.getConnectedValue(r); return new Ir(t(n.x), t(n.y), t(n.z), t(n.w)); }; break; } } return this; } serialize() { const e = super.serialize(); return e.operation = this.operation, e; } _deserialize(e) { super._deserialize(e), this.operation = e.operation; } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.operation = BABYLON.GeometryTrigonometryBlockOperations.${Bs[this.operation]}; `; } } C([ rn("Operation", Gr.List, "ADVANCED", { notifiers: { rebuild: !0 }, options: [ { label: "Cos", value: Bs.Cos }, { label: "Sin", value: Bs.Sin }, { label: "Abs", value: Bs.Abs }, { label: "Exp", value: Bs.Exp }, { label: "Round", value: Bs.Round }, { label: "Floor", value: Bs.Floor }, { label: "Ceiling", value: Bs.Ceiling }, { label: "Sqrt", value: Bs.Sqrt }, { label: "Log", value: Bs.Log }, { label: "Tan", value: Bs.Tan }, { label: "ArcTan", value: Bs.ArcTan }, { label: "ArcCos", value: Bs.ArcCos }, { label: "ArcSin", value: Bs.ArcSin }, { label: "Sign", value: Bs.Sign }, { label: "Negate", value: Bs.Negate }, { label: "OneMinus", value: Bs.OneMinus }, { label: "Reciprocal", value: Bs.Reciprocal }, { label: "ToDegrees", value: Bs.ToDegrees }, { label: "ToRadians", value: Bs.ToRadians } ] }) ], SY.prototype, "operation", void 0); Ue("BABYLON.GeometryTrigonometryBlock", SY); class UY extends yi { /** * Create a new GeometryTransformBlock * @param name defines the block name */ constructor(e) { super(e), this._rotationMatrix = new he(), this._scalingMatrix = new he(), this._translationMatrix = new he(), this._scalingRotationMatrix = new he(), this._transformMatrix = new he(), this.evaluateContext = !0, this.registerInput("value", Me.AutoDetect), this.registerInput("matrix", Me.Matrix, !0), this.registerInput("translation", Me.Vector3, !0, S.Zero()), this.registerInput("rotation", Me.Vector3, !0, S.Zero()), this.registerInput("scaling", Me.Vector3, !0, S.One()), this.registerOutput("output", Me.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._inputs[0].excludedConnectionPointTypes.push(Me.Float), this._inputs[0].excludedConnectionPointTypes.push(Me.Matrix), this._inputs[0].excludedConnectionPointTypes.push(Me.Texture); } /** * Gets the current class name * @returns the class name */ getClassName() { return "GeometryTransformBlock"; } /** * Gets the value input component */ get value() { return this._inputs[0]; } /** * Gets the matrix input component */ get matrix() { return this._inputs[1]; } /** * Gets the translation input component */ get translation() { return this._inputs[2]; } /** * Gets the rotation input component */ get rotation() { return this._inputs[3]; } /** * Gets the scaling input component */ get scaling() { return this._inputs[4]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { if (!this.value.isConnected) { this.output._storedFunction = null, this.output._storedValue = null; return; } const t = (r) => { const n = this.value.getConnectedValue(r); if (!n) return null; let i; if (this.matrix.isConnected) i = this.matrix.getConnectedValue(r); else { const s = this.scaling.getConnectedValue(r), a = this.rotation.getConnectedValue(r), f = this.translation.getConnectedValue(r); he.ScalingToRef(s.x, s.y, s.z, this._scalingMatrix), he.RotationYawPitchRollToRef(a.y, a.x, a.z, this._rotationMatrix), he.TranslationToRef(f.x, f.y, f.z, this._translationMatrix), this._scalingMatrix.multiplyToRef(this._rotationMatrix, this._scalingRotationMatrix), this._scalingRotationMatrix.multiplyToRef(this._translationMatrix, this._transformMatrix), i = this._transformMatrix; } switch (this.value.type) { case Me.Geometry: { const s = n.clone(); return s.transform(i), s; } case Me.Vector2: return at.Transform(n, i); case Me.Vector3: return S.TransformCoordinates(n, i); case Me.Vector4: return Ir.TransformCoordinates(n, i); } return null; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), e.evaluateContext !== void 0 && (this.evaluateContext = e.evaluateContext); } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], UY.prototype, "evaluateContext", void 0); Ue("BABYLON.GeometryTransformBlock", UY); class mne extends yi { /** * Create a new RotationXBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("angle", Me.Float, !1, 0), this.registerOutput("matrix", Me.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "RotationXBlock"; } /** * Gets the angle input component */ get angle() { return this._inputs[0]; } /** * Gets the matrix output component */ get matrix() { return this._outputs[0]; } autoConfigure() { if (!this.angle.isConnected) { const e = new xf("Angle"); e.value = 0, e.output.connectTo(this.angle); } } _buildBlock(e) { super._buildBlock(e), this.matrix._storedFunction = (t) => he.RotationX(this.angle.getConnectedValue(t)); } } Ue("BABYLON.RotationXBlock", mne); class Bne extends yi { /** * Create a new RotationYBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("angle", Me.Float, !1, 0), this.registerOutput("matrix", Me.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "RotationYBlock"; } /** * Gets the angle input component */ get angle() { return this._inputs[0]; } /** * Gets the matrix output component */ get matrix() { return this._outputs[0]; } autoConfigure() { if (!this.angle.isConnected) { const e = new xf("Angle"); e.value = 0, e.output.connectTo(this.angle); } } _buildBlock(e) { super._buildBlock(e), this.matrix._storedFunction = (t) => he.RotationY(this.angle.getConnectedValue(t)); } } Ue("BABYLON.RotationYBlock", Bne); class Wne extends yi { /** * Create a new RotationZBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("angle", Me.Float, !1, 0), this.registerOutput("matrix", Me.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "RotationZBlock"; } /** * Gets the angle input component */ get angle() { return this._inputs[0]; } /** * Gets the matrix output component */ get matrix() { return this._outputs[0]; } autoConfigure() { if (!this.angle.isConnected) { const e = new xf("Angle"); e.value = 0, e.output.connectTo(this.angle); } } _buildBlock(e) { super._buildBlock(e), this.matrix._storedFunction = (t) => he.RotationZ(this.angle.getConnectedValue(t)); } } Ue("BABYLON.RotationZBlock", Wne); class Sne extends yi { /** * Create a new ScalingBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("scale", Me.Vector3, !1, S.One()), this.registerOutput("matrix", Me.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "ScalingBlock"; } /** * Gets the scale input component */ get scale() { return this._inputs[0]; } /** * Gets the matrix output component */ get matrix() { return this._outputs[0]; } autoConfigure() { if (!this.scale.isConnected) { const e = new xf("Scale"); e.value = new S(1, 1, 1), e.output.connectTo(this.scale); } } _buildBlock(e) { super._buildBlock(e), this.matrix._storedFunction = (t) => { const r = this.scale.getConnectedValue(t); return he.Scaling(r.x, r.y, r.z); }; } } Ue("BABYLON.ScalingBlock", Sne); class Une extends yi { /** * Create a new AlignBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("source", Me.Vector3, !0, S.Up()), this.registerInput("target", Me.Vector3, !0, S.Left()), this.registerOutput("matrix", Me.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "AlignBlock"; } /** * Gets the source input component */ get source() { return this._inputs[0]; } /** * Gets the target input component */ get target() { return this._inputs[1]; } /** * Gets the matrix output component */ get matrix() { return this._outputs[0]; } _buildBlock(e) { super._buildBlock(e), this.matrix._storedFunction = (t) => { const r = this.source.getConnectedValue(t).clone(), n = this.target.getConnectedValue(t).clone(), i = new he(); return r.normalize(), n.normalize(), he.RotationAlignToRef(r, n, i, !0), i; }; } } Ue("BABYLON.AlignBlock", Une); class Ine extends yi { /** * Create a new TranslationBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("translation", Me.Vector3, !1, S.Zero()), this.registerOutput("matrix", Me.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "TranslationBlock"; } /** * Gets the translation input component */ get translation() { return this._inputs[0]; } /** * Gets the matrix output component */ get matrix() { return this._outputs[0]; } autoConfigure() { if (!this.translation.isConnected) { const e = new xf("Translation"); e.value = new S(0, 0, 0), e.output.connectTo(this.translation); } } _buildBlock(e) { super._buildBlock(e), this.matrix._storedFunction = (t) => { const r = this.translation.getConnectedValue(t); return he.Translation(r.x, r.y, r.z); }; } } Ue("BABYLON.TranslationBlock", Ine); class by extends yi { /** * Create a new InstantiateOnVerticesBlock * @param name defines the block name */ constructor(e) { super(e), this._indexTranslation = null, this.evaluateContext = !0, this.removeDuplicatedPositions = !0, this.registerInput("geometry", Me.Geometry), this.registerInput("instance", Me.Geometry, !0), this.registerInput("density", Me.Float, !0, 1, 0, 1), this.registerInput("matrix", Me.Matrix, !0), this.registerInput("rotation", Me.Vector3, !0, S.Zero()), this.registerInput("scaling", Me.Vector3, !0, S.One()), this.scaling.acceptedConnectionPointTypes.push(Me.Float), this.registerOutput("output", Me.Geometry); } /** * Gets the current instance index in the current flow * @returns the current index */ getInstanceIndex() { return this._currentLoopIndex; } /** * Gets the current index in the current flow * @returns the current index */ getExecutionIndex() { return this._indexTranslation ? this._indexTranslation[this._currentIndex] : this._currentIndex; } /** * Gets the current loop index in the current flow * @returns the current loop index */ getExecutionLoopIndex() { return this._currentLoopIndex; } /** * Gets the current face index in the current flow * @returns the current face index */ getExecutionFaceIndex() { return 0; } /** * Gets the current class name * @returns the class name */ getClassName() { return "InstantiateOnVerticesBlock"; } /** * Gets the geometry input component */ get geometry() { return this._inputs[0]; } /** * Gets the instance input component */ get instance() { return this._inputs[1]; } /** * Gets the density input component */ get density() { return this._inputs[2]; } /** * Gets the matrix input component */ get matrix() { return this._inputs[3]; } /** * Gets the rotation input component */ get rotation() { return this._inputs[4]; } /** * Gets the scaling input component */ get scaling() { return this._inputs[5]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } _buildBlock(e) { const t = (r) => { if (r.pushExecutionContext(this), r.pushInstancingContext(this), this._vertexData = this.geometry.getConnectedValue(r), r.pushGeometryContext(this._vertexData), !this._vertexData || !this._vertexData.positions || !this.instance.isConnected) { r.restoreExecutionContext(), r.restoreInstancingContext(), r.restoreGeometryContext(), this.output._storedValue = null; return; } let n = this._vertexData.positions.length / 3; const i = [], s = new S(), a = []; let f = this._vertexData.positions; if (this._currentLoopIndex = 0, this.removeDuplicatedPositions) { for (this._indexTranslation = {}, this._currentIndex = 0; this._currentIndex < n; this._currentIndex++) { const o = f[this._currentIndex * 3], d = f[this._currentIndex * 3 + 1], v = f[this._currentIndex * 3 + 2]; let u = !1; for (let l = 0; l < a.length; l += 3) if (Math.abs(a[l] - o) < Dn && Math.abs(a[l + 1] - d) < Dn && Math.abs(a[l + 2] - v) < Dn) { u = !0; break; } u || (this._indexTranslation[a.length / 3] = this._currentIndex, a.push(o, d, v)); } f = a, n = f.length / 3; } else this._indexTranslation = null; for (this._currentIndex = 0; this._currentIndex < n; this._currentIndex++) { const o = this.instance.getConnectedValue(r); if (!o || !o.positions || o.positions.length === 0) continue; const d = this.density.getConnectedValue(r); if (d < 1 && Math.random() > d) continue; s.fromArray(f, this._currentIndex * 3); const v = o.clone(); if (this.matrix.isConnected) { const u = this.matrix.getConnectedValue(r); r._instantiateWithPositionAndMatrix(v, s, u, i); } else { const u = r.adaptInput(this.scaling, Me.Vector3, S.OneReadOnly), l = this.rotation.getConnectedValue(r) || S.ZeroReadOnly; r._instantiate(v, s, l, u, i); } this._currentLoopIndex++; } if (r.restoreGeometryContext(), r.restoreExecutionContext(), r.restoreInstancingContext(), i.length) if (i.length === 1) this._vertexData = i[0]; else { const o = i.splice(0, 1)[0]; this._vertexData = o.merge(i, !0, !1, !0, !0); } else return null; return this._vertexData; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } _dumpPropertiesCode() { let e = super._dumpPropertiesCode() + `${this._codeVariableName}.removeDuplicatedPositions = ${this.removeDuplicatedPositions ? "true" : "false"}; `; return e += `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `, e; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.removeDuplicatedPositions = this.removeDuplicatedPositions, e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), this.removeDuplicatedPositions = e.removeDuplicatedPositions, e.evaluateContext !== void 0 && (this.evaluateContext = e.evaluateContext); } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], by.prototype, "evaluateContext", void 0); C([ rn("Remove duplicated positions", Gr.Boolean, "ADVANCED", { notifiers: { update: !0 } }) ], by.prototype, "removeDuplicatedPositions", void 0); Ue("BABYLON.InstantiateOnVerticesBlock", by); class IY extends yi { /** * Create a new InstantiateOnFacesBlock * @param name defines the block name */ constructor(e) { super(e), this._currentPosition = new S(), this._currentUV = new at(), this._vertex0 = new S(), this._vertex1 = new S(), this._vertex2 = new S(), this._tempVector0 = new S(), this._tempVector1 = new S(), this._uv0 = new at(), this._uv1 = new at(), this._uv2 = new at(), this.evaluateContext = !0, this.registerInput("geometry", Me.Geometry), this.registerInput("instance", Me.Geometry, !0), this.registerInput("count", Me.Int, !0, 256), this.registerInput("matrix", Me.Matrix, !0), this.registerInput("rotation", Me.Vector3, !0, S.Zero()), this.registerInput("scaling", Me.Vector3, !0, S.One()), this.scaling.acceptedConnectionPointTypes.push(Me.Float), this.registerOutput("output", Me.Geometry); } /** * Gets the current instance index in the current flow * @returns the current index */ getInstanceIndex() { return this._currentLoopIndex; } /** * Gets the current index in the current flow * @returns the current index */ getExecutionIndex() { return 0; } /** * Gets the current face index in the current flow * @returns the current face index */ getExecutionFaceIndex() { return this._currentFaceIndex; } /** * Gets the current loop index in the current flow * @returns the current loop index */ getExecutionLoopIndex() { return this._currentLoopIndex; } /** * Gets the value associated with a contextual positions * @returns the value associated with the source */ getOverridePositionsContextualValue() { return this._currentPosition; } /** * Gets the value associated with a contextual normals * @returns the value associated with the source */ getOverrideNormalsContextualValue() { return this._vertex1.subtractToRef(this._vertex0, this._tempVector0), this._vertex2.subtractToRef(this._vertex1, this._tempVector1), this._tempVector0.normalize(), this._tempVector1.normalize(), S.Cross(this._tempVector1, this._tempVector0); } /** * Gets the value associated with a contextual UV1 se * @returns the value associated with the source */ getOverrideUVs1ContextualValue() { return this._currentUV; } /** * Gets the current class name * @returns the class name */ getClassName() { return "InstantiateOnFacesBlock"; } /** * Gets the geometry input component */ get geometry() { return this._inputs[0]; } /** * Gets the instance input component */ get instance() { return this._inputs[1]; } /** * Gets the count input component */ get count() { return this._inputs[2]; } /** * Gets the matrix input component */ get matrix() { return this._inputs[3]; } /** * Gets the rotation input component */ get rotation() { return this._inputs[4]; } /** * Gets the scaling input component */ get scaling() { return this._inputs[5]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } _buildBlock(e) { const t = (r) => { if (r.pushExecutionContext(this), r.pushInstancingContext(this), this._vertexData = this.geometry.getConnectedValue(r), r.pushGeometryContext(this._vertexData), !this._vertexData || !this._vertexData.positions || !this._vertexData.indices || !this.instance.isConnected) { r.restoreExecutionContext(), r.restoreInstancingContext(), r.restoreGeometryContext(), this.output._storedValue = null; return; } let n = null; const i = this.count.getConnectedValue(r), s = this._vertexData.indices.length / 3, a = i / s; let f = 0; const o = []; let d = 0; for (this._currentLoopIndex = 0, this._currentFaceIndex = 0; this._currentFaceIndex < s; this._currentFaceIndex++) { f += a; const v = (f | 0) - d; if (v < 1) continue; const u = this._vertexData.indices[this._currentFaceIndex * 3], l = this._vertexData.indices[this._currentFaceIndex * 3 + 1], P = this._vertexData.indices[this._currentFaceIndex * 3 + 2]; this._vertex0.fromArray(this._vertexData.positions, u * 3), this._vertex1.fromArray(this._vertexData.positions, l * 3), this._vertex2.fromArray(this._vertexData.positions, P * 3), this._vertexData.uvs && (this._uv0.fromArray(this._vertexData.uvs, u * 2), this._uv1.fromArray(this._vertexData.uvs, l * 2), this._uv2.fromArray(this._vertexData.uvs, P * 2)); for (let p = 0; p < v && !(d >= i); p++) { let c = Math.random(), H = Math.random(); if (c > H) { const w = c; c = H, H = w; } const T = c, q = H - c, b = 1 - T - q; if (this._currentPosition.set(T * this._vertex0.x + q * this._vertex1.x + b * this._vertex2.x, T * this._vertex0.y + q * this._vertex1.y + b * this._vertex2.y, T * this._vertex0.z + q * this._vertex1.z + b * this._vertex2.z), this._vertexData.uvs && this._currentUV.set(T * this._uv0.x + q * this._uv1.x + b * this._uv2.x, T * this._uv0.y + q * this._uv1.y + b * this._uv2.y), n = this.instance.getConnectedValue(r), !n || !n.positions || n.positions.length === 0) { f -= a; continue; } const j = n.clone(); if (this.matrix.isConnected) { const w = this.matrix.getConnectedValue(r); r._instantiateWithPositionAndMatrix(j, this._currentPosition, w, o); } else { const w = r.adaptInput(this.scaling, Me.Vector3, S.OneReadOnly), m = this.rotation.getConnectedValue(r) || S.ZeroReadOnly; r._instantiate(j, this._currentPosition, m, w, o); } d++, this._currentLoopIndex++; } } if (o.length) if (o.length === 1) this._vertexData = o[0]; else { const v = o.splice(0, 1)[0]; this._vertexData = v.merge(o, !0, !1, !0, !0); } return r.restoreExecutionContext(), r.restoreInstancingContext(), r.restoreGeometryContext(), this._vertexData; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), e.evaluateContext !== void 0 && (this.evaluateContext = e.evaluateContext); } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], IY.prototype, "evaluateContext", void 0); Ue("BABYLON.InstantiateOnFacesBlock", IY); class RY extends yi { /** * Create a new InstantiateOnVolumeBlock * @param name defines the block name */ constructor(e) { super(e), this._currentPosition = new S(), this._vertex0 = new S(), this._vertex1 = new S(), this._vertex2 = new S(), this.evaluateContext = !0, this.registerInput("geometry", Me.Geometry), this.registerInput("instance", Me.Geometry, !0), this.registerInput("count", Me.Int, !0, 256), this.registerInput("matrix", Me.Matrix, !0), this.registerInput("rotation", Me.Vector3, !0, S.Zero()), this.registerInput("scaling", Me.Vector3, !0, S.One()), this.scaling.acceptedConnectionPointTypes.push(Me.Float), this.registerOutput("output", Me.Geometry); } /** * Gets the current instance index in the current flow * @returns the current index */ getInstanceIndex() { return this._currentLoopIndex; } /** * Gets the current index in the current flow * @returns the current index */ getExecutionIndex() { return 0; } /** * Gets the current face index in the current flow * @returns the current face index */ getExecutionFaceIndex() { return 0; } /** * Gets the current loop index in the current flow * @returns the current loop index */ getExecutionLoopIndex() { return this._currentLoopIndex; } /** * Gets the value associated with a contextual positions * @returns the value associated with the source */ getOverridePositionsContextualValue() { return this._currentPosition; } /** * Gets the current class name * @returns the class name */ getClassName() { return "InstantiateOnVolumeBlock"; } /** * Gets the geometry input component */ get geometry() { return this._inputs[0]; } /** * Gets the instance input component */ get instance() { return this._inputs[1]; } /** * Gets the count input component */ get count() { return this._inputs[2]; } /** * Gets the matrix input component */ get matrix() { return this._inputs[3]; } /** * Gets the rotation input component */ get rotation() { return this._inputs[4]; } /** * Gets the scaling input component */ get scaling() { return this._inputs[5]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } _buildBlock(e) { const t = (r) => { if (r.pushExecutionContext(this), r.pushInstancingContext(this), this._vertexData = this.geometry.getConnectedValue(r), r.pushGeometryContext(this._vertexData), !this._vertexData || !this._vertexData.positions || !this._vertexData.indices || !this.instance.isConnected) { r.restoreExecutionContext(), r.restoreInstancingContext(), r.restoreGeometryContext(), this.output._storedValue = null; return; } let n = null; const i = this.count.getConnectedValue(r), s = [], a = WS(this._vertexData.positions, 0, this._vertexData.positions.length / 3), f = a.minimum, o = a.maximum, d = new S(1, 0, 0), v = this._vertexData.indices.length / 3; this._currentLoopIndex = 0; for (let u = 0; u < i; u++) { this._currentPosition.set(Math.random() * (o.x - f.x) + f.x, Math.random() * (o.y - f.y) + f.y, Math.random() * (o.z - f.z) + f.z); const l = new Hi(this._currentPosition, d); let P = 0; for (let c = 0; c < v; c++) { this._vertex0.fromArray(this._vertexData.positions, this._vertexData.indices[c * 3] * 3), this._vertex1.fromArray(this._vertexData.positions, this._vertexData.indices[c * 3 + 1] * 3), this._vertex2.fromArray(this._vertexData.positions, this._vertexData.indices[c * 3 + 2] * 3); const H = l.intersectsTriangle(this._vertex0, this._vertex1, this._vertex2); H && H.distance > 0 && P++; } if (P % 2 === 0) { u--; continue; } if (n = this.instance.getConnectedValue(r), !n || !n.positions || n.positions.length === 0) continue; const p = n.clone(); if (this.matrix.isConnected) { const c = this.matrix.getConnectedValue(r); r._instantiateWithPositionAndMatrix(p, this._currentPosition, c, s); } else { const c = r.adaptInput(this.scaling, Me.Vector3, S.OneReadOnly), H = this.rotation.getConnectedValue(r) || S.ZeroReadOnly; r._instantiate(p, this._currentPosition, H, c, s); } this._currentLoopIndex++; } if (s.length) if (s.length === 1) this._vertexData = s[0]; else { const u = s.splice(0, 1)[0]; this._vertexData = u.merge(s, !0, !1, !0, !0); } return r.restoreGeometryContext(), r.restoreExecutionContext(), r.restoreInstancingContext(), this._vertexData; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), e.evaluateContext !== void 0 && (this.evaluateContext = e.evaluateContext); } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], RY.prototype, "evaluateContext", void 0); Ue("BABYLON.InstantiateOnVolumeBlock", RY); class xy extends yi { /** * Create a new InstantiateBaseBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !0, this.registerInput("instance", Me.Geometry, !0), this.registerInput("count", Me.Int, !0, 1), this.registerOutput("output", Me.Geometry); } /** * Gets the current instance index in the current flow * @returns the current index */ getInstanceIndex() { return this._currentIndex; } /** * Gets the current index in the current flow * @returns the current index */ getExecutionIndex() { return this._currentIndex; } /** * Gets the current loop index in the current flow * @returns the current loop index */ getExecutionLoopIndex() { return this._currentIndex; } /** * Gets the current face index in the current flow * @returns the current face index */ getExecutionFaceIndex() { return 0; } /** * Gets the current class name * @returns the class name */ getClassName() { return "InstantiateBaseBlock"; } /** * Gets the instance input component */ get instance() { return this._inputs[0]; } /** * Gets the count input component */ get count() { return this._inputs[1]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e; } _deserialize(e) { super._deserialize(e), e.evaluateContext !== void 0 && (this.evaluateContext = e.evaluateContext); } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], xy.prototype, "evaluateContext", void 0); class Rne extends xy { /** * Create a new InstantiateBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("matrix", Me.Matrix, !0), this.registerInput("position", Me.Vector3, !0, S.Zero()), this.registerInput("rotation", Me.Vector3, !0, S.Zero()), this.registerInput("scaling", Me.Vector3, !0, S.One()), this.scaling.acceptedConnectionPointTypes.push(Me.Float); } /** * Gets the current instance index in the current flow * @returns the current index */ getInstanceIndex() { return this._currentIndex; } /** * Gets the current index in the current flow * @returns the current index */ getExecutionIndex() { return this._currentIndex; } /** * Gets the current loop index in the current flow * @returns the current loop index */ getExecutionLoopIndex() { return this._currentIndex; } /** * Gets the current face index in the current flow * @returns the current face index */ getExecutionFaceIndex() { return 0; } /** * Gets the current class name * @returns the class name */ getClassName() { return "InstantiateBlock"; } /** * Gets the matrix input component */ get matrix() { return this._inputs[2]; } /** * Gets the position input component */ get position() { return this._inputs[3]; } /** * Gets the rotation input component */ get rotation() { return this._inputs[4]; } /** * Gets the scaling input component */ get scaling() { return this._inputs[5]; } _buildBlock(e) { const t = (r) => { r.pushExecutionContext(this), r.pushInstancingContext(this); const n = this.count.getConnectedValue(r), i = []; for (this._currentIndex = 0; this._currentIndex < n; this._currentIndex++) { const s = this.instance.getConnectedValue(r); if (!s || !s.positions || s.positions.length === 0) continue; const a = s.clone(); if (this.matrix.isConnected) { const f = this.matrix.getConnectedValue(r); r._instantiateWithMatrix(a, f, i); } else { const f = this.position.getConnectedValue(r) || S.ZeroReadOnly, o = r.adaptInput(this.scaling, Me.Vector3, S.OneReadOnly), d = this.rotation.getConnectedValue(r) || S.ZeroReadOnly; r._instantiate(a, f, d, o, i); } } if (i.length) if (i.length === 1) this._vertexData = i[0]; else { const s = i.splice(0, 1)[0]; this._vertexData = s.merge(i, !0, !1, !0, !0); } return r.restoreExecutionContext(), r.restoreInstancingContext(), this._vertexData; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } } Ue("BABYLON.InstantiateBlock", Rne); class Vne extends xy { /** * Create a new Instantiate Linear Block * @param name defines the block name */ constructor(e) { super(e), this.registerInput("direction", Me.Vector3, !0, new S(1, 0, 0)), this.registerInput("rotation", Me.Vector3, !0, new S(0, 0, 0)), this.registerInput("scaling", Me.Vector3, !0, new S(0, 0, 0)), this.scaling.acceptedConnectionPointTypes.push(Me.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "InstantiateLinearBlock"; } /** * Gets the direction input component */ get direction() { return this._inputs[2]; } /** * Gets the rotation input component */ get rotation() { return this._inputs[3]; } /** * Gets the scaling input component */ get scaling() { return this._inputs[4]; } _buildBlock(e) { const t = (r) => { r.pushExecutionContext(this), r.pushInstancingContext(this); const n = this.count.getConnectedValue(r), i = [], s = he.Identity(), a = S.Zero(), f = S.Zero(), o = S.Zero(); for (this._currentIndex = 0; this._currentIndex < n; this._currentIndex++) { const d = this.instance.getConnectedValue(r); if (!d || !d.positions || d.positions.length === 0) continue; const v = d.clone(), u = this.direction.getConnectedValue(r), l = this.rotation.getConnectedValue(r), P = r.adaptInput(this.scaling, Me.Vector3, S.OneReadOnly); a.copyFrom(u.clone().scale(this._currentIndex)), f.copyFrom(l.clone().scale(this._currentIndex)), o.copyFrom(P.clone().scale(this._currentIndex)), o.addInPlaceFromFloats(1, 1, 1), he.ComposeToRef(o, Ze.FromEulerAngles(f.x, f.y, f.z), a, s), r._instantiateWithMatrix(v, s, i); } if (i.length) if (i.length === 1) this._vertexData = i[0]; else { const d = i.splice(0, 1)[0]; this._vertexData = d.merge(i, !0, !1, !0, !0); } return r.restoreExecutionContext(), r.restoreInstancingContext(), this._vertexData; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } } Ue("BABYLON.InstantiateLinearBlock", Vne); class Cne extends xy { /** * Create a new InstantiateRadialBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("radius", Me.Int, !0, 0, 0), this.registerInput("angleStart", Me.Float, !0, 0), this.registerInput("angleEnd", Me.Float, !0, Math.PI * 2), this.registerInput("transform", Me.Vector3, !0, new S(0, 0, 0)), this.registerInput("rotation", Me.Vector3, !0, new S(0, 0, 0)), this.registerInput("scaling", Me.Vector3, !0, new S(0, 0, 0)), this.scaling.acceptedConnectionPointTypes.push(Me.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "InstantiateRadialBlock"; } /** * Gets the direction input component */ get radius() { return this._inputs[2]; } /** * Gets the direction input component */ get angleStart() { return this._inputs[3]; } /** * Gets the direction input component */ get angleEnd() { return this._inputs[4]; } /** * Gets the transform input component */ get transform() { return this._inputs[5]; } /** * Gets the rotation input component */ get rotation() { return this._inputs[6]; } /** * Gets the scaling input component */ get scaling() { return this._inputs[7]; } _buildBlock(e) { const t = (r) => { r.pushExecutionContext(this), r.pushInstancingContext(this); const n = this.count.getConnectedValue(r), i = [], s = he.Identity(), a = he.Identity(), f = he.Identity(), o = S.Zero(), d = S.Zero(), v = S.Zero(); for (this._currentIndex = 0; this._currentIndex < n; this._currentIndex++) { const u = this.instance.getConnectedValue(r); if (!u || !u.positions || u.positions.length === 0) continue; const l = u.clone(), P = this.radius.getConnectedValue(r), p = this.angleStart.getConnectedValue(r), c = this.angleEnd.getConnectedValue(r), H = this.transform.getConnectedValue(r), T = this.rotation.getConnectedValue(r), q = r.adaptInput(this.scaling, Me.Vector3, S.OneReadOnly), j = (c - p) / n, w = p + j * this._currentIndex, m = Ze.FromEulerAngles(0, w, 0); o.copyFrom(H.clone().scale(this._currentIndex)), d.copyFrom(T.clone().scale(this._currentIndex)), v.copyFrom(q.clone().scale(this._currentIndex)), v.addInPlaceFromFloats(1, 1, 1), he.RotationYawPitchRollToRef(d.y, d.x, d.z, s), a.setTranslationFromFloats(0, 0, P), he.ComposeToRef(v, m, o, f), s.multiplyToRef(a, a), a.multiplyToRef(f, f), r._instantiateWithMatrix(l, f, i); } if (i.length) if (i.length === 1) this._vertexData = i[0]; else { const u = i.splice(0, 1)[0]; this._vertexData = u.merge(i, !0, !1, !0, !0); } return r.restoreExecutionContext(), r.restoreInstancingContext(), this._vertexData; }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } } Ue("BABYLON.InstantiateRadialBlock", Cne); class One extends yi { /** * Create a new IntFloatConverterBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("float ", Me.Float, !0), this.registerInput("int ", Me.Int, !0), this.registerOutput("float", Me.Float), this.registerOutput("int", Me.Int); } /** * Gets the current class name * @returns the class name */ getClassName() { return "IntFloatConverterBlock"; } /** * Gets the float input component */ get floatIn() { return this._inputs[0]; } /** * Gets the int input component */ get intIn() { return this._inputs[1]; } /** * Gets the float output component */ get floatOut() { return this._outputs[0]; } /** * Gets the int output component */ get intOut() { return this._outputs[1]; } _inputRename(e) { return e === "float " ? "floatIn" : e === "int " ? "intIn" : e; } _buildBlock() { this.floatOut._storedFunction = (e) => this.floatIn.isConnected ? this.floatIn.getConnectedValue(e) : this.intIn.isConnected ? this.intIn.getConnectedValue(e) : 0, this.intOut._storedFunction = (e) => this.floatIn.isConnected ? Math.floor(this.floatIn.getConnectedValue(e)) : this.intIn.isConnected ? Math.floor(this.intIn.getConnectedValue(e)) : 0; } } Ue("BABYLON.IntFloatConverterBlock", One); class yne extends yi { /** * Create a new DebugBlock * @param name defines the block name */ constructor(e) { super(e), this.log = [], this._isDebug = !0, this.registerInput("input", Me.AutoDetect), this.registerOutput("output", Me.BasedOnInput), this._outputs[0]._typeConnectionSource = this._inputs[0], this._inputs[0].excludedConnectionPointTypes.push(Me.Geometry), this._inputs[0].excludedConnectionPointTypes.push(Me.Texture); } /** * Gets the time spent to build this block (in ms) */ get buildExecutionTime() { return 0; } /** * Gets the current class name * @returns the class name */ getClassName() { return "DebugBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock(e) { if (!this.input.isConnected) { this.output._storedFunction = null, this.output._storedValue = null; return; } this.log = []; const t = (r) => { const n = this.input.getConnectedValue(r); return n == null ? (this.log.push("null"), n) : (this.log.push(n.toString()), n); }; this.output.isConnected ? this.output._storedFunction = t : this.output._storedValue = t(e); } } Ue("BABYLON.DebugBlock", yne); class kne extends yi { /** * Create a new GeometryInfoBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("geometry", Me.Geometry), this.registerOutput("output", Me.Geometry), this.registerOutput("id", Me.Int), this.registerOutput("collectionId", Me.Int), this.registerOutput("verticesCount", Me.Int), this.registerOutput("facesCount", Me.Int); } /** * Gets the current class name * @returns the class name */ getClassName() { return "GeometryInfoBlock"; } /** * Gets the geometry input component */ get geometry() { return this._inputs[0]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } /** * Gets the id output component */ get id() { return this._outputs[1]; } /** * Gets the collectionId output component */ get collectionId() { return this._outputs[2]; } /** * Gets the verticesCount output component */ get verticesCount() { return this._outputs[3]; } /** * Gets the facesCount output component */ get facesCount() { return this._outputs[4]; } _buildBlock() { if (!this.geometry.isConnected) { this.id._storedValue = 0, this.collectionId._storedValue = 0, this.verticesCount._storedValue = 0, this.facesCount._storedValue = 0, this.output._storedValue = 0, this.id._storedFunction = null, this.collectionId._storedFunction = null, this.verticesCount._storedFunction = null, this.facesCount._storedFunction = null, this.output._storedFunction = null; return; } this.output._storedFunction = (e) => (this._currentVertexData = this.geometry.getConnectedValue(e), this._currentVertexData), this.id._storedFunction = (e) => (this._currentVertexData = this._currentVertexData || this.geometry.getConnectedValue(e), this._currentVertexData.uniqueId), this.collectionId._storedFunction = (e) => (this._currentVertexData = this._currentVertexData || this.geometry.getConnectedValue(e), this._currentVertexData.metadata ? this._currentVertexData.metadata.collectionId : 0), this.verticesCount._storedFunction = (e) => (this._currentVertexData = this._currentVertexData || this.geometry.getConnectedValue(e), this._currentVertexData.positions ? this._currentVertexData.positions.length / 3 : 0), this.facesCount._storedFunction = (e) => (this._currentVertexData = this._currentVertexData || this.geometry.getConnectedValue(e), this._currentVertexData.indices ? this._currentVertexData.indices.length / 3 : 0); } } Ue("BABYLON.GeometryInfoBlock", kne); var GH; (function(A) { A[A.Spherical = 0] = "Spherical", A[A.Cylindrical = 1] = "Cylindrical", A[A.Cubic = 2] = "Cubic"; })(GH || (GH = {})); class VY extends yi { /** * Create a new MappingBlock * @param name defines the block name */ constructor(e) { super(e), this.mapping = GH.Spherical, this.registerInput("position", Me.Vector3), this.registerInput("normal", Me.Vector3), this.registerInput("center", Me.Vector3, !0, S.Zero()), this.registerOutput("uv", Me.Vector2); } /** * Gets the current class name * @returns the class name */ getClassName() { return "MappingBlock"; } /** * Gets the position input component */ get position() { return this._inputs[0]; } /** * Gets the normal input component */ get normal() { return this._inputs[1]; } /** * Gets the center input component */ get center() { return this._inputs[2]; } /** * Gets the output component */ get uv() { return this._outputs[0]; } _buildBlock() { if (!this.position.isConnected) { this.uv._storedFunction = null, this.uv._storedValue = null; return; } const e = S.Zero(), t = (r) => { const n = this.position.getConnectedValue(r) || S.Zero(), i = this.normal.getConnectedValue(r) || S.Zero(), s = this.center.getConnectedValue(r), a = at.Zero(); switch (this.mapping) { case GH.Spherical: { n.subtractToRef(s, e); const f = e.length(); f > 0 && (a.x = Math.acos(e.y / f) / Math.PI, (e.x !== 0 || e.z !== 0) && (a.y = Math.atan2(e.x, e.z) / (Math.PI * 2))); break; } case GH.Cylindrical: { n.subtractToRef(s, e); const f = e.length(); f > 0 && (a.x = Math.atan2(e.x / f, e.z / f) / (Math.PI * 2), a.y = (e.y + 1) / 2); break; } case GH.Cubic: { const f = Math.abs(i.x), o = Math.abs(i.y), d = Math.abs(i.z), v = Math.max(Math.abs(n.x), Math.abs(n.y), Math.abs(n.z)); let u = 0, l = 0; f >= o && f >= d ? (u = n.y / v - s.y, l = n.z / v - s.z) : o >= f && o >= d ? (u = n.x / v - s.x, l = n.z / v - s.z) : (u = n.x / v - s.x, l = n.y / v - s.y), a.x = (u + 1) / 2, a.y = (l + 1) / 2; } } return a; }; this.uv._storedFunction = (r) => t(r); } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.mapping = BABYLON.MappingTypes.${GH[this.mapping]}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.mapping = this.mapping, e; } _deserialize(e) { super._deserialize(e), this.mapping = e.mapping; } } C([ rn("Mapping", Gr.List, "ADVANCED", { notifiers: { rebuild: !0 }, options: [ { label: "Spherical", value: GH.Spherical }, { label: "Cylindrical", value: GH.Cylindrical }, { label: "Cubic", value: GH.Cubic } ] }) ], VY.prototype, "mapping", void 0); Ue("BABYLON.MappingBlock", VY); class Ene extends yi { /** * Create a new MatrixComposeBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("matrix0", Me.Matrix), this.registerInput("matrix1", Me.Matrix), this.registerOutput("output", Me.Matrix); } /** * Gets the current class name * @returns the class name */ getClassName() { return "MatrixComposeBlock"; } /** * Gets the matrix0 input component */ get matrix0() { return this._inputs[0]; } /** * Gets the matrix1 input component */ get matrix1() { return this._inputs[1]; } /** * Gets the output component */ get output() { return this._outputs[0]; } _buildBlock() { this.output._storedFunction = (e) => { if (!this.matrix0.isConnected || !this.matrix1.isConnected) return null; const t = this.matrix0.getConnectedValue(e), r = this.matrix1.getConnectedValue(e); return !t || !r ? null : t.multiply(r); }; } } Ue("BABYLON.MatrixComposeBlock", Ene); class Fne extends yi { /** Gets the list of attached endpoints */ get endpoints() { return this._endpoints; } /** * Create a new TeleportInBlock * @param name defines the block name */ constructor(e) { super(e), this._endpoints = [], this._isTeleportIn = !0, this.registerInput("input", Me.AutoDetect); } /** * Gets the current class name * @returns the class name */ getClassName() { return "TeleportInBlock"; } /** * Gets the input component */ get input() { return this._inputs[0]; } _dumpCode(e, t) { let r = super._dumpCode(e, t); for (const n of this.endpoints) t.indexOf(n) === -1 && (r += n._dumpCode(e, t)); return r; } /** * Checks if the current block is an ancestor of a given type * @param type defines the potential type to check * @returns true if block is a descendant */ isAnAncestorOfType(e) { if (this.getClassName() === e) return !0; for (const t of this.endpoints) if (t.isAnAncestorOfType(e)) return !0; return !1; } /** * Checks if the current block is an ancestor of a given block * @param block defines the potential descendant block to check * @returns true if block is a descendant */ isAnAncestorOf(e) { for (const t of this.endpoints) if (t === e || t.isAnAncestorOf(e)) return !0; return !1; } /** * Get the first descendant using a predicate * @param predicate defines the predicate to check * @returns descendant or null if none found */ getDescendantOfPredicate(e) { if (e(this)) return this; for (const t of this.endpoints) { const r = t.getDescendantOfPredicate(e); if (r) return r; } return null; } /** * Add an enpoint to this block * @param endpoint define the endpoint to attach to */ attachToEndpoint(e) { e.detach(), this._endpoints.push(e), e._entryPoint = this, e._outputs[0]._typeConnectionSource = this._inputs[0], e._tempEntryPointUniqueId = null, e.name = "> " + this.name; } /** * Remove enpoint from this block * @param endpoint define the endpoint to remove */ detachFromEndpoint(e) { const t = this._endpoints.indexOf(e); t !== -1 && (this._endpoints.splice(t, 1), e._outputs[0]._typeConnectionSource = null, e._entryPoint = null); } _buildBlock() { for (const e of this._endpoints) e.output._storedFunction = (t) => this.input.getConnectedValue(t); } } Ue("BABYLON.TeleportInBlock", Fne); class Nne extends yi { /** * Create a new TeleportOutBlock * @param name defines the block name */ constructor(e) { super(e), this._entryPoint = null, this._tempEntryPointUniqueId = null, this._isTeleportOut = !0, this.registerOutput("output", Me.BasedOnInput); } /** * Gets the entry point */ get entryPoint() { return this._entryPoint; } /** * Gets the current class name * @returns the class name */ getClassName() { return "TeleportOutBlock"; } /** * Gets the output component */ get output() { return this._outputs[0]; } /** Detach from entry point */ detach() { this._entryPoint && this._entryPoint.detachFromEndpoint(this); } _buildBlock() { } _customBuildStep(e) { this.entryPoint && this.entryPoint.build(e); } _dumpCode(e, t) { let r = ""; return this.entryPoint && t.indexOf(this.entryPoint) === -1 && (r += this.entryPoint._dumpCode(e, t)), r + super._dumpCode(e, t); } _dumpCodeForOutputConnections(e) { let t = super._dumpCodeForOutputConnections(e); return this.entryPoint && (t += this.entryPoint._dumpCodeForOutputConnections(e)), t; } /** * Clone the current block to a new identical block * @returns a copy of the current block */ clone() { const e = super.clone(); return this.entryPoint && this.entryPoint.attachToEndpoint(e), e; } _dumpPropertiesCode() { let e = super._dumpPropertiesCode(); return this.entryPoint && (e += `${this.entryPoint._codeVariableName}.attachToEndpoint(${this._codeVariableName}); `), e; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { var e, t; const r = super.serialize(); return r.entryPoint = (t = (e = this.entryPoint) === null || e === void 0 ? void 0 : e.uniqueId) !== null && t !== void 0 ? t : "", r; } _deserialize(e) { super._deserialize(e), this._tempEntryPointUniqueId = e.entryPoint; } } Ue("BABYLON.TeleportOutBlock", Nne); class CY extends yi { /** * Gets the texture data */ get textureData() { return this._data; } /** * Gets the texture width */ get textureWidth() { return this._width; } /** * Gets the texture height */ get textureHeight() { return this._height; } /** * Creates a new GeometryTextureBlock * @param name defines the block name */ constructor(e) { super(e), this._data = null, this.serializedCachedData = !1, this.registerOutput("texture", Me.Texture); } /** * Gets the current class name * @returns the class name */ getClassName() { return "GeometryTextureBlock"; } /** * Gets the texture component */ get texture() { return this._outputs[0]; } _prepareImgToLoadAsync(e) { return new Promise((t, r) => { const n = new Image(), i = document.createElement("canvas"), s = i.getContext("2d"); n.onload = () => { i.width = n.width, i.height = n.height, s.drawImage(n, 0, 0); const f = s.getImageData(0, 0, n.width, n.height).data, o = new Float32Array(f.length); for (let d = 0; d < f.length; d++) o[d] = f[d] / 255; this._data = o, this._width = n.width, this._height = n.height, t(); }, n.onerror = () => { this._data = null, r(); }, n.src = e; }); } /** * Remove stored data */ cleanData() { this._data = null; } /** * Load the texture data * @param imageFile defines the file to load data from * @returns a promise fulfilled when image data is loaded */ loadTextureFromFileAsync(e) { return this._prepareImgToLoadAsync(URL.createObjectURL(e)); } /** * Load the texture data * @param url defines the url to load data from * @returns a promise fulfilled when image data is loaded */ loadTextureFromUrlAsync(e) { return this._prepareImgToLoadAsync(e); } /** * Load the texture data * @param url defines the url to load data from * @returns a promise fulfilled when image data is loaded */ extractFromTextureAsync(e) { return new Promise((t, r) => { if (!e.isReady()) { e.onLoadObservable.addOnce(() => this.extractFromTextureAsync(e).then(t).catch(r)); return; } const n = e.getSize(); Pte.GetTextureDataAsync(e, n.width, n.height).then(async (i) => { const s = new Float32Array(i.length); for (let a = 0; a < i.length; a++) s[a] = i[a] / 255; this._data = s, this._width = n.width, this._height = n.height, t(); }).catch(r); }); } _buildBlock() { if (!this._data) { this.texture._storedValue = null; return; } const e = { data: this._data, width: this._width, height: this._height }; this.texture._storedValue = e; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.width = this._width, e.height = this._height, e.serializedCachedData = this.serializedCachedData, this._data && this.serializedCachedData && (e.data = Array.from(this._data)), e; } _deserialize(e) { super._deserialize(e), this._width = e.width, this._height = e.height, e.data ? (this._data = new Float32Array(e.data), this.serializedCachedData = !0) : this.serializedCachedData = !!e.serializedCachedData; } } C([ rn("Serialize cached data", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], CY.prototype, "serializedCachedData", void 0); Ue("BABYLON.GeometryTextureBlock", CY); class OY extends yi { /** * Creates a new GeometryTextureFetchBlock * @param name defines the block name */ constructor(e) { super(e), this.clampCoordinates = !0, this.registerInput("texture", Me.Texture), this.registerInput("coordinates", Me.Vector2), this.registerOutput("rgba", Me.Vector4), this.registerOutput("rgb", Me.Vector3), this.registerOutput("r", Me.Float), this.registerOutput("g", Me.Float), this.registerOutput("b", Me.Float), this.registerOutput("a", Me.Float); } /** * Gets the current class name * @returns the class name */ getClassName() { return "GeometryTextureFetchBlock"; } /** * Gets the texture component */ get texture() { return this.inputs[0]; } /** * Gets the coordinates component */ get coordinates() { return this.inputs[1]; } /** * Gets the rgba component */ get rgba() { return this._outputs[0]; } /** * Gets the rgb component */ get rgb() { return this._outputs[1]; } /** * Gets the r component */ get r() { return this._outputs[2]; } /** * Gets the g component */ get g() { return this._outputs[3]; } /** * Gets the b component */ get b() { return this._outputs[4]; } /** * Gets the a component */ get a() { return this._outputs[5]; } _repeatClamp(e) { return e >= 0 ? e % 1 : 1 - Math.abs(e) % 1; } _buildBlock() { const e = (t) => { const r = this.texture.getConnectedValue(t); if (!r || !r.data) return null; const n = this.coordinates.getConnectedValue(t); if (!n) return null; const i = this.clampCoordinates ? Math.max(0, Math.min(n.x, 1)) : this._repeatClamp(n.x), s = this.clampCoordinates ? Math.max(0, Math.min(n.y, 1)) : this._repeatClamp(n.y), a = Math.floor(i * (r.width - 1)), f = Math.floor(s * (r.height - 1)), o = a + r.width * f; return Ir.FromArray(r.data, o * 4); }; this.rgba._storedFunction = (t) => e(t), this.rgb._storedFunction = (t) => { const r = e(t); return r ? r.toVector3() : null; }, this.r._storedFunction = (t) => { const r = e(t); return r ? r.x : null; }, this.g._storedFunction = (t) => { const r = e(t); return r ? r.y : null; }, this.b._storedFunction = (t) => { const r = e(t); return r ? r.z : null; }, this.a._storedFunction = (t) => { const r = e(t); return r ? r.w : null; }; } _dumpPropertiesCode() { return super._dumpPropertiesCode() + `${this._codeVariableName}.clampCoordinates = ${this.clampCoordinates}; `; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.clampCoordinates = this.clampCoordinates, e; } _deserialize(e) { super._deserialize(e), this.clampCoordinates = e.clampCoordinates; } } C([ rn("Clamp Coordinates", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], OY.prototype, "clampCoordinates", void 0); Ue("BABYLON.GeometryTextureFetchBlock", OY); class Qne extends yi { /** * Create a new BoundingBlock * @param name defines the block name */ constructor(e) { super(e), this.registerInput("geometry", Me.Geometry), this.registerOutput("min", Me.Vector3), this.registerOutput("max", Me.Vector3); } /** * Gets the current class name * @returns the class name */ getClassName() { return "BoundingBlock"; } /** * Gets the geometry input component */ get geometry() { return this._inputs[0]; } /** * Gets the min output component */ get min() { return this._outputs[0]; } /** * Gets the max output component */ get max() { return this._outputs[1]; } _buildBlock() { this.min._storedFunction = (e) => { const t = this.geometry.getConnectedValue(e); return t ? WS(t.positions, 0, t.positions.length / 3).minimum : null; }, this.max._storedFunction = (e) => { const t = this.geometry.getConnectedValue(e); return t ? WS(t.positions, 0, t.positions.length / 3).maximum : null; }; } } Ue("BABYLON.BoundingBlock", Qne); var ZH; (function(A) { A[A.Intersect = 0] = "Intersect", A[A.Subtract = 1] = "Subtract", A[A.Union = 2] = "Union"; })(ZH || (ZH = {})); class Dy extends yi { /** * Create a new BooleanGeometryBlock * @param name defines the block name */ constructor(e) { super(e), this.evaluateContext = !1, this.operation = ZH.Intersect, this.registerInput("geometry0", Me.Geometry), this.registerInput("geometry1", Me.Geometry), this.registerOutput("output", Me.Geometry); } /** * Gets the current class name * @returns the class name */ getClassName() { return "BooleanGeometryBlock"; } /** * Gets the geometry0 input component */ get geometry0() { return this._inputs[0]; } /** * Gets the geometry1 input component */ get geometry1() { return this._inputs[1]; } /** * Gets the geometry output component */ get output() { return this._outputs[0]; } _buildBlock(e) { const t = (r) => { const n = this.geometry0.getConnectedValue(r), i = this.geometry1.getConnectedValue(r); if (!n || !i) return null; const s = n.positions.length / 3; !n.normals && i.normals && (n.normals = new Array(n.positions.length)), !i.normals && n.normals && (i.normals = new Array(i.positions.length)), !n.uvs && i.uvs && (n.uvs = new Array(s * 2)), !i.uvs && n.uvs && (i.uvs = new Array(s * 2)), !n.colors && i.colors && (n.colors = new Array(s * 4)), !i.colors && n.colors && (i.colors = new Array(s * 4)); const a = YH.FromVertexData(n), f = YH.FromVertexData(i); let o; switch (this.operation) { case ZH.Intersect: o = a.intersect(f); break; case ZH.Subtract: o = a.subtract(f); break; case ZH.Union: o = a.union(f); break; } return o.toVertexData(); }; this.evaluateContext ? this.output._storedFunction = t : (this.output._storedFunction = null, this.output._storedValue = t(e)); } _dumpPropertiesCode() { let e = super._dumpPropertiesCode() + `${this._codeVariableName}.evaluateContext = ${this.evaluateContext ? "true" : "false"}; `; return e += `${this._codeVariableName}.operation = BABYLON.BooleanGeometryOperations.${ZH[this.operation]}; `, e; } /** * Serializes this block in a JSON representation * @returns the serialized block object */ serialize() { const e = super.serialize(); return e.evaluateContext = this.evaluateContext, e.operation = this.operation, e; } _deserialize(e) { super._deserialize(e), this.evaluateContext = e.evaluateContext, e.operation && (this.operation = e.operation); } } C([ rn("Evaluate context", Gr.Boolean, "ADVANCED", { notifiers: { rebuild: !0 } }) ], Dy.prototype, "evaluateContext", void 0); C([ rn("Operation", Gr.List, "ADVANCED", { notifiers: { rebuild: !0 }, options: [ { label: "Intersect", value: ZH.Intersect }, { label: "Subtract", value: ZH.Subtract }, { label: "Union", value: ZH.Union } ] }) ], Dy.prototype, "operation", void 0); Ue("BABYLON.BooleanGeometryBlock", Dy); class Bhe { /** * Initializes the recastJS plugin * @param recastInjection can be used to inject your own recast reference */ constructor(e = Recast) { if (this.bjsRECAST = {}, this.name = "RecastJSPlugin", this._maximumSubStepCount = 10, this._timeStep = 1 / 60, this._timeFactor = 1, this._worker = null, typeof e == "function" ? Se.Error("RecastJS is not ready. Please make sure you await Recast() before using the plugin.") : this.bjsRECAST = e, !this.isSupported()) { Se.Error("RecastJS is not available. Please make sure you included the js file."); return; } this.setTimeStep(), this._tempVec1 = new this.bjsRECAST.Vec3(), this._tempVec2 = new this.bjsRECAST.Vec3(); } /** * Set worker URL to be used when generating a new navmesh * @param workerURL url string * @returns boolean indicating if worker is created */ setWorkerURL(e) { return window && window.Worker ? (this._worker = new Worker(e), !0) : !1; } /** * Set the time step of the navigation tick update. * Default is 1/60. * A value of 0 will disable fixed time update * @param newTimeStep the new timestep to apply to this world. */ setTimeStep(e = 1 / 60) { this._timeStep = e; } /** * Get the time step of the navigation tick update. * @returns the current time step */ getTimeStep() { return this._timeStep; } /** * If delta time in navigation tick update is greater than the time step * a number of sub iterations are done. If more iterations are need to reach deltatime * they will be discarded. * A value of 0 will set to no maximum and update will use as many substeps as needed * @param newStepCount the maximum number of iterations */ setMaximumSubStepCount(e = 10) { this._maximumSubStepCount = e; } /** * Get the maximum number of iterations per navigation tick update * @returns the maximum number of iterations */ getMaximumSubStepCount() { return this._maximumSubStepCount; } /** * Time factor applied when updating crowd agents (default 1). A value of 0 will pause crowd updates. * @param value the time factor applied at update */ set timeFactor(e) { this._timeFactor = Math.max(e, 0); } /** * Get the time factor used for crowd agent update * @returns the time factor */ get timeFactor() { return this._timeFactor; } /** * Creates a navigation mesh * @param meshes array of all the geometry used to compute the navigation mesh * @param parameters bunch of parameters used to filter geometry * @param completion callback when data is available from the worker. Not used without a worker */ createNavMesh(e, t, r) { this._worker && !r ? console.warn("A worker is avaible but no completion callback. Defaulting to blocking navmesh creation") : !this._worker && r && console.warn("A completion callback is avaible but no worker. Defaulting to blocking navmesh creation"), this.navMesh = new this.bjsRECAST.NavMesh(); let n, i, s; const a = [], f = []; let o = 0; for (n = 0; n < e.length; n++) if (e[n]) { const d = e[n], v = d.getIndices(); if (!v) continue; const u = d.getVerticesData(J.PositionKind, !1, !1); if (!u) continue; const l = [], P = d.computeWorldMatrix(!0); if (d.hasThinInstances) { const p = d.thinInstanceGetWorldMatrices(); for (let c = 0; c < p.length; c++) { const H = new he(); p[c].multiplyToRef(P, H), l.push(H); } } else l.push(P); for (let p = 0; p < l.length; p++) { const c = l[p]; for (i = 0; i < v.length; i++) a.push(v[i] + o); const H = S.Zero(), T = S.Zero(); for (s = 0; s < u.length; s += 3) S.FromArrayToRef(u, s, T), S.TransformCoordinatesToRef(T, c, H), f.push(H.x, H.y, H.z); o += u.length / 3; } } if (this._worker && r) this._worker.postMessage([f, o, a, a.length, t]), this._worker.onmessage = function(d) { r(d.data); }; else { const d = new this.bjsRECAST.rcConfig(); d.cs = t.cs, d.ch = t.ch, d.borderSize = t.borderSize ? t.borderSize : 0, d.tileSize = t.tileSize ? t.tileSize : 0, d.walkableSlopeAngle = t.walkableSlopeAngle, d.walkableHeight = t.walkableHeight, d.walkableClimb = t.walkableClimb, d.walkableRadius = t.walkableRadius, d.maxEdgeLen = t.maxEdgeLen, d.maxSimplificationError = t.maxSimplificationError, d.minRegionArea = t.minRegionArea, d.mergeRegionArea = t.mergeRegionArea, d.maxVertsPerPoly = t.maxVertsPerPoly, d.detailSampleDist = t.detailSampleDist, d.detailSampleMaxError = t.detailSampleMaxError, this.navMesh.build(f, o, a, a.length, d); } } /** * Create a navigation mesh debug mesh * @param scene is where the mesh will be added * @returns debug display mesh */ createDebugNavMesh(e) { let t, r; const n = this.navMesh.getDebugNavMesh(), i = n.getTriangleCount(), s = [], a = []; for (t = 0; t < i * 3; t++) s.push(t); for (t = 0; t < i; t++) for (r = 0; r < 3; r++) { const d = n.getTriangle(t).getPoint(r); a.push(d.x, d.y, d.z); } const f = new Ee("NavMeshDebug", e), o = new Ut(); return o.indices = s, o.positions = a, o.applyToMesh(f, !1), f; } /** * Get a navigation mesh constrained position, closest to the parameter position * @param position world position * @returns the closest point to position constrained by the navigation mesh */ getClosestPoint(e) { this._tempVec1.x = e.x, this._tempVec1.y = e.y, this._tempVec1.z = e.z; const t = this.navMesh.getClosestPoint(this._tempVec1); return new S(t.x, t.y, t.z); } /** * Get a navigation mesh constrained position, closest to the parameter position * @param position world position * @param result output the closest point to position constrained by the navigation mesh */ getClosestPointToRef(e, t) { this._tempVec1.x = e.x, this._tempVec1.y = e.y, this._tempVec1.z = e.z; const r = this.navMesh.getClosestPoint(this._tempVec1); t.set(r.x, r.y, r.z); } /** * Get a navigation mesh constrained position, within a particular radius * @param position world position * @param maxRadius the maximum distance to the constrained world position * @returns the closest point to position constrained by the navigation mesh */ getRandomPointAround(e, t) { this._tempVec1.x = e.x, this._tempVec1.y = e.y, this._tempVec1.z = e.z; const r = this.navMesh.getRandomPointAround(this._tempVec1, t); return new S(r.x, r.y, r.z); } /** * Get a navigation mesh constrained position, within a particular radius * @param position world position * @param maxRadius the maximum distance to the constrained world position * @param result output the closest point to position constrained by the navigation mesh */ getRandomPointAroundToRef(e, t, r) { this._tempVec1.x = e.x, this._tempVec1.y = e.y, this._tempVec1.z = e.z; const n = this.navMesh.getRandomPointAround(this._tempVec1, t); r.set(n.x, n.y, n.z); } /** * Compute the final position from a segment made of destination-position * @param position world position * @param destination world position * @returns the resulting point along the navmesh */ moveAlong(e, t) { this._tempVec1.x = e.x, this._tempVec1.y = e.y, this._tempVec1.z = e.z, this._tempVec2.x = t.x, this._tempVec2.y = t.y, this._tempVec2.z = t.z; const r = this.navMesh.moveAlong(this._tempVec1, this._tempVec2); return new S(r.x, r.y, r.z); } /** * Compute the final position from a segment made of destination-position * @param position world position * @param destination world position * @param result output the resulting point along the navmesh */ moveAlongToRef(e, t, r) { this._tempVec1.x = e.x, this._tempVec1.y = e.y, this._tempVec1.z = e.z, this._tempVec2.x = t.x, this._tempVec2.y = t.y, this._tempVec2.z = t.z; const n = this.navMesh.moveAlong(this._tempVec1, this._tempVec2); r.set(n.x, n.y, n.z); } /** * Compute a navigation path from start to end. Returns an empty array if no path can be computed * @param start world position * @param end world position * @returns array containing world position composing the path */ computePath(e, t) { let r; this._tempVec1.x = e.x, this._tempVec1.y = e.y, this._tempVec1.z = e.z, this._tempVec2.x = t.x, this._tempVec2.y = t.y, this._tempVec2.z = t.z; const n = this.navMesh.computePath(this._tempVec1, this._tempVec2), i = n.getPointCount(), s = []; for (r = 0; r < i; r++) { const a = n.getPoint(r); s.push(new S(a.x, a.y, a.z)); } return s; } /** * Create a new Crowd so you can add agents * @param maxAgents the maximum agent count in the crowd * @param maxAgentRadius the maximum radius an agent can have * @param scene to attach the crowd to * @returns the crowd you can add agents to */ createCrowd(e, t, r) { return new Yne(this, e, t, r); } /** * Set the Bounding box extent for doing spatial queries (getClosestPoint, getRandomPointAround, ...) * The queries will try to find a solution within those bounds * default is (1,1,1) * @param extent x,y,z value that define the extent around the queries point of reference */ setDefaultQueryExtent(e) { this._tempVec1.x = e.x, this._tempVec1.y = e.y, this._tempVec1.z = e.z, this.navMesh.setDefaultQueryExtent(this._tempVec1); } /** * Get the Bounding box extent specified by setDefaultQueryExtent * @returns the box extent values */ getDefaultQueryExtent() { const e = this.navMesh.getDefaultQueryExtent(); return new S(e.x, e.y, e.z); } /** * build the navmesh from a previously saved state using getNavmeshData * @param data the Uint8Array returned by getNavmeshData */ buildFromNavmeshData(e) { const t = e.length * e.BYTES_PER_ELEMENT, r = this.bjsRECAST._malloc(t), n = new Uint8Array(this.bjsRECAST.HEAPU8.buffer, r, t); n.set(e); const i = new this.bjsRECAST.NavmeshData(); i.dataPointer = n.byteOffset, i.size = e.length, this.navMesh = new this.bjsRECAST.NavMesh(), this.navMesh.buildFromNavmeshData(i), this.bjsRECAST._free(n.byteOffset); } /** * returns the navmesh data that can be used later. The navmesh must be built before retrieving the data * @returns data the Uint8Array that can be saved and reused */ getNavmeshData() { const e = this.navMesh.getNavmeshData(), t = new Uint8Array(this.bjsRECAST.HEAPU8.buffer, e.dataPointer, e.size), r = new Uint8Array(e.size); return r.set(t), this.navMesh.freeNavmeshData(e), r; } /** * Get the Bounding box extent result specified by setDefaultQueryExtent * @param result output the box extent values */ getDefaultQueryExtentToRef(e) { const t = this.navMesh.getDefaultQueryExtent(); e.set(t.x, t.y, t.z); } /** * Disposes */ dispose() { } /** * Creates a cylinder obstacle and add it to the navigation * @param position world position * @param radius cylinder radius * @param height cylinder height * @returns the obstacle freshly created */ addCylinderObstacle(e, t, r) { return this._tempVec1.x = e.x, this._tempVec1.y = e.y, this._tempVec1.z = e.z, this.navMesh.addCylinderObstacle(this._tempVec1, t, r); } /** * Creates an oriented box obstacle and add it to the navigation * @param position world position * @param extent box size * @param angle angle in radians of the box orientation on Y axis * @returns the obstacle freshly created */ addBoxObstacle(e, t, r) { return this._tempVec1.x = e.x, this._tempVec1.y = e.y, this._tempVec1.z = e.z, this._tempVec2.x = t.x, this._tempVec2.y = t.y, this._tempVec2.z = t.z, this.navMesh.addBoxObstacle(this._tempVec1, this._tempVec2, r); } /** * Removes an obstacle created by addCylinderObstacle or addBoxObstacle * @param obstacle obstacle to remove from the navigation */ removeObstacle(e) { this.navMesh.removeObstacle(e); } /** * If this plugin is supported * @returns true if plugin is supported */ isSupported() { return this.bjsRECAST !== void 0; } } class Yne { /** * Constructor * @param plugin recastJS plugin * @param maxAgents the maximum agent count in the crowd * @param maxAgentRadius the maximum radius an agent can have * @param scene to attach the crowd to * @returns the crowd you can add agents to */ constructor(e, t, r, n) { this.recastCrowd = {}, this.transforms = new Array(), this.agents = new Array(), this.reachRadii = new Array(), this._agentDestinationArmed = new Array(), this._agentDestination = new Array(), this._onBeforeAnimationsObserver = null, this.onReachTargetObservable = new Oe(), this.bjsRECASTPlugin = e, this.recastCrowd = new this.bjsRECASTPlugin.bjsRECAST.Crowd(t, r, this.bjsRECASTPlugin.navMesh.getNavMesh()), this._scene = n, this._onBeforeAnimationsObserver = n.onBeforeAnimationsObservable.add(() => { this.update(n.getEngine().getDeltaTime() * 1e-3 * e.timeFactor); }); } /** * Add a new agent to the crowd with the specified parameter a corresponding transformNode. * You can attach anything to that node. The node position is updated in the scene update tick. * @param pos world position that will be constrained by the navigation mesh * @param parameters agent parameters * @param transform hooked to the agent that will be update by the scene * @returns agent index */ addAgent(e, t, r) { const n = new this.bjsRECASTPlugin.bjsRECAST.dtCrowdAgentParams(); n.radius = t.radius, n.height = t.height, n.maxAcceleration = t.maxAcceleration, n.maxSpeed = t.maxSpeed, n.collisionQueryRange = t.collisionQueryRange, n.pathOptimizationRange = t.pathOptimizationRange, n.separationWeight = t.separationWeight, n.updateFlags = 7, n.obstacleAvoidanceType = 0, n.queryFilterType = 0, n.userData = 0; const i = this.recastCrowd.addAgent(new this.bjsRECASTPlugin.bjsRECAST.Vec3(e.x, e.y, e.z), n); return this.transforms.push(r), this.agents.push(i), this.reachRadii.push(t.reachRadius ? t.reachRadius : t.radius), this._agentDestinationArmed.push(!1), this._agentDestination.push(new S(0, 0, 0)), i; } /** * Returns the agent position in world space * @param index agent index returned by addAgent * @returns world space position */ getAgentPosition(e) { const t = this.recastCrowd.getAgentPosition(e); return new S(t.x, t.y, t.z); } /** * Returns the agent position result in world space * @param index agent index returned by addAgent * @param result output world space position */ getAgentPositionToRef(e, t) { const r = this.recastCrowd.getAgentPosition(e); t.set(r.x, r.y, r.z); } /** * Returns the agent velocity in world space * @param index agent index returned by addAgent * @returns world space velocity */ getAgentVelocity(e) { const t = this.recastCrowd.getAgentVelocity(e); return new S(t.x, t.y, t.z); } /** * Returns the agent velocity result in world space * @param index agent index returned by addAgent * @param result output world space velocity */ getAgentVelocityToRef(e, t) { const r = this.recastCrowd.getAgentVelocity(e); t.set(r.x, r.y, r.z); } /** * Returns the agent next target point on the path * @param index agent index returned by addAgent * @returns world space position */ getAgentNextTargetPath(e) { const t = this.recastCrowd.getAgentNextTargetPath(e); return new S(t.x, t.y, t.z); } /** * Returns the agent next target point on the path * @param index agent index returned by addAgent * @param result output world space position */ getAgentNextTargetPathToRef(e, t) { const r = this.recastCrowd.getAgentNextTargetPath(e); t.set(r.x, r.y, r.z); } /** * Gets the agent state * @param index agent index returned by addAgent * @returns agent state */ getAgentState(e) { return this.recastCrowd.getAgentState(e); } /** * returns true if the agent in over an off mesh link connection * @param index agent index returned by addAgent * @returns true if over an off mesh link connection */ overOffmeshConnection(e) { return this.recastCrowd.overOffmeshConnection(e); } /** * Asks a particular agent to go to a destination. That destination is constrained by the navigation mesh * @param index agent index returned by addAgent * @param destination targeted world position */ agentGoto(e, t) { this.recastCrowd.agentGoto(e, new this.bjsRECASTPlugin.bjsRECAST.Vec3(t.x, t.y, t.z)); const r = this.agents.indexOf(e); r > -1 && (this._agentDestinationArmed[r] = !0, this._agentDestination[r].set(t.x, t.y, t.z)); } /** * Teleport the agent to a new position * @param index agent index returned by addAgent * @param destination targeted world position */ agentTeleport(e, t) { this.recastCrowd.agentTeleport(e, new this.bjsRECASTPlugin.bjsRECAST.Vec3(t.x, t.y, t.z)); } /** * Update agent parameters * @param index agent index returned by addAgent * @param parameters agent parameters */ updateAgentParameters(e, t) { const r = this.recastCrowd.getAgentParameters(e); t.radius !== void 0 && (r.radius = t.radius), t.height !== void 0 && (r.height = t.height), t.maxAcceleration !== void 0 && (r.maxAcceleration = t.maxAcceleration), t.maxSpeed !== void 0 && (r.maxSpeed = t.maxSpeed), t.collisionQueryRange !== void 0 && (r.collisionQueryRange = t.collisionQueryRange), t.pathOptimizationRange !== void 0 && (r.pathOptimizationRange = t.pathOptimizationRange), t.separationWeight !== void 0 && (r.separationWeight = t.separationWeight), this.recastCrowd.setAgentParameters(e, r); } /** * remove a particular agent previously created * @param index agent index returned by addAgent */ removeAgent(e) { this.recastCrowd.removeAgent(e); const t = this.agents.indexOf(e); t > -1 && (this.agents.splice(t, 1), this.transforms.splice(t, 1), this.reachRadii.splice(t, 1), this._agentDestinationArmed.splice(t, 1), this._agentDestination.splice(t, 1)); } /** * get the list of all agents attached to this crowd * @returns list of agent indices */ getAgents() { return this.agents; } /** * Tick update done by the Scene. Agent position/velocity/acceleration is updated by this function * @param deltaTime in seconds */ update(e) { if (this.bjsRECASTPlugin.navMesh.update(), e <= Dn) return; const t = this.bjsRECASTPlugin.getTimeStep(), r = this.bjsRECASTPlugin.getMaximumSubStepCount(); if (t <= Dn) this.recastCrowd.update(e); else { let n = Math.floor(e / t); r && n > r && (n = r), n < 1 && (n = 1); const i = e / n; for (let s = 0; s < n; s++) this.recastCrowd.update(i); } for (let n = 0; n < this.agents.length; n++) { const i = this.agents[n], s = this.getAgentPosition(i); if (this.transforms[n].position = s, this._agentDestinationArmed[n]) { const a = s.x - this._agentDestination[n].x, f = s.z - this._agentDestination[n].z, o = this.reachRadii[n], d = this._agentDestination[n].y - this.reachRadii[n], v = this._agentDestination[n].y + this.reachRadii[n], u = a * a + f * f; s.y > d && s.y < v && u < o * o && (this.onReachTargetObservable.notifyObservers({ agentIndex: i, destination: this._agentDestination[n] }), this._agentDestinationArmed[n] = !1); } } } /** * Set the Bounding box extent for doing spatial queries (getClosestPoint, getRandomPointAround, ...) * The queries will try to find a solution within those bounds * default is (1,1,1) * @param extent x,y,z value that define the extent around the queries point of reference */ setDefaultQueryExtent(e) { const t = new this.bjsRECASTPlugin.bjsRECAST.Vec3(e.x, e.y, e.z); this.recastCrowd.setDefaultQueryExtent(t); } /** * Get the Bounding box extent specified by setDefaultQueryExtent * @returns the box extent values */ getDefaultQueryExtent() { const e = this.recastCrowd.getDefaultQueryExtent(); return new S(e.x, e.y, e.z); } /** * Get the Bounding box extent result specified by setDefaultQueryExtent * @param result output the box extent values */ getDefaultQueryExtentToRef(e) { const t = this.recastCrowd.getDefaultQueryExtent(); e.set(t.x, t.y, t.z); } /** * Get the next corner points composing the path (max 4 points) * @param index agent index returned by addAgent * @returns array containing world position composing the path */ getCorners(e) { let t; const r = this.recastCrowd.getCorners(e), n = r.getPointCount(), i = []; for (t = 0; t < n; t++) { const s = r.getPoint(t); i.push(new S(s.x, s.y, s.z)); } return i; } /** * Release all resources */ dispose() { this.recastCrowd.destroy(), this._scene.onBeforeAnimationsObservable.remove(this._onBeforeAnimationsObserver), this._onBeforeAnimationsObserver = null, this.onReachTargetObservable.clear(); } } Ge.OfflineProviderFactory = (A, e, t = !1) => new Cv(A, e, t); class Cv { /** * Gets a boolean indicating if scene must be saved in the database */ get enableSceneOffline() { return this._enableSceneOffline; } /** * Gets a boolean indicating if textures must be saved in the database */ get enableTexturesOffline() { return this._enableTexturesOffline; } /** * Creates a new Database * @param urlToScene defines the url to load the scene * @param callbackManifestChecked defines the callback to use when manifest is checked * @param disableManifestCheck defines a boolean indicating that we want to skip the manifest validation (it will be considered validated and up to date) */ constructor(e, t, r = !1) { this._idbFactory = typeof indexedDB < "u" ? indexedDB : void 0, this._currentSceneUrl = Cv._ReturnFullUrlLocation(e), this._db = null, this._enableSceneOffline = !1, this._enableTexturesOffline = !1, this._manifestVersionFound = 0, this._mustUpdateRessources = !1, this._hasReachedQuota = !1, Cv.IDBStorageEnabled ? r ? (this._enableSceneOffline = !0, this._enableTexturesOffline = !0, this._manifestVersionFound = 1, ye.SetImmediate(() => { t(!0); })) : this._checkManifestFile(t) : t(!0); } _checkManifestFile(e) { const t = () => { this._enableSceneOffline = !1, this._enableTexturesOffline = !1, e(!1); }, r = () => { try { if (typeof URL == "function" && this._currentSceneUrl.indexOf("http") === 0) { const a = new URL(this._currentSceneUrl); return a.pathname += ".manifest", a.toString(); } } catch { } return `${this._currentSceneUrl}.manifest`; }; let n = !1, i = r(); const s = new ho(); navigator.onLine && (n = !0, i = i + (i.match(/\?/) == null ? "?" : "&") + Date.now()), s.open("GET", i), s.addEventListener("load", () => { if (s.status === 200 || Cv._ValidateXHRData(s, 1)) try { const a = JSON.parse(s.response); this._enableSceneOffline = a.enableSceneOffline, this._enableTexturesOffline = a.enableTexturesOffline && Cv._IsUASupportingBlobStorage, a.version && !isNaN(parseInt(a.version)) && (this._manifestVersionFound = a.version), e(!0); } catch { t(); } else t(); }, !1), s.addEventListener("error", () => { if (n) { n = !1; const a = r(); s.open("GET", a), s.send(); } else t(); }, !1); try { s.send(); } catch { Se.Error("Error on XHR send request."), e(!1); } } /** * Open the database and make it available * @param successCallback defines the callback to call on success * @param errorCallback defines the callback to call on error */ open(e, t) { const r = () => { this._isSupported = !1, t && t(); }; if (!this._idbFactory || !(this._enableSceneOffline || this._enableTexturesOffline)) this._isSupported = !1, t && t(); else if (this._db) e && e(); else { this._hasReachedQuota = !1, this._isSupported = !0; const n = this._idbFactory.open("babylonjs", 1); n.onerror = () => { r(); }, n.onblocked = () => { Se.Error("IDB request blocked. Please reload the page."), r(); }, n.onsuccess = () => { this._db = n.result, e(); }, n.onupgradeneeded = (i) => { if (this._db = i.target.result, this._db) try { this._db.createObjectStore("scenes", { keyPath: "sceneUrl" }), this._db.createObjectStore("versions", { keyPath: "sceneUrl" }), this._db.createObjectStore("textures", { keyPath: "textureUrl" }); } catch (s) { Se.Error("Error while creating object stores. Exception: " + s.message), r(); } }; } } /** * Loads an image from the database * @param url defines the url to load from * @param image defines the target DOM image */ loadImage(e, t) { const r = Cv._ReturnFullUrlLocation(e), n = () => { !this._hasReachedQuota && this._db !== null ? this._saveImageIntoDBAsync(r, t) : t.src = e; }; this._mustUpdateRessources ? n() : this._loadImageFromDBAsync(r, t, n); } _loadImageFromDBAsync(e, t, r) { if (this._isSupported && this._db !== null) { let n; const i = this._db.transaction(["textures"]); i.onabort = () => { t.src = e; }, i.oncomplete = () => { let a; n && typeof URL == "function" ? (a = URL.createObjectURL(n.data), t.onerror = () => { Se.Error("Error loading image from blob URL: " + a + " switching back to web url: " + e), t.src = e; }, t.src = a) : r(); }; const s = i.objectStore("textures").get(e); s.onsuccess = (a) => { n = a.target.result; }, s.onerror = () => { Se.Error("Error loading texture " + e + " from DB."), t.src = e; }; } else Se.Error("Error: IndexedDB not supported by your browser or BabylonJS Database is not open."), t.src = e; } _saveImageIntoDBAsync(e, t) { let r; if (this._isSupported) { const n = () => { let i; if (r && typeof URL == "function") try { i = URL.createObjectURL(r); } catch { i = URL.createObjectURL(r); } i && (t.src = i); }; if (Cv._IsUASupportingBlobStorage) { const i = new ho(); i.open("GET", e), i.responseType = "blob", i.addEventListener("load", () => { if (i.status === 200 && this._db) { r = i.response; const s = this._db.transaction(["textures"], "readwrite"); s.onabort = (f) => { try { const d = f.target.error; d && d.name === "QuotaExceededError" && (this._hasReachedQuota = !0); } catch { } n(); }, s.oncomplete = () => { n(); }; const a = { textureUrl: e, data: r }; try { const f = s.objectStore("textures").put(a); f.onsuccess = () => { }, f.onerror = () => { n(); }; } catch (f) { f.code === 25 && (Cv._IsUASupportingBlobStorage = !1, this._enableTexturesOffline = !1), t.src = e; } } else t.src = e; }, !1), i.addEventListener("error", () => { Se.Error("Error in XHR request in BABYLON.Database."), t.src = e; }, !1), i.send(); } else t.src = e; } else Se.Error("Error: IndexedDB not supported by your browser or Babylon.js database is not open."), t.src = e; } _checkVersionFromDB(e, t) { const r = () => { this._saveVersionIntoDBAsync(e, t); }; this._loadVersionFromDBAsync(e, t, r); } _loadVersionFromDBAsync(e, t, r) { if (this._isSupported && this._db) { let n; try { const i = this._db.transaction(["versions"]); i.oncomplete = () => { n ? this._manifestVersionFound !== n.data ? (this._mustUpdateRessources = !0, r()) : t(n.data) : (this._mustUpdateRessources = !0, r()); }, i.onabort = () => { t(-1); }; const s = i.objectStore("versions").get(e); s.onsuccess = (a) => { n = a.target.result; }, s.onerror = () => { Se.Error("Error loading version for scene " + e + " from DB."), t(-1); }; } catch (i) { Se.Error("Error while accessing 'versions' object store (READ OP). Exception: " + i.message), t(-1); } } else Se.Error("Error: IndexedDB not supported by your browser or Babylon.js database is not open."), t(-1); } _saveVersionIntoDBAsync(e, t) { if (this._isSupported && !this._hasReachedQuota && this._db) try { const r = this._db.transaction(["versions"], "readwrite"); r.onabort = (s) => { try { const a = s.target.error; a && a.name === "QuotaExceededError" && (this._hasReachedQuota = !0); } catch { } t(-1); }, r.oncomplete = () => { t(this._manifestVersionFound); }; const n = { sceneUrl: e, data: this._manifestVersionFound }, i = r.objectStore("versions").put(n); i.onsuccess = () => { }, i.onerror = () => { Se.Error("Error in DB add version request in BABYLON.Database."); }; } catch (r) { Se.Error("Error while accessing 'versions' object store (WRITE OP). Exception: " + r.message), t(-1); } else t(-1); } /** * Loads a file from database * @param url defines the URL to load from * @param sceneLoaded defines a callback to call on success * @param progressCallBack defines a callback to call when progress changed * @param errorCallback defines a callback to call on error * @param useArrayBuffer defines a boolean to use array buffer instead of text string */ loadFile(e, t, r, n, i) { const s = Cv._ReturnFullUrlLocation(e), a = () => { this._saveFileAsync(s, t, r, i, n); }; this._checkVersionFromDB(s, (f) => { f !== -1 ? this._mustUpdateRessources ? this._saveFileAsync(s, t, r, i, n) : this._loadFileAsync(s, t, a) : n && n(); }); } _loadFileAsync(e, t, r) { if (this._isSupported && this._db) { let n; e.indexOf(".babylon") !== -1 ? n = "scenes" : n = "textures"; let i; const s = this._db.transaction([n]); s.oncomplete = () => { i ? t(i.data) : r(); }, s.onabort = () => { r(); }; const a = s.objectStore(n).get(e); a.onsuccess = (f) => { i = f.target.result; }, a.onerror = () => { Se.Error("Error loading file " + e + " from DB."), r(); }; } else Se.Error("Error: IndexedDB not supported by your browser or BabylonJS Database is not open."), t(); } _saveFileAsync(e, t, r, n, i) { if (this._isSupported) { let s; e.indexOf(".babylon") !== -1 ? s = "scenes" : s = "textures"; const a = new ho(); let f; a.open("GET", e + (e.match(/\?/) == null ? "?" : "&") + Date.now()), n && (a.responseType = "arraybuffer"), r && (a.onprogress = r), a.addEventListener("load", () => { if (a.status === 200 || a.status < 400 && Cv._ValidateXHRData(a, n ? 6 : 1)) if (f = n ? a.response : a.responseText, !this._hasReachedQuota && this._db) { const o = this._db.transaction([s], "readwrite"); o.onabort = (v) => { try { const u = v.target.error; u && u.name === "QuotaExceededError" && (this._hasReachedQuota = !0); } catch { } t(f); }, o.oncomplete = () => { t(f); }; let d; s === "scenes" ? d = { sceneUrl: e, data: f, version: this._manifestVersionFound } : d = { textureUrl: e, data: f }; try { const v = o.objectStore(s).put(d); v.onsuccess = () => { }, v.onerror = () => { Se.Error("Error in DB add file request in BABYLON.Database."); }; } catch { t(f); } } else t(f); else a.status >= 400 && i ? i(a) : t(); }, !1), a.addEventListener("error", () => { Se.Error("error on XHR request."), i && i(); }, !1), a.send(); } else Se.Error("Error: IndexedDB not supported by your browser or Babylon.js database is not open."), i && i(); } /** * Validates if xhr data is correct * @param xhr defines the request to validate * @param dataType defines the expected data type * @returns true if data is correct */ static _ValidateXHRData(e, t = 7) { try { if (t & 1) { if (e.responseText && e.responseText.length > 0) return !0; if (t === 1) return !1; } if (t & 2) { const r = sV(e.response); if (r.width && r.height && r.width > 0 && r.height > 0) return !0; if (t === 2) return !1; } if (t & 4) { const r = new Uint8Array(e.response, 0, 3); return r[0] === 68 && r[1] === 68 && r[2] === 83; } } catch { } return !1; } } Cv._IsUASupportingBlobStorage = !0; Cv.IDBStorageEnabled = !1; Cv._ParseURL = (A) => { const e = document.createElement("a"); e.href = A; const t = A.substring(0, A.lastIndexOf("#")), r = A.substring(t.lastIndexOf("/") + 1, A.length); return A.substring(0, A.indexOf(r, 0)); }; Cv._ReturnFullUrlLocation = (A) => A.indexOf("http:/") === -1 && A.indexOf("https:/") === -1 && typeof window < "u" ? Cv._ParseURL(window.location.href) + A : A; class Mne { _isUbo(e) { return e.addUniform !== void 0; } constructor(e) { this._isUbo(e) ? (this.setMatrix3x3 = e.updateMatrix3x3.bind(e), this.setMatrix2x2 = e.updateMatrix2x2.bind(e), this.setFloat = e.updateFloat.bind(e), this.setFloat2 = e.updateFloat2.bind(e), this.setFloat3 = e.updateFloat3.bind(e), this.setFloat4 = e.updateFloat4.bind(e), this.setFloatArray = e.updateFloatArray.bind(e), this.setArray = e.updateArray.bind(e), this.setIntArray = e.updateIntArray.bind(e), this.setMatrix = e.updateMatrix.bind(e), this.setMatrices = e.updateMatrices.bind(e), this.setVector3 = e.updateVector3.bind(e), this.setVector4 = e.updateVector4.bind(e), this.setColor3 = e.updateColor3.bind(e), this.setColor4 = e.updateColor4.bind(e), this.setDirectColor4 = e.updateDirectColor4.bind(e), this.setInt = e.updateInt.bind(e), this.setInt2 = e.updateInt2.bind(e), this.setInt3 = e.updateInt3.bind(e), this.setInt4 = e.updateInt4.bind(e)) : (this.setMatrix3x3 = e.setMatrix3x3.bind(e), this.setMatrix2x2 = e.setMatrix2x2.bind(e), this.setFloat = e.setFloat.bind(e), this.setFloat2 = e.setFloat2.bind(e), this.setFloat3 = e.setFloat3.bind(e), this.setFloat4 = e.setFloat4.bind(e), this.setFloatArray = e.setFloatArray.bind(e), this.setArray = e.setArray.bind(e), this.setIntArray = e.setIntArray.bind(e), this.setMatrix = e.setMatrix.bind(e), this.setMatrices = e.setMatrices.bind(e), this.setVector3 = e.setVector3.bind(e), this.setVector4 = e.setVector4.bind(e), this.setColor3 = e.setColor3.bind(e), this.setColor4 = e.setColor4.bind(e), this.setDirectColor4 = e.setDirectColor4.bind(e), this.setInt = e.setInt.bind(e), this.setInt2 = e.setInt2.bind(e), this.setInt3 = e.setInt3.bind(e), this.setInt4 = e.setInt4.bind(e)); } } const Whe = "gpuUpdateParticlesPixelShader", She = `#version 300 es void main() {discard;} `; Le.ShadersStore[Whe] = She; const Uhe = "gpuUpdateParticlesVertexShader", Ihe = `#version 300 es #define PI 3.14159 uniform float currentCount;uniform float timeDelta;uniform float stopFactor; #ifndef LOCAL uniform mat4 emitterWM; #endif uniform vec2 lifeTime;uniform vec2 emitPower;uniform vec2 sizeRange;uniform vec4 scaleRange; #ifndef COLORGRADIENTS uniform vec4 color1;uniform vec4 color2; #endif uniform vec3 gravity;uniform sampler2D randomSampler;uniform sampler2D randomSampler2;uniform vec4 angleRange; #ifdef BOXEMITTER uniform vec3 direction1;uniform vec3 direction2;uniform vec3 minEmitBox;uniform vec3 maxEmitBox; #endif #ifdef POINTEMITTER uniform vec3 direction1;uniform vec3 direction2; #endif #ifdef HEMISPHERICEMITTER uniform float radius;uniform float radiusRange;uniform float directionRandomizer; #endif #ifdef SPHEREEMITTER uniform float radius;uniform float radiusRange; #ifdef DIRECTEDSPHEREEMITTER uniform vec3 direction1;uniform vec3 direction2; #else uniform float directionRandomizer; #endif #endif #ifdef CYLINDEREMITTER uniform float radius;uniform float height;uniform float radiusRange; #ifdef DIRECTEDCYLINDEREMITTER uniform vec3 direction1;uniform vec3 direction2; #else uniform float directionRandomizer; #endif #endif #ifdef CONEEMITTER uniform vec2 radius;uniform float coneAngle;uniform vec2 height;uniform float directionRandomizer; #endif in vec3 position; #ifdef CUSTOMEMITTER in vec3 initialPosition; #endif in float age;in float life;in vec4 seed;in vec3 size; #ifndef COLORGRADIENTS in vec4 color; #endif in vec3 direction; #ifndef BILLBOARD in vec3 initialDirection; #endif #ifdef ANGULARSPEEDGRADIENTS in float angle; #else in vec2 angle; #endif #ifdef ANIMATESHEET in float cellIndex; #ifdef ANIMATESHEETRANDOMSTART in float cellStartOffset; #endif #endif #ifdef NOISE in vec3 noiseCoordinates1;in vec3 noiseCoordinates2; #endif out vec3 outPosition; #ifdef CUSTOMEMITTER out vec3 outInitialPosition; #endif out float outAge;out float outLife;out vec4 outSeed;out vec3 outSize; #ifndef COLORGRADIENTS out vec4 outColor; #endif out vec3 outDirection; #ifndef BILLBOARD out vec3 outInitialDirection; #endif #ifdef ANGULARSPEEDGRADIENTS out float outAngle; #else out vec2 outAngle; #endif #ifdef ANIMATESHEET out float outCellIndex; #ifdef ANIMATESHEETRANDOMSTART out float outCellStartOffset; #endif #endif #ifdef NOISE out vec3 outNoiseCoordinates1;out vec3 outNoiseCoordinates2; #endif #ifdef SIZEGRADIENTS uniform sampler2D sizeGradientSampler; #endif #ifdef ANGULARSPEEDGRADIENTS uniform sampler2D angularSpeedGradientSampler; #endif #ifdef VELOCITYGRADIENTS uniform sampler2D velocityGradientSampler; #endif #ifdef LIMITVELOCITYGRADIENTS uniform sampler2D limitVelocityGradientSampler;uniform float limitVelocityDamping; #endif #ifdef DRAGGRADIENTS uniform sampler2D dragGradientSampler; #endif #ifdef NOISE uniform vec3 noiseStrength;uniform sampler2D noiseSampler; #endif #ifdef ANIMATESHEET uniform vec4 cellInfos; #endif vec3 getRandomVec3(float offset) {return texture(randomSampler2,vec2(float(gl_VertexID)*offset/currentCount,0)).rgb;} vec4 getRandomVec4(float offset) {return texture(randomSampler,vec2(float(gl_VertexID)*offset/currentCount,0));} void main() {float newAge=age+timeDelta; if (newAge>=life && stopFactor != 0.) {vec3 newPosition;vec3 newDirection;vec4 randoms=getRandomVec4(seed.x);outLife=lifeTime.x+(lifeTime.y-lifeTime.x)*randoms.r;outAge=newAge-life;outSeed=seed; #ifdef SIZEGRADIENTS outSize.x=texture(sizeGradientSampler,vec2(0,0)).r; #else outSize.x=sizeRange.x+(sizeRange.y-sizeRange.x)*randoms.g; #endif outSize.y=scaleRange.x+(scaleRange.y-scaleRange.x)*randoms.b;outSize.z=scaleRange.z+(scaleRange.w-scaleRange.z)*randoms.a; #ifndef COLORGRADIENTS outColor=color1+(color2-color1)*randoms.b; #endif #ifndef ANGULARSPEEDGRADIENTS outAngle.y=angleRange.x+(angleRange.y-angleRange.x)*randoms.a;outAngle.x=angleRange.z+(angleRange.w-angleRange.z)*randoms.r; #else outAngle=angleRange.z+(angleRange.w-angleRange.z)*randoms.r; #endif #ifdef POINTEMITTER vec3 randoms2=getRandomVec3(seed.y);vec3 randoms3=getRandomVec3(seed.z);newPosition=vec3(0,0,0);newDirection=direction1+(direction2-direction1)*randoms3; #elif defined(BOXEMITTER) vec3 randoms2=getRandomVec3(seed.y);vec3 randoms3=getRandomVec3(seed.z);newPosition=minEmitBox+(maxEmitBox-minEmitBox)*randoms2;newDirection=direction1+(direction2-direction1)*randoms3; #elif defined(HEMISPHERICEMITTER) vec3 randoms2=getRandomVec3(seed.y);vec3 randoms3=getRandomVec3(seed.z);float phi=2.0*PI*randoms2.x;float theta=acos(2.0*randoms2.y-1.0);float randX=cos(phi)*sin(theta);float randY=cos(theta);float randZ=sin(phi)*sin(theta);newPosition=(radius-(radius*radiusRange*randoms2.z))*vec3(randX,abs(randY),randZ);newDirection=newPosition+directionRandomizer*randoms3; #elif defined(SPHEREEMITTER) vec3 randoms2=getRandomVec3(seed.y);vec3 randoms3=getRandomVec3(seed.z);float phi=2.0*PI*randoms2.x;float theta=acos(2.0*randoms2.y-1.0);float randX=cos(phi)*sin(theta);float randY=cos(theta);float randZ=sin(phi)*sin(theta);newPosition=(radius-(radius*radiusRange*randoms2.z))*vec3(randX,randY,randZ); #ifdef DIRECTEDSPHEREEMITTER newDirection=normalize(direction1+(direction2-direction1)*randoms3); #else newDirection=normalize(newPosition+directionRandomizer*randoms3); #endif #elif defined(CYLINDEREMITTER) vec3 randoms2=getRandomVec3(seed.y);vec3 randoms3=getRandomVec3(seed.z);float yPos=(randoms2.x-0.5)*height;float angle=randoms2.y*PI*2.;float inverseRadiusRangeSquared=((1.-radiusRange)*(1.-radiusRange));float positionRadius=radius*sqrt(inverseRadiusRangeSquared+(randoms2.z*(1.-inverseRadiusRangeSquared)));float xPos=positionRadius*cos(angle);float zPos=positionRadius*sin(angle);newPosition=vec3(xPos,yPos,zPos); #ifdef DIRECTEDCYLINDEREMITTER newDirection=direction1+(direction2-direction1)*randoms3; #else angle=angle+((randoms3.x-0.5)*PI)*directionRandomizer;newDirection=vec3(cos(angle),(randoms3.y-0.5)*directionRandomizer,sin(angle));newDirection=normalize(newDirection); #endif #elif defined(CONEEMITTER) vec3 randoms2=getRandomVec3(seed.y);float s=2.0*PI*randoms2.x; #ifdef CONEEMITTERSPAWNPOINT float h=0.0001; #else float h=randoms2.y*height.y;h=1.-h*h; #endif float lRadius=radius.x-radius.x*randoms2.z*radius.y;lRadius=lRadius*h;float randX=lRadius*sin(s);float randZ=lRadius*cos(s);float randY=h *height.x;newPosition=vec3(randX,randY,randZ); if (abs(cos(coneAngle))==1.0) {newDirection=vec3(0.,1.0,0.);} else {vec3 randoms3=getRandomVec3(seed.z);newDirection=normalize(newPosition+directionRandomizer*randoms3); } #elif defined(CUSTOMEMITTER) newPosition=initialPosition;outInitialPosition=initialPosition; #else newPosition=vec3(0.,0.,0.);newDirection=2.0*(getRandomVec3(seed.w)-vec3(0.5,0.5,0.5)); #endif float power=emitPower.x+(emitPower.y-emitPower.x)*randoms.a; #ifdef LOCAL outPosition=newPosition; #else outPosition=(emitterWM*vec4(newPosition,1.)).xyz; #endif #ifdef CUSTOMEMITTER outDirection=direction; #ifndef BILLBOARD outInitialDirection=direction; #endif #else #ifdef LOCAL vec3 initial=newDirection; #else vec3 initial=(emitterWM*vec4(newDirection,0.)).xyz; #endif outDirection=initial*power; #ifndef BILLBOARD outInitialDirection=initial; #endif #endif #ifdef ANIMATESHEET outCellIndex=cellInfos.x; #ifdef ANIMATESHEETRANDOMSTART outCellStartOffset=randoms.a*outLife; #endif #endif #ifdef NOISE outNoiseCoordinates1=noiseCoordinates1;outNoiseCoordinates2=noiseCoordinates2; #endif } else {float directionScale=timeDelta;outAge=newAge;float ageGradient=newAge/life; #ifdef VELOCITYGRADIENTS directionScale*=texture(velocityGradientSampler,vec2(ageGradient,0)).r; #endif #ifdef DRAGGRADIENTS directionScale*=1.0-texture(dragGradientSampler,vec2(ageGradient,0)).r; #endif #if defined(CUSTOMEMITTER) outPosition=position+(direction-position)*ageGradient; outInitialPosition=initialPosition; #else outPosition=position+direction*directionScale; #endif outLife=life;outSeed=seed; #ifndef COLORGRADIENTS outColor=color; #endif #ifdef SIZEGRADIENTS outSize.x=texture(sizeGradientSampler,vec2(ageGradient,0)).r;outSize.yz=size.yz; #else outSize=size; #endif #ifndef BILLBOARD outInitialDirection=initialDirection; #endif #ifdef CUSTOMEMITTER outDirection=direction; #else vec3 updatedDirection=direction+gravity*timeDelta; #ifdef LIMITVELOCITYGRADIENTS float limitVelocity=texture(limitVelocityGradientSampler,vec2(ageGradient,0)).r;float currentVelocity=length(updatedDirection);if (currentVelocity>limitVelocity) {updatedDirection=updatedDirection*limitVelocityDamping;} #endif outDirection=updatedDirection; #ifdef NOISE float fetchedR=texture(noiseSampler,vec2(noiseCoordinates1.x,noiseCoordinates1.y)*vec2(0.5)+vec2(0.5)).r;float fetchedG=texture(noiseSampler,vec2(noiseCoordinates1.z,noiseCoordinates2.x)*vec2(0.5)+vec2(0.5)).r;float fetchedB=texture(noiseSampler,vec2(noiseCoordinates2.y,noiseCoordinates2.z)*vec2(0.5)+vec2(0.5)).r;vec3 force=vec3(2.*fetchedR-1.,2.*fetchedG-1.,2.*fetchedB-1.)*noiseStrength;outDirection=outDirection+force*timeDelta;outNoiseCoordinates1=noiseCoordinates1;outNoiseCoordinates2=noiseCoordinates2; #endif #endif #ifdef ANGULARSPEEDGRADIENTS float angularSpeed=texture(angularSpeedGradientSampler,vec2(ageGradient,0)).r;outAngle=angle+angularSpeed*timeDelta; #else outAngle=vec2(angle.x+angle.y*timeDelta,angle.y); #endif #ifdef ANIMATESHEET float offsetAge=outAge;float dist=cellInfos.y-cellInfos.x; #ifdef ANIMATESHEETRANDOMSTART outCellStartOffset=cellStartOffset;offsetAge+=cellStartOffset; #else float cellStartOffset=0.; #endif float ratio=0.;if (cellInfos.w==1.0) {ratio=clamp(mod(cellStartOffset+cellInfos.z*offsetAge,life)/life,0.,1.0);} else {ratio=clamp(cellStartOffset+cellInfos.z*offsetAge/life,0.,1.0);} outCellIndex=float(int(cellInfos.x+ratio*dist)); #endif }}`; Le.ShadersStore[Uhe] = Ihe; class Lne { constructor(e, t) { this._renderVAO = [], this._updateVAO = [], this.alignDataInBuffer = !1, this._parent = e, this._engine = t, this._updateEffectOptions = { attributes: [ "position", "initialPosition", "age", "life", "seed", "size", "color", "direction", "initialDirection", "angle", "cellIndex", "cellStartOffset", "noiseCoordinates1", "noiseCoordinates2" ], uniformsNames: [ "currentCount", "timeDelta", "emitterWM", "lifeTime", "color1", "color2", "sizeRange", "scaleRange", "gravity", "emitPower", "direction1", "direction2", "minEmitBox", "maxEmitBox", "radius", "directionRandomizer", "height", "coneAngle", "stopFactor", "angleRange", "radiusRange", "cellInfos", "noiseStrength", "limitVelocityDamping" ], uniformBuffersNames: [], samplers: [ "randomSampler", "randomSampler2", "sizeGradientSampler", "angularSpeedGradientSampler", "velocityGradientSampler", "limitVelocityGradientSampler", "noiseSampler", "dragGradientSampler" ], defines: "", fallbacks: null, onCompiled: null, onError: null, indexParameters: null, maxSimultaneousLights: 0, transformFeedbackVaryings: [] }; } contextLost() { this._updateEffect = void 0, this._renderVAO.length = 0, this._updateVAO.length = 0; } isUpdateBufferCreated() { return !!this._updateEffect; } isUpdateBufferReady() { var e, t; return (t = (e = this._updateEffect) === null || e === void 0 ? void 0 : e.isReady()) !== null && t !== void 0 ? t : !1; } createUpdateBuffer(e) { return this._updateEffectOptions.transformFeedbackVaryings = ["outPosition"], this._updateEffectOptions.transformFeedbackVaryings.push("outAge"), this._updateEffectOptions.transformFeedbackVaryings.push("outSize"), this._updateEffectOptions.transformFeedbackVaryings.push("outLife"), this._updateEffectOptions.transformFeedbackVaryings.push("outSeed"), this._updateEffectOptions.transformFeedbackVaryings.push("outDirection"), this._parent.particleEmitterType instanceof dD && this._updateEffectOptions.transformFeedbackVaryings.push("outInitialPosition"), this._parent._colorGradientsTexture || this._updateEffectOptions.transformFeedbackVaryings.push("outColor"), this._parent._isBillboardBased || this._updateEffectOptions.transformFeedbackVaryings.push("outInitialDirection"), this._parent.noiseTexture && (this._updateEffectOptions.transformFeedbackVaryings.push("outNoiseCoordinates1"), this._updateEffectOptions.transformFeedbackVaryings.push("outNoiseCoordinates2")), this._updateEffectOptions.transformFeedbackVaryings.push("outAngle"), this._parent.isAnimationSheetEnabled && (this._updateEffectOptions.transformFeedbackVaryings.push("outCellIndex"), this._parent.spriteRandomStartCell && this._updateEffectOptions.transformFeedbackVaryings.push("outCellStartOffset")), this._updateEffectOptions.defines = e, this._updateEffect = new An("gpuUpdateParticles", this._updateEffectOptions, this._engine), new Mne(this._updateEffect); } createVertexBuffers(e, t) { this._updateVAO.push(this._createUpdateVAO(e)), this._renderVAO.push(this._engine.recordVertexArrayObject(t, null, this._parent._getWrapper(this._parent.blendMode).effect)), this._engine.bindArrayBuffer(null), this._renderVertexBuffers = t; } createParticleBuffer(e) { return e; } bindDrawBuffers(e, t, r) { r ? this._engine.bindBuffers(this._renderVertexBuffers, r, t) : this._engine.bindVertexArrayObject(this._renderVAO[e], null); } preUpdateParticleBuffer() { const e = this._engine; if (this._engine.enableEffect(this._updateEffect), !e.setState) throw new Error("GPU particles cannot work without a full Engine. ThinEngine is not supported"); } updateParticleBuffer(e, t, r) { this._updateEffect.setTexture("randomSampler", this._parent._randomTexture), this._updateEffect.setTexture("randomSampler2", this._parent._randomTexture2), this._parent._sizeGradientsTexture && this._updateEffect.setTexture("sizeGradientSampler", this._parent._sizeGradientsTexture), this._parent._angularSpeedGradientsTexture && this._updateEffect.setTexture("angularSpeedGradientSampler", this._parent._angularSpeedGradientsTexture), this._parent._velocityGradientsTexture && this._updateEffect.setTexture("velocityGradientSampler", this._parent._velocityGradientsTexture), this._parent._limitVelocityGradientsTexture && this._updateEffect.setTexture("limitVelocityGradientSampler", this._parent._limitVelocityGradientsTexture), this._parent._dragGradientsTexture && this._updateEffect.setTexture("dragGradientSampler", this._parent._dragGradientsTexture), this._parent.noiseTexture && this._updateEffect.setTexture("noiseSampler", this._parent.noiseTexture), this._engine.bindVertexArrayObject(this._updateVAO[e], null); const n = this._engine; n.bindTransformFeedbackBuffer(t.getBuffer()), n.setRasterizerState(!1), n.beginTransformFeedback(!0), n.drawArraysType(3, 0, r), n.endTransformFeedback(), n.setRasterizerState(!0), n.bindTransformFeedbackBuffer(null); } releaseBuffers() { } releaseVertexBuffers() { for (let e = 0; e < this._updateVAO.length; e++) this._engine.releaseVertexArrayObject(this._updateVAO[e]); this._updateVAO.length = 0; for (let e = 0; e < this._renderVAO.length; e++) this._engine.releaseVertexArrayObject(this._renderVAO[e]); this._renderVAO.length = 0; } _createUpdateVAO(e) { const t = {}; t.position = e.createVertexBuffer("position", 0, 3); let r = 3; t.age = e.createVertexBuffer("age", r, 1), r += 1, t.size = e.createVertexBuffer("size", r, 3), r += 3, t.life = e.createVertexBuffer("life", r, 1), r += 1, t.seed = e.createVertexBuffer("seed", r, 4), r += 4, t.direction = e.createVertexBuffer("direction", r, 3), r += 3, this._parent.particleEmitterType instanceof dD && (t.initialPosition = e.createVertexBuffer("initialPosition", r, 3), r += 3), this._parent._colorGradientsTexture || (t.color = e.createVertexBuffer("color", r, 4), r += 4), this._parent._isBillboardBased || (t.initialDirection = e.createVertexBuffer("initialDirection", r, 3), r += 3), this._parent.noiseTexture && (t.noiseCoordinates1 = e.createVertexBuffer("noiseCoordinates1", r, 3), r += 3, t.noiseCoordinates2 = e.createVertexBuffer("noiseCoordinates2", r, 3), r += 3), this._parent._angularSpeedGradientsTexture ? (t.angle = e.createVertexBuffer("angle", r, 1), r += 1) : (t.angle = e.createVertexBuffer("angle", r, 2), r += 2), this._parent._isAnimationSheetEnabled && (t.cellIndex = e.createVertexBuffer("cellIndex", r, 1), r += 1, this._parent.spriteRandomStartCell && (t.cellStartOffset = e.createVertexBuffer("cellStartOffset", r, 1), r += 1)); const n = this._engine.recordVertexArrayObject(t, null, this._updateEffect); return this._engine.bindArrayBuffer(null), n; } } Ue("BABYLON.WebGL2ParticleSystem", Lne); const Rhe = "gpuUpdateParticlesComputeShader", Vhe = `struct Particle {position : vec3, age : f32, size : vec3, life : f32, seed : vec4, direction : vec3, dummy0: f32, #ifdef CUSTOMEMITTER initialPosition : vec3, dummy1: f32, #endif #ifndef COLORGRADIENTS color : vec4, #endif #ifndef BILLBOARD initialDirection : vec3, dummy2: f32, #endif #ifdef NOISE noiseCoordinates1 : vec3, dummy3: f32, noiseCoordinates2 : vec3, dummy4: f32, #endif #ifdef ANGULARSPEEDGRADIENTS angle : f32, #else angle : vec2, #endif #ifdef ANIMATESHEET cellIndex : f32, #ifdef ANIMATESHEETRANDOMSTART cellStartOffset : f32, #endif #endif };struct Particles {particles : array,};struct SimParams {currentCount : f32, timeDelta : f32, stopFactor : f32, randomTextureSize: i32, lifeTime : vec2, emitPower : vec2, #ifndef COLORGRADIENTS color1 : vec4, color2 : vec4, #endif sizeRange : vec2, scaleRange : vec4, angleRange : vec4, gravity : vec3, #ifdef LIMITVELOCITYGRADIENTS limitVelocityDamping : f32, #endif #ifdef ANIMATESHEET cellInfos : vec4, #endif #ifdef NOISE noiseStrength : vec3, #endif #ifndef LOCAL emitterWM : mat4x4, #endif #ifdef BOXEMITTER direction1 : vec3, direction2 : vec3, minEmitBox : vec3, maxEmitBox : vec3, #endif #ifdef CONEEMITTER radius : vec2, coneAngle : f32, height : vec2, directionRandomizer : f32, #endif #ifdef CYLINDEREMITTER radius : f32, height : f32, radiusRange : f32, #ifdef DIRECTEDCYLINDEREMITTER direction1 : vec3, direction2 : vec3, #else directionRandomizer : f32, #endif #endif #ifdef HEMISPHERICEMITTER radius : f32, radiusRange : f32, directionRandomizer : f32, #endif #ifdef POINTEMITTER direction1 : vec3, direction2 : vec3, #endif #ifdef SPHEREEMITTER radius : f32, radiusRange : f32, #ifdef DIRECTEDSPHEREEMITTER direction1 : vec3, direction2 : vec3, #else directionRandomizer : f32, #endif #endif };@binding(0) @group(0) var params : SimParams;@binding(1) @group(0) var particlesIn : Particles;@binding(2) @group(0) var particlesOut : Particles;@binding(3) @group(0) var randomTexture : texture_2d;@binding(4) @group(0) var randomTexture2 : texture_2d; #ifdef SIZEGRADIENTS @binding(0) @group(1) var sizeGradientSampler : sampler;@binding(1) @group(1) var sizeGradientTexture : texture_2d; #endif #ifdef ANGULARSPEEDGRADIENTS @binding(2) @group(1) var angularSpeedGradientSampler : sampler;@binding(3) @group(1) var angularSpeedGradientTexture : texture_2d; #endif #ifdef VELOCITYGRADIENTS @binding(4) @group(1) var velocityGradientSampler : sampler;@binding(5) @group(1) var velocityGradientTexture : texture_2d; #endif #ifdef LIMITVELOCITYGRADIENTS @binding(6) @group(1) var limitVelocityGradientSampler : sampler;@binding(7) @group(1) var limitVelocityGradientTexture : texture_2d; #endif #ifdef DRAGGRADIENTS @binding(8) @group(1) var dragGradientSampler : sampler;@binding(9) @group(1) var dragGradientTexture : texture_2d; #endif #ifdef NOISE @binding(10) @group(1) var noiseSampler : sampler;@binding(11) @group(1) var noiseTexture : texture_2d; #endif fn getRandomVec3(offset : f32,vertexID : f32)->vec3 {return textureLoad(randomTexture2,vec2(i32(vertexID*offset/params.currentCount*f32(params.randomTextureSize)) % params.randomTextureSize,0),0).rgb;} fn getRandomVec4(offset : f32,vertexID : f32)->vec4 {return textureLoad(randomTexture,vec2(i32(vertexID*offset/params.currentCount*f32(params.randomTextureSize)) % params.randomTextureSize,0),0);} @compute @workgroup_size(64) fn main(@builtin(global_invocation_id) GlobalInvocationID : vec3) {let index : u32=GlobalInvocationID.x;let vertexID : f32=f32(index);if (index>=u32(params.currentCount)) {return;} let PI : f32=3.14159;let timeDelta : f32=params.timeDelta;let newAge : f32=particlesIn.particles[index].age+timeDelta;let life : f32=particlesIn.particles[index].life;let seed : vec4=particlesIn.particles[index].seed;let direction : vec3=particlesIn.particles[index].direction;if (newAge>=life && params.stopFactor != 0.) {var newPosition : vec3;var newDirection : vec3;let randoms : vec4=getRandomVec4(seed.x,vertexID);let outLife : f32=params.lifeTime.x+(params.lifeTime.y-params.lifeTime.x)*randoms.r;particlesOut.particles[index].life=outLife;particlesOut.particles[index].age=newAge-life;particlesOut.particles[index].seed=seed;var sizex : f32; #ifdef SIZEGRADIENTS sizex=textureSampleLevel(sizeGradientTexture,sizeGradientSampler,vec2(0.,0.),0.).r; #else sizex=params.sizeRange.x+(params.sizeRange.y-params.sizeRange.x)*randoms.g; #endif particlesOut.particles[index].size=vec3( sizex, params.scaleRange.x+(params.scaleRange.y-params.scaleRange.x)*randoms.b, params.scaleRange.z+(params.scaleRange.w-params.scaleRange.z)*randoms.a); #ifndef COLORGRADIENTS particlesOut.particles[index].color=params.color1+(params.color2-params.color1)*randoms.b; #endif #ifndef ANGULARSPEEDGRADIENTS particlesOut.particles[index].angle=vec2( params.angleRange.z+(params.angleRange.w-params.angleRange.z)*randoms.r, params.angleRange.x+(params.angleRange.y-params.angleRange.x)*randoms.a); #else particlesOut.particles[index].angle=params.angleRange.z+(params.angleRange.w-params.angleRange.z)*randoms.r; #endif #if defined(POINTEMITTER) let randoms2 : vec3=getRandomVec3(seed.y,vertexID);let randoms3 : vec3=getRandomVec3(seed.z,vertexID);newPosition=vec3(0.,0.,0.);newDirection=params.direction1+(params.direction2-params.direction1)*randoms3; #elif defined(BOXEMITTER) let randoms2 : vec3=getRandomVec3(seed.y,vertexID);let randoms3 : vec3=getRandomVec3(seed.z,vertexID);newPosition=params.minEmitBox+(params.maxEmitBox-params.minEmitBox)*randoms2;newDirection=params.direction1+(params.direction2-params.direction1)*randoms3; #elif defined(HEMISPHERICEMITTER) let randoms2 : vec3=getRandomVec3(seed.y,vertexID);let randoms3 : vec3=getRandomVec3(seed.z,vertexID);let phi : f32=2.0*PI*randoms2.x;let theta : f32=acos(-1.0+2.0*randoms2.y);let randX : f32=cos(phi)*sin(theta);let randY : f32=cos(theta);let randZ : f32=sin(phi)*sin(theta);newPosition=(params.radius-(params.radius*params.radiusRange*randoms2.z))*vec3(randX,abs(randY),randZ);newDirection=normalize(newPosition+params.directionRandomizer*randoms3); #elif defined(SPHEREEMITTER) let randoms2 : vec3=getRandomVec3(seed.y,vertexID);let randoms3 : vec3=getRandomVec3(seed.z,vertexID);let phi : f32=2.0*PI*randoms2.x;let theta : f32=acos(-1.0+2.0*randoms2.y);let randX : f32=cos(phi)*sin(theta);let randY : f32=cos(theta);let randZ : f32=sin(phi)*sin(theta);newPosition=(params.radius-(params.radius*params.radiusRange*randoms2.z))*vec3(randX,randY,randZ); #ifdef DIRECTEDSPHEREEMITTER newDirection=normalize(params.direction1+(params.direction2-params.direction1)*randoms3); #else newDirection=normalize(newPosition+params.directionRandomizer*randoms3); #endif #elif defined(CYLINDEREMITTER) let randoms2 : vec3=getRandomVec3(seed.y,vertexID);let randoms3 : vec3=getRandomVec3(seed.z,vertexID);let yPos : f32=(-0.5+randoms2.x)*params.height;var angle : f32=randoms2.y*PI*2.;let inverseRadiusRangeSquared : f32=(1.-params.radiusRange)*(1.-params.radiusRange);let positionRadius : f32=params.radius*sqrt(inverseRadiusRangeSquared+randoms2.z*(1.-inverseRadiusRangeSquared));let xPos : f32=positionRadius*cos(angle);let zPos : f32=positionRadius*sin(angle);newPosition=vec3(xPos,yPos,zPos); #ifdef DIRECTEDCYLINDEREMITTER newDirection=params.direction1+(params.direction2-params.direction1)*randoms3; #else angle=angle+(-0.5+randoms3.x)*PI*params.directionRandomizer;newDirection=vec3(cos(angle),(-0.5+randoms3.y)*params.directionRandomizer,sin(angle));newDirection=normalize(newDirection); #endif #elif defined(CONEEMITTER) let randoms2 : vec3=getRandomVec3(seed.y,vertexID);let s : f32=2.0*PI*randoms2.x; #ifdef CONEEMITTERSPAWNPOINT let h : f32=0.0001; #else var h : f32=randoms2.y*params.height.y;h=1.-h*h; #endif var lRadius : f32=params.radius.x-params.radius.x*randoms2.z*params.radius.y;lRadius=lRadius*h;let randX : f32=lRadius*sin(s);let randZ : f32=lRadius*cos(s);let randY : f32=h *params.height.x;newPosition=vec3(randX,randY,randZ); if (abs(cos(params.coneAngle))==1.0) {newDirection=vec3(0.,1.0,0.);} else {let randoms3 : vec3=getRandomVec3(seed.z,vertexID);newDirection=normalize(newPosition+params.directionRandomizer*randoms3); } #elif defined(CUSTOMEMITTER) newPosition=particlesIn.particles[index].initialPosition;particlesOut.particles[index].initialPosition=newPosition; #else newPosition=vec3(0.,0.,0.);newDirection=2.0*(getRandomVec3(seed.w,vertexID)-vec3(0.5,0.5,0.5)); #endif let power : f32=params.emitPower.x+(params.emitPower.y-params.emitPower.x)*randoms.a; #ifdef LOCAL particlesOut.particles[index].position=newPosition; #else particlesOut.particles[index].position=(params.emitterWM*vec4(newPosition,1.)).xyz; #endif #ifdef CUSTOMEMITTER particlesOut.particles[index].direction=direction; #ifndef BILLBOARD particlesOut.particles[index].initialDirection=direction; #endif #else #ifdef LOCAL let initial : vec3=newDirection; #else let initial : vec3=(params.emitterWM*vec4(newDirection,0.)).xyz; #endif particlesOut.particles[index].direction=initial*power; #ifndef BILLBOARD particlesOut.particles[index].initialDirection=initial; #endif #endif #ifdef ANIMATESHEET particlesOut.particles[index].cellIndex=params.cellInfos.x; #ifdef ANIMATESHEETRANDOMSTART particlesOut.particles[index].cellStartOffset=randoms.a*outLife; #endif #endif #ifdef NOISE particlesOut.particles[index].noiseCoordinates1=particlesIn.particles[index].noiseCoordinates1;particlesOut.particles[index].noiseCoordinates2=particlesIn.particles[index].noiseCoordinates2; #endif } else {var directionScale : f32=timeDelta;particlesOut.particles[index].age=newAge;let ageGradient : f32=newAge/life; #ifdef VELOCITYGRADIENTS directionScale=directionScale*textureSampleLevel(velocityGradientTexture,velocityGradientSampler,vec2(ageGradient,0.),0.).r; #endif #ifdef DRAGGRADIENTS directionScale=directionScale*(1.0-textureSampleLevel(dragGradientTexture,dragGradientSampler,vec2(ageGradient,0.),0.).r); #endif let position : vec3=particlesIn.particles[index].position; #if defined(CUSTOMEMITTER) particlesOut.particles[index].position=position+(direction-position)*ageGradient; particlesOut.particles[index].initialPosition=particlesIn.particles[index].initialPosition; #else particlesOut.particles[index].position=position+direction*directionScale; #endif particlesOut.particles[index].life=life;particlesOut.particles[index].seed=seed; #ifndef COLORGRADIENTS particlesOut.particles[index].color=particlesIn.particles[index].color; #endif #ifdef SIZEGRADIENTS particlesOut.particles[index].size=vec3( textureSampleLevel(sizeGradientTexture,sizeGradientSampler,vec2(ageGradient,0.),0.).r, particlesIn.particles[index].size.yz); #else particlesOut.particles[index].size=particlesIn.particles[index].size; #endif #ifndef BILLBOARD particlesOut.particles[index].initialDirection=particlesIn.particles[index].initialDirection; #endif #ifdef CUSTOMEMITTER particlesOut.particles[index].direction=direction; #else var updatedDirection : vec3=direction+params.gravity*timeDelta; #ifdef LIMITVELOCITYGRADIENTS let limitVelocity : f32=textureSampleLevel(limitVelocityGradientTexture,limitVelocityGradientSampler,vec2(ageGradient,0.),0.).r;let currentVelocity : f32=length(updatedDirection);if (currentVelocity>limitVelocity) {updatedDirection=updatedDirection*params.limitVelocityDamping;} #endif particlesOut.particles[index].direction=updatedDirection; #ifdef NOISE let noiseCoordinates1 : vec3=particlesIn.particles[index].noiseCoordinates1;let noiseCoordinates2 : vec3=particlesIn.particles[index].noiseCoordinates2;let fetchedR : f32=textureSampleLevel(noiseTexture,noiseSampler,vec2(noiseCoordinates1.x,noiseCoordinates1.y)*vec2(0.5,0.5)+vec2(0.5,0.5),0.).r;let fetchedG : f32=textureSampleLevel(noiseTexture,noiseSampler,vec2(noiseCoordinates1.z,noiseCoordinates2.x)*vec2(0.5,0.5)+vec2(0.5,0.5),0.).r;let fetchedB : f32=textureSampleLevel(noiseTexture,noiseSampler,vec2(noiseCoordinates2.y,noiseCoordinates2.z)*vec2(0.5,0.5)+vec2(0.5,0.5),0.).r;let force : vec3=vec3(-1.+2.*fetchedR,-1.+2.*fetchedG,-1.+2.*fetchedB)*params.noiseStrength;particlesOut.particles[index].direction=particlesOut.particles[index].direction+force*timeDelta;particlesOut.particles[index].noiseCoordinates1=noiseCoordinates1;particlesOut.particles[index].noiseCoordinates2=noiseCoordinates2; #endif #endif #ifdef ANGULARSPEEDGRADIENTS let angularSpeed : f32=textureSampleLevel(angularSpeedGradientTexture,angularSpeedGradientSampler,vec2(ageGradient,0.),0.).r;particlesOut.particles[index].angle=particlesIn.particles[index].angle+angularSpeed*timeDelta; #else let angle : vec2=particlesIn.particles[index].angle;particlesOut.particles[index].angle=vec2(angle.x+angle.y*timeDelta,angle.y); #endif #ifdef ANIMATESHEET var offsetAge : f32=particlesOut.particles[index].age;let dist : f32=params.cellInfos.y-params.cellInfos.x; #ifdef ANIMATESHEETRANDOMSTART let cellStartOffset : f32=particlesIn.particles[index].cellStartOffset;particlesOut.particles[index].cellStartOffset=cellStartOffset;offsetAge=offsetAge+cellStartOffset; #else let cellStartOffset : f32=0.; #endif var ratio : f32;if (params.cellInfos.w==1.0) {ratio=clamp(((cellStartOffset+params.cellInfos.z*offsetAge) % life)/life,0.,1.0);} else {ratio=clamp((cellStartOffset+params.cellInfos.z*offsetAge)/life,0.,1.0);} particlesOut.particles[index].cellIndex=f32(i32(params.cellInfos.x+ratio*dist)); #endif }} `; Le.ShadersStoreWGSL[Rhe] = Vhe; class Kne { constructor(e, t) { this._bufferComputeShader = [], this._renderVertexBuffers = [], this.alignDataInBuffer = !0, this._parent = e, this._engine = t; } contextLost() { this._updateComputeShader = void 0, this._bufferComputeShader.length = 0, this._renderVertexBuffers.length = 0; } isUpdateBufferCreated() { return !!this._updateComputeShader; } isUpdateBufferReady() { var e, t; return (t = (e = this._updateComputeShader) === null || e === void 0 ? void 0 : e.isReady()) !== null && t !== void 0 ? t : !1; } createUpdateBuffer(e) { var t; const r = { params: { group: 0, binding: 0 }, particlesIn: { group: 0, binding: 1 }, particlesOut: { group: 0, binding: 2 }, randomTexture: { group: 0, binding: 3 }, randomTexture2: { group: 0, binding: 4 } }; return this._parent._sizeGradientsTexture && (r.sizeGradientTexture = { group: 1, binding: 1 }), this._parent._angularSpeedGradientsTexture && (r.angularSpeedGradientTexture = { group: 1, binding: 3 }), this._parent._velocityGradientsTexture && (r.velocityGradientTexture = { group: 1, binding: 5 }), this._parent._limitVelocityGradientsTexture && (r.limitVelocityGradientTexture = { group: 1, binding: 7 }), this._parent._dragGradientsTexture && (r.dragGradientTexture = { group: 1, binding: 9 }), this._parent.noiseTexture && (r.noiseTexture = { group: 1, binding: 11 }), this._updateComputeShader = new vU("updateParticles", this._engine, "gpuUpdateParticles", { bindingsMapping: r, defines: e.split(` `) }), (t = this._simParamsComputeShader) === null || t === void 0 || t.dispose(), this._simParamsComputeShader = new yr(this._engine), this._simParamsComputeShader.addUniform("currentCount", 1), this._simParamsComputeShader.addUniform("timeDelta", 1), this._simParamsComputeShader.addUniform("stopFactor", 1), this._simParamsComputeShader.addUniform("randomTextureSize", 1), this._simParamsComputeShader.addUniform("lifeTime", 2), this._simParamsComputeShader.addUniform("emitPower", 2), this._parent._colorGradientsTexture || (this._simParamsComputeShader.addUniform("color1", 4), this._simParamsComputeShader.addUniform("color2", 4)), this._simParamsComputeShader.addUniform("sizeRange", 2), this._simParamsComputeShader.addUniform("scaleRange", 4), this._simParamsComputeShader.addUniform("angleRange", 4), this._simParamsComputeShader.addUniform("gravity", 3), this._parent._limitVelocityGradientsTexture && this._simParamsComputeShader.addUniform("limitVelocityDamping", 1), this._parent.isAnimationSheetEnabled && this._simParamsComputeShader.addUniform("cellInfos", 4), this._parent.noiseTexture && this._simParamsComputeShader.addUniform("noiseStrength", 3), this._parent.isLocal || this._simParamsComputeShader.addUniform("emitterWM", 16), this._parent.particleEmitterType && this._parent.particleEmitterType.buildUniformLayout(this._simParamsComputeShader), this._updateComputeShader.setUniformBuffer("params", this._simParamsComputeShader), new Mne(this._simParamsComputeShader); } createVertexBuffers(e, t) { this._renderVertexBuffers.push(t); } createParticleBuffer(e) { const t = new bee(this._engine, e.length * 4, 11); return t.update(e), this._bufferComputeShader.push(t), t.getBuffer(); } bindDrawBuffers(e, t, r) { this._engine.bindBuffers(this._renderVertexBuffers[e], r, t); } preUpdateParticleBuffer() { } updateParticleBuffer(e, t, r) { this._simParamsComputeShader.update(), this._updateComputeShader.setTexture("randomTexture", this._parent._randomTexture, !1), this._updateComputeShader.setTexture("randomTexture2", this._parent._randomTexture2, !1), this._parent._sizeGradientsTexture && this._updateComputeShader.setTexture("sizeGradientTexture", this._parent._sizeGradientsTexture), this._parent._angularSpeedGradientsTexture && this._updateComputeShader.setTexture("angularSpeedGradientTexture", this._parent._angularSpeedGradientsTexture), this._parent._velocityGradientsTexture && this._updateComputeShader.setTexture("velocityGradientTexture", this._parent._velocityGradientsTexture), this._parent._limitVelocityGradientsTexture && this._updateComputeShader.setTexture("limitVelocityGradientTexture", this._parent._limitVelocityGradientsTexture), this._parent._dragGradientsTexture && this._updateComputeShader.setTexture("dragGradientTexture", this._parent._dragGradientsTexture), this._parent.noiseTexture && this._updateComputeShader.setTexture("noiseTexture", this._parent.noiseTexture), this._updateComputeShader.setStorageBuffer("particlesIn", this._bufferComputeShader[e]), this._updateComputeShader.setStorageBuffer("particlesOut", this._bufferComputeShader[e ^ 1]), this._updateComputeShader.dispatch(Math.ceil(r / 64)); } releaseBuffers() { var e; for (let t = 0; t < this._bufferComputeShader.length; ++t) this._bufferComputeShader[t].dispose(); this._bufferComputeShader.length = 0, (e = this._simParamsComputeShader) === null || e === void 0 || e.dispose(), this._simParamsComputeShader = null, this._updateComputeShader = null; } releaseVertexBuffers() { this._renderVertexBuffers.length = 0; } } Ue("BABYLON.ComputeShaderParticleSystem", Kne); class yY { /** * Creates a new color4 gradient * @param gradient gets or sets the gradient value (between 0 and 1) * @param color1 gets or sets first associated color * @param color2 gets or sets first second color */ constructor(e, t, r) { this.gradient = e, this.color1 = t, this.color2 = r; } /** * Will get a color picked randomly between color1 and color2. * If color2 is undefined then color1 will be used * @param result defines the target Color4 to store the result in */ getColorToRef(e) { if (!this.color2) { e.copyFrom(this.color1); return; } xt.LerpToRef(this.color1, this.color2, Math.random(), e); } } class Jne { /** * Creates a new color3 gradient * @param gradient gets or sets the gradient value (between 0 and 1) * @param color gets or sets associated color */ constructor(e, t) { this.gradient = e, this.color = t; } } class kY { /** * Creates a new factor gradient * @param gradient gets or sets the gradient value (between 0 and 1) * @param factor1 gets or sets first associated factor * @param factor2 gets or sets second associated factor */ constructor(e, t, r) { this.gradient = e, this.factor1 = t, this.factor2 = r; } /** * Will get a number picked randomly between factor1 and factor2. * If factor2 is undefined then factor1 will be used * @returns the picked number */ getFactor() { return this.factor2 === void 0 || this.factor2 === this.factor1 ? this.factor1 : this.factor1 + (this.factor2 - this.factor1) * Math.random(); } } class kl { /** * Gets the current gradient from an array of IValueGradient * @param ratio defines the current ratio to get * @param gradients defines the array of IValueGradient * @param updateFunc defines the callback function used to get the final value from the selected gradients */ static GetCurrentGradient(e, t, r) { if (t[0].gradient > e) { r(t[0], t[0], 1); return; } for (let i = 0; i < t.length - 1; i++) { const s = t[i], a = t[i + 1]; if (e >= s.gradient && e <= a.gradient) { const f = (e - s.gradient) / (a.gradient - s.gradient); r(s, a, f); return; } } const n = t.length - 1; r(t[n], t[n], 1); } } class FS { /** * Creates a new instance Particle * @param particleSystem the particle system the particle belongs to */ constructor(e) { this.particleSystem = e, this.position = S.Zero(), this.direction = S.Zero(), this.color = new xt(0, 0, 0, 0), this.colorStep = new xt(0, 0, 0, 0), this.lifeTime = 1, this.age = 0, this.size = 0, this.scale = new at(1, 1), this.angle = 0, this.angularSpeed = 0, this.cellIndex = 0, this._attachedSubEmitters = null, this._currentColor1 = new xt(0, 0, 0, 0), this._currentColor2 = new xt(0, 0, 0, 0), this._currentSize1 = 0, this._currentSize2 = 0, this._currentAngularSpeed1 = 0, this._currentAngularSpeed2 = 0, this._currentVelocity1 = 0, this._currentVelocity2 = 0, this._currentLimitVelocity1 = 0, this._currentLimitVelocity2 = 0, this._currentDrag1 = 0, this._currentDrag2 = 0, this.id = FS._Count++, this.particleSystem.isAnimationSheetEnabled && this._updateCellInfoFromSystem(); } _updateCellInfoFromSystem() { this.cellIndex = this.particleSystem.startSpriteCellID; } /** * Defines how the sprite cell index is updated for the particle */ updateCellIndex() { let e = this.age, t = this.particleSystem.spriteCellChangeSpeed; this.particleSystem.spriteRandomStartCell && (this._randomCellOffset === void 0 && (this._randomCellOffset = Math.random() * this.lifeTime), t === 0 ? (t = 1, e = this._randomCellOffset) : e += this._randomCellOffset); const r = this._initialEndSpriteCellID - this._initialStartSpriteCellID; let n; this._initialSpriteCellLoop ? n = Xt.Clamp(e * t % this.lifeTime / this.lifeTime) : n = Xt.Clamp(e * t / this.lifeTime), this.cellIndex = this._initialStartSpriteCellID + n * r | 0; } /** * @internal */ _inheritParticleInfoToSubEmitter(e) { if (e.particleSystem.emitter.position) { const t = e.particleSystem.emitter; if (t.position.copyFrom(this.position), e.inheritDirection) { const r = ue.Vector3[0]; this.direction.normalizeToRef(r), t.setDirection(r, 0, Math.PI / 2); } } else e.particleSystem.emitter.copyFrom(this.position); this.direction.scaleToRef(e.inheritedVelocityAmount / 2, ue.Vector3[0]), e.particleSystem._inheritedVelocityOffset.copyFrom(ue.Vector3[0]); } /** @internal */ _inheritParticleInfoToSubEmitters() { this._attachedSubEmitters && this._attachedSubEmitters.length > 0 && this._attachedSubEmitters.forEach((e) => { this._inheritParticleInfoToSubEmitter(e); }); } /** @internal */ _reset() { this.age = 0, this.id = FS._Count++, this._currentColorGradient = null, this._currentSizeGradient = null, this._currentAngularSpeedGradient = null, this._currentVelocityGradient = null, this._currentLimitVelocityGradient = null, this._currentDragGradient = null, this.cellIndex = this.particleSystem.startSpriteCellID, this._randomCellOffset = void 0; } /** * Copy the properties of particle to another one. * @param other the particle to copy the information to. */ copyTo(e) { e.position.copyFrom(this.position), this._initialDirection ? e._initialDirection ? e._initialDirection.copyFrom(this._initialDirection) : e._initialDirection = this._initialDirection.clone() : e._initialDirection = null, e.direction.copyFrom(this.direction), this._localPosition && (e._localPosition ? e._localPosition.copyFrom(this._localPosition) : e._localPosition = this._localPosition.clone()), e.color.copyFrom(this.color), e.colorStep.copyFrom(this.colorStep), e.lifeTime = this.lifeTime, e.age = this.age, e._randomCellOffset = this._randomCellOffset, e.size = this.size, e.scale.copyFrom(this.scale), e.angle = this.angle, e.angularSpeed = this.angularSpeed, e.particleSystem = this.particleSystem, e.cellIndex = this.cellIndex, e.id = this.id, e._attachedSubEmitters = this._attachedSubEmitters, this._currentColorGradient && (e._currentColorGradient = this._currentColorGradient, e._currentColor1.copyFrom(this._currentColor1), e._currentColor2.copyFrom(this._currentColor2)), this._currentSizeGradient && (e._currentSizeGradient = this._currentSizeGradient, e._currentSize1 = this._currentSize1, e._currentSize2 = this._currentSize2), this._currentAngularSpeedGradient && (e._currentAngularSpeedGradient = this._currentAngularSpeedGradient, e._currentAngularSpeed1 = this._currentAngularSpeed1, e._currentAngularSpeed2 = this._currentAngularSpeed2), this._currentVelocityGradient && (e._currentVelocityGradient = this._currentVelocityGradient, e._currentVelocity1 = this._currentVelocity1, e._currentVelocity2 = this._currentVelocity2), this._currentLimitVelocityGradient && (e._currentLimitVelocityGradient = this._currentLimitVelocityGradient, e._currentLimitVelocity1 = this._currentLimitVelocity1, e._currentLimitVelocity2 = this._currentLimitVelocity2), this._currentDragGradient && (e._currentDragGradient = this._currentDragGradient, e._currentDrag1 = this._currentDrag1, e._currentDrag2 = this._currentDrag2), this.particleSystem.isAnimationSheetEnabled && (e._initialStartSpriteCellID = this._initialStartSpriteCellID, e._initialEndSpriteCellID = this._initialEndSpriteCellID, e._initialSpriteCellLoop = this._initialSpriteCellLoop), this.particleSystem.useRampGradients && (e.remapData && this.remapData ? e.remapData.copyFrom(this.remapData) : e.remapData = new Ir(0, 0, 0, 0)), this._randomNoiseCoordinates1 && (e._randomNoiseCoordinates1 ? (e._randomNoiseCoordinates1.copyFrom(this._randomNoiseCoordinates1), e._randomNoiseCoordinates2.copyFrom(this._randomNoiseCoordinates2)) : (e._randomNoiseCoordinates1 = this._randomNoiseCoordinates1.clone(), e._randomNoiseCoordinates2 = this._randomNoiseCoordinates2.clone())); } } FS._Count = 0; var NS; (function(A) { A[A.ATTACHED = 0] = "ATTACHED", A[A.END = 1] = "END"; })(NS || (NS = {})); class vq { /** * Creates a sub emitter * @param particleSystem the particle system to be used by the sub emitter */ constructor(e) { if (this.particleSystem = e, this.type = NS.END, this.inheritDirection = !1, this.inheritedVelocityAmount = 0, !e.emitter || !e.emitter.dispose) { const t = Jo("BABYLON.AbstractMesh"); e.emitter = new t("SubemitterSystemEmitter", e.getScene()), e._disposeEmitterOnDispose = !0; } } /** * Clones the sub emitter * @returns the cloned sub emitter */ clone() { let e = this.particleSystem.emitter; if (!e) e = new S(); else if (e instanceof S) e = e.clone(); else if (e.getClassName().indexOf("Mesh") !== -1) { const r = Jo("BABYLON.Mesh"); e = new r("", e.getScene()), e.isVisible = !1; } const t = new vq(this.particleSystem.clone(this.particleSystem.name, e)); return t.particleSystem.name += "Clone", t.type = this.type, t.inheritDirection = this.inheritDirection, t.inheritedVelocityAmount = this.inheritedVelocityAmount, t.particleSystem._disposeEmitterOnDispose = !0, t.particleSystem.disposeOnStop = !0, t; } /** * Serialize current object to a JSON object * @param serializeTexture defines if the texture must be serialized as well * @returns the serialized object */ serialize(e = !1) { const t = {}; return t.type = this.type, t.inheritDirection = this.inheritDirection, t.inheritedVelocityAmount = this.inheritedVelocityAmount, t.particleSystem = this.particleSystem.serialize(e), t; } /** * @internal */ // eslint-disable-next-line @typescript-eslint/no-unused-vars static _ParseParticleSystem(e, t, r, n = !1) { throw qn("ParseParticle"); } /** * Creates a new SubEmitter from a serialized JSON version * @param serializationObject defines the JSON object to read from * @param sceneOrEngine defines the hosting scene or the hosting engine * @param rootUrl defines the rootUrl for data loading * @returns a new SubEmitter */ static Parse(e, t, r) { const n = e.particleSystem, i = new vq(vq._ParseParticleSystem(n, t, r, !0)); return i.type = e.type, i.inheritDirection = e.inheritDirection, i.inheritedVelocityAmount = e.inheritedVelocityAmount, i.particleSystem._isSubEmitter = !0, i; } /** Release associated resources */ dispose() { this.particleSystem.dispose(); } } const Che = "particlesPixelShader", Ohe = `#ifdef LOGARITHMICDEPTH #extension GL_EXT_frag_depth : enable #endif varying vec2 vUV;varying vec4 vColor;uniform vec4 textureMask;uniform sampler2D diffuseSampler; #include #include #include #include #include #ifdef RAMPGRADIENT varying vec4 remapRanges;uniform sampler2D rampSampler; #endif #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #define CUSTOM_FRAGMENT_MAIN_BEGIN #include vec4 textureColor=texture2D(diffuseSampler,vUV);vec4 baseColor=(textureColor*textureMask+(vec4(1.,1.,1.,1.)-textureMask))*vColor; #ifdef RAMPGRADIENT float alpha=baseColor.a;float remappedColorIndex=clamp((alpha-remapRanges.x)/remapRanges.y,0.0,1.0);vec4 rampColor=texture2D(rampSampler,vec2(1.0-remappedColorIndex,0.));baseColor.rgb*=rampColor.rgb;float finalAlpha=baseColor.a;baseColor.a=clamp((alpha*rampColor.a-remapRanges.z)/remapRanges.w,0.0,1.0); #endif #ifdef BLENDMULTIPLYMODE float sourceAlpha=vColor.a*textureColor.a;baseColor.rgb=baseColor.rgb*sourceAlpha+vec3(1.0)*(1.0-sourceAlpha); #endif #include #ifdef IMAGEPROCESSINGPOSTPROCESS baseColor.rgb=toLinearSpace(baseColor.rgb); #else #ifdef IMAGEPROCESSING baseColor.rgb=toLinearSpace(baseColor.rgb);baseColor=applyImageProcessing(baseColor); #endif #endif gl_FragColor=baseColor; #define CUSTOM_FRAGMENT_MAIN_END }`; Le.ShadersStore[Che] = Ohe; const yhe = "particlesVertexShader", khe = `attribute vec3 position;attribute vec4 color;attribute float angle;attribute vec2 size; #ifdef ANIMATESHEET attribute float cellIndex; #endif #ifndef BILLBOARD attribute vec3 direction; #endif #ifdef BILLBOARDSTRETCHED attribute vec3 direction; #endif #ifdef RAMPGRADIENT attribute vec4 remapData; #endif attribute vec2 offset;uniform mat4 view;uniform mat4 projection;uniform vec2 translationPivot; #ifdef ANIMATESHEET uniform vec3 particlesInfos; #endif varying vec2 vUV;varying vec4 vColor;varying vec3 vPositionW; #ifdef RAMPGRADIENT varying vec4 remapRanges; #endif #if defined(BILLBOARD) && !defined(BILLBOARDY) && !defined(BILLBOARDSTRETCHED) uniform mat4 invView; #endif #include #include #ifdef BILLBOARD uniform vec3 eyePosition; #endif vec3 rotate(vec3 yaxis,vec3 rotatedCorner) {vec3 xaxis=normalize(cross(vec3(0.,1.0,0.),yaxis));vec3 zaxis=normalize(cross(yaxis,xaxis));vec3 row0=vec3(xaxis.x,xaxis.y,xaxis.z);vec3 row1=vec3(yaxis.x,yaxis.y,yaxis.z);vec3 row2=vec3(zaxis.x,zaxis.y,zaxis.z);mat3 rotMatrix= mat3(row0,row1,row2);vec3 alignedCorner=rotMatrix*rotatedCorner;return position+alignedCorner;} #ifdef BILLBOARDSTRETCHED vec3 rotateAlign(vec3 toCamera,vec3 rotatedCorner) {vec3 normalizedToCamera=normalize(toCamera);vec3 normalizedCrossDirToCamera=normalize(cross(normalize(direction),normalizedToCamera));vec3 row0=vec3(normalizedCrossDirToCamera.x,normalizedCrossDirToCamera.y,normalizedCrossDirToCamera.z);vec3 row2=vec3(normalizedToCamera.x,normalizedToCamera.y,normalizedToCamera.z); #ifdef BILLBOARDSTRETCHED_LOCAL vec3 row1=direction; #else vec3 crossProduct=normalize(cross(normalizedToCamera,normalizedCrossDirToCamera));vec3 row1=vec3(crossProduct.x,crossProduct.y,crossProduct.z); #endif mat3 rotMatrix= mat3(row0,row1,row2);vec3 alignedCorner=rotMatrix*rotatedCorner;return position+alignedCorner;} #endif #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN vec2 cornerPos;cornerPos=(vec2(offset.x-0.5,offset.y -0.5)-translationPivot)*size; #ifdef BILLBOARD vec3 rotatedCorner; #ifdef BILLBOARDY rotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.z=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.y=0.;rotatedCorner.xz+=translationPivot;vec3 yaxis=position-eyePosition;yaxis.y=0.;vPositionW=rotate(normalize(yaxis),rotatedCorner);vec3 viewPos=(view*vec4(vPositionW,1.0)).xyz; #elif defined(BILLBOARDSTRETCHED) rotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.y=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.z=0.;rotatedCorner.xy+=translationPivot;vec3 toCamera=position-eyePosition;vPositionW=rotateAlign(toCamera,rotatedCorner);vec3 viewPos=(view*vec4(vPositionW,1.0)).xyz; #else rotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.y=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.z=0.;rotatedCorner.xy+=translationPivot;vec3 viewPos=(view*vec4(position,1.0)).xyz+rotatedCorner;vPositionW=(invView*vec4(viewPos,1)).xyz; #endif #ifdef RAMPGRADIENT remapRanges=remapData; #endif gl_Position=projection*vec4(viewPos,1.0); #else vec3 rotatedCorner;rotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.z=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.y=0.;rotatedCorner.xz+=translationPivot;vec3 yaxis=normalize(direction);vPositionW=rotate(yaxis,rotatedCorner);gl_Position=projection*view*vec4(vPositionW,1.0); #endif vColor=color; #ifdef ANIMATESHEET float rowOffset=floor(cellIndex*particlesInfos.z);float columnOffset=cellIndex-rowOffset/particlesInfos.z;vec2 uvScale=particlesInfos.xy;vec2 uvOffset=vec2(offset.x ,1.0-offset.y);vUV=(uvOffset+vec2(columnOffset,rowOffset))*uvScale; #else vUV=offset; #endif #if defined(CLIPPLANE) || defined(CLIPPLANE2) || defined(CLIPPLANE3) || defined(CLIPPLANE4) || defined(CLIPPLANE5) || defined(CLIPPLANE6) vec4 worldPos=vec4(vPositionW,1.0); #endif #include #include #define CUSTOM_VERTEX_MAIN_END }`; Le.ShadersStore[yhe] = khe; class ti extends N0 { /** * Sets a callback that will be triggered when the system is disposed */ set onDispose(e) { this._onDisposeObserver && this.onDisposeObservable.remove(this._onDisposeObserver), this._onDisposeObserver = this.onDisposeObservable.add(e); } /** Gets or sets a boolean indicating that ramp gradients must be used * @see https://doc.babylonjs.com/features/featuresDeepDive/particles/particle_system/particle_system_intro#ramp-gradients */ get useRampGradients() { return this._useRampGradients; } set useRampGradients(e) { this._useRampGradients !== e && (this._useRampGradients = e, this._resetEffect()); } //end of Sub-emitter /** * Gets the current list of active particles */ get particles() { return this._particles; } /** * Gets the number of particles active at the same time. * @returns The number of active particles. */ getActiveCount() { return this._particles.length; } /** * Returns the string "ParticleSystem" * @returns a string containing the class name */ getClassName() { return "ParticleSystem"; } /** * Gets a boolean indicating that the system is stopping * @returns true if the system is currently stopping */ isStopping() { return this._stopped && this.isAlive(); } /** * Gets the custom effect used to render the particles * @param blendMode Blend mode for which the effect should be retrieved * @returns The effect */ getCustomEffect(e = 0) { var t, r; return (r = (t = this._customWrappers[e]) === null || t === void 0 ? void 0 : t.effect) !== null && r !== void 0 ? r : this._customWrappers[0].effect; } _getCustomDrawWrapper(e = 0) { var t; return (t = this._customWrappers[e]) !== null && t !== void 0 ? t : this._customWrappers[0]; } /** * Sets the custom effect used to render the particles * @param effect The effect to set * @param blendMode Blend mode for which the effect should be set */ setCustomEffect(e, t = 0) { this._customWrappers[t] = new zo(this._engine), this._customWrappers[t].effect = e, this._customWrappers[t].drawContext && (this._customWrappers[t].drawContext.useInstancing = this._useInstancing); } /** * Observable that will be called just before the particles are drawn */ get onBeforeDrawParticlesObservable() { return this._onBeforeDrawParticlesObservable || (this._onBeforeDrawParticlesObservable = new Oe()), this._onBeforeDrawParticlesObservable; } /** * Gets the name of the particle vertex shader */ get vertexShaderName() { return "particles"; } /** * Gets the vertex buffers used by the particle system */ get vertexBuffers() { return this._vertexBuffers; } /** * Gets the index buffer used by the particle system (or null if no index buffer is used (if _useInstancing=true)) */ get indexBuffer() { return this._indexBuffer; } /** * Instantiates a particle system. * Particles are often small sprites used to simulate hard-to-reproduce phenomena like fire, smoke, water, or abstract visual effects like magic glitter and faery dust. * @param name The name of the particle system * @param capacity The max number of particles alive at the same time * @param sceneOrEngine The scene the particle system belongs to or the engine to use if no scene * @param customEffect a custom effect used to change the way particles are rendered by default * @param isAnimationSheetEnabled Must be true if using a spritesheet to animate the particles texture * @param epsilon Offset used to render the particles */ constructor(e, t, r, n = null, i = !1, s = 0.01) { super(e), this._emitterInverseWorldMatrix = he.Identity(), this._inheritedVelocityOffset = new S(), this.onDisposeObservable = new Oe(), this.onStoppedObservable = new Oe(), this._particles = new Array(), this._stockParticles = new Array(), this._newPartsExcess = 0, this._vertexBuffers = {}, this._scaledColorStep = new xt(0, 0, 0, 0), this._colorDiff = new xt(0, 0, 0, 0), this._scaledDirection = S.Zero(), this._scaledGravity = S.Zero(), this._currentRenderId = -1, this._useInstancing = !1, this._started = !1, this._stopped = !1, this._actualFrame = 0, this._currentEmitRate1 = 0, this._currentEmitRate2 = 0, this._currentStartSize1 = 0, this._currentStartSize2 = 0, this.updateInAnimate = !0, this._rawTextureWidth = 256, this._useRampGradients = !1, this._disposeEmitterOnDispose = !1, this.isLocal = !1, this.isGPU = !1, this._onBeforeDrawParticlesObservable = null, this.recycleParticle = (f) => { const o = this._particles.pop(); o !== f && o.copyTo(f), this._stockParticles.push(o); }, this._createParticle = () => { let f; if (this._stockParticles.length !== 0 ? (f = this._stockParticles.pop(), f._reset()) : f = new FS(this), this._subEmitters && this._subEmitters.length > 0) { const o = this._subEmitters[Math.floor(Math.random() * this._subEmitters.length)]; f._attachedSubEmitters = [], o.forEach((d) => { if (d.type === NS.ATTACHED) { const v = d.clone(); f._attachedSubEmitters.push(v), v.particleSystem.start(); } }); } return f; }, this._emitFromParticle = (f) => { if (!this._subEmitters || this._subEmitters.length === 0) return; const o = Math.floor(Math.random() * this._subEmitters.length); this._subEmitters[o].forEach((d) => { if (d.type === NS.END) { const v = d.clone(); f._inheritParticleInfoToSubEmitter(v), v.particleSystem._rootParticleSystem = this, this.activeSubSystems.push(v.particleSystem), v.particleSystem.start(); } }); }, this._capacity = t, this._epsilon = s, this._isAnimationSheetEnabled = i, !r || r.getClassName() === "Scene" ? (this._scene = r || gr.LastCreatedScene, this._engine = this._scene.getEngine(), this.uniqueId = this._scene.getUniqueId(), this._scene.particleSystems.push(this)) : (this._engine = r, this.defaultProjectionMatrix = he.PerspectiveFovLH(0.8, 1, 0.1, 100, this._engine.isNDCHalfZRange)), this._engine.getCaps().vertexArrayObject && (this._vertexArrayObject = null), this._attachImageProcessingConfiguration(null), this._customWrappers = { 0: new zo(this._engine) }, this._customWrappers[0].effect = n, this._drawWrappers = [], this._useInstancing = this._engine.getCaps().instancedArrays, this._createIndexBuffer(), this._createVertexBuffers(), this.particleEmitterType = new AD(); let a = null; this.updateFunction = (f) => { var o; let d = null; this.noiseTexture && (d = this.noiseTexture.getSize(), (o = this.noiseTexture.getContent()) === null || o === void 0 || o.then((u) => { a = u; })); const v = f === this._particles; for (let u = 0; u < f.length; u++) { const l = f[u]; let P = this._scaledUpdateSpeed; const p = l.age; if (l.age += P, l.age > l.lifeTime) { const T = l.age - p; P = (l.lifeTime - p) * P / T, l.age = l.lifeTime; } const c = l.age / l.lifeTime; this._colorGradients && this._colorGradients.length > 0 ? kl.GetCurrentGradient(c, this._colorGradients, (T, q, b) => { T !== l._currentColorGradient && (l._currentColor1.copyFrom(l._currentColor2), q.getColorToRef(l._currentColor2), l._currentColorGradient = T), xt.LerpToRef(l._currentColor1, l._currentColor2, b, l.color); }) : (l.colorStep.scaleToRef(P, this._scaledColorStep), l.color.addInPlace(this._scaledColorStep), l.color.a < 0 && (l.color.a = 0)), this._angularSpeedGradients && this._angularSpeedGradients.length > 0 && kl.GetCurrentGradient(c, this._angularSpeedGradients, (T, q, b) => { T !== l._currentAngularSpeedGradient && (l._currentAngularSpeed1 = l._currentAngularSpeed2, l._currentAngularSpeed2 = q.getFactor(), l._currentAngularSpeedGradient = T), l.angularSpeed = Xt.Lerp(l._currentAngularSpeed1, l._currentAngularSpeed2, b); }), l.angle += l.angularSpeed * P; let H = P; if (this._velocityGradients && this._velocityGradients.length > 0 && kl.GetCurrentGradient(c, this._velocityGradients, (T, q, b) => { T !== l._currentVelocityGradient && (l._currentVelocity1 = l._currentVelocity2, l._currentVelocity2 = q.getFactor(), l._currentVelocityGradient = T), H *= Xt.Lerp(l._currentVelocity1, l._currentVelocity2, b); }), l.direction.scaleToRef(H, this._scaledDirection), this._limitVelocityGradients && this._limitVelocityGradients.length > 0 && kl.GetCurrentGradient(c, this._limitVelocityGradients, (T, q, b) => { T !== l._currentLimitVelocityGradient && (l._currentLimitVelocity1 = l._currentLimitVelocity2, l._currentLimitVelocity2 = q.getFactor(), l._currentLimitVelocityGradient = T); const j = Xt.Lerp(l._currentLimitVelocity1, l._currentLimitVelocity2, b); l.direction.length() > j && l.direction.scaleInPlace(this.limitVelocityDamping); }), this._dragGradients && this._dragGradients.length > 0 && kl.GetCurrentGradient(c, this._dragGradients, (T, q, b) => { T !== l._currentDragGradient && (l._currentDrag1 = l._currentDrag2, l._currentDrag2 = q.getFactor(), l._currentDragGradient = T); const j = Xt.Lerp(l._currentDrag1, l._currentDrag2, b); this._scaledDirection.scaleInPlace(1 - j); }), this.isLocal && l._localPosition ? (l._localPosition.addInPlace(this._scaledDirection), S.TransformCoordinatesToRef(l._localPosition, this._emitterWorldMatrix, l.position)) : l.position.addInPlace(this._scaledDirection), a && d && l._randomNoiseCoordinates1) { const T = this._fetchR(l._randomNoiseCoordinates1.x, l._randomNoiseCoordinates1.y, d.width, d.height, a), q = this._fetchR(l._randomNoiseCoordinates1.z, l._randomNoiseCoordinates2.x, d.width, d.height, a), b = this._fetchR(l._randomNoiseCoordinates2.y, l._randomNoiseCoordinates2.z, d.width, d.height, a), j = ue.Vector3[0], w = ue.Vector3[1]; j.copyFromFloats((2 * T - 1) * this.noiseStrength.x, (2 * q - 1) * this.noiseStrength.y, (2 * b - 1) * this.noiseStrength.z), j.scaleToRef(P, w), l.direction.addInPlace(w); } if (this.gravity.scaleToRef(P, this._scaledGravity), l.direction.addInPlace(this._scaledGravity), this._sizeGradients && this._sizeGradients.length > 0 && kl.GetCurrentGradient(c, this._sizeGradients, (T, q, b) => { T !== l._currentSizeGradient && (l._currentSize1 = l._currentSize2, l._currentSize2 = q.getFactor(), l._currentSizeGradient = T), l.size = Xt.Lerp(l._currentSize1, l._currentSize2, b); }), this._useRampGradients && (this._colorRemapGradients && this._colorRemapGradients.length > 0 && kl.GetCurrentGradient(c, this._colorRemapGradients, (T, q, b) => { const j = Xt.Lerp(T.factor1, q.factor1, b), w = Xt.Lerp(T.factor2, q.factor2, b); l.remapData.x = j, l.remapData.y = w - j; }), this._alphaRemapGradients && this._alphaRemapGradients.length > 0 && kl.GetCurrentGradient(c, this._alphaRemapGradients, (T, q, b) => { const j = Xt.Lerp(T.factor1, q.factor1, b), w = Xt.Lerp(T.factor2, q.factor2, b); l.remapData.z = j, l.remapData.w = w - j; })), this._isAnimationSheetEnabled && l.updateCellIndex(), l._inheritParticleInfoToSubEmitters(), l.age >= l.lifeTime) { this._emitFromParticle(l), l._attachedSubEmitters && (l._attachedSubEmitters.forEach((T) => { T.particleSystem.disposeOnStop = !0, T.particleSystem.stop(); }), l._attachedSubEmitters = null), this.recycleParticle(l), v && u--; continue; } } }; } _addFactorGradient(e, t, r, n) { const i = new kY(t, r, n); e.push(i), e.sort((s, a) => s.gradient < a.gradient ? -1 : s.gradient > a.gradient ? 1 : 0); } _removeFactorGradient(e, t) { if (!e) return; let r = 0; for (const n of e) { if (n.gradient === t) { e.splice(r, 1); break; } r++; } } /** * Adds a new life time gradient * @param gradient defines the gradient to use (between 0 and 1) * @param factor defines the life time factor to affect to the specified gradient * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from * @returns the current particle system */ addLifeTimeGradient(e, t, r) { return this._lifeTimeGradients || (this._lifeTimeGradients = []), this._addFactorGradient(this._lifeTimeGradients, e, t, r), this; } /** * Remove a specific life time gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeLifeTimeGradient(e) { return this._removeFactorGradient(this._lifeTimeGradients, e), this; } /** * Adds a new size gradient * @param gradient defines the gradient to use (between 0 and 1) * @param factor defines the size factor to affect to the specified gradient * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from * @returns the current particle system */ addSizeGradient(e, t, r) { return this._sizeGradients || (this._sizeGradients = []), this._addFactorGradient(this._sizeGradients, e, t, r), this; } /** * Remove a specific size gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeSizeGradient(e) { return this._removeFactorGradient(this._sizeGradients, e), this; } /** * Adds a new color remap gradient * @param gradient defines the gradient to use (between 0 and 1) * @param min defines the color remap minimal range * @param max defines the color remap maximal range * @returns the current particle system */ addColorRemapGradient(e, t, r) { return this._colorRemapGradients || (this._colorRemapGradients = []), this._addFactorGradient(this._colorRemapGradients, e, t, r), this; } /** * Remove a specific color remap gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeColorRemapGradient(e) { return this._removeFactorGradient(this._colorRemapGradients, e), this; } /** * Adds a new alpha remap gradient * @param gradient defines the gradient to use (between 0 and 1) * @param min defines the alpha remap minimal range * @param max defines the alpha remap maximal range * @returns the current particle system */ addAlphaRemapGradient(e, t, r) { return this._alphaRemapGradients || (this._alphaRemapGradients = []), this._addFactorGradient(this._alphaRemapGradients, e, t, r), this; } /** * Remove a specific alpha remap gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeAlphaRemapGradient(e) { return this._removeFactorGradient(this._alphaRemapGradients, e), this; } /** * Adds a new angular speed gradient * @param gradient defines the gradient to use (between 0 and 1) * @param factor defines the angular speed to affect to the specified gradient * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from * @returns the current particle system */ addAngularSpeedGradient(e, t, r) { return this._angularSpeedGradients || (this._angularSpeedGradients = []), this._addFactorGradient(this._angularSpeedGradients, e, t, r), this; } /** * Remove a specific angular speed gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeAngularSpeedGradient(e) { return this._removeFactorGradient(this._angularSpeedGradients, e), this; } /** * Adds a new velocity gradient * @param gradient defines the gradient to use (between 0 and 1) * @param factor defines the velocity to affect to the specified gradient * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from * @returns the current particle system */ addVelocityGradient(e, t, r) { return this._velocityGradients || (this._velocityGradients = []), this._addFactorGradient(this._velocityGradients, e, t, r), this; } /** * Remove a specific velocity gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeVelocityGradient(e) { return this._removeFactorGradient(this._velocityGradients, e), this; } /** * Adds a new limit velocity gradient * @param gradient defines the gradient to use (between 0 and 1) * @param factor defines the limit velocity value to affect to the specified gradient * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from * @returns the current particle system */ addLimitVelocityGradient(e, t, r) { return this._limitVelocityGradients || (this._limitVelocityGradients = []), this._addFactorGradient(this._limitVelocityGradients, e, t, r), this; } /** * Remove a specific limit velocity gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeLimitVelocityGradient(e) { return this._removeFactorGradient(this._limitVelocityGradients, e), this; } /** * Adds a new drag gradient * @param gradient defines the gradient to use (between 0 and 1) * @param factor defines the drag value to affect to the specified gradient * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from * @returns the current particle system */ addDragGradient(e, t, r) { return this._dragGradients || (this._dragGradients = []), this._addFactorGradient(this._dragGradients, e, t, r), this; } /** * Remove a specific drag gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeDragGradient(e) { return this._removeFactorGradient(this._dragGradients, e), this; } /** * Adds a new emit rate gradient (please note that this will only work if you set the targetStopDuration property) * @param gradient defines the gradient to use (between 0 and 1) * @param factor defines the emit rate value to affect to the specified gradient * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from * @returns the current particle system */ addEmitRateGradient(e, t, r) { return this._emitRateGradients || (this._emitRateGradients = []), this._addFactorGradient(this._emitRateGradients, e, t, r), this; } /** * Remove a specific emit rate gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeEmitRateGradient(e) { return this._removeFactorGradient(this._emitRateGradients, e), this; } /** * Adds a new start size gradient (please note that this will only work if you set the targetStopDuration property) * @param gradient defines the gradient to use (between 0 and 1) * @param factor defines the start size value to affect to the specified gradient * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from * @returns the current particle system */ addStartSizeGradient(e, t, r) { return this._startSizeGradients || (this._startSizeGradients = []), this._addFactorGradient(this._startSizeGradients, e, t, r), this; } /** * Remove a specific start size gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeStartSizeGradient(e) { return this._removeFactorGradient(this._startSizeGradients, e), this; } _createRampGradientTexture() { if (!this._rampGradients || !this._rampGradients.length || this._rampGradientsTexture || !this._scene) return; const e = new Uint8Array(this._rawTextureWidth * 4), t = Hs.Color3[0]; for (let r = 0; r < this._rawTextureWidth; r++) { const n = r / this._rawTextureWidth; kl.GetCurrentGradient(n, this._rampGradients, (i, s, a) => { Ne.LerpToRef(i.color, s.color, a, t), e[r * 4] = t.r * 255, e[r * 4 + 1] = t.g * 255, e[r * 4 + 2] = t.b * 255, e[r * 4 + 3] = 255; }); } this._rampGradientsTexture = Bo.CreateRGBATexture(e, this._rawTextureWidth, 1, this._scene, !1, !1, 1); } /** * Gets the current list of ramp gradients. * You must use addRampGradient and removeRampGradient to update this list * @returns the list of ramp gradients */ getRampGradients() { return this._rampGradients; } /** Force the system to rebuild all gradients that need to be resync */ forceRefreshGradients() { this._syncRampGradientTexture(); } _syncRampGradientTexture() { this._rampGradients && (this._rampGradients.sort((e, t) => e.gradient < t.gradient ? -1 : e.gradient > t.gradient ? 1 : 0), this._rampGradientsTexture && (this._rampGradientsTexture.dispose(), this._rampGradientsTexture = null), this._createRampGradientTexture()); } /** * Adds a new ramp gradient used to remap particle colors * @param gradient defines the gradient to use (between 0 and 1) * @param color defines the color to affect to the specified gradient * @returns the current particle system */ addRampGradient(e, t) { this._rampGradients || (this._rampGradients = []); const r = new Jne(e, t); return this._rampGradients.push(r), this._syncRampGradientTexture(), this; } /** * Remove a specific ramp gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeRampGradient(e) { return this._removeGradientAndTexture(e, this._rampGradients, this._rampGradientsTexture), this._rampGradientsTexture = null, this._rampGradients && this._rampGradients.length > 0 && this._createRampGradientTexture(), this; } /** * Adds a new color gradient * @param gradient defines the gradient to use (between 0 and 1) * @param color1 defines the color to affect to the specified gradient * @param color2 defines an additional color used to define a range ([color, color2]) with main color to pick the final color from * @returns this particle system */ addColorGradient(e, t, r) { this._colorGradients || (this._colorGradients = []); const n = new yY(e, t, r); return this._colorGradients.push(n), this._colorGradients.sort((i, s) => i.gradient < s.gradient ? -1 : i.gradient > s.gradient ? 1 : 0), this; } /** * Remove a specific color gradient * @param gradient defines the gradient to remove * @returns this particle system */ removeColorGradient(e) { if (!this._colorGradients) return this; let t = 0; for (const r of this._colorGradients) { if (r.gradient === e) { this._colorGradients.splice(t, 1); break; } t++; } return this; } /** * Resets the draw wrappers cache */ resetDrawCache() { for (const e of this._drawWrappers) if (e) for (const t of e) t == null || t.dispose(); this._drawWrappers = []; } _fetchR(e, t, r, n, i) { e = Math.abs(e) * 0.5 + 0.5, t = Math.abs(t) * 0.5 + 0.5; const s = e * r % r | 0, a = t * n % n | 0, f = (s + a * r) * 4; return i[f] / 255; } _reset() { this._resetEffect(); } _resetEffect() { this._vertexBuffer && (this._vertexBuffer.dispose(), this._vertexBuffer = null), this._spriteBuffer && (this._spriteBuffer.dispose(), this._spriteBuffer = null), this._vertexArrayObject && (this._engine.releaseVertexArrayObject(this._vertexArrayObject), this._vertexArrayObject = null), this._createVertexBuffers(); } _createVertexBuffers() { this._vertexBufferSize = this._useInstancing ? 10 : 12, this._isAnimationSheetEnabled && (this._vertexBufferSize += 1), (!this._isBillboardBased || this.billboardMode === ti.BILLBOARDMODE_STRETCHED || this.billboardMode === ti.BILLBOARDMODE_STRETCHED_LOCAL) && (this._vertexBufferSize += 3), this._useRampGradients && (this._vertexBufferSize += 4); const e = this._engine, t = this._vertexBufferSize * (this._useInstancing ? 1 : 4); this._vertexData = new Float32Array(this._capacity * t), this._vertexBuffer = new P9(e, this._vertexData, !0, t); let r = 0; const n = this._vertexBuffer.createVertexBuffer(J.PositionKind, r, 3, this._vertexBufferSize, this._useInstancing); this._vertexBuffers[J.PositionKind] = n, r += 3; const i = this._vertexBuffer.createVertexBuffer(J.ColorKind, r, 4, this._vertexBufferSize, this._useInstancing); this._vertexBuffers[J.ColorKind] = i, r += 4; const s = this._vertexBuffer.createVertexBuffer("angle", r, 1, this._vertexBufferSize, this._useInstancing); this._vertexBuffers.angle = s, r += 1; const a = this._vertexBuffer.createVertexBuffer("size", r, 2, this._vertexBufferSize, this._useInstancing); if (this._vertexBuffers.size = a, r += 2, this._isAnimationSheetEnabled) { const o = this._vertexBuffer.createVertexBuffer("cellIndex", r, 1, this._vertexBufferSize, this._useInstancing); this._vertexBuffers.cellIndex = o, r += 1; } if (!this._isBillboardBased || this.billboardMode === ti.BILLBOARDMODE_STRETCHED || this.billboardMode === ti.BILLBOARDMODE_STRETCHED_LOCAL) { const o = this._vertexBuffer.createVertexBuffer("direction", r, 3, this._vertexBufferSize, this._useInstancing); this._vertexBuffers.direction = o, r += 3; } if (this._useRampGradients) { const o = this._vertexBuffer.createVertexBuffer("remapData", r, 4, this._vertexBufferSize, this._useInstancing); this._vertexBuffers.remapData = o, r += 4; } let f; if (this._useInstancing) { const o = new Float32Array([0, 0, 1, 0, 0, 1, 1, 1]); this._spriteBuffer = new P9(e, o, !1, 2), f = this._spriteBuffer.createVertexBuffer("offset", 0, 2); } else f = this._vertexBuffer.createVertexBuffer("offset", r, 2, this._vertexBufferSize, this._useInstancing), r += 2; this._vertexBuffers.offset = f, this.resetDrawCache(); } _createIndexBuffer() { if (this._useInstancing) { this._linesIndexBufferUseInstancing = this._engine.createIndexBuffer(new Uint32Array([0, 1, 1, 3, 3, 2, 2, 0, 0, 3])); return; } const e = [], t = []; let r = 0; for (let n = 0; n < this._capacity; n++) e.push(r), e.push(r + 1), e.push(r + 2), e.push(r), e.push(r + 2), e.push(r + 3), t.push(r, r + 1, r + 1, r + 2, r + 2, r + 3, r + 3, r, r, r + 3), r += 4; this._indexBuffer = this._engine.createIndexBuffer(e), this._linesIndexBuffer = this._engine.createIndexBuffer(t); } /** * Gets the maximum number of particles active at the same time. * @returns The max number of active particles. */ getCapacity() { return this._capacity; } /** * Gets whether there are still active particles in the system. * @returns True if it is alive, otherwise false. */ isAlive() { return this._alive; } /** * Gets if the system has been started. (Note: this will still be true after stop is called) * @returns True if it has been started, otherwise false. */ isStarted() { return this._started; } _prepareSubEmitterInternalArray() { this._subEmitters = new Array(), this.subEmitters && this.subEmitters.forEach((e) => { e instanceof ti ? this._subEmitters.push([new vq(e)]) : e instanceof vq ? this._subEmitters.push([e]) : e instanceof Array && this._subEmitters.push(e); }); } /** * Starts the particle system and begins to emit * @param delay defines the delay in milliseconds before starting the system (this.startDelay by default) */ start(e = this.startDelay) { var t; if (!this.targetStopDuration && this._hasTargetStopDurationDependantGradient()) throw "Particle system started with a targetStopDuration dependant gradient (eg. startSizeGradients) but no targetStopDuration set"; if (e) { setTimeout(() => { this.start(0); }, e); return; } if (this._prepareSubEmitterInternalArray(), this._started = !0, this._stopped = !1, this._actualFrame = 0, this._subEmitters && this._subEmitters.length != 0 && (this.activeSubSystems = []), this._emitRateGradients && (this._emitRateGradients.length > 0 && (this._currentEmitRateGradient = this._emitRateGradients[0], this._currentEmitRate1 = this._currentEmitRateGradient.getFactor(), this._currentEmitRate2 = this._currentEmitRate1), this._emitRateGradients.length > 1 && (this._currentEmitRate2 = this._emitRateGradients[1].getFactor())), this._startSizeGradients && (this._startSizeGradients.length > 0 && (this._currentStartSizeGradient = this._startSizeGradients[0], this._currentStartSize1 = this._currentStartSizeGradient.getFactor(), this._currentStartSize2 = this._currentStartSize1), this._startSizeGradients.length > 1 && (this._currentStartSize2 = this._startSizeGradients[1].getFactor())), this.preWarmCycles) { ((t = this.emitter) === null || t === void 0 ? void 0 : t.getClassName().indexOf("Mesh")) !== -1 && this.emitter.computeWorldMatrix(!0); const r = this.noiseTexture; if (r && r.onGeneratedObservable) r.onGeneratedObservable.addOnce(() => { setTimeout(() => { for (let n = 0; n < this.preWarmCycles; n++) this.animate(!0), r.render(); }); }); else for (let n = 0; n < this.preWarmCycles; n++) this.animate(!0); } this.beginAnimationOnStart && this.animations && this.animations.length > 0 && this._scene && this._scene.beginAnimation(this, this.beginAnimationFrom, this.beginAnimationTo, this.beginAnimationLoop); } /** * Stops the particle system. * @param stopSubEmitters if true it will stop the current system and all created sub-Systems if false it will stop the current root system only, this param is used by the root particle system only. the default value is true. */ stop(e = !0) { this._stopped || (this.onStoppedObservable.notifyObservers(this), this._stopped = !0, e && this._stopSubEmitters()); } // animation sheet /** * Remove all active particles */ reset() { this._stockParticles.length = 0, this._particles.length = 0; } /** * @internal (for internal use only) */ _appendParticleVertex(e, t, r, n) { let i = e * this._vertexBufferSize; if (this._vertexData[i++] = t.position.x + this.worldOffset.x, this._vertexData[i++] = t.position.y + this.worldOffset.y, this._vertexData[i++] = t.position.z + this.worldOffset.z, this._vertexData[i++] = t.color.r, this._vertexData[i++] = t.color.g, this._vertexData[i++] = t.color.b, this._vertexData[i++] = t.color.a, this._vertexData[i++] = t.angle, this._vertexData[i++] = t.scale.x * t.size, this._vertexData[i++] = t.scale.y * t.size, this._isAnimationSheetEnabled && (this._vertexData[i++] = t.cellIndex), this._isBillboardBased) (this.billboardMode === ti.BILLBOARDMODE_STRETCHED || this.billboardMode === ti.BILLBOARDMODE_STRETCHED_LOCAL) && (this._vertexData[i++] = t.direction.x, this._vertexData[i++] = t.direction.y, this._vertexData[i++] = t.direction.z); else if (t._initialDirection) { let s = t._initialDirection; this.isLocal && (S.TransformNormalToRef(s, this._emitterWorldMatrix, ue.Vector3[0]), s = ue.Vector3[0]), s.x === 0 && s.z === 0 && (s.x = 1e-3), this._vertexData[i++] = s.x, this._vertexData[i++] = s.y, this._vertexData[i++] = s.z; } else { let s = t.direction; this.isLocal && (S.TransformNormalToRef(s, this._emitterWorldMatrix, ue.Vector3[0]), s = ue.Vector3[0]), s.x === 0 && s.z === 0 && (s.x = 1e-3), this._vertexData[i++] = s.x, this._vertexData[i++] = s.y, this._vertexData[i++] = s.z; } this._useRampGradients && t.remapData && (this._vertexData[i++] = t.remapData.x, this._vertexData[i++] = t.remapData.y, this._vertexData[i++] = t.remapData.z, this._vertexData[i++] = t.remapData.w), this._useInstancing || (this._isAnimationSheetEnabled && (r === 0 ? r = this._epsilon : r === 1 && (r = 1 - this._epsilon), n === 0 ? n = this._epsilon : n === 1 && (n = 1 - this._epsilon)), this._vertexData[i++] = r, this._vertexData[i++] = n); } _stopSubEmitters() { this.activeSubSystems && (this.activeSubSystems.forEach((e) => { e.stop(!0); }), this.activeSubSystems = []); } _removeFromRoot() { if (!this._rootParticleSystem) return; const e = this._rootParticleSystem.activeSubSystems.indexOf(this); e !== -1 && this._rootParticleSystem.activeSubSystems.splice(e, 1), this._rootParticleSystem = null; } // End of sub system methods _update(e) { if (this._alive = this._particles.length > 0, this.emitter.position) { const r = this.emitter; this._emitterWorldMatrix = r.getWorldMatrix(); } else { const r = this.emitter; this._emitterWorldMatrix = he.Translation(r.x, r.y, r.z); } this._emitterWorldMatrix.invertToRef(this._emitterInverseWorldMatrix), this.updateFunction(this._particles); let t; for (let r = 0; r < e && this._particles.length !== this._capacity; r++) { if (t = this._createParticle(), this._particles.push(t), this.targetStopDuration && this._lifeTimeGradients && this._lifeTimeGradients.length > 0) { const i = Xt.Clamp(this._actualFrame / this.targetStopDuration); kl.GetCurrentGradient(i, this._lifeTimeGradients, (s, a) => { const f = s, o = a, d = f.getFactor(), v = o.getFactor(), u = (i - f.gradient) / (o.gradient - f.gradient); t.lifeTime = Xt.Lerp(d, v, u); }); } else t.lifeTime = Xt.RandomRange(this.minLifeTime, this.maxLifeTime); const n = Xt.RandomRange(this.minEmitPower, this.maxEmitPower); if (this.startPositionFunction ? this.startPositionFunction(this._emitterWorldMatrix, t.position, t, this.isLocal) : this.particleEmitterType.startPositionFunction(this._emitterWorldMatrix, t.position, t, this.isLocal), this.isLocal && (t._localPosition ? t._localPosition.copyFrom(t.position) : t._localPosition = t.position.clone(), S.TransformCoordinatesToRef(t._localPosition, this._emitterWorldMatrix, t.position)), this.startDirectionFunction ? this.startDirectionFunction(this._emitterWorldMatrix, t.direction, t, this.isLocal) : this.particleEmitterType.startDirectionFunction(this._emitterWorldMatrix, t.direction, t, this.isLocal, this._emitterInverseWorldMatrix), n === 0 ? t._initialDirection ? t._initialDirection.copyFrom(t.direction) : t._initialDirection = t.direction.clone() : t._initialDirection = null, t.direction.scaleInPlace(n), !this._sizeGradients || this._sizeGradients.length === 0 ? t.size = Xt.RandomRange(this.minSize, this.maxSize) : (t._currentSizeGradient = this._sizeGradients[0], t._currentSize1 = t._currentSizeGradient.getFactor(), t.size = t._currentSize1, this._sizeGradients.length > 1 ? t._currentSize2 = this._sizeGradients[1].getFactor() : t._currentSize2 = t._currentSize1), t.scale.copyFromFloats(Xt.RandomRange(this.minScaleX, this.maxScaleX), Xt.RandomRange(this.minScaleY, this.maxScaleY)), this._startSizeGradients && this._startSizeGradients[0] && this.targetStopDuration) { const i = this._actualFrame / this.targetStopDuration; kl.GetCurrentGradient(i, this._startSizeGradients, (s, a, f) => { s !== this._currentStartSizeGradient && (this._currentStartSize1 = this._currentStartSize2, this._currentStartSize2 = a.getFactor(), this._currentStartSizeGradient = s); const o = Xt.Lerp(this._currentStartSize1, this._currentStartSize2, f); t.scale.scaleInPlace(o); }); } if (!this._angularSpeedGradients || this._angularSpeedGradients.length === 0 ? t.angularSpeed = Xt.RandomRange(this.minAngularSpeed, this.maxAngularSpeed) : (t._currentAngularSpeedGradient = this._angularSpeedGradients[0], t.angularSpeed = t._currentAngularSpeedGradient.getFactor(), t._currentAngularSpeed1 = t.angularSpeed, this._angularSpeedGradients.length > 1 ? t._currentAngularSpeed2 = this._angularSpeedGradients[1].getFactor() : t._currentAngularSpeed2 = t._currentAngularSpeed1), t.angle = Xt.RandomRange(this.minInitialRotation, this.maxInitialRotation), this._velocityGradients && this._velocityGradients.length > 0 && (t._currentVelocityGradient = this._velocityGradients[0], t._currentVelocity1 = t._currentVelocityGradient.getFactor(), this._velocityGradients.length > 1 ? t._currentVelocity2 = this._velocityGradients[1].getFactor() : t._currentVelocity2 = t._currentVelocity1), this._limitVelocityGradients && this._limitVelocityGradients.length > 0 && (t._currentLimitVelocityGradient = this._limitVelocityGradients[0], t._currentLimitVelocity1 = t._currentLimitVelocityGradient.getFactor(), this._limitVelocityGradients.length > 1 ? t._currentLimitVelocity2 = this._limitVelocityGradients[1].getFactor() : t._currentLimitVelocity2 = t._currentLimitVelocity1), this._dragGradients && this._dragGradients.length > 0 && (t._currentDragGradient = this._dragGradients[0], t._currentDrag1 = t._currentDragGradient.getFactor(), this._dragGradients.length > 1 ? t._currentDrag2 = this._dragGradients[1].getFactor() : t._currentDrag2 = t._currentDrag1), !this._colorGradients || this._colorGradients.length === 0) { const i = Xt.RandomRange(0, 1); xt.LerpToRef(this.color1, this.color2, i, t.color), this.colorDead.subtractToRef(t.color, this._colorDiff), this._colorDiff.scaleToRef(1 / t.lifeTime, t.colorStep); } else t._currentColorGradient = this._colorGradients[0], t._currentColorGradient.getColorToRef(t.color), t._currentColor1.copyFrom(t.color), this._colorGradients.length > 1 ? this._colorGradients[1].getColorToRef(t._currentColor2) : t._currentColor2.copyFrom(t.color); this._isAnimationSheetEnabled && (t._initialStartSpriteCellID = this.startSpriteCellID, t._initialEndSpriteCellID = this.endSpriteCellID, t._initialSpriteCellLoop = this.spriteCellLoop), t.direction.addInPlace(this._inheritedVelocityOffset), this._useRampGradients && (t.remapData = new Ir(0, 1, 0, 1)), this.noiseTexture && (t._randomNoiseCoordinates1 ? (t._randomNoiseCoordinates1.copyFromFloats(Math.random(), Math.random(), Math.random()), t._randomNoiseCoordinates2.copyFromFloats(Math.random(), Math.random(), Math.random())) : (t._randomNoiseCoordinates1 = new S(Math.random(), Math.random(), Math.random()), t._randomNoiseCoordinates2 = new S(Math.random(), Math.random(), Math.random()))), t._inheritParticleInfoToSubEmitters(); } } /** * @internal */ static _GetAttributeNamesOrOptions(e = !1, t = !1, r = !1) { const n = [J.PositionKind, J.ColorKind, "angle", "offset", "size"]; return e && n.push("cellIndex"), t || n.push("direction"), r && n.push("remapData"), n; } /** * @internal */ static _GetEffectCreationOptions(e = !1, t = !1) { const r = ["invView", "view", "projection", "textureMask", "translationPivot", "eyePosition"]; return Mf(r), e && r.push("particlesInfos"), t && r.push("logarithmicDepthConstant"), r; } /** * Fill the defines array according to the current settings of the particle system * @param defines Array to be updated * @param blendMode blend mode to take into account when updating the array */ fillDefines(e, t) { if (this._scene && xq(this, this._scene, e), this._isAnimationSheetEnabled && e.push("#define ANIMATESHEET"), this.useLogarithmicDepth && e.push("#define LOGARITHMICDEPTH"), t === ti.BLENDMODE_MULTIPLY && e.push("#define BLENDMULTIPLYMODE"), this._useRampGradients && e.push("#define RAMPGRADIENT"), this._isBillboardBased) switch (e.push("#define BILLBOARD"), this.billboardMode) { case ti.BILLBOARDMODE_Y: e.push("#define BILLBOARDY"); break; case ti.BILLBOARDMODE_STRETCHED: case ti.BILLBOARDMODE_STRETCHED_LOCAL: e.push("#define BILLBOARDSTRETCHED"), this.billboardMode === ti.BILLBOARDMODE_STRETCHED_LOCAL && e.push("#define BILLBOARDSTRETCHED_LOCAL"); break; case ti.BILLBOARDMODE_ALL: e.push("#define BILLBOARDMODE_ALL"); break; } this._imageProcessingConfiguration && (this._imageProcessingConfiguration.prepareDefines(this._imageProcessingConfigurationDefines), e.push(this._imageProcessingConfigurationDefines.toString())); } /** * Fill the uniforms, attributes and samplers arrays according to the current settings of the particle system * @param uniforms Uniforms array to fill * @param attributes Attributes array to fill * @param samplers Samplers array to fill */ fillUniformsAttributesAndSamplerNames(e, t, r) { t.push(...ti._GetAttributeNamesOrOptions(this._isAnimationSheetEnabled, this._isBillboardBased && this.billboardMode !== ti.BILLBOARDMODE_STRETCHED && this.billboardMode !== ti.BILLBOARDMODE_STRETCHED_LOCAL, this._useRampGradients)), e.push(...ti._GetEffectCreationOptions(this._isAnimationSheetEnabled, this.useLogarithmicDepth)), r.push("diffuseSampler", "rampSampler"), this._imageProcessingConfiguration && (Ui.PrepareUniforms(e, this._imageProcessingConfigurationDefines), Ui.PrepareSamplers(r, this._imageProcessingConfigurationDefines)); } /** * @internal */ _getWrapper(e) { const t = this._getCustomDrawWrapper(e); if (t != null && t.effect) return t; const r = []; this.fillDefines(r, e); const n = this._engine._features.supportRenderPasses ? this._engine.currentRenderPassId : 0; let i = this._drawWrappers[n]; i || (i = this._drawWrappers[n] = []); let s = i[e]; s || (s = new zo(this._engine), s.drawContext && (s.drawContext.useInstancing = this._useInstancing), i[e] = s); const a = r.join(` `); if (s.defines !== a) { const f = [], o = [], d = []; this.fillUniformsAttributesAndSamplerNames(o, f, d), s.setEffect(this._engine.createEffect("particles", f, o, d, a), a); } return s; } /** * Animates the particle system for the current frame by emitting new particles and or animating the living ones. * @param preWarmOnly will prevent the system from updating the vertex buffer (default is false) */ animate(e = !1) { var t; if (!this._started) return; if (!e && this._scene) { if (!this.isReady() || this._currentRenderId === this._scene.getFrameId()) return; this._currentRenderId = this._scene.getFrameId(); } this._scaledUpdateSpeed = this.updateSpeed * (e ? this.preWarmStepOffset : ((t = this._scene) === null || t === void 0 ? void 0 : t.getAnimationRatio()) || 1); let r; if (this.manualEmitCount > -1) r = this.manualEmitCount, this._newPartsExcess = 0, this.manualEmitCount = 0; else { let n = this.emitRate; if (this._emitRateGradients && this._emitRateGradients.length > 0 && this.targetStopDuration) { const i = this._actualFrame / this.targetStopDuration; kl.GetCurrentGradient(i, this._emitRateGradients, (s, a, f) => { s !== this._currentEmitRateGradient && (this._currentEmitRate1 = this._currentEmitRate2, this._currentEmitRate2 = a.getFactor(), this._currentEmitRateGradient = s), n = Xt.Lerp(this._currentEmitRate1, this._currentEmitRate2, f); }); } r = n * this._scaledUpdateSpeed >> 0, this._newPartsExcess += n * this._scaledUpdateSpeed - r; } if (this._newPartsExcess > 1 && (r += this._newPartsExcess >> 0, this._newPartsExcess -= this._newPartsExcess >> 0), this._alive = !1, this._stopped ? r = 0 : (this._actualFrame += this._scaledUpdateSpeed, this.targetStopDuration && this._actualFrame >= this.targetStopDuration && this.stop()), this._update(r), this._stopped && (this._alive || (this._started = !1, this.onAnimationEnd && this.onAnimationEnd(), this.disposeOnStop && this._scene && this._scene._toBeDisposed.push(this))), !e) { let n = 0; for (let i = 0; i < this._particles.length; i++) { const s = this._particles[i]; this._appendParticleVertices(n, s), n += this._useInstancing ? 1 : 4; } this._vertexBuffer && this._vertexBuffer.updateDirectly(this._vertexData, 0, this._particles.length); } this.manualEmitCount === 0 && this.disposeOnStop && this.stop(); } _appendParticleVertices(e, t) { this._appendParticleVertex(e++, t, 0, 0), this._useInstancing || (this._appendParticleVertex(e++, t, 1, 0), this._appendParticleVertex(e++, t, 1, 1), this._appendParticleVertex(e++, t, 0, 1)); } /** * Rebuilds the particle system. */ rebuild() { var e, t; this._engine.getCaps().vertexArrayObject && (this._vertexArrayObject = null), this._createIndexBuffer(), (e = this._spriteBuffer) === null || e === void 0 || e._rebuild(), (t = this._vertexBuffer) === null || t === void 0 || t._rebuild(); for (const r in this._vertexBuffers) this._vertexBuffers[r]._rebuild(); this.resetDrawCache(); } /** * Is this system ready to be used/rendered * @returns true if the system is ready */ isReady() { if (!this.emitter || this._imageProcessingConfiguration && !this._imageProcessingConfiguration.isReady() || !this.particleTexture || !this.particleTexture.isReady()) return !1; if (this.blendMode !== ti.BLENDMODE_MULTIPLYADD) { if (!this._getWrapper(this.blendMode).effect.isReady()) return !1; } else if (!this._getWrapper(ti.BLENDMODE_MULTIPLY).effect.isReady() || !this._getWrapper(ti.BLENDMODE_ADD).effect.isReady()) return !1; return !0; } _render(e) { var t, r, n, i, s, a, f, o; const d = this._getWrapper(e), v = d.effect, u = this._engine; u.enableEffect(d); const l = (t = this.defaultViewMatrix) !== null && t !== void 0 ? t : this._scene.getViewMatrix(); if (v.setTexture("diffuseSampler", this.particleTexture), v.setMatrix("view", l), v.setMatrix("projection", (r = this.defaultProjectionMatrix) !== null && r !== void 0 ? r : this._scene.getProjectionMatrix()), this._isAnimationSheetEnabled && this.particleTexture) { const p = this.particleTexture.getBaseSize(); v.setFloat3("particlesInfos", this.spriteCellWidth / p.width, this.spriteCellHeight / p.height, this.spriteCellWidth / p.width); } if (v.setVector2("translationPivot", this.translationPivot), v.setFloat4("textureMask", this.textureMask.r, this.textureMask.g, this.textureMask.b, this.textureMask.a), this._isBillboardBased && this._scene) { const p = this._scene.activeCamera; v.setVector3("eyePosition", p.globalPosition); } this._rampGradientsTexture && ((!this._rampGradients || !this._rampGradients.length) && (this._rampGradientsTexture.dispose(), this._rampGradientsTexture = null), v.setTexture("rampSampler", this._rampGradientsTexture)); const P = v.defines; switch (this._scene && Df(v, this, this._scene), P.indexOf("#define BILLBOARDMODE_ALL") >= 0 && (l.invertToRef(ue.Matrix[0]), v.setMatrix("invView", ue.Matrix[0])), this._vertexArrayObject !== void 0 ? !((n = this._scene) === null || n === void 0) && n.forceWireframe ? u.bindBuffers(this._vertexBuffers, this._linesIndexBufferUseInstancing, v) : (this._vertexArrayObject || (this._vertexArrayObject = this._engine.recordVertexArrayObject(this._vertexBuffers, null, v)), this._engine.bindVertexArrayObject(this._vertexArrayObject, !((i = this._scene) === null || i === void 0) && i.forceWireframe ? this._linesIndexBufferUseInstancing : this._indexBuffer)) : this._indexBuffer ? u.bindBuffers(this._vertexBuffers, !((a = this._scene) === null || a === void 0) && a.forceWireframe ? this._linesIndexBuffer : this._indexBuffer, v) : u.bindBuffers(this._vertexBuffers, !((s = this._scene) === null || s === void 0) && s.forceWireframe ? this._linesIndexBufferUseInstancing : null, v), this.useLogarithmicDepth && this._scene && Ye.BindLogDepth(P, v, this._scene), this._imageProcessingConfiguration && !this._imageProcessingConfiguration.applyByPostProcess && this._imageProcessingConfiguration.bind(v), e) { case ti.BLENDMODE_ADD: u.setAlphaMode(1); break; case ti.BLENDMODE_ONEONE: u.setAlphaMode(6); break; case ti.BLENDMODE_STANDARD: u.setAlphaMode(2); break; case ti.BLENDMODE_MULTIPLY: u.setAlphaMode(4); break; } return this._onBeforeDrawParticlesObservable && this._onBeforeDrawParticlesObservable.notifyObservers(v), this._useInstancing ? !((f = this._scene) === null || f === void 0) && f.forceWireframe ? u.drawElementsType(6, 0, 10, this._particles.length) : u.drawArraysType(7, 0, 4, this._particles.length) : !((o = this._scene) === null || o === void 0) && o.forceWireframe ? u.drawElementsType(1, 0, this._particles.length * 10) : u.drawElementsType(0, 0, this._particles.length * 6), this._particles.length; } /** * Renders the particle system in its current state. * @returns the current number of particles */ render() { if (!this.isReady() || !this._particles.length) return 0; const e = this._engine; e.setState && (e.setState(!1), this.forceDepthWrite && e.setDepthWrite(!0)); let t = 0; return this.blendMode === ti.BLENDMODE_MULTIPLYADD ? t = this._render(ti.BLENDMODE_MULTIPLY) + this._render(ti.BLENDMODE_ADD) : t = this._render(this.blendMode), this._engine.unbindInstanceAttributes(), this._engine.setAlphaMode(0), t; } /** * Disposes the particle system and free the associated resources * @param disposeTexture defines if the particle texture must be disposed as well (true by default) */ dispose(e = !0) { if (this.resetDrawCache(), this._vertexBuffer && (this._vertexBuffer.dispose(), this._vertexBuffer = null), this._spriteBuffer && (this._spriteBuffer.dispose(), this._spriteBuffer = null), this._indexBuffer && (this._engine._releaseBuffer(this._indexBuffer), this._indexBuffer = null), this._linesIndexBuffer && (this._engine._releaseBuffer(this._linesIndexBuffer), this._linesIndexBuffer = null), this._linesIndexBufferUseInstancing && (this._engine._releaseBuffer(this._linesIndexBufferUseInstancing), this._linesIndexBufferUseInstancing = null), this._vertexArrayObject && (this._engine.releaseVertexArrayObject(this._vertexArrayObject), this._vertexArrayObject = null), e && this.particleTexture && (this.particleTexture.dispose(), this.particleTexture = null), e && this.noiseTexture && (this.noiseTexture.dispose(), this.noiseTexture = null), this._rampGradientsTexture && (this._rampGradientsTexture.dispose(), this._rampGradientsTexture = null), this._removeFromRoot(), this.subEmitters && !this._subEmitters && this._prepareSubEmitterInternalArray(), this._subEmitters && this._subEmitters.length) { for (let t = 0; t < this._subEmitters.length; t++) for (const r of this._subEmitters[t]) r.dispose(); this._subEmitters = [], this.subEmitters = []; } if (this._disposeEmitterOnDispose && this.emitter && this.emitter.dispose && this.emitter.dispose(!0), this._onBeforeDrawParticlesObservable && this._onBeforeDrawParticlesObservable.clear(), this._scene) { const t = this._scene.particleSystems.indexOf(this); t > -1 && this._scene.particleSystems.splice(t, 1), this._scene._activeParticleSystems.dispose(); } this.onDisposeObservable.notifyObservers(this), this.onDisposeObservable.clear(), this.onStoppedObservable.clear(), this.reset(); } // Clone /** * Clones the particle system. * @param name The name of the cloned object * @param newEmitter The new emitter to use * @param cloneTexture Also clone the textures if true * @returns the cloned particle system */ clone(e, t, r = !1) { const n = Object.assign({}, this._customWrappers); let i = null; const s = this._engine; if (s.createEffectForParticles && this.customShader != null) { i = this.customShader; const o = i.shaderOptions.defines.length > 0 ? i.shaderOptions.defines.join(` `) : "", d = s.createEffectForParticles(i.shaderPath.fragmentElement, i.shaderOptions.uniforms, i.shaderOptions.samplers, o); n[0] ? n[0].effect = d : this.setCustomEffect(d, 0); } const a = this.serialize(r), f = ti.Parse(a, this._scene || this._engine, this._rootUrl); return f.name = e, f.customShader = i, f._customWrappers = n, t === void 0 && (t = this.emitter), this.noiseTexture && (f.noiseTexture = this.noiseTexture.clone()), f.emitter = t, this.preventAutoStart || f.start(), f; } /** * Serializes the particle system to a JSON object * @param serializeTexture defines if the texture must be serialized as well * @returns the JSON object */ serialize(e = !1) { const t = {}; if (ti._Serialize(t, this, e), t.textureMask = this.textureMask.asArray(), t.customShader = this.customShader, t.preventAutoStart = this.preventAutoStart, this.subEmitters) { t.subEmitters = [], this._subEmitters || this._prepareSubEmitterInternalArray(); for (const r of this._subEmitters) { const n = []; for (const i of r) n.push(i.serialize(e)); t.subEmitters.push(n); } } return t; } /** * @internal */ static _Serialize(e, t, r) { if (e.name = t.name, e.id = t.id, e.capacity = t.getCapacity(), e.disposeOnStop = t.disposeOnStop, e.manualEmitCount = t.manualEmitCount, t.emitter.position) { const c = t.emitter; e.emitterId = c.id; } else { const c = t.emitter; e.emitter = c.asArray(); } t.particleEmitterType && (e.particleEmitterType = t.particleEmitterType.serialize()), t.particleTexture && (r ? e.texture = t.particleTexture.serialize() : (e.textureName = t.particleTexture.name, e.invertY = !!t.particleTexture._invertY)), e.isLocal = t.isLocal, jt.AppendSerializedAnimations(t, e), e.beginAnimationOnStart = t.beginAnimationOnStart, e.beginAnimationFrom = t.beginAnimationFrom, e.beginAnimationTo = t.beginAnimationTo, e.beginAnimationLoop = t.beginAnimationLoop, e.startDelay = t.startDelay, e.renderingGroupId = t.renderingGroupId, e.isBillboardBased = t.isBillboardBased, e.billboardMode = t.billboardMode, e.minAngularSpeed = t.minAngularSpeed, e.maxAngularSpeed = t.maxAngularSpeed, e.minSize = t.minSize, e.maxSize = t.maxSize, e.minScaleX = t.minScaleX, e.maxScaleX = t.maxScaleX, e.minScaleY = t.minScaleY, e.maxScaleY = t.maxScaleY, e.minEmitPower = t.minEmitPower, e.maxEmitPower = t.maxEmitPower, e.minLifeTime = t.minLifeTime, e.maxLifeTime = t.maxLifeTime, e.emitRate = t.emitRate, e.gravity = t.gravity.asArray(), e.noiseStrength = t.noiseStrength.asArray(), e.color1 = t.color1.asArray(), e.color2 = t.color2.asArray(), e.colorDead = t.colorDead.asArray(), e.updateSpeed = t.updateSpeed, e.targetStopDuration = t.targetStopDuration, e.blendMode = t.blendMode, e.preWarmCycles = t.preWarmCycles, e.preWarmStepOffset = t.preWarmStepOffset, e.minInitialRotation = t.minInitialRotation, e.maxInitialRotation = t.maxInitialRotation, e.startSpriteCellID = t.startSpriteCellID, e.spriteCellLoop = t.spriteCellLoop, e.endSpriteCellID = t.endSpriteCellID, e.spriteCellChangeSpeed = t.spriteCellChangeSpeed, e.spriteCellWidth = t.spriteCellWidth, e.spriteCellHeight = t.spriteCellHeight, e.spriteRandomStartCell = t.spriteRandomStartCell, e.isAnimationSheetEnabled = t.isAnimationSheetEnabled, e.useLogarithmicDepth = t.useLogarithmicDepth; const n = t.getColorGradients(); if (n) { e.colorGradients = []; for (const c of n) { const H = { gradient: c.gradient, color1: c.color1.asArray() }; c.color2 ? H.color2 = c.color2.asArray() : H.color2 = c.color1.asArray(), e.colorGradients.push(H); } } const i = t.getRampGradients(); if (i) { e.rampGradients = []; for (const c of i) { const H = { gradient: c.gradient, color: c.color.asArray() }; e.rampGradients.push(H); } e.useRampGradients = t.useRampGradients; } const s = t.getColorRemapGradients(); if (s) { e.colorRemapGradients = []; for (const c of s) { const H = { gradient: c.gradient, factor1: c.factor1 }; c.factor2 !== void 0 ? H.factor2 = c.factor2 : H.factor2 = c.factor1, e.colorRemapGradients.push(H); } } const a = t.getAlphaRemapGradients(); if (a) { e.alphaRemapGradients = []; for (const c of a) { const H = { gradient: c.gradient, factor1: c.factor1 }; c.factor2 !== void 0 ? H.factor2 = c.factor2 : H.factor2 = c.factor1, e.alphaRemapGradients.push(H); } } const f = t.getSizeGradients(); if (f) { e.sizeGradients = []; for (const c of f) { const H = { gradient: c.gradient, factor1: c.factor1 }; c.factor2 !== void 0 ? H.factor2 = c.factor2 : H.factor2 = c.factor1, e.sizeGradients.push(H); } } const o = t.getAngularSpeedGradients(); if (o) { e.angularSpeedGradients = []; for (const c of o) { const H = { gradient: c.gradient, factor1: c.factor1 }; c.factor2 !== void 0 ? H.factor2 = c.factor2 : H.factor2 = c.factor1, e.angularSpeedGradients.push(H); } } const d = t.getVelocityGradients(); if (d) { e.velocityGradients = []; for (const c of d) { const H = { gradient: c.gradient, factor1: c.factor1 }; c.factor2 !== void 0 ? H.factor2 = c.factor2 : H.factor2 = c.factor1, e.velocityGradients.push(H); } } const v = t.getDragGradients(); if (v) { e.dragGradients = []; for (const c of v) { const H = { gradient: c.gradient, factor1: c.factor1 }; c.factor2 !== void 0 ? H.factor2 = c.factor2 : H.factor2 = c.factor1, e.dragGradients.push(H); } } const u = t.getEmitRateGradients(); if (u) { e.emitRateGradients = []; for (const c of u) { const H = { gradient: c.gradient, factor1: c.factor1 }; c.factor2 !== void 0 ? H.factor2 = c.factor2 : H.factor2 = c.factor1, e.emitRateGradients.push(H); } } const l = t.getStartSizeGradients(); if (l) { e.startSizeGradients = []; for (const c of l) { const H = { gradient: c.gradient, factor1: c.factor1 }; c.factor2 !== void 0 ? H.factor2 = c.factor2 : H.factor2 = c.factor1, e.startSizeGradients.push(H); } } const P = t.getLifeTimeGradients(); if (P) { e.lifeTimeGradients = []; for (const c of P) { const H = { gradient: c.gradient, factor1: c.factor1 }; c.factor2 !== void 0 ? H.factor2 = c.factor2 : H.factor2 = c.factor1, e.lifeTimeGradients.push(H); } } const p = t.getLimitVelocityGradients(); if (p) { e.limitVelocityGradients = []; for (const c of p) { const H = { gradient: c.gradient, factor1: c.factor1 }; c.factor2 !== void 0 ? H.factor2 = c.factor2 : H.factor2 = c.factor1, e.limitVelocityGradients.push(H); } e.limitVelocityDamping = t.limitVelocityDamping; } t.noiseTexture && (e.noiseTexture = t.noiseTexture.serialize()); } /** * @internal */ static _Parse(e, t, r, n) { var i, s, a; let f; r instanceof hr ? f = null : f = r; const o = Jo("BABYLON.Texture"); if (o && f && (e.texture ? t.particleTexture = o.Parse(e.texture, f, n) : e.textureName && (t.particleTexture = new o(n + e.textureName, f, !1, e.invertY !== void 0 ? e.invertY : !0), t.particleTexture.name = e.textureName)), !e.emitterId && e.emitterId !== 0 && e.emitter === void 0 ? t.emitter = S.Zero() : e.emitterId && f ? t.emitter = f.getLastMeshById(e.emitterId) : t.emitter = S.FromArray(e.emitter), t.isLocal = !!e.isLocal, e.renderingGroupId !== void 0 && (t.renderingGroupId = e.renderingGroupId), e.isBillboardBased !== void 0 && (t.isBillboardBased = e.isBillboardBased), e.billboardMode !== void 0 && (t.billboardMode = e.billboardMode), e.useLogarithmicDepth !== void 0 && (t.useLogarithmicDepth = e.useLogarithmicDepth), e.animations) { for (let v = 0; v < e.animations.length; v++) { const u = e.animations[v], l = Jo("BABYLON.Animation"); l && t.animations.push(l.Parse(u)); } t.beginAnimationOnStart = e.beginAnimationOnStart, t.beginAnimationFrom = e.beginAnimationFrom, t.beginAnimationTo = e.beginAnimationTo, t.beginAnimationLoop = e.beginAnimationLoop; } if (e.autoAnimate && f && f.beginAnimation(t, e.autoAnimateFrom, e.autoAnimateTo, e.autoAnimateLoop, e.autoAnimateSpeed || 1), t.startDelay = e.startDelay | 0, t.minAngularSpeed = e.minAngularSpeed, t.maxAngularSpeed = e.maxAngularSpeed, t.minSize = e.minSize, t.maxSize = e.maxSize, e.minScaleX && (t.minScaleX = e.minScaleX, t.maxScaleX = e.maxScaleX, t.minScaleY = e.minScaleY, t.maxScaleY = e.maxScaleY), e.preWarmCycles !== void 0 && (t.preWarmCycles = e.preWarmCycles, t.preWarmStepOffset = e.preWarmStepOffset), e.minInitialRotation !== void 0 && (t.minInitialRotation = e.minInitialRotation, t.maxInitialRotation = e.maxInitialRotation), t.minLifeTime = e.minLifeTime, t.maxLifeTime = e.maxLifeTime, t.minEmitPower = e.minEmitPower, t.maxEmitPower = e.maxEmitPower, t.emitRate = e.emitRate, t.gravity = S.FromArray(e.gravity), e.noiseStrength && (t.noiseStrength = S.FromArray(e.noiseStrength)), t.color1 = xt.FromArray(e.color1), t.color2 = xt.FromArray(e.color2), t.colorDead = xt.FromArray(e.colorDead), t.updateSpeed = e.updateSpeed, t.targetStopDuration = e.targetStopDuration, t.blendMode = e.blendMode, e.colorGradients) for (const v of e.colorGradients) t.addColorGradient(v.gradient, xt.FromArray(v.color1), v.color2 ? xt.FromArray(v.color2) : void 0); if (e.rampGradients) { for (const v of e.rampGradients) t.addRampGradient(v.gradient, Ne.FromArray(v.color)); t.useRampGradients = e.useRampGradients; } if (e.colorRemapGradients) for (const v of e.colorRemapGradients) t.addColorRemapGradient(v.gradient, v.factor1 !== void 0 ? v.factor1 : v.factor, v.factor2); if (e.alphaRemapGradients) for (const v of e.alphaRemapGradients) t.addAlphaRemapGradient(v.gradient, v.factor1 !== void 0 ? v.factor1 : v.factor, v.factor2); if (e.sizeGradients) for (const v of e.sizeGradients) t.addSizeGradient(v.gradient, v.factor1 !== void 0 ? v.factor1 : v.factor, v.factor2); if (e.angularSpeedGradients) for (const v of e.angularSpeedGradients) t.addAngularSpeedGradient(v.gradient, v.factor1 !== void 0 ? v.factor1 : v.factor, v.factor2); if (e.velocityGradients) for (const v of e.velocityGradients) t.addVelocityGradient(v.gradient, v.factor1 !== void 0 ? v.factor1 : v.factor, v.factor2); if (e.dragGradients) for (const v of e.dragGradients) t.addDragGradient(v.gradient, v.factor1 !== void 0 ? v.factor1 : v.factor, v.factor2); if (e.emitRateGradients) for (const v of e.emitRateGradients) t.addEmitRateGradient(v.gradient, v.factor1 !== void 0 ? v.factor1 : v.factor, v.factor2); if (e.startSizeGradients) for (const v of e.startSizeGradients) t.addStartSizeGradient(v.gradient, v.factor1 !== void 0 ? v.factor1 : v.factor, v.factor2); if (e.lifeTimeGradients) for (const v of e.lifeTimeGradients) t.addLifeTimeGradient(v.gradient, v.factor1 !== void 0 ? v.factor1 : v.factor, v.factor2); if (e.limitVelocityGradients) { for (const v of e.limitVelocityGradients) t.addLimitVelocityGradient(v.gradient, v.factor1 !== void 0 ? v.factor1 : v.factor, v.factor2); t.limitVelocityDamping = e.limitVelocityDamping; } if (e.noiseTexture && f) { const v = Jo("BABYLON.ProceduralTexture"); t.noiseTexture = v.Parse(e.noiseTexture, f, n); } let d; if (e.particleEmitterType) { switch (e.particleEmitterType.type) { case "SphereParticleEmitter": d = new oU(); break; case "SphereDirectedParticleEmitter": d = new SR(); break; case "ConeEmitter": case "ConeParticleEmitter": d = new wR(); break; case "CylinderParticleEmitter": d = new aU(); break; case "CylinderDirectedParticleEmitter": d = new mR(); break; case "HemisphericParticleEmitter": d = new BR(); break; case "PointParticleEmitter": d = new WR(); break; case "MeshParticleEmitter": d = new IO(); break; case "BoxEmitter": case "BoxParticleEmitter": default: d = new AD(); break; } d.parse(e.particleEmitterType, f); } else d = new AD(), d.parse(e, f); t.particleEmitterType = d, t.startSpriteCellID = e.startSpriteCellID, t.endSpriteCellID = e.endSpriteCellID, t.spriteCellLoop = (i = e.spriteCellLoop) !== null && i !== void 0 ? i : !0, t.spriteCellWidth = e.spriteCellWidth, t.spriteCellHeight = e.spriteCellHeight, t.spriteCellChangeSpeed = e.spriteCellChangeSpeed, t.spriteRandomStartCell = e.spriteRandomStartCell, t.disposeOnStop = (s = e.disposeOnStop) !== null && s !== void 0 ? s : !1, t.manualEmitCount = (a = e.manualEmitCount) !== null && a !== void 0 ? a : -1; } /** * Parses a JSON object to create a particle system. * @param parsedParticleSystem The JSON object to parse * @param sceneOrEngine The scene or the engine to create the particle system in * @param rootUrl The root url to use to load external dependencies like texture * @param doNotStart Ignore the preventAutoStart attribute and does not start * @param capacity defines the system capacity (if null or undefined the sotred capacity will be used) * @returns the Parsed particle system */ static Parse(e, t, r, n = !1, i) { const s = e.name; let a = null, f = null, o, d; if (t instanceof hr ? o = t : (d = t, o = d.getEngine()), e.customShader && o.createEffectForParticles) { f = e.customShader; const u = f.shaderOptions.defines.length > 0 ? f.shaderOptions.defines.join(` `) : ""; a = o.createEffectForParticles(f.shaderPath.fragmentElement, f.shaderOptions.uniforms, f.shaderOptions.samplers, u); } const v = new ti(s, i || e.capacity, t, a, e.isAnimationSheetEnabled); if (v.customShader = f, v._rootUrl = r, e.id && (v.id = e.id), e.subEmitters) { v.subEmitters = []; for (const u of e.subEmitters) { const l = []; for (const P of u) l.push(vq.Parse(P, t, r)); v.subEmitters.push(l); } } return ti._Parse(e, v, t, r), e.textureMask && (v.textureMask = xt.FromArray(e.textureMask)), e.preventAutoStart && (v.preventAutoStart = e.preventAutoStart), !n && !v.preventAutoStart && v.start(), v; } } ti.BILLBOARDMODE_Y = 2; ti.BILLBOARDMODE_ALL = 7; ti.BILLBOARDMODE_STRETCHED = 8; ti.BILLBOARDMODE_STRETCHED_LOCAL = 9; vq._ParseParticleSystem = ti.Parse; const Ehe = "clipPlaneFragmentDeclaration2", Fhe = `#ifdef CLIPPLANE in float fClipDistance; #endif #ifdef CLIPPLANE2 in float fClipDistance2; #endif #ifdef CLIPPLANE3 in float fClipDistance3; #endif #ifdef CLIPPLANE4 in float fClipDistance4; #endif #ifdef CLIPPLANE5 in float fClipDistance5; #endif #ifdef CLIPPLANE6 in float fClipDistance6; #endif `; Le.IncludesShadersStore[Ehe] = Fhe; const Nhe = "gpuRenderParticlesPixelShader", Qhe = `precision highp float; #ifdef LOGARITHMICDEPTH #extension GL_EXT_frag_depth : enable #endif uniform sampler2D diffuseSampler;varying vec2 vUV;varying vec4 vColor; #include #include #include #include #include void main() { #include vec4 textureColor=texture2D(diffuseSampler,vUV);gl_FragColor=textureColor*vColor; #ifdef BLENDMULTIPLYMODE float alpha=vColor.a*textureColor.a;gl_FragColor.rgb=gl_FragColor.rgb*alpha+vec3(1.0)*(1.0-alpha); #endif #include #ifdef IMAGEPROCESSINGPOSTPROCESS gl_FragColor.rgb=toLinearSpace(gl_FragColor.rgb); #else #ifdef IMAGEPROCESSING gl_FragColor.rgb=toLinearSpace(gl_FragColor.rgb);gl_FragColor=applyImageProcessing(gl_FragColor); #endif #endif } `; Le.ShadersStore[Nhe] = Qhe; const Yhe = "clipPlaneVertexDeclaration2", Mhe = `#ifdef CLIPPLANE uniform vec4 vClipPlane;out float fClipDistance; #endif #ifdef CLIPPLANE2 uniform vec4 vClipPlane2;out float fClipDistance2; #endif #ifdef CLIPPLANE3 uniform vec4 vClipPlane3;out float fClipDistance3; #endif #ifdef CLIPPLANE4 uniform vec4 vClipPlane4;out float fClipDistance4; #endif #ifdef CLIPPLANE5 uniform vec4 vClipPlane5;out float fClipDistance5; #endif #ifdef CLIPPLANE6 uniform vec4 vClipPlane6;out float fClipDistance6; #endif `; Le.IncludesShadersStore[Yhe] = Mhe; const Lhe = "gpuRenderParticlesVertexShader", Khe = `precision highp float;uniform mat4 view;uniform mat4 projection;uniform vec2 translationPivot;uniform vec3 worldOffset; #ifdef LOCAL uniform mat4 emitterWM; #endif attribute vec3 position;attribute float age;attribute float life;attribute vec3 size; #ifndef BILLBOARD attribute vec3 initialDirection; #endif #ifdef BILLBOARDSTRETCHED attribute vec3 direction; #endif attribute float angle; #ifdef ANIMATESHEET attribute float cellIndex; #endif attribute vec2 offset;attribute vec2 uv;varying vec2 vUV;varying vec4 vColor;varying vec3 vPositionW; #if defined(BILLBOARD) && !defined(BILLBOARDY) && !defined(BILLBOARDSTRETCHED) uniform mat4 invView; #endif #include #include #ifdef COLORGRADIENTS uniform sampler2D colorGradientSampler; #else uniform vec4 colorDead;attribute vec4 color; #endif #ifdef ANIMATESHEET uniform vec3 sheetInfos; #endif #ifdef BILLBOARD uniform vec3 eyePosition; #endif vec3 rotate(vec3 yaxis,vec3 rotatedCorner) {vec3 xaxis=normalize(cross(vec3(0.,1.0,0.),yaxis));vec3 zaxis=normalize(cross(yaxis,xaxis));vec3 row0=vec3(xaxis.x,xaxis.y,xaxis.z);vec3 row1=vec3(yaxis.x,yaxis.y,yaxis.z);vec3 row2=vec3(zaxis.x,zaxis.y,zaxis.z);mat3 rotMatrix= mat3(row0,row1,row2);vec3 alignedCorner=rotMatrix*rotatedCorner; #ifdef LOCAL return ((emitterWM*vec4(position,1.0)).xyz+worldOffset)+alignedCorner; #else return (position+worldOffset)+alignedCorner; #endif } #ifdef BILLBOARDSTRETCHED vec3 rotateAlign(vec3 toCamera,vec3 rotatedCorner) {vec3 normalizedToCamera=normalize(toCamera);vec3 normalizedCrossDirToCamera=normalize(cross(normalize(direction),normalizedToCamera));vec3 crossProduct=normalize(cross(normalizedToCamera,normalizedCrossDirToCamera));vec3 row0=vec3(normalizedCrossDirToCamera.x,normalizedCrossDirToCamera.y,normalizedCrossDirToCamera.z);vec3 row1=vec3(crossProduct.x,crossProduct.y,crossProduct.z);vec3 row2=vec3(normalizedToCamera.x,normalizedToCamera.y,normalizedToCamera.z);mat3 rotMatrix= mat3(row0,row1,row2);vec3 alignedCorner=rotMatrix*rotatedCorner; #ifdef LOCAL return ((emitterWM*vec4(position,1.0)).xyz+worldOffset)+alignedCorner; #else return (position+worldOffset)+alignedCorner; #endif } #endif void main() { #ifdef ANIMATESHEET float rowOffset=floor(cellIndex/sheetInfos.z);float columnOffset=cellIndex-rowOffset*sheetInfos.z;vec2 uvScale=sheetInfos.xy;vec2 uvOffset=vec2(uv.x ,1.0-uv.y);vUV=(uvOffset+vec2(columnOffset,rowOffset))*uvScale; #else vUV=uv; #endif float ratio=age/life; #ifdef COLORGRADIENTS vColor=texture2D(colorGradientSampler,vec2(ratio,0)); #else vColor=color*vec4(1.0-ratio)+colorDead*vec4(ratio); #endif vec2 cornerPos=(offset-translationPivot)*size.yz*size.x; #ifdef BILLBOARD vec4 rotatedCorner;rotatedCorner.w=0.; #ifdef BILLBOARDY rotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.z=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.y=0.;rotatedCorner.xz+=translationPivot;vec3 yaxis=(position+worldOffset)-eyePosition;yaxis.y=0.;vPositionW=rotate(normalize(yaxis),rotatedCorner.xyz);vec4 viewPosition=(view*vec4(vPositionW,1.0)); #elif defined(BILLBOARDSTRETCHED) rotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.y=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.z=0.;rotatedCorner.xy+=translationPivot;vec3 toCamera=(position+worldOffset)-eyePosition;vPositionW=rotateAlign(toCamera,rotatedCorner.xyz);vec4 viewPosition=(view*vec4(vPositionW,1.0)); #else rotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.y=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.z=0.;rotatedCorner.xy+=translationPivot; #ifdef LOCAL vec4 viewPosition=view*vec4(((emitterWM*vec4(position,1.0)).xyz+worldOffset),1.0)+rotatedCorner; #else vec4 viewPosition=view*vec4((position+worldOffset),1.0)+rotatedCorner; #endif vPositionW=(invView*viewPosition).xyz; #endif #else vec3 rotatedCorner;rotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.y=0.;rotatedCorner.z=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.xz+=translationPivot;vec3 yaxis=normalize(initialDirection);vPositionW=rotate(yaxis,rotatedCorner);vec4 viewPosition=view*vec4(vPositionW,1.0); #endif gl_Position=projection*viewPosition; #if defined(CLIPPLANE) || defined(CLIPPLANE2) || defined(CLIPPLANE3) || defined(CLIPPLANE4) || defined(CLIPPLANE5) || defined(CLIPPLANE6) vec4 worldPos=vec4(vPositionW,1.0); #endif #include #include }`; Le.ShadersStore[Lhe] = Khe; class bc extends N0 { /** * Gets a boolean indicating if the GPU particles can be rendered on current browser */ static get IsSupported() { if (!gr.LastCreatedEngine) return !1; const e = gr.LastCreatedEngine.getCaps(); return e.supportTransformFeedbacks || e.supportComputeShaders; } _createIndexBuffer() { this._linesIndexBufferUseInstancing = this._engine.createIndexBuffer(new Uint32Array([0, 1, 1, 3, 3, 2, 2, 0, 0, 3])); } /** * Gets the maximum number of particles active at the same time. * @returns The max number of active particles. */ getCapacity() { return this._capacity; } /** * Gets or set the number of active particles * The value cannot be greater than "capacity" (if it is, it will be limited to "capacity"). */ get maxActiveParticleCount() { return this._maxActiveParticleCount; } set maxActiveParticleCount(e) { this._maxActiveParticleCount = Math.min(e, this._capacity); } /** * Gets or set the number of active particles * @deprecated Please use maxActiveParticleCount instead. */ get activeParticleCount() { return this.maxActiveParticleCount; } set activeParticleCount(e) { this.maxActiveParticleCount = e; } /** * Is this system ready to be used/rendered * @returns true if the system is ready */ isReady() { if (!this.emitter || this._imageProcessingConfiguration && !this._imageProcessingConfiguration.isReady() || !this.particleTexture || !this.particleTexture.isReady()) return !1; if (this.blendMode !== ti.BLENDMODE_MULTIPLYADD) { if (!this._getWrapper(this.blendMode).effect.isReady()) return !1; } else if (!this._getWrapper(ti.BLENDMODE_MULTIPLY).effect.isReady() || !this._getWrapper(ti.BLENDMODE_ADD).effect.isReady()) return !1; return this._platform.isUpdateBufferCreated() ? this._platform.isUpdateBufferReady() : (this._recreateUpdateEffect(), !1); } /** * Gets if the system has been started. (Note: this will still be true after stop is called) * @returns True if it has been started, otherwise false. */ isStarted() { return this._started; } /** * Gets if the system has been stopped. (Note: rendering is still happening but the system is frozen) * @returns True if it has been stopped, otherwise false. */ isStopped() { return this._stopped; } /** * Gets a boolean indicating that the system is stopping * @returns true if the system is currently stopping */ isStopping() { return !1; } /** * Gets the number of particles active at the same time. * @returns The number of active particles. */ getActiveCount() { return this._currentActiveCount; } /** * Starts the particle system and begins to emit * @param delay defines the delay in milliseconds before starting the system (this.startDelay by default) */ start(e = this.startDelay) { if (!this.targetStopDuration && this._hasTargetStopDurationDependantGradient()) throw "Particle system started with a targetStopDuration dependant gradient (eg. startSizeGradients) but no targetStopDuration set"; if (e) { setTimeout(() => { this.start(0); }, e); return; } this._started = !0, this._stopped = !1, this._preWarmDone = !1, this.beginAnimationOnStart && this.animations && this.animations.length > 0 && this._scene && this._scene.beginAnimation(this, this.beginAnimationFrom, this.beginAnimationTo, this.beginAnimationLoop); } /** * Stops the particle system. */ stop() { this._stopped || (this._stopped = !0); } /** * Remove all active particles */ reset() { this._releaseBuffers(), this._platform.releaseVertexBuffers(), this._currentActiveCount = 0, this._targetIndex = 0; } /** * Returns the string "GPUParticleSystem" * @returns a string containing the class name */ getClassName() { return "GPUParticleSystem"; } /** * Gets the custom effect used to render the particles * @param blendMode Blend mode for which the effect should be retrieved * @returns The effect */ getCustomEffect(e = 0) { var t, r; return (r = (t = this._customWrappers[e]) === null || t === void 0 ? void 0 : t.effect) !== null && r !== void 0 ? r : this._customWrappers[0].effect; } _getCustomDrawWrapper(e = 0) { var t; return (t = this._customWrappers[e]) !== null && t !== void 0 ? t : this._customWrappers[0]; } /** * Sets the custom effect used to render the particles * @param effect The effect to set * @param blendMode Blend mode for which the effect should be set */ setCustomEffect(e, t = 0) { this._customWrappers[t] = new zo(this._engine), this._customWrappers[t].effect = e; } /** * Observable that will be called just before the particles are drawn */ get onBeforeDrawParticlesObservable() { return this._onBeforeDrawParticlesObservable || (this._onBeforeDrawParticlesObservable = new Oe()), this._onBeforeDrawParticlesObservable; } /** * Gets the name of the particle vertex shader */ get vertexShaderName() { return "gpuRenderParticles"; } /** * Gets the vertex buffers used by the particle system * Should be called after render() has been called for the current frame so that the buffers returned are the ones that have been updated * in the current frame (there's a ping-pong between two sets of buffers - for a given frame, one set is used as the source and the other as the destination) */ get vertexBuffers() { return this._renderVertexBuffers[this._targetIndex ^ 1]; } /** * Gets the index buffer used by the particle system (null for GPU particle systems) */ get indexBuffer() { return null; } _removeGradientAndTexture(e, t, r) { return super._removeGradientAndTexture(e, t, r), this._releaseBuffers(), this; } /** * Adds a new color gradient * @param gradient defines the gradient to use (between 0 and 1) * @param color1 defines the color to affect to the specified gradient * @returns the current particle system */ addColorGradient(e, t) { this._colorGradients || (this._colorGradients = []); const r = new yY(e, t); return this._colorGradients.push(r), this._refreshColorGradient(!0), this._releaseBuffers(), this; } _refreshColorGradient(e = !1) { this._colorGradients && (e && this._colorGradients.sort((t, r) => t.gradient < r.gradient ? -1 : t.gradient > r.gradient ? 1 : 0), this._colorGradientsTexture && (this._colorGradientsTexture.dispose(), this._colorGradientsTexture = null)); } /** Force the system to rebuild all gradients that need to be resync */ forceRefreshGradients() { this._refreshColorGradient(), this._refreshFactorGradient(this._sizeGradients, "_sizeGradientsTexture"), this._refreshFactorGradient(this._angularSpeedGradients, "_angularSpeedGradientsTexture"), this._refreshFactorGradient(this._velocityGradients, "_velocityGradientsTexture"), this._refreshFactorGradient(this._limitVelocityGradients, "_limitVelocityGradientsTexture"), this._refreshFactorGradient(this._dragGradients, "_dragGradientsTexture"), this.reset(); } /** * Remove a specific color gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeColorGradient(e) { return this._removeGradientAndTexture(e, this._colorGradients, this._colorGradientsTexture), this._colorGradientsTexture = null, this; } /** * Resets the draw wrappers cache */ resetDrawCache() { var e; for (const t in this._drawWrappers) (e = this._drawWrappers[t].drawContext) === null || e === void 0 || e.reset(); } _addFactorGradient(e, t, r) { const n = new kY(t, r); e.push(n), this._releaseBuffers(); } /** * Adds a new size gradient * @param gradient defines the gradient to use (between 0 and 1) * @param factor defines the size factor to affect to the specified gradient * @returns the current particle system */ addSizeGradient(e, t) { return this._sizeGradients || (this._sizeGradients = []), this._addFactorGradient(this._sizeGradients, e, t), this._refreshFactorGradient(this._sizeGradients, "_sizeGradientsTexture", !0), this._releaseBuffers(), this; } /** * Remove a specific size gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeSizeGradient(e) { return this._removeGradientAndTexture(e, this._sizeGradients, this._sizeGradientsTexture), this._sizeGradientsTexture = null, this; } _refreshFactorGradient(e, t, r = !1) { if (!e) return; r && e.sort((i, s) => i.gradient < s.gradient ? -1 : i.gradient > s.gradient ? 1 : 0); const n = this; n[t] && (n[t].dispose(), n[t] = null); } /** * Adds a new angular speed gradient * @param gradient defines the gradient to use (between 0 and 1) * @param factor defines the angular speed to affect to the specified gradient * @returns the current particle system */ addAngularSpeedGradient(e, t) { return this._angularSpeedGradients || (this._angularSpeedGradients = []), this._addFactorGradient(this._angularSpeedGradients, e, t), this._refreshFactorGradient(this._angularSpeedGradients, "_angularSpeedGradientsTexture", !0), this._releaseBuffers(), this; } /** * Remove a specific angular speed gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeAngularSpeedGradient(e) { return this._removeGradientAndTexture(e, this._angularSpeedGradients, this._angularSpeedGradientsTexture), this._angularSpeedGradientsTexture = null, this; } /** * Adds a new velocity gradient * @param gradient defines the gradient to use (between 0 and 1) * @param factor defines the velocity to affect to the specified gradient * @returns the current particle system */ addVelocityGradient(e, t) { return this._velocityGradients || (this._velocityGradients = []), this._addFactorGradient(this._velocityGradients, e, t), this._refreshFactorGradient(this._velocityGradients, "_velocityGradientsTexture", !0), this._releaseBuffers(), this; } /** * Remove a specific velocity gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeVelocityGradient(e) { return this._removeGradientAndTexture(e, this._velocityGradients, this._velocityGradientsTexture), this._velocityGradientsTexture = null, this; } /** * Adds a new limit velocity gradient * @param gradient defines the gradient to use (between 0 and 1) * @param factor defines the limit velocity value to affect to the specified gradient * @returns the current particle system */ addLimitVelocityGradient(e, t) { return this._limitVelocityGradients || (this._limitVelocityGradients = []), this._addFactorGradient(this._limitVelocityGradients, e, t), this._refreshFactorGradient(this._limitVelocityGradients, "_limitVelocityGradientsTexture", !0), this._releaseBuffers(), this; } /** * Remove a specific limit velocity gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeLimitVelocityGradient(e) { return this._removeGradientAndTexture(e, this._limitVelocityGradients, this._limitVelocityGradientsTexture), this._limitVelocityGradientsTexture = null, this; } /** * Adds a new drag gradient * @param gradient defines the gradient to use (between 0 and 1) * @param factor defines the drag value to affect to the specified gradient * @returns the current particle system */ addDragGradient(e, t) { return this._dragGradients || (this._dragGradients = []), this._addFactorGradient(this._dragGradients, e, t), this._refreshFactorGradient(this._dragGradients, "_dragGradientsTexture", !0), this._releaseBuffers(), this; } /** * Remove a specific drag gradient * @param gradient defines the gradient to remove * @returns the current particle system */ removeDragGradient(e) { return this._removeGradientAndTexture(e, this._dragGradients, this._dragGradientsTexture), this._dragGradientsTexture = null, this; } /** * Not supported by GPUParticleSystem * @returns the current particle system */ addEmitRateGradient() { return this; } /** * Not supported by GPUParticleSystem * @returns the current particle system */ removeEmitRateGradient() { return this; } /** * Not supported by GPUParticleSystem * @returns the current particle system */ addStartSizeGradient() { return this; } /** * Not supported by GPUParticleSystem * @returns the current particle system */ removeStartSizeGradient() { return this; } /** * Not supported by GPUParticleSystem * @returns the current particle system */ addColorRemapGradient() { return this; } /** * Not supported by GPUParticleSystem * @returns the current particle system */ removeColorRemapGradient() { return this; } /** * Not supported by GPUParticleSystem * @returns the current particle system */ addAlphaRemapGradient() { return this; } /** * Not supported by GPUParticleSystem * @returns the current particle system */ removeAlphaRemapGradient() { return this; } /** * Not supported by GPUParticleSystem * @returns the current particle system */ addRampGradient() { return this; } /** * Not supported by GPUParticleSystem * @returns the current particle system */ removeRampGradient() { return this; } /** * Not supported by GPUParticleSystem * @returns the list of ramp gradients */ getRampGradients() { return null; } /** * Not supported by GPUParticleSystem * Gets or sets a boolean indicating that ramp gradients must be used * @see https://doc.babylonjs.com/features/featuresDeepDive/particles/particle_system/particle_system_intro#ramp-gradients */ get useRampGradients() { return !1; } set useRampGradients(e) { } /** * Not supported by GPUParticleSystem * @returns the current particle system */ addLifeTimeGradient() { return this; } /** * Not supported by GPUParticleSystem * @returns the current particle system */ removeLifeTimeGradient() { return this; } /** * Instantiates a GPU particle system. * Particles are often small sprites used to simulate hard-to-reproduce phenomena like fire, smoke, water, or abstract visual effects like magic glitter and faery dust. * @param name The name of the particle system * @param options The options used to create the system * @param sceneOrEngine The scene the particle system belongs to or the engine to use if no scene * @param customEffect a custom effect used to change the way particles are rendered by default * @param isAnimationSheetEnabled Must be true if using a spritesheet to animate the particles texture */ constructor(e, t, r, n = null, i = !1) { if (super(e), this.layerMask = 268435455, this._accumulatedCount = 0, this._renderVertexBuffers = [], this._targetIndex = 0, this._currentRenderId = -1, this._currentRenderingCameraUniqueId = -1, this._started = !1, this._stopped = !1, this._timeDelta = 0, this.updateInAnimate = !1, this._actualFrame = 0, this._rawTextureWidth = 256, this.onDisposeObservable = new Oe(), this.onStoppedObservable = new Oe(), this.forceDepthWrite = !1, this._preWarmDone = !1, this.isLocal = !1, this.isGPU = !0, this._onBeforeDrawParticlesObservable = null, !r || r.getClassName() === "Scene" ? (this._scene = r || gr.LastCreatedScene, this._engine = this._scene.getEngine(), this.uniqueId = this._scene.getUniqueId(), this._scene.particleSystems.push(this)) : (this._engine = r, this.defaultProjectionMatrix = he.PerspectiveFovLH(0.8, 1, 0.1, 100, this._engine.isNDCHalfZRange)), this._engine.getCaps().supportComputeShaders) { if (!Jo("BABYLON.ComputeShaderParticleSystem")) throw new Error("The ComputeShaderParticleSystem class is not available! Make sure you have imported it."); this._platform = new (Jo("BABYLON.ComputeShaderParticleSystem"))(this, this._engine); } else { if (!Jo("BABYLON.WebGL2ParticleSystem")) throw new Error("The WebGL2ParticleSystem class is not available! Make sure you have imported it."); this._platform = new (Jo("BABYLON.WebGL2ParticleSystem"))(this, this._engine); } this._customWrappers = { 0: new zo(this._engine) }, this._customWrappers[0].effect = n, this._drawWrappers = { 0: new zo(this._engine) }, this._drawWrappers[0].drawContext && (this._drawWrappers[0].drawContext.useInstancing = !0), this._createIndexBuffer(), this._attachImageProcessingConfiguration(null), t = t ?? {}, t.randomTextureSize || delete t.randomTextureSize; const s = Object.assign({ capacity: 5e4, randomTextureSize: this._engine.getCaps().maxTextureSize }, t), a = t; isFinite(a) && (s.capacity = a), this._capacity = s.capacity, this._maxActiveParticleCount = s.capacity, this._currentActiveCount = 0, this._isAnimationSheetEnabled = i, this.particleEmitterType = new AD(); const f = Math.min(this._engine.getCaps().maxTextureSize, s.randomTextureSize); let o = []; for (let d = 0; d < f; ++d) o.push(Math.random()), o.push(Math.random()), o.push(Math.random()), o.push(Math.random()); this._randomTexture = new Bo(new Float32Array(o), f, 1, 5, r, !1, !1, 1, 1), this._randomTexture.name = "GPUParticleSystem_random1", this._randomTexture.wrapU = 1, this._randomTexture.wrapV = 1, o = []; for (let d = 0; d < f; ++d) o.push(Math.random()), o.push(Math.random()), o.push(Math.random()), o.push(Math.random()); this._randomTexture2 = new Bo(new Float32Array(o), f, 1, 5, r, !1, !1, 1, 1), this._randomTexture2.name = "GPUParticleSystem_random2", this._randomTexture2.wrapU = 1, this._randomTexture2.wrapV = 1, this._randomTextureSize = f; } _reset() { this._releaseBuffers(); } _createVertexBuffers(e, t, r) { const n = {}; n.position = t.createVertexBuffer("position", 0, 3, this._attributesStrideSize, !0); let i = 3; n.age = t.createVertexBuffer("age", i, 1, this._attributesStrideSize, !0), i += 1, n.size = t.createVertexBuffer("size", i, 3, this._attributesStrideSize, !0), i += 3, n.life = t.createVertexBuffer("life", i, 1, this._attributesStrideSize, !0), i += 1, i += 4, this.billboardMode === ti.BILLBOARDMODE_STRETCHED && (n.direction = t.createVertexBuffer("direction", i, 3, this._attributesStrideSize, !0)), i += 3, this._platform.alignDataInBuffer && (i += 1), this.particleEmitterType instanceof dD && (i += 3, this._platform.alignDataInBuffer && (i += 1)), this._colorGradientsTexture || (n.color = t.createVertexBuffer("color", i, 4, this._attributesStrideSize, !0), i += 4), this._isBillboardBased || (n.initialDirection = t.createVertexBuffer("initialDirection", i, 3, this._attributesStrideSize, !0), i += 3, this._platform.alignDataInBuffer && (i += 1)), this.noiseTexture && (n.noiseCoordinates1 = t.createVertexBuffer("noiseCoordinates1", i, 3, this._attributesStrideSize, !0), i += 3, this._platform.alignDataInBuffer && (i += 1), n.noiseCoordinates2 = t.createVertexBuffer("noiseCoordinates2", i, 3, this._attributesStrideSize, !0), i += 3, this._platform.alignDataInBuffer && (i += 1)), n.angle = t.createVertexBuffer("angle", i, 1, this._attributesStrideSize, !0), this._angularSpeedGradientsTexture ? i++ : i += 2, this._isAnimationSheetEnabled && (n.cellIndex = t.createVertexBuffer("cellIndex", i, 1, this._attributesStrideSize, !0), i += 1, this.spriteRandomStartCell && (n.cellStartOffset = t.createVertexBuffer("cellStartOffset", i, 1, this._attributesStrideSize, !0), i += 1)), n.offset = r.createVertexBuffer("offset", 0, 2), n.uv = r.createVertexBuffer("uv", 2, 2), this._renderVertexBuffers.push(n), this._platform.createVertexBuffers(e, n), this.resetDrawCache(); } _initialize(e = !1) { if (this._buffer0 && !e) return; const t = this._engine, r = []; this._attributesStrideSize = 21, this._targetIndex = 0, this._platform.alignDataInBuffer && (this._attributesStrideSize += 1), this.particleEmitterType instanceof dD && (this._attributesStrideSize += 3, this._platform.alignDataInBuffer && (this._attributesStrideSize += 1)), this.isBillboardBased || (this._attributesStrideSize += 3, this._platform.alignDataInBuffer && (this._attributesStrideSize += 1)), this._colorGradientsTexture && (this._attributesStrideSize -= 4), this._angularSpeedGradientsTexture && (this._attributesStrideSize -= 1), this._isAnimationSheetEnabled && (this._attributesStrideSize += 1, this.spriteRandomStartCell && (this._attributesStrideSize += 1)), this.noiseTexture && (this._attributesStrideSize += 6, this._platform.alignDataInBuffer && (this._attributesStrideSize += 2)), this._platform.alignDataInBuffer && (this._attributesStrideSize += 3 - (this._attributesStrideSize + 3 & 3)); const n = this.particleEmitterType instanceof dD, i = ue.Vector3[0]; let s = 0; for (let d = 0; d < this._capacity; d++) if (r.push(0), r.push(0), r.push(0), r.push(0), r.push(0), r.push(0), r.push(0), r.push(0), r.push(Math.random()), r.push(Math.random()), r.push(Math.random()), r.push(Math.random()), n ? (this.particleEmitterType.particleDestinationGenerator(d, null, i), r.push(i.x), r.push(i.y), r.push(i.z)) : (r.push(0), r.push(0), r.push(0)), this._platform.alignDataInBuffer && r.push(0), s += 16, n && (this.particleEmitterType.particlePositionGenerator(d, null, i), r.push(i.x), r.push(i.y), r.push(i.z), this._platform.alignDataInBuffer && r.push(0), s += 4), this._colorGradientsTexture || (r.push(0), r.push(0), r.push(0), r.push(0), s += 4), this.isBillboardBased || (r.push(0), r.push(0), r.push(0), this._platform.alignDataInBuffer && r.push(0), s += 4), this.noiseTexture && (r.push(Math.random()), r.push(Math.random()), r.push(Math.random()), this._platform.alignDataInBuffer && r.push(0), r.push(Math.random()), r.push(Math.random()), r.push(Math.random()), this._platform.alignDataInBuffer && r.push(0), s += 8), r.push(0), s += 1, this._angularSpeedGradientsTexture || (r.push(0), s += 1), this._isAnimationSheetEnabled && (r.push(0), s += 1, this.spriteRandomStartCell && (r.push(0), s += 1)), this._platform.alignDataInBuffer) { let v = 3 - (s + 3 & 3); for (s += v; v-- > 0; ) r.push(0); } const a = new Float32Array([0.5, 0.5, 1, 1, -0.5, 0.5, 0, 1, 0.5, -0.5, 1, 0, -0.5, -0.5, 0, 0]), f = this._platform.createParticleBuffer(r), o = this._platform.createParticleBuffer(r); this._buffer0 = new P9(t, f, !1, this._attributesStrideSize), this._buffer1 = new P9(t, o, !1, this._attributesStrideSize), this._spriteBuffer = new P9(t, a, !1, 4), this._renderVertexBuffers = [], this._createVertexBuffers(this._buffer0, this._buffer1, this._spriteBuffer), this._createVertexBuffers(this._buffer1, this._buffer0, this._spriteBuffer), this._sourceBuffer = this._buffer0, this._targetBuffer = this._buffer1; } /** @internal */ _recreateUpdateEffect() { this._createColorGradientTexture(), this._createSizeGradientTexture(), this._createAngularSpeedGradientTexture(), this._createVelocityGradientTexture(), this._createLimitVelocityGradientTexture(), this._createDragGradientTexture(); let e = this.particleEmitterType ? this.particleEmitterType.getEffectDefines() : ""; return this._isBillboardBased && (e += ` #define BILLBOARD`), this._colorGradientsTexture && (e += ` #define COLORGRADIENTS`), this._sizeGradientsTexture && (e += ` #define SIZEGRADIENTS`), this._angularSpeedGradientsTexture && (e += ` #define ANGULARSPEEDGRADIENTS`), this._velocityGradientsTexture && (e += ` #define VELOCITYGRADIENTS`), this._limitVelocityGradientsTexture && (e += ` #define LIMITVELOCITYGRADIENTS`), this._dragGradientsTexture && (e += ` #define DRAGGRADIENTS`), this.isAnimationSheetEnabled && (e += ` #define ANIMATESHEET`, this.spriteRandomStartCell && (e += ` #define ANIMATESHEETRANDOMSTART`)), this.noiseTexture && (e += ` #define NOISE`), this.isLocal && (e += ` #define LOCAL`), this._platform.isUpdateBufferCreated() && this._cachedUpdateDefines === e ? !0 : (this._cachedUpdateDefines = e, this._updateBuffer = this._platform.createUpdateBuffer(e), this._platform.isUpdateBufferReady()); } /** * @internal */ _getWrapper(e) { const t = this._getCustomDrawWrapper(e); if (t != null && t.effect) return t; const r = []; this.fillDefines(r, e); let n = this._drawWrappers[e]; n || (n = new zo(this._engine), n.drawContext && (n.drawContext.useInstancing = !0), this._drawWrappers[e] = n); const i = r.join(` `); if (n.defines !== i) { const s = [], a = [], f = []; this.fillUniformsAttributesAndSamplerNames(a, s, f), n.setEffect(this._engine.createEffect("gpuRenderParticles", s, a, f, i), i); } return n; } /** * @internal */ static _GetAttributeNamesOrOptions(e = !1, t = !1, r = !1, n = !1) { const i = [J.PositionKind, "age", "life", "size", "angle"]; return e || i.push(J.ColorKind), t && i.push("cellIndex"), r || i.push("initialDirection"), n && i.push("direction"), i.push("offset", J.UVKind), i; } /** * @internal */ static _GetEffectCreationOptions(e = !1, t = !1) { const r = ["emitterWM", "worldOffset", "view", "projection", "colorDead", "invView", "translationPivot", "eyePosition"]; return Mf(r), e && r.push("sheetInfos"), t && r.push("logarithmicDepthConstant"), r; } /** * Fill the defines array according to the current settings of the particle system * @param defines Array to be updated * @param blendMode blend mode to take into account when updating the array */ fillDefines(e, t = 0) { if (this._scene && xq(this, this._scene, e), t === ti.BLENDMODE_MULTIPLY && e.push("#define BLENDMULTIPLYMODE"), this.isLocal && e.push("#define LOCAL"), this.useLogarithmicDepth && e.push("#define LOGARITHMICDEPTH"), this._isBillboardBased) switch (e.push("#define BILLBOARD"), this.billboardMode) { case ti.BILLBOARDMODE_Y: e.push("#define BILLBOARDY"); break; case ti.BILLBOARDMODE_STRETCHED: e.push("#define BILLBOARDSTRETCHED"); break; case ti.BILLBOARDMODE_ALL: e.push("#define BILLBOARDMODE_ALL"); break; } this._colorGradientsTexture && e.push("#define COLORGRADIENTS"), this.isAnimationSheetEnabled && e.push("#define ANIMATESHEET"), this._imageProcessingConfiguration && (this._imageProcessingConfiguration.prepareDefines(this._imageProcessingConfigurationDefines), e.push("" + this._imageProcessingConfigurationDefines.toString())); } /** * Fill the uniforms, attributes and samplers arrays according to the current settings of the particle system * @param uniforms Uniforms array to fill * @param attributes Attributes array to fill * @param samplers Samplers array to fill */ fillUniformsAttributesAndSamplerNames(e, t, r) { t.push(...bc._GetAttributeNamesOrOptions(!!this._colorGradientsTexture, this._isAnimationSheetEnabled, this._isBillboardBased, this._isBillboardBased && this.billboardMode === ti.BILLBOARDMODE_STRETCHED)), e.push(...bc._GetEffectCreationOptions(this._isAnimationSheetEnabled, this.useLogarithmicDepth)), r.push("diffuseSampler", "colorGradientSampler"), this._imageProcessingConfiguration && (Ui.PrepareUniforms(e, this._imageProcessingConfigurationDefines), Ui.PrepareSamplers(r, this._imageProcessingConfigurationDefines)); } /** * Animates the particle system for the current frame by emitting new particles and or animating the living ones. * @param preWarm defines if we are in the pre-warmimg phase */ animate(e = !1) { var t; this._timeDelta = this.updateSpeed * (e ? this.preWarmStepOffset : ((t = this._scene) === null || t === void 0 ? void 0 : t.getAnimationRatio()) || 1), this._actualFrame += this._timeDelta, this._stopped || this.targetStopDuration && this._actualFrame >= this.targetStopDuration && this.stop(), this.updateInAnimate && this._update(); } _createFactorGradientTexture(e, t) { const r = this[t]; if (!e || !e.length || r) return; const n = new Float32Array(this._rawTextureWidth); for (let i = 0; i < this._rawTextureWidth; i++) { const s = i / this._rawTextureWidth; kl.GetCurrentGradient(s, e, (a, f, o) => { n[i] = Xt.Lerp(a.factor1, f.factor1, o); }); } this[t] = Bo.CreateRTexture(n, this._rawTextureWidth, 1, this._scene || this._engine, !1, !1, 1), this[t].name = t.substring(1); } _createSizeGradientTexture() { this._createFactorGradientTexture(this._sizeGradients, "_sizeGradientsTexture"); } _createAngularSpeedGradientTexture() { this._createFactorGradientTexture(this._angularSpeedGradients, "_angularSpeedGradientsTexture"); } _createVelocityGradientTexture() { this._createFactorGradientTexture(this._velocityGradients, "_velocityGradientsTexture"); } _createLimitVelocityGradientTexture() { this._createFactorGradientTexture(this._limitVelocityGradients, "_limitVelocityGradientsTexture"); } _createDragGradientTexture() { this._createFactorGradientTexture(this._dragGradients, "_dragGradientsTexture"); } _createColorGradientTexture() { if (!this._colorGradients || !this._colorGradients.length || this._colorGradientsTexture) return; const e = new Uint8Array(this._rawTextureWidth * 4), t = Hs.Color4[0]; for (let r = 0; r < this._rawTextureWidth; r++) { const n = r / this._rawTextureWidth; kl.GetCurrentGradient(n, this._colorGradients, (i, s, a) => { xt.LerpToRef(i.color1, s.color1, a, t), e[r * 4] = t.r * 255, e[r * 4 + 1] = t.g * 255, e[r * 4 + 2] = t.b * 255, e[r * 4 + 3] = t.a * 255; }); } this._colorGradientsTexture = Bo.CreateRGBATexture(e, this._rawTextureWidth, 1, this._scene, !1, !1, 1), this._colorGradientsTexture.name = "colorGradients"; } _render(e, t) { var r, n, i, s, a; const f = this._getWrapper(e), o = f.effect; this._engine.enableEffect(f); const d = ((r = this._scene) === null || r === void 0 ? void 0 : r.getViewMatrix()) || he.IdentityReadOnly; if (o.setMatrix("view", d), o.setMatrix("projection", (n = this.defaultProjectionMatrix) !== null && n !== void 0 ? n : this._scene.getProjectionMatrix()), o.setTexture("diffuseSampler", this.particleTexture), o.setVector2("translationPivot", this.translationPivot), o.setVector3("worldOffset", this.worldOffset), this.isLocal && o.setMatrix("emitterWM", t), this._colorGradientsTexture ? o.setTexture("colorGradientSampler", this._colorGradientsTexture) : o.setDirectColor4("colorDead", this.colorDead), this._isAnimationSheetEnabled && this.particleTexture) { const u = this.particleTexture.getBaseSize(); o.setFloat3("sheetInfos", this.spriteCellWidth / u.width, this.spriteCellHeight / u.height, u.width / this.spriteCellWidth); } if (this._isBillboardBased && this._scene) { const u = this._scene.activeCamera; o.setVector3("eyePosition", u.globalPosition); } const v = o.defines; if (this._scene && Df(o, this, this._scene), v.indexOf("#define BILLBOARDMODE_ALL") >= 0) { const u = d.clone(); u.invert(), o.setMatrix("invView", u); } switch (this.useLogarithmicDepth && this._scene && Ye.BindLogDepth(v, o, this._scene), this._imageProcessingConfiguration && !this._imageProcessingConfiguration.applyByPostProcess && this._imageProcessingConfiguration.bind(o), e) { case ti.BLENDMODE_ADD: this._engine.setAlphaMode(1); break; case ti.BLENDMODE_ONEONE: this._engine.setAlphaMode(6); break; case ti.BLENDMODE_STANDARD: this._engine.setAlphaMode(2); break; case ti.BLENDMODE_MULTIPLY: this._engine.setAlphaMode(4); break; } return this._platform.bindDrawBuffers(this._targetIndex, o, !((i = this._scene) === null || i === void 0) && i.forceWireframe ? this._linesIndexBufferUseInstancing : null), this._onBeforeDrawParticlesObservable && this._onBeforeDrawParticlesObservable.notifyObservers(o), !((s = this._scene) === null || s === void 0) && s.forceWireframe ? this._engine.drawElementsType(6, 0, 10, this._currentActiveCount) : this._engine.drawArraysType(7, 0, 4, this._currentActiveCount), this._engine.setAlphaMode(0), !((a = this._scene) === null || a === void 0) && a.forceWireframe && this._engine.unbindInstanceAttributes(), this._currentActiveCount; } /** @internal */ _update(e) { if (!this.emitter || !this._targetBuffer || !this._recreateUpdateEffect()) return; if (!e) if (this.emitter.position) e = this.emitter.getWorldMatrix(); else { const r = this.emitter; e = ue.Matrix[0], he.TranslationToRef(r.x, r.y, r.z, e); } this._platform.preUpdateParticleBuffer(), this._updateBuffer.setFloat("currentCount", this._currentActiveCount), this._updateBuffer.setFloat("timeDelta", this._timeDelta), this._updateBuffer.setFloat("stopFactor", this._stopped ? 0 : 1), this._updateBuffer.setInt("randomTextureSize", this._randomTextureSize), this._updateBuffer.setFloat2("lifeTime", this.minLifeTime, this.maxLifeTime), this._updateBuffer.setFloat2("emitPower", this.minEmitPower, this.maxEmitPower), this._colorGradientsTexture || (this._updateBuffer.setDirectColor4("color1", this.color1), this._updateBuffer.setDirectColor4("color2", this.color2)), this._updateBuffer.setFloat2("sizeRange", this.minSize, this.maxSize), this._updateBuffer.setFloat4("scaleRange", this.minScaleX, this.maxScaleX, this.minScaleY, this.maxScaleY), this._updateBuffer.setFloat4("angleRange", this.minAngularSpeed, this.maxAngularSpeed, this.minInitialRotation, this.maxInitialRotation), this._updateBuffer.setVector3("gravity", this.gravity), this._limitVelocityGradientsTexture && this._updateBuffer.setFloat("limitVelocityDamping", this.limitVelocityDamping), this.particleEmitterType && this.particleEmitterType.applyToShader(this._updateBuffer), this._isAnimationSheetEnabled && this._updateBuffer.setFloat4("cellInfos", this.startSpriteCellID, this.endSpriteCellID, this.spriteCellChangeSpeed, this.spriteCellLoop ? 1 : 0), this.noiseTexture && this._updateBuffer.setVector3("noiseStrength", this.noiseStrength), this.isLocal || this._updateBuffer.setMatrix("emitterWM", e), this._platform.updateParticleBuffer(this._targetIndex, this._targetBuffer, this._currentActiveCount), this._targetIndex++, this._targetIndex === 2 && (this._targetIndex = 0); const t = this._sourceBuffer; this._sourceBuffer = this._targetBuffer, this._targetBuffer = t; } /** * Renders the particle system in its current state * @param preWarm defines if the system should only update the particles but not render them * @param forceUpdateOnly if true, force to only update the particles and never display them (meaning, even if preWarm=false, when forceUpdateOnly=true the particles won't be displayed) * @returns the current number of particles */ render(e = !1, t = !1) { if (!this._started || !this.isReady()) return 0; if (!e && this._scene) { if (!this._preWarmDone && this.preWarmCycles) { for (let s = 0; s < this.preWarmCycles; s++) this.animate(!0), this.render(!0, !0); this._preWarmDone = !0; } if (this._currentRenderId === this._scene.getRenderId() && (!this._scene.activeCamera || this._scene.activeCamera && this._currentRenderingCameraUniqueId === this._scene.activeCamera.uniqueId)) return 0; this._currentRenderId = this._scene.getRenderId(), this._scene.activeCamera && (this._currentRenderingCameraUniqueId = this._scene.activeCamera.uniqueId); } if (this._initialize(), this._accumulatedCount += this.emitRate * this._timeDelta, this._accumulatedCount > 1) { const s = this._accumulatedCount | 0; this._accumulatedCount -= s, this._currentActiveCount += s; } if (this._currentActiveCount = Math.min(this._maxActiveParticleCount, this._currentActiveCount), !this._currentActiveCount) return 0; let r; if (this.emitter.position) r = this.emitter.getWorldMatrix(); else { const s = this.emitter; r = ue.Matrix[0], he.TranslationToRef(s.x, s.y, s.z, r); } const n = this._engine; this.updateInAnimate || this._update(r); let i = 0; return !e && !t && (n.setState(!1), this.forceDepthWrite && n.setDepthWrite(!0), this.blendMode === ti.BLENDMODE_MULTIPLYADD ? i = this._render(ti.BLENDMODE_MULTIPLY, r) + this._render(ti.BLENDMODE_ADD, r) : i = this._render(this.blendMode, r), this._engine.setAlphaMode(0)), i; } /** * Rebuilds the particle system */ rebuild() { const e = () => { !this._recreateUpdateEffect() || !this._platform.isUpdateBufferReady() ? setTimeout(e, 10) : this._initialize(!0); }; this._createIndexBuffer(), this._cachedUpdateDefines = "", this._platform.contextLost(), e(); } _releaseBuffers() { this._buffer0 && (this._buffer0.dispose(), this._buffer0 = null), this._buffer1 && (this._buffer1.dispose(), this._buffer1 = null), this._spriteBuffer && (this._spriteBuffer.dispose(), this._spriteBuffer = null), this._platform.releaseBuffers(); } /** * Disposes the particle system and free the associated resources * @param disposeTexture defines if the particule texture must be disposed as well (true by default) */ dispose(e = !0) { for (const t in this._drawWrappers) this._drawWrappers[t].dispose(); if (this._drawWrappers = {}, this._scene) { const t = this._scene.particleSystems.indexOf(this); t > -1 && this._scene.particleSystems.splice(t, 1); } this._releaseBuffers(), this._platform.releaseVertexBuffers(); for (let t = 0; t < this._renderVertexBuffers.length; ++t) { const r = this._renderVertexBuffers[t]; for (const n in r) r[n].dispose(); } this._renderVertexBuffers = [], this._colorGradientsTexture && (this._colorGradientsTexture.dispose(), this._colorGradientsTexture = null), this._sizeGradientsTexture && (this._sizeGradientsTexture.dispose(), this._sizeGradientsTexture = null), this._angularSpeedGradientsTexture && (this._angularSpeedGradientsTexture.dispose(), this._angularSpeedGradientsTexture = null), this._velocityGradientsTexture && (this._velocityGradientsTexture.dispose(), this._velocityGradientsTexture = null), this._limitVelocityGradientsTexture && (this._limitVelocityGradientsTexture.dispose(), this._limitVelocityGradientsTexture = null), this._dragGradientsTexture && (this._dragGradientsTexture.dispose(), this._dragGradientsTexture = null), this._randomTexture && (this._randomTexture.dispose(), this._randomTexture = null), this._randomTexture2 && (this._randomTexture2.dispose(), this._randomTexture2 = null), e && this.particleTexture && (this.particleTexture.dispose(), this.particleTexture = null), e && this.noiseTexture && (this.noiseTexture.dispose(), this.noiseTexture = null), this.onStoppedObservable.clear(), this.onDisposeObservable.notifyObservers(this), this.onDisposeObservable.clear(); } /** * Clones the particle system. * @param name The name of the cloned object * @param newEmitter The new emitter to use * @param cloneTexture Also clone the textures if true * @returns the cloned particle system */ clone(e, t, r = !1) { const n = Object.assign({}, this._customWrappers); let i = null; const s = this._engine; if (s.createEffectForParticles && this.customShader != null) { i = this.customShader; const o = i.shaderOptions.defines.length > 0 ? i.shaderOptions.defines.join(` `) : ""; n[0] = s.createEffectForParticles(i.shaderPath.fragmentElement, i.shaderOptions.uniforms, i.shaderOptions.samplers, o, void 0, void 0, void 0, this); } const a = this.serialize(r), f = bc.Parse(a, this._scene || this._engine, this._rootUrl); return f.name = e, f.customShader = i, f._customWrappers = n, t === void 0 && (t = this.emitter), this.noiseTexture && (f.noiseTexture = this.noiseTexture.clone()), f.emitter = t, f; } /** * Serializes the particle system to a JSON object * @param serializeTexture defines if the texture must be serialized as well * @returns the JSON object */ serialize(e = !1) { const t = {}; return ti._Serialize(t, this, e), t.activeParticleCount = this.activeParticleCount, t.randomTextureSize = this._randomTextureSize, t.customShader = this.customShader, t; } /** * Parses a JSON object to create a GPU particle system. * @param parsedParticleSystem The JSON object to parse * @param sceneOrEngine The scene or the engine to create the particle system in * @param rootUrl The root url to use to load external dependencies like texture * @param doNotStart Ignore the preventAutoStart attribute and does not start * @param capacity defines the system capacity (if null or undefined the sotred capacity will be used) * @returns the parsed GPU particle system */ static Parse(e, t, r, n = !1, i) { const s = e.name; let a, f; t instanceof hr ? a = t : (f = t, a = f.getEngine()); const o = new bc(s, { capacity: i || e.capacity, randomTextureSize: e.randomTextureSize }, t, null, e.isAnimationSheetEnabled); if (o._rootUrl = r, e.customShader && a.createEffectForParticles) { const d = e.customShader, v = d.shaderOptions.defines.length > 0 ? d.shaderOptions.defines.join(` `) : "", u = a.createEffectForParticles(d.shaderPath.fragmentElement, d.shaderOptions.uniforms, d.shaderOptions.samplers, v, void 0, void 0, void 0, o); o.setCustomEffect(u, 0), o.customShader = d; } return e.id && (o.id = e.id), e.activeParticleCount && (o.activeParticleCount = e.activeParticleCount), ti._Parse(e, o, t, r), e.preventAutoStart && (o.preventAutoStart = e.preventAutoStart), !n && !o.preventAutoStart && o.start(), o; } } class bq { constructor() { this._emitterNodeIsOwned = !0, this.systems = []; } /** * Gets or sets the emitter node used with this set */ get emitterNode() { return this._emitterNode; } set emitterNode(e) { this._emitterNodeIsOwned && this._emitterNode && (this._emitterNode.dispose && this._emitterNode.dispose(), this._emitterNodeIsOwned = !1); for (const t of this.systems) t.emitter = e; this._emitterNode = e; } /** * Creates a new emitter mesh as a sphere * @param options defines the options used to create the sphere * @param options.diameter * @param options.segments * @param options.color * @param renderingGroupId defines the renderingGroupId to use for the sphere * @param scene defines the hosting scene */ setEmitterAsSphere(e, t, r) { this._emitterNodeIsOwned && this._emitterNode && this._emitterNode.dispose && this._emitterNode.dispose(), this._emitterNodeIsOwned = !0, this._emitterCreationOptions = { kind: "Sphere", options: e, renderingGroupId: t }; const n = UA("emitterSphere", { diameter: e.diameter, segments: e.segments }, r); n.renderingGroupId = t; const i = new Wt("emitterSphereMaterial", r); i.emissiveColor = e.color, n.material = i; for (const s of this.systems) s.emitter = n; this._emitterNode = n; } /** * Starts all particle systems of the set * @param emitter defines an optional mesh to use as emitter for the particle systems */ start(e) { for (const t of this.systems) e && (t.emitter = e), t.start(); } /** * Release all associated resources */ dispose() { for (const e of this.systems) e.dispose(); this.systems.length = 0, this._emitterNode && (this._emitterNode.dispose && this._emitterNode.dispose(), this._emitterNode = null); } /** * Serialize the set into a JSON compatible object * @param serializeTexture defines if the texture must be serialized as well * @returns a JSON compatible representation of the set */ serialize(e = !1) { const t = {}; t.systems = []; for (const r of this.systems) t.systems.push(r.serialize(e)); return this._emitterNode && (t.emitter = this._emitterCreationOptions), t; } /** * Parse a new ParticleSystemSet from a serialized source * @param data defines a JSON compatible representation of the set * @param scene defines the hosting scene * @param gpu defines if we want GPU particles or CPU particles * @param capacity defines the system capacity (if null or undefined the sotred capacity will be used) * @returns a new ParticleSystemSet */ static Parse(e, t, r = !1, n) { const i = new bq(), s = this.BaseAssetsUrl + "/textures/"; t = t || gr.LastCreatedScene; for (const a of e.systems) i.systems.push(r ? bc.Parse(a, t, s, !0, n) : ti.Parse(a, t, s, !0, n)); if (e.emitter) { const a = e.emitter.options; switch (e.emitter.kind) { case "Sphere": i.setEmitterAsSphere({ diameter: a.diameter, segments: a.segments, color: Ne.FromArray(a.color) }, e.emitter.renderingGroupId, t); break; } } return i; } } bq.BaseAssetsUrl = "https://assets.babylonjs.com/particles"; class Tm { /** * Create a default particle system that you can tweak * @param emitter defines the emitter to use * @param capacity defines the system capacity (default is 500 particles) * @param scene defines the hosting scene * @param useGPU defines if a GPUParticleSystem must be created (default is false) * @returns the new Particle system */ static CreateDefault(e, t = 500, r, n = !1) { let i; return n ? i = new bc("default system", { capacity: t }, r) : i = new ti("default system", t, r), i.emitter = e, i.particleTexture = new We("https://assets.babylonjs.com/textures/flare.png", i.getScene()), i.createConeEmitter(0.1, Math.PI / 4), i.color1 = new xt(1, 1, 1, 1), i.color2 = new xt(1, 1, 1, 1), i.colorDead = new xt(1, 1, 1, 0), i.minSize = 0.1, i.maxSize = 0.1, i.minEmitPower = 2, i.maxEmitPower = 2, i.updateSpeed = 1 / 60, i.emitRate = 30, i; } /** * This is the main static method (one-liner) of this helper to create different particle systems * @param type This string represents the type to the particle system to create * @param scene The scene where the particle system should live * @param gpu If the system will use gpu * @param capacity defines the system capacity (if null or undefined the sotred capacity will be used) * @returns the ParticleSystemSet created */ static CreateAsync(e, t, r = !1, n) { t || (t = gr.LastCreatedScene); const i = {}; return t.addPendingData(i), new Promise((s, a) => { if (r && !bc.IsSupported) return t.removePendingData(i), a("Particle system with GPU is not supported."); ye.LoadFile(`${Tm.BaseAssetsUrl}/systems/${e}.json`, (f) => { t.removePendingData(i); const o = JSON.parse(f.toString()); return s(bq.Parse(o, t, r, n)); }, void 0, void 0, void 0, () => (t.removePendingData(i), a(`An error occurred with the creation of your particle system. Check if your type '${e}' exists.`))); }); } /** * Static function used to export a particle system to a ParticleSystemSet variable. * Please note that the emitter shape is not exported * @param systems defines the particle systems to export * @returns the created particle system set */ static ExportSet(e) { const t = new bq(); for (const r of e) t.systems.push(r); return t; } /** * Creates a particle system from a snippet saved in a remote file * @param name defines the name of the particle system to create (can be null or empty to use the one from the json data) * @param url defines the url to load from * @param scene defines the hosting scene * @param gpu If the system will use gpu * @param rootUrl defines the root URL to use to load textures and relative dependencies * @param capacity defines the system capacity (if null or undefined the sotred capacity will be used) * @returns a promise that will resolve to the new particle system */ static ParseFromFileAsync(e, t, r, n = !1, i = "", s) { return new Promise((a, f) => { const o = new ho(); o.addEventListener("readystatechange", () => { if (o.readyState == 4) if (o.status == 200) { const d = JSON.parse(o.responseText); let v; n ? v = bc.Parse(d, r, i, !1, s) : v = ti.Parse(d, r, i, !1, s), e && (v.name = e), a(v); } else f("Unable to load the particle system"); }), o.open("GET", t), o.send(); }); } /** * Creates a particle system from a snippet saved by the particle system editor * @param snippetId defines the snippet to load (can be set to _BLANK to create a default one) * @param scene defines the hosting scene * @param gpu If the system will use gpu * @param rootUrl defines the root URL to use to load textures and relative dependencies * @param capacity defines the system capacity (if null or undefined the sotred capacity will be used) * @returns a promise that will resolve to the new particle system */ static ParseFromSnippetAsync(e, t, r = !1, n = "", i) { if (e === "_BLANK") { const s = this.CreateDefault(null); return s.start(), Promise.resolve(s); } return new Promise((s, a) => { const f = new ho(); f.addEventListener("readystatechange", () => { if (f.readyState == 4) if (f.status == 200) { const o = JSON.parse(JSON.parse(f.responseText).jsonPayload), d = JSON.parse(o.particleSystem); let v; r ? v = bc.Parse(d, t, n, !1, i) : v = ti.Parse(d, t, n, !1, i), v.snippetId = e, s(v); } else a("Unable to load the snippet " + e); }), f.open("GET", this.SnippetUrl + "/" + e.replace(/#/g, "/")), f.send(); }); } } Tm.BaseAssetsUrl = bq.BaseAssetsUrl; Tm.SnippetUrl = "https://snippet.babylonjs.com"; Tm.CreateFromSnippetAsync = Tm.ParseFromSnippetAsync; J1.AddParser(Ot.NAME_PARTICLESYSTEM, (A, e, t, r) => { const n = J1.GetIndividualParser(Ot.NAME_PARTICLESYSTEM); if (n && A.particleSystems !== void 0 && A.particleSystems !== null) for (let i = 0, s = A.particleSystems.length; i < s; i++) { const a = A.particleSystems[i]; t.particleSystems.push(n(a, e, r)); } }); J1.AddIndividualParser(Ot.NAME_PARTICLESYSTEM, (A, e, t) => A.activeParticleCount ? bc.Parse(A, e, t) : ti.Parse(A, e, t)); Ge.prototype.createEffectForParticles = function(A, e = [], t = [], r = "", n, i, s, a) { var f; let o = [], d = []; const v = []; return a ? a.fillUniformsAttributesAndSamplerNames(d, o, v) : (o = ti._GetAttributeNamesOrOptions(), d = ti._GetEffectCreationOptions()), r.indexOf(" BILLBOARD") === -1 && (r += ` #define BILLBOARD `), a != null && a.isAnimationSheetEnabled && r.indexOf(" ANIMATESHEET") === -1 && (r += ` #define ANIMATESHEET `), t.indexOf("diffuseSampler") === -1 && t.push("diffuseSampler"), this.createEffect({ vertex: (f = a == null ? void 0 : a.vertexShaderName) !== null && f !== void 0 ? f : "particles", fragmentElement: A }, o, d.concat(e), v.concat(t), r, n, i, s); }; Ee.prototype.getEmittedParticleSystems = function() { const A = []; for (let e = 0; e < this.getScene().particleSystems.length; e++) { const t = this.getScene().particleSystems[e]; t.emitter === this && A.push(t); } return A; }; Ee.prototype.getHierarchyEmittedParticleSystems = function() { const A = [], e = this.getDescendants(); e.push(this); for (let t = 0; t < this.getScene().particleSystems.length; t++) { const r = this.getScene().particleSystems[t], n = r.emitter; n.position && e.indexOf(n) !== -1 && A.push(r); } return A; }; class bF { /** * Particle BoundingInfo object * @returns a BoundingInfo */ getBoundingInfo() { return this._boundingInfo; } /** * Returns true if there is already a bounding info */ get hasBoundingInfo() { return this._boundingInfo !== null; } /** * Creates a Solid Particle object. * Don't create particles manually, use instead the Solid Particle System internal tools like _addParticle() * @param particleIndex (integer) is the particle index in the Solid Particle System pool. * @param particleId (integer) is the particle identifier. Unless some particles are removed from the SPS, it's the same value than the particle idx. * @param positionIndex (integer) is the starting index of the particle vertices in the SPS "positions" array. * @param indiceIndex (integer) is the starting index of the particle indices in the SPS "indices" array. * @param model (ModelShape) is a reference to the model shape on what the particle is designed. * @param shapeId (integer) is the model shape identifier in the SPS. * @param idxInShape (integer) is the index of the particle in the current model (ex: the 10th box of addShape(box, 30)) * @param sps defines the sps it is associated to * @param modelBoundingInfo is the reference to the model BoundingInfo used for intersection computations. * @param materialIndex is the particle material identifier (integer) when the MultiMaterials are enabled in the SPS. */ constructor(e, t, r, n, i, s, a, f, o = null, d = null) { this.idx = 0, this.id = 0, this.color = new xt(1, 1, 1, 1), this.position = S.Zero(), this.rotation = S.Zero(), this.scaling = S.One(), this.uvs = new Ir(0, 0, 1, 1), this.velocity = S.Zero(), this.pivot = S.Zero(), this.translateFromPivot = !1, this.alive = !0, this.isVisible = !0, this._pos = 0, this._ind = 0, this.shapeId = 0, this.idxInShape = 0, this._stillInvisible = !1, this._rotationMatrix = [1, 0, 0, 0, 1, 0, 0, 0, 1], this.parentId = null, this.materialIndex = null, this.props = null, this.cullingStrategy = jn.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY, this._globalPosition = S.Zero(), this.idx = e, this.id = t, this._pos = r, this._ind = n, this._model = i, this.shapeId = s, this.idxInShape = a, this._sps = f, o && (this._modelBoundingInfo = o, this._boundingInfo = new Md(o.minimum, o.maximum)), d !== null && (this.materialIndex = d); } /** * Copies the particle property values into the existing target : position, rotation, scaling, uvs, colors, pivot, parent, visibility, alive * @param target the particle target * @returns the current particle */ copyToRef(e) { return e.position.copyFrom(this.position), e.rotation.copyFrom(this.rotation), this.rotationQuaternion && (e.rotationQuaternion ? e.rotationQuaternion.copyFrom(this.rotationQuaternion) : e.rotationQuaternion = this.rotationQuaternion.clone()), e.scaling.copyFrom(this.scaling), this.color && (e.color ? e.color.copyFrom(this.color) : e.color = this.color.clone()), e.uvs.copyFrom(this.uvs), e.velocity.copyFrom(this.velocity), e.pivot.copyFrom(this.pivot), e.translateFromPivot = this.translateFromPivot, e.alive = this.alive, e.isVisible = this.isVisible, e.parentId = this.parentId, e.cullingStrategy = this.cullingStrategy, this.materialIndex !== null && (e.materialIndex = this.materialIndex), this; } /** * Legacy support, changed scale to scaling */ get scale() { return this.scaling; } /** * Legacy support, changed scale to scaling */ set scale(e) { this.scaling = e; } /** * Legacy support, changed quaternion to rotationQuaternion */ get quaternion() { return this.rotationQuaternion; } /** * Legacy support, changed quaternion to rotationQuaternion */ set quaternion(e) { this.rotationQuaternion = e; } /** * Returns a boolean. True if the particle intersects another particle or another mesh, else false. * The intersection is computed on the particle bounding sphere and Axis Aligned Bounding Box (AABB) * @param target is the object (solid particle or mesh) what the intersection is computed against. * @returns true if it intersects */ intersectsMesh(e) { return !this._boundingInfo || !e.hasBoundingInfo ? !1 : this._sps._bSphereOnly ? rg.Intersects(this._boundingInfo.boundingSphere, e.getBoundingInfo().boundingSphere) : this._boundingInfo.intersects(e.getBoundingInfo(), !1); } /** * Returns `true` if the solid particle is within the frustum defined by the passed array of planes. * A particle is in the frustum if its bounding box intersects the frustum * @param frustumPlanes defines the frustum to test * @returns true if the particle is in the frustum planes */ isInFrustum(e) { return this._boundingInfo !== null && this._boundingInfo.isInFrustum(e, this.cullingStrategy); } /** * get the rotation matrix of the particle * @internal */ getRotationMatrix(e) { let t; if (this.rotationQuaternion) t = this.rotationQuaternion; else { t = ue.Quaternion[0]; const r = this.rotation; Ze.RotationYawPitchRollToRef(r.y, r.x, r.z, t); } t.toRotationMatrix(e); } } class xF { /** * Get or set the shapeId * @deprecated Please use shapeId instead */ get shapeID() { return this.shapeId; } set shapeID(e) { this.shapeId = e; } /** * Creates a ModelShape object. This is an internal simplified reference to a mesh used as for a model to replicate particles from by the SPS. * SPS internal tool, don't use it manually. * @internal */ constructor(e, t, r, n, i, s, a, f, o) { this._indicesLength = 0, this.shapeId = e, this._shape = t, this._indices = r, this._indicesLength = r.length, this._shapeUV = s, this._shapeColors = i, this._normals = n, this._positionFunction = a, this._vertexFunction = f, this._material = o; } } class zne { /** * Creates a new sorted particle * @param idx * @param ind * @param indLength * @param materialIndex */ constructor(e, t, r, n) { this.idx = 0, this.ind = 0, this.indicesLength = 0, this.sqDistance = 0, this.materialIndex = 0, this.idx = e, this.ind = t, this.indicesLength = r, this.materialIndex = n; } } class Gne { /** * Creates a new solid particle vertex */ constructor() { this.position = S.Zero(), this.color = new xt(1, 1, 1, 1), this.uv = at.Zero(); } // Getters and Setters for back-compatibility /** Vertex x coordinate */ get x() { return this.position.x; } set x(e) { this.position.x = e; } /** Vertex y coordinate */ get y() { return this.position.y; } set y(e) { this.position.y = e; } /** Vertex z coordinate */ get z() { return this.position.z; } set z(e) { this.position.z = e; } } class Jhe { /** * Creates a SPS (Solid Particle System) object. * @param name (String) is the SPS name, this will be the underlying mesh name. * @param scene (Scene) is the scene in which the SPS is added. * @param options defines the options of the sps e.g. * * updatable (optional boolean, default true) : if the SPS must be updatable or immutable. * * isPickable (optional boolean, default false) : if the solid particles must be pickable. * * enableDepthSort (optional boolean, default false) : if the solid particles must be sorted in the geometry according to their distance to the camera. * * useModelMaterial (optional boolean, default false) : if the model materials must be used to create the SPS multimaterial. This enables the multimaterial supports of the SPS. * * enableMultiMaterial (optional boolean, default false) : if the solid particles can be given different materials. * * expandable (optional boolean, default false) : if particles can still be added after the initial SPS mesh creation. * * particleIntersection (optional boolean, default false) : if the solid particle intersections must be computed. * * boundingSphereOnly (optional boolean, default false) : if the particle intersection must be computed only with the bounding sphere (no bounding box computation, so faster). * * bSphereRadiusFactor (optional float, default 1.0) : a number to multiply the bounding sphere radius by in order to reduce it for instance. * * computeBoundingBox (optional boolean, default false): if the bounding box of the entire SPS will be computed (for occlusion detection, for example). If it is false, the bounding box will be the bounding box of the first particle. * * autoFixFaceOrientation (optional boolean, default false): if the particle face orientations will be flipped for transformations that change orientation (scale (-1, 1, 1), for example) * @param options.updatable * @param options.isPickable * @param options.enableDepthSort * @param options.particleIntersection * @param options.boundingSphereOnly * @param options.bSphereRadiusFactor * @param options.expandable * @param options.useModelMaterial * @param options.enableMultiMaterial * @param options.computeBoundingBox * @param options.autoFixFaceOrientation * @example bSphereRadiusFactor = 1.0 / Math.sqrt(3.0) => the bounding sphere exactly matches a spherical mesh. */ constructor(e, t, r) { this.particles = new Array(), this.nbParticles = 0, this.billboard = !1, this.recomputeNormals = !1, this.counter = 0, this.vars = {}, this._bSphereOnly = !1, this._bSphereRadiusFactor = 1, this._positions = new Array(), this._indices = new Array(), this._normals = new Array(), this._colors = new Array(), this._uvs = new Array(), this._index = 0, this._updatable = !0, this._pickable = !1, this._isVisibilityBoxLocked = !1, this._alwaysVisible = !1, this._depthSort = !1, this._expandable = !1, this._shapeCounter = 0, this._copy = new bF(0, 0, 0, 0, null, 0, 0, this), this._color = new xt(0, 0, 0, 0), this._computeParticleColor = !0, this._computeParticleTexture = !0, this._computeParticleRotation = !0, this._computeParticleVertex = !1, this._computeBoundingBox = !1, this._autoFixFaceOrientation = !1, this._depthSortParticles = !0, this._mustUnrotateFixedNormals = !1, this._particlesIntersect = !1, this._needs32Bits = !1, this._isNotBuilt = !0, this._lastParticleId = 0, this._idxOfId = [], this._multimaterialEnabled = !1, this._useModelMaterial = !1, this._depthSortFunction = (n, i) => i.sqDistance - n.sqDistance, this._materialSortFunction = (n, i) => n.materialIndex - i.materialIndex, this._autoUpdateSubMeshes = !1, this._recomputeInvisibles = !1, this.name = e, this._scene = t || gr.LastCreatedScene, this._camera = t.activeCamera, this._pickable = r ? r.isPickable : !1, this._depthSort = r ? r.enableDepthSort : !1, this._multimaterialEnabled = r ? r.enableMultiMaterial : !1, this._useModelMaterial = r ? r.useModelMaterial : !1, this._multimaterialEnabled = this._useModelMaterial ? !0 : this._multimaterialEnabled, this._expandable = r ? r.expandable : !1, this._particlesIntersect = r ? r.particleIntersection : !1, this._bSphereOnly = r ? r.boundingSphereOnly : !1, this._bSphereRadiusFactor = r && r.bSphereRadiusFactor ? r.bSphereRadiusFactor : 1, this._computeBoundingBox = r != null && r.computeBoundingBox ? r.computeBoundingBox : !1, this._autoFixFaceOrientation = r != null && r.autoFixFaceOrientation ? r.autoFixFaceOrientation : !1, r && r.updatable !== void 0 ? this._updatable = r.updatable : this._updatable = !0, this._pickable && (this.pickedBySubMesh = [[]], this.pickedParticles = this.pickedBySubMesh[0]), (this._depthSort || this._multimaterialEnabled) && (this.depthSortedParticles = []), this._multimaterialEnabled && (this._multimaterial = new Dc(this.name + "MultiMaterial", this._scene), this._materials = [], this._materialIndexesById = {}), this._tmpVertex = new Gne(); } /** * Builds the SPS underlying mesh. Returns a standard Mesh. * If no model shape was added to the SPS, the returned mesh is just a single triangular plane. * @returns the created mesh */ buildMesh() { if (!this._isNotBuilt && this.mesh) return this.mesh; if (this.nbParticles === 0 && !this.mesh) { const t = uU("", { radius: 1, tessellation: 3 }, this._scene); this.addShape(t, 1), t.dispose(); } if (this._indices32 = this._needs32Bits ? new Uint32Array(this._indices) : new Uint16Array(this._indices), this._positions32 = new Float32Array(this._positions), this._uvs32 = new Float32Array(this._uvs), this._colors32 = new Float32Array(this._colors), !this.mesh) { const t = new Ee(this.name, this._scene); this.mesh = t; } !this._updatable && this._multimaterialEnabled && this._sortParticlesByMaterial(), this.recomputeNormals && Ut.ComputeNormals(this._positions32, this._indices32, this._normals), this._normals32 = new Float32Array(this._normals), this._fixedNormal32 = new Float32Array(this._normals), this._mustUnrotateFixedNormals && this._unrotateFixedNormals(); const e = new Ut(); if (e.indices = this._depthSort ? this._indices : this._indices32, e.set(this._positions32, J.PositionKind), e.set(this._normals32, J.NormalKind), this._uvs32.length > 0 && e.set(this._uvs32, J.UVKind), this._colors32.length > 0 && e.set(this._colors32, J.ColorKind), e.applyToMesh(this.mesh, this._updatable), this.mesh.isPickable = this._pickable, this._pickable) { let t = 0; for (let r = 0; r < this.nbParticles; r++) { const n = this.particles[r], i = n._model._indicesLength; for (let s = 0; s < i; s++) if (s % 3 == 0) { const f = { idx: n.idx, faceId: t }; this.pickedParticles[t] = f, t++; } } } return this._multimaterialEnabled && this.setMultiMaterial(this._materials), this._expandable || (!this._depthSort && !this._multimaterialEnabled && !this._autoFixFaceOrientation && (this._indices = null), this._positions = null, this._normals = null, this._uvs = null, this._colors = null, this._updatable || (this.particles.length = 0)), this._isNotBuilt = !1, this.recomputeNormals = !1, this._recomputeInvisibles = !0, this.mesh; } _getUVKind(e, t) { var r, n; return t === -1 && (!((r = e.material) === null || r === void 0) && r.diffuseTexture ? t = e.material.diffuseTexture.coordinatesIndex : !((n = e.material) === null || n === void 0) && n.albedoTexture && (t = e.material.albedoTexture.coordinatesIndex)), "uv" + (t ? t + 1 : ""); } /** * Digests the mesh and generates as many solid particles in the system as wanted. Returns the SPS. * These particles will have the same geometry than the mesh parts and will be positioned at the same localisation than the mesh original places. * Thus the particles generated from `digest()` have their property `position` set yet. * @param mesh ( Mesh ) is the mesh to be digested * @param options {facetNb} (optional integer, default 1) is the number of mesh facets per particle, this parameter is overridden by the parameter `number` if any * {delta} (optional integer, default 0) is the random extra number of facets per particle , each particle will have between `facetNb` and `facetNb + delta` facets * {number} (optional positive integer) is the wanted number of particles : each particle is built with `mesh_total_facets / number` facets * {storage} (optional existing array) is an array where the particles will be stored for a further use instead of being inserted in the SPS. * {uvKind} (optional positive integer, default 0) is the kind of UV to read from. Use -1 to deduce it from the diffuse/albedo texture (if any) of the mesh material * @param options.facetNb * @param options.number * @param options.delta * @param options.storage * @param options.uvKind * @returns the current SPS */ digest(e, t) { var r; let n = t && t.facetNb || 1, i = t && t.number || 0, s = t && t.delta || 0; const a = e.getVerticesData(J.PositionKind), f = e.getIndices(), o = e.getVerticesData(this._getUVKind(e, (r = t == null ? void 0 : t.uvKind) !== null && r !== void 0 ? r : 0)), d = e.getVerticesData(J.ColorKind), v = e.getVerticesData(J.NormalKind), u = t && t.storage ? t.storage : null; let l = 0; const P = f.length / 3; i ? (i = i > P ? P : i, n = Math.round(P / i), s = 0) : n = n > P ? P : n; const p = [], c = [], H = [], T = [], q = [], b = S.Zero(), j = n; for (; l < P; ) { n = j + Math.floor((1 + s) * Math.random()), l > P - n && (n = P - l), p.length = 0, c.length = 0, H.length = 0, T.length = 0, q.length = 0; let w = 0; for (let L = l * 3; L < (l + n) * 3; L++) { H.push(w); const $ = f[L], ae = $ * 3; if (p.push(a[ae], a[ae + 1], a[ae + 2]), c.push(v[ae], v[ae + 1], v[ae + 2]), o) { const Pe = $ * 2; T.push(o[Pe], o[Pe + 1]); } if (d) { const Pe = $ * 4; q.push(d[Pe], d[Pe + 1], d[Pe + 2], d[Pe + 3]); } w++; } let m = this.nbParticles; const I = this._posToShape(p), N = this._uvsToShapeUV(T), k = H.slice(), R = q.slice(), y = c.slice(); b.copyFromFloats(0, 0, 0); let O; for (O = 0; O < I.length; O++) b.addInPlace(I[O]); b.scaleInPlace(1 / I.length); const Y = new S(1 / 0, 1 / 0, 1 / 0), ee = new S(-1 / 0, -1 / 0, -1 / 0); for (O = 0; O < I.length; O++) I[O].subtractInPlace(b), Y.minimizeInPlaceFromFloats(I[O].x, I[O].y, I[O].z), ee.maximizeInPlaceFromFloats(I[O].x, I[O].y, I[O].z); let Z; this._particlesIntersect && (Z = new Md(Y, ee)); let te = null; this._useModelMaterial && (te = e.material ? e.material : this._setDefaultMaterial()); const fe = new xF(this._shapeCounter, I, k, y, R, N, null, null, te), _ = this._positions.length, G = this._indices.length; this._meshBuilder(this._index, G, I, this._positions, k, this._indices, T, this._uvs, R, this._colors, y, this._normals, m, 0, null, fe), this._addParticle(m, this._lastParticleId, _, G, fe, this._shapeCounter, 0, Z, u), this.particles[this.nbParticles].position.addInPlace(b), u || (this._index += I.length, m++, this.nbParticles++, this._lastParticleId++), this._shapeCounter++, l += n; } return this._isNotBuilt = !0, this; } /** * Unrotate the fixed normals in case the mesh was built with pre-rotated particles, ex : use of positionFunction in addShape() * @internal */ _unrotateFixedNormals() { let e = 0, t = 0; const r = ue.Vector3[0], n = ue.Quaternion[0], i = ue.Matrix[0]; for (let s = 0; s < this.particles.length; s++) { const a = this.particles[s], f = a._model._shape; if (a.rotationQuaternion) a.rotationQuaternion.conjugateToRef(n); else { const o = a.rotation; Ze.RotationYawPitchRollToRef(o.y, o.x, o.z, n), n.conjugateInPlace(); } n.toRotationMatrix(i); for (let o = 0; o < f.length; o++) t = e + o * 3, S.TransformNormalFromFloatsToRef(this._normals32[t], this._normals32[t + 1], this._normals32[t + 2], i, r), r.toArray(this._fixedNormal32, t); e = t + 3; } } /** * Resets the temporary working copy particle * @internal */ _resetCopy() { const e = this._copy; e.position.setAll(0), e.rotation.setAll(0), e.rotationQuaternion = null, e.scaling.setAll(1), e.uvs.copyFromFloats(0, 0, 1, 1), e.color = null, e.translateFromPivot = !1, e.shapeId = 0, e.materialIndex = null; } /** * Inserts the shape model geometry in the global SPS mesh by updating the positions, indices, normals, colors, uvs arrays * @param p the current index in the positions array to be updated * @param ind the current index in the indices array * @param shape a Vector3 array, the shape geometry * @param positions the positions array to be updated * @param meshInd the shape indices array * @param indices the indices array to be updated * @param meshUV the shape uv array * @param uvs the uv array to be updated * @param meshCol the shape color array * @param colors the color array to be updated * @param meshNor the shape normals array * @param normals the normals array to be updated * @param idx the particle index * @param idxInShape the particle index in its shape * @param options the addShape() method passed options * @param model * @model the particle model * @internal */ _meshBuilder(e, t, r, n, i, s, a, f, o, d, v, u, l, P, p, c) { let H, T = 0, q = 0, b = 0; this._resetCopy(); const j = this._copy, w = !!(p && p.storage); if (j.idx = l, j.idxInShape = P, j.shapeId = c.shapeId, this._useModelMaterial) { const Z = c._material.uniqueId, te = this._materialIndexesById; Object.prototype.hasOwnProperty.call(te, Z) || (te[Z] = this._materials.length, this._materials.push(c._material)); const fe = te[Z]; j.materialIndex = fe; } if (p && p.positionFunction && (p.positionFunction(j, l, P), this._mustUnrotateFixedNormals = !0), w) return j; const m = ue.Matrix[0], I = this._tmpVertex, N = I.position, k = I.color, R = I.uv, y = ue.Vector3[1], O = ue.Vector3[2], Y = ue.Vector3[3]; he.IdentityToRef(m), j.getRotationMatrix(m), j.pivot.multiplyToRef(j.scaling, Y), j.translateFromPivot ? O.setAll(0) : O.copyFrom(Y); const ee = p && p.vertexFunction; for (H = 0; H < r.length; H++) { if (N.copyFrom(r[H]), j.color && k.copyFrom(j.color), a && R.copyFromFloats(a[T], a[T + 1]), ee && p.vertexFunction(j, I, H), N.multiplyInPlace(j.scaling).subtractInPlace(Y), S.TransformCoordinatesToRef(N, m, y), y.addInPlace(O).addInPlace(j.position), n.push(y.x, y.y, y.z), a) { const Z = j.uvs; f.push((Z.z - Z.x) * R.x + Z.x, (Z.w - Z.y) * R.y + Z.y), T += 2; } if (j.color) this._color.copyFrom(k); else { const Z = this._color; o && o[q] !== void 0 ? (Z.r = o[q], Z.g = o[q + 1], Z.b = o[q + 2], Z.a = o[q + 3]) : (Z.r = 1, Z.g = 1, Z.b = 1, Z.a = 1); } d.push(this._color.r, this._color.g, this._color.b, this._color.a), q += 4, !this.recomputeNormals && v && (S.TransformNormalFromFloatsToRef(v[b], v[b + 1], v[b + 2], m, N), u.push(N.x, N.y, N.z), b += 3); } for (H = 0; H < i.length; H++) { const Z = e + i[H]; s.push(Z), Z > 65535 && (this._needs32Bits = !0); } if (this._depthSort || this._multimaterialEnabled) { const Z = j.materialIndex !== null ? j.materialIndex : 0; this.depthSortedParticles.push(new zne(l, t, i.length, Z)); } return j; } /** * Returns a shape Vector3 array from positions float array * @param positions float array * @returns a vector3 array * @internal */ _posToShape(e) { const t = []; for (let r = 0; r < e.length; r += 3) t.push(S.FromArray(e, r)); return t; } /** * Returns a shapeUV array from a float uvs (array deep copy) * @param uvs as a float array * @returns a shapeUV array * @internal */ _uvsToShapeUV(e) { const t = []; if (e) for (let r = 0; r < e.length; r++) t.push(e[r]); return t; } /** * Adds a new particle object in the particles array * @param idx particle index in particles array * @param id particle id * @param idxpos positionIndex : the starting index of the particle vertices in the SPS "positions" array * @param idxind indiceIndex : he starting index of the particle indices in the SPS "indices" array * @param model particle ModelShape object * @param shapeId model shape identifier * @param idxInShape index of the particle in the current model * @param bInfo model bounding info object * @param storage target storage array, if any * @internal */ _addParticle(e, t, r, n, i, s, a, f = null, o = null) { const d = new bF(e, t, r, n, i, s, a, this, f); return (o || this.particles).push(d), d; } /** * Adds some particles to the SPS from the model shape. Returns the shape id. * Please read the doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/immutable_sps * @param mesh is any Mesh object that will be used as a model for the solid particles. If the mesh does not have vertex normals, it will turn on the recomputeNormals attribute. * @param nb (positive integer) the number of particles to be created from this model * @param options {positionFunction} is an optional javascript function to called for each particle on SPS creation. * {vertexFunction} is an optional javascript function to called for each vertex of each particle on SPS creation * {storage} (optional existing array) is an array where the particles will be stored for a further use instead of being inserted in the SPS. * @param options.positionFunction * @param options.vertexFunction * @param options.storage * @returns the number of shapes in the system */ addShape(e, t, r) { const n = e.getVerticesData(J.PositionKind), i = e.getIndices(), s = e.getVerticesData(J.UVKind), a = e.getVerticesData(J.ColorKind), f = e.getVerticesData(J.NormalKind); this.recomputeNormals = !f; const o = Array.from(i), d = f ? Array.from(f) : [], v = a ? Array.from(a) : [], u = r && r.storage ? r.storage : null; let l = null; this._particlesIntersect && (l = e.getBoundingInfo()); const P = this._posToShape(n), p = this._uvsToShapeUV(s), c = r ? r.positionFunction : null, H = r ? r.vertexFunction : null; let T = null; this._useModelMaterial && (T = e.material ? e.material : this._setDefaultMaterial()); const q = new xF(this._shapeCounter, P, o, d, v, p, c, H, T); for (let b = 0; b < t; b++) this._insertNewParticle(this.nbParticles, b, q, P, i, s, a, f, l, u, r); return this._shapeCounter++, this._isNotBuilt = !0, this._shapeCounter - 1; } /** * Rebuilds a particle back to its just built status : if needed, recomputes the custom positions and vertices * @internal */ _rebuildParticle(e, t = !1) { this._resetCopy(); const r = this._copy; e._model._positionFunction && e._model._positionFunction(r, e.idx, e.idxInShape); const n = ue.Matrix[0], i = ue.Vector3[0], s = ue.Vector3[1], a = ue.Vector3[2], f = ue.Vector3[3]; r.getRotationMatrix(n), e.pivot.multiplyToRef(e.scaling, f), r.translateFromPivot ? a.copyFromFloats(0, 0, 0) : a.copyFrom(f); const o = e._model._shape; for (let d = 0; d < o.length; d++) i.copyFrom(o[d]), e._model._vertexFunction && e._model._vertexFunction(r, i, d), i.multiplyInPlace(r.scaling).subtractInPlace(f), S.TransformCoordinatesToRef(i, n, s), s.addInPlace(a).addInPlace(r.position).toArray(this._positions32, e._pos + d * 3); t && (e.position.setAll(0), e.rotation.setAll(0), e.rotationQuaternion = null, e.scaling.setAll(1), e.uvs.setAll(0), e.pivot.setAll(0), e.translateFromPivot = !1, e.parentId = null); } /** * Rebuilds the whole mesh and updates the VBO : custom positions and vertices are recomputed if needed. * @param reset boolean, default false : if the particles must be reset at position and rotation zero, scaling 1, color white, initial UVs and not parented. * @returns the SPS. */ rebuildMesh(e = !1) { for (let t = 0; t < this.particles.length; t++) this._rebuildParticle(this.particles[t], e); return this.mesh.updateVerticesData(J.PositionKind, this._positions32, !1, !1), this; } /** Removes the particles from the start-th to the end-th included from an expandable SPS (required). * Returns an array with the removed particles. * If the number of particles to remove is lower than zero or greater than the global remaining particle number, then an empty array is returned. * The SPS can't be empty so at least one particle needs to remain in place. * Under the hood, the VertexData array, so the VBO buffer, is recreated each call. * @param start index of the first particle to remove * @param end index of the last particle to remove (included) * @returns an array populated with the removed particles */ removeParticles(e, t) { const r = t - e + 1; if (!this._expandable || r <= 0 || r >= this.nbParticles || !this._updatable) return []; const n = this.particles, i = this.nbParticles; if (t < i - 1) { const o = t + 1, d = n[o]._pos - n[e]._pos, v = n[o]._ind - n[e]._ind; for (let u = o; u < i; u++) { const l = n[u]; l._pos -= d, l._ind -= v; } } const s = n.splice(e, r); this._positions.length = 0, this._indices.length = 0, this._colors.length = 0, this._uvs.length = 0, this._normals.length = 0, this._index = 0, this._idxOfId.length = 0, (this._depthSort || this._multimaterialEnabled) && (this.depthSortedParticles = []); let a = 0; const f = n.length; for (let o = 0; o < f; o++) { const d = n[o], v = d._model, u = v._shape, l = v._indices, P = v._normals, p = v._shapeColors, c = v._shapeUV; d.idx = o, this._idxOfId[d.id] = o, this._meshBuilder(this._index, a, u, this._positions, l, this._indices, c, this._uvs, p, this._colors, P, this._normals, d.idx, d.idxInShape, null, v), this._index += u.length, a += l.length; } return this.nbParticles -= r, this._isNotBuilt = !0, s; } /** * Inserts some pre-created particles in the solid particle system so that they can be managed by setParticles(). * @param solidParticleArray an array populated with Solid Particles objects * @returns the SPS */ insertParticlesFromArray(e) { if (!this._expandable) return this; let t = 0, r = e[0].shapeId; const n = e.length; for (let i = 0; i < n; i++) { const s = e[i], a = s._model, f = a._shape, o = a._indices, d = a._shapeUV, v = a._shapeColors, u = a._normals, l = !u; this.recomputeNormals = l || this.recomputeNormals; const P = s.getBoundingInfo(), p = this._insertNewParticle(this.nbParticles, t, a, f, o, d, v, u, P, null, null); s.copyToRef(p), t++, r != s.shapeId && (r = s.shapeId, t = 0); } return this._isNotBuilt = !0, this; } /** * Creates a new particle and modifies the SPS mesh geometry : * - calls _meshBuilder() to increase the SPS mesh geometry step by step * - calls _addParticle() to populate the particle array * factorized code from addShape() and insertParticlesFromArray() * @param idx particle index in the particles array * @param i particle index in its shape * @param modelShape particle ModelShape object * @param shape shape vertex array * @param meshInd shape indices array * @param meshUV shape uv array * @param meshCol shape color array * @param meshNor shape normals array * @param bbInfo shape bounding info * @param storage target particle storage * @param options * @options addShape() passed options * @internal */ _insertNewParticle(e, t, r, n, i, s, a, f, o, d, v) { const u = this._positions.length, l = this._indices.length, P = this._meshBuilder(this._index, l, n, this._positions, i, this._indices, s, this._uvs, a, this._colors, f, this._normals, e, t, v, r); let p = null; return this._updatable && (p = this._addParticle(this.nbParticles, this._lastParticleId, u, l, r, this._shapeCounter, t, o, d), p.position.copyFrom(P.position), p.rotation.copyFrom(P.rotation), P.rotationQuaternion && (p.rotationQuaternion ? p.rotationQuaternion.copyFrom(P.rotationQuaternion) : p.rotationQuaternion = P.rotationQuaternion.clone()), P.color && (p.color ? p.color.copyFrom(P.color) : p.color = P.color.clone()), p.scaling.copyFrom(P.scaling), p.uvs.copyFrom(P.uvs), P.materialIndex !== null && (p.materialIndex = P.materialIndex), this.expandable && (this._idxOfId[p.id] = p.idx)), d || (this._index += n.length, this.nbParticles++, this._lastParticleId++), p; } /** * Sets all the particles : this method actually really updates the mesh according to the particle positions, rotations, colors, textures, etc. * This method calls `updateParticle()` for each particle of the SPS. * For an animated SPS, it is usually called within the render loop. * This methods does nothing if called on a non updatable or not yet built SPS. Example : buildMesh() not called after having added or removed particles from an expandable SPS. * @param start The particle index in the particle array where to start to compute the particle property values _(default 0)_ * @param end The particle index in the particle array where to stop to compute the particle property values _(default nbParticle - 1)_ * @param update If the mesh must be finally updated on this call after all the particle computations _(default true)_ * @returns the SPS. */ setParticles(e = 0, t = this.nbParticles - 1, r = !0) { if (!this._updatable || this._isNotBuilt) return this; this.beforeUpdateParticles(e, t, r); const n = ue.Matrix[0], i = ue.Matrix[1], s = this.mesh, a = this._colors32, f = this._positions32, o = this._normals32, d = this._uvs32, v = this._indices32, u = this._indices, l = this._fixedNormal32, P = this._depthSort && this._depthSortParticles, p = ue.Vector3, c = p[5].copyFromFloats(1, 0, 0), H = p[6].copyFromFloats(0, 1, 0), T = p[7].copyFromFloats(0, 0, 1), q = p[8].setAll(Number.MAX_VALUE), b = p[9].setAll(-Number.MAX_VALUE), j = p[10].setAll(0), w = this._tmpVertex, m = w.position, I = w.color, N = w.uv; if ((this.billboard || this._depthSort) && (this.mesh.computeWorldMatrix(!0), this.mesh._worldMatrix.invertToRef(i)), this.billboard) { const fe = p[0]; this._camera.getDirectionToRef(bf.Z, fe), S.TransformNormalToRef(fe, i, T), T.normalize(); const _ = this._camera.getViewMatrix(!0); S.TransformNormalFromFloatsToRef(_.m[1], _.m[5], _.m[9], i, H), S.CrossToRef(H, T, c), H.normalize(), c.normalize(); } this._depthSort && S.TransformCoordinatesToRef(this._camera.globalPosition, i, j), he.IdentityToRef(n); let k = 0, R = 0, y = 0, O = 0, Y = 0, ee = 0, Z = 0; if (this.mesh.isFacetDataEnabled && (this._computeBoundingBox = !0), t = t >= this.nbParticles ? this.nbParticles - 1 : t, this._computeBoundingBox && (e != 0 || t != this.nbParticles - 1)) { const fe = this.mesh.getBoundingInfo(); fe && (q.copyFrom(fe.minimum), b.copyFrom(fe.maximum)); } R = this.particles[e]._pos; const te = R / 3 | 0; O = te * 4, ee = te * 2; for (let fe = e; fe <= t; fe++) { const _ = this.particles[fe]; this.updateParticle(_); const G = _._model._shape, L = _._model._shapeUV, $ = _._rotationMatrix, ae = _.position, Pe = _.rotation, ge = _.scaling, me = _._globalPosition; if (P) { const Xe = this.depthSortedParticles[fe]; Xe.idx = _.idx, Xe.ind = _._ind, Xe.indicesLength = _._model._indicesLength, Xe.sqDistance = S.DistanceSquared(_.position, j); } if (!_.alive || _._stillInvisible && !_.isVisible && !this._recomputeInvisibles) { Z = G.length, R += Z * 3, O += Z * 4, ee += Z * 2; continue; } if (_.isVisible) { _._stillInvisible = !1; const Xe = p[12]; if (_.pivot.multiplyToRef(ge, Xe), this.billboard && (Pe.x = 0, Pe.y = 0), (this._computeParticleRotation || this.billboard) && _.getRotationMatrix(n), _.parentId !== null) { const re = this.getParticleById(_.parentId); if (re) { const ve = re._rotationMatrix, qe = re._globalPosition, ke = ae.x * ve[1] + ae.y * ve[4] + ae.z * ve[7], be = ae.x * ve[0] + ae.y * ve[3] + ae.z * ve[6], Fe = ae.x * ve[2] + ae.y * ve[5] + ae.z * ve[8]; if (me.x = qe.x + be, me.y = qe.y + ke, me.z = qe.z + Fe, this._computeParticleRotation || this.billboard) { const Ke = n.m; $[0] = Ke[0] * ve[0] + Ke[1] * ve[3] + Ke[2] * ve[6], $[1] = Ke[0] * ve[1] + Ke[1] * ve[4] + Ke[2] * ve[7], $[2] = Ke[0] * ve[2] + Ke[1] * ve[5] + Ke[2] * ve[8], $[3] = Ke[4] * ve[0] + Ke[5] * ve[3] + Ke[6] * ve[6], $[4] = Ke[4] * ve[1] + Ke[5] * ve[4] + Ke[6] * ve[7], $[5] = Ke[4] * ve[2] + Ke[5] * ve[5] + Ke[6] * ve[8], $[6] = Ke[8] * ve[0] + Ke[9] * ve[3] + Ke[10] * ve[6], $[7] = Ke[8] * ve[1] + Ke[9] * ve[4] + Ke[10] * ve[7], $[8] = Ke[8] * ve[2] + Ke[9] * ve[5] + Ke[10] * ve[8]; } } else _.parentId = null; } else if (me.x = ae.x, me.y = ae.y, me.z = ae.z, this._computeParticleRotation || this.billboard) { const re = n.m; $[0] = re[0], $[1] = re[1], $[2] = re[2], $[3] = re[4], $[4] = re[5], $[5] = re[6], $[6] = re[8], $[7] = re[9], $[8] = re[10]; } const ne = p[11]; for (_.translateFromPivot ? ne.setAll(0) : ne.copyFrom(Xe), Z = 0; Z < G.length; Z++) { k = R + Z * 3, y = O + Z * 4, Y = ee + Z * 2; const re = 2 * Z, ve = re + 1; m.copyFrom(G[Z]), this._computeParticleColor && _.color && I.copyFrom(_.color), this._computeParticleTexture && N.copyFromFloats(L[re], L[ve]), this._computeParticleVertex && this.updateParticleVertex(_, w, Z); const qe = m.x * ge.x - Xe.x, ke = m.y * ge.y - Xe.y, be = m.z * ge.z - Xe.z; let Fe = qe * $[0] + ke * $[3] + be * $[6], Ke = qe * $[1] + ke * $[4] + be * $[7], nt = qe * $[2] + ke * $[5] + be * $[8]; Fe += ne.x, Ke += ne.y, nt += ne.z; const ut = f[k] = me.x + c.x * Fe + H.x * Ke + T.x * nt, bt = f[k + 1] = me.y + c.y * Fe + H.y * Ke + T.y * nt, wt = f[k + 2] = me.z + c.z * Fe + H.z * Ke + T.z * nt; if (this._computeBoundingBox && (q.minimizeInPlaceFromFloats(ut, bt, wt), b.maximizeInPlaceFromFloats(ut, bt, wt)), !this._computeParticleVertex) { const Tt = l[k], lr = l[k + 1], Qt = l[k + 2], tr = Tt * $[0] + lr * $[3] + Qt * $[6], br = Tt * $[1] + lr * $[4] + Qt * $[7], Xn = Tt * $[2] + lr * $[5] + Qt * $[8]; o[k] = c.x * tr + H.x * br + T.x * Xn, o[k + 1] = c.y * tr + H.y * br + T.y * Xn, o[k + 2] = c.z * tr + H.z * br + T.z * Xn; } if (this._computeParticleColor && _.color) { const Tt = this._colors32; Tt[y] = I.r, Tt[y + 1] = I.g, Tt[y + 2] = I.b, Tt[y + 3] = I.a; } if (this._computeParticleTexture) { const Tt = _.uvs; d[Y] = N.x * (Tt.z - Tt.x) + Tt.x, d[Y + 1] = N.y * (Tt.w - Tt.y) + Tt.y; } } } else for (_._stillInvisible = !0, Z = 0; Z < G.length; Z++) { if (k = R + Z * 3, y = O + Z * 4, Y = ee + Z * 2, f[k] = f[k + 1] = f[k + 2] = 0, o[k] = o[k + 1] = o[k + 2] = 0, this._computeParticleColor && _.color) { const Xe = _.color; a[y] = Xe.r, a[y + 1] = Xe.g, a[y + 2] = Xe.b, a[y + 3] = Xe.a; } if (this._computeParticleTexture) { const Xe = _.uvs; d[Y] = L[Z * 2] * (Xe.z - Xe.x) + Xe.x, d[Y + 1] = L[Z * 2 + 1] * (Xe.w - Xe.y) + Xe.y; } } if (this._particlesIntersect) { const Xe = _.getBoundingInfo(), De = Xe.boundingBox, ne = Xe.boundingSphere, re = _._modelBoundingInfo; if (!this._bSphereOnly) { const nt = re.boundingBox.vectors, ut = p[1], bt = p[2]; ut.setAll(Number.MAX_VALUE), bt.setAll(-Number.MAX_VALUE); for (let wt = 0; wt < 8; wt++) { const Tt = nt[wt].x * ge.x, lr = nt[wt].y * ge.y, Qt = nt[wt].z * ge.z, tr = Tt * $[0] + lr * $[3] + Qt * $[6], br = Tt * $[1] + lr * $[4] + Qt * $[7], Xn = Tt * $[2] + lr * $[5] + Qt * $[8], qr = ae.x + c.x * tr + H.x * br + T.x * Xn, En = ae.y + c.y * tr + H.y * br + T.y * Xn, Bn = ae.z + c.z * tr + H.z * br + T.z * Xn; ut.minimizeInPlaceFromFloats(qr, En, Bn), bt.maximizeInPlaceFromFloats(qr, En, Bn); } De.reConstruct(ut, bt, s._worldMatrix); } const ve = re.minimum.multiplyToRef(ge, p[1]), qe = re.maximum.multiplyToRef(ge, p[2]), ke = qe.addToRef(ve, p[3]).scaleInPlace(0.5).addInPlace(me), be = qe.subtractToRef(ve, p[4]).scaleInPlace(0.5 * this._bSphereRadiusFactor), Fe = ke.subtractToRef(be, p[1]), Ke = ke.addToRef(be, p[2]); ne.reConstruct(Fe, Ke, s._worldMatrix); } R = k + 3, O = y + 4, ee = Y + 2; } if (r) { if (this._computeParticleColor) { const _ = s.getVertexBuffer(J.ColorKind); _ && !s.isPickable ? _.updateDirectly(a, 0) : s.updateVerticesData(J.ColorKind, a, !1, !1); } if (this._computeParticleTexture) { const _ = s.getVertexBuffer(J.UVKind); _ && !s.isPickable ? _.updateDirectly(d, 0) : s.updateVerticesData(J.UVKind, d, !1, !1); } const fe = s.getVertexBuffer(J.PositionKind); if (fe && !s.isPickable ? fe.updateDirectly(f, 0) : s.updateVerticesData(J.PositionKind, f, !1, !1), !s.areNormalsFrozen || s.isFacetDataEnabled) { if (this._computeParticleVertex || s.isFacetDataEnabled) { const _ = s.isFacetDataEnabled ? s.getFacetDataParameters() : null; Ut.ComputeNormals(f, v, o, _); for (let G = 0; G < o.length; G++) l[G] = o[G]; } if (!s.areNormalsFrozen) { const _ = s.getVertexBuffer(J.NormalKind); _ && !s.isPickable ? _.updateDirectly(o, 0) : s.updateVerticesData(J.NormalKind, o, !1, !1); } } if (P) { const _ = this.depthSortedParticles; _.sort(this._depthSortFunction); const G = _.length; let L = 0, $ = 0; for (let ae = 0; ae < G; ae++) { const Pe = _[ae], ge = Pe.indicesLength, me = Pe.ind; for (let Xe = 0; Xe < ge; Xe++) if (v[L] = u[me + Xe], L++, this._pickable && Xe % 3 == 0) { const ne = this.pickedParticles[$]; ne.idx = Pe.idx, ne.faceId = $, $++; } } } if (this._autoFixFaceOrientation) { let _ = 0; for (let G = 0; G < this.particles.length; G++) { const L = P ? this.particles[this.depthSortedParticles[G].idx] : this.particles[G]; if (L.scale.x * L.scale.y * L.scale.z < 0) for (let ae = 0; ae < L._model._indicesLength; ae += 3) { const Pe = u[L._ind + ae]; v[_ + ae] = u[L._ind + ae + 1], v[_ + ae + 1] = Pe; } _ += L._model._indicesLength; } } (P || this._autoFixFaceOrientation) && s.updateIndices(v); } return this._computeBoundingBox && (s.hasBoundingInfo ? s.getBoundingInfo().reConstruct(q, b, s._worldMatrix) : s.buildBoundingInfo(q, b, s._worldMatrix)), this._autoUpdateSubMeshes && this.computeSubMeshes(), this._recomputeInvisibles = !1, this.afterUpdateParticles(e, t, r), this; } /** * Disposes the SPS. */ dispose() { this.mesh.dispose(), this.vars = null, this._positions = null, this._indices = null, this._normals = null, this._uvs = null, this._colors = null, this._indices32 = null, this._positions32 = null, this._normals32 = null, this._fixedNormal32 = null, this._uvs32 = null, this._colors32 = null, this.pickedParticles = null, this.pickedBySubMesh = null, this._materials = null, this._materialIndexes = null, this._indicesByMaterial = null, this._idxOfId = null; } /** Returns an object {idx: number faceId: number} for the picked particle from the passed pickingInfo object. * idx is the particle index in the SPS * faceId is the picked face index counted within this particle. * Returns null if the pickInfo can't identify a picked particle. * @param pickingInfo (PickingInfo object) * @returns {idx: number, faceId: number} or null */ pickedParticle(e) { if (e.hit) { const t = e.subMeshId, r = e.faceId - this.mesh.subMeshes[t].indexStart / 3, n = this.pickedBySubMesh; if (n[t] && n[t][r]) return n[t][r]; } return null; } /** * Returns a SolidParticle object from its identifier : particle.id * @param id (integer) the particle Id * @returns the searched particle or null if not found in the SPS. */ getParticleById(e) { const t = this.particles[e]; if (t && t.id == e) return t; const r = this.particles, n = this._idxOfId[e]; if (n !== void 0) return r[n]; let i = 0; const s = this.nbParticles; for (; i < s; ) { const a = r[i]; if (a.id == e) return a; i++; } return null; } /** * Returns a new array populated with the particles having the passed shapeId. * @param shapeId (integer) the shape identifier * @returns a new solid particle array */ getParticlesByShapeId(e) { const t = []; return this.getParticlesByShapeIdToRef(e, t), t; } /** * Populates the passed array "ref" with the particles having the passed shapeId. * @param shapeId the shape identifier * @returns the SPS * @param ref */ getParticlesByShapeIdToRef(e, t) { t.length = 0; for (let r = 0; r < this.nbParticles; r++) { const n = this.particles[r]; n.shapeId == e && t.push(n); } return this; } /** * Computes the required SubMeshes according the materials assigned to the particles. * @returns the solid particle system. * Does nothing if called before the SPS mesh is built. */ computeSubMeshes() { if (!this.mesh || !this._multimaterialEnabled) return this; const e = this.depthSortedParticles; if (this.particles.length > 0) for (let s = 0; s < this.particles.length; s++) { const a = this.particles[s]; a.materialIndex || (a.materialIndex = 0); const f = e[s]; f.materialIndex = a.materialIndex, f.ind = a._ind, f.indicesLength = a._model._indicesLength, f.idx = a.idx; } this._sortParticlesByMaterial(); const t = this._indicesByMaterial, r = this._materialIndexes, n = this.mesh; n.subMeshes = []; const i = n.getTotalVertices(); for (let s = 0; s < r.length; s++) { const a = t[s], f = t[s + 1] - a, o = r[s]; new rA(o, 0, i, a, f, n); } return this; } /** * Sorts the solid particles by material when MultiMaterial is enabled. * Updates the indices32 array. * Updates the indicesByMaterial array. * Updates the mesh indices array. * @returns the SPS * @internal */ _sortParticlesByMaterial() { const e = [0]; this._indicesByMaterial = e; const t = []; this._materialIndexes = t; const r = this.depthSortedParticles; r.sort(this._materialSortFunction); const n = r.length, i = this._indices32, s = this._indices; let a = 0, f = 0, o = 0, d = r[0].materialIndex; t.push(d), this._pickable && (this.pickedBySubMesh = [[]], this.pickedParticles = this.pickedBySubMesh[0]); for (let v = 0; v < n; v++) { const u = r[v], l = u.indicesLength, P = u.ind; u.materialIndex !== d && (d = u.materialIndex, e.push(o), t.push(d), this._pickable && (a++, this.pickedBySubMesh[a] = [], f = 0)); let p = 0; for (let c = 0; c < l; c++) { if (i[o] = s[P + c], this._pickable && c % 3 == 0) { const T = this.pickedBySubMesh[a][f]; T ? (T.idx = u.idx, T.faceId = p) : this.pickedBySubMesh[a][f] = { idx: u.idx, faceId: p }, f++, p++; } o++; } } return e.push(i.length), this._updatable && this.mesh.updateIndices(i), this; } /** * Sets the material indexes by id materialIndexesById[id] = materialIndex * @internal */ _setMaterialIndexesById() { this._materialIndexesById = {}; for (let e = 0; e < this._materials.length; e++) { const t = this._materials[e].uniqueId; this._materialIndexesById[t] = e; } } /** * Returns an array with unique values of Materials from the passed array * @param array the material array to be checked and filtered * @internal */ _filterUniqueMaterialId(e) { return e.filter(function(r, n, i) { return i.indexOf(r) === n; }); } /** * Sets a new Standard Material as _defaultMaterial if not already set. * @internal */ _setDefaultMaterial() { return this._defaultMaterial || (this._defaultMaterial = new Wt(this.name + "DefaultMaterial", this._scene)), this._defaultMaterial; } /** * Visibility helper : Recomputes the visible size according to the mesh bounding box * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility * @returns the SPS. */ refreshVisibleSize() { return this._isVisibilityBoxLocked || this.mesh.refreshBoundingInfo(), this; } /** * Visibility helper : Sets the size of a visibility box, this sets the underlying mesh bounding box. * @param size the size (float) of the visibility box * note : this doesn't lock the SPS mesh bounding box. * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility */ setVisibilityBox(e) { const t = e / 2; this.mesh.buildBoundingInfo(new S(-t, -t, -t), new S(t, t, t)); } /** * Gets whether the SPS as always visible or not * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility */ get isAlwaysVisible() { return this._alwaysVisible; } /** * Sets the SPS as always visible or not * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility */ set isAlwaysVisible(e) { this._alwaysVisible = e, this.mesh.alwaysSelectAsActiveMesh = e; } /** * Sets the SPS visibility box as locked or not. This enables/disables the underlying mesh bounding box updates. * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility */ set isVisibilityBoxLocked(e) { this._isVisibilityBoxLocked = e; const t = this.mesh.getBoundingInfo(); t.isLocked = e; } /** * Gets if the SPS visibility box as locked or not. This enables/disables the underlying mesh bounding box updates. * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility */ get isVisibilityBoxLocked() { return this._isVisibilityBoxLocked; } /** * Tells to `setParticles()` to compute the particle rotations or not. * Default value : true. The SPS is faster when it's set to false. * Note : the particle rotations aren't stored values, so setting `computeParticleRotation` to false will prevents the particle to rotate. */ set computeParticleRotation(e) { this._computeParticleRotation = e; } /** * Tells to `setParticles()` to compute the particle colors or not. * Default value : true. The SPS is faster when it's set to false. * Note : the particle colors are stored values, so setting `computeParticleColor` to false will keep yet the last colors set. */ set computeParticleColor(e) { this._computeParticleColor = e; } set computeParticleTexture(e) { this._computeParticleTexture = e; } /** * Tells to `setParticles()` to call the vertex function for each vertex of each particle, or not. * Default value : false. The SPS is faster when it's set to false. * Note : the particle custom vertex positions aren't stored values. */ set computeParticleVertex(e) { this._computeParticleVertex = e; } /** * Tells to `setParticles()` to compute or not the mesh bounding box when computing the particle positions. */ set computeBoundingBox(e) { this._computeBoundingBox = e; } /** * Tells to `setParticles()` to sort or not the distance between each particle and the camera. * Skipped when `enableDepthSort` is set to `false` (default) at construction time. * Default : `true` */ set depthSortParticles(e) { this._depthSortParticles = e; } /** * Gets if `setParticles()` computes the particle rotations or not. * Default value : true. The SPS is faster when it's set to false. * Note : the particle rotations aren't stored values, so setting `computeParticleRotation` to false will prevents the particle to rotate. */ get computeParticleRotation() { return this._computeParticleRotation; } /** * Gets if `setParticles()` computes the particle colors or not. * Default value : true. The SPS is faster when it's set to false. * Note : the particle colors are stored values, so setting `computeParticleColor` to false will keep yet the last colors set. */ get computeParticleColor() { return this._computeParticleColor; } /** * Gets if `setParticles()` computes the particle textures or not. * Default value : true. The SPS is faster when it's set to false. * Note : the particle textures are stored values, so setting `computeParticleTexture` to false will keep yet the last colors set. */ get computeParticleTexture() { return this._computeParticleTexture; } /** * Gets if `setParticles()` calls the vertex function for each vertex of each particle, or not. * Default value : false. The SPS is faster when it's set to false. * Note : the particle custom vertex positions aren't stored values. */ get computeParticleVertex() { return this._computeParticleVertex; } /** * Gets if `setParticles()` computes or not the mesh bounding box when computing the particle positions. */ get computeBoundingBox() { return this._computeBoundingBox; } /** * Gets if `setParticles()` sorts or not the distance between each particle and the camera. * Skipped when `enableDepthSort` is set to `false` (default) at construction time. * Default : `true` */ get depthSortParticles() { return this._depthSortParticles; } /** * Gets if the SPS is created as expandable at construction time. * Default : `false` */ get expandable() { return this._expandable; } /** * Gets if the SPS supports the Multi Materials */ get multimaterialEnabled() { return this._multimaterialEnabled; } /** * Gets if the SPS uses the model materials for its own multimaterial. */ get useModelMaterial() { return this._useModelMaterial; } /** * The SPS used material array. */ get materials() { return this._materials; } /** * Sets the SPS MultiMaterial from the passed materials. * Note : the passed array is internally copied and not used then by reference. * @param materials an array of material objects. This array indexes are the materialIndex values of the particles. */ setMultiMaterial(e) { this._materials = this._filterUniqueMaterialId(e), this._setMaterialIndexesById(), this._multimaterial && this._multimaterial.dispose(), this._multimaterial = new Dc(this.name + "MultiMaterial", this._scene); for (let t = 0; t < this._materials.length; t++) this._multimaterial.subMaterials.push(this._materials[t]); this.computeSubMeshes(), this.mesh.material = this._multimaterial; } /** * The SPS computed multimaterial object */ get multimaterial() { return this._multimaterial; } set multimaterial(e) { this._multimaterial = e; } /** * If the subMeshes must be updated on the next call to setParticles() */ get autoUpdateSubMeshes() { return this._autoUpdateSubMeshes; } set autoUpdateSubMeshes(e) { this._autoUpdateSubMeshes = e; } // ======================================================================= // Particle behavior logic // these following methods may be overwritten by the user to fit his needs /** * This function does nothing. It may be overwritten to set all the particle first values. * The SPS doesn't call this function, you may have to call it by your own. * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/manage_sps_particles */ initParticles() { } /** * This function does nothing. It may be overwritten to recycle a particle. * The SPS doesn't call this function, you may have to call it by your own. * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/manage_sps_particles * @param particle The particle to recycle * @returns the recycled particle */ recycleParticle(e) { return e; } /** * Updates a particle : this function should be overwritten by the user. * It is called on each particle by `setParticles()`. This is the place to code each particle behavior. * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/manage_sps_particles * @example : just set a particle position or velocity and recycle conditions * @param particle The particle to update * @returns the updated particle */ updateParticle(e) { return e; } /** * Updates a vertex of a particle : it can be overwritten by the user. * This will be called on each vertex particle by `setParticles()` if `computeParticleVertex` is set to true only. * @param particle the current particle * @param vertex the current vertex of the current particle : a SolidParticleVertex object * @param pt the index of the current vertex in the particle shape * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_vertices * @example : just set a vertex particle position or color * @returns the sps */ // eslint-disable-next-line @typescript-eslint/no-unused-vars updateParticleVertex(e, t, r) { return this; } /** * This will be called before any other treatment by `setParticles()` and will be passed three parameters. * This does nothing and may be overwritten by the user. * @param start the particle index in the particle array where to stop to iterate, same than the value passed to setParticle() * @param stop the particle index in the particle array where to stop to iterate, same than the value passed to setParticle() * @param update the boolean update value actually passed to setParticles() */ // eslint-disable-next-line @typescript-eslint/no-unused-vars beforeUpdateParticles(e, t, r) { } /** * This will be called by `setParticles()` after all the other treatments and just before the actual mesh update. * This will be passed three parameters. * This does nothing and may be overwritten by the user. * @param start the particle index in the particle array where to stop to iterate, same than the value passed to setParticle() * @param stop the particle index in the particle array where to stop to iterate, same than the value passed to setParticle() * @param update the boolean update value actually passed to setParticles() */ // eslint-disable-next-line @typescript-eslint/no-unused-vars afterUpdateParticles(e, t, r) { } } class Zne { /** * Creates a Point Cloud object. * Don't create particles manually, use instead the PCS internal tools like _addParticle() * @param particleIndex (integer) is the particle index in the PCS pool. It's also the particle identifier. * @param group (PointsGroup) is the group the particle belongs to * @param groupId (integer) is the group identifier in the PCS. * @param idxInGroup (integer) is the index of the particle in the current point group (ex: the 10th point of addPoints(30)) * @param pcs defines the PCS it is associated to */ constructor(e, t, r, n, i) { this.idx = 0, this.color = new xt(1, 1, 1, 1), this.position = S.Zero(), this.rotation = S.Zero(), this.uv = new at(0, 0), this.velocity = S.Zero(), this.pivot = S.Zero(), this.translateFromPivot = !1, this._pos = 0, this._ind = 0, this.groupId = 0, this.idxInGroup = 0, this._stillInvisible = !1, this._rotationMatrix = [1, 0, 0, 0, 1, 0, 0, 0, 1], this.parentId = null, this._globalPosition = S.Zero(), this.idx = e, this._group = t, this.groupId = r, this.idxInGroup = n, this._pcs = i; } /** * get point size */ get size() { return this.size; } /** * Set point size */ set size(e) { this.size = e; } /** * Legacy support, changed quaternion to rotationQuaternion */ get quaternion() { return this.rotationQuaternion; } /** * Legacy support, changed quaternion to rotationQuaternion */ set quaternion(e) { this.rotationQuaternion = e; } /** * Returns a boolean. True if the particle intersects a mesh, else false * The intersection is computed on the particle position and Axis Aligned Bounding Box (AABB) or Sphere * @param target is the object (point or mesh) what the intersection is computed against * @param isSphere is boolean flag when false (default) bounding box of mesh is used, when true the bounding sphere is used * @returns true if it intersects */ intersectsMesh(e, t) { if (!e.hasBoundingInfo) return !1; if (!this._pcs.mesh) throw new Error("Point Cloud System doesnt contain the Mesh"); if (t) return e.getBoundingInfo().boundingSphere.intersectsPoint(this.position.add(this._pcs.mesh.position)); const r = e.getBoundingInfo().boundingBox, n = r.maximumWorld.x, i = r.minimumWorld.x, s = r.maximumWorld.y, a = r.minimumWorld.y, f = r.maximumWorld.z, o = r.minimumWorld.z, d = this.position.x + this._pcs.mesh.position.x, v = this.position.y + this._pcs.mesh.position.y, u = this.position.z + this._pcs.mesh.position.z; return i <= d && d <= n && a <= v && v <= s && o <= u && u <= f; } /** * get the rotation matrix of the particle * @internal */ getRotationMatrix(e) { let t; if (this.rotationQuaternion) t = this.rotationQuaternion; else { t = ue.Quaternion[0]; const r = this.rotation; Ze.RotationYawPitchRollToRef(r.y, r.x, r.z, t); } t.toRotationMatrix(e); } } class WC { /** * Get or set the groupId * @deprecated Please use groupId instead */ get groupID() { return this.groupId; } set groupID(e) { this.groupId = e; } /** * Creates a points group object. This is an internal reference to produce particles for the PCS. * PCS internal tool, don't use it manually. * @internal */ constructor(e, t) { this.groupId = e, this._positionFunction = t; } } var Iv; (function(A) { A[A.Color = 2] = "Color", A[A.UV = 1] = "UV", A[A.Random = 0] = "Random", A[A.Stated = 3] = "Stated"; })(Iv || (Iv = {})); class _ne { /** * Gets the particle positions computed by the Point Cloud System */ get positions() { return this._positions32; } /** * Gets the particle colors computed by the Point Cloud System */ get colors() { return this._colors32; } /** * Gets the particle uvs computed by the Point Cloud System */ get uvs() { return this._uvs32; } /** * Creates a PCS (Points Cloud System) object * @param name (String) is the PCS name, this will be the underlying mesh name * @param pointSize (number) is the size for each point. Has no effect on a WebGPU engine. * @param scene (Scene) is the scene in which the PCS is added * @param options defines the options of the PCS e.g. * * updatable (optional boolean, default true) : if the PCS must be updatable or immutable * @param options.updatable */ constructor(e, t, r, n) { this.particles = new Array(), this.nbParticles = 0, this.counter = 0, this.vars = {}, this._promises = [], this._positions = new Array(), this._indices = new Array(), this._normals = new Array(), this._colors = new Array(), this._uvs = new Array(), this._updatable = !0, this._isVisibilityBoxLocked = !1, this._alwaysVisible = !1, this._groups = new Array(), this._groupCounter = 0, this._computeParticleColor = !0, this._computeParticleTexture = !0, this._computeParticleRotation = !0, this._computeBoundingBox = !1, this._isReady = !1, this.name = e, this._size = t, this._scene = r || gr.LastCreatedScene, n && n.updatable !== void 0 ? this._updatable = n.updatable : this._updatable = !0; } /** * Builds the PCS underlying mesh. Returns a standard Mesh. * If no points were added to the PCS, the returned mesh is just a single point. * @param material The material to use to render the mesh. If not provided, will create a default one * @returns a promise for the created mesh */ buildMeshAsync(e) { return Promise.all(this._promises).then(() => (this._isReady = !0, this._buildMesh(e))); } /** * @internal */ _buildMesh(e) { this.nbParticles === 0 && this.addPoints(1), this._positions32 = new Float32Array(this._positions), this._uvs32 = new Float32Array(this._uvs), this._colors32 = new Float32Array(this._colors); const t = new Ut(); t.set(this._positions32, J.PositionKind), this._uvs32.length > 0 && t.set(this._uvs32, J.UVKind); let r = 0; this._colors32.length > 0 && (r = 1, t.set(this._colors32, J.ColorKind)); const n = new Ee(this.name, this._scene); t.applyToMesh(n, this._updatable), this.mesh = n, this._positions = null, this._uvs = null, this._colors = null, this._updatable || (this.particles.length = 0); let i = e; return i || (i = new Wt("point cloud material", this._scene), i.emissiveColor = new Ne(r, r, r), i.disableLighting = !0, i.pointsCloud = !0, i.pointSize = this._size), n.material = i, new Promise((s) => s(n)); } // adds a new particle object in the particles array _addParticle(e, t, r, n) { const i = new Zne(e, t, r, n, this); return this.particles.push(i), i; } _randomUnitVector(e) { e.position = new S(Math.random(), Math.random(), Math.random()), e.color = new xt(1, 1, 1, 1); } _getColorIndicesForCoord(e, t, r, n) { const i = e._groupImageData, s = r * (n * 4) + t * 4, a = [s, s + 1, s + 2, s + 3], f = a[0], o = a[1], d = a[2], v = a[3], u = i[f], l = i[o], P = i[d], p = i[v]; return new xt(u / 255, l / 255, P / 255, p); } _setPointsColorOrUV(e, t, r, n, i, s, a, f) { f = f ?? 0, r && e.updateFacetData(); const d = 2 * e.getBoundingInfo().boundingSphere.radius; let v = e.getVerticesData(J.PositionKind); const u = e.getIndices(), l = e.getVerticesData(J.UVKind + (f ? f + 1 : "")), P = e.getVerticesData(J.ColorKind), p = S.Zero(); e.computeWorldMatrix(); const c = e.getWorldMatrix(); if (!c.isIdentity()) { v = v.slice(0); for (let Er = 0; Er < v.length / 3; Er++) S.TransformCoordinatesFromFloatsToRef(v[3 * Er], v[3 * Er + 1], v[3 * Er + 2], c, p), v[3 * Er] = p.x, v[3 * Er + 1] = p.y, v[3 * Er + 2] = p.z; } let H = 0, T = 0, q = 0, b = 0, j = 0, w = 0, m = 0, I = 0, N = 0, k = 0, R = 0, y = 0, O = 0; const Y = S.Zero(), ee = S.Zero(), Z = S.Zero(), te = S.Zero(), fe = S.Zero(); let _ = 0, G = 0, L = 0, $ = 0, ae = 0, Pe = 0; const ge = at.Zero(), me = at.Zero(), Xe = at.Zero(), De = at.Zero(), ne = at.Zero(); let re = 0, ve = 0, qe = 0, ke = 0, be = 0, Fe = 0, Ke = 0, nt = 0, ut = 0, bt = 0, wt = 0, Tt = 0; const lr = Ir.Zero(), Qt = Ir.Zero(), tr = Ir.Zero(), br = Ir.Zero(), Xn = Ir.Zero(); let qr = 0, En = 0; a = a || 0; let Bn, Fi, Gt = new Ir(0, 0, 0, 0), xr = S.Zero(), Br = S.Zero(), nn = S.Zero(), sn = 0, Pn = S.Zero(), an = 0, Wn = 0; const rr = new Hi(S.Zero(), new S(1, 0, 0)); let Sr, nr = S.Zero(); for (let Er = 0; Er < u.length / 3; Er++) { T = u[3 * Er], q = u[3 * Er + 1], b = u[3 * Er + 2], j = v[3 * T], w = v[3 * T + 1], m = v[3 * T + 2], I = v[3 * q], N = v[3 * q + 1], k = v[3 * q + 2], R = v[3 * b], y = v[3 * b + 1], O = v[3 * b + 2], Y.set(j, w, m), ee.set(I, N, k), Z.set(R, y, O), ee.subtractToRef(Y, te), Z.subtractToRef(ee, fe), l && (_ = l[2 * T], G = l[2 * T + 1], L = l[2 * q], $ = l[2 * q + 1], ae = l[2 * b], Pe = l[2 * b + 1], ge.set(_, G), me.set(L, $), Xe.set(ae, Pe), me.subtractToRef(ge, De), Xe.subtractToRef(me, ne)), P && n && (re = P[4 * T], ve = P[4 * T + 1], qe = P[4 * T + 2], ke = P[4 * T + 3], be = P[4 * q], Fe = P[4 * q + 1], Ke = P[4 * q + 2], nt = P[4 * q + 3], ut = P[4 * b], bt = P[4 * b + 1], wt = P[4 * b + 2], Tt = P[4 * b + 3], lr.set(re, ve, qe, ke), Qt.set(be, Fe, Ke, nt), tr.set(ut, bt, wt, Tt), Qt.subtractToRef(lr, br), tr.subtractToRef(Qt, Xn)); let Rn, Vn, ki, as, Rr, Ii, is, ji; const _i = new Ne(0, 0, 0), oi = new Ne(0, 0, 0); let vr, Fr; for (let pr = 0; pr < t._groupDensity[Er]; pr++) H = this.particles.length, this._addParticle(H, t, this._groupCounter, Er + pr), Fr = this.particles[H], qr = Xt.RandomRange(0, 1), En = Xt.RandomRange(0, 1), Bn = Y.add(te.scale(qr)).add(fe.scale(qr * En)), r && (xr = e.getFacetNormal(Er).normalize().scale(-1), Br = te.clone().normalize(), nn = S.Cross(xr, Br), sn = Xt.RandomRange(0, 2 * Math.PI), Pn = Br.scale(Math.cos(sn)).add(nn.scale(Math.sin(sn))), sn = Xt.RandomRange(0.1, Math.PI / 2), nr = Pn.scale(Math.cos(sn)).add(xr.scale(Math.sin(sn))), rr.origin = Bn.add(nr.scale(1e-5)), rr.direction = nr, rr.length = d, Sr = rr.intersectsMesh(e), Sr.hit && (Wn = Sr.pickedPoint.subtract(Bn).length(), an = Xt.RandomRange(0, 1) * Wn, Bn.addInPlace(nr.scale(an)))), Fr.position = Bn.clone(), this._positions.push(Fr.position.x, Fr.position.y, Fr.position.z), n !== void 0 ? l && (Fi = ge.add(De.scale(qr)).add(ne.scale(qr * En)), n ? i && t._groupImageData !== null ? (Rn = t._groupImgWidth, Vn = t._groupImgHeight, vr = this._getColorIndicesForCoord(t, Math.round(Fi.x * Rn), Math.round(Fi.y * Vn), Rn), Fr.color = vr, this._colors.push(vr.r, vr.g, vr.b, vr.a)) : P ? (Gt = lr.add(br.scale(qr)).add(Xn.scale(qr * En)), Fr.color = new xt(Gt.x, Gt.y, Gt.z, Gt.w), this._colors.push(Gt.x, Gt.y, Gt.z, Gt.w)) : (Gt = lr.set(Math.random(), Math.random(), Math.random(), 1), Fr.color = new xt(Gt.x, Gt.y, Gt.z, Gt.w), this._colors.push(Gt.x, Gt.y, Gt.z, Gt.w)) : (Fr.uv = Fi.clone(), this._uvs.push(Fr.uv.x, Fr.uv.y))) : (s ? (_i.set(s.r, s.g, s.b), ki = Xt.RandomRange(-a, a), as = Xt.RandomRange(-a, a), ji = _i.toHSV(), Rr = ji.r, Ii = ji.g + ki, is = ji.b + as, Ii < 0 && (Ii = 0), Ii > 1 && (Ii = 1), is < 0 && (is = 0), is > 1 && (is = 1), Ne.HSVtoRGBToRef(Rr, Ii, is, oi), Gt.set(oi.r, oi.g, oi.b, 1)) : Gt = lr.set(Math.random(), Math.random(), Math.random(), 1), Fr.color = new xt(Gt.x, Gt.y, Gt.z, Gt.w), this._colors.push(Gt.x, Gt.y, Gt.z, Gt.w)); } } // stores mesh texture in dynamic texture for color pixel retrieval // when pointColor type is color for surface points _colorFromTexture(e, t, r) { if (e.material === null) { Se.Warn(e.name + "has no material."), t._groupImageData = null, this._setPointsColorOrUV(e, t, r, !0, !1); return; } const i = e.material.getActiveTextures(); if (i.length === 0) { Se.Warn(e.name + "has no usable texture."), t._groupImageData = null, this._setPointsColorOrUV(e, t, r, !0, !1); return; } const s = e.clone(); s.setEnabled(!1), this._promises.push(new Promise((a) => { ls.WhenAllReady(i, () => { let f = t._textureNb; f < 0 && (f = 0), f > i.length - 1 && (f = i.length - 1); const o = () => { t._groupImgWidth = i[f].getSize().width, t._groupImgHeight = i[f].getSize().height, this._setPointsColorOrUV(s, t, r, !0, !0, void 0, void 0, i[f].coordinatesIndex), s.dispose(), a(); }; t._groupImageData = null; const d = i[f].readPixels(); d ? d.then((v) => { t._groupImageData = v, o(); }) : o(); }); })); } // calculates the point density per facet of a mesh for surface points _calculateDensity(e, t, r) { let n = new Array(), i, s, a, f, o, d, v, u, l, P, p, c; const H = S.Zero(), T = S.Zero(), q = S.Zero(), b = S.Zero(), j = S.Zero(), w = S.Zero(); let m, I, N, k, R; const y = new Array(); let O = 0; const Y = r.length / 3; for (let _ = 0; _ < Y; _++) i = r[3 * _], s = r[3 * _ + 1], a = r[3 * _ + 2], f = t[3 * i], o = t[3 * i + 1], d = t[3 * i + 2], v = t[3 * s], u = t[3 * s + 1], l = t[3 * s + 2], P = t[3 * a], p = t[3 * a + 1], c = t[3 * a + 2], H.set(f, o, d), T.set(v, u, l), q.set(P, p, c), T.subtractToRef(H, b), q.subtractToRef(T, j), q.subtractToRef(H, w), m = b.length(), I = j.length(), N = w.length(), k = (m + I + N) / 2, R = Math.sqrt(k * (k - m) * (k - I) * (k - N)), O += R, y[_] = R; let ee = 0; for (let _ = 0; _ < Y; _++) n[_] = Math.floor(e * y[_] / O), ee += n[_]; const Z = e - ee, te = Math.floor(Z / Y), fe = Z % Y; te > 0 && (n = n.map((_) => _ + te)); for (let _ = 0; _ < fe; _++) n[_] += 1; return n; } /** * Adds points to the PCS in random positions within a unit sphere * @param nb (positive integer) the number of particles to be created from this model * @param pointFunction is an optional javascript function to be called for each particle on PCS creation * @returns the number of groups in the system */ addPoints(e, t = this._randomUnitVector) { const r = new WC(this._groupCounter, t); let n, i = this.nbParticles; for (let s = 0; s < e; s++) n = this._addParticle(i, r, this._groupCounter, s), r && r._positionFunction && r._positionFunction(n, i, s), this._positions.push(n.position.x, n.position.y, n.position.z), n.color && this._colors.push(n.color.r, n.color.g, n.color.b, n.color.a), n.uv && this._uvs.push(n.uv.x, n.uv.y), i++; return this.nbParticles += e, this._groupCounter++, this._groupCounter; } /** * Adds points to the PCS from the surface of the model shape * @param mesh is any Mesh object that will be used as a surface model for the points * @param nb (positive integer) the number of particles to be created from this model * @param colorWith determines whether a point is colored using color (default), uv, random, stated or none (invisible) * @param color (color4) to be used when colorWith is stated or color (number) when used to specify texture position * @param range (number from 0 to 1) to determine the variation in shape and tone for a stated color * @returns the number of groups in the system */ addSurfacePoints(e, t, r, n, i) { let s = r || Iv.Random; (isNaN(s) || s < 0 || s > 3) && (s = Iv.Random); const a = e.getVerticesData(J.PositionKind), f = e.getIndices(); this._groups.push(this._groupCounter); const o = new WC(this._groupCounter, null); switch (o._groupDensity = this._calculateDensity(t, a, f), s === Iv.Color ? o._textureNb = n || 0 : n = n || new xt(1, 1, 1, 1), s) { case Iv.Color: this._colorFromTexture(e, o, !1); break; case Iv.UV: this._setPointsColorOrUV(e, o, !1, !1, !1); break; case Iv.Random: this._setPointsColorOrUV(e, o, !1); break; case Iv.Stated: this._setPointsColorOrUV(e, o, !1, void 0, void 0, n, i); break; } return this.nbParticles += t, this._groupCounter++, this._groupCounter - 1; } /** * Adds points to the PCS inside the model shape * @param mesh is any Mesh object that will be used as a surface model for the points * @param nb (positive integer) the number of particles to be created from this model * @param colorWith determines whether a point is colored using color (default), uv, random, stated or none (invisible) * @param color (color4) to be used when colorWith is stated or color (number) when used to specify texture position * @param range (number from 0 to 1) to determine the variation in shape and tone for a stated color * @returns the number of groups in the system */ addVolumePoints(e, t, r, n, i) { let s = r || Iv.Random; (isNaN(s) || s < 0 || s > 3) && (s = Iv.Random); const a = e.getVerticesData(J.PositionKind), f = e.getIndices(); this._groups.push(this._groupCounter); const o = new WC(this._groupCounter, null); switch (o._groupDensity = this._calculateDensity(t, a, f), s === Iv.Color ? o._textureNb = n || 0 : n = n || new xt(1, 1, 1, 1), s) { case Iv.Color: this._colorFromTexture(e, o, !0); break; case Iv.UV: this._setPointsColorOrUV(e, o, !0, !1, !1); break; case Iv.Random: this._setPointsColorOrUV(e, o, !0); break; case Iv.Stated: this._setPointsColorOrUV(e, o, !0, void 0, void 0, n, i); break; } return this.nbParticles += t, this._groupCounter++, this._groupCounter - 1; } /** * Sets all the particles : this method actually really updates the mesh according to the particle positions, rotations, colors, textures, etc. * This method calls `updateParticle()` for each particle of the SPS. * For an animated SPS, it is usually called within the render loop. * @param start The particle index in the particle array where to start to compute the particle property values _(default 0)_ * @param end The particle index in the particle array where to stop to compute the particle property values _(default nbParticle - 1)_ * @param update If the mesh must be finally updated on this call after all the particle computations _(default true)_ * @returns the PCS. */ setParticles(e = 0, t = this.nbParticles - 1, r = !0) { var n, i; if (!this._updatable || !this._isReady) return this; this.beforeUpdateParticles(e, t, r); const s = ue.Matrix[0], a = this.mesh, f = this._colors32, o = this._positions32, d = this._uvs32, v = ue.Vector3, u = v[5].copyFromFloats(1, 0, 0), l = v[6].copyFromFloats(0, 1, 0), P = v[7].copyFromFloats(0, 0, 1), p = v[8].setAll(Number.MAX_VALUE), c = v[9].setAll(-Number.MAX_VALUE); he.IdentityToRef(s); let H = 0; if (!((n = this.mesh) === null || n === void 0) && n.isFacetDataEnabled && (this._computeBoundingBox = !0), t = t >= this.nbParticles ? this.nbParticles - 1 : t, this._computeBoundingBox && (e != 0 || t != this.nbParticles - 1)) { const j = (i = this.mesh) === null || i === void 0 ? void 0 : i.getBoundingInfo(); j && (p.copyFrom(j.minimum), c.copyFrom(j.maximum)); } H = 0; let T = 0, q = 0, b = 0; for (let j = e; j <= t; j++) { const w = this.particles[j]; H = w.idx, T = 3 * H, q = 4 * H, b = 2 * H, this.updateParticle(w); const m = w._rotationMatrix, I = w.position, N = w._globalPosition; if (this._computeParticleRotation && w.getRotationMatrix(s), w.parentId !== null) { const $ = this.particles[w.parentId], ae = $._rotationMatrix, Pe = $._globalPosition, ge = I.x * ae[1] + I.y * ae[4] + I.z * ae[7], me = I.x * ae[0] + I.y * ae[3] + I.z * ae[6], Xe = I.x * ae[2] + I.y * ae[5] + I.z * ae[8]; if (N.x = Pe.x + me, N.y = Pe.y + ge, N.z = Pe.z + Xe, this._computeParticleRotation) { const De = s.m; m[0] = De[0] * ae[0] + De[1] * ae[3] + De[2] * ae[6], m[1] = De[0] * ae[1] + De[1] * ae[4] + De[2] * ae[7], m[2] = De[0] * ae[2] + De[1] * ae[5] + De[2] * ae[8], m[3] = De[4] * ae[0] + De[5] * ae[3] + De[6] * ae[6], m[4] = De[4] * ae[1] + De[5] * ae[4] + De[6] * ae[7], m[5] = De[4] * ae[2] + De[5] * ae[5] + De[6] * ae[8], m[6] = De[8] * ae[0] + De[9] * ae[3] + De[10] * ae[6], m[7] = De[8] * ae[1] + De[9] * ae[4] + De[10] * ae[7], m[8] = De[8] * ae[2] + De[9] * ae[5] + De[10] * ae[8]; } } else if (N.x = 0, N.y = 0, N.z = 0, this._computeParticleRotation) { const $ = s.m; m[0] = $[0], m[1] = $[1], m[2] = $[2], m[3] = $[4], m[4] = $[5], m[5] = $[6], m[6] = $[8], m[7] = $[9], m[8] = $[10]; } const R = v[11]; w.translateFromPivot ? R.setAll(0) : R.copyFrom(w.pivot); const y = v[0]; y.copyFrom(w.position); const O = y.x - w.pivot.x, Y = y.y - w.pivot.y, ee = y.z - w.pivot.z; let Z = O * m[0] + Y * m[3] + ee * m[6], te = O * m[1] + Y * m[4] + ee * m[7], fe = O * m[2] + Y * m[5] + ee * m[8]; Z += R.x, te += R.y, fe += R.z; const _ = o[T] = N.x + u.x * Z + l.x * te + P.x * fe, G = o[T + 1] = N.y + u.y * Z + l.y * te + P.y * fe, L = o[T + 2] = N.z + u.z * Z + l.z * te + P.z * fe; if (this._computeBoundingBox && (p.minimizeInPlaceFromFloats(_, G, L), c.maximizeInPlaceFromFloats(_, G, L)), this._computeParticleColor && w.color) { const $ = w.color, ae = this._colors32; ae[q] = $.r, ae[q + 1] = $.g, ae[q + 2] = $.b, ae[q + 3] = $.a; } if (this._computeParticleTexture && w.uv) { const $ = w.uv, ae = this._uvs32; ae[b] = $.x, ae[b + 1] = $.y; } } return a && (r && (this._computeParticleColor && a.updateVerticesData(J.ColorKind, f, !1, !1), this._computeParticleTexture && a.updateVerticesData(J.UVKind, d, !1, !1), a.updateVerticesData(J.PositionKind, o, !1, !1)), this._computeBoundingBox && (a.hasBoundingInfo ? a.getBoundingInfo().reConstruct(p, c, a._worldMatrix) : a.buildBoundingInfo(p, c, a._worldMatrix))), this.afterUpdateParticles(e, t, r), this; } /** * Disposes the PCS. */ dispose() { var e; (e = this.mesh) === null || e === void 0 || e.dispose(), this.vars = null, this._positions = null, this._indices = null, this._normals = null, this._uvs = null, this._colors = null, this._indices32 = null, this._positions32 = null, this._uvs32 = null, this._colors32 = null; } /** * Visibility helper : Recomputes the visible size according to the mesh bounding box * doc : * @returns the PCS. */ refreshVisibleSize() { var e; return this._isVisibilityBoxLocked || (e = this.mesh) === null || e === void 0 || e.refreshBoundingInfo(), this; } /** * Visibility helper : Sets the size of a visibility box, this sets the underlying mesh bounding box. * @param size the size (float) of the visibility box * note : this doesn't lock the PCS mesh bounding box. * doc : */ setVisibilityBox(e) { if (!this.mesh) return; const t = e / 2; this.mesh.buildBoundingInfo(new S(-t, -t, -t), new S(t, t, t)); } /** * Gets whether the PCS is always visible or not * doc : */ get isAlwaysVisible() { return this._alwaysVisible; } /** * Sets the PCS as always visible or not * doc : */ set isAlwaysVisible(e) { this.mesh && (this._alwaysVisible = e, this.mesh.alwaysSelectAsActiveMesh = e); } /** * Tells to `setParticles()` to compute the particle rotations or not * Default value : false. The PCS is faster when it's set to false * Note : particle rotations are only applied to parent particles * Note : the particle rotations aren't stored values, so setting `computeParticleRotation` to false will prevents the particle to rotate */ set computeParticleRotation(e) { this._computeParticleRotation = e; } /** * Tells to `setParticles()` to compute the particle colors or not. * Default value : true. The PCS is faster when it's set to false. * Note : the particle colors are stored values, so setting `computeParticleColor` to false will keep yet the last colors set. */ set computeParticleColor(e) { this._computeParticleColor = e; } set computeParticleTexture(e) { this._computeParticleTexture = e; } /** * Gets if `setParticles()` computes the particle colors or not. * Default value : false. The PCS is faster when it's set to false. * Note : the particle colors are stored values, so setting `computeParticleColor` to false will keep yet the last colors set. */ get computeParticleColor() { return this._computeParticleColor; } /** * Gets if `setParticles()` computes the particle textures or not. * Default value : false. The PCS is faster when it's set to false. * Note : the particle textures are stored values, so setting `computeParticleTexture` to false will keep yet the last colors set. */ get computeParticleTexture() { return this._computeParticleTexture; } /** * Tells to `setParticles()` to compute or not the mesh bounding box when computing the particle positions. */ set computeBoundingBox(e) { this._computeBoundingBox = e; } /** * Gets if `setParticles()` computes or not the mesh bounding box when computing the particle positions. */ get computeBoundingBox() { return this._computeBoundingBox; } // ======================================================================= // Particle behavior logic // these following methods may be overwritten by users to fit their needs /** * This function does nothing. It may be overwritten to set all the particle first values. * The PCS doesn't call this function, you may have to call it by your own. * doc : */ initParticles() { } /** * This function does nothing. It may be overwritten to recycle a particle * The PCS doesn't call this function, you can to call it * doc : * @param particle The particle to recycle * @returns the recycled particle */ recycleParticle(e) { return e; } /** * Updates a particle : this function should be overwritten by the user. * It is called on each particle by `setParticles()`. This is the place to code each particle behavior. * doc : * @example : just set a particle position or velocity and recycle conditions * @param particle The particle to update * @returns the updated particle */ updateParticle(e) { return e; } /** * This will be called before any other treatment by `setParticles()` and will be passed three parameters. * This does nothing and may be overwritten by the user. * @param start the particle index in the particle array where to start to iterate, same than the value passed to setParticle() * @param stop the particle index in the particle array where to stop to iterate, same than the value passed to setParticle() * @param update the boolean update value actually passed to setParticles() */ // eslint-disable-next-line @typescript-eslint/no-unused-vars beforeUpdateParticles(e, t, r) { } /** * This will be called by `setParticles()` after all the other treatments and just before the actual mesh update. * This will be passed three parameters. * This does nothing and may be overwritten by the user. * @param start the particle index in the particle array where to start to iterate, same than the value passed to setParticle() * @param stop the particle index in the particle array where to stop to iterate, same than the value passed to setParticle() * @param update the boolean update value actually passed to setParticles() */ // eslint-disable-next-line @typescript-eslint/no-unused-vars afterUpdateParticles(e, t, r) { } } Object.defineProperty(jn.prototype, "physicsImpostor", { get: function() { return this._physicsImpostor; }, set: function(A) { this._physicsImpostor !== A && (this._disposePhysicsObserver && this.onDisposeObservable.remove(this._disposePhysicsObserver), this._physicsImpostor = A, A && (this._disposePhysicsObserver = this.onDisposeObservable.add(() => { this.physicsImpostor && (this.physicsImpostor.dispose( /*!doNotRecurse*/ ), this.physicsImpostor = null); }))); }, enumerable: !0, configurable: !0 }); jn.prototype.getPhysicsImpostor = function() { return this.physicsImpostor; }; jn.prototype.applyImpulse = function(A, e) { return this.physicsImpostor ? (this.physicsImpostor.applyImpulse(A, e), this) : this; }; jn.prototype.setPhysicsLinkWith = function(A, e, t, r) { return !this.physicsImpostor || !A.physicsImpostor ? this : (this.physicsImpostor.createJoint(A.physicsImpostor, ta.HingeJoint, { mainPivot: e, connectedPivot: t, nativeParams: r }), this); }; class jy { /** * * @returns physics plugin version */ getPluginVersion() { return this._physicsPlugin.getPluginVersion(); } /** * Factory used to create the default physics plugin. * @returns The default physics plugin */ static DefaultPluginFactory() { throw qn(""); } /** * Creates a new Physics Engine * @param gravity defines the gravity vector used by the simulation * @param _physicsPlugin defines the plugin to use (CannonJS by default) */ constructor(e, t = jy.DefaultPluginFactory()) { this._physicsPlugin = t, this._physicsBodies = [], this._subTimeStep = 0, e = e || new S(0, -9.807, 0), this.setGravity(e), this.setTimeStep(); } /** * Sets the gravity vector used by the simulation * @param gravity defines the gravity vector to use */ setGravity(e) { this.gravity = e, this._physicsPlugin.setGravity(this.gravity); } /** * Set the time step of the physics engine. * Default is 1/60. * To slow it down, enter 1/600 for example. * To speed it up, 1/30 * @param newTimeStep defines the new timestep to apply to this world. */ setTimeStep(e = 1 / 60) { this._physicsPlugin.setTimeStep(e); } /** * Get the time step of the physics engine. * @returns the current time step */ getTimeStep() { return this._physicsPlugin.getTimeStep(); } /** * Set the sub time step of the physics engine. * Default is 0 meaning there is no sub steps * To increase physics resolution precision, set a small value (like 1 ms) * @param subTimeStep defines the new sub timestep used for physics resolution. */ setSubTimeStep(e = 0) { this._subTimeStep = e; } /** * Get the sub time step of the physics engine. * @returns the current sub time step */ getSubTimeStep() { return this._subTimeStep; } /** * Release all resources */ dispose() { this._physicsPlugin.dispose(); } /** * Gets the name of the current physics plugin * @returns the name of the plugin */ getPhysicsPluginName() { return this._physicsPlugin.name; } /** * Adding a new impostor for the impostor tracking. * This will be done by the impostor itself. * @param impostor the impostor to add */ /** * Called by the scene. No need to call it. * @param delta defines the timespan between frames */ _step(e) { e > 0.1 ? e = 0.1 : e <= 0 && (e = 1 / 60), this._physicsPlugin.executeStep(e, this._physicsBodies); } /** * Add a body as an active component of this engine * @param body */ addBody(e) { this._physicsBodies.push(e); } /** * Removes a particular body from this engine */ removeBody(e) { const t = this._physicsBodies.indexOf(e); t > -1 && this._physicsBodies.splice(t, 1); } /** * Returns an array of bodies added to this engine */ getBodies() { return this._physicsBodies; } /** * Gets the current plugin used to run the simulation * @returns current plugin */ getPhysicsPlugin() { return this._physicsPlugin; } /** * Does a raycast in the physics world * @param from when should the ray start? * @param to when should the ray end? * @param result resulting PhysicsRaycastResult */ raycastToRef(e, t, r, n) { this._physicsPlugin.raycast(e, t, r, n); } /** * Does a raycast in the physics world * @param from when should the ray start? * @param to when should the ray end? * @returns PhysicsRaycastResult */ raycast(e, t, r) { const n = new iV(); return this._physicsPlugin.raycast(e, t, n, r), n; } } class wy { /** * Constructs a new physics body for the given node. * @param transformNode - The Transform Node to construct the physics body for. For better performance, it is advised that this node does not have a parent. * @param motionType - The motion type of the physics body. The options are: * - PhysicsMotionType.STATIC - Static bodies are not moving and unaffected by forces or collisions. They are good for level boundaries or terrain. * - PhysicsMotionType.DYNAMIC - Dynamic bodies are fully simulated. They can move and collide with other objects. * - PhysicsMotionType.ANIMATED - They behave like dynamic bodies, but they won't be affected by other bodies, but still push other bodies out of the way. * @param startsAsleep - Whether the physics body should start in a sleeping state (not a guarantee). Defaults to false. * @param scene - The scene containing the physics engine. * * This code is useful for creating a physics body for a given Transform Node in a scene. * It checks the version of the physics engine and the physics plugin, and initializes the body accordingly. * It also sets the node's rotation quaternion if it is not already set. Finally, it adds the body to the physics engine. */ constructor(e, t, r, n) { if (this._pluginData = void 0, this._pluginDataInstances = [], this._collisionCBEnabled = !1, this._collisionEndedCBEnabled = !1, this.disablePreStep = !0, this._isDisposed = !1, !n) return; const i = n.getPhysicsEngine(); if (!i) throw new Error("No Physics Engine available."); if (this._physicsEngine = i, i.getPluginVersion() != 2) throw new Error("Plugin version is incorrect. Expected version 2."); const s = i.getPhysicsPlugin(); if (!s) throw new Error("No Physics Plugin available."); this._physicsPlugin = s, e.rotationQuaternion || (e.rotationQuaternion = Ze.FromEulerAngles(e.rotation.x, e.rotation.y, e.rotation.z)), this.startAsleep = r; const a = e; a.hasThinInstances ? this._physicsPlugin.initBodyInstances(this, t, a) : (e.parent && e.computeWorldMatrix(!0), this._physicsPlugin.initBody(this, t, e.absolutePosition, e.absoluteRotationQuaternion)), this.transformNode = e, e.physicsBody = this, i.addBody(this), this._nodeDisposeObserver = e.onDisposeObservable.add(() => { this.dispose(); }); } /** * Returns the string "PhysicsBody". * @returns "PhysicsBody" */ getClassName() { return "PhysicsBody"; } /** * Clone the PhysicsBody to a new body and assign it to the transformNode parameter * @param transformNode transformNode that will be used for the cloned PhysicsBody * @returns the newly cloned PhysicsBody */ clone(e) { const t = new wy(e, this.getMotionType(), this.startAsleep, this.transformNode.getScene()); return t.shape = this.shape, t.setMassProperties(this.getMassProperties()), t.setLinearDamping(this.getLinearDamping()), t.setAngularDamping(this.getAngularDamping()), t; } /** * If a physics body is connected to an instanced node, update the number physic instances to match the number of node instances. */ updateBodyInstances() { const e = this.transformNode; e.hasThinInstances && this._physicsPlugin.updateBodyInstances(this, e); } /** * This returns the number of internal instances of the physics body */ get numInstances() { return this._pluginDataInstances.length; } /** * Sets the shape of the physics body. * @param shape - The shape of the physics body. * * This method is useful for setting the shape of the physics body, which is necessary for the physics engine to accurately simulate the body's behavior. * The shape is used to calculate the body's mass, inertia, and other properties. */ set shape(e) { this._physicsPlugin.setShape(this, e); } /** * Retrieves the physics shape associated with this object. * * @returns The physics shape associated with this object, or `undefined` if no * shape is associated. * * This method is useful for retrieving the physics shape associated with this object, * which can be used to apply physical forces to the object or to detect collisions. */ get shape() { return this._physicsPlugin.getShape(this); } /** * Sets the event mask for the physics engine. * * @param eventMask - A bitmask that determines which events will be sent to the physics engine. * * This method is useful for setting the event mask for the physics engine, which determines which events * will be sent to the physics engine. This allows the user to control which events the physics engine will respond to. */ setEventMask(e, t) { this._physicsPlugin.setEventMask(this, e, t); } /** * Gets the event mask of the physics engine. * * @returns The event mask of the physics engine. * * This method is useful for getting the event mask of the physics engine, * which is used to determine which events the engine will respond to. * This is important for ensuring that the engine is responding to the correct events and not * wasting resources on unnecessary events. */ getEventMask(e) { return this._physicsPlugin.getEventMask(this, e); } /** * Sets the motion type of the physics body. Can be STATIC, DYNAMIC, or ANIMATED. */ setMotionType(e, t) { this._physicsPlugin.setMotionType(this, e, t); } /** * Gets the motion type of the physics body. Can be STATIC, DYNAMIC, or ANIMATED. */ getMotionType(e) { return this._physicsPlugin.getMotionType(this, e); } /** * Computes the mass properties of the physics object, based on the set of physics shapes this body uses. * This method is useful for computing the initial mass properties of a physics object, such as its mass, * inertia, and center of mass; these values are important for accurately simulating the physics of the * object in the physics engine, and computing values based on the shape will provide you with reasonable * intial values, which you can then customize. */ computeMassProperties(e) { return this._physicsPlugin.computeMassProperties(this, e); } /** * Sets the mass properties of the physics object. * * @param massProps - The mass properties to set. * @param instanceIndex - The index of the instance to set the mass properties for. If not defined, the mass properties will be set for all instances. * * This method is useful for setting the mass properties of a physics object, such as its mass, * inertia, and center of mass. This is important for accurately simulating the physics of the object in the physics engine. */ setMassProperties(e, t) { this._physicsPlugin.setMassProperties(this, e, t); } /** * Retrieves the mass properties of the object. * * @returns The mass properties of the object. * * This method is useful for physics simulations, as it allows the user to * retrieve the mass properties of the object, such as its mass, center of mass, * and moment of inertia. This information is necessary for accurate physics * simulations. */ getMassProperties(e) { return this._physicsPlugin.getMassProperties(this, e); } /** * Sets the linear damping of the physics body. * * @param damping - The linear damping value. * * This method is useful for controlling the linear damping of the physics body, * which is the rate at which the body's velocity decreases over time. This is useful for simulating * the effects of air resistance or other forms of friction. */ setLinearDamping(e, t) { this._physicsPlugin.setLinearDamping(this, e, t); } /** * Gets the linear damping of the physics body. * @returns The linear damping of the physics body. * * This method is useful for retrieving the linear damping of the physics body, which is the amount of * resistance the body has to linear motion. This is useful for simulating realistic physics behavior * in a game. */ getLinearDamping(e) { return this._physicsPlugin.getLinearDamping(this, e); } /** * Sets the angular damping of the physics body. * @param damping The angular damping of the body. * * This method is useful for controlling the angular velocity of a physics body. * By setting the damping, the body's angular velocity will be reduced over time, simulating the effect of friction. * This can be used to create realistic physical behavior in a physics engine. */ setAngularDamping(e, t) { this._physicsPlugin.setAngularDamping(this, e, t); } /** * Gets the angular damping of the physics body. * * @returns The angular damping of the physics body. * * This method is useful for getting the angular damping of the physics body, * which is the rate of reduction of the angular velocity over time. * This is important for simulating realistic physics behavior in a game. */ getAngularDamping(e) { return this._physicsPlugin.getAngularDamping(this, e); } /** * Sets the linear velocity of the physics object. * @param linVel - The linear velocity to set. * * This method is useful for setting the linear velocity of a physics object, * which is necessary for simulating realistic physics in a game engine. * By setting the linear velocity, the physics object will move in the direction and speed specified by the vector. * This allows for realistic physics simulations, such as simulating the motion of a ball rolling down a hill. */ setLinearVelocity(e, t) { this._physicsPlugin.setLinearVelocity(this, e, t); } /** * Gets the linear velocity of the physics body and stores it in the given vector3. * @param linVel - The vector3 to store the linear velocity in. * * This method is useful for getting the linear velocity of a physics body in a physics engine. * This can be used to determine the speed and direction of the body, which can be used to calculate the motion of the body. */ getLinearVelocityToRef(e, t) { return this._physicsPlugin.getLinearVelocityToRef(this, e, t); } /** * Gets the linear velocity of the physics body as a new vector3. * @returns The linear velocity of the physics body. * * This method is useful for getting the linear velocity of a physics body in a physics engine. * This can be used to determine the speed and direction of the body, which can be used to calculate the motion of the body. */ getLinearVelocity(e) { const t = new S(); return this.getLinearVelocityToRef(t, e), t; } /** * Sets the angular velocity of the physics object. * @param angVel - The angular velocity to set. * * This method is useful for setting the angular velocity of a physics object, which is necessary for * simulating realistic physics behavior. The angular velocity is used to determine the rate of rotation of the object, * which is important for simulating realistic motion. */ setAngularVelocity(e, t) { this._physicsPlugin.setAngularVelocity(this, e, t); } /** * Gets the angular velocity of the physics body and stores it in the given vector3. * @param angVel - The vector3 to store the angular velocity in. * * This method is useful for getting the angular velocity of a physics body, which can be used to determine the body's * rotational speed. This information can be used to create realistic physics simulations. */ getAngularVelocityToRef(e, t) { return this._physicsPlugin.getAngularVelocityToRef(this, e, t); } /** * Gets the angular velocity of the physics body as a new vector3. * @returns The angular velocity of the physics body. * * This method is useful for getting the angular velocity of a physics body, which can be used to determine the body's * rotational speed. This information can be used to create realistic physics simulations. */ getAngularVelocity(e) { const t = new S(); return this.getAngularVelocityToRef(t, e), t; } /** * Applies an impulse to the physics object. * * @param impulse The impulse vector. * @param location The location of the impulse. * @param instanceIndex For a instanced body, the instance to where the impulse should be applied. If not specified, the impulse is applied to all instances. * * This method is useful for applying an impulse to a physics object, which can be used to simulate physical forces such as gravity, * collisions, and explosions. This can be used to create realistic physics simulations in a game or other application. */ applyImpulse(e, t, r) { this._physicsPlugin.applyImpulse(this, e, t, r); } /** * Applies a force to the physics object. * * @param force The force vector. * @param location The location of the force. * @param instanceIndex For a instanced body, the instance to where the force should be applied. If not specified, the force is applied to all instances. * * This method is useful for applying a force to a physics object, which can be used to simulate physical forces such as gravity, * collisions, and explosions. This can be used to create realistic physics simulations in a game or other application. */ applyForce(e, t, r) { this._physicsPlugin.applyForce(this, e, t, r); } /** * Retrieves the geometry of the body from the physics plugin. * * @returns The geometry of the body. * * This method is useful for retrieving the geometry of the body from the physics plugin, which can be used for various physics calculations. */ getGeometry() { return this._physicsPlugin.getBodyGeometry(this); } /** * Returns an observable that will be notified for when a collision starts or continues for this PhysicsBody * @returns Observable */ getCollisionObservable() { return this._physicsPlugin.getCollisionObservable(this); } /** * Returns an observable that will be notified when the body has finished colliding with another body * @returns */ getCollisionEndedObservable() { return this._physicsPlugin.getCollisionEndedObservable(this); } /** * Enable or disable collision callback for this PhysicsBody. * @param enabled true if PhysicsBody's collision will rise a collision event and notifies the observable */ setCollisionCallbackEnabled(e) { this._collisionCBEnabled = e, this._physicsPlugin.setCollisionCallbackEnabled(this, e); } setCollisionEndedCallbackEnabled(e) { this._collisionEndedCBEnabled = e, this._physicsPlugin.setCollisionEndedCallbackEnabled(this, e); } /* * Get the center of the object in world space. * @param instanceIndex - If this body is instanced, the index of the instance to get the center for. * @returns geometric center of the associated mesh */ getObjectCenterWorld(e) { const t = new S(); return this.getObjectCenterWorldToRef(t, e); } /* * Get the center of the object in world space. * @param ref - The vector3 to store the result in. * @param instanceIndex - If this body is instanced, the index of the instance to get the center for. * @returns geometric center of the associated mesh */ getObjectCenterWorldToRef(e, t) { var r; if (((r = this._pluginDataInstances) === null || r === void 0 ? void 0 : r.length) > 0) { const n = t || 0, i = this.transformNode._thinInstanceDataStorage.matrixData; i && e.set(i[n * 16 + 12], i[n * 16 + 13], i[n * 16 + 14]); } else e.copyFrom(this.transformNode.position); return e; } /** * Adds a constraint to the physics engine. * * @param childBody - The body to which the constraint will be applied. * @param constraint - The constraint to be applied. * @param instanceIndex - If this body is instanced, the index of the instance to which the constraint will be applied. If not specified, no constraint will be applied. * @param childInstanceIndex - If the child body is instanced, the index of the instance to which the constraint will be applied. If not specified, no constraint will be applied. * */ addConstraint(e, t, r, n) { this._physicsPlugin.addConstraint(this, e, t, r, n); } /** * Sync with a bone * @param bone The bone that the impostor will be synced to. * @param boneMesh The mesh that the bone is influencing. * @param jointPivot The pivot of the joint / bone in local space. * @param distToJoint Optional distance from the impostor to the joint. * @param adjustRotation Optional quaternion for adjusting the local rotation of the bone. * @param boneAxis Optional vector3 axis the bone is aligned with */ syncWithBone(e, t, r, n, i, s) { const a = this.transformNode; if (a.rotationQuaternion) if (i) { const d = ue.Quaternion[0]; e.getRotationQuaternionToRef(ai.WORLD, t, d), d.multiplyToRef(i, a.rotationQuaternion); } else e.getRotationQuaternionToRef(ai.WORLD, t, a.rotationQuaternion); const f = ue.Vector3[0], o = ue.Vector3[1]; s || (s = ue.Vector3[2], s.x = 0, s.y = 1, s.z = 0), e.getDirectionToRef(s, t, o), e.getAbsolutePositionToRef(t, f), n == null && r && (n = r.length()), n != null && (f.x += o.x * n, f.y += o.y * n, f.z += o.z * n), a.setAbsolutePosition(f); } /** * Executes a callback on the body or all of the instances of a body * @param callback the callback to execute */ iterateOverAllInstances(e) { var t; if (((t = this._pluginDataInstances) === null || t === void 0 ? void 0 : t.length) > 0) for (let r = 0; r < this._pluginDataInstances.length; r++) e(this, r); else e(this, void 0); } /** * Sets the gravity factor of the physics body * @param factor the gravity factor to set * @param instanceIndex the instance of the body to set, if undefined all instances will be set */ setGravityFactor(e, t) { this._physicsPlugin.setGravityFactor(this, e, t); } /** * Gets the gravity factor of the physics body * @param instanceIndex the instance of the body to get, if undefined the value of first instance will be returned * @returns the gravity factor */ getGravityFactor(e) { return this._physicsPlugin.getGravityFactor(this, e); } /** * Set the target transformation (position and rotation) of the body, such that the body will set its velocity to reach that target * @param position The target position * @param rotation The target rotation * @param instanceIndex The index of the instance in an instanced body */ setTargetTransform(e, t, r) { this._physicsPlugin.setTargetTransform(this, e, t, r); } /** * Disposes the body from the physics engine. * * This method is useful for cleaning up the physics engine when a body is no longer needed. Disposing the body will free up resources and prevent memory leaks. */ dispose() { this._isDisposed || (this._collisionCBEnabled && this.setCollisionCallbackEnabled(!1), this._collisionEndedCBEnabled && this.setCollisionEndedCallbackEnabled(!1), this._nodeDisposeObserver && (this.transformNode.onDisposeObservable.remove(this._nodeDisposeObserver), this._nodeDisposeObserver = null), this._physicsEngine.removeBody(this), this._physicsPlugin.removeBody(this), this._physicsPlugin.disposeBody(this), this.transformNode.physicsBody = null, this._pluginData = null, this._pluginDataInstances.length = 0, this._isDisposed = !0); } } var I2; (function(A) { A[A.FREE = 0] = "FREE", A[A.LIMITED = 1] = "LIMITED", A[A.LOCKED = 2] = "LOCKED"; })(I2 || (I2 = {})); var MH; (function(A) { A[A.LINEAR_X = 0] = "LINEAR_X", A[A.LINEAR_Y = 1] = "LINEAR_Y", A[A.LINEAR_Z = 2] = "LINEAR_Z", A[A.ANGULAR_X = 3] = "ANGULAR_X", A[A.ANGULAR_Y = 4] = "ANGULAR_Y", A[A.ANGULAR_Z = 5] = "ANGULAR_Z", A[A.LINEAR_DISTANCE = 6] = "LINEAR_DISTANCE"; })(MH || (MH = {})); var kv; (function(A) { A[A.BALL_AND_SOCKET = 1] = "BALL_AND_SOCKET", A[A.DISTANCE = 2] = "DISTANCE", A[A.HINGE = 3] = "HINGE", A[A.SLIDER = 4] = "SLIDER", A[A.LOCK = 5] = "LOCK", A[A.PRISMATIC = 6] = "PRISMATIC", A[A.SIX_DOF = 7] = "SIX_DOF"; })(kv || (kv = {})); var Ff; (function(A) { A[A.SPHERE = 0] = "SPHERE", A[A.CAPSULE = 1] = "CAPSULE", A[A.CYLINDER = 2] = "CYLINDER", A[A.BOX = 3] = "BOX", A[A.CONVEX_HULL = 4] = "CONVEX_HULL", A[A.CONTAINER = 5] = "CONTAINER", A[A.MESH = 6] = "MESH", A[A.HEIGHTFIELD = 7] = "HEIGHTFIELD"; })(Ff || (Ff = {})); var _x; (function(A) { A[A.NONE = 0] = "NONE", A[A.VELOCITY = 1] = "VELOCITY", A[A.POSITION = 2] = "POSITION"; })(_x || (_x = {})); var q0; (function(A) { A.COLLISION_STARTED = "COLLISION_STARTED", A.COLLISION_CONTINUED = "COLLISION_CONTINUED", A.COLLISION_FINISHED = "COLLISION_FINISHED", A.TRIGGER_ENTERED = "TRIGGER_ENTERED", A.TRIGGER_EXITED = "TRIGGER_EXITED"; })(q0 || (q0 = {})); var W0; (function(A) { A[A.STATIC = 0] = "STATIC", A[A.ANIMATED = 1] = "ANIMATED", A[A.DYNAMIC = 2] = "DYNAMIC"; })(W0 || (W0 = {})); class P4 { /** * Constructs a new physics shape. * @param options The options for the physics shape. These are: * * type: The type of the shape. This can be one of the following: SPHERE, BOX, CAPSULE, CYLINDER, CONVEX_HULL, MESH, HEIGHTFIELD, CONTAINER * * parameters: The parameters of the shape. * * pluginData: The plugin data of the shape. This is used if you already have a reference to the object on the plugin side. * You need to specify either type or pluginData. * @param scene The scene the shape belongs to. * * This code is useful for creating a new physics shape with the given type, options, and scene. * It also checks that the physics engine and plugin version are correct. * If not, it throws an error. This ensures that the shape is created with the correct parameters and is compatible with the physics engine. */ constructor(e, t) { var r; if (this._pluginData = void 0, this._isTrigger = !1, this._isDisposed = !1, !t) return; const n = t.getPhysicsEngine(); if (!n) throw new Error("No Physics Engine available."); if (n.getPluginVersion() != 2) throw new Error("Plugin version is incorrect. Expected version 2."); const i = n.getPhysicsPlugin(); if (!i) throw new Error("No Physics Plugin available."); if (this._physicsPlugin = i, e.pluginData !== void 0 && e.pluginData !== null) this._pluginData = e.pluginData, this._type = this._physicsPlugin.getShapeType(this); else if (e.type !== void 0 && e.type !== null) { this._type = e.type; const s = (r = e.parameters) !== null && r !== void 0 ? r : {}; this._physicsPlugin.initShape(this, e.type, s); } } /** * Returns the string "PhysicsShape". * @returns "PhysicsShape" */ getClassName() { return "PhysicsShape"; } /** * Returns the type of the physics shape. * @returns The type of the physics shape. */ get type() { return this._type; } /** * Set the membership mask of a shape. This is a bitfield of arbitrary * "categories" to which the shape is a member. This is used in combination * with the collide mask to determine if this shape should collide with * another. * * @param membershipMask Bitfield of categories of this shape. */ set filterMembershipMask(e) { this._physicsPlugin.setShapeFilterMembershipMask(this, e); } /** * Get the membership mask of a shape. * @returns Bitmask of categories which this shape is a member of. */ get filterMembershipMask() { return this._physicsPlugin.getShapeFilterMembershipMask(this); } /** * Sets the collide mask of a shape. This is a bitfield of arbitrary * "categories" to which this shape collides with. Given two shapes, * the engine will check if the collide mask and membership overlap: * shapeA.filterMembershipMask & shapeB.filterCollideMask * * If this value is zero (i.e. shapeB only collides with categories * which shapeA is _not_ a member of) then the shapes will not collide. * * Note, the engine will also perform the same test with shapeA and * shapeB swapped; the shapes will not collide if either shape has * a collideMask which prevents collision with the other shape. * * @param collideMask Bitmask of categories this shape should collide with */ set filterCollideMask(e) { this._physicsPlugin.setShapeFilterCollideMask(this, e); } /** * * @returns Bitmask of categories that this shape should collide with */ get filterCollideMask() { return this._physicsPlugin.getShapeFilterCollideMask(this); } /** * * @param material */ set material(e) { this._physicsPlugin.setMaterial(this, e), this._material = e; } /** * Returns the material of the physics shape. * @returns The material of the physics shape. */ get material() { return this._material; } /** * Sets the density of the physics shape. * @param density The density of the physics shape. */ set density(e) { this._physicsPlugin.setDensity(this, e); } /** * Returns the density of the physics shape. * @returns The density of the physics shape. */ get density() { return this._physicsPlugin.getDensity(this); } /** * Utility to add a child shape to this container, * automatically computing the relative transform between * the container shape and the child instance. * * @param parentTransform The transform node associated with this shape * @param newChild The new PhysicsShape to add * @param childTransform The transform node associated with the child shape */ addChildFromParent(e, t, r) { const n = r.computeWorldMatrix(!0), i = e.computeWorldMatrix(!0), s = ue.Matrix[0]; n.multiplyToRef(he.Invert(i), s); const a = ue.Vector3[0], f = ue.Quaternion[0], o = ue.Vector3[1]; s.decompose(o, f, a), this._physicsPlugin.addChild(this, t, a, f, o); } /** * Adds a child shape to a container with an optional transform * @param newChild The new PhysicsShape to add * @param translation Optional position of the child shape relative to this shape * @param rotation Optional rotation of the child shape relative to this shape * @param scale Optional scale of the child shape relative to this shape */ addChild(e, t, r, n) { this._physicsPlugin.addChild(this, e, t, r, n); } /** * Removes a child shape from this shape. * @param childIndex The index of the child shape to remove */ removeChild(e) { this._physicsPlugin.removeChild(this, e); } /** * Returns the number of children of a physics shape. * @returns The number of children of a physics shape. */ getNumChildren() { return this._physicsPlugin.getNumChildren(this); } /** * Returns the bounding box of the physics shape. * @returns The bounding box of the physics shape. */ getBoundingBox() { return this._physicsPlugin.getBoundingBox(this); } set isTrigger(e) { this._isTrigger !== e && (this._isTrigger = e, this._physicsPlugin.setTrigger(this, e)); } get isTrigger() { return this._isTrigger; } /** * Dispose the shape and release its associated resources. */ dispose() { this._isDisposed || (this._physicsPlugin.disposeShape(this), this._isDisposed = !0); } } class EY extends P4 { /** * Constructor for the Sphere Shape * @param center local center of the sphere * @param radius radius * @param scene scene to attach to */ constructor(e, t, r) { super({ type: Ff.SPHERE, parameters: { center: e, radius: t } }, r); } /** * Derive an approximate sphere from the mesh. * @param mesh node from which to derive the sphere shape * @returns PhysicsShapeSphere */ static FromMesh(e) { const t = e.getBoundingInfo(), r = t.boundingSphere.center, n = t.boundingBox.extendSize, i = Math.max(n.x, n.y, n.z); return new EY(r, i, e.getScene()); } } class FY extends P4 { /** * * @param pointA Starting point that defines the capsule segment * @param pointB ending point of that same segment * @param radius radius * @param scene scene to attach to */ constructor(e, t, r, n) { super({ type: Ff.CAPSULE, parameters: { pointA: e, pointB: t, radius: r } }, n); } /** * Derive an approximate capsule from the mesh. Note, this is * not the optimal bounding capsule. * @param mesh Node from which to derive a cylinder shape */ static FromMesh(e) { const t = e.getBoundingInfo(), r = t.boundingBox.extendSize.x, n = new S(0, t.boundingBox.extendSize.y - r, 0), i = t.boundingBox.center.add(n), s = t.boundingBox.center.subtract(n); return new FY(i, s, r, e.getScene()); } } class NY extends P4 { /** * * @param pointA Starting point that defines the cylinder segment * @param pointB ending point of that same segment * @param radius radius * @param scene scene to attach to */ constructor(e, t, r, n) { super({ type: Ff.CYLINDER, parameters: { pointA: e, pointB: t, radius: r } }, n); } /** * Derive an approximate cylinder from the mesh. Note, this is * not the optimal bounding cylinder. * @param mesh Node from which to derive a cylinder shape */ static FromMesh(e) { const t = e.getBoundingInfo(), r = t.boundingBox.extendSize.x, n = new S(0, t.boundingBox.extendSize.y, 0), i = t.boundingBox.center.add(n), s = t.boundingBox.center.subtract(n); return new NY(i, s, r, e.getScene()); } } class QY extends P4 { /** * * @param center local center of the box * @param rotation local orientation * @param extents size of the box in each direction * @param scene scene to attach to */ constructor(e, t, r, n) { super({ type: Ff.BOX, parameters: { center: e, rotation: t, extents: r } }, n); } /** * * @param mesh * @returns PhysicsShapeBox */ static FromMesh(e) { const t = e.getBoundingInfo(), r = t.boundingBox.center, n = t.boundingBox.extendSize.scale(2); return new QY(r, Ze.Identity(), n, e.getScene()); } } class zhe extends P4 { /** * * @param mesh the mesh to be used as topology infos for the convex hull * @param scene scene to attach to */ constructor(e, t) { super({ type: Ff.CONVEX_HULL, parameters: { mesh: e } }, t); } } class Ghe extends P4 { /** * * @param mesh the mesh topology that will be used to create the shape * @param scene scene to attach to */ constructor(e, t) { super({ type: Ff.MESH, parameters: { mesh: e } }, t); } } class Zhe extends P4 { /** * Constructor of the Shape container * @param scene scene to attach to */ constructor(e) { super({ type: Ff.CONTAINER, parameters: {} }, e); } } class mD { /** * Constructs a new constraint for the physics constraint. * @param type The type of constraint to create. * @param options The options for the constraint. * @param scene The scene the constraint belongs to. * * This code is useful for creating a new constraint for the physics engine. It checks if the scene has a physics engine, and if the plugin version is correct. * If all checks pass, it initializes the constraint with the given type and options. */ constructor(e, t, r) { if (this._pluginData = void 0, !r) throw new Error("Missing scene parameter for constraint constructor."); const n = r.getPhysicsEngine(); if (!n) throw new Error("No Physics Engine available."); if (n.getPluginVersion() != 2) throw new Error("Plugin version is incorrect. Expected version 2."); const i = n.getPhysicsPlugin(); if (!i) throw new Error("No Physics Plugin available."); this._physicsPlugin = i, this._options = t, this._type = e; } /** * Gets the type of the constraint. * * @returns The type of the constraint. * */ get type() { return this._type; } /** * Retrieves the options of the physics constraint. * * @returns The physics constraint parameters. * */ get options() { return this._options; } /** * Enable/disable the constraint * @param isEnabled value for the constraint */ set isEnabled(e) { this._physicsPlugin.setEnabled(this, e); } /** * * @returns true if constraint is enabled */ get isEnabled() { return this._physicsPlugin.getEnabled(this); } /** * Enables or disables collisions for the physics engine. * * @param isEnabled - A boolean value indicating whether collisions should be enabled or disabled. * */ set isCollisionsEnabled(e) { this._physicsPlugin.setCollisionsEnabled(this, e); } /** * Gets whether collisions are enabled for this physics object. * * @returns `true` if collisions are enabled, `false` otherwise. * */ get isCollisionsEnabled() { return this._physicsPlugin.getCollisionsEnabled(this); } /** * Gets all bodies that are using this constraint * @returns */ getBodiesUsingConstraint() { return this._physicsPlugin.getBodiesUsingConstraint(this); } /** * Disposes the constraint from the physics engine. * * This method is useful for cleaning up the physics engine when a body is no longer needed. Disposing the body will free up resources and prevent memory leaks. */ dispose() { this._physicsPlugin.disposeConstraint(this); } } class _he { } class $ne extends mD { constructor(e, t, r) { super(kv.SIX_DOF, e, r), this.limits = t; } /** * Sets the friction of the given axis of the physics engine. * @param axis - The axis of the physics engine to set the friction for. * @param friction - The friction to set for the given axis. * */ setAxisFriction(e, t) { this._physicsPlugin.setAxisFriction(this, e, t); } /** * Gets the friction of the given axis of the physics engine. * @param axis - The axis of the physics engine. * @returns The friction of the given axis, or null if the constraint hasn't been initialized yet. * */ getAxisFriction(e) { return this._physicsPlugin.getAxisFriction(this, e); } /** * Sets the limit mode for the given axis of the constraint. * @param axis The axis to set the limit mode for. * @param limitMode The limit mode to set. * * This method is useful for setting the limit mode for a given axis of the constraint. This is important for * controlling the behavior of the physics engine when the constraint is reached. By setting the limit mode, * the engine can be configured to either stop the motion of the objects, or to allow them to continue * moving beyond the constraint. */ setAxisMode(e, t) { this._physicsPlugin.setAxisMode(this, e, t); } /** * Gets the limit mode of the given axis of the constraint. * * @param axis - The axis of the constraint. * @returns The limit mode of the given axis, or null if the constraint hasn't been initialized yet. * */ getAxisMode(e) { return this._physicsPlugin.getAxisMode(this, e); } /** * Sets the minimum limit of a given axis of a constraint. * @param axis - The axis of the constraint. * @param minLimit - The minimum limit of the axis. * */ setAxisMinLimit(e, t) { this._physicsPlugin.setAxisMinLimit(this, e, t); } /** * Gets the minimum limit of the given axis of the physics engine. * @param axis - The axis of the physics engine. * @returns The minimum limit of the given axis, or null if the constraint hasn't been initialized yet. * */ getAxisMinLimit(e) { return this._physicsPlugin.getAxisMinLimit(this, e); } /** * Sets the maximum limit of the given axis for the physics engine. * @param axis - The axis to set the limit for. * @param limit - The maximum limit of the axis. * * This method is useful for setting the maximum limit of the given axis for the physics engine, * which can be used to control the movement of the physics object. This helps to ensure that the * physics object does not move beyond the given limit. */ setAxisMaxLimit(e, t) { this._physicsPlugin.setAxisMaxLimit(this, e, t); } /** * Gets the maximum limit of the given axis of the physics engine. * @param axis - The axis of the physics engine. * @returns The maximum limit of the given axis, or null if the constraint hasn't been initialized yet. * */ getAxisMaxLimit(e) { return this._physicsPlugin.getAxisMaxLimit(this, e); } /** * Sets the motor type of the given axis of the constraint. * @param axis - The axis of the constraint. * @param motorType - The type of motor to use. * @returns void * */ setAxisMotorType(e, t) { this._physicsPlugin.setAxisMotorType(this, e, t); } /** * Gets the motor type of the specified axis of the constraint. * * @param axis - The axis of the constraint. * @returns The motor type of the specified axis, or null if the constraint hasn't been initialized yet. * */ getAxisMotorType(e) { return this._physicsPlugin.getAxisMotorType(this, e); } /** * Sets the target velocity of the motor associated with the given axis of the constraint. * @param axis - The axis of the constraint. * @param target - The target velocity of the motor. * * This method is useful for setting the target velocity of the motor associated with the given axis of the constraint. */ setAxisMotorTarget(e, t) { this._physicsPlugin.setAxisMotorTarget(this, e, t); } /** * Gets the target velocity of the motor associated to the given constraint axis. * @param axis - The constraint axis associated to the motor. * @returns The target velocity of the motor, or null if the constraint hasn't been initialized yet. * */ getAxisMotorTarget(e) { return this._physicsPlugin.getAxisMotorTarget(this, e); } /** * Sets the maximum force of the motor of the given axis of the constraint. * @param axis - The axis of the constraint. * @param maxForce - The maximum force of the motor. * */ setAxisMotorMaxForce(e, t) { this._physicsPlugin.setAxisMotorMaxForce(this, e, t); } /** * Gets the maximum force of the motor of the given axis of the constraint. * @param axis - The axis of the constraint. * @returns The maximum force of the motor, or null if the constraint hasn't been initialized yet. * */ getAxisMotorMaxForce(e) { return this._physicsPlugin.getAxisMotorMaxForce(this, e); } } class $he extends mD { constructor(e, t, r, n, i) { super(kv.BALL_AND_SOCKET, { pivotA: e, pivotB: t, axisA: r, axisB: n }, i); } } class e0e extends mD { constructor(e, t) { super(kv.DISTANCE, { maxDistance: e }, t); } } class t0e extends mD { constructor(e, t, r, n, i) { super(kv.HINGE, { pivotA: e, pivotB: t, axisA: r, axisB: n }, i); } } class r0e extends mD { constructor(e, t, r, n, i) { super(kv.SLIDER, { pivotA: e, pivotB: t, axisA: r, axisB: n }, i); } } class n0e extends mD { constructor(e, t, r, n, i) { super(kv.LOCK, { pivotA: e, pivotB: t, axisA: r, axisB: n }, i); } } class i0e extends mD { constructor(e, t, r, n, i) { super(kv.PRISMATIC, { pivotA: e, pivotB: t, axisA: r, axisB: n }, i); } } class s0e extends $ne { constructor(e, t, r, n, i, s, a, f, o) { super({ pivotA: e, pivotB: t, axisA: r, axisB: n }, [{ axis: MH.LINEAR_DISTANCE, minLimit: i, maxLimit: s, stiffness: a, damping: f }], o); } } var R2; (function(A) { A[A.GEOMETRIC_MEAN = 0] = "GEOMETRIC_MEAN", A[A.MINIMUM = 1] = "MINIMUM", A[A.MAXIMUM = 2] = "MAXIMUM", A[A.ARITHMETIC_MEAN = 3] = "ARITHMETIC_MEAN", A[A.MULTIPLY = 4] = "MULTIPLY"; })(R2 || (R2 = {})); class a0e { constructor(e, t, r = { mass: 0 }, n) { var i; if (this.transformNode = e, this.type = t, this._options = r, this._scene = n, this._disposeShapeWhenDisposed = !0, !this.transformNode) { Se.Error("No object was provided. A physics object is obligatory"); return; } const s = e; if (this.transformNode.parent && this._options.mass !== 0 && s.hasThinInstances && Se.Warn("A physics body has been created for an object which has a parent and thin instances. Babylon physics currently works in local space so unexpected issues may occur."), !this._scene && e.getScene && (this._scene = e.getScene()), !this._scene) return; this._options.mass = r.mass === void 0 ? 0 : r.mass, this._options.friction = r.friction === void 0 ? 0.2 : r.friction, this._options.restitution = r.restitution === void 0 ? 0.2 : r.restitution; const a = this._options.mass === 0 ? W0.STATIC : W0.DYNAMIC, f = (i = this._options.startAsleep) !== null && i !== void 0 ? i : !1; this.body = new wy(e, a, f, this._scene), this._addSizeOptions(), t.getClassName && t.getClassName() === "PhysicsShape" ? (this.shape = t, this._disposeShapeWhenDisposed = !1) : this.shape = new P4({ type: t, parameters: this._options }, this._scene), this._options.isTriggerShape && (this.shape.isTrigger = !0), this.material = { friction: this._options.friction, restitution: this._options.restitution }, this.body.shape = this.shape, this.shape.material = this.material, this.body.setMassProperties({ mass: this._options.mass }), this._nodeDisposeObserver = this.transformNode.onDisposeObservable.add(() => { this.dispose(); }); } _getObjectBoundingBox() { return this.transformNode.getRawBoundingInfo ? this.transformNode.getRawBoundingInfo().boundingBox : new cp(new S(-0.5, -0.5, -0.5), new S(0.5, 0.5, 0.5)); } _hasVertices(e) { return (e == null ? void 0 : e.getTotalVertices()) > 0; } _addSizeOptions() { var e, t, r, n, i, s, a, f; this.transformNode.computeWorldMatrix(!0); const o = this._getObjectBoundingBox(), d = ue.Vector3[0]; d.copyFrom(o.extendSize), d.scaleInPlace(2), d.multiplyInPlace(this.transformNode.scaling), d.x = Math.abs(d.x), d.y = Math.abs(d.y), d.z = Math.abs(d.z); const v = ue.Vector3[1]; if (v.copyFrom(o.minimum), v.multiplyInPlace(this.transformNode.scaling), !this._options.center) { const u = new S(); u.copyFrom(o.center), u.multiplyInPlace(this.transformNode.scaling), this._options.center = u; } switch (this.type) { case Ff.SPHERE: !this._options.radius && Xt.WithinEpsilon(d.x, d.y, 1e-4) && Xt.WithinEpsilon(d.x, d.z, 1e-4) ? this._options.radius = d.x / 2 : this._options.radius || (Se.Warn("Non uniform scaling is unsupported for sphere shapes. Setting the radius to the biggest bounding box extent."), this._options.radius = Math.max(d.x, d.y, d.z) / 2); break; case Ff.CAPSULE: { const u = d.x / 2; this._options.radius = (e = this._options.radius) !== null && e !== void 0 ? e : u, this._options.pointA = (t = this._options.pointA) !== null && t !== void 0 ? t : new S(0, v.y + u, 0), this._options.pointB = (r = this._options.pointB) !== null && r !== void 0 ? r : new S(0, v.y + d.y - u, 0); } break; case Ff.CYLINDER: { const u = d.x / 2; this._options.radius = (n = this._options.radius) !== null && n !== void 0 ? n : u, this._options.pointA = (i = this._options.pointA) !== null && i !== void 0 ? i : new S(0, v.y, 0), this._options.pointB = (s = this._options.pointB) !== null && s !== void 0 ? s : new S(0, v.y + d.y, 0); } break; case Ff.MESH: case Ff.CONVEX_HULL: if (!this._options.mesh && this._hasVertices(this.transformNode)) this._options.mesh = this.transformNode; else if (!this._options.mesh || !this._hasVertices(this._options.mesh)) throw new Error("No valid mesh was provided for mesh or convex hull shape parameter. Please provide a mesh with valid geometry (number of vertices greater than 0)."); break; case Ff.BOX: this._options.extents = (a = this._options.extents) !== null && a !== void 0 ? a : new S(d.x, d.y, d.z), this._options.rotation = (f = this._options.rotation) !== null && f !== void 0 ? f : Ze.Identity(); break; } } /** * Releases the body, shape and material */ dispose() { this._nodeDisposeObserver && (this.body.transformNode.onDisposeObservable.remove(this._nodeDisposeObserver), this._nodeDisposeObserver = null), this.body.dispose(), this._disposeShapeWhenDisposed && this.shape.dispose(); } } class o0e { /** * Constructor of the mesh accumulator * @param mesh - The mesh used to compute the world matrix. * @param collectIndices - use mesh indices * @param scene - The scene used to determine the right handed system. * * Merge mesh and its children so whole hierarchy can be used as a mesh shape or convex hull */ constructor(e, t, r) { this._vertices = [], this._indices = [], this._isRightHanded = r.useRightHandedSystem, this._collectIndices = t; } /** * Adds a mesh to the physics engine. * @param mesh The mesh to add. * @param includeChildren Whether to include the children of the mesh. * * This method adds a mesh to the physics engine by computing the world matrix, * multiplying it with the body from world matrix, and then transforming the * coordinates of the mesh's vertices. It also adds the indices of the mesh * to the physics engine. If includeChildren is true, it will also add the * children of the mesh to the physics engine, ignoring any children which * have a physics impostor. This is useful for creating a physics engine * that accurately reflects the mesh and its children. */ addNodeMeshes(e, t) { e.computeWorldMatrix(!0); const r = ue.Matrix[0]; if (he.ScalingToRef(e.absoluteScaling.x, e.absoluteScaling.y, e.absoluteScaling.z, r), e instanceof Ee ? this._addMesh(e, r) : e instanceof bp && this._addMesh(e.sourceMesh, r), t) { const n = ue.Matrix[1]; e.computeWorldMatrix().invertToRef(n); const i = ue.Matrix[2]; n.multiplyToRef(r, i), e.getChildMeshes(!1).filter((a) => !a.physicsBody).forEach((a) => { const f = a.computeWorldMatrix(), o = ue.Matrix[3]; f.multiplyToRef(i, o), a instanceof Ee ? this._addMesh(a, o) : a instanceof bp && this._addMesh(a.sourceMesh, o); }); } } _addMesh(e, t) { const r = e.getVerticesData(J.PositionKind) || [], n = r.length / 3, i = this._vertices.length; for (let s = 0; s < n; s++) { const a = new S(r[s * 3 + 0], r[s * 3 + 1], r[s * 3 + 2]); this._vertices.push(S.TransformCoordinates(a, t)); } if (this._collectIndices) { const s = e.getIndices(); if (s) for (let a = 0; a < s.length; a += 3) this._isRightHanded ? (this._indices.push(s[a + 0] + i), this._indices.push(s[a + 1] + i), this._indices.push(s[a + 2] + i)) : (this._indices.push(s[a + 2] + i), this._indices.push(s[a + 1] + i), this._indices.push(s[a + 0] + i)); } } /** * Allocate and populate the vertex positions inside the physics plugin. * * @returns An array of floats, whose backing memory is inside the plugin. The array contains the * positions of the mesh vertices, where a position is defined by three floats. You must call * freeBuffer() on the returned array once you have finished with it, in order to free the * memory inside the plugin.. */ getVertices(e) { const t = this._vertices.length * 3, n = t * 4, i = e._malloc(n), s = new Float32Array(e.HEAPU8.buffer, i, t); for (let a = 0; a < this._vertices.length; a++) s[a * 3 + 0] = this._vertices[a].x, s[a * 3 + 1] = this._vertices[a].y, s[a * 3 + 2] = this._vertices[a].z; return { offset: i, numObjects: t }; } freeBuffer(e, t) { e._free(t.offset); } /** * Allocate and populate the triangle indices inside the physics plugin * * @returns A new Int32Array, whose backing memory is inside the plugin. The array contains the indices * of the triangle positions, where a single triangle is defined by three indices. You must call * freeBuffer() on this array once you have finished with it, to free the memory inside the plugin.. */ getTriangles(e) { const r = this._indices.length * 4, n = e._malloc(r), i = new Int32Array(e.HEAPU8.buffer, n, this._indices.length); for (let s = 0; s < this._indices.length; s++) i[s] = this._indices[s]; return { offset: n, numObjects: this._indices.length }; } } class JG { constructor(e) { this.hpBodyId = e, this.userMassProps = { centerOfMass: void 0, mass: void 0, inertia: void 0, inertiaOrientation: void 0 }; } } class zG { constructor() { this.bodyId = BigInt(0), this.position = new S(), this.normal = new S(); } } class GG { constructor() { this.contactOnA = new zG(), this.contactOnB = new zG(), this.impulseApplied = 0, this.type = 0; } static readToRef(e, t, r) { const n = new Int32Array(e, t), i = new Float32Array(e, t), s = 2; r.contactOnA.bodyId = BigInt(n[s]), r.contactOnA.position.set(i[s + 8], i[s + 9], i[s + 10]), r.contactOnA.normal.set(i[s + 11], i[s + 12], i[s + 13]); const a = 18; r.contactOnB.bodyId = BigInt(n[a]), r.contactOnB.position.set(i[a + 8], i[a + 9], i[a + 10]), r.contactOnB.normal.set(i[a + 11], i[a + 12], i[a + 13]), r.impulseApplied = i[a + 13 + 3], r.type = n[0]; } } class ZG { constructor() { this.bodyIdA = BigInt(0), this.bodyIdB = BigInt(0), this.type = 0; } static readToRef(e, t, r) { const n = new Int32Array(e, t); r.type = n[0], r.bodyIdA = BigInt(n[2]), r.bodyIdB = BigInt(n[6]); } } class f0e { constructor(e = !0, t = HK) { if (this._useDeltaForWorldStep = e, this._hknp = {}, this.name = "HavokPlugin", this._fixedTimeStep = 1 / 60, this._timeStep = 1 / 60, this._tmpVec3 = Nf.BuildArray(3, S.Zero), this._bodies = /* @__PURE__ */ new Map(), this._bodyCollisionObservable = /* @__PURE__ */ new Map(), this._constraintToBodyIdPair = /* @__PURE__ */ new Map(), this._bodyCollisionEndedObservable = /* @__PURE__ */ new Map(), this.onCollisionObservable = new Oe(), this.onCollisionEndedObservable = new Oe(), this.onTriggerCollisionObservable = new Oe(), typeof t == "function") { Se.Error("Havok is not ready. Please make sure you await HK() before using the plugin."); return; } else this._hknp = t; if (!this.isSupported()) { Se.Error("Havok is not available. Please make sure you included the js file."); return; } this.world = this._hknp.HP_World_Create()[1], this._queryCollector = this._hknp.HP_QueryCollector_Create(1)[1]; } /** * If this plugin is supported * @returns true if its supported */ isSupported() { return this._hknp !== void 0; } /** * Sets the gravity of the physics world. * * @param gravity - The gravity vector to set. * */ setGravity(e) { this._hknp.HP_World_SetGravity(this.world, this._bVecToV3(e)); } /** * Sets the fixed time step for the physics engine. * * @param timeStep - The fixed time step to use for the physics engine. * */ setTimeStep(e) { this._fixedTimeStep = e; } /** * Gets the fixed time step used by the physics engine. * * @returns The fixed time step used by the physics engine. * */ getTimeStep() { return this._fixedTimeStep; } /** * Executes a single step of the physics engine. * * @param delta The time delta in seconds since the last step. * @param physicsBodies An array of physics bodies to be simulated. * @returns void * * This method is useful for simulating the physics engine. It sets the physics body transformation, * steps the world, syncs the physics body, and notifies collisions. This allows for the physics engine * to accurately simulate the physics bodies in the world. */ executeStep(e, t) { for (const r of t) r.disablePreStep || this.setPhysicsBodyTransformation(r, r.transformNode); this._hknp.HP_World_Step(this.world, this._useDeltaForWorldStep ? e : this._timeStep), this._bodyBuffer = this._hknp.HP_World_GetBodyBuffer(this.world)[1]; for (const r of t) this.sync(r); this._notifyCollisions(), this._notifyTriggers(); } /** * Returns the version of the physics engine plugin. * * @returns The version of the physics engine plugin. * * This method is useful for determining the version of the physics engine plugin that is currently running. */ getPluginVersion() { return 2; } /** * Initializes a physics body with the given position and orientation. * * @param body - The physics body to initialize. * @param motionType - The motion type of the body. * @param position - The position of the body. * @param orientation - The orientation of the body. * This code is useful for initializing a physics body with the given position and orientation. * It creates a plugin data for the body and adds it to the world. It then converts the position * and orientation to a transform and sets the body's transform to the given values. */ initBody(e, t, r, n) { e._pluginData = new JG(this._hknp.HP_Body_Create()[1]), this._internalSetMotionType(e._pluginData, t); const i = [this._bVecToV3(r), this._bQuatToV4(n)]; this._hknp.HP_Body_SetQTransform(e._pluginData.hpBodyId, i), this._hknp.HP_World_AddBody(this.world, e._pluginData.hpBodyId, e.startAsleep), this._bodies.set(e._pluginData.hpBodyId[0], { body: e, index: 0 }); } /** * Removes a body from the world. To dispose of a body, it is necessary to remove it from the world first. * * @param body - The body to remove. */ removeBody(e) { if (e._pluginDataInstances && e._pluginDataInstances.length > 0) for (const t of e._pluginDataInstances) this._bodyCollisionObservable.delete(t.hpBodyId[0]), this._hknp.HP_World_RemoveBody(this.world, t.hpBodyId), this._bodies.delete(t.hpBodyId[0]); e._pluginData && (this._bodyCollisionObservable.delete(e._pluginData.hpBodyId[0]), this._hknp.HP_World_RemoveBody(this.world, e._pluginData.hpBodyId), this._bodies.delete(e._pluginData.hpBodyId[0])); } /** * Initializes the body instances for a given physics body and mesh. * * @param body - The physics body to initialize. * @param motionType - How the body will be handled by the engine * @param mesh - The mesh to initialize. * * This code is useful for creating a physics body from a mesh. It creates a * body instance for each instance of the mesh and adds it to the world. It also * sets the position of the body instance to the position of the mesh instance. * This allows for the physics engine to accurately simulate the mesh in the * world. */ initBodyInstances(e, t, r) { var n, i; const s = (i = (n = r._thinInstanceDataStorage) === null || n === void 0 ? void 0 : n.instancesCount) !== null && i !== void 0 ? i : 0, a = r._thinInstanceDataStorage.matrixData; a && (this._createOrUpdateBodyInstances(e, t, a, 0, s, !1), e._pluginDataInstances.forEach((f, o) => { this._bodies.set(f.hpBodyId[0], { body: e, index: o }); })); } _createOrUpdateBodyInstances(e, t, r, n, i, s) { const a = ue.Quaternion[0], f = he.Identity(); for (let o = n; o < i; o++) { const d = [r[o * 16 + 12], r[o * 16 + 13], r[o * 16 + 14]]; let v; s ? v = e._pluginDataInstances[o].hpBodyId : v = this._hknp.HP_Body_Create()[1], f.setRowFromFloats(0, r[o * 16 + 0], r[o * 16 + 1], r[o * 16 + 2], 0), f.setRowFromFloats(1, r[o * 16 + 4], r[o * 16 + 5], r[o * 16 + 6], 0), f.setRowFromFloats(2, r[o * 16 + 8], r[o * 16 + 9], r[o * 16 + 10], 0), Ze.FromRotationMatrixToRef(f, a); const u = [d, [a.x, a.y, a.z, a.w]]; if (this._hknp.HP_Body_SetQTransform(v, u), !s) { const l = new JG(v); e._pluginDataInstances.length && (l.userMassProps = e._pluginDataInstances[0].userMassProps), this._internalSetMotionType(l, t), this._internalUpdateMassProperties(l), e._pluginDataInstances.push(l), this._hknp.HP_World_AddBody(this.world, v, e.startAsleep), l.worldTransformOffset = this._hknp.HP_Body_GetWorldTransformOffset(v)[1]; } } } /** * Update the internal body instances for a given physics body to match the instances in a mesh. * @param body the body that will be updated * @param mesh the mesh with reference instances */ updateBodyInstances(e, t) { var r, n; const i = (n = (r = t._thinInstanceDataStorage) === null || r === void 0 ? void 0 : r.instancesCount) !== null && n !== void 0 ? n : 0, s = t._thinInstanceDataStorage.matrixData; if (!s) return; const a = e._pluginDataInstances.length, f = this.getMotionType(e); if (i > a) { this._createOrUpdateBodyInstances(e, f, s, a, i, !1); const o = this._hknp.HP_Body_GetShape(e._pluginDataInstances[0].hpBodyId)[1]; for (let d = a; d < i; d++) this._hknp.HP_Body_SetShape(e._pluginDataInstances[d].hpBodyId, o), this._internalUpdateMassProperties(e._pluginDataInstances[d]), this._bodies.set(e._pluginDataInstances[d].hpBodyId[0], { body: e, index: d }); } else if (i < a) { const o = a - i; for (let d = 0; d < o; d++) { const v = e._pluginDataInstances.pop(); this._bodies.delete(v.hpBodyId[0]), this._hknp.HP_World_RemoveBody(this.world, v.hpBodyId), this._hknp.HP_Body_Release(v.hpBodyId); } this._createOrUpdateBodyInstances(e, f, s, 0, i, !0); } } /** * Synchronizes the transform of a physics body with its transform node. * @param body - The physics body to synchronize. * * This function is useful for keeping the physics body's transform in sync with its transform node. * This is important for ensuring that the physics body is accurately represented in the physics engine. */ sync(e) { this.syncTransform(e, e.transformNode); } /** * Synchronizes the transform of a physics body with the transform of its * corresponding transform node. * * @param body - The physics body to synchronize. * @param transformNode - The destination Transform Node. * * This code is useful for synchronizing the position and orientation of a * physics body with the position and orientation of its corresponding * transform node. This is important for ensuring that the physics body and * the transform node are in the same position and orientation in the scene. * This is necessary for the physics engine to accurately simulate the * physical behavior of the body. */ syncTransform(e, t) { var r; if (e._pluginDataInstances.length) { const n = t, i = n._thinInstanceDataStorage.matrixData; if (!i) return; const s = e._pluginDataInstances.length; for (let a = 0; a < s; a++) { const f = e._pluginDataInstances[a].worldTransformOffset, o = new Float32Array(this._hknp.HEAPU8.buffer, this._bodyBuffer + f, 16), d = a * 16; for (let v = 0; v < 15; v++) (v & 3) != 3 && (i[d + v] = o[v]); i[d + 15] = 1; } n.thinInstanceBufferUpdated("matrix"); } else try { const n = this._hknp.HP_Body_GetQTransform(e._pluginData.hpBodyId)[1], i = n[0], s = n[1], a = ue.Quaternion[0]; a.set(s[0], s[1], s[2], s[3]); const f = t.parent; if (f && !f.getWorldMatrix().isIdentity()) { f.computeWorldMatrix(!0), a.normalize(); const o = ue.Matrix[0], d = ue.Vector3[0]; d.copyFromFloats(i[0], i[1], i[2]), he.ComposeToRef(t.absoluteScaling, a, d, o); const v = ue.Matrix[1]; f.getWorldMatrix().invertToRef(v); const u = ue.Matrix[2]; o.multiplyToRef(v, u), u.decomposeToTransformNode(t), (r = t.rotationQuaternion) === null || r === void 0 || r.normalize(); } else t.position.set(i[0], i[1], i[2]), t.rotationQuaternion ? t.rotationQuaternion.copyFrom(a) : a.toEulerAnglesToRef(t.rotation); } catch (n) { console.log(`Syncing transform failed for node ${t.name}: ${n.message}...`); } } /** * Sets the shape of a physics body. * @param body - The physics body to set the shape for. * @param shape - The physics shape to set. * * This function is used to set the shape of a physics body. It is useful for * creating a physics body with a specific shape, such as a box or a sphere, * which can then be used to simulate physical interactions in a physics engine. * This function is especially useful for meshes with multiple instances, as it * will set the shape for each instance of the mesh. */ setShape(e, t) { var r, n, i; const s = t && t._pluginData ? t._pluginData : BigInt(0); if (!(e.transformNode instanceof Ee) || !(!((r = e.transformNode._thinInstanceDataStorage) === null || r === void 0) && r.matrixData)) { this._hknp.HP_Body_SetShape(e._pluginData.hpBodyId, s), this._internalUpdateMassProperties(e._pluginData); return; } const f = (i = (n = e.transformNode._thinInstanceDataStorage) === null || n === void 0 ? void 0 : n.instancesCount) !== null && i !== void 0 ? i : 0; for (let o = 0; o < f; o++) this._hknp.HP_Body_SetShape(e._pluginDataInstances[o].hpBodyId, s), this._internalUpdateMassProperties(e._pluginDataInstances[o]); } /** * Returns a reference to the first instance of the plugin data for a physics body. * @param body * @param instanceIndex * @returns a reference to the first instance */ _getPluginReference(e, t) { var r; return !((r = e._pluginDataInstances) === null || r === void 0) && r.length ? e._pluginDataInstances[t ?? 0] : e._pluginData; } /** * Gets the shape of a physics body. This will create a new shape object * * @param body - The physics body. * @returns The shape of the physics body. * */ getShape(e) { const t = this._getPluginReference(e), r = this._hknp.HP_Body_GetShape(t.hpBodyId)[1]; if (r != 0) { const n = e.transformNode.getScene(); return new P4({ pluginData: r }, n); } return null; } /** * Gets the type of a physics shape. * @param shape - The physics shape to get the type for. * @returns The type of the physics shape. * */ getShapeType(e) { return e.type ? e.type : this._hknp.HP_Shape_GetType(e._pluginData); } /** * Sets the event mask of a physics body. * @param body - The physics body to set the event mask for. * @param eventMask - The event mask to set. * * This function is useful for setting the event mask of a physics body, which is used to determine which events the body will respond to. This is important for ensuring that the physics engine is able to accurately simulate the behavior of the body in the game world. */ setEventMask(e, t, r) { this._applyToBodyOrInstances(e, (n) => { this._hknp.HP_Body_SetEventMask(n.hpBodyId, t); }, r); } /** * Retrieves the event mask of a physics body. * * @param body - The physics body to retrieve the event mask from. * @returns The event mask of the physics body. * */ getEventMask(e, t) { const r = this._getPluginReference(e, t); return this._hknp.HP_Body_GetEventMask(r.hpBodyId)[1]; } _fromMassPropertiesTuple(e) { return { centerOfMass: S.FromArray(e[0]), mass: e[1], inertia: S.FromArray(e[2]), inertiaOrientation: Ze.FromArray(e[3]) }; } _internalUpdateMassProperties(e) { const t = this._internalComputeMassProperties(e), r = e.userMassProps; r.centerOfMass && (t[0] = r.centerOfMass.asArray()), r.mass != null && (t[1] = r.mass), r.inertia && (t[2] = r.inertia.asArray()), r.inertiaOrientation && (t[3] = r.inertiaOrientation.asArray()), this._hknp.HP_Body_SetMassProperties(e.hpBodyId, t); } _internalSetMotionType(e, t) { switch (t) { case W0.STATIC: this._hknp.HP_Body_SetMotionType(e.hpBodyId, this._hknp.MotionType.STATIC); break; case W0.ANIMATED: this._hknp.HP_Body_SetMotionType(e.hpBodyId, this._hknp.MotionType.KINEMATIC); break; case W0.DYNAMIC: this._hknp.HP_Body_SetMotionType(e.hpBodyId, this._hknp.MotionType.DYNAMIC); break; } } setMotionType(e, t, r) { this._applyToBodyOrInstances(e, (n) => { this._internalSetMotionType(n, t); }, r); } getMotionType(e, t) { const r = this._getPluginReference(e, t), n = this._hknp.HP_Body_GetMotionType(r.hpBodyId)[1]; switch (n) { case this._hknp.MotionType.STATIC: return W0.STATIC; case this._hknp.MotionType.KINEMATIC: return W0.ANIMATED; case this._hknp.MotionType.DYNAMIC: return W0.DYNAMIC; } throw new Error("Unknown motion type: " + n); } _internalComputeMassProperties(e) { const t = this._hknp.HP_Body_GetShape(e.hpBodyId); if (t[0] == this._hknp.Result.RESULT_OK) { const r = this._hknp.HP_Shape_BuildMassProperties(t[1]); if (r[0] == this._hknp.Result.RESULT_OK) return r[1]; } return [[0, 0, 0], 1, [1, 1, 1], [0, 0, 0, 1]]; } /** * Computes the mass properties of a physics body, from it's shape * * @param body - The physics body to copmute the mass properties of */ computeMassProperties(e, t) { const r = this._getPluginReference(e, t), n = this._internalComputeMassProperties(r); return this._fromMassPropertiesTuple(n); } /** * Sets the mass properties of a physics body. * * @param body - The physics body to set the mass properties of. * @param massProps - The mass properties to set. * @param instanceIndex - The index of the instance to set the mass properties of. If undefined, the mass properties of all the bodies will be set. * This function is useful for setting the mass properties of a physics body, * such as its mass, inertia, and center of mass. This is important for * accurately simulating the physics of the body in the physics engine. * */ setMassProperties(e, t, r) { this._applyToBodyOrInstances(e, (n) => { n.userMassProps = t, this._internalUpdateMassProperties(n); }, r); } /** * */ getMassProperties(e, t) { const r = this._getPluginReference(e, t), n = this._hknp.HP_Body_GetMassProperties(r.hpBodyId)[1]; return this._fromMassPropertiesTuple(n); } /** * Sets the linear damping of the given body. * @param body - The body to set the linear damping for. * @param damping - The linear damping to set. * * This method is useful for controlling the linear damping of a body in a physics engine. * Linear damping is a force that opposes the motion of the body, and is proportional to the velocity of the body. * This method allows the user to set the linear damping of a body, which can be used to control the motion of the body. */ setLinearDamping(e, t, r) { this._applyToBodyOrInstances(e, (n) => { this._hknp.HP_Body_SetLinearDamping(n.hpBodyId, t); }, r); } /** * Gets the linear damping of the given body. * @param body - The body to get the linear damping from. * @returns The linear damping of the given body. * * This method is useful for getting the linear damping of a body in a physics engine. * Linear damping is a force that opposes the motion of the body and is proportional to the velocity of the body. * It is used to simulate the effects of air resistance and other forms of friction. */ getLinearDamping(e, t) { const r = this._getPluginReference(e, t); return this._hknp.HP_Body_GetLinearDamping(r.hpBodyId)[1]; } /** * Sets the angular damping of a physics body. * @param body - The physics body to set the angular damping for. * @param damping - The angular damping value to set. * * This function is useful for controlling the angular velocity of a physics body. * By setting the angular damping, the body's angular velocity will be reduced over time, allowing for more realistic physics simulations. */ setAngularDamping(e, t, r) { this._applyToBodyOrInstances(e, (n) => { this._hknp.HP_Body_SetAngularDamping(n.hpBodyId, t); }, r); } /** * Gets the angular damping of a physics body. * @param body - The physics body to get the angular damping from. * @returns The angular damping of the body. * * This function is useful for retrieving the angular damping of a physics body, * which is used to control the rotational motion of the body. The angular damping is a value between 0 and 1, where 0 is no damping and 1 is full damping. */ getAngularDamping(e, t) { const r = this._getPluginReference(e, t); return this._hknp.HP_Body_GetAngularDamping(r.hpBodyId)[1]; } /** * Sets the linear velocity of a physics body. * @param body - The physics body to set the linear velocity of. * @param linVel - The linear velocity to set. * * This function is useful for setting the linear velocity of a physics body, which is necessary for simulating * motion in a physics engine. The linear velocity is the speed and direction of the body's movement. */ setLinearVelocity(e, t, r) { this._applyToBodyOrInstances(e, (n) => { this._hknp.HP_Body_SetLinearVelocity(n.hpBodyId, this._bVecToV3(t)); }, r); } /** * Gets the linear velocity of a physics body and stores it in a given vector. * @param body - The physics body to get the linear velocity from. * @param linVel - The vector to store the linear velocity in. * * This function is useful for retrieving the linear velocity of a physics body, * which can be used to determine the speed and direction of the body. This * information can be used to simulate realistic physics behavior in a game. */ getLinearVelocityToRef(e, t, r) { const n = this._getPluginReference(e, r), i = this._hknp.HP_Body_GetLinearVelocity(n.hpBodyId)[1]; this._v3ToBvecRef(i, t); } /* * Apply an operation either to all instances of a body, if instanceIndex is not specified, or to a specific instance. */ _applyToBodyOrInstances(e, t, r) { var n; if (((n = e._pluginDataInstances) === null || n === void 0 ? void 0 : n.length) > 0 && r === void 0) for (let i = 0; i < e._pluginDataInstances.length; i++) t(e._pluginDataInstances[i]); else t(this._getPluginReference(e, r)); } /** * Applies an impulse to a physics body at a given location. * @param body - The physics body to apply the impulse to. * @param impulse - The impulse vector to apply. * @param location - The location in world space to apply the impulse. * @param instanceIndex - The index of the instance to apply the impulse to. If not specified, the impulse will be applied to all instances. * * This method is useful for applying an impulse to a physics body at a given location. * This can be used to simulate physical forces such as explosions, collisions, and gravity. */ applyImpulse(e, t, r, n) { this._applyToBodyOrInstances(e, (i) => { this._hknp.HP_Body_ApplyImpulse(i.hpBodyId, this._bVecToV3(r), this._bVecToV3(t)); }, n); } /** * Applies a force to a physics body at a given location. * @param body - The physics body to apply the impulse to. * @param force - The force vector to apply. * @param location - The location in world space to apply the impulse. * @param instanceIndex - The index of the instance to apply the force to. If not specified, the force will be applied to all instances. * * This method is useful for applying a force to a physics body at a given location. * This can be used to simulate physical forces such as explosions, collisions, and gravity. */ applyForce(e, t, r, n) { t.scaleToRef(this.getTimeStep(), this._tmpVec3[0]), this.applyImpulse(e, this._tmpVec3[0], r, n); } /** * Sets the angular velocity of a physics body. * * @param body - The physics body to set the angular velocity of. * @param angVel - The angular velocity to set. * * This function is useful for setting the angular velocity of a physics body in a physics engine. * This allows for more realistic simulations of physical objects, as they can be given a rotational velocity. */ setAngularVelocity(e, t, r) { this._applyToBodyOrInstances(e, (n) => { this._hknp.HP_Body_SetAngularVelocity(n.hpBodyId, this._bVecToV3(t)); }, r); } /** * Gets the angular velocity of a body. * @param body - The body to get the angular velocity from. * @param angVel - The vector3 to store the angular velocity. * * This method is useful for getting the angular velocity of a body in a physics engine. It * takes the body and a vector3 as parameters and stores the angular velocity of the body * in the vector3. This is useful for getting the angular velocity of a body in order to * calculate the motion of the body in the physics engine. */ getAngularVelocityToRef(e, t, r) { const n = this._getPluginReference(e, r), i = this._hknp.HP_Body_GetAngularVelocity(n.hpBodyId)[1]; this._v3ToBvecRef(i, t); } /** * Sets the transformation of the given physics body to the given transform node. * @param body The physics body to set the transformation for. * @param node The transform node to set the transformation from. * Sets the transformation of the given physics body to the given transform node. * * This function is useful for setting the transformation of a physics body to a * transform node, which is necessary for the physics engine to accurately simulate * the motion of the body. It also takes into account instances of the transform * node, which is necessary for accurate simulation of multiple bodies with the * same transformation. */ setPhysicsBodyTransformation(e, t) { const r = e.transformNode; if (e.numInstances > 0) { const i = r._thinInstanceDataStorage.matrixData; if (!i) return; const s = e.numInstances; this._createOrUpdateBodyInstances(e, e.getMotionType(), i, 0, s, !0); } else this._hknp.HP_Body_SetQTransform(e._pluginData.hpBodyId, this._getTransformInfos(t)); } /** * Set the target transformation (position and rotation) of the body, such that the body will set its velocity to reach that target * @param body The physics body to set the target transformation for. * @param position The target position * @param rotation The target rotation * @param instanceIndex The index of the instance in an instanced body */ setTargetTransform(e, t, r, n) { this._applyToBodyOrInstances(e, (i) => { this._hknp.HP_Body_SetTargetQTransform(i.hpBodyId, [this._bVecToV3(t), this._bQuatToV4(r)]); }, n); } /** * Sets the gravity factor of a body * @param body the physics body to set the gravity factor for * @param factor the gravity factor * @param instanceIndex the index of the instance in an instanced body */ setGravityFactor(e, t, r) { this._applyToBodyOrInstances(e, (n) => { this._hknp.HP_Body_SetGravityFactor(n.hpBodyId, t); }, r); } /** * Get the gravity factor of a body * @param body the physics body to get the gravity factor from * @param instanceIndex the index of the instance in an instanced body. If not specified, the gravity factor of the first instance will be returned. * @returns the gravity factor */ getGravityFactor(e, t) { const r = this._getPluginReference(e, t); return this._hknp.HP_Body_GetGravityFactor(r.hpBodyId)[1]; } /** * Disposes a physics body. * * @param body - The physics body to dispose. * * This method is useful for releasing the resources associated with a physics body when it is no longer needed. * This is important for avoiding memory leaks in the physics engine. */ disposeBody(e) { if (e._pluginDataInstances && e._pluginDataInstances.length > 0) for (const t of e._pluginDataInstances) this._hknp.HP_Body_Release(t.hpBodyId), t.hpBodyId = void 0; e._pluginData && (this._hknp.HP_Body_Release(e._pluginData.hpBodyId), e._pluginData.hpBodyId = void 0); } /** * Initializes a physics shape with the given type and parameters. * @param shape - The physics shape to initialize. * @param type - The type of shape to initialize. * @param options - The parameters for the shape. * * This code is useful for initializing a physics shape with the given type and parameters. * It allows for the creation of a sphere, box, capsule, container, cylinder, mesh, and heightfield. * Depending on the type of shape, different parameters are required. * For example, a sphere requires a radius, while a box requires extents and a rotation. */ initShape(e, t, r) { switch (t) { case Ff.SPHERE: { const n = r.radius || 1, i = r.center ? this._bVecToV3(r.center) : [0, 0, 0]; e._pluginData = this._hknp.HP_Shape_CreateSphere(i, n)[1]; } break; case Ff.BOX: { const n = r.rotation ? this._bQuatToV4(r.rotation) : [0, 0, 0, 1], i = r.extents ? this._bVecToV3(r.extents) : [1, 1, 1], s = r.center ? this._bVecToV3(r.center) : [0, 0, 0]; e._pluginData = this._hknp.HP_Shape_CreateBox(s, n, i)[1]; } break; case Ff.CAPSULE: { const n = r.pointA ? this._bVecToV3(r.pointA) : [0, 0, 0], i = r.pointB ? this._bVecToV3(r.pointB) : [0, 1, 0], s = r.radius || 0; e._pluginData = this._hknp.HP_Shape_CreateCapsule(n, i, s)[1]; } break; case Ff.CONTAINER: e._pluginData = this._hknp.HP_Shape_CreateContainer()[1]; break; case Ff.CYLINDER: { const n = r.pointA ? this._bVecToV3(r.pointA) : [0, 0, 0], i = r.pointB ? this._bVecToV3(r.pointB) : [0, 1, 0], s = r.radius || 0; e._pluginData = this._hknp.HP_Shape_CreateCylinder(n, i, s)[1]; } break; case Ff.CONVEX_HULL: case Ff.MESH: { const n = r.mesh; if (n) { const i = !!r.includeChildMeshes, s = t != Ff.CONVEX_HULL, a = new o0e(n, s, n == null ? void 0 : n.getScene()); a.addNodeMeshes(n, i); const f = a.getVertices(this._hknp), o = f.numObjects / 3; if (t == Ff.CONVEX_HULL) e._pluginData = this._hknp.HP_Shape_CreateConvexHull(f.offset, o)[1]; else { const d = a.getTriangles(this._hknp), v = d.numObjects / 3; e._pluginData = this._hknp.HP_Shape_CreateMesh(f.offset, o, d.offset, v)[1], a.freeBuffer(this._hknp, d); } a.freeBuffer(this._hknp, f); } else throw new Error("No mesh provided to create physics shape."); } break; default: throw new Error("Unsupported Shape Type."); } } setShapeFilterMembershipMask(e, t) { const r = this._hknp.HP_Shape_GetFilterInfo(e._pluginData)[1][1]; this._hknp.HP_Shape_SetFilterInfo(e._pluginData, [t, r]); } getShapeFilterMembershipMask(e) { return this._hknp.HP_Shape_GetFilterInfo(e._pluginData)[1][0]; } setShapeFilterCollideMask(e, t) { const r = this._hknp.HP_Shape_GetFilterInfo(e._pluginData)[1][0]; this._hknp.HP_Shape_SetFilterInfo(e._pluginData, [r, t]); } getShapeFilterCollideMask(e) { return this._hknp.HP_Shape_GetFilterInfo(e._pluginData)[1][1]; } /** * Sets the material of a physics shape. * @param shape - The physics shape to set the material of. * @param material - The material to set. * */ setMaterial(e, t) { var r, n, i, s, a; const f = (r = t.friction) !== null && r !== void 0 ? r : 0.5, o = (n = t.staticFriction) !== null && n !== void 0 ? n : f, d = (i = t.restitution) !== null && i !== void 0 ? i : 0, v = (s = t.frictionCombine) !== null && s !== void 0 ? s : R2.MINIMUM, u = (a = t.restitutionCombine) !== null && a !== void 0 ? a : R2.MAXIMUM, l = [o, f, d, this._materialCombineToNative(v), this._materialCombineToNative(u)]; this._hknp.HP_Shape_SetMaterial(e._pluginData, l); } /** * Sets the density of a physics shape. * @param shape - The physics shape to set the density of. * @param density - The density to set. * */ setDensity(e, t) { this._hknp.HP_Shape_SetDensity(e._pluginData, t); } /** * Calculates the density of a given physics shape. * * @param shape - The physics shape to calculate the density of. * @returns The density of the given physics shape. * */ getDensity(e) { return this._hknp.HP_Shape_GetDensity(e._pluginData)[1]; } /** * Gets the transform infos of a given transform node. * @param node - The transform node. * @returns An array containing the position and orientation of the node. * This code is useful for getting the position and orientation of a given transform node. * It first checks if the node has a rotation quaternion, and if not, it creates one from the node's rotation. * It then creates an array containing the position and orientation of the node and returns it. */ _getTransformInfos(e) { if (e.parent) return e.computeWorldMatrix(!0), [this._bVecToV3(e.absolutePosition), this._bQuatToV4(e.absoluteRotationQuaternion)]; let t = ue.Quaternion[0]; if (e.rotationQuaternion) t = e.rotationQuaternion; else { const n = e.rotation; Ze.FromEulerAnglesToRef(n.x, n.y, n.z, t); } return [this._bVecToV3(e.position), this._bQuatToV4(t)]; } /** * Adds a child shape to the given shape. * @param shape - The parent shape. * @param newChild - The child shape to add. * @param translation - The relative translation of the child from the parent shape * @param rotation - The relative rotation of the child from the parent shape * @param scale - The relative scale scale of the child from the parent shaep * */ addChild(e, t, r, n, i) { const s = [ r ? this._bVecToV3(r) : [0, 0, 0], n ? this._bQuatToV4(n) : [0, 0, 0, 1], i ? this._bVecToV3(i) : [1, 1, 1] ]; this._hknp.HP_Shape_AddChild(e._pluginData, t._pluginData, s); } /** * Removes a child shape from a parent shape. * @param shape - The parent shape. * @param childIndex - The index of the child shape to remove. * */ removeChild(e, t) { this._hknp.HP_Shape_RemoveChild(e._pluginData, t); } /** * Returns the number of children of the given shape. * * @param shape - The shape to get the number of children from. * @returns The number of children of the given shape. * */ getNumChildren(e) { return this._hknp.HP_Shape_GetNumChildren(e._pluginData)[1]; } /** * Marks the shape as a trigger * @param shape the shape to mark as a trigger * @param isTrigger if the shape is a trigger */ setTrigger(e, t) { this._hknp.HP_Shape_SetTrigger(e._pluginData, t); } /** * Calculates the bounding box of a given physics shape. * * @param _shape - The physics shape to calculate the bounding box for. * @returns The calculated bounding box. * * This method is useful for physics engines as it allows to calculate the * boundaries of a given shape. Knowing the boundaries of a shape is important * for collision detection and other physics calculations. */ getBoundingBox(e) { return {}; } /** * Gets the geometry of a physics body. * * @param body - The physics body. * @returns An object containing the positions and indices of the body's geometry. * */ getBodyGeometry(e) { var t; const r = ((t = e._pluginDataInstances) === null || t === void 0 ? void 0 : t.length) > 0 ? e._pluginDataInstances[0] : e._pluginData, n = this._hknp.HP_Body_GetShape(r.hpBodyId)[1], i = this._hknp.HP_Shape_CreateDebugDisplayGeometry(n); if (i[0] != this._hknp.Result.RESULT_OK) return { positions: [], indices: [] }; const s = this._hknp.HP_DebugGeometry_GetInfo(i[1])[1], a = new Float32Array(this._hknp.HEAPU8.buffer, s[0], s[1] * 3), f = new Uint32Array(this._hknp.HEAPU8.buffer, s[2], s[3] * 3), o = a.slice(0), d = f.slice(0); return this._hknp.HP_DebugGeometry_Release(i[1]), { positions: o, indices: d }; } /** * Releases a physics shape from the physics engine. * * @param shape - The physics shape to be released. * @returns void * * This method is useful for releasing a physics shape from the physics engine, freeing up resources and preventing memory leaks. */ disposeShape(e) { this._hknp.HP_Shape_Release(e._pluginData), e._pluginData = void 0; } // constraint /** * Initializes a physics constraint with the given parameters. * * @param constraint - The physics constraint to be initialized. * @param body - The main body * @param childBody - The child body. * @param instanceIndex - If this body is instanced, the index of the instance to which the constraint will be applied. If not specified, no constraint will be applied. * @param childInstanceIndex - If the child body is instanced, the index of the instance to which the constraint will be applied. If not specified, no constraint will be applied. * * This function is useful for setting up a physics constraint in a physics engine. */ initConstraint(e, t, r, n, i) { var s, a, f, o, d; const v = e.type, u = e.options; if (!v || !u) { Se.Warn("No constraint type or options. Constraint is invalid."); return; } if (t._pluginDataInstances.length > 0 && n === void 0 || r._pluginDataInstances.length > 0 && i === void 0) { Se.Warn("Body is instanced but no instance index was specified. Constraint will not be applied."); return; } e._pluginData = (s = e._pluginData) !== null && s !== void 0 ? s : []; const l = this._hknp.HP_Constraint_Create()[1]; e._pluginData.push(l); const P = this._getPluginReference(t, n).hpBodyId, p = this._getPluginReference(r, i).hpBodyId; this._hknp.HP_Constraint_SetParentBody(l, P), this._hknp.HP_Constraint_SetChildBody(l, p), this._constraintToBodyIdPair.set(l[0], [P[0], p[0]]); const c = u.pivotA ? this._bVecToV3(u.pivotA) : this._bVecToV3(S.Zero()), H = (a = u.axisA) !== null && a !== void 0 ? a : new S(1, 0, 0), T = this._tmpVec3[0]; u.perpAxisA ? T.copyFrom(u.perpAxisA) : H.getNormalToRef(T), this._hknp.HP_Constraint_SetAnchorInParent(l, c, this._bVecToV3(H), this._bVecToV3(T)); const q = u.pivotB ? this._bVecToV3(u.pivotB) : this._bVecToV3(S.Zero()), b = (f = u.axisB) !== null && f !== void 0 ? f : new S(1, 0, 0), j = this._tmpVec3[0]; if (u.perpAxisB ? j.copyFrom(u.perpAxisB) : b.getNormalToRef(j), this._hknp.HP_Constraint_SetAnchorInChild(l, q, this._bVecToV3(b), this._bVecToV3(j)), e._initOptions || (e._initOptions = { axisA: H.clone(), axisB: b.clone(), perpAxisA: T.clone(), perpAxisB: j.clone(), pivotA: new S(c[0], c[1], c[2]), pivotB: new S(q[0], q[1], q[2]) }), v == kv.LOCK) this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.LINEAR_X, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.LINEAR_Y, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.LINEAR_Z, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.ANGULAR_X, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.ANGULAR_Y, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.ANGULAR_Z, this._hknp.ConstraintAxisLimitMode.LOCKED); else if (v == kv.DISTANCE) { const m = u.maxDistance || 0, I = this._hknp.ConstraintAxis.LINEAR_DISTANCE; this._hknp.HP_Constraint_SetAxisMode(l, I, this._hknp.ConstraintAxisLimitMode.LIMITED), this._hknp.HP_Constraint_SetAxisMinLimit(l, I, m), this._hknp.HP_Constraint_SetAxisMaxLimit(l, I, m); } else if (v == kv.HINGE) this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.LINEAR_X, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.LINEAR_Y, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.LINEAR_Z, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.ANGULAR_Y, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.ANGULAR_Z, this._hknp.ConstraintAxisLimitMode.LOCKED); else if (v == kv.PRISMATIC) this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.LINEAR_Y, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.LINEAR_Z, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.ANGULAR_X, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.ANGULAR_Y, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.ANGULAR_Z, this._hknp.ConstraintAxisLimitMode.LOCKED); else if (v == kv.SLIDER) this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.LINEAR_Y, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.LINEAR_Z, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.ANGULAR_Y, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.ANGULAR_Z, this._hknp.ConstraintAxisLimitMode.LOCKED); else if (v == kv.BALL_AND_SOCKET) this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.LINEAR_X, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.LINEAR_Y, this._hknp.ConstraintAxisLimitMode.LOCKED), this._hknp.HP_Constraint_SetAxisMode(l, this._hknp.ConstraintAxis.LINEAR_Z, this._hknp.ConstraintAxisLimitMode.LOCKED); else if (v == kv.SIX_DOF) { const m = e; for (const I of m.limits) { const N = this._constraintAxisToNative(I.axis); ((o = I.minLimit) !== null && o !== void 0 ? o : -1) == 0 && ((d = I.maxLimit) !== null && d !== void 0 ? d : -1) == 0 ? this._hknp.HP_Constraint_SetAxisMode(l, N, this._hknp.ConstraintAxisLimitMode.LOCKED) : (I.minLimit != null && (this._hknp.HP_Constraint_SetAxisMode(l, N, this._hknp.ConstraintAxisLimitMode.LIMITED), this._hknp.HP_Constraint_SetAxisMinLimit(l, N, I.minLimit)), I.maxLimit != null && (this._hknp.HP_Constraint_SetAxisMode(l, N, this._hknp.ConstraintAxisLimitMode.LIMITED), this._hknp.HP_Constraint_SetAxisMaxLimit(l, N, I.maxLimit))), I.stiffness && this._hknp.HP_Constraint_SetAxisStiffness(l, N, I.stiffness), I.damping && this._hknp.HP_Constraint_SetAxisDamping(l, N, I.damping); } } else throw new Error("Unsupported Constraint Type."); const w = !!u.collision; this._hknp.HP_Constraint_SetCollisionsEnabled(l, w), this._hknp.HP_Constraint_SetEnabled(l, !0); } /** * Get a list of all the pairs of bodies that are connected by this constraint. * @param constraint the constraint to search from * @returns a list of parent, child pairs */ getBodiesUsingConstraint(e) { const t = []; for (const r of e._pluginData) { const n = this._constraintToBodyIdPair.get(r[0]); if (n) { const i = this._bodies.get(n[0]), s = this._bodies.get(n[1]); i && s && t.push({ parentBody: i.body, parentBodyIndex: i.index, childBody: s.body, childBodyIndex: s.index }); } } return t; } /** * Adds a constraint to the physics engine. * * @param body - The main body to which the constraint is applied. * @param childBody - The body to which the constraint is applied. * @param constraint - The constraint to be applied. * @param instanceIndex - If this body is instanced, the index of the instance to which the constraint will be applied. If not specified, no constraint will be applied. * @param childInstanceIndex - If the child body is instanced, the index of the instance to which the constraint will be applied. If not specified, no constraint will be applied. */ addConstraint(e, t, r, n, i) { this.initConstraint(r, e, t, n, i); } /** * Enables or disables a constraint in the physics engine. * @param constraint - The constraint to enable or disable. * @param isEnabled - Whether the constraint should be enabled or disabled. * */ setEnabled(e, t) { for (const r of e._pluginData) this._hknp.HP_Constraint_SetEnabled(r, t); } /** * Gets the enabled state of the given constraint. * @param constraint - The constraint to get the enabled state from. * @returns The enabled state of the given constraint. * */ getEnabled(e) { const t = e._pluginData && e._pluginData[0]; return t ? this._hknp.HP_Constraint_GetEnabled(t)[1] : !1; } /** * Enables or disables collisions for the given constraint. * @param constraint - The constraint to enable or disable collisions for. * @param isEnabled - Whether collisions should be enabled or disabled. * */ setCollisionsEnabled(e, t) { for (const r of e._pluginData) this._hknp.HP_Constraint_SetCollisionsEnabled(r, t); } /** * Gets whether collisions are enabled for the given constraint. * @param constraint - The constraint to get collisions enabled for. * @returns Whether collisions are enabled for the given constraint. * */ getCollisionsEnabled(e) { const t = e._pluginData && e._pluginData[0]; return t ? this._hknp.HP_Constraint_GetCollisionsEnabled(t)[1] : !1; } /** * Sets the friction of the given axis of the given constraint. * * @param constraint - The constraint to set the friction of. * @param axis - The axis of the constraint to set the friction of. * @param friction - The friction to set. * @returns void * */ setAxisFriction(e, t, r) { for (const n of e._pluginData) this._hknp.HP_Constraint_SetAxisFriction(n, this._constraintAxisToNative(t), r); } /** * Gets the friction value of the specified axis of the given constraint. * * @param constraint - The constraint to get the axis friction from. * @param axis - The axis to get the friction from. * @returns The friction value of the specified axis. * */ getAxisFriction(e, t) { const r = e._pluginData && e._pluginData[0]; return r ? this._hknp.HP_Constraint_GetAxisFriction(r, this._constraintAxisToNative(t))[1] : null; } /** * Sets the limit mode of the specified axis of the given constraint. * @param constraint - The constraint to set the axis mode of. * @param axis - The axis to set the limit mode of. * @param limitMode - The limit mode to set. */ setAxisMode(e, t, r) { for (const n of e._pluginData) this._hknp.HP_Constraint_SetAxisMode(n, this._constraintAxisToNative(t), this._limitModeToNative(r)); } /** * Gets the axis limit mode of the given constraint. * * @param constraint - The constraint to get the axis limit mode from. * @param axis - The axis to get the limit mode from. * @returns The axis limit mode of the given constraint. * */ getAxisMode(e, t) { const r = e._pluginData && e._pluginData[0]; if (r) { const n = this._hknp.HP_Constraint_GetAxisMode(r, this._constraintAxisToNative(t))[1]; return this._nativeToLimitMode(n); } return null; } /** * Sets the minimum limit of the given axis of the given constraint. * @param constraint - The constraint to set the minimum limit of. * @param axis - The axis to set the minimum limit of. * @param limit - The minimum limit to set. * */ setAxisMinLimit(e, t, r) { for (const n of e._pluginData) this._hknp.HP_Constraint_SetAxisMinLimit(n, this._constraintAxisToNative(t), r); } /** * Gets the minimum limit of the specified axis of the given constraint. * @param constraint - The constraint to get the minimum limit from. * @param axis - The axis to get the minimum limit from. * @returns The minimum limit of the specified axis of the given constraint. * */ getAxisMinLimit(e, t) { const r = e._pluginData && e._pluginData[0]; return r ? this._hknp.HP_Constraint_GetAxisMinLimit(r, this._constraintAxisToNative(t))[1] : null; } /** * Sets the maximum limit of the given axis of the given constraint. * @param constraint - The constraint to set the maximum limit of the given axis. * @param axis - The axis to set the maximum limit of. * @param limit - The maximum limit to set. * */ setAxisMaxLimit(e, t, r) { for (const n of e._pluginData) this._hknp.HP_Constraint_SetAxisMaxLimit(n, this._constraintAxisToNative(t), r); } /** * Gets the maximum limit of the given axis of the given constraint. * * @param constraint - The constraint to get the maximum limit from. * @param axis - The axis to get the maximum limit from. * @returns The maximum limit of the given axis of the given constraint. * */ getAxisMaxLimit(e, t) { const r = e._pluginData && e._pluginData[0]; return r ? this._hknp.HP_Constraint_GetAxisMaxLimit(r, this._constraintAxisToNative(t))[1] : null; } /** * Sets the motor type of the given axis of the given constraint. * @param constraint - The constraint to set the motor type of. * @param axis - The axis of the constraint to set the motor type of. * @param motorType - The motor type to set. * @returns void * */ setAxisMotorType(e, t, r) { for (const n of e._pluginData) this._hknp.HP_Constraint_SetAxisMotorType(n, this._constraintAxisToNative(t), this._constraintMotorTypeToNative(r)); } /** * Gets the motor type of the specified axis of the given constraint. * @param constraint - The constraint to get the motor type from. * @param axis - The axis of the constraint to get the motor type from. * @returns The motor type of the specified axis of the given constraint. * */ getAxisMotorType(e, t) { const r = e._pluginData && e._pluginData[0]; return r ? this._nativeToMotorType(this._hknp.HP_Constraint_GetAxisMotorType(r, this._constraintAxisToNative(t))[1]) : null; } /** * Sets the target of an axis motor of a constraint. * * @param constraint - The constraint to set the axis motor target of. * @param axis - The axis of the constraint to set the motor target of. * @param target - The target of the axis motor. * */ setAxisMotorTarget(e, t, r) { for (const n of e._pluginData) this._hknp.HP_Constraint_SetAxisMotorTarget(n, this._constraintAxisToNative(t), r); } /** * Gets the target of the motor of the given axis of the given constraint. * * @param constraint - The constraint to get the motor target from. * @param axis - The axis of the constraint to get the motor target from. * @returns The target of the motor of the given axis of the given constraint. * */ getAxisMotorTarget(e, t) { return e._pluginData && e._pluginData[0] ? this._hknp.HP_Constraint_GetAxisMotorTarget(e._pluginData, this._constraintAxisToNative(t))[1] : null; } /** * Sets the maximum force that can be applied by the motor of the given constraint axis. * @param constraint - The constraint to set the motor max force for. * @param axis - The axis of the constraint to set the motor max force for. * @param maxForce - The maximum force that can be applied by the motor. * */ setAxisMotorMaxForce(e, t, r) { for (const n of e._pluginData) this._hknp.HP_Constraint_SetAxisMotorMaxForce(n, this._constraintAxisToNative(t), r); } /** * Gets the maximum force of the motor of the given constraint axis. * * @param constraint - The constraint to get the motor maximum force from. * @param axis - The axis of the constraint to get the motor maximum force from. * @returns The maximum force of the motor of the given constraint axis. * */ getAxisMotorMaxForce(e, t) { const r = e._pluginData && e._pluginData[0]; return r ? this._hknp.HP_Constraint_GetAxisMotorMaxForce(r, this._constraintAxisToNative(t))[1] : null; } /** * Disposes a physics constraint. * * @param constraint - The physics constraint to dispose. * * This method is useful for releasing the resources associated with a physics constraint, such as * the Havok constraint, when it is no longer needed. This is important for avoiding memory leaks. */ disposeConstraint(e) { for (const t of e._pluginData) this._hknp.HP_Constraint_SetEnabled(t, !1), this._hknp.HP_Constraint_Release(t); e._pluginData.length = 0; } /** * Performs a raycast from a given start point to a given end point and stores the result in a given PhysicsRaycastResult object. * * @param from - The start point of the raycast. * @param to - The end point of the raycast. * @param result - The PhysicsRaycastResult object to store the result of the raycast. * @param query - The raycast query options. See [[IRaycastQuery]] for more information. * * Performs a raycast. It takes in two points, from and to, and a PhysicsRaycastResult object to store the result of the raycast. * It then performs the raycast and stores the hit data in the PhysicsRaycastResult object. */ raycast(e, t, r, n) { var i, s; const a = (i = n == null ? void 0 : n.membership) !== null && i !== void 0 ? i : -1, f = (s = n == null ? void 0 : n.collideWith) !== null && s !== void 0 ? s : -1; r.reset(e, t); const o = !1, d = [BigInt(0)], v = [this._bVecToV3(e), this._bVecToV3(t), [a, f], o, d]; if (this._hknp.HP_World_CastRayWithCollector(this.world, this._queryCollector, v), this._hknp.HP_QueryCollector_GetNumHits(this._queryCollector)[1] > 0) { const u = this._hknp.HP_QueryCollector_GetCastRayResult(this._queryCollector, 0)[1], l = u[1][3], P = u[1][4], p = u[1][5]; r.setHitData({ x: P[0], y: P[1], z: P[2] }, { x: l[0], y: l[1], z: l[2] }, p), r.calculateHitDistance(); const c = this._bodies.get(u[1][0][0]); r.body = c == null ? void 0 : c.body, r.bodyIndex = c == null ? void 0 : c.index; } } /** * Return the collision observable for a particular physics body. * @param body the physics body */ getCollisionObservable(e) { const t = e._pluginData.hpBodyId[0]; let r = this._bodyCollisionObservable.get(t); return r || (r = new Oe(), this._bodyCollisionObservable.set(t, r)), r; } /** * Return the collision ended observable for a particular physics body. * @param body the physics body * @returns */ getCollisionEndedObservable(e) { const t = e._pluginData.hpBodyId[0]; let r = this._bodyCollisionEndedObservable.get(t); return r || (r = new Oe(), this._bodyCollisionEndedObservable.set(t, r)), r; } /** * Enable collision to be reported for a body when a callback is setup on the world * @param body the physics body * @param enabled */ setCollisionCallbackEnabled(e, t) { const r = this._hknp.EventType.COLLISION_STARTED.value | this._hknp.EventType.COLLISION_CONTINUED.value | this._hknp.EventType.COLLISION_FINISHED.value; e._pluginDataInstances && e._pluginDataInstances.length ? e._pluginDataInstances.forEach((n) => { this._hknp.HP_Body_SetEventMask(n.hpBodyId, t ? r : 0); }) : e._pluginData && this._hknp.HP_Body_SetEventMask(e._pluginData.hpBodyId, t ? r : 0); } /** * Enable collision ended to be reported for a body when a callback is setup on the world * @param body * @param enabled */ setCollisionEndedCallbackEnabled(e, t) { const r = this._getPluginReference(e); let n = this._hknp.HP_Body_GetEventMask(r.hpBodyId)[1]; n = t ? n | this._hknp.EventType.COLLISION_FINISHED.value : n & ~this._hknp.EventType.COLLISION_FINISHED.value, e._pluginDataInstances && e._pluginDataInstances.length ? e._pluginDataInstances.forEach((i) => { this._hknp.HP_Body_SetEventMask(i.hpBodyId, n); }) : e._pluginData && this._hknp.HP_Body_SetEventMask(e._pluginData.hpBodyId, n); } _notifyTriggers() { let e = this._hknp.HP_World_GetTriggerEvents(this.world)[1]; const t = new ZG(); for (; e; ) { ZG.readToRef(this._hknp.HEAPU8.buffer, e, t); const r = this._bodies.get(t.bodyIdA), n = this._bodies.get(t.bodyIdB); if (r && n) { const i = { collider: r.body, colliderIndex: r.index, collidedAgainst: n.body, collidedAgainstIndex: n.index, type: this._nativeTriggerCollisionValueToCollisionType(t.type) }; this.onTriggerCollisionObservable.notifyObservers(i); } e = this._hknp.HP_World_GetNextTriggerEvent(this.world, e); } } /** * Runs thru all detected collisions and filter by body */ _notifyCollisions() { let e = this._hknp.HP_World_GetCollisionEvents(this.world)[1]; const t = new GG(), r = Number(this.world); for (; e; ) { GG.readToRef(this._hknp.HEAPU8.buffer, e, t); const n = this._bodies.get(t.contactOnA.bodyId), i = this._bodies.get(t.contactOnB.bodyId); if (n && i) { const s = { collider: n.body, colliderIndex: n.index, collidedAgainst: i.body, collidedAgainstIndex: i.index, type: this._nativeCollisionValueToCollisionType(t.type) }; if (s.type === q0.COLLISION_FINISHED) this.onCollisionEndedObservable.notifyObservers(s); else { t.contactOnB.position.subtractToRef(t.contactOnA.position, this._tmpVec3[0]); const a = S.Dot(this._tmpVec3[0], t.contactOnA.normal); s.point = t.contactOnA.position, s.distance = a, s.impulse = t.impulseApplied, s.normal = t.contactOnA.normal, this.onCollisionObservable.notifyObservers(s); } if (this._bodyCollisionObservable.size && s.type !== q0.COLLISION_FINISHED) { const a = this._bodyCollisionObservable.get(t.contactOnA.bodyId), f = this._bodyCollisionObservable.get(t.contactOnB.bodyId); a ? a.notifyObservers(s) : f && (s.collider = i.body, s.colliderIndex = i.index, s.collidedAgainst = n.body, s.collidedAgainstIndex = n.index, s.normal = t.contactOnB.normal, f.notifyObservers(s)); } else if (this._bodyCollisionEndedObservable.size) { const a = this._bodyCollisionEndedObservable.get(t.contactOnA.bodyId), f = this._bodyCollisionEndedObservable.get(t.contactOnB.bodyId); a ? a.notifyObservers(s) : f && (s.collider = i.body, s.colliderIndex = i.index, s.collidedAgainst = n.body, s.collidedAgainstIndex = n.index, s.normal = t.contactOnB.normal, f.notifyObservers(s)); } } e = this._hknp.HP_World_GetNextCollisionEvent(r, e); } } /** * Gets the number of bodies in the world */ get numBodies() { return this._hknp.HP_World_GetNumBodies(this.world)[1]; } /** * Dispose the world and free resources */ dispose() { this._hknp.HP_QueryCollector_Release(this._queryCollector), this._queryCollector = BigInt(0), this._hknp.HP_World_Release(this.world), this.world = void 0; } _v3ToBvecRef(e, t) { t.set(e[0], e[1], e[2]); } _bVecToV3(e) { return [e._x, e._y, e._z]; } _bQuatToV4(e) { return [e._x, e._y, e._z, e._w]; } _constraintMotorTypeToNative(e) { switch (e) { case _x.POSITION: return this._hknp.ConstraintMotorType.POSITION; case _x.VELOCITY: return this._hknp.ConstraintMotorType.VELOCITY; } return this._hknp.ConstraintMotorType.NONE; } _nativeToMotorType(e) { switch (e) { case this._hknp.ConstraintMotorType.POSITION: return _x.POSITION; case this._hknp.ConstraintMotorType.VELOCITY: return _x.VELOCITY; } return _x.NONE; } _materialCombineToNative(e) { switch (e) { case R2.GEOMETRIC_MEAN: return this._hknp.MaterialCombine.GEOMETRIC_MEAN; case R2.MINIMUM: return this._hknp.MaterialCombine.MINIMUM; case R2.MAXIMUM: return this._hknp.MaterialCombine.MAXIMUM; case R2.ARITHMETIC_MEAN: return this._hknp.MaterialCombine.ARITHMETIC_MEAN; case R2.MULTIPLY: return this._hknp.MaterialCombine.MULTIPLY; } } _constraintAxisToNative(e) { switch (e) { case MH.LINEAR_X: return this._hknp.ConstraintAxis.LINEAR_X; case MH.LINEAR_Y: return this._hknp.ConstraintAxis.LINEAR_Y; case MH.LINEAR_Z: return this._hknp.ConstraintAxis.LINEAR_Z; case MH.ANGULAR_X: return this._hknp.ConstraintAxis.ANGULAR_X; case MH.ANGULAR_Y: return this._hknp.ConstraintAxis.ANGULAR_Y; case MH.ANGULAR_Z: return this._hknp.ConstraintAxis.ANGULAR_Z; case MH.LINEAR_DISTANCE: return this._hknp.ConstraintAxis.LINEAR_DISTANCE; } } _nativeToLimitMode(e) { switch (e) { case this._hknp.ConstraintAxisLimitMode.FREE: return I2.FREE; case this._hknp.ConstraintAxisLimitMode.LIMITED: return I2.LIMITED; case this._hknp.ConstraintAxisLimitMode.LOCKED: return I2.LOCKED; } return I2.FREE; } _limitModeToNative(e) { switch (e) { case I2.FREE: return this._hknp.ConstraintAxisLimitMode.FREE; case I2.LIMITED: return this._hknp.ConstraintAxisLimitMode.LIMITED; case I2.LOCKED: return this._hknp.ConstraintAxisLimitMode.LOCKED; } } _nativeCollisionValueToCollisionType(e) { switch (e) { case this._hknp.EventType.COLLISION_STARTED.value: return q0.COLLISION_STARTED; case this._hknp.EventType.COLLISION_FINISHED.value: return q0.COLLISION_FINISHED; case this._hknp.EventType.COLLISION_CONTINUED.value: return q0.COLLISION_CONTINUED; } return q0.COLLISION_STARTED; } _nativeTriggerCollisionValueToCollisionType(e) { switch (e) { case 8: return q0.TRIGGER_ENTERED; case 16: return q0.TRIGGER_EXITED; } return q0.TRIGGER_ENTERED; } } sr.prototype.getPhysicsEngine = function() { return this._physicsEngine; }; sr.prototype.enablePhysics = function(A = null, e) { if (this._physicsEngine) return !0; let t = this._getComponent(Ot.NAME_PHYSICSENGINE); t || (t = new A0e(this), this._addComponent(t)); try { if (!e || (e == null ? void 0 : e.getPluginVersion()) === 1) this._physicsEngine = new $Q(A, e); else if ((e == null ? void 0 : e.getPluginVersion()) === 2) this._physicsEngine = new jy(A, e); else throw new Error("Unsupported Physics plugin version."); return this._physicsTimeAccumulator = 0, !0; } catch (r) { return Se.Error(r.message), !1; } }; sr.prototype.disablePhysicsEngine = function() { this._physicsEngine && (this._physicsEngine.dispose(), this._physicsEngine = null); }; sr.prototype.isPhysicsEnabled = function() { return this._physicsEngine !== void 0; }; sr.prototype.deleteCompoundImpostor = function(A) { const e = A.parts[0].mesh; e.physicsImpostor && (e.physicsImpostor.dispose( /*true*/ ), e.physicsImpostor = null); }; sr.prototype._advancePhysicsEngineStep = function(A) { if (this._physicsEngine) { const e = this._physicsEngine.getSubTimeStep(); if (e > 0) for (this._physicsTimeAccumulator += A; this._physicsTimeAccumulator > e; ) this.onBeforePhysicsObservable.notifyObservers(this), this._physicsEngine._step(e / 1e3), this.onAfterPhysicsObservable.notifyObservers(this), this._physicsTimeAccumulator -= e; else this.onBeforePhysicsObservable.notifyObservers(this), this._physicsEngine._step(A / 1e3), this.onAfterPhysicsObservable.notifyObservers(this); } }; class A0e { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_PHYSICSENGINE, this.scene = e, this.scene.onBeforePhysicsObservable = new Oe(), this.scene.onAfterPhysicsObservable = new Oe(), this.scene.getDeterministicFrameTime = () => this.scene._physicsEngine ? this.scene._physicsEngine.getTimeStep() * 1e3 : 1e3 / 60; } /** * Registers the component in a given scene */ register() { } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { } /** * Disposes the component and the associated resources */ dispose() { this.scene.onBeforePhysicsObservable.clear(), this.scene.onAfterPhysicsObservable.clear(), this.scene._physicsEngine && this.scene.disablePhysicsEngine(); } } Object.defineProperty(Hr.prototype, "physicsBody", { get: function() { return this._physicsBody; }, set: function(A) { this._physicsBody !== A && (this._disposePhysicsObserver && this.onDisposeObservable.remove(this._disposePhysicsObserver), this._physicsBody = A, A && (this._disposePhysicsObserver = this.onDisposeObservable.add(() => { this.physicsBody && (this.physicsBody.dispose( /*!doNotRecurse*/ ), this.physicsBody = null); }))); }, enumerable: !0, configurable: !0 }); Hr.prototype.getPhysicsBody = function() { return this.physicsBody; }; Hr.prototype.applyImpulse = function(A, e) { if (!this.physicsBody) throw new Error("No Physics Body for TransformNode"); return this.physicsBody.applyImpulse(A, e), this; }; class sm { /* * Gets the hit contact point between a mesh and a ray. The method varies between * the different plugin versions; V1 uses a mesh intersection, V2 uses the physics body instance/object center (to avoid a raycast and improve perf). */ static GetContactPointToRef(e, t, r, n, i) { const s = e.getScene().getPhysicsEngine(), a = s == null ? void 0 : s.getPluginVersion(); if (a === 1) { const o = new Hi(t, r).intersectsMesh(e); if (o.hit && o.pickedPoint) return n.copyFrom(o.pickedPoint), !0; } else if (a === 2) return e.physicsBody.getObjectCenterWorldToRef(n, i), !0; return !1; } /** * Checks if a body will be affected by forces * @param body the body to check * @param instanceIndex for instanced bodies, the index of the instance to check * @returns */ static HasAppliedForces(e, t) { var r, n, i; return e.getMotionType(t) === W0.STATIC || ((n = (r = e.getMassProperties(t)) === null || r === void 0 ? void 0 : r.mass) !== null && n !== void 0 ? n : 0) === 0 || ((i = e.transformNode) === null || i === void 0 ? void 0 : i.getTotalVertices()) === 0; } /** * Checks if a point is inside a cylinder * @param point point to check * @param origin cylinder origin on the bottom * @param radius cylinder radius * @param height cylinder height * @returns */ static IsInsideCylinder(e, t, r, n) { const i = ue.Vector3[0]; return e.subtractToRef(t, i), Math.abs(i.x) <= r && Math.abs(i.z) <= r && i.y >= 0 && i.y <= n; } } class d0e { /** * Initializes the Physics helper * @param scene Babylon.js scene */ constructor(e) { if (this._hitData = { force: new S(), contactPoint: new S(), distanceFromOrigin: 0 }, this._scene = e, this._physicsEngine = this._scene.getPhysicsEngine(), !this._physicsEngine) { Se.Warn("Physics engine not enabled. Please enable the physics before you can use the methods."); return; } } /** * Applies a radial explosion impulse * @param origin the origin of the explosion * @param radiusOrEventOptions the radius or the options of radial explosion * @param strength the explosion strength * @param falloff possible options: Constant & Linear. Defaults to Constant * @returns A physics radial explosion event, or null */ applyRadialExplosionImpulse(e, t, r, n) { if (!this._physicsEngine) return Se.Warn("Physics engine not enabled. Please enable the physics before you call this method."), null; if (this._physicsEngine.getPluginVersion() === 1 && this._physicsEngine.getImpostors().length === 0 || this._physicsEngine.getPluginVersion() === 2 && this._physicsEngine.getBodies().length === 0) return null; let i = !1; if (typeof t == "number") { const f = t; t = new sS(), t.radius = f, t.strength = r ?? t.strength, t.falloff = n ?? t.falloff; } else i = !!(t.affectedImpostorsCallback || t.affectedBodiesCallback); const s = new _G(this._scene, t), a = this._hitData; if (this._physicsEngine.getPluginVersion() === 1) { const f = Array(); this._physicsEngine.getImpostors().forEach((d) => { s.getImpostorHitData(d, e, a) && (d.applyImpulse(a.force, a.contactPoint), i && f.push({ impostor: d, hitData: this._copyPhysicsHitData(a) })); }), s.triggerAffectedImpostorsCallback(f); } else this._applicationForBodies(s, e, a, i, (f, o) => { f.applyImpulse(o.force, o.contactPoint, o.instanceIndex); }); return s.dispose(!1), s; } /** * Applies a radial explosion force * @param origin the origin of the explosion * @param radiusOrEventOptions the radius or the options of radial explosion * @param strength the explosion strength * @param falloff possible options: Constant & Linear. Defaults to Constant * @returns A physics radial explosion event, or null */ applyRadialExplosionForce(e, t, r, n) { if (!this._physicsEngine) return Se.Warn("Physics engine not enabled. Please enable the physics before you call the PhysicsHelper."), null; if (this._physicsEngine.getPluginVersion() === 1 && this._physicsEngine.getImpostors().length === 0 || this._physicsEngine.getPluginVersion() === 2 && this._physicsEngine.getBodies().length === 0) return null; let i = !1; if (typeof t == "number") { const f = t; t = new sS(), t.radius = f, t.strength = r ?? t.strength, t.falloff = n ?? t.falloff; } else i = !!(t.affectedImpostorsCallback || t.affectedBodiesCallback); const s = new _G(this._scene, t), a = this._hitData; if (this._physicsEngine.getPluginVersion() === 1) { const f = Array(); this._physicsEngine.getImpostors().forEach((d) => { s.getImpostorHitData(d, e, a) && (d.applyForce(a.force, a.contactPoint), i && f.push({ impostor: d, hitData: this._copyPhysicsHitData(a) })); }), s.triggerAffectedImpostorsCallback(f); } else this._applicationForBodies(s, e, a, i, (f, o) => { f.applyForce(o.force, o.contactPoint, o.instanceIndex); }); return s.dispose(!1), s; } _applicationForBodies(e, t, r, n, i) { const s = Array(), a = this._physicsEngine.getBodies(); for (const f of a) f.iterateOverAllInstances((o, d) => { e.getBodyHitData(o, t, r, d) && (i(o, r), n && s.push({ body: o, hitData: this._copyPhysicsHitData(r) })); }); e.triggerAffectedBodiesCallback(s); } /** * Creates a gravitational field * @param origin the origin of the gravitational field * @param radiusOrEventOptions the radius or the options of radial gravitational field * @param strength the gravitational field strength * @param falloff possible options: Constant & Linear. Defaults to Constant * @returns A physics gravitational field event, or null */ gravitationalField(e, t, r, n) { if (!this._physicsEngine) return Se.Warn("Physics engine not enabled. Please enable the physics before you call the PhysicsHelper."), null; if (this._physicsEngine.getPluginVersion() === 1 && this._physicsEngine.getImpostors().length === 0 || this._physicsEngine.getPluginVersion() === 2 && this._physicsEngine.getBodies().length === 0) return null; if (typeof t == "number") { const s = t; t = new sS(), t.radius = s, t.strength = r ?? t.strength, t.falloff = n ?? t.falloff; } const i = new v0e(this, this._scene, e, t); return i.dispose(!1), i; } /** * Creates a physics updraft event * @param origin the origin of the updraft * @param radiusOrEventOptions the radius or the options of the updraft * @param strength the strength of the updraft * @param height the height of the updraft * @param updraftMode possible options: Center & Perpendicular. Defaults to Center * @returns A physics updraft event, or null */ updraft(e, t, r, n, i) { if (!this._physicsEngine) return Se.Warn("Physics engine not enabled. Please enable the physics before you call the PhysicsHelper."), null; if (this._physicsEngine.getPluginVersion() === 1 && this._physicsEngine.getImpostors().length === 0 || this._physicsEngine.getPluginVersion() === 2 && this._physicsEngine.getBodies().length === 0) return null; if (typeof t == "number") { const a = t; t = new YY(), t.radius = a, t.strength = r ?? t.strength, t.height = n ?? t.height, t.updraftMode = i ?? t.updraftMode; } const s = new my(this._scene, e, t); return s.dispose(!1), s; } /** * Creates a physics vortex event * @param origin the of the vortex * @param radiusOrEventOptions the radius or the options of the vortex * @param strength the strength of the vortex * @param height the height of the vortex * @returns a Physics vortex event, or null * A physics vortex event or null */ vortex(e, t, r, n) { if (!this._physicsEngine) return Se.Warn("Physics engine not enabled. Please enable the physics before you call the PhysicsHelper."), null; if (this._physicsEngine.getPluginVersion() === 1 && this._physicsEngine.getImpostors().length === 0 || this._physicsEngine.getPluginVersion() === 2 && this._physicsEngine.getBodies().length === 0) return null; if (typeof t == "number") { const s = t; t = new MY(), t.radius = s, t.strength = r ?? t.strength, t.height = n ?? t.height; } const i = new QS(this._scene, e, t); return i.dispose(!1), i; } _copyPhysicsHitData(e) { return { force: e.force.clone(), contactPoint: e.contactPoint.clone(), distanceFromOrigin: e.distanceFromOrigin, instanceIndex: e.instanceIndex }; } } class _G { /** * Initializes a radial explosion event * @param _scene BabylonJS scene * @param _options The options for the vortex event */ constructor(e, t) { this._scene = e, this._options = t, this._dataFetched = !1, this._options = Object.assign(Object.assign({}, new sS()), this._options); } /** * Returns the data related to the radial explosion event (sphere). * @returns The radial explosion event data */ getData() { return this._dataFetched = !0, { sphere: this._sphere }; } _getHitData(e, t, r, n) { const i = ue.Vector3[0]; i.copyFrom(t).subtractInPlace(r); const s = ue.Vector3[1]; if (!sm.GetContactPointToRef(e, r, i, s, n.instanceIndex)) return !1; const f = S.Distance(r, s); if (f > this._options.radius) return !1; const o = this._options.falloff === iR.Constant ? this._options.strength : this._options.strength * (1 - f / this._options.radius); return i.scaleInPlace(o), n.force.copyFrom(i), n.contactPoint.copyFrom(s), n.distanceFromOrigin = f, !0; } /** * Returns the force and contact point of the body or false, if the body is not affected by the force/impulse. * @param body A physics body where the transform node is an AbstractMesh * @param origin the origin of the explosion * @param data the data of the hit * @param instanceIndex the instance index of the body * @returns if there was a hit */ getBodyHitData(e, t, r, n) { if (sm.HasAppliedForces(e, n)) return !1; const i = e.transformNode, s = e.getObjectCenterWorld(n); return r.instanceIndex = n, this._getHitData(i, s, t, r); } /** * Returns the force and contact point of the impostor or false, if the impostor is not affected by the force/impulse. * @param impostor A physics imposter * @param origin the origin of the explosion * @returns A physics force and contact point, or null */ getImpostorHitData(e, t, r) { if (e.mass === 0 || e.object.getClassName() !== "Mesh" && e.object.getClassName() !== "InstancedMesh") return !1; const n = e.object; if (!this._intersectsWithSphere(n, t, this._options.radius)) return !1; const i = e.getObjectCenter(); return this._getHitData(n, i, t, r), !0; } /** * Triggers affected impostors callbacks * @param affectedImpostorsWithData defines the list of affected impostors (including associated data) */ triggerAffectedImpostorsCallback(e) { this._options.affectedImpostorsCallback && this._options.affectedImpostorsCallback(e); } /** * Triggers affected bodies callbacks * @param affectedBodiesWithData defines the list of affected bodies (including associated data) */ triggerAffectedBodiesCallback(e) { this._options.affectedBodiesCallback && this._options.affectedBodiesCallback(e); } /** * Disposes the sphere. * @param force Specifies if the sphere should be disposed by force */ dispose(e = !0) { this._sphere && (e ? this._sphere.dispose() : setTimeout(() => { this._dataFetched || this._sphere.dispose(); }, 0)); } /*** Helpers ***/ _prepareSphere() { this._sphere || (this._sphere = UA("radialExplosionEventSphere", this._options.sphere, this._scene), this._sphere.isVisible = !1); } _intersectsWithSphere(e, t, r) { return this._prepareSphere(), this._sphere.position = t, this._sphere.scaling.setAll(r * 2), this._sphere._updateBoundingInfo(), this._sphere.computeWorldMatrix(!0), this._sphere.intersectsMesh(e, !0); } } class v0e { /** * Initializes the physics gravitational field event * @param _physicsHelper A physics helper * @param _scene BabylonJS scene * @param _origin The origin position of the gravitational field event * @param _options The options for the vortex event */ constructor(e, t, r, n) { this._physicsHelper = e, this._scene = t, this._origin = r, this._options = n, this._dataFetched = !1, this._options = Object.assign(Object.assign({}, new sS()), this._options), this._tickCallback = () => this._tick(), this._options.strength = this._options.strength * -1; } /** * Returns the data related to the gravitational field event (sphere). * @returns A gravitational field event */ getData() { return this._dataFetched = !0, { sphere: this._sphere }; } /** * Enables the gravitational field. */ enable() { this._tickCallback.call(this), this._scene.registerBeforeRender(this._tickCallback); } /** * Disables the gravitational field. */ disable() { this._scene.unregisterBeforeRender(this._tickCallback); } /** * Disposes the sphere. * @param force The force to dispose from the gravitational field event */ dispose(e = !0) { this._sphere && (e ? this._sphere.dispose() : setTimeout(() => { this._dataFetched || this._sphere.dispose(); }, 0)); } _tick() { var e; if (this._sphere) this._physicsHelper.applyRadialExplosionForce(this._origin, this._options); else { const t = this._physicsHelper.applyRadialExplosionForce(this._origin, this._options); t && (this._sphere = (e = t.getData().sphere) === null || e === void 0 ? void 0 : e.clone("radialExplosionEventSphereClone")); } } } class my { /** * Initializes the physics updraft event * @param _scene BabylonJS scene * @param _origin The origin position of the updraft * @param _options The options for the updraft event */ constructor(e, t, r) { this._scene = e, this._origin = t, this._options = r, this._originTop = S.Zero(), this._originDirection = S.Zero(), this._cylinderPosition = S.Zero(), this._dataFetched = !1, this._physicsEngine = this._scene.getPhysicsEngine(), this._options = Object.assign(Object.assign({}, new YY()), this._options), this._origin.addToRef(new S(0, this._options.height / 2, 0), this._cylinderPosition), this._origin.addToRef(new S(0, this._options.height, 0), this._originTop), this._options.updraftMode === YS.Perpendicular && (this._originDirection = this._origin.subtract(this._originTop).normalize()), this._tickCallback = () => this._tick(), this._physicsEngine.getPluginVersion() === 1 && this._prepareCylinder(); } /** * Returns the data related to the updraft event (cylinder). * @returns A physics updraft event */ getData() { return this._dataFetched = !0, { cylinder: this._cylinder }; } /** * Enables the updraft. */ enable() { this._tickCallback.call(this), this._scene.registerBeforeRender(this._tickCallback); } /** * Disables the updraft. */ disable() { this._scene.unregisterBeforeRender(this._tickCallback); } /** * Disposes the cylinder. * @param force Specifies if the updraft should be disposed by force */ dispose(e = !0) { this._cylinder && (e ? (this._cylinder.dispose(), this._cylinder = void 0) : setTimeout(() => { !this._dataFetched && this._cylinder && (this._cylinder.dispose(), this._cylinder = void 0); }, 0)); } _getHitData(e, t) { let r; this._options.updraftMode === YS.Perpendicular ? r = this._originDirection : r = e.subtract(this._originTop); const n = S.Distance(this._origin, e), i = this._options.strength * -1, s = r.multiplyByFloats(i, i, i); t.force.copyFrom(s), t.contactPoint.copyFrom(e), t.distanceFromOrigin = n; } _getBodyHitData(e, t, r) { if (sm.HasAppliedForces(e)) return !1; const n = e.getObjectCenterWorld(r); return sm.IsInsideCylinder(n, this._origin, this._options.radius, this._options.height) ? (t.instanceIndex = r, this._getHitData(n, t), !0) : !1; } _getImpostorHitData(e, t) { if (e.mass === 0) return !1; const r = e.object; if (!this._intersectsWithCylinder(r)) return !1; const n = e.getObjectCenter(); return this._getHitData(n, t), !0; } _tick() { const e = my._HitData; this._physicsEngine.getPluginVersion() === 1 ? this._physicsEngine.getImpostors().forEach((t) => { this._getImpostorHitData(t, e) && t.applyForce(e.force, e.contactPoint); }) : this._physicsEngine.getBodies().forEach((t) => { t.iterateOverAllInstances((r, n) => { this._getBodyHitData(r, e, n) && r.applyForce(e.force, e.contactPoint, e.instanceIndex); }); }); } /*** Helpers ***/ _prepareCylinder() { this._cylinder || (this._cylinder = Ld("updraftEventCylinder", { height: this._options.height, diameter: this._options.radius * 2 }, this._scene), this._cylinder.isVisible = !1); } _intersectsWithCylinder(e) { return this._cylinder ? (this._cylinder.position = this._cylinderPosition, this._cylinder.intersectsMesh(e, !0)) : !1; } } my._HitData = { force: new S(), contactPoint: new S(), distanceFromOrigin: 0 }; class QS { /** * Initializes the physics vortex event * @param _scene The BabylonJS scene * @param _origin The origin position of the vortex * @param _options The options for the vortex event */ constructor(e, t, r) { this._scene = e, this._origin = t, this._options = r, this._originTop = S.Zero(), this._cylinderPosition = S.Zero(), this._dataFetched = !1, this._physicsEngine = this._scene.getPhysicsEngine(), this._options = Object.assign(Object.assign({}, new MY()), this._options), this._origin.addToRef(new S(0, this._options.height / 2, 0), this._cylinderPosition), this._origin.addToRef(new S(0, this._options.height, 0), this._originTop), this._tickCallback = () => this._tick(), this._physicsEngine.getPluginVersion() === 1 && this._prepareCylinder(); } /** * Returns the data related to the vortex event (cylinder). * @returns The physics vortex event data */ getData() { return this._dataFetched = !0, { cylinder: this._cylinder }; } /** * Enables the vortex. */ enable() { this._tickCallback.call(this), this._scene.registerBeforeRender(this._tickCallback); } /** * Disables the cortex. */ disable() { this._scene.unregisterBeforeRender(this._tickCallback); } /** * Disposes the sphere. * @param force */ dispose(e = !0) { this._cylinder && (e ? this._cylinder.dispose() : setTimeout(() => { this._dataFetched || this._cylinder.dispose(); }, 0)); } _getHitData(e, t, r) { const n = QS.originOnPlane; n.set(this._origin.x, t.y, this._origin.z); const i = ue.Vector3[0]; t.subtractToRef(n, i); const s = ue.Vector3[1]; if (!sm.GetContactPointToRef(e, n, i, s, r.instanceIndex)) return !1; const o = S.Distance(s, n) / this._options.radius, d = ue.Vector3[2]; s.normalizeToRef(d), o > this._options.centripetalForceThreshold && d.negateInPlace(); let v, u, l; if (o > this._options.centripetalForceThreshold) v = d.x * this._options.centripetalForceMultiplier, u = d.y * this._options.updraftForceMultiplier, l = d.z * this._options.centripetalForceMultiplier; else { const p = S.Cross(n, t).normalize(); v = (p.x + d.x) * this._options.centrifugalForceMultiplier, u = this._originTop.y * this._options.updraftForceMultiplier, l = (p.z + d.z) * this._options.centrifugalForceMultiplier; } const P = ue.Vector3[3]; return P.set(v, u, l), P.scaleInPlace(this._options.strength), r.force.copyFrom(P), r.contactPoint.copyFrom(t), r.distanceFromOrigin = o, !0; } _getBodyHitData(e, t, r) { if (sm.HasAppliedForces(e, r)) return !1; const n = e.transformNode, i = e.getObjectCenterWorld(r); return sm.IsInsideCylinder(i, this._origin, this._options.radius, this._options.height) ? (t.instanceIndex = r, this._getHitData(n, i, t)) : !1; } _getImpostorHitData(e, t) { if (e.mass === 0 || e.object.getClassName() !== "Mesh" && e.object.getClassName() !== "InstancedMesh") return !1; const r = e.object; if (!this._intersectsWithCylinder(r)) return !1; const n = e.getObjectCenter(); return this._getHitData(r, n, t), !0; } _tick() { const e = QS.hitData; this._physicsEngine.getPluginVersion() === 1 ? this._physicsEngine.getImpostors().forEach((t) => { this._getImpostorHitData(t, e) && t.applyForce(e.force, e.contactPoint); }) : this._physicsEngine.getBodies().forEach((t) => { t.iterateOverAllInstances((r, n) => { this._getBodyHitData(r, e, n) && r.applyForce(e.force, e.contactPoint, e.instanceIndex); }); }); } /*** Helpers ***/ _prepareCylinder() { this._cylinder || (this._cylinder = Ld("vortexEventCylinder", { height: this._options.height, diameter: this._options.radius * 2 }, this._scene), this._cylinder.isVisible = !1); } _intersectsWithCylinder(e) { return this._cylinder.position = this._cylinderPosition, this._cylinder.intersectsMesh(e, !0); } } QS.originOnPlane = S.Zero(); QS.hitData = { force: new S(), contactPoint: new S(), distanceFromOrigin: 0 }; class sS { constructor() { this.radius = 5, this.strength = 10, this.falloff = iR.Constant, this.sphere = { segments: 32, diameter: 1 }; } } class YY { constructor() { this.radius = 5, this.strength = 10, this.height = 10, this.updraftMode = YS.Center; } } class MY { constructor() { this.radius = 5, this.strength = 10, this.height = 10, this.centripetalForceThreshold = 0.7, this.centripetalForceMultiplier = 5, this.centrifugalForceMultiplier = 0.5, this.updraftForceMultiplier = 0.02; } } var iR; (function(A) { A[A.Constant = 0] = "Constant", A[A.Linear = 1] = "Linear"; })(iR || (iR = {})); var YS; (function(A) { A[A.Center = 0] = "Center", A[A.Perpendicular = 1] = "Perpendicular"; })(YS || (YS = {})); const u0e = "blackAndWhitePixelShader", l0e = `varying vec2 vUV;uniform sampler2D textureSampler;uniform float degree; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {vec3 color=texture2D(textureSampler,vUV).rgb;float luminance=dot(color,vec3(0.3,0.59,0.11)); vec3 blackAndWhite=vec3(luminance,luminance,luminance);gl_FragColor=vec4(color-((color-blackAndWhite)*degree),1.0);}`; Le.ShadersStore[u0e] = l0e; class AV extends kr { /** * Gets a string identifying the name of the class * @returns "BlackAndWhitePostProcess" string */ getClassName() { return "BlackAndWhitePostProcess"; } /** * Creates a black and white post process * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/usePostProcesses#black-and-white * @param name The name of the effect. * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) */ constructor(e, t, r, n, i, s) { super(e, "blackAndWhite", ["degree"], null, t, r, n, i, s), this.degree = 1, this.onApplyObservable.add((a) => { a.setFloat("degree", this.degree); }); } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new AV(e.name, e.options, t, e.renderTargetSamplingMode, r.getEngine(), e.reusable), e, r, n); } } C([ M() ], AV.prototype, "degree", void 0); Ue("BABYLON.BlackAndWhitePostProcess", AV); class gs { /** * Instantiates a post process render effect. * A post process can be used to apply a shader to a texture after it is rendered. * @param engine The engine the effect is tied to * @param name The name of the effect * @param getPostProcesses A function that returns a set of post processes which the effect will run in order to be run. * @param singleInstance False if this post process can be run on multiple cameras. (default: true) */ constructor(e, t, r, n) { this._name = t, this._singleInstance = n || !0, this._getPostProcesses = r, this._cameras = {}, this._indicesForCamera = {}, this._postProcesses = {}; } /** * Checks if all the post processes in the effect are supported. */ get isSupported() { for (const e in this._postProcesses) if (Object.prototype.hasOwnProperty.call(this._postProcesses, e)) { const t = this._postProcesses[e]; for (let r = 0; r < t.length; r++) if (!t[r].isSupported) return !1; } return !0; } /** * Updates the current state of the effect * @internal */ _update() { } /** * Attaches the effect on cameras * @param cameras The camera to attach to. * @internal */ _attachCameras(e) { let t; const r = ye.MakeArray(e || this._cameras); if (r) for (let n = 0; n < r.length; n++) { const i = r[n]; if (!i) continue; const s = i.name; if (this._singleInstance ? t = 0 : t = s, !this._postProcesses[t]) { const a = this._getPostProcesses(); a && (this._postProcesses[t] = Array.isArray(a) ? a : [a]); } this._indicesForCamera[s] || (this._indicesForCamera[s] = []), this._postProcesses[t].forEach((a) => { const f = i.attachPostProcess(a); this._indicesForCamera[s].push(f); }), this._cameras[s] || (this._cameras[s] = i); } } /** * Detaches the effect on cameras * @param cameras The camera to detach from. * @internal */ _detachCameras(e) { const t = ye.MakeArray(e || this._cameras); if (t) for (let r = 0; r < t.length; r++) { const n = t[r], i = n.name, s = this._postProcesses[this._singleInstance ? 0 : i]; s && s.forEach((a) => { n.detachPostProcess(a); }), this._cameras[i] && (this._cameras[i] = null), delete this._indicesForCamera[i]; } } /** * Enables the effect on given cameras * @param cameras The camera to enable. * @internal */ _enable(e) { const t = ye.MakeArray(e || this._cameras); if (t) for (let r = 0; r < t.length; r++) { const n = t[r], i = n.name, s = this._singleInstance ? 0 : i; for (let a = 0; a < this._indicesForCamera[i].length; a++) { const f = this._indicesForCamera[i][a], o = n._postProcesses[f]; o == null && t[r].attachPostProcess(this._postProcesses[s][a], f); } } } /** * Disables the effect on the given cameras * @param cameras The camera to disable. * @internal */ _disable(e) { const t = ye.MakeArray(e || this._cameras); if (t) for (let r = 0; r < t.length; r++) { const n = t[r], i = n.name; this._postProcesses[this._singleInstance ? 0 : i].forEach((s) => { n.detachPostProcess(s); }); } } /** * Gets a list of the post processes contained in the effect. * @param camera The camera to get the post processes on. * @returns The list of the post processes in the effect. */ getPostProcesses(e) { return this._singleInstance ? this._postProcesses[0] : e ? this._postProcesses[e.name] : null; } } const P0e = "extractHighlightsPixelShader", c0e = `#include varying vec2 vUV;uniform sampler2D textureSampler;uniform float threshold;uniform float exposure; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {gl_FragColor=texture2D(textureSampler,vUV);float luma=dot(LuminanceEncodeApprox,gl_FragColor.rgb*exposure);gl_FragColor.rgb=step(threshold,luma)*gl_FragColor.rgb;}`; Le.ShadersStore[P0e] = c0e; class By extends kr { /** * Gets a string identifying the name of the class * @returns "ExtractHighlightsPostProcess" string */ getClassName() { return "ExtractHighlightsPostProcess"; } constructor(e, t, r, n, i, s, a = 0, f = !1) { super(e, "extractHighlights", ["threshold", "exposure"], null, t, r, n, i, s, null, a, void 0, null, f), this.threshold = 0.9, this._exposure = 1, this._inputPostProcess = null, this.onApplyObservable.add((o) => { this.externalTextureSamplerBinding = !!this._inputPostProcess, this._inputPostProcess && o.setTextureFromPostProcess("textureSampler", this._inputPostProcess), o.setFloat("threshold", Math.pow(this.threshold, JW)), o.setFloat("exposure", this._exposure); }); } } C([ M() ], By.prototype, "threshold", void 0); Ue("BABYLON.ExtractHighlightsPostProcess", By); const p0e = "bloomMergePixelShader", h0e = `uniform sampler2D textureSampler;uniform sampler2D bloomBlur;varying vec2 vUV;uniform float bloomWeight; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {gl_FragColor=texture2D(textureSampler,vUV);vec3 blurred=texture2D(bloomBlur,vUV).rgb;gl_FragColor.rgb=gl_FragColor.rgb+(blurred.rgb*bloomWeight); } `; Le.ShadersStore[p0e] = h0e; class Wy extends kr { /** * Gets a string identifying the name of the class * @returns "BloomMergePostProcess" string */ getClassName() { return "BloomMergePostProcess"; } /** * Creates a new instance of @see BloomMergePostProcess * @param name The name of the effect. * @param originalFromInput Post process which's input will be used for the merge. * @param blurred Blurred highlights post process which's output will be used. * @param weight Weight of the bloom to be added to the original input. * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) * @param textureType Type of textures used when performing the post process. (default: 0) * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false) */ constructor(e, t, r, n, i, s, a, f, o, d = 0, v = !1) { super(e, "bloomMerge", ["bloomWeight"], ["bloomBlur"], i, s, a, f, o, null, d, void 0, null, !0), this.weight = 1, this.weight = n, this.externalTextureSamplerBinding = !0, this.onApplyObservable.add((u) => { u.setTextureFromPostProcess("textureSampler", t), u.setTextureFromPostProcessOutput("bloomBlur", r), u.setFloat("bloomWeight", this.weight); }), v || this.updateEffect(); } } C([ M() ], Wy.prototype, "weight", void 0); Ue("BABYLON.BloomMergePostProcess", Wy); class DF extends gs { /** * The luminance threshold to find bright areas of the image to bloom. */ get threshold() { return this._downscale.threshold; } set threshold(e) { this._downscale.threshold = e; } /** * The strength of the bloom. */ get weight() { return this._merge.weight; } set weight(e) { this._merge.weight = e; } /** * Specifies the size of the bloom blur kernel, relative to the final output size */ get kernel() { return this._blurX.kernel / this._bloomScale; } set kernel(e) { this._blurX.kernel = e * this._bloomScale, this._blurY.kernel = e * this._bloomScale; } /** * Creates a new instance of @see BloomEffect * @param scene The scene the effect belongs to. * @param _bloomScale The ratio of the blur texture to the input texture that should be used to compute the bloom. * @param bloomWeight The strength of bloom. * @param bloomKernel The size of the kernel to be used when applying the blur. * @param pipelineTextureType The type of texture to be used when performing the post processing. * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false) */ constructor(e, t, r, n, i = 0, s = !1) { super(e.getEngine(), "bloom", () => this._effects, !0), this._bloomScale = t, this._effects = [], this._downscale = new By("highlights", 1, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, i, s), this._blurX = new c9("horizontal blur", new at(1, 0), 10, t, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, i, void 0, s), this._blurX.alwaysForcePOT = !0, this._blurX.autoClear = !1, this._blurY = new c9("vertical blur", new at(0, 1), 10, t, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, i, void 0, s), this._blurY.alwaysForcePOT = !0, this._blurY.autoClear = !1, this.kernel = n, this._effects = [this._downscale, this._blurX, this._blurY], this._merge = new Wy("bloomMerge", this._downscale, this._blurY, r, t, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, i, s), this._merge.autoClear = !1, this._effects.push(this._merge); } /** * Disposes each of the internal effects for a given camera. * @param camera The camera to dispose the effect on. */ disposeEffects(e) { for (let t = 0; t < this._effects.length; t++) this._effects[t].dispose(e); } /** * @internal Internal */ _updateEffects() { for (let e = 0; e < this._effects.length; e++) this._effects[e].updateEffect(); } /** * Internal * @returns if all the contained post processes are ready. * @internal */ _isReady() { for (let e = 0; e < this._effects.length; e++) if (!this._effects[e].isReady()) return !1; return !0; } } const H0e = "chromaticAberrationPixelShader", g0e = `uniform sampler2D textureSampler; uniform float chromatic_aberration;uniform float radialIntensity;uniform vec2 direction;uniform vec2 centerPosition;uniform float screen_width;uniform float screen_height;varying vec2 vUV; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {vec2 centered_screen_pos=vec2(vUV.x-centerPosition.x,vUV.y-centerPosition.y);vec2 directionOfEffect=direction;if(directionOfEffect.x==0. && directionOfEffect.y==0.){directionOfEffect=normalize(centered_screen_pos);} float radius2=centered_screen_pos.x*centered_screen_pos.x + centered_screen_pos.y*centered_screen_pos.y;float radius=sqrt(radius2);vec4 original=texture2D(textureSampler,vUV);vec3 ref_indices=vec3(-0.3,0.0,0.3);float ref_shiftX=chromatic_aberration*pow(radius,radialIntensity)*directionOfEffect.x/screen_width;float ref_shiftY=chromatic_aberration*pow(radius,radialIntensity)*directionOfEffect.y/screen_height;vec2 ref_coords_r=vec2(vUV.x+ref_indices.r*ref_shiftX,vUV.y+ref_indices.r*ref_shiftY*0.5);vec2 ref_coords_g=vec2(vUV.x+ref_indices.g*ref_shiftX,vUV.y+ref_indices.g*ref_shiftY*0.5);vec2 ref_coords_b=vec2(vUV.x+ref_indices.b*ref_shiftX,vUV.y+ref_indices.b*ref_shiftY*0.5);original.r=texture2D(textureSampler,ref_coords_r).r;original.g=texture2D(textureSampler,ref_coords_g).g;original.b=texture2D(textureSampler,ref_coords_b).b;original.a=clamp(texture2D(textureSampler,ref_coords_r).a+texture2D(textureSampler,ref_coords_g).a+texture2D(textureSampler,ref_coords_b).a,0.,1.);gl_FragColor=original;}`; Le.ShadersStore[H0e] = g0e; class Pg extends kr { /** * Gets a string identifying the name of the class * @returns "ChromaticAberrationPostProcess" string */ getClassName() { return "ChromaticAberrationPostProcess"; } /** * Creates a new instance ChromaticAberrationPostProcess * @param name The name of the effect. * @param screenWidth The width of the screen to apply the effect on. * @param screenHeight The height of the screen to apply the effect on. * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) * @param textureType Type of textures used when performing the post process. (default: 0) * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false) */ constructor(e, t, r, n, i, s, a, f, o = 0, d = !1) { super(e, "chromaticAberration", ["chromatic_aberration", "screen_width", "screen_height", "direction", "radialIntensity", "centerPosition"], [], n, i, s, a, f, null, o, void 0, null, d), this.aberrationAmount = 30, this.radialIntensity = 0, this.direction = new at(0.707, 0.707), this.centerPosition = new at(0.5, 0.5), this.screenWidth = t, this.screenHeight = r, this.onApplyObservable.add((v) => { v.setFloat("chromatic_aberration", this.aberrationAmount), v.setFloat("screen_width", t), v.setFloat("screen_height", r), v.setFloat("radialIntensity", this.radialIntensity), v.setFloat2("direction", this.direction.x, this.direction.y), v.setFloat2("centerPosition", this.centerPosition.x, this.centerPosition.y); }); } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new Pg(e.name, e.screenWidth, e.screenHeight, e.options, t, e.renderTargetSamplingMode, r.getEngine(), e.reusable, e.textureType, !1), e, r, n); } } C([ M() ], Pg.prototype, "aberrationAmount", void 0); C([ M() ], Pg.prototype, "radialIntensity", void 0); C([ M() ], Pg.prototype, "direction", void 0); C([ M() ], Pg.prototype, "centerPosition", void 0); C([ M() ], Pg.prototype, "screenWidth", void 0); C([ M() ], Pg.prototype, "screenHeight", void 0); Ue("BABYLON.ChromaticAberrationPostProcess", Pg); const X0e = "circleOfConfusionPixelShader", T0e = `uniform sampler2D depthSampler;varying vec2 vUV;uniform vec2 cameraMinMaxZ;uniform float focusDistance;uniform float cocPrecalculation; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {float depth=texture2D(depthSampler,vUV).r; #define CUSTOM_COC_DEPTH float pixelDistance=(cameraMinMaxZ.x+cameraMinMaxZ.y*depth)*1000.0; #define CUSTOM_COC_PIXELDISTANCE float coc=abs(cocPrecalculation*((focusDistance-pixelDistance)/pixelDistance));coc=clamp(coc,0.0,1.0);gl_FragColor=vec4(coc,coc,coc,1.0);} `; Le.ShadersStore[X0e] = T0e; class Om extends kr { /** * Gets a string identifying the name of the class * @returns "CircleOfConfusionPostProcess" string */ getClassName() { return "CircleOfConfusionPostProcess"; } /** * Creates a new instance CircleOfConfusionPostProcess * @param name The name of the effect. * @param depthTexture The depth texture of the scene to compute the circle of confusion. This must be set in order for this to function but may be set after initialization if needed. * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) * @param textureType Type of textures used when performing the post process. (default: 0) * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false) */ constructor(e, t, r, n, i, s, a, f = 0, o = !1) { super(e, "circleOfConfusion", ["cameraMinMaxZ", "focusDistance", "cocPrecalculation"], ["depthSampler"], r, n, i, s, a, null, f, void 0, null, o), this.lensSize = 50, this.fStop = 1.4, this.focusDistance = 2e3, this.focalLength = 50, this._depthTexture = null, this._depthTexture = t, this.onApplyObservable.add((d) => { if (!this._depthTexture) { Se.Warn("No depth texture set on CircleOfConfusionPostProcess"); return; } d.setTexture("depthSampler", this._depthTexture); const u = this.lensSize / this.fStop * this.focalLength / (this.focusDistance - this.focalLength); d.setFloat("focusDistance", this.focusDistance), d.setFloat("cocPrecalculation", u); const l = this._depthTexture.activeCamera; d.setFloat2("cameraMinMaxZ", l.minZ, l.maxZ - l.minZ); }); } /** * Depth texture to be used to compute the circle of confusion. This must be set here or in the constructor in order for the post process to function. */ set depthTexture(e) { this._depthTexture = e; } } C([ M() ], Om.prototype, "lensSize", void 0); C([ M() ], Om.prototype, "fStop", void 0); C([ M() ], Om.prototype, "focusDistance", void 0); C([ M() ], Om.prototype, "focalLength", void 0); Ue("BABYLON.CircleOfConfusionPostProcess", Om); const q0e = "colorCorrectionPixelShader", b0e = `uniform sampler2D textureSampler; uniform sampler2D colorTable; varying vec2 vUV;const float SLICE_COUNT=16.0; vec4 sampleAs3DTexture(sampler2D textureSampler,vec3 uv,float width) {float sliceSize=1.0/width; float slicePixelSize=sliceSize/width; float sliceInnerSize=slicePixelSize*(width-1.0); float zSlice0=min(floor(uv.z*width),width-1.0);float zSlice1=min(zSlice0+1.0,width-1.0);float xOffset=slicePixelSize*0.5+uv.x*sliceInnerSize;float s0=xOffset+(zSlice0*sliceSize);float s1=xOffset+(zSlice1*sliceSize);vec4 slice0Color=texture2D(textureSampler,vec2(s0,uv.y));vec4 slice1Color=texture2D(textureSampler,vec2(s1,uv.y));float zOffset=mod(uv.z*width,1.0);vec4 result=mix(slice0Color,slice1Color,zOffset);return result;} #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {vec4 screen_color=texture2D(textureSampler,vUV);gl_FragColor=sampleAs3DTexture(colorTable,screen_color.rgb,SLICE_COUNT);}`; Le.ShadersStore[q0e] = b0e; class dV extends kr { /** * Gets a string identifying the name of the class * @returns "ColorCorrectionPostProcess" string */ getClassName() { return "ColorCorrectionPostProcess"; } constructor(e, t, r, n, i, s, a) { super(e, "colorCorrection", null, ["colorTable"], r, n, i, s, a); const f = (n == null ? void 0 : n.getScene()) || null; this._colorTableTexture = new We(t, f, !0, !1, We.TRILINEAR_SAMPLINGMODE), this._colorTableTexture.anisotropicFilteringLevel = 1, this._colorTableTexture.wrapU = We.CLAMP_ADDRESSMODE, this._colorTableTexture.wrapV = We.CLAMP_ADDRESSMODE, this.colorTableUrl = t, this.onApply = (o) => { o.setTexture("colorTable", this._colorTableTexture); }; } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new dV(e.name, e.colorTableUrl, e.options, t, e.renderTargetSamplingMode, r.getEngine(), e.reusable), e, r, n); } } C([ M() ], dV.prototype, "colorTableUrl", void 0); Ue("BABYLON.ColorCorrectionPostProcess", dV); const x0e = "convolutionPixelShader", D0e = `varying vec2 vUV;uniform sampler2D textureSampler;uniform vec2 screenSize;uniform float kernel[9]; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {vec2 onePixel=vec2(1.0,1.0)/screenSize;vec4 colorSum = texture2D(textureSampler,vUV+onePixel*vec2(-1,-1))*kernel[0] + texture2D(textureSampler,vUV+onePixel*vec2(0,-1))*kernel[1] + texture2D(textureSampler,vUV+onePixel*vec2(1,-1))*kernel[2] + texture2D(textureSampler,vUV+onePixel*vec2(-1,0))*kernel[3] + texture2D(textureSampler,vUV+onePixel*vec2(0,0))*kernel[4] + texture2D(textureSampler,vUV+onePixel*vec2(1,0))*kernel[5] + texture2D(textureSampler,vUV+onePixel*vec2(-1,1))*kernel[6] + texture2D(textureSampler,vUV+onePixel*vec2(0,1))*kernel[7] + texture2D(textureSampler,vUV+onePixel*vec2(1,1))*kernel[8];float kernelWeight = kernel[0] + kernel[1] + kernel[2] + kernel[3] + kernel[4] + kernel[5] + kernel[6] + kernel[7] + kernel[8];if (kernelWeight<=0.0) {kernelWeight=1.0;} gl_FragColor=vec4((colorSum/kernelWeight).rgb,1);}`; Le.ShadersStore[x0e] = D0e; class cg extends kr { /** * Gets a string identifying the name of the class * @returns "ConvolutionPostProcess" string */ getClassName() { return "ConvolutionPostProcess"; } /** * Creates a new instance ConvolutionPostProcess * @param name The name of the effect. * @param kernel Array of 9 values corresponding to the 3x3 kernel to be applied * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) * @param textureType Type of textures used when performing the post process. (default: 0) */ constructor(e, t, r, n, i, s, a, f = 0) { super(e, "convolution", ["kernel", "screenSize"], null, r, n, i, s, a, null, f), this.kernel = t, this.onApply = (o) => { o.setFloat2("screenSize", this.width, this.height), o.setArray("kernel", this.kernel); }; } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new cg(e.name, e.kernel, e.options, t, e.renderTargetSamplingMode, r.getEngine(), e.reusable, e.textureType), e, r, n); } } cg.EdgeDetect0Kernel = [1, 0, -1, 0, 0, 0, -1, 0, 1]; cg.EdgeDetect1Kernel = [0, 1, 0, 1, -4, 1, 0, 1, 0]; cg.EdgeDetect2Kernel = [-1, -1, -1, -1, 8, -1, -1, -1, -1]; cg.SharpenKernel = [0, -1, 0, -1, 5, -1, 0, -1, 0]; cg.EmbossKernel = [-2, -1, 0, -1, 1, 1, 0, 1, 2]; cg.GaussianKernel = [0, 1, 0, 1, 1, 1, 0, 1, 0]; C([ M() ], cg.prototype, "kernel", void 0); Ue("BABYLON.ConvolutionPostProcess", cg); class sR extends c9 { /** * Gets a string identifying the name of the class * @returns "DepthOfFieldBlurPostProcess" string */ getClassName() { return "DepthOfFieldBlurPostProcess"; } /** * Creates a new instance DepthOfFieldBlurPostProcess * @param name The name of the effect. * @param scene The scene the effect belongs to. * @param direction The direction the blur should be applied. * @param kernel The size of the kernel used to blur. * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param circleOfConfusion The circle of confusion + depth map to be used to avoid blurring across edges * @param imageToBlur The image to apply the blur to (default: Current rendered frame) * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) * @param textureType Type of textures used when performing the post process. (default: 0) * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false) * @param textureFormat Format of textures used when performing the post process. (default: TEXTUREFORMAT_RGBA) */ constructor(e, t, r, n, i, s, a, f = null, o = We.BILINEAR_SAMPLINGMODE, d, v, u = 0, l = !1, P = 5) { super( e, r, n, i, s, // eslint-disable-next-line @typescript-eslint/no-unused-vars o = 2, d, v, u, `#define DOF 1 `, l, P ), this.direction = r, this.externalTextureSamplerBinding = !!f, this.onApplyObservable.add((p) => { f != null && p.setTextureFromPostProcess("textureSampler", f), p.setTextureFromPostProcessOutput("circleOfConfusionSampler", a); }); } } C([ M() ], sR.prototype, "direction", void 0); Ue("BABYLON.DepthOfFieldBlurPostProcess", sR); const j0e = "depthOfFieldMergePixelShader", w0e = `#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE) #define TEXTUREFUNC(s,c,lod) texture2DLodEXT(s,c,lod) #else #define TEXTUREFUNC(s,c,bias) texture2D(s,c,bias) #endif uniform sampler2D textureSampler;varying vec2 vUV;uniform sampler2D circleOfConfusionSampler;uniform sampler2D blurStep0; #if BLUR_LEVEL>0 uniform sampler2D blurStep1; #endif #if BLUR_LEVEL>1 uniform sampler2D blurStep2; #endif #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {float coc=TEXTUREFUNC(circleOfConfusionSampler,vUV,0.0).r; #if BLUR_LEVEL==0 vec4 original=TEXTUREFUNC(textureSampler,vUV,0.0);vec4 blurred0=TEXTUREFUNC(blurStep0,vUV,0.0);gl_FragColor=mix(original,blurred0,coc); #endif #if BLUR_LEVEL==1 if(coc<0.5){vec4 original=TEXTUREFUNC(textureSampler,vUV,0.0);vec4 blurred1=TEXTUREFUNC(blurStep1,vUV,0.0);gl_FragColor=mix(original,blurred1,coc/0.5);}else{vec4 blurred0=TEXTUREFUNC(blurStep0,vUV,0.0);vec4 blurred1=TEXTUREFUNC(blurStep1,vUV,0.0);gl_FragColor=mix(blurred1,blurred0,(coc-0.5)/0.5);} #endif #if BLUR_LEVEL==2 if(coc<0.33){vec4 original=TEXTUREFUNC(textureSampler,vUV,0.0);vec4 blurred2=TEXTUREFUNC(blurStep2,vUV,0.0);gl_FragColor=mix(original,blurred2,coc/0.33);}else if(coc<0.66){vec4 blurred1=TEXTUREFUNC(blurStep1,vUV,0.0);vec4 blurred2=TEXTUREFUNC(blurStep2,vUV,0.0);gl_FragColor=mix(blurred2,blurred1,(coc-0.33)/0.33);}else{vec4 blurred0=TEXTUREFUNC(blurStep0,vUV,0.0);vec4 blurred1=TEXTUREFUNC(blurStep1,vUV,0.0);gl_FragColor=mix(blurred1,blurred0,(coc-0.66)/0.34);} #endif } `; Le.ShadersStore[j0e] = w0e; class eie extends kr { /** * Gets a string identifying the name of the class * @returns "DepthOfFieldMergePostProcess" string */ getClassName() { return "DepthOfFieldMergePostProcess"; } /** * Creates a new instance of DepthOfFieldMergePostProcess * @param name The name of the effect. * @param originalFromInput Post process which's input will be used for the merge. * @param circleOfConfusion Circle of confusion post process which's output will be used to blur each pixel. * @param _blurSteps Blur post processes from low to high which will be mixed with the original image. * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) * @param textureType Type of textures used when performing the post process. (default: 0) * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false) */ constructor(e, t, r, n, i, s, a, f, o, d = 0, v = !1) { super(e, "depthOfFieldMerge", [], ["circleOfConfusionSampler", "blurStep0", "blurStep1", "blurStep2"], i, s, a, f, o, null, d, void 0, null, !0), this._blurSteps = n, this.externalTextureSamplerBinding = !0, this.onApplyObservable.add((u) => { u.setTextureFromPostProcess("textureSampler", t), u.setTextureFromPostProcessOutput("circleOfConfusionSampler", r), n.forEach((l, P) => { u.setTextureFromPostProcessOutput("blurStep" + (n.length - P - 1), l); }); }), v || this.updateEffect(); } /** * Updates the effect with the current post process compile time values and recompiles the shader. * @param defines Define statements that should be added at the beginning of the shader. (default: null) * @param uniforms Set of uniform variables that will be passed to the shader. (default: null) * @param samplers Set of Texture2D variables that will be passed to the shader. (default: null) * @param indexParameters The index parameters to be used for babylons include syntax "#include[0..varyingCount]". (default: undefined) See usage in babylon.blurPostProcess.ts and kernelBlur.vertex.fx * @param onCompiled Called when the shader has been compiled. * @param onError Called if there is an error when compiling a shader. */ updateEffect(e = null, t = null, r = null, n, i, s) { e || (e = "", e += "#define BLUR_LEVEL " + (this._blurSteps.length - 1) + ` `), super.updateEffect(e, t, r, n, i, s); } } var am; (function(A) { A[A.Low = 0] = "Low", A[A.Medium = 1] = "Medium", A[A.High = 2] = "High"; })(am || (am = {})); class jF extends gs { /** * The focal the length of the camera used in the effect in scene units/1000 (eg. millimeter) */ set focalLength(e) { this._circleOfConfusion.focalLength = e; } get focalLength() { return this._circleOfConfusion.focalLength; } /** * F-Stop of the effect's camera. The diameter of the resulting aperture can be computed by lensSize/fStop. (default: 1.4) */ set fStop(e) { this._circleOfConfusion.fStop = e; } get fStop() { return this._circleOfConfusion.fStop; } /** * Distance away from the camera to focus on in scene units/1000 (eg. millimeter). (default: 2000) */ set focusDistance(e) { this._circleOfConfusion.focusDistance = e; } get focusDistance() { return this._circleOfConfusion.focusDistance; } /** * Max lens size in scene units/1000 (eg. millimeter). Standard cameras are 50mm. (default: 50) The diameter of the resulting aperture can be computed by lensSize/fStop. */ set lensSize(e) { this._circleOfConfusion.lensSize = e; } get lensSize() { return this._circleOfConfusion.lensSize; } /** * Creates a new instance DepthOfFieldEffect * @param scene The scene the effect belongs to. * @param depthTexture The depth texture of the scene to compute the circle of confusion.This must be set in order for this to function but may be set after initialization if needed. * @param blurLevel * @param pipelineTextureType The type of texture to be used when performing the post processing. * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false) */ constructor(e, t, r = am.Low, n = 0, i = !1) { super(e.getEngine(), "depth of field", () => this._effects, !0), this._effects = []; const s = e.getEngine(), a = s.isWebGPU || s.webGLVersion > 1 ? 6 : 5; this._circleOfConfusion = new Om("circleOfConfusion", t, 1, null, We.BILINEAR_SAMPLINGMODE, s, !1, n, i), this._depthOfFieldBlurY = [], this._depthOfFieldBlurX = []; let f = 1, o = 15; switch (r) { case am.High: { f = 3, o = 51; break; } case am.Medium: { f = 2, o = 31; break; } default: { o = 15, f = 1; break; } } const d = o / Math.pow(2, f - 1); let v = 1; for (let u = 0; u < f; u++) { const l = new sR("vertical blur", e, new at(0, 1), d, v, null, this._circleOfConfusion, u == 0 ? this._circleOfConfusion : null, We.BILINEAR_SAMPLINGMODE, s, !1, n, i, u == 0 ? a : 5); l.autoClear = !1, v = 0.75 / Math.pow(2, u); const P = new sR("horizontal blur", e, new at(1, 0), d, v, null, this._circleOfConfusion, null, We.BILINEAR_SAMPLINGMODE, s, !1, n, i); P.autoClear = !1, this._depthOfFieldBlurY.push(l), this._depthOfFieldBlurX.push(P); } this._effects = [this._circleOfConfusion]; for (let u = 0; u < this._depthOfFieldBlurX.length; u++) this._effects.push(this._depthOfFieldBlurY[u]), this._effects.push(this._depthOfFieldBlurX[u]); this._dofMerge = new eie("dofMerge", this._circleOfConfusion, this._circleOfConfusion, this._depthOfFieldBlurX, v, null, We.BILINEAR_SAMPLINGMODE, s, !1, n, i), this._dofMerge.autoClear = !1, this._effects.push(this._dofMerge); } /** * Get the current class name of the current effect * @returns "DepthOfFieldEffect" */ getClassName() { return "DepthOfFieldEffect"; } /** * Depth texture to be used to compute the circle of confusion. This must be set here or in the constructor in order for the post process to function. */ set depthTexture(e) { this._circleOfConfusion.depthTexture = e; } /** * Disposes each of the internal effects for a given camera. * @param camera The camera to dispose the effect on. */ disposeEffects(e) { for (let t = 0; t < this._effects.length; t++) this._effects[t].dispose(e); } /** * @internal Internal */ _updateEffects() { for (let e = 0; e < this._effects.length; e++) this._effects[e].updateEffect(); } /** * Internal * @returns if all the contained post processes are ready. * @internal */ _isReady() { for (let e = 0; e < this._effects.length; e++) if (!this._effects[e].isReady()) return !1; return !0; } } const m0e = "displayPassPixelShader", B0e = `varying vec2 vUV;uniform sampler2D textureSampler;uniform sampler2D passSampler; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {gl_FragColor=texture2D(passSampler,vUV);}`; Le.ShadersStore[m0e] = B0e; class Sy extends kr { /** * Gets a string identifying the name of the class * @returns "DisplayPassPostProcess" string */ getClassName() { return "DisplayPassPostProcess"; } /** * Creates the DisplayPassPostProcess * @param name The name of the effect. * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) */ constructor(e, t, r, n, i, s) { super(e, "displayPass", ["passSampler"], ["passSampler"], t, r, n, i, s); } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new Sy(e.name, e.options, t, e.renderTargetSamplingMode, r.getEngine(), e.reusable), e, r, n); } } Ue("BABYLON.DisplayPassPostProcess", Sy); const W0e = "filterPixelShader", S0e = `varying vec2 vUV;uniform sampler2D textureSampler;uniform mat4 kernelMatrix; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {vec3 baseColor=texture2D(textureSampler,vUV).rgb;vec3 updatedColor=(kernelMatrix*vec4(baseColor,1.0)).rgb;gl_FragColor=vec4(updatedColor,1.0);}`; Le.ShadersStore[W0e] = S0e; class vV extends kr { /** * Gets a string identifying the name of the class * @returns "FilterPostProcess" string */ getClassName() { return "FilterPostProcess"; } /** * * @param name The name of the effect. * @param kernelMatrix The matrix to be applied to the image * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) */ constructor(e, t, r, n, i, s, a) { super(e, "filter", ["kernelMatrix"], null, r, n, i, s, a), this.kernelMatrix = t, this.onApply = (f) => { f.setMatrix("kernelMatrix", this.kernelMatrix); }; } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new vV(e.name, e.kernelMatrix, e.options, t, e.renderTargetSamplingMode, r.getEngine(), e.reusable), e, r, n); } } C([ qO() ], vV.prototype, "kernelMatrix", void 0); Ue("BABYLON.FilterPostProcess", vV); const U0e = "fxaaPixelShader", I0e = `#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE) #define TEXTUREFUNC(s,c,l) texture2DLodEXT(s,c,l) #else #define TEXTUREFUNC(s,c,b) texture2D(s,c,b) #endif uniform sampler2D textureSampler;uniform vec2 texelSize;varying vec2 vUV;varying vec2 sampleCoordS;varying vec2 sampleCoordE;varying vec2 sampleCoordN;varying vec2 sampleCoordW;varying vec2 sampleCoordNW;varying vec2 sampleCoordSE;varying vec2 sampleCoordNE;varying vec2 sampleCoordSW;const float fxaaQualitySubpix=1.0;const float fxaaQualityEdgeThreshold=0.166;const float fxaaQualityEdgeThresholdMin=0.0833;const vec3 kLumaCoefficients=vec3(0.2126,0.7152,0.0722); #define FxaaLuma(rgba) dot(rgba.rgb,kLumaCoefficients) void main(){vec2 posM;posM.x=vUV.x;posM.y=vUV.y;vec4 rgbyM=TEXTUREFUNC(textureSampler,vUV,0.0);float lumaM=FxaaLuma(rgbyM);float lumaS=FxaaLuma(TEXTUREFUNC(textureSampler,sampleCoordS,0.0));float lumaE=FxaaLuma(TEXTUREFUNC(textureSampler,sampleCoordE,0.0));float lumaN=FxaaLuma(TEXTUREFUNC(textureSampler,sampleCoordN,0.0));float lumaW=FxaaLuma(TEXTUREFUNC(textureSampler,sampleCoordW,0.0));float maxSM=max(lumaS,lumaM);float minSM=min(lumaS,lumaM);float maxESM=max(lumaE,maxSM);float minESM=min(lumaE,minSM);float maxWN=max(lumaN,lumaW);float minWN=min(lumaN,lumaW);float rangeMax=max(maxWN,maxESM);float rangeMin=min(minWN,minESM);float rangeMaxScaled=rangeMax*fxaaQualityEdgeThreshold;float range=rangeMax-rangeMin;float rangeMaxClamped=max(fxaaQualityEdgeThresholdMin,rangeMaxScaled); #ifndef MALI if(range=edgeVert;float subpixA=subpixNSWE*2.0+subpixNWSWNESE;if (!horzSpan) {lumaN=lumaW;} if (!horzSpan) {lumaS=lumaE;} if (horzSpan) {lengthSign=texelSize.y;} float subpixB=(subpixA*(1.0/12.0))-lumaM;float gradientN=lumaN-lumaM;float gradientS=lumaS-lumaM;float lumaNN=lumaN+lumaM;float lumaSS=lumaS+lumaM;bool pairN=abs(gradientN)>=abs(gradientS);float gradient=max(abs(gradientN),abs(gradientS));if (pairN) {lengthSign=-lengthSign;} float subpixC=clamp(abs(subpixB)*subpixRcpRange,0.0,1.0);vec2 posB;posB.x=posM.x;posB.y=posM.y;vec2 offNP;offNP.x=(!horzSpan) ? 0.0 : texelSize.x;offNP.y=(horzSpan) ? 0.0 : texelSize.y;if (!horzSpan) {posB.x+=lengthSign*0.5;} if (horzSpan) {posB.y+=lengthSign*0.5;} vec2 posN;posN.x=posB.x-offNP.x*1.5;posN.y=posB.y-offNP.y*1.5;vec2 posP;posP.x=posB.x+offNP.x*1.5;posP.y=posB.y+offNP.y*1.5;float subpixD=((-2.0)*subpixC)+3.0;float lumaEndN=FxaaLuma(TEXTUREFUNC(textureSampler,posN,0.0));float subpixE=subpixC*subpixC;float lumaEndP=FxaaLuma(TEXTUREFUNC(textureSampler,posP,0.0));if (!pairN) {lumaNN=lumaSS;} float gradientScaled=gradient*1.0/4.0;float lumaMM=lumaM-lumaNN*0.5;float subpixF=subpixD*subpixE;bool lumaMLTZero=lumaMM<0.0;lumaEndN-=lumaNN*0.5;lumaEndP-=lumaNN*0.5;bool doneN=abs(lumaEndN)>=gradientScaled;bool doneP=abs(lumaEndP)>=gradientScaled;if (!doneN) {posN.x-=offNP.x*3.0;} if (!doneN) {posN.y-=offNP.y*3.0;} bool doneNP=(!doneN) || (!doneP);if (!doneP) {posP.x+=offNP.x*3.0;} if (!doneP) {posP.y+=offNP.y*3.0;} if (doneNP) {if (!doneN) lumaEndN=FxaaLuma(TEXTUREFUNC(textureSampler,posN.xy,0.0));if (!doneP) lumaEndP=FxaaLuma(TEXTUREFUNC(textureSampler,posP.xy,0.0));if (!doneN) lumaEndN=lumaEndN-lumaNN*0.5;if (!doneP) lumaEndP=lumaEndP-lumaNN*0.5;doneN=abs(lumaEndN)>=gradientScaled;doneP=abs(lumaEndP)>=gradientScaled;if (!doneN) posN.x-=offNP.x*12.0;if (!doneN) posN.y-=offNP.y*12.0;doneNP=(!doneN) || (!doneP);if (!doneP) posP.x+=offNP.x*12.0;if (!doneP) posP.y+=offNP.y*12.0;} float dstN=posM.x-posN.x;float dstP=posP.x-posM.x;if (!horzSpan) {dstN=posM.y-posN.y;} if (!horzSpan) {dstP=posP.y-posM.y;} bool goodSpanN=(lumaEndN<0.0) != lumaMLTZero;float spanLength=(dstP+dstN);bool goodSpanP=(lumaEndP<0.0) != lumaMLTZero;float spanLengthRcp=1.0/spanLength;bool directionN=dstN { const d = this.texelSize; o.setFloat2("texelSize", d.x, d.y); }); } _getDefines() { const e = this.getEngine(); if (!e) return null; const t = e.getGlInfo(); return t && t.renderer && t.renderer.toLowerCase().indexOf("mali") > -1 ? `#define MALI 1 ` : null; } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new ym(e.name, e.options, t, e.renderTargetSamplingMode, r.getEngine(), e.reusable), e, r, n); } } Ue("BABYLON.FxaaPostProcess", ym); const C0e = "grainPixelShader", O0e = `#include uniform sampler2D textureSampler; uniform float intensity;uniform float animatedSeed;varying vec2 vUV; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {gl_FragColor=texture2D(textureSampler,vUV);vec2 seed=vUV*(animatedSeed);float grain=dither(seed,intensity);float lum=getLuminance(gl_FragColor.rgb);float grainAmount=(cos(-PI+(lum*PI*2.))+1.)/2.;gl_FragColor.rgb+=grain*grainAmount;gl_FragColor.rgb=max(gl_FragColor.rgb,0.0);}`; Le.ShadersStore[C0e] = O0e; class km extends kr { /** * Gets a string identifying the name of the class * @returns "GrainPostProcess" string */ getClassName() { return "GrainPostProcess"; } /** * Creates a new instance of @see GrainPostProcess * @param name The name of the effect. * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) * @param textureType Type of textures used when performing the post process. (default: 0) * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false) */ constructor(e, t, r, n, i, s, a = 0, f = !1) { super(e, "grain", ["intensity", "animatedSeed"], [], t, r, n, i, s, null, a, void 0, null, f), this.intensity = 30, this.animated = !1, this.onApplyObservable.add((o) => { o.setFloat("intensity", this.intensity), o.setFloat("animatedSeed", this.animated ? Math.random() + 1 : 1); }); } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new km(e.name, e.options, t, e.renderTargetSamplingMode, r.getEngine(), e.reusable), e, r, n); } } C([ M() ], km.prototype, "intensity", void 0); C([ M() ], km.prototype, "animated", void 0); Ue("BABYLON.GrainPostProcess", km); const y0e = "highlightsPixelShader", k0e = `varying vec2 vUV;uniform sampler2D textureSampler;const vec3 RGBLuminanceCoefficients=vec3(0.2126,0.7152,0.0722); #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {vec4 tex=texture2D(textureSampler,vUV);vec3 c=tex.rgb;float luma=dot(c.rgb,RGBLuminanceCoefficients);gl_FragColor=vec4(pow(c,vec3(25.0-luma*15.0)),tex.a); }`; Le.ShadersStore[y0e] = k0e; class E0e extends kr { /** * Gets a string identifying the name of the class * @returns "HighlightsPostProcess" string */ getClassName() { return "HighlightsPostProcess"; } /** * Extracts highlights from the image * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/usePostProcesses * @param name The name of the effect. * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) * @param textureType Type of texture for the post process (default: Engine.TEXTURETYPE_UNSIGNED_INT) */ constructor(e, t, r, n, i, s, a = 0) { super(e, "highlights", null, null, t, r, n, i, s, null, a); } } const F0e = "imageProcessingPixelShader", N0e = `varying vec2 vUV;uniform sampler2D textureSampler; #include #include #include #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {vec4 result=texture2D(textureSampler,vUV); #ifdef IMAGEPROCESSING #ifndef FROMLINEARSPACE result.rgb=toLinearSpace(result.rgb); #endif result=applyImageProcessing(result); #else #ifdef FROMLINEARSPACE result=applyImageProcessing(result); #endif #endif gl_FragColor=result;}`; Le.ShadersStore[F0e] = N0e; class Uy extends kr { /** * Gets the image processing configuration used either in this material. */ get imageProcessingConfiguration() { return this._imageProcessingConfiguration; } /** * Sets the Default image processing configuration used either in the this material. * * If sets to null, the scene one is in use. */ set imageProcessingConfiguration(e) { e.applyByPostProcess = !0, this._attachImageProcessingConfiguration(e); } /** * Attaches a new image processing configuration to the PBR Material. * @param configuration * @param doNotBuild */ _attachImageProcessingConfiguration(e, t = !1) { if (e !== this._imageProcessingConfiguration) { if (this._imageProcessingConfiguration && this._imageProcessingObserver && this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver), e) this._imageProcessingConfiguration = e; else { let r = null; const n = this.getEngine(), i = this.getCamera(); if (i) r = i.getScene(); else if (n && n.scenes) { const s = n.scenes; r = s[s.length - 1]; } else r = gr.LastCreatedScene; r ? this._imageProcessingConfiguration = r.imageProcessingConfiguration : this._imageProcessingConfiguration = new Ui(); } this._imageProcessingConfiguration && (this._imageProcessingObserver = this._imageProcessingConfiguration.onUpdateParameters.add(() => { this._updateParameters(); })), t || this._updateParameters(); } } /** * If the post process is supported. */ get isSupported() { const e = this.getEffect(); return !e || e.isSupported; } /** * Gets Color curves setup used in the effect if colorCurvesEnabled is set to true . */ get colorCurves() { return this.imageProcessingConfiguration.colorCurves; } /** * Sets Color curves setup used in the effect if colorCurvesEnabled is set to true . */ set colorCurves(e) { this.imageProcessingConfiguration.colorCurves = e; } /** * Gets whether the color curves effect is enabled. */ get colorCurvesEnabled() { return this.imageProcessingConfiguration.colorCurvesEnabled; } /** * Sets whether the color curves effect is enabled. */ set colorCurvesEnabled(e) { this.imageProcessingConfiguration.colorCurvesEnabled = e; } /** * Gets Color grading LUT texture used in the effect if colorGradingEnabled is set to true. */ get colorGradingTexture() { return this.imageProcessingConfiguration.colorGradingTexture; } /** * Sets Color grading LUT texture used in the effect if colorGradingEnabled is set to true. */ set colorGradingTexture(e) { this.imageProcessingConfiguration.colorGradingTexture = e; } /** * Gets whether the color grading effect is enabled. */ get colorGradingEnabled() { return this.imageProcessingConfiguration.colorGradingEnabled; } /** * Gets whether the color grading effect is enabled. */ set colorGradingEnabled(e) { this.imageProcessingConfiguration.colorGradingEnabled = e; } /** * Gets exposure used in the effect. */ get exposure() { return this.imageProcessingConfiguration.exposure; } /** * Sets exposure used in the effect. */ set exposure(e) { this.imageProcessingConfiguration.exposure = e; } /** * Gets whether tonemapping is enabled or not. */ get toneMappingEnabled() { return this._imageProcessingConfiguration.toneMappingEnabled; } /** * Sets whether tonemapping is enabled or not */ set toneMappingEnabled(e) { this._imageProcessingConfiguration.toneMappingEnabled = e; } /** * Gets the type of tone mapping effect. */ get toneMappingType() { return this._imageProcessingConfiguration.toneMappingType; } /** * Sets the type of tone mapping effect. */ set toneMappingType(e) { this._imageProcessingConfiguration.toneMappingType = e; } /** * Gets contrast used in the effect. */ get contrast() { return this.imageProcessingConfiguration.contrast; } /** * Sets contrast used in the effect. */ set contrast(e) { this.imageProcessingConfiguration.contrast = e; } /** * Gets Vignette stretch size. */ get vignetteStretch() { return this.imageProcessingConfiguration.vignetteStretch; } /** * Sets Vignette stretch size. */ set vignetteStretch(e) { this.imageProcessingConfiguration.vignetteStretch = e; } /** * Gets Vignette center X Offset. * @deprecated use vignetteCenterX instead */ get vignetteCentreX() { return this.imageProcessingConfiguration.vignetteCenterX; } /** * Sets Vignette center X Offset. * @deprecated use vignetteCenterX instead */ set vignetteCentreX(e) { this.imageProcessingConfiguration.vignetteCenterX = e; } /** * Gets Vignette center Y Offset. * @deprecated use vignetteCenterY instead */ get vignetteCentreY() { return this.imageProcessingConfiguration.vignetteCenterY; } /** * Sets Vignette center Y Offset. * @deprecated use vignetteCenterY instead */ set vignetteCentreY(e) { this.imageProcessingConfiguration.vignetteCenterY = e; } /** * Vignette center Y Offset. */ get vignetteCenterY() { return this.imageProcessingConfiguration.vignetteCenterY; } set vignetteCenterY(e) { this.imageProcessingConfiguration.vignetteCenterY = e; } /** * Vignette center X Offset. */ get vignetteCenterX() { return this.imageProcessingConfiguration.vignetteCenterX; } set vignetteCenterX(e) { this.imageProcessingConfiguration.vignetteCenterX = e; } /** * Gets Vignette weight or intensity of the vignette effect. */ get vignetteWeight() { return this.imageProcessingConfiguration.vignetteWeight; } /** * Sets Vignette weight or intensity of the vignette effect. */ set vignetteWeight(e) { this.imageProcessingConfiguration.vignetteWeight = e; } /** * Gets Color of the vignette applied on the screen through the chosen blend mode (vignetteBlendMode) * if vignetteEnabled is set to true. */ get vignetteColor() { return this.imageProcessingConfiguration.vignetteColor; } /** * Sets Color of the vignette applied on the screen through the chosen blend mode (vignetteBlendMode) * if vignetteEnabled is set to true. */ set vignetteColor(e) { this.imageProcessingConfiguration.vignetteColor = e; } /** * Gets Camera field of view used by the Vignette effect. */ get vignetteCameraFov() { return this.imageProcessingConfiguration.vignetteCameraFov; } /** * Sets Camera field of view used by the Vignette effect. */ set vignetteCameraFov(e) { this.imageProcessingConfiguration.vignetteCameraFov = e; } /** * Gets the vignette blend mode allowing different kind of effect. */ get vignetteBlendMode() { return this.imageProcessingConfiguration.vignetteBlendMode; } /** * Sets the vignette blend mode allowing different kind of effect. */ set vignetteBlendMode(e) { this.imageProcessingConfiguration.vignetteBlendMode = e; } /** * Gets whether the vignette effect is enabled. */ get vignetteEnabled() { return this.imageProcessingConfiguration.vignetteEnabled; } /** * Sets whether the vignette effect is enabled. */ set vignetteEnabled(e) { this.imageProcessingConfiguration.vignetteEnabled = e; } /** * Gets intensity of the dithering effect. */ get ditheringIntensity() { return this.imageProcessingConfiguration.ditheringIntensity; } /** * Sets intensity of the dithering effect. */ set ditheringIntensity(e) { this.imageProcessingConfiguration.ditheringIntensity = e; } /** * Gets whether the dithering effect is enabled. */ get ditheringEnabled() { return this.imageProcessingConfiguration.ditheringEnabled; } /** * Sets whether the dithering effect is enabled. */ set ditheringEnabled(e) { this.imageProcessingConfiguration.ditheringEnabled = e; } /** * Gets whether the input of the processing is in Gamma or Linear Space. */ get fromLinearSpace() { return this._fromLinearSpace; } /** * Sets whether the input of the processing is in Gamma or Linear Space. */ set fromLinearSpace(e) { this._fromLinearSpace !== e && (this._fromLinearSpace = e, this._updateParameters()); } constructor(e, t, r = null, n, i, s, a = 0, f) { super(e, "imageProcessing", [], [], t, r, n, i, s, null, a, "postprocess", null, !0), this._fromLinearSpace = !0, this._defines = { IMAGEPROCESSING: !1, VIGNETTE: !1, VIGNETTEBLENDMODEMULTIPLY: !1, VIGNETTEBLENDMODEOPAQUE: !1, TONEMAPPING: !1, TONEMAPPING_ACES: !1, CONTRAST: !1, COLORCURVES: !1, COLORGRADING: !1, COLORGRADING3D: !1, FROMLINEARSPACE: !1, SAMPLER3DGREENDEPTH: !1, SAMPLER3DBGRMAP: !1, DITHER: !1, IMAGEPROCESSINGPOSTPROCESS: !1, EXPOSURE: !1, SKIPFINALCOLORCLAMP: !1 }, f ? (f.applyByPostProcess = !0, this._attachImageProcessingConfiguration(f, !0), this._updateParameters()) : (this._attachImageProcessingConfiguration(null, !0), this.imageProcessingConfiguration.applyByPostProcess = !0), this.onApply = (o) => { this.imageProcessingConfiguration.bind(o, this.aspectRatio); }; } /** * "ImageProcessingPostProcess" * @returns "ImageProcessingPostProcess" */ getClassName() { return "ImageProcessingPostProcess"; } /** * @internal */ _updateParameters() { this._defines.FROMLINEARSPACE = this._fromLinearSpace, this.imageProcessingConfiguration.prepareDefines(this._defines, !0); let e = ""; for (const n in this._defines) this._defines[n] && (e += `#define ${n}; `); const t = ["textureSampler"], r = ["scale"]; Ui && (Ui.PrepareSamplers(t, this._defines), Ui.PrepareUniforms(r, this._defines)), this.updateEffect(e, r, t); } dispose(e) { super.dispose(e), this._imageProcessingConfiguration && this._imageProcessingObserver && this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver), this._imageProcessingConfiguration && (this.imageProcessingConfiguration.applyByPostProcess = !1); } } C([ M() ], Uy.prototype, "_fromLinearSpace", void 0); const Q0e = "mrtFragmentDeclaration", Y0e = `#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE) layout(location=0) out vec4 glFragData[{X}]; #endif `; Le.IncludesShadersStore[Q0e] = Y0e; const M0e = "geometryPixelShader", L0e = `#extension GL_EXT_draw_buffers : require #if defined(BUMP) || !defined(NORMAL) #extension GL_OES_standard_derivatives : enable #endif precision highp float; #ifdef BUMP varying mat4 vWorldView;varying vec3 vNormalW; #else varying vec3 vNormalV; #endif varying vec4 vViewPos; #if defined(POSITION) || defined(BUMP) varying vec3 vPositionW; #endif #ifdef VELOCITY varying vec4 vCurrentPosition;varying vec4 vPreviousPosition; #endif #ifdef NEED_UV varying vec2 vUV; #endif #ifdef BUMP uniform vec3 vBumpInfos;uniform vec2 vTangentSpaceParams; #endif #if defined(REFLECTIVITY) #if defined(ORMTEXTURE) || defined(SPECULARGLOSSINESSTEXTURE) || defined(REFLECTIVITYTEXTURE) uniform sampler2D reflectivitySampler;varying vec2 vReflectivityUV; #endif #ifdef ALBEDOTEXTURE varying vec2 vAlbedoUV;uniform sampler2D albedoSampler; #endif #ifdef REFLECTIVITYCOLOR uniform vec3 reflectivityColor; #endif #ifdef ALBEDOCOLOR uniform vec3 albedoColor; #endif #ifdef METALLIC uniform float metallic; #endif #if defined(ROUGHNESS) || defined(GLOSSINESS) uniform float glossiness; #endif #endif #if defined(ALPHATEST) && defined(NEED_UV) uniform sampler2D diffuseSampler; #endif #include #include[RENDER_TARGET_COUNT] #include #include #include void main() { #include #ifdef ALPHATEST if (texture2D(diffuseSampler,vUV).a<0.4) discard; #endif vec3 normalOutput; #ifdef BUMP vec3 normalW=normalize(vNormalW); #include #ifdef NORMAL_WORLDSPACE normalOutput=normalW; #else normalOutput=normalize(vec3(vWorldView*vec4(normalW,0.0))); #endif #else normalOutput=normalize(vNormalV); #endif #ifdef PREPASS #ifdef PREPASS_DEPTH gl_FragData[DEPTH_INDEX]=vec4(vViewPos.z/vViewPos.w,0.0,0.0,1.0); #endif #ifdef PREPASS_NORMAL gl_FragData[NORMAL_INDEX]=vec4(normalOutput,1.0); #endif #else gl_FragData[0]=vec4(vViewPos.z/vViewPos.w,0.0,0.0,1.0);gl_FragData[1]=vec4(normalOutput,1.0); #endif #ifdef POSITION gl_FragData[POSITION_INDEX]=vec4(vPositionW,1.0); #endif #ifdef VELOCITY vec2 a=(vCurrentPosition.xy/vCurrentPosition.w)*0.5+0.5;vec2 b=(vPreviousPosition.xy/vPreviousPosition.w)*0.5+0.5;vec2 velocity=abs(a-b);velocity=vec2(pow(velocity.x,1.0/3.0),pow(velocity.y,1.0/3.0))*sign(a-b)*0.5+0.5;gl_FragData[VELOCITY_INDEX]=vec4(velocity,0.0,1.0); #endif #ifdef REFLECTIVITY vec4 reflectivity=vec4(0.0,0.0,0.0,1.0); #ifdef METALLICWORKFLOW float metal=1.0;float roughness=1.0; #ifdef ORMTEXTURE metal*=texture2D(reflectivitySampler,vReflectivityUV).b;roughness*=texture2D(reflectivitySampler,vReflectivityUV).g; #endif #ifdef METALLIC metal*=metallic; #endif #ifdef ROUGHNESS roughness*=(1.0-glossiness); #endif reflectivity.a-=roughness;vec3 color=vec3(1.0); #ifdef ALBEDOTEXTURE color=texture2D(albedoSampler,vAlbedoUV).rgb; #ifdef GAMMAALBEDO color=toLinearSpace(color); #endif #endif #ifdef ALBEDOCOLOR color*=albedoColor.xyz; #endif reflectivity.rgb=mix(vec3(0.04),color,metal); #else #if defined(SPECULARGLOSSINESSTEXTURE) || defined(REFLECTIVITYTEXTURE) reflectivity=texture2D(reflectivitySampler,vReflectivityUV); #ifdef GAMMAREFLECTIVITYTEXTURE reflectivity.rgb=toLinearSpace(reflectivity.rgb); #endif #else #ifdef REFLECTIVITYCOLOR reflectivity.rgb=toLinearSpace(reflectivityColor.xyz);reflectivity.a=1.0; #endif #endif #ifdef GLOSSINESSS reflectivity.a*=glossiness; #endif #endif gl_FragData[REFLECTIVITY_INDEX]=reflectivity; #endif } `; Le.ShadersStore[M0e] = L0e; const K0e = "geometryVertexDeclaration", J0e = "uniform mat4 viewProjection;uniform mat4 view;"; Le.IncludesShadersStore[K0e] = J0e; const z0e = "geometryUboDeclaration", G0e = `#include `; Le.IncludesShadersStore[z0e] = G0e; const Z0e = "geometryVertexShader", _0e = `precision highp float; #include #include #include #include[0..maxSimultaneousMorphTargets] #include #include<__decl__geometryVertex> #include attribute vec3 position;attribute vec3 normal; #ifdef NEED_UV varying vec2 vUV; #ifdef ALPHATEST uniform mat4 diffuseMatrix; #endif #ifdef BUMP uniform mat4 bumpMatrix;varying vec2 vBumpUV; #endif #ifdef REFLECTIVITY uniform mat4 reflectivityMatrix;uniform mat4 albedoMatrix;varying vec2 vReflectivityUV;varying vec2 vAlbedoUV; #endif #ifdef UV1 attribute vec2 uv; #endif #ifdef UV2 attribute vec2 uv2; #endif #endif #ifdef BUMP varying mat4 vWorldView; #endif #ifdef BUMP varying vec3 vNormalW; #else varying vec3 vNormalV; #endif varying vec4 vViewPos; #if defined(POSITION) || defined(BUMP) varying vec3 vPositionW; #endif #ifdef VELOCITY uniform mat4 previousViewProjection;varying vec4 vCurrentPosition;varying vec4 vPreviousPosition; #endif #define CUSTOM_VERTEX_DEFINITIONS void main(void) {vec3 positionUpdated=position;vec3 normalUpdated=normal; #ifdef UV1 vec2 uvUpdated=uv; #endif #include #include[0..maxSimultaneousMorphTargets] #include #if defined(VELOCITY) && !defined(BONES_VELOCITY_ENABLED) vCurrentPosition=viewProjection*finalWorld*vec4(positionUpdated,1.0);vPreviousPosition=previousViewProjection*finalPreviousWorld*vec4(positionUpdated,1.0); #endif #include #include vec4 worldPos=vec4(finalWorld*vec4(positionUpdated,1.0)); #ifdef BUMP vWorldView=view*finalWorld;vNormalW=normalUpdated; #else #ifdef NORMAL_WORLDSPACE vNormalV=normalize(vec3(finalWorld*vec4(normalUpdated,0.0))); #else vNormalV=normalize(vec3((view*finalWorld)*vec4(normalUpdated,0.0))); #endif #endif vViewPos=view*worldPos; #if defined(VELOCITY) && defined(BONES_VELOCITY_ENABLED) vCurrentPosition=viewProjection*finalWorld*vec4(positionUpdated,1.0); #if NUM_BONE_INFLUENCERS>0 mat4 previousInfluence;previousInfluence=mPreviousBones[int(matricesIndices[0])]*matricesWeights[0]; #if NUM_BONE_INFLUENCERS>1 previousInfluence+=mPreviousBones[int(matricesIndices[1])]*matricesWeights[1]; #endif #if NUM_BONE_INFLUENCERS>2 previousInfluence+=mPreviousBones[int(matricesIndices[2])]*matricesWeights[2]; #endif #if NUM_BONE_INFLUENCERS>3 previousInfluence+=mPreviousBones[int(matricesIndices[3])]*matricesWeights[3]; #endif #if NUM_BONE_INFLUENCERS>4 previousInfluence+=mPreviousBones[int(matricesIndicesExtra[0])]*matricesWeightsExtra[0]; #endif #if NUM_BONE_INFLUENCERS>5 previousInfluence+=mPreviousBones[int(matricesIndicesExtra[1])]*matricesWeightsExtra[1]; #endif #if NUM_BONE_INFLUENCERS>6 previousInfluence+=mPreviousBones[int(matricesIndicesExtra[2])]*matricesWeightsExtra[2]; #endif #if NUM_BONE_INFLUENCERS>7 previousInfluence+=mPreviousBones[int(matricesIndicesExtra[3])]*matricesWeightsExtra[3]; #endif vPreviousPosition=previousViewProjection*finalPreviousWorld*previousInfluence*vec4(positionUpdated,1.0); #else vPreviousPosition=previousViewProjection*finalPreviousWorld*vec4(positionUpdated,1.0); #endif #endif #if defined(POSITION) || defined(BUMP) vPositionW=worldPos.xyz/worldPos.w; #endif gl_Position=viewProjection*finalWorld*vec4(positionUpdated,1.0); #include #ifdef NEED_UV #ifdef UV1 #if defined(ALPHATEST) && defined(ALPHATEST_UV1) vUV=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0)); #else vUV=uv; #endif #ifdef BUMP_UV1 vBumpUV=vec2(bumpMatrix*vec4(uvUpdated,1.0,0.0)); #endif #ifdef REFLECTIVITY_UV1 vReflectivityUV=vec2(reflectivityMatrix*vec4(uvUpdated,1.0,0.0)); #endif #ifdef ALBEDO_UV1 vAlbedoUV=vec2(albedoMatrix*vec4(uvUpdated,1.0,0.0)); #endif #endif #ifdef UV2 #if defined(ALPHATEST) && defined(ALPHATEST_UV2) vUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0)); #else vUV=uv2; #endif #ifdef BUMP_UV2 vBumpUV=vec2(bumpMatrix*vec4(uv2,1.0,0.0)); #endif #ifdef REFLECTIVITY_UV2 vReflectivityUV=vec2(reflectivityMatrix*vec4(uv2,1.0,0.0)); #endif #ifdef ALBEDO_UV2 vAlbedoUV=vec2(albedoMatrix*vec4(uv2,1.0,0.0)); #endif #endif #endif #include } `; Le.ShadersStore[Z0e] = _0e; const tie = [ "world", "mBones", "viewProjection", "diffuseMatrix", "view", "previousWorld", "previousViewProjection", "mPreviousBones", "bumpMatrix", "reflectivityMatrix", "albedoMatrix", "reflectivityColor", "albedoColor", "metallic", "glossiness", "vTangentSpaceParams", "vBumpInfos", "morphTargetInfluences", "morphTargetTextureInfo", "morphTargetTextureIndices", "boneTextureWidth" ]; Mf(tie); class po { /** * @internal * Sets up internal structures to share outputs with PrePassRenderer * This method should only be called by the PrePassRenderer itself */ _linkPrePassRenderer(e) { this._linkedWithPrePass = !0, this._prePassRenderer = e, this._multiRenderTarget && (this._multiRenderTarget.onClearObservable.clear(), this._multiRenderTarget.onClearObservable.add(() => { })); } /** * @internal * Separates internal structures from PrePassRenderer so the geometry buffer can now operate by itself. * This method should only be called by the PrePassRenderer itself */ _unlinkPrePassRenderer() { this._linkedWithPrePass = !1, this._createRenderTargets(); } /** * @internal * Resets the geometry buffer layout */ _resetLayout() { this._enablePosition = !1, this._enableReflectivity = !1, this._enableVelocity = !1, this._attachmentsFromPrePass = []; } /** * @internal * Replaces a texture in the geometry buffer renderer * Useful when linking textures of the prepass renderer */ _forceTextureType(e, t) { e === po.POSITION_TEXTURE_TYPE ? (this._positionIndex = t, this._enablePosition = !0) : e === po.VELOCITY_TEXTURE_TYPE ? (this._velocityIndex = t, this._enableVelocity = !0) : e === po.REFLECTIVITY_TEXTURE_TYPE ? (this._reflectivityIndex = t, this._enableReflectivity = !0) : e === po.DEPTH_TEXTURE_TYPE ? this._depthIndex = t : e === po.NORMAL_TEXTURE_TYPE && (this._normalIndex = t); } /** * @internal * Sets texture attachments * Useful when linking textures of the prepass renderer */ _setAttachments(e) { this._attachmentsFromPrePass = e; } /** * @internal * Replaces the first texture which is hard coded as a depth texture in the geometry buffer * Useful when linking textures of the prepass renderer */ _linkInternalTexture(e) { this._multiRenderTarget.setInternalTexture(e, 0, !1); } /** * Gets the render list (meshes to be rendered) used in the G buffer. */ get renderList() { return this._multiRenderTarget.renderList; } /** * Set the render list (meshes to be rendered) used in the G buffer. */ set renderList(e) { this._multiRenderTarget.renderList = e; } /** * Gets whether or not G buffer are supported by the running hardware. * This requires draw buffer supports */ get isSupported() { return this._multiRenderTarget.isSupported; } /** * Returns the index of the given texture type in the G-Buffer textures array * @param textureType The texture type constant. For example GeometryBufferRenderer.POSITION_TEXTURE_INDEX * @returns the index of the given texture type in the G-Buffer textures array */ getTextureIndex(e) { switch (e) { case po.POSITION_TEXTURE_TYPE: return this._positionIndex; case po.VELOCITY_TEXTURE_TYPE: return this._velocityIndex; case po.REFLECTIVITY_TEXTURE_TYPE: return this._reflectivityIndex; case po.DEPTH_TEXTURE_TYPE: return this._linkedWithPrePass ? this._depthIndex : 0; case po.NORMAL_TEXTURE_TYPE: return this._linkedWithPrePass ? this._normalIndex : 1; default: return -1; } } /** * Gets a boolean indicating if objects positions are enabled for the G buffer. */ get enablePosition() { return this._enablePosition; } /** * Sets whether or not objects positions are enabled for the G buffer. */ set enablePosition(e) { this._enablePosition = e, this._linkedWithPrePass || (this.dispose(), this._createRenderTargets()); } /** * Gets a boolean indicating if objects velocities are enabled for the G buffer. */ get enableVelocity() { return this._enableVelocity; } /** * Sets whether or not objects velocities are enabled for the G buffer. */ set enableVelocity(e) { this._enableVelocity = e, e || (this._previousTransformationMatrices = {}), this._linkedWithPrePass || (this.dispose(), this._createRenderTargets()), this._scene.needsPreviousWorldMatrices = e; } /** * Gets a boolean indicating if objects reflectivity are enabled in the G buffer. */ get enableReflectivity() { return this._enableReflectivity; } /** * Sets whether or not objects reflectivity are enabled for the G buffer. * For Metallic-Roughness workflow with ORM texture, we assume that ORM texture is defined according to the default layout: * pbr.useRoughnessFromMetallicTextureAlpha = false; * pbr.useRoughnessFromMetallicTextureGreen = true; * pbr.useMetallnessFromMetallicTextureBlue = true; */ set enableReflectivity(e) { this._enableReflectivity = e, this._linkedWithPrePass || (this.dispose(), this._createRenderTargets()); } /** * Gets the scene associated with the buffer. */ get scene() { return this._scene; } /** * Gets the ratio used by the buffer during its creation. * How big is the buffer related to the main canvas. */ get ratio() { return typeof this._ratioOrDimensions == "object" ? 1 : this._ratioOrDimensions; } /** * Creates a new G Buffer for the scene * @param scene The scene the buffer belongs to * @param ratioOrDimensions How big is the buffer related to the main canvas (default: 1). You can also directly pass a width and height for the generated textures @since * @param depthFormat Format of the depth texture (default: 15) */ constructor(e, t = 1, r = 15) { this._previousTransformationMatrices = {}, this._previousBonesTransformationMatrices = {}, this.excludedSkinnedMeshesFromVelocity = [], this.renderTransparentMeshes = !0, this.generateNormalsInWorldSpace = !1, this._resizeObserver = null, this._enablePosition = !1, this._enableVelocity = !1, this._enableReflectivity = !1, this._clearColor = new xt(0, 0, 0, 0), this._clearDepthColor = new xt(1e8, 0, 0, 1), this._positionIndex = -1, this._velocityIndex = -1, this._reflectivityIndex = -1, this._depthIndex = -1, this._normalIndex = -1, this._linkedWithPrePass = !1, this.useSpecificClearForDepthTexture = !1, this._scene = e, this._ratioOrDimensions = t, this._useUbo = e.getEngine().supportsUniformBuffers, this._depthFormat = r, po._SceneComponentInitialization(this._scene), this._createRenderTargets(); } /** * Checks whether everything is ready to render a submesh to the G buffer. * @param subMesh the submesh to check readiness for * @param useInstances is the mesh drawn using instance or not * @returns true if ready otherwise false */ isReady(e, t) { const r = e.getMaterial(); if (r && r.disableDepthWrite) return !1; const n = [], i = [J.PositionKind, J.NormalKind], s = e.getMesh(); if (r) { let l = !1; if (r.needAlphaTesting() && r.getAlphaTestTexture() && (n.push("#define ALPHATEST"), n.push(`#define ALPHATEST_UV${r.getAlphaTestTexture().coordinatesIndex + 1}`), l = !0), r.bumpTexture && Dt.BumpTextureEnabled && (n.push("#define BUMP"), n.push(`#define BUMP_UV${r.bumpTexture.coordinatesIndex + 1}`), l = !0), this._enableReflectivity) { let P = !1; r.getClassName() === "PBRMetallicRoughnessMaterial" ? (r.metallicRoughnessTexture !== null && (n.push("#define ORMTEXTURE"), n.push(`#define REFLECTIVITY_UV${r.metallicRoughnessTexture.coordinatesIndex + 1}`), n.push("#define METALLICWORKFLOW"), l = !0, P = !0), r.metallic !== null && (n.push("#define METALLIC"), n.push("#define METALLICWORKFLOW"), P = !0), r.roughness !== null && (n.push("#define ROUGHNESS"), n.push("#define METALLICWORKFLOW"), P = !0), P && (r.baseTexture !== null && (n.push("#define ALBEDOTEXTURE"), n.push(`#define ALBEDO_UV${r.baseTexture.coordinatesIndex + 1}`), r.baseTexture.gammaSpace && n.push("#define GAMMAALBEDO"), l = !0), r.baseColor !== null && n.push("#define ALBEDOCOLOR"))) : r.getClassName() === "PBRSpecularGlossinessMaterial" ? (r.specularGlossinessTexture !== null ? (n.push("#define SPECULARGLOSSINESSTEXTURE"), n.push(`#define REFLECTIVITY_UV${r.specularGlossinessTexture.coordinatesIndex + 1}`), l = !0, r.specularGlossinessTexture.gammaSpace && n.push("#define GAMMAREFLECTIVITYTEXTURE")) : r.specularColor !== null && n.push("#define REFLECTIVITYCOLOR"), r.glossiness !== null && n.push("#define GLOSSINESS")) : r.getClassName() === "PBRMaterial" ? (r.metallicTexture !== null && (n.push("#define ORMTEXTURE"), n.push(`#define REFLECTIVITY_UV${r.metallicTexture.coordinatesIndex + 1}`), n.push("#define METALLICWORKFLOW"), l = !0, P = !0), r.metallic !== null && (n.push("#define METALLIC"), n.push("#define METALLICWORKFLOW"), P = !0), r.roughness !== null && (n.push("#define ROUGHNESS"), n.push("#define METALLICWORKFLOW"), P = !0), P ? (r.albedoTexture !== null && (n.push("#define ALBEDOTEXTURE"), n.push(`#define ALBEDO_UV${r.albedoTexture.coordinatesIndex + 1}`), r.albedoTexture.gammaSpace && n.push("#define GAMMAALBEDO"), l = !0), r.albedoColor !== null && n.push("#define ALBEDOCOLOR")) : (r.reflectivityTexture !== null ? (n.push("#define SPECULARGLOSSINESSTEXTURE"), n.push(`#define REFLECTIVITY_UV${r.reflectivityTexture.coordinatesIndex + 1}`), r.reflectivityTexture.gammaSpace && n.push("#define GAMMAREFLECTIVITYTEXTURE"), l = !0) : r.reflectivityColor !== null && n.push("#define REFLECTIVITYCOLOR"), r.microSurface !== null && n.push("#define GLOSSINESS"))) : r.getClassName() === "StandardMaterial" && (r.specularTexture !== null && (n.push("#define REFLECTIVITYTEXTURE"), n.push(`#define REFLECTIVITY_UV${r.specularTexture.coordinatesIndex + 1}`), r.specularTexture.gammaSpace && n.push("#define GAMMAREFLECTIVITYTEXTURE"), l = !0), r.specularColor !== null && n.push("#define REFLECTIVITYCOLOR")); } l && (n.push("#define NEED_UV"), s.isVerticesDataPresent(J.UVKind) && (i.push(J.UVKind), n.push("#define UV1")), s.isVerticesDataPresent(J.UV2Kind) && (i.push(J.UV2Kind), n.push("#define UV2"))); } this._linkedWithPrePass && (n.push("#define PREPASS"), this._depthIndex !== -1 && (n.push("#define DEPTH_INDEX " + this._depthIndex), n.push("#define PREPASS_DEPTH")), this._normalIndex !== -1 && (n.push("#define NORMAL_INDEX " + this._normalIndex), n.push("#define PREPASS_NORMAL"))), this._enablePosition && (n.push("#define POSITION"), n.push("#define POSITION_INDEX " + this._positionIndex)), this._enableVelocity && (n.push("#define VELOCITY"), n.push("#define VELOCITY_INDEX " + this._velocityIndex), this.excludedSkinnedMeshesFromVelocity.indexOf(s) === -1 && n.push("#define BONES_VELOCITY_ENABLED")), this._enableReflectivity && (n.push("#define REFLECTIVITY"), n.push("#define REFLECTIVITY_INDEX " + this._reflectivityIndex)), this.generateNormalsInWorldSpace && n.push("#define NORMAL_WORLDSPACE"), s.useBones && s.computeBonesUsingShaders && s.skeleton ? (i.push(J.MatricesIndicesKind), i.push(J.MatricesWeightsKind), s.numBoneInfluencers > 4 && (i.push(J.MatricesIndicesExtraKind), i.push(J.MatricesWeightsExtraKind)), n.push("#define NUM_BONE_INFLUENCERS " + s.numBoneInfluencers), n.push("#define BONETEXTURE " + s.skeleton.isUsingTextureForMatrices), n.push("#define BonesPerMesh " + (s.skeleton.bones.length + 1))) : (n.push("#define NUM_BONE_INFLUENCERS 0"), n.push("#define BONETEXTURE false"), n.push("#define BonesPerMesh 0")); const a = s.morphTargetManager; let f = 0; a && a.numInfluencers > 0 && (f = a.numInfluencers, n.push("#define MORPHTARGETS"), n.push("#define NUM_MORPH_INFLUENCERS " + f), a.isUsingTextureForTargets && n.push("#define MORPHTARGETS_TEXTURE"), Ye.PrepareAttributesForMorphTargetsInfluencers(i, s, f)), t && (n.push("#define INSTANCES"), Ye.PushAttributesForInstances(i, this._enableVelocity), e.getRenderingMesh().hasThinInstances && n.push("#define THIN_INSTANCES")), this._linkedWithPrePass ? n.push("#define RENDER_TARGET_COUNT " + this._attachmentsFromPrePass.length) : n.push("#define RENDER_TARGET_COUNT " + this._multiRenderTarget.textures.length), xq(r, this._scene, n); const o = this._scene.getEngine(), d = e._getDrawWrapper(void 0, !0), v = d.defines, u = n.join(` `); return v !== u && d.setEffect(o.createEffect("geometry", { attributes: i, uniformsNames: tie, samplers: ["diffuseSampler", "bumpSampler", "reflectivitySampler", "albedoSampler", "morphTargets", "boneSampler"], defines: u, onCompiled: null, fallbacks: null, onError: null, uniformBuffersNames: ["Scene"], indexParameters: { buffersCount: this._multiRenderTarget.textures.length - 1, maxSimultaneousMorphTargets: f } }, o), u), d.effect.isReady(); } /** * Gets the current underlying G Buffer. * @returns the buffer */ getGBuffer() { return this._multiRenderTarget; } /** * Gets the number of samples used to render the buffer (anti aliasing). */ get samples() { return this._multiRenderTarget.samples; } /** * Sets the number of samples used to render the buffer (anti aliasing). */ set samples(e) { this._multiRenderTarget.samples = e; } /** * Disposes the renderer and frees up associated resources. */ dispose() { this._resizeObserver && (this._scene.getEngine().onResizeObservable.remove(this._resizeObserver), this._resizeObserver = null), this.getGBuffer().dispose(); } _assignRenderTargetIndices() { const e = []; let t = 2; return e.push("gBuffer_Depth", "gBuffer_Normal"), this._enablePosition && (this._positionIndex = t, t++, e.push("gBuffer_Position")), this._enableVelocity && (this._velocityIndex = t, t++, e.push("gBuffer_Velocity")), this._enableReflectivity && (this._reflectivityIndex = t, t++, e.push("gBuffer_Reflectivity")), [t, e]; } _createRenderTargets() { const e = this._scene.getEngine(), [t, r] = this._assignRenderTargetIndices(); let n = 0; e._caps.textureFloat && e._caps.textureFloatLinearFiltering ? n = 1 : e._caps.textureHalfFloat && e._caps.textureHalfFloatLinearFiltering && (n = 2); const i = this._ratioOrDimensions.width !== void 0 ? this._ratioOrDimensions : { width: e.getRenderWidth() * this._ratioOrDimensions, height: e.getRenderHeight() * this._ratioOrDimensions }; if (this._multiRenderTarget = new Zx("gBuffer", i, t, this._scene, { generateMipMaps: !1, generateDepthTexture: !0, defaultType: n, depthTextureFormat: this._depthFormat }, r.concat("gBuffer_DepthBuffer")), !this.isSupported) return; this._multiRenderTarget.wrapU = We.CLAMP_ADDRESSMODE, this._multiRenderTarget.wrapV = We.CLAMP_ADDRESSMODE, this._multiRenderTarget.refreshRate = 1, this._multiRenderTarget.renderParticles = !1, this._multiRenderTarget.renderList = null; const s = [!0], a = [!1], f = [!0]; for (let l = 1; l < t; ++l) s.push(!0), f.push(!1), a.push(!0); const o = e.buildTextureLayout(s), d = e.buildTextureLayout(a), v = e.buildTextureLayout(f); this._multiRenderTarget.onClearObservable.add((l) => { l.bindAttachments(this.useSpecificClearForDepthTexture ? d : o), l.clear(this._clearColor, !0, !0, !0), this.useSpecificClearForDepthTexture && (l.bindAttachments(v), l.clear(this._clearDepthColor, !0, !0, !0)), l.bindAttachments(o); }), this._resizeObserver = e.onResizeObservable.add(() => { if (this._multiRenderTarget) { const l = this._ratioOrDimensions.width !== void 0 ? this._ratioOrDimensions : { width: e.getRenderWidth() * this._ratioOrDimensions, height: e.getRenderHeight() * this._ratioOrDimensions }; this._multiRenderTarget.resize(l); } }); const u = (l) => { const P = l.getRenderingMesh(), p = l.getEffectiveMesh(), c = this._scene, H = c.getEngine(), T = l.getMaterial(); if (!T) return; if (p._internalAbstractMeshDataInfo._isActiveIntermediate = !1, this._enableVelocity && !this._previousTransformationMatrices[p.uniqueId] && (this._previousTransformationMatrices[p.uniqueId] = { world: he.Identity(), viewProjection: c.getTransformMatrix() }, P.skeleton)) { const w = P.skeleton.getTransformMatrices(P); this._previousBonesTransformationMatrices[P.uniqueId] = this._copyBonesTransformationMatrices(w, new Float32Array(w.length)); } const q = P._getInstancesRenderList(l._id, !!l.getReplacementMesh()); if (q.mustReturn) return; const b = H.getCaps().instancedArrays && (q.visibleInstances[l._id] !== null || P.hasThinInstances), j = p.getWorldMatrix(); if (this.isReady(l, b)) { const w = l._getDrawWrapper(); if (!w) return; const m = w.effect; H.enableEffect(w), b || P._bind(l, m, T.fillMode), this._useUbo ? (Ye.BindSceneUniformBuffer(m, this._scene.getSceneUniformBuffer()), this._scene.finalizeSceneUbo()) : (m.setMatrix("viewProjection", c.getTransformMatrix()), m.setMatrix("view", c.getViewMatrix())); let I; const N = P._instanceDataStorage; if (!N.isFrozen && (T.backFaceCulling || P.overrideMaterialSideOrientation !== null)) { const k = p._getWorldMatrixDeterminant(); I = P.overrideMaterialSideOrientation, I === null && (I = T.sideOrientation), k < 0 && (I = I === gt.ClockWiseSideOrientation ? gt.CounterClockWiseSideOrientation : gt.ClockWiseSideOrientation); } else I = N.sideOrientation; if (T._preBind(w, I), T.needAlphaTesting()) { const k = T.getAlphaTestTexture(); k && (m.setTexture("diffuseSampler", k), m.setMatrix("diffuseMatrix", k.getTextureMatrix())); } if (T.bumpTexture && c.getEngine().getCaps().standardDerivatives && Dt.BumpTextureEnabled && (m.setFloat3("vBumpInfos", T.bumpTexture.coordinatesIndex, 1 / T.bumpTexture.level, T.parallaxScaleBias), m.setMatrix("bumpMatrix", T.bumpTexture.getTextureMatrix()), m.setTexture("bumpSampler", T.bumpTexture), m.setFloat2("vTangentSpaceParams", T.invertNormalMapX ? -1 : 1, T.invertNormalMapY ? -1 : 1)), this._enableReflectivity && (T.getClassName() === "PBRMetallicRoughnessMaterial" ? (T.metallicRoughnessTexture !== null && (m.setTexture("reflectivitySampler", T.metallicRoughnessTexture), m.setMatrix("reflectivityMatrix", T.metallicRoughnessTexture.getTextureMatrix())), T.metallic !== null && m.setFloat("metallic", T.metallic), T.roughness !== null && m.setFloat("glossiness", 1 - T.roughness), T.baseTexture !== null && (m.setTexture("albedoSampler", T.baseTexture), m.setMatrix("albedoMatrix", T.baseTexture.getTextureMatrix())), T.baseColor !== null && m.setColor3("albedoColor", T.baseColor)) : T.getClassName() === "PBRSpecularGlossinessMaterial" ? (T.specularGlossinessTexture !== null ? (m.setTexture("reflectivitySampler", T.specularGlossinessTexture), m.setMatrix("reflectivityMatrix", T.specularGlossinessTexture.getTextureMatrix())) : T.specularColor !== null && m.setColor3("reflectivityColor", T.specularColor), T.glossiness !== null && m.setFloat("glossiness", T.glossiness)) : T.getClassName() === "PBRMaterial" ? (T.metallicTexture !== null && (m.setTexture("reflectivitySampler", T.metallicTexture), m.setMatrix("reflectivityMatrix", T.metallicTexture.getTextureMatrix())), T.metallic !== null && m.setFloat("metallic", T.metallic), T.roughness !== null && m.setFloat("glossiness", 1 - T.roughness), T.roughness !== null || T.metallic !== null || T.metallicTexture !== null ? (T.albedoTexture !== null && (m.setTexture("albedoSampler", T.albedoTexture), m.setMatrix("albedoMatrix", T.albedoTexture.getTextureMatrix())), T.albedoColor !== null && m.setColor3("albedoColor", T.albedoColor)) : (T.reflectivityTexture !== null ? (m.setTexture("reflectivitySampler", T.reflectivityTexture), m.setMatrix("reflectivityMatrix", T.reflectivityTexture.getTextureMatrix())) : T.reflectivityColor !== null && m.setColor3("reflectivityColor", T.reflectivityColor), T.microSurface !== null && m.setFloat("glossiness", T.microSurface))) : T.getClassName() === "StandardMaterial" && (T.specularTexture !== null && (m.setTexture("reflectivitySampler", T.specularTexture), m.setMatrix("reflectivityMatrix", T.specularTexture.getTextureMatrix())), T.specularColor !== null && m.setColor3("reflectivityColor", T.specularColor))), Df(m, T, this._scene), P.useBones && P.computeBonesUsingShaders && P.skeleton) { const k = P.skeleton; if (k.isUsingTextureForMatrices && m.getUniformIndex("boneTextureWidth") > -1) { const R = k.getTransformMatrixTexture(P); m.setTexture("boneSampler", R), m.setFloat("boneTextureWidth", 4 * (k.bones.length + 1)); } else m.setMatrices("mBones", P.skeleton.getTransformMatrices(P)); this._enableVelocity && m.setMatrices("mPreviousBones", this._previousBonesTransformationMatrices[P.uniqueId]); } Ye.BindMorphTargetParameters(P, m), P.morphTargetManager && P.morphTargetManager.isUsingTextureForTargets && P.morphTargetManager._bind(m), this._enableVelocity && (m.setMatrix("previousWorld", this._previousTransformationMatrices[p.uniqueId].world), m.setMatrix("previousViewProjection", this._previousTransformationMatrices[p.uniqueId].viewProjection)), b && P.hasThinInstances && m.setMatrix("world", j), P._processRendering(p, l, m, T.fillMode, q, b, (k, R) => { k || m.setMatrix("world", R); }); } this._enableVelocity && (this._previousTransformationMatrices[p.uniqueId].world = j.clone(), this._previousTransformationMatrices[p.uniqueId].viewProjection = this._scene.getTransformMatrix().clone(), P.skeleton && this._copyBonesTransformationMatrices(P.skeleton.getTransformMatrices(P), this._previousBonesTransformationMatrices[p.uniqueId])); }; this._multiRenderTarget.customIsReadyFunction = (l, P, p) => { if ((p || P === 0) && l.subMeshes) for (let c = 0; c < l.subMeshes.length; ++c) { const H = l.subMeshes[c], T = H.getMaterial(), q = H.getRenderingMesh(); if (!T) continue; const b = q._getInstancesRenderList(H._id, !!H.getReplacementMesh()), j = e.getCaps().instancedArrays && (b.visibleInstances[H._id] !== null || q.hasThinInstances); if (!this.isReady(H, j)) return !1; } return !0; }, this._multiRenderTarget.customRenderFunction = (l, P, p, c) => { let H; if (this._linkedWithPrePass) { if (!this._prePassRenderer.enabled) return; this._scene.getEngine().bindAttachments(this._attachmentsFromPrePass); } if (c.length) { for (e.setColorWrite(!1), H = 0; H < c.length; H++) u(c.data[H]); e.setColorWrite(!0); } for (H = 0; H < l.length; H++) u(l.data[H]); for (e.setDepthWrite(!1), H = 0; H < P.length; H++) u(P.data[H]); if (this.renderTransparentMeshes) for (H = 0; H < p.length; H++) u(p.data[H]); e.setDepthWrite(!0); }; } // Copies the bones transformation matrices into the target array and returns the target's reference _copyBonesTransformationMatrices(e, t) { for (let r = 0; r < e.length; r++) t[r] = e[r]; return t; } } po.DEPTH_TEXTURE_TYPE = 0; po.NORMAL_TEXTURE_TYPE = 1; po.POSITION_TEXTURE_TYPE = 2; po.VELOCITY_TEXTURE_TYPE = 3; po.REFLECTIVITY_TEXTURE_TYPE = 4; po._SceneComponentInitialization = (A) => { throw qn("GeometryBufferRendererSceneComponent"); }; class $0e { constructor() { this.enabled = !1, this.name = "motionBlur", this.texturesRequired = [2]; } } Object.defineProperty(sr.prototype, "geometryBufferRenderer", { get: function() { return this._geometryBufferRenderer; }, set: function(A) { A && A.isSupported && (this._geometryBufferRenderer = A); }, enumerable: !0, configurable: !0 }); sr.prototype.enableGeometryBufferRenderer = function(A = 1, e = 15) { return this._geometryBufferRenderer ? this._geometryBufferRenderer : (this._geometryBufferRenderer = new po(this, A, e), this._geometryBufferRenderer.isSupported || (this._geometryBufferRenderer = null), this._geometryBufferRenderer); }; sr.prototype.disableGeometryBufferRenderer = function() { this._geometryBufferRenderer && (this._geometryBufferRenderer.dispose(), this._geometryBufferRenderer = null); }; class rie { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_GEOMETRYBUFFERRENDERER, this.scene = e; } /** * Registers the component in a given scene */ register() { this.scene._gatherRenderTargetsStage.registerStep(Ot.STEP_GATHERRENDERTARGETS_GEOMETRYBUFFERRENDERER, this, this._gatherRenderTargets); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { } /** * Disposes the component and the associated resources */ dispose() { } _gatherRenderTargets(e) { this.scene._geometryBufferRenderer && e.push(this.scene._geometryBufferRenderer.getGBuffer()); } } po._SceneComponentInitialization = (A) => { let e = A._getComponent(Ot.NAME_GEOMETRYBUFFERRENDERER); e || (e = new rie(A), A._addComponent(e)); }; const e7e = "motionBlurPixelShader", t7e = `varying vec2 vUV;uniform sampler2D textureSampler;uniform float motionStrength;uniform float motionScale;uniform vec2 screenSize; #ifdef OBJECT_BASED uniform sampler2D velocitySampler; #else uniform sampler2D depthSampler;uniform mat4 inverseViewProjection;uniform mat4 prevViewProjection;uniform mat4 projection; #endif #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #ifdef GEOMETRY_SUPPORTED #ifdef OBJECT_BASED vec2 texelSize=1.0/screenSize;vec4 velocityColor=texture2D(velocitySampler,vUV);velocityColor.rg=velocityColor.rg*2.0-vec2(1.0);vec2 velocity=vec2(pow(velocityColor.r,3.0),pow(velocityColor.g,3.0))*velocityColor.a;velocity*=motionScale*motionStrength;float speed=length(velocity/texelSize);int samplesCount=int(clamp(speed,1.0,SAMPLES));velocity=normalize(velocity)*texelSize;float hlim=float(-samplesCount)*0.5+0.5;vec4 result=texture2D(textureSampler,vUV);for (int i=1; i=samplesCount) break;vec2 offset=vUV+velocity*(hlim+float(i)); #if defined(WEBGPU) result+=texture2DLodEXT(textureSampler,offset,0.0); #else result+=texture2D(textureSampler,offset); #endif } gl_FragColor=result/float(samplesCount);gl_FragColor.a=1.0; #else vec2 texelSize=1.0/screenSize;float depth=texture2D(depthSampler,vUV).r;depth=projection[2].z+projection[3].z/depth; vec4 cpos=vec4(vUV*2.0-1.0,depth,1.0);cpos=inverseViewProjection*cpos;cpos/=cpos.w;vec4 ppos=prevViewProjection*cpos;ppos/=ppos.w;ppos.xy=ppos.xy*0.5+0.5;vec2 velocity=(ppos.xy-vUV)*motionScale*motionStrength;float speed=length(velocity/texelSize);int nSamples=int(clamp(speed,1.0,SAMPLES));vec4 result=texture2D(textureSampler,vUV);for (int i=1; i=nSamples) break;vec2 offset1=vUV+velocity*(float(i)/float(nSamples-1)-0.5); #if defined(WEBGPU) result+=texture2DLodEXT(textureSampler,offset1,0.0); #else result+=texture2D(textureSampler,offset1); #endif } gl_FragColor=result/float(nSamples); #endif #else gl_FragColor=texture2D(textureSampler,vUV); #endif } `; Le.ShadersStore[e7e] = t7e; class BD extends kr { /** * Gets the number of iterations are used for motion blur quality. Default value is equal to 32 */ get motionBlurSamples() { return this._motionBlurSamples; } /** * Sets the number of iterations to be used for motion blur quality */ set motionBlurSamples(e) { this._motionBlurSamples = e, this._updateEffect(); } /** * Gets whether or not the motion blur post-process is in object based mode. */ get isObjectBased() { return this._isObjectBased; } /** * Sets whether or not the motion blur post-process is in object based mode. */ set isObjectBased(e) { this._isObjectBased !== e && (this._isObjectBased = e, this._applyMode()); } get _geometryBufferRenderer() { return this._forceGeometryBuffer ? this._scene.geometryBufferRenderer : null; } get _prePassRenderer() { return this._forceGeometryBuffer ? null : this._scene.prePassRenderer; } /** * Gets a string identifying the name of the class * @returns "MotionBlurPostProcess" string */ getClassName() { return "MotionBlurPostProcess"; } /** * Creates a new instance MotionBlurPostProcess * @param name The name of the effect. * @param scene The scene containing the objects to blur according to their velocity. * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) * @param textureType Type of textures used when performing the post process. (default: 0) * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: true) * @param forceGeometryBuffer If this post process should use geometry buffer instead of prepass (default: false) */ constructor(e, t, r, n, i, s, a, f = 0, o = !1, d = !1) { super(e, "motionBlur", ["motionStrength", "motionScale", "screenSize", "inverseViewProjection", "prevViewProjection", "projection"], ["velocitySampler", "depthSampler"], r, n, i, s, a, `#define GEOMETRY_SUPPORTED #define SAMPLES 64.0 #define OBJECT_BASED`, f, void 0, null, o), this.motionStrength = 1, this._motionBlurSamples = 32, this._isObjectBased = !0, this._forceGeometryBuffer = !1, this._invViewProjection = null, this._previousViewProjection = null, this._forceGeometryBuffer = d, this._forceGeometryBuffer ? (t.enableGeometryBufferRenderer(), this._geometryBufferRenderer && (this._geometryBufferRenderer.enableVelocity = this._isObjectBased)) : (t.enablePrePassRenderer(), this._prePassRenderer && (this._prePassRenderer.markAsDirty(), this._prePassEffectConfiguration = new $0e())), this._applyMode(); } /** * Excludes the given skinned mesh from computing bones velocities. * Computing bones velocities can have a cost and that cost. The cost can be saved by calling this function and by passing the skinned mesh reference to ignore. * @param skinnedMesh The mesh containing the skeleton to ignore when computing the velocity map. */ excludeSkinnedMesh(e) { if (e.skeleton) { let t; if (this._geometryBufferRenderer) t = this._geometryBufferRenderer.excludedSkinnedMeshesFromVelocity; else if (this._prePassRenderer) t = this._prePassRenderer.excludedSkinnedMesh; else return; t.push(e); } } /** * Removes the given skinned mesh from the excluded meshes to integrate bones velocities while rendering the velocity map. * @param skinnedMesh The mesh containing the skeleton that has been ignored previously. * @see excludeSkinnedMesh to exclude a skinned mesh from bones velocity computation. */ removeExcludedSkinnedMesh(e) { if (e.skeleton) { let t; if (this._geometryBufferRenderer) t = this._geometryBufferRenderer.excludedSkinnedMeshesFromVelocity; else if (this._prePassRenderer) t = this._prePassRenderer.excludedSkinnedMesh; else return; const r = t.indexOf(e); r !== -1 && t.splice(r, 1); } } /** * Disposes the post process. * @param camera The camera to dispose the post process on. */ dispose(e) { this._geometryBufferRenderer && (this._geometryBufferRenderer._previousTransformationMatrices = {}, this._geometryBufferRenderer._previousBonesTransformationMatrices = {}, this._geometryBufferRenderer.excludedSkinnedMeshesFromVelocity = []), super.dispose(e); } /** * Called on the mode changed (object based or screen based). */ _applyMode() { if (!this._geometryBufferRenderer && !this._prePassRenderer) return Se.Warn("Multiple Render Target support needed to compute object based motion blur"), this.updateEffect(); this._geometryBufferRenderer && (this._geometryBufferRenderer.enableVelocity = this._isObjectBased), this._updateEffect(), this._invViewProjection = null, this._previousViewProjection = null, this.isObjectBased ? (this._prePassRenderer && this._prePassEffectConfiguration && (this._prePassEffectConfiguration.texturesRequired[0] = 2), this.onApply = (e) => this._onApplyObjectBased(e)) : (this._invViewProjection = he.Identity(), this._previousViewProjection = this._scene.getTransformMatrix().clone(), this._prePassRenderer && this._prePassEffectConfiguration && (this._prePassEffectConfiguration.texturesRequired[0] = 5), this.onApply = (e) => this._onApplyScreenBased(e)); } /** * Called on the effect is applied when the motion blur post-process is in object based mode. * @param effect */ _onApplyObjectBased(e) { if (e.setVector2("screenSize", new at(this.width, this.height)), e.setFloat("motionScale", this._scene.getAnimationRatio()), e.setFloat("motionStrength", this.motionStrength), this._geometryBufferRenderer) { const t = this._geometryBufferRenderer.getTextureIndex(po.VELOCITY_TEXTURE_TYPE); e.setTexture("velocitySampler", this._geometryBufferRenderer.getGBuffer().textures[t]); } else if (this._prePassRenderer) { const t = this._prePassRenderer.getIndex(2); e.setTexture("velocitySampler", this._prePassRenderer.getRenderTarget().textures[t]); } } /** * Called on the effect is applied when the motion blur post-process is in screen based mode. * @param effect */ _onApplyScreenBased(e) { const t = ue.Matrix[0]; if (t.copyFrom(this._scene.getTransformMatrix()), t.invertToRef(this._invViewProjection), e.setMatrix("inverseViewProjection", this._invViewProjection), e.setMatrix("prevViewProjection", this._previousViewProjection), this._previousViewProjection.copyFrom(t), e.setMatrix("projection", this._scene.getProjectionMatrix()), e.setVector2("screenSize", new at(this.width, this.height)), e.setFloat("motionScale", this._scene.getAnimationRatio()), e.setFloat("motionStrength", this.motionStrength), this._geometryBufferRenderer) { const r = this._geometryBufferRenderer.getTextureIndex(po.DEPTH_TEXTURE_TYPE); e.setTexture("depthSampler", this._geometryBufferRenderer.getGBuffer().textures[r]); } else if (this._prePassRenderer) { const r = this._prePassRenderer.getIndex(5); e.setTexture("depthSampler", this._prePassRenderer.getRenderTarget().textures[r]); } } /** * Called on the effect must be updated (changed mode, samples count, etc.). */ _updateEffect() { if (this._geometryBufferRenderer || this._prePassRenderer) { const e = [ "#define GEOMETRY_SUPPORTED", "#define SAMPLES " + this._motionBlurSamples.toFixed(1), this._isObjectBased ? "#define OBJECT_BASED" : "#define SCREEN_BASED" ]; this.updateEffect(e.join(` `)); } } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new BD(e.name, r, e.options, t, e.renderTargetSamplingMode, r.getEngine(), e.reusable, e.textureType, !1), e, r, n); } } C([ M() ], BD.prototype, "motionStrength", void 0); C([ M() ], BD.prototype, "motionBlurSamples", null); C([ M() ], BD.prototype, "isObjectBased", null); Ue("BABYLON.MotionBlurPostProcess", BD); const r7e = "refractionPixelShader", n7e = "varying vec2 vUV;uniform sampler2D textureSampler;uniform sampler2D refractionSampler;uniform vec3 baseColor;uniform float depth;uniform float colorLevel;void main() {float ref=1.0-texture2D(refractionSampler,vUV).r;vec2 uv=vUV-vec2(0.5);vec2 offset=uv*depth*ref;vec3 sourceColor=texture2D(textureSampler,vUV-offset).rgb;gl_FragColor=vec4(sourceColor+sourceColor*ref*colorLevel,1.0);}"; Le.ShadersStore[r7e] = n7e; class WD extends kr { /** * Gets or sets the refraction texture * Please note that you are responsible for disposing the texture if you set it manually */ get refractionTexture() { return this._refTexture; } set refractionTexture(e) { this._refTexture && this._ownRefractionTexture && this._refTexture.dispose(), this._refTexture = e, this._ownRefractionTexture = !1; } /** * Gets a string identifying the name of the class * @returns "RefractionPostProcess" string */ getClassName() { return "RefractionPostProcess"; } /** * Initializes the RefractionPostProcess * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/usePostProcesses#refraction * @param name The name of the effect. * @param refractionTextureUrl Url of the refraction texture to use * @param color the base color of the refraction (used to taint the rendering) * @param depth simulated refraction depth * @param colorLevel the coefficient of the base color (0 to remove base color tainting) * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) */ constructor(e, t, r, n, i, s, a, f, o, d) { super(e, "refraction", ["baseColor", "depth", "colorLevel"], ["refractionSampler"], s, a, f, o, d), this._ownRefractionTexture = !0, this.color = r, this.depth = n, this.colorLevel = i, this.refractionTextureUrl = t, this.onActivateObservable.add((v) => { this._refTexture = this._refTexture || new We(t, v.getScene()); }), this.onApplyObservable.add((v) => { v.setColor3("baseColor", this.color), v.setFloat("depth", this.depth), v.setFloat("colorLevel", this.colorLevel), v.setTexture("refractionSampler", this._refTexture); }); } // Methods /** * Disposes of the post process * @param camera Camera to dispose post process on */ dispose(e) { this._refTexture && this._ownRefractionTexture && (this._refTexture.dispose(), this._refTexture = null), super.dispose(e); } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new WD(e.name, e.refractionTextureUrl, e.color, e.depth, e.colorLevel, e.options, t, e.renderTargetSamplingMode, r.getEngine(), e.reusable), e, r, n); } } C([ M() ], WD.prototype, "color", void 0); C([ M() ], WD.prototype, "depth", void 0); C([ M() ], WD.prototype, "colorLevel", void 0); C([ M() ], WD.prototype, "refractionTextureUrl", void 0); Ue("BABYLON.RefractionPostProcess", WD); const i7e = "sharpenPixelShader", s7e = `varying vec2 vUV;uniform sampler2D textureSampler;uniform vec2 screenSize;uniform vec2 sharpnessAmounts; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {vec2 onePixel=vec2(1.0,1.0)/screenSize;vec4 color=texture2D(textureSampler,vUV);vec4 edgeDetection=texture2D(textureSampler,vUV+onePixel*vec2(0,-1)) + texture2D(textureSampler,vUV+onePixel*vec2(-1,0)) + texture2D(textureSampler,vUV+onePixel*vec2(1,0)) + texture2D(textureSampler,vUV+onePixel*vec2(0,1)) - color*4.0;gl_FragColor=max(vec4(color.rgb*sharpnessAmounts.y,color.a)-(sharpnessAmounts.x*vec4(edgeDetection.rgb,0)),0.);}`; Le.ShadersStore[i7e] = s7e; class Em extends kr { /** * Gets a string identifying the name of the class * @returns "SharpenPostProcess" string */ getClassName() { return "SharpenPostProcess"; } /** * Creates a new instance ConvolutionPostProcess * @param name The name of the effect. * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) * @param textureType Type of textures used when performing the post process. (default: 0) * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false) */ constructor(e, t, r, n, i, s, a = 0, f = !1) { super(e, "sharpen", ["sharpnessAmounts", "screenSize"], null, t, r, n, i, s, null, a, void 0, null, f), this.colorAmount = 1, this.edgeAmount = 0.3, this.onApply = (o) => { o.setFloat2("screenSize", this.width, this.height), o.setFloat2("sharpnessAmounts", this.edgeAmount, this.colorAmount); }; } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new Em(e.name, e.options, t, e.renderTargetSamplingMode, r.getEngine(), e.textureType, e.reusable), e, r, n); } } C([ M() ], Em.prototype, "colorAmount", void 0); C([ M() ], Em.prototype, "edgeAmount", void 0); Ue("BABYLON.SharpenPostProcess", Em); class SD { /** * Gets pipeline name */ get name() { return this._name; } /** Gets the list of attached cameras */ get cameras() { return this._cameras; } /** * Initializes a PostProcessRenderPipeline * @param _engine engine to add the pipeline to * @param name name of the pipeline */ constructor(e, t) { this._engine = e, this._name = t, this._renderEffects = {}, this._renderEffectsForIsolatedPass = new Array(), this._cameras = []; } /** * Gets the class name * @returns "PostProcessRenderPipeline" */ getClassName() { return "PostProcessRenderPipeline"; } /** * If all the render effects in the pipeline are supported */ get isSupported() { for (const e in this._renderEffects) if (Object.prototype.hasOwnProperty.call(this._renderEffects, e) && !this._renderEffects[e].isSupported) return !1; return !0; } /** * Adds an effect to the pipeline * @param renderEffect the effect to add */ addEffect(e) { this._renderEffects[e._name] = e; } // private /** @internal */ _rebuild() { } /** * @internal */ _enableEffect(e, t) { const r = this._renderEffects[e]; r && r._enable(ye.MakeArray(t || this._cameras)); } /** * @internal */ _disableEffect(e, t) { const r = this._renderEffects[e]; r && r._disable(ye.MakeArray(t || this._cameras)); } /** * @internal */ _attachCameras(e, t) { const r = ye.MakeArray(e || this._cameras); if (!r) return; const n = []; let i; for (i = 0; i < r.length; i++) { const s = r[i]; s && (this._cameras.indexOf(s) === -1 ? this._cameras.push(s) : t && n.push(i)); } for (i = 0; i < n.length; i++) r.splice(n[i], 1); for (const s in this._renderEffects) Object.prototype.hasOwnProperty.call(this._renderEffects, s) && this._renderEffects[s]._attachCameras(r); } /** * @internal */ _detachCameras(e) { const t = ye.MakeArray(e || this._cameras); if (t) { for (const r in this._renderEffects) Object.prototype.hasOwnProperty.call(this._renderEffects, r) && this._renderEffects[r]._detachCameras(t); for (let r = 0; r < t.length; r++) this._cameras.splice(this._cameras.indexOf(t[r]), 1); } } /** @internal */ _update() { for (const e in this._renderEffects) Object.prototype.hasOwnProperty.call(this._renderEffects, e) && this._renderEffects[e]._update(); for (let e = 0; e < this._cameras.length; e++) { if (!this._cameras[e]) continue; const t = this._cameras[e].name; this._renderEffectsForIsolatedPass[t] && this._renderEffectsForIsolatedPass[t]._update(); } } /** @internal */ _reset() { this._renderEffects = {}, this._renderEffectsForIsolatedPass = new Array(); } _enableMSAAOnFirstPostProcess(e) { if (!this._engine._features.supportMSAA) return !1; const t = Object.keys(this._renderEffects); if (t.length > 0) { const r = this._renderEffects[t[0]].getPostProcesses(); r && (r[0].samples = e); } return !0; } /** * Ensures that all post processes in the pipeline are the correct size according to the * the viewport's required size */ _adaptPostProcessesToViewPort() { const e = Object.keys(this._renderEffects); for (const t of e) { const r = this._renderEffects[t].getPostProcesses(); if (r) for (const n of r) n.adaptScaleToCurrentViewport = !0; } } /** * Sets the required values to the prepass renderer. * @param prePassRenderer defines the prepass renderer to setup. * @returns true if the pre pass is needed. */ // eslint-disable-next-line @typescript-eslint/no-unused-vars setPrePassRenderer(e) { return !1; } /** * Disposes of the pipeline */ dispose() { } } C([ M() ], SD.prototype, "_name", void 0); class nie { /** * Initializes a PostProcessRenderPipelineManager * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/postProcessRenderPipeline */ constructor() { this._renderPipelines = {}; } /** * Gets the list of supported render pipelines */ get supportedPipelines() { const e = []; for (const t in this._renderPipelines) if (Object.prototype.hasOwnProperty.call(this._renderPipelines, t)) { const r = this._renderPipelines[t]; r.isSupported && e.push(r); } return e; } /** * Adds a pipeline to the manager * @param renderPipeline The pipeline to add */ addPipeline(e) { this._renderPipelines[e._name] = e; } /** * Remove the pipeline from the manager * @param renderPipelineName the name of the pipeline to remove */ removePipeline(e) { delete this._renderPipelines[e]; } /** * Attaches a camera to the pipeline * @param renderPipelineName The name of the pipeline to attach to * @param cameras the camera to attach * @param unique if the camera can be attached multiple times to the pipeline */ attachCamerasToRenderPipeline(e, t, r = !1) { const n = this._renderPipelines[e]; n && n._attachCameras(t, r); } /** * Detaches a camera from the pipeline * @param renderPipelineName The name of the pipeline to detach from * @param cameras the camera to detach */ detachCamerasFromRenderPipeline(e, t) { const r = this._renderPipelines[e]; r && r._detachCameras(t); } /** * Enables an effect by name on a pipeline * @param renderPipelineName the name of the pipeline to enable the effect in * @param renderEffectName the name of the effect to enable * @param cameras the cameras that the effect should be enabled on */ enableEffectInPipeline(e, t, r) { const n = this._renderPipelines[e]; n && n._enableEffect(t, r); } /** * Disables an effect by name on a pipeline * @param renderPipelineName the name of the pipeline to disable the effect in * @param renderEffectName the name of the effect to disable * @param cameras the cameras that the effect should be disabled on */ disableEffectInPipeline(e, t, r) { const n = this._renderPipelines[e]; n && n._disableEffect(t, r); } /** * Updates the state of all contained render pipelines and disposes of any non supported pipelines */ update() { for (const e in this._renderPipelines) if (Object.prototype.hasOwnProperty.call(this._renderPipelines, e)) { const t = this._renderPipelines[e]; t.isSupported ? t._update() : (t.dispose(), delete this._renderPipelines[e]); } } /** @internal */ _rebuild() { for (const e in this._renderPipelines) Object.prototype.hasOwnProperty.call(this._renderPipelines, e) && this._renderPipelines[e]._rebuild(); } /** * Disposes of the manager and pipelines */ dispose() { for (const e in this._renderPipelines) Object.prototype.hasOwnProperty.call(this._renderPipelines, e) && this._renderPipelines[e].dispose(); } } Object.defineProperty(sr.prototype, "postProcessRenderPipelineManager", { get: function() { if (!this._postProcessRenderPipelineManager) { let A = this._getComponent(Ot.NAME_POSTPROCESSRENDERPIPELINEMANAGER); A || (A = new iie(this), this._addComponent(A)), this._postProcessRenderPipelineManager = new nie(); } return this._postProcessRenderPipelineManager; }, enumerable: !0, configurable: !0 }); class iie { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_POSTPROCESSRENDERPIPELINEMANAGER, this.scene = e; } /** * Registers the component in a given scene */ register() { this.scene._gatherRenderTargetsStage.registerStep(Ot.STEP_GATHERRENDERTARGETS_POSTPROCESSRENDERPIPELINEMANAGER, this, this._gatherRenderTargets); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { this.scene._postProcessRenderPipelineManager && this.scene._postProcessRenderPipelineManager._rebuild(); } /** * Disposes the component and the associated resources */ dispose() { this.scene._postProcessRenderPipelineManager && this.scene._postProcessRenderPipelineManager.dispose(); } _gatherRenderTargets() { this.scene._postProcessRenderPipelineManager && this.scene._postProcessRenderPipelineManager.update(); } } class K9 extends SD { /** * Enable or disable automatic building of the pipeline when effects are enabled and disabled. * If false, you will have to manually call prepare() to update the pipeline. */ get automaticBuild() { return this._buildAllowed; } set automaticBuild(e) { this._buildAllowed = e; } /** * Gets active scene */ get scene() { return this._scene; } /** * Enable or disable the sharpen process from the pipeline */ set sharpenEnabled(e) { this._sharpenEnabled !== e && (this._sharpenEnabled = e, this._buildPipeline()); } get sharpenEnabled() { return this._sharpenEnabled; } /** * Specifies the size of the bloom blur kernel, relative to the final output size */ get bloomKernel() { return this._bloomKernel; } set bloomKernel(e) { this._bloomKernel = e, this.bloom.kernel = e / this._hardwareScaleLevel; } /** * The strength of the bloom. */ set bloomWeight(e) { this._bloomWeight !== e && (this.bloom.weight = e, this._bloomWeight = e); } get bloomWeight() { return this._bloomWeight; } /** * The luminance threshold to find bright areas of the image to bloom. */ set bloomThreshold(e) { this._bloomThreshold !== e && (this.bloom.threshold = e, this._bloomThreshold = e); } get bloomThreshold() { return this._bloomThreshold; } /** * The scale of the bloom, lower value will provide better performance. */ set bloomScale(e) { this._bloomScale !== e && (this._bloomScale = e, this._rebuildBloom(), this._buildPipeline()); } get bloomScale() { return this._bloomScale; } /** * Enable or disable the bloom from the pipeline */ set bloomEnabled(e) { this._bloomEnabled !== e && (this._bloomEnabled = e, this._buildPipeline()); } get bloomEnabled() { return this._bloomEnabled; } _rebuildBloom() { const e = this.bloom; this.bloom = new DF(this._scene, this.bloomScale, this._bloomWeight, this.bloomKernel / this._hardwareScaleLevel, this._defaultPipelineTextureType, !1), this.bloom.threshold = e.threshold; for (let t = 0; t < this._cameras.length; t++) e.disposeEffects(this._cameras[t]); } /** * If the depth of field is enabled. */ get depthOfFieldEnabled() { return this._depthOfFieldEnabled; } set depthOfFieldEnabled(e) { this._depthOfFieldEnabled !== e && (this._depthOfFieldEnabled = e, this._buildPipeline()); } /** * Blur level of the depth of field effect. (Higher blur will effect performance) */ get depthOfFieldBlurLevel() { return this._depthOfFieldBlurLevel; } set depthOfFieldBlurLevel(e) { if (this._depthOfFieldBlurLevel === e) return; this._depthOfFieldBlurLevel = e; const t = this.depthOfField; this.depthOfField = new jF(this._scene, null, this._depthOfFieldBlurLevel, this._defaultPipelineTextureType, !1), this.depthOfField.focalLength = t.focalLength, this.depthOfField.focusDistance = t.focusDistance, this.depthOfField.fStop = t.fStop, this.depthOfField.lensSize = t.lensSize; for (let r = 0; r < this._cameras.length; r++) t.disposeEffects(this._cameras[r]); this._buildPipeline(); } /** * If the anti aliasing is enabled. */ set fxaaEnabled(e) { this._fxaaEnabled !== e && (this._fxaaEnabled = e, this._buildPipeline()); } get fxaaEnabled() { return this._fxaaEnabled; } /** * MSAA sample count, setting this to 4 will provide 4x anti aliasing. (default: 1) */ set samples(e) { this._samples !== e && (this._samples = e, this._buildPipeline()); } get samples() { return this._samples; } /** * If image processing is enabled. */ set imageProcessingEnabled(e) { this._imageProcessingEnabled !== e && (this._scene.imageProcessingConfiguration.isEnabled = e); } get imageProcessingEnabled() { return this._imageProcessingEnabled; } /** * If glow layer is enabled. (Adds a glow effect to emmissive materials) */ set glowLayerEnabled(e) { e && !this._glowLayer ? this._glowLayer = new zl("", this._scene) : !e && this._glowLayer && (this._glowLayer.dispose(), this._glowLayer = null); } get glowLayerEnabled() { return this._glowLayer != null; } /** * Gets the glow layer (or null if not defined) */ get glowLayer() { return this._glowLayer; } /** * Enable or disable the chromaticAberration process from the pipeline */ set chromaticAberrationEnabled(e) { this._chromaticAberrationEnabled !== e && (this._chromaticAberrationEnabled = e, this._buildPipeline()); } get chromaticAberrationEnabled() { return this._chromaticAberrationEnabled; } /** * Enable or disable the grain process from the pipeline */ set grainEnabled(e) { this._grainEnabled !== e && (this._grainEnabled = e, this._buildPipeline()); } get grainEnabled() { return this._grainEnabled; } /** * Instantiates a DefaultRenderingPipeline. * @param name The rendering pipeline name (default: "") * @param hdr If high dynamic range textures should be used (default: true) * @param scene The scene linked to this pipeline (default: the last created scene) * @param cameras The array of cameras that the rendering pipeline will be attached to (default: scene.cameras) * @param automaticBuild If false, you will have to manually call prepare() to update the pipeline (default: true) */ constructor(e = "", t = !0, r = gr.LastCreatedScene, n, i = !0) { super(r.getEngine(), e), this._camerasToBeAttached = [], this.SharpenPostProcessId = "SharpenPostProcessEffect", this.ImageProcessingPostProcessId = "ImageProcessingPostProcessEffect", this.FxaaPostProcessId = "FxaaPostProcessEffect", this.ChromaticAberrationPostProcessId = "ChromaticAberrationPostProcessEffect", this.GrainPostProcessId = "GrainPostProcessEffect", this._glowLayer = null, this.animations = [], this._imageProcessingConfigurationObserver = null, this._sharpenEnabled = !1, this._bloomEnabled = !1, this._depthOfFieldEnabled = !1, this._depthOfFieldBlurLevel = am.Low, this._fxaaEnabled = !1, this._imageProcessingEnabled = !0, this._bloomScale = 0.5, this._chromaticAberrationEnabled = !1, this._grainEnabled = !1, this._buildAllowed = !0, this.onBuildObservable = new Oe(), this._resizeObserver = null, this._hardwareScaleLevel = 1, this._bloomKernel = 64, this._bloomWeight = 0.15, this._bloomThreshold = 0.9, this._samples = 1, this._hasCleared = !1, this._prevPostProcess = null, this._prevPrevPostProcess = null, this._depthOfFieldSceneObserver = null, this._activeCameraChangedObserver = null, this._activeCamerasChangedObserver = null, this._cameras = n || r.cameras, this._cameras = this._cameras.slice(), this._camerasToBeAttached = this._cameras.slice(), this._buildAllowed = i, this._scene = r; const s = this._scene.getEngine().getCaps(); this._hdr = t && (s.textureHalfFloatRender || s.textureFloatRender), this._hdr ? s.textureHalfFloatRender ? this._defaultPipelineTextureType = 2 : s.textureFloatRender && (this._defaultPipelineTextureType = 1) : this._defaultPipelineTextureType = 0, r.postProcessRenderPipelineManager.addPipeline(this); const a = this._scene.getEngine(); this.sharpen = new Em("sharpen", 1, null, We.BILINEAR_SAMPLINGMODE, a, !1, this._defaultPipelineTextureType, !0), this._sharpenEffect = new gs(a, this.SharpenPostProcessId, () => this.sharpen, !0), this.depthOfField = new jF(this._scene, null, this._depthOfFieldBlurLevel, this._defaultPipelineTextureType, !0), this._hardwareScaleLevel = a.getHardwareScalingLevel(), this._resizeObserver = a.onResizeObservable.add(() => { this._hardwareScaleLevel = a.getHardwareScalingLevel(), this.bloomKernel = this._bloomKernel; }), this.bloom = new DF(this._scene, this._bloomScale, this._bloomWeight, this.bloomKernel / this._hardwareScaleLevel, this._defaultPipelineTextureType, !0), this.chromaticAberration = new Pg("ChromaticAberration", a.getRenderWidth(), a.getRenderHeight(), 1, null, We.BILINEAR_SAMPLINGMODE, a, !1, this._defaultPipelineTextureType, !0), this._chromaticAberrationEffect = new gs(a, this.ChromaticAberrationPostProcessId, () => this.chromaticAberration, !0), this.grain = new km("Grain", 1, null, We.BILINEAR_SAMPLINGMODE, a, !1, this._defaultPipelineTextureType, !0), this._grainEffect = new gs(a, this.GrainPostProcessId, () => this.grain, !0); let f = !0; this._imageProcessingConfigurationObserver = this._scene.imageProcessingConfiguration.onUpdateParameters.add(() => { this.bloom._downscale._exposure = this._scene.imageProcessingConfiguration.exposure, this.imageProcessingEnabled !== this._scene.imageProcessingConfiguration.isEnabled && (this._imageProcessingEnabled = this._scene.imageProcessingConfiguration.isEnabled, f ? ye.SetImmediate(() => { this._buildPipeline(); }) : this._buildPipeline()); }), this._buildPipeline(), f = !1; } /** * Get the class name * @returns "DefaultRenderingPipeline" */ getClassName() { return "DefaultRenderingPipeline"; } /** * Force the compilation of the entire pipeline. */ prepare() { const e = this._buildAllowed; this._buildAllowed = !0, this._buildPipeline(), this._buildAllowed = e; } _setAutoClearAndTextureSharing(e, t = !1) { this._hasCleared ? e.autoClear = !1 : (e.autoClear = !0, this._scene.autoClear = !1, this._hasCleared = !0), t || (this._prevPrevPostProcess ? e.shareOutputWith(this._prevPrevPostProcess) : e.useOwnOutput(), this._prevPostProcess && (this._prevPrevPostProcess = this._prevPostProcess), this._prevPostProcess = e); } _buildPipeline() { if (!this._buildAllowed) return; this._scene.autoClear = !0; const e = this._scene.getEngine(); if (this._disposePostProcesses(), this._cameras !== null && (this._scene.postProcessRenderPipelineManager.detachCamerasFromRenderPipeline(this._name, this._cameras), this._cameras = this._camerasToBeAttached.slice()), this._reset(), this._prevPostProcess = null, this._prevPrevPostProcess = null, this._hasCleared = !1, this.depthOfFieldEnabled) { if (this._cameras.length > 1) { for (const t of this._cameras) { const r = this._scene.enableDepthRenderer(t); r.useOnlyInActiveCamera = !0; } this._depthOfFieldSceneObserver = this._scene.onAfterRenderTargetsRenderObservable.add((t) => { this._cameras.indexOf(t.activeCamera) > -1 && (this.depthOfField.depthTexture = t.enableDepthRenderer(t.activeCamera).getDepthMap()); }); } else { this._scene.onAfterRenderTargetsRenderObservable.remove(this._depthOfFieldSceneObserver); const t = this._scene.enableDepthRenderer(this._cameras[0]); this.depthOfField.depthTexture = t.getDepthMap(); } this.depthOfField._isReady() || this.depthOfField._updateEffects(), this.addEffect(this.depthOfField), this._setAutoClearAndTextureSharing(this.depthOfField._effects[0], !0); } else this._scene.onAfterRenderTargetsRenderObservable.remove(this._depthOfFieldSceneObserver); this.bloomEnabled && (this.bloom._isReady() || this.bloom._updateEffects(), this.addEffect(this.bloom), this._setAutoClearAndTextureSharing(this.bloom._effects[0], !0)), this._imageProcessingEnabled && (this.imageProcessing = new Uy("imageProcessing", 1, null, We.BILINEAR_SAMPLINGMODE, e, !1, this._defaultPipelineTextureType, this.scene.imageProcessingConfiguration), this._hdr ? (this.addEffect(new gs(e, this.ImageProcessingPostProcessId, () => this.imageProcessing, !0)), this._setAutoClearAndTextureSharing(this.imageProcessing)) : this._scene.imageProcessingConfiguration.applyByPostProcess = !1, (!this._cameras || this._cameras.length === 0) && (this._scene.imageProcessingConfiguration.applyByPostProcess = !1), this.imageProcessing.getEffect() || this.imageProcessing._updateParameters()), this.sharpenEnabled && (this.sharpen.isReady() || this.sharpen.updateEffect(), this.addEffect(this._sharpenEffect), this._setAutoClearAndTextureSharing(this.sharpen)), this.grainEnabled && (this.grain.isReady() || this.grain.updateEffect(), this.addEffect(this._grainEffect), this._setAutoClearAndTextureSharing(this.grain)), this.chromaticAberrationEnabled && (this.chromaticAberration.isReady() || this.chromaticAberration.updateEffect(), this.addEffect(this._chromaticAberrationEffect), this._setAutoClearAndTextureSharing(this.chromaticAberration)), this.fxaaEnabled && (this.fxaa = new ym("fxaa", 1, null, We.BILINEAR_SAMPLINGMODE, e, !1, this._defaultPipelineTextureType), this.addEffect(new gs(e, this.FxaaPostProcessId, () => this.fxaa, !0)), this._setAutoClearAndTextureSharing(this.fxaa, !0)), this._cameras !== null && this._scene.postProcessRenderPipelineManager.attachCamerasToRenderPipeline(this._name, this._cameras), (this._scene.activeCameras && this._scene.activeCameras.length > 1 || this._scene.activeCamera && this._cameras.indexOf(this._scene.activeCamera) === -1) && (this._scene.autoClear = !0), this._activeCameraChangedObserver || (this._activeCameraChangedObserver = this._scene.onActiveCameraChanged.add(() => { this._scene.activeCamera && this._cameras.indexOf(this._scene.activeCamera) === -1 && (this._scene.autoClear = !0); })), this._activeCamerasChangedObserver || (this._activeCamerasChangedObserver = this._scene.onActiveCamerasChanged.add(() => { this._scene.activeCameras && this._scene.activeCameras.length > 1 && (this._scene.autoClear = !0); })), this._adaptPostProcessesToViewPort(), !this._enableMSAAOnFirstPostProcess(this.samples) && this.samples > 1 && Se.Warn("MSAA failed to enable, MSAA is only supported in browsers that support webGL >= 2.0"), this.onBuildObservable.notifyObservers(this); } _disposePostProcesses(e = !1) { for (let t = 0; t < this._cameras.length; t++) { const r = this._cameras[t]; this.imageProcessing && this.imageProcessing.dispose(r), this.fxaa && this.fxaa.dispose(r), e && (this.sharpen && this.sharpen.dispose(r), this.depthOfField && (this._scene.onAfterRenderTargetsRenderObservable.remove(this._depthOfFieldSceneObserver), this.depthOfField.disposeEffects(r)), this.bloom && this.bloom.disposeEffects(r), this.chromaticAberration && this.chromaticAberration.dispose(r), this.grain && this.grain.dispose(r), this._glowLayer && this._glowLayer.dispose()); } this.imageProcessing = null, this.fxaa = null, e && (this.sharpen = null, this._sharpenEffect = null, this.depthOfField = null, this.bloom = null, this.chromaticAberration = null, this._chromaticAberrationEffect = null, this.grain = null, this._grainEffect = null, this._glowLayer = null); } /** * Adds a camera to the pipeline * @param camera the camera to be added */ addCamera(e) { this._camerasToBeAttached.push(e), this._buildPipeline(); } /** * Removes a camera from the pipeline * @param camera the camera to remove */ removeCamera(e) { const t = this._camerasToBeAttached.indexOf(e); this._camerasToBeAttached.splice(t, 1), this._buildPipeline(); } /** * Dispose of the pipeline and stop all post processes */ dispose() { this._buildAllowed = !1, this.onBuildObservable.clear(), this._disposePostProcesses(!0), this._scene.postProcessRenderPipelineManager.detachCamerasFromRenderPipeline(this._name, this._cameras), this._scene._postProcessRenderPipelineManager.removePipeline(this.name), this._scene.autoClear = !0, this._resizeObserver && (this._scene.getEngine().onResizeObservable.remove(this._resizeObserver), this._resizeObserver = null), this._scene.onActiveCameraChanged.remove(this._activeCameraChangedObserver), this._scene.onActiveCamerasChanged.remove(this._activeCamerasChangedObserver), this._scene.imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingConfigurationObserver), super.dispose(); } /** * Serialize the rendering pipeline (Used when exporting) * @returns the serialized object */ serialize() { const e = jt.Serialize(this); return e.customType = "DefaultRenderingPipeline", e; } /** * Parse the serialized pipeline * @param source Source pipeline. * @param scene The scene to load the pipeline to. * @param rootUrl The URL of the serialized pipeline. * @returns An instantiated pipeline from the serialized object. */ static Parse(e, t, r) { return jt.Parse(() => new K9(e._name, e._name._hdr, t), e, t, r); } } C([ M() ], K9.prototype, "sharpenEnabled", null); C([ M() ], K9.prototype, "bloomKernel", null); C([ M() ], K9.prototype, "_bloomWeight", void 0); C([ M() ], K9.prototype, "_bloomThreshold", void 0); C([ M() ], K9.prototype, "_hdr", void 0); C([ M() ], K9.prototype, "bloomWeight", null); C([ M() ], K9.prototype, "bloomThreshold", null); C([ M() ], K9.prototype, "bloomScale", null); C([ M() ], K9.prototype, "bloomEnabled", null); C([ M() ], K9.prototype, "depthOfFieldEnabled", null); C([ M() ], K9.prototype, "depthOfFieldBlurLevel", null); C([ M() ], K9.prototype, "fxaaEnabled", null); C([ M() ], K9.prototype, "samples", null); C([ M() ], K9.prototype, "imageProcessingEnabled", null); C([ M() ], K9.prototype, "glowLayerEnabled", null); C([ M() ], K9.prototype, "chromaticAberrationEnabled", null); C([ M() ], K9.prototype, "grainEnabled", null); Ue("BABYLON.DefaultRenderingPipeline", K9); const a7e = "lensHighlightsPixelShader", o7e = `uniform sampler2D textureSampler; uniform float gain;uniform float threshold;uniform float screen_width;uniform float screen_height;varying vec2 vUV;vec4 highlightColor(vec4 color) {vec4 highlight=color;float luminance=dot(highlight.rgb,vec3(0.2125,0.7154,0.0721));float lum_threshold;if (threshold>1.0) { lum_threshold=0.94+0.01*threshold; } else { lum_threshold=0.5+0.44*threshold; } luminance=clamp((luminance-lum_threshold)*(1.0/(1.0-lum_threshold)),0.0,1.0);highlight*=luminance*gain;highlight.a=1.0;return highlight;} #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {vec4 original=texture2D(textureSampler,vUV);if (gain==-1.0) {gl_FragColor=vec4(0.0,0.0,0.0,1.0);return;} float w=2.0/screen_width;float h=2.0/screen_height;float weight=1.0;vec4 blurred=vec4(0.0,0.0,0.0,0.0); #ifdef PENTAGON blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.84*w,0.43*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.48*w,-1.29*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.61*w,1.51*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.55*w,-0.74*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.71*w,-0.52*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.94*w,1.59*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.40*w,-1.87*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.62*w,1.16*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.09*w,0.25*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.46*w,-1.71*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.08*w,2.42*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.85*w,-1.89*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.89*w,0.16*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.29*w,1.88*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.40*w,-2.81*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.54*w,2.26*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.60*w,-0.61*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.31*w,-1.30*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.83*w,2.53*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.12*w,-2.48*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.60*w,1.11*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.82*w,0.99*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.50*w,-2.81*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.85*w,3.33*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.94*w,-1.92*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(3.27*w,-0.53*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.95*w,2.48*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.23*w,-3.04*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.17*w,2.05*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.97*w,-0.04*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.25*w,-2.00*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.31*w,3.08*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.94*w,-2.59*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(3.37*w,0.64*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-3.13*w,1.93*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.03*w,-3.65*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.60*w,3.17*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-3.14*w,-1.19*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(3.00*w,-1.19*h))); #else blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.85*w,0.36*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.52*w,-1.14*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.46*w,1.42*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.46*w,-0.83*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.79*w,-0.42*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.11*w,1.62*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.29*w,-2.07*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.69*w,1.39*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.28*w,0.12*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.65*w,-1.69*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.08*w,2.44*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.63*w,-1.90*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.55*w,0.31*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.13*w,1.52*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.56*w,-2.61*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.38*w,2.34*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.64*w,-0.81*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.53*w,-1.21*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.06*w,2.63*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.00*w,-2.69*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.59*w,1.32*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.82*w,0.78*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.57*w,-2.50*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.54*w,2.93*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.39*w,-1.81*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(3.01*w,-0.28*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.04*w,2.25*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.02*w,-3.05*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.09*w,2.25*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-3.07*w,-0.25*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.44*w,-1.90*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.52*w,3.05*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.68*w,-2.61*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(3.01*w,0.79*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.76*w,1.46*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.05*w,-2.94*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.21*w,2.88*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.84*w,-1.30*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.98*w,-0.96*h))); #endif blurred/=39.0;gl_FragColor=blurred;}`; Le.ShadersStore[a7e] = o7e; const f7e = "depthOfFieldPixelShader", A7e = `uniform sampler2D textureSampler;uniform sampler2D highlightsSampler;uniform sampler2D depthSampler;uniform sampler2D grainSampler;uniform float grain_amount;uniform bool blur_noise;uniform float screen_width;uniform float screen_height;uniform float distortion;uniform bool dof_enabled;uniform float screen_distance; uniform float aperture;uniform float darken;uniform float edge_blur;uniform bool highlights;uniform float near;uniform float far;varying vec2 vUV; #define PI 3.14159265 #define TWOPI 6.28318530 #define inverse_focal_length 0.1 vec2 centered_screen_pos;vec2 distorted_coords;float radius2;float radius;vec2 rand(vec2 co) {float noise1=(fract(sin(dot(co,vec2(12.9898,78.233)))*43758.5453));float noise2=(fract(sin(dot(co,vec2(12.9898,78.233)*2.0))*43758.5453));return clamp(vec2(noise1,noise2),0.0,1.0);} vec2 getDistortedCoords(vec2 coords) {if (distortion==0.0) { return coords; } vec2 direction=1.0*normalize(centered_screen_pos);vec2 dist_coords=vec2(0.5,0.5);dist_coords.x=0.5+direction.x*radius2*1.0;dist_coords.y=0.5+direction.y*radius2*1.0;float dist_amount=clamp(distortion*0.23,0.0,1.0);dist_coords=mix(coords,dist_coords,dist_amount);return dist_coords;} float sampleScreen(inout vec4 color,in vec2 offset,in float weight) {vec2 coords=distorted_coords;float angle=rand(coords*100.0).x*TWOPI;coords+=vec2(offset.x*cos(angle)-offset.y*sin(angle),offset.x*sin(angle)+offset.y*cos(angle));color+=texture2D(textureSampler,coords)*weight;return weight;} float getBlurLevel(float size) {return min(3.0,ceil(size/1.0));} vec4 getBlurColor(float size) {vec4 col=texture2D(textureSampler,distorted_coords);float blur_level=getBlurLevel(size);float w=(size/screen_width);float h=(size/screen_height);float total_weight=1.0;vec2 sample_coords;total_weight+=sampleScreen(col,vec2(-0.50*w,0.24*h),0.93);total_weight+=sampleScreen(col,vec2(0.30*w,-0.75*h),0.90);total_weight+=sampleScreen(col,vec2(0.36*w,0.96*h),0.87);total_weight+=sampleScreen(col,vec2(-1.08*w,-0.55*h),0.85);total_weight+=sampleScreen(col,vec2(1.33*w,-0.37*h),0.83);total_weight+=sampleScreen(col,vec2(-0.82*w,1.31*h),0.80);total_weight+=sampleScreen(col,vec2(-0.31*w,-1.67*h),0.78);total_weight+=sampleScreen(col,vec2(1.47*w,1.11*h),0.76);total_weight+=sampleScreen(col,vec2(-1.97*w,0.19*h),0.74);total_weight+=sampleScreen(col,vec2(1.42*w,-1.57*h),0.72);if (blur_level>1.0) {total_weight+=sampleScreen(col,vec2(0.01*w,2.25*h),0.70);total_weight+=sampleScreen(col,vec2(-1.62*w,-1.74*h),0.67);total_weight+=sampleScreen(col,vec2(2.49*w,0.20*h),0.65);total_weight+=sampleScreen(col,vec2(-2.07*w,1.61*h),0.63);total_weight+=sampleScreen(col,vec2(0.46*w,-2.70*h),0.61);total_weight+=sampleScreen(col,vec2(1.55*w,2.40*h),0.59);total_weight+=sampleScreen(col,vec2(-2.88*w,-0.75*h),0.56);total_weight+=sampleScreen(col,vec2(2.73*w,-1.44*h),0.54);total_weight+=sampleScreen(col,vec2(-1.08*w,3.02*h),0.52);total_weight+=sampleScreen(col,vec2(-1.28*w,-3.05*h),0.49);} if (blur_level>2.0) {total_weight+=sampleScreen(col,vec2(3.11*w,1.43*h),0.46);total_weight+=sampleScreen(col,vec2(-3.36*w,1.08*h),0.44);total_weight+=sampleScreen(col,vec2(1.80*w,-3.16*h),0.41);total_weight+=sampleScreen(col,vec2(0.83*w,3.65*h),0.38);total_weight+=sampleScreen(col,vec2(-3.16*w,-2.19*h),0.34);total_weight+=sampleScreen(col,vec2(3.92*w,-0.53*h),0.31);total_weight+=sampleScreen(col,vec2(-2.59*w,3.12*h),0.26);total_weight+=sampleScreen(col,vec2(-0.20*w,-4.15*h),0.22);total_weight+=sampleScreen(col,vec2(3.02*w,3.00*h),0.15);} col/=total_weight; if (darken>0.0) {col.rgb*=clamp(0.3,1.0,1.05-size*0.5*darken);} return col;} #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {centered_screen_pos=vec2(vUV.x-0.5,vUV.y-0.5);radius2=centered_screen_pos.x*centered_screen_pos.x+centered_screen_pos.y*centered_screen_pos.y;radius=sqrt(radius2);distorted_coords=getDistortedCoords(vUV); vec2 texels_coords=vec2(vUV.x*screen_width,vUV.y*screen_height); float depth=texture2D(depthSampler,distorted_coords).r; float distance=near+(far-near)*depth; vec4 color=texture2D(textureSampler,vUV); float coc=abs(aperture*(screen_distance*(inverse_focal_length-1.0/distance)-1.0));if (dof_enabled==false || coc<0.07) { coc=0.0; } float edge_blur_amount=0.0;if (edge_blur>0.0) {edge_blur_amount=clamp((radius*2.0-1.0+0.15*edge_blur)*1.5,0.0,1.0)*1.3;} float blur_amount=max(edge_blur_amount,coc);if (blur_amount==0.0) {gl_FragColor=texture2D(textureSampler,distorted_coords);} else {gl_FragColor=getBlurColor(blur_amount*1.7);if (highlights) {gl_FragColor.rgb+=clamp(coc,0.0,1.0)*texture2D(highlightsSampler,distorted_coords).rgb;} if (blur_noise) {vec2 noise=rand(distorted_coords)*0.01*blur_amount;vec2 blurred_coord=vec2(distorted_coords.x+noise.x,distorted_coords.y+noise.y);gl_FragColor=0.04*texture2D(textureSampler,blurred_coord)+0.96*gl_FragColor;}} if (grain_amount>0.0) {vec4 grain_color=texture2D(grainSampler,texels_coords*0.003);gl_FragColor.rgb+=(-0.5+grain_color.rgb)*0.30*grain_amount;}} `; Le.ShadersStore[f7e] = A7e; class d7e extends SD { /** * @constructor * * Effect parameters are as follow: * { * chromatic_aberration: number; // from 0 to x (1 for realism) * edge_blur: number; // from 0 to x (1 for realism) * distortion: number; // from 0 to x (1 for realism), note that this will effect the pointer position precision * grain_amount: number; // from 0 to 1 * grain_texture: BABYLON.Texture; // texture to use for grain effect; if unset, use random B&W noise * dof_focus_distance: number; // depth-of-field: focus distance; unset to disable (disabled by default) * dof_aperture: number; // depth-of-field: focus blur bias (default: 1) * dof_darken: number; // depth-of-field: darken that which is out of focus (from 0 to 1, disabled by default) * dof_pentagon: boolean; // depth-of-field: makes a pentagon-like "bokeh" effect * dof_gain: number; // depth-of-field: highlights gain; unset to disable (disabled by default) * dof_threshold: number; // depth-of-field: highlights threshold (default: 1) * blur_noise: boolean; // add a little bit of noise to the blur (default: true) * } * Note: if an effect parameter is unset, effect is disabled * * @param name The rendering pipeline name * @param parameters - An object containing all parameters (see above) * @param scene The scene linked to this pipeline * @param ratio The size of the postprocesses (0.5 means that your postprocess will have a width = canvas.width 0.5 and a height = canvas.height 0.5) * @param cameras The array of cameras that the rendering pipeline will be attached to */ constructor(e, t, r, n = 1, i) { super(r.getEngine(), e), this.LensChromaticAberrationEffect = "LensChromaticAberrationEffect", this.HighlightsEnhancingEffect = "HighlightsEnhancingEffect", this.LensDepthOfFieldEffect = "LensDepthOfFieldEffect", this._pentagonBokehIsEnabled = !1, this._scene = r, this._depthTexture = r.enableDepthRenderer().getDepthMap(), t.grain_texture ? this._grainTexture = t.grain_texture : this._createGrainTexture(), this._edgeBlur = t.edge_blur ? t.edge_blur : 0, this._grainAmount = t.grain_amount ? t.grain_amount : 0, this._chromaticAberration = t.chromatic_aberration ? t.chromatic_aberration : 0, this._distortion = t.distortion ? t.distortion : 0, this._highlightsGain = t.dof_gain !== void 0 ? t.dof_gain : -1, this._highlightsThreshold = t.dof_threshold ? t.dof_threshold : 1, this._dofDistance = t.dof_focus_distance !== void 0 ? t.dof_focus_distance : -1, this._dofAperture = t.dof_aperture ? t.dof_aperture : 1, this._dofDarken = t.dof_darken ? t.dof_darken : 0, this._dofPentagon = t.dof_pentagon !== void 0 ? t.dof_pentagon : !0, this._blurNoise = t.blur_noise !== void 0 ? t.blur_noise : !0, this._createChromaticAberrationPostProcess(n), this._createHighlightsPostProcess(n), this._createDepthOfFieldPostProcess(n / 4), this.addEffect(new gs(r.getEngine(), this.LensChromaticAberrationEffect, () => this._chromaticAberrationPostProcess, !0)), this.addEffect(new gs(r.getEngine(), this.HighlightsEnhancingEffect, () => this._highlightsPostProcess, !0)), this.addEffect(new gs(r.getEngine(), this.LensDepthOfFieldEffect, () => this._depthOfFieldPostProcess, !0)), this._highlightsGain === -1 && this._disableEffect(this.HighlightsEnhancingEffect, null), r.postProcessRenderPipelineManager.addPipeline(this), i && r.postProcessRenderPipelineManager.attachCamerasToRenderPipeline(e, i); } /** * Get the class name * @returns "LensRenderingPipeline" */ getClassName() { return "LensRenderingPipeline"; } // Properties /** * Gets associated scene */ get scene() { return this._scene; } /** * Gets or sets the edge blur */ get edgeBlur() { return this._edgeBlur; } set edgeBlur(e) { this.setEdgeBlur(e); } /** * Gets or sets the grain amount */ get grainAmount() { return this._grainAmount; } set grainAmount(e) { this.setGrainAmount(e); } /** * Gets or sets the chromatic aberration amount */ get chromaticAberration() { return this._chromaticAberration; } set chromaticAberration(e) { this.setChromaticAberration(e); } /** * Gets or sets the depth of field aperture */ get dofAperture() { return this._dofAperture; } set dofAperture(e) { this.setAperture(e); } /** * Gets or sets the edge distortion */ get edgeDistortion() { return this._distortion; } set edgeDistortion(e) { this.setEdgeDistortion(e); } /** * Gets or sets the depth of field distortion */ get dofDistortion() { return this._dofDistance; } set dofDistortion(e) { this.setFocusDistance(e); } /** * Gets or sets the darken out of focus amount */ get darkenOutOfFocus() { return this._dofDarken; } set darkenOutOfFocus(e) { this.setDarkenOutOfFocus(e); } /** * Gets or sets a boolean indicating if blur noise is enabled */ get blurNoise() { return this._blurNoise; } set blurNoise(e) { this._blurNoise = e; } /** * Gets or sets a boolean indicating if pentagon bokeh is enabled */ get pentagonBokeh() { return this._pentagonBokehIsEnabled; } set pentagonBokeh(e) { e ? this.enablePentagonBokeh() : this.disablePentagonBokeh(); } /** * Gets or sets the highlight grain amount */ get highlightsGain() { return this._highlightsGain; } set highlightsGain(e) { this.setHighlightsGain(e); } /** * Gets or sets the highlight threshold */ get highlightsThreshold() { return this._highlightsThreshold; } set highlightsThreshold(e) { this.setHighlightsThreshold(e); } // public methods (self explanatory) /** * Sets the amount of blur at the edges * @param amount blur amount */ setEdgeBlur(e) { this._edgeBlur = e; } /** * Sets edge blur to 0 */ disableEdgeBlur() { this._edgeBlur = 0; } /** * Sets the amount of grain * @param amount Amount of grain */ setGrainAmount(e) { this._grainAmount = e; } /** * Set grain amount to 0 */ disableGrain() { this._grainAmount = 0; } /** * Sets the chromatic aberration amount * @param amount amount of chromatic aberration */ setChromaticAberration(e) { this._chromaticAberration = e; } /** * Sets chromatic aberration amount to 0 */ disableChromaticAberration() { this._chromaticAberration = 0; } /** * Sets the EdgeDistortion amount * @param amount amount of EdgeDistortion */ setEdgeDistortion(e) { this._distortion = e; } /** * Sets edge distortion to 0 */ disableEdgeDistortion() { this._distortion = 0; } /** * Sets the FocusDistance amount * @param amount amount of FocusDistance */ setFocusDistance(e) { this._dofDistance = e; } /** * Disables depth of field */ disableDepthOfField() { this._dofDistance = -1; } /** * Sets the Aperture amount * @param amount amount of Aperture */ setAperture(e) { this._dofAperture = e; } /** * Sets the DarkenOutOfFocus amount * @param amount amount of DarkenOutOfFocus */ setDarkenOutOfFocus(e) { this._dofDarken = e; } /** * Creates a pentagon bokeh effect */ enablePentagonBokeh() { this._highlightsPostProcess.updateEffect(`#define PENTAGON `), this._pentagonBokehIsEnabled = !0; } /** * Disables the pentagon bokeh effect */ disablePentagonBokeh() { this._pentagonBokehIsEnabled = !1, this._highlightsPostProcess.updateEffect(); } /** * Enables noise blur */ enableNoiseBlur() { this._blurNoise = !0; } /** * Disables noise blur */ disableNoiseBlur() { this._blurNoise = !1; } /** * Sets the HighlightsGain amount * @param amount amount of HighlightsGain */ setHighlightsGain(e) { this._highlightsGain = e; } /** * Sets the HighlightsThreshold amount * @param amount amount of HighlightsThreshold */ setHighlightsThreshold(e) { this._highlightsGain === -1 && (this._highlightsGain = 1), this._highlightsThreshold = e; } /** * Disables highlights */ disableHighlights() { this._highlightsGain = -1; } /** * Removes the internal pipeline assets and detaches the pipeline from the scene cameras * @param disableDepthRender If the scene's depth rendering should be disabled (default: false) */ dispose(e = !1) { this._scene.postProcessRenderPipelineManager.detachCamerasFromRenderPipeline(this._name, this._scene.cameras), this._chromaticAberrationPostProcess = null, this._highlightsPostProcess = null, this._depthOfFieldPostProcess = null, this._grainTexture.dispose(), e && this._scene.disableDepthRenderer(); } // colors shifting and distortion _createChromaticAberrationPostProcess(e) { this._chromaticAberrationPostProcess = new kr( "LensChromaticAberration", "chromaticAberration", ["chromatic_aberration", "screen_width", "screen_height", "direction", "radialIntensity", "centerPosition"], // uniforms [], // samplers e, null, We.TRILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1 ), this._chromaticAberrationPostProcess.onApply = (t) => { t.setFloat("chromatic_aberration", this._chromaticAberration), t.setFloat("screen_width", this._scene.getEngine().getRenderWidth()), t.setFloat("screen_height", this._scene.getEngine().getRenderHeight()), t.setFloat("radialIntensity", 1), t.setFloat2("direction", 17, 17), t.setFloat2("centerPosition", 0.5, 0.5); }; } // highlights enhancing _createHighlightsPostProcess(e) { this._highlightsPostProcess = new kr( "LensHighlights", "lensHighlights", ["gain", "threshold", "screen_width", "screen_height"], // uniforms [], // samplers e, null, We.TRILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1, this._dofPentagon ? `#define PENTAGON ` : "" ), this._highlightsPostProcess.externalTextureSamplerBinding = !0, this._highlightsPostProcess.onApply = (t) => { t.setFloat("gain", this._highlightsGain), t.setFloat("threshold", this._highlightsThreshold), t.setTextureFromPostProcess("textureSampler", this._chromaticAberrationPostProcess), t.setFloat("screen_width", this._scene.getEngine().getRenderWidth()), t.setFloat("screen_height", this._scene.getEngine().getRenderHeight()); }; } // colors shifting and distortion _createDepthOfFieldPostProcess(e) { this._depthOfFieldPostProcess = new kr("LensDepthOfField", "depthOfField", [ "grain_amount", "blur_noise", "screen_width", "screen_height", "distortion", "dof_enabled", "screen_distance", "aperture", "darken", "edge_blur", "highlights", "near", "far" ], ["depthSampler", "grainSampler", "highlightsSampler"], e, null, We.TRILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1), this._depthOfFieldPostProcess.externalTextureSamplerBinding = !0, this._depthOfFieldPostProcess.onApply = (t) => { t.setTexture("depthSampler", this._depthTexture), t.setTexture("grainSampler", this._grainTexture), t.setTextureFromPostProcess("textureSampler", this._highlightsPostProcess), t.setTextureFromPostProcess("highlightsSampler", this._depthOfFieldPostProcess), t.setFloat("grain_amount", this._grainAmount), t.setBool("blur_noise", this._blurNoise), t.setFloat("screen_width", this._scene.getEngine().getRenderWidth()), t.setFloat("screen_height", this._scene.getEngine().getRenderHeight()), t.setFloat("distortion", this._distortion), t.setBool("dof_enabled", this._dofDistance !== -1), t.setFloat("screen_distance", 1 / (0.1 - 1 / this._dofDistance)), t.setFloat("aperture", this._dofAperture), t.setFloat("darken", this._dofDarken), t.setFloat("edge_blur", this._edgeBlur), t.setBool("highlights", this._highlightsGain !== -1), this._scene.activeCamera && (t.setFloat("near", this._scene.activeCamera.minZ), t.setFloat("far", this._scene.activeCamera.maxZ)); }; } // creates a black and white random noise texture, 512x512 _createGrainTexture() { const t = new Uint8Array(1048576); for (let n = 0; n < t.length; ) { const i = Math.floor(Xt.RandomRange(0.42, 0.58) * 255); t[n++] = i, t[n++] = i, t[n++] = i, t[n++] = 255; } const r = Bo.CreateRGBATexture(t, 512, 512, this._scene, !1, !1, 2); r.name = "LensNoiseTexture", r.wrapU = We.WRAP_ADDRESSMODE, r.wrapV = We.WRAP_ADDRESSMODE, this._grainTexture = r; } } class v7e { constructor() { this.enabled = !1, this.name = "ssao2", this.texturesRequired = [6, 5]; } } const u7e = "ssao2PixelShader", l7e = `precision highp float;uniform sampler2D textureSampler;varying vec2 vUV; #ifdef SSAO float scales[16]=float[16]( 0.1, 0.11406250000000001, 0.131640625, 0.15625, 0.187890625, 0.2265625, 0.272265625, 0.325, 0.384765625, 0.4515625, 0.525390625, 0.60625, 0.694140625, 0.7890625, 0.891015625, 1.0 );uniform float near;uniform float radius;uniform sampler2D depthSampler;uniform sampler2D randomSampler;uniform sampler2D normalSampler;uniform float randTextureTiles;uniform float samplesFactor;uniform vec3 sampleSphere[SAMPLES];uniform float totalStrength;uniform float base;uniform float xViewport;uniform float yViewport;uniform mat3 depthProjection;uniform float maxZ;uniform float minZAspect;uniform vec2 texelSize;uniform mat4 projection;void main() {vec3 random=textureLod(randomSampler,vUV*randTextureTiles,0.0).rgb;float depth=textureLod(depthSampler,vUV,0.0).r;float depthSign=depth/abs(depth);depth=depth*depthSign;vec3 normal=textureLod(normalSampler,vUV,0.0).rgb;float occlusion=0.0;float correctedRadius=min(radius,minZAspect*depth/near);vec3 vViewRay=vec3((vUV.x*2.0-1.0)*xViewport,(vUV.y*2.0-1.0)*yViewport,depthSign);vec3 vDepthFactor=depthProjection*vec3(1.0,1.0,depth);vec3 origin=vViewRay*vDepthFactor;vec3 rvec=random*2.0-1.0;rvec.z=0.0;float dotProduct=dot(rvec,normal);rvec=1.0-abs(dotProduct)>1e-2 ? rvec : vec3(-rvec.y,0.0,rvec.x);vec3 tangent=normalize(rvec-normal*dot(rvec,normal));vec3 bitangent=cross(normal,tangent);mat3 tbn=mat3(tangent,bitangent,normal);float difference;for (int i=0; i1.0 || offset.y>1.0) {continue;} float sampleDepth=abs(textureLod(depthSampler,offset.xy,0.0).r);difference=depthSign*samplePosition.z-sampleDepth;float rangeCheck=1.0-smoothstep(correctedRadius*0.5,correctedRadius,difference);occlusion+=step(EPSILON,difference)*rangeCheck;} occlusion=occlusion*(1.0-smoothstep(maxZ*0.75,maxZ,depth));float ao=1.0-totalStrength*occlusion*samplesFactor;float result=clamp(ao+base,0.0,1.0);gl_FragColor=vec4(vec3(result),1.0);} #endif #ifdef BLUR uniform float outSize;uniform float soften;uniform float tolerance;uniform int samples; #ifndef BLUR_BYPASS uniform sampler2D depthSampler; #ifdef BLUR_LEGACY #define inline float blur13Bilateral(sampler2D image,vec2 uv,vec2 step) {float result=0.0;vec2 off1=vec2(1.411764705882353)*step;vec2 off2=vec2(3.2941176470588234)*step;vec2 off3=vec2(5.176470588235294)*step;float compareDepth=abs(textureLod(depthSampler,uv,0.0).r);float sampleDepth;float weight;float weightSum=30.0;result+=textureLod(image,uv,0.0).r*30.0;sampleDepth=abs(textureLod(depthSampler,uv+off1,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+= weight;result+=textureLod(image,uv+off1,0.0).r*weight;sampleDepth=abs(textureLod(depthSampler,uv-off1,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+= weight;result+=textureLod(image,uv-off1,0.0).r*weight;sampleDepth=abs(textureLod(depthSampler,uv+off2,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+=weight;result+=textureLod(image,uv+off2,0.0).r*weight;sampleDepth=abs(textureLod(depthSampler,uv-off2,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+=weight;result+=textureLod(image,uv-off2,0.0).r*weight;sampleDepth=abs(textureLod(depthSampler,uv+off3,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+=weight;result+=textureLod(image,uv+off3,0.0).r*weight;sampleDepth=abs(textureLod(depthSampler,uv-off3,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+=weight;result+=textureLod(image,uv-off3,0.0).r*weight;return result/weightSum;} #endif #endif void main() {float result=0.0; #ifdef BLUR_BYPASS result=textureLod(textureSampler,vUV,0.0).r; #else #ifdef BLUR_H vec2 step=vec2(1.0/outSize,0.0); #else vec2 step=vec2(0.0,1.0/outSize); #endif #ifdef BLUR_LEGACY result=blur13Bilateral(textureSampler,vUV,step); #else float compareDepth=abs(textureLod(depthSampler,vUV,0.0).r);float weightSum=0.0;for (int i=-samples; i this._originalColorPostProcess, !0)), this.addEffect(new gs(t.getEngine(), this.SSAORenderEffect, () => this._ssaoPostProcess, !0)), this.addEffect(new gs(t.getEngine(), this.SSAOBlurHRenderEffect, () => this._blurHPostProcess, !0)), this.addEffect(new gs(t.getEngine(), this.SSAOBlurVRenderEffect, () => this._blurVPostProcess, !0)), this.addEffect(new gs(t.getEngine(), this.SSAOCombineRenderEffect, () => this._ssaoCombinePostProcess, !0)), t.postProcessRenderPipelineManager.addPipeline(this), n && t.postProcessRenderPipelineManager.attachCamerasToRenderPipeline(e, n); } // Public Methods /** * Get the class name * @returns "SSAO2RenderingPipeline" */ getClassName() { return "SSAO2RenderingPipeline"; } /** * Removes the internal pipeline assets and detaches the pipeline from the scene cameras * @param disableGeometryBufferRenderer */ dispose(e = !1) { for (let t = 0; t < this._scene.cameras.length; t++) { const r = this._scene.cameras[t]; this._originalColorPostProcess.dispose(r), this._ssaoPostProcess.dispose(r), this._blurHPostProcess.dispose(r), this._blurVPostProcess.dispose(r), this._ssaoCombinePostProcess.dispose(r); } this._randomTexture.dispose(), e && this._scene.disableGeometryBufferRenderer(), this._scene.postProcessRenderPipelineManager.detachCamerasFromRenderPipeline(this._name, this._scene.cameras), super.dispose(); } // Private Methods /** @internal */ _rebuild() { super._rebuild(); } _getSamplersForBlur(e) { return e ? ["textureSampler"] : ["textureSampler", "depthSampler"]; } _getDefinesForBlur(e, t) { let r = `#define BLUR `; return t && (r += `#define BLUR_BYPASS `), e || (r += `#define BLUR_LEGACY `), { h: r + `#define BLUR_H `, v: r }; } _createBlurPostProcess(e, t, r) { const n = this._getDefinesForBlur(this.expensiveBlur, this.bypassBlur), i = this._getSamplersForBlur(this.bypassBlur); this._blurHPostProcess = this._createBlurFilter("BlurH", i, e, n.h, r, !0), this._blurVPostProcess = this._createBlurFilter("BlurV", i, t, n.v, r, !1); } _createBlurFilter(e, t, r, n, i, s) { const a = new kr(e, "ssao2", ["outSize", "samples", "soften", "tolerance"], t, r, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1, n, i); return a.onApply = (f) => { if (!this._scene.activeCamera) return; const o = s ? this._ssaoCombinePostProcess.width : this._ssaoCombinePostProcess.height, d = s ? this._originalColorPostProcess.width : this._originalColorPostProcess.height; f.setFloat("outSize", o > 0 ? o : d), f.setInt("samples", this.bilateralSamples), f.setFloat("soften", this.bilateralSoften), f.setFloat("tolerance", this.bilateralTolerance), this._geometryBufferRenderer ? f.setTexture("depthSampler", this._geometryBufferRenderer.getGBuffer().textures[0]) : this._prePassRenderer && f.setTexture("depthSampler", this._prePassRenderer.getRenderTarget().textures[this._prePassRenderer.getIndex(5)]); }, a.samples = this.textureSamples, a; } //Van der Corput radical inverse _radicalInverse_VdC(e) { return this._bits[0] = e, this._bits[0] = (this._bits[0] << 16 | this._bits[0] >> 16) >>> 0, this._bits[0] = (this._bits[0] & 1431655765) << 1 | (this._bits[0] & 2863311530) >>> 1 >>> 0, this._bits[0] = (this._bits[0] & 858993459) << 2 | (this._bits[0] & 3435973836) >>> 2 >>> 0, this._bits[0] = (this._bits[0] & 252645135) << 4 | (this._bits[0] & 4042322160) >>> 4 >>> 0, this._bits[0] = (this._bits[0] & 16711935) << 8 | (this._bits[0] & 4278255360) >>> 8 >>> 0, this._bits[0] * 23283064365386963e-26; } _hammersley(e, t) { return [e / t, this._radicalInverse_VdC(e)]; } _hemisphereSample_uniform(e, t) { const r = t * 2 * Math.PI, n = 1 - e * 0.85, i = Math.sqrt(1 - n * n); return new S(Math.cos(r) * i, Math.sin(r) * i, n); } _generateHemisphere() { const e = this.samples, t = []; let r, n = 0; for (; n < e; ) { if (e < 16) r = this._hemisphereSample_uniform(Math.random(), Math.random()); else { const i = this._hammersley(n, e); r = this._hemisphereSample_uniform(i[0], i[1]); } t.push(r.x, r.y, r.z), n++; } return t; } _getDefinesForSSAO() { return `#define SSAO #define SAMPLES ${this.samples} #define EPSILON ${this.epsilon.toFixed(4)}`; } _createSSAOPostProcess(e, t) { this._sampleSphere = this._generateHemisphere(); const r = this._getDefinesForSSAO(), n = ["randomSampler", "depthSampler", "normalSampler"]; this._ssaoPostProcess = new kr("ssao2", "ssao2", [ "sampleSphere", "samplesFactor", "randTextureTiles", "totalStrength", "radius", "base", "range", "projection", "near", "texelSize", "xViewport", "yViewport", "maxZ", "minZAspect", "depthProjection" ], n, e, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1, r, t), this._ssaoPostProcess.onApply = (i) => { var s, a, f, o; if (this._scene.activeCamera) { if (i.setArray3("sampleSphere", this._sampleSphere), i.setFloat("randTextureTiles", 32), i.setFloat("samplesFactor", 1 / this.samples), i.setFloat("totalStrength", this.totalStrength), i.setFloat2("texelSize", 1 / this._ssaoPostProcess.width, 1 / this._ssaoPostProcess.height), i.setFloat("radius", this.radius), i.setFloat("maxZ", this.maxZ), i.setFloat("minZAspect", this.minZAspect), i.setFloat("base", this.base), i.setFloat("near", this._scene.activeCamera.minZ), this._scene.activeCamera.mode === Tr.PERSPECTIVE_CAMERA) i.setMatrix3x3("depthProjection", H9.PERSPECTIVE_DEPTH_PROJECTION), i.setFloat("xViewport", Math.tan(this._scene.activeCamera.fov / 2) * this._scene.getEngine().getAspectRatio(this._scene.activeCamera, !0)), i.setFloat("yViewport", Math.tan(this._scene.activeCamera.fov / 2)); else { const d = this._scene.getEngine().getRenderWidth() / 2, v = this._scene.getEngine().getRenderHeight() / 2, u = (s = this._scene.activeCamera.orthoLeft) !== null && s !== void 0 ? s : -d, l = (a = this._scene.activeCamera.orthoRight) !== null && a !== void 0 ? a : d, P = (f = this._scene.activeCamera.orthoBottom) !== null && f !== void 0 ? f : -v, p = (o = this._scene.activeCamera.orthoTop) !== null && o !== void 0 ? o : v; i.setMatrix3x3("depthProjection", H9.ORTHO_DEPTH_PROJECTION), i.setFloat("xViewport", (l - u) * 0.5), i.setFloat("yViewport", (p - P) * 0.5); } i.setMatrix("projection", this._scene.getProjectionMatrix()), this._geometryBufferRenderer ? (i.setTexture("depthSampler", this._geometryBufferRenderer.getGBuffer().textures[0]), i.setTexture("normalSampler", this._geometryBufferRenderer.getGBuffer().textures[1])) : this._prePassRenderer && (i.setTexture("depthSampler", this._prePassRenderer.getRenderTarget().textures[this._prePassRenderer.getIndex(5)]), i.setTexture("normalSampler", this._prePassRenderer.getRenderTarget().textures[this._prePassRenderer.getIndex(6)])), i.setTexture("randomSampler", this._randomTexture); } }, this._ssaoPostProcess.samples = this.textureSamples, this._forceGeometryBuffer || (this._ssaoPostProcess._prePassEffectConfiguration = new v7e()); } _createSSAOCombinePostProcess(e, t) { this._ssaoCombinePostProcess = new kr("ssaoCombine", "ssaoCombine", [], ["originalColor", "viewport"], e, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1, void 0, t), this._ssaoCombinePostProcess.onApply = (r) => { const n = this._scene.activeCamera.viewport; r.setVector4("viewport", ue.Vector4[0].copyFromFloats(n.x, n.y, n.width, n.height)), r.setTextureFromPostProcessOutput("originalColor", this._originalColorPostProcess); }, this._ssaoCombinePostProcess.samples = this.textureSamples; } _createRandomTexture() { const t = new Uint8Array(65536), r = at.Zero(); for (let i = 0; i < t.length; ) r.set(Xt.RandomRange(0, 1), Xt.RandomRange(0, 1)).normalize().scaleInPlace(255), t[i++] = Math.floor(r.x), t[i++] = Math.floor(r.y), t[i++] = 0, t[i++] = 255; const n = Bo.CreateRGBATexture(t, 128, 128, this._scene, !1, !1, 2); n.name = "SSAORandomTexture", n.wrapU = We.WRAP_ADDRESSMODE, n.wrapV = We.WRAP_ADDRESSMODE, this._randomTexture = n; } /** * Serialize the rendering pipeline (Used when exporting) * @returns the serialized object */ serialize() { const e = jt.Serialize(this); return e.customType = "SSAO2RenderingPipeline", e; } /** * Parse the serialized pipeline * @param source Source pipeline. * @param scene The scene to load the pipeline to. * @param rootUrl The URL of the serialized pipeline. * @returns An instantiated pipeline from the serialized object. */ static Parse(e, t, r) { return jt.Parse(() => new H9(e._name, t, e._ratio, void 0, e._forceGeometryBuffer, e._textureType), e, t, r); } } H9.ORTHO_DEPTH_PROJECTION = [1, 0, 0, 0, 1, 0, 0, 0, 1]; H9.PERSPECTIVE_DEPTH_PROJECTION = [0, 0, 0, 0, 0, 0, 1, 1, 1]; C([ M() ], H9.prototype, "totalStrength", void 0); C([ M() ], H9.prototype, "maxZ", void 0); C([ M() ], H9.prototype, "minZAspect", void 0); C([ M("epsilon") ], H9.prototype, "_epsilon", void 0); C([ M("samples") ], H9.prototype, "_samples", void 0); C([ M("textureSamples") ], H9.prototype, "_textureSamples", void 0); C([ M() ], H9.prototype, "_forceGeometryBuffer", void 0); C([ M() ], H9.prototype, "_ratio", void 0); C([ M() ], H9.prototype, "_textureType", void 0); C([ M() ], H9.prototype, "radius", void 0); C([ M() ], H9.prototype, "base", void 0); C([ M("bypassBlur") ], H9.prototype, "_bypassBlur", void 0); C([ M("expensiveBlur") ], H9.prototype, "_expensiveBlur", void 0); C([ M() ], H9.prototype, "bilateralSamples", void 0); C([ M() ], H9.prototype, "bilateralSoften", void 0); C([ M() ], H9.prototype, "bilateralTolerance", void 0); Ue("BABYLON.SSAO2RenderingPipeline", H9); const p7e = "ssaoPixelShader", h7e = `uniform sampler2D textureSampler;varying vec2 vUV; #ifdef SSAO uniform sampler2D randomSampler;uniform float randTextureTiles;uniform float samplesFactor;uniform vec3 sampleSphere[SAMPLES];uniform float totalStrength;uniform float radius;uniform float area;uniform float fallOff;uniform float base;vec3 normalFromDepth(float depth,vec2 coords) {vec2 offset1=vec2(0.0,radius);vec2 offset2=vec2(radius,0.0);float depth1=texture2D(textureSampler,coords+offset1).r;float depth2=texture2D(textureSampler,coords+offset2).r;vec3 p1=vec3(offset1,depth1-depth);vec3 p2=vec3(offset2,depth2-depth);vec3 normal=cross(p1,p2);normal.z=-normal.z;return normalize(normal);} void main() {vec3 random=normalize(texture2D(randomSampler,vUV*randTextureTiles).rgb);float depth=texture2D(textureSampler,vUV).r;vec3 position=vec3(vUV,depth);vec3 normal=normalFromDepth(depth,vUV);float radiusDepth=radius/depth;float occlusion=0.0;vec3 ray;vec3 hemiRay;float occlusionDepth;float difference;for (int i=0; i this._originalColorPostProcess, !0)), this.addEffect(new gs(t.getEngine(), this.SSAORenderEffect, () => this._ssaoPostProcess, !0)), this.addEffect(new gs(t.getEngine(), this.SSAOBlurHRenderEffect, () => this._blurHPostProcess, !0)), this.addEffect(new gs(t.getEngine(), this.SSAOBlurVRenderEffect, () => this._blurVPostProcess, !0)), this.addEffect(new gs(t.getEngine(), this.SSAOCombineRenderEffect, () => this._ssaoCombinePostProcess, !0)), t.postProcessRenderPipelineManager.addPipeline(this), n && t.postProcessRenderPipelineManager.attachCamerasToRenderPipeline(e, n); } /** * @internal */ _attachCameras(e, t) { super._attachCameras(e, t); for (const r of this._cameras) this._scene.enableDepthRenderer(r).getDepthMap(); } // Public Methods /** * Get the class name * @returns "SSAORenderingPipeline" */ getClassName() { return "SSAORenderingPipeline"; } /** * Removes the internal pipeline assets and detaches the pipeline from the scene cameras * @param disableDepthRender */ dispose(e = !1) { for (let t = 0; t < this._scene.cameras.length; t++) { const r = this._scene.cameras[t]; this._originalColorPostProcess.dispose(r), this._ssaoPostProcess.dispose(r), this._blurHPostProcess.dispose(r), this._blurVPostProcess.dispose(r), this._ssaoCombinePostProcess.dispose(r); } this._randomTexture.dispose(), e && this._scene.disableDepthRenderer(), this._scene.postProcessRenderPipelineManager.detachCamerasFromRenderPipeline(this._name, this._scene.cameras), super.dispose(); } // Private Methods _createBlurPostProcess(e) { this._blurHPostProcess = new c9("BlurH", new at(1, 0), 16, e, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1, 0), this._blurVPostProcess = new c9("BlurV", new at(0, 1), 16, e, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1, 0), this._blurHPostProcess.onActivateObservable.add(() => { const r = this._blurHPostProcess.width / this._scene.getEngine().getRenderWidth(); this._blurHPostProcess.kernel = 16 * r; }), this._blurVPostProcess.onActivateObservable.add(() => { const r = this._blurVPostProcess.height / this._scene.getEngine().getRenderHeight(); this._blurVPostProcess.kernel = 16 * r; }); } /** @internal */ _rebuild() { this._firstUpdate = !0, super._rebuild(); } _createSSAOPostProcess(e) { const r = [ 0.5381, 0.1856, -0.4319, 0.1379, 0.2486, 0.443, 0.3371, 0.5679, -57e-4, -0.6999, -0.0451, -19e-4, 0.0689, -0.1598, -0.8547, 0.056, 69e-4, -0.1843, -0.0146, 0.1402, 0.0762, 0.01, -0.1924, -0.0344, -0.3577, -0.5301, -0.4358, -0.3169, 0.1063, 0.0158, 0.0103, -0.5869, 46e-4, -0.0897, -0.494, 0.3287, 0.7119, -0.0154, -0.0918, -0.0533, 0.0596, -0.5411, 0.0352, -0.0631, 0.546, -0.4776, 0.2847, -0.0271 ], n = 1 / 16; this._ssaoPostProcess = new kr("ssao", "ssao", ["sampleSphere", "samplesFactor", "randTextureTiles", "totalStrength", "radius", "area", "fallOff", "base", "range", "viewport"], ["randomSampler"], e, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1, `#define SAMPLES 16 #define SSAO`), this._ssaoPostProcess.externalTextureSamplerBinding = !0, this._ssaoPostProcess.onApply = (i) => { this._firstUpdate && (i.setArray3("sampleSphere", r), i.setFloat("samplesFactor", n), i.setFloat("randTextureTiles", 4)), i.setFloat("totalStrength", this.totalStrength), i.setFloat("radius", this.radius), i.setFloat("area", this.area), i.setFloat("fallOff", this.fallOff), i.setFloat("base", this.base), i.setTexture("textureSampler", this._scene.enableDepthRenderer(this._scene.activeCamera).getDepthMap()), i.setTexture("randomSampler", this._randomTexture); }; } _createSSAOCombinePostProcess(e) { this._ssaoCombinePostProcess = new kr("ssaoCombine", "ssaoCombine", [], ["originalColor", "viewport"], e, null, We.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), !1), this._ssaoCombinePostProcess.onApply = (t) => { t.setVector4("viewport", ue.Vector4[0].copyFromFloats(0, 0, 1, 1)), t.setTextureFromPostProcess("originalColor", this._originalColorPostProcess); }; } _createRandomTexture() { const t = new Uint8Array(1048576); for (let n = 0; n < t.length; ) t[n++] = Math.floor(Math.max(0, Xt.RandomRange(-1, 1)) * 255), t[n++] = Math.floor(Math.max(0, Xt.RandomRange(-1, 1)) * 255), t[n++] = Math.floor(Math.max(0, Xt.RandomRange(-1, 1)) * 255), t[n++] = 255; const r = Bo.CreateRGBATexture(t, 512, 512, this._scene, !1, !1, 2); r.name = "SSAORandomTexture", r.wrapU = We.WRAP_ADDRESSMODE, r.wrapV = We.WRAP_ADDRESSMODE, this._randomTexture = r; } } C([ M() ], pU.prototype, "totalStrength", void 0); C([ M() ], pU.prototype, "radius", void 0); C([ M() ], pU.prototype, "area", void 0); C([ M() ], pU.prototype, "fallOff", void 0); C([ M() ], pU.prototype, "base", void 0); class H7e { constructor() { this.enabled = !1, this.name = "screenSpaceReflections", this.texturesRequired = [6, 3, 1]; } } const g7e = "screenSpaceReflectionPixelShader", X7e = `uniform sampler2D textureSampler; #ifdef SSR_SUPPORTED uniform sampler2D reflectivitySampler;uniform sampler2D normalSampler;uniform sampler2D positionSampler; #endif uniform mat4 view;uniform mat4 projection;uniform float stepSize;uniform float strength;uniform float threshold;uniform float roughnessFactor;uniform float reflectionSpecularFalloffExponent;varying vec2 vUV; #ifdef SSR_SUPPORTED struct ReflectionInfo {vec3 color;vec4 coords;};/** * According to specular,see https: */ vec3 fresnelSchlick(float cosTheta,vec3 F0) {return F0+(1.0-F0)*pow(1.0-cosTheta,5.0);} /** * Once the pixel's coordinates has been found,let's adjust (smooth) a little bit * by sampling multiple reflection pixels. */ ReflectionInfo smoothReflectionInfo(vec3 dir,vec3 hitCoord) {ReflectionInfo info;info.color=vec3(0.0);vec4 projectedCoord;float sampledDepth;for(int i=0; i0.0) hitCoord-=dir;else hitCoord+=dir;info.color+=texture2D(textureSampler,projectedCoord.xy).rgb;} projectedCoord=projection*vec4(hitCoord,1.0);projectedCoord.xy/=projectedCoord.w;projectedCoord.xy=0.5*projectedCoord.xy+vec2(0.5);info.coords=vec4(projectedCoord.xy,sampledDepth,1.0);info.color+=texture2D(textureSampler,projectedCoord.xy).rgb;info.color/=float(SMOOTH_STEPS+1);return info;} /** * Tests the given world position (hitCoord) according to the given reflection vector (dir) * until it finds a collision (means that depth is enough close to say "it's the pixel to sample!"). */ ReflectionInfo getReflectionInfo(vec3 dir,vec3 hitCoord) {ReflectionInfo info;vec4 projectedCoord;float sampledDepth;dir*=stepSize;for(int i=0; i { const u = this._geometryBufferRenderer, l = this._prePassRenderer; if (!l && !u) return; if (u) { const H = u.getTextureIndex(po.POSITION_TEXTURE_TYPE), T = u.getTextureIndex(po.REFLECTIVITY_TEXTURE_TYPE); v.setTexture("normalSampler", u.getGBuffer().textures[1]), v.setTexture("positionSampler", u.getGBuffer().textures[H]), v.setTexture("reflectivitySampler", u.getGBuffer().textures[T]); } else if (l) { const H = l.getIndex(1), T = l.getIndex(3), q = l.getIndex(6); v.setTexture("normalSampler", l.getRenderTarget().textures[q]), v.setTexture("positionSampler", l.getRenderTarget().textures[H]), v.setTexture("reflectivitySampler", l.getRenderTarget().textures[T]); } const P = t.activeCamera; if (!P) return; const p = P.getViewMatrix(!0), c = P.getProjectionMatrix(!0); v.setMatrix("projection", c), v.setMatrix("view", p), v.setFloat("threshold", this.threshold), v.setFloat("reflectionSpecularFalloffExponent", this.reflectionSpecularFalloffExponent), v.setFloat("strength", this.strength), v.setFloat("stepSize", this.step), v.setFloat("roughnessFactor", this.roughnessFactor); }, this._isSceneRightHanded = t.useRightHandedSystem; } /** * Gets whether or not smoothing reflections is enabled. * Enabling smoothing will require more GPU power and can generate a drop in FPS. */ get enableSmoothReflections() { return this._enableSmoothReflections; } /** * Sets whether or not smoothing reflections is enabled. * Enabling smoothing will require more GPU power and can generate a drop in FPS. */ set enableSmoothReflections(e) { e !== this._enableSmoothReflections && (this._enableSmoothReflections = e, this._updateEffectDefines()); } /** * Gets the number of samples taken while computing reflections. More samples count is high, * more the post-process wil require GPU power and can generate a drop in FPS. Basically in interval [25, 100]. */ get reflectionSamples() { return this._reflectionSamples; } /** * Sets the number of samples taken while computing reflections. More samples count is high, * more the post-process wil require GPU power and can generate a drop in FPS. Basically in interval [25, 100]. */ set reflectionSamples(e) { e !== this._reflectionSamples && (this._reflectionSamples = e, this._updateEffectDefines()); } /** * Gets the number of samples taken while smoothing reflections. More samples count is high, * more the post-process will require GPU power and can generate a drop in FPS. * Default value (5.0) work pretty well in all cases but can be adjusted. */ get smoothSteps() { return this._smoothSteps; } /* * Sets the number of samples taken while smoothing reflections. More samples count is high, * more the post-process will require GPU power and can generate a drop in FPS. * Default value (5.0) work pretty well in all cases but can be adjusted. */ set smoothSteps(e) { e !== this._smoothSteps && (this._smoothSteps = e, this._updateEffectDefines()); } _updateEffectDefines() { const e = []; (this._geometryBufferRenderer || this._prePassRenderer) && e.push("#define SSR_SUPPORTED"), this._enableSmoothReflections && e.push("#define ENABLE_SMOOTH_REFLECTIONS"), this._isSceneRightHanded && e.push("#define RIGHT_HANDED_SCENE"), e.push("#define REFLECTION_SAMPLES " + (this._reflectionSamples >> 0)), e.push("#define SMOOTH_STEPS " + (this._smoothSteps >> 0)), this.updateEffect(e.join(` `)); } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new jp(e.name, r, e.options, t, e.renderTargetSamplingMode, r.getEngine(), e.textureType, e.reusable), e, r, n); } } C([ M() ], jp.prototype, "threshold", void 0); C([ M() ], jp.prototype, "strength", void 0); C([ M() ], jp.prototype, "reflectionSpecularFalloffExponent", void 0); C([ M() ], jp.prototype, "step", void 0); C([ M() ], jp.prototype, "roughnessFactor", void 0); C([ M() ], jp.prototype, "enableSmoothReflections", null); C([ M() ], jp.prototype, "reflectionSamples", null); C([ M() ], jp.prototype, "smoothSteps", null); Ue("BABYLON.ScreenSpaceReflectionPostProcess", jp); const T7e = "standardPixelShader", q7e = `uniform sampler2D textureSampler;varying vec2 vUV; #define CUSTOM_FRAGMENT_DEFINITIONS #if defined(PASS_POST_PROCESS) void main(void) {vec4 color=texture2D(textureSampler,vUV);gl_FragColor=color;} #endif #if defined(DOWN_SAMPLE_X4) uniform vec2 dsOffsets[16];void main(void) {vec4 average=vec4(0.0,0.0,0.0,0.0);average=texture2D(textureSampler,vUV+dsOffsets[0]);average+=texture2D(textureSampler,vUV+dsOffsets[1]);average+=texture2D(textureSampler,vUV+dsOffsets[2]);average+=texture2D(textureSampler,vUV+dsOffsets[3]);average+=texture2D(textureSampler,vUV+dsOffsets[4]);average+=texture2D(textureSampler,vUV+dsOffsets[5]);average+=texture2D(textureSampler,vUV+dsOffsets[6]);average+=texture2D(textureSampler,vUV+dsOffsets[7]);average+=texture2D(textureSampler,vUV+dsOffsets[8]);average+=texture2D(textureSampler,vUV+dsOffsets[9]);average+=texture2D(textureSampler,vUV+dsOffsets[10]);average+=texture2D(textureSampler,vUV+dsOffsets[11]);average+=texture2D(textureSampler,vUV+dsOffsets[12]);average+=texture2D(textureSampler,vUV+dsOffsets[13]);average+=texture2D(textureSampler,vUV+dsOffsets[14]);average+=texture2D(textureSampler,vUV+dsOffsets[15]);average/=16.0;gl_FragColor=average;} #endif #if defined(BRIGHT_PASS) uniform vec2 dsOffsets[4];uniform float brightThreshold;void main(void) {vec4 average=vec4(0.0,0.0,0.0,0.0);average=texture2D(textureSampler,vUV+vec2(dsOffsets[0].x,dsOffsets[0].y));average+=texture2D(textureSampler,vUV+vec2(dsOffsets[1].x,dsOffsets[1].y));average+=texture2D(textureSampler,vUV+vec2(dsOffsets[2].x,dsOffsets[2].y));average+=texture2D(textureSampler,vUV+vec2(dsOffsets[3].x,dsOffsets[3].y));average*=0.25;float luminance=length(average.rgb);if (luminanceshadowPixelDepth) accumFog+=sunColor*computeScattering(dot(rayDirection,sunDirection));currentPosition+=stepL;} accumFog/=NB_STEPS;vec3 color=accumFog*scatteringPower;gl_FragColor=vec4(color*exp(color) ,1.0);} #endif #if defined(VLSMERGE) uniform sampler2D originalSampler;void main(void) {gl_FragColor=texture2D(originalSampler,vUV)+texture2D(textureSampler,vUV);} #endif #if defined(LUMINANCE) uniform vec2 lumOffsets[4];void main() {float average=0.0;vec4 color=vec4(0.0);float maximum=-1e20;vec3 weight=vec3(0.299,0.587,0.114);for (int i=0; i<4; i++) {color=texture2D(textureSampler,vUV+ lumOffsets[i]);float GreyValue=dot(color.rgb,vec3(0.33,0.33,0.33)); #ifdef WEIGHTED_AVERAGE float GreyValue=dot(color.rgb,weight); #endif #ifdef BRIGHTNESS float GreyValue=max(color.r,max(color.g,color.b)); #endif #ifdef HSL_COMPONENT float GreyValue=0.5*(max(color.r,max(color.g,color.b))+min(color.r,min(color.g,color.b))); #endif #ifdef MAGNITUDE float GreyValue=length(color.rgb); #endif maximum=max(maximum,GreyValue);average+=(0.25*log(1e-5+GreyValue));} average=exp(average);gl_FragColor=vec4(average,maximum,0.0,1.0);} #endif #if defined(LUMINANCE_DOWN_SAMPLE) uniform vec2 dsOffsets[9];uniform float halfDestPixelSize; #ifdef FINAL_DOWN_SAMPLER #include #endif void main() {vec4 color=vec4(0.0);float average=0.0;for (int i=0; i<9; i++) {color=texture2D(textureSampler,vUV+vec2(halfDestPixelSize,halfDestPixelSize)+dsOffsets[i]);average+=color.r;} average/=9.0; #ifdef FINAL_DOWN_SAMPLER gl_FragColor=pack(average); #else gl_FragColor=vec4(average,average,0.0,1.0); #endif } #endif #if defined(HDR) uniform sampler2D textureAdderSampler;uniform float averageLuminance;void main() {vec4 color=texture2D(textureAdderSampler,vUV); #ifndef AUTO_EXPOSURE vec4 adjustedColor=color/averageLuminance;color=adjustedColor;color.a=1.0; #endif gl_FragColor=color;} #endif #if defined(LENS_FLARE) #define GHOSTS 3 uniform sampler2D lensColorSampler;uniform float strength;uniform float ghostDispersal;uniform float haloWidth;uniform vec2 resolution;uniform float distortionStrength;float hash(vec2 p) {float h=dot(p,vec2(127.1,311.7));return -1.0+2.0*fract(sin(h)*43758.5453123);} float noise(in vec2 p) {vec2 i=floor(p);vec2 f=fract(p);vec2 u=f*f*(3.0-2.0*f);return mix(mix(hash(i+vec2(0.0,0.0)), hash(i+vec2(1.0,0.0)),u.x), mix(hash(i+vec2(0.0,1.0)), hash(i+vec2(1.0,1.0)),u.x),u.y);} float fbm(vec2 p) {float f=0.0;f+=0.5000*noise(p); p*=2.02;f+=0.2500*noise(p); p*=2.03;f+=0.1250*noise(p); p*=2.01;f+=0.0625*noise(p); p*=2.04;f/=0.9375;return f;} vec3 pattern(vec2 uv) {vec2 p=-1.0+2.0*uv;float p2=dot(p,p);float f=fbm(vec2(15.0*p2))/2.0;float r=0.2+0.6*sin(12.5*length(uv-vec2(0.5)));float g=0.2+0.6*sin(20.5*length(uv-vec2(0.5)));float b=0.2+0.6*sin(17.2*length(uv-vec2(0.5)));return (1.0-f)*vec3(r,g,b);} float luminance(vec3 color) {return dot(color.rgb,vec3(0.2126,0.7152,0.0722));} vec4 textureDistorted(sampler2D tex,vec2 texcoord,vec2 direction,vec3 distortion) {return vec4( texture2D(tex,texcoord+direction*distortion.r).r, texture2D(tex,texcoord+direction*distortion.g).g, texture2D(tex,texcoord+direction*distortion.b).b, 1.0 );} void main(void) {vec2 uv=-vUV+vec2(1.0);vec2 ghostDir=(vec2(0.5)-uv)*ghostDispersal;vec2 texelSize=1.0/resolution;vec3 distortion=vec3(-texelSize.x*distortionStrength,0.0,texelSize.x*distortionStrength);vec4 result=vec4(0.0);float ghostIndice=1.0;for (int i=0; i=nSamples) break;vec2 offset1=vUV+velocity*(float(i)/float(nSamples-1)-0.5);result+=texture2D(textureSampler,offset1);} gl_FragColor=result/float(nSamples);} #endif `; Le.ShadersStore[T7e] = q7e; class Ss extends SD { /** * Gets the overall exposure used by the pipeline */ get exposure() { return this._fixedExposure; } /** * Sets the overall exposure used by the pipeline */ set exposure(e) { this._fixedExposure = e, this._currentExposure = e; } /** * Gets whether or not the exposure of the overall pipeline should be automatically adjusted by the HDR post-process */ get hdrAutoExposure() { return this._hdrAutoExposure; } /** * Sets whether or not the exposure of the overall pipeline should be automatically adjusted by the HDR post-process */ set hdrAutoExposure(e) { if (this._hdrAutoExposure = e, this.hdrPostProcess) { const t = ["#define HDR"]; e && t.push("#define AUTO_EXPOSURE"), this.hdrPostProcess.updateEffect(t.join(` `)); } } /** * Gets how much the image is blurred by the movement while using the motion blur post-process */ get motionStrength() { return this._motionStrength; } /** * Sets how much the image is blurred by the movement while using the motion blur post-process */ set motionStrength(e) { this._motionStrength = e, this._isObjectBasedMotionBlur && this.motionBlurPostProcess && (this.motionBlurPostProcess.motionStrength = e); } /** * Gets whether or not the motion blur post-process is object based or screen based. */ get objectBasedMotionBlur() { return this._isObjectBasedMotionBlur; } /** * Sets whether or not the motion blur post-process should be object based or screen based */ set objectBasedMotionBlur(e) { const t = this._isObjectBasedMotionBlur !== e; this._isObjectBasedMotionBlur = e, t && this._buildPipeline(); } /** * @ignore * Specifies if the bloom pipeline is enabled */ get BloomEnabled() { return this._bloomEnabled; } set BloomEnabled(e) { this._bloomEnabled !== e && (this._bloomEnabled = e, this._buildPipeline()); } /** * @ignore * Specifies if the depth of field pipeline is enabled */ get DepthOfFieldEnabled() { return this._depthOfFieldEnabled; } set DepthOfFieldEnabled(e) { this._depthOfFieldEnabled !== e && (this._depthOfFieldEnabled = e, this._buildPipeline()); } /** * @ignore * Specifies if the lens flare pipeline is enabled */ get LensFlareEnabled() { return this._lensFlareEnabled; } set LensFlareEnabled(e) { this._lensFlareEnabled !== e && (this._lensFlareEnabled = e, this._buildPipeline()); } /** * @ignore * Specifies if the HDR pipeline is enabled */ get HDREnabled() { return this._hdrEnabled; } set HDREnabled(e) { this._hdrEnabled !== e && (this._hdrEnabled = e, this._buildPipeline()); } /** * @ignore * Specifies if the volumetric lights scattering effect is enabled */ get VLSEnabled() { return this._vlsEnabled; } // eslint-disable-next-line @typescript-eslint/naming-convention set VLSEnabled(e) { if (this._vlsEnabled !== e) { if (e && !this._scene.enableGeometryBufferRenderer()) { Se.Warn("Geometry renderer is not supported, cannot create volumetric lights in Standard Rendering Pipeline"); return; } this._vlsEnabled = e, this._buildPipeline(); } } /** * @ignore * Specifies if the motion blur effect is enabled */ get MotionBlurEnabled() { return this._motionBlurEnabled; } set MotionBlurEnabled(e) { this._motionBlurEnabled !== e && (this._motionBlurEnabled = e, this._buildPipeline()); } /** * Specifies if anti-aliasing is enabled */ get fxaaEnabled() { return this._fxaaEnabled; } set fxaaEnabled(e) { this._fxaaEnabled !== e && (this._fxaaEnabled = e, this._buildPipeline()); } /** * Specifies if screen space reflections are enabled. */ get screenSpaceReflectionsEnabled() { return this._screenSpaceReflectionsEnabled; } set screenSpaceReflectionsEnabled(e) { this._screenSpaceReflectionsEnabled !== e && (this._screenSpaceReflectionsEnabled = e, this._buildPipeline()); } /** * Specifies the number of steps used to calculate the volumetric lights * Typically in interval [50, 200] */ get volumetricLightStepsCount() { return this._volumetricLightStepsCount; } set volumetricLightStepsCount(e) { this.volumetricLightPostProcess && this.volumetricLightPostProcess.updateEffect(`#define VLS #define NB_STEPS ` + e.toFixed(1)), this._volumetricLightStepsCount = e; } /** * Specifies the number of samples used for the motion blur effect * Typically in interval [16, 64] */ get motionBlurSamples() { return this._motionBlurSamples; } set motionBlurSamples(e) { this.motionBlurPostProcess && (this._isObjectBasedMotionBlur ? this.motionBlurPostProcess.motionBlurSamples = e : this.motionBlurPostProcess.updateEffect(`#define MOTION_BLUR #define MAX_MOTION_SAMPLES ` + e.toFixed(1))), this._motionBlurSamples = e; } /** * Specifies MSAA sample count, setting this to 4 will provide 4x anti aliasing. (default: 1) */ get samples() { return this._samples; } set samples(e) { this._samples !== e && (this._samples = e, this._buildPipeline()); } /** * Default pipeline should be used going forward but the standard pipeline will be kept for backwards compatibility. * @constructor * @param name The rendering pipeline name * @param scene The scene linked to this pipeline * @param ratio The size of the postprocesses (0.5 means that your postprocess will have a width = canvas.width 0.5 and a height = canvas.height 0.5) * @param originalPostProcess the custom original color post-process. Must be "reusable". Can be null. * @param cameras The array of cameras that the rendering pipeline will be attached to */ constructor(e, t, r, n = null, i) { super(t.getEngine(), e), this.downSampleX4PostProcess = null, this.brightPassPostProcess = null, this.blurHPostProcesses = [], this.blurVPostProcesses = [], this.textureAdderPostProcess = null, this.volumetricLightPostProcess = null, this.volumetricLightSmoothXPostProcess = null, this.volumetricLightSmoothYPostProcess = null, this.volumetricLightMergePostProces = null, this.volumetricLightFinalPostProcess = null, this.luminancePostProcess = null, this.luminanceDownSamplePostProcesses = [], this.hdrPostProcess = null, this.textureAdderFinalPostProcess = null, this.lensFlareFinalPostProcess = null, this.hdrFinalPostProcess = null, this.lensFlarePostProcess = null, this.lensFlareComposePostProcess = null, this.motionBlurPostProcess = null, this.depthOfFieldPostProcess = null, this.fxaaPostProcess = null, this.screenSpaceReflectionPostProcess = null, this.brightThreshold = 1, this.blurWidth = 512, this.horizontalBlur = !1, this.lensTexture = null, this.volumetricLightCoefficient = 0.2, this.volumetricLightPower = 4, this.volumetricLightBlurScale = 64, this.sourceLight = null, this.hdrMinimumLuminance = 1, this.hdrDecreaseRate = 0.5, this.hdrIncreaseRate = 0.5, this.lensColorTexture = null, this.lensFlareStrength = 20, this.lensFlareGhostDispersal = 1.4, this.lensFlareHaloWidth = 0.7, this.lensFlareDistortionStrength = 16, this.lensFlareBlurWidth = 512, this.lensStarTexture = null, this.lensFlareDirtTexture = null, this.depthOfFieldDistance = 10, this.depthOfFieldBlurWidth = 64, this.animations = [], this._currentDepthOfFieldSource = null, this._fixedExposure = 1, this._currentExposure = 1, this._hdrAutoExposure = !1, this._hdrCurrentLuminance = 1, this._motionStrength = 1, this._isObjectBasedMotionBlur = !1, this._camerasToBeAttached = [], this._bloomEnabled = !1, this._depthOfFieldEnabled = !1, this._vlsEnabled = !1, this._lensFlareEnabled = !1, this._hdrEnabled = !1, this._motionBlurEnabled = !1, this._fxaaEnabled = !1, this._screenSpaceReflectionsEnabled = !1, this._motionBlurSamples = 64, this._volumetricLightStepsCount = 50, this._samples = 1, this._cameras = i || t.cameras, this._cameras = this._cameras.slice(), this._camerasToBeAttached = this._cameras.slice(), this._scene = t, this._basePostProcess = n, this._ratio = r, this._floatTextureType = t.getEngine().getCaps().textureFloatRender ? 1 : 2, t.postProcessRenderPipelineManager.addPipeline(this), this._buildPipeline(); } _buildPipeline() { const e = this._ratio, t = this._scene; this._disposePostProcesses(), this._cameras !== null && (this._scene.postProcessRenderPipelineManager.detachCamerasFromRenderPipeline(this._name, this._cameras), this._cameras = this._camerasToBeAttached.slice()), this._reset(), this._screenSpaceReflectionsEnabled && (this.screenSpaceReflectionPostProcess = new jp("HDRPass", t, e, null, We.BILINEAR_SAMPLINGMODE, t.getEngine(), !1, this._floatTextureType), this.screenSpaceReflectionPostProcess.onApplyObservable.add(() => { this._currentDepthOfFieldSource = this.screenSpaceReflectionPostProcess; }), this.addEffect(new gs(t.getEngine(), "HDRScreenSpaceReflections", () => this.screenSpaceReflectionPostProcess, !0))), this._basePostProcess ? this.originalPostProcess = this._basePostProcess : this.originalPostProcess = new kr("HDRPass", "standard", [], [], e, null, We.BILINEAR_SAMPLINGMODE, t.getEngine(), !1, "#define PASS_POST_PROCESS", this._floatTextureType), this.originalPostProcess.autoClear = !this.screenSpaceReflectionPostProcess, this.originalPostProcess.onApplyObservable.add(() => { this._currentDepthOfFieldSource = this.originalPostProcess; }), this.addEffect(new gs(t.getEngine(), "HDRPassPostProcess", () => this.originalPostProcess, !0)), this._bloomEnabled && (this._createDownSampleX4PostProcess(t, e / 4), this._createBrightPassPostProcess(t, e / 4), this._createBlurPostProcesses(t, e / 4, 1), this._createTextureAdderPostProcess(t, e), this.textureAdderFinalPostProcess = new kr("HDRDepthOfFieldSource", "standard", [], [], e, null, We.BILINEAR_SAMPLINGMODE, t.getEngine(), !1, "#define PASS_POST_PROCESS", 0), this.addEffect(new gs(t.getEngine(), "HDRBaseDepthOfFieldSource", () => this.textureAdderFinalPostProcess, !0))), this._vlsEnabled && (this._createVolumetricLightPostProcess(t, e), this.volumetricLightFinalPostProcess = new kr("HDRVLSFinal", "standard", [], [], e, null, We.BILINEAR_SAMPLINGMODE, t.getEngine(), !1, "#define PASS_POST_PROCESS", 0), this.addEffect(new gs(t.getEngine(), "HDRVLSFinal", () => this.volumetricLightFinalPostProcess, !0))), this._lensFlareEnabled && (this._createLensFlarePostProcess(t, e), this.lensFlareFinalPostProcess = new kr("HDRPostLensFlareDepthOfFieldSource", "standard", [], [], e, null, We.BILINEAR_SAMPLINGMODE, t.getEngine(), !1, "#define PASS_POST_PROCESS", 0), this.addEffect(new gs(t.getEngine(), "HDRPostLensFlareDepthOfFieldSource", () => this.lensFlareFinalPostProcess, !0))), this._hdrEnabled && (this._createLuminancePostProcesses(t, this._floatTextureType), this._createHdrPostProcess(t, e), this.hdrFinalPostProcess = new kr("HDRPostHDReDepthOfFieldSource", "standard", [], [], e, null, We.BILINEAR_SAMPLINGMODE, t.getEngine(), !1, "#define PASS_POST_PROCESS", 0), this.addEffect(new gs(t.getEngine(), "HDRPostHDReDepthOfFieldSource", () => this.hdrFinalPostProcess, !0))), this._depthOfFieldEnabled && (this._createBlurPostProcesses(t, e / 2, 3, "depthOfFieldBlurWidth"), this._createDepthOfFieldPostProcess(t, e)), this._motionBlurEnabled && this._createMotionBlurPostProcess(t, e), this._fxaaEnabled && (this.fxaaPostProcess = new ym("fxaa", 1, null, We.BILINEAR_SAMPLINGMODE, t.getEngine(), !1, 0), this.addEffect(new gs(t.getEngine(), "HDRFxaa", () => this.fxaaPostProcess, !0))), this._cameras !== null && this._scene.postProcessRenderPipelineManager.attachCamerasToRenderPipeline(this._name, this._cameras), !this._enableMSAAOnFirstPostProcess(this._samples) && this._samples > 1 && Se.Warn("MSAA failed to enable, MSAA is only supported in browsers that support webGL >= 2.0"); } // Down Sample X4 Post-Process _createDownSampleX4PostProcess(e, t) { const r = new Array(32); this.downSampleX4PostProcess = new kr("HDRDownSampleX4", "standard", ["dsOffsets"], [], t, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, "#define DOWN_SAMPLE_X4", this._floatTextureType), this.downSampleX4PostProcess.onApply = (n) => { let i = 0; const s = this.downSampleX4PostProcess.width, a = this.downSampleX4PostProcess.height; for (let f = -2; f < 2; f++) for (let o = -2; o < 2; o++) r[i] = (f + 0.5) * (1 / s), r[i + 1] = (o + 0.5) * (1 / a), i += 2; n.setArray2("dsOffsets", r); }, this.addEffect(new gs(e.getEngine(), "HDRDownSampleX4", () => this.downSampleX4PostProcess, !0)); } // Brightpass Post-Process _createBrightPassPostProcess(e, t) { const r = new Array(8); this.brightPassPostProcess = new kr("HDRBrightPass", "standard", ["dsOffsets", "brightThreshold"], [], t, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, "#define BRIGHT_PASS", this._floatTextureType), this.brightPassPostProcess.onApply = (n) => { const i = 1 / this.brightPassPostProcess.width, s = 1 / this.brightPassPostProcess.height; r[0] = -0.5 * i, r[1] = 0.5 * s, r[2] = 0.5 * i, r[3] = 0.5 * s, r[4] = -0.5 * i, r[5] = -0.5 * s, r[6] = 0.5 * i, r[7] = -0.5 * s, n.setArray2("dsOffsets", r), n.setFloat("brightThreshold", this.brightThreshold); }, this.addEffect(new gs(e.getEngine(), "HDRBrightPass", () => this.brightPassPostProcess, !0)); } // Create blur H&V post-processes _createBlurPostProcesses(e, t, r, n = "blurWidth") { const i = e.getEngine(), s = new c9("HDRBlurH_" + r, new at(1, 0), this[n], t, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, this._floatTextureType), a = new c9("HDRBlurV_" + r, new at(0, 1), this[n], t, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, this._floatTextureType); s.onActivateObservable.add(() => { const f = s.width / i.getRenderWidth(); s.kernel = this[n] * f; }), a.onActivateObservable.add(() => { const f = a.height / i.getRenderHeight(); a.kernel = this.horizontalBlur ? 64 * f : this[n] * f; }), this.addEffect(new gs(e.getEngine(), "HDRBlurH" + r, () => s, !0)), this.addEffect(new gs(e.getEngine(), "HDRBlurV" + r, () => a, !0)), this.blurHPostProcesses.push(s), this.blurVPostProcesses.push(a); } // Create texture adder post-process _createTextureAdderPostProcess(e, t) { this.textureAdderPostProcess = new kr("HDRTextureAdder", "standard", ["exposure"], ["otherSampler", "lensSampler"], t, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, "#define TEXTURE_ADDER", this._floatTextureType), this.textureAdderPostProcess.onApply = (r) => { r.setTextureFromPostProcess("otherSampler", this._vlsEnabled ? this._currentDepthOfFieldSource : this.originalPostProcess), r.setTexture("lensSampler", this.lensTexture), r.setFloat("exposure", this._currentExposure), this._currentDepthOfFieldSource = this.textureAdderFinalPostProcess; }, this.addEffect(new gs(e.getEngine(), "HDRTextureAdder", () => this.textureAdderPostProcess, !0)); } _createVolumetricLightPostProcess(e, t) { const r = e.enableGeometryBufferRenderer(); r.enablePosition = !0; const n = r.getGBuffer(); this.volumetricLightPostProcess = new kr("HDRVLS", "standard", ["shadowViewProjection", "cameraPosition", "sunDirection", "sunColor", "scatteringCoefficient", "scatteringPower", "depthValues"], ["shadowMapSampler", "positionSampler"], t / 8, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, `#define VLS #define NB_STEPS ` + this._volumetricLightStepsCount.toFixed(1)); const i = at.Zero(); this.volumetricLightPostProcess.onApply = (s) => { if (this.sourceLight && this.sourceLight.getShadowGenerator() && this._scene.activeCamera) { const a = this.sourceLight.getShadowGenerator(); s.setTexture("shadowMapSampler", a.getShadowMap()), s.setTexture("positionSampler", n.textures[2]), s.setColor3("sunColor", this.sourceLight.diffuse), s.setVector3("sunDirection", this.sourceLight.getShadowDirection()), s.setVector3("cameraPosition", this._scene.activeCamera.globalPosition), s.setMatrix("shadowViewProjection", a.getTransformMatrix()), s.setFloat("scatteringCoefficient", this.volumetricLightCoefficient), s.setFloat("scatteringPower", this.volumetricLightPower), i.x = this.sourceLight.getDepthMinZ(this._scene.activeCamera), i.y = this.sourceLight.getDepthMaxZ(this._scene.activeCamera), s.setVector2("depthValues", i); } }, this.addEffect(new gs(e.getEngine(), "HDRVLS", () => this.volumetricLightPostProcess, !0)), this._createBlurPostProcesses(e, t / 4, 0, "volumetricLightBlurScale"), this.volumetricLightMergePostProces = new kr("HDRVLSMerge", "standard", [], ["originalSampler"], t, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, "#define VLSMERGE"), this.volumetricLightMergePostProces.onApply = (s) => { s.setTextureFromPostProcess("originalSampler", this._bloomEnabled ? this.textureAdderFinalPostProcess : this.originalPostProcess), this._currentDepthOfFieldSource = this.volumetricLightFinalPostProcess; }, this.addEffect(new gs(e.getEngine(), "HDRVLSMerge", () => this.volumetricLightMergePostProces, !0)); } // Create luminance _createLuminancePostProcesses(e, t) { let r = Math.pow(3, Ss.LuminanceSteps); this.luminancePostProcess = new kr("HDRLuminance", "standard", ["lumOffsets"], [], { width: r, height: r }, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, "#define LUMINANCE", t); const n = []; this.luminancePostProcess.onApply = (s) => { const a = 1 / this.luminancePostProcess.width, f = 1 / this.luminancePostProcess.height; n[0] = -0.5 * a, n[1] = 0.5 * f, n[2] = 0.5 * a, n[3] = 0.5 * f, n[4] = -0.5 * a, n[5] = -0.5 * f, n[6] = 0.5 * a, n[7] = -0.5 * f, s.setArray2("lumOffsets", n); }, this.addEffect(new gs(e.getEngine(), "HDRLuminance", () => this.luminancePostProcess, !0)); for (let s = Ss.LuminanceSteps - 1; s >= 0; s--) { r = Math.pow(3, s); let a = `#define LUMINANCE_DOWN_SAMPLE `; s === 0 && (a += "#define FINAL_DOWN_SAMPLER"); const f = new kr("HDRLuminanceDownSample" + s, "standard", ["dsOffsets", "halfDestPixelSize"], [], { width: r, height: r }, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, a, t); this.luminanceDownSamplePostProcesses.push(f); } let i = this.luminancePostProcess; this.luminanceDownSamplePostProcesses.forEach((s, a) => { const f = new Array(18); s.onApply = (o) => { if (!i) return; let d = 0; for (let v = -1; v < 2; v++) for (let u = -1; u < 2; u++) f[d] = v / i.width, f[d + 1] = u / i.height, d += 2; o.setArray2("dsOffsets", f), o.setFloat("halfDestPixelSize", 0.5 / i.width), a === this.luminanceDownSamplePostProcesses.length - 1 ? i = this.luminancePostProcess : i = s; }, a === this.luminanceDownSamplePostProcesses.length - 1 && (s.onAfterRender = () => { const o = e.getEngine().readPixels(0, 0, 1, 1), d = new Ir(1 / (255 * 255 * 255), 1 / (255 * 255), 1 / 255, 1); o.then((v) => { const u = new Uint8Array(v.buffer); this._hdrCurrentLuminance = (u[0] * d.x + u[1] * d.y + u[2] * d.z + u[3] * d.w) / 100; }); }), this.addEffect(new gs(e.getEngine(), "HDRLuminanceDownSample" + a, () => s, !0)); }); } // Create HDR post-process _createHdrPostProcess(e, t) { const r = ["#define HDR"]; this._hdrAutoExposure && r.push("#define AUTO_EXPOSURE"), this.hdrPostProcess = new kr("HDR", "standard", ["averageLuminance"], ["textureAdderSampler"], t, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, r.join(` `), 0); let n = 1, i = 0, s = 0; this.hdrPostProcess.onApply = (a) => { if (a.setTextureFromPostProcess("textureAdderSampler", this._currentDepthOfFieldSource), i += e.getEngine().getDeltaTime(), n < 0) n = this._hdrCurrentLuminance; else { const f = (s - i) / 1e3; this._hdrCurrentLuminance < n + this.hdrDecreaseRate * f ? n += this.hdrDecreaseRate * f : this._hdrCurrentLuminance > n - this.hdrIncreaseRate * f ? n -= this.hdrIncreaseRate * f : n = this._hdrCurrentLuminance; } this.hdrAutoExposure ? this._currentExposure = this._fixedExposure / n : (n = Xt.Clamp(n, this.hdrMinimumLuminance, 1e20), a.setFloat("averageLuminance", n)), s = i, this._currentDepthOfFieldSource = this.hdrFinalPostProcess; }, this.addEffect(new gs(e.getEngine(), "HDR", () => this.hdrPostProcess, !0)); } // Create lens flare post-process _createLensFlarePostProcess(e, t) { this.lensFlarePostProcess = new kr("HDRLensFlare", "standard", ["strength", "ghostDispersal", "haloWidth", "resolution", "distortionStrength"], ["lensColorSampler"], t / 2, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, "#define LENS_FLARE", 0), this.addEffect(new gs(e.getEngine(), "HDRLensFlare", () => this.lensFlarePostProcess, !0)), this._createBlurPostProcesses(e, t / 4, 2, "lensFlareBlurWidth"), this.lensFlareComposePostProcess = new kr("HDRLensFlareCompose", "standard", ["lensStarMatrix"], ["otherSampler", "lensDirtSampler", "lensStarSampler"], t, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, "#define LENS_FLARE_COMPOSE", 0), this.addEffect(new gs(e.getEngine(), "HDRLensFlareCompose", () => this.lensFlareComposePostProcess, !0)); const r = new at(0, 0); this.lensFlarePostProcess.externalTextureSamplerBinding = !0, this.lensFlarePostProcess.onApply = (s) => { s.setTextureFromPostProcess("textureSampler", this._bloomEnabled ? this.blurHPostProcesses[0] : this.originalPostProcess), s.setTexture("lensColorSampler", this.lensColorTexture), s.setFloat("strength", this.lensFlareStrength), s.setFloat("ghostDispersal", this.lensFlareGhostDispersal), s.setFloat("haloWidth", this.lensFlareHaloWidth), r.x = this.lensFlarePostProcess.width, r.y = this.lensFlarePostProcess.height, s.setVector2("resolution", r), s.setFloat("distortionStrength", this.lensFlareDistortionStrength); }; const n = he.FromValues(2, 0, -1, 0, 0, 2, -1, 0, 0, 0, 1, 0, 0, 0, 0, 1), i = he.FromValues(0.5, 0, 0.5, 0, 0, 0.5, 0.5, 0, 0, 0, 1, 0, 0, 0, 0, 1); this.lensFlareComposePostProcess.onApply = (s) => { if (!this._scene.activeCamera) return; s.setTextureFromPostProcess("otherSampler", this.lensFlarePostProcess), s.setTexture("lensDirtSampler", this.lensFlareDirtTexture), s.setTexture("lensStarSampler", this.lensStarTexture); const a = this._scene.activeCamera.getViewMatrix().getRow(0), f = this._scene.activeCamera.getViewMatrix().getRow(2); let o = S.Dot(a.toVector3(), new S(1, 0, 0)) + S.Dot(f.toVector3(), new S(0, 0, 1)); o *= 4; const d = he.FromValues(Math.cos(o) * 0.5, -Math.sin(o), 0, 0, Math.sin(o), Math.cos(o) * 0.5, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1), v = i.multiply(d).multiply(n); s.setMatrix("lensStarMatrix", v), this._currentDepthOfFieldSource = this.lensFlareFinalPostProcess; }; } // Create depth-of-field post-process _createDepthOfFieldPostProcess(e, t) { this.depthOfFieldPostProcess = new kr("HDRDepthOfField", "standard", ["distance"], ["otherSampler", "depthSampler"], t, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, "#define DEPTH_OF_FIELD", 0), this.depthOfFieldPostProcess.onApply = (r) => { r.setTextureFromPostProcess("otherSampler", this._currentDepthOfFieldSource), r.setTexture("depthSampler", this._getDepthTexture()), r.setFloat("distance", this.depthOfFieldDistance); }, this.addEffect(new gs(e.getEngine(), "HDRDepthOfField", () => this.depthOfFieldPostProcess, !0)); } // Create motion blur post-process _createMotionBlurPostProcess(e, t) { if (this._isObjectBasedMotionBlur) { const r = new BD("HDRMotionBlur", e, t, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, 0); r.motionStrength = this.motionStrength, r.motionBlurSamples = this.motionBlurSamples, this.motionBlurPostProcess = r; } else { this.motionBlurPostProcess = new kr("HDRMotionBlur", "standard", ["inverseViewProjection", "prevViewProjection", "screenSize", "motionScale", "motionStrength"], ["depthSampler"], t, null, We.BILINEAR_SAMPLINGMODE, e.getEngine(), !1, `#define MOTION_BLUR #define MAX_MOTION_SAMPLES ` + this.motionBlurSamples.toFixed(1), 0); let r = 0, n = he.Identity(); const i = he.Identity(); let s = he.Identity(); const a = at.Zero(); this.motionBlurPostProcess.onApply = (f) => { s = e.getProjectionMatrix().multiply(e.getViewMatrix()), s.invertToRef(i), f.setMatrix("inverseViewProjection", i), f.setMatrix("prevViewProjection", n), n = s, a.x = this.motionBlurPostProcess.width, a.y = this.motionBlurPostProcess.height, f.setVector2("screenSize", a), r = e.getEngine().getFps() / 60, f.setFloat("motionScale", r), f.setFloat("motionStrength", this.motionStrength), f.setTexture("depthSampler", this._getDepthTexture()); }; } this.addEffect(new gs(e.getEngine(), "HDRMotionBlur", () => this.motionBlurPostProcess, !0)); } _getDepthTexture() { return this._scene.getEngine().getCaps().drawBuffersExtension ? this._scene.enableGeometryBufferRenderer().getGBuffer().textures[0] : this._scene.enableDepthRenderer().getDepthMap(); } _disposePostProcesses() { for (let e = 0; e < this._cameras.length; e++) { const t = this._cameras[e]; this.originalPostProcess && this.originalPostProcess.dispose(t), this.screenSpaceReflectionPostProcess && this.screenSpaceReflectionPostProcess.dispose(t), this.downSampleX4PostProcess && this.downSampleX4PostProcess.dispose(t), this.brightPassPostProcess && this.brightPassPostProcess.dispose(t), this.textureAdderPostProcess && this.textureAdderPostProcess.dispose(t), this.volumetricLightPostProcess && this.volumetricLightPostProcess.dispose(t), this.volumetricLightSmoothXPostProcess && this.volumetricLightSmoothXPostProcess.dispose(t), this.volumetricLightSmoothYPostProcess && this.volumetricLightSmoothYPostProcess.dispose(t), this.volumetricLightMergePostProces && this.volumetricLightMergePostProces.dispose(t), this.volumetricLightFinalPostProcess && this.volumetricLightFinalPostProcess.dispose(t), this.lensFlarePostProcess && this.lensFlarePostProcess.dispose(t), this.lensFlareComposePostProcess && this.lensFlareComposePostProcess.dispose(t); for (let r = 0; r < this.luminanceDownSamplePostProcesses.length; r++) this.luminanceDownSamplePostProcesses[r].dispose(t); this.luminancePostProcess && this.luminancePostProcess.dispose(t), this.hdrPostProcess && this.hdrPostProcess.dispose(t), this.hdrFinalPostProcess && this.hdrFinalPostProcess.dispose(t), this.depthOfFieldPostProcess && this.depthOfFieldPostProcess.dispose(t), this.motionBlurPostProcess && this.motionBlurPostProcess.dispose(t), this.fxaaPostProcess && this.fxaaPostProcess.dispose(t); for (let r = 0; r < this.blurHPostProcesses.length; r++) this.blurHPostProcesses[r].dispose(t); for (let r = 0; r < this.blurVPostProcesses.length; r++) this.blurVPostProcesses[r].dispose(t); } this.originalPostProcess = null, this.downSampleX4PostProcess = null, this.brightPassPostProcess = null, this.textureAdderPostProcess = null, this.textureAdderFinalPostProcess = null, this.volumetricLightPostProcess = null, this.volumetricLightSmoothXPostProcess = null, this.volumetricLightSmoothYPostProcess = null, this.volumetricLightMergePostProces = null, this.volumetricLightFinalPostProcess = null, this.lensFlarePostProcess = null, this.lensFlareComposePostProcess = null, this.luminancePostProcess = null, this.hdrPostProcess = null, this.hdrFinalPostProcess = null, this.depthOfFieldPostProcess = null, this.motionBlurPostProcess = null, this.fxaaPostProcess = null, this.screenSpaceReflectionPostProcess = null, this.luminanceDownSamplePostProcesses.length = 0, this.blurHPostProcesses.length = 0, this.blurVPostProcesses.length = 0; } /** * Dispose of the pipeline and stop all post processes */ dispose() { this._disposePostProcesses(), this._scene.postProcessRenderPipelineManager.detachCamerasFromRenderPipeline(this._name, this._cameras), super.dispose(); } /** * Serialize the rendering pipeline (Used when exporting) * @returns the serialized object */ serialize() { const e = jt.Serialize(this); return this.sourceLight && (e.sourceLightId = this.sourceLight.id), this.screenSpaceReflectionPostProcess && (e.screenSpaceReflectionPostProcess = jt.Serialize(this.screenSpaceReflectionPostProcess)), e.customType = "StandardRenderingPipeline", e; } /** * Parse the serialized pipeline * @param source Source pipeline. * @param scene The scene to load the pipeline to. * @param rootUrl The URL of the serialized pipeline. * @returns An instantiated pipeline from the serialized object. */ static Parse(e, t, r) { const n = jt.Parse(() => new Ss(e._name, t, e._ratio), e, t, r); return e.sourceLightId && (n.sourceLight = t.getLightById(e.sourceLightId)), e.screenSpaceReflectionPostProcess && jt.Parse(() => n.screenSpaceReflectionPostProcess, e.screenSpaceReflectionPostProcess, t, r), n; } } Ss.LuminanceSteps = 6; C([ M() ], Ss.prototype, "brightThreshold", void 0); C([ M() ], Ss.prototype, "blurWidth", void 0); C([ M() ], Ss.prototype, "horizontalBlur", void 0); C([ M() ], Ss.prototype, "exposure", null); C([ en("lensTexture") ], Ss.prototype, "lensTexture", void 0); C([ M() ], Ss.prototype, "volumetricLightCoefficient", void 0); C([ M() ], Ss.prototype, "volumetricLightPower", void 0); C([ M() ], Ss.prototype, "volumetricLightBlurScale", void 0); C([ M() ], Ss.prototype, "hdrMinimumLuminance", void 0); C([ M() ], Ss.prototype, "hdrDecreaseRate", void 0); C([ M() ], Ss.prototype, "hdrIncreaseRate", void 0); C([ M() ], Ss.prototype, "hdrAutoExposure", null); C([ en("lensColorTexture") ], Ss.prototype, "lensColorTexture", void 0); C([ M() ], Ss.prototype, "lensFlareStrength", void 0); C([ M() ], Ss.prototype, "lensFlareGhostDispersal", void 0); C([ M() ], Ss.prototype, "lensFlareHaloWidth", void 0); C([ M() ], Ss.prototype, "lensFlareDistortionStrength", void 0); C([ M() ], Ss.prototype, "lensFlareBlurWidth", void 0); C([ en("lensStarTexture") ], Ss.prototype, "lensStarTexture", void 0); C([ en("lensFlareDirtTexture") ], Ss.prototype, "lensFlareDirtTexture", void 0); C([ M() ], Ss.prototype, "depthOfFieldDistance", void 0); C([ M() ], Ss.prototype, "depthOfFieldBlurWidth", void 0); C([ M() ], Ss.prototype, "motionStrength", null); C([ M() ], Ss.prototype, "objectBasedMotionBlur", null); C([ M() ], Ss.prototype, "_ratio", void 0); C([ M() ], Ss.prototype, "BloomEnabled", null); C([ M() ], Ss.prototype, "DepthOfFieldEnabled", null); C([ M() ], Ss.prototype, "LensFlareEnabled", null); C([ M() ], Ss.prototype, "HDREnabled", null); C([ M() // eslint-disable-next-line @typescript-eslint/naming-convention ], Ss.prototype, "VLSEnabled", null); C([ M() ], Ss.prototype, "MotionBlurEnabled", null); C([ M() ], Ss.prototype, "fxaaEnabled", null); C([ M() ], Ss.prototype, "screenSpaceReflectionsEnabled", null); C([ M() ], Ss.prototype, "volumetricLightStepsCount", null); C([ M() ], Ss.prototype, "motionBlurSamples", null); C([ M() ], Ss.prototype, "samples", null); Ue("BABYLON.StandardRenderingPipeline", Ss); class b7e { constructor() { this.enabled = !1, this.name = "screenSpaceReflections2", this.texturesRequired = [6, 3, 5]; } } const x7e = "screenSpaceRayTrace", D7e = `float distanceSquared(vec2 a,vec2 b) { a-=b; return dot(a,a); } /** param csOrigin Camera-space ray origin,which must be within the view volume and must have z>0.01 and project within the valid screen rectangle param csDirection Unit length camera-space ray direction param projectToPixelMatrix A projection matrix that maps to **pixel** coordinates (**not** [-1,+1] normalized device coordinates). param csZBuffer The camera-space Z buffer param csZBufferSize Dimensions of csZBuffer param csZThickness Camera space csZThickness to ascribe to each pixel in the depth buffer param nearPlaneZ Positive number. Doesn't have to be THE actual near plane,just a reasonable value for clipping rays headed towards the camera param stride Step in horizontal or vertical pixels between samples. This is a float because integer math is slow on GPUs,but should be set to an integer>=1 param jitterFraction Number between 0 and 1 for how far to bump the ray in stride units to conceal banding artifacts,plus the stride ray offset. param maxSteps Maximum number of iterations. Higher gives better images but may be slow param maxRayTraceDistance Maximum camera-space distance to trace before returning a miss param selfCollisionNumSkip Number of steps to skip at start when raytracing to avoid self collisions. 1 is a reasonable value,depending on the scene you may need to set this value to 2 param hitPixel Pixel coordinates of the first intersection with the scene param numIterations number of iterations performed param csHitPoint Camera space location of the ray hit */ #define inline bool traceScreenSpaceRay1( vec3 csOrigin, vec3 csDirection, mat4 projectToPixelMatrix, sampler2D csZBuffer, vec2 csZBufferSize, #ifdef SSRAYTRACE_USE_BACK_DEPTHBUFFER sampler2D csZBackBuffer, float csZBackSizeFactor, #endif float csZThickness, float nearPlaneZ, float stride, float jitterFraction, float maxSteps, float maxRayTraceDistance, float selfCollisionNumSkip, out vec2 startPixel, out vec2 hitPixel, out vec3 csHitPoint, out float numIterations #ifdef SSRAYTRACE_DEBUG ,out vec3 debugColor #endif ) { #ifdef SSRAYTRACE_RIGHT_HANDED_SCENE float rayLength=(csOrigin.z+csDirection.z*maxRayTraceDistance)>-nearPlaneZ ? (-nearPlaneZ-csOrigin.z)/csDirection.z : maxRayTraceDistance; #else float rayLength=(csOrigin.z+csDirection.z*maxRayTraceDistance)yMax) || (P1.yyMax) ? yMax : yMin))/(P1.y-P0.y);} if ((P1.x>xMax) || (P1.xxMax) ? xMax : xMin))/(P1.x-P0.x));} P1=mix(P1,P0,alpha); k1=mix(k1,k0,alpha); Q1=mix(Q1,Q0,alpha); #endif P1+=vec2((distanceSquared(P0,P1)<0.0001) ? 0.01 : 0.0);vec2 delta=P1-P0;bool permute=false;if (abs(delta.x)rayZMax) { float t=rayZMin; rayZMin=rayZMax; rayZMax=t;} sceneZMax=texelFetch(csZBuffer,ivec2(hitPixel),0).r; #ifdef SSRAYTRACE_RIGHT_HANDED_SCENE #ifdef SSRAYTRACE_USE_BACK_DEPTHBUFFER float sceneBackZ=texelFetch(csZBackBuffer,ivec2(hitPixel/csZBackSizeFactor),0).r;hit=(rayZMax>=sceneBackZ-csZThickness) && (rayZMin<=sceneZMax); #else hit=(rayZMax>=sceneZMax-csZThickness) && (rayZMin<=sceneZMax); #endif #else #ifdef SSRAYTRACE_USE_BACK_DEPTHBUFFER float sceneBackZ=texelFetch(csZBackBuffer,ivec2(hitPixel/csZBackSizeFactor),0).r;hit=(rayZMin<=sceneBackZ+csZThickness) && (rayZMax>=sceneZMax) && (sceneZMax != 0.0); #else hit=(rayZMin<=sceneZMax+csZThickness) && (rayZMax>=sceneZMax); #endif #endif } pqk-=dPQK;stepCount-=1.0;if (((pqk.x+dPQK.x)*stepDirection)>end || (stepCount+1.0)>=maxSteps || sceneZMax==0.0) {hit=false;} #ifdef SSRAYTRACE_ENABLE_REFINEMENT if (stride>1.0 && hit) {pqk-=dPQK;stepCount-=1.0;float invStride=1.0/stride;dPQK*=invStride;float refinementStepCount=0.0;prevZMaxEstimate=pqk.z/pqk.w;rayZMax=prevZMaxEstimate;sceneZMax=rayZMax+1e7;for (;refinementStepCount<=1.0 || (refinementStepCount<=stride*1.4) && (rayZMaxend) {debugColor=vec3(0,0,1);} else if ((stepCount+1.0)>=maxSteps) {debugColor=vec3(1,0,0);} else if (sceneZMax==0.0) {debugColor=vec3(1,1,0);} else {debugColor=vec3(0,stepCount/maxSteps,0);} #endif return hit;} /** texCoord: in the [0,1] range depth: depth in view space (range [znear,zfar]]) */ vec3 computeViewPosFromUVDepth(vec2 texCoord,float depth,mat4 projection,mat4 invProjectionMatrix) {vec4 ndc;ndc.xy=texCoord*2.0-1.0; #ifdef SSRAYTRACE_RIGHT_HANDED_SCENE ndc.z=-projection[2].z-projection[3].z/depth; #else ndc.z=projection[2].z+projection[3].z/depth; #endif ndc.w=1.0;vec4 eyePos=invProjectionMatrix*ndc;eyePos.xyz/=eyePos.w;return eyePos.xyz;} `; Le.IncludesShadersStore[x7e] = D7e; const j7e = "screenSpaceReflection2PixelShader", w7e = `#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE) #define TEXTUREFUNC(s,c,lod) texture2DLodEXT(s,c,lod) #define TEXTURECUBEFUNC(s,c,lod) textureLod(s,c,lod) #else #define TEXTUREFUNC(s,c,bias) texture2D(s,c,bias) #define TEXTURECUBEFUNC(s,c,bias) textureCube(s,c,bias) #endif uniform sampler2D textureSampler;varying vec2 vUV; #ifdef SSR_SUPPORTED uniform sampler2D reflectivitySampler;uniform sampler2D normalSampler;uniform sampler2D depthSampler; #ifdef SSRAYTRACE_USE_BACK_DEPTHBUFFER uniform sampler2D backDepthSampler;uniform float backSizeFactor; #endif #ifdef SSR_USE_ENVIRONMENT_CUBE uniform samplerCube envCubeSampler; #ifdef SSR_USE_LOCAL_REFLECTIONMAP_CUBIC uniform vec3 vReflectionPosition;uniform vec3 vReflectionSize; #endif #endif uniform mat4 view;uniform mat4 invView;uniform mat4 projection;uniform mat4 invProjectionMatrix;uniform mat4 projectionPixel;uniform float nearPlaneZ;uniform float stepSize;uniform float maxSteps;uniform float strength;uniform float thickness;uniform float roughnessFactor;uniform float reflectionSpecularFalloffExponent;uniform float maxDistance;uniform float selfCollisionNumSkip;uniform float reflectivityThreshold; #include #include #include vec3 hash(vec3 a) {a=fract(a*0.8);a+=dot(a,a.yxz+19.19);return fract((a.xxy+a.yxx)*a.zyx);} float computeAttenuationForIntersection(ivec2 hitPixel,vec2 hitUV,vec3 vsRayOrigin,vec3 vsHitPoint,vec3 reflectionVector,float maxRayDistance,float numIterations) {float attenuation=1.0; #ifdef SSR_ATTENUATE_SCREEN_BORDERS vec2 dCoords=smoothstep(0.2,0.6,abs(vec2(0.5,0.5)-hitUV.xy));attenuation*=clamp(1.0-(dCoords.x+dCoords.y),0.0,1.0); #endif #ifdef SSR_ATTENUATE_INTERSECTION_DISTANCE attenuation*=1.0-clamp(distance(vsRayOrigin,vsHitPoint)/maxRayDistance,0.0,1.0); #endif #ifdef SSR_ATTENUATE_INTERSECTION_NUMITERATIONS attenuation*=1.0-(numIterations/maxSteps); #endif #ifdef SSR_ATTENUATE_BACKFACE_REFLECTION vec3 reflectionNormal=texelFetch(normalSampler,hitPixel,0).xyz;float directionBasedAttenuation=smoothstep(-0.17,0.0,dot(reflectionNormal,-reflectionVector));attenuation*=directionBasedAttenuation; #endif return attenuation;} #endif void main() { #ifdef SSR_SUPPORTED vec4 colorFull=TEXTUREFUNC(textureSampler,vUV,0.0);vec3 color=colorFull.rgb;vec4 reflectivity=TEXTUREFUNC(reflectivitySampler,vUV,0.0); #ifndef SSR_DISABLE_REFLECTIVITY_TEST if (max(reflectivity.r,max(reflectivity.g,reflectivity.b))<=reflectivityThreshold) { #ifdef SSR_USE_BLUR gl_FragColor=vec4(0.); #else gl_FragColor=colorFull; #endif return;} #endif #ifdef SSR_INPUT_IS_GAMMA_SPACE color=toLinearSpace(color); #endif vec2 texSize=vec2(textureSize(depthSampler,0));vec3 csNormal=texelFetch(normalSampler,ivec2(vUV*texSize),0).xyz; float depth=texelFetch(depthSampler,ivec2(vUV*texSize),0).r;vec3 csPosition=computeViewPosFromUVDepth(vUV,depth,projection,invProjectionMatrix);vec3 csViewDirection=normalize(csPosition);vec3 csReflectedVector=reflect(csViewDirection,csNormal); #ifdef SSR_USE_ENVIRONMENT_CUBE vec3 wReflectedVector=vec3(invView*vec4(csReflectedVector,0.0)); #ifdef SSR_USE_LOCAL_REFLECTIONMAP_CUBIC vec4 worldPos=invView*vec4(csPosition,1.0);wReflectedVector=parallaxCorrectNormal(worldPos.xyz,normalize(wReflectedVector),vReflectionSize,vReflectionPosition); #endif #ifdef SSR_INVERTCUBICMAP wReflectedVector.y*=-1.0; #endif #ifdef SSRAYTRACE_RIGHT_HANDED_SCENE wReflectedVector.z*=-1.0; #endif vec3 envColor=TEXTURECUBEFUNC(envCubeSampler,wReflectedVector,0.0).xyz; #ifdef SSR_ENVIRONMENT_CUBE_IS_GAMMASPACE envColor=toLinearSpace(envColor); #endif #else vec3 envColor=color; #endif float reflectionAttenuation=1.0;bool rayHasHit=false;vec2 startPixel;vec2 hitPixel;vec3 hitPoint;float numIterations; #ifdef SSRAYTRACE_DEBUG vec3 debugColor; #endif #ifdef SSR_ATTENUATE_FACING_CAMERA reflectionAttenuation*=1.0-smoothstep(0.25,0.5,dot(-csViewDirection,csReflectedVector)); #endif if (reflectionAttenuation>0.0) { #ifdef SSR_USE_BLUR vec3 jitt=vec3(0.); #else float roughness=1.0-reflectivity.a;vec3 jitt=mix(vec3(0.0),hash(csPosition)-vec3(0.5),roughness)*roughnessFactor; #endif vec2 uv2=vUV*texSize;float c=(uv2.x+uv2.y)*0.25;float jitter=mod(c,1.0); rayHasHit=traceScreenSpaceRay1( csPosition, normalize(csReflectedVector+jitt), projectionPixel, depthSampler, texSize, #ifdef SSRAYTRACE_USE_BACK_DEPTHBUFFER backDepthSampler, backSizeFactor, #endif thickness, nearPlaneZ, stepSize, jitter, maxSteps, maxDistance, selfCollisionNumSkip, startPixel, hitPixel, hitPoint, numIterations #ifdef SSRAYTRACE_DEBUG ,debugColor #endif );} #ifdef SSRAYTRACE_DEBUG gl_FragColor=vec4(debugColor,1.);return; #endif vec3 F0=reflectivity.rgb;vec3 fresnel=fresnelSchlickGGX(max(dot(csNormal,-csViewDirection),0.0),F0,vec3(1.));vec3 SSR=envColor;if (rayHasHit) {vec3 reflectedColor=texelFetch(textureSampler,ivec2(hitPixel),0).rgb; #ifdef SSR_INPUT_IS_GAMMA_SPACE reflectedColor=toLinearSpace(reflectedColor); #endif reflectionAttenuation*=computeAttenuationForIntersection(ivec2(hitPixel),hitPixel/texSize,csPosition,hitPoint,csReflectedVector,maxDistance,numIterations);SSR=reflectedColor*reflectionAttenuation+(1.0-reflectionAttenuation)*envColor;} #ifndef SSR_BLEND_WITH_FRESNEL SSR*=fresnel; #endif #ifdef SSR_USE_BLUR float blur_radius=0.0;float roughness=1.0-reflectivity.a*(1.0-roughnessFactor);if (roughness>0.001) {float cone_angle=min(roughness,0.999)*3.14159265*0.5;float cone_len=distance(startPixel,hitPixel);float op_len=2.0*tan(cone_angle)*cone_len; float a=op_len;float h=cone_len;float a2=a*a;float fh2=4.0f*h*h;blur_radius=(a*(sqrt(a2+fh2)-a))/(4.0f*h);} gl_FragColor=vec4(SSR,blur_radius/255.0); #else #ifdef SSR_BLEND_WITH_FRESNEL vec3 reflectionMultiplier=clamp(pow(fresnel*strength,vec3(reflectionSpecularFalloffExponent)),0.0,1.0); #else vec3 reflectionMultiplier=clamp(pow(reflectivity.rgb*strength,vec3(reflectionSpecularFalloffExponent)),0.0,1.0); #endif vec3 colorMultiplier=1.0-reflectionMultiplier;vec3 finalColor=(color*colorMultiplier)+(SSR*reflectionMultiplier); #ifdef SSR_OUTPUT_IS_GAMMA_SPACE finalColor=toGammaSpace(finalColor); #endif gl_FragColor=vec4(finalColor,colorFull.a); #endif #else gl_FragColor=TEXTUREFUNC(textureSampler,vUV,0.0); #endif } `; Le.ShadersStore[j7e] = w7e; const m7e = "screenSpaceReflection2BlurPixelShader", B7e = `#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE) #define TEXTUREFUNC(s,c,lod) texture2DLodEXT(s,c,lod) #else #define TEXTUREFUNC(s,c,bias) texture2D(s,c,bias) #endif uniform sampler2D textureSampler;varying vec2 vUV;uniform vec2 texelOffsetScale;const float weights[8]=float[8] (0.071303,0.131514,0.189879,0.321392,0.452906, 0.584419,0.715932,0.847445);void processSample(vec2 uv,float i,vec2 stepSize,inout vec4 accumulator,inout float denominator) {vec2 offsetUV=stepSize*i+uv;float coefficient=weights[int(2.0-abs(i))];accumulator+=TEXTUREFUNC(textureSampler,offsetUV,0.0)*coefficient;denominator+=coefficient;} void main() {vec4 colorFull=TEXTUREFUNC(textureSampler,vUV,0.0);if (dot(colorFull,vec4(1.0))==0.0) {gl_FragColor=colorFull;return;} float blurRadius=colorFull.a*255.0; vec2 stepSize=texelOffsetScale.xy*blurRadius;vec4 accumulator=TEXTUREFUNC(textureSampler,vUV,0.0)*0.214607;float denominator=0.214607;processSample(vUV,1.0,stepSize,accumulator,denominator);processSample(vUV,1.0*0.2,stepSize,accumulator,denominator);processSample(vUV,1.0*0.4,stepSize,accumulator,denominator);processSample(vUV,1.0*0.6,stepSize,accumulator,denominator);processSample(vUV,1.0*0.8,stepSize,accumulator,denominator);processSample(vUV,1.0*1.2,stepSize,accumulator,denominator);processSample(vUV,1.0*1.4,stepSize,accumulator,denominator);processSample(vUV,1.0*1.6,stepSize,accumulator,denominator);processSample(vUV,1.0*1.8,stepSize,accumulator,denominator);processSample(vUV,1.0*2.0,stepSize,accumulator,denominator);processSample(vUV,-1.0,stepSize,accumulator,denominator);processSample(vUV,-1.0*0.2,stepSize,accumulator,denominator);processSample(vUV,-1.0*0.4,stepSize,accumulator,denominator);processSample(vUV,-1.0*0.6,stepSize,accumulator,denominator);processSample(vUV,-1.0*0.8,stepSize,accumulator,denominator);processSample(vUV,-1.0*1.2,stepSize,accumulator,denominator);processSample(vUV,-1.0*1.4,stepSize,accumulator,denominator);processSample(vUV,-1.0*1.6,stepSize,accumulator,denominator);processSample(vUV,-1.0*1.8,stepSize,accumulator,denominator);processSample(vUV,-1.0*2.0,stepSize,accumulator,denominator);gl_FragColor=vec4(accumulator.rgb/denominator,colorFull.a);} `; Le.ShadersStore[m7e] = B7e; const W7e = "screenSpaceReflection2BlurCombinerPixelShader", S7e = `uniform sampler2D textureSampler; uniform sampler2D mainSampler;uniform sampler2D reflectivitySampler;uniform float strength;uniform float reflectionSpecularFalloffExponent;uniform float reflectivityThreshold;varying vec2 vUV; #include #ifdef SSR_BLEND_WITH_FRESNEL #include #include uniform mat4 projection;uniform mat4 invProjectionMatrix;uniform sampler2D normalSampler;uniform sampler2D depthSampler; #endif void main() { #ifdef SSRAYTRACE_DEBUG gl_FragColor=texture2D(textureSampler,vUV); #else vec3 SSR=texture2D(textureSampler,vUV).rgb;vec4 color=texture2D(mainSampler,vUV);vec4 reflectivity=texture2D(reflectivitySampler,vUV); #ifndef SSR_DISABLE_REFLECTIVITY_TEST if (max(reflectivity.r,max(reflectivity.g,reflectivity.b))<=reflectivityThreshold) {gl_FragColor=color;return;} #endif #ifdef SSR_INPUT_IS_GAMMA_SPACE color=toLinearSpace(color); #endif #ifdef SSR_BLEND_WITH_FRESNEL vec2 texSize=vec2(textureSize(depthSampler,0));vec3 csNormal=texelFetch(normalSampler,ivec2(vUV*texSize),0).xyz;float depth=texelFetch(depthSampler,ivec2(vUV*texSize),0).r;vec3 csPosition=computeViewPosFromUVDepth(vUV,depth,projection,invProjectionMatrix);vec3 csViewDirection=normalize(csPosition);vec3 F0=reflectivity.rgb;vec3 fresnel=fresnelSchlickGGX(max(dot(csNormal,-csViewDirection),0.0),F0,vec3(1.));vec3 reflectionMultiplier=clamp(pow(fresnel*strength,vec3(reflectionSpecularFalloffExponent)),0.0,1.0); #else vec3 reflectionMultiplier=clamp(pow(reflectivity.rgb*strength,vec3(reflectionSpecularFalloffExponent)),0.0,1.0); #endif vec3 colorMultiplier=1.0-reflectionMultiplier;vec3 finalColor=(color.rgb*colorMultiplier)+(SSR*reflectionMultiplier); #ifdef SSR_OUTPUT_IS_GAMMA_SPACE finalColor=toGammaSpace(finalColor); #endif gl_FragColor=vec4(finalColor,color.a); #endif } `; Le.ShadersStore[W7e] = S7e; const U7e = he.Compose(new S(0.5, 0.5, 0.5), Ze.Identity(), new S(0.5, 0.5, 0.5)), I7e = he.Compose(new S(0.5, 0.5, 1), Ze.Identity(), new S(0.5, 0.5, 0)); class Wa extends SD { /** * MSAA sample count, setting this to 4 will provide 4x anti aliasing. (default: 1) */ set samples(e) { this._samples !== e && (this._samples = e, this._buildPipeline()); } get samples() { return this._samples; } /** * Gets or sets the minimum value for one of the reflectivity component of the material to consider it for SSR (default: 0.04). * If all r/g/b components of the reflectivity is below or equal this value, the pixel will not be considered reflective and SSR won't be applied. */ get reflectivityThreshold() { return this._reflectivityThreshold; } set reflectivityThreshold(e) { e !== this._reflectivityThreshold && (e === 0 && this._reflectivityThreshold !== 0 || e !== 0 && this._reflectivityThreshold === 0 ? (this._reflectivityThreshold = e, this._buildPipeline()) : this._reflectivityThreshold = e); } /** * Gets or sets the downsample factor used to reduce the size of the texture used to compute the SSR contribution (default: 0). * Use 0 to render the SSR contribution at full resolution, 1 to render at half resolution, 2 to render at 1/3 resolution, etc. * Note that it is used only when blurring is enabled (blurDispersionStrength \> 0), because in that mode the SSR contribution is generated in a separate texture. */ get ssrDownsample() { return this._ssrDownsample; } set ssrDownsample(e) { e !== this._ssrDownsample && (this._ssrDownsample = e, this._buildPipeline()); } /** * Gets or sets the blur dispersion strength. Set this value to 0 to disable blurring (default: 0.05) * The reflections are blurred based on the roughness of the surface and the distance between the pixel shaded and the reflected pixel: the higher the distance the more blurry the reflection is. * blurDispersionStrength allows to increase or decrease this effect. */ get blurDispersionStrength() { return this._blurDispersionStrength; } set blurDispersionStrength(e) { if (e === this._blurDispersionStrength) return; const t = e === 0 && this._blurDispersionStrength !== 0 || e !== 0 && this._blurDispersionStrength === 0; this._blurDispersionStrength = e, t && this._buildPipeline(); } _useBlur() { return this._blurDispersionStrength > 0; } /** * Gets or sets the downsample factor used to reduce the size of the textures used to blur the reflection effect (default: 0). * Use 0 to blur at full resolution, 1 to render at half resolution, 2 to render at 1/3 resolution, etc. */ get blurDownsample() { return this._blurDownsample; } set blurDownsample(e) { e !== this._blurDownsample && (this._blurDownsample = e, this._buildPipeline()); } /** * Gets or sets whether or not smoothing reflections is enabled (default: false) * Enabling smoothing will require more GPU power. * Note that this setting has no effect if step = 1: it's only used if step \> 1. */ get enableSmoothReflections() { return this._enableSmoothReflections; } set enableSmoothReflections(e) { e !== this._enableSmoothReflections && (this._enableSmoothReflections = e, this._updateEffectDefines()); } /** * Gets or sets the environment cube texture used to define the reflection when the reflected rays of SSR leave the view space or when the maxDistance/maxSteps is reached. */ get environmentTexture() { return this._environmentTexture; } set environmentTexture(e) { this._environmentTexture = e, this._updateEffectDefines(); } /** * Gets or sets the boolean defining if the environment texture is a standard cubemap (false) or a probe (true). Default value is false. * Note: a probe cube texture is treated differently than an ordinary cube texture because the Y axis is reversed. */ get environmentTextureIsProbe() { return this._environmentTextureIsProbe; } set environmentTextureIsProbe(e) { this._environmentTextureIsProbe = e, this._updateEffectDefines(); } /** * Gets or sets a boolean indicating if the reflections should be attenuated at the screen borders (default: true). */ get attenuateScreenBorders() { return this._attenuateScreenBorders; } set attenuateScreenBorders(e) { this._attenuateScreenBorders !== e && (this._attenuateScreenBorders = e, this._updateEffectDefines()); } /** * Gets or sets a boolean indicating if the reflections should be attenuated according to the distance of the intersection (default: true). */ get attenuateIntersectionDistance() { return this._attenuateIntersectionDistance; } set attenuateIntersectionDistance(e) { this._attenuateIntersectionDistance !== e && (this._attenuateIntersectionDistance = e, this._updateEffectDefines()); } /** * Gets or sets a boolean indicating if the reflections should be attenuated according to the number of iterations performed to find the intersection (default: true). */ get attenuateIntersectionIterations() { return this._attenuateIntersectionIterations; } set attenuateIntersectionIterations(e) { this._attenuateIntersectionIterations !== e && (this._attenuateIntersectionIterations = e, this._updateEffectDefines()); } /** * Gets or sets a boolean indicating if the reflections should be attenuated when the reflection ray is facing the camera (the view direction) (default: false). */ get attenuateFacingCamera() { return this._attenuateFacingCamera; } set attenuateFacingCamera(e) { this._attenuateFacingCamera !== e && (this._attenuateFacingCamera = e, this._updateEffectDefines()); } /** * Gets or sets a boolean indicating if the backface reflections should be attenuated (default: false). */ get attenuateBackfaceReflection() { return this._attenuateBackfaceReflection; } set attenuateBackfaceReflection(e) { this._attenuateBackfaceReflection !== e && (this._attenuateBackfaceReflection = e, this._updateEffectDefines()); } /** * Gets or sets a boolean indicating if the ray should be clipped to the frustum (default: true). * You can try to set this parameter to false to save some performances: it may produce some artefacts in some cases, but generally they won't really be visible */ get clipToFrustum() { return this._clipToFrustum; } set clipToFrustum(e) { this._clipToFrustum !== e && (this._clipToFrustum = e, this._updateEffectDefines()); } /** * Gets or sets a boolean indicating whether the blending between the current color pixel and the reflection color should be done with a Fresnel coefficient (default: false). * It is more physically accurate to use the Fresnel coefficient (otherwise it uses the reflectivity of the material for blending), but it is also more expensive when you use blur (when blurDispersionStrength \> 0). */ get useFresnel() { return this._useFresnel; } set useFresnel(e) { this._useFresnel !== e && (this._useFresnel = e, this._buildPipeline()); } /** * Gets or sets a boolean defining if geometry thickness should be computed automatically (default: false). * When enabled, a depth renderer is created which will render the back faces of the scene to a depth texture (meaning additional work for the GPU). * In that mode, the "thickness" property is still used as an offset to compute the ray intersection, but you can typically use a much lower * value than when enableAutomaticThicknessComputation is false (it's even possible to use a value of 0 when using low values for "step") * Note that for performance reasons, this option will only apply to the first camera to which the rendering pipeline is attached! */ get enableAutomaticThicknessComputation() { return this._enableAutomaticThicknessComputation; } set enableAutomaticThicknessComputation(e) { this._enableAutomaticThicknessComputation !== e && (this._enableAutomaticThicknessComputation = e, this._buildPipeline()); } /** * Gets the depth renderer used to render the back faces of the scene to a depth texture. */ get backfaceDepthRenderer() { return this._depthRenderer; } /** * Gets or sets the downsample factor (default: 0) used to create the backface depth texture - used only if enableAutomaticThicknessComputation = true. * Use 0 to render the depth at full resolution, 1 to render at half resolution, 2 to render at 1/4 resolution, etc. * Note that you will get rendering artefacts when using a value different from 0: it's a tradeoff between image quality and performances. */ get backfaceDepthTextureDownsample() { return this._backfaceDepthTextureDownsample; } set backfaceDepthTextureDownsample(e) { this._backfaceDepthTextureDownsample !== e && (this._backfaceDepthTextureDownsample = e, this._resizeDepthRenderer()); } /** * Gets or sets a boolean (default: true) indicating if the depth of transparent meshes should be written to the backface depth texture (when automatic thickness computation is enabled). */ get backfaceForceDepthWriteTransparentMeshes() { return this._backfaceForceDepthWriteTransparentMeshes; } set backfaceForceDepthWriteTransparentMeshes(e) { this._backfaceForceDepthWriteTransparentMeshes !== e && (this._backfaceForceDepthWriteTransparentMeshes = e, this._depthRenderer && (this._depthRenderer.forceDepthWriteTransparentMeshes = e)); } /** * Gets or sets a boolean indicating if the effect is enabled (default: true). */ get isEnabled() { return this._isEnabled; } set isEnabled(e) { this._isEnabled !== e && (this._isEnabled = e, e ? e && (this._isDirty ? this._buildPipeline() : this._cameras !== null && this._scene.postProcessRenderPipelineManager.attachCamerasToRenderPipeline(this._name, this._cameras)) : this._cameras !== null && (this._scene.postProcessRenderPipelineManager.detachCamerasFromRenderPipeline(this._name, this._cameras), this._cameras = this._camerasToBeAttached.slice())); } /** * Gets or sets a boolean defining if the input color texture is in gamma space (default: true) * The SSR effect works in linear space, so if the input texture is in gamma space, we must convert the texture to linear space before applying the effect */ get inputTextureColorIsInGammaSpace() { return this._inputTextureColorIsInGammaSpace; } set inputTextureColorIsInGammaSpace(e) { this._inputTextureColorIsInGammaSpace !== e && (this._inputTextureColorIsInGammaSpace = e, this._buildPipeline()); } /** * Gets or sets a boolean defining if the output color texture generated by the SSR pipeline should be in gamma space (default: true) * If you have a post-process that comes after the SSR and that post-process needs the input to be in a linear space, you must disable generateOutputInGammaSpace */ get generateOutputInGammaSpace() { return this._generateOutputInGammaSpace; } set generateOutputInGammaSpace(e) { this._generateOutputInGammaSpace !== e && (this._generateOutputInGammaSpace = e, this._buildPipeline()); } /** * Gets or sets a boolean indicating if the effect should be rendered in debug mode (default: false). * In this mode, colors have this meaning: * - blue: the ray hit the max distance (we reached maxDistance) * - red: the ray ran out of steps (we reached maxSteps) * - yellow: the ray went off screen * - green: the ray hit a surface. The brightness of the green color is proportional to the distance between the ray origin and the intersection point: A brighter green means more computation than a darker green. * In the first 3 cases, the final color is calculated by mixing the skybox color with the pixel color (if environmentTexture is defined), otherwise the pixel color is not modified * You should try to get as few blue/red/yellow pixels as possible, as this means that the ray has gone further than if it had hit a surface. */ get debug() { return this._debug; } set debug(e) { this._debug !== e && (this._debug = e, this._buildPipeline()); } /** * Gets the scene the effect belongs to. * @returns the scene the effect belongs to. */ getScene() { return this._scene; } get _geometryBufferRenderer() { return this._forceGeometryBuffer ? this._scene.geometryBufferRenderer : null; } get _prePassRenderer() { return this._forceGeometryBuffer ? null : this._scene.prePassRenderer; } /** * Gets active scene */ get scene() { return this._scene; } /** * Returns true if SSR is supported by the running hardware */ get isSupported() { const e = this._scene.getEngine().getCaps(); return e.drawBuffersExtension && e.texelFetch; } /** * Constructor of the SSR rendering pipeline * @param name The rendering pipeline name * @param scene The scene linked to this pipeline * @param cameras The array of cameras that the rendering pipeline will be attached to (default: scene.cameras) * @param forceGeometryBuffer Set to true if you want to use the legacy geometry buffer renderer (default: false) * @param textureType The texture type used by the different post processes created by SSR (default: 0) */ constructor(e, t, r, n = !1, i = 0) { if (super(t.getEngine(), e), this.SSRRenderEffect = "SSRRenderEffect", this.SSRBlurRenderEffect = "SSRBlurRenderEffect", this.SSRCombineRenderEffect = "SSRCombineRenderEffect", this._samples = 1, this.maxDistance = 1e3, this.step = 1, this.thickness = 0.5, this.strength = 1, this.reflectionSpecularFalloffExponent = 1, this.maxSteps = 1e3, this.roughnessFactor = 0.2, this.selfCollisionNumSkip = 1, this._reflectivityThreshold = 0.04, this._ssrDownsample = 0, this._blurDispersionStrength = 0.03, this._blurDownsample = 0, this._enableSmoothReflections = !1, this._environmentTextureIsProbe = !1, this._attenuateScreenBorders = !0, this._attenuateIntersectionDistance = !0, this._attenuateIntersectionIterations = !0, this._attenuateFacingCamera = !1, this._attenuateBackfaceReflection = !1, this._clipToFrustum = !0, this._useFresnel = !1, this._enableAutomaticThicknessComputation = !1, this._backfaceDepthTextureDownsample = 0, this._backfaceForceDepthWriteTransparentMeshes = !0, this._isEnabled = !0, this._inputTextureColorIsInGammaSpace = !0, this._generateOutputInGammaSpace = !0, this._debug = !1, this._forceGeometryBuffer = !1, this._isDirty = !1, this._camerasToBeAttached = [], this._cameras = r || t.cameras, this._cameras = this._cameras.slice(), this._camerasToBeAttached = this._cameras.slice(), this._scene = t, this._textureType = i, this._forceGeometryBuffer = n, this.isSupported) { if (t.postProcessRenderPipelineManager.addPipeline(this), this._forceGeometryBuffer) { const s = t.enableGeometryBufferRenderer(); s && (s.enableReflectivity = !0, s.useSpecificClearForDepthTexture = !0, s.generateNormalsInWorldSpace && console.error("SSRRenderingPipeline does not support generateNormalsInWorldSpace=true for the geometry buffer renderer!")); } else { const s = t.enablePrePassRenderer(); s && (s.useSpecificClearForDepthTexture = !0, s.markAsDirty(), s.generateNormalsInWorldSpace && console.error("SSRRenderingPipeline does not support generateNormalsInWorldSpace=true for the prepass renderer!")); } this._buildPipeline(); } } /** * Get the class name * @returns "SSRRenderingPipeline" */ getClassName() { return "SSRRenderingPipeline"; } /** * Adds a camera to the pipeline * @param camera the camera to be added */ addCamera(e) { this._camerasToBeAttached.push(e), this._buildPipeline(); } /** * Removes a camera from the pipeline * @param camera the camera to remove */ removeCamera(e) { const t = this._camerasToBeAttached.indexOf(e); this._camerasToBeAttached.splice(t, 1), this._buildPipeline(); } /** * Removes the internal pipeline assets and detaches the pipeline from the scene cameras * @param disableGeometryBufferRenderer */ dispose(e = !1) { this._disposeDepthRenderer(), this._disposePostProcesses(), e && this._scene.disableGeometryBufferRenderer(), this._scene.postProcessRenderPipelineManager.detachCamerasFromRenderPipeline(this._name, this._cameras), super.dispose(); } _getTextureSize() { var e, t; const r = this._scene.getEngine(), n = this._prePassRenderer; let i = { width: r.getRenderWidth(), height: r.getRenderHeight() }; if (n && ((e = this._scene.activeCamera) === null || e === void 0 ? void 0 : e._getFirstPostProcess()) === this._ssrPostProcess) { const s = n.getRenderTarget(); s && s.textures && (i = s.textures[n.getIndex(4)].getSize()); } else !((t = this._ssrPostProcess) === null || t === void 0) && t.inputTexture && (i.width = this._ssrPostProcess.inputTexture.width, i.height = this._ssrPostProcess.inputTexture.height); return i; } _updateEffectDefines() { var e; const t = []; (this._geometryBufferRenderer || this._prePassRenderer) && t.push("#define SSR_SUPPORTED"), this._enableSmoothReflections && t.push("#define SSRAYTRACE_ENABLE_REFINEMENT"), this._scene.useRightHandedSystem && t.push("#define SSRAYTRACE_RIGHT_HANDED_SCENE"), this._environmentTexture && (t.push("#define SSR_USE_ENVIRONMENT_CUBE"), this._environmentTexture.boundingBoxSize && t.push("#define SSR_USE_LOCAL_REFLECTIONMAP_CUBIC"), this._environmentTexture.gammaSpace && t.push("#define SSR_ENVIRONMENT_CUBE_IS_GAMMASPACE")), this._environmentTextureIsProbe && t.push("#define SSR_INVERTCUBICMAP"), this._enableAutomaticThicknessComputation && t.push("#define SSRAYTRACE_USE_BACK_DEPTHBUFFER"), this._attenuateScreenBorders && t.push("#define SSR_ATTENUATE_SCREEN_BORDERS"), this._attenuateIntersectionDistance && t.push("#define SSR_ATTENUATE_INTERSECTION_DISTANCE"), this._attenuateIntersectionIterations && t.push("#define SSR_ATTENUATE_INTERSECTION_NUMITERATIONS"), this._attenuateFacingCamera && t.push("#define SSR_ATTENUATE_FACING_CAMERA"), this._attenuateBackfaceReflection && t.push("#define SSR_ATTENUATE_BACKFACE_REFLECTION"), this._clipToFrustum && t.push("#define SSRAYTRACE_CLIP_TO_FRUSTUM"), this._useBlur() && t.push("#define SSR_USE_BLUR"), this._debug && t.push("#define SSRAYTRACE_DEBUG"), this._inputTextureColorIsInGammaSpace && t.push("#define SSR_INPUT_IS_GAMMA_SPACE"), this._generateOutputInGammaSpace && t.push("#define SSR_OUTPUT_IS_GAMMA_SPACE"), this._useFresnel && t.push("#define SSR_BLEND_WITH_FRESNEL"), this._reflectivityThreshold === 0 && t.push("#define SSR_DISABLE_REFLECTIVITY_TEST"), (e = this._ssrPostProcess) === null || e === void 0 || e.updateEffect(t.join(` `)); } _buildPipeline() { var e; if (!this.isSupported) return; if (!this._isEnabled) { this._isDirty = !0; return; } this._isDirty = !1; const t = this._scene.getEngine(); if (this._disposeDepthRenderer(), this._disposePostProcesses(), this._cameras !== null && (this._scene.postProcessRenderPipelineManager.detachCamerasFromRenderPipeline(this._name, this._cameras), this._cameras = this._camerasToBeAttached.slice()), this._reset(), this._enableAutomaticThicknessComputation) { const r = (e = this._cameras) === null || e === void 0 ? void 0 : e[0]; r && (this._depthRendererCamera = r, this._depthRenderer = new jD(this._scene, void 0, void 0, void 0, 1, !0, "SSRBackDepth"), this._depthRenderer.clearColor.r = 1e8, this._depthRenderer.reverseCulling = !0, this._depthRenderer.forceDepthWriteTransparentMeshes = this._backfaceForceDepthWriteTransparentMeshes, this._resizeDepthRenderer(), r.customRenderTargets.push(this._depthRenderer.getDepthMap())); } this._createSSRPostProcess(), this.addEffect(new gs(t, this.SSRRenderEffect, () => this._ssrPostProcess, !0)), this._useBlur() && (this._createBlurAndCombinerPostProcesses(), this.addEffect(new gs(t, this.SSRBlurRenderEffect, () => [this._blurPostProcessX, this._blurPostProcessY], !0)), this.addEffect(new gs(t, this.SSRCombineRenderEffect, () => this._blurCombinerPostProcess, !0))), this._cameras !== null && this._scene.postProcessRenderPipelineManager.attachCamerasToRenderPipeline(this._name, this._cameras); } _resizeDepthRenderer() { if (!this._depthRenderer) return; const e = this._getTextureSize(), t = this._depthRenderer.getDepthMap().getSize(), r = Math.floor(e.width / (this._backfaceDepthTextureDownsample + 1)), n = Math.floor(e.height / (this._backfaceDepthTextureDownsample + 1)); (t.width !== r || t.height !== n) && this._depthRenderer.getDepthMap().resize({ width: r, height: n }); } _disposeDepthRenderer() { var e; if (this._depthRenderer) { if (this._depthRendererCamera) { const t = (e = this._depthRendererCamera.customRenderTargets.indexOf(this._depthRenderer.getDepthMap())) !== null && e !== void 0 ? e : -1; t !== -1 && this._depthRendererCamera.customRenderTargets.splice(t, 1); } this._depthRendererCamera = null, this._depthRenderer.getDepthMap().dispose(); } this._depthRenderer = null; } _disposePostProcesses() { var e, t, r, n; for (let i = 0; i < this._cameras.length; i++) { const s = this._cameras[i]; (e = this._ssrPostProcess) === null || e === void 0 || e.dispose(s), (t = this._blurPostProcessX) === null || t === void 0 || t.dispose(s), (r = this._blurPostProcessY) === null || r === void 0 || r.dispose(s), (n = this._blurCombinerPostProcess) === null || n === void 0 || n.dispose(s); } this._ssrPostProcess = null, this._blurPostProcessX = null, this._blurPostProcessY = null, this._blurCombinerPostProcess = null; } _createSSRPostProcess() { this._ssrPostProcess = new kr("ssr", "screenSpaceReflection2", [ "projection", "invProjectionMatrix", "view", "invView", "thickness", "reflectionSpecularFalloffExponent", "strength", "stepSize", "maxSteps", "roughnessFactor", "projectionPixel", "nearPlaneZ", "maxDistance", "selfCollisionNumSkip", "vReflectionPosition", "vReflectionSize", "backSizeFactor", "reflectivityThreshold" ], ["textureSampler", "normalSampler", "reflectivitySampler", "depthSampler", "envCubeSampler", "backDepthSampler"], 1, null, this._textureType, this._scene.getEngine(), !1, "", this._textureType), this._updateEffectDefines(), this._ssrPostProcess.onApply = (e) => { this._resizeDepthRenderer(); const t = this._geometryBufferRenderer, r = this._prePassRenderer; if (!r && !t) return; if (t) { const f = t.getTextureIndex(po.REFLECTIVITY_TEXTURE_TYPE); e.setTexture("normalSampler", t.getGBuffer().textures[1]), e.setTexture("reflectivitySampler", t.getGBuffer().textures[f]), e.setTexture("depthSampler", t.getGBuffer().textures[0]); } else if (r) { const f = r.getIndex(5), o = r.getIndex(3), d = r.getIndex(6); e.setTexture("normalSampler", r.getRenderTarget().textures[d]), e.setTexture("depthSampler", r.getRenderTarget().textures[f]), e.setTexture("reflectivitySampler", r.getRenderTarget().textures[o]); } this._enableAutomaticThicknessComputation && this._depthRenderer && (e.setTexture("backDepthSampler", this._depthRenderer.getDepthMap()), e.setFloat("backSizeFactor", this._backfaceDepthTextureDownsample + 1)); const n = this._scene.activeCamera; if (!n) return; const i = n.getViewMatrix(!0), s = n.getProjectionMatrix(!0); s.invertToRef(ue.Matrix[0]), i.invertToRef(ue.Matrix[1]), e.setMatrix("projection", s), e.setMatrix("view", i), e.setMatrix("invView", ue.Matrix[1]), e.setMatrix("invProjectionMatrix", ue.Matrix[0]), e.setFloat("thickness", this.thickness), e.setFloat("reflectionSpecularFalloffExponent", this.reflectionSpecularFalloffExponent), e.setFloat("strength", this.strength), e.setFloat("stepSize", this.step), e.setFloat("maxSteps", this.maxSteps), e.setFloat("roughnessFactor", this.roughnessFactor), e.setFloat("nearPlaneZ", n.minZ), e.setFloat("maxDistance", this.maxDistance), e.setFloat("selfCollisionNumSkip", this.selfCollisionNumSkip), e.setFloat("reflectivityThreshold", this._reflectivityThreshold); const a = this._getTextureSize(); he.ScalingToRef(a.width, a.height, 1, ue.Matrix[2]), s.multiplyToRef(this._scene.getEngine().isWebGPU ? I7e : U7e, ue.Matrix[3]), ue.Matrix[3].multiplyToRef(ue.Matrix[2], ue.Matrix[4]), e.setMatrix("projectionPixel", ue.Matrix[4]), this._environmentTexture && (e.setTexture("envCubeSampler", this._environmentTexture), this._environmentTexture.boundingBoxSize && (e.setVector3("vReflectionPosition", this._environmentTexture.boundingBoxPosition), e.setVector3("vReflectionSize", this._environmentTexture.boundingBoxSize))); }, this._ssrPostProcess.samples = this.samples, this._forceGeometryBuffer || (this._ssrPostProcess._prePassEffectConfiguration = new b7e()); } _createBlurAndCombinerPostProcesses() { const e = this._scene.getEngine(); this._blurPostProcessX = new kr("SSRblurX", "screenSpaceReflection2Blur", ["texelOffsetScale"], ["textureSampler"], this._useBlur() ? 1 / (this._ssrDownsample + 1) : 1, null, 2, e, !1, "", this._textureType), this._blurPostProcessX.autoClear = !1, this._blurPostProcessX.onApplyObservable.add((i) => { var s, a; const f = (a = (s = this._blurPostProcessX) === null || s === void 0 ? void 0 : s.inputTexture.width) !== null && a !== void 0 ? a : this._scene.getEngine().getRenderWidth(); i.setFloat2("texelOffsetScale", this._blurDispersionStrength / f, 0); }), this._blurPostProcessY = new kr("SSRblurY", "screenSpaceReflection2Blur", ["texelOffsetScale"], ["textureSampler"], this._useBlur() ? 1 / (this._blurDownsample + 1) : 1, null, 2, e, !1, "", this._textureType), this._blurPostProcessY.autoClear = !1, this._blurPostProcessY.onApplyObservable.add((i) => { var s, a; const f = (a = (s = this._blurPostProcessY) === null || s === void 0 ? void 0 : s.inputTexture.height) !== null && a !== void 0 ? a : this._scene.getEngine().getRenderHeight(); i.setFloat2("texelOffsetScale", 0, this._blurDispersionStrength / f); }); const t = ["strength", "reflectionSpecularFalloffExponent", "reflectivityThreshold"], r = ["textureSampler", "mainSampler", "reflectivitySampler"]; let n = ""; this._debug && (n += `#define SSRAYTRACE_DEBUG `), this._inputTextureColorIsInGammaSpace && (n += `#define SSR_INPUT_IS_GAMMA_SPACE `), this._generateOutputInGammaSpace && (n += `#define SSR_OUTPUT_IS_GAMMA_SPACE `), this.useFresnel && (n += `#define SSR_BLEND_WITH_FRESNEL `, t.push("projection", "invProjectionMatrix"), r.push("depthSampler", "normalSampler")), this._reflectivityThreshold === 0 && (n += "#define SSR_DISABLE_REFLECTIVITY_TEST"), this._blurCombinerPostProcess = new kr("SSRblurCombiner", "screenSpaceReflection2BlurCombiner", t, r, this._useBlur() ? 1 / (this._blurDownsample + 1) : 1, null, 1, e, !1, n, this._textureType), this._blurCombinerPostProcess.autoClear = !1, this._blurCombinerPostProcess.onApplyObservable.add((i) => { var s; const a = this._geometryBufferRenderer, f = this._prePassRenderer; if (!(!f && !a)) { if (f && ((s = this._scene.activeCamera) === null || s === void 0 ? void 0 : s._getFirstPostProcess()) === this._ssrPostProcess) { const o = f.getRenderTarget(); o && o.textures && i.setTexture("mainSampler", o.textures[f.getIndex(4)]); } else i.setTextureFromPostProcess("mainSampler", this._ssrPostProcess); if (a) { const o = a.getTextureIndex(po.REFLECTIVITY_TEXTURE_TYPE); i.setTexture("reflectivitySampler", a.getGBuffer().textures[o]), this.useFresnel && (i.setTexture("normalSampler", a.getGBuffer().textures[1]), i.setTexture("depthSampler", a.getGBuffer().textures[0])); } else if (f) { const o = f.getIndex(3); if (i.setTexture("reflectivitySampler", f.getRenderTarget().textures[o]), this.useFresnel) { const d = f.getIndex(5), v = f.getIndex(6); i.setTexture("normalSampler", f.getRenderTarget().textures[v]), i.setTexture("depthSampler", f.getRenderTarget().textures[d]); } } if (i.setFloat("strength", this.strength), i.setFloat("reflectionSpecularFalloffExponent", this.reflectionSpecularFalloffExponent), i.setFloat("reflectivityThreshold", this._reflectivityThreshold), this.useFresnel) { const o = this._scene.activeCamera; if (o) { const d = o.getProjectionMatrix(); d.invertToRef(ue.Matrix[0]), i.setMatrix("projection", d), i.setMatrix("invProjectionMatrix", ue.Matrix[0]); } } } }); } /** * Serializes the rendering pipeline (Used when exporting) * @returns the serialized object */ serialize() { const e = jt.Serialize(this); return e.customType = "SSRRenderingPipeline", e; } /** * Parse the serialized pipeline * @param source Source pipeline. * @param scene The scene to load the pipeline to. * @param rootUrl The URL of the serialized pipeline. * @returns An instantiated pipeline from the serialized object. */ static Parse(e, t, r) { return jt.Parse(() => new Wa(e._name, t, e._ratio), e, t, r); } } C([ M() ], Wa.prototype, "samples", null); C([ M() ], Wa.prototype, "maxDistance", void 0); C([ M() ], Wa.prototype, "step", void 0); C([ M() ], Wa.prototype, "thickness", void 0); C([ M() ], Wa.prototype, "strength", void 0); C([ M() ], Wa.prototype, "reflectionSpecularFalloffExponent", void 0); C([ M() ], Wa.prototype, "maxSteps", void 0); C([ M() ], Wa.prototype, "roughnessFactor", void 0); C([ M() ], Wa.prototype, "selfCollisionNumSkip", void 0); C([ M() ], Wa.prototype, "_reflectivityThreshold", void 0); C([ M("_ssrDownsample") ], Wa.prototype, "_ssrDownsample", void 0); C([ M() ], Wa.prototype, "ssrDownsample", null); C([ M("blurDispersionStrength") ], Wa.prototype, "_blurDispersionStrength", void 0); C([ M("blurDownsample") ], Wa.prototype, "_blurDownsample", void 0); C([ M("enableSmoothReflections") ], Wa.prototype, "_enableSmoothReflections", void 0); C([ M("environmentTexture") ], Wa.prototype, "_environmentTexture", void 0); C([ M("environmentTextureIsProbe") ], Wa.prototype, "_environmentTextureIsProbe", void 0); C([ M("attenuateScreenBorders") ], Wa.prototype, "_attenuateScreenBorders", void 0); C([ M("attenuateIntersectionDistance") ], Wa.prototype, "_attenuateIntersectionDistance", void 0); C([ M("attenuateIntersectionIterations") ], Wa.prototype, "_attenuateIntersectionIterations", void 0); C([ M("attenuateFacingCamera") ], Wa.prototype, "_attenuateFacingCamera", void 0); C([ M("attenuateBackfaceReflection") ], Wa.prototype, "_attenuateBackfaceReflection", void 0); C([ M("clipToFrustum") ], Wa.prototype, "_clipToFrustum", void 0); C([ M("useFresnel") ], Wa.prototype, "_useFresnel", void 0); C([ M("enableAutomaticThicknessComputation") ], Wa.prototype, "_enableAutomaticThicknessComputation", void 0); C([ M("backfaceDepthTextureDownsample") ], Wa.prototype, "_backfaceDepthTextureDownsample", void 0); C([ M("backfaceForceDepthWriteTransparentMeshes") ], Wa.prototype, "_backfaceForceDepthWriteTransparentMeshes", void 0); C([ M("isEnabled") ], Wa.prototype, "_isEnabled", void 0); C([ M("inputTextureColorIsInGammaSpace") ], Wa.prototype, "_inputTextureColorIsInGammaSpace", void 0); C([ M("generateOutputInGammaSpace") ], Wa.prototype, "_generateOutputInGammaSpace", void 0); C([ M("debug") ], Wa.prototype, "_debug", void 0); Ue("BABYLON.SSRRenderingPipeline", Wa); const R7e = "tonemapPixelShader", V7e = `varying vec2 vUV;uniform sampler2D textureSampler;uniform float _ExposureAdjustment; #if defined(HABLE_TONEMAPPING) const float A=0.15;const float B=0.50;const float C=0.10;const float D=0.20;const float E=0.02;const float F=0.30;const float W=11.2; #endif float Luminance(vec3 c) {return dot(c,vec3(0.22,0.707,0.071));} #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {vec3 colour=texture2D(textureSampler,vUV).rgb; #if defined(REINHARD_TONEMAPPING) float lum=Luminance(colour.rgb); float lumTm=lum*_ExposureAdjustment;float scale=lumTm/(1.0+lumTm); colour*=scale/lum; #elif defined(HABLE_TONEMAPPING) colour*=_ExposureAdjustment;const float ExposureBias=2.0;vec3 x=ExposureBias*colour;vec3 curr=((x*(A*x+C*B)+D*E)/(x*(A*x+B)+D*F))-E/F;x=vec3(W,W,W);vec3 whiteScale=1.0/(((x*(A*x+C*B)+D*E)/(x*(A*x+B)+D*F))-E/F);colour=curr*whiteScale; #elif defined(OPTIMIZED_HEJIDAWSON_TONEMAPPING) colour*=_ExposureAdjustment;vec3 X=max(vec3(0.0,0.0,0.0),colour-0.004);vec3 retColor=(X*(6.2*X+0.5))/(X*(6.2*X+1.7)+0.06);colour=retColor*retColor; #elif defined(PHOTOGRAPHIC_TONEMAPPING) colour= vec3(1.0,1.0,1.0)-exp2(-_ExposureAdjustment*colour); #endif gl_FragColor=vec4(colour.rgb,1.0);}`; Le.ShadersStore[R7e] = V7e; var _w; (function(A) { A[A.Hable = 0] = "Hable", A[A.Reinhard = 1] = "Reinhard", A[A.HejiDawson = 2] = "HejiDawson", A[A.Photographic = 3] = "Photographic"; })(_w || (_w = {})); class C7e extends kr { /** * Gets a string identifying the name of the class * @returns "TonemapPostProcess" string */ getClassName() { return "TonemapPostProcess"; } /** * Creates a new TonemapPostProcess * @param name defines the name of the postprocess * @param _operator defines the operator to use * @param exposureAdjustment defines the required exposure adjustment * @param camera defines the camera to use (can be null) * @param samplingMode defines the required sampling mode (BABYLON.Texture.BILINEAR_SAMPLINGMODE by default) * @param engine defines the hosting engine (can be ignore if camera is set) * @param textureFormat defines the texture format to use (BABYLON.Engine.TEXTURETYPE_UNSIGNED_INT by default) * @param reusable If the post process can be reused on the same frame. (default: false) */ constructor(e, t, r, n, i = 2, s, a = 0, f) { super(e, "tonemap", ["_ExposureAdjustment"], null, 1, n, i, s, f, null, a), this._operator = t, this.exposureAdjustment = r; let o = "#define "; this._operator === _w.Hable ? o += "HABLE_TONEMAPPING" : this._operator === _w.Reinhard ? o += "REINHARD_TONEMAPPING" : this._operator === _w.HejiDawson ? o += "OPTIMIZED_HEJIDAWSON_TONEMAPPING" : this._operator === _w.Photographic && (o += "PHOTOGRAPHIC_TONEMAPPING"), this.updateEffect(o), this.onApply = (d) => { d.setFloat("_ExposureAdjustment", this.exposureAdjustment); }; } } const O7e = "volumetricLightScatteringPixelShader", y7e = `uniform sampler2D textureSampler;uniform sampler2D lightScatteringSampler;uniform float decay;uniform float exposure;uniform float weight;uniform float density;uniform vec2 meshPositionOnScreen;varying vec2 vUV; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #define CUSTOM_FRAGMENT_MAIN_BEGIN vec2 tc=vUV;vec2 deltaTexCoord=(tc-meshPositionOnScreen.xy);deltaTexCoord*=1.0/float(NUM_SAMPLES)*density;float illuminationDecay=1.0;vec4 color=texture2D(lightScatteringSampler,tc)*0.4;for(int i=0; i #include #include #include[0..maxSimultaneousMorphTargets] #include uniform mat4 viewProjection;uniform vec2 depthValues; #if defined(ALPHATEST) || defined(NEED_UV) varying vec2 vUV;uniform mat4 diffuseMatrix; #ifdef UV1 attribute vec2 uv; #endif #ifdef UV2 attribute vec2 uv2; #endif #endif #define CUSTOM_VERTEX_DEFINITIONS void main(void) {vec3 positionUpdated=position; #if (defined(ALPHATEST) || defined(NEED_UV)) && defined(UV1) vec2 uvUpdated=uv; #endif #include #include[0..maxSimultaneousMorphTargets] #include #include #include gl_Position=viewProjection*finalWorld*vec4(positionUpdated,1.0); #if defined(ALPHATEST) || defined(BASIC_RENDER) #ifdef UV1 vUV=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0)); #endif #ifdef UV2 vUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0)); #endif #endif } `; Le.ShadersStore[k7e] = E7e; const F7e = "volumetricLightScatteringPassPixelShader", N7e = `#if defined(ALPHATEST) || defined(NEED_UV) varying vec2 vUV; #endif #if defined(ALPHATEST) uniform sampler2D diffuseSampler; #endif #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #if defined(ALPHATEST) vec4 diffuseColor=texture2D(diffuseSampler,vUV);if (diffuseColor.a<0.4) discard; #endif gl_FragColor=vec4(0.0,0.0,0.0,1.0);} `; Le.ShadersStore[F7e] = N7e; class Bc extends kr { /** * @internal * VolumetricLightScatteringPostProcess.useDiffuseColor is no longer used, use the mesh material directly instead */ get useDiffuseColor() { return Se.Warn("VolumetricLightScatteringPostProcess.useDiffuseColor is no longer used, use the mesh material directly instead"), !1; } set useDiffuseColor(e) { Se.Warn("VolumetricLightScatteringPostProcess.useDiffuseColor is no longer used, use the mesh material directly instead"); } /** * @constructor * @param name The post-process name * @param ratio The size of the post-process and/or internal pass (0.5 means that your postprocess will have a width = canvas.width 0.5 and a height = canvas.height 0.5) * @param camera The camera that the post-process will be attached to * @param mesh The mesh used to create the light scattering * @param samples The post-process quality, default 100 * @param samplingMode The post-process filtering mode * @param engine The babylon engine * @param reusable If the post-process is reusable * @param scene The constructor needs a scene reference to initialize internal components. If "camera" is null a "scene" must be provided */ constructor(e, t, r, n, i = 100, s = We.BILINEAR_SAMPLINGMODE, a, f, o) { var d, v; super(e, "volumetricLightScattering", ["decay", "exposure", "weight", "meshPositionOnScreen", "density"], ["lightScatteringSampler"], t.postProcessRatio || t, r, s, a, f, "#define NUM_SAMPLES " + i), this._screenCoordinates = at.Zero(), this.customMeshPosition = S.Zero(), this.useCustomMeshPosition = !1, this.invert = !0, this.excludedMeshes = [], this.includedMeshes = [], this.exposure = 0.3, this.decay = 0.96815, this.weight = 0.58767, this.density = 0.926, o = (v = (d = r == null ? void 0 : r.getScene()) !== null && d !== void 0 ? d : o) !== null && v !== void 0 ? v : this._scene, a = o.getEngine(), this._viewPort = new WA(0, 0, 1, 1).toGlobal(a.getRenderWidth(), a.getRenderHeight()), this.mesh = n ?? Bc.CreateDefaultMesh("VolumetricLightScatteringMesh", o), this._createPass(o, t.passRatio || t), this.onActivate = (u) => { this.isSupported || this.dispose(u), this.onActivate = null; }, this.onApplyObservable.add((u) => { this._updateMeshScreenCoordinates(o), u.setTexture("lightScatteringSampler", this._volumetricLightScatteringRTT), u.setFloat("exposure", this.exposure), u.setFloat("decay", this.decay), u.setFloat("weight", this.weight), u.setFloat("density", this.density), u.setVector2("meshPositionOnScreen", this._screenCoordinates); }); } /** * Returns the string "VolumetricLightScatteringPostProcess" * @returns "VolumetricLightScatteringPostProcess" */ getClassName() { return "VolumetricLightScatteringPostProcess"; } _isReady(e, t) { var r; const n = e.getMesh(); if (n === this.mesh && n.material) return n.material.isReady(n); const i = (r = n._internalAbstractMeshDataInfo._materialForRenderPass) === null || r === void 0 ? void 0 : r[this._scene.getEngine().currentRenderPassId]; if (i) return i.isReadyForSubMesh(n, e, t); const s = [], a = [J.PositionKind], f = e.getMaterial(); f && (f.needAlphaTesting() && s.push("#define ALPHATEST"), n.isVerticesDataPresent(J.UVKind) && (a.push(J.UVKind), s.push("#define UV1")), n.isVerticesDataPresent(J.UV2Kind) && (a.push(J.UV2Kind), s.push("#define UV2"))), n.useBones && n.computeBonesUsingShaders ? (a.push(J.MatricesIndicesKind), a.push(J.MatricesWeightsKind), s.push("#define NUM_BONE_INFLUENCERS " + n.numBoneInfluencers), s.push("#define BonesPerMesh " + (n.skeleton ? n.skeleton.bones.length + 1 : 0))) : s.push("#define NUM_BONE_INFLUENCERS 0"), t && (s.push("#define INSTANCES"), Ye.PushAttributesForInstances(a), e.getRenderingMesh().hasThinInstances && s.push("#define THIN_INSTANCES")); const o = e._getDrawWrapper(void 0, !0), d = o.defines, v = s.join(` `); return d !== v && o.setEffect(n.getScene().getEngine().createEffect("volumetricLightScatteringPass", a, ["world", "mBones", "viewProjection", "diffuseMatrix"], ["diffuseSampler"], v, void 0, void 0, void 0, { maxSimultaneousMorphTargets: n.numBoneInfluencers }), v), o.effect.isReady(); } /** * Sets the new light position for light scattering effect * @param position The new custom light position */ setCustomMeshPosition(e) { this.customMeshPosition = e; } /** * Returns the light position for light scattering effect * @returns Vector3 The custom light position */ getCustomMeshPosition() { return this.customMeshPosition; } /** * Disposes the internal assets and detaches the post-process from the camera * @param camera */ dispose(e) { const t = e.getScene().customRenderTargets.indexOf(this._volumetricLightScatteringRTT); t !== -1 && e.getScene().customRenderTargets.splice(t, 1), this._volumetricLightScatteringRTT.dispose(), super.dispose(e); } /** * Returns the render target texture used by the post-process * @returns the render target texture used by the post-process */ getPass() { return this._volumetricLightScatteringRTT; } // Private methods _meshExcluded(e) { return this.includedMeshes.length > 0 && this.includedMeshes.indexOf(e) === -1 || this.excludedMeshes.length > 0 && this.excludedMeshes.indexOf(e) !== -1; } _createPass(e, t) { const r = e.getEngine(); this._volumetricLightScatteringRTT = new Ta("volumetricLightScatteringMap", { width: r.getRenderWidth() * t, height: r.getRenderHeight() * t }, e, !1, !0, 0), this._volumetricLightScatteringRTT.wrapU = We.CLAMP_ADDRESSMODE, this._volumetricLightScatteringRTT.wrapV = We.CLAMP_ADDRESSMODE, this._volumetricLightScatteringRTT.renderList = null, this._volumetricLightScatteringRTT.renderParticles = !1, this._volumetricLightScatteringRTT.ignoreCameraViewport = !0; const n = this.getCamera(); n ? n.customRenderTargets.push(this._volumetricLightScatteringRTT) : e.customRenderTargets.push(this._volumetricLightScatteringRTT); const i = (f) => { var o; const d = f.getRenderingMesh(), v = f.getEffectiveMesh(); if (this._meshExcluded(d)) return; v._internalAbstractMeshDataInfo._isActiveIntermediate = !1; const u = f.getMaterial(); if (!u) return; const l = d.getScene(), P = l.getEngine(); P.setState(u.backFaceCulling, void 0, void 0, void 0, u.cullBackFaces); const p = d._getInstancesRenderList(f._id, !!f.getReplacementMesh()); if (p.mustReturn) return; const c = P.getCaps().instancedArrays && (p.visibleInstances[f._id] !== null || d.hasThinInstances); if (this._isReady(f, c)) { const H = (o = v._internalAbstractMeshDataInfo._materialForRenderPass) === null || o === void 0 ? void 0 : o[P.currentRenderPassId]; let T = f._getDrawWrapper(); if (d === this.mesh && !T && (T = u._getDrawWrapper()), !T) return; const q = T.effect; if (P.enableEffect(T), c || d._bind(f, q, u.fillMode), d === this.mesh) u.bind(v.getWorldMatrix(), d); else if (H) H.bindForSubMesh(v.getWorldMatrix(), v, f); else { if (q.setMatrix("viewProjection", l.getTransformMatrix()), u && u.needAlphaTesting()) { const b = u.getAlphaTestTexture(); q.setTexture("diffuseSampler", b), b && q.setMatrix("diffuseMatrix", b.getTextureMatrix()); } d.useBones && d.computeBonesUsingShaders && d.skeleton && q.setMatrices("mBones", d.skeleton.getTransformMatrices(d)); } c && d.hasThinInstances && q.setMatrix("world", v.getWorldMatrix()), d._processRendering(v, f, q, gt.TriangleFillMode, p, c, (b, j) => { b || q.setMatrix("world", j); }); } }; let s; const a = new xt(0, 0, 0, 1); this._volumetricLightScatteringRTT.onBeforeRenderObservable.add(() => { s = e.clearColor, e.clearColor = a; }), this._volumetricLightScatteringRTT.onAfterRenderObservable.add(() => { e.clearColor = s; }), this._volumetricLightScatteringRTT.customIsReadyFunction = (f, o, d) => { if ((d || o === 0) && f.subMeshes) for (let v = 0; v < f.subMeshes.length; ++v) { const u = f.subMeshes[v], l = u.getMaterial(), P = u.getRenderingMesh(); if (!l) continue; const p = P._getInstancesRenderList(u._id, !!u.getReplacementMesh()), c = r.getCaps().instancedArrays && (p.visibleInstances[u._id] !== null || P.hasThinInstances); if (!this._isReady(u, c)) return !1; } return !0; }, this._volumetricLightScatteringRTT.customRenderFunction = (f, o, d, v) => { const u = e.getEngine(); let l; if (v.length) { for (u.setColorWrite(!1), l = 0; l < v.length; l++) i(v.data[l]); u.setColorWrite(!0); } for (l = 0; l < f.length; l++) i(f.data[l]); for (l = 0; l < o.length; l++) i(o.data[l]); if (d.length) { for (l = 0; l < d.length; l++) { const p = d.data[l], c = p.getBoundingInfo(); c && e.activeCamera && (p._alphaIndex = p.getMesh().alphaIndex, p._distanceToCamera = c.boundingSphere.centerWorld.subtract(e.activeCamera.position).length()); } const P = d.data.slice(0, d.length); for (P.sort((p, c) => p._alphaIndex > c._alphaIndex ? 1 : p._alphaIndex < c._alphaIndex ? -1 : p._distanceToCamera < c._distanceToCamera ? 1 : p._distanceToCamera > c._distanceToCamera ? -1 : 0), u.setAlphaMode(2), l = 0; l < P.length; l++) i(P[l]); u.setAlphaMode(0); } }; } _updateMeshScreenCoordinates(e) { const t = e.getTransformMatrix(); let r; this.useCustomMeshPosition ? r = this.customMeshPosition : this.attachedNode ? r = this.attachedNode.position : r = this.mesh.parent ? this.mesh.getAbsolutePosition() : this.mesh.position; const n = S.Project(r, he.Identity(), t, this._viewPort); this._screenCoordinates.x = n.x / this._viewPort.width, this._screenCoordinates.y = n.y / this._viewPort.height, this.invert && (this._screenCoordinates.y = 1 - this._screenCoordinates.y); } // Static methods /** * Creates a default mesh for the Volumeric Light Scattering post-process * @param name The mesh name * @param scene The scene where to create the mesh * @returns the default mesh */ static CreateDefaultMesh(e, t) { const r = u4(e, { size: 1 }, t); r.billboardMode = jn.BILLBOARDMODE_ALL; const n = new Wt(e + "Material", t); return n.emissiveColor = new Ne(1, 1, 1), r.material = n, r; } } C([ fo() ], Bc.prototype, "customMeshPosition", void 0); C([ M() ], Bc.prototype, "useCustomMeshPosition", void 0); C([ M() ], Bc.prototype, "invert", void 0); C([ tU() ], Bc.prototype, "mesh", void 0); C([ M() ], Bc.prototype, "excludedMeshes", void 0); C([ M() ], Bc.prototype, "includedMeshes", void 0); C([ M() ], Bc.prototype, "exposure", void 0); C([ M() ], Bc.prototype, "decay", void 0); C([ M() ], Bc.prototype, "weight", void 0); C([ M() ], Bc.prototype, "density", void 0); Ue("BABYLON.VolumetricLightScatteringPostProcess", Bc); const Q7e = "screenSpaceCurvaturePixelShader", Y7e = `precision highp float;varying vec2 vUV;uniform sampler2D textureSampler;uniform sampler2D normalSampler;uniform float curvature_ridge;uniform float curvature_valley; #ifndef CURVATURE_OFFSET #define CURVATURE_OFFSET 1 #endif float curvature_soft_clamp(float curvature,float control) {if (curvature<0.5/control) return curvature*(1.0-curvature*control);return 0.25/control;} float calculate_curvature(ivec2 texel,float ridge,float valley) {vec2 normal_up =texelFetch(normalSampler,texel+ivec2(0, CURVATURE_OFFSET),0).rb;vec2 normal_down =texelFetch(normalSampler,texel+ivec2(0,-CURVATURE_OFFSET),0).rb;vec2 normal_left =texelFetch(normalSampler,texel+ivec2(-CURVATURE_OFFSET,0),0).rb;vec2 normal_right=texelFetch(normalSampler,texel+ivec2( CURVATURE_OFFSET,0),0).rb;float normal_diff=((normal_up.g-normal_down.g)+(normal_right.r-normal_left.r));if (normal_diff<0.0) return -2.0*curvature_soft_clamp(-normal_diff,valley);return 2.0*curvature_soft_clamp(normal_diff,ridge);} #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {ivec2 texel=ivec2(gl_FragCoord.xy);vec4 baseColor=texture2D(textureSampler,vUV);float curvature=calculate_curvature(texel,curvature_ridge,curvature_valley);baseColor.rgb*=curvature+1.0;gl_FragColor=baseColor;}`; Le.ShadersStore[Q7e] = Y7e; class hU extends kr { /** * Gets a string identifying the name of the class * @returns "ScreenSpaceCurvaturePostProcess" string */ getClassName() { return "ScreenSpaceCurvaturePostProcess"; } /** * Creates a new instance ScreenSpaceCurvaturePostProcess * @param name The name of the effect. * @param scene The scene containing the objects to blur according to their velocity. * @param options The required width/height ratio to downsize to before computing the render pass. * @param camera The camera to apply the render pass to. * @param samplingMode The sampling mode to be used when computing the pass. (default: 0) * @param engine The engine which the post process will be applied. (default: current engine) * @param reusable If the post process can be reused on the same frame. (default: false) * @param textureType Type of textures used when performing the post process. (default: 0) * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false) */ constructor(e, t, r, n, i, s, a, f = 0, o = !1) { super(e, "screenSpaceCurvature", ["curvature_ridge", "curvature_valley"], ["textureSampler", "normalSampler"], r, n, i, s, a, void 0, f, void 0, null, o), this.ridge = 1, this.valley = 1, this._geometryBufferRenderer = t.enableGeometryBufferRenderer(), this._geometryBufferRenderer ? (this._geometryBufferRenderer.generateNormalsInWorldSpace && console.error("ScreenSpaceCurvaturePostProcess does not support generateNormalsInWorldSpace=true for the geometry buffer renderer!"), this.onApply = (d) => { d.setFloat("curvature_ridge", 0.5 / Math.max(this.ridge * this.ridge, 1e-4)), d.setFloat("curvature_valley", 0.7 / Math.max(this.valley * this.valley, 1e-4)); const v = this._geometryBufferRenderer.getGBuffer().textures[1]; d.setTexture("normalSampler", v); }) : Se.Error("Multiple Render Target support needed for screen space curvature post process. Please use IsSupported test first."); } /** * Support test. */ static get IsSupported() { const e = gr.LastCreatedEngine; return e ? e.getCaps().drawBuffersExtension : !1; } /** * @internal */ static _Parse(e, t, r, n) { return jt.Parse(() => new hU(e.name, r, e.options, t, e.renderTargetSamplingMode, r.getEngine(), e.textureType, e.reusable), e, r, n); } } C([ M() ], hU.prototype, "ridge", void 0); C([ M() ], hU.prototype, "valley", void 0); Ue("BABYLON.ScreenSpaceCurvaturePostProcess", hU); const M7e = "boundingBoxRendererFragmentDeclaration", L7e = `uniform vec4 color; `; Le.IncludesShadersStore[M7e] = L7e; const K7e = "boundingBoxRendererUboDeclaration", J7e = `#ifdef WEBGL2 uniform vec4 color;uniform mat4 world;uniform mat4 viewProjection; #ifdef MULTIVIEW uniform mat4 viewProjectionR; #endif #else layout(std140,column_major) uniform;uniform BoundingBoxRenderer {vec4 color;mat4 world;mat4 viewProjection;mat4 viewProjectionR;}; #endif `; Le.IncludesShadersStore[K7e] = J7e; const z7e = "boundingBoxRendererPixelShader", G7e = `#include<__decl__boundingBoxRendererFragment> #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #define CUSTOM_FRAGMENT_MAIN_BEGIN gl_FragColor=color; #define CUSTOM_FRAGMENT_MAIN_END }`; Le.ShadersStore[z7e] = G7e; const Z7e = "boundingBoxRendererVertexDeclaration", _7e = `uniform mat4 world;uniform mat4 viewProjection; #ifdef MULTIVIEW uniform mat4 viewProjectionR; #endif `; Le.IncludesShadersStore[Z7e] = _7e; const $7e = "boundingBoxRendererVertexShader", eHe = `attribute vec3 position; #include<__decl__boundingBoxRendererVertex> #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN vec4 worldPos=world*vec4(position,1.0); #ifdef MULTIVIEW if (gl_ViewID_OVR==0u) {gl_Position=viewProjection*worldPos;} else {gl_Position=viewProjectionR*worldPos;} #else gl_Position=viewProjection*worldPos; #endif #define CUSTOM_VERTEX_MAIN_END } `; Le.ShadersStore[$7e] = eHe; Object.defineProperty(sr.prototype, "forceShowBoundingBoxes", { get: function() { return this._forceShowBoundingBoxes || !1; }, set: function(A) { this._forceShowBoundingBoxes = A, A && this.getBoundingBoxRenderer(); }, enumerable: !0, configurable: !0 }); sr.prototype.getBoundingBoxRenderer = function() { return this._boundingBoxRenderer || (this._boundingBoxRenderer = new sie(this)), this._boundingBoxRenderer; }; Object.defineProperty(jn.prototype, "showBoundingBox", { get: function() { return this._showBoundingBox || !1; }, set: function(A) { this._showBoundingBox = A, A && this.getScene().getBoundingBoxRenderer(); }, enumerable: !0, configurable: !0 }); class sie { /** * Instantiates a new bounding box renderer in a scene. * @param scene the scene the renderer renders in */ constructor(e) { this.name = Ot.NAME_BOUNDINGBOXRENDERER, this.frontColor = new Ne(1, 1, 1), this.backColor = new Ne(0.1, 0.1, 0.1), this.showBackLines = !0, this.onBeforeBoxRenderingObservable = new Oe(), this.onAfterBoxRenderingObservable = new Oe(), this.onResourcesReadyObservable = new Oe(), this.enabled = !0, this.renderList = new qf(32), this._vertexBuffers = {}, this._fillIndexBuffer = null, this._fillIndexData = null, this.scene = e, e._addComponent(this), this._uniformBufferFront = new yr(this.scene.getEngine(), void 0, void 0, "BoundingBoxRendererFront", !this.scene.getEngine().isWebGPU), this._buildUniformLayout(this._uniformBufferFront), this._uniformBufferBack = new yr(this.scene.getEngine(), void 0, void 0, "BoundingBoxRendererBack", !this.scene.getEngine().isWebGPU), this._buildUniformLayout(this._uniformBufferBack); } _buildUniformLayout(e) { e.addUniform("color", 4), e.addUniform("world", 16), e.addUniform("viewProjection", 16), e.addUniform("viewProjectionR", 16), e.create(); } /** * Registers the component in a given scene */ register() { this.scene._beforeEvaluateActiveMeshStage.registerStep(Ot.STEP_BEFOREEVALUATEACTIVEMESH_BOUNDINGBOXRENDERER, this, this.reset), this.scene._preActiveMeshStage.registerStep(Ot.STEP_PREACTIVEMESH_BOUNDINGBOXRENDERER, this, this._preActiveMesh), this.scene._evaluateSubMeshStage.registerStep(Ot.STEP_EVALUATESUBMESH_BOUNDINGBOXRENDERER, this, this._evaluateSubMesh), this.scene._afterRenderingGroupDrawStage.registerStep(Ot.STEP_AFTERRENDERINGGROUPDRAW_BOUNDINGBOXRENDERER, this, this.render); } _evaluateSubMesh(e, t) { if (e.showSubMeshesBoundingBox) { const r = t.getBoundingInfo(); r != null && (r.boundingBox._tag = e.renderingGroupId, this.renderList.push(r.boundingBox)); } } _preActiveMesh(e) { if (e.showBoundingBox || this.scene.forceShowBoundingBoxes) { const t = e.getBoundingInfo(); t.boundingBox._tag = e.renderingGroupId, this.renderList.push(t.boundingBox); } } _prepareResources() { if (this._colorShader) return; this._colorShader = new Zo("colorShader", this.scene, "boundingBoxRenderer", { attributes: [J.PositionKind], uniforms: ["world", "viewProjection", "viewProjectionR", "color"], uniformBuffers: ["BoundingBoxRenderer"] }, !1), this._colorShader.doNotSerialize = !0, this._colorShader.reservedDataStore = { hidden: !0 }, this._colorShaderForOcclusionQuery = new Zo("colorShaderOccQuery", this.scene, "boundingBoxRenderer", { attributes: [J.PositionKind], uniforms: ["world", "viewProjection", "viewProjectionR", "color"], uniformBuffers: ["BoundingBoxRenderer"] }, !0), this._colorShaderForOcclusionQuery.doNotSerialize = !0, this._colorShaderForOcclusionQuery.reservedDataStore = { hidden: !0 }; const e = this.scene.getEngine(), t = LO({ size: 1 }); this._vertexBuffers[J.PositionKind] = new J(e, t.positions, J.PositionKind, !1), this._createIndexBuffer(), this._fillIndexData = t.indices, this.onResourcesReadyObservable.notifyObservers(this); } _createIndexBuffer() { const e = this.scene.getEngine(); this._indexBuffer = e.createIndexBuffer([0, 1, 1, 2, 2, 3, 3, 0, 4, 5, 5, 6, 6, 7, 7, 4, 0, 7, 1, 6, 2, 5, 3, 4]); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { const e = this._vertexBuffers[J.PositionKind]; e && e._rebuild(), this._createIndexBuffer(); } /** * @internal */ reset() { this.renderList.reset(); } /** * Render the bounding boxes of a specific rendering group * @param renderingGroupId defines the rendering group to render */ render(e) { var t, r; if (this.renderList.length === 0 || !this.enabled || (this._prepareResources(), !this._colorShader.isReady())) return; const n = this.scene.getEngine(); n.setDepthWrite(!1); const i = this.scene.getTransformMatrix(); for (let s = 0; s < this.renderList.length; s++) { const a = this.renderList.data[s]; if (a._tag !== e) continue; this._createWrappersForBoundingBox(a), this.onBeforeBoxRenderingObservable.notifyObservers(a); const f = a.minimum, d = a.maximum.subtract(f), v = f.add(d.scale(0.5)), u = he.Scaling(d.x, d.y, d.z).multiply(he.Translation(v.x, v.y, v.z)).multiply(a.getWorldMatrix()), l = n.useReverseDepthBuffer; if (this.showBackLines) { const p = (t = a._drawWrapperBack) !== null && t !== void 0 ? t : this._colorShader._getDrawWrapper(); this._colorShader._preBind(p), n.bindBuffers(this._vertexBuffers, this._indexBuffer, this._colorShader.getEffect()), l ? n.setDepthFunctionToLessOrEqual() : n.setDepthFunctionToGreaterOrEqual(), this._uniformBufferBack.bindToEffect(p.effect, "BoundingBoxRenderer"), this._uniformBufferBack.updateColor4("color", this.backColor, 1), this._uniformBufferBack.updateMatrix("world", u), this._uniformBufferBack.updateMatrix("viewProjection", i), this._uniformBufferBack.update(), n.drawElementsType(gt.LineListDrawMode, 0, 24); } const P = (r = a._drawWrapperFront) !== null && r !== void 0 ? r : this._colorShader._getDrawWrapper(); this._colorShader._preBind(P), n.bindBuffers(this._vertexBuffers, this._indexBuffer, this._colorShader.getEffect()), l ? n.setDepthFunctionToGreater() : n.setDepthFunctionToLess(), this._uniformBufferFront.bindToEffect(P.effect, "BoundingBoxRenderer"), this._uniformBufferFront.updateColor4("color", this.frontColor, 1), this._uniformBufferFront.updateMatrix("world", u), this._uniformBufferFront.updateMatrix("viewProjection", i), this._uniformBufferFront.update(), n.drawElementsType(gt.LineListDrawMode, 0, 24), this.onAfterBoxRenderingObservable.notifyObservers(a); } this._colorShader.unbind(), n.setDepthFunctionToLessOrEqual(), n.setDepthWrite(!0); } _createWrappersForBoundingBox(e) { if (!e._drawWrapperFront) { const t = this.scene.getEngine(); e._drawWrapperFront = new zo(t), e._drawWrapperBack = new zo(t), e._drawWrapperFront.setEffect(this._colorShader.getEffect()), e._drawWrapperBack.setEffect(this._colorShader.getEffect()); } } /** * In case of occlusion queries, we can render the occlusion bounding box through this method * @param mesh Define the mesh to render the occlusion bounding box for */ renderOcclusionBoundingBox(e) { const t = this.scene.getEngine(); this._renderPassIdForOcclusionQuery === void 0 && (this._renderPassIdForOcclusionQuery = t.createRenderPassId("Render pass for occlusion query")); const r = t.currentRenderPassId; t.currentRenderPassId = this._renderPassIdForOcclusionQuery, this._prepareResources(); const n = e.subMeshes[0]; if (!this._colorShaderForOcclusionQuery.isReady(e, void 0, n) || !e.hasBoundingInfo) { t.currentRenderPassId = r; return; } this._fillIndexBuffer || (this._fillIndexBuffer = t.createIndexBuffer(this._fillIndexData)); const i = t.useReverseDepthBuffer; t.setDepthWrite(!1), t.setColorWrite(!1); const s = e.getBoundingInfo().boundingBox, a = s.minimum, o = s.maximum.subtract(a), d = a.add(o.scale(0.5)), v = he.Scaling(o.x, o.y, o.z).multiply(he.Translation(d.x, d.y, d.z)).multiply(s.getWorldMatrix()), u = n._drawWrapper; this._colorShaderForOcclusionQuery._preBind(u), t.bindBuffers(this._vertexBuffers, this._fillIndexBuffer, u.effect), i ? t.setDepthFunctionToGreater() : t.setDepthFunctionToLess(), this.scene.resetCachedMaterial(), this._uniformBufferFront.bindToEffect(u.effect, "BoundingBoxRenderer"), this._uniformBufferFront.updateMatrix("world", v), this._uniformBufferFront.updateMatrix("viewProjection", this.scene.getTransformMatrix()), this._uniformBufferFront.update(), t.drawElementsType(gt.TriangleFillMode, 0, 36), this._colorShaderForOcclusionQuery.unbind(), t.setDepthFunctionToLessOrEqual(), t.setDepthWrite(!0), t.setColorWrite(!0), t.currentRenderPassId = r; } /** * Dispose and release the resources attached to this renderer. */ dispose() { if (this._renderPassIdForOcclusionQuery !== void 0 && (this.scene.getEngine().releaseRenderPassId(this._renderPassIdForOcclusionQuery), this._renderPassIdForOcclusionQuery = void 0), !this._colorShader) return; this.onBeforeBoxRenderingObservable.clear(), this.onAfterBoxRenderingObservable.clear(), this.onResourcesReadyObservable.clear(), this.renderList.dispose(), this._colorShader.dispose(), this._colorShaderForOcclusionQuery.dispose(), this._uniformBufferFront.dispose(), this._uniformBufferBack.dispose(); const e = this._vertexBuffers[J.PositionKind]; e && (e.dispose(), this._vertexBuffers[J.PositionKind] = null), this.scene.getEngine()._releaseBuffer(this._indexBuffer), this._fillIndexBuffer && (this.scene.getEngine()._releaseBuffer(this._fillIndexBuffer), this._fillIndexBuffer = null); } } sr.prototype.enableDepthRenderer = function(A, e = !1, t = !1, r = 3, n = !1) { if (A = A || this.activeCamera, !A) throw "No camera available to enable depth renderer"; if (this._depthRenderer || (this._depthRenderer = {}), !this._depthRenderer[A.id]) { const i = !!this.getEngine().getCaps().textureFloatRender; let s = 0; this.getEngine().getCaps().textureHalfFloatRender && (!t || !i) ? s = 2 : i ? s = 1 : s = 0, this._depthRenderer[A.id] = new jD(this, s, A, e, r, n); } return this._depthRenderer[A.id]; }; sr.prototype.disableDepthRenderer = function(A) { A = A || this.activeCamera, !(!A || !this._depthRenderer || !this._depthRenderer[A.id]) && this._depthRenderer[A.id].dispose(); }; class aie { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_DEPTHRENDERER, this.scene = e; } /** * Registers the component in a given scene */ register() { this.scene._gatherRenderTargetsStage.registerStep(Ot.STEP_GATHERRENDERTARGETS_DEPTHRENDERER, this, this._gatherRenderTargets), this.scene._gatherActiveCameraRenderTargetsStage.registerStep(Ot.STEP_GATHERACTIVECAMERARENDERTARGETS_DEPTHRENDERER, this, this._gatherActiveCameraRenderTargets); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { } /** * Disposes the component and the associated resources */ dispose() { for (const e in this.scene._depthRenderer) this.scene._depthRenderer[e].dispose(); } _gatherRenderTargets(e) { if (this.scene._depthRenderer) for (const t in this.scene._depthRenderer) { const r = this.scene._depthRenderer[t]; r.enabled && !r.useOnlyInActiveCamera && e.push(r.getDepthMap()); } } _gatherActiveCameraRenderTargets(e) { if (this.scene._depthRenderer) for (const t in this.scene._depthRenderer) { const r = this.scene._depthRenderer[t]; r.enabled && r.useOnlyInActiveCamera && this.scene.activeCamera.id === t && e.push(r.getDepthMap()); } } } jD._SceneComponentInitialization = (A) => { let e = A._getComponent(Ot.NAME_DEPTHRENDERER); e || (e = new aie(A), A._addComponent(e)); }; const tHe = "oitFinalPixelShader", rHe = `precision highp float;uniform sampler2D uFrontColor;uniform sampler2D uBackColor;void main() {ivec2 fragCoord=ivec2(gl_FragCoord.xy);vec4 frontColor=texelFetch(uFrontColor,fragCoord,0);vec4 backColor=texelFetch(uBackColor,fragCoord,0);float alphaMultiplier=1.0-frontColor.a;glFragColor=vec4( frontColor.rgb+alphaMultiplier*backColor.rgb, frontColor.a+backColor.a );}`; Le.ShadersStore[tHe] = rHe; const nHe = "oitBackBlendPixelShader", iHe = `precision highp float;uniform sampler2D uBackColor;void main() {glFragColor=texelFetch(uBackColor,ivec2(gl_FragCoord.xy),0);if (glFragColor.a==0.0) { discard;}}`; Le.ShadersStore[nHe] = iHe; class sHe { constructor() { this.enabled = !0, this.name = "depthPeeling", this.texturesRequired = [4]; } } class J2 { /** * Number of depth peeling passes. As we are using dual depth peeling, each pass two levels of transparency are processed. */ get passCount() { return this._passCount; } set passCount(e) { this._passCount !== e && (this._passCount = e, this._createRenderPassIds()); } /** * Instructs the renderer to use render passes. It is an optimization that makes the rendering faster for some engines (like WebGPU) but that consumes more memory, so it is disabled by default. */ get useRenderPasses() { return this._useRenderPasses; } set useRenderPasses(e) { this._useRenderPasses !== e && (this._useRenderPasses = e, this._createRenderPassIds()); } /** * Add a mesh in the exclusion list to prevent it to be handled by the depth peeling renderer * @param mesh The mesh to exclude from the depth peeling renderer */ addExcludedMesh(e) { this._excludedMeshes.indexOf(e.uniqueId) === -1 && this._excludedMeshes.push(e.uniqueId); } /** * Remove a mesh from the exclusion list of the depth peeling renderer * @param mesh The mesh to remove */ removeExcludedMesh(e) { const t = this._excludedMeshes.indexOf(e.uniqueId); t !== -1 && this._excludedMeshes.splice(t, 1); } /** * Instanciates the depth peeling renderer * @param scene Scene to attach to * @param passCount Number of depth layers to peel * @returns The depth peeling renderer */ constructor(e, t = 5) { if (this._thinTextures = [], this._currentPingPongState = 0, this._layoutCacheFormat = [[!0], [!0, !0], [!0, !0, !0]], this._layoutCache = [], this._candidateSubMeshes = new qf(10), this._excludedSubMeshes = new qf(10), this._excludedMeshes = [], this._colorCache = [ new xt(J2._DEPTH_CLEAR_VALUE, J2._DEPTH_CLEAR_VALUE, 0, 0), new xt(-J2._MIN_DEPTH, J2._MAX_DEPTH, 0, 0), new xt(0, 0, 0, 0) ], this._scene = e, this._engine = e.getEngine(), this._passCount = t, !e.enablePrePassRenderer()) { Se.Warn("Depth peeling for order independant transparency could not enable PrePass, aborting."); return; } for (let r = 0; r < this._layoutCacheFormat.length; ++r) this._layoutCache[r] = this._engine.buildTextureLayout(this._layoutCacheFormat[r]); this._renderPassIds = [], this.useRenderPasses = !1, this._prePassEffectConfiguration = new sHe(), this._createTextures(), this._createEffects(); } _createRenderPassIds() { if (this._releaseRenderPassIds(), this._useRenderPasses) for (let e = 0; e < this._passCount + 1; ++e) this._renderPassIds[e] || (this._renderPassIds[e] = this._engine.createRenderPassId(`DepthPeelingRenderer - pass #${e}`)); } _releaseRenderPassIds() { for (let e = 0; e < this._renderPassIds.length; ++e) this._engine.releaseRenderPassId(this._renderPassIds[e]); this._renderPassIds = []; } _createTextures() { const e = { width: this._engine.getRenderWidth(), height: this._engine.getRenderHeight() }; this._depthMrts = [ new Zx("depthPeelingDepth0MRT", e, 3, this._scene, void 0, [ "depthPeelingDepth0MRT_depth", "depthPeelingDepth0MRT_frontColor", "depthPeelingDepth0MRT_backColor" ]), new Zx("depthPeelingDepth1MRT", e, 3, this._scene, void 0, [ "depthPeelingDepth1MRT_depth", "depthPeelingDepth1MRT_frontColor", "depthPeelingDepth1MRT_backColor" ]) ], this._colorMrts = [ new Zx("depthPeelingColor0MRT", e, 2, this._scene, { generateDepthBuffer: !1 }, [ "depthPeelingColor0MRT_frontColor", "depthPeelingColor0MRT_backColor" ]), new Zx("depthPeelingColor1MRT", e, 2, this._scene, { generateDepthBuffer: !1 }, [ "depthPeelingColor1MRT_frontColor", "depthPeelingColor1MRT_backColor" ]) ], this._blendBackMrt = new Zx("depthPeelingBackMRT", e, 1, this._scene, { generateDepthBuffer: !1 }, ["depthPeelingBackMRT_blendBack"]), this._outputRT = new Ta("depthPeelingOutputRTT", e, this._scene, !1); const t = [ { format: 7, samplingMode: 1, type: this._engine.getCaps().textureFloatLinearFiltering ? 1 : 2, label: "DepthPeelingRenderer-DepthTexture" }, { format: 5, samplingMode: 1, type: 2, label: "DepthPeelingRenderer-ColorTexture" } ]; for (let r = 0; r < 2; r++) { const n = this._engine._createInternalTexture(e, t[0], !1), i = this._engine._createInternalTexture(e, t[1], !1), s = this._engine._createInternalTexture(e, t[1], !1); this._depthMrts[r].setInternalTexture(n, 0), this._depthMrts[r].setInternalTexture(i, 1), this._depthMrts[r].setInternalTexture(s, 2), this._colorMrts[r].setInternalTexture(i, 0), this._colorMrts[r].setInternalTexture(s, 1), this._thinTextures.push(new nq(n), new nq(i), new nq(s)); } } // TODO : explore again MSAA with depth peeling when // we are able to fetch individual samples in a multisampled renderbuffer // public set samples(value: number) { // for (let i = 0; i < 2; i++) { // this._depthMrts[i].samples = value; // this._colorMrts[i].samples = value; // } // this._scene.prePassRenderer!.samples = value; // } _disposeTextures() { for (let e = 0; e < this._thinTextures.length; e++) e !== 6 && this._thinTextures[e].dispose(); for (let e = 0; e < 2; e++) this._depthMrts[e].dispose(!0), this._colorMrts[e].dispose(!0), this._blendBackMrt.dispose(!0); this._outputRT.dispose(), this._thinTextures = [], this._colorMrts = [], this._depthMrts = []; } _updateTextures() { return (this._depthMrts[0].getSize().width !== this._engine.getRenderWidth() || this._depthMrts[0].getSize().height !== this._engine.getRenderHeight()) && (this._disposeTextures(), this._createTextures()), this._updateTextureReferences(); } _updateTextureReferences() { var e; const t = this._scene.prePassRenderer; if (!t) return !1; const r = t.getIndex(4), n = !((e = t.defaultRT.textures) === null || e === void 0) && e.length ? t.defaultRT.textures[r].getInternalTexture() : null; return n ? (this._blendBackTexture !== n && (this._blendBackTexture = n, this._blendBackMrt.setInternalTexture(this._blendBackTexture, 0), this._thinTextures[6] && this._thinTextures[6].dispose(), this._thinTextures[6] = new nq(this._blendBackTexture), t.defaultRT.renderTarget._shareDepth(this._depthMrts[0].renderTarget)), !0) : !1; } _createEffects() { this._blendBackEffectWrapper = new ng({ fragmentShader: "oitBackBlend", useShaderStore: !0, engine: this._engine, samplerNames: ["uBackColor"], uniformNames: [] }), this._blendBackEffectWrapperPingPong = new ng({ fragmentShader: "oitBackBlend", useShaderStore: !0, engine: this._engine, samplerNames: ["uBackColor"], uniformNames: [] }), this._finalEffectWrapper = new ng({ fragmentShader: "oitFinal", useShaderStore: !0, engine: this._engine, samplerNames: ["uFrontColor", "uBackColor"], uniformNames: [] }), this._effectRenderer = new fU(this._engine); } /** * Links to the prepass renderer * @param prePassRenderer The scene PrePassRenderer */ setPrePassRenderer(e) { e.addEffectConfiguration(this._prePassEffectConfiguration); } /** * Binds depth peeling textures on an effect * @param effect The effect to bind textures on */ bind(e) { e.setTexture("oitDepthSampler", this._thinTextures[this._currentPingPongState * 3]), e.setTexture("oitFrontColorSampler", this._thinTextures[this._currentPingPongState * 3 + 1]); } _renderSubMeshes(e) { let t; this._useRenderPasses && (t = {}); for (let r = 0; r < e.length; r++) { const n = e.data[r].getMaterial(); let i = !0, s = !1; const a = e.data[r]; let f, o = !1; if (this._useRenderPasses && (f = a._getDrawWrapper(), o = !f), n && (i = n.allowShaderHotSwapping, s = n.backFaceCulling, n.allowShaderHotSwapping = !1, n.backFaceCulling = !1), a.render(!1), o && (f = a._getDrawWrapper(), f.materialContext)) { let d = t[f.materialContext.uniqueId]; d || (d = t[f.materialContext.uniqueId] = this._engine.createMaterialContext()), a._getDrawWrapper().materialContext = d; } n && (n.allowShaderHotSwapping = i, n.backFaceCulling = s); } } _finalCompose(e) { var t; ((t = this._scene.prePassRenderer) === null || t === void 0 ? void 0 : t.setCustomOutput(this._outputRT)) ? this._engine.bindFramebuffer(this._outputRT.renderTarget) : this._engine.restoreDefaultFramebuffer(), this._engine.setAlphaMode(0), this._engine.applyStates(), this._engine.enableEffect(this._finalEffectWrapper._drawWrapper), this._finalEffectWrapper.effect.setTexture("uFrontColor", this._thinTextures[e * 3 + 1]), this._finalEffectWrapper.effect.setTexture("uBackColor", this._thinTextures[6]), this._effectRenderer.render(this._finalEffectWrapper); } /** * Checks if the depth peeling renderer is ready to render transparent meshes * @returns true if the depth peeling renderer is ready to render the transparent meshes */ isReady() { return this._blendBackEffectWrapper.effect.isReady() && this._blendBackEffectWrapperPingPong.effect.isReady() && this._finalEffectWrapper.effect.isReady() && this._updateTextures(); } /** * Renders transparent submeshes with depth peeling * @param transparentSubMeshes List of transparent meshes to render * @returns The array of submeshes that could not be handled by this renderer */ render(e) { if (this._candidateSubMeshes.length = 0, this._excludedSubMeshes.length = 0, !this.isReady()) return this._excludedSubMeshes; this._scene.activeCamera && this._engine.setViewport(this._scene.activeCamera.viewport); for (let i = 0; i < e.length; i++) { const s = e.data[i], a = s.getMaterial(), f = a && s.getRenderingMesh()._getRenderingFillMode(a.fillMode); a && (f === gt.TriangleFanDrawMode || f === gt.TriangleFillMode || f === gt.TriangleStripDrawMode) && this._excludedMeshes.indexOf(s.getMesh().uniqueId) === -1 ? this._candidateSubMeshes.push(s) : this._excludedSubMeshes.push(s); } if (!this._candidateSubMeshes.length) return this._engine.bindFramebuffer(this._colorMrts[1].renderTarget), this._engine.bindAttachments(this._layoutCache[1]), this._engine.clear(this._colorCache[2], !0, !1, !1), this._engine.unBindFramebuffer(this._colorMrts[1].renderTarget), this._finalCompose(1), this._excludedSubMeshes; const t = this._engine.currentRenderPassId; this._scene.prePassRenderer._enabled = !1, this._useRenderPasses && (this._engine.currentRenderPassId = this._renderPassIds[0]), this._engine.bindFramebuffer(this._depthMrts[0].renderTarget), this._engine.bindAttachments(this._layoutCache[0]), this._engine.clear(this._colorCache[0], !0, !1, !1), this._engine.unBindFramebuffer(this._depthMrts[0].renderTarget), this._engine.bindFramebuffer(this._depthMrts[1].renderTarget), this._engine.bindAttachments(this._layoutCache[0]), this._engine.clear(this._colorCache[1], !0, !1, !1), this._engine.unBindFramebuffer(this._depthMrts[1].renderTarget), this._engine.bindFramebuffer(this._colorMrts[0].renderTarget), this._engine.bindAttachments(this._layoutCache[1]), this._engine.clear(this._colorCache[2], !0, !1, !1), this._engine.unBindFramebuffer(this._colorMrts[0].renderTarget), this._engine.bindFramebuffer(this._colorMrts[1].renderTarget), this._engine.bindAttachments(this._layoutCache[1]), this._engine.clear(this._colorCache[2], !0, !1, !1), this._engine.unBindFramebuffer(this._colorMrts[1].renderTarget), this._engine.bindFramebuffer(this._depthMrts[0].renderTarget), this._engine.bindAttachments(this._layoutCache[0]), this._engine.setAlphaMode(11), this._engine.setAlphaEquation(3), this._engine.depthCullingState.depthMask = !1, this._engine.depthCullingState.depthTest = !0, this._engine.applyStates(), this._currentPingPongState = 1, this._renderSubMeshes(this._candidateSubMeshes), this._engine.unBindFramebuffer(this._depthMrts[0].renderTarget), this._scene.resetCachedMaterial(); let r = 0, n = 0; for (let i = 0; i < this._passCount; i++) { r = i % 2, n = 1 - r, this._currentPingPongState = r, this._useRenderPasses && (this._engine.currentRenderPassId = this._renderPassIds[i + 1]), this._scene.activeCamera && this._engine.setViewport(this._scene.activeCamera.viewport), this._engine.bindFramebuffer(this._depthMrts[n].renderTarget), this._engine.bindAttachments(this._layoutCache[0]), this._engine.clear(this._colorCache[0], !0, !1, !1), this._engine.unBindFramebuffer(this._depthMrts[n].renderTarget), this._engine.bindFramebuffer(this._colorMrts[n].renderTarget), this._engine.bindAttachments(this._layoutCache[1]), this._engine.clear(this._colorCache[2], !0, !1, !1), this._engine.unBindFramebuffer(this._colorMrts[n].renderTarget), this._engine.bindFramebuffer(this._depthMrts[n].renderTarget), this._engine.bindAttachments(this._layoutCache[2]), this._engine.setAlphaMode(11), this._engine.setAlphaEquation(3), this._engine.depthCullingState.depthTest = !1, this._engine.applyStates(), this._renderSubMeshes(this._candidateSubMeshes), this._engine.unBindFramebuffer(this._depthMrts[n].renderTarget), this._scene.resetCachedMaterial(), this._engine.bindFramebuffer(this._blendBackMrt.renderTarget), this._engine.bindAttachments(this._layoutCache[0]), this._engine.setAlphaEquation(0), this._engine.setAlphaMode(17), this._engine.applyStates(); const s = n === 0 || !this._useRenderPasses ? this._blendBackEffectWrapper : this._blendBackEffectWrapperPingPong; this._engine.enableEffect(s._drawWrapper), s.effect.setTexture("uBackColor", this._thinTextures[n * 3 + 2]), this._effectRenderer.render(s), this._engine.unBindFramebuffer(this._blendBackMrt.renderTarget); } return this._engine.currentRenderPassId = t, this._finalCompose(n), this._scene.prePassRenderer._enabled = !0, this._engine.depthCullingState.depthMask = !0, this._engine.depthCullingState.depthTest = !0, this._excludedSubMeshes; } /** * Disposes the depth peeling renderer and associated ressources */ dispose() { this._disposeTextures(), this._blendBackEffectWrapper.dispose(), this._finalEffectWrapper.dispose(), this._effectRenderer.dispose(), this._releaseRenderPassIds(); } } J2._DEPTH_CLEAR_VALUE = -99999; J2._MIN_DEPTH = 0; J2._MAX_DEPTH = 1; Object.defineProperty(sr.prototype, "depthPeelingRenderer", { get: function() { if (!this._depthPeelingRenderer) { let A = this._getComponent(Ot.NAME_DEPTHPEELINGRENDERER); A || (A = new oie(this), this._addComponent(A)); } return this._depthPeelingRenderer; }, set: function(A) { this._depthPeelingRenderer = A; }, enumerable: !0, configurable: !0 }); Object.defineProperty(sr.prototype, "useOrderIndependentTransparency", { get: function() { return this._useOrderIndependentTransparency; }, set: function(A) { var e; this._useOrderIndependentTransparency !== A && (this._useOrderIndependentTransparency = A, this.markAllMaterialsAsDirty(63), (e = this.prePassRenderer) === null || e === void 0 || e.markAsDirty()); }, enumerable: !0, configurable: !0 }); class oie { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_DEPTHPEELINGRENDERER, this.scene = e, e.depthPeelingRenderer = new J2(e); } /** * Registers the component in a given scene */ register() { } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { } /** * Disposes the component and the associated resources. */ dispose() { var e; (e = this.scene.depthPeelingRenderer) === null || e === void 0 || e.dispose(), this.scene.depthPeelingRenderer = null; } } const aHe = "linePixelShader", oHe = `#include uniform vec4 color; #ifdef LOGARITHMICDEPTH #extension GL_EXT_frag_depth : enable #endif #include #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #define CUSTOM_FRAGMENT_MAIN_BEGIN #include #include gl_FragColor=color; #define CUSTOM_FRAGMENT_MAIN_END }`; Le.ShadersStore[aHe] = oHe; const fHe = "lineVertexShader", AHe = `#include #include attribute vec3 position;attribute vec4 normal;uniform mat4 viewProjection;uniform float width;uniform float aspectRatio; #include #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN #include mat4 worldViewProjection=viewProjection*finalWorld;vec4 viewPosition=worldViewProjection*vec4(position,1.0);vec4 viewPositionNext=worldViewProjection*vec4(normal.xyz,1.0);vec2 currentScreen=viewPosition.xy/viewPosition.w;vec2 nextScreen=viewPositionNext.xy/viewPositionNext.w;currentScreen.x*=aspectRatio;nextScreen.x*=aspectRatio;vec2 dir=normalize(nextScreen-currentScreen);vec2 normalDir=vec2(-dir.y,dir.x);normalDir*=width/2.0;normalDir.x/=aspectRatio;vec4 offset=vec4(normalDir*normal.w,0.0,0.0);gl_Position=viewPosition+offset; #if defined(CLIPPLANE) || defined(CLIPPLANE2) || defined(CLIPPLANE3) || defined(CLIPPLANE4) || defined(CLIPPLANE5) || defined(CLIPPLANE6) vec4 worldPos=finalWorld*vec4(position,1.0); #include #endif #include #define CUSTOM_VERTEX_MAIN_END }`; Le.ShadersStore[fHe] = AHe; jn.prototype.disableEdgesRendering = function() { return this._edgesRenderer && (this._edgesRenderer.dispose(), this._edgesRenderer = null), this; }; jn.prototype.enableEdgesRendering = function(A = 0.95, e = !1, t) { return this.disableEdgesRendering(), this._edgesRenderer = new uV(this, A, e, !0, t), this; }; Object.defineProperty(jn.prototype, "edgesRenderer", { get: function() { return this._edgesRenderer; }, enumerable: !0, configurable: !0 }); Tp.prototype.enableEdgesRendering = function(A = 0.95, e = !1) { return this.disableEdgesRendering(), this._edgesRenderer = new fie(this, A, e), this; }; SQ.prototype.enableEdgesRendering = function(A = 0.95, e = !1) { return Tp.prototype.enableEdgesRendering.apply(this, arguments), this; }; class dHe { constructor() { this.edges = [], this.edgesConnectedCount = 0; } } class uV { /** Gets the vertices generated by the edge renderer */ get linesPositions() { return this._linesPositions; } /** Gets the normals generated by the edge renderer */ get linesNormals() { return this._linesNormals; } /** Gets the indices generated by the edge renderer */ get linesIndices() { return this._linesIndices; } /** * Gets or sets the shader used to draw the lines */ get lineShader() { return this._lineShader; } set lineShader(e) { this._lineShader = e; } static _GetShader(e) { if (!e._edgeRenderLineShader) { const t = new Zo("lineShader", e, "line", { attributes: ["position", "normal"], uniforms: ["world", "viewProjection", "color", "width", "aspectRatio"] }, !1); t.disableDepthWrite = !0, t.backFaceCulling = !1, t.checkReadyOnEveryCall = e.getEngine().isWebGPU, e._edgeRenderLineShader = t; } return e._edgeRenderLineShader; } /** * Creates an instance of the EdgesRenderer. It is primarily use to display edges of a mesh. * Beware when you use this class with complex objects as the adjacencies computation can be really long * @param source Mesh used to create edges * @param epsilon sum of angles in adjacency to check for edge * @param checkVerticesInsteadOfIndices bases the edges detection on vertices vs indices. Note that this parameter is not used if options.useAlternateEdgeFinder = true * @param generateEdgesLines - should generate Lines or only prepare resources. * @param options The options to apply when generating the edges */ constructor(e, t = 0.95, r = !1, n = !0, i) { var s; this.edgesWidthScalerForOrthographic = 1e3, this.edgesWidthScalerForPerspective = 50, this._linesPositions = new Array(), this._linesNormals = new Array(), this._linesIndices = new Array(), this._buffers = {}, this._buffersForInstances = {}, this._checkVerticesInsteadOfIndices = !1, this.isEnabled = !0, this.customInstances = new qf(32), this._source = e, this._checkVerticesInsteadOfIndices = r, this._options = i ?? null, this._epsilon = t, this._source.getScene().getEngine().isWebGPU && (this._drawWrapper = new zo(e.getEngine())), this._prepareRessources(), n && (!((s = i == null ? void 0 : i.useAlternateEdgeFinder) !== null && s !== void 0) || s ? this._generateEdgesLinesAlternate() : this._generateEdgesLines()), this._meshRebuildObserver = this._source.onRebuildObservable.add(() => { this._rebuild(); }), this._meshDisposeObserver = this._source.onDisposeObservable.add(() => { this.dispose(); }); } _prepareRessources() { this._lineShader || (this._lineShader = uV._GetShader(this._source.getScene())); } /** @internal */ _rebuild() { let e = this._buffers[J.PositionKind]; e && e._rebuild(), e = this._buffers[J.NormalKind], e && e._rebuild(); const r = this._source.getScene().getEngine(); this._ib = r.createIndexBuffer(this._linesIndices); } /** * Releases the required resources for the edges renderer */ dispose() { var e; this._source.onRebuildObservable.remove(this._meshRebuildObserver), this._source.onDisposeObservable.remove(this._meshDisposeObserver); let t = this._buffers[J.PositionKind]; t && (t.dispose(), this._buffers[J.PositionKind] = null), t = this._buffers[J.NormalKind], t && (t.dispose(), this._buffers[J.NormalKind] = null), this._ib && this._source.getScene().getEngine()._releaseBuffer(this._ib), this._lineShader.dispose(), (e = this._drawWrapper) === null || e === void 0 || e.dispose(); } _processEdgeForAdjacencies(e, t, r, n, i) { return e === r && t === n || e === n && t === r ? 0 : e === n && t === i || e === i && t === n ? 1 : e === i && t === r || e === r && t === i ? 2 : -1; } _processEdgeForAdjacenciesWithVertices(e, t, r, n, i) { return e.equalsWithEpsilon(r, 1e-10) && t.equalsWithEpsilon(n, 1e-10) || e.equalsWithEpsilon(n, 1e-10) && t.equalsWithEpsilon(r, 1e-10) ? 0 : e.equalsWithEpsilon(n, 1e-10) && t.equalsWithEpsilon(i, 1e-10) || e.equalsWithEpsilon(i, 1e-10) && t.equalsWithEpsilon(n, 1e-10) ? 1 : e.equalsWithEpsilon(i, 1e-10) && t.equalsWithEpsilon(r, 1e-10) || e.equalsWithEpsilon(r, 1e-10) && t.equalsWithEpsilon(i, 1e-10) ? 2 : -1; } /** * Checks if the pair of p0 and p1 is en edge * @param faceIndex * @param edge * @param faceNormals * @param p0 * @param p1 * @private */ _checkEdge(e, t, r, n, i) { let s; t === void 0 ? s = !0 : s = S.Dot(r[e], r[t]) < this._epsilon, s && this.createLine(n, i, this._linesPositions.length / 3); } /** * push line into the position, normal and index buffer * @param p0 * @param p1 * @param offset * @protected */ // eslint-disable-next-line @typescript-eslint/naming-convention createLine(e, t, r) { this._linesPositions.push(e.x, e.y, e.z, e.x, e.y, e.z, t.x, t.y, t.z, t.x, t.y, t.z), this._linesNormals.push(t.x, t.y, t.z, -1, t.x, t.y, t.z, 1, e.x, e.y, e.z, -1, e.x, e.y, e.z, 1), this._linesIndices.push(r, r + 1, r + 2, r, r + 2, r + 3); } /** * See https://playground.babylonjs.com/#R3JR6V#1 for a visual display of the algorithm * @param edgePoints * @param indexTriangle * @param indices * @param remapVertexIndices */ _tessellateTriangle(e, t, r, n) { const i = (I, N, k) => { k >= 0 && N.push(k); for (let R = 0; R < I.length; ++R) N.push(I[R][0]); }; let s = 0; e[1].length >= e[0].length && e[1].length >= e[2].length ? s = 1 : e[2].length >= e[0].length && e[2].length >= e[1].length && (s = 2); for (let I = 0; I < 3; ++I) I === s ? e[I].sort((N, k) => N[1] < k[1] ? -1 : N[1] > k[1] ? 1 : 0) : e[I].sort((N, k) => N[1] > k[1] ? -1 : N[1] < k[1] ? 1 : 0); const a = [], f = []; i(e[s], a, -1); const o = a.length; for (let I = s + 2; I >= s + 1; --I) i(e[I % 3], f, I !== s + 2 ? n[r[t + (I + 1) % 3]] : -1); const d = f.length, v = 0, u = 0; r.push(n[r[t + s]], a[0], f[0]), r.push(n[r[t + (s + 1) % 3]], f[d - 1], a[o - 1]); const l = o <= d, P = l ? o : d, p = l ? d : o, c = l ? o - 1 : d - 1, H = l ? 0 : 1; let T = o + d - 2, q = l ? v : u, b = l ? u : v; const j = l ? a : f, w = l ? f : a; let m = 0; for (; T-- > 0; ) { H ? r.push(j[q], w[b]) : r.push(w[b], j[q]), m += P; let I; m >= p && q < c ? (I = j[++q], m -= p) : I = w[++b], r.push(I); } r[t + 0] = r[r.length - 3], r[t + 1] = r[r.length - 2], r[t + 2] = r[r.length - 1], r.length = r.length - 3; } _generateEdgesLinesAlternate() { var e, t, r, n, i, s, a, f, o, d; const v = this._source.getVerticesData(J.PositionKind); let u = this._source.getIndices(); if (!u || !v) return; Array.isArray(u) || (u = Array.from(u)); const l = (t = (e = this._options) === null || e === void 0 ? void 0 : e.useFastVertexMerger) !== null && t !== void 0 ? t : !0, P = l ? Math.round(-Math.log((n = (r = this._options) === null || r === void 0 ? void 0 : r.epsilonVertexMerge) !== null && n !== void 0 ? n : 1e-6) / Math.log(10)) : (s = (i = this._options) === null || i === void 0 ? void 0 : i.epsilonVertexMerge) !== null && s !== void 0 ? s : 1e-6, p = [], c = []; if (l) { const q = {}; for (let b = 0; b < v.length; b += 3) { const j = v[b + 0], w = v[b + 1], m = v[b + 2], I = j.toFixed(P) + "|" + w.toFixed(P) + "|" + m.toFixed(P); if (q[I] !== void 0) p.push(q[I]); else { const N = b / 3; q[I] = N, p.push(N), c.push(N); } } } else for (let q = 0; q < v.length; q += 3) { const b = v[q + 0], j = v[q + 1], w = v[q + 2]; let m = !1; for (let I = 0; I < q && !m; I += 3) { const N = v[I + 0], k = v[I + 1], R = v[I + 2]; if (Math.abs(b - N) < P && Math.abs(j - k) < P && Math.abs(w - R) < P) { p.push(I / 3), m = !0; break; } } m || (p.push(q / 3), c.push(q / 3)); } if (!((a = this._options) === null || a === void 0) && a.applyTessellation) { const q = (o = (f = this._options) === null || f === void 0 ? void 0 : f.epsilonVertexAligned) !== null && o !== void 0 ? o : 1e-6, b = []; for (let j = 0; j < u.length; j += 3) { let w; for (let m = 0; m < 3; ++m) { const I = p[u[j + m]], N = p[u[j + (m + 1) % 3]], k = p[u[j + (m + 2) % 3]]; if (I === N) continue; const R = v[I * 3 + 0], y = v[I * 3 + 1], O = v[I * 3 + 2], Y = v[N * 3 + 0], ee = v[N * 3 + 1], Z = v[N * 3 + 2], te = Math.sqrt((Y - R) * (Y - R) + (ee - y) * (ee - y) + (Z - O) * (Z - O)); for (let fe = 0; fe < c.length - 1; fe++) { const _ = c[fe]; if (_ === I || _ === N || _ === k) continue; const G = v[_ * 3 + 0], L = v[_ * 3 + 1], $ = v[_ * 3 + 2], ae = Math.sqrt((G - R) * (G - R) + (L - y) * (L - y) + ($ - O) * ($ - O)), Pe = Math.sqrt((G - Y) * (G - Y) + (L - ee) * (L - ee) + ($ - Z) * ($ - Z)); Math.abs(ae + Pe - te) < q && (w || (w = { index: j, edgesPoints: [[], [], []] }, b.push(w)), w.edgesPoints[m].push([_, ae])); } } } for (let j = 0; j < b.length; ++j) { const w = b[j]; this._tessellateTriangle(w.edgesPoints, w.index, u, p); } b.length = 0; } const H = {}; for (let q = 0; q < u.length; q += 3) { let b; for (let j = 0; j < 3; ++j) { let w = p[u[q + j]], m = p[u[q + (j + 1) % 3]]; const I = p[u[q + (j + 2) % 3]]; if (w === m || (w === I || m === I) && (!((d = this._options) === null || d === void 0) && d.removeDegeneratedTriangles)) continue; if (ue.Vector3[0].copyFromFloats(v[w * 3 + 0], v[w * 3 + 1], v[w * 3 + 2]), ue.Vector3[1].copyFromFloats(v[m * 3 + 0], v[m * 3 + 1], v[m * 3 + 2]), ue.Vector3[2].copyFromFloats(v[I * 3 + 0], v[I * 3 + 1], v[I * 3 + 2]), b || (ue.Vector3[1].subtractToRef(ue.Vector3[0], ue.Vector3[3]), ue.Vector3[2].subtractToRef(ue.Vector3[1], ue.Vector3[4]), b = S.Cross(ue.Vector3[3], ue.Vector3[4]), b.normalize()), w > m) { const R = w; w = m, m = R; } const N = w + "_" + m, k = H[N]; k ? k.done || (S.Dot(b, k.normal) < this._epsilon && this.createLine(ue.Vector3[0], ue.Vector3[1], this._linesPositions.length / 3), k.done = !0) : H[N] = { normal: b, done: !1, index: q, i: j }; } } for (const q in H) { const b = H[q]; if (!b.done) { const j = p[u[b.index + b.i]], w = p[u[b.index + (b.i + 1) % 3]]; ue.Vector3[0].copyFromFloats(v[j * 3 + 0], v[j * 3 + 1], v[j * 3 + 2]), ue.Vector3[1].copyFromFloats(v[w * 3 + 0], v[w * 3 + 1], v[w * 3 + 2]), this.createLine(ue.Vector3[0], ue.Vector3[1], this._linesPositions.length / 3); } } const T = this._source.getScene().getEngine(); this._buffers[J.PositionKind] = new J(T, this._linesPositions, J.PositionKind, !1), this._buffers[J.NormalKind] = new J(T, this._linesNormals, J.NormalKind, !1, !1, 4), this._buffersForInstances[J.PositionKind] = this._buffers[J.PositionKind], this._buffersForInstances[J.NormalKind] = this._buffers[J.NormalKind], this._ib = T.createIndexBuffer(this._linesIndices), this._indicesCount = this._linesIndices.length; } /** * Generates lines edges from adjacencjes * @private */ _generateEdgesLines() { const e = this._source.getVerticesData(J.PositionKind), t = this._source.getIndices(); if (!t || !e) return; const r = [], n = []; let i, s; for (i = 0; i < t.length; i += 3) { s = new dHe(); const f = t[i], o = t[i + 1], d = t[i + 2]; s.p0 = new S(e[f * 3], e[f * 3 + 1], e[f * 3 + 2]), s.p1 = new S(e[o * 3], e[o * 3 + 1], e[o * 3 + 2]), s.p2 = new S(e[d * 3], e[d * 3 + 1], e[d * 3 + 2]); const v = S.Cross(s.p1.subtract(s.p0), s.p2.subtract(s.p1)); v.normalize(), n.push(v), r.push(s); } for (i = 0; i < r.length; i++) { s = r[i]; for (let f = i + 1; f < r.length; f++) { const o = r[f]; if (s.edgesConnectedCount === 3) break; if (o.edgesConnectedCount === 3) continue; const d = t[f * 3], v = t[f * 3 + 1], u = t[f * 3 + 2]; for (let l = 0; l < 3; l++) { let P = 0; if (s.edges[l] === void 0) { switch (l) { case 0: this._checkVerticesInsteadOfIndices ? P = this._processEdgeForAdjacenciesWithVertices(s.p0, s.p1, o.p0, o.p1, o.p2) : P = this._processEdgeForAdjacencies(t[i * 3], t[i * 3 + 1], d, v, u); break; case 1: this._checkVerticesInsteadOfIndices ? P = this._processEdgeForAdjacenciesWithVertices(s.p1, s.p2, o.p0, o.p1, o.p2) : P = this._processEdgeForAdjacencies(t[i * 3 + 1], t[i * 3 + 2], d, v, u); break; case 2: this._checkVerticesInsteadOfIndices ? P = this._processEdgeForAdjacenciesWithVertices(s.p2, s.p0, o.p0, o.p1, o.p2) : P = this._processEdgeForAdjacencies(t[i * 3 + 2], t[i * 3], d, v, u); break; } if (P !== -1 && (s.edges[l] = f, o.edges[P] = i, s.edgesConnectedCount++, o.edgesConnectedCount++, s.edgesConnectedCount === 3)) break; } } } } for (i = 0; i < r.length; i++) { const f = r[i]; this._checkEdge(i, f.edges[0], n, f.p0, f.p1), this._checkEdge(i, f.edges[1], n, f.p1, f.p2), this._checkEdge(i, f.edges[2], n, f.p2, f.p0); } const a = this._source.getScene().getEngine(); this._buffers[J.PositionKind] = new J(a, this._linesPositions, J.PositionKind, !1), this._buffers[J.NormalKind] = new J(a, this._linesNormals, J.NormalKind, !1, !1, 4), this._buffersForInstances[J.PositionKind] = this._buffers[J.PositionKind], this._buffersForInstances[J.NormalKind] = this._buffers[J.NormalKind], this._ib = a.createIndexBuffer(this._linesIndices), this._indicesCount = this._linesIndices.length; } /** * Checks whether or not the edges renderer is ready to render. * @returns true if ready, otherwise false. */ isReady() { return this._lineShader.isReady(this._source, this._source.hasInstances && this.customInstances.length > 0 || this._source.hasThinInstances); } /** * Renders the edges of the attached mesh, */ render() { const e = this._source.getScene(), t = this._lineShader._getDrawWrapper(); if (this._drawWrapper && this._lineShader._setDrawWrapper(this._drawWrapper), !this.isReady() || !e.activeCamera) { this._lineShader._setDrawWrapper(t); return; } const r = this._source.hasInstances && this.customInstances.length > 0, n = r || this._source.hasThinInstances; let i = 0; if (n) if (this._buffersForInstances.world0 = this._source.getVertexBuffer("world0"), this._buffersForInstances.world1 = this._source.getVertexBuffer("world1"), this._buffersForInstances.world2 = this._source.getVertexBuffer("world2"), this._buffersForInstances.world3 = this._source.getVertexBuffer("world3"), r) { const a = this._source._instanceDataStorage; if (i = this.customInstances.length, !a.instancesData) { this._source.getScene()._activeMeshesFrozen || this.customInstances.reset(); return; } if (!a.isFrozen) { let f = 0; for (let o = 0; o < i; ++o) this.customInstances.data[o].copyToArray(a.instancesData, f), f += 16; a.instancesBuffer.updateDirectly(a.instancesData, 0, i); } } else i = this._source.thinInstanceCount; const s = e.getEngine(); this._lineShader._preBind(), this._source.edgesColor.a !== 1 ? s.setAlphaMode(2) : s.setAlphaMode(0), s.bindBuffers(n ? this._buffersForInstances : this._buffers, this._ib, this._lineShader.getEffect()), e.resetCachedMaterial(), this._lineShader.setColor4("color", this._source.edgesColor), e.activeCamera.mode === Tr.ORTHOGRAPHIC_CAMERA ? this._lineShader.setFloat("width", this._source.edgesWidth / this.edgesWidthScalerForOrthographic) : this._lineShader.setFloat("width", this._source.edgesWidth / this.edgesWidthScalerForPerspective), this._lineShader.setFloat("aspectRatio", s.getAspectRatio(e.activeCamera)), this._lineShader.bind(this._source.getWorldMatrix()), s.drawElementsType(gt.TriangleFillMode, 0, this._indicesCount, i), this._lineShader.unbind(), n && s.unbindInstanceAttributes(), this._source.getScene()._activeMeshesFrozen || this.customInstances.reset(), this._lineShader._setDrawWrapper(t); } } class fie extends uV { /** * This constructor turns off auto generating edges line in Edges Renderer to make it here. * @param source LineMesh used to generate edges * @param epsilon not important (specified angle for edge detection) * @param checkVerticesInsteadOfIndices not important for LineMesh */ constructor(e, t = 0.95, r = !1) { super(e, t, r, !1), this._generateEdgesLines(); } /** * Generate edges for each line in LinesMesh. Every Line should be rendered as edge. */ _generateEdgesLines() { const e = this._source.getVerticesData(J.PositionKind), t = this._source.getIndices(); if (!t || !e) return; const r = ue.Vector3[0], n = ue.Vector3[1], i = t.length - 1; for (let a = 0, f = 0; a < i; a += 2, f += 4) S.FromArrayToRef(e, 3 * t[a], r), S.FromArrayToRef(e, 3 * t[a + 1], n), this.createLine(r, n, f); const s = this._source.getScene().getEngine(); this._buffers[J.PositionKind] = new J(s, this._linesPositions, J.PositionKind, !1), this._buffers[J.NormalKind] = new J(s, this._linesNormals, J.NormalKind, !1, !1, 4), this._ib = s.createIndexBuffer(this._linesIndices), this._indicesCount = this._linesIndices.length; } } class vHe extends Zx { constructor(e, t, r, n, i, s) { super(e, r, n, i, s), this._beforeCompositionPostProcesses = [], this._internalTextureDirty = !1, this.enabled = !1, this.renderTargetTexture = null, this.renderTargetTexture = t; } /** * Creates a composition effect for this RT * @internal */ _createCompositionEffect() { this.imageProcessingPostProcess = new Uy("prePassComposition", 1, null, void 0, this._engine), this.imageProcessingPostProcess._updateParameters(); } /** * Checks that the size of this RT is still adapted to the desired render size. * @internal */ _checkSize() { const e = this._engine.getRenderWidth(!0), t = this._engine.getRenderHeight(!0), r = this.getRenderWidth(), n = this.getRenderHeight(); (r !== e || n !== t) && (this.resize({ width: e, height: t }), this._internalTextureDirty = !0); } /** * Changes the number of render targets in this MRT * Be careful as it will recreate all the data in the new texture. * @param count new texture count * @param options Specifies texture types and sampling modes for new textures * @param textureNames Specifies the names of the textures (optional) */ updateCount(e, t, r) { super.updateCount(e, t, r), this._internalTextureDirty = !0; } /** * Resets the post processes chains applied to this RT. * @internal */ _resetPostProcessChain() { this._beforeCompositionPostProcesses.length = 0; } /** * Diposes this render target */ dispose() { const e = this._scene; if (super.dispose(), e && e.prePassRenderer) { const t = e.prePassRenderer.renderTargets.indexOf(this); t !== -1 && e.prePassRenderer.renderTargets.splice(t, 1); } this.imageProcessingPostProcess && this.imageProcessingPostProcess.dispose(), this.renderTargetTexture && (this.renderTargetTexture._prePassRenderTarget = null), this._outputPostProcess && (this._outputPostProcess.autoClear = !0, this._outputPostProcess.restoreDefaultInputTexture()); } } class mA { /** * Indicates if the prepass renderer is generating normals in world space or camera space (default: camera space) */ get generateNormalsInWorldSpace() { return this._generateNormalsInWorldSpace; } set generateNormalsInWorldSpace(e) { this._generateNormalsInWorldSpace !== e && (this._generateNormalsInWorldSpace = e, this._markAllMaterialsAsPrePassDirty()); } /** * Returns the index of a texture in the multi render target texture array. * @param type Texture type * @returns The index */ getIndex(e) { return this._textureIndices[e]; } /** * How many samples are used for MSAA of the scene render target */ get samples() { return this.defaultRT.samples; } set samples(e) { this.defaultRT.samples = e; } /** * If set to true (default: false), the depth texture will be cleared with the depth value corresponding to the far plane (1 in normal mode, 0 in reverse depth buffer mode) * If set to false, the depth texture is always cleared with 0. */ get useSpecificClearForDepthTexture() { return this._useSpecificClearForDepthTexture; } set useSpecificClearForDepthTexture(e) { this._useSpecificClearForDepthTexture !== e && (this._useSpecificClearForDepthTexture = e, this._isDirty = !0); } /** * @returns the prepass render target for the rendering pass. * If we are currently rendering a render target, it returns the PrePassRenderTarget * associated with that render target. Otherwise, it returns the scene default PrePassRenderTarget */ getRenderTarget() { return this._currentTarget; } /** * @internal * Managed by the scene component * @param prePassRenderTarget */ _setRenderTarget(e) { var t, r; e ? this._currentTarget = e : (this._currentTarget = this.defaultRT, this._engine.currentRenderPassId = (r = (t = this._scene.activeCamera) === null || t === void 0 ? void 0 : t.renderPassId) !== null && r !== void 0 ? r : this._currentTarget.renderPassId); } /** * Returns true if the currently rendered prePassRenderTarget is the one * associated with the scene. */ get currentRTisSceneRT() { return this._currentTarget === this.defaultRT; } _refreshGeometryBufferRendererLink() { if (this.doNotUseGeometryRendererFallback) this._geometryBuffer && this._geometryBuffer._unlinkPrePassRenderer(), this._geometryBuffer = null, this._scene.disableGeometryBufferRenderer(); else { if (this._geometryBuffer = this._scene.enableGeometryBufferRenderer(), !this._geometryBuffer) { this.doNotUseGeometryRendererFallback = !0; return; } this._geometryBuffer._linkPrePassRenderer(this); } } /** * Indicates if the prepass is enabled */ get enabled() { return this._enabled; } /** * Instantiates a prepass renderer * @param scene The scene */ constructor(e) { this.excludedSkinnedMesh = [], this.excludedMaterials = [], this.mrtCount = 0, this._mrtTypes = [], this._mrtFormats = [], this._mrtLayout = [], this._mrtNames = [], this._textureIndices = [], this._generateNormalsInWorldSpace = !1, this._useSpecificClearForDepthTexture = !1, this._isDirty = !0, this._effectConfigurations = [], this.doNotUseGeometryRendererFallback = !0, this.renderTargets = [], this._clearColor = new xt(0, 0, 0, 0), this._clearDepthColor = new xt(1e8, 0, 0, 1), this._enabled = !1, this._needsCompositionForThisPass = !1, this.disableGammaTransform = !1, this._scene = e, this._engine = e.getEngine(); let t = 0; this._engine._caps.textureFloat && this._engine._caps.textureFloatLinearFiltering ? t = 1 : this._engine._caps.textureHalfFloat && this._engine._caps.textureHalfFloatLinearFiltering && (t = 2); for (let r = 0; r < mA.TextureFormats.length; ++r) { const n = mA.TextureFormats[r].format; mA.TextureFormats[r].type === 1 && (mA.TextureFormats[5].type = t, (n === 6 || n === 7 || n === 5) && !this._engine._caps.supportFloatTexturesResolve && (mA.TextureFormats[5].type = 2)); } mA._SceneComponentInitialization(this._scene), this.defaultRT = this._createRenderTarget("sceneprePassRT", null), this._currentTarget = this.defaultRT; } /** * Creates a new PrePassRenderTarget * This should be the only way to instantiate a `PrePassRenderTarget` * @param name Name of the `PrePassRenderTarget` * @param renderTargetTexture RenderTarget the `PrePassRenderTarget` will be attached to. * Can be `null` if the created `PrePassRenderTarget` is attached to the scene (default framebuffer). * @internal */ _createRenderTarget(e, t) { const r = new vHe(e, t, { width: this._engine.getRenderWidth(), height: this._engine.getRenderHeight() }, 0, this._scene, { generateMipMaps: !1, generateStencilBuffer: this._engine.isStencilEnable, defaultType: 0, types: [], drawOnlyOnFirstAttachmentByDefault: !0 }); return this.renderTargets.push(r), this._enabled && this._update(), r; } /** * Indicates if rendering a prepass is supported */ get isSupported() { return this._scene.getEngine().getCaps().drawBuffersExtension; } /** * Sets the proper output textures to draw in the engine. * @param effect The effect that is drawn. It can be or not be compatible with drawing to several output textures. * @param subMesh Submesh on which the effect is applied */ bindAttachmentsForEffect(e, t) { const r = t.getMaterial(), n = r && r.isPrePassCapable, i = r && this.excludedMaterials.indexOf(r) !== -1; this.enabled && this._currentTarget.enabled && (e._multiTarget && n && !i ? this._engine.bindAttachments(this._multiRenderAttachments) : (this._engine._currentRenderTarget ? this._engine.bindAttachments(this._defaultAttachments) : this._engine.restoreSingleAttachment(), this._geometryBuffer && this.currentRTisSceneRT && !i && this._geometryBuffer.renderList.push(t.getRenderingMesh()))); } _reinitializeAttachments() { const e = [], t = [!1], r = [!1], n = [!0]; for (let i = 0; i < this.mrtCount; i++) e.push(!0), i > 0 && (this._useSpecificClearForDepthTexture && this._mrtLayout[i] === 5 ? (t.push(!1), r.push(!0)) : (t.push(!0), r.push(!1)), n.push(!1)); this._multiRenderAttachments = this._engine.buildTextureLayout(e), this._clearAttachments = this._engine.buildTextureLayout(t), this._clearDepthAttachments = this._engine.buildTextureLayout(r), this._defaultAttachments = this._engine.buildTextureLayout(n); } _resetLayout() { for (let e = 0; e < mA.TextureFormats.length; e++) this._textureIndices[mA.TextureFormats[e].purpose] = -1; this._textureIndices[4] = 0, this._mrtLayout = [4], this._mrtTypes = [mA.TextureFormats[4].type], this._mrtFormats = [mA.TextureFormats[4].format], this._mrtNames = [mA.TextureFormats[4].name], this.mrtCount = 1; } _updateGeometryBufferLayout() { if (this._refreshGeometryBufferRendererLink(), this._geometryBuffer) { this._geometryBuffer._resetLayout(); const e = []; for (let r = 0; r < this._mrtLayout.length; r++) e.push(!1); this._geometryBuffer._linkInternalTexture(this.defaultRT.getInternalTexture()); const t = [ { prePassConstant: 5, geometryBufferConstant: po.DEPTH_TEXTURE_TYPE }, { prePassConstant: 6, geometryBufferConstant: po.NORMAL_TEXTURE_TYPE }, { prePassConstant: 1, geometryBufferConstant: po.POSITION_TEXTURE_TYPE }, { prePassConstant: 3, geometryBufferConstant: po.REFLECTIVITY_TEXTURE_TYPE }, { prePassConstant: 2, geometryBufferConstant: po.VELOCITY_TEXTURE_TYPE } ]; for (let r = 0; r < t.length; r++) { const n = this._mrtLayout.indexOf(t[r].prePassConstant); n !== -1 && (this._geometryBuffer._forceTextureType(t[r].geometryBufferConstant, n), e[n] = !0); } this._geometryBuffer._setAttachments(this._engine.buildTextureLayout(e)); } } /** * Restores attachments for single texture draw. */ restoreAttachments() { this.enabled && this._currentTarget.enabled && this._defaultAttachments && (this._engine._currentRenderTarget ? this._engine.bindAttachments(this._defaultAttachments) : this._engine.restoreSingleAttachment()); } /** * @internal */ // eslint-disable-next-line @typescript-eslint/no-unused-vars _beforeDraw(e, t, r) { this._isDirty && this._update(), !(!this._enabled || !this._currentTarget.enabled) && (this._geometryBuffer && (this._geometryBuffer.renderList = []), this._setupOutputForThisPass(this._currentTarget, e)); } _prepareFrame(e, t, r) { e.renderTargetTexture ? e.renderTargetTexture._prepareFrame(this._scene, t, r, e.renderTargetTexture.useCameraPostProcesses) : this._postProcessesSourceForThisPass.length ? this._scene.postProcessManager._prepareFrame() : this._engine.restoreDefaultFramebuffer(); } /** * Sets an intermediary texture between prepass and postprocesses. This texture * will be used as input for post processes * @param rt * @returns true if there are postprocesses that will use this texture, * false if there is no postprocesses - and the function has no effect */ setCustomOutput(e) { const t = this._postProcessesSourceForThisPass[0]; return t ? (t.inputTexture = e.renderTarget, !0) : !1; } _renderPostProcesses(e, t) { var r; const n = this._postProcessesSourceForThisPass[0], i = n ? n.inputTexture : e.renderTargetTexture ? e.renderTargetTexture.renderTarget : null; let s = this._currentTarget._beforeCompositionPostProcesses; this._needsCompositionForThisPass && (s = s.concat([this._currentTarget.imageProcessingPostProcess])), s.length && (this._scene.postProcessManager._prepareFrame((r = this._currentTarget.renderTarget) === null || r === void 0 ? void 0 : r.texture, s), this._scene.postProcessManager.directRender(s, i, !1, t)); } /** * @internal */ _afterDraw(e, t) { this._enabled && this._currentTarget.enabled && (this._prepareFrame(this._currentTarget, e, t), this._renderPostProcesses(this._currentTarget, e)); } /** * Clears the current prepass render target (in the sense of settings pixels to the scene clear color value) * @internal */ _clear() { this._enabled && this._currentTarget.enabled && (this._bindFrameBuffer(), this._engine.bindAttachments(this._clearAttachments), this._engine.clear(this._clearColor, !0, !1, !1), this._useSpecificClearForDepthTexture && (this._engine.bindAttachments(this._clearDepthAttachments), this._engine.clear(this._clearDepthColor, !0, !1, !1)), this._engine.bindAttachments(this._defaultAttachments)); } // eslint-disable-next-line @typescript-eslint/no-unused-vars _bindFrameBuffer() { if (this._enabled && this._currentTarget.enabled) { this._currentTarget._checkSize(); const e = this._currentTarget.renderTarget; e && this._engine.bindFramebuffer(e); } } _setEnabled(e) { this._enabled = e; } _setRenderTargetEnabled(e, t) { e.enabled = t, t || this._unlinkInternalTexture(e); } /** * Adds an effect configuration to the prepass render target. * If an effect has already been added, it won't add it twice and will return the configuration * already present. * @param cfg the effect configuration * @returns the effect configuration now used by the prepass */ addEffectConfiguration(e) { for (let t = 0; t < this._effectConfigurations.length; t++) if (this._effectConfigurations[t].name === e.name) return this._effectConfigurations[t]; return this._effectConfigurations.push(e), e; } /** * Retrieves an effect configuration by name * @param name * @returns the effect configuration, or null if not present */ getEffectConfiguration(e) { for (let t = 0; t < this._effectConfigurations.length; t++) if (this._effectConfigurations[t].name === e) return this._effectConfigurations[t]; return null; } _enable() { const e = this.mrtCount; for (let t = 0; t < this._effectConfigurations.length; t++) this._effectConfigurations[t].enabled && this._enableTextures(this._effectConfigurations[t].texturesRequired); for (let t = 0; t < this.renderTargets.length; t++) { (this.mrtCount !== e || this.renderTargets[t].count !== this.mrtCount) && this.renderTargets[t].updateCount(this.mrtCount, { types: this._mrtTypes, formats: this._mrtFormats }, this._mrtNames.concat("prePass_DepthBuffer")), this.renderTargets[t]._resetPostProcessChain(); for (let r = 0; r < this._effectConfigurations.length; r++) this._effectConfigurations[r].enabled && (!this._effectConfigurations[r].postProcess && this._effectConfigurations[r].createPostProcess && this._effectConfigurations[r].createPostProcess(), this._effectConfigurations[r].postProcess && this.renderTargets[t]._beforeCompositionPostProcesses.push(this._effectConfigurations[r].postProcess)); } this._reinitializeAttachments(), this._setEnabled(!0), this._updateGeometryBufferLayout(); } _disable() { this._setEnabled(!1); for (let e = 0; e < this.renderTargets.length; e++) this._setRenderTargetEnabled(this.renderTargets[e], !1); this._resetLayout(); for (let e = 0; e < this._effectConfigurations.length; e++) this._effectConfigurations[e].enabled = !1; } _getPostProcessesSource(e, t) { if (t) return t._postProcesses; if (e.renderTargetTexture) if (e.renderTargetTexture.useCameraPostProcesses) { const r = e.renderTargetTexture.activeCamera ? e.renderTargetTexture.activeCamera : this._scene.activeCamera; return r ? r._postProcesses : []; } else return e.renderTargetTexture.postProcesses ? e.renderTargetTexture.postProcesses : []; else return this._scene.activeCamera ? this._scene.activeCamera._postProcesses : []; } _setupOutputForThisPass(e, t) { const r = t && this._scene.activeCameras && !!this._scene.activeCameras.length && this._scene.activeCameras.indexOf(t) !== 0; this._postProcessesSourceForThisPass = this._getPostProcessesSource(e, t), this._postProcessesSourceForThisPass = this._postProcessesSourceForThisPass.filter((f) => f != null), this._scene.autoClear = !0; const n = this._hasImageProcessing(this._postProcessesSourceForThisPass); this._needsCompositionForThisPass = !n && !this.disableGammaTransform && this._needsImageProcessing() && !r; const i = this._getFirstPostProcess(this._postProcessesSourceForThisPass), s = e._beforeCompositionPostProcesses && e._beforeCompositionPostProcesses[0]; let a = null; this._scene.imageProcessingConfiguration.applyByPostProcess = this._needsCompositionForThisPass || n, this._needsCompositionForThisPass && !e.imageProcessingPostProcess && e._createCompositionEffect(), s ? a = s : this._needsCompositionForThisPass ? a = e.imageProcessingPostProcess : i && (a = i), this._bindFrameBuffer(), this._linkInternalTexture(e, a); } _linkInternalTexture(e, t) { t && (t.autoClear = !1, t.inputTexture = e.renderTarget), e._outputPostProcess !== t && (e._outputPostProcess && this._unlinkInternalTexture(e), e._outputPostProcess = t), e._internalTextureDirty && (this._updateGeometryBufferLayout(), e._internalTextureDirty = !1); } /** * @internal */ _unlinkInternalTexture(e) { e._outputPostProcess && (e._outputPostProcess.autoClear = !0, e._outputPostProcess.restoreDefaultInputTexture(), e._outputPostProcess = null); } _needsImageProcessing() { for (let e = 0; e < this._effectConfigurations.length; e++) if (this._effectConfigurations[e].enabled && this._effectConfigurations[e].needsImageProcessing) return !0; return !1; } _hasImageProcessing(e) { var t; let r = !1; if (e) { for (let n = 0; n < e.length; n++) if (((t = e[n]) === null || t === void 0 ? void 0 : t.getClassName()) === "ImageProcessingPostProcess") { r = !0; break; } } return r; } /** * Internal, gets the first post proces. * @param postProcesses * @returns the first post process to be run on this camera. */ _getFirstPostProcess(e) { for (let t = 0; t < e.length; t++) if (e[t] !== null) return e[t]; return null; } /** * Marks the prepass renderer as dirty, triggering a check if the prepass is necessary for the next rendering. */ markAsDirty() { this._isDirty = !0; } /** * Enables a texture on the MultiRenderTarget for prepass * @param types */ _enableTextures(e) { this._scene.needsPreviousWorldMatrices = !1; for (let t = 0; t < e.length; t++) { const r = e[t]; this._textureIndices[r] === -1 && (this._textureIndices[r] = this._mrtLayout.length, this._mrtLayout.push(r), this._mrtTypes.push(mA.TextureFormats[r].type), this._mrtFormats.push(mA.TextureFormats[r].format), this._mrtNames.push(mA.TextureFormats[r].name), this.mrtCount++), r === 2 && (this._scene.needsPreviousWorldMatrices = !0); } } /** * Makes sure that the prepass renderer is up to date if it has been dirtified. */ update() { this._isDirty && this._update(); } _update() { this._disable(); let e = !1; this._scene.imageProcessingConfiguration.applyByPostProcess = !1, this._scene._depthPeelingRenderer && this._scene.useOrderIndependentTransparency && (this._scene._depthPeelingRenderer.setPrePassRenderer(this), e = !0); for (let r = 0; r < this._scene.materials.length; r++) this._scene.materials[r].setPrePassRenderer(this) && (e = !0); e && this._setRenderTargetEnabled(this.defaultRT, !0); let t; for (let r = 0; r < this.renderTargets.length; r++) { if (this.renderTargets[r].renderTargetTexture) t = this._getPostProcessesSource(this.renderTargets[r]); else { const n = this._scene.activeCamera; if (!n) continue; t = n._postProcesses; } if (t && (t = t.filter((n) => n != null), t)) { for (let n = 0; n < t.length; n++) t[n].setPrePassRenderer(this) && (this._setRenderTargetEnabled(this.renderTargets[r], !0), e = !0); this._hasImageProcessing(t) && (this._scene.imageProcessingConfiguration.applyByPostProcess = !0); } } this._markAllMaterialsAsPrePassDirty(), this._isDirty = !1, e && this._enable(); } _markAllMaterialsAsPrePassDirty() { const e = this._scene.materials; for (let t = 0; t < e.length; t++) e[t].markAsDirty(gt.PrePassDirtyFlag); } /** * Disposes the prepass renderer. */ dispose() { for (let e = this.renderTargets.length - 1; e >= 0; e--) this.renderTargets[e].dispose(); for (let e = 0; e < this._effectConfigurations.length; e++) this._effectConfigurations[e].dispose && this._effectConfigurations[e].dispose(); } } mA._SceneComponentInitialization = (A) => { throw qn("PrePassRendererSceneComponent"); }; mA.TextureFormats = [ { purpose: 0, type: 2, format: 5, name: "prePass_Irradiance" }, { purpose: 1, type: 2, format: 5, name: "prePass_Position" }, { purpose: 2, type: 0, format: 5, name: "prePass_Velocity" }, { purpose: 3, type: 0, format: 5, name: "prePass_Reflectivity" }, { purpose: 4, type: 2, format: 5, name: "prePass_Color" }, { purpose: 5, type: 1, format: 6, name: "prePass_Depth" }, { purpose: 6, type: 2, format: 5, name: "prePass_Normal" }, { purpose: 7, type: 0, format: 5, name: "prePass_Albedo" } ]; Object.defineProperty(sr.prototype, "prePassRenderer", { get: function() { return this._prePassRenderer; }, set: function(A) { A && A.isSupported && (this._prePassRenderer = A); }, enumerable: !0, configurable: !0 }); sr.prototype.enablePrePassRenderer = function() { return this._prePassRenderer ? this._prePassRenderer : (this._prePassRenderer = new mA(this), this._prePassRenderer.isSupported || (this._prePassRenderer = null, Se.Error(`PrePassRenderer needs WebGL 2 support. Maybe you tried to use the following features that need the PrePassRenderer : + Subsurface Scattering`)), this._prePassRenderer); }; sr.prototype.disablePrePassRenderer = function() { this._prePassRenderer && (this._prePassRenderer.dispose(), this._prePassRenderer = null); }; class Aie { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_PREPASSRENDERER, this.scene = e; } /** * Registers the component in a given scene */ register() { this.scene._beforeCameraDrawStage.registerStep(Ot.STEP_BEFORECAMERADRAW_PREPASS, this, this._beforeCameraDraw), this.scene._afterCameraDrawStage.registerStep(Ot.STEP_AFTERCAMERADRAW_PREPASS, this, this._afterCameraDraw), this.scene._beforeRenderTargetDrawStage.registerStep(Ot.STEP_BEFORERENDERTARGETDRAW_PREPASS, this, this._beforeRenderTargetDraw), this.scene._afterRenderTargetDrawStage.registerStep(Ot.STEP_AFTERCAMERADRAW_PREPASS, this, this._afterRenderTargetDraw), this.scene._beforeClearStage.registerStep(Ot.STEP_BEFORECLEAR_PREPASS, this, this._beforeClearStage), this.scene._beforeRenderTargetClearStage.registerStep(Ot.STEP_BEFORERENDERTARGETCLEAR_PREPASS, this, this._beforeRenderTargetClearStage), this.scene._beforeRenderingMeshStage.registerStep(Ot.STEP_BEFORERENDERINGMESH_PREPASS, this, this._beforeRenderingMeshStage), this.scene._afterRenderingMeshStage.registerStep(Ot.STEP_AFTERRENDERINGMESH_PREPASS, this, this._afterRenderingMeshStage); } _beforeRenderTargetDraw(e, t, r) { this.scene.prePassRenderer && !e.noPrePassRenderer && (this.scene.prePassRenderer._setRenderTarget(e._prePassRenderTarget), this.scene.prePassRenderer._beforeDraw(void 0, t, r)); } _afterRenderTargetDraw(e, t, r) { this.scene.prePassRenderer && !e.noPrePassRenderer && this.scene.prePassRenderer._afterDraw(t, r); } _beforeRenderTargetClearStage(e) { this.scene.prePassRenderer && !e.noPrePassRenderer && (e._prePassRenderTarget || (e._prePassRenderTarget = this.scene.prePassRenderer._createRenderTarget(e.name + "_prePassRTT", e)), this.scene.prePassRenderer._setRenderTarget(e._prePassRenderTarget), this.scene.prePassRenderer._clear()); } _beforeCameraDraw(e) { this.scene.prePassRenderer && (this.scene.prePassRenderer._setRenderTarget(null), this.scene.prePassRenderer._beforeDraw(e)); } _afterCameraDraw() { this.scene.prePassRenderer && this.scene.prePassRenderer._afterDraw(); } _beforeClearStage() { this.scene.prePassRenderer && (this.scene.prePassRenderer._setRenderTarget(null), this.scene.prePassRenderer._clear()); } _beforeRenderingMeshStage(e, t, r, n) { if (!n) return; const i = e.getScene(); i.prePassRenderer && i.prePassRenderer.bindAttachmentsForEffect(n, t); } _afterRenderingMeshStage(e) { const t = e.getScene(); t.prePassRenderer && t.prePassRenderer.restoreAttachments(); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { this.scene.disablePrePassRenderer(), this.scene.enablePrePassRenderer(); } /** * Disposes the component and the associated resources */ dispose() { this.scene.disablePrePassRenderer(); } } mA._SceneComponentInitialization = (A) => { let e = A._getComponent(Ot.NAME_PREPASSRENDERER); e || (e = new Aie(A), A._addComponent(e)); }; const uHe = "fibonacci", lHe = `#define rcp(x) 1./x #define GOLDEN_RATIO 1.618033988749895 #define TWO_PI 6.2831855 vec2 Golden2dSeq(int i,float n) {return vec2(float(i)/n+(0.5/n),fract(float(i)*rcp(GOLDEN_RATIO)));} vec2 SampleDiskGolden(int i,int sampleCount) {vec2 f=Golden2dSeq(i,float(sampleCount));return vec2(sqrt(f.x),TWO_PI*f.y);}`; Le.IncludesShadersStore[uHe] = lHe; const PHe = "diffusionProfile", cHe = "uniform vec3 diffusionS[5];uniform float diffusionD[5];uniform float filterRadii[5];"; Le.IncludesShadersStore[PHe] = cHe; const pHe = "subSurfaceScatteringPixelShader", hHe = `#include #include #include #include varying vec2 vUV;uniform vec2 texelSize;uniform sampler2D textureSampler;uniform sampler2D irradianceSampler;uniform sampler2D depthSampler;uniform sampler2D albedoSampler;uniform vec2 viewportSize;uniform float metersPerUnit;const float LOG2_E=1.4426950408889634;const float SSS_PIXELS_PER_SAMPLE=4.;const int _SssSampleBudget=40; #define rcp(x) 1./x #define Sq(x) x*x #define SSS_BILATERAL_FILTER true vec3 EvalBurleyDiffusionProfile(float r,vec3 S) {vec3 exp_13=exp2(((LOG2_E*(-1.0/3.0))*r)*S); vec3 expSum=exp_13*(1.+exp_13*exp_13); return (S*rcp(8.*PI))*expSum; } vec2 SampleBurleyDiffusionProfile(float u,float rcpS) {u=1.-u; float g=1.+(4.*u)*(2.*u+sqrt(1.+(4.*u)*u));float n=exp2(log2(g)*(-1.0/3.0)); float p=(g*n)*n; float c=1.+p+n; float d=(3./LOG2_E*2.)+(3./LOG2_E)*log2(u); float x=(3./LOG2_E)*log2(c)-d; float rcpExp=((c*c)*c)*rcp((4.*u)*((c*c)+(4.*u)*(4.*u)));float r=x*rcpS;float rcpPdf=(8.*PI*rcpS)*rcpExp; return vec2(r,rcpPdf);} vec3 ComputeBilateralWeight(float xy2,float z,float mmPerUnit,vec3 S,float rcpPdf) { #ifndef SSS_BILATERAL_FILTER z=0.; #endif float r=sqrt(xy2+(z*mmPerUnit)*(z*mmPerUnit));float area=rcpPdf; #if SSS_CLAMP_ARTIFACT return clamp(EvalBurleyDiffusionProfile(r,S)*area,0.0,1.0); #else return EvalBurleyDiffusionProfile(r,S)*area; #endif } void EvaluateSample(int i,int n,vec3 S,float d,vec3 centerPosVS,float mmPerUnit,float pixelsPerMm, float phase,inout vec3 totalIrradiance,inout vec3 totalWeight) {float scale =rcp(float(n));float offset=rcp(float(n))*0.5;float sinPhase,cosPhase;sinPhase=sin(phase);cosPhase=cos(phase);vec2 bdp=SampleBurleyDiffusionProfile(float(i)*scale+offset,d);float r=bdp.x;float rcpPdf=bdp.y;float phi=SampleDiskGolden(i,n).y;float sinPhi,cosPhi;sinPhi=sin(phi);cosPhi=cos(phi);float sinPsi=cosPhase*sinPhi+sinPhase*cosPhi; float cosPsi=cosPhase*cosPhi-sinPhase*sinPhi; vec2 vec=r*vec2(cosPsi,sinPsi);vec2 position; float xy2;position=vUV+round((pixelsPerMm*r)*vec2(cosPsi,sinPsi))*texelSize;xy2 =r*r;vec4 textureSample=texture2D(irradianceSampler,position);float viewZ=texture2D(depthSampler,position).r;vec3 irradiance =textureSample.rgb;if (testLightingForSSS(textureSample.a)) {float relZ=viewZ-centerPosVS.z;vec3 weight=ComputeBilateralWeight(xy2,relZ,mmPerUnit,S,rcpPdf);totalIrradiance+=weight*irradiance;totalWeight +=weight;} else {}} #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) {vec4 irradianceAndDiffusionProfile =texture2D(irradianceSampler,vUV);vec3 centerIrradiance=irradianceAndDiffusionProfile.rgb;int diffusionProfileIndex=int(round(irradianceAndDiffusionProfile.a*255.));float centerDepth =0.;vec4 inputColor=texture2D(textureSampler,vUV);bool passedStencilTest=testLightingForSSS(irradianceAndDiffusionProfile.a);if (passedStencilTest) {centerDepth=texture2D(depthSampler,vUV).r;} if (!passedStencilTest) { gl_FragColor=inputColor;return;} float distScale =1.;vec3 S =diffusionS[diffusionProfileIndex];float d =diffusionD[diffusionProfileIndex];float filterRadius=filterRadii[diffusionProfileIndex];vec2 centerPosNDC=vUV;vec2 cornerPosNDC=vUV+0.5*texelSize;vec3 centerPosVS =vec3(centerPosNDC*viewportSize,1.0)*centerDepth; vec3 cornerPosVS =vec3(cornerPosNDC*viewportSize,1.0)*centerDepth; float mmPerUnit =1000.*(metersPerUnit*rcp(distScale));float unitsPerMm=rcp(mmPerUnit);float unitsPerPixel=2.*abs(cornerPosVS.x-centerPosVS.x);float pixelsPerMm =rcp(unitsPerPixel)*unitsPerMm;float filterArea =PI*Sq(filterRadius*pixelsPerMm);int sampleCount =int(filterArea*rcp(SSS_PIXELS_PER_SAMPLE));int sampleBudget=_SssSampleBudget;int texturingMode=0;vec3 albedo =texture2D(albedoSampler,vUV).rgb;if (distScale==0. || sampleCount<1) { #ifdef DEBUG_SSS_SAMPLES vec3 green=vec3(0.,1.,0.);gl_FragColor=vec4(green,1.0);return; #endif gl_FragColor=vec4(inputColor.rgb+albedo*centerIrradiance,1.0);return;} #ifdef DEBUG_SSS_SAMPLES vec3 red =vec3(1.,0.,0.);vec3 blue=vec3(0.,0.,1.);gl_FragColor=vec4(mix(blue,red,clamp(float(sampleCount)/float(sampleBudget),0.0,1.0)),1.0);return; #endif float phase=0.;int n=min(sampleCount,sampleBudget);vec3 centerWeight =vec3(0.); vec3 totalIrradiance=vec3(0.);vec3 totalWeight =vec3(0.);for (int i=0; i { if (!t.prePassRenderer || !t.subSurfaceConfiguration) { Se.Error("PrePass and subsurface configuration needs to be enabled for subsurface scattering."); return; } const d = this.texelSize; o.setFloat("metersPerUnit", t.subSurfaceConfiguration.metersPerUnit), o.setFloat2("texelSize", d.x, d.y), o.setTexture("irradianceSampler", t.prePassRenderer.getRenderTarget().textures[t.prePassRenderer.getIndex(0)]), o.setTexture("depthSampler", t.prePassRenderer.getRenderTarget().textures[t.prePassRenderer.getIndex(5)]), o.setTexture("albedoSampler", t.prePassRenderer.getRenderTarget().textures[t.prePassRenderer.getIndex(7)]), o.setFloat2("viewportSize", Math.tan(t.activeCamera.fov / 2) * t.getEngine().getAspectRatio(t.activeCamera, !0), Math.tan(t.activeCamera.fov / 2)), o.setArray3("diffusionS", t.subSurfaceConfiguration.ssDiffusionS), o.setArray("diffusionD", t.subSurfaceConfiguration.ssDiffusionD), o.setArray("filterRadii", t.subSurfaceConfiguration.ssFilterRadii); }); } } class lV { /** * Diffusion profile color for subsurface scattering */ get ssDiffusionS() { return this._ssDiffusionS; } /** * Diffusion profile max color channel value for subsurface scattering */ get ssDiffusionD() { return this._ssDiffusionD; } /** * Diffusion profile filter radius for subsurface scattering */ get ssFilterRadii() { return this._ssFilterRadii; } /** * Builds a subsurface configuration object * @param scene The scene */ constructor(e) { this._ssDiffusionS = [], this._ssFilterRadii = [], this._ssDiffusionD = [], this.enabled = !1, this.needsImageProcessing = !0, this.name = Ot.NAME_SUBSURFACE, this.ssDiffusionProfileColors = [], this.metersPerUnit = 1, this.texturesRequired = [ 5, 7, 4, 0 ], this.addDiffusionProfile(new Ne(1, 1, 1)), this._scene = e, lV._SceneComponentInitialization(this._scene); } /** * Adds a new diffusion profile. * Useful for more realistic subsurface scattering on diverse materials. * @param color The color of the diffusion profile. Should be the average color of the material. * @returns The index of the diffusion profile for the material subsurface configuration */ addDiffusionProfile(e) { if (this.ssDiffusionD.length >= 5) return Se.Error("You already reached the maximum number of diffusion profiles."), 0; for (let t = 0; t < this._ssDiffusionS.length / 3; t++) if (this._ssDiffusionS[t * 3] === e.r && this._ssDiffusionS[t * 3 + 1] === e.g && this._ssDiffusionS[t * 3 + 2] === e.b) return t; return this._ssDiffusionS.push(e.r, e.b, e.g), this._ssDiffusionD.push(Math.max(Math.max(e.r, e.b), e.g)), this._ssFilterRadii.push(this.getDiffusionProfileParameters(e)), this.ssDiffusionProfileColors.push(e), this._ssDiffusionD.length - 1; } /** * Creates the sss post process * @returns The created post process */ createPostProcess() { return this.postProcess = new HHe("subSurfaceScattering", this._scene, 1, null, void 0, this._scene.getEngine()), this.postProcess.autoClear = !1, this.postProcess; } /** * Deletes all diffusion profiles. * Note that in order to render subsurface scattering, you should have at least 1 diffusion profile. */ clearAllDiffusionProfiles() { this._ssDiffusionD = [], this._ssDiffusionS = [], this._ssFilterRadii = [], this.ssDiffusionProfileColors = []; } /** * Disposes this object */ dispose() { this.clearAllDiffusionProfiles(), this.postProcess && this.postProcess.dispose(); } /** * @internal * https://zero-radiance.github.io/post/sampling-diffusion/ * * Importance sample the normalized diffuse reflectance profile for the computed value of 's'. * ------------------------------------------------------------------------------------ * R[r, phi, s] = s * (Exp[-r * s] + Exp[-r * s / 3]) / (8 * Pi * r) * PDF[r, phi, s] = r * R[r, phi, s] * CDF[r, s] = 1 - 1/4 * Exp[-r * s] - 3/4 * Exp[-r * s / 3] * ------------------------------------------------------------------------------------ * We importance sample the color channel with the widest scattering distance. */ getDiffusionProfileParameters(e) { const r = Math.max(e.r, e.g, e.b); return this._sampleBurleyDiffusionProfile(0.997, r); } /** * Performs sampling of a Normalized Burley diffusion profile in polar coordinates. * 'u' is the random number (the value of the CDF): [0, 1). * rcp(s) = 1 / ShapeParam = ScatteringDistance. * Returns the sampled radial distance, s.t. (u = 0 -> r = 0) and (u = 1 -> r = Inf). * @param u * @param rcpS */ _sampleBurleyDiffusionProfile(e, t) { e = 1 - e; const r = 1 + 4 * e * (2 * e + Math.sqrt(1 + 4 * e * e)), n = Math.pow(r, -1 / 3), s = 1 + r * n * n + n; return 3 * Math.log(s / (4 * e)) * t; } } lV._SceneComponentInitialization = (A) => { throw qn("SubSurfaceSceneComponent"); }; J1.AddParser(Ot.NAME_SUBSURFACE, (A, e) => { if (A.ssDiffusionProfileColors !== void 0 && A.ssDiffusionProfileColors !== null && (e.enableSubSurfaceForPrePass(), e.subSurfaceConfiguration)) for (let t = 0, r = A.ssDiffusionProfileColors.length; t < r; t++) { const n = A.ssDiffusionProfileColors[t]; e.subSurfaceConfiguration.addDiffusionProfile(new Ne(n.r, n.g, n.b)); } }); Object.defineProperty(sr.prototype, "subSurfaceConfiguration", { get: function() { return this._subSurfaceConfiguration; }, set: function(A) { A && this.enablePrePassRenderer() && (this._subSurfaceConfiguration = A); }, enumerable: !0, configurable: !0 }); sr.prototype.enableSubSurfaceForPrePass = function() { if (this._subSurfaceConfiguration) return this._subSurfaceConfiguration; const A = this.enablePrePassRenderer(); return A ? (this._subSurfaceConfiguration = new lV(this), A.addEffectConfiguration(this._subSurfaceConfiguration), this._subSurfaceConfiguration) : null; }; sr.prototype.disableSubSurfaceForPrePass = function() { this._subSurfaceConfiguration && (this._subSurfaceConfiguration.dispose(), this._subSurfaceConfiguration = null); }; class die { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_PREPASSRENDERER, this.scene = e; } /** * Registers the component in a given scene */ register() { } /** * Serializes the component data to the specified json object * @param serializationObject The object to serialize to */ serialize(e) { if (!this.scene.subSurfaceConfiguration) return; const t = this.scene.subSurfaceConfiguration.ssDiffusionProfileColors; e.ssDiffusionProfileColors = []; for (let r = 0; r < t.length; r++) e.ssDiffusionProfileColors.push({ r: t[r].r, g: t[r].g, b: t[r].b }); } /** * Adds all the elements from the container to the scene */ addFromContainer() { } /** * Removes all the elements in the container from the scene */ removeFromContainer() { this.scene.prePassRenderer && this.scene.subSurfaceConfiguration && this.scene.subSurfaceConfiguration.clearAllDiffusionProfiles(); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { } /** * Disposes the component and the associated resources */ dispose() { } } lV._SceneComponentInitialization = (A) => { let e = A._getComponent(Ot.NAME_SUBSURFACE); e || (e = new die(A), A._addComponent(e)); }; const gHe = "outlinePixelShader", XHe = `#ifdef LOGARITHMICDEPTH #extension GL_EXT_frag_depth : enable #endif uniform vec4 color; #ifdef ALPHATEST varying vec2 vUV;uniform sampler2D diffuseSampler; #endif #include #include #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #define CUSTOM_FRAGMENT_MAIN_BEGIN #include #ifdef ALPHATEST if (texture2D(diffuseSampler,vUV).a<0.4) discard; #endif #include gl_FragColor=color; #define CUSTOM_FRAGMENT_MAIN_END }`; Le.ShadersStore[gHe] = XHe; const THe = "outlineVertexShader", qHe = `attribute vec3 position;attribute vec3 normal; #include #include #include #include[0..maxSimultaneousMorphTargets] #include uniform float offset; #include uniform mat4 viewProjection; #ifdef ALPHATEST varying vec2 vUV;uniform mat4 diffuseMatrix; #ifdef UV1 attribute vec2 uv; #endif #ifdef UV2 attribute vec2 uv2; #endif #endif #include #define CUSTOM_VERTEX_DEFINITIONS void main(void) {vec3 positionUpdated=position;vec3 normalUpdated=normal; #ifdef UV1 vec2 uvUpdated=uv; #endif #include #include[0..maxSimultaneousMorphTargets] vec3 offsetPosition=positionUpdated+(normalUpdated*offset); #include #include #include vec4 worldPos=finalWorld*vec4(offsetPosition,1.0);gl_Position=viewProjection*worldPos; #ifdef ALPHATEST #ifdef UV1 vUV=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0)); #endif #ifdef UV2 vUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0)); #endif #endif #include #include } `; Le.ShadersStore[THe] = qHe; sr.prototype.getOutlineRenderer = function() { return this._outlineRenderer || (this._outlineRenderer = new MS(this)), this._outlineRenderer; }; Object.defineProperty(Ee.prototype, "renderOutline", { get: function() { return this._renderOutline; }, set: function(A) { A && this.getScene().getOutlineRenderer(), this._renderOutline = A; }, enumerable: !0, configurable: !0 }); Object.defineProperty(Ee.prototype, "renderOverlay", { get: function() { return this._renderOverlay; }, set: function(A) { A && this.getScene().getOutlineRenderer(), this._renderOverlay = A; }, enumerable: !0, configurable: !0 }); class MS { /** * Instantiates a new outline renderer. (There could be only one per scene). * @param scene Defines the scene it belongs to */ constructor(e) { this.name = Ot.NAME_OUTLINERENDERER, this.zOffset = 1, this.zOffsetUnits = 4, this.scene = e, this._engine = e.getEngine(), this.scene._addComponent(this), this._passIdForDrawWrapper = []; for (let t = 0; t < 4; ++t) this._passIdForDrawWrapper[t] = this._engine.createRenderPassId(`Outline Renderer (${t})`); } /** * Register the component to one instance of a scene. */ register() { this.scene._beforeRenderingMeshStage.registerStep(Ot.STEP_BEFORERENDERINGMESH_OUTLINE, this, this._beforeRenderingMesh), this.scene._afterRenderingMeshStage.registerStep(Ot.STEP_AFTERRENDERINGMESH_OUTLINE, this, this._afterRenderingMesh); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { } /** * Disposes the component and the associated resources. */ dispose() { for (let e = 0; e < this._passIdForDrawWrapper.length; ++e) this._engine.releaseRenderPassId(this._passIdForDrawWrapper[e]); } /** * Renders the outline in the canvas. * @param subMesh Defines the sumesh to render * @param batch Defines the batch of meshes in case of instances * @param useOverlay Defines if the rendering is for the overlay or the outline * @param renderPassId Render pass id to use to render the mesh */ render(e, t, r = !1, n) { n = n ?? this._passIdForDrawWrapper[0]; const i = this.scene, s = i.getEngine(), a = s.getCaps().instancedArrays && (t.visibleInstances[e._id] !== null && t.visibleInstances[e._id] !== void 0 || e.getRenderingMesh().hasThinInstances); if (!this.isReady(e, a, n)) return; const f = e.getMesh(), o = f._internalAbstractMeshDataInfo._actAsRegularMesh ? f : null, d = e.getRenderingMesh(), v = o || d, u = e.getMaterial(); if (!u || !i.activeCamera) return; const l = e._getDrawWrapper(n), P = zo.GetEffect(l); if (s.enableEffect(l), u.useLogarithmicDepth && P.setFloat("logarithmicDepthConstant", 2 / (Math.log(i.activeCamera.maxZ + 1) / Math.LN2)), P.setFloat("offset", r ? 0 : d.outlineWidth), P.setColor4("color", r ? d.overlayColor : d.outlineColor, r ? d.overlayAlpha : u.alpha), P.setMatrix("viewProjection", i.getTransformMatrix()), P.setMatrix("world", v.getWorldMatrix()), d.useBones && d.computeBonesUsingShaders && d.skeleton && P.setMatrices("mBones", d.skeleton.getTransformMatrices(d)), d.morphTargetManager && d.morphTargetManager.isUsingTextureForTargets && d.morphTargetManager._bind(P), Ye.BindMorphTargetParameters(d, P), a || d._bind(e, P, u.fillMode), u && u.needAlphaTesting()) { const p = u.getAlphaTestTexture(); p && (P.setTexture("diffuseSampler", p), P.setMatrix("diffuseMatrix", p.getTextureMatrix())); } Df(P, u, i), s.setZOffset(-this.zOffset), s.setZOffsetUnits(-this.zOffsetUnits), d._processRendering(v, e, P, u.fillMode, t, a, (p, c) => { P.setMatrix("world", c); }), s.setZOffset(0), s.setZOffsetUnits(0); } /** * Returns whether or not the outline renderer is ready for a given submesh. * All the dependencies e.g. submeshes, texture, effect... mus be ready * @param subMesh Defines the submesh to check readiness for * @param useInstances Defines whether wee are trying to render instances or not * @param renderPassId Render pass id to use to render the mesh * @returns true if ready otherwise false */ isReady(e, t, r) { r = r ?? this._passIdForDrawWrapper[0]; const n = [], i = [J.PositionKind, J.NormalKind], s = e.getMesh(), a = e.getMaterial(); if (!a) return !1; const f = s.getScene(); a.needAlphaTesting() && (n.push("#define ALPHATEST"), s.isVerticesDataPresent(J.UVKind) && (i.push(J.UVKind), n.push("#define UV1")), s.isVerticesDataPresent(J.UV2Kind) && (i.push(J.UV2Kind), n.push("#define UV2"))), a.useLogarithmicDepth && n.push("#define LOGARITHMICDEPTH"), xq(a, f, n), s.useBones && s.computeBonesUsingShaders ? (i.push(J.MatricesIndicesKind), i.push(J.MatricesWeightsKind), s.numBoneInfluencers > 4 && (i.push(J.MatricesIndicesExtraKind), i.push(J.MatricesWeightsExtraKind)), n.push("#define NUM_BONE_INFLUENCERS " + s.numBoneInfluencers), n.push("#define BonesPerMesh " + (s.skeleton ? s.skeleton.bones.length + 1 : 0))) : n.push("#define NUM_BONE_INFLUENCERS 0"); const o = s.morphTargetManager; let d = 0; o && o.numInfluencers > 0 && (d = o.numInfluencers, n.push("#define MORPHTARGETS"), n.push("#define NUM_MORPH_INFLUENCERS " + d), o.isUsingTextureForTargets && n.push("#define MORPHTARGETS_TEXTURE"), Ye.PrepareAttributesForMorphTargetsInfluencers(i, s, d)), t && (n.push("#define INSTANCES"), Ye.PushAttributesForInstances(i), e.getRenderingMesh().hasThinInstances && n.push("#define THIN_INSTANCES")); const v = e._getDrawWrapper(r, !0), u = v.defines, l = n.join(` `); if (u !== l) { const P = [ "world", "mBones", "viewProjection", "diffuseMatrix", "offset", "color", "logarithmicDepthConstant", "morphTargetInfluences", "morphTargetTextureInfo", "morphTargetTextureIndices" ]; Mf(P), v.setEffect(this.scene.getEngine().createEffect("outline", i, P, ["diffuseSampler", "morphTargets"], l, void 0, void 0, void 0, { maxSimultaneousMorphTargets: d }), l); } return v.effect.isReady(); } _beforeRenderingMesh(e, t, r) { if (this._savedDepthWrite = this._engine.getDepthWrite(), e.renderOutline) { const n = t.getMaterial(); n && n.needAlphaBlendingForMesh(e) && (this._engine.cacheStencilState(), this._engine.setDepthWrite(!1), this._engine.setColorWrite(!1), this._engine.setStencilBuffer(!0), this._engine.setStencilOperationPass(7681), this._engine.setStencilFunction(519), this._engine.setStencilMask(MS._StencilReference), this._engine.setStencilFunctionReference(MS._StencilReference), this._engine.stencilStateComposer.useStencilGlobalOnly = !0, this.render( t, r, /* This sets offset to 0 */ !0, this._passIdForDrawWrapper[1] ), this._engine.setColorWrite(!0), this._engine.setStencilFunction(517)), this._engine.setDepthWrite(!1), this.render(t, r, !1, this._passIdForDrawWrapper[0]), this._engine.setDepthWrite(this._savedDepthWrite), n && n.needAlphaBlendingForMesh(e) && (this._engine.stencilStateComposer.useStencilGlobalOnly = !1, this._engine.restoreStencilState()); } } _afterRenderingMesh(e, t, r) { if (e.renderOverlay) { const n = this._engine.getAlphaMode(), i = this._engine.alphaState.alphaBlend; this._engine.setAlphaMode(2), this.render(t, r, !0, this._passIdForDrawWrapper[3]), this._engine.setAlphaMode(n), this._engine.setDepthWrite(this._savedDepthWrite), this._engine.alphaState.alphaBlend = i; } e.renderOutline && this._savedDepthWrite && (this._engine.setDepthWrite(!0), this._engine.setColorWrite(!1), this.render(t, r, !1, this._passIdForDrawWrapper[2]), this._engine.setColorWrite(!0)); } } MS._StencilReference = 4; class LY { /** Gets or sets the size of the particle */ get particleSize() { return this._particleSize; } set particleSize(e) { e !== this._particleSize && (this._particleSize = e, this.onParticleSizeChanged.notifyObservers(this)); } /** Indicates if the object uses instancing or not */ get useInstancing() { return !this.indexBuffer; } /** Indicates if velocity of particles should be used when rendering the object. The vertex buffer set must contain a "velocity" buffer for this to work! */ get useVelocity() { return this._useVelocity; } set useVelocity(e) { this._useVelocity === e || !this._hasVelocity() || (this._useVelocity = e, this._effectsAreDirty = !0); } _hasVelocity() { var e; return !!(!((e = this.vertexBuffers) === null || e === void 0) && e.velocity); } /** * Gets the index buffer (or null if the object is using instancing) */ get indexBuffer() { return null; } /** * Gets the name of the class */ getClassName() { return "FluidRenderingObject"; } /** * Instantiates a fluid rendering object * @param scene The scene the object is part of */ constructor(e) { this.priority = 0, this._particleSize = 0.1, this.onParticleSizeChanged = new Oe(), this.particleThicknessAlpha = 0.05, this._useVelocity = !1, this._scene = e, this._engine = e.getEngine(), this._effectsAreDirty = !0, this._depthEffectWrapper = null, this._thicknessEffectWrapper = null; } _createEffects() { const e = ["view", "projection", "particleRadius", "size"], t = ["position", "offset"], r = []; this._effectsAreDirty = !1, this.useVelocity && (t.push("velocity"), r.push("#define FLUIDRENDERING_VELOCITY")), this._scene.useRightHandedSystem && r.push("#define FLUIDRENDERING_RHS"), this._depthEffectWrapper = new ng({ engine: this._engine, useShaderStore: !0, vertexShader: "fluidRenderingParticleDepth", fragmentShader: "fluidRenderingParticleDepth", attributeNames: t, uniformNames: e, samplerNames: [], defines: r }), e.push("particleAlpha"), this._thicknessEffectWrapper = new ng({ engine: this._engine, useShaderStore: !0, vertexShader: "fluidRenderingParticleThickness", fragmentShader: "fluidRenderingParticleThickness", attributeNames: ["position", "offset"], uniformNames: e, samplerNames: [] }); } /** * Indicates if the object is ready to be rendered * @returns True if everything is ready for the object to be rendered, otherwise false */ isReady() { if (this._effectsAreDirty && this._createEffects(), !this._depthEffectWrapper || !this._thicknessEffectWrapper) return !1; const e = this._depthEffectWrapper._drawWrapper.effect, t = this._thicknessEffectWrapper._drawWrapper.effect; return e.isReady() && t.isReady(); } /** * Render the depth texture for this object */ renderDepthTexture() { const e = this.numParticles; if (!this._depthEffectWrapper || e === 0) return; const t = this._depthEffectWrapper._drawWrapper, r = t.effect; this._engine.enableEffect(t), this._engine.bindBuffers(this.vertexBuffers, this.indexBuffer, r), r.setMatrix("view", this._scene.getViewMatrix()), r.setMatrix("projection", this._scene.getProjectionMatrix()), r.setFloat2("size", this._particleSize, this._particleSize), r.setFloat("particleRadius", this._particleSize / 2), this.useInstancing ? this._engine.drawArraysType(7, 0, 4, e) : this._engine.drawElementsType(0, 0, e); } /** * Render the thickness texture for this object */ renderThicknessTexture() { const e = this.numParticles; if (!this._thicknessEffectWrapper || e === 0) return; const t = this._thicknessEffectWrapper._drawWrapper, r = t.effect; this._engine.setAlphaMode(6), this._engine.setDepthWrite(!1), this._engine.enableEffect(t), this._engine.bindBuffers(this.vertexBuffers, this.indexBuffer, r), r.setMatrix("view", this._scene.getViewMatrix()), r.setMatrix("projection", this._scene.getProjectionMatrix()), r.setFloat("particleAlpha", this.particleThicknessAlpha), r.setFloat2("size", this._particleSize, this._particleSize), this.useInstancing ? this._engine.drawArraysType(7, 0, 4, e) : this._engine.drawElementsType(0, 0, e), this._engine.setDepthWrite(!0), this._engine.setAlphaMode(0); } /** * Render the diffuse texture for this object */ renderDiffuseTexture() { } /** * Releases the ressources used by the class */ dispose() { var e, t; (e = this._depthEffectWrapper) === null || e === void 0 || e.dispose(), (t = this._thicknessEffectWrapper) === null || t === void 0 || t.dispose(); } } class vie extends LY { /** Gets the particle system */ get particleSystem() { return this._particleSystem; } /** * Gets the name of the class */ getClassName() { return "FluidRenderingObjectParticleSystem"; } /** * Gets or sets a boolean indicating that the diffuse texture should be generated based on the regular rendering of the particle system (default: true). * Sometimes, generating the diffuse texture this way may be sub-optimal. In that case, you can disable this property, in which case the particle system will be * rendered using a ALPHA_COMBINE mode instead of the one used by the particle system. */ get useTrueRenderingForDiffuseTexture() { return this._useTrueRenderingForDiffuseTexture; } set useTrueRenderingForDiffuseTexture(e) { this._useTrueRenderingForDiffuseTexture !== e && (this._useTrueRenderingForDiffuseTexture = e, e ? (this._particleSystem.blendMode = this._blendMode, this._particleSystem.onBeforeDrawParticlesObservable.remove(this._onBeforeDrawParticleObserver), this._onBeforeDrawParticleObserver = null) : (this._particleSystem.blendMode = -1, this._onBeforeDrawParticleObserver = this._particleSystem.onBeforeDrawParticlesObservable.add(() => { this._engine.setAlphaMode(2); }))); } /** * Gets the vertex buffers */ get vertexBuffers() { return this._particleSystem.vertexBuffers; } /** * Gets the index buffer (or null if the object is using instancing) */ get indexBuffer() { return this._particleSystem.indexBuffer; } /** * Creates a new instance of the class * @param scene The scene the particle system is part of * @param ps The particle system */ constructor(e, t) { super(e), this._useTrueRenderingForDiffuseTexture = !0, this._particleSystem = t, this._originalRender = t.render.bind(t), this._blendMode = t.blendMode, this._onBeforeDrawParticleObserver = null, this._updateInAnimate = this._particleSystem.updateInAnimate, this._particleSystem.updateInAnimate = !0, this._particleSystem.render = () => 0, this.particleSize = (t.minSize + t.maxSize) / 2, this.useTrueRenderingForDiffuseTexture = !1; } /** * Indicates if the object is ready to be rendered * @returns True if everything is ready for the object to be rendered, otherwise false */ isReady() { return super.isReady() && this._particleSystem.isReady(); } /** * Gets the number of particles in this particle system * @returns The number of particles */ get numParticles() { return this._particleSystem.getActiveCount(); } /** * Render the diffuse texture for this object */ renderDiffuseTexture() { this._originalRender(); } /** * Releases the ressources used by the class */ dispose() { super.dispose(), this._particleSystem.onBeforeDrawParticlesObservable.remove(this._onBeforeDrawParticleObserver), this._onBeforeDrawParticleObserver = null, this._particleSystem.render = this._originalRender, this._particleSystem.blendMode = this._blendMode, this._particleSystem.updateInAnimate = this._updateInAnimate; } } class IE { get blurNumIterations() { return this._blurNumIterations; } set blurNumIterations(e) { if (this._blurNumIterations !== e && (this._blurNumIterations = e, this._blurPostProcesses !== null)) { const t = this._blurPostProcesses[0], r = this._blurPostProcesses[1]; this._blurPostProcesses = []; for (let n = 0; n < this._blurNumIterations * 2; ++n) this._blurPostProcesses[n] = n & 1 ? r : t; } } get renderTarget() { return this._rt; } get renderTargetBlur() { return this._rtBlur; } get texture() { return this._texture; } get textureBlur() { return this._textureBlurred; } constructor(e, t, r, n, i, s, a = 1, f = 6, o = 1, d = 6, v = !1, u = null, l = !0, P = 1) { this.enableBlur = !0, this.blurSizeDivisor = 1, this.blurFilterSize = 7, this._blurNumIterations = 3, this.blurMaxFilterSize = 100, this.blurDepthScale = 10, this.particleSize = 0.02, this.onDisposeObservable = new Oe(), this._name = e, this._scene = t, this._camera = u, this._engine = t.getEngine(), this._width = r, this._height = n, this._blurTextureSizeX = i, this._blurTextureSizeY = s, this._textureType = a, this._textureFormat = f, this._blurTextureType = o, this._blurTextureFormat = d, this._useStandardBlur = v, this._generateDepthBuffer = l, this._samples = P, this._postProcessRunningIndex = 0, this.enableBlur = i !== 0 && s !== 0, this._rt = null, this._texture = null, this._rtBlur = null, this._textureBlurred = null, this._blurPostProcesses = null; } initialize() { if (this.dispose(), this._createRenderTarget(), this.enableBlur && this._texture) { const [e, t, r] = this._createBlurPostProcesses(this._texture, this._blurTextureType, this._blurTextureFormat, this.blurSizeDivisor, this._name, this._useStandardBlur); this._rtBlur = e, this._textureBlurred = t, this._blurPostProcesses = r; } } applyBlurPostProcesses() { this.enableBlur && this._blurPostProcesses && (this._postProcessRunningIndex = 0, this._scene.postProcessManager.directRender(this._blurPostProcesses, this._rtBlur, !0), this._engine.unBindFramebuffer(this._rtBlur)); } _createRenderTarget() { this._rt = this._engine.createRenderTargetTexture({ width: this._width, height: this._height }, { generateMipMaps: !1, type: this._textureType, format: this._textureFormat, samplingMode: 1, generateDepthBuffer: this._generateDepthBuffer, generateStencilBuffer: !1, samples: this._samples, label: `FluidRenderingRTT-${this._name}` }); const e = this._rt.texture; e.incrementReferences(), this._texture = new We(null, this._scene), this._texture.name = "rtt" + this._name, this._texture._texture = e, this._texture.wrapU = We.CLAMP_ADDRESSMODE, this._texture.wrapV = We.CLAMP_ADDRESSMODE, this._texture.anisotropicFilteringLevel = 1; } _createBlurPostProcesses(e, t, r, n, i, s = !1) { const a = this._scene.getEngine(), f = new at(Math.floor(this._blurTextureSizeX / n), Math.floor(this._blurTextureSizeY / n)), o = t === 1 && a.getCaps().textureFloatLinearFiltering || t === 2 && a.getCaps().textureHalfFloatLinearFiltering, d = this._engine.createRenderTargetTexture({ width: f.x, height: f.y }, { generateMipMaps: !1, type: t, format: r, samplingMode: o ? 2 : 1, generateDepthBuffer: !1, generateStencilBuffer: !1, samples: this._samples, label: `FluidRenderingRTTBlur-${i}` }), v = d.texture; v.incrementReferences(); const u = new We(null, this._scene); if (u.name = "rttBlurred" + i, u._texture = v, u.wrapU = We.CLAMP_ADDRESSMODE, u.wrapV = We.CLAMP_ADDRESSMODE, u.anisotropicFilteringLevel = 1, s) { const l = new kr("BilateralBlurX", "fluidRenderingStandardBlur", ["filterSize", "blurDir"], null, 1, null, 1, a, !0, null, t, void 0, void 0, void 0, r); l.samples = this._samples, l.externalTextureSamplerBinding = !0, l.onApplyObservable.add((c) => { this._postProcessRunningIndex === 0 ? c.setTexture("textureSampler", e) : c._bindTexture("textureSampler", l.inputTexture.texture), c.setInt("filterSize", this.blurFilterSize), c.setFloat2("blurDir", 1 / this._blurTextureSizeX, 0), this._postProcessRunningIndex++; }), l.onSizeChangedObservable.add(() => { l._textures.forEach((c) => { c.texture.wrapU = We.CLAMP_ADDRESSMODE, c.texture.wrapV = We.CLAMP_ADDRESSMODE; }); }), this._fixReusablePostProcess(l); const P = new kr("BilateralBlurY", "fluidRenderingStandardBlur", ["filterSize", "blurDir"], null, 1, null, 1, a, !0, null, t, void 0, void 0, void 0, r); P.samples = this._samples, P.onApplyObservable.add((c) => { c.setInt("filterSize", this.blurFilterSize), c.setFloat2("blurDir", 0, 1 / this._blurTextureSizeY), this._postProcessRunningIndex++; }), P.onSizeChangedObservable.add(() => { P._textures.forEach((c) => { c.texture.wrapU = We.CLAMP_ADDRESSMODE, c.texture.wrapV = We.CLAMP_ADDRESSMODE; }); }), this._fixReusablePostProcess(P), l.autoClear = !1, P.autoClear = !1; const p = []; for (let c = 0; c < this._blurNumIterations * 2; ++c) p[c] = c & 1 ? P : l; return [d, u, p]; } else { const l = ["maxFilterSize", "blurDir", "projectedParticleConstant", "depthThreshold"], P = new kr("BilateralBlurX", "fluidRenderingBilateralBlur", l, null, 1, null, 1, a, !0, null, t, void 0, void 0, void 0, r); P.samples = this._samples, P.externalTextureSamplerBinding = !0, P.onApplyObservable.add((H) => { this._postProcessRunningIndex === 0 ? H.setTexture("textureSampler", e) : H._bindTexture("textureSampler", P.inputTexture.texture), H.setInt("maxFilterSize", this.blurMaxFilterSize), H.setFloat2("blurDir", 1 / this._blurTextureSizeX, 0), H.setFloat("projectedParticleConstant", this._getProjectedParticleConstant()), H.setFloat("depthThreshold", this._getDepthThreshold()), this._postProcessRunningIndex++; }), P.onSizeChangedObservable.add(() => { P._textures.forEach((H) => { H.texture.wrapU = We.CLAMP_ADDRESSMODE, H.texture.wrapV = We.CLAMP_ADDRESSMODE; }); }), this._fixReusablePostProcess(P); const p = new kr("BilateralBlurY", "fluidRenderingBilateralBlur", l, null, 1, null, 1, a, !0, null, t, void 0, void 0, void 0, r); p.samples = this._samples, p.onApplyObservable.add((H) => { H.setInt("maxFilterSize", this.blurMaxFilterSize), H.setFloat2("blurDir", 0, 1 / this._blurTextureSizeY), H.setFloat("projectedParticleConstant", this._getProjectedParticleConstant()), H.setFloat("depthThreshold", this._getDepthThreshold()), this._postProcessRunningIndex++; }), p.onSizeChangedObservable.add(() => { p._textures.forEach((H) => { H.texture.wrapU = We.CLAMP_ADDRESSMODE, H.texture.wrapV = We.CLAMP_ADDRESSMODE; }); }), this._fixReusablePostProcess(p), P.autoClear = !1, p.autoClear = !1; const c = []; for (let H = 0; H < this._blurNumIterations * 2; ++H) c[H] = H & 1 ? p : P; return [d, u, c]; } } _fixReusablePostProcess(e) { e.isReusable() && (e.onActivateObservable.add(() => { e._currentRenderTextureInd = (e._currentRenderTextureInd + 1) % 2; }), e.onApplyObservable.add(() => { e._currentRenderTextureInd = (e._currentRenderTextureInd + 1) % 2; })); } _getProjectedParticleConstant() { var e, t; return this.blurFilterSize * this.particleSize * 0.05 * (this._height / 2) / Math.tan(((t = (e = this._camera) === null || e === void 0 ? void 0 : e.fov) !== null && t !== void 0 ? t : 45 * Math.PI / 180) / 2); } _getDepthThreshold() { return this.particleSize / 2 * this.blurDepthScale; } dispose() { var e, t, r, n; this.onDisposeObservable.hasObservers() && this.onDisposeObservable.notifyObservers(this), (e = this._rt) === null || e === void 0 || e.dispose(), this._rt = null, (t = this._texture) === null || t === void 0 || t.dispose(), this._texture = null, (r = this._rtBlur) === null || r === void 0 || r.dispose(), this._rtBlur = null, (n = this._textureBlurred) === null || n === void 0 || n.dispose(), this._textureBlurred = null, this._blurPostProcesses && (this._blurPostProcesses[0].dispose(), this._blurPostProcesses[1].dispose()), this._blurPostProcesses = null; } } var Pc; (function(A) { A[A.DepthTexture = 0] = "DepthTexture", A[A.DepthBlurredTexture = 1] = "DepthBlurredTexture", A[A.ThicknessTexture = 2] = "ThicknessTexture", A[A.ThicknessBlurredTexture = 3] = "ThicknessBlurredTexture", A[A.DiffuseTexture = 4] = "DiffuseTexture", A[A.Normals = 5] = "Normals", A[A.DiffuseRendering = 6] = "DiffuseRendering"; })(Pc || (Pc = {})); class wF { /** * Returns true if the class needs to be reinitialized (because of changes in parameterization) */ get needInitialization() { return this._needInitialization; } /** * Gets or sets a boolean indicating that the diffuse texture should be generated and used for the rendering */ get generateDiffuseTexture() { return this._generateDiffuseTexture; } set generateDiffuseTexture(e) { this._generateDiffuseTexture !== e && (this._generateDiffuseTexture = e, this._needInitialization = !0); } /** * Gets or sets the feature (texture) to be debugged. Not used if debug is false */ get debugFeature() { return this._debugFeature; } set debugFeature(e) { this._debugFeature !== e && (this._needInitialization = !0, this._debugFeature = e); } /** * Gets or sets a boolean indicating if we should display a specific texture (given by debugFeature) for debugging purpose */ get debug() { return this._debug; } set debug(e) { this._debug !== e && (this._debug = e, this._needInitialization = !0); } /** * Gets or sets the environment map used for the reflection part of the shading * If null, no map will be used. If undefined, the scene.environmentMap will be used (if defined) */ get environmentMap() { return this._environmentMap; } set environmentMap(e) { this._environmentMap !== e && (this._needInitialization = !0, this._environmentMap = e); } /** * Gets or sets a boolean indicating that the depth texture should be blurred */ get enableBlurDepth() { return this._enableBlurDepth; } set enableBlurDepth(e) { this._enableBlurDepth !== e && (this._enableBlurDepth = e, this._needInitialization = !0); } /** * Gets or sets the depth size divisor (positive number, generally between 1 and 4), which is used as a divisor when creating the texture used for blurring the depth * For eg. if blurDepthSizeDivisor=2, the texture used to blur the depth will be half the size of the depth texture */ get blurDepthSizeDivisor() { return this._blurDepthSizeDivisor; } set blurDepthSizeDivisor(e) { this._blurDepthSizeDivisor !== e && (this._blurDepthSizeDivisor = e, this._needInitialization = !0); } /** * Size of the kernel used to filter the depth blur texture (positive number, generally between 1 and 20 - higher values will require more processing power from the GPU) */ get blurDepthFilterSize() { return this._blurDepthFilterSize; } set blurDepthFilterSize(e) { this._blurDepthFilterSize !== e && (this._blurDepthFilterSize = e, this._setBlurParameters()); } /** * Number of blurring iterations used to generate the depth blur texture (positive number, generally between 1 and 10 - higher values will require more processing power from the GPU) */ get blurDepthNumIterations() { return this._blurDepthNumIterations; } set blurDepthNumIterations(e) { this._blurDepthNumIterations !== e && (this._blurDepthNumIterations = e, this._setBlurParameters()); } /** * Maximum size of the kernel used to blur the depth texture (positive number, generally between 1 and 200 - higher values will require more processing power from the GPU when the particles are larger on screen) */ get blurDepthMaxFilterSize() { return this._blurDepthMaxFilterSize; } set blurDepthMaxFilterSize(e) { this._blurDepthMaxFilterSize !== e && (this._blurDepthMaxFilterSize = e, this._setBlurParameters()); } /** * Depth weight in the calculation when applying the bilateral blur to generate the depth blur texture (positive number, generally between 0 and 100) */ get blurDepthDepthScale() { return this._blurDepthDepthScale; } set blurDepthDepthScale(e) { this._blurDepthDepthScale !== e && (this._blurDepthDepthScale = e, this._setBlurParameters()); } /** * Gets or sets a boolean indicating that the thickness texture should be blurred */ get enableBlurThickness() { return this._enableBlurThickness; } set enableBlurThickness(e) { this._enableBlurThickness !== e && (this._enableBlurThickness = e, this._needInitialization = !0); } /** * Gets or sets the thickness size divisor (positive number, generally between 1 and 4), which is used as a divisor when creating the texture used for blurring the thickness * For eg. if blurThicknessSizeDivisor=2, the texture used to blur the thickness will be half the size of the thickness texture */ get blurThicknessSizeDivisor() { return this._blurThicknessSizeDivisor; } set blurThicknessSizeDivisor(e) { this._blurThicknessSizeDivisor !== e && (this._blurThicknessSizeDivisor = e, this._needInitialization = !0); } /** * Size of the kernel used to filter the thickness blur texture (positive number, generally between 1 and 20 - higher values will require more processing power from the GPU) */ get blurThicknessFilterSize() { return this._blurThicknessFilterSize; } set blurThicknessFilterSize(e) { this._blurThicknessFilterSize !== e && (this._blurThicknessFilterSize = e, this._setBlurParameters()); } /** * Number of blurring iterations used to generate the thickness blur texture (positive number, generally between 1 and 10 - higher values will require more processing power from the GPU) */ get blurThicknessNumIterations() { return this._blurThicknessNumIterations; } set blurThicknessNumIterations(e) { this._blurThicknessNumIterations !== e && (this._blurThicknessNumIterations = e, this._setBlurParameters()); } /** * Gets or sets a boolean indicating that a fixed thickness should be used instead of generating a thickness texture */ get useFixedThickness() { return this._useFixedThickness; } set useFixedThickness(e) { this._useFixedThickness !== e && (this._useFixedThickness = e, this._needInitialization = !0); } /** * Gets or sets a boolean indicating that the velocity should be used when rendering the particles as a fluid. * Note: the vertex buffers must contain a "velocity" buffer for this to work! */ get useVelocity() { return this._useVelocity; } set useVelocity(e) { this._useVelocity !== e && (this._useVelocity = e, this._needInitialization = !0, this._onUseVelocityChanged.notifyObservers(this)); } /** * Defines the size of the depth texture. * If null, the texture will have the size of the screen */ get depthMapSize() { return this._depthMapSize; } set depthMapSize(e) { this._depthMapSize !== e && (this._depthMapSize = e, this._needInitialization = !0); } /** * Defines the size of the thickness texture. * If null, the texture will have the size of the screen */ get thicknessMapSize() { return this._thicknessMapSize; } set thicknessMapSize(e) { this._thicknessMapSize !== e && (this._thicknessMapSize = e, this._needInitialization = !0); } /** * Defines the size of the diffuse texture. * If null, the texture will have the size of the screen */ get diffuseMapSize() { return this._diffuseMapSize; } set diffuseMapSize(e) { this._diffuseMapSize !== e && (this._diffuseMapSize = e, this._needInitialization = !0); } /** * Gets or sets the number of samples used by MSAA * Note: changing this value in WebGL does not work because depth/stencil textures can't be created with MSAA (see https://github.com/BabylonJS/Babylon.js/issues/12444) */ get samples() { return this._samples; } set samples(e) { this._samples !== e && (this._samples = e, this._needInitialization = !0); } /** * Gets the camera used for the rendering */ get camera() { return this._camera; } /** * Creates an instance of the class * @param scene Scene used to render the fluid object into * @param camera Camera used to render the fluid object. If not provided, use the active camera of the scene instead */ constructor(e, t) { this._generateDiffuseTexture = !1, this.fluidColor = new Ne(0.085, 0.6375, 0.765), this.density = 2, this.refractionStrength = 0.1, this.fresnelClamp = 1, this.specularPower = 250, this.minimumThickness = 0, this.dirLight = new S(-2, -1, 1).normalize(), this._debugFeature = Pc.DepthBlurredTexture, this._debug = !1, this._enableBlurDepth = !0, this._blurDepthSizeDivisor = 1, this._blurDepthFilterSize = 7, this._blurDepthNumIterations = 3, this._blurDepthMaxFilterSize = 100, this._blurDepthDepthScale = 10, this._enableBlurThickness = !0, this._blurThicknessSizeDivisor = 1, this._blurThicknessFilterSize = 5, this._blurThicknessNumIterations = 1, this._useFixedThickness = !1, this._onUseVelocityChanged = new Oe(), this._useVelocity = !1, this._depthMapSize = null, this._thicknessMapSize = null, this._diffuseMapSize = null, this._samples = 1, this._scene = e, this._engine = e.getEngine(), this._camera = t ?? e.activeCamera, this._needInitialization = !0, this._bgDepthTexture = null, this._invProjectionMatrix = new he(), this._depthClearColor = new xt(1e6, 1e6, 1e6, 1), this._thicknessClearColor = new xt(0, 0, 0, 1), this._depthRenderTarget = null, this._diffuseRenderTarget = null, this._thicknessRenderTarget = null, this._renderPostProcess = null; } /** @internal */ _initialize() { var e, t, r; this.dispose(), this._needInitialization = !1; const n = (e = this._depthMapSize) !== null && e !== void 0 ? e : this._engine.getRenderWidth(), i = this._depthMapSize !== null ? Math.round(this._depthMapSize * this._engine.getRenderHeight() / this._engine.getRenderWidth()) : this._engine.getRenderHeight(); if (this._depthRenderTarget = new IE("Depth", this._scene, n, i, n, i, 1, 7, 1, 7, !1, this._camera, !0, this._samples), this._initializeRenderTarget(this._depthRenderTarget), this.generateDiffuseTexture) { const f = (t = this._diffuseMapSize) !== null && t !== void 0 ? t : this._engine.getRenderWidth(), o = this._diffuseMapSize !== null ? Math.round(this._diffuseMapSize * this._engine.getRenderHeight() / this._engine.getRenderWidth()) : this._engine.getRenderHeight(); this._diffuseRenderTarget = new IE("Diffuse", this._scene, f, o, 0, 0, 0, 5, 0, 5, !0, this._camera, !0, this._samples), this._initializeRenderTarget(this._diffuseRenderTarget); } const s = (r = this._thicknessMapSize) !== null && r !== void 0 ? r : this._engine.getRenderWidth(), a = this._thicknessMapSize !== null ? Math.round(this._thicknessMapSize * this._engine.getRenderHeight() / this._engine.getRenderWidth()) : this._engine.getRenderHeight(); this._useFixedThickness || (this._thicknessRenderTarget = new IE("Thickness", this._scene, s, a, s, a, 2, 6, 2, 6, !0, this._camera, !1, this._samples), this._initializeRenderTarget(this._thicknessRenderTarget)), this._createLiquidRenderingPostProcess(); } _setBlurParameters(e = null) { (e === null || e === this._depthRenderTarget) && this._setBlurDepthParameters(), (e === null || e === this._thicknessRenderTarget) && this._setBlurThicknessParameters(); } _setBlurDepthParameters() { this._depthRenderTarget && (this._depthRenderTarget.blurFilterSize = this.blurDepthFilterSize, this._depthRenderTarget.blurMaxFilterSize = this.blurDepthMaxFilterSize, this._depthRenderTarget.blurNumIterations = this.blurDepthNumIterations, this._depthRenderTarget.blurDepthScale = this.blurDepthDepthScale); } _setBlurThicknessParameters() { this._thicknessRenderTarget && (this._thicknessRenderTarget.blurFilterSize = this.blurThicknessFilterSize, this._thicknessRenderTarget.blurNumIterations = this.blurThicknessNumIterations); } _initializeRenderTarget(e) { e !== this._diffuseRenderTarget && (e.enableBlur = e === this._depthRenderTarget ? this.enableBlurDepth : this.enableBlurThickness, e.blurSizeDivisor = e === this._depthRenderTarget ? this.blurDepthSizeDivisor : this.blurThicknessSizeDivisor), this._setBlurParameters(e), e.initialize(); } _createLiquidRenderingPostProcess() { var e; const t = this._scene.getEngine(), r = [ "viewMatrix", "projectionMatrix", "invProjectionMatrix", "texelSize", "dirLight", "cameraFar", "density", "refractionStrength", "fresnelClamp", "specularPower" ], n = ["depthSampler"], i = []; if (this.dispose(!0), !this._camera) return; const s = this._depthRenderTarget.enableBlur ? this._depthRenderTarget.textureBlur : this._depthRenderTarget.texture, a = new at(1 / s.getSize().width, 1 / s.getSize().height); this._scene.useRightHandedSystem && i.push("#define FLUIDRENDERING_RHS"), this._environmentMap !== null && ((e = this._environmentMap) !== null && e !== void 0 ? e : this._scene.environmentTexture) && (n.push("reflectionSampler"), i.push("#define FLUIDRENDERING_ENVIRONMENT")), this._diffuseRenderTarget ? (n.push("diffuseSampler"), i.push("#define FLUIDRENDERING_DIFFUSETEXTURE")) : r.push("diffuseColor"), this._useVelocity && (n.push("velocitySampler"), i.push("#define FLUIDRENDERING_VELOCITY")), this._useFixedThickness ? (r.push("thickness"), n.push("bgDepthSampler"), i.push("#define FLUIDRENDERING_FIXED_THICKNESS")) : (r.push("minimumThickness"), n.push("thicknessSampler")), this._debug && (i.push("#define FLUIDRENDERING_DEBUG"), this._debugFeature === Pc.Normals ? i.push("#define FLUIDRENDERING_DEBUG_SHOWNORMAL") : this._debugFeature === Pc.DiffuseRendering ? i.push("#define FLUIDRENDERING_DEBUG_DIFFUSERENDERING") : (i.push("#define FLUIDRENDERING_DEBUG_TEXTURE"), n.push("debugSampler"), (this._debugFeature === Pc.DepthTexture || this._debugFeature === Pc.DepthBlurredTexture) && i.push("#define FLUIDRENDERING_DEBUG_DEPTH"))), this._renderPostProcess = new kr("FluidRendering", "fluidRenderingRender", r, n, 1, null, 2, t, !1, null, 0, void 0, void 0, !0, void 0), this._renderPostProcess.updateEffect(i.join(` `)), this._renderPostProcess.samples = this._samples, this._renderPostProcess.onApplyObservable.add((f) => { var o, d, v, u, l, P, p, c, H, T, q, b, j, w, m, I, N, k, R, y, O, Y, ee; if (this._invProjectionMatrix.copyFrom(this._scene.getProjectionMatrix()), this._invProjectionMatrix.invert(), t.isWebGPU && f.setTextureSampler("textureSamplerSampler", this._renderPostProcess.inputTexture.texture), this._depthRenderTarget.enableBlur ? (f.setTexture("depthSampler", this._depthRenderTarget.textureBlur), t.isWebGPU && f.setTextureSampler("depthSamplerSampler", (u = (v = this._depthRenderTarget.textureBlur) === null || v === void 0 ? void 0 : v.getInternalTexture()) !== null && u !== void 0 ? u : null)) : (f.setTexture("depthSampler", this._depthRenderTarget.texture), t.isWebGPU && f.setTextureSampler("depthSamplerSampler", (d = (o = this._depthRenderTarget.texture) === null || o === void 0 ? void 0 : o.getInternalTexture()) !== null && d !== void 0 ? d : null)), this._diffuseRenderTarget ? this._diffuseRenderTarget.enableBlur ? (f.setTexture("diffuseSampler", this._diffuseRenderTarget.textureBlur), t.isWebGPU && f.setTextureSampler("diffuseSamplerSampler", (c = (p = this._diffuseRenderTarget.textureBlur) === null || p === void 0 ? void 0 : p.getInternalTexture()) !== null && c !== void 0 ? c : null)) : (f.setTexture("diffuseSampler", this._diffuseRenderTarget.texture), t.isWebGPU && f.setTextureSampler("diffuseSamplerSampler", (P = (l = this._diffuseRenderTarget.texture) === null || l === void 0 ? void 0 : l.getInternalTexture()) !== null && P !== void 0 ? P : null)) : f.setColor3("diffuseColor", this.fluidColor), this._useFixedThickness ? (f.setFloat("thickness", this.minimumThickness), f._bindTexture("bgDepthSampler", this._bgDepthTexture), t.isWebGPU && f.setTextureSampler("bgDepthSamplerSampler", (H = this._bgDepthTexture) !== null && H !== void 0 ? H : null)) : (this._thicknessRenderTarget.enableBlur ? (f.setTexture("thicknessSampler", this._thicknessRenderTarget.textureBlur), t.isWebGPU && f.setTextureSampler("thicknessSamplerSampler", (j = (b = this._thicknessRenderTarget.textureBlur) === null || b === void 0 ? void 0 : b.getInternalTexture()) !== null && j !== void 0 ? j : null)) : (f.setTexture("thicknessSampler", this._thicknessRenderTarget.texture), t.isWebGPU && f.setTextureSampler("thicknessSamplerSampler", (q = (T = this._thicknessRenderTarget.texture) === null || T === void 0 ? void 0 : T.getInternalTexture()) !== null && q !== void 0 ? q : null)), f.setFloat("minimumThickness", this.minimumThickness)), this._environmentMap !== null) { const Z = (w = this._environmentMap) !== null && w !== void 0 ? w : this._scene.environmentTexture; Z && (f.setTexture("reflectionSampler", Z), t.isWebGPU && f.setTextureSampler("reflectionSamplerSampler", (m = Z == null ? void 0 : Z.getInternalTexture()) !== null && m !== void 0 ? m : null)); } if (f.setMatrix("viewMatrix", this._scene.getViewMatrix()), f.setMatrix("invProjectionMatrix", this._invProjectionMatrix), f.setMatrix("projectionMatrix", this._scene.getProjectionMatrix()), f.setVector2("texelSize", a), f.setFloat("density", this.density), f.setFloat("refractionStrength", this.refractionStrength), f.setFloat("fresnelClamp", this.fresnelClamp), f.setFloat("specularPower", this.specularPower), f.setVector3("dirLight", this.dirLight), f.setFloat("cameraFar", this._camera.maxZ), this._debug) { let Z = null; switch (this._debugFeature) { case Pc.DepthTexture: Z = this._depthRenderTarget.texture; break; case Pc.DepthBlurredTexture: Z = this._depthRenderTarget.enableBlur ? this._depthRenderTarget.textureBlur : this._depthRenderTarget.texture; break; case Pc.ThicknessTexture: Z = (N = (I = this._thicknessRenderTarget) === null || I === void 0 ? void 0 : I.texture) !== null && N !== void 0 ? N : null; break; case Pc.ThicknessBlurredTexture: Z = !((k = this._thicknessRenderTarget) === null || k === void 0) && k.enableBlur ? (y = (R = this._thicknessRenderTarget) === null || R === void 0 ? void 0 : R.textureBlur) !== null && y !== void 0 ? y : null : (Y = (O = this._thicknessRenderTarget) === null || O === void 0 ? void 0 : O.texture) !== null && Y !== void 0 ? Y : null; break; case Pc.DiffuseTexture: this._diffuseRenderTarget && (Z = this._diffuseRenderTarget.texture); break; } this._debugFeature !== Pc.Normals && (f.setTexture("debugSampler", Z), t.isWebGPU && f.setTextureSampler("debugSamplerSampler", (ee = Z == null ? void 0 : Z.getInternalTexture()) !== null && ee !== void 0 ? ee : null)); } }); } /** @internal */ _clearTargets() { var e, t, r; !((e = this._depthRenderTarget) === null || e === void 0) && e.renderTarget && (this._engine.bindFramebuffer(this._depthRenderTarget.renderTarget), this._engine.clear(this._depthClearColor, !0, !0, !1), this._engine.unBindFramebuffer(this._depthRenderTarget.renderTarget)), !((t = this._diffuseRenderTarget) === null || t === void 0) && t.renderTarget && (this._engine.bindFramebuffer(this._diffuseRenderTarget.renderTarget), this._engine.clear(this._thicknessClearColor, !0, !0, !1), this._engine.unBindFramebuffer(this._diffuseRenderTarget.renderTarget)), !((r = this._thicknessRenderTarget) === null || r === void 0) && r.renderTarget && (this._engine.bindFramebuffer(this._thicknessRenderTarget.renderTarget), this._engine.clear(this._thicknessClearColor, !0, !1, !1), this._engine.unBindFramebuffer(this._thicknessRenderTarget.renderTarget)); } /** @internal */ _render(e) { var t, r, n, i, s, a; if (this._needInitialization || !e.isReady()) return; const f = this._engine._currentRenderTarget; this._engine.setState(!1, void 0, void 0, void 0, !0), this._engine.setDepthBuffer(!0), this._engine.setDepthWrite(!0), this._engine.setAlphaMode(0), !((t = this._depthRenderTarget) === null || t === void 0) && t.renderTarget && (this._engine.bindFramebuffer(this._depthRenderTarget.renderTarget), e.renderDepthTexture(), this._engine.unbindInstanceAttributes(), this._engine.unBindFramebuffer(this._depthRenderTarget.renderTarget)), !((r = this._diffuseRenderTarget) === null || r === void 0) && r.renderTarget && (this._engine.bindFramebuffer(this._diffuseRenderTarget.renderTarget), e.renderDiffuseTexture(), this._engine.unbindInstanceAttributes(), this._engine.unBindFramebuffer(this._diffuseRenderTarget.renderTarget)), !((n = this._thicknessRenderTarget) === null || n === void 0) && n.renderTarget && (this._engine.bindFramebuffer(this._thicknessRenderTarget.renderTarget), e.renderThicknessTexture(), this._engine.unbindInstanceAttributes(), this._engine.unBindFramebuffer(this._thicknessRenderTarget.renderTarget)), (i = this._depthRenderTarget) === null || i === void 0 || i.applyBlurPostProcesses(), (s = this._diffuseRenderTarget) === null || s === void 0 || s.applyBlurPostProcesses(), (a = this._thicknessRenderTarget) === null || a === void 0 || a.applyBlurPostProcesses(), f && this._engine.bindFramebuffer(f); } /** * Releases all the ressources used by the class * @param onlyPostProcesses If true, releases only the ressources used by the render post processes */ dispose(e = !1) { var t, r, n, i; e || ((t = this._depthRenderTarget) === null || t === void 0 || t.dispose(), this._depthRenderTarget = null, (r = this._diffuseRenderTarget) === null || r === void 0 || r.dispose(), this._diffuseRenderTarget = null, (n = this._thicknessRenderTarget) === null || n === void 0 || n.dispose(), this._thicknessRenderTarget = null), this._renderPostProcess && this._camera && this._camera.detachPostProcess(this._renderPostProcess), (i = this._renderPostProcess) === null || i === void 0 || i.dispose(), this._renderPostProcess = null, this._needInitialization = !1; } } class uie extends LY { /** * Gets the name of the class */ getClassName() { return "FluidRenderingObjectCustomParticles"; } /** * Gets the vertex buffers */ get vertexBuffers() { return this._vertexBuffers; } /** * Creates a new instance of the class * @param scene The scene the particles should be rendered into * @param buffers The list of buffers (must contain at least one "position" buffer!). Note that you don't have to pass all (or any!) buffers at once in the constructor, you can use the addBuffers method to add more later. * @param numParticles Number of vertices to take into account from the buffers */ constructor(e, t, r) { super(e), this._numParticles = r, this._diffuseEffectWrapper = null, this._vertexBuffers = {}, this.addBuffers(t); } /** * Add some new buffers * @param buffers List of buffers */ addBuffers(e) { for (const t in e) { let r, n = !0; switch (t) { case "velocity": r = 3; break; case "offset": n = !1; break; } this._vertexBuffers[t] = new J(this._engine, e[t], t, !0, !1, r, n); } } _createEffects() { super._createEffects(); const e = ["view", "projection", "size"], t = ["position", "offset", "color"]; this._diffuseEffectWrapper = new ng({ engine: this._engine, useShaderStore: !0, vertexShader: "fluidRenderingParticleDiffuse", fragmentShader: "fluidRenderingParticleDiffuse", attributeNames: t, uniformNames: e, samplerNames: [] }); } /** * Indicates if the object is ready to be rendered * @returns True if everything is ready for the object to be rendered, otherwise false */ isReady() { var e, t; return this._vertexBuffers.offset || (this._vertexBuffers.offset = new J(this._engine, [0, 0, 1, 0, 0, 1, 1, 1], "offset", !1, !1, 2)), super.isReady() && ((t = (e = this._diffuseEffectWrapper) === null || e === void 0 ? void 0 : e.effect.isReady()) !== null && t !== void 0 ? t : !1); } /** * Gets the number of particles in this object * @returns The number of particles */ get numParticles() { return this._numParticles; } /** * Sets the number of particles in this object * @param num The number of particles to take into account */ setNumParticles(e) { this._numParticles = e; } /** * Render the diffuse texture for this object */ renderDiffuseTexture() { const e = this.numParticles; if (!this._diffuseEffectWrapper || e === 0) return; const t = this._diffuseEffectWrapper._drawWrapper, r = t.effect; this._engine.enableEffect(t), this._engine.bindBuffers(this.vertexBuffers, this.indexBuffer, r), r.setMatrix("view", this._scene.getViewMatrix()), r.setMatrix("projection", this._scene.getProjectionMatrix()), this._particleSize !== null && r.setFloat2("size", this._particleSize, this._particleSize), this.useInstancing ? this._engine.drawArraysType(7, 0, 4, e) : this._engine.drawElementsType(0, 0, e); } /** * Releases the ressources used by the class */ dispose() { var e; super.dispose(), (e = this._diffuseEffectWrapper) === null || e === void 0 || e.dispose(); for (const t in this._vertexBuffers) this._vertexBuffers[t].dispose(); this._vertexBuffers = {}; } } const bHe = "copyTextureToTexturePixelShader", xHe = `uniform float conversion;uniform sampler2D textureSampler;varying vec2 vUV; #include void main(void) {vec4 color=texture2D(textureSampler,vUV); #ifdef DEPTH_TEXTURE gl_FragDepth=color.r; #else if (conversion==1.) {color=toLinearSpace(color);} else if (conversion==2.) {color=toGammaSpace(color);} gl_FragColor=color; #endif } `; Le.ShadersStore[bHe] = xHe; var dO; (function(A) { A[A.None = 0] = "None", A[A.ToLinearSpace = 1] = "ToLinearSpace", A[A.ToGammaSpace = 2] = "ToGammaSpace"; })(dO || (dO = {})); class lie { _textureIsInternal(e) { return e.getInternalTexture === void 0; } /** * Constructs a new instance of the class * @param engine The engine to use for the copy * @param isDepthTexture True means that we should write (using gl_FragDepth) into the depth texture attached to the destination (default: false) */ constructor(e, t = !1) { this._engine = e, this._isDepthTexture = t, this._renderer = new fU(e), this._effectWrapper = new ng({ engine: e, name: "CopyTextureToTexture", fragmentShader: "copyTextureToTexture", useShaderStore: !0, uniformNames: ["conversion"], samplerNames: ["textureSampler"], defines: t ? ["#define DEPTH_TEXTURE"] : [] }), this._effectWrapper.onApplyObservable.add(() => { t && (e.setState(!1), e.setDepthBuffer(!0), e.depthCullingState.depthMask = !0, e.depthCullingState.depthFunc = 519), this._textureIsInternal(this._source) ? this._effectWrapper.effect._bindTexture("textureSampler", this._source) : this._effectWrapper.effect.setTexture("textureSampler", this._source), this._effectWrapper.effect.setFloat("conversion", this._conversion); }); } /** * Indicates if the effect is ready to be used for the copy * @returns true if "copy" can be called without delay, else false */ isReady() { return this._effectWrapper.effect.isReady(); } /** * Copy one texture into another * @param source The source texture * @param destination The destination texture * @param conversion The conversion mode that should be applied when copying * @returns */ copy(e, t, r = dO.None) { if (!this.isReady()) return !1; this._source = e, this._conversion = r; const n = this._engine.depthCullingState.depthFunc; return this._renderer.render(this._effectWrapper, t), this._isDepthTexture && n && (this._engine.depthCullingState.depthFunc = n), !0; } /** * Releases all the resources used by the class */ dispose() { this._effectWrapper.dispose(), this._renderer.dispose(); } } class DHe { get depthRTWrapper() { return this._depthRTWrapper; } constructor(e, t, r, n = 1) { this._engine = e, this._copyTextureToTexture = new lie(e, !0), this._depthRTWrapper = this._engine.createRenderTargetTexture({ width: t, height: r }, { generateMipMaps: !1, type: 0, format: 6, samplingMode: 1, generateDepthBuffer: !0, generateStencilBuffer: !1, samples: n, noColorAttachment: !0, label: "FluidRenderingDepthTextureCopyRTT" }); const i = this._depthRTWrapper.createDepthStencilTexture(0, !1, !1, 1, void 0, "FluidRenderingDepthTextureCopyRTTDepthStencil"); i.label = `FluidDepthTextureCopy${t}x${r}x${n}`; } copy(e) { return this._copyTextureToTexture.copy(e, this._depthRTWrapper); } dispose() { this._depthRTWrapper.dispose(), this._copyTextureToTexture.dispose(); } } const jHe = "fluidRenderingParticleDepthVertexShader", wHe = `attribute vec3 position;attribute vec2 offset;uniform mat4 view;uniform mat4 projection;uniform vec2 size;varying vec2 uv;varying vec3 viewPos;varying float sphereRadius; #ifdef FLUIDRENDERING_VELOCITY attribute vec3 velocity;varying float velocityNorm; #endif void main(void) {vec3 cornerPos;cornerPos.xy=vec2(offset.x-0.5,offset.y-0.5)*size;cornerPos.z=0.0;viewPos=(view*vec4(position,1.0)).xyz;gl_Position=projection*vec4(viewPos+cornerPos,1.0);uv=offset;sphereRadius=size.x/2.0; #ifdef FLUIDRENDERING_VELOCITY velocityNorm=length(velocity); #endif } `; Le.ShadersStore[jHe] = wHe; const mHe = "fluidRenderingParticleDepthPixelShader", BHe = `uniform mat4 projection;varying vec2 uv;varying vec3 viewPos;varying float sphereRadius; #ifdef FLUIDRENDERING_VELOCITY varying float velocityNorm; #endif void main(void) {vec3 normal;normal.xy=uv*2.0-1.0;float r2=dot(normal.xy,normal.xy);if (r2>1.0) discard;normal.z=sqrt(1.0-r2); #ifndef FLUIDRENDERING_RHS normal.z=-normal.z; #endif vec4 realViewPos=vec4(viewPos+normal*sphereRadius,1.0);vec4 clipSpacePos=projection*realViewPos; #ifdef WEBGPU gl_FragDepth=clipSpacePos.z/clipSpacePos.w; #else gl_FragDepth=(clipSpacePos.z/clipSpacePos.w)*0.5+0.5; #endif #ifdef FLUIDRENDERING_RHS realViewPos.z=-realViewPos.z; #endif #ifdef FLUIDRENDERING_VELOCITY glFragColor=vec4(realViewPos.z,velocityNorm,0.,1.); #else glFragColor=vec4(realViewPos.z,0.,0.,1.); #endif } `; Le.ShadersStore[mHe] = BHe; const WHe = "fluidRenderingParticleThicknessVertexShader", SHe = `attribute vec3 position;attribute vec2 offset;uniform mat4 view;uniform mat4 projection;uniform vec2 size;varying vec2 uv;void main(void) {vec3 cornerPos;cornerPos.xy=vec2(offset.x-0.5,offset.y-0.5)*size;cornerPos.z=0.0;vec3 viewPos=(view*vec4(position,1.0)).xyz+cornerPos;gl_Position=projection*vec4(viewPos,1.0);uv=offset;} `; Le.ShadersStore[WHe] = SHe; const UHe = "fluidRenderingParticleThicknessPixelShader", IHe = `uniform float particleAlpha;varying vec2 uv;void main(void) {vec3 normal;normal.xy=uv*2.0-1.0;float r2=dot(normal.xy,normal.xy);if (r2>1.0) discard;float thickness=sqrt(1.0-r2);glFragColor=vec4(vec3(particleAlpha*thickness),1.0);} `; Le.ShadersStore[UHe] = IHe; const RHe = "fluidRenderingParticleDiffuseVertexShader", VHe = `attribute vec3 position;attribute vec2 offset;attribute vec4 color;uniform mat4 view;uniform mat4 projection;uniform vec2 size;varying vec2 uv;varying vec3 diffuseColor;void main(void) {vec3 cornerPos;cornerPos.xy=vec2(offset.x-0.5,offset.y-0.5)*size;cornerPos.z=0.0;vec3 viewPos=(view*vec4(position,1.0)).xyz+cornerPos;gl_Position=projection*vec4(viewPos,1.0);uv=offset;diffuseColor=color.rgb;} `; Le.ShadersStore[RHe] = VHe; const CHe = "fluidRenderingParticleDiffusePixelShader", OHe = `uniform float particleAlpha;varying vec2 uv;varying vec3 diffuseColor;void main(void) {vec3 normal;normal.xy=uv*2.0-1.0;float r2=dot(normal.xy,normal.xy);if (r2>1.0) discard;glFragColor=vec4(diffuseColor,1.0);} `; Le.ShadersStore[CHe] = OHe; const yHe = "fluidRenderingBilateralBlurPixelShader", kHe = `uniform sampler2D textureSampler;uniform int maxFilterSize;uniform vec2 blurDir;uniform float projectedParticleConstant;uniform float depthThreshold;varying vec2 vUV;void main(void) {float depth=textureLod(textureSampler,vUV,0.).x;if (depth>=1e6 || depth<=0.) {glFragColor=vec4(vec3(depth),1.);return;} int filterSize=min(maxFilterSize,int(ceil(projectedParticleConstant/depth)));float sigma=float(filterSize)/3.0;float two_sigma2=2.0*sigma*sigma;float sigmaDepth=depthThreshold/3.0;float two_sigmaDepth2=2.0*sigmaDepth*sigmaDepth;float sum=0.;float wsum=0.;float sumVel=0.;for (int x=-filterSize; x<=filterSize; ++x) {vec2 coords=vec2(x);vec2 sampleDepthVel=textureLod(textureSampler,vUV+coords*blurDir,0.).rg;float r=dot(coords,coords);float w=exp(-r/two_sigma2);float rDepth=sampleDepthVel.r-depth;float wd=exp(-rDepth*rDepth/two_sigmaDepth2);sum+=sampleDepthVel.r*w*wd;sumVel+=sampleDepthVel.g*w*wd;wsum+=w*wd;} glFragColor=vec4(sum/wsum,sumVel/wsum,0.,1.);} `; Le.ShadersStore[yHe] = kHe; const EHe = "fluidRenderingStandardBlurPixelShader", FHe = `uniform sampler2D textureSampler;uniform int filterSize;uniform vec2 blurDir;varying vec2 vUV;void main(void) {vec4 s=textureLod(textureSampler,vUV,0.);if (s.r==0.) {glFragColor=vec4(0.,0.,0.,1.);return;} float sigma=float(filterSize)/3.0;float twoSigma2=2.0*sigma*sigma;vec4 sum=vec4(0.);float wsum=0.;for (int x=-filterSize; x<=filterSize; ++x) {vec2 coords=vec2(x);vec4 sampl=textureLod(textureSampler,vUV+coords*blurDir,0.);float w=exp(-coords.x*coords.x/twoSigma2);sum+=sampl*w;wsum+=w;} sum/=wsum;glFragColor=vec4(sum.rgb,1.);} `; Le.ShadersStore[EHe] = FHe; const NHe = "fluidRenderingRenderPixelShader", QHe = `/* disable_uniformity_analysis */ #define IOR 1.333 #define ETA 1.0/IOR #define F0 0.02 uniform sampler2D textureSampler;uniform sampler2D depthSampler; #ifdef FLUIDRENDERING_DIFFUSETEXTURE uniform sampler2D diffuseSampler; #else uniform vec3 diffuseColor; #endif #ifdef FLUIDRENDERING_FIXED_THICKNESS uniform float thickness;uniform sampler2D bgDepthSampler; #else uniform float minimumThickness;uniform sampler2D thicknessSampler; #endif #ifdef FLUIDRENDERING_ENVIRONMENT uniform samplerCube reflectionSampler; #endif #if defined(FLUIDRENDERING_DEBUG) && defined(FLUIDRENDERING_DEBUG_TEXTURE) uniform sampler2D debugSampler; #endif uniform mat4 viewMatrix;uniform mat4 projectionMatrix;uniform mat4 invProjectionMatrix;uniform vec2 texelSize;uniform vec3 dirLight;uniform float cameraFar;uniform float density;uniform float refractionStrength;uniform float fresnelClamp;uniform float specularPower;varying vec2 vUV;vec3 computeViewPosFromUVDepth(vec2 texCoord,float depth) {vec4 ndc;ndc.xy=texCoord*2.0-1.0; #ifdef FLUIDRENDERING_RHS ndc.z=-projectionMatrix[2].z+projectionMatrix[3].z/depth; #else ndc.z=projectionMatrix[2].z+projectionMatrix[3].z/depth; #endif ndc.w=1.0;vec4 eyePos=invProjectionMatrix*ndc;eyePos.xyz/=eyePos.w;return eyePos.xyz;} vec3 getViewPosFromTexCoord(vec2 texCoord) {float depth=textureLod(depthSampler,texCoord,0.).x;return computeViewPosFromUVDepth(texCoord,depth);} void main(void) {vec2 texCoord=vUV; #if defined(FLUIDRENDERING_DEBUG) && defined(FLUIDRENDERING_DEBUG_TEXTURE) vec4 color=texture2D(debugSampler,texCoord); #ifdef FLUIDRENDERING_DEBUG_DEPTH glFragColor=vec4(color.rgb/vec3(2.0),1.);if (color.r>0.999 && color.g>0.999) {glFragColor=texture2D(textureSampler,texCoord);} #else glFragColor=vec4(color.rgb,1.);if (color.r<0.001 && color.g<0.001 && color.b<0.001) {glFragColor=texture2D(textureSampler,texCoord);} #endif return; #endif vec2 depthVel=textureLod(depthSampler,texCoord,0.).rg;float depth=depthVel.r; #ifndef FLUIDRENDERING_FIXED_THICKNESS float thickness=texture2D(thicknessSampler,texCoord).x; #else float bgDepth=texture2D(bgDepthSampler,texCoord).x;float depthNonLinear=projectionMatrix[2].z+projectionMatrix[3].z/depth;depthNonLinear=depthNonLinear*0.5+0.5; #endif vec4 backColor=texture2D(textureSampler,texCoord); #ifndef FLUIDRENDERING_FIXED_THICKNESS if (depth>=cameraFar || depth<=0. || thickness<=minimumThickness) { #else if (depth>=cameraFar || depth<=0. || bgDepth<=depthNonLinear) { #endif glFragColor=backColor;return;} vec3 viewPos=computeViewPosFromUVDepth(texCoord,depth);vec3 ddx=getViewPosFromTexCoord(texCoord+vec2(texelSize.x,0.))-viewPos;vec3 ddy=getViewPosFromTexCoord(texCoord+vec2(0.,texelSize.y))-viewPos;vec3 ddx2=viewPos-getViewPosFromTexCoord(texCoord+vec2(-texelSize.x,0.));if (abs(ddx.z)>abs(ddx2.z)) {ddx=ddx2;} vec3 ddy2=viewPos-getViewPosFromTexCoord(texCoord+vec2(0.,-texelSize.y));if (abs(ddy.z)>abs(ddy2.z)) {ddy=ddy2;} vec3 normal=normalize(cross(ddy,ddx)); #ifdef FLUIDRENDERING_RHS normal=-normal; #endif #ifndef WEBGPU if(isnan(normal.x) || isnan(normal.y) || isnan(normal.z) || isinf(normal.x) || isinf(normal.y) || isinf(normal.z)) {normal=vec3(0.,0.,-1.);} #endif #if defined(FLUIDRENDERING_DEBUG) && defined(FLUIDRENDERING_DEBUG_SHOWNORMAL) glFragColor=vec4(normal*0.5+0.5,1.0);return; #endif vec3 rayDir=normalize(viewPos); #ifdef FLUIDRENDERING_DIFFUSETEXTURE vec3 diffuseColor=textureLod(diffuseSampler,texCoord,0.0).rgb; #endif vec3 lightDir=normalize(vec3(viewMatrix*vec4(-dirLight,0.)));vec3 H =normalize(lightDir-rayDir);float specular=pow(max(0.0,dot(H,normal)),specularPower); #ifdef FLUIDRENDERING_DEBUG_DIFFUSERENDERING float diffuse =max(0.0,dot(lightDir,normal))*1.0;glFragColor=vec4(vec3(0.1) /*ambient*/+vec3(0.42,0.50,1.00)*diffuse+vec3(0,0,0.2)+specular,1.);return; #endif vec3 refractionDir=refract(rayDir,normal,ETA);vec4 transmitted=textureLod(textureSampler,vec2(texCoord+refractionDir.xy*thickness*refractionStrength),0.0);vec3 transmittance=exp(-density*thickness*(1.0-diffuseColor)); vec3 refractionColor=transmitted.rgb*transmittance; #ifdef FLUIDRENDERING_ENVIRONMENT vec3 reflectionDir=reflect(rayDir,normal);vec3 reflectionColor=(textureCube(reflectionSampler,reflectionDir).rgb);float fresnel=clamp(F0+(1.0-F0)*pow(1.0-dot(normal,-rayDir),5.0),0.,fresnelClamp);vec3 finalColor=mix(refractionColor,reflectionColor,fresnel)+specular; #else vec3 finalColor=refractionColor+specular; #endif #ifdef FLUIDRENDERING_VELOCITY float velocity=depthVel.g;finalColor=mix(finalColor,vec3(1.0),smoothstep(0.3,1.0,velocity/6.0)); #endif glFragColor=vec4(finalColor,transmitted.a);} `; Le.ShadersStore[NHe] = QHe; Object.defineProperty(sr.prototype, "fluidRenderer", { get: function() { return this._fluidRenderer; }, set: function(A) { this._fluidRenderer = A; }, enumerable: !0, configurable: !0 }); sr.prototype.enableFluidRenderer = function() { return this._fluidRenderer ? this._fluidRenderer : (this._fluidRenderer = new Iy(this), this._fluidRenderer); }; sr.prototype.disableFluidRenderer = function() { var A; (A = this._fluidRenderer) === null || A === void 0 || A.dispose(), this._fluidRenderer = null; }; function YHe(A) { return !!A.particleSystem; } class Pie { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_FLUIDRENDERER, this.scene = e; } /** * Registers the component in a given scene */ register() { this.scene._gatherActiveCameraRenderTargetsStage.registerStep(Ot.STEP_GATHERACTIVECAMERARENDERTARGETS_FLUIDRENDERER, this, this._gatherActiveCameraRenderTargets), this.scene._afterCameraDrawStage.registerStep(Ot.STEP_AFTERCAMERADRAW_FLUIDRENDERER, this, this._afterCameraDraw); } _gatherActiveCameraRenderTargets(e) { var t; (t = this.scene.fluidRenderer) === null || t === void 0 || t._prepareRendering(); } _afterCameraDraw(e) { var t; (t = this.scene.fluidRenderer) === null || t === void 0 || t._render(e); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { this.scene._fluidRenderer && (this.scene.disableFluidRenderer(), this.scene.enableFluidRenderer()); } /** * Disposes the component and the associated resources */ dispose() { this.scene.disableFluidRenderer(); } } class Iy { /** @internal */ static _SceneComponentInitialization(e) { let t = e._getComponent(Ot.NAME_FLUIDRENDERER); t || (t = new Pie(e), e._addComponent(t)); } /** * Initializes the class * @param scene Scene in which the objects are part of */ constructor(e) { this._scene = e, this._engine = e.getEngine(), this._onEngineResizeObserver = null, this.renderObjects = [], this.targetRenderers = [], this._cameras = /* @__PURE__ */ new Map(), Iy._SceneComponentInitialization(this._scene), this._onEngineResizeObserver = this._engine.onResizeObservable.add(() => { this._initialize(); }); } /** * Reinitializes the class * Can be used if you change the object priority (FluidRenderingObject.priority), to make sure the objects are rendered in the right order */ recreate() { this._sortRenderingObjects(), this._initialize(); } /** * Gets the render object corresponding to a particle system (null if the particle system is not rendered as a fluid) * @param ps The particle system * @returns the render object corresponding to this particle system if any, otherwise null */ getRenderObjectFromParticleSystem(e) { const t = this._getParticleSystemIndex(e); return t !== -1 ? this.renderObjects[t] : null; } /** * Adds a particle system to the fluid renderer. * @param ps particle system * @param generateDiffuseTexture True if you want to generate a diffuse texture from the particle system and use it as part of the fluid rendering (default: false) * @param targetRenderer The target renderer used to display the particle system as a fluid. If not provided, the method will create a new one * @param camera The camera used by the target renderer (if the target renderer is created by the method) * @returns the render object corresponding to the particle system */ addParticleSystem(e, t, r, n) { const i = new vie(this._scene, e); i.onParticleSizeChanged.add(() => this._setParticleSizeForRenderTargets()), r || (r = new wF(this._scene, n), this.targetRenderers.push(r)), r._onUseVelocityChanged.hasObservers() || r._onUseVelocityChanged.add(() => this._setUseVelocityForRenderObject()), t !== void 0 && (r.generateDiffuseTexture = t); const s = { object: i, targetRenderer: r }; return this.renderObjects.push(s), this._sortRenderingObjects(), this._setParticleSizeForRenderTargets(), s; } /** * Adds a custom particle set to the fluid renderer. * @param buffers The list of buffers (should contain at least a "position" buffer!) * @param numParticles Number of particles in each buffer * @param generateDiffuseTexture True if you want to generate a diffuse texture from buffers and use it as part of the fluid rendering (default: false). For the texture to be generated correctly, you need a "color" buffer in the set! * @param targetRenderer The target renderer used to display the particle system as a fluid. If not provided, the method will create a new one * @param camera The camera used by the target renderer (if the target renderer is created by the method) * @returns the render object corresponding to the custom particle set */ addCustomParticles(e, t, r, n, i) { const s = new uie(this._scene, e, t); s.onParticleSizeChanged.add(() => this._setParticleSizeForRenderTargets()), n || (n = new wF(this._scene, i), this.targetRenderers.push(n)), n._onUseVelocityChanged.hasObservers() || n._onUseVelocityChanged.add(() => this._setUseVelocityForRenderObject()), r !== void 0 && (n.generateDiffuseTexture = r); const a = { object: s, targetRenderer: n }; return this.renderObjects.push(a), this._sortRenderingObjects(), this._setParticleSizeForRenderTargets(), a; } /** * Removes a render object from the fluid renderer * @param renderObject the render object to remove * @param removeUnusedTargetRenderer True to remove/dispose of the target renderer if it's not used anymore (default: true) * @returns True if the render object has been found and released, else false */ removeRenderObject(e, t = !0) { const r = this.renderObjects.indexOf(e); return r === -1 ? !1 : (e.object.dispose(), this.renderObjects.splice(r, 1), t && this._removeUnusedTargetRenderers() ? this._initialize() : this._setParticleSizeForRenderTargets(), !0); } _sortRenderingObjects() { this.renderObjects.sort((e, t) => e.object.priority < t.object.priority ? -1 : e.object.priority > t.object.priority ? 1 : 0); } _removeUnusedTargetRenderers() { const e = {}; for (let n = 0; n < this.renderObjects.length; ++n) { const i = this.renderObjects[n].targetRenderer; e[this.targetRenderers.indexOf(i)] = !0; } let t = !1; const r = []; for (let n = 0; n < this.targetRenderers.length; ++n) e[n] ? r.push(this.targetRenderers[n]) : (this.targetRenderers[n].dispose(), t = !0); return t && (this.targetRenderers.length = 0, this.targetRenderers.push(...r)), t; } _getParticleSystemIndex(e) { for (let t = 0; t < this.renderObjects.length; ++t) { const r = this.renderObjects[t].object; if (YHe(r) && r.particleSystem === e) return t; } return -1; } _initialize() { for (let r = 0; r < this.targetRenderers.length; ++r) this.targetRenderers[r].dispose(); const e = /* @__PURE__ */ new Map(); for (let r = 0; r < this.targetRenderers.length; ++r) { const n = this.targetRenderers[r]; if (n._initialize(), n.camera && n._renderPostProcess) { let i = e.get(n.camera); i || (i = [[], {}], e.set(n.camera, i)), i[0].push(n), n.camera.attachPostProcess(n._renderPostProcess, r); } } let t = e.keys(); for (let r = t.next(); r.done !== !0; r = t.next()) { const n = r.value, i = e.get(n), s = n._getFirstPostProcess(); if (!s) continue; const [a, f] = i; s.onSizeChangedObservable.add(() => { var o; s.inputTexture.depthStencilTexture || s.inputTexture.createDepthStencilTexture(0, !0, this._engine.isStencilEnable, a[0].samples, this._engine.isStencilEnable ? 13 : 14, `PostProcessRTTDepthStencil-${s.name}`); for (const d of a) { const v = (o = d._thicknessRenderTarget) === null || o === void 0 ? void 0 : o.renderTarget, u = v == null ? void 0 : v.texture; if (v && u) { const l = u.width + "_" + u.height; let P = f[l]; P || (P = f[l] = new DHe(this._engine, u.width, u.height)), P.depthRTWrapper._shareDepth(v); } } }); } t = this._cameras.keys(); for (let r = t.next(); r.done !== !0; r = t.next()) { const n = r.value, s = this._cameras.get(n)[1], a = e.get(n); if (a) for (const f in s) a[1][f] || s[f].dispose(); else for (const f in s) s[f].dispose(); } this._cameras.clear(), this._cameras = e, this._setParticleSizeForRenderTargets(); } _setParticleSizeForRenderTargets() { const e = /* @__PURE__ */ new Map(); for (let t = 0; t < this.renderObjects.length; ++t) { const r = this.renderObjects[t]; let n = e.get(r.targetRenderer); n === void 0 && (n = 0), e.set(r.targetRenderer, Math.max(n, r.object.particleSize)); } e.forEach((t, r) => { r._depthRenderTarget && (r._depthRenderTarget.particleSize = t); }); } _setUseVelocityForRenderObject() { for (const e of this.renderObjects) e.object.useVelocity = e.targetRenderer.useVelocity; } /** @internal */ _prepareRendering() { for (const e of this.targetRenderers) if (e.needInitialization) { this._initialize(); return; } } /** @internal */ _render(e) { var t; for (let n = 0; n < this.targetRenderers.length; ++n) (!e || this.targetRenderers[n].camera === e) && this.targetRenderers[n]._clearTargets(); const r = this._cameras.keys(); for (let n = r.next(); n.done !== !0; n = r.next()) { const i = n.value, s = this._cameras.get(i); if (e && i !== e) continue; const a = i._getFirstPostProcess(); if (!a) continue; const f = (t = a.inputTexture) === null || t === void 0 ? void 0 : t.depthStencilTexture; if (f) { const [o, d] = s; for (const v of o) v._bgDepthTexture = f; for (const v in d) d[v].copy(f); } } for (let n = 0; n < this.renderObjects.length; ++n) { const i = this.renderObjects[n]; (!e || i.targetRenderer.camera === e) && i.targetRenderer._render(i.object); } } /** * Disposes of all the ressources used by the class */ dispose() { this._engine.onResizeObservable.remove(this._onEngineResizeObserver), this._onEngineResizeObserver = null; for (let e = 0; e < this.renderObjects.length; ++e) this.renderObjects[e].object.dispose(); for (let e = 0; e < this.targetRenderers.length; ++e) this.targetRenderers[e].dispose(); this._cameras.forEach((e) => { const t = e[1]; for (const r in t) t[r].dispose(); }), this.renderObjects = [], this.targetRenderers = [], this._cameras.clear(); } } class Fd { /** * Return the number of splattings used */ get vertexCount() { return this._vertexCount; } /** * Shader material with alpha blending * @param scene parent scene */ _createMaterial(e) { An.ShadersStore.customVertexShader = Fd._VertexShaderSource, An.ShadersStore.customFragmentShader = Fd._FragmentShaderSource; const t = new Zo("GaussianSplattingShader", e, { vertex: "custom", fragment: "custom" }, { attributes: ["position"], uniforms: ["projection", "modelView"] }); t.backFaceCulling = !1, t.alpha = 0.9999, this._material = t; } /** * * @param scene parent scene * @returns A simple 2 triangles quad */ _getMesh(e) { const t = new Ee(this.name, e), r = new Ut(); return r.positions = [-2, -2, 0, 2, -2, 0, 2, 2, 0, -2, 2, 0], r.indices = [0, 1, 2, 0, 2, 3], r.applyToMesh(t), t.material = this._material, t.alwaysSelectAsActiveMesh = !0, t; } _setData(e) { this._vertexCount = e.length / 32; const r = this._vertexCount; this._positions = new Float32Array(3 * r), this._covA = new Float32Array(3 * r), this._covB = new Float32Array(3 * r); const n = new Float32Array(e.buffer); this._uBuffer = new Uint8Array(e.buffer); const i = he.Zero(), s = he.Zero(), a = Ze.Identity(); for (let f = 0; f < r; f++) { this._positions[3 * f + 0] = n[8 * f + 0], this._positions[3 * f + 1] = -n[8 * f + 1], this._positions[3 * f + 2] = n[8 * f + 2], a.set((this._uBuffer[32 * f + 28 + 1] - 128) / 128, (this._uBuffer[32 * f + 28 + 2] - 128) / 128, (this._uBuffer[32 * f + 28 + 3] - 128) / 128, -(this._uBuffer[32 * f + 28 + 0] - 128) / 128), a.toRotationMatrix(i), he.ScalingToRef(n[8 * f + 3 + 0] * 2, n[8 * f + 3 + 1] * 2, n[8 * f + 3 + 2] * 2, s); const o = i.multiplyToRef(s, ue.Matrix[0]).m; this._covA[f * 3 + 0] = o[0] * o[0] + o[1] * o[1] + o[2] * o[2], this._covA[f * 3 + 1] = o[0] * o[4] + o[1] * o[5] + o[2] * o[6], this._covA[f * 3 + 2] = o[0] * o[8] + o[1] * o[9] + o[2] * o[10], this._covB[f * 3 + 0] = o[4] * o[4] + o[5] * o[5] + o[6] * o[6], this._covB[f * 3 + 1] = o[4] * o[8] + o[5] * o[9] + o[6] * o[10], this._covB[f * 3 + 2] = o[8] * o[8] + o[9] * o[9] + o[10] * o[10]; } } /** * Construct a Gaussian Splatting proxy object * @param name name of the mesh used for rendering * @param scene scene it belongs to */ constructor(e, t) { var r; this._vertexCount = 0, this._modelViewMatrix = he.Identity(), this.scene = t, this.name = e, this._createMaterial(t), (r = Fd._Worker) === null || r === void 0 || r.terminate(), Fd._Worker = null; } /** * Loads a .splat Gaussian Splatting file asynchronously * @param url path to the splat file to load * @returns a promise that resolves when the operation is complete */ loadAsync(e) { return ye.LoadFileAsync(e, !0).then((t) => { this.mesh && this.dispose(), this._setData(new Uint8Array(t)); const r = new Float32Array(this.vertexCount * 16), n = (i) => { var s; for (let a = 0; a < this.vertexCount; a++) { const f = i[2 * a], o = a * 16; r[o + 0] = this._positions[f * 3 + 0], r[o + 1] = this._positions[f * 3 + 1], r[o + 2] = this._positions[f * 3 + 2], r[o + 4] = this._uBuffer[32 * f + 24 + 0] / 255, r[o + 5] = this._uBuffer[32 * f + 24 + 1] / 255, r[o + 6] = this._uBuffer[32 * f + 24 + 2] / 255, r[o + 7] = this._uBuffer[32 * f + 24 + 3] / 255, r[o + 8] = this._covA[f * 3 + 0], r[o + 9] = this._covA[f * 3 + 1], r[o + 10] = this._covA[f * 3 + 2], r[o + 12] = this._covB[f * 3 + 0], r[o + 13] = this._covB[f * 3 + 1], r[o + 14] = this._covB[f * 3 + 2]; } (s = this.mesh) === null || s === void 0 || s.thinInstanceBufferUpdated("matrix"); }; this.mesh = this._getMesh(this.scene), this.mesh.thinInstanceSetBuffer("matrix", r, 16, !1), Fd._Worker && (console.warn("Only one web worker possible. Previous Gaussian Splatting instance might not be rendered correctly."), Fd._Worker.terminate()), Fd._Worker = new Worker(URL.createObjectURL(new Blob(["(", Fd._CreateWorker.toString(), ")(self)"], { type: "application/javascript" }))), Fd._Worker.onmessage = (i) => { const s = new Uint32Array(i.data.depthMix.buffer); n(s); }, this._sceneBeforeRenderObserver = this.scene.onBeforeRenderObservable.add(() => { var i; const s = this.scene.getEngine(); this._material.setVector2("viewport", new at(s.getRenderWidth(), s.getRenderHeight())), this.mesh.getWorldMatrix().multiplyToRef(this.scene.activeCamera.getViewMatrix(), this._modelViewMatrix), this._material.setMatrix("modelView", this._modelViewMatrix), (i = Fd._Worker) === null || i === void 0 || i.postMessage({ view: this._modelViewMatrix.m, positions: this._positions }); }), this._sceneDisposeObserver = this.scene.onDisposeObservable.add(() => { this.dispose(); }); }); } /** * Clear datas used for Gaussian Splatting and associated resources */ dispose() { var e, t; this.scene.onDisposeObservable.remove(this._sceneDisposeObserver), this.scene.onBeforeRenderObservable.remove(this._sceneBeforeRenderObserver), (e = Fd._Worker) === null || e === void 0 || e.terminate(), Fd._Worker = null, (t = this.mesh) === null || t === void 0 || t.dispose(), this.mesh = null; } } Fd._Worker = null; Fd._VertexShaderSource = ` precision mediump float; attribute vec2 position; attribute vec4 world0; attribute vec4 world1; attribute vec4 world2; attribute vec4 world3; uniform mat4 projection, modelView; uniform vec2 viewport; varying vec4 vColor; varying vec2 vPosition; void main () { vec3 center = world0.xyz; vec4 color = world1; vec3 covA = world2.xyz; vec3 covB = world3.xyz; vec4 camspace = modelView * vec4(center, 1); vec4 pos2d = projection * camspace; float bounds = 1.2 * pos2d.w; if (pos2d.z < -pos2d.w || pos2d.x < -bounds || pos2d.x > bounds || pos2d.y < -bounds || pos2d.y > bounds) { gl_Position = vec4(0.0, 0.0, 2.0, 1.0); return; } mat3 Vrk = mat3( covA.x, covA.y, covA.z, covA.y, covB.x, covB.y, covA.z, covB.y, covB.z ); vec2 focal = vec2(1132., 1132.); mat3 J = mat3( focal.x / camspace.z, 0., -(focal.x * camspace.x) / (camspace.z * camspace.z), 0., focal.y / camspace.z, -(focal.y * camspace.y) / (camspace.z * camspace.z), 0., 0., 0. ); mat3 invy = mat3(1,0,0, 0,-1,0,0,0,1); mat3 T = invy * transpose(mat3(modelView)) * J; mat3 cov2d = transpose(T) * Vrk * T; float mid = (cov2d[0][0] + cov2d[1][1]) / 2.0; float radius = length(vec2((cov2d[0][0] - cov2d[1][1]) / 2.0, cov2d[0][1])); float lambda1 = mid + radius, lambda2 = mid - radius; if(lambda2 < 0.0) return; vec2 diagonalVector = normalize(vec2(cov2d[0][1], lambda1 - cov2d[0][0])); vec2 majorAxis = min(sqrt(2.0 * lambda1), 1024.0) * diagonalVector; vec2 minorAxis = min(sqrt(2.0 * lambda2), 1024.0) * vec2(diagonalVector.y, -diagonalVector.x); vColor = color; vPosition = position; vec2 vCenter = vec2(pos2d); gl_Position = vec4( vCenter + (position.x * majorAxis * 1. / viewport + position.y * minorAxis * 1. / viewport) * pos2d.w, pos2d.zw); }`; Fd._FragmentShaderSource = ` precision highp float; varying vec4 vColor; varying vec2 vPosition; void main () { float A = -dot(vPosition, vPosition); if (A < -4.0) discard; float B = exp(A) * vColor.a; gl_FragColor = vec4(vColor.rgb, B); }`; Fd._CreateWorker = function(A) { let e, t = [], r = 0, n; const i = (f) => { r = n.length; const o = new BigInt64Array(r), d = new Uint32Array(o.buffer); for (let u = 0; u < r; u++) d[2 * u] = u; const v = new Float32Array(o.buffer); for (let u = 0; u < r; u++) v[2 * u + 1] = 1e4 - (f[2] * n[3 * u + 0] + f[6] * n[3 * u + 1] + f[10] * n[3 * u + 2]); t = f, o.sort(), A.postMessage({ depthMix: o }, [o.buffer]); }; let s = !1; const a = () => { if (!s) { s = !0; const f = e; i(f), setTimeout(() => { s = !1, f !== e && a(); }, 0); } }; A.onmessage = (f) => { e = f.data.view; const o = t[2] * e[2] + t[6] * e[6] + t[10] * e[10]; Math.abs(o - 1) < 0.01 || (n = f.data.positions, a()); }; }; class MHe { /** * Returns a boolean indicating if the animation is started */ get animationStarted() { return this._animationStarted; } /** Gets the initial key for the animation (setting it will restart the animation) */ get fromIndex() { return this._fromIndex; } /** Gets or sets the end key for the animation (setting it will restart the animation) */ get toIndex() { return this._toIndex; } /** Gets or sets a boolean indicating if the animation is looping (setting it will restart the animation) */ get loopAnimation() { return this._loopAnimation; } /** Gets or sets the delay between cell changes (setting it will restart the animation) */ get delay() { return Math.max(this._delay, 1); } /** * Creates a new Thin Sprite */ constructor() { this.width = 1, this.height = 1, this.angle = 0, this.invertU = !1, this.invertV = !1, this.isVisible = !0, this._animationStarted = !1, this._loopAnimation = !1, this._fromIndex = 0, this._toIndex = 0, this._delay = 0, this._direction = 1, this._time = 0, this._onBaseAnimationEnd = null, this.position = { x: 1, y: 1, z: 1 }, this.color = { r: 1, g: 1, b: 1, a: 1 }; } /** * Starts an animation * @param from defines the initial key * @param to defines the end key * @param loop defines if the animation must loop * @param delay defines the start delay (in ms) * @param onAnimationEnd defines a callback for when the animation ends */ playAnimation(e, t, r, n, i) { this._fromIndex = e, this._toIndex = t, this._loopAnimation = r, this._delay = n || 1, this._animationStarted = !0, this._onBaseAnimationEnd = i, e < t ? this._direction = 1 : (this._direction = -1, this._toIndex = e, this._fromIndex = t), this.cellIndex = e, this._time = 0; } /** Stops current animation (if any) */ stopAnimation() { this._animationStarted = !1; } /** * @internal */ _animate(e) { this._animationStarted && (this._time += e, this._time > this._delay && (this._time = this._time % this._delay, this.cellIndex += this._direction, (this._direction > 0 && this.cellIndex > this._toIndex || this._direction < 0 && this.cellIndex < this._fromIndex) && (this._loopAnimation ? this.cellIndex = this._direction > 0 ? this._fromIndex : this._toIndex : (this.cellIndex = this._toIndex, this._animationStarted = !1, this._onBaseAnimationEnd && this._onBaseAnimationEnd())))); } } class Ry extends MHe { /** * Gets or sets the sprite size */ get size() { return this.width; } set size(e) { this.width = e, this.height = e; } /** * Gets the manager of this sprite */ get manager() { return this._manager; } /** * Creates a new Sprite * @param name defines the name * @param manager defines the manager */ constructor(e, t) { super(), this.name = e, this.animations = new Array(), this.isPickable = !1, this.useAlphaForPicking = !1, this.onDisposeObservable = new Oe(), this._onAnimationEnd = null, this._endAnimation = () => { this._onAnimationEnd && this._onAnimationEnd(), this.disposeWhenFinishedAnimating && this.dispose(); }, this.color = new xt(1, 1, 1, 1), this.position = S.Zero(), this._manager = t, this._manager.sprites.push(this), this.uniqueId = this._manager.scene.getUniqueId(); } /** * Returns the string "Sprite" * @returns "Sprite" */ getClassName() { return "Sprite"; } /** Gets or sets the initial key for the animation (setting it will restart the animation) */ get fromIndex() { return this._fromIndex; } set fromIndex(e) { this.playAnimation(e, this._toIndex, this._loopAnimation, this._delay, this._onAnimationEnd); } /** Gets or sets the end key for the animation (setting it will restart the animation) */ get toIndex() { return this._toIndex; } set toIndex(e) { this.playAnimation(this._fromIndex, e, this._loopAnimation, this._delay, this._onAnimationEnd); } /** Gets or sets a boolean indicating if the animation is looping (setting it will restart the animation) */ get loopAnimation() { return this._loopAnimation; } set loopAnimation(e) { this.playAnimation(this._fromIndex, this._toIndex, e, this._delay, this._onAnimationEnd); } /** Gets or sets the delay between cell changes (setting it will restart the animation) */ get delay() { return Math.max(this._delay, 1); } set delay(e) { this.playAnimation(this._fromIndex, this._toIndex, this._loopAnimation, e, this._onAnimationEnd); } /** * Starts an animation * @param from defines the initial key * @param to defines the end key * @param loop defines if the animation must loop * @param delay defines the start delay (in ms) * @param onAnimationEnd defines a callback to call when animation ends */ playAnimation(e, t, r, n, i = null) { this._onAnimationEnd = i, super.playAnimation(e, t, r, n, this._endAnimation); } /** Release associated resources */ dispose() { for (let e = 0; e < this._manager.sprites.length; e++) this._manager.sprites[e] == this && this._manager.sprites.splice(e, 1); this.onDisposeObservable.notifyObservers(this), this.onDisposeObservable.clear(); } /** * Serializes the sprite to a JSON object * @returns the JSON object */ serialize() { const e = {}; return e.name = this.name, e.position = this.position.asArray(), e.color = this.color.asArray(), e.width = this.width, e.height = this.height, e.angle = this.angle, e.cellIndex = this.cellIndex, e.cellRef = this.cellRef, e.invertU = this.invertU, e.invertV = this.invertV, e.disposeWhenFinishedAnimating = this.disposeWhenFinishedAnimating, e.isPickable = this.isPickable, e.isVisible = this.isVisible, e.useAlphaForPicking = this.useAlphaForPicking, e.animationStarted = this.animationStarted, e.fromIndex = this.fromIndex, e.toIndex = this.toIndex, e.loopAnimation = this.loopAnimation, e.delay = this.delay, e; } /** * Parses a JSON object to create a new sprite * @param parsedSprite The JSON object to parse * @param manager defines the hosting manager * @returns the new sprite */ static Parse(e, t) { const r = new Ry(e.name, t); return r.position = S.FromArray(e.position), r.color = xt.FromArray(e.color), r.width = e.width, r.height = e.height, r.angle = e.angle, r.cellIndex = e.cellIndex, r.cellRef = e.cellRef, r.invertU = e.invertU, r.invertV = e.invertV, r.disposeWhenFinishedAnimating = e.disposeWhenFinishedAnimating, r.isPickable = e.isPickable, r.isVisible = e.isVisible, r.useAlphaForPicking = e.useAlphaForPicking, r._fromIndex = e.fromIndex, r._toIndex = e.toIndex, r._loopAnimation = e.loopAnimation, r._delay = e.delay, e.animationStarted && r.playAnimation(r.fromIndex, r.toIndex, r.loopAnimation, r.delay), r; } } sr.prototype._internalPickSprites = function(A, e, t, r) { if (!F9) return null; let n = null; if (!r) { if (!this.activeCamera) return null; r = this.activeCamera; } if (this.spriteManagers && this.spriteManagers.length > 0) for (let i = 0; i < this.spriteManagers.length; i++) { const s = this.spriteManagers[i]; if (!s.isPickable) continue; const a = s.intersects(A, r, e, t); if (!(!a || !a.hit) && !(!t && n != null && a.distance >= n.distance) && (n = a, t)) break; } return n || new F9(); }; sr.prototype._internalMultiPickSprites = function(A, e, t) { if (!F9) return null; let r = []; if (!t) { if (!this.activeCamera) return null; t = this.activeCamera; } if (this.spriteManagers && this.spriteManagers.length > 0) for (let n = 0; n < this.spriteManagers.length; n++) { const i = this.spriteManagers[n]; if (!i.isPickable) continue; const s = i.multiIntersects(A, t, e); s !== null && (r = r.concat(s)); } return r; }; sr.prototype.pickSprite = function(A, e, t, r, n) { if (!this._tempSpritePickingRay) return null; this.createPickingRayInCameraSpaceToRef(A, e, this._tempSpritePickingRay, n); const i = this._internalPickSprites(this._tempSpritePickingRay, t, r, n); return i && (i.ray = this.createPickingRayInCameraSpace(A, e, n)), i; }; sr.prototype.pickSpriteWithRay = function(A, e, t, r) { if (!this._tempSpritePickingRay) return null; if (!r) { if (!this.activeCamera) return null; r = this.activeCamera; } Hi.TransformToRef(A, r.getViewMatrix(), this._tempSpritePickingRay); const n = this._internalPickSprites(this._tempSpritePickingRay, e, t, r); return n && (n.ray = A), n; }; sr.prototype.multiPickSprite = function(A, e, t, r) { return this.createPickingRayInCameraSpaceToRef(A, e, this._tempSpritePickingRay, r), this._internalMultiPickSprites(this._tempSpritePickingRay, t, r); }; sr.prototype.multiPickSpriteWithRay = function(A, e, t) { if (!this._tempSpritePickingRay) return null; if (!t) { if (!this.activeCamera) return null; t = this.activeCamera; } return Hi.TransformToRef(A, t.getViewMatrix(), this._tempSpritePickingRay), this._internalMultiPickSprites(this._tempSpritePickingRay, e, t); }; sr.prototype.setPointerOverSprite = function(A) { this._pointerOverSprite !== A && (this._pointerOverSprite && this._pointerOverSprite.actionManager && this._pointerOverSprite.actionManager.processTrigger(10, mo.CreateNewFromSprite(this._pointerOverSprite, this)), this._pointerOverSprite = A, this._pointerOverSprite && this._pointerOverSprite.actionManager && this._pointerOverSprite.actionManager.processTrigger(9, mo.CreateNewFromSprite(this._pointerOverSprite, this))); }; sr.prototype.getPointerOverSprite = function() { return this._pointerOverSprite; }; class cie { /** * Creates a new instance of the component for the given scene * @param scene Defines the scene to register the component in */ constructor(e) { this.name = Ot.NAME_SPRITE, this.scene = e, this.scene.spriteManagers = [], this.scene._tempSpritePickingRay = Hi ? Hi.Zero() : null, this.scene.onBeforeSpritesRenderingObservable = new Oe(), this.scene.onAfterSpritesRenderingObservable = new Oe(), this._spritePredicate = (t) => t.actionManager ? t.isPickable && t.actionManager.hasPointerTriggers : !1; } /** * Registers the component in a given scene */ register() { this.scene._pointerMoveStage.registerStep(Ot.STEP_POINTERMOVE_SPRITE, this, this._pointerMove), this.scene._pointerDownStage.registerStep(Ot.STEP_POINTERDOWN_SPRITE, this, this._pointerDown), this.scene._pointerUpStage.registerStep(Ot.STEP_POINTERUP_SPRITE, this, this._pointerUp); } /** * Rebuilds the elements related to this component in case of * context lost for instance. */ rebuild() { } /** * Disposes the component and the associated resources. */ dispose() { this.scene.onBeforeSpritesRenderingObservable.clear(), this.scene.onAfterSpritesRenderingObservable.clear(); const e = this.scene.spriteManagers; if (e) for (; e.length; ) e[0].dispose(); } _pickSpriteButKeepRay(e, t, r, n, i) { const s = this.scene.pickSprite(t, r, this._spritePredicate, n, i); return s && (s.ray = e ? e.ray : null), s; } _pointerMove(e, t, r, n, i) { const s = this.scene; return n ? s.setPointerOverSprite(null) : (r = this._pickSpriteButKeepRay(r, e, t, !1, s.cameraToUseForPointers || void 0), r && r.hit && r.pickedSprite ? (s.setPointerOverSprite(r.pickedSprite), !s.doNotHandleCursors && i && (s._pointerOverSprite && s._pointerOverSprite.actionManager && s._pointerOverSprite.actionManager.hoverCursor ? i.style.cursor = s._pointerOverSprite.actionManager.hoverCursor : i.style.cursor = s.hoverCursor)) : s.setPointerOverSprite(null)), r; } _pointerDown(e, t, r, n) { const i = this.scene; if (i._pickedDownSprite = null, i.spriteManagers && i.spriteManagers.length > 0 && (r = i.pickSprite(e, t, this._spritePredicate, !1, i.cameraToUseForPointers || void 0), r && r.hit && r.pickedSprite && r.pickedSprite.actionManager)) { switch (i._pickedDownSprite = r.pickedSprite, n.button) { case 0: r.pickedSprite.actionManager.processTrigger(2, mo.CreateNewFromSprite(r.pickedSprite, i, n)); break; case 1: r.pickedSprite.actionManager.processTrigger(4, mo.CreateNewFromSprite(r.pickedSprite, i, n)); break; case 2: r.pickedSprite.actionManager.processTrigger(3, mo.CreateNewFromSprite(r.pickedSprite, i, n)); break; } r.pickedSprite.actionManager && r.pickedSprite.actionManager.processTrigger(5, mo.CreateNewFromSprite(r.pickedSprite, i, n)); } return r; } _pointerUp(e, t, r, n, i) { const s = this.scene; if (s.spriteManagers && s.spriteManagers.length > 0) { const a = s.pickSprite(e, t, this._spritePredicate, !1, s.cameraToUseForPointers || void 0); a && (a.hit && a.pickedSprite && a.pickedSprite.actionManager && (a.pickedSprite.actionManager.processTrigger(7, mo.CreateNewFromSprite(a.pickedSprite, s, n)), a.pickedSprite.actionManager && (this.scene._inputManager._isPointerSwiping() || a.pickedSprite.actionManager.processTrigger(1, mo.CreateNewFromSprite(a.pickedSprite, s, n)), i && a.pickedSprite.actionManager.processTrigger(6, mo.CreateNewFromSprite(a.pickedSprite, s, n)))), s._pickedDownSprite && s._pickedDownSprite.actionManager && s._pickedDownSprite !== a.pickedSprite && s._pickedDownSprite.actionManager.processTrigger(16, mo.CreateNewFromSprite(s._pickedDownSprite, s, n))); } return r; } } const LHe = "imageProcessingCompatibility", KHe = `#ifdef IMAGEPROCESSINGPOSTPROCESS gl_FragColor.rgb=pow(gl_FragColor.rgb,vec3(2.2)); #endif `; Le.IncludesShadersStore[LHe] = KHe; const JHe = "spritesPixelShader", zHe = `uniform bool alphaTest;varying vec4 vColor;varying vec2 vUV;uniform sampler2D diffuseSampler; #include #define CUSTOM_FRAGMENT_DEFINITIONS #ifdef PIXEL_PERFECT vec2 uvPixelPerfect(vec2 uv) {vec2 res=vec2(textureSize(diffuseSampler,0));uv=uv*res;vec2 seam=floor(uv+0.5);uv=seam+clamp((uv-seam)/fwidth(uv),-0.5,0.5);return uv/res;} #endif void main(void) { #define CUSTOM_FRAGMENT_MAIN_BEGIN #ifdef PIXEL_PERFECT vec2 uv=uvPixelPerfect(vUV); #else vec2 uv=vUV; #endif vec4 color=texture2D(diffuseSampler,uv);float fAlphaTest=float(alphaTest);if (fAlphaTest != 0.) {if (color.a<0.95) discard;} color*=vColor; #include gl_FragColor=color; #include #define CUSTOM_FRAGMENT_MAIN_END }`; Le.ShadersStore[JHe] = zHe; const GHe = "spritesVertexShader", ZHe = `attribute vec4 position;attribute vec2 options;attribute vec2 offsets;attribute vec2 inverts;attribute vec4 cellInfo;attribute vec4 color;uniform mat4 view;uniform mat4 projection;varying vec2 vUV;varying vec4 vColor; #include #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN vec3 viewPos=(view*vec4(position.xyz,1.0)).xyz; vec2 cornerPos;float angle=position.w;vec2 size=vec2(options.x,options.y);vec2 offset=offsets.xy;cornerPos=vec2(offset.x-0.5,offset.y -0.5)*size;vec3 rotatedCorner;rotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.y=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.z=0.;viewPos+=rotatedCorner;gl_Position=projection*vec4(viewPos,1.0); vColor=color;vec2 uvOffset=vec2(abs(offset.x-inverts.x),abs(1.0-offset.y-inverts.y));vec2 uvPlace=cellInfo.xy;vec2 uvSize=cellInfo.zw;vUV.x=uvPlace.x+uvSize.x*uvOffset.x;vUV.y=uvPlace.y+uvSize.y*uvOffset.y; #ifdef FOG vFogDistance=viewPos; #endif #define CUSTOM_VERTEX_MAIN_END }`; Le.ShadersStore[GHe] = ZHe; class _He { /** * Gets the capacity of the manager */ get capacity() { return this._capacity; } /** * Gets or sets a boolean indicating if the renderer must render sprites with pixel perfect rendering * Note that pixel perfect mode is not supported in WebGL 1 */ get pixelPerfect() { return this._pixelPerfect; } set pixelPerfect(e) { this._pixelPerfect !== e && (this._pixelPerfect = e, this._createEffects()); } /** * Creates a new sprite Renderer * @param engine defines the engine the renderer works with * @param capacity defines the maximum allowed number of sprites * @param epsilon defines the epsilon value to align texture (0.01 by default) * @param scene defines the hosting scene */ constructor(e, t, r = 0.01, n = null) { this.blendMode = 2, this.autoResetAlpha = !0, this.disableDepthWrite = !1, this.fogEnabled = !0, this._pixelPerfect = !1, this._useVAO = !1, this._useInstancing = !1, this._vertexBuffers = {}, this._capacity = t, this._epsilon = r, this._engine = e, this._useInstancing = e.getCaps().instancedArrays && e._features.supportSpriteInstancing, this._useVAO = e.getCaps().vertexArrayObject && !e.disableVertexArrayObjects, this._scene = n, this._useInstancing || this._buildIndexBuffer(), this._vertexBufferSize = this._useInstancing ? 16 : 18, this._vertexData = new Float32Array(t * this._vertexBufferSize * (this._useInstancing ? 1 : 4)), this._buffer = new P9(e, this._vertexData, !0, this._vertexBufferSize); const i = this._buffer.createVertexBuffer(J.PositionKind, 0, 4, this._vertexBufferSize, this._useInstancing), s = this._buffer.createVertexBuffer("options", 4, 2, this._vertexBufferSize, this._useInstancing); let a = 6, f; if (this._useInstancing) { const u = new Float32Array([0, 0, 1, 0, 0, 1, 1, 1]); this._spriteBuffer = new P9(e, u, !1, 2), f = this._spriteBuffer.createVertexBuffer("offsets", 0, 2); } else f = this._buffer.createVertexBuffer("offsets", a, 2, this._vertexBufferSize, this._useInstancing), a += 2; const o = this._buffer.createVertexBuffer("inverts", a, 2, this._vertexBufferSize, this._useInstancing), d = this._buffer.createVertexBuffer("cellInfo", a + 2, 4, this._vertexBufferSize, this._useInstancing), v = this._buffer.createVertexBuffer(J.ColorKind, a + 6, 4, this._vertexBufferSize, this._useInstancing); this._vertexBuffers[J.PositionKind] = i, this._vertexBuffers.options = s, this._vertexBuffers.offsets = f, this._vertexBuffers.inverts = o, this._vertexBuffers.cellInfo = d, this._vertexBuffers[J.ColorKind] = v, this._createEffects(); } _createEffects() { var e, t, r, n; (e = this._drawWrapperBase) === null || e === void 0 || e.dispose(), (t = this._drawWrapperFog) === null || t === void 0 || t.dispose(), (r = this._drawWrapperDepth) === null || r === void 0 || r.dispose(), (n = this._drawWrapperFogDepth) === null || n === void 0 || n.dispose(), this._drawWrapperBase = new zo(this._engine), this._drawWrapperFog = new zo(this._engine), this._drawWrapperDepth = new zo(this._engine, !1), this._drawWrapperFogDepth = new zo(this._engine, !1), this._drawWrapperBase.drawContext && (this._drawWrapperBase.drawContext.useInstancing = this._useInstancing), this._drawWrapperFog.drawContext && (this._drawWrapperFog.drawContext.useInstancing = this._useInstancing), this._drawWrapperDepth.drawContext && (this._drawWrapperDepth.drawContext.useInstancing = this._useInstancing), this._drawWrapperFogDepth.drawContext && (this._drawWrapperFogDepth.drawContext.useInstancing = this._useInstancing); const i = this._pixelPerfect ? `#define PIXEL_PERFECT ` : ""; this._drawWrapperBase.effect = this._engine.createEffect("sprites", [J.PositionKind, "options", "offsets", "inverts", "cellInfo", J.ColorKind], ["view", "projection", "textureInfos", "alphaTest"], ["diffuseSampler"], i), this._drawWrapperDepth.effect = this._drawWrapperBase.effect, this._drawWrapperDepth.materialContext = this._drawWrapperBase.materialContext, this._scene && (this._drawWrapperFog.effect = this._scene.getEngine().createEffect("sprites", [J.PositionKind, "options", "offsets", "inverts", "cellInfo", J.ColorKind], ["view", "projection", "textureInfos", "alphaTest", "vFogInfos", "vFogColor"], ["diffuseSampler"], i + "#define FOG"), this._drawWrapperFogDepth.effect = this._drawWrapperFog.effect, this._drawWrapperFogDepth.materialContext = this._drawWrapperFog.materialContext); } /** * Render all child sprites * @param sprites defines the list of sprites to render * @param deltaTime defines the time since last frame * @param viewMatrix defines the viewMatrix to use to render the sprites * @param projectionMatrix defines the projectionMatrix to use to render the sprites * @param customSpriteUpdate defines a custom function to update the sprites data before they render */ render(e, t, r, n, i = null) { if (!this.texture || !this.texture.isReady() || !e.length) return; let s = this._drawWrapperBase, a = this._drawWrapperDepth, f = !1; this.fogEnabled && this._scene && this._scene.fogEnabled && this._scene.fogMode !== 0 && (s = this._drawWrapperFog, a = this._drawWrapperFogDepth, f = !0); const o = s.effect; if (!o.isReady()) return; const d = this._engine, v = !!(this._scene && this._scene.useRightHandedSystem), u = this.texture.getBaseSize(), l = Math.min(this._capacity, e.length); let P = 0, p = !0; for (let q = 0; q < l; q++) { const b = e[q]; !b || !b.isVisible || (p = !1, b._animate(t), this._appendSpriteVertex(P++, b, 0, 0, u, v, i), this._useInstancing || (this._appendSpriteVertex(P++, b, 1, 0, u, v, i), this._appendSpriteVertex(P++, b, 1, 1, u, v, i), this._appendSpriteVertex(P++, b, 0, 1, u, v, i))); } if (p) return; this._buffer.update(this._vertexData); const c = !!d.depthCullingState.cull, H = d.depthCullingState.zOffset, T = d.depthCullingState.zOffsetUnits; if (d.setState(c, H, !1, !1, void 0, void 0, T), d.enableEffect(s), o.setTexture("diffuseSampler", this.texture), o.setMatrix("view", r), o.setMatrix("projection", n), f) { const q = this._scene; o.setFloat4("vFogInfos", q.fogMode, q.fogStart, q.fogEnd, q.fogDensity), o.setColor3("vFogColor", q.fogColor); } this._useVAO ? (this._vertexArrayObject || (this._vertexArrayObject = d.recordVertexArrayObject(this._vertexBuffers, this._indexBuffer, o)), d.bindVertexArrayObject(this._vertexArrayObject, this._indexBuffer)) : d.bindBuffers(this._vertexBuffers, this._indexBuffer, o), d.depthCullingState.depthFunc = d.useReverseDepthBuffer ? 518 : 515, this.disableDepthWrite || (o.setBool("alphaTest", !0), d.setColorWrite(!1), d.enableEffect(a), this._useInstancing ? d.drawArraysType(7, 0, 4, P) : d.drawElementsType(0, 0, P / 4 * 6), d.enableEffect(s), d.setColorWrite(!0), o.setBool("alphaTest", !1)), d.setAlphaMode(this.blendMode), this._useInstancing ? d.drawArraysType(7, 0, 4, P) : d.drawElementsType(0, 0, P / 4 * 6), this.autoResetAlpha && d.setAlphaMode(0), v && this._scene.getEngine().setState(c, H, !1, !0, void 0, void 0, T), d.unbindInstanceAttributes(); } _appendSpriteVertex(e, t, r, n, i, s, a) { let f = e * this._vertexBufferSize; if (r === 0 ? r = this._epsilon : r === 1 && (r = 1 - this._epsilon), n === 0 ? n = this._epsilon : n === 1 && (n = 1 - this._epsilon), a) a(t, i); else { t.cellIndex || (t.cellIndex = 0); const o = i.width / this.cellWidth, d = t.cellIndex / o >> 0; t._xOffset = (t.cellIndex - d * o) * this.cellWidth / i.width, t._yOffset = d * this.cellHeight / i.height, t._xSize = this.cellWidth, t._ySize = this.cellHeight; } this._vertexData[f] = t.position.x, this._vertexData[f + 1] = t.position.y, this._vertexData[f + 2] = t.position.z, this._vertexData[f + 3] = t.angle, this._vertexData[f + 4] = t.width, this._vertexData[f + 5] = t.height, this._useInstancing ? f -= 2 : (this._vertexData[f + 6] = r, this._vertexData[f + 7] = n), s ? this._vertexData[f + 8] = t.invertU ? 0 : 1 : this._vertexData[f + 8] = t.invertU ? 1 : 0, this._vertexData[f + 9] = t.invertV ? 1 : 0, this._vertexData[f + 10] = t._xOffset, this._vertexData[f + 11] = t._yOffset, this._vertexData[f + 12] = t._xSize / i.width, this._vertexData[f + 13] = t._ySize / i.height, this._vertexData[f + 14] = t.color.r, this._vertexData[f + 15] = t.color.g, this._vertexData[f + 16] = t.color.b, this._vertexData[f + 17] = t.color.a; } _buildIndexBuffer() { const e = []; let t = 0; for (let r = 0; r < this._capacity; r++) e.push(t), e.push(t + 1), e.push(t + 2), e.push(t), e.push(t + 2), e.push(t + 3), t += 4; this._indexBuffer = this._engine.createIndexBuffer(e); } /** * Rebuilds the renderer (after a context lost, for eg) */ rebuild() { var e; this._indexBuffer && this._buildIndexBuffer(), this._useVAO && (this._vertexArrayObject = void 0), this._buffer._rebuild(); for (const t in this._vertexBuffers) this._vertexBuffers[t]._rebuild(); (e = this._spriteBuffer) === null || e === void 0 || e._rebuild(); } /** * Release associated resources */ dispose() { this._buffer && (this._buffer.dispose(), this._buffer = null), this._spriteBuffer && (this._spriteBuffer.dispose(), this._spriteBuffer = null), this._indexBuffer && (this._engine._releaseBuffer(this._indexBuffer), this._indexBuffer = null), this._vertexArrayObject && (this._engine.releaseVertexArrayObject(this._vertexArrayObject), this._vertexArrayObject = null), this.texture && (this.texture.dispose(), this.texture = null), this._drawWrapperBase.dispose(), this._drawWrapperFog.dispose(), this._drawWrapperDepth.dispose(), this._drawWrapperFogDepth.dispose(); } } class z2 { /** * Callback called when the manager is disposed */ set onDispose(e) { this._onDisposeObserver && this.onDisposeObservable.remove(this._onDisposeObserver), this._onDisposeObserver = this.onDisposeObservable.add(e); } /** * Gets the array of sprites */ get children() { return this.sprites; } /** * Gets the hosting scene */ get scene() { return this._scene; } /** * Gets the capacity of the manager */ get capacity() { return this._spriteRenderer.capacity; } /** * Gets or sets the spritesheet texture */ get texture() { return this._spriteRenderer.texture; } set texture(e) { e.wrapU = We.CLAMP_ADDRESSMODE, e.wrapV = We.CLAMP_ADDRESSMODE, this._spriteRenderer.texture = e, this._textureContent = null; } /** Defines the default width of a cell in the spritesheet */ get cellWidth() { return this._spriteRenderer.cellWidth; } set cellWidth(e) { this._spriteRenderer.cellWidth = e; } /** Defines the default height of a cell in the spritesheet */ get cellHeight() { return this._spriteRenderer.cellHeight; } set cellHeight(e) { this._spriteRenderer.cellHeight = e; } /** Gets or sets a boolean indicating if the manager must consider scene fog when rendering */ get fogEnabled() { return this._spriteRenderer.fogEnabled; } set fogEnabled(e) { this._spriteRenderer.fogEnabled = e; } /** * Blend mode use to render the particle, it can be any of * the static undefined properties provided in this class. * Default value is 2 */ get blendMode() { return this._spriteRenderer.blendMode; } set blendMode(e) { this._spriteRenderer.blendMode = e; } /** Disables writing to the depth buffer when rendering the sprites. * It can be handy to disable depth writing when using textures without alpha channel * and setting some specific blend modes. */ get disableDepthWrite() { return this._disableDepthWrite; } set disableDepthWrite(e) { this._disableDepthWrite = e, this._spriteRenderer.disableDepthWrite = e; } /** * Gets or sets a boolean indicating if the renderer must render sprites with pixel perfect rendering * In this mode, sprites are rendered as "pixel art", which means that they appear as pixelated but remain stable when moving or when rotated or scaled. * Note that for this mode to work as expected, the sprite texture must use the BILINEAR sampling mode, not NEAREST! */ get pixelPerfect() { return this._spriteRenderer.pixelPerfect; } set pixelPerfect(e) { this._spriteRenderer.pixelPerfect = e, e && this.texture.samplingMode !== 3 && this.texture.updateSamplingMode(3); } /** * Creates a new sprite manager * @param name defines the manager's name * @param imgUrl defines the sprite sheet url * @param capacity defines the maximum allowed number of sprites * @param cellSize defines the size of a sprite cell * @param scene defines the hosting scene * @param epsilon defines the epsilon value to align texture (0.01 by default) * @param samplingMode defines the sampling mode to use with spritesheet * @param fromPacked set to false; do not alter * @param spriteJSON null otherwise a JSON object defining sprite sheet data; do not alter */ constructor(e, t, r, n, i, s = 0.01, a = We.TRILINEAR_SAMPLINGMODE, f = !1, o = null) { this.name = e, this.sprites = [], this.renderingGroupId = 0, this.layerMask = 268435455, this.isPickable = !1, this.metadata = null, this._wasDispatched = !1, this.onDisposeObservable = new Oe(), this._disableDepthWrite = !1, this._packedAndReady = !1, this._customUpdate = (v, u) => { v.cellRef || (v.cellIndex = 0); const l = v.cellIndex; typeof l == "number" && isFinite(l) && Math.floor(l) === l && (v.cellRef = this._spriteMap[v.cellIndex]), v._xOffset = this._cellData[v.cellRef].frame.x / u.width, v._yOffset = this._cellData[v.cellRef].frame.y / u.height, v._xSize = this._cellData[v.cellRef].frame.w, v._ySize = this._cellData[v.cellRef].frame.h; }, i || (i = gr.LastCreatedScene), i._getComponent(Ot.NAME_SPRITE) || i._addComponent(new cie(i)), this._fromPacked = f, this._scene = i; const d = this._scene.getEngine(); if (this._spriteRenderer = new _He(d, r, s, i), n.width && n.height) this.cellWidth = n.width, this.cellHeight = n.height; else if (n !== void 0) this.cellWidth = n, this.cellHeight = n; else { this._spriteRenderer = null; return; } this._scene.spriteManagers && this._scene.spriteManagers.push(this), this.uniqueId = this.scene.getUniqueId(), t && (this.texture = new We(t, i, !0, !1, a)), this._fromPacked && this._makePacked(t, o); } /** * Returns the string "SpriteManager" * @returns "SpriteManager" */ getClassName() { return "SpriteManager"; } _makePacked(e, t) { if (t !== null) try { let r; if (typeof t == "string" ? r = JSON.parse(t) : r = t, r.frames.length) { const i = {}; for (let s = 0; s < r.frames.length; s++) { const a = r.frames[s]; if (typeof Object.keys(a)[0] != "string") throw new Error("Invalid JSON Format. Check the frame values and make sure the name is the first parameter."); const f = a[Object.keys(a)[0]]; i[f] = a; } r.frames = i; } const n = Reflect.ownKeys(r.frames); this._spriteMap = n, this._packedAndReady = !0, this._cellData = r.frames; } catch { throw this._fromPacked = !1, this._packedAndReady = !1, new Error("Invalid JSON from string. Spritesheet managed with constant cell size."); } else { const r = /\./g; let n; do n = r.lastIndex, r.test(e); while (r.lastIndex > 0); const i = e.substring(0, n - 1) + ".json", s = () => { Se.Error("JSON ERROR: Unable to load JSON file."), this._fromPacked = !1, this._packedAndReady = !1; }, a = (f) => { try { const o = JSON.parse(f), d = Reflect.ownKeys(o.frames); this._spriteMap = d, this._packedAndReady = !0, this._cellData = o.frames; } catch { throw this._fromPacked = !1, this._packedAndReady = !1, new Error("Invalid JSON format. Please check documentation for format specifications."); } }; ye.LoadFile(i, a, void 0, void 0, !1, s); } } _checkTextureAlpha(e, t, r, n, i) { if (!e.useAlphaForPicking || !this.texture) return !0; const s = this.texture.getSize(); this._textureContent || (this._textureContent = new Uint8Array(s.width * s.height * 4), this.texture.readPixels(0, 0, this._textureContent)); const a = ue.Vector3[0]; a.copyFrom(t.direction), a.normalize(), a.scaleInPlace(r), a.addInPlace(t.origin); const f = (a.x - n.x) / (i.x - n.x), o = 1 - (a.y - n.y) / (i.y - n.y), d = e._xOffset * s.width + f * e._xSize | 0, v = e._yOffset * s.height + o * e._ySize | 0; return this._textureContent[(d + v * s.width) * 4 + 3] > 0.5; } /** * Intersects the sprites with a ray * @param ray defines the ray to intersect with * @param camera defines the current active camera * @param predicate defines a predicate used to select candidate sprites * @param fastCheck defines if a fast check only must be done (the first potential sprite is will be used and not the closer) * @returns null if no hit or a PickingInfo */ intersects(e, t, r, n) { const i = Math.min(this.capacity, this.sprites.length), s = S.Zero(), a = S.Zero(); let f = Number.MAX_VALUE, o = null; const d = ue.Vector3[0], v = ue.Vector3[1], u = t.getViewMatrix(); let l = e, P = e; for (let p = 0; p < i; p++) { const c = this.sprites[p]; if (c) { if (r) { if (!r(c)) continue; } else if (!c.isPickable) continue; if (S.TransformCoordinatesToRef(c.position, u, v), c.angle ? (he.TranslationToRef(-v.x, -v.y, 0, ue.Matrix[1]), he.TranslationToRef(v.x, v.y, 0, ue.Matrix[2]), he.RotationZToRef(-c.angle, ue.Matrix[3]), ue.Matrix[1].multiplyToRef(ue.Matrix[3], ue.Matrix[4]), ue.Matrix[4].multiplyToRef(ue.Matrix[2], ue.Matrix[0]), l = e.clone(), S.TransformCoordinatesToRef(e.origin, ue.Matrix[0], l.origin), S.TransformNormalToRef(e.direction, ue.Matrix[0], l.direction)) : l = e, s.copyFromFloats(v.x - c.width / 2, v.y - c.height / 2, v.z), a.copyFromFloats(v.x + c.width / 2, v.y + c.height / 2, v.z), l.intersectsBoxMinMax(s, a)) { const H = S.Distance(v, l.origin); if (f > H) { if (!this._checkTextureAlpha(c, l, H, s, a)) continue; if (P = l, f = H, o = c, n) break; } } } } if (o) { const p = new F9(); u.invertToRef(ue.Matrix[0]), p.hit = !0, p.pickedSprite = o, p.distance = f; const c = ue.Vector3[2]; return c.copyFrom(P.direction), c.normalize(), c.scaleInPlace(f), P.origin.addToRef(c, d), p.pickedPoint = S.TransformCoordinates(d, ue.Matrix[0]), p; } return null; } /** * Intersects the sprites with a ray * @param ray defines the ray to intersect with * @param camera defines the current active camera * @param predicate defines a predicate used to select candidate sprites * @returns null if no hit or a PickingInfo array */ multiIntersects(e, t, r) { const n = Math.min(this.capacity, this.sprites.length), i = S.Zero(), s = S.Zero(); let a; const f = [], o = ue.Vector3[0].copyFromFloats(0, 0, 0), d = ue.Vector3[1].copyFromFloats(0, 0, 0), v = t.getViewMatrix(); for (let u = 0; u < n; u++) { const l = this.sprites[u]; if (l) { if (r) { if (!r(l)) continue; } else if (!l.isPickable) continue; if (S.TransformCoordinatesToRef(l.position, v, d), i.copyFromFloats(d.x - l.width / 2, d.y - l.height / 2, d.z), s.copyFromFloats(d.x + l.width / 2, d.y + l.height / 2, d.z), e.intersectsBoxMinMax(i, s)) { if (a = S.Distance(d, e.origin), !this._checkTextureAlpha(l, e, a, i, s)) continue; const P = new F9(); f.push(P), v.invertToRef(ue.Matrix[0]), P.hit = !0, P.pickedSprite = l, P.distance = a; const p = ue.Vector3[2]; p.copyFrom(e.direction), p.normalize(), p.scaleInPlace(a), e.origin.addToRef(p, o), P.pickedPoint = S.TransformCoordinates(o, ue.Matrix[0]); } } } return f; } /** * Render all child sprites */ render() { if (this._fromPacked && (!this._packedAndReady || !this._spriteMap || !this._cellData)) return; const t = this._scene.getEngine().getDeltaTime(); this._packedAndReady ? this._spriteRenderer.render(this.sprites, t, this._scene.getViewMatrix(), this._scene.getProjectionMatrix(), this._customUpdate) : this._spriteRenderer.render(this.sprites, t, this._scene.getViewMatrix(), this._scene.getProjectionMatrix()); } /** * Rebuilds the manager (after a context lost, for eg) */ rebuild() { var e; (e = this._spriteRenderer) === null || e === void 0 || e.rebuild(); } /** * Release associated resources */ dispose() { if (this._spriteRenderer && (this._spriteRenderer.dispose(), this._spriteRenderer = null), this._textureContent = null, this._scene.spriteManagers) { const e = this._scene.spriteManagers.indexOf(this); this._scene.spriteManagers.splice(e, 1); } this.onDisposeObservable.notifyObservers(this), this.onDisposeObservable.clear(), this.metadata = null; } /** * Serializes the sprite manager to a JSON object * @param serializeTexture defines if the texture must be serialized as well * @returns the JSON object */ serialize(e = !1) { const t = {}; t.name = this.name, t.capacity = this.capacity, t.cellWidth = this.cellWidth, t.cellHeight = this.cellHeight, t.fogEnabled = this.fogEnabled, t.blendMode = this.blendMode, t.disableDepthWrite = this.disableDepthWrite, t.pixelPerfect = this.pixelPerfect, this.texture && (e ? t.texture = this.texture.serialize() : (t.textureUrl = this.texture.name, t.invertY = this.texture._invertY)), t.sprites = []; for (const r of this.sprites) t.sprites.push(r.serialize()); return t.metadata = this.metadata, t; } /** * Parses a JSON object to create a new sprite manager. * @param parsedManager The JSON object to parse * @param scene The scene to create the sprite manager * @param rootUrl The root url to use to load external dependencies like texture * @returns the new sprite manager */ static Parse(e, t, r) { const n = new z2(e.name, "", e.capacity, { width: e.cellWidth, height: e.cellHeight }, t); e.fogEnabled !== void 0 && (n.fogEnabled = e.fogEnabled), e.blendMode !== void 0 && (n.blendMode = e.blendMode), e.disableDepthWrite !== void 0 && (n.disableDepthWrite = e.disableDepthWrite), e.pixelPerfect !== void 0 && (n.pixelPerfect = e.pixelPerfect), e.metadata !== void 0 && (n.metadata = e.metadata), e.texture ? n.texture = We.Parse(e.texture, t, r) : e.textureName && (n.texture = new We(r + e.textureUrl, t, !1, e.invertY !== void 0 ? e.invertY : !0)); for (const i of e.sprites) Ry.Parse(i, n); return n; } /** * Creates a sprite manager from a snippet saved in a remote file * @param name defines the name of the sprite manager to create (can be null or empty to use the one from the json data) * @param url defines the url to load from * @param scene defines the hosting scene * @param rootUrl defines the root URL to use to load textures and relative dependencies * @returns a promise that will resolve to the new sprite manager */ static ParseFromFileAsync(e, t, r, n = "") { return new Promise((i, s) => { const a = new ho(); a.addEventListener("readystatechange", () => { if (a.readyState == 4) if (a.status == 200) { const f = JSON.parse(a.responseText), o = z2.Parse(f, r || gr.LastCreatedScene, n); e && (o.name = e), i(o); } else s("Unable to load the sprite manager"); }), a.open("GET", t), a.send(); }); } /** * Creates a sprite manager from a snippet saved by the sprite editor * @param snippetId defines the snippet to load (can be set to _BLANK to create a default one) * @param scene defines the hosting scene * @param rootUrl defines the root URL to use to load textures and relative dependencies * @returns a promise that will resolve to the new sprite manager */ static ParseFromSnippetAsync(e, t, r = "") { return e === "_BLANK" ? Promise.resolve(new z2("Default sprite manager", "//playground.babylonjs.com/textures/player.png", 500, 64, t)) : new Promise((n, i) => { const s = new ho(); s.addEventListener("readystatechange", () => { if (s.readyState == 4) if (s.status == 200) { const a = JSON.parse(JSON.parse(s.responseText).jsonPayload), f = JSON.parse(a.spriteManager), o = z2.Parse(f, t || gr.LastCreatedScene, r); o.snippetId = e, n(o); } else i("Unable to load the snippet " + e); }), s.open("GET", this.SnippetUrl + "/" + e.replace(/#/g, "/")), s.send(); }); } } z2.SnippetUrl = "https://snippet.babylonjs.com"; z2.CreateFromSnippetAsync = z2.ParseFromSnippetAsync; const $He = "spriteMapPixelShader", ege = `#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE) #define TEXTUREFUNC(s,c,l) texture2DLodEXT(s,c,l) #else #define TEXTUREFUNC(s,c,b) texture2D(s,c,b) #endif precision highp float;varying vec3 vPosition;varying vec2 vUV;varying vec2 tUV;uniform float time;uniform float spriteCount;uniform sampler2D spriteSheet;uniform vec2 spriteMapSize;uniform vec2 outputSize;uniform vec2 stageSize;uniform sampler2D frameMap;uniform sampler2D tileMaps[LAYERS];uniform sampler2D animationMap;uniform vec3 colorMul;float mt;const float fdStep=1./4.;const float aFrameSteps=1./MAX_ANIMATION_FRAMES;mat4 getFrameData(float frameID){float fX=frameID/spriteCount;return mat4( texture2D(frameMap,vec2(fX,0.),0.), texture2D(frameMap,vec2(fX,fdStep*1.),0.), texture2D(frameMap,vec2(fX,fdStep*2.),0.), vec4(0.) );} void main(){vec4 color=vec4(0.);vec2 tileUV=fract(tUV); #ifdef FLIPU tileUV.y=1.0-tileUV.y; #endif vec2 tileID=floor(tUV);vec2 sheetUnits=1./spriteMapSize;float spriteUnits=1./spriteCount;vec2 stageUnits=1./stageSize;for(int i=0; i0.) {mt=mod(time*animationData.z,1.0);for(float f=0.; fmt){frameID=animationData.x;break;} animationData=TEXTUREFUNC(animationMap,vec2((frameID+0.5)/spriteCount,aFrameSteps*f),0.);}} mat4 frameData=getFrameData(frameID+0.5);vec2 frameSize=(frameData[0].zw)/spriteMapSize;vec2 offset=frameData[0].xy*sheetUnits;vec2 ratio=frameData[2].xy/frameData[0].zw;if (frameData[2].z==1.){tileUV.xy=tileUV.yx;} vec4 nc=texture2D(spriteSheet,tileUV*frameSize+offset);if (i==0){color=nc;} else {float alpha=min(color.a+nc.a,1.0);vec3 mixed=mix(color.xyz,nc.xyz,nc.a);color=vec4(mixed,alpha);}} color.xyz*=colorMul;gl_FragColor=color;}`; Le.ShadersStore[$He] = ege; const tge = "spriteMapVertexShader", rge = `precision highp float;attribute vec3 position;attribute vec3 normal;attribute vec2 uv;varying vec3 vPosition;varying vec2 vUV;varying vec2 tUV;varying vec2 stageUnits;varying vec2 levelUnits;varying vec2 tileID;uniform float time;uniform mat4 worldViewProjection;uniform vec2 outputSize;uniform vec2 stageSize;uniform vec2 spriteMapSize;uniform float stageScale;void main() {vec4 p=vec4( position,1. );vPosition=p.xyz;vUV=uv;tUV=uv*stageSize; gl_Position=worldViewProjection*p;}`; Le.ShadersStore[tge] = rge; class nge { /** Returns the Number of Sprites in the System */ get spriteCount() { return this.sprites.length; } /** Returns the Position of Output Plane*/ get position() { return this._output.position; } /** Returns the Position of Output Plane*/ set position(e) { this._output.position = e; } /** Returns the Rotation of Output Plane*/ get rotation() { return this._output.rotation; } /** Returns the Rotation of Output Plane*/ set rotation(e) { this._output.rotation = e; } /** Sets the AnimationMap*/ get animationMap() { return this._animationMap; } /** Sets the AnimationMap*/ set animationMap(e) { const t = e._texture._bufferView, r = this._createTileAnimationBuffer(t); this._animationMap.dispose(), this._animationMap = r, this._material.setTexture("animationMap", this._animationMap); } /** * Creates a new SpriteMap * @param name defines the SpriteMaps Name * @param atlasJSON is the JSON file that controls the Sprites Frames and Meta * @param spriteSheet is the Texture that the Sprites are on. * @param options a basic deployment configuration * @param scene The Scene that the map is deployed on */ constructor(e, t, r, n, i) { this.name = e, this.sprites = [], this.atlasJSON = t, this.sprites = this.atlasJSON.frames, this.spriteSheet = r, this.options = n, n.stageSize = n.stageSize || new at(1, 1), n.outputSize = n.outputSize || n.stageSize, n.outputPosition = n.outputPosition || S.Zero(), n.outputRotation = n.outputRotation || S.Zero(), n.layerCount = n.layerCount || 1, n.maxAnimationFrames = n.maxAnimationFrames || 0, n.baseTile = n.baseTile || 0, n.flipU = n.flipU || !1, n.colorMultiply = n.colorMultiply || new S(1, 1, 1), this._scene = i, this._frameMap = this._createFrameBuffer(), this._tileMaps = new Array(); for (let u = 0; u < n.layerCount; u++) this._tileMaps.push(this._createTileBuffer(null, u)); this._animationMap = this._createTileAnimationBuffer(null); const s = []; s.push("#define LAYERS " + n.layerCount), n.flipU && s.push("#define FLIPU"), s.push(`#define MAX_ANIMATION_FRAMES ${n.maxAnimationFrames}.0`); const a = An.ShadersStore.spriteMapPixelShader; let f; if (i.getEngine()._features.supportSwitchCaseInShader) { f = "switch(i) {"; for (let u = 0; u < n.layerCount; u++) f += "case " + u + " : frameID = texture(tileMaps[" + u + "], (tileID + 0.5) / stageSize, 0.).x;", f += "break;"; f += "}"; } else { f = ""; for (let u = 0; u < n.layerCount; u++) f += `if (${u} == i) { frameID = texture2D(tileMaps[${u}], (tileID + 0.5) / stageSize, 0.).x; }`; } An.ShadersStore["spriteMap" + this.name + "PixelShader"] = a.replace("#define LAYER_ID_SWITCH", f), this._material = new Zo("spriteMap:" + this.name, this._scene, { vertex: "spriteMap", fragment: "spriteMap" + this.name }, { defines: s, attributes: ["position", "normal", "uv"], uniforms: ["worldViewProjection", "time", "stageSize", "outputSize", "spriteMapSize", "spriteCount", "time", "colorMul", "mousePosition", "curTile", "flipU"], samplers: ["spriteSheet", "frameMap", "tileMaps", "animationMap"], needAlphaBlending: !0 }), this._time = 0, this._material.setFloat("spriteCount", this.spriteCount), this._material.setVector2("stageSize", n.stageSize), this._material.setVector2("outputSize", n.outputSize), this._material.setTexture("spriteSheet", this.spriteSheet), this._material.setVector2("spriteMapSize", new at(1, 1)), this._material.setVector3("colorMul", n.colorMultiply); let o = 0; const d = () => { if (this.spriteSheet && this.spriteSheet.isReady() && this.spriteSheet._texture) { this._material.setVector2("spriteMapSize", new at(this.spriteSheet._texture.baseWidth || 1, this.spriteSheet._texture.baseHeight || 1)); return; } o < 100 && setTimeout(() => { o++, d(); }, 100); }; d(), this._material.setVector3("colorMul", n.colorMultiply), this._material.setTexture("frameMap", this._frameMap), this._material.setTextureArray("tileMaps", this._tileMaps), this._material.setTexture("animationMap", this._animationMap), this._material.setFloat("time", this._time), this._output = u4(e + ":output", { size: 1, updatable: !0 }, i), this._output.scaling.x = n.outputSize.x, this._output.scaling.y = n.outputSize.y, this.position = n.outputPosition, this.rotation = n.outputRotation; const v = () => { this._time += this._scene.getEngine().getDeltaTime(), this._material.setFloat("time", this._time); }; this._scene.onBeforeRenderObservable.add(v), this._output.material = this._material; } /** * Returns tileID location * @returns Vector2 the cell position ID */ getTileID() { const e = this.getMousePosition(); return e.multiplyInPlace(this.options.stageSize || at.Zero()), e.x = Math.floor(e.x), e.y = Math.floor(e.y), e; } /** * Gets the UV location of the mouse over the SpriteMap. * @returns Vector2 the UV position of the mouse interaction */ getMousePosition() { const e = this._output, t = this._scene.pick(this._scene.pointerX, this._scene.pointerY, (n) => n === e); if (!t || !t.hit || !t.getTextureCoordinates) return new at(-1, -1); const r = t.getTextureCoordinates(); return r || new at(-1, -1); } /** * Creates the "frame" texture Buffer * ------------------------------------- * Structure of frames * "filename": "Falling-Water-2.png", * "frame": {"x":69,"y":103,"w":24,"h":32}, * "rotated": true, * "trimmed": true, * "spriteSourceSize": {"x":4,"y":0,"w":24,"h":32}, * "sourceSize": {"w":32,"h":32} * @returns RawTexture of the frameMap */ _createFrameBuffer() { const e = []; for (let n = 0; n < this.spriteCount; n++) e.push(0, 0, 0, 0), e.push(0, 0, 0, 0), e.push(0, 0, 0, 0), e.push(0, 0, 0, 0); for (let n = 0; n < this.spriteCount; n++) { const i = this.sprites[n].frame, s = this.sprites[n].spriteSourceSize, a = this.sprites[n].sourceSize, f = this.sprites[n].rotated ? 1 : 0, o = this.sprites[n].trimmed ? 1 : 0; e[n * 4] = i.x, e[n * 4 + 1] = i.y, e[n * 4 + 2] = i.w, e[n * 4 + 3] = i.h, e[n * 4 + this.spriteCount * 4] = s.x, e[n * 4 + 1 + this.spriteCount * 4] = s.y, e[n * 4 + 3 + this.spriteCount * 4] = s.h, e[n * 4 + this.spriteCount * 8] = a.w, e[n * 4 + 1 + this.spriteCount * 8] = a.h, e[n * 4 + 2 + this.spriteCount * 8] = f, e[n * 4 + 3 + this.spriteCount * 8] = o; } const t = new Float32Array(e); return Bo.CreateRGBATexture(t, this.spriteCount, 4, this._scene, !1, !1, We.NEAREST_NEAREST, Ge.TEXTURETYPE_FLOAT); } /** * Creates the tileMap texture Buffer * @param buffer normally and array of numbers, or a false to generate from scratch * @param _layer indicates what layer for a logic trigger dealing with the baseTile. The system uses this * @returns RawTexture of the tileMap */ _createTileBuffer(e, t = 0) { let r = []; const n = this.options.stageSize.y || 0, i = this.options.stageSize.x || 0; if (e) r = e; else { let f = this.options.baseTile; t != 0 && (f = 0); for (let o = 0; o < n; o++) for (let d = 0; d < i * 4; d += 4) r.push(f, 0, 0, 0); } const s = new Float32Array(r); return Bo.CreateRGBATexture(s, i, n, this._scene, !1, !1, We.NEAREST_NEAREST, Ge.TEXTURETYPE_FLOAT); } /** * Modifies the data of the tileMaps * @param _layer is the ID of the layer you want to edit on the SpriteMap * @param pos is the iVector2 Coordinates of the Tile * @param tile The SpriteIndex of the new Tile */ changeTiles(e = 0, t, r = 0) { const n = this._tileMaps[e]._texture._bufferView; if (n === null) return; let i = []; t instanceof at ? i.push(t) : i = t; const s = this.options.stageSize.x || 0; for (let f = 0; f < i.length; f++) { const o = i[f]; o.x = Math.floor(o.x), o.y = Math.floor(o.y); const d = o.x * 4 + o.y * (s * 4); n[d] = r; } const a = this._createTileBuffer(n); this._tileMaps[e].dispose(), this._tileMaps[e] = a, this._material.setTextureArray("tileMap", this._tileMaps); } /** * Creates the animationMap texture Buffer * @param buffer normally and array of numbers, or a false to generate from scratch * @returns RawTexture of the animationMap */ _createTileAnimationBuffer(e) { const t = []; let r; if (e) r = e; else { for (let i = 0; i < this.spriteCount; i++) { t.push(0, 0, 0, 0); let s = 1; for (; s < (this.options.maxAnimationFrames || 4); ) t.push(0, 0, 0, 0), s++; } r = new Float32Array(t); } return Bo.CreateRGBATexture(r, this.spriteCount, this.options.maxAnimationFrames || 4, this._scene, !1, !1, We.NEAREST_NEAREST, Ge.TEXTURETYPE_FLOAT); } /** * Modifies the data of the animationMap * @param cellID is the Index of the Sprite * @param _frame is the target Animation frame * @param toCell is the Target Index of the next frame of the animation * @param time is a value between 0-1 that is the trigger for when the frame should change tiles * @param speed is a global scalar of the time variable on the map. */ addAnimationToTile(e = 0, t = 0, r = 0, n = 0, i = 1) { const s = this._animationMap._texture._bufferView, a = e * 4 + this.spriteCount * 4 * t; if (!s) return; s[a] = r, s[a + 1] = n, s[a + 2] = i; const f = this._createTileAnimationBuffer(s); this._animationMap.dispose(), this._animationMap = f, this._material.setTexture("animationMap", this._animationMap); } /** * Exports the .tilemaps file */ saveTileMaps() { let e = ""; for (let r = 0; r < this._tileMaps.length; r++) r > 0 && (e += ` \r`), e += this._tileMaps[r]._texture._bufferView.toString(); const t = document.createElement("a"); t.href = "data:octet/stream;charset=utf-8," + encodeURI(e), t.target = "_blank", t.download = this.name + ".tilemaps", t.click(), t.remove(); } /** * Imports the .tilemaps file * @param url of the .tilemaps file */ loadTileMaps(e) { const t = new XMLHttpRequest(); t.open("GET", e); const r = this.options.layerCount || 0; t.onload = () => { const n = t.response.split(` \r`); for (let i = 0; i < r; i++) { const s = n[i].split(",").map(Number), a = this._createTileBuffer(s); this._tileMaps[i].dispose(), this._tileMaps[i] = a; } this._material.setTextureArray("tileMap", this._tileMaps); }, t.send(); } /** * Release associated resources */ dispose() { this._output.dispose(), this._material.dispose(), this._animationMap.dispose(), this._tileMaps.forEach((e) => { e.dispose(); }), this._frameMap.dispose(); } } class ige extends z2 { /** * Creates a new sprite manager from a packed sprite sheet * @param name defines the manager's name * @param imgUrl defines the sprite sheet url * @param capacity defines the maximum allowed number of sprites * @param scene defines the hosting scene * @param spriteJSON null otherwise a JSON object defining sprite sheet data * @param epsilon defines the epsilon value to align texture (0.01 by default) * @param samplingMode defines the sampling mode to use with spritesheet * @param fromPacked set to true; do not alter */ constructor(e, t, r, n, i = null, s = 0.01, a = We.TRILINEAR_SAMPLINGMODE) { super(e, t, r, 64, n, s, a, !0, i), this.name = e; } } var Y2; (function(A) { A[A.INIT = 0] = "INIT", A[A.RUNNING = 1] = "RUNNING", A[A.DONE = 2] = "DONE", A[A.ERROR = 3] = "ERROR"; })(Y2 || (Y2 = {})); class pg { /** * Creates a new AssetsManager * @param name defines the name of the task */ constructor(e) { this.name = e, this._isCompleted = !1, this._taskState = Y2.INIT; } /** * Get if the task is completed */ get isCompleted() { return this._isCompleted; } /** * Gets the current state of the task */ get taskState() { return this._taskState; } /** * Gets the current error object (if task is in error) */ get errorObject() { return this._errorObject; } /** * Internal only * @internal */ _setErrorObject(e, t) { this._errorObject || (this._errorObject = { message: e, exception: t }); } /** * Execute the current task * @param scene defines the scene where you want your assets to be loaded * @param onSuccess is a callback called when the task is successfully executed * @param onError is a callback called if an error occurs */ run(e, t, r) { this._taskState = Y2.RUNNING, this.runTask(e, () => { this._onDoneCallback(t, r); }, (n, i) => { this._onErrorCallback(r, n, i); }); } /** * Execute the current task * @param scene defines the scene where you want your assets to be loaded * @param onSuccess is a callback called when the task is successfully executed * @param onError is a callback called if an error occurs */ // eslint-disable-next-line @typescript-eslint/no-unused-vars runTask(e, t, r) { throw new Error("runTask is not implemented"); } /** * Reset will set the task state back to INIT, so the next load call of the assets manager will execute this task again. * This can be used with failed tasks that have the reason for failure fixed. */ reset() { this._taskState = Y2.INIT; } _onErrorCallback(e, t, r) { this._taskState = Y2.ERROR, this._errorObject = { message: t, exception: r }, this.onError && this.onError(this, t, r), e(); } _onDoneCallback(e, t) { try { this._taskState = Y2.DONE, this._isCompleted = !0, this.onSuccess && this.onSuccess(this), e(); } catch (r) { this._onErrorCallback(t, "Task is done, error executing success callback(s)", r); } } } class pie { /** * Creates a AssetsProgressEvent * @param remainingCount defines the number of remaining tasks to process * @param totalCount defines the total number of tasks * @param task defines the task that was just processed */ constructor(e, t, r) { this.remainingCount = e, this.totalCount = t, this.task = r; } } class hie extends pg { /** * Creates a new ContainerAssetTask * @param name defines the name of the task * @param meshesNames defines the list of mesh's names you want to load * @param rootUrl defines the root url to use as a base to load your meshes and associated resources * @param sceneFilename defines the filename or File of the scene to load from */ constructor(e, t, r, n, i) { super(e), this.name = e, this.meshesNames = t, this.rootUrl = r, this.sceneFilename = n, this.extension = i; } /** * Execute the current task * @param scene defines the scene where you want your assets to be loaded * @param onSuccess is a callback called when the task is successfully executed * @param onError is a callback called if an error occurs */ runTask(e, t, r) { Hn.LoadAssetContainer(this.rootUrl, this.sceneFilename, e, (n) => { this.loadedContainer = n, this.loadedMeshes = n.meshes, this.loadedTransformNodes = n.transformNodes, this.loadedParticleSystems = n.particleSystems, this.loadedSkeletons = n.skeletons, this.loadedAnimationGroups = n.animationGroups, t(); }, null, (n, i, s) => { r(i, s); }, this.extension); } } class Hie extends pg { /** * Creates a new MeshAssetTask * @param name defines the name of the task * @param meshesNames defines the list of mesh's names you want to load * @param rootUrl defines the root url to use as a base to load your meshes and associated resources * @param sceneFilename defines the filename or File of the scene to load from */ constructor(e, t, r, n, i) { super(e), this.name = e, this.meshesNames = t, this.rootUrl = r, this.sceneFilename = n, this.extension = i; } /** * Execute the current task * @param scene defines the scene where you want your assets to be loaded * @param onSuccess is a callback called when the task is successfully executed * @param onError is a callback called if an error occurs */ runTask(e, t, r) { Hn.ImportMesh(this.meshesNames, this.rootUrl, this.sceneFilename, e, (n, i, s, a, f) => { this.loadedMeshes = n, this.loadedTransformNodes = f, this.loadedParticleSystems = i, this.loadedSkeletons = s, this.loadedAnimationGroups = a, t(); }, null, (n, i, s) => { r(i, s); }, this.extension); } } class sge extends pg { /** * Creates a new AnimationAssetTask * @param name defines the name of the task * @param rootUrl defines the root url to use as a base to load your meshes and associated resources * @param filename defines the filename or File of the scene to load from * @param targetConverter defines a function used to convert animation targets from loaded scene to current scene (default: search node by name) */ constructor(e, t, r, n, i) { super(e), this.name = e, this.rootUrl = t, this.filename = r, this.targetConverter = n, this.extension = i; } /** * Execute the current task * @param scene defines the scene where you want your assets to be loaded * @param onSuccess is a callback called when the task is successfully executed * @param onError is a callback called if an error occurs */ runTask(e, t, r) { const n = e.animatables.length, i = e.animationGroups.length; this.loadedAnimatables = [], this.loadedAnimationGroups = [], Hn.ImportAnimations(this.rootUrl, this.filename, e, !1, O2.NoSync, this.targetConverter, () => { this.loadedAnimatables = e.animatables.slice(n), this.loadedAnimationGroups = e.animationGroups.slice(i), t(); }, null, (s, a, f) => { r(a, f); }, this.extension); } } class gie extends pg { /** * Creates a new TextFileAssetTask object * @param name defines the name of the task * @param url defines the location of the file to load */ constructor(e, t) { super(e), this.name = e, this.url = t; } /** * Execute the current task * @param scene defines the scene where you want your assets to be loaded * @param onSuccess is a callback called when the task is successfully executed * @param onError is a callback called if an error occurs */ runTask(e, t, r) { e._loadFile(this.url, (n) => { this.text = n, t(); }, void 0, !1, !1, (n, i) => { n && r(n.status + " " + n.statusText, i); }); } } class Xie extends pg { /** * Creates a new BinaryFileAssetTask object * @param name defines the name of the new task * @param url defines the location of the file to load */ constructor(e, t) { super(e), this.name = e, this.url = t; } /** * Execute the current task * @param scene defines the scene where you want your assets to be loaded * @param onSuccess is a callback called when the task is successfully executed * @param onError is a callback called if an error occurs */ runTask(e, t, r) { e._loadFile(this.url, (n) => { this.data = n, t(); }, void 0, !0, !0, (n, i) => { n && r(n.status + " " + n.statusText, i); }); } } class Tie extends pg { /** * Creates a new ImageAssetTask * @param name defines the name of the task * @param url defines the location of the image to load */ constructor(e, t) { super(e), this.name = e, this.url = t; } /** * Execute the current task * @param scene defines the scene where you want your assets to be loaded * @param onSuccess is a callback called when the task is successfully executed * @param onError is a callback called if an error occurs */ runTask(e, t, r) { const n = new Image(); ye.SetCorsBehavior(this.url, n), n.onload = () => { this.image = n, t(); }, n.onerror = (i) => { r("Error loading image", i); }, n.src = this.url; } } class qie extends pg { /** * Creates a new TextureAssetTask object * @param name defines the name of the task * @param url defines the location of the file to load * @param noMipmap defines if mipmap should not be generated (default is false) * @param invertY defines if texture must be inverted on Y axis (default is true) * @param samplingMode defines the sampling mode to use (default is Texture.TRILINEAR_SAMPLINGMODE) */ constructor(e, t, r, n = !0, i = We.TRILINEAR_SAMPLINGMODE) { super(e), this.name = e, this.url = t, this.noMipmap = r, this.invertY = n, this.samplingMode = i; } /** * Execute the current task * @param scene defines the scene where you want your assets to be loaded * @param onSuccess is a callback called when the task is successfully executed * @param onError is a callback called if an error occurs */ runTask(e, t, r) { const n = () => { t(); }, i = (s, a) => { r(s, a); }; this.texture = new We(this.url, e, this.noMipmap, this.invertY, this.samplingMode, n, i); } } class bie extends pg { /** * Creates a new CubeTextureAssetTask * @param name defines the name of the task * @param url defines the location of the files to load (You have to specify the folder where the files are + filename with no extension) * @param extensions defines the extensions to use to load files (["_px", "_py", "_pz", "_nx", "_ny", "_nz"] by default) * @param noMipmap defines if mipmaps should not be generated (default is false) * @param files defines the explicit list of files (undefined by default) * @param prefiltered */ constructor(e, t, r, n, i, s) { super(e), this.name = e, this.url = t, this.extensions = r, this.noMipmap = n, this.files = i, this.prefiltered = s; } /** * Execute the current task * @param scene defines the scene where you want your assets to be loaded * @param onSuccess is a callback called when the task is successfully executed * @param onError is a callback called if an error occurs */ runTask(e, t, r) { const n = () => { t(); }, i = (s, a) => { r(s, a); }; this.texture = new v1(this.url, e, this.extensions, this.noMipmap, this.files, n, i, void 0, this.prefiltered); } } class xie extends pg { /** * Creates a new HDRCubeTextureAssetTask object * @param name defines the name of the task * @param url defines the location of the file to load * @param size defines the desired size (the more it increases the longer the generation will be) If the size is omitted this implies you are using a preprocessed cubemap. * @param noMipmap defines if mipmaps should not be generated (default is false) * @param generateHarmonics specifies whether you want to extract the polynomial harmonics during the generation process (default is true) * @param gammaSpace specifies if the texture will be use in gamma or linear space (the PBR material requires those texture in linear space, but the standard material would require them in Gamma space) (default is false) * @param reserved Internal use only */ constructor(e, t, r, n = !1, i = !0, s = !1, a = !1) { super(e), this.name = e, this.url = t, this.size = r, this.noMipmap = n, this.generateHarmonics = i, this.gammaSpace = s, this.reserved = a; } /** * Execute the current task * @param scene defines the scene where you want your assets to be loaded * @param onSuccess is a callback called when the task is successfully executed * @param onError is a callback called if an error occurs */ runTask(e, t, r) { const n = () => { t(); }, i = (s, a) => { r(s, a); }; this.texture = new _2(this.url, e, this.size, this.noMipmap, this.generateHarmonics, this.gammaSpace, this.reserved, n, i); } } class Die extends pg { /** * Creates a new EquiRectangularCubeTextureAssetTask object * @param name defines the name of the task * @param url defines the location of the file to load * @param size defines the desired size (the more it increases the longer the generation will be) * If the size is omitted this implies you are using a preprocessed cubemap. * @param noMipmap defines if mipmaps should not be generated (default is false) * @param gammaSpace specifies if the texture will be used in gamma or linear space * (the PBR material requires those texture in linear space, but the standard material would require them in Gamma space) * (default is true) */ constructor(e, t, r, n = !1, i = !0) { super(e), this.name = e, this.url = t, this.size = r, this.noMipmap = n, this.gammaSpace = i; } /** * Execute the current task * @param scene defines the scene where you want your assets to be loaded * @param onSuccess is a callback called when the task is successfully executed * @param onError is a callback called if an error occurs */ runTask(e, t, r) { const n = () => { t(); }, i = (s, a) => { r(s, a); }; this.texture = new VS(this.url, e, this.size, this.noMipmap, this.gammaSpace, n, i); } } class age { /** * Creates a new AssetsManager * @param scene defines the scene to work on */ constructor(e) { this._isLoading = !1, this._tasks = new Array(), this._waitingTasksCount = 0, this._totalTasksCount = 0, this.onTaskSuccessObservable = new Oe(), this.onTaskErrorObservable = new Oe(), this.onTasksDoneObservable = new Oe(), this.onProgressObservable = new Oe(), this.useDefaultLoadingScreen = !0, this.autoHideLoadingUI = !0, this._scene = e || gr.LastCreatedScene; } /** * Add a ContainerAssetTask to the list of active tasks * @param taskName defines the name of the new task * @param meshesNames defines the name of meshes to load * @param rootUrl defines the root url to use to locate files * @param sceneFilename defines the filename of the scene file or the File itself * @param extension defines the extension to use to load the file * @returns a new ContainerAssetTask object */ addContainerTask(e, t, r, n, i) { const s = new hie(e, t, r, n, i); return this._tasks.push(s), s; } /** * Add a MeshAssetTask to the list of active tasks * @param taskName defines the name of the new task * @param meshesNames defines the name of meshes to load * @param rootUrl defines the root url to use to locate files * @param sceneFilename defines the filename of the scene file or the File itself * @param extension defines the extension to use to load the file * @returns a new MeshAssetTask object */ addMeshTask(e, t, r, n, i) { const s = new Hie(e, t, r, n, i); return this._tasks.push(s), s; } /** * Add a TextFileAssetTask to the list of active tasks * @param taskName defines the name of the new task * @param url defines the url of the file to load * @returns a new TextFileAssetTask object */ addTextFileTask(e, t) { const r = new gie(e, t); return this._tasks.push(r), r; } /** * Add a BinaryFileAssetTask to the list of active tasks * @param taskName defines the name of the new task * @param url defines the url of the file to load * @returns a new BinaryFileAssetTask object */ addBinaryFileTask(e, t) { const r = new Xie(e, t); return this._tasks.push(r), r; } /** * Add a ImageAssetTask to the list of active tasks * @param taskName defines the name of the new task * @param url defines the url of the file to load * @returns a new ImageAssetTask object */ addImageTask(e, t) { const r = new Tie(e, t); return this._tasks.push(r), r; } /** * Add a TextureAssetTask to the list of active tasks * @param taskName defines the name of the new task * @param url defines the url of the file to load * @param noMipmap defines if the texture must not receive mipmaps (false by default) * @param invertY defines if you want to invert Y axis of the loaded texture (true by default) * @param samplingMode defines the sampling mode to use (Texture.TRILINEAR_SAMPLINGMODE by default) * @returns a new TextureAssetTask object */ addTextureTask(e, t, r, n, i = We.TRILINEAR_SAMPLINGMODE) { const s = new qie(e, t, r, n, i); return this._tasks.push(s), s; } /** * Add a CubeTextureAssetTask to the list of active tasks * @param taskName defines the name of the new task * @param url defines the url of the file to load * @param extensions defines the extension to use to load the cube map (can be null) * @param noMipmap defines if the texture must not receive mipmaps (false by default) * @param files defines the list of files to load (can be null) * @param prefiltered defines the prefiltered texture option (default is false) * @returns a new CubeTextureAssetTask object */ addCubeTextureTask(e, t, r, n, i, s) { const a = new bie(e, t, r, n, i, s); return this._tasks.push(a), a; } /** * * Add a HDRCubeTextureAssetTask to the list of active tasks * @param taskName defines the name of the new task * @param url defines the url of the file to load * @param size defines the size you want for the cubemap (can be null) * @param noMipmap defines if the texture must not receive mipmaps (false by default) * @param generateHarmonics defines if you want to automatically generate (true by default) * @param gammaSpace specifies if the texture will be use in gamma or linear space (the PBR material requires those texture in linear space, but the standard material would require them in Gamma space) (default is false) * @param reserved Internal use only * @returns a new HDRCubeTextureAssetTask object */ addHDRCubeTextureTask(e, t, r, n = !1, i = !0, s = !1, a = !1) { const f = new xie(e, t, r, n, i, s, a); return this._tasks.push(f), f; } /** * * Add a EquiRectangularCubeTextureAssetTask to the list of active tasks * @param taskName defines the name of the new task * @param url defines the url of the file to load * @param size defines the size you want for the cubemap (can be null) * @param noMipmap defines if the texture must not receive mipmaps (false by default) * @param gammaSpace Specifies if the texture will be used in gamma or linear space * (the PBR material requires those textures in linear space, but the standard material would require them in Gamma space) * @returns a new EquiRectangularCubeTextureAssetTask object */ addEquiRectangularCubeTextureAssetTask(e, t, r, n = !1, i = !0) { const s = new Die(e, t, r, n, i); return this._tasks.push(s), s; } /** * Remove a task from the assets manager. * @param task the task to remove */ removeTask(e) { const t = this._tasks.indexOf(e); t > -1 && this._tasks.splice(t, 1); } _decreaseWaitingTasksCount(e) { this._waitingTasksCount--; try { this.onProgress && this.onProgress(this._waitingTasksCount, this._totalTasksCount, e), this.onProgressObservable.notifyObservers(new pie(this._waitingTasksCount, this._totalTasksCount, e)); } catch (t) { Se.Error("Error running progress callbacks."), console.log(t); } if (this._waitingTasksCount === 0) { try { const t = this._tasks.slice(); this.onFinish && this.onFinish(t); for (const r of t) if (r.taskState === Y2.DONE) { const n = this._tasks.indexOf(r); n > -1 && this._tasks.splice(n, 1); } this.onTasksDoneObservable.notifyObservers(this._tasks); } catch (t) { Se.Error("Error running tasks-done callbacks."), console.log(t); } this._isLoading = !1, this.autoHideLoadingUI && this._scene.getEngine().hideLoadingUI(); } } _runTask(e) { const t = () => { try { this.onTaskSuccess && this.onTaskSuccess(e), this.onTaskSuccessObservable.notifyObservers(e), this._decreaseWaitingTasksCount(e); } catch (n) { r("Error executing task success callbacks", n); } }, r = (n, i) => { e._setErrorObject(n, i), this.onTaskError ? this.onTaskError(e) : e.onError || Se.Error(this._formatTaskErrorMessage(e)), this.onTaskErrorObservable.notifyObservers(e), this._decreaseWaitingTasksCount(e); }; e.run(this._scene, t, r); } _formatTaskErrorMessage(e) { let t = "Unable to complete task " + e.name; return e.errorObject.message && (t += `: ${e.errorObject.message}`), e.errorObject.exception && (t += `: ${e.errorObject.exception}`), t; } /** * Reset the AssetsManager and remove all tasks * @returns the current instance of the AssetsManager */ reset() { return this._isLoading = !1, this._tasks = new Array(), this; } /** * Start the loading process * @returns the current instance of the AssetsManager */ load() { if (this._isLoading) return this; if (this._isLoading = !0, this._waitingTasksCount = this._tasks.length, this._totalTasksCount = this._tasks.length, this._waitingTasksCount === 0) return this._isLoading = !1, this.onFinish && this.onFinish(this._tasks), this.onTasksDoneObservable.notifyObservers(this._tasks), this; this.useDefaultLoadingScreen && this._scene.getEngine().displayLoadingUI(); for (let e = 0; e < this._tasks.length; e++) { const t = this._tasks[e]; t.taskState === Y2.INIT && this._runTask(t); } return this; } /** * Start the loading process as an async operation * @returns a promise returning the list of failed tasks */ loadAsync() { return new Promise((e, t) => { if (this._isLoading) { e(); return; } this.onTasksDoneObservable.addOnce((r) => { r && r.length ? t(r) : e(); }), this.load(); }); } } class MW { /** * The resolve method of the promise associated with this deferred object. */ get resolve() { return this._resolve; } /** * The reject method of the promise associated with this deferred object. */ get reject() { return this._reject; } /** * Constructor for this deferred object. */ constructor() { this.promise = new Promise((e, t) => { this._resolve = e, this._reject = t; }); } } class oge { /** * Explodes meshes from a center mesh. * @param meshes The meshes to explode. * @param centerMesh The mesh to be center of explosion. */ constructor(e, t) { this._meshesOrigins = [], this._toCenterVectors = [], this._scaledDirection = new S(1, 1, 1), this._newPosition = S.Zero(), this._centerPosition = S.Zero(), this._meshes = e.slice(), t ? this._centerMesh = t : this._setCenterMesh(), this._centerMesh.computeWorldMatrix(!0); const r = this._meshes.indexOf(this._centerMesh); r >= 0 && this._meshes.splice(r, 1), this._centerPosition = this._centerMesh.getAbsolutePosition().clone(); for (let n = 0; n < this._meshes.length; n++) if (this._meshes[n]) { const i = this._meshes[n]; this._meshesOrigins[n] = i.getAbsolutePosition().clone(), this._toCenterVectors[n] = S.Zero(), i.hasBoundingInfo && this._centerMesh.hasBoundingInfo && (i.computeWorldMatrix(!0), i.getBoundingInfo().boundingBox.centerWorld.subtractToRef(this._centerMesh.getBoundingInfo().boundingBox.centerWorld, this._toCenterVectors[n])); } } _setCenterMesh() { let e = S.Zero(); const t = S.Zero(); let r = Number.MAX_VALUE; for (let n = 0; n < this._meshes.length; n++) if (this._meshes[n]) { const s = this._meshes[n].getBoundingInfo(); s && t.addInPlace(s.boundingBox.centerWorld); } e = t.scale(1 / this._meshes.length); for (let n = 0; n < this._meshes.length; n++) if (this._meshes[n]) { const i = this._meshes[n], s = i.getBoundingInfo(); if (s) { const a = s.boundingBox.centerWorld.subtract(e).lengthSquared(); a < r && (this._centerMesh = i, r = a); } } } /** * Get class name * @returns "MeshExploder" */ getClassName() { return "MeshExploder"; } /** * "Exploded meshes" * @returns Array of meshes with the centerMesh at index 0. */ getMeshes() { const e = this._meshes.slice(); return e.unshift(this._centerMesh), e; } /** * Explodes meshes giving a specific direction * @param direction Number to multiply distance of each mesh's origin from center. Use a negative number to implode, or zero to reset. */ explode(e = 1) { for (let t = 0; t < this._meshes.length; t++) this._meshes[t] && this._meshesOrigins[t] && this._toCenterVectors[t] && (this._toCenterVectors[t].scaleToRef(e, this._scaledDirection), this._meshesOrigins[t].addToRef(this._scaledDirection, this._newPosition), this._meshes[t].setAbsolutePosition(this._newPosition)); this._centerMesh.setAbsolutePosition(this._centerPosition); } } class KY { /** * List of files ready to be loaded */ static get FilesToLoad() { return em.FilesToLoad; } /** * Creates a new FilesInput * @param engine defines the rendering engine * @param scene defines the hosting scene * @param sceneLoadedCallback callback called when scene (files provided) is loaded * @param progressCallback callback called to track progress * @param additionalRenderLoopLogicCallback callback called to add user logic to the rendering loop * @param textureLoadingCallback callback called when a texture is loading * @param startingProcessingFilesCallback callback called when the system is about to process all files * @param onReloadCallback callback called when a reload is requested * @param errorCallback callback call if an error occurs * @param useAppend defines if the file loaded must be appended (true) or have the scene replaced (false, default behavior) */ constructor(e, t, r, n, i, s, a, f, o, d = !1) { this.useAppend = d, this.onProcessFileCallback = () => !0, this.displayLoadingUI = !0, this.loadAsync = (v, u) => this.useAppend ? Hn.AppendAsync("file:", v, this._currentScene, u) : Hn.LoadAsync("file:", v, this._engine, u), this._engine = e, this._currentScene = t, this._sceneLoadedCallback = r, this._progressCallback = n, this._additionalRenderLoopLogicCallback = i, this._textureLoadingCallback = s, this._startingProcessingFilesCallback = a, this._onReloadCallback = f, this._errorCallback = o; } /** * Calls this function to listen to drag'n'drop events on a specific DOM element * @param elementToMonitor defines the DOM element to track */ monitorElementForDragNDrop(e) { e && (this._elementToMonitor = e, this._dragEnterHandler = (t) => { this._drag(t); }, this._dragOverHandler = (t) => { this._drag(t); }, this._dropHandler = (t) => { this._drop(t); }, this._elementToMonitor.addEventListener("dragenter", this._dragEnterHandler, !1), this._elementToMonitor.addEventListener("dragover", this._dragOverHandler, !1), this._elementToMonitor.addEventListener("drop", this._dropHandler, !1)); } /** Gets the current list of files to load */ get filesToLoad() { return this._filesToLoad; } /** * Release all associated resources */ dispose() { this._elementToMonitor && (this._elementToMonitor.removeEventListener("dragenter", this._dragEnterHandler), this._elementToMonitor.removeEventListener("dragover", this._dragOverHandler), this._elementToMonitor.removeEventListener("drop", this._dropHandler)); } _renderFunction() { if (this._additionalRenderLoopLogicCallback && this._additionalRenderLoopLogicCallback(), this._currentScene) { if (this._textureLoadingCallback) { const e = this._currentScene.getWaitingItemsCount(); e > 0 && this._textureLoadingCallback(e); } this._currentScene.render(); } } _drag(e) { e.stopPropagation(), e.preventDefault(); } _drop(e) { e.stopPropagation(), e.preventDefault(), this.loadFiles(e); } _traverseFolder(e, t, r, n) { const i = e.createReader(), s = e.fullPath.replace(/^\//, "").replace(/(.+?)\/?$/, "$1/"); i.readEntries((a) => { r.count += a.length; for (const f of a) f.isFile ? f.file((o) => { o.correctName = s + o.name, t.push(o), --r.count === 0 && n(); }) : f.isDirectory && this._traverseFolder(f, t, r, n); --r.count === 0 && n(); }); } _processFiles(e) { for (let t = 0; t < e.length; t++) { const r = e[t].correctName.toLowerCase(), n = r.split(".").pop(); this.onProcessFileCallback(e[t], r, n, (i) => this._sceneFileToLoad = i) && (Hn.IsPluginForExtensionAvailable("." + n) && (this._sceneFileToLoad = e[t]), KY.FilesToLoad[r] = e[t]); } } /** * Load files from a drop event * @param event defines the drop event to use as source */ loadFiles(e) { if (e && e.dataTransfer && e.dataTransfer.files && (this._filesToLoad = e.dataTransfer.files), e && e.target && e.target.files && (this._filesToLoad = e.target.files), !(!this._filesToLoad || this._filesToLoad.length === 0) && (this._startingProcessingFilesCallback && this._startingProcessingFilesCallback(this._filesToLoad), this._filesToLoad && this._filesToLoad.length > 0)) { const t = [], r = [], n = e.dataTransfer ? e.dataTransfer.items : null; for (let i = 0; i < this._filesToLoad.length; i++) { const s = this._filesToLoad[i], a = s.name.toLowerCase(); let f; if (s.correctName = a, n) { const o = n[i]; o.getAsEntry ? f = o.getAsEntry() : o.webkitGetAsEntry && (f = o.webkitGetAsEntry()); } f && f.isDirectory ? r.push(f) : t.push(s); } if (r.length === 0) this._processFiles(t), this._processReload(); else { const i = { count: r.length }; for (const s of r) this._traverseFolder(s, t, i, () => { this._processFiles(t), i.count === 0 && this._processReload(); }); } } } _processReload() { this._onReloadCallback ? this._onReloadCallback(this._sceneFileToLoad) : this.reload(); } /** * Reload the current scene from the loaded files */ reload() { this._sceneFileToLoad ? (this.useAppend || this._currentScene && (Se.errorsCount > 0 && Se.ClearLogCache(), this._engine.stopRenderLoop()), Hn.ShowLoadingScreen = !1, this.displayLoadingUI && this._engine.displayLoadingUI(), this.loadAsync(this._sceneFileToLoad, this._progressCallback).then((e) => { this.useAppend ? this.displayLoadingUI && this._engine.hideLoadingUI() : (this._currentScene && this._currentScene.dispose(), this._currentScene = e, this._currentScene.executeWhenReady(() => { this.displayLoadingUI && this._engine.hideLoadingUI(), this._engine.runRenderLoop(() => { this._renderFunction(); }); })), this._sceneLoadedCallback && this._currentScene && this._sceneLoadedCallback(this._sceneFileToLoad, this._currentScene); }).catch((e) => { this.displayLoadingUI && this._engine.hideLoadingUI(), this._errorCallback && this._errorCallback(this._sceneFileToLoad, this._currentScene, e.message); })) : Se.Error("Please provide a valid .babylon file."); } } class JY { /** * Release associated resources */ dispose() { if (this._observers && this._observables) for (let e = 0; e < this._observers.length; e++) this._observables[e].remove(this._observers[e]); this._observers = null, this._observables = null; } /** * Raise a callback when one of the observable will notify * @param observables defines a list of observables to watch * @param callback defines the callback to call on notification * @param mask defines the mask used to filter notifications * @param scope defines the current scope used to restore the JS context * @returns the new MultiObserver */ static Watch(e, t, r = -1, n = null) { const i = new JY(); i._observers = new Array(), i._observables = e; for (const s of e) { const a = s.add(t, r, !1, n); a && i._observers.push(a); } return i; } } Oe.prototype.notifyObserversWithPromise = async function(A, e = -1, t, r, n) { let i = Promise.resolve(A); if (!this.observers.length) return i; const s = this._eventState; return s.mask = e, s.target = t, s.currentTarget = r, s.skipNextObservers = !1, s.userInfo = n, this.observers.forEach((a) => { s.skipNextObservers || a._willBeUnregistered || a.mask & e && (a.scope ? i = i.then((f) => (s.lastReturnValue = f, a.callback.apply(a.scope, [A, s]))) : i = i.then((f) => (s.lastReturnValue = f, a.callback(A, s))), a.unregisterOnNextCall && this._deferUnregister(a)); }), await i, A; }; class c4 { /** * Gets a string describing the action executed by the current optimization * @returns description string */ getDescription() { return ""; } /** * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization * @param scene defines the current scene where to apply this optimization * @param optimizer defines the current optimizer * @returns true if everything that can be done was applied */ apply(e, t) { return !0; } /** * Creates the SceneOptimization object * @param priority defines the priority of this optimization (0 by default which means first in the list) */ constructor(e = 0) { this.priority = e; } } class SC extends c4 { /** * Gets a string describing the action executed by the current optimization * @returns description string */ getDescription() { return "Reducing render target texture size to " + this.maximumSize; } /** * Creates the TextureOptimization object * @param priority defines the priority of this optimization (0 by default which means first in the list) * @param maximumSize defines the maximum sized allowed for textures (1024 is the default value). If a texture is bigger, it will be scaled down using a factor defined by the step parameter * @param step defines the factor (0.5 by default) used to scale down textures bigger than maximum sized allowed. */ constructor(e = 0, t = 1024, r = 0.5) { super(e), this.priority = e, this.maximumSize = t, this.step = r; } /** * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization * @param scene defines the current scene where to apply this optimization * @param optimizer defines the current optimizer * @returns true if everything that can be done was applied */ apply(e, t) { let r = !0; for (let n = 0; n < e.textures.length; n++) { const i = e.textures[n]; if (!i.canRescale || i.getContext) continue; const s = i.getSize(); Math.max(s.width, s.height) > this.maximumSize && (i.scale(this.step), r = !1); } return r; } } class mF extends c4 { /** * Gets a string describing the action executed by the current optimization * @returns description string */ getDescription() { return "Setting hardware scaling level to " + this._currentScale; } /** * Creates the HardwareScalingOptimization object * @param priority defines the priority of this optimization (0 by default which means first in the list) * @param maximumScale defines the maximum scale to use (2 by default) * @param step defines the step to use between two passes (0.5 by default) */ constructor(e = 0, t = 2, r = 0.25) { super(e), this.priority = e, this.maximumScale = t, this.step = r, this._currentScale = -1, this._directionOffset = 1; } /** * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization * @param scene defines the current scene where to apply this optimization * @param optimizer defines the current optimizer * @returns true if everything that can be done was applied */ apply(e, t) { return this._currentScale === -1 && (this._currentScale = e.getEngine().getHardwareScalingLevel(), this._currentScale > this.maximumScale && (this._directionOffset = -1)), this._currentScale += this._directionOffset * this.step, e.getEngine().setHardwareScalingLevel(this._currentScale), this._directionOffset === 1 ? this._currentScale >= this.maximumScale : this._currentScale <= this.maximumScale; } } class UC extends c4 { /** * Gets a string describing the action executed by the current optimization * @returns description string */ getDescription() { return "Turning shadows on/off"; } /** * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization * @param scene defines the current scene where to apply this optimization * @param optimizer defines the current optimizer * @returns true if everything that can be done was applied */ apply(e, t) { return e.shadowsEnabled = t.isInImprovementMode, !0; } } class IC extends c4 { /** * Gets a string describing the action executed by the current optimization * @returns description string */ getDescription() { return "Turning post-processes on/off"; } /** * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization * @param scene defines the current scene where to apply this optimization * @param optimizer defines the current optimizer * @returns true if everything that can be done was applied */ apply(e, t) { return e.postProcessesEnabled = t.isInImprovementMode, !0; } } class RC extends c4 { /** * Gets a string describing the action executed by the current optimization * @returns description string */ getDescription() { return "Turning lens flares on/off"; } /** * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization * @param scene defines the current scene where to apply this optimization * @param optimizer defines the current optimizer * @returns true if everything that can be done was applied */ apply(e, t) { return e.lensFlaresEnabled = t.isInImprovementMode, !0; } } class jie extends c4 { /** * Gets a string describing the action executed by the current optimization * @returns description string */ getDescription() { return this.onGetDescription ? this.onGetDescription() : "Running user defined callback"; } /** * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization * @param scene defines the current scene where to apply this optimization * @param optimizer defines the current optimizer * @returns true if everything that can be done was applied */ apply(e, t) { return this.onApply ? this.onApply(e, t) : !0; } } class VC extends c4 { /** * Gets a string describing the action executed by the current optimization * @returns description string */ getDescription() { return "Turning particles on/off"; } /** * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization * @param scene defines the current scene where to apply this optimization * @param optimizer defines the current optimizer * @returns true if everything that can be done was applied */ apply(e, t) { return e.particlesEnabled = t.isInImprovementMode, !0; } } class BF extends c4 { /** * Gets a string describing the action executed by the current optimization * @returns description string */ getDescription() { return "Turning render targets off"; } /** * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization * @param scene defines the current scene where to apply this optimization * @param optimizer defines the current optimizer * @returns true if everything that can be done was applied */ apply(e, t) { return e.renderTargetsEnabled = t.isInImprovementMode, !0; } } class uq extends c4 { constructor() { super(...arguments), this._canBeMerged = (e) => { if (!(e instanceof Ee)) return !1; const t = e; return !(t.isDisposed() || !t.isVisible || !t.isEnabled() || t.instances.length > 0 || t.skeleton || t.hasLODLevels || t.getTotalVertices() === 0); }; } /** * Gets or sets a boolean which defines if optimization octree has to be updated */ static get UpdateSelectionTree() { return uq._UpdateSelectionTree; } /** * Gets or sets a boolean which defines if optimization octree has to be updated */ static set UpdateSelectionTree(e) { uq._UpdateSelectionTree = e; } /** * Gets a string describing the action executed by the current optimization * @returns description string */ getDescription() { return "Merging similar meshes together"; } /** * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization * @param scene defines the current scene where to apply this optimization * @param optimizer defines the current optimizer * @param updateSelectionTree defines that the selection octree has to be updated (false by default) * @returns true if everything that can be done was applied */ apply(e, t, r) { const n = e.meshes.slice(0); let i = n.length; for (let a = 0; a < i; a++) { const f = [], o = n[a]; if (this._canBeMerged(o)) { f.push(o); for (let d = a + 1; d < i; d++) { const v = n[d]; this._canBeMerged(v) && v.material === o.material && v.checkCollisions === o.checkCollisions && (f.push(v), i--, n.splice(d, 1), d--); } f.length < 2 || Ee.MergeMeshes(f, void 0, !0); } } const s = e; return s.createOrUpdateSelectionOctree && (r != null ? r && s.createOrUpdateSelectionOctree() : uq.UpdateSelectionTree && s.createOrUpdateSelectionOctree()), !0; } } uq._UpdateSelectionTree = !1; class om { /** * Creates a new list of options used by SceneOptimizer * @param targetFrameRate defines the target frame rate to reach (60 by default) * @param trackerDuration defines the interval between two checks (2000ms by default) */ constructor(e = 60, t = 2e3) { this.targetFrameRate = e, this.trackerDuration = t, this.optimizations = []; } /** * Add a new optimization * @param optimization defines the SceneOptimization to add to the list of active optimizations * @returns the current SceneOptimizerOptions */ addOptimization(e) { return this.optimizations.push(e), this; } /** * Add a new custom optimization * @param onApply defines the callback called to apply the custom optimization (true if everything that can be done was applied) * @param onGetDescription defines the callback called to get the description attached with the optimization. * @param priority defines the priority of this optimization (0 by default which means first in the list) * @returns the current SceneOptimizerOptions */ addCustomOptimization(e, t, r = 0) { const n = new jie(r); return n.onApply = e, n.onGetDescription = t, this.optimizations.push(n), this; } /** * Creates a list of pre-defined optimizations aimed to reduce the visual impact on the scene * @param targetFrameRate defines the target frame rate (60 by default) * @returns a SceneOptimizerOptions object */ static LowDegradationAllowed(e) { const t = new om(e); let r = 0; return t.addOptimization(new uq(r)), t.addOptimization(new UC(r)), t.addOptimization(new RC(r)), r++, t.addOptimization(new IC(r)), t.addOptimization(new VC(r)), r++, t.addOptimization(new SC(r, 1024)), t; } /** * Creates a list of pre-defined optimizations aimed to have a moderate impact on the scene visual * @param targetFrameRate defines the target frame rate (60 by default) * @returns a SceneOptimizerOptions object */ static ModerateDegradationAllowed(e) { const t = new om(e); let r = 0; return t.addOptimization(new uq(r)), t.addOptimization(new UC(r)), t.addOptimization(new RC(r)), r++, t.addOptimization(new IC(r)), t.addOptimization(new VC(r)), r++, t.addOptimization(new SC(r, 512)), r++, t.addOptimization(new BF(r)), r++, t.addOptimization(new mF(r, 2)), t; } /** * Creates a list of pre-defined optimizations aimed to have a big impact on the scene visual * @param targetFrameRate defines the target frame rate (60 by default) * @returns a SceneOptimizerOptions object */ static HighDegradationAllowed(e) { const t = new om(e); let r = 0; return t.addOptimization(new uq(r)), t.addOptimization(new UC(r)), t.addOptimization(new RC(r)), r++, t.addOptimization(new IC(r)), t.addOptimization(new VC(r)), r++, t.addOptimization(new SC(r, 256)), r++, t.addOptimization(new BF(r)), r++, t.addOptimization(new mF(r, 4)), t; } } class zY { /** * Gets or sets a boolean indicating if the optimizer is in improvement mode */ get isInImprovementMode() { return this._improvementMode; } set isInImprovementMode(e) { this._improvementMode = e; } /** * Gets the current priority level (0 at start) */ get currentPriorityLevel() { return this._currentPriorityLevel; } /** * Gets the current frame rate checked by the SceneOptimizer */ get currentFrameRate() { return this._currentFrameRate; } /** * Gets or sets the current target frame rate (60 by default) */ get targetFrameRate() { return this._targetFrameRate; } /** * Gets or sets the current target frame rate (60 by default) */ set targetFrameRate(e) { this._targetFrameRate = e; } /** * Gets or sets the current interval between two checks (every 2000ms by default) */ get trackerDuration() { return this._trackerDuration; } /** * Gets or sets the current interval between two checks (every 2000ms by default) */ set trackerDuration(e) { this._trackerDuration = e; } /** * Gets the list of active optimizations */ get optimizations() { return this._options.optimizations; } /** * Creates a new SceneOptimizer * @param scene defines the scene to work on * @param options defines the options to use with the SceneOptimizer * @param autoGeneratePriorities defines if priorities must be generated and not read from SceneOptimization property (true by default) * @param improvementMode defines if the scene optimizer must run the maximum optimization while staying over a target frame instead of trying to reach the target framerate (false by default) */ constructor(e, t, r = !0, n = !1) { if (this._isRunning = !1, this._currentPriorityLevel = 0, this._targetFrameRate = 60, this._trackerDuration = 2e3, this._currentFrameRate = 0, this._improvementMode = !1, this.onSuccessObservable = new Oe(), this.onNewOptimizationAppliedObservable = new Oe(), this.onFailureObservable = new Oe(), t ? this._options = t : this._options = new om(), this._options.targetFrameRate && (this._targetFrameRate = this._options.targetFrameRate), this._options.trackerDuration && (this._trackerDuration = this._options.trackerDuration), r) { let i = 0; for (const s of this._options.optimizations) s.priority = i++; } this._improvementMode = n, this._scene = e || gr.LastCreatedScene, this._sceneDisposeObserver = this._scene.onDisposeObservable.add(() => { this._sceneDisposeObserver = null, this.dispose(); }); } /** * Stops the current optimizer */ stop() { this._isRunning = !1; } /** * Reset the optimizer to initial step (current priority level = 0) */ reset() { this._currentPriorityLevel = 0; } /** * Start the optimizer. By default it will try to reach a specific framerate * but if the optimizer is set with improvementMode === true then it will run all optimization while frame rate is above the target frame rate */ start() { this._isRunning || (this._isRunning = !0, this._scene.executeWhenReady(() => { setTimeout(() => { this._checkCurrentState(); }, this._trackerDuration); })); } _checkCurrentState() { if (!this._isRunning) return; const e = this._scene, t = this._options; if (this._currentFrameRate = Math.round(e.getEngine().getFps()), this._improvementMode && this._currentFrameRate <= this._targetFrameRate || !this._improvementMode && this._currentFrameRate >= this._targetFrameRate) { this._isRunning = !1, this.onSuccessObservable.notifyObservers(this); return; } let r = !0, n = !0; for (let i = 0; i < t.optimizations.length; i++) { const s = t.optimizations[i]; s.priority === this._currentPriorityLevel && (n = !1, r = r && s.apply(e, this), this.onNewOptimizationAppliedObservable.notifyObservers(s)); } if (n) { this._isRunning = !1, this.onFailureObservable.notifyObservers(this); return; } r && this._currentPriorityLevel++, e.executeWhenReady(() => { setTimeout(() => { this._checkCurrentState(); }, this._trackerDuration); }); } /** * Release all resources */ dispose() { this.stop(), this.onSuccessObservable.clear(), this.onFailureObservable.clear(), this.onNewOptimizationAppliedObservable.clear(), this._sceneDisposeObserver && this._scene.onDisposeObservable.remove(this._sceneDisposeObserver); } /** * Helper function to create a SceneOptimizer with one single line of code * @param scene defines the scene to work on * @param options defines the options to use with the SceneOptimizer * @param onSuccess defines a callback to call on success * @param onFailure defines a callback to call on failure * @returns the new SceneOptimizer object */ static OptimizeAsync(e, t, r, n) { const i = new zY(e, t || om.ModerateDegradationAllowed(), !1); return r && i.onSuccessObservable.add(() => { r(); }), n && i.onFailureObservable.add(() => { n(); }), i.start(), i; } } let WF = []; const GY = (A, e) => { A.doNotSerialize || (e.vertexData.push(A.serializeVerticeData()), WF[A.id] = !0); }, wie = (A, e) => { const t = {}, r = A._geometry; return r && (A.getScene().getGeometryById(r.id) || GY(r, e.geometries)), A.serialize && A.serialize(t), t; }, fge = (A, e) => { if (A._isMesh) { const t = A; if (t.delayLoadState === 1 || t.delayLoadState === 0) { const r = (i) => { e.materials = e.materials || [], t.material && !e.materials.some((s) => s.id === t.material.id) && e.materials.push(i.serialize()); }; if (t.material && !t.material.doNotSerialize) if (t.material instanceof Dc) { if (e.multiMaterials = e.multiMaterials || [], !e.multiMaterials.some((i) => i.id === t.material.id)) { e.multiMaterials.push(t.material.serialize()); for (const i of t.material.subMaterials) i && r(i); } } else r(t.material); else t.material || r(t.getScene().defaultMaterial); const n = t._geometry; n && (e.geometries || (e.geometries = {}, e.geometries.boxes = [], e.geometries.spheres = [], e.geometries.cylinders = [], e.geometries.toruses = [], e.geometries.grounds = [], e.geometries.planes = [], e.geometries.torusKnots = [], e.geometries.vertexData = []), GY(n, e.geometries)), t.skeleton && !t.skeleton.doNotSerialize && (e.skeletons = e.skeletons || [], e.skeletons.push(t.skeleton.serialize())), e.meshes = e.meshes || [], e.meshes.push(wie(t, e)); } } else if (A.getClassName() === "TransformNode") { const t = A; e.transformNodes.push(t.serialize()); } else if (A.getClassName().indexOf("Camera") !== -1) { const t = A; e.cameras.push(t.serialize()); } else if (A.getClassName().indexOf("Light") !== -1) { const t = A; e.lights.push(t.serialize()); } }; class G2 { /** * Clear cache used by a previous serialization */ static ClearCache() { WF = []; } /** * Serialize a scene into a JSON compatible object * Note that if the current engine does not support synchronous texture reading (like WebGPU), you should use SerializeAsync instead * as else you may not retrieve the proper base64 encoded texture data (when using the Texture.ForceSerializeBuffers flag) * @param scene defines the scene to serialize * @returns a JSON compatible object */ static Serialize(e) { return G2._Serialize(e); } static _Serialize(e, t = !0) { const r = {}; if (t && !e.getEngine()._features.supportSyncTextureRead && We.ForceSerializeBuffers && console.warn("The serialization object may not contain the proper base64 encoded texture data! You should use the SerializeAsync method instead."), G2.ClearCache(), r.useDelayedTextureLoading = e.useDelayedTextureLoading, r.autoClear = e.autoClear, r.clearColor = e.clearColor.asArray(), r.ambientColor = e.ambientColor.asArray(), r.gravity = e.gravity.asArray(), r.collisionsEnabled = e.collisionsEnabled, r.useRightHandedSystem = e.useRightHandedSystem, e.fogMode && e.fogMode !== 0 && (r.fogMode = e.fogMode, r.fogColor = e.fogColor.asArray(), r.fogStart = e.fogStart, r.fogEnd = e.fogEnd, r.fogDensity = e.fogDensity), e.isPhysicsEnabled && e.isPhysicsEnabled()) { const f = e.getPhysicsEngine(); f && (r.physicsEnabled = !0, r.physicsGravity = f.gravity.asArray(), r.physicsEngine = f.getPhysicsPluginName()); } e.metadata && (r.metadata = e.metadata), r.morphTargetManagers = []; for (const f of e.meshes) { const o = f.morphTargetManager; o && r.morphTargetManagers.push(o.serialize()); } r.lights = []; let n, i; for (n = 0; n < e.lights.length; n++) i = e.lights[n], i.doNotSerialize || r.lights.push(i.serialize()); for (r.cameras = [], n = 0; n < e.cameras.length; n++) { const f = e.cameras[n]; f.doNotSerialize || r.cameras.push(f.serialize()); } if (e.activeCamera && (r.activeCameraID = e.activeCamera.id), jt.AppendSerializedAnimations(e, r), e.animationGroups && e.animationGroups.length > 0) { r.animationGroups = []; for (let f = 0; f < e.animationGroups.length; f++) { const o = e.animationGroups[f]; r.animationGroups.push(o.serialize()); } } if (e.reflectionProbes && e.reflectionProbes.length > 0) for (r.reflectionProbes = [], n = 0; n < e.reflectionProbes.length; n++) { const f = e.reflectionProbes[n]; r.reflectionProbes.push(f.serialize()); } r.materials = [], r.multiMaterials = []; let s; for (n = 0; n < e.materials.length; n++) s = e.materials[n], s.doNotSerialize || r.materials.push(s.serialize()); for (r.multiMaterials = [], n = 0; n < e.multiMaterials.length; n++) { const f = e.multiMaterials[n]; r.multiMaterials.push(f.serialize()); } for (e.environmentTexture && (e.environmentTexture._files ? r.environmentTexture = e.environmentTexture.serialize() : (r.environmentTexture = e.environmentTexture.name, r.environmentTextureRotationY = e.environmentTexture.rotationY)), r.environmentIntensity = e.environmentIntensity, r.skeletons = [], n = 0; n < e.skeletons.length; n++) { const f = e.skeletons[n]; f.doNotSerialize || r.skeletons.push(f.serialize()); } for (r.transformNodes = [], n = 0; n < e.transformNodes.length; n++) e.transformNodes[n].doNotSerialize || r.transformNodes.push(e.transformNodes[n].serialize()); r.geometries = {}, r.geometries.boxes = [], r.geometries.spheres = [], r.geometries.cylinders = [], r.geometries.toruses = [], r.geometries.grounds = [], r.geometries.planes = [], r.geometries.torusKnots = [], r.geometries.vertexData = [], WF = []; const a = e.getGeometries(); for (n = 0; n < a.length; n++) { const f = a[n]; f.isReady() && GY(f, r.geometries); } for (r.meshes = [], n = 0; n < e.meshes.length; n++) { const f = e.meshes[n]; if (f instanceof Ee) { const o = f; o.doNotSerialize || (o.delayLoadState === 1 || o.delayLoadState === 0) && r.meshes.push(wie(o, r)); } } for (r.particleSystems = [], n = 0; n < e.particleSystems.length; n++) r.particleSystems.push(e.particleSystems[n].serialize(!1)); for (r.postProcesses = [], n = 0; n < e.postProcesses.length; n++) r.postProcesses.push(e.postProcesses[n].serialize()); e.actionManager && (r.actions = e.actionManager.serialize("scene")); for (const f of e._serializableComponents) f.serialize(r); return r; } /** * Serialize a scene into a JSON compatible object * @param scene defines the scene to serialize * @returns a JSON promise compatible object */ static SerializeAsync(e) { const t = G2._Serialize(e, !1), r = []; return this._CollectPromises(t, r), Promise.all(r).then(() => t); } static _CollectPromises(e, t) { if (Array.isArray(e)) for (let r = 0; r < e.length; ++r) { const n = e[r]; n instanceof Promise ? t.push(n.then((i) => e[r] = i)) : (n instanceof Object || Array.isArray(n)) && this._CollectPromises(n, t); } else if (e instanceof Object) { for (const r in e) if (Object.prototype.hasOwnProperty.call(e, r)) { const n = e[r]; n instanceof Promise ? t.push(n.then((i) => e[r] = i)) : (n instanceof Object || Array.isArray(n)) && this._CollectPromises(n, t); } } } /** * Serialize a mesh into a JSON compatible object * @param toSerialize defines the mesh to serialize * @param withParents defines if parents must be serialized as well * @param withChildren defines if children must be serialized as well * @returns a JSON compatible object */ static SerializeMesh(e, t = !1, r = !1) { const n = {}; if (n.meshes = [], n.transformNodes = [], n.cameras = [], n.lights = [], G2.ClearCache(), e = e instanceof Array ? e : [e], t || r) for (let i = 0; i < e.length; ++i) r && e[i].getDescendants().forEach((s) => { e.indexOf(s) < 0 && !s.doNotSerialize && e.push(s); }), t && e[i].parent && e.indexOf(e[i].parent) < 0 && !e[i].parent.doNotSerialize && e.push(e[i].parent); return e.forEach((i) => { fge(i, n); }), n; } } class aR { /** * Returns whether or not the VideoRecorder is available in your browser. * @param engine Defines the Babylon Engine. * @returns true if supported otherwise false. */ static IsSupported(e) { const t = e.getRenderingCanvas(); return !!t && typeof t.captureStream == "function"; } /** * True when a recording is already in progress. */ get isRecording() { return !!this._canvas && this._canvas.isRecording; } /** * Create a new VideoCapture object which can help converting what you see in Babylon to a video file. * @param engine Defines the BabylonJS Engine you wish to record. * @param options Defines options that can be used to customize the capture. */ constructor(e, t = {}) { if (!aR.IsSupported(e)) throw "Your browser does not support recording so far."; const r = e.getRenderingCanvas(); if (!r) throw "The babylon engine must have a canvas to be recorded"; this._canvas = r, this._canvas.isRecording = !1, this._options = Object.assign(Object.assign({}, aR._DefaultOptions), t); const n = this._canvas.captureStream(this._options.fps); if (this._options.audioTracks) for (const i of this._options.audioTracks) n.addTrack(i); this._mediaRecorder = new MediaRecorder(n, { mimeType: this._options.mimeType }), this._mediaRecorder.ondataavailable = (i) => this._handleDataAvailable(i), this._mediaRecorder.onerror = (i) => this._handleError(i), this._mediaRecorder.onstop = () => this._handleStop(); } /** * Stops the current recording before the default capture timeout passed in the startRecording function. */ stopRecording() { !this._canvas || !this._mediaRecorder || this.isRecording && (this._canvas.isRecording = !1, this._mediaRecorder.stop()); } /** * Starts recording the canvas for a max duration specified in parameters. * @param fileName Defines the name of the file to be downloaded when the recording stop. * If null no automatic download will start and you can rely on the promise to get the data back. * @param maxDuration Defines the maximum recording time in seconds. * It defaults to 7 seconds. A value of zero will not stop automatically, you would need to call stopRecording manually. * @returns A promise callback at the end of the recording with the video data in Blob. */ startRecording(e = "babylonjs.webm", t = 7) { if (!this._canvas || !this._mediaRecorder) throw "Recorder has already been disposed"; if (this.isRecording) throw "Recording already in progress"; return t > 0 && setTimeout(() => { this.stopRecording(); }, t * 1e3), this._fileName = e, this._recordedChunks = [], this._resolve = null, this._reject = null, this._canvas.isRecording = !0, this._mediaRecorder.start(this._options.recordChunckSize), new Promise((r, n) => { this._resolve = r, this._reject = n; }); } /** * Releases internal resources used during the recording. */ dispose() { this._canvas = null, this._mediaRecorder = null, this._recordedChunks = [], this._fileName = null, this._resolve = null, this._reject = null; } _handleDataAvailable(e) { e.data.size > 0 && this._recordedChunks.push(e.data); } _handleError(e) { if (this.stopRecording(), this._reject) this._reject(e.error); else throw new e.error(); } _handleStop() { this.stopRecording(); const e = new Blob(this._recordedChunks); this._resolve && this._resolve(e), window.URL.createObjectURL(e), this._fileName && ye.Download(e, this._fileName); } } aR._DefaultOptions = { mimeType: "video/webm", fps: 25, recordChunckSize: 3e3 }; let Ex = null; function PV(A, e, t, r, n = "image/png", i = !1, s) { const { height: a, width: f } = Bie(A, e, t); if (!(a && f)) { Se.Error("Invalid 'size' parameter !"); return; } Ex || (Ex = document.createElement("canvas")), Ex.width = f, Ex.height = a; const o = Ex.getContext("2d"), d = A.getRenderWidth() / A.getRenderHeight(); let v = f, u = v / d; u > a && (u = a, v = u * d); const l = Math.max(0, f - v) / 2, P = Math.max(0, a - u) / 2; e.getScene().activeCamera !== e ? cV(A, e, t, (c) => { if (i) { const H = new Blob([c]); ye.DownloadBlob(H), r && r(""); } else r && r(c); }, n, 1, A.getCreationOptions().antialias, void 0, void 0, void 0, void 0, s) : A.onEndFrameObservable.addOnce(() => { const c = A.getRenderingCanvas(); o && c && o.drawImage(c, l, P, v, u), Ex && (i ? (ye.EncodeScreenshotCanvasData(Ex, void 0, n, void 0, s), r && r("")) : ye.EncodeScreenshotCanvasData(Ex, r, n, void 0, s)); }); } function ZY(A, e, t, r = "image/png", n) { return new Promise((i, s) => { PV(A, e, t, (a) => { typeof a < "u" ? i(a) : s(new Error("Data is undefined")); }, r, void 0, n); }); } function mie(A, e, t, r, n = "image/png", i) { return new Promise((s) => { PV(A, e, { width: t, height: r }, () => { s(); }, n, !0, i); }); } function cV(A, e, t, r, n = "image/png", i = 1, s = !1, a, f = !1, o = !1, d = !0, v, u) { const { height: l, width: P, finalWidth: p, finalHeight: c } = Bie(A, e, t), H = { width: P, height: l }; if (!(l && P)) { Se.Error("Invalid 'size' parameter !"); return; } const T = { width: A.getRenderWidth(), height: A.getRenderHeight() }; A.setSize(P, l); const q = e.getScene(), b = new Ta("screenShot", H, q, !1, !1, 0, !1, We.BILINEAR_SAMPLINGMODE, void 0, o, void 0, void 0, void 0, i); b.renderList = q.meshes.slice(), b.samples = i, b.renderSprites = f, b.activeCamera = e, b.forceLayerMaskCheck = d, u == null || u(b); const j = () => { b.isReadyForRendering() && e.isReady(!0) ? (A.onEndFrameObservable.addOnce(() => { p === P && c === l ? b.readPixels(void 0, void 0, void 0, !1).then((m) => { eA.DumpData(P, l, m, r, n, a, !0, void 0, v), b.dispose(); }) : ty("pass", b.getInternalTexture(), q, void 0, void 0, void 0, p, c).then((m) => { A._readTexturePixels(m, p, c, -1, 0, null, !0, !1, 0, 0).then((I) => { eA.DumpData(p, c, I, r, n, a, !0, void 0, v), m.dispose(); }); }); }), b.render(!0), q.incrementRenderId(), q.resetCachedMaterial(), A.setSize(T.width, T.height), e.getProjectionMatrix(!0), q.render()) : setTimeout(j, 16); }, w = () => { q.incrementRenderId(), q.resetCachedMaterial(), j(); }; if (s) { const m = new ym("antialiasing", 1, q.activeCamera); b.addPostProcess(m), m.getEffect().isReady() ? w() : m.getEffect().onCompiled = () => { w(); }; } else w(); } function _Y(A, e, t, r = "image/png", n = 1, i = !1, s, a = !1, f = !1, o = !0, d) { return new Promise((v, u) => { cV(A, e, t, (l) => { typeof l < "u" ? v(l) : u(new Error("Data is undefined")); }, r, n, i, s, a, f, o, d); }); } function Bie(A, e, t) { let r = 0, n = 0, i = 0, s = 0; if (typeof t == "object") { const a = t.precision ? Math.abs(t.precision) : 1; t.width && t.height ? (r = t.height * a, n = t.width * a) : t.width && !t.height ? (n = t.width * a, r = Math.round(n / A.getAspectRatio(e))) : t.height && !t.width ? (r = t.height * a, n = Math.round(r * A.getAspectRatio(e))) : (n = Math.round(A.getRenderWidth() * a), r = Math.round(n / A.getAspectRatio(e))), t.finalWidth && t.finalHeight ? (s = t.finalHeight, i = t.finalWidth) : t.finalWidth && !t.finalHeight ? (i = t.finalWidth, s = Math.round(i / A.getAspectRatio(e))) : t.finalHeight && !t.finalWidth ? (s = t.finalHeight, i = Math.round(s * A.getAspectRatio(e))) : (i = n, s = r); } else isNaN(t) || (r = t, n = t, i = t, s = t); return n && (n = Math.floor(n)), r && (r = Math.floor(r)), i && (i = Math.floor(i)), s && (s = Math.floor(s)), { height: r | 0, width: n | 0, finalWidth: i | 0, finalHeight: s | 0 }; } const Age = { /** * Captures a screenshot of the current rendering * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG * @param engine defines the rendering engine * @param camera defines the source camera * @param size This parameter can be set to a single number or to an object with the * following (optional) properties: precision, width, height. If a single number is passed, * it will be used for both width and height. If an object is passed, the screenshot size * will be derived from the parameters. The precision property is a multiplier allowing * rendering at a higher or lower resolution * @param successCallback defines the callback receives a single parameter which contains the * screenshot as a string of base64-encoded characters. This string can be assigned to the * src parameter of an to display it * @param mimeType defines the MIME type of the screenshot image (default: image/png). * Check your browser for supported MIME types * @param forceDownload force the system to download the image even if a successCallback is provided * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. */ CreateScreenshot: PV, /** * Captures a screenshot of the current rendering * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG * @param engine defines the rendering engine * @param camera defines the source camera * @param size This parameter can be set to a single number or to an object with the * following (optional) properties: precision, width, height. If a single number is passed, * it will be used for both width and height. If an object is passed, the screenshot size * will be derived from the parameters. The precision property is a multiplier allowing * rendering at a higher or lower resolution * @param mimeType defines the MIME type of the screenshot image (default: image/png). * Check your browser for supported MIME types * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. * @returns screenshot as a string of base64-encoded characters. This string can be assigned * to the src parameter of an to display it */ CreateScreenshotAsync: ZY, /** * Captures a screenshot of the current rendering for a specific size. This will render the entire canvas but will generate a blink (due to canvas resize) * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG * @param engine defines the rendering engine * @param camera defines the source camera * @param width defines the expected width * @param height defines the expected height * @param mimeType defines the MIME type of the screenshot image (default: image/png). * Check your browser for supported MIME types * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. * @returns screenshot as a string of base64-encoded characters. This string can be assigned * to the src parameter of an to display it */ CreateScreenshotWithResizeAsync: mie, /** * Generates an image screenshot from the specified camera. * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG * @param engine The engine to use for rendering * @param camera The camera to use for rendering * @param size This parameter can be set to a single number or to an object with the * following (optional) properties: precision, width, height. If a single number is passed, * it will be used for both width and height. If an object is passed, the screenshot size * will be derived from the parameters. The precision property is a multiplier allowing * rendering at a higher or lower resolution * @param successCallback The callback receives a single parameter which contains the * screenshot as a string of base64-encoded characters. This string can be assigned to the * src parameter of an to display it * @param mimeType The MIME type of the screenshot image (default: image/png). * Check your browser for supported MIME types * @param samples Texture samples (default: 1) * @param antialiasing Whether antialiasing should be turned on or not (default: false) * @param fileName A name for for the downloaded file. * @param renderSprites Whether the sprites should be rendered or not (default: false) * @param enableStencilBuffer Whether the stencil buffer should be enabled or not (default: false) * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. */ CreateScreenshotUsingRenderTarget: cV, /** * Generates an image screenshot from the specified camera. * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG * @param engine The engine to use for rendering * @param camera The camera to use for rendering * @param size This parameter can be set to a single number or to an object with the * following (optional) properties: precision, width, height. If a single number is passed, * it will be used for both width and height. If an object is passed, the screenshot size * will be derived from the parameters. The precision property is a multiplier allowing * rendering at a higher or lower resolution * @param mimeType The MIME type of the screenshot image (default: image/png). * Check your browser for supported MIME types * @param samples Texture samples (default: 1) * @param antialiasing Whether antialiasing should be turned on or not (default: false) * @param fileName A name for for the downloaded file. * @param renderSprites Whether the sprites should be rendered or not (default: false) * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter. * @returns screenshot as a string of base64-encoded characters. This string can be assigned * to the src parameter of an to display it */ CreateScreenshotUsingRenderTargetAsync: _Y }, dge = () => { ye.CreateScreenshot = PV, ye.CreateScreenshotAsync = ZY, ye.CreateScreenshotUsingRenderTarget = cV, ye.CreateScreenshotUsingRenderTargetAsync = _Y; }; dge(); var SF; (function(A) { A[A.Checkbox = 0] = "Checkbox", A[A.Slider = 1] = "Slider", A[A.Vector3 = 2] = "Vector3", A[A.Quaternion = 3] = "Quaternion", A[A.Color3 = 4] = "Color3", A[A.String = 5] = "String", A[A.Button = 6] = "Button", A[A.Options = 7] = "Options", A[A.Tab = 8] = "Tab", A[A.FileButton = 9] = "FileButton", A[A.Vector2 = 10] = "Vector2"; })(SF || (SF = {})); class gI { /** * Constructor * @param buffer The buffer to read */ constructor(e) { this.byteOffset = 0, this.buffer = e; } /** * Loads the given byte length. * @param byteLength The byte length to load * @returns A promise that resolves when the load is complete */ loadAsync(e) { return this.buffer.readAsync(this.byteOffset, e).then((t) => { this._dataView = new DataView(t.buffer, t.byteOffset, t.byteLength), this._dataByteOffset = 0; }); } /** * Read a unsigned 32-bit integer from the currently loaded data range. * @returns The 32-bit integer read */ readUint32() { const e = this._dataView.getUint32(this._dataByteOffset, !0); return this._dataByteOffset += 4, this.byteOffset += 4, e; } /** * Read a byte array from the currently loaded data range. * @param byteLength The byte length to read * @returns The byte array read */ readUint8Array(e) { const t = new Uint8Array(this._dataView.buffer, this._dataView.byteOffset + this._dataByteOffset, e); return this._dataByteOffset += e, this.byteOffset += e, t; } /** * Read a string from the currently loaded data range. * @param byteLength The byte length to read * @returns The string read */ readString(e) { return QN(this.readUint8Array(e)); } /** * Skips the given byte length the currently loaded data range. * @param byteLength The byte length to skip */ skipBytes(e) { this._dataByteOffset += e, this.byteOffset += e; } } class UF { static _GetStorage() { try { return localStorage.setItem("test", ""), localStorage.removeItem("test"), localStorage; } catch { const t = {}; return { getItem: (r) => { const n = t[r]; return n === void 0 ? null : n; }, setItem: (r, n) => { t[r] = n; } }; } } /** * Reads a string from the data storage * @param key The key to read * @param defaultValue The value if the key doesn't exist * @returns The string value */ static ReadString(e, t) { const r = this._Storage.getItem(e); return r !== null ? r : t; } /** * Writes a string to the data storage * @param key The key to write * @param value The value to write */ static WriteString(e, t) { this._Storage.setItem(e, t); } /** * Reads a boolean from the data storage * @param key The key to read * @param defaultValue The value if the key doesn't exist * @returns The boolean value */ static ReadBoolean(e, t) { const r = this._Storage.getItem(e); return r !== null ? r === "true" : t; } /** * Writes a boolean to the data storage * @param key The key to write * @param value The value to write */ static WriteBoolean(e, t) { this._Storage.setItem(e, t ? "true" : "false"); } /** * Reads a number from the data storage * @param key The key to read * @param defaultValue The value if the key doesn't exist * @returns The number value */ static ReadNumber(e, t) { const r = this._Storage.getItem(e); return r !== null ? parseFloat(r) : t; } /** * Writes a number to the data storage * @param key The key to write * @param value The value to write */ static WriteNumber(e, t) { this._Storage.setItem(e, t.toString()); } } UF._Storage = UF._GetStorage(); class vge { constructor() { this._trackedScene = null; } /** * Track a given scene. This means the current scene state will be considered the original state * @param scene defines the scene to track */ track(e) { this._trackedScene = e, jt.AllowLoadingUniqueId = !0, this._savedJSON = G2.Serialize(e), jt.AllowLoadingUniqueId = !1; } /** * Get the delta between current state and original state * @returns a any containing the delta */ getDelta() { if (!this._trackedScene) return null; const e = We.ForceSerializeBuffers; We.ForceSerializeBuffers = !1, jt.AllowLoadingUniqueId = !0; const t = G2.Serialize(this._trackedScene); jt.AllowLoadingUniqueId = !1; const r = {}; for (const n in t) this._compareCollections(n, this._savedJSON[n], t[n], r); return We.ForceSerializeBuffers = e, r; } _compareArray(e, t, r, n) { if (t.length === 0 && r.length === 0) return !0; if (t.length && !isNaN(t[0]) || r.length && !isNaN(r[0])) { if (t.length !== r.length) return !1; if (t.length === 0) return !0; for (let s = 0; s < t.length; s++) if (t[s] !== r[s]) return n[e] = r, !1; return !0; } const i = []; for (let s = 0; s < t.length; s++) { const a = t[s], f = a.uniqueId; i.push(f); const o = r.filter((d) => d.uniqueId === f); if (o.length) { const d = o[0], v = {}; this._compareObjects(a, d, v) || (n[e] || (n[e] = []), v.__state = { id: d.id || d.name }, n[e].push(v)); } else { const d = { __state: { deleteId: a.id || a.name } }; n[e] || (n[e] = []), n[e].push(d); } } for (let s = 0; s < r.length; s++) { const a = r[s], f = a.uniqueId; i.indexOf(f) === -1 && (n[e] || (n[e] = []), n[e].push(a)); } return !0; } _compareObjects(e, t, r) { let n = !1; for (const i in e) { if (!Object.prototype.hasOwnProperty.call(e, i)) continue; const s = e[i], a = t[i]; let f = !1; if (Array.isArray(s)) f = JSON.stringify(s) !== JSON.stringify(a); else if (!isNaN(s) || Object.prototype.toString.call(s) == "[object String]") f = s !== a; else if (typeof s == "object" && typeof a == "object") { const o = {}; this._compareObjects(s, a, o) || (r[i] = o, n = !0); } f && (n = !0, r[i] = a); } return !n; } _compareCollections(e, t, r, n) { if (t !== r && t && r) { if (Array.isArray(t) && Array.isArray(r)) { if (this._compareArray(e, t, r, n)) return; } else if (typeof t == "object" && typeof r == "object") { const i = {}; this._compareObjects(t, r, i) || (n[e] = i); return; } } } static GetShadowGeneratorById(e, t) { const r = e.lights.map((n) => n.getShadowGenerators()); for (const n of r) if (n) { const i = n.values(); for (let s = i.next(); s.done !== !0; s = i.next()) { const a = s.value; if (a && a.id === t) return a; } } return null; } /** * Apply a given delta to a given scene * @param deltaJSON defines the JSON containing the delta * @param scene defines the scene to apply the delta to */ static ApplyDelta(e, t) { typeof e == "string" && (e = JSON.parse(e)); const r = t; for (const n in e) { const i = e[n], s = r[n]; if (Array.isArray(s) || n === "shadowGenerators") switch (n) { case "cameras": this._ApplyDeltaForEntity(i, t, t.getCameraById.bind(t), (a) => Tr.Parse(a, t)); break; case "lights": this._ApplyDeltaForEntity(i, t, t.getLightById.bind(t), (a) => ci.Parse(a, t)); break; case "shadowGenerators": this._ApplyDeltaForEntity(i, t, (a) => this.GetShadowGeneratorById(t, a), (a) => ln.Parse(a, t)); break; case "meshes": this._ApplyDeltaForEntity(i, t, t.getMeshById.bind(t), (a) => Ee.Parse(a, t, "")); break; case "skeletons": this._ApplyDeltaForEntity(i, t, t.getSkeletonById.bind(t), (a) => r4.Parse(a, t)); break; case "materials": this._ApplyDeltaForEntity(i, t, t.getMaterialById.bind(t), (a) => gt.Parse(a, t, "")); break; case "multiMaterials": this._ApplyDeltaForEntity(i, t, t.getMaterialById.bind(t), (a) => Dc.Parse(a, t, "")); break; case "transformNodes": this._ApplyDeltaForEntity(i, t, t.getTransformNodeById.bind(t), (a) => Hr.Parse(a, t, "")); break; case "particleSystems": this._ApplyDeltaForEntity(i, t, t.getParticleSystemById.bind(t), (a) => ti.Parse(a, t, "")); break; case "morphTargetManagers": this._ApplyDeltaForEntity(i, t, t.getMorphTargetById.bind(t), (a) => U0.Parse(a, t)); break; case "postProcesses": this._ApplyDeltaForEntity(i, t, t.getPostProcessByName.bind(t), (a) => kr.Parse(a, t, "")); break; } else isNaN(s) ? s.fromArray && s.fromArray(i) : r[n] = i; } } static _ApplyPropertiesToEntity(e, t) { for (const r in e) { const n = e[r], i = t[r]; i !== void 0 && (!isNaN(i) || Array.isArray(i) ? t[r] = n : i.fromArray ? i.fromArray(n) : typeof i == "object" && i !== null && this._ApplyPropertiesToEntity(n, i)); } } static _ApplyDeltaForEntity(e, t, r, n) { for (const i of e) if (i.__state && i.__state.id !== void 0) { const s = r(i.__state.id); s && (this._ApplyPropertiesToEntity(i, s), jt.ParseProperties(i, s, t, null)); } else if (i.__state && i.__state.deleteId !== void 0) { const s = r(i.__state.deleteId); s == null || s.dispose(); } else n(i); } } var LS; (function(A) { class e { /** * Serialize the Alphabet to JSON string. * @returns JSON serialization */ serialize() { const n = {}, i = new Array(this._characterToIdx.size); return this._characterToIdx.forEach((s, a) => { i[s] = a; }), n.characters = i, n.insertionCosts = this._insertionCosts, n.deletionCosts = this._deletionCosts, n.substitutionCosts = this._substitutionCosts, JSON.stringify(n); } /** * Parse an Alphabet from a JSON serialization. * @param json JSON string to deserialize * @returns deserialized Alphabet */ static Deserialize(n) { const i = JSON.parse(n), s = new e(i.characters); return s._insertionCosts = i.insertionCosts, s._deletionCosts = i.deletionCosts, s._substitutionCosts = i.substitutionCosts, s; } /** * Create a new Alphabet. * @param characters characters of the alphabet * @param charToInsertionCost function mapping characters to insertion costs * @param charToDeletionCost function mapping characters to deletion costs * @param charsToSubstitutionCost function mapping character pairs to substitution costs */ constructor(n, i = null, s = null, a = null) { i = i ?? (() => 1), s = s ?? (() => 1), a = a ?? ((o, d) => o === d ? 0 : 1), this._characterToIdx = /* @__PURE__ */ new Map(), this._insertionCosts = new Array(n.length), this._deletionCosts = new Array(n.length), this._substitutionCosts = new Array(n.length); let f; for (let o = 0; o < n.length; ++o) { f = n[o], this._characterToIdx.set(f, o), this._insertionCosts[o] = i(f), this._deletionCosts[o] = s(f), this._substitutionCosts[o] = new Array(n.length); for (let d = o; d < n.length; ++d) this._substitutionCosts[o][d] = a(f, n[d]); } } /** * Get the index (internally-assigned number) for a character. * @param char character * @returns index */ getCharacterIdx(n) { return this._characterToIdx.get(n); } /** * Get the insertion cost of a character from its index. * @param idx character index * @returns insertion cost */ getInsertionCost(n) { return this._insertionCosts[n]; } /** * Get the deletion cost of a character from its index. * @param idx character index * @returns deletion cost */ getDeletionCost(n) { return this._deletionCosts[n]; } /** * Gets the cost to substitute two characters. NOTE: this cost is * required to be bi-directional, meaning it cannot matter which of * the provided characters is being removed and which is being inserted. * @param idx1 the first character index * @param idx2 the second character index * @returns substitution cost */ getSubstitutionCost(n, i) { const s = Math.min(n, i), a = Math.max(n, i); return this._substitutionCosts[s][a]; } } A.Alphabet = e; class t { /** * Serialize to JSON string. JSON representation does NOT include the Alphabet * from which this Sequence was created; Alphabet must be independently * serialized. * @returns JSON string */ serialize() { return JSON.stringify(this._characters); } /** * Deserialize from JSON string and Alphabet. This should be the same Alphabet * from which the Sequence was originally created, which must be serialized and * deserialized independently so that it can be passed in here. * @param json JSON string representation of Sequence * @param alphabet Alphabet from which Sequence was originally created * @returns Sequence */ static Deserialize(n, i) { const s = new t([], i); return s._characters = JSON.parse(n), s; } /** * Create a new Sequence. * @param characters characters in the new Sequence * @param alphabet Alphabet, which must include all used characters */ constructor(n, i) { if (n.length > t._MAX_SEQUENCE_LENGTH) throw new Error("Sequences longer than " + t._MAX_SEQUENCE_LENGTH + " not supported."); this._alphabet = i, this._characters = n.map((s) => this._alphabet.getCharacterIdx(s)); } /** * Get the distance between this Sequence and another. * @param other sequence to compare to * @returns Levenshtein distance */ distance(n) { return t._Distance(this, n); } /** * Compute the Levenshtein distance between two Sequences. * @param a first Sequence * @param b second Sequence * @returns Levenshtein distance */ static _Distance(n, i) { const s = n._alphabet; if (s !== i._alphabet) throw new Error("Cannot Levenshtein compare Sequences built from different alphabets."); const a = n._characters, f = i._characters, o = a.length, d = f.length, v = t._CostMatrix; v[0][0] = 0; for (let u = 0; u < o; ++u) v[u + 1][0] = v[u][0] + s.getInsertionCost(a[u]); for (let u = 0; u < d; ++u) v[0][u + 1] = v[0][u] + s.getInsertionCost(f[u]); for (let u = 0; u < o; ++u) for (let l = 0; l < d; ++l) t._InsertionCost = v[u + 1][l] + s.getInsertionCost(f[l]), t._DeletionCost = v[u][l + 1] + s.getDeletionCost(a[u]), t._SubstitutionCost = v[u][l] + s.getSubstitutionCost(a[u], f[l]), v[u + 1][l + 1] = Math.min(t._InsertionCost, t._DeletionCost, t._SubstitutionCost); return v[o][d]; } } t._MAX_SEQUENCE_LENGTH = 256, t._CostMatrix = [...Array(t._MAX_SEQUENCE_LENGTH + 1)].map(() => new Array(t._MAX_SEQUENCE_LENGTH + 1)), A.Sequence = t; })(LS || (LS = {})); class fa { /** * Serialize to JSON. * @returns serialized JSON string */ serialize() { return JSON.stringify(this); } /** * Deserialize from JSON. * @param json serialized JSON string * @returns deserialized Trajectory */ static Deserialize(e) { const t = JSON.parse(e), r = new fa(t._segmentLength); return r._points = t._points.map((n) => new S(n._x, n._y, n._z)), r; } /** * Create a new empty Trajectory. * @param segmentLength radius of discretization for Trajectory points */ constructor(e = 0.01) { this._points = [], this._segmentLength = e; } /** * Get the length of the Trajectory. * @returns length of the Trajectory */ getLength() { return this._points.length * this._segmentLength; } /** * Append a new point to the Trajectory. * NOTE: This implementation has many allocations. * @param point point to append to the Trajectory */ add(e) { let t = this._points.length; if (t === 0) this._points.push(e.clone()); else { const r = () => this._segmentLength / S.Distance(this._points[t - 1], e); for (let n = r(); n <= 1; n = r()) { const i = this._points[t - 1].scale(1 - n); e.scaleAndAddToRef(n, i), this._points.push(i), ++t; } } } /** * Create a new Trajectory with a segment length chosen to make it * probable that the new Trajectory will have a specified number of * segments. This operation is imprecise. * @param targetResolution number of segments desired * @returns new Trajectory with approximately the requested number of segments */ resampleAtTargetResolution(e) { const t = new fa(this.getLength() / e); return this._points.forEach((r) => { t.add(r); }), t; } /** * Convert Trajectory segments into tokenized representation. This * representation is an array of numbers where each nth number is the * index of the token which is most similar to the nth segment of the * Trajectory. * @param tokens list of vectors which serve as discrete tokens * @returns list of indices of most similar token per segment */ tokenize(e) { const t = [], r = new S(); for (let n = 2; n < this._points.length; ++n) fa._TransformSegmentDirToRef(this._points[n - 2], this._points[n - 1], this._points[n], r) && t.push(fa._TokenizeSegment(r, e)); return t; } /** * Transform the rotation (i.e., direction) of a segment to isolate * the relative transformation represented by the segment. This operation * may or may not succeed due to singularities in the equations that define * motion relativity in this context. * @param priorVec the origin of the prior segment * @param fromVec the origin of the current segment * @param toVec the destination of the current segment * @param result reference to output variable * @returns whether or not transformation was successful */ static _TransformSegmentDirToRef(e, t, r, n) { return t.subtractToRef(e, fa._ForwardDir), fa._ForwardDir.normalize(), t.scaleToRef(-1, fa._InverseFromVec), fa._InverseFromVec.normalize(), Math.abs(S.Dot(fa._ForwardDir, fa._InverseFromVec)) > 0.98 ? !1 : (S.CrossToRef(fa._ForwardDir, fa._InverseFromVec, fa._UpDir), fa._UpDir.normalize(), he.LookAtLHToRef(e, t, fa._UpDir, fa._LookMatrix), r.subtractToRef(t, fa._FromToVec), fa._FromToVec.normalize(), S.TransformNormalToRef(fa._FromToVec, fa._LookMatrix, n), !0); } /** * Determine which token vector is most similar to the * segment vector. * @param segment segment vector * @param tokens token vector list * @returns index of the most similar token to the segment */ static _TokenizeSegment(e, t) { fa._BestMatch = 0, fa._Score = S.Dot(e, t[0]), fa._BestScore = fa._Score; for (let r = 1; r < t.length; ++r) fa._Score = S.Dot(e, t[r]), fa._Score > fa._BestScore && (fa._BestMatch = r, fa._BestScore = fa._Score); return fa._BestMatch; } } fa._ForwardDir = new S(); fa._InverseFromVec = new S(); fa._UpDir = new S(); fa._FromToVec = new S(); fa._LookMatrix = new he(); class oR { /** * Helper method to create new "spikeball" Vector3Alphabets. Uses a naive * optimize-from-random strategy to space points around the unit sphere * surface as a simple alternative to really doing the math to tile the * sphere. * @param alphabetSize size of the desired alphabet * @param iterations number of iterations over which to optimize the "spikeball" * @param startingStepSize distance factor to move points in early optimization iterations * @param endingStepSize distance factor to move points in late optimization iterations * @param fixedValues alphabet "characters" that are required and cannot be moved by optimization * @returns a new randomly generated and optimized Vector3Alphabet of the specified size */ static Generate(e = 64, t = 256, r = 0.1, n = 1e-3, i = []) { const f = new oR(e); for (let P = 0; P < e; ++P) f.chars[P] = new S(Math.random() - 0.5, Math.random() - 0.5, Math.random() - 0.5), f.chars[P].normalize(); for (let P = 0; P < i.length; ++P) f.chars[P].copyFrom(i[P]); let o, d; const v = new S(), u = new S(), l = (P, p, c) => (1 - c) * P + c * p; for (let P = 0; P < t; ++P) { o = l(r, n, P / (t - 1)); for (let p = i.length; p < f.chars.length; ++p) v.copyFromFloats(0, 0, 0), f.chars.forEach((c) => { f.chars[p].subtractToRef(c, u), d = u.lengthSquared(), d > 1e-6 && u.scaleAndAddToRef(1 / (u.lengthSquared() * d), v); }), v.scaleInPlace(o), f.chars[p].addInPlace(v), f.chars[p].normalize(); } return f; } /** * Serialize to JSON. * @returns JSON serialization */ serialize() { return JSON.stringify(this.chars); } /** * Deserialize from JSON. * @param json JSON serialization * @returns deserialized Vector3Alphabet */ static Deserialize(e) { const t = JSON.parse(e), r = new oR(t.length); for (let n = 0; n < t.length; ++n) r.chars[n] = new S(t[n]._x, t[n]._y, t[n]._z); return r; } constructor(e) { this.chars = new Array(e); } } class M2 { /** * Serialize to JSON. * @returns JSON serialization */ serialize() { return JSON.stringify(this._sequences.map((e) => e.serialize())); } /** * Deserialize from JSON string and Alphabet. This should be the same Alphabet * from which the descriptor was originally created, which must be serialized and * deserialized independently so that it can be passed in here. * @param json JSON serialization * @param alphabet Alphabet from which descriptor was originally created * @returns deserialized TrajectoryDescriptor */ static Deserialize(e, t) { const r = new M2(); return r._sequences = JSON.parse(e).map((n) => LS.Sequence.Deserialize(n, t)), r; } /** * Create a new TrajectoryDescriptor to describe a provided Trajectory according * to the provided alphabets. * @param trajectory Trajectory to be described * @param vector3Alphabet Vector3Alphabet to be used to tokenize the Trajectory * @param levenshteinAlphabet Levenshtein.Alphabet to be used as basis for comparison with other descriptors * @returns TrajectoryDescriptor describing provided Trajectory */ static CreateFromTrajectory(e, t, r) { return M2.CreateFromTokenizationPyramid(M2._GetTokenizationPyramid(e, t), r); } /** * Create a new TrajectoryDescriptor from a pre-existing pyramid of tokens. * NOTE: This function exists to support an outdated serialization mechanism and should * be deleted if it is no longer useful. * @param pyramid tokenization pyramid * @param levenshteinAlphabet Levenshtein.Alphabet to be uses as basis for comparison with other descriptors * @returns TrajectoryDescriptor describing the Trajectory from which the pyramid was built */ static CreateFromTokenizationPyramid(e, t) { const r = new M2(); return r._sequences = e.map((n) => new LS.Sequence(n, t)), r; } constructor() { this._sequences = []; } /** * Create the tokenization pyramid for the provided Trajectory according to the given * Vector3Alphabet. * @param trajectory Trajectory to be tokenized * @param alphabet Vector3Alphabet containing tokens * @param targetResolution finest resolution of descriptor * @returns tokenization pyramid for Trajectory */ static _GetTokenizationPyramid(e, t, r = M2._FINEST_DESCRIPTOR_RESOLUTION) { const n = []; for (let i = r; i > 4; i = Math.floor(i / 2)) n.push(e.resampleAtTargetResolution(i).tokenize(t.chars)); return n; } /** * Calculate a distance metric between this TrajectoryDescriptor and another. This is * essentially a similarity score and does not directly represent Euclidean distance, * edit distance, or any other formal distance metric. * @param other TrajectoryDescriptor from which to determine distance * @returns distance, a nonnegative similarity score where larger values indicate dissimilarity */ distance(e) { let t = 0, r; for (let n = 0; n < this._sequences.length; ++n) r = Math.pow(2, n), t += r * this._sequences[n].distance(e._sequences[n]); return t; } } M2._FINEST_DESCRIPTOR_RESOLUTION = 32; class KS { /** * Serialize to JSON. * @returns JSON serialization */ serialize() { const e = {}; return e.descriptors = this._descriptors.map((t) => t.serialize()), e.centroidIdx = this._centroidIdx, e.averageDistance = this._averageDistance, JSON.stringify(e); } /** * Deserialize from JSON string and Alphabet. This should be the same Alphabet * from which the descriptors were originally created, which must be serialized and * deserialized independently so that it can be passed in here. * @param json JSON string representation * @param alphabet Alphabet from which TrajectoryDescriptors were originally created * @returns deserialized TrajectoryDescriptor */ static Deserialize(e, t) { const r = JSON.parse(e), n = new KS(); return n._descriptors = r.descriptors.map((i) => M2.Deserialize(i, t)), n._centroidIdx = r.centroidIdx, n._averageDistance = r.averageDistance, n; } /** * Create a new DescribedTrajectory. * @param descriptors currently-known TrajectoryDescriptors, if any */ constructor(e = []) { this._descriptors = e, this._centroidIdx = -1, this._averageDistance = 0, this._refreshDescription(); } /** * Add a new TrajectoryDescriptor to the list of descriptors known to describe * this same DescribedTrajectory. * @param descriptor descriptor to be added */ add(e) { this._descriptors.push(e), this._refreshDescription(); } /** * Compute the cost, which is inversely related to the likelihood that the provided * TrajectoryDescriptor describes a Trajectory that is considered to be the same as * the class represented by this DescribedTrajectory. * @param descriptor the descriptor to be costed * @returns cost of the match, which is a nonnegative similarity metric where larger values indicate dissimilarity */ getMatchCost(e) { return e.distance(this._descriptors[this._centroidIdx]) / this._averageDistance; } /** * Compute the minimum distance between the queried TrajectoryDescriptor and a * descriptor which is a member of this collection. This is an alternative way of * conceptualizing match cost from getMatchCost(), and it serves a different function. * @param descriptor the descriptor to find the minimum distance to * @returns minimum descriptor distance to a member descriptor of this DescribedTrajectory */ getMatchMinimumDistance(e) { return Math.min(...this._descriptors.map((t) => t.distance(e))); } /** * Refreshes the internal representation of this DescribedTrajectory. */ _refreshDescription() { this._centroidIdx = -1; let e; const t = this._descriptors.map((r) => (e = 0, this._descriptors.forEach((n) => { e += r.distance(n); }), e)); for (let r = 0; r < t.length; ++r) (this._centroidIdx < 0 || t[r] < t[this._centroidIdx]) && (this._centroidIdx = r); this._averageDistance = 0, this._descriptors.forEach((r) => { this._averageDistance += r.distance(this._descriptors[this._centroidIdx]); }), this._descriptors.length > 0 && (this._averageDistance = Math.max(this._averageDistance / this._descriptors.length, KS._MIN_AVERAGE_DISTANCE)); } } KS._MIN_AVERAGE_DISTANCE = 1; class vO { /** * Serialize to JSON. * @returns JSON serialization */ serialize() { const e = {}; return e.maximumAllowableMatchCost = this._maximumAllowableMatchCost, e.vector3Alphabet = this._vector3Alphabet.serialize(), e.levenshteinAlphabet = this._levenshteinAlphabet.serialize(), e.nameToDescribedTrajectory = [], this._nameToDescribedTrajectory.forEach((t, r) => { e.nameToDescribedTrajectory.push(r), e.nameToDescribedTrajectory.push(t.serialize()); }), JSON.stringify(e); } /** * Deserialize from JSON. * @param json JSON serialization * @returns deserialized TrajectorySet */ static Deserialize(e) { const t = JSON.parse(e), r = new vO(); r._maximumAllowableMatchCost = t.maximumAllowableMatchCost, r._vector3Alphabet = oR.Deserialize(t.vector3Alphabet), r._levenshteinAlphabet = LS.Alphabet.Deserialize(t.levenshteinAlphabet); for (let n = 0; n < t.nameToDescribedTrajectory.length; n += 2) r._nameToDescribedTrajectory.set(t.nameToDescribedTrajectory[n], KS.Deserialize(t.nameToDescribedTrajectory[n + 1], r._levenshteinAlphabet)); return r; } /** * Initialize a new empty TrajectorySet with auto-generated Alphabets. * VERY naive, need to be generating these things from known * sets. Better version later, probably eliminating this one. * @returns auto-generated TrajectorySet */ static Generate() { const e = oR.Generate(64, 256, 0.1, 1e-3, [S.Forward()]), t = new Array(e.chars.length); for (let i = 0; i < t.length; ++i) t[i] = i; const r = new LS.Alphabet(t, (i) => i === 0 ? 0 : 1, (i) => i === 0 ? 0 : 1, (i, s) => Math.min(1 - S.Dot(e.chars[i], e.chars[s]), 1)), n = new vO(); return n._vector3Alphabet = e, n._levenshteinAlphabet = r, n; } constructor() { this._maximumAllowableMatchCost = 4, this._nameToDescribedTrajectory = /* @__PURE__ */ new Map(); } /** * Add a new Trajectory to the set with a given name. * @param trajectory new Trajectory to be added * @param classification name to which to add the Trajectory */ addTrajectoryToClassification(e, t) { this._nameToDescribedTrajectory.has(t) || this._nameToDescribedTrajectory.set(t, new KS()), this._nameToDescribedTrajectory.get(t).add(M2.CreateFromTrajectory(e, this._vector3Alphabet, this._levenshteinAlphabet)); } /** * Remove a known named trajectory and all Trajectories associated with it. * @param classification name to remove * @returns whether anything was removed */ deleteClassification(e) { return this._nameToDescribedTrajectory.delete(e); } /** * Attempt to recognize a Trajectory from among all the classifications * already known to the classifier. * @param trajectory Trajectory to be recognized * @returns classification of Trajectory if recognized, null otherwise */ classifyTrajectory(e) { const t = M2.CreateFromTrajectory(e, this._vector3Alphabet, this._levenshteinAlphabet), r = []; if (this._nameToDescribedTrajectory.forEach((a, f) => { a.getMatchCost(t) < this._maximumAllowableMatchCost && r.push(f); }), r.length === 0) return null; let n = 0, i = this._nameToDescribedTrajectory.get(r[n]).getMatchMinimumDistance(t), s; for (let a = 0; a < r.length; ++a) s = this._nameToDescribedTrajectory.get(r[a]).getMatchMinimumDistance(t), s < i && (i = s, n = a); return r[n]; } } class fR { /** * Constructs a reflector object. * @param scene The scene to use * @param hostname The hostname of the reflector bridge * @param port The port of the reflector bridge */ constructor(e, t, r) { this._scene = e, Se.Log(`[Reflector] Connecting to ws://${t}:${r}`), this._webSocket = new WebSocket(`ws://${t}:${r}`), this._webSocket.onmessage = (n) => { const i = n.data; if (i.startsWith(fR._SERVER_PREFIX)) { const s = i.substr(fR._SERVER_PREFIX.length); Se.Log(`[Reflector] Received server message: ${s.substr(0, 64)}`), this._handleServerMessage(s); return; } else Se.Log(`[Reflector] Received client message: ${i.substr(0, 64)}`), this._handleClientMessage(); }, this._webSocket.onclose = (n) => { Se.Log(`[Reflector] Disconnected ${n.code} ${n.reason}`); }; } /** * Closes the reflector connection */ close() { this._webSocket.close(); } _handleServerMessage(e) { switch (e) { case "connected": { G2.SerializeAsync(this._scene).then((t) => { this._webSocket.send(`load|${JSON.stringify(t)}`); }); break; } } } _handleClientMessage() { } } fR._SERVER_PREFIX = "$$"; class Vy { /** * A pressure observer will call this callback, whenever these thresholds are met. * @param options An object containing the thresholds used to decide what value to to return for each update property (average of start and end of a threshold boundary). */ constructor(e) { this._observer = null, this._currentState = [], this.onPressureChanged = new Oe(), Vy.IsAvailable && (this._observer = new PressureObserver((t) => { this._currentState = t, this.onPressureChanged.notifyObservers(t); }, e)); } /** * Returns true if PressureObserver is available for use, false otherwise. */ static get IsAvailable() { return typeof PressureObserver < "u" && PressureObserver.supportedSources.includes("cpu"); } /** * Method that must be called to begin observing changes, and triggering callbacks. * @param source defines the source to observe */ observe(e) { var t; try { (t = this._observer) === null || t === void 0 || t.observe(e), this.onPressureChanged.notifyObservers(this._currentState); } catch { } } /** * Method that must be called to stop observing changes and triggering callbacks (cleanup function). * @param source defines the source to unobserve */ unobserve(e) { var t; try { (t = this._observer) === null || t === void 0 || t.unobserve(e); } catch { } } /** * Release the associated resources. */ dispose() { var e; (e = this._observer) === null || e === void 0 || e.disconnect(), this._observer = null, this.onPressureChanged.clear(); } } const uge = 1.5; class M8 { /** * Creates a new DynamicFloat32Array with the desired item capacity. * @param itemCapacity The initial item capacity you would like to set for the array. */ constructor(e) { this._view = new Float32Array(e), this._itemLength = 0; } /** * The number of items currently in the array. */ get itemLength() { return this._itemLength; } /** * Gets value at index, NaN if no such index exists. * @param index the index to get the value at. * @returns the value at the index provided. */ at(e) { return e < 0 || e >= this._itemLength ? NaN : this._view[e]; } /** * Gets a view of the original array from start to end (exclusive of end). * @param start starting index. * @param end ending index. * @returns a subarray of the original array. */ subarray(e, t) { return e >= t || e < 0 ? new Float32Array(0) : (t > this._itemLength && (t = this._itemLength), this._view.subarray(e, t)); } /** * Pushes items to the end of the array. * @param item The item to push into the array. */ push(e) { this._view[this._itemLength] = e, this._itemLength++, this._itemLength >= this._view.length && this._growArray(); } /** * Grows the array by the growth factor when necessary. */ _growArray() { const e = Math.floor(this._view.length * uge), t = new Float32Array(e); t.set(this._view), this._view = t; } } const Fx = 1800, lge = 24, Pge = "0", $G = "timestamp", eZ = "numPoints", cge = /\r/g, RE = "@"; class b0 { /** * The offset for when actual data values start appearing inside a slice. */ static get SliceDataOffset() { return 2; } /** * The offset for the value of the number of points inside a slice. */ static get NumberOfPointsOffset() { return 1; } /** * Handles the creation of a performance viewer collector. * @param _scene the scene to collect on. * @param _enabledStrategyCallbacks the list of data to collect with callbacks for initialization purposes. */ constructor(e, t) { this._scene = e, this._collectDataAtFrame = () => { const r = Yi.Now - this._startingTimestamp, n = this.datasets.ids.length, i = this.datasets.startingIndices.itemLength; let s = 0; if (i > 0) { const a = this.datasets.startingIndices.at(i - 1); s = a + this.datasets.data.at(a + b0.NumberOfPointsOffset) + b0.SliceDataOffset; } if (this.datasets.startingIndices.push(s), this.datasets.data.push(r), this.datasets.data.push(n), this.datasets.ids.forEach((a) => { const f = this._strategies.get(a); f && this.datasets.data.push(f.getData()); }), this.datasetObservable.hasObservers()) { const a = [r, n]; for (let f = 0; f < n; f++) a.push(this.datasets.data.at(s + b0.SliceDataOffset + f)); this.datasetObservable.notifyObservers(a); } }, this.datasets = { ids: [], data: new M8(Fx), startingIndices: new M8(Fx) }, this._strategies = /* @__PURE__ */ new Map(), this._datasetMeta = /* @__PURE__ */ new Map(), this._eventRestoreSet = /* @__PURE__ */ new Set(), this._customEventObservable = new Oe(), this.datasetObservable = new Oe(), this.metadataObservable = new Oe((r) => r.callback(this._datasetMeta, new IN(0))), t && this.addCollectionStrategies(...t); } /** * Registers a custom string event which will be callable via sendEvent. This method returns an event object which will contain the id of the event. * The user can set a value optionally, which will be used in the sendEvent method. If the value is set, we will record this value at the end of each frame, * if not we will increment our counter and record the value of the counter at the end of each frame. The value recorded is 0 if no sendEvent method is called, within a frame. * @param name The name of the event to register * @param forceUpdate if the code should force add an event, and replace the last one. * @param category the category for that event * @returns The event registered, used in sendEvent */ registerEvent(e, t, r) { var n; if (this._strategies.has(e) && !t) return; this._strategies.has(e) && t && ((n = this._strategies.get(e)) === null || n === void 0 || n.dispose(), this._strategies.delete(e)); const i = (a) => { let f = 0, o = 0; const d = a.onAfterRenderObservable.add(() => { o = f, f = 0; }), v = this._customEventObservable.add((u) => { e === u.name && (u.value !== void 0 ? f = u.value : f++); }); return { id: e, getData: () => o, dispose: () => { a.onAfterRenderObservable.remove(d), this._customEventObservable.remove(v); } }; }, s = { name: e }; return this._eventRestoreSet.add(e), this.addCollectionStrategies({ strategyCallback: i, category: r }), s; } /** * Lets the perf collector handle an event, occurences or event value depending on if the event.value params is set. * @param event the event to handle an occurence for */ sendEvent(e) { this._customEventObservable.notifyObservers(e); } /** * This event restores all custom string events if necessary. */ _restoreStringEvents() { this._eventRestoreSet.size !== this._customEventObservable.observers.length && this._eventRestoreSet.forEach((e) => { this.registerEvent(e, !0); }); } /** * This method adds additional collection strategies for data collection purposes. * @param strategyCallbacks the list of data to collect with callbacks. */ addCollectionStrategies(...e) { for (let { strategyCallback: t, category: r, hidden: n } of e) { const i = t(this._scene); if (this._strategies.has(i.id)) { i.dispose(); continue; } this.datasets.ids.push(i.id), r && (r = r.replace(new RegExp(RE, "g"), "")), this._datasetMeta.set(i.id, { color: this._getHexColorFromId(i.id), category: r, hidden: n }), this._strategies.set(i.id, i); } this.metadataObservable.notifyObservers(this._datasetMeta); } /** * Gets a 6 character hexcode representing the colour from a passed in string. * @param id the string to get a hex code for. * @returns a hexcode hashed from the id. */ _getHexColorFromId(e) { let t = 0; for (let n = 0; n < e.length; n++) t = e.charCodeAt(n) + ((t << 5) - t); let r = "#"; for (let n = 0; n < lge; n += 8) { const i = t >> n & 255; r += (Pge + i.toString(16)).substr(-2); } return r; } /** * Collects and then sends the latest slice to any observers by using the appropriate strategy when the user wants. * The slice will be of the form [timestamp, numberOfPoints, value1, value2...] * This method does not add onto the collected data accessible via the datasets variable. */ getCurrentSlice() { const e = Yi.Now - this._startingTimestamp, t = this.datasets.ids.length, r = [e, t]; this.datasets.ids.forEach((n) => { const i = this._strategies.get(n); i && this.datasetObservable.hasObservers() && r.push(i.getData()); }), this.datasetObservable.hasObservers() && this.datasetObservable.notifyObservers(r); } /** * Updates a property for a dataset's metadata with the value provided. * @param id the id of the dataset which needs its metadata updated. * @param prop the property to update. * @param value the value to update the property with. */ updateMetadata(e, t, r) { const n = this._datasetMeta.get(e); n && (n[t] = r, this.metadataObservable.notifyObservers(this._datasetMeta)); } /** * Completely clear, data, ids, and strategies saved to this performance collector. * @param preserveStringEventsRestore if it should preserve the string events, by default will clear string events registered when called. */ clear(e) { this.datasets.data = new M8(Fx), this.datasets.ids.length = 0, this.datasets.startingIndices = new M8(Fx), this._datasetMeta.clear(), this._strategies.forEach((t) => t.dispose()), this._strategies.clear(), e || this._eventRestoreSet.clear(), this._hasLoadedData = !1; } /** * Accessor which lets the caller know if the performance collector has data loaded from a file or not! * Call clear() to reset this value. * @returns true if the data is loaded from a file, false otherwise. */ get hasLoadedData() { return this._hasLoadedData; } /** * Given a string containing file data, this function parses the file data into the datasets object. * It returns a boolean to indicate if this object was successfully loaded with the data. * @param data string content representing the file data. * @param keepDatasetMeta if it should use reuse the existing dataset metadata * @returns true if the data was successfully loaded, false otherwise. */ loadFromFileData(e, t) { const r = e.replace(cge, "").split(` `).map((v) => v.split(",").filter((u) => u.length > 0)).filter((v) => v.length > 0), n = 0, i = b0.NumberOfPointsOffset; if (r.length < 2) return !1; const s = { ids: [], data: new M8(Fx), startingIndices: new M8(Fx) }, [a, ...f] = r; if (a.length < 2 || a[n] !== $G || a[i] !== eZ) return !1; const o = /* @__PURE__ */ new Map(); for (let v = b0.SliceDataOffset; v < a.length; v++) { const [u, l] = a[v].split(RE); s.ids.push(u), o.set(u, l); } let d = 0; for (const v of f) { if (v.length < 2) return !1; const u = parseFloat(v[n]), l = parseInt(v[i]); if (isNaN(l) || isNaN(u) || (s.data.push(u), s.data.push(l), l + b0.SliceDataOffset !== v.length)) return !1; for (let P = b0.SliceDataOffset; P < v.length; P++) { const p = parseFloat(v[P]); if (isNaN(p)) return !1; s.data.push(p); } s.startingIndices.push(d), d += v.length; } if (this.datasets.ids = s.ids, this.datasets.data = s.data, this.datasets.startingIndices = s.startingIndices, t || this._datasetMeta.clear(), this._strategies.forEach((v) => v.dispose()), this._strategies.clear(), !t) for (const v of this.datasets.ids) { const u = o.get(v); this._datasetMeta.set(v, { category: u, color: this._getHexColorFromId(v) }); } return this.metadataObservable.notifyObservers(this._datasetMeta), this._hasLoadedData = !0, !0; } /** * Exports the datasets inside of the collector to a csv. */ exportDataToCsv() { let e = ""; e += `${$G},${eZ}`; for (let r = 0; r < this.datasets.ids.length; r++) if (e += `,${this.datasets.ids[r]}`, this._datasetMeta) { const n = this._datasetMeta.get(this.datasets.ids[r]); n != null && n.category && (e += `${RE}${n.category}`); } e += ` `; for (let r = 0; r < this.datasets.startingIndices.itemLength; r++) { const n = this.datasets.startingIndices.at(r), i = this.datasets.data.at(n), s = this.datasets.data.at(n + b0.NumberOfPointsOffset); e += `${i},${s}`; for (let a = 0; a < s; a++) e += `,${this.datasets.data.at(n + b0.SliceDataOffset + a)}`; for (let a = 0; a < this.datasets.ids.length - s; a++) e += ","; e += ` `; } const t = `${(/* @__PURE__ */ new Date()).toISOString()}-perfdata.csv`; ye.Download(new Blob([e], { type: "text/csv" }), t); } /** * Starts the realtime collection of data. * @param shouldPreserve optional boolean param, if set will preserve the dataset between calls of start. */ start(e) { e ? this._startingTimestamp === void 0 && (this._startingTimestamp = Yi.Now) : (this.datasets.data = new M8(Fx), this.datasets.startingIndices = new M8(Fx), this._startingTimestamp = Yi.Now), this._scene.onAfterRenderObservable.add(this._collectDataAtFrame), this._restoreStringEvents(), this._isStarted = !0; } /** * Stops the collection of data. */ stop() { this._scene.onAfterRenderObservable.removeCallback(this._collectDataAtFrame), this._isStarted = !1; } /** * Returns if the perf collector has been started or not. */ get isStarted() { return this._isStarted; } /** * Disposes of the object */ dispose() { this._scene.onAfterRenderObservable.removeCallback(this._collectDataAtFrame), this._datasetMeta.clear(), this._strategies.forEach((e) => { e.dispose(); }), this.datasetObservable.clear(), this.metadataObservable.clear(), this._isStarted = !1, this.datasets = null; } } const h0 = () => { }; class pge { /** * Gets the initializer for the strategy used for collection of fps metrics * @returns the initializer for the fps strategy */ static FpsStrategy() { return (e) => { const t = e.getEngine(); return { id: "FPS", getData: () => t.getFps(), dispose: h0 }; }; } /** * Gets the initializer for the strategy used for collection of thermal utilization metrics. * Needs the experimental pressure API. * @returns the initializer for the thermal utilization strategy */ static ThermalStrategy() { return this._PressureStrategy("Thermal utilization", "thermal"); } /** * Gets the initializer for the strategy used for collection of power supply utilization metrics. * Needs the experimental pressure API. * @returns the initializer for the power supply utilization strategy */ static PowerSupplyStrategy() { return this._PressureStrategy("Power supply utilization", "power-supply"); } /** * Gets the initializer for the strategy used for collection of pressure metrics. * Needs the experimental pressure API. * @returns the initializer for the pressure strategy */ static PressureStrategy() { return this._PressureStrategy("Pressure"); } static _PressureStrategy(e, t = null) { return () => { let r = 0; const n = new Vy(); return n.observe("cpu"), n.onPressureChanged.add((i) => { var s, a; for (const f of i) if (t && f.factors.includes(t) || !t && ((a = (s = f.factors) === null || s === void 0 ? void 0 : s.length) !== null && a !== void 0 ? a : 0) === 0) switch (f.state) { case "nominal": r = 0; break; case "fair": r = 0.25; break; case "serious": r = 0.5; break; case "critical": r = 1; break; } }), { id: e, getData: () => r, dispose: () => n.dispose() }; }; } /** * Gets the initializer for the strategy used for collection of total meshes metrics. * @returns the initializer for the total meshes strategy */ static TotalMeshesStrategy() { return (e) => ({ id: "Total meshes", getData: () => e.meshes.length, dispose: h0 }); } /** * Gets the initializer for the strategy used for collection of active meshes metrics. * @returns the initializer for the active meshes strategy */ static ActiveMeshesStrategy() { return (e) => ({ id: "Active meshes", getData: () => e.getActiveMeshes().length, dispose: h0 }); } /** * Gets the initializer for the strategy used for collection of active indices metrics. * @returns the initializer for the active indices strategy */ static ActiveIndicesStrategy() { return (e) => ({ id: "Active indices", getData: () => e.getActiveIndices(), dispose: h0 }); } /** * Gets the initializer for the strategy used for collection of active faces metrics. * @returns the initializer for the active faces strategy */ static ActiveFacesStrategy() { return (e) => ({ id: "Active faces", getData: () => e.getActiveIndices() / 3, dispose: h0 }); } /** * Gets the initializer for the strategy used for collection of active bones metrics. * @returns the initializer for the active bones strategy */ static ActiveBonesStrategy() { return (e) => ({ id: "Active bones", getData: () => e.getActiveBones(), dispose: h0 }); } /** * Gets the initializer for the strategy used for collection of active particles metrics. * @returns the initializer for the active particles strategy */ static ActiveParticlesStrategy() { return (e) => ({ id: "Active particles", getData: () => e.getActiveParticles(), dispose: h0 }); } /** * Gets the initializer for the strategy used for collection of draw calls metrics. * @returns the initializer for the draw calls strategy */ static DrawCallsStrategy() { return (e) => { let t = 0; const r = e.onBeforeAnimationsObservable.add(() => { e.getEngine()._drawCalls.fetchNewFrame(); }), n = e.onAfterRenderObservable.add(() => { t = e.getEngine()._drawCalls.current; }); return { id: "Draw calls", getData: () => t, dispose: () => { e.onBeforeAnimationsObservable.remove(r), e.onAfterRenderObservable.remove(n); } }; }; } /** * Gets the initializer for the strategy used for collection of total lights metrics. * @returns the initializer for the total lights strategy */ static TotalLightsStrategy() { return (e) => ({ id: "Total lights", getData: () => e.lights.length, dispose: h0 }); } /** * Gets the initializer for the strategy used for collection of total vertices metrics. * @returns the initializer for the total vertices strategy */ static TotalVerticesStrategy() { return (e) => ({ id: "Total vertices", getData: () => e.getTotalVertices(), dispose: h0 }); } /** * Gets the initializer for the strategy used for collection of total materials metrics. * @returns the initializer for the total materials strategy */ static TotalMaterialsStrategy() { return (e) => ({ id: "Total materials", getData: () => e.materials.length, dispose: h0 }); } /** * Gets the initializer for the strategy used for collection of total textures metrics. * @returns the initializer for the total textures strategy */ static TotalTexturesStrategy() { return (e) => ({ id: "Total textures", getData: () => e.textures.length, dispose: h0 }); } /** * Gets the initializer for the strategy used for collection of absolute fps metrics. * @returns the initializer for the absolute fps strategy */ static AbsoluteFpsStrategy() { return (e) => { const t = new Fte(e); return t.captureFrameTime = !0, { id: "Absolute FPS", getData: () => 1e3 / t.frameTimeCounter.lastSecAverage, dispose: h0 }; }; } /** * Gets the initializer for the strategy used for collection of meshes selection time metrics. * @returns the initializer for the meshes selection time strategy */ static MeshesSelectionStrategy() { return (e) => { let t = Yi.Now, r = 0; const n = e.onBeforeActiveMeshesEvaluationObservable.add(() => { t = Yi.Now; }), i = e.onAfterActiveMeshesEvaluationObservable.add(() => { r = Yi.Now - t; }); return { id: "Meshes Selection", getData: () => r, dispose: () => { e.onBeforeActiveMeshesEvaluationObservable.remove(n), e.onAfterActiveMeshesEvaluationObservable.remove(i); } }; }; } /** * Gets the initializer for the strategy used for collection of render targets time metrics. * @returns the initializer for the render targets time strategy */ static RenderTargetsStrategy() { return (e) => { let t = Yi.Now, r = 0; const n = e.onBeforeRenderTargetsRenderObservable.add(() => { t = Yi.Now; }), i = e.onAfterRenderTargetsRenderObservable.add(() => { r = Yi.Now - t; }); return { id: "Render Targets", getData: () => r, dispose: () => { e.onBeforeRenderTargetsRenderObservable.remove(n), e.onAfterRenderTargetsRenderObservable.remove(i); } }; }; } /** * Gets the initializer for the strategy used for collection of particles time metrics. * @returns the initializer for the particles time strategy */ static ParticlesStrategy() { return (e) => { let t = Yi.Now, r = 0; const n = e.onBeforeParticlesRenderingObservable.add(() => { t = Yi.Now; }), i = e.onAfterParticlesRenderingObservable.add(() => { r = Yi.Now - t; }); return { id: "Particles", getData: () => r, dispose: () => { e.onBeforeParticlesRenderingObservable.remove(n), e.onAfterParticlesRenderingObservable.remove(i); } }; }; } /** * Gets the initializer for the strategy used for collection of sprites time metrics. * @returns the initializer for the sprites time strategy */ static SpritesStrategy() { return (e) => { var t, r; let n = Yi.Now, i = 0; const s = (t = e.onBeforeSpritesRenderingObservable) === null || t === void 0 ? void 0 : t.add(() => { n = Yi.Now; }), a = (r = e.onAfterSpritesRenderingObservable) === null || r === void 0 ? void 0 : r.add(() => { i = Yi.Now - n; }); return { id: "Sprites", getData: () => i, dispose: () => { var f, o; (f = e.onBeforeSpritesRenderingObservable) === null || f === void 0 || f.remove(s), (o = e.onAfterSpritesRenderingObservable) === null || o === void 0 || o.remove(a); } }; }; } /** * Gets the initializer for the strategy used for collection of animations time metrics. * @returns the initializer for the animations time strategy */ static AnimationsStrategy() { return (e) => { let t = Yi.Now, r = 0; const n = e.onBeforeAnimationsObservable.add(() => { t = Yi.Now; }), i = e.onAfterAnimationsObservable.add(() => { r = Yi.Now - t; }); return { id: "Animations", getData: () => r, dispose: () => { e.onBeforeAnimationsObservable.remove(n), e.onAfterAnimationsObservable.remove(i); } }; }; } /** * Gets the initializer for the strategy used for collection of physics time metrics. * @returns the initializer for the physics time strategy */ static PhysicsStrategy() { return (e) => { var t, r; let n = Yi.Now, i = 0; const s = (t = e.onBeforePhysicsObservable) === null || t === void 0 ? void 0 : t.add(() => { n = Yi.Now; }), a = (r = e.onAfterPhysicsObservable) === null || r === void 0 ? void 0 : r.add(() => { i = Yi.Now - n; }); return { id: "Physics", getData: () => i, dispose: () => { var f, o; (f = e.onBeforePhysicsObservable) === null || f === void 0 || f.remove(s), (o = e.onAfterPhysicsObservable) === null || o === void 0 || o.remove(a); } }; }; } /** * Gets the initializer for the strategy used for collection of render time metrics. * @returns the initializer for the render time strategy */ static RenderStrategy() { return (e) => { let t = Yi.Now, r = 0; const n = e.onBeforeDrawPhaseObservable.add(() => { t = Yi.Now; }), i = e.onAfterDrawPhaseObservable.add(() => { r = Yi.Now - t; }); return { id: "Render", getData: () => r, dispose: () => { e.onBeforeDrawPhaseObservable.remove(n), e.onAfterDrawPhaseObservable.remove(i); } }; }; } /** * Gets the initializer for the strategy used for collection of total frame time metrics. * @returns the initializer for the total frame time strategy */ static FrameTotalStrategy() { return (e) => { let t = Yi.Now, r = 0; const n = e.onBeforeAnimationsObservable.add(() => { t = Yi.Now; }), i = e.onAfterRenderObservable.add(() => { r = Yi.Now - t; }); return { id: "Frame Total", getData: () => r, dispose: () => { e.onBeforeAnimationsObservable.remove(n), e.onAfterRenderObservable.remove(i); } }; }; } /** * Gets the initializer for the strategy used for collection of inter-frame time metrics. * @returns the initializer for the inter-frame time strategy */ static InterFrameStrategy() { return (e) => { let t = Yi.Now, r = 0; const n = e.onBeforeAnimationsObservable.add(() => { r = Yi.Now - t; }), i = e.onAfterRenderObservable.add(() => { t = Yi.Now; }); return { id: "Inter-frame", getData: () => r, dispose: () => { e.onBeforeAnimationsObservable.remove(n), e.onAfterRenderObservable.remove(i); } }; }; } /** * Gets the initializer for the strategy used for collection of gpu frame time metrics. * @returns the initializer for the gpu frame time strategy */ static GpuFrameTimeStrategy() { return (e) => { const t = new Ete(e.getEngine()); return t.captureGPUFrameTime = !0, { id: "GPU frame time", getData: () => Math.max(t.gpuFrameTimeCounter.current * 1e-6, 0), dispose: () => { t.dispose(); } }; }; } } sr.prototype.getPerfCollector = function() { return this._perfCollector || (this._perfCollector = new b0(this)), this._perfCollector; }; function hge(A) { const e = new Array(), t = new Array(), r = new Array(), n = A.add(() => { const s = e.length; for (let a = 0; a < s; a++) FI(e.shift(), t.shift(), r.shift()); }); return { scheduler: (s, a, f) => { e.push(s), t.push(a), r.push(f); }, dispose: () => { A.remove(n); } }; } Oe.prototype.runCoroutineAsync = function(A) { if (!this._coroutineScheduler) { const e = hge(this); this._coroutineScheduler = e.scheduler, this._coroutineSchedulerDispose = e.dispose; } return BO(A, this._coroutineScheduler); }; Oe.prototype.cancelAllCoroutines = function() { this._coroutineSchedulerDispose && this._coroutineSchedulerDispose(), this._coroutineScheduler = void 0, this._coroutineSchedulerDispose = void 0; }; const Hge = "equirectangularPanoramaPixelShader", gge = `#ifdef GL_ES precision highp float; #endif #define M_PI 3.1415926535897932384626433832795 varying vec2 vUV;uniform samplerCube cubeMap;void main(void) {vec2 uv=vUV;float longitude=uv.x*2.*M_PI-M_PI+M_PI/2.;float latitude=(1.-uv.y)*M_PI;vec3 dir=vec3( - sin( longitude )*sin( latitude ), cos( latitude ), - cos( longitude )*sin( latitude ) );normalize( dir );gl_FragColor=textureCube( cubeMap,dir );}`; Le.ShadersStore[Hge] = gge; async function Xge(A, e) { var t, r; const n = (t = e.probe) !== null && t !== void 0 ? t : new Im("tempProbe", e.size, A), i = !!e.probe; i || (e.position ? n.position = e.position.clone() : A.activeCamera && (n.position = A.activeCamera.position.clone())); const s = e.meshesFilter ? A.meshes.filter(e.meshesFilter) : A.meshes; (r = n.renderList) === null || r === void 0 || r.push(...s), n.refreshRate = Ta.REFRESHRATE_RENDER_ONCE, n.cubeTexture.render(); const a = new rre("tempProceduralTexture", "equirectangularPanorama", { width: e.size * 2, height: e.size }, A); return a.setTexture("cubeMap", n.cubeTexture), new Promise((f, o) => { a.onGeneratedObservable.addOnce(() => { const d = a.readPixels(); if (!d) { o(new Error("No Pixel Data found on procedural texture")), a.dispose(), i || n.dispose(); return; } d.then((v) => { a.dispose(), i || n.dispose(), e.filename ? (eA.DumpData(e.size * 2, e.size, v, void 0, "image/png", e.filename), f(null)) : f(v); }); }); }); } class lq extends L9 { /** * Creates a new instance of the (legacy version) hit test feature * @param _xrSessionManager an instance of WebXRSessionManager * @param options options to use when constructing this feature */ constructor(e, t = {}) { super(e), this.options = t, this._direction = new S(0, 0, -1), this._mat = new he(), this._onSelectEnabled = !1, this._origin = new S(0, 0, 0), this.lastNativeXRHitResults = [], this.onHitTestResultObservable = new Oe(), this._onHitTestResults = (r) => { const n = r.map((i) => { const s = he.FromArray(i.hitMatrix); return this._xrSessionManager.scene.useRightHandedSystem || s.toggleModelMatrixHandInPlace(), this.options.worldParentNode && s.multiplyToRef(this.options.worldParentNode.getWorldMatrix(), s), { xrHitResult: i, transformationMatrix: s }; }); this.lastNativeXRHitResults = r, this.onHitTestResultObservable.notifyObservers(n); }, this._onSelect = (r) => { this._onSelectEnabled && lq.XRHitTestWithSelectEvent(r, this._xrSessionManager.referenceSpace); }, this.xrNativeFeatureName = "hit-test", ye.Warn("A newer version of this plugin is available"); } /** * execute a hit test with an XR Ray * * @param xrSession a native xrSession that will execute this hit test * @param xrRay the ray (position and direction) to use for ray-casting * @param referenceSpace native XR reference space to use for the hit-test * @param filter filter function that will filter the results * @returns a promise that resolves with an array of native XR hit result in xr coordinates system */ static XRHitTestWithRay(e, t, r, n) { return e.requestHitTest(t, r).then((i) => { const s = n || ((a) => !!a.hitMatrix); return i.filter(s); }); } /** * Execute a hit test on the current running session using a select event returned from a transient input (such as touch) * @param event the (select) event to use to select with * @param referenceSpace the reference space to use for this hit test * @returns a promise that resolves with an array of native XR hit result in xr coordinates system */ static XRHitTestWithSelectEvent(e, t) { const r = e.frame.getPose(e.inputSource.targetRaySpace, t); if (!r) return Promise.resolve([]); const n = new XRRay(r.transform); return this.XRHitTestWithRay(e.frame.session, n, t); } /** * attach this feature * Will usually be called by the features manager * * @returns true if successful. */ attach() { return super.attach() ? (this.options.testOnPointerDownOnly && this._xrSessionManager.session.addEventListener("select", this._onSelect, !1), !0) : !1; } /** * detach this feature. * Will usually be called by the features manager * * @returns true if successful. */ detach() { return super.detach() ? (this._onSelectEnabled = !1, this._xrSessionManager.session.removeEventListener("select", this._onSelect), !0) : !1; } /** * Dispose this feature and all of the resources attached */ dispose() { super.dispose(), this.onHitTestResultObservable.clear(); } _onXRFrame(e) { if (!this.attached || this.options.testOnPointerDownOnly) return; const t = e.getViewerPose(this._xrSessionManager.referenceSpace); if (!t) return; he.FromArrayToRef(t.transform.matrix, 0, this._mat), S.TransformCoordinatesFromFloatsToRef(0, 0, 0, this._mat, this._origin), S.TransformCoordinatesFromFloatsToRef(0, 0, -1, this._mat, this._direction), this._direction.subtractInPlace(this._origin), this._direction.normalize(); const r = new XRRay({ x: this._origin.x, y: this._origin.y, z: this._origin.z, w: 0 }, { x: this._direction.x, y: this._direction.y, z: this._direction.z, w: 0 }); lq.XRHitTestWithRay(this._xrSessionManager.session, r, this._xrSessionManager.referenceSpace).then(this._onHitTestResults); } } lq.Name = Gi.HIT_TEST; lq.Version = 1; So.AddWebXRFeature(lq.Name, (A, e) => () => new lq(A, e), lq.Version, !1); let Tge = 0; class aS extends L9 { /** * Set the reference space to use for anchor creation, when not using a hit test. * Will default to the session's reference space if not defined */ set referenceSpaceForFrameAnchors(e) { this._referenceSpaceForFrameAnchors = e; } /** * constructs a new anchor system * @param _xrSessionManager an instance of WebXRSessionManager * @param _options configuration object for this feature */ constructor(e, t = {}) { super(e), this._options = t, this._lastFrameDetected = /* @__PURE__ */ new Set(), this._trackedAnchors = [], this._futureAnchors = [], this.onAnchorAddedObservable = new Oe(), this.onAnchorRemovedObservable = new Oe(), this.onAnchorUpdatedObservable = new Oe(), this._tmpVector = new S(), this._tmpQuaternion = new Ze(), this.xrNativeFeatureName = "anchors"; } _populateTmpTransformation(e, t) { return this._tmpVector.copyFrom(e), this._tmpQuaternion.copyFrom(t), this._xrSessionManager.scene.useRightHandedSystem || (this._tmpVector.z *= -1, this._tmpQuaternion.z *= -1, this._tmpQuaternion.w *= -1), { position: this._tmpVector, rotationQuaternion: this._tmpQuaternion }; } /** * Create a new anchor point using a hit test result at a specific point in the scene * An anchor is tracked only after it is added to the trackerAnchors in xrFrame. The promise returned here does not yet guaranty that. * Use onAnchorAddedObservable to get newly added anchors if you require tracking guaranty. * * @param hitTestResult The hit test result to use for this anchor creation * @param position an optional position offset for this anchor * @param rotationQuaternion an optional rotation offset for this anchor * @returns A promise that fulfills when babylon has created the corresponding WebXRAnchor object and tracking has begun */ async addAnchorPointUsingHitTestResultAsync(e, t = new S(), r = new Ze()) { this._populateTmpTransformation(t, r); const n = new XRRigidTransform({ x: this._tmpVector.x, y: this._tmpVector.y, z: this._tmpVector.z }, { x: this._tmpQuaternion.x, y: this._tmpQuaternion.y, z: this._tmpQuaternion.z, w: this._tmpQuaternion.w }); if (e.xrHitResult.createAnchor) try { const i = await e.xrHitResult.createAnchor(n); return new Promise((s, a) => { this._futureAnchors.push({ nativeAnchor: i, resolved: !1, submitted: !0, xrTransformation: n, resolve: s, reject: a }); }); } catch (i) { throw new Error(i); } else throw this.detach(), new Error("Anchors not enabled in this environment/browser"); } /** * Add a new anchor at a specific position and rotation * This function will add a new anchor per default in the next available frame. Unless forced, the createAnchor function * will be called in the next xrFrame loop to make sure that the anchor can be created correctly. * An anchor is tracked only after it is added to the trackerAnchors in xrFrame. The promise returned here does not yet guaranty that. * Use onAnchorAddedObservable to get newly added anchors if you require tracking guaranty. * * @param position the position in which to add an anchor * @param rotationQuaternion an optional rotation for the anchor transformation * @param forceCreateInCurrentFrame force the creation of this anchor in the current frame. Must be called inside xrFrame loop! * @returns A promise that fulfills when babylon has created the corresponding WebXRAnchor object and tracking has begun */ async addAnchorAtPositionAndRotationAsync(e, t = new Ze(), r = !1) { this._populateTmpTransformation(e, t); const n = new XRRigidTransform({ x: this._tmpVector.x, y: this._tmpVector.y, z: this._tmpVector.z }, { x: this._tmpQuaternion.x, y: this._tmpQuaternion.y, z: this._tmpQuaternion.z, w: this._tmpQuaternion.w }), i = r && this.attached && this._xrSessionManager.currentFrame ? await this._createAnchorAtTransformation(n, this._xrSessionManager.currentFrame) : void 0; return new Promise((s, a) => { this._futureAnchors.push({ nativeAnchor: i, resolved: !1, submitted: !1, xrTransformation: n, resolve: s, reject: a }); }); } /** * Get the list of anchors currently being tracked by the system */ get anchors() { return this._trackedAnchors; } /** * detach this feature. * Will usually be called by the features manager * * @returns true if successful. */ detach() { if (!super.detach()) return !1; if (!this._options.doNotRemoveAnchorsOnSessionEnded) for (; this._trackedAnchors.length; ) { const e = this._trackedAnchors.pop(); if (e) { try { e.remove(); } catch { } this.onAnchorRemovedObservable.notifyObservers(e); } } return !0; } /** * Dispose this feature and all of the resources attached */ dispose() { this._futureAnchors.length = 0, super.dispose(), this.onAnchorAddedObservable.clear(), this.onAnchorRemovedObservable.clear(), this.onAnchorUpdatedObservable.clear(); } _onXRFrame(e) { if (!this.attached || !e) return; const t = e.trackedAnchors; if (t) { const r = this._trackedAnchors.filter((i) => !t.has(i.xrAnchor)).map((i) => this._trackedAnchors.indexOf(i)); let n = 0; r.forEach((i) => { const s = this._trackedAnchors.splice(i - n, 1)[0]; this.onAnchorRemovedObservable.notifyObservers(s), n++; }), t.forEach((i) => { if (this._lastFrameDetected.has(i)) { const s = this._findIndexInAnchorArray(i), a = this._trackedAnchors[s]; try { this._updateAnchorWithXRFrame(i, a, e), a.attachedNode && (a.attachedNode.rotationQuaternion = a.attachedNode.rotationQuaternion || new Ze(), a.transformationMatrix.decompose(a.attachedNode.scaling, a.attachedNode.rotationQuaternion, a.attachedNode.position)), this.onAnchorUpdatedObservable.notifyObservers(a); } catch { ye.Warn("Anchor could not be updated"); } } else { const s = { id: Tge++, xrAnchor: i, remove: () => i.delete() }, a = this._updateAnchorWithXRFrame(i, s, e); this._trackedAnchors.push(a), this.onAnchorAddedObservable.notifyObservers(a); const o = this._futureAnchors.filter((d) => d.nativeAnchor === i)[0]; o && (o.resolve(a), o.resolved = !0); } }), this._lastFrameDetected = t; } this._futureAnchors.forEach((r) => { !r.resolved && !r.submitted && (this._createAnchorAtTransformation(r.xrTransformation, e).then((n) => { r.nativeAnchor = n; }, (n) => { r.resolved = !0, r.reject(n); }), r.submitted = !0); }); } /** * avoiding using Array.find for global support. * @param xrAnchor the plane to find in the array */ _findIndexInAnchorArray(e) { for (let t = 0; t < this._trackedAnchors.length; ++t) if (this._trackedAnchors[t].xrAnchor === e) return t; return -1; } _updateAnchorWithXRFrame(e, t, r) { const n = r.getPose(e.anchorSpace, this._xrSessionManager.referenceSpace); if (n) { const i = t.transformationMatrix || new he(); he.FromArrayToRef(n.transform.matrix, 0, i), this._xrSessionManager.scene.useRightHandedSystem || i.toggleModelMatrixHandInPlace(), t.transformationMatrix = i, this._options.worldParentNode && i.multiplyToRef(this._options.worldParentNode.getWorldMatrix(), i); } return t; } async _createAnchorAtTransformation(e, t) { var r; if (t.createAnchor) try { return t.createAnchor(e, (r = this._referenceSpaceForFrameAnchors) !== null && r !== void 0 ? r : this._xrSessionManager.referenceSpace); } catch (n) { throw new Error(n); } else throw this.detach(), new Error("Anchors are not enabled in your browser"); } } aS.Name = Gi.ANCHOR_SYSTEM; aS.Version = 1; So.AddWebXRFeature(aS.Name, (A, e) => () => new aS(A, e), aS.Version); let qge = 0; class oS extends L9 { /** * construct a new Plane Detector * @param _xrSessionManager an instance of xr Session manager * @param _options configuration to use when constructing this feature */ constructor(e, t = {}) { super(e), this._options = t, this._detectedPlanes = [], this._enabled = !1, this._lastFrameDetected = /* @__PURE__ */ new Set(), this.onPlaneAddedObservable = new Oe(), this.onPlaneRemovedObservable = new Oe(), this.onPlaneUpdatedObservable = new Oe(), this.xrNativeFeatureName = "plane-detection", this._xrSessionManager.session ? this._init() : this._xrSessionManager.onXRSessionInit.addOnce(() => { this._init(); }); } /** * detach this feature. * Will usually be called by the features manager * * @returns true if successful. */ detach() { if (!super.detach()) return !1; if (!this._options.doNotRemovePlanesOnSessionEnded) for (; this._detectedPlanes.length; ) { const e = this._detectedPlanes.pop(); e && this.onPlaneRemovedObservable.notifyObservers(e); } return !0; } /** * Dispose this feature and all of the resources attached */ dispose() { super.dispose(), this.onPlaneAddedObservable.clear(), this.onPlaneRemovedObservable.clear(), this.onPlaneUpdatedObservable.clear(); } /** * Check if the needed objects are defined. * This does not mean that the feature is enabled, but that the objects needed are well defined. */ isCompatible() { return typeof XRPlane < "u"; } /** * Enable room capture mode. * When enabled and supported by the system, * the detectedPlanes array will be populated with the detected room boundaries * @see https://immersive-web.github.io/real-world-geometry/plane-detection.html#dom-xrsession-initiateroomcapture * @returns true if plane detection is enabled and supported. Will reject if not supported. */ async initiateRoomCapture() { return this._xrSessionManager.session.initiateRoomCapture ? this._xrSessionManager.session.initiateRoomCapture() : Promise.reject("initiateRoomCapture is not supported on this session"); } _onXRFrame(e) { var t; if (!this.attached || !this._enabled || !e) return; const r = e.detectedPlanes || ((t = e.worldInformation) === null || t === void 0 ? void 0 : t.detectedPlanes); if (r) { for (let n = 0; n < this._detectedPlanes.length; n++) { const i = this._detectedPlanes[n]; r.has(i.xrPlane) || (this._detectedPlanes.splice(n--, 1), this.onPlaneRemovedObservable.notifyObservers(i)); } r.forEach((n) => { if (this._lastFrameDetected.has(n)) { if (n.lastChangedTime === this._xrSessionManager.currentTimestamp) { const i = this._findIndexInPlaneArray(n), s = this._detectedPlanes[i]; this._updatePlaneWithXRPlane(n, s, e), this.onPlaneUpdatedObservable.notifyObservers(s); } } else { const i = { id: qge++, xrPlane: n, polygonDefinition: [] }, s = this._updatePlaneWithXRPlane(n, i, e); this._detectedPlanes.push(s), this.onPlaneAddedObservable.notifyObservers(s); } }), this._lastFrameDetected = r; } } _init() { const e = () => { this._enabled = !0, this._detectedPlanes.length && (this._detectedPlanes.length = 0); }; if (this._xrSessionManager.isNative && this._options.preferredDetectorOptions && this._xrSessionManager.session.trySetPreferredPlaneDetectorOptions && this._xrSessionManager.session.trySetPreferredPlaneDetectorOptions(this._options.preferredDetectorOptions), !this._xrSessionManager.session.updateWorldTrackingState) { e(); return; } this._xrSessionManager.session.updateWorldTrackingState({ planeDetectionState: { enabled: !0 } }), e(); } _updatePlaneWithXRPlane(e, t, r) { t.polygonDefinition = e.polygon.map((i) => { const s = this._xrSessionManager.scene.useRightHandedSystem ? 1 : -1; return new S(i.x, i.y, i.z * s); }); const n = r.getPose(e.planeSpace, this._xrSessionManager.referenceSpace); if (n) { const i = t.transformationMatrix || new he(); he.FromArrayToRef(n.transform.matrix, 0, i), this._xrSessionManager.scene.useRightHandedSystem || i.toggleModelMatrixHandInPlace(), t.transformationMatrix = i, this._options.worldParentNode && i.multiplyToRef(this._options.worldParentNode.getWorldMatrix(), i); } return t; } /** * avoiding using Array.find for global support. * @param xrPlane the plane to find in the array */ _findIndexInPlaneArray(e) { for (let t = 0; t < this._detectedPlanes.length; ++t) if (this._detectedPlanes[t].xrPlane === e) return t; return -1; } } oS.Name = Gi.PLANE_DETECTION; oS.Version = 1; So.AddWebXRFeature(oS.Name, (A, e) => () => new oS(A, e), oS.Version); class fS extends L9 { /** * constructs a new background remover module * @param _xrSessionManager the session manager for this module * @param options read-only options to be used in this module */ constructor(e, t = {}) { super(e), this.options = t, this.onBackgroundStateChangedObservable = new Oe(); } /** * attach this feature * Will usually be called by the features manager * * @returns true if successful. */ attach() { return this._setBackgroundState(!1), super.attach(); } /** * detach this feature. * Will usually be called by the features manager * * @returns true if successful. */ detach() { return this._setBackgroundState(!0), super.detach(); } /** * Dispose this feature and all of the resources attached */ dispose() { super.dispose(), this.onBackgroundStateChangedObservable.clear(); } _onXRFrame(e) { } _setBackgroundState(e) { const t = this._xrSessionManager.scene; if (!this.options.ignoreEnvironmentHelper) if (this.options.environmentHelperRemovalFlags) { if (this.options.environmentHelperRemovalFlags.skyBox) { const r = t.getMeshByName("BackgroundSkybox"); r && r.setEnabled(e); } if (this.options.environmentHelperRemovalFlags.ground) { const r = t.getMeshByName("BackgroundPlane"); r && r.setEnabled(e); } } else { const r = t.getMeshByName("BackgroundHelper"); r && r.setEnabled(e); } this.options.backgroundMeshes && this.options.backgroundMeshes.forEach((r) => r.setEnabled(e)), this.onBackgroundStateChangedObservable.notifyObservers(e); } } fS.Name = Gi.BACKGROUND_REMOVER; fS.Version = 1; So.AddWebXRFeature(fS.Name, (A, e) => () => new fS(A, e), fS.Version, !0); class bge { } class AS extends L9 { _createPhysicsImpostor(e) { const t = this._options.physicsProperties.impostorType || tn.SphereImpostor, r = this._options.physicsProperties.impostorSize || 0.1, n = UA("impostor-mesh-" + e.uniqueId, { diameterX: typeof r == "number" ? r : r.width, diameterY: typeof r == "number" ? r : r.height, diameterZ: typeof r == "number" ? r : r.depth }); n.isVisible = this._debugMode, n.isPickable = !1, n.rotationQuaternion = new Ze(); const i = e.grip || e.pointer; n.position.copyFrom(i.position), n.rotationQuaternion.copyFrom(i.rotationQuaternion); const s = new tn(n, t, Object.assign({ mass: 0 }, this._options.physicsProperties)); this._controllers[e.uniqueId] = { xrController: e, impostor: s, impostorMesh: n }; } /** * Construct a new Controller Physics Feature * @param _xrSessionManager the corresponding xr session manager * @param _options options to create this feature with */ constructor(e, t) { super(e), this._options = t, this._attachController = (r) => { this._controllers[r.uniqueId] || (this._xrSessionManager.scene.isPhysicsEnabled() || Se.Warn("physics engine not enabled, skipped. Please add this controller manually."), this._options.physicsProperties.useControllerMesh && r.inputSource.gamepad ? r.onMotionControllerInitObservable.addOnce((n) => { n._doNotLoadControllerMesh ? this._createPhysicsImpostor(r) : n.onModelLoadedObservable.addOnce(() => { const i = new tn(n.rootMesh, tn.MeshImpostor, Object.assign({ mass: 0 }, this._options.physicsProperties)), s = r.grip || r.pointer; this._controllers[r.uniqueId] = { xrController: r, impostor: i, oldPos: s.position.clone(), oldRotation: s.rotationQuaternion.clone() }; }); }) : this._createPhysicsImpostor(r)); }, this._controllers = {}, this._debugMode = !1, this._delta = 0, this._lastTimestamp = 0, this._tmpQuaternion = new Ze(), this._tmpVector = new S(), this._options.physicsProperties || (this._options.physicsProperties = {}); } /** * @internal * enable debugging - will show console outputs and the impostor mesh */ _enablePhysicsDebug() { this._debugMode = !0, Object.keys(this._controllers).forEach((e) => { const t = this._controllers[e]; t.impostorMesh && (t.impostorMesh.isVisible = !0); }); } /** * Manually add a controller (if no xrInput was provided or physics engine was not enabled) * @param xrController the controller to add */ addController(e) { this._attachController(e); } /** * attach this feature * Will usually be called by the features manager * * @returns true if successful. */ attach() { if (!super.attach()) return !1; if (!this._options.xrInput) return !0; if (this._options.xrInput.controllers.forEach(this._attachController), this._addNewAttachObserver(this._options.xrInput.onControllerAddedObservable, this._attachController), this._addNewAttachObserver(this._options.xrInput.onControllerRemovedObservable, (e) => { this._detachController(e.uniqueId); }), this._options.enableHeadsetImpostor) { const e = this._options.headsetImpostorParams || { impostorType: tn.SphereImpostor, restitution: 0.8, impostorSize: 0.3 }, t = e.impostorSize || 0.3; this._headsetMesh = UA("headset-mesh", { diameterX: typeof t == "number" ? t : t.width, diameterY: typeof t == "number" ? t : t.height, diameterZ: typeof t == "number" ? t : t.depth }), this._headsetMesh.rotationQuaternion = new Ze(), this._headsetMesh.isVisible = !1, this._headsetImpostor = new tn(this._headsetMesh, e.impostorType, Object.assign({ mass: 0 }, e)); } return !0; } /** * detach this feature. * Will usually be called by the features manager * * @returns true if successful. */ detach() { return super.detach() ? (Object.keys(this._controllers).forEach((e) => { this._detachController(e); }), this._headsetMesh && this._headsetMesh.dispose(), !0) : !1; } /** * Get the headset impostor, if enabled * @returns the impostor */ getHeadsetImpostor() { return this._headsetImpostor; } /** * Get the physics impostor of a specific controller. * The impostor is not attached to a mesh because a mesh for each controller is not obligatory * @param controller the controller or the controller id of which to get the impostor * @returns the impostor or null */ getImpostorForController(e) { const t = typeof e == "string" ? e : e.uniqueId; return this._controllers[t] ? this._controllers[t].impostor : null; } /** * Update the physics properties provided in the constructor * @param newProperties the new properties object * @param newProperties.impostorType * @param newProperties.impostorSize * @param newProperties.friction * @param newProperties.restitution */ setPhysicsProperties(e) { this._options.physicsProperties = Object.assign(Object.assign({}, this._options.physicsProperties), e); } _onXRFrame(e) { var t, r; if (this._delta = this._xrSessionManager.currentTimestamp - this._lastTimestamp, this._lastTimestamp = this._xrSessionManager.currentTimestamp, this._headsetMesh && this._headsetImpostor) { if (this._headsetMesh.position.copyFrom(this._options.xrInput.xrCamera.globalPosition), this._headsetMesh.rotationQuaternion.copyFrom(this._options.xrInput.xrCamera.absoluteRotation), !((t = this._options.xrInput.xrCamera._lastXRViewerPose) === null || t === void 0) && t.linearVelocity) { const n = this._options.xrInput.xrCamera._lastXRViewerPose.linearVelocity; this._tmpVector.set(n.x, n.y, n.z), this._headsetImpostor.setLinearVelocity(this._tmpVector); } if (!((r = this._options.xrInput.xrCamera._lastXRViewerPose) === null || r === void 0) && r.angularVelocity) { const n = this._options.xrInput.xrCamera._lastXRViewerPose.angularVelocity; this._tmpVector.set(n.x, n.y, n.z), this._headsetImpostor.setAngularVelocity(this._tmpVector); } } Object.keys(this._controllers).forEach((n) => { var i, s; const a = this._controllers[n], f = a.xrController.grip || a.xrController.pointer, o = a.oldPos || a.impostorMesh.position; if (!((i = a.xrController._lastXRPose) === null || i === void 0) && i.linearVelocity) { const v = a.xrController._lastXRPose.linearVelocity; this._tmpVector.set(v.x, v.y, v.z), a.impostor.setLinearVelocity(this._tmpVector); } else f.position.subtractToRef(o, this._tmpVector), this._tmpVector.scaleInPlace(1e3 / this._delta), a.impostor.setLinearVelocity(this._tmpVector); o.copyFrom(f.position), this._debugMode && console.log(this._tmpVector, "linear"); const d = a.oldRotation || a.impostorMesh.rotationQuaternion; if (!((s = a.xrController._lastXRPose) === null || s === void 0) && s.angularVelocity) { const v = a.xrController._lastXRPose.angularVelocity; this._tmpVector.set(v.x, v.y, v.z), a.impostor.setAngularVelocity(this._tmpVector); } else if (!d.equalsWithEpsilon(f.rotationQuaternion)) { d.conjugateInPlace().multiplyToRef(f.rotationQuaternion, this._tmpQuaternion); const v = Math.sqrt(this._tmpQuaternion.x * this._tmpQuaternion.x + this._tmpQuaternion.y * this._tmpQuaternion.y + this._tmpQuaternion.z * this._tmpQuaternion.z); if (this._tmpVector.set(this._tmpQuaternion.x, this._tmpQuaternion.y, this._tmpQuaternion.z), v < 1e-3) this._tmpVector.scaleInPlace(2); else { const u = 2 * Math.atan2(v, this._tmpQuaternion.w); this._tmpVector.scaleInPlace(u / (v * (this._delta / 1e3))); } a.impostor.setAngularVelocity(this._tmpVector); } d.copyFrom(f.rotationQuaternion), this._debugMode && console.log(this._tmpVector, this._tmpQuaternion, "angular"); }); } _detachController(e) { const t = this._controllers[e]; t && (t.impostorMesh && t.impostorMesh.dispose(), delete this._controllers[e]); } } AS.Name = Gi.PHYSICS_CONTROLLERS; AS.Version = 1; So.AddWebXRFeature(AS.Name, (A, e) => () => new AS(A, e), AS.Version, !0); class dS extends L9 { /** * Creates a new instance of the hit test feature * @param _xrSessionManager an instance of WebXRSessionManager * @param options options to use when constructing this feature */ constructor(e, t = {}) { super(e), this.options = t, this._tmpMat = new he(), this._tmpPos = new S(), this._tmpQuat = new Ze(), this._initHitTestSource = (r) => { if (!r) return; const n = new XRRay(this.options.offsetRay || {}), i = { space: this.options.useReferenceSpace ? r : this._xrSessionManager.viewerReferenceSpace, offsetRay: n }; if (this.options.entityTypes && (i.entityTypes = this.options.entityTypes), !i.space) { ye.Warn("waiting for viewer reference space to initialize"); return; } this._xrSessionManager.session.requestHitTestSource(i).then((s) => { this._xrHitTestSource && this._xrHitTestSource.cancel(), this._xrHitTestSource = s; }); }, this.autoCloneTransformation = !1, this.onHitTestResultObservable = new Oe(), this.paused = !1, this.xrNativeFeatureName = "hit-test", ye.Warn("Hit test is an experimental and unstable feature."); } /** * attach this feature * Will usually be called by the features manager * * @returns true if successful. */ attach() { if (!super.attach() || !this._xrSessionManager.session.requestHitTestSource) return !1; if (this.options.disablePermanentHitTest || (this._xrSessionManager.referenceSpace && this._initHitTestSource(this._xrSessionManager.referenceSpace), this._xrSessionManager.onXRReferenceSpaceChanged.add(this._initHitTestSource)), this.options.enableTransientHitTest) { const e = new XRRay(this.options.transientOffsetRay || {}); this._xrSessionManager.session.requestHitTestSourceForTransientInput({ profile: this.options.transientHitTestProfile || "generic-touchscreen", offsetRay: e, entityTypes: this.options.entityTypes }).then((t) => { this._transientXrHitTestSource = t; }); } return !0; } /** * detach this feature. * Will usually be called by the features manager * * @returns true if successful. */ detach() { return super.detach() ? (this._xrHitTestSource && (this._xrHitTestSource.cancel(), this._xrHitTestSource = null), this._xrSessionManager.onXRReferenceSpaceChanged.removeCallback(this._initHitTestSource), this._transientXrHitTestSource && (this._transientXrHitTestSource.cancel(), this._transientXrHitTestSource = null), !0) : !1; } /** * Dispose this feature and all of the resources attached */ dispose() { super.dispose(), this.onHitTestResultObservable.clear(); } _onXRFrame(e) { if (!(!this.attached || this.paused)) { if (this._xrHitTestSource) { const t = e.getHitTestResults(this._xrHitTestSource); this._processWebXRHitTestResult(t); } this._transientXrHitTestSource && e.getHitTestResultsForTransientInput(this._transientXrHitTestSource).forEach((r) => { this._processWebXRHitTestResult(r.results, r.inputSource); }); } } _processWebXRHitTestResult(e, t) { const r = []; e.forEach((n) => { const i = n.getPose(this._xrSessionManager.referenceSpace); if (!i) return; const s = i.transform.position, a = i.transform.orientation; this._tmpPos.set(s.x, s.y, s.z), this._tmpQuat.set(a.x, a.y, a.z, a.w), he.FromFloat32ArrayToRefScaled(i.transform.matrix, 0, 1, this._tmpMat), this._xrSessionManager.scene.useRightHandedSystem || (this._tmpPos.z *= -1, this._tmpQuat.z *= -1, this._tmpQuat.w *= -1, this._tmpMat.toggleModelMatrixHandInPlace()); const f = { position: this.autoCloneTransformation ? this._tmpPos.clone() : this._tmpPos, rotationQuaternion: this.autoCloneTransformation ? this._tmpQuat.clone() : this._tmpQuat, transformationMatrix: this.autoCloneTransformation ? this._tmpMat.clone() : this._tmpMat, inputSource: t, isTransient: !!t, xrHitResult: n }; r.push(f); }), this.onHitTestResultObservable.notifyObservers(r); } } dS.Name = Gi.HIT_TEST; dS.Version = 2; So.AddWebXRFeature(dS.Name, (A, e) => () => new dS(A, e), dS.Version, !1); class vS extends L9 { /** * The current feature point cloud maintained across frames. */ get featurePointCloud() { return this._featurePointCloud; } /** * construct the feature point system * @param _xrSessionManager an instance of xr Session manager */ constructor(e) { super(e), this._enabled = !1, this._featurePointCloud = [], this.onFeaturePointsAddedObservable = new Oe(), this.onFeaturePointsUpdatedObservable = new Oe(), this.xrNativeFeatureName = "bjsfeature-points", this._xrSessionManager.session ? this._init() : this._xrSessionManager.onXRSessionInit.addOnce(() => { this._init(); }); } /** * Detach this feature. * Will usually be called by the features manager * * @returns true if successful. */ detach() { return super.detach() ? (this.featurePointCloud.length = 0, !0) : !1; } /** * Dispose this feature and all of the resources attached */ dispose() { super.dispose(), this._featurePointCloud.length = 0, this.onFeaturePointsUpdatedObservable.clear(), this.onFeaturePointsAddedObservable.clear(); } /** * On receiving a new XR frame if this feature is attached notify observers new feature point data is available. * @param frame */ _onXRFrame(e) { if (!this.attached || !this._enabled || !e) return; const t = e.featurePointCloud; if (!(!t || t.length === 0)) { if (t.length % 5 !== 0) throw new Error("Received malformed feature point cloud of length: " + t.length); const r = t.length / 5, n = [], i = []; for (let s = 0; s < r; s++) { const a = s * 5, f = t[a + 4]; this._featurePointCloud[f] ? n.push(f) : (this._featurePointCloud[f] = { position: new S(), confidenceValue: 0 }, i.push(f)), this._featurePointCloud[f].position.x = t[a], this._featurePointCloud[f].position.y = t[a + 1], this._featurePointCloud[f].position.z = t[a + 2], this._featurePointCloud[f].confidenceValue = t[a + 3]; } i.length > 0 && this.onFeaturePointsAddedObservable.notifyObservers(i), n.length > 0 && this.onFeaturePointsUpdatedObservable.notifyObservers(n); } } /** * Initializes the feature. If the feature point feature is not available for this environment do not mark the feature as enabled. */ _init() { !this._xrSessionManager.session.trySetFeaturePointCloudEnabled || !this._xrSessionManager.session.trySetFeaturePointCloudEnabled(!0) || (this._enabled = !0); } } vS.Name = Gi.FEATURE_POINTS; vS.Version = 1; So.AddWebXRFeature(vS.Name, (A) => () => new vS(A), vS.Version); let xge = 0; class uS extends L9 { constructor(e, t = {}) { super(e), this._options = t, this._detectedMeshes = /* @__PURE__ */ new Map(), this.onMeshAddedObservable = new Oe(), this.onMeshRemovedObservable = new Oe(), this.onMeshUpdatedObservable = new Oe(), this.xrNativeFeatureName = "mesh-detection", this._xrSessionManager.session ? this._init() : this._xrSessionManager.onXRSessionInit.addOnce(() => { this._init(); }); } detach() { return super.detach() ? (this._xrSessionManager.isNative && this._xrSessionManager.session.trySetMeshDetectorEnabled && this._xrSessionManager.session.trySetMeshDetectorEnabled(!1), this._options.doNotRemoveMeshesOnSessionEnded || (this._detectedMeshes.forEach((e) => { this.onMeshRemovedObservable.notifyObservers(e); }), this._detectedMeshes.clear()), !0) : !1; } dispose() { super.dispose(), this.onMeshAddedObservable.clear(), this.onMeshRemovedObservable.clear(), this.onMeshUpdatedObservable.clear(); } _onXRFrame(e) { var t; try { if (!this.attached || !e) return; const r = (t = e.worldInformation) === null || t === void 0 ? void 0 : t.detectedMeshes; if (r) { const n = /* @__PURE__ */ new Set(); this._detectedMeshes.forEach((i, s) => { r.has(s) || n.add(s); }), n.forEach((i) => { const s = this._detectedMeshes.get(i); s && (this.onMeshRemovedObservable.notifyObservers(s), this._detectedMeshes.delete(i)); }), r.forEach((i) => { if (this._detectedMeshes.has(i)) { if (i.lastChangedTime === this._xrSessionManager.currentTimestamp) { const s = this._detectedMeshes.get(i); s && (this._updateVertexDataWithXRMesh(i, s, e), this.onMeshUpdatedObservable.notifyObservers(s)); } } else { const s = { id: xge++, xrMesh: i }, a = this._updateVertexDataWithXRMesh(i, s, e); this._detectedMeshes.set(i, a), this.onMeshAddedObservable.notifyObservers(a); } }); } } catch (r) { console.log(r.stack); } } _init() { this._xrSessionManager.isNative && (this._xrSessionManager.session.trySetMeshDetectorEnabled && this._xrSessionManager.session.trySetMeshDetectorEnabled(!0), this._options.preferredDetectorOptions && this._xrSessionManager.session.trySetPreferredMeshDetectorOptions && this._xrSessionManager.session.trySetPreferredMeshDetectorOptions(this._options.preferredDetectorOptions)); } _updateVertexDataWithXRMesh(e, t, r) { if (t.xrMesh = e, t.worldParentNode = this._options.worldParentNode, this._options.convertCoordinateSystems) { if (this._xrSessionManager.scene.useRightHandedSystem) t.positions = e.positions, t.normals = e.normals; else { t.positions = new Float32Array(e.positions.length); for (let i = 0; i < e.positions.length; i += 3) t.positions[i] = e.positions[i], t.positions[i + 1] = e.positions[i + 1], t.positions[i + 2] = -1 * e.positions[i + 2]; if (e.normals) { t.normals = new Float32Array(e.normals.length); for (let i = 0; i < e.normals.length; i += 3) t.normals[i] = e.normals[i], t.normals[i + 1] = e.normals[i + 1], t.normals[i + 2] = -1 * e.normals[i + 2]; } } t.indices = e.indices; const n = r.getPose(e.meshSpace, this._xrSessionManager.referenceSpace); if (n) { const i = t.transformationMatrix || new he(); he.FromArrayToRef(n.transform.matrix, 0, i), this._xrSessionManager.scene.useRightHandedSystem || i.toggleModelMatrixHandInPlace(), t.transformationMatrix = i, this._options.worldParentNode && i.multiplyToRef(this._options.worldParentNode.getWorldMatrix(), i); } } return t; } } uS.Name = Gi.MESH_DETECTION; uS.Version = 1; So.AddWebXRFeature(uS.Name, (A, e) => () => new uS(A, e), uS.Version, !1); var W2; (function(A) { A[A.NotReceived = 0] = "NotReceived", A[A.Waiting = 1] = "Waiting", A[A.Received = 2] = "Received"; })(W2 || (W2 = {})); class lS extends L9 { /** * constructs the image tracking feature * @param _xrSessionManager the session manager for this module * @param options read-only options to be used in this module */ constructor(e, t) { super(e), this.options = t, this.onUntrackableImageFoundObservable = new Oe(), this.onTrackableImageFoundObservable = new Oe(), this.onTrackedImageUpdatedObservable = new Oe(), this._trackableScoreStatus = W2.NotReceived, this._trackedImages = [], this.xrNativeFeatureName = "image-tracking"; } /** * attach this feature * Will usually be called by the features manager * * @returns true if successful. */ attach() { return super.attach(); } /** * detach this feature. * Will usually be called by the features manager * * @returns true if successful. */ detach() { return super.detach(); } /** * Get a tracked image by its ID. * * @param id the id of the image to load (position in the init array) * @returns a trackable image, if exists in this location */ getTrackedImageById(e) { return this._trackedImages[e] || null; } /** * Dispose this feature and all of the resources attached */ dispose() { super.dispose(), this._trackedImages.forEach((e) => { e.originalBitmap.close(); }), this._trackedImages.length = 0, this.onTrackableImageFoundObservable.clear(), this.onUntrackableImageFoundObservable.clear(), this.onTrackedImageUpdatedObservable.clear(); } /** * Extends the session init object if needed * @returns augmentation object fo the xr session init object. */ async getXRSessionInitExtension() { if (!this.options.images || !this.options.images.length) return {}; const e = this.options.images.map((t) => typeof t.src == "string" ? this._xrSessionManager.scene.getEngine()._createImageBitmapFromSource(t.src) : Promise.resolve(t.src)); try { const t = await Promise.all(e); return this._originalTrackingRequest = t.map((r, n) => ({ image: r, widthInMeters: this.options.images[n].estimatedRealWorldWidth })), { trackedImages: this._originalTrackingRequest }; } catch { return ye.Error("Error loading images for tracking, WebXRImageTracking disabled for this session."), {}; } } _onXRFrame(e) { if (!e.getImageTrackingResults || this._trackableScoreStatus === W2.Waiting) return; if (this._trackableScoreStatus === W2.NotReceived) { this._checkScoresAsync(); return; } const t = e.getImageTrackingResults(); for (const r of t) { let n = !1; const i = r.index, s = this._trackedImages[i]; if (!s) continue; s.xrTrackingResult = r, s.realWorldWidth !== r.measuredWidthInMeters && (s.realWorldWidth = r.measuredWidthInMeters, n = !0); const a = e.getPose(r.imageSpace, this._xrSessionManager.referenceSpace); if (a) { const d = s.transformationMatrix; he.FromArrayToRef(a.transform.matrix, 0, d), this._xrSessionManager.scene.useRightHandedSystem || d.toggleModelMatrixHandInPlace(), n = !0; } const o = r.trackingState === "emulated"; s.emulated !== o && (s.emulated = o, n = !0), n && this.onTrackedImageUpdatedObservable.notifyObservers(s); } } async _checkScoresAsync() { if (!this._xrSessionManager.session.getTrackedImageScores || this._trackableScoreStatus !== W2.NotReceived) return; this._trackableScoreStatus = W2.Waiting; const e = await this._xrSessionManager.session.getTrackedImageScores(); if (!e || e.length === 0) { this._trackableScoreStatus = W2.NotReceived; return; } for (let t = 0; t < e.length; ++t) if (e[t] == "untrackable") this.onUntrackableImageFoundObservable.notifyObservers(t); else { const r = this._originalTrackingRequest[t].image, n = { id: t, originalBitmap: r, transformationMatrix: new he(), ratio: r.width / r.height }; this._trackedImages[t] = n, this.onTrackableImageFoundObservable.notifyObservers(n); } this._trackableScoreStatus = e.length > 0 ? W2.Received : W2.NotReceived; } } lS.Name = Gi.IMAGE_TRACKING; lS.Version = 1; So.AddWebXRFeature(lS.Name, (A, e) => () => new lS(A, e), lS.Version, !1); class PS extends L9 { /** * Creates a new instance of the dom-overlay feature * @param _xrSessionManager an instance of WebXRSessionManager * @param options options to use when constructing this feature */ constructor(e, t) { super(e), this.options = t, this._domOverlayType = null, this._beforeXRSelectListener = null, this._element = null, this.xrNativeFeatureName = "dom-overlay", ye.Warn("dom-overlay is an experimental and unstable feature."); } /** * attach this feature * Will usually be called by the features manager * * @returns true if successful. */ attach() { return !super.attach() || !this._xrSessionManager.session.domOverlayState || this._xrSessionManager.session.domOverlayState.type === null ? !1 : (this._domOverlayType = this._xrSessionManager.session.domOverlayState.type, this._element !== null && this.options.supressXRSelectEvents === !0 && (this._beforeXRSelectListener = (e) => { e.preventDefault(); }, this._element.addEventListener("beforexrselect", this._beforeXRSelectListener)), !0); } /** * The type of DOM overlay (null when not supported). Provided by UA and remains unchanged for duration of session. */ get domOverlayType() { return this._domOverlayType; } /** * Dispose this feature and all of the resources attached */ dispose() { super.dispose(), this._element !== null && this._beforeXRSelectListener && this._element.removeEventListener("beforexrselect", this._beforeXRSelectListener); } _onXRFrame(e) { } /** * Extends the session init object if needed * @returns augmentation object for the xr session init object. */ async getXRSessionInitExtension() { if (this.options.element === void 0) return ye.Warn('"element" option must be provided to attach xr-dom-overlay feature.'), {}; if (typeof this.options.element == "string") { const e = document.querySelector(this.options.element); if (e === null) return ye.Warn(`element not found '${this.options.element}' (not requesting xr-dom-overlay)`), {}; this._element = e; } else this._element = this.options.element; return { domOverlay: { root: this._element } }; } } PS.Name = Gi.DOM_OVERLAY; PS.Version = 1; So.AddWebXRFeature(PS.Name, (A, e) => () => new PS(A, e), PS.Version, !1); class Pq extends L9 { /** * Current movement direction. Will be null before XR Frames have been processed. */ get movementDirection() { return this._movementDirection; } /** * Is movement enabled */ get movementEnabled() { return this._featureContext.movementEnabled; } /** * Sets whether movement is enabled or not * @param enabled is movement enabled */ set movementEnabled(e) { this._featureContext.movementEnabled = e; } /** * If movement follows viewer pose */ get movementOrientationFollowsViewerPose() { return this._featureContext.movementOrientationFollowsViewerPose; } /** * Sets whether movement follows viewer pose * @param followsPose is movement should follow viewer pose */ set movementOrientationFollowsViewerPose(e) { this._featureContext.movementOrientationFollowsViewerPose = e; } /** * Gets movement speed */ get movementSpeed() { return this._featureContext.movementSpeed; } /** * Sets movement speed * @param movementSpeed movement speed */ set movementSpeed(e) { this._featureContext.movementSpeed = e; } /** * Gets minimum threshold the controller's thumbstick/touchpad must pass before being recognized for movement (avoids jitter/unintentional movement) */ get movementThreshold() { return this._featureContext.movementThreshold; } /** * Sets minimum threshold the controller's thumbstick/touchpad must pass before being recognized for movement (avoids jitter/unintentional movement) * @param movementThreshold new threshold */ set movementThreshold(e) { this._featureContext.movementThreshold = e; } /** * Is rotation enabled */ get rotationEnabled() { return this._featureContext.rotationEnabled; } /** * Sets whether rotation is enabled or not * @param enabled is rotation enabled */ set rotationEnabled(e) { this._featureContext.rotationEnabled = e; } /** * Gets rotation speed factor */ get rotationSpeed() { return this._featureContext.rotationSpeed; } /** * Sets rotation speed factor (1.0 is default) * @param rotationSpeed new rotation speed factor */ set rotationSpeed(e) { this._featureContext.rotationSpeed = e; } /** * Gets minimum threshold the controller's thumbstick/touchpad must pass before being recognized for rotation (avoids jitter/unintentional rotation) */ get rotationThreshold() { return this._featureContext.rotationThreshold; } /** * Sets minimum threshold the controller's thumbstick/touchpad must pass before being recognized for rotation (avoids jitter/unintentional rotation) * @param threshold new threshold */ set rotationThreshold(e) { this._featureContext.rotationThreshold = e; } /** * constructs a new movement controller system * @param _xrSessionManager an instance of WebXRSessionManager * @param options configuration object for this feature */ constructor(e, t) { var r, n, i, s, a, f; if (super(e), this._controllers = {}, this._currentRegistrationConfigurations = [], this._movementDirection = new Ze(), this._tmpRotationMatrix = he.Identity(), this._tmpTranslationDirection = new S(), this._tmpMovementTranslation = new S(), this._tempCacheQuaternion = new Ze(), this._attachController = (o) => { if (this._controllers[o.uniqueId]) return; this._controllers[o.uniqueId] = { xrController: o, registeredComponents: [] }; const d = this._controllers[o.uniqueId]; if (d.xrController.inputSource.targetRayMode === "tracked-pointer" && d.xrController.inputSource.gamepad) { const v = () => { if (o.motionController) for (const u of this._currentRegistrationConfigurations) { let l = null; if (u.allowedComponentTypes) for (const p of u.allowedComponentTypes) { const c = o.motionController.getComponentOfType(p); if (c !== null) { l = c; break; } } if (u.mainComponentOnly) { const p = o.motionController.getMainComponent(); if (p === null) continue; l = p; } if (typeof u.componentSelectionPredicate == "function" && (l = u.componentSelectionPredicate(o)), l && u.forceHandedness && o.inputSource.handedness !== u.forceHandedness || l === null) continue; const P = { registrationConfiguration: u, component: l }; d.registeredComponents.push(P), "axisChangedHandler" in u && (P.onAxisChangedObserver = l.onAxisValueChangedObservable.add((p) => { u.axisChangedHandler(p, this._movementState, this._featureContext, this._xrInput); })), "buttonChangedhandler" in u && (P.onButtonChangedObserver = l.onButtonStateChangedObservable.add(() => { l.changes.pressed && u.buttonChangedhandler(l.changes.pressed, this._movementState, this._featureContext, this._xrInput); })); } }; o.motionController ? v() : o.onMotionControllerInitObservable.addOnce(() => { v(); }); } }, !t || t.xrInput === void 0) { ye.Error('WebXRControllerMovement feature requires "xrInput" option.'); return; } Array.isArray(t.customRegistrationConfigurations) ? this._currentRegistrationConfigurations = t.customRegistrationConfigurations : this._currentRegistrationConfigurations = Pq.REGISTRATIONS.default, this._featureContext = { movementEnabled: t.movementEnabled || !0, movementOrientationFollowsViewerPose: (r = t.movementOrientationFollowsViewerPose) !== null && r !== void 0 ? r : !0, movementSpeed: (n = t.movementSpeed) !== null && n !== void 0 ? n : 1, movementThreshold: (i = t.movementThreshold) !== null && i !== void 0 ? i : 0.25, rotationEnabled: (s = t.rotationEnabled) !== null && s !== void 0 ? s : !0, rotationSpeed: (a = t.rotationSpeed) !== null && a !== void 0 ? a : 1, rotationThreshold: (f = t.rotationThreshold) !== null && f !== void 0 ? f : 0.25 }, this._movementState = { moveX: 0, moveY: 0, rotateX: 0, rotateY: 0 }, this._xrInput = t.xrInput; } attach() { return super.attach() ? (this._xrInput.controllers.forEach(this._attachController), this._addNewAttachObserver(this._xrInput.onControllerAddedObservable, this._attachController), this._addNewAttachObserver(this._xrInput.onControllerRemovedObservable, (e) => { this._detachController(e.uniqueId); }), !0) : !1; } detach() { return super.detach() ? (Object.keys(this._controllers).forEach((e) => { this._detachController(e); }), this._controllers = {}, !0) : !1; } /** * Occurs on every XR frame. * @param _xrFrame */ _onXRFrame(e) { if (this.attached) { if (this._movementState.rotateX !== 0 && this._featureContext.rotationEnabled) { const r = this._xrSessionManager.scene.getEngine().getDeltaTime() * 1e-3 * this._featureContext.rotationSpeed * this._movementState.rotateX * (this._xrSessionManager.scene.useRightHandedSystem ? -1 : 1); this._featureContext.movementOrientationFollowsViewerPose ? (this._xrInput.xrCamera.cameraRotation.y += r, Ze.RotationYawPitchRollToRef(r, 0, 0, this._tempCacheQuaternion), this._xrInput.xrCamera.rotationQuaternion.multiplyToRef(this._tempCacheQuaternion, this._movementDirection)) : (Ze.RotationYawPitchRollToRef(r * 3, 0, 0, this._tempCacheQuaternion), this._movementDirection.multiplyInPlace(this._tempCacheQuaternion)); } else this._featureContext.movementOrientationFollowsViewerPose && this._movementDirection.copyFrom(this._xrInput.xrCamera.rotationQuaternion); (this._movementState.moveX || this._movementState.moveY) && this._featureContext.movementEnabled && (he.FromQuaternionToRef(this._movementDirection, this._tmpRotationMatrix), this._tmpTranslationDirection.set(this._movementState.moveX, 0, this._movementState.moveY * (this._xrSessionManager.scene.useRightHandedSystem ? 1 : -1)), S.TransformCoordinatesToRef(this._tmpTranslationDirection, this._tmpRotationMatrix, this._tmpMovementTranslation), this._tmpMovementTranslation.scaleInPlace(this._xrInput.xrCamera._computeLocalCameraSpeed() * this._featureContext.movementSpeed), this._xrInput.xrCamera.cameraDirection.addInPlace(this._tmpMovementTranslation)); } } _detachController(e) { const t = this._controllers[e]; if (t) { for (const r of t.registeredComponents) r.onAxisChangedObserver && r.component.onAxisValueChangedObservable.remove(r.onAxisChangedObserver), r.onButtonChangedObserver && r.component.onButtonStateChangedObservable.remove(r.onButtonChangedObserver); delete this._controllers[e]; } } } Pq.Name = Gi.MOVEMENT; Pq.REGISTRATIONS = { default: [ { allowedComponentTypes: [Jl.THUMBSTICK_TYPE, Jl.TOUCHPAD_TYPE], forceHandedness: "left", axisChangedHandler: (A, e, t) => { e.rotateX = Math.abs(A.x) > t.rotationThreshold ? A.x : 0, e.rotateY = Math.abs(A.y) > t.rotationThreshold ? A.y : 0; } }, { allowedComponentTypes: [Jl.THUMBSTICK_TYPE, Jl.TOUCHPAD_TYPE], forceHandedness: "right", axisChangedHandler: (A, e, t) => { e.moveX = Math.abs(A.x) > t.movementThreshold ? A.x : 0, e.moveY = Math.abs(A.y) > t.movementThreshold ? A.y : 0; } } ] }; Pq.Version = 1; So.AddWebXRFeature(Pq.Name, (A, e) => () => new Pq(A, e), Pq.Version, !0); class cS extends L9 { /** * Creates a new instance of the light estimation feature * @param _xrSessionManager an instance of WebXRSessionManager * @param options options to use when constructing this feature */ constructor(e, t) { super(e), this.options = t, this._canvasContext = null, this._reflectionCubeMap = null, this._xrLightEstimate = null, this._xrLightProbe = null, this._xrWebGLBinding = null, this._lightDirection = S.Up().negateInPlace(), this._lightColor = Ne.White(), this._intensity = 1, this._sphericalHarmonics = new HD(), this._cubeMapPollTime = Date.now(), this._lightEstimationPollTime = Date.now(), this._reflectionCubeMapTextureSize = 16, this.directionalLight = null, this.onReflectionCubeMapUpdatedObservable = new Oe(), this._updateReflectionCubeMap = () => { var r; if (!this._xrLightProbe) return; if (this.options.cubeMapPollInterval) { const i = Date.now(); if (i - this._cubeMapPollTime < this.options.cubeMapPollInterval) return; this._cubeMapPollTime = i; } const n = this._getXRGLBinding().getReflectionCubeMap(this._xrLightProbe); if (n && this._reflectionCubeMap) { if (this._reflectionCubeMap._texture) (r = this._reflectionCubeMap._texture._hardwareTexture) === null || r === void 0 || r.set(n), this._reflectionCubeMap._texture.getEngine().resetTextureCache(); else { const i = new As(this._xrSessionManager.scene.getEngine(), ri.Unknown); i.isCube = !0, i.invertY = !1, i._useSRGBBuffer = this.options.reflectionFormat === "srgba8", i.format = 5, i.generateMipMaps = !0, i.type = this.options.reflectionFormat !== "srgba8" ? 2 : 0, i.samplingMode = 3, i.width = this._reflectionCubeMapTextureSize, i.height = this._reflectionCubeMapTextureSize, i._cachedWrapU = 1, i._cachedWrapV = 1, i._hardwareTexture = new jm(n, this._getCanvasContext()), this._reflectionCubeMap._texture = i; } this._reflectionCubeMap._texture.isReady = !0, this.options.disablePreFiltering ? (this._xrSessionManager.scene.markAllMaterialsAsDirty(1), this.onReflectionCubeMapUpdatedObservable.notifyObservers(this._reflectionCubeMap)) : (this._xrLightProbe.removeEventListener("reflectionchange", this._updateReflectionCubeMap), this._hdrFilter.prefilter(this._reflectionCubeMap).then(() => { this._xrSessionManager.scene.markAllMaterialsAsDirty(1), this.onReflectionCubeMapUpdatedObservable.notifyObservers(this._reflectionCubeMap), this._xrLightProbe.addEventListener("reflectionchange", this._updateReflectionCubeMap); })); } }, this.xrNativeFeatureName = "light-estimation", this.options.createDirectionalLightSource && (this.directionalLight = new IA("light estimation directional", this._lightDirection, this._xrSessionManager.scene), this.directionalLight.position = new S(0, 8, 0), this.directionalLight.intensity = 0, this.directionalLight.falloffType = ra.FALLOFF_GLTF), this._hdrFilter = new _Q(this._xrSessionManager.scene.getEngine()), ye.Warn("light-estimation is an experimental and unstable feature."); } /** * While the estimated cube map is expected to update over time to better reflect the user's environment as they move around those changes are unlikely to happen with every XRFrame. * Since creating and processing the cube map is potentially expensive, especially if mip maps are needed, you can listen to the onReflectionCubeMapUpdatedObservable to determine * when it has been updated. */ get reflectionCubeMapTexture() { return this._reflectionCubeMap; } /** * The most recent light estimate. Available starting on the first frame where the device provides a light probe. */ get xrLightingEstimate() { return this._xrLightEstimate ? { lightColor: this._lightColor, lightDirection: this._lightDirection, lightIntensity: this._intensity, sphericalHarmonics: this._sphericalHarmonics } : this._xrLightEstimate; } _getCanvasContext() { return this._canvasContext === null && (this._canvasContext = this._xrSessionManager.scene.getEngine()._gl), this._canvasContext; } _getXRGLBinding() { if (this._xrWebGLBinding === null) { const e = this._getCanvasContext(); this._xrWebGLBinding = new XRWebGLBinding(this._xrSessionManager.session, e); } return this._xrWebGLBinding; } /** * attach this feature * Will usually be called by the features manager * * @returns true if successful. */ attach() { var e; if (!super.attach()) return !1; const t = (e = this.options.reflectionFormat) !== null && e !== void 0 ? e : this._xrSessionManager.session.preferredReflectionFormat || "srgba8"; return this.options.reflectionFormat = t, this._xrSessionManager.session.requestLightProbe({ reflectionFormat: t }).then((r) => { this._xrLightProbe = r, this.options.disableCubeMapReflection || (this._reflectionCubeMap || (this._reflectionCubeMap = new ls(this._xrSessionManager.scene), this._reflectionCubeMap._isCube = !0, this._reflectionCubeMap.coordinatesMode = 3, this.options.setSceneEnvironmentTexture && (this._xrSessionManager.scene.environmentTexture = this._reflectionCubeMap)), this._xrLightProbe.addEventListener("reflectionchange", this._updateReflectionCubeMap)); }), !0; } /** * detach this feature. * Will usually be called by the features manager * * @returns true if successful. */ detach() { const e = super.detach(); return this._xrLightProbe !== null && !this.options.disableCubeMapReflection && (this._xrLightProbe.removeEventListener("reflectionchange", this._updateReflectionCubeMap), this._xrLightProbe = null), this._canvasContext = null, this._xrLightEstimate = null, this._xrWebGLBinding = null, e; } /** * Dispose this feature and all of the resources attached */ dispose() { super.dispose(), this.onReflectionCubeMapUpdatedObservable.clear(), this.directionalLight && (this.directionalLight.dispose(), this.directionalLight = null), this._reflectionCubeMap !== null && (this._reflectionCubeMap._texture && this._reflectionCubeMap._texture.dispose(), this._reflectionCubeMap.dispose(), this._reflectionCubeMap = null); } _onXRFrame(e) { var t; if (this._xrLightProbe !== null) { if (this.options.lightEstimationPollInterval) { const r = Date.now(); if (r - this._lightEstimationPollTime < this.options.lightEstimationPollInterval) return; this._lightEstimationPollTime = r; } if (this._xrLightEstimate = e.getLightEstimate(this._xrLightProbe), this._xrLightEstimate) { this._intensity = Math.max(1, this._xrLightEstimate.primaryLightIntensity.x, this._xrLightEstimate.primaryLightIntensity.y, this._xrLightEstimate.primaryLightIntensity.z); const r = this._xrSessionManager.scene.useRightHandedSystem ? 1 : -1; this.options.disableVectorReuse && (this._lightDirection = new S(), this._lightColor = new Ne(), this.directionalLight && (this.directionalLight.direction = this._lightDirection, this.directionalLight.diffuse = this._lightColor)), this._lightDirection.copyFromFloats(this._xrLightEstimate.primaryLightDirection.x, this._xrLightEstimate.primaryLightDirection.y, this._xrLightEstimate.primaryLightDirection.z * r), this._lightColor.copyFromFloats(this._xrLightEstimate.primaryLightIntensity.x / this._intensity, this._xrLightEstimate.primaryLightIntensity.y / this._intensity, this._xrLightEstimate.primaryLightIntensity.z / this._intensity), this._sphericalHarmonics.updateFromFloatsArray(this._xrLightEstimate.sphericalHarmonicsCoefficients), this._reflectionCubeMap && !this.options.disableSphericalPolynomial && (this._reflectionCubeMap.sphericalPolynomial = this._reflectionCubeMap.sphericalPolynomial || new i4(), (t = this._reflectionCubeMap.sphericalPolynomial) === null || t === void 0 || t.updateFromHarmonics(this._sphericalHarmonics)), this._lightDirection.negateInPlace(), this.directionalLight && (this.directionalLight.direction.copyFrom(this._lightDirection), this.directionalLight.intensity = Math.min(this._intensity, 1), this.directionalLight.diffuse.copyFrom(this._lightColor)); } } } } cS.Name = Gi.LIGHT_ESTIMATION; cS.Version = 1; So.AddWebXRFeature(cS.Name, (A, e) => () => new cS(A, e), cS.Version, !1); class pS extends L9 { /** * Creates a new instance of the XR eye tracking feature. * @param _xrSessionManager An instance of WebXRSessionManager. */ constructor(e) { super(e), this.onEyeTrackingStartedObservable = new Oe(), this.onEyeTrackingEndedObservable = new Oe(), this.onEyeTrackingFrameUpdateObservable = new Oe(), this._eyeTrackingStartListener = (t) => { this._latestEyeSpace = t.gazeSpace, this._gazeRay = new Hi(S.Zero(), S.Forward()), this.onEyeTrackingStartedObservable.notifyObservers(this._gazeRay); }, this._eyeTrackingEndListener = () => { this._latestEyeSpace = null, this._gazeRay = null, this.onEyeTrackingEndedObservable.notifyObservers(); }, this.xrNativeFeatureName = "eye-tracking", this._xrSessionManager.session ? this._init() : this._xrSessionManager.onXRSessionInit.addOnce(() => { this._init(); }); } /** * Dispose this feature and all of the resources attached. */ dispose() { super.dispose(), this._xrSessionManager.session.removeEventListener("eyetrackingstart", this._eyeTrackingStartListener), this._xrSessionManager.session.removeEventListener("eyetrackingend", this._eyeTrackingEndListener), this.onEyeTrackingStartedObservable.clear(), this.onEyeTrackingEndedObservable.clear(), this.onEyeTrackingFrameUpdateObservable.clear(); } /** * Returns whether the gaze data is valid or not * @returns true if the data is valid */ get isEyeGazeValid() { return !!this._gazeRay; } /** * Get a reference to the gaze ray. This data is valid while eye tracking persists, and will be set to null when gaze data is no longer available * @returns a reference to the gaze ray if it exists and is valid, returns null otherwise. */ getEyeGaze() { return this._gazeRay; } _onXRFrame(e) { if (!(!this.attached || !e) && this._latestEyeSpace && this._gazeRay) { const t = e.getPose(this._latestEyeSpace, this._xrSessionManager.referenceSpace); if (t) { this._gazeRay.origin.set(t.transform.position.x, t.transform.position.y, t.transform.position.z); const r = t.transform.orientation; ue.Quaternion[0].set(r.x, r.y, r.z, r.w), this._xrSessionManager.scene.useRightHandedSystem ? S.RightHandedForwardReadOnly.rotateByQuaternionToRef(ue.Quaternion[0], this._gazeRay.direction) : (this._gazeRay.origin.z *= -1, ue.Quaternion[0].z *= -1, ue.Quaternion[0].w *= -1, S.LeftHandedForwardReadOnly.rotateByQuaternionToRef(ue.Quaternion[0], this._gazeRay.direction)), this.onEyeTrackingFrameUpdateObservable.notifyObservers(this._gazeRay); } } } _init() { this._xrSessionManager.isNative && (this._xrSessionManager.session.addEventListener("eyetrackingstart", this._eyeTrackingStartListener), this._xrSessionManager.session.addEventListener("eyetrackingend", this._eyeTrackingEndListener)); } } pS.Name = Gi.EYE_TRACKING; pS.Version = 1; So.AddWebXRFeature(pS.Name, (A) => () => new pS(A), pS.Version, !1); class Dge { constructor(e, t) { this._samples = [], this._idx = 0; for (let r = 0; r < e; ++r) this._samples.push(t ? t() : at.Zero()); } get length() { return this._samples.length; } push(e, t) { this._idx = (this._idx + this._samples.length - 1) % this._samples.length, this.at(0).copyFromFloats(e, t); } at(e) { if (e >= this._samples.length) throw new Error("Index out of bounds"); return this._samples[(this._idx + e) % this._samples.length]; } } class jge { constructor() { this._samples = new Dge(20), this._entropy = 0, this.onFirstStepDetected = new Oe(); } update(e, t, r, n) { this._samples.push(e, t); const i = this._samples.at(0); if (this._entropy *= this._entropyDecayFactor, this._entropy += at.Distance(i, this._samples.at(1)), this._entropy > this._entropyThreshold) return; let s; for (s = this._samePointCheckStartIdx; s < this._samples.length && !(at.DistanceSquared(i, this._samples.at(s)) < this._samePointSquaredDistanceThreshold); ++s) ; if (s === this._samples.length) return; let a = -1, f = 0; for (let b, j = 1; j < s; ++j) b = at.DistanceSquared(i, this._samples.at(j)), b > a && (f = j, a = b); if (a < this._apexSquaredDistanceThreshold) return; const o = this._samples.at(f), d = o.subtract(i); d.normalize(); const v = ue.Vector2[0]; let u, l, P = 0; for (let b = 1; b < s; ++b) l = this._samples.at(b), l.subtractToRef(i, v), u = at.Dot(d, v), P += v.lengthSquared() - u * u; if (P > s * this._squaredProjectionDistanceThreshold) return; const p = ue.Vector3[0]; p.set(r, n, 0); const c = ue.Vector3[1]; c.set(d.x, d.y, 0); const H = S.Cross(p, c).z > 0, T = i.clone(), q = i.clone(); o.subtractToRef(i, d), H ? (d.scaleAndAddToRef(this._axisToApexShrinkFactor, T), d.scaleAndAddToRef(this._axisToApexExtendFactor, q)) : (d.scaleAndAddToRef(this._axisToApexExtendFactor, T), d.scaleAndAddToRef(this._axisToApexShrinkFactor, q)), this.onFirstStepDetected.notifyObservers({ leftApex: T, rightApex: q, currentPosition: i, currentStepDirection: H ? "right" : "left" }); } reset() { for (let e = 0; e < this._samples.length; ++e) this._samples.at(e).copyFromFloats(0, 0); } get _samePointCheckStartIdx() { return Math.floor(this._samples.length / 3); } get _samePointSquaredDistanceThreshold() { return 0.03 * 0.03; } get _apexSquaredDistanceThreshold() { return 0.09 * 0.09; } get _squaredProjectionDistanceThreshold() { return 0.03 * 0.03; } get _axisToApexShrinkFactor() { return 0.8; } get _axisToApexExtendFactor() { return -1.6; } get _entropyDecayFactor() { return 0.93; } get _entropyThreshold() { return 0.4; } } class wge { constructor(e, t, r, n) { this._leftApex = new at(), this._rightApex = new at(), this._currentPosition = new at(), this._axis = new at(), this._axisLength = -1, this._forward = new at(), this._steppingLeft = !1, this._t = -1, this._maxT = -1, this._maxTPosition = new at(), this._vitality = 0, this.onMovement = new Oe(), this.onFootfall = new Oe(), this._reset(e, t, r, n === "left"); } _reset(e, t, r, n) { this._leftApex.copyFrom(e), this._rightApex.copyFrom(t), this._steppingLeft = n, this._steppingLeft ? (this._leftApex.subtractToRef(this._rightApex, this._axis), this._forward.copyFromFloats(-this._axis.y, this._axis.x)) : (this._rightApex.subtractToRef(this._leftApex, this._axis), this._forward.copyFromFloats(this._axis.y, -this._axis.x)), this._axisLength = this._axis.length(), this._forward.scaleInPlace(1 / this._axisLength), this._updateTAndVitality(r.x, r.y), this._maxT = this._t, this._maxTPosition.copyFrom(r), this._vitality = 1; } _updateTAndVitality(e, t) { this._currentPosition.copyFromFloats(e, t), this._steppingLeft ? this._currentPosition.subtractInPlace(this._rightApex) : this._currentPosition.subtractInPlace(this._leftApex); const r = this._t, n = at.Dot(this._currentPosition, this._axis); this._t = n / (this._axisLength * this._axisLength); const i = this._currentPosition.lengthSquared() - n / this._axisLength * (n / this._axisLength); this._vitality *= 0.92 - 100 * Math.max(i - 16e-4, 0) + Math.max(this._t - r, 0); } update(e, t) { if (this._vitality < this._vitalityThreshold) return !1; const r = this._t; return this._updateTAndVitality(e, t), this._t > this._maxT && (this._maxT = this._t, this._maxTPosition.copyFromFloats(e, t)), !(this._vitality < this._vitalityThreshold || (this._t > r && (this.onMovement.notifyObservers({ deltaT: this._t - r }), r < 0.5 && this._t >= 0.5 && this.onFootfall.notifyObservers({ foot: this._steppingLeft ? "left" : "right" })), this._t < 0.95 * this._maxT && (this._currentPosition.copyFromFloats(e, t), this._steppingLeft ? this._leftApex.copyFrom(this._maxTPosition) : this._rightApex.copyFrom(this._maxTPosition), this._reset(this._leftApex, this._rightApex, this._currentPosition, !this._steppingLeft)), this._axisLength < 0.03)); } get _vitalityThreshold() { return 0.1; } get forward() { return this._forward; } } class UI { static get _MillisecondsPerUpdate() { return 1e3 / 15; } constructor(e) { this._detector = new jge(), this._walker = null, this._movement = new at(), this._millisecondsSinceLastUpdate = UI._MillisecondsPerUpdate, this.movementThisFrame = S.Zero(), this._engine = e, this._detector.onFirstStepDetected.add((t) => { this._walker || (this._walker = new wge(t.leftApex, t.rightApex, t.currentPosition, t.currentStepDirection), this._walker.onFootfall.add(() => { console.log("Footfall!"); }), this._walker.onMovement.add((r) => { this._walker.forward.scaleAndAddToRef(0.024 * r.deltaT, this._movement); })); }); } update(e, t) { t.y = 0, t.normalize(), this._millisecondsSinceLastUpdate += this._engine.getDeltaTime(), this._millisecondsSinceLastUpdate >= UI._MillisecondsPerUpdate && (this._millisecondsSinceLastUpdate -= UI._MillisecondsPerUpdate, this._detector.update(e.x, e.z, t.x, t.z), this._walker && (this._walker.update(e.x, e.z) || (this._walker = null)), this._movement.scaleInPlace(0.85)), this.movementThisFrame.set(this._movement.x, 0, this._movement.y); } } class CC extends L9 { /** * The module's name. */ static get Name() { return Gi.WALKING_LOCOMOTION; } /** * The (Babylon) version of this module. * This is an integer representing the implementation version. * This number has no external basis. */ static get Version() { return 1; } /** * The target to be articulated by walking locomotion. * When the walking locomotion feature detects walking in place, this element's * X and Z coordinates will be modified to reflect locomotion. This target should * be either the XR space's origin (i.e., the parent node of the WebXRCamera) or * the WebXRCamera itself. Note that the WebXRCamera path will modify the position * of the WebXRCamera directly and is thus discouraged. */ get locomotionTarget() { return this._locomotionTarget; } /** * The target to be articulated by walking locomotion. * When the walking locomotion feature detects walking in place, this element's * X and Z coordinates will be modified to reflect locomotion. This target should * be either the XR space's origin (i.e., the parent node of the WebXRCamera) or * the WebXRCamera itself. Note that the WebXRCamera path will modify the position * of the WebXRCamera directly and is thus discouraged. */ set locomotionTarget(e) { this._locomotionTarget = e, this._isLocomotionTargetWebXRCamera = this._locomotionTarget.getClassName() === "WebXRCamera"; } /** * Construct a new Walking Locomotion feature. * @param sessionManager manager for the current XR session * @param options creation options, prominently including the vector target for locomotion */ constructor(e, t) { super(e), this._up = new S(), this._forward = new S(), this._position = new S(), this._movement = new S(), this._sessionManager = e, this.locomotionTarget = t.locomotionTarget, this._isLocomotionTargetWebXRCamera && Se.Warn("Using walking locomotion directly on a WebXRCamera may have unintended interactions with other XR techniques. Using an XR space parent is highly recommended"); } /** * Checks whether this feature is compatible with the current WebXR session. * Walking locomotion is only compatible with "immersive-vr" sessions. * @returns true if compatible, false otherwise */ isCompatible() { return this._sessionManager.sessionMode === void 0 || this._sessionManager.sessionMode === "immersive-vr"; } /** * Attaches the feature. * Typically called automatically by the features manager. * @returns true if attach succeeded, false otherwise */ attach() { return !this.isCompatible || !super.attach() ? !1 : (this._walker = new UI(this._sessionManager.scene.getEngine()), !0); } /** * Detaches the feature. * Typically called automatically by the features manager. * @returns true if detach succeeded, false otherwise */ detach() { return super.detach() ? (this._walker = null, !0) : !1; } _onXRFrame(e) { const t = e.getViewerPose(this._sessionManager.baseReferenceSpace); if (!t) return; const r = this.locomotionTarget.getScene().useRightHandedSystem ? 1 : -1, n = t.transform.matrix; this._up.copyFromFloats(n[4], n[5], r * n[6]), this._forward.copyFromFloats(n[8], n[9], r * n[10]), this._position.copyFromFloats(n[12], n[13], r * n[14]), this._forward.scaleAndAddToRef(0.05, this._position), this._up.scaleAndAddToRef(-0.05, this._position), this._walker.update(this._position, this._forward), this._movement.copyFrom(this._walker.movementThisFrame), this._isLocomotionTargetWebXRCamera || S.TransformNormalToRef(this._movement, this.locomotionTarget.getWorldMatrix(), this._movement), this.locomotionTarget.position.addInPlace(this._movement); } } So.AddWebXRFeature(CC.Name, (A, e) => () => new CC(A, e), CC.Version, !1); class Wie extends XQ { constructor(e, t, r, n, i, s) { super(e, t, r, n, s), this.getWidth = e, this.getHeight = t, this.layer = r, this.layerType = n, this.isMultiview = i, this.createRTTProvider = s; } } class mge extends TQ { constructor(e, t, r) { super(e.scene, r), this._xrSessionManager = e, this._xrWebGLBinding = t, this.layerWrapper = r, this._lastSubImages = /* @__PURE__ */ new Map(), this._compositionLayer = r.layer; } _getRenderTargetForSubImage(e, t) { var r, n, i, s; const a = this._lastSubImages.get(t), f = t == "left" ? 0 : 1, o = (r = e.colorTextureWidth) !== null && r !== void 0 ? r : e.textureWidth, d = (n = e.colorTextureHeight) !== null && n !== void 0 ? n : e.textureHeight; if (!this._renderTargetTextures[f] || (a == null ? void 0 : a.textureWidth) !== o || (a == null ? void 0 : a.textureHeight) !== d) { let v; const u = (i = e.depthStencilTextureWidth) !== null && i !== void 0 ? i : o, l = (s = e.depthStencilTextureHeight) !== null && s !== void 0 ? s : d; (o === u || d === l) && (v = e.depthStencilTexture), this._renderTargetTextures[f] = this._createRenderTargetTexture(o, d, null, e.colorTexture, v, this.layerWrapper.isMultiview), this._framebufferDimensions = { framebufferWidth: o, framebufferHeight: d }; } return this._lastSubImages.set(t, e), this._renderTargetTextures[f]; } _getSubImageForEye(e) { const t = this._xrSessionManager.currentFrame; return t ? this._xrWebGLBinding.getSubImage(this._compositionLayer, t, e) : null; } getRenderTargetTextureForEye(e) { const t = this._getSubImageForEye(e); return t ? this._getRenderTargetForSubImage(t, e) : null; } getRenderTargetTextureForView(e) { return this.getRenderTargetTextureForEye(e.eye); } _setViewportForSubImage(e, t) { var r, n; const i = (r = t.colorTextureWidth) !== null && r !== void 0 ? r : t.textureWidth, s = (n = t.colorTextureWidth) !== null && n !== void 0 ? n : t.textureHeight, a = t.viewport; e.x = a.x / i, e.y = a.y / s, e.width = a.width / i, e.height = a.height / s; } trySetViewportForView(e, t) { const r = this._lastSubImages.get(t.eye) || this._getSubImageForEye(t.eye); return r ? (this._setViewportForSubImage(e, r), !0) : !1; } } class Sie extends Wie { constructor(e, t, r) { super(() => e.textureWidth, () => e.textureHeight, e, "XRProjectionLayer", t, (n) => new Bge(n, r, this)), this.layer = e; } } class Bge extends mge { constructor(e, t, r) { super(e, t, r), this.layerWrapper = r, this._projectionLayer = r.layer; } _getSubImageForView(e) { return this._xrWebGLBinding.getViewSubImage(this._projectionLayer, e); } getRenderTargetTextureForView(e) { return this._getRenderTargetForSubImage(this._getSubImageForView(e), e.eye); } getRenderTargetTextureForEye(e) { const t = this._lastSubImages.get(e); return t ? this._getRenderTargetForSubImage(t, e) : null; } trySetViewportForView(e, t) { const r = this._lastSubImages.get(t.eye) || this._getSubImageForView(t); return r ? (this._setViewportForSubImage(e, r), !0) : !1; } } const Wge = {}, tZ = { textureType: "texture", colorFormat: 6408, depthFormat: 35056, scaleFactor: 1 }; class hS extends L9 { constructor(e, t = {}) { super(e), this._options = t, this._existingLayers = [], this.xrNativeFeatureName = "layers"; } /** * Attach this feature. * Will usually be called by the features manager. * * @returns true if successful. */ attach() { if (!super.attach()) return !1; const e = this._xrSessionManager.scene.getEngine(); this._glContext = e._gl, this._xrWebGLBinding = new XRWebGLBinding(this._xrSessionManager.session, this._glContext), this._existingLayers.length = 0; const t = Object.assign({}, tZ), r = this._options.preferMultiviewOnInit && e.getCaps().multiview; return r && (t.textureType = "texture-array"), this.addXRSessionLayer(this.createProjectionLayer(t, r)), !0; } detach() { return super.detach() ? (this._existingLayers.length = 0, !0) : !1; } /** * Creates a new XRWebGLLayer. * @param params an object providing configuration options for the new XRWebGLLayer * @returns the XRWebGLLayer */ createXRWebGLLayer(e = Wge) { const t = new XRWebGLLayer(this._xrSessionManager.session, this._glContext, e); return new qQ(t); } /** * Creates a new XRProjectionLayer. * @param params an object providing configuration options for the new XRProjectionLayer. * @param multiview whether the projection layer should render with multiview. * @returns the projection layer */ createProjectionLayer(e = tZ, t = !1) { if (t && e.textureType !== "texture-array") throw new Error("Projection layers can only be made multiview if they use texture arrays. Set the textureType parameter to 'texture-array'."); if (!t && e.textureType === "texture-array") throw new Error("We currently only support multiview rendering when the textureType parameter is set to 'texture-array'."); const r = this._xrWebGLBinding.createProjectionLayer(e); return new Sie(r, t, this._xrWebGLBinding); } /** * Add a new layer to the already-existing list of layers * @param wrappedLayer the new layer to add to the existing ones */ addXRSessionLayer(e) { this.setXRSessionLayers([...this._existingLayers, e]); } /** * Sets the layers to be used by the XR session. * Note that you must call this function with any layers you wish to render to * since it adds them to the XR session's render state * (replacing any layers that were added in a previous call to setXRSessionLayers or updateRenderState). * This method also sets up the session manager's render target texture provider * as the first layer in the array, which feeds the WebXR camera(s) attached to the session. * @param wrappedLayers An array of WebXRLayerWrapper, usually returned from the WebXRLayers createLayer functions. */ setXRSessionLayers(e) { this._existingLayers = e; const t = Object.assign({}, this._xrSessionManager.session.renderState); t.baseLayer = void 0, t.layers = e.map((r) => r.layer), this._xrSessionManager.updateRenderState(t), this._xrSessionManager._setBaseLayerWrapper(e.length > 0 ? e[0] : null); } isCompatible() { return !this._xrSessionManager.isNative && typeof XRWebGLBinding < "u" && !!XRWebGLBinding.prototype.createProjectionLayer; } /** * Dispose this feature and all of the resources attached. */ dispose() { super.dispose(); } _onXRFrame(e) { } } hS.Name = Gi.LAYERS; hS.Version = 1; So.AddWebXRFeature(hS.Name, (A, e) => () => new hS(A, e), hS.Version, !1); class HS extends L9 { /** * Width of depth data. If depth data is not exist, returns null. */ get width() { return this._width; } /** * Height of depth data. If depth data is not exist, returns null. */ get height() { return this._height; } /** * Scale factor by which the raw depth values must be multiplied in order to get the depths in meters. */ get rawValueToMeters() { return this._rawValueToMeters; } /** * An XRRigidTransform that needs to be applied when indexing into the depth buffer. */ get normDepthBufferFromNormView() { return this._normDepthBufferFromNormView; } /** * Describes which depth-sensing usage ("cpu" or "gpu") is used. */ get depthUsage() { switch (this._xrSessionManager.session.depthUsage) { case "cpu-optimized": return "cpu"; case "gpu-optimized": return "gpu"; } } /** * Describes which depth sensing data format ("ushort" or "float") is used. */ get depthDataFormat() { switch (this._xrSessionManager.session.depthDataFormat) { case "luminance-alpha": return "ushort"; case "float32": return "float"; } } /** * Latest cached InternalTexture which containing depth buffer information. * This can be used when the depth usage is "gpu". */ get latestInternalTexture() { var e, t; if (!this._cachedWebGLTexture) return null; const r = this._xrSessionManager.scene.getEngine(), n = new As(r, ri.Unknown); return n.isCube = !1, n.invertY = !1, n._useSRGBBuffer = !1, n.format = this.depthDataFormat === "ushort" ? 2 : 5, n.generateMipMaps = !1, n.type = this.depthDataFormat === "ushort" ? 5 : 1, n.samplingMode = 7, n.width = (e = this.width) !== null && e !== void 0 ? e : 0, n.height = (t = this.height) !== null && t !== void 0 ? t : 0, n._cachedWrapU = 1, n._cachedWrapV = 1, n._hardwareTexture = new jm(this._cachedWebGLTexture, r._gl), n; } /** * cached depth buffer */ get latestDepthBuffer() { return this._cachedDepthBuffer ? this.depthDataFormat === "ushort" ? new Uint16Array(this._cachedDepthBuffer) : new Float32Array(this._cachedDepthBuffer) : null; } /** * Latest cached Texture of depth image which is made from the depth buffer data. */ get latestDepthImageTexture() { return this._cachedDepthImageTexture; } /** * Creates a new instance of the depth sensing feature * @param _xrSessionManager the WebXRSessionManager * @param options options for WebXR Depth Sensing Feature */ constructor(e, t) { super(e), this.options = t, this._width = null, this._height = null, this._rawValueToMeters = null, this._normDepthBufferFromNormView = null, this._cachedDepthBuffer = null, this._cachedWebGLTexture = null, this._cachedDepthImageTexture = null, this.onGetDepthInMetersAvailable = new Oe(), this.xrNativeFeatureName = "depth-sensing", ye.Warn("depth-sensing is an experimental and unstable feature."); } /** * attach this feature * Will usually be called by the features manager * * @returns true if successful. */ attach(e) { return !super.attach(e) || this._xrSessionManager.session.depthDataFormat == null || this._xrSessionManager.session.depthUsage == null ? !1 : (this._glBinding = new XRWebGLBinding(this._xrSessionManager.session, this._xrSessionManager.scene.getEngine()._gl), !0); } /** * Dispose this feature and all of the resources attached */ dispose() { var e; (e = this._cachedDepthImageTexture) === null || e === void 0 || e.dispose(); } _onXRFrame(e) { const t = this._xrSessionManager.referenceSpace, r = e.getViewerPose(t); if (r != null) for (const n of r.views) switch (this.depthUsage) { case "cpu": this._updateDepthInformationAndTextureCPUDepthUsage(e, n, this.depthDataFormat); break; case "gpu": if (!this._glBinding) break; this._updateDepthInformationAndTextureWebGLDepthUsage(this._glBinding, n, this.depthDataFormat); break; default: ye.Error("Unknown depth usage"), this.detach(); break; } } _updateDepthInformationAndTextureCPUDepthUsage(e, t, r) { const n = e.getDepthInformation(t); if (n === null) return; const { data: i, width: s, height: a, rawValueToMeters: f, getDepthInMeters: o } = n; switch (this._width = s, this._height = a, this._rawValueToMeters = f, this._cachedDepthBuffer = i, this.onGetDepthInMetersAvailable.notifyObservers(o.bind(n)), this._cachedDepthImageTexture || (this._cachedDepthImageTexture = Bo.CreateRTexture(null, s, a, this._xrSessionManager.scene, !1, !0, We.NEAREST_SAMPLINGMODE, Ge.TEXTURETYPE_FLOAT)), r) { case "ushort": this._cachedDepthImageTexture.update(Float32Array.from(new Uint16Array(i)).map((d) => d * f)); break; case "float": this._cachedDepthImageTexture.update(new Float32Array(i).map((d) => d * f)); break; } } _updateDepthInformationAndTextureWebGLDepthUsage(e, t, r) { const n = e.getDepthInformation(t); if (n === null) return; const { texture: i, width: s, height: a } = n; this._width = s, this._height = a, this._cachedWebGLTexture = i; const f = this._xrSessionManager.scene, d = f.getEngine().wrapWebGLTexture(i); this._cachedDepthImageTexture || (this._cachedDepthImageTexture = Bo.CreateRTexture(null, s, a, f, !1, !0, We.NEAREST_SAMPLINGMODE, r === "ushort" ? Ge.TEXTURETYPE_UNSIGNED_BYTE : Ge.TEXTURETYPE_FLOAT)), this._cachedDepthImageTexture._texture = d; } /** * Extends the session init object if needed * @returns augmentation object for the xr session init object. */ getXRSessionInitExtension() { const e = this.options.usagePreference != null && this.options.usagePreference.length !== 0, t = this.options.dataFormatPreference != null && this.options.dataFormatPreference.length !== 0; return new Promise((r) => { if (e && t) { const n = this.options.usagePreference.map((s) => { switch (s) { case "cpu": return "cpu-optimized"; case "gpu": return "gpu-optimized"; } }), i = this.options.dataFormatPreference.map((s) => { switch (s) { case "ushort": return "luminance-alpha"; case "float": return "float32"; } }); r({ depthSensing: { usagePreference: n, dataFormatPreference: i } }); } else r({}); }); } } HS.Name = Gi.DEPTH_SENSING; HS.Version = 1; So.AddWebXRFeature(HS.Name, (A, e) => () => new HS(A, e), HS.Version, !1); const Sge = "velocityPixelShader", Uge = `precision highp float; #define CUSTOM_FRAGMENT_BEGIN varying vec4 clipPos;varying vec4 previousClipPos; #define CUSTOM_FRAGMENT_DEFINITIONS void main(void) { #define CUSTOM_FRAGMENT_MAIN_BEGIN highp vec4 motionVector=( clipPos/clipPos.w-previousClipPos/previousClipPos.w );gl_FragColor=motionVector; #define CUSTOM_FRAGMENT_MAIN_END }`; Le.ShadersStore[Sge] = Uge; const Ige = "velocityVertexShader", Rge = `#define CUSTOM_VERTEX_BEGIN #define VELOCITY attribute vec3 position; #include uniform mat4 viewProjection;uniform mat4 previousViewProjection; #ifdef MULTIVIEW uniform mat4 viewProjectionR;uniform mat4 previousViewProjectionR; #endif varying vec4 clipPos;varying vec4 previousClipPos; #define CUSTOM_VERTEX_DEFINITIONS void main(void) { #define CUSTOM_VERTEX_MAIN_BEGIN vec3 positionUpdated=position; #include vec4 worldPos=finalWorld*vec4(positionUpdated,1.0);vec4 previousWorldPos=finalPreviousWorld*vec4(positionUpdated,1.0); #ifdef MULTIVIEW if (gl_ViewID_OVR==0u) {clipPos=viewProjection*worldPos;previousClipPos=previousViewProjection*previousWorldPos;gl_Position=clipPos;} else {clipPos=viewProjectionR*worldPos;previousClipPos=previousViewProjectionR*previousWorldPos;gl_Position=clipPos;} #elif clipPos=viewProjection*worldPos;previousClipPos=previousViewProjection*previousWorldPos;gl_Position=clipPos; #endif #define CUSTOM_VERTEX_MAIN_END }`; Le.ShadersStore[Ige] = Rge; class Uie extends Ta { /** * Creates a Space Warp render target * @param motionVectorTexture WebGLTexture provided by WebGLSubImage * @param depthStencilTexture WebGLTexture provided by WebGLSubImage * @param scene scene used with the render target * @param size the size of the render target (used for each view) */ constructor(e, t, r, n = 512) { super("spacewarp rtt", n, r, !1, !0, 2, !1, void 0, !1, !1, !0, void 0, !0), this._originalPairing = [], this._previousWorldMatrices = [], this._previousTransforms = [he.Identity(), he.Identity()], this._renderTarget = this.getScene().getEngine().createMultiviewRenderTargetTexture(this.getRenderWidth(), this.getRenderHeight(), e, t), this._renderTarget._disposeOnlyFramebuffers = !0, this._texture = this._renderTarget.texture, this._texture.isMultiview = !0, this._texture.format = 5, r && (this._velocityMaterial = new Zo("velocity shader material", r, { vertex: "velocity", fragment: "velocity" }, { uniforms: ["world", "previousWorld", "viewProjection", "viewProjectionR", "previousViewProjection", "previousViewProjectionR"] }), this._velocityMaterial._materialHelperNeedsPreviousMatrices = !0, this._velocityMaterial.onBindObservable.add((i) => { this._previousWorldMatrices[i.uniqueId] = this._previousWorldMatrices[i.uniqueId] || i.getWorldMatrix(), this._velocityMaterial.getEffect().setMatrix("previousWorld", this._previousWorldMatrices[i.uniqueId]), this._previousWorldMatrices[i.uniqueId] = i.getWorldMatrix(), this._velocityMaterial.getEffect().setMatrix("previousViewProjection", this._previousTransforms[0]), this._velocityMaterial.getEffect().setMatrix("previousViewProjectionR", this._previousTransforms[1]), this._previousTransforms[0].copyFrom(r.getTransformMatrix()), this._previousTransforms[1].copyFrom(r._transformMatrixR); }), this._velocityMaterial.freeze()); } /** * {@inheritDoc} */ render(e = !1, t = !1) { this._originalPairing.length = 0; const r = this.getScene(); r && this._velocityMaterial && r.getActiveMeshes().forEach((n) => { this._originalPairing.push([n, n.material]), n.material = this._velocityMaterial; }), super.render(e, t), this._originalPairing.forEach((n) => { n[0].material = n[1]; }); } /** * @internal */ _bindFrameBuffer() { this._renderTarget && this.getScene().getEngine().bindSpaceWarpFramebuffer(this._renderTarget); } /** * Gets the number of views the corresponding to the texture (eg. a SpaceWarpRenderTarget will have > 1) * @returns the view count */ getViewCount() { return 2; } /** * {@inheritdoc} */ dispose() { super.dispose(), this._velocityMaterial.dispose(), this._previousTransforms.length = 0, this._previousWorldMatrices.length = 0, this._originalPairing.length = 0; } } class Iie { constructor(e, t, r) { this._scene = e, this._xrSessionManager = t, this._xrWebGLBinding = r, this._lastSubImages = /* @__PURE__ */ new Map(), this._renderTargetTextures = /* @__PURE__ */ new Map(), this._engine = e.getEngine(); } _getSubImageForView(e) { const t = this._xrSessionManager._getBaseLayerWrapper(); if (!t) throw new Error("For Space Warp, the base layer should be a WebXR Projection Layer."); if (t.layerType !== "XRProjectionLayer") throw new Error('For Space Warp, the base layer type should "XRProjectionLayer".'); const r = t.layer; return this._xrWebGLBinding.getViewSubImage(r, e); } _setViewportForSubImage(e, t) { e.x = 0, e.y = 0, e.width = t.motionVectorTextureWidth, e.height = t.motionVectorTextureHeight; } _createRenderTargetTexture(e, t, r, n, i) { if (!this._engine) throw new Error("Engine is disposed"); const s = { width: e, height: t }, a = new Uie(n, i, this._scene, s), f = a.renderTarget; return r && (f._framebuffer = r), f._colorTextureArray = n, f._depthStencilTextureArray = i, a.disableRescaling(), a.renderListPredicate = () => !0, a; } _getRenderTargetForSubImage(e, t) { const r = this._lastSubImages.get(t); let n = this._renderTargetTextures.get(t.eye); const i = e.motionVectorTextureWidth, s = e.motionVectorTextureHeight; return (!n || (r == null ? void 0 : r.textureWidth) !== i || (r == null ? void 0 : r.textureHeight) != s) && (n = this._createRenderTargetTexture(i, s, null, e.motionVectorTexture, e.depthStencilTexture), this._renderTargetTextures.set(t.eye, n), this._framebufferDimensions = { framebufferWidth: i, framebufferHeight: s }), this._lastSubImages.set(t, e), n; } /** * {@inheritDoc} */ trySetViewportForView(e, t) { const r = this._lastSubImages.get(t) || this._getSubImageForView(t); return r ? (this._setViewportForSubImage(e, r), !0) : !1; } /** * Access the motion vector (which will turn on Space Warp) * @param view the view to access the motion vector texture for */ accessMotionVector(e) { const t = this._getSubImageForView(e); t && (t.motionVectorTexture, t.depthStencilTexture); } /** * {@inheritDoc} */ getRenderTargetTextureForEye(e) { return null; } /** * {@inheritDoc} */ getRenderTargetTextureForView(e) { const t = this._getSubImageForView(e); return t ? this._getRenderTargetForSubImage(t, e) : null; } /** * {@inheritDoc} */ dispose() { this._renderTargetTextures.forEach((e) => e.dispose()), this._renderTargetTextures.clear(); } } class gS extends L9 { /** * constructor for the space warp feature * @param _xrSessionManager the xr session manager for this feature */ constructor(e) { super(e), this._onAfterRenderObserver = null, this.dependsOn = [Gi.LAYERS], this.xrNativeFeatureName = "space-warp", this._xrSessionManager.scene.needsPreviousWorldMatrices = !0; } /** * Attach this feature. * Will usually be called by the features manager. * * @returns true if successful. */ attach() { if (!super.attach()) return !1; const e = this._xrSessionManager.scene.getEngine(); return this._glContext = e._gl, this._xrWebGLBinding = new XRWebGLBinding(this._xrSessionManager.session, this._glContext), this.spaceWarpRTTProvider = new Iie(this._xrSessionManager.scene, this._xrSessionManager, this._xrWebGLBinding), this._onAfterRenderObserver = this._xrSessionManager.scene.onAfterRenderObservable.add(() => this._onAfterRender()), !0; } detach() { return this._xrSessionManager.scene.onAfterRenderObservable.remove(this._onAfterRenderObserver), super.detach(); } _onAfterRender() { this.attached && this._renderTargetTexture && this._renderTargetTexture.render(!1, !1); } /** * {@inheritdoc} */ isCompatible() { return this._xrSessionManager.scene.getEngine().getCaps().colorBufferHalfFloat || !1; } /** * {@inheritdoc} */ dispose() { super.dispose(); } _onXRFrame(e) { const t = e.getViewerPose(this._xrSessionManager.referenceSpace); if (!t) return; const r = t.views[0]; this._renderTargetTexture = this._renderTargetTexture || this.spaceWarpRTTProvider.getRenderTargetTextureForView(r), this.spaceWarpRTTProvider.accessMotionVector(r); } } gS.Name = Gi.SPACE_WARP; gS.Version = 1; So.AddWebXRFeature(gS.Name, (A) => () => new gS(A), gS.Version, !1); class XS extends L9 { /** * Creates a new instance of the feature * @param _xrSessionManager the WebXRSessionManager * @param options options for the Feature */ constructor(e, t = {}) { super(e), this.options = t, this._cachedInternalTextures = [], this.texturesData = [], this.viewIndex = [], this.cameraIntrinsics = [], this.onTexturesUpdatedObservable = new Oe(), this.xrNativeFeatureName = "camera-access"; } attach(e) { return super.attach(e) ? (this._glContext = this._xrSessionManager.scene.getEngine()._gl, this._glBinding = new XRWebGLBinding(this._xrSessionManager.session, this._glContext), !0) : !1; } detach() { return super.detach() ? (this._glBinding = void 0, this.options.doNotDisposeOnDetach || (this._cachedInternalTextures.forEach((e) => e.dispose()), this.texturesData.forEach((e) => e.dispose()), this._cachedInternalTextures.length = 0, this.texturesData.length = 0, this.cameraIntrinsics.length = 0), !0) : !1; } /** * Dispose this feature and all of the resources attached */ dispose() { super.dispose(), this.onTexturesUpdatedObservable.clear(); } /** * @see https://github.com/immersive-web/raw-camera-access/blob/main/explainer.md */ _updateCameraIntrinsics(e, t) { const r = { width: e.camera.width, height: e.camera.height, x: 0, y: 0 }, n = e.projectionMatrix, i = (1 - n[8]) * r.width / 2 + r.x, s = (1 - n[9]) * r.height / 2 + r.y, a = r.width / 2 * n[0], f = r.height / 2 * n[5], o = r.width / 2 * n[4]; this.cameraIntrinsics[t] = { u0: i, v0: s, ax: a, ay: f, gamma: o, width: r.width, height: r.height, viewportX: r.x, viewportY: r.y }; } _updateInternalTextures(e, t = 0) { var r, n; if (!e.camera) return !1; this.viewIndex[t] = e.eye; const i = (r = this._glBinding) === null || r === void 0 ? void 0 : r.getCameraImage(e.camera); if (this._cachedInternalTextures[t]) (n = this._cachedInternalTextures[t]._hardwareTexture) === null || n === void 0 || n.set(i); else { const s = new As(this._xrSessionManager.scene.getEngine(), ri.Unknown, !0); s.isCube = !0, s.invertY = !1, s.format = 5, s.generateMipMaps = !0, s.type = 1, s.samplingMode = 3, s.width = e.camera.width, s.height = e.camera.height, s._cachedWrapU = 1, s._cachedWrapV = 1, s._hardwareTexture = new jm(i, this._glContext), this._cachedInternalTextures[t] = s; const a = new ls(this._xrSessionManager.scene); a.name = `WebXR Raw Camera Access (${t})`, a._texture = this._cachedInternalTextures[t], this.texturesData[t] = a, this._updateCameraIntrinsics(e, t); } return this._cachedInternalTextures[t].isReady = !0, !0; } _onXRFrame(e) { const t = this._xrSessionManager.referenceSpace, r = e.getViewerPose(t); if (!r || !r.views) return; let n = !0; r.views.forEach((i, s) => { n = n && this._updateInternalTextures(i, s); }), n && this.onTexturesUpdatedObservable.notifyObservers(this.texturesData); } } XS.Name = Gi.RAW_CAMERA_ACCESS; XS.Version = 1; So.AddWebXRFeature(XS.Name, (A, e) => () => new XS(A, e), XS.Version, !1); class Rie extends Um { /** * Create a new hand controller object, without loading a controller model * @param scene the scene to use to create this controller * @param gamepadObject the corresponding gamepad object * @param handedness the handedness of the controller */ constructor(e, t, r) { super(e, Vge[r], t, r, !0), this.profileId = "generic-hand-select-grasp"; } _getFilenameAndPath() { return { filename: "generic.babylon", path: "https://controllers.babylonjs.com/generic/" }; } _getModelLoadingConstraints() { return !0; } _processLoadedModel(e) { } _setRootMesh(e) { } _updateModel() { } } E9.RegisterController("generic-hand-select-grasp", (A, e) => new Rie(e, A.gamepad, A.handedness)); const Vge = { left: { selectComponentId: "xr-standard-trigger", components: { // eslint-disable-next-line @typescript-eslint/naming-convention "xr-standard-trigger": { type: "trigger", gamepadIndices: { button: 0 }, rootNodeName: "xr-standard-trigger", visualResponses: {} }, grasp: { type: "trigger", gamepadIndices: { button: 4 }, rootNodeName: "grasp", visualResponses: {} } }, gamepadMapping: "xr-standard", rootNodeName: "generic-hand-select-grasp-left", assetPath: "left.glb" }, right: { selectComponentId: "xr-standard-trigger", components: { // eslint-disable-next-line @typescript-eslint/naming-convention "xr-standard-trigger": { type: "trigger", gamepadIndices: { button: 0 }, rootNodeName: "xr-standard-trigger", visualResponses: {} }, grasp: { type: "trigger", gamepadIndices: { button: 4 }, rootNodeName: "grasp", visualResponses: {} } }, gamepadMapping: "xr-standard", rootNodeName: "generic-hand-select-grasp-right", assetPath: "right.glb" }, none: { selectComponentId: "xr-standard-trigger", components: { // eslint-disable-next-line @typescript-eslint/naming-convention "xr-standard-trigger": { type: "trigger", gamepadIndices: { button: 0 }, rootNodeName: "xr-standard-trigger", visualResponses: {} }, grasp: { type: "trigger", gamepadIndices: { button: 4 }, rootNodeName: "grasp", visualResponses: {} } }, gamepadMapping: "xr-standard", rootNodeName: "generic-hand-select-grasp-none", assetPath: "none.glb" } }; class cq extends Um { constructor(e, t, r) { super(e, Cge["left-right"], t, r), this._mapping = { defaultButton: { valueNodeName: "VALUE", unpressedNodeName: "UNPRESSED", pressedNodeName: "PRESSED" }, defaultAxis: { valueNodeName: "VALUE", minNodeName: "MIN", maxNodeName: "MAX" }, buttons: { "xr-standard-trigger": { rootNodeName: "SELECT", componentProperty: "button", states: ["default", "touched", "pressed"] }, "xr-standard-squeeze": { rootNodeName: "GRASP", componentProperty: "state", states: ["pressed"] }, "xr-standard-touchpad": { rootNodeName: "TOUCHPAD_PRESS", labelAnchorNodeName: "squeeze-label", touchPointNodeName: "TOUCH" // TODO - use this for visual feedback }, "xr-standard-thumbstick": { rootNodeName: "THUMBSTICK_PRESS", componentProperty: "state", states: ["pressed"] } }, axes: { "xr-standard-touchpad": { "x-axis": { rootNodeName: "TOUCHPAD_TOUCH_X" }, "y-axis": { rootNodeName: "TOUCHPAD_TOUCH_Y" } }, "xr-standard-thumbstick": { "x-axis": { rootNodeName: "THUMBSTICK_X" }, "y-axis": { rootNodeName: "THUMBSTICK_Y" } } } }, this.profileId = "microsoft-mixed-reality"; } _getFilenameAndPath() { let e = ""; this.handedness === "left" ? e = cq.MODEL_LEFT_FILENAME : e = cq.MODEL_RIGHT_FILENAME; const t = "default", r = cq.MODEL_BASE_URL + t + "/"; return { filename: e, path: r }; } _getModelLoadingConstraints() { const e = Hn.IsPluginForExtensionAvailable(".glb"); return e || Se.Warn("glTF / glb loaded was not registered, using generic controller instead"), e; } _processLoadedModel(e) { this.rootMesh && (this.getComponentIds().forEach((t, r) => { if (!this.disableAnimation && t && this.rootMesh) { const n = this._mapping.buttons[t], i = n.rootNodeName; if (!i) { Se.Log("Skipping unknown button at index: " + r + " with mapped name: " + t); return; } const s = this._getChildByName(this.rootMesh, i); if (!s) { Se.Warn("Missing button mesh with name: " + i); return; } if (n.valueMesh = this._getImmediateChildByName(s, this._mapping.defaultButton.valueNodeName), n.pressedMesh = this._getImmediateChildByName(s, this._mapping.defaultButton.pressedNodeName), n.unpressedMesh = this._getImmediateChildByName(s, this._mapping.defaultButton.unpressedNodeName), n.valueMesh && n.pressedMesh && n.unpressedMesh) { const a = this.getComponent(t); a && a.onButtonStateChangedObservable.add((f) => { this._lerpTransform(n, f.value); }, void 0, !0); } else Se.Warn("Missing button submesh under mesh with name: " + i); } }), this.getComponentIds().forEach((t) => { const r = this.getComponent(t); r.isAxes() && ["x-axis", "y-axis"].forEach((n) => { if (!this.rootMesh) return; const i = this._mapping.axes[t][n], s = this._getChildByName(this.rootMesh, i.rootNodeName); if (!s) { Se.Warn("Missing axis mesh with name: " + i.rootNodeName); return; } i.valueMesh = this._getImmediateChildByName(s, this._mapping.defaultAxis.valueNodeName), i.minMesh = this._getImmediateChildByName(s, this._mapping.defaultAxis.minNodeName), i.maxMesh = this._getImmediateChildByName(s, this._mapping.defaultAxis.maxNodeName), i.valueMesh && i.minMesh && i.maxMesh ? r && r.onAxisValueChangedObservable.add((a) => { const f = n === "x-axis" ? a.x : a.y; this._lerpTransform(i, f, !0); }, void 0, !0) : Se.Warn("Missing axis submesh under mesh with name: " + i.rootNodeName); }); })); } _setRootMesh(e) { this.rootMesh = new Ee(this.profileId + " " + this.handedness, this.scene), this.rootMesh.isPickable = !1; let t; for (let r = 0; r < e.length; r++) { const n = e[r]; n.isPickable = !1, n.parent || (t = n); } t && t.setParent(this.rootMesh), this.scene.useRightHandedSystem || (this.rootMesh.rotationQuaternion = Ze.FromEulerAngles(0, Math.PI, 0)); } _updateModel() { } } cq.MODEL_BASE_URL = "https://controllers.babylonjs.com/microsoft/"; cq.MODEL_LEFT_FILENAME = "left.glb"; cq.MODEL_RIGHT_FILENAME = "right.glb"; E9.RegisterController("windows-mixed-reality", (A, e) => new cq(e, A.gamepad, A.handedness)); const Cge = { left: { selectComponentId: "xr-standard-trigger", components: { "xr-standard-trigger": { type: "trigger", gamepadIndices: { button: 0 }, rootNodeName: "xr_standard_trigger", visualResponses: { xr_standard_trigger_pressed: { componentProperty: "button", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_trigger_pressed_value", minNodeName: "xr_standard_trigger_pressed_min", maxNodeName: "xr_standard_trigger_pressed_max" } } }, "xr-standard-squeeze": { type: "squeeze", gamepadIndices: { button: 1 }, rootNodeName: "xr_standard_squeeze", visualResponses: { xr_standard_squeeze_pressed: { componentProperty: "button", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_squeeze_pressed_value", minNodeName: "xr_standard_squeeze_pressed_min", maxNodeName: "xr_standard_squeeze_pressed_max" } } }, "xr-standard-touchpad": { type: "touchpad", gamepadIndices: { button: 2, xAxis: 0, yAxis: 1 }, rootNodeName: "xr_standard_touchpad", visualResponses: { xr_standard_touchpad_pressed: { componentProperty: "button", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_touchpad_pressed_value", minNodeName: "xr_standard_touchpad_pressed_min", maxNodeName: "xr_standard_touchpad_pressed_max" }, xr_standard_touchpad_xaxis_pressed: { componentProperty: "xAxis", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_touchpad_xaxis_pressed_value", minNodeName: "xr_standard_touchpad_xaxis_pressed_min", maxNodeName: "xr_standard_touchpad_xaxis_pressed_max" }, xr_standard_touchpad_yaxis_pressed: { componentProperty: "yAxis", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_touchpad_yaxis_pressed_value", minNodeName: "xr_standard_touchpad_yaxis_pressed_min", maxNodeName: "xr_standard_touchpad_yaxis_pressed_max" }, xr_standard_touchpad_xaxis_touched: { componentProperty: "xAxis", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_touchpad_xaxis_touched_value", minNodeName: "xr_standard_touchpad_xaxis_touched_min", maxNodeName: "xr_standard_touchpad_xaxis_touched_max" }, xr_standard_touchpad_yaxis_touched: { componentProperty: "yAxis", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_touchpad_yaxis_touched_value", minNodeName: "xr_standard_touchpad_yaxis_touched_min", maxNodeName: "xr_standard_touchpad_yaxis_touched_max" }, xr_standard_touchpad_axes_touched: { componentProperty: "state", states: ["touched", "pressed"], valueNodeProperty: "visibility", valueNodeName: "xr_standard_touchpad_axes_touched_value" } }, touchPointNodeName: "xr_standard_touchpad_axes_touched_value" }, "xr-standard-thumbstick": { type: "thumbstick", gamepadIndices: { button: 3, xAxis: 2, yAxis: 3 }, rootNodeName: "xr_standard_thumbstick", visualResponses: { xr_standard_thumbstick_pressed: { componentProperty: "button", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_thumbstick_pressed_value", minNodeName: "xr_standard_thumbstick_pressed_min", maxNodeName: "xr_standard_thumbstick_pressed_max" }, xr_standard_thumbstick_xaxis_pressed: { componentProperty: "xAxis", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_thumbstick_xaxis_pressed_value", minNodeName: "xr_standard_thumbstick_xaxis_pressed_min", maxNodeName: "xr_standard_thumbstick_xaxis_pressed_max" }, xr_standard_thumbstick_yaxis_pressed: { componentProperty: "yAxis", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_thumbstick_yaxis_pressed_value", minNodeName: "xr_standard_thumbstick_yaxis_pressed_min", maxNodeName: "xr_standard_thumbstick_yaxis_pressed_max" } } } }, gamepadMapping: "xr-standard", rootNodeName: "microsoft-mixed-reality-left", assetPath: "left.glb" }, right: { selectComponentId: "xr-standard-trigger", components: { "xr-standard-trigger": { type: "trigger", gamepadIndices: { button: 0 }, rootNodeName: "xr_standard_trigger", visualResponses: { xr_standard_trigger_pressed: { componentProperty: "button", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_trigger_pressed_value", minNodeName: "xr_standard_trigger_pressed_min", maxNodeName: "xr_standard_trigger_pressed_max" } } }, "xr-standard-squeeze": { type: "squeeze", gamepadIndices: { button: 1 }, rootNodeName: "xr_standard_squeeze", visualResponses: { xr_standard_squeeze_pressed: { componentProperty: "button", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_squeeze_pressed_value", minNodeName: "xr_standard_squeeze_pressed_min", maxNodeName: "xr_standard_squeeze_pressed_max" } } }, "xr-standard-touchpad": { type: "touchpad", gamepadIndices: { button: 2, xAxis: 0, yAxis: 1 }, rootNodeName: "xr_standard_touchpad", visualResponses: { xr_standard_touchpad_pressed: { componentProperty: "button", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_touchpad_pressed_value", minNodeName: "xr_standard_touchpad_pressed_min", maxNodeName: "xr_standard_touchpad_pressed_max" }, xr_standard_touchpad_xaxis_pressed: { componentProperty: "xAxis", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_touchpad_xaxis_pressed_value", minNodeName: "xr_standard_touchpad_xaxis_pressed_min", maxNodeName: "xr_standard_touchpad_xaxis_pressed_max" }, xr_standard_touchpad_yaxis_pressed: { componentProperty: "yAxis", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_touchpad_yaxis_pressed_value", minNodeName: "xr_standard_touchpad_yaxis_pressed_min", maxNodeName: "xr_standard_touchpad_yaxis_pressed_max" }, xr_standard_touchpad_xaxis_touched: { componentProperty: "xAxis", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_touchpad_xaxis_touched_value", minNodeName: "xr_standard_touchpad_xaxis_touched_min", maxNodeName: "xr_standard_touchpad_xaxis_touched_max" }, xr_standard_touchpad_yaxis_touched: { componentProperty: "yAxis", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_touchpad_yaxis_touched_value", minNodeName: "xr_standard_touchpad_yaxis_touched_min", maxNodeName: "xr_standard_touchpad_yaxis_touched_max" }, xr_standard_touchpad_axes_touched: { componentProperty: "state", states: ["touched", "pressed"], valueNodeProperty: "visibility", valueNodeName: "xr_standard_touchpad_axes_touched_value" } }, touchPointNodeName: "xr_standard_touchpad_axes_touched_value" }, "xr-standard-thumbstick": { type: "thumbstick", gamepadIndices: { button: 3, xAxis: 2, yAxis: 3 }, rootNodeName: "xr_standard_thumbstick", visualResponses: { xr_standard_thumbstick_pressed: { componentProperty: "button", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_thumbstick_pressed_value", minNodeName: "xr_standard_thumbstick_pressed_min", maxNodeName: "xr_standard_thumbstick_pressed_max" }, xr_standard_thumbstick_xaxis_pressed: { componentProperty: "xAxis", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_thumbstick_xaxis_pressed_value", minNodeName: "xr_standard_thumbstick_xaxis_pressed_min", maxNodeName: "xr_standard_thumbstick_xaxis_pressed_max" }, xr_standard_thumbstick_yaxis_pressed: { componentProperty: "yAxis", states: ["default", "touched", "pressed"], valueNodeProperty: "transform", valueNodeName: "xr_standard_thumbstick_yaxis_pressed_value", minNodeName: "xr_standard_thumbstick_yaxis_pressed_min", maxNodeName: "xr_standard_thumbstick_yaxis_pressed_max" } } } }, gamepadMapping: "xr-standard", rootNodeName: "microsoft-mixed-reality-right", assetPath: "right.glb" } }; class R0 extends Um { constructor(e, t, r, n = !1, i = !1) { super(e, Oge[r], t, r), this._forceLegacyControllers = i, this.profileId = "oculus-touch"; } _getFilenameAndPath() { let e = ""; this.handedness === "left" ? e = R0.MODEL_LEFT_FILENAME : e = R0.MODEL_RIGHT_FILENAME; const t = this._isQuest() ? R0.QUEST_MODEL_BASE_URL : R0.MODEL_BASE_URL; return { filename: e, path: t }; } _getModelLoadingConstraints() { return !0; } _processLoadedModel(e) { const t = this._isQuest(), r = this.handedness === "right" ? -1 : 1; this.getComponentIds().forEach((n) => { const i = n && this.getComponent(n); i && i.onButtonStateChangedObservable.add((s) => { if (!(!this.rootMesh || this.disableAnimation)) switch (n) { case "xr-standard-trigger": t || (this._modelRootNode.getChildren()[3].rotation.x = -s.value * 0.2, this._modelRootNode.getChildren()[3].position.y = -s.value * 5e-3, this._modelRootNode.getChildren()[3].position.z = -s.value * 5e-3); return; case "xr-standard-squeeze": t || (this._modelRootNode.getChildren()[4].position.x = r * s.value * 35e-4); return; case "xr-standard-thumbstick": return; case "a-button": case "x-button": t || (s.pressed ? this._modelRootNode.getChildren()[1].position.y = -1e-3 : this._modelRootNode.getChildren()[1].position.y = 0); return; case "b-button": case "y-button": t || (s.pressed ? this._modelRootNode.getChildren()[2].position.y = -1e-3 : this._modelRootNode.getChildren()[2].position.y = 0); return; } }, void 0, !0); }); } _setRootMesh(e) { this.rootMesh = new Ee(this.profileId + " " + this.handedness, this.scene), this.scene.useRightHandedSystem || (this.rootMesh.rotationQuaternion = Ze.FromEulerAngles(0, Math.PI, 0)), e.forEach((t) => { t.isPickable = !1; }), this._isQuest() ? this._modelRootNode = e[0] : (this._modelRootNode = e[1], this.rootMesh.position.y = 0.034, this.rootMesh.position.z = 0.052), this._modelRootNode.parent = this.rootMesh; } _updateModel() { } /** * Is this the new type of oculus touch. At the moment both have the same profile and it is impossible to differentiate * between the touch and touch 2. */ _isQuest() { return !!navigator.userAgent.match(/Quest/gi) && !this._forceLegacyControllers; } } R0.MODEL_BASE_URL = "https://controllers.babylonjs.com/oculus/"; R0.MODEL_LEFT_FILENAME = "left.babylon"; R0.MODEL_RIGHT_FILENAME = "right.babylon"; R0.QUEST_MODEL_BASE_URL = "https://controllers.babylonjs.com/oculusQuest/"; E9.RegisterController("oculus-touch", (A, e) => new R0(e, A.gamepad, A.handedness)); E9.RegisterController("oculus-touch-legacy", (A, e) => new R0(e, A.gamepad, A.handedness, !0)); const Oge = { left: { selectComponentId: "xr-standard-trigger", components: { "xr-standard-trigger": { type: "trigger", gamepadIndices: { button: 0 }, rootNodeName: "xr_standard_trigger", visualResponses: {} }, "xr-standard-squeeze": { type: "squeeze", gamepadIndices: { button: 1 }, rootNodeName: "xr_standard_squeeze", visualResponses: {} }, "xr-standard-thumbstick": { type: "thumbstick", gamepadIndices: { button: 3, xAxis: 2, yAxis: 3 }, rootNodeName: "xr_standard_thumbstick", visualResponses: {} }, "x-button": { type: "button", gamepadIndices: { button: 4 }, rootNodeName: "x_button", visualResponses: {} }, "y-button": { type: "button", gamepadIndices: { button: 5 }, rootNodeName: "y_button", visualResponses: {} }, thumbrest: { type: "button", gamepadIndices: { button: 6 }, rootNodeName: "thumbrest", visualResponses: {} } }, gamepadMapping: "xr-standard", rootNodeName: "oculus-touch-v2-left", assetPath: "left.glb" }, right: { selectComponentId: "xr-standard-trigger", components: { "xr-standard-trigger": { type: "trigger", gamepadIndices: { button: 0 }, rootNodeName: "xr_standard_trigger", visualResponses: {} }, "xr-standard-squeeze": { type: "squeeze", gamepadIndices: { button: 1 }, rootNodeName: "xr_standard_squeeze", visualResponses: {} }, "xr-standard-thumbstick": { type: "thumbstick", gamepadIndices: { button: 3, xAxis: 2, yAxis: 3 }, rootNodeName: "xr_standard_thumbstick", visualResponses: {} }, "a-button": { type: "button", gamepadIndices: { button: 4 }, rootNodeName: "a_button", visualResponses: {} }, "b-button": { type: "button", gamepadIndices: { button: 5 }, rootNodeName: "b_button", visualResponses: {} }, thumbrest: { type: "button", gamepadIndices: { button: 6 }, rootNodeName: "thumbrest", visualResponses: {} } }, gamepadMapping: "xr-standard", rootNodeName: "oculus-touch-v2-right", assetPath: "right.glb" } }; class qm extends Um { /** * Create a new Vive motion controller object * @param scene the scene to use to create this controller * @param gamepadObject the corresponding gamepad object * @param handedness the handedness of the controller */ constructor(e, t, r) { super(e, yge[r], t, r), this.profileId = "htc-vive"; } _getFilenameAndPath() { const e = qm.MODEL_FILENAME, t = qm.MODEL_BASE_URL; return { filename: e, path: t }; } _getModelLoadingConstraints() { return !0; } _processLoadedModel(e) { this.getComponentIds().forEach((t) => { const r = t && this.getComponent(t); r && r.onButtonStateChangedObservable.add((n) => { if (!(!this.rootMesh || this.disableAnimation)) switch (t) { case "xr-standard-trigger": this._modelRootNode.getChildren()[6].rotation.x = -n.value * 0.15; return; case "xr-standard-touchpad": return; case "xr-standard-squeeze": return; } }, void 0, !0); }); } _setRootMesh(e) { this.rootMesh = new Ee(this.profileId + " " + this.handedness, this.scene), e.forEach((t) => { t.isPickable = !1; }), this._modelRootNode = e[1], this._modelRootNode.parent = this.rootMesh, this.scene.useRightHandedSystem || (this.rootMesh.rotationQuaternion = Ze.FromEulerAngles(0, Math.PI, 0)); } _updateModel() { } } qm.MODEL_BASE_URL = "https://controllers.babylonjs.com/vive/"; qm.MODEL_FILENAME = "wand.babylon"; E9.RegisterController("htc-vive", (A, e) => new qm(e, A.gamepad, A.handedness)); const yge = { left: { selectComponentId: "xr-standard-trigger", components: { "xr-standard-trigger": { type: "trigger", gamepadIndices: { button: 0 }, rootNodeName: "xr_standard_trigger", visualResponses: {} }, "xr-standard-squeeze": { type: "squeeze", gamepadIndices: { button: 1 }, rootNodeName: "xr_standard_squeeze", visualResponses: {} }, "xr-standard-touchpad": { type: "touchpad", gamepadIndices: { button: 2, xAxis: 0, yAxis: 1 }, rootNodeName: "xr_standard_touchpad", visualResponses: {} }, menu: { type: "button", gamepadIndices: { button: 4 }, rootNodeName: "menu", visualResponses: {} } }, gamepadMapping: "xr-standard", rootNodeName: "htc_vive_none", assetPath: "none.glb" }, right: { selectComponentId: "xr-standard-trigger", components: { "xr-standard-trigger": { type: "trigger", gamepadIndices: { button: 0 }, rootNodeName: "xr_standard_trigger", visualResponses: {} }, "xr-standard-squeeze": { type: "squeeze", gamepadIndices: { button: 1 }, rootNodeName: "xr_standard_squeeze", visualResponses: {} }, "xr-standard-touchpad": { type: "touchpad", gamepadIndices: { button: 2, xAxis: 0, yAxis: 1 }, rootNodeName: "xr_standard_touchpad", visualResponses: {} }, menu: { type: "button", gamepadIndices: { button: 4 }, rootNodeName: "menu", visualResponses: {} } }, gamepadMapping: "xr-standard", rootNodeName: "htc_vive_none", assetPath: "none.glb" }, none: { selectComponentId: "xr-standard-trigger", components: { "xr-standard-trigger": { type: "trigger", gamepadIndices: { button: 0 }, rootNodeName: "xr_standard_trigger", visualResponses: {} }, "xr-standard-squeeze": { type: "squeeze", gamepadIndices: { button: 1 }, rootNodeName: "xr_standard_squeeze", visualResponses: {} }, "xr-standard-touchpad": { type: "touchpad", gamepadIndices: { button: 2, xAxis: 0, yAxis: 1 }, rootNodeName: "xr_standard_touchpad", visualResponses: {} }, menu: { type: "button", gamepadIndices: { button: 4 }, rootNodeName: "menu", visualResponses: {} } }, gamepadMapping: "xr-standard", rootNodeName: "htc-vive-none", assetPath: "none.glb" } }; class Vie { get session() { return this._nativeImpl.session; } constructor(e) { this._nativeImpl = e, this._xrTransform = new XRRigidTransform(), this._xrPose = { transform: this._xrTransform, emulatedPosition: !1 }, this._xrPoseVectorData = new Float32Array(8), this.fillPoses = this._nativeImpl.fillPoses.bind(this._nativeImpl), this.getViewerPose = this._nativeImpl.getViewerPose.bind(this._nativeImpl), this.getHitTestResults = this._nativeImpl.getHitTestResults.bind(this._nativeImpl), this.getHitTestResultsForTransientInput = () => { throw new Error("XRFrame.getHitTestResultsForTransientInput not supported on native."); }, this.createAnchor = this._nativeImpl.createAnchor.bind(this._nativeImpl), this.getJointPose = this._nativeImpl.getJointPose.bind(this._nativeImpl), this.fillJointRadii = this._nativeImpl.fillJointRadii.bind(this._nativeImpl), this.getLightEstimate = () => { throw new Error("XRFrame.getLightEstimate not supported on native."); }, this.getImageTrackingResults = () => { var t; return (t = this._nativeImpl._imageTrackingResults) !== null && t !== void 0 ? t : []; }; } getPose(e, t) { if (!this._nativeImpl.getPoseData(e, t, this._xrPoseVectorData.buffer, this._xrTransform.matrix.buffer)) return; const r = this._xrTransform.position; r.x = this._xrPoseVectorData[0], r.y = this._xrPoseVectorData[1], r.z = this._xrPoseVectorData[2], r.w = this._xrPoseVectorData[3]; const n = this._xrTransform.orientation; return n.x = this._xrPoseVectorData[4], n.y = this._xrPoseVectorData[5], n.z = this._xrPoseVectorData[6], n.w = this._xrPoseVectorData[7], this._xrPose; } get trackedAnchors() { return this._nativeImpl.trackedAnchors; } get worldInformation() { return this._nativeImpl.worldInformation; } get detectedPlanes() { return this._nativeImpl.detectedPlanes; } get featurePointCloud() { return this._nativeImpl.featurePointCloud; } getDepthInformation(e) { throw new Error("This function is not available in Babylon Native"); } } gte("NativeXRFrame", Vie); var f4; (function(A) { A[A.Input = 0] = "Input", A[A.Output = 1] = "Output"; })(f4 || (f4 = {})); class IF { constructor(e, t, r) { this._ownerBlock = r, this._connectedPoint = [], this.uniqueId = v4(), this.connectedPointIds = [], this.name = e, this._connectionType = t; } /** * The type of the connection */ get connectionType() { return this._connectionType; } /** * @internal * Override this to indicate if a point can connect to more than one point. */ _isSingularConnection() { return !0; } /** * Returns if a point is connected to any other point. * @returns boolean indicating if the point is connected. */ isConnected() { return this._connectedPoint.length > 0; } /** * Connects two points together. * @param point */ connectTo(e) { if (this._connectionType === e._connectionType) throw new Error(`Cannot connect two points of type ${this.connectionType}`); if (this._isSingularConnection() && this._connectedPoint.length > 0 || e._isSingularConnection() && e._connectedPoint.length > 0) throw new Error("Max number of connections for point reached"); this._connectedPoint.push(e), e._connectedPoint.push(this); } /** * Saves the connection to a JSON object. */ serialize(e = {}) { e.uniqueId = this.uniqueId, e.name = this.name, e._connectionType = this._connectionType, e.connectedPointIds = [], e.className = this.getClassName(); for (const t of this._connectedPoint) e.connectedPointIds.push(t.uniqueId); } getClassName() { return "FGConnection"; } /** * Deserialize from a object into this * @param serializationObject */ deserialize(e) { this.uniqueId = e.uniqueId, this.name = e.name, this._connectionType = e._connectionType, this.connectedPointIds = e.connectedPointIds; } /** * Parses a connection from an object * @param serializationObject * @param ownerBlock * @returns */ static Parse(e = {}, t) { const r = ye.Instantiate(e.className), n = new r(e.name, e._connectionType, t); return n.deserialize(e), n; } } class Zl { constructor(e, t) { this.typeName = e, this.defaultValue = t; } serialize(e) { e.typeName = this.typeName, e.defaultValue = this.defaultValue; } static Parse(e) { return new Zl(e.typeName, e.defaultValue); } } const w6 = new Zl("any", void 0), pV = new Zl("string", ""), Yt = new Zl("number", 0), m6 = new Zl("boolean", !1), Y9 = new Zl("Vector2", at.Zero()), ma = new Zl("Vector3", S.Zero()), iA = new Zl("Vector4", Ir.Zero()), D1 = new Zl("Matrix", he.Identity()), Cie = new Zl("Color3", Ne.Black()), Oie = new Zl("Color4", new xt(0, 0, 0, 0)), hV = new Zl("Quaternion", Ze.Identity()); function yie(A) { switch (typeof A) { case "string": return pV; case "number": return Yt; case "boolean": return m6; case "object": return A instanceof at ? Y9 : A instanceof S ? ma : A instanceof Ir ? iA : A instanceof Ne ? Cie : A instanceof xt ? Oie : A instanceof Ze ? hV : w6; default: return w6; } } class RF extends IF { /** * Create a new data connection point. * @param name * @param connectionType * @param ownerBlock * @param richType */ constructor(e, t, r, n) { super(e, t, r), this.richType = n; } /** * An output data block can connect to multiple input data blocks, * but an input data block can only connect to one output data block. */ _isSingularConnection() { return this.connectionType === f4.Input; } /** * Set the value of the connection in a specific context. * @param value the value to set * @param context the context to which the value is set */ setValue(e, t) { t._setConnectionValue(this, e); } connectTo(e) { super.connectTo(e); } _getValueOrDefault(e) { return e._hasConnectionValue(this) ? e._getConnectionValue(this) : this.richType.defaultValue; } /** * Gets the value of the connection in a specific context. * @param context the context from which the value is retrieved * @returns the value of the connection */ getValue(e) { return this.connectionType === f4.Output ? (e._notifyExecuteNode(this._ownerBlock), this._ownerBlock._updateOutputs(e), this._getValueOrDefault(e)) : this.isConnected() ? this._connectedPoint[0].getValue(e) : this._getValueOrDefault(e); } getClassName() { return "FGDataConnection"; } serialize(e = {}) { super.serialize(e), e.richType = {}, this.richType.serialize(e.richType); } static Parse(e, t) { const r = IF.Parse(e, t); return r.richType = Zl.Parse(e.richType), r; } } Ue("FGDataConnection", RF); class CA { /** Constructor is protected so only subclasses can be instantiated */ constructor(e) { this.config = e, this.uniqueId = v4(), this.configure(); } configure() { var e, t; this.name = (t = (e = this.config) === null || e === void 0 ? void 0 : e.name) !== null && t !== void 0 ? t : this.getClassName(), this.dataInputs = [], this.dataOutputs = []; } /** * @internal */ _updateOutputs(e) { } registerDataInput(e, t) { const r = new RF(e, f4.Input, this, t); return this.dataInputs.push(r), r; } registerDataOutput(e, t) { const r = new RF(e, f4.Output, this, t); return this.dataOutputs.push(r), r; } getDataInput(e) { return this.dataInputs.find((t) => t.name === e); } serialize(e = {}) { e.uniqueId = this.uniqueId, e.config = {}, this.config && (e.config.name = this.config.name), e.dataInputs = [], e.dataOutputs = [], e.className = this.getClassName(); for (const t of this.dataInputs) { const r = {}; t.serialize(r), e.dataInputs.push(r); } for (const t of this.dataOutputs) { const r = {}; t.serialize(r), e.dataOutputs.push(r); } } getClassName() { return "FGBlock"; } static Parse(e) { const t = ye.Instantiate(e.className), r = {}; if (e.config) for (const i in e.config) { const s = e.config[i]; if (s && s.className) { const a = ye.Instantiate(s.className); r[i] = a.prototype.Parse(s); } else r[i] = s; } const n = new t(r); n.uniqueId = e.uniqueId; for (let i = 0; i < e.dataInputs.length; i++) n.dataInputs[i].deserialize(e.dataInputs[i]); for (let i = 0; i < e.dataOutputs.length; i++) n.dataOutputs[i].deserialize(e.dataOutputs[i]); if (n instanceof og) { for (let i = 0; i < e.signalInputs.length; i++) n.signalInputs[i].deserialize(e.signalInputs[i]); for (let i = 0; i < e.signalOutputs.length; i++) n.signalOutputs[i].deserialize(e.signalOutputs[i]); } return n; } } class uO extends IF { /** * A signal input can be connected to more than one signal output, * but a signal output can only connect to one signal input */ _isSingularConnection() { return this.connectionType === f4.Output; } /** * @internal */ _activateSignal(e) { var t; this.connectionType === f4.Input ? (e._notifyExecuteNode(this._ownerBlock), this._ownerBlock._execute(e, this), e._increaseExecutionId()) : (t = this._connectedPoint[0]) === null || t === void 0 || t._activateSignal(e); } } Ue("FlowGraphSignalConnection", uO); class og extends CA { constructor(e) { super(e), this.onStart = this._registerSignalInput("onStart"); } configure() { super.configure(), this.signalInputs = [], this.signalOutputs = []; } _registerSignalInput(e) { const t = new uO(e, f4.Input, this); return this.signalInputs.push(t), t; } _registerSignalOutput(e) { const t = new uO(e, f4.Output, this); return this.signalOutputs.push(t), t; } serialize(e = {}) { super.serialize(e), e.signalInputs = [], e.signalOutputs = []; for (const t of this.signalInputs) { const r = {}; t.serialize(r), e.signalInputs.push(r); } for (const t of this.signalOutputs) { const r = {}; t.serialize(r), e.signalOutputs.push(r); } } getClassName() { return "FGExecutionBlock"; } static Parse(e = {}) { const t = super.Parse(e); for (let r = 0; r < e.signalInputs.length; r++) t.signalInputs[r].deserialize(e.signalInputs[r]); for (let r = 0; r < e.signalOutputs.length; r++) t.signalOutputs[r].deserialize(e.signalOutputs[r]); return t; } } class $Y extends og { constructor(e) { super(e), this.onDone = this._registerSignalOutput("onDone"); } /** * @internal * @param context */ _startPendingTasks(e) { this._preparePendingTasks(e), e._addPendingBlock(this); } } class HU extends $Y { /** * @internal */ _execute(e) { e._notifyExecuteNode(this), this.onDone._activateSignal(e); } } function kie(A) { return A === "Mesh" || A === "AbstractMesh" || A === "GroundMesh" || A === "InstanceMesh" || A === "LinesMesh" || A === "GoldbergMesh" || A === "GreasedLineMesh" || A === "TrailMesh"; } function Eie(A) { return A === "Vector2" || A === "Vector3" || A === "Vector4" || A === "Quaternion"; } function kge(A, e) { if (A === "Vector2") return at.FromArray(e); if (A === "Vector3") return S.FromArray(e); if (A === "Vector4") return Ir.FromArray(e); if (A === "Quaternion") return Ze.FromArray(e); throw new Error(`Unknown vector class name ${A}`); } function Ege(A, e, t) { var r, n; const i = (n = (r = e == null ? void 0 : e.getClassName) === null || r === void 0 ? void 0 : r.call(e)) !== null && n !== void 0 ? n : ""; kie(i) ? t[A] = { name: e.name, className: i } : Eie(i) ? t[A] = { value: e.asArray(), className: i } : t[A] = e; } function Fge(A, e, t) { const r = e[A]; let n; const i = r == null ? void 0 : r.className; return kie(i) ? n = t.getMeshByName(r.name) : Eie(i) ? n = kge(i, r.value) : n = r, n; } class lO { constructor(e) { this.uniqueId = v4(), this._userVariables = {}, this._executionVariables = {}, this._connectionValues = {}, this._pendingBlocks = [], this._executionId = 0, this.onNodeExecutedObservable = new Oe(), this._configuration = e; } /** * Check if a user-defined variable is defined. * @param name * @returns */ hasVariable(e) { return e in this._userVariables; } /** * Set a user-defined variable. * @param name * @param value */ setVariable(e, t) { this._userVariables[e] = t; } /** * Get a user-defined variable. * @param name * @returns */ getVariable(e) { return this._userVariables[e]; } _getUniqueIdPrefixedName(e, t) { return `${e.uniqueId}_${t}`; } /** * Set an internal execution variable * @internal * @param name * @param value */ _setExecutionVariable(e, t, r) { this._executionVariables[this._getUniqueIdPrefixedName(e, t)] = r; } /** * Get an internal execution variable * @internal * @param name * @returns */ _getExecutionVariable(e, t, r) { return this._hasExecutionVariable(e, t) ? this._executionVariables[this._getUniqueIdPrefixedName(e, t)] : r; } /** * Delete an internal execution variable * @internal * @param block * @param name */ _deleteExecutionVariable(e, t) { delete this._executionVariables[this._getUniqueIdPrefixedName(e, t)]; } /** * Check if an internal execution variable is defined * @internal * @param block * @param name * @returns */ _hasExecutionVariable(e, t) { return this._getUniqueIdPrefixedName(e, t) in this._executionVariables; } /** * Check if a connection value is defined * @internal * @param connectionPoint * @returns */ _hasConnectionValue(e) { return e.uniqueId in this._connectionValues; } /** * Set a connection value * @internal * @param connectionPoint * @param value */ _setConnectionValue(e, t) { this._connectionValues[e.uniqueId] = t; } /** * Get a connection value * @internal * @param connectionPoint * @returns */ _getConnectionValue(e) { return this._connectionValues[e.uniqueId]; } /** * Get the configuration * @internal * @param name * @param value */ get configuration() { return this._configuration; } /** * Add a block to the list of blocks that have pending tasks. * @internal * @param block */ _addPendingBlock(e) { this._pendingBlocks.push(e); } /** * Remove a block from the list of blocks that have pending tasks. * @internal * @param block */ _removePendingBlock(e) { const t = this._pendingBlocks.indexOf(e); t !== -1 && this._pendingBlocks.splice(t, 1); } /** * Clear all pending blocks. * @internal */ _clearPendingBlocks() { for (const e of this._pendingBlocks) e._cancelPendingTasks(this); this._pendingBlocks.length = 0; } /** * @internal * Function that notifies the node executed observable * @param node */ _notifyExecuteNode(e) { this.onNodeExecutedObservable.notifyObservers(e); } /** * @internal */ _increaseExecutionId() { this._executionId++; } /** * A monotonically increasing ID for each execution. * Incremented for every block executed. */ get executionId() { return this._executionId; } /** * Serializes a context * @param serializationObject the object to write the values in * @param valueSerializationFunction a function to serialize complex values */ serialize(e = {}, t = Ege) { e.uniqueId = this.uniqueId, e._userVariables = {}; for (const r in this._userVariables) t(r, this._userVariables[r], e._userVariables); e._connectionValues = {}; for (const r in this._connectionValues) t(r, this._connectionValues[r], e._connectionValues); } getClassName() { return "FGContext"; } /** * Parses a context * @param serializationObject the object containing the context serialization values * @param graph the graph to which the context should belong * @param valueParseFunction a function to parse complex values * @returns */ static Parse(e = {}, t, r = Fge) { const n = t.createContext(); n.uniqueId = e.uniqueId; for (const i in e._userVariables) { const s = r(i, e._userVariables, n._configuration.scene); n._userVariables[i] = s; } for (const i in e._connectionValues) { const s = r(i, e._connectionValues, n._configuration.scene); n._connectionValues[i] = s; } return n; } } C([ M() ], lO.prototype, "uniqueId", void 0); function eM(A, e) { return !!(A.parent && (A.parent === e || eM(A.parent, e))); } class bm extends HU { constructor(e) { e.path.hasTemplateStrings && ye.Warn("Template strings are not supported in the path of mesh pick event blocks."), super(e), this.config = e; } _getReferencedMesh(e) { return this.config.path.getProperty(e); } /** * @internal */ _preparePendingTasks(e) { let t = e._getExecutionVariable(this, "meshPickObserver"); if (!t) { const r = this.config.path.getProperty(e); if (!r || !(r instanceof jn)) throw new Error("Mesh pick event block requires a valid mesh"); e._setExecutionVariable(this, "mesh", r), t = r.getScene().onPointerObservable.add((i) => { var s, a, f; i.type === ir.POINTERPICK && (!((s = i.pickInfo) === null || s === void 0) && s.pickedMesh) && (((a = i.pickInfo) === null || a === void 0 ? void 0 : a.pickedMesh) === r || eM((f = i.pickInfo) === null || f === void 0 ? void 0 : f.pickedMesh, r)) && this._execute(e); }); const n = r.onDisposeObservable.add(() => this._onDispose); e._setExecutionVariable(this, "meshPickObserver", t), e._setExecutionVariable(this, "meshDisposeObserver", n); } } _onDispose(e) { this._cancelPendingTasks(e), e._removePendingBlock(this); } /** * @internal */ _cancelPendingTasks(e) { const t = e._getExecutionVariable(this, "mesh"), r = e._getExecutionVariable(this, "meshPickObserver"), n = e._getExecutionVariable(this, "meshDisposeObserver"); t.getScene().onPointerObservable.remove(r), t.onDisposeObservable.remove(n), e._deleteExecutionVariable(this, "mesh"), e._deleteExecutionVariable(this, "meshPickObserver"), e._deleteExecutionVariable(this, "meshDisposeObserver"); } getClassName() { return bm.ClassName; } serialize(e) { super.serialize(e), e.config.path = this.config.path.serialize(); } } bm.ClassName = "FGMeshPickEventBlock"; Ue(bm.ClassName, bm); var $x; (function(A) { A[A.Stopped = 0] = "Stopped", A[A.Started = 1] = "Started"; })($x || ($x = {})); class JS { /** * Construct a Flow Graph * @param params construction parameters. currently only the scene */ constructor(e) { this._eventBlocks = [], this._executionContexts = [], this.state = $x.Stopped, this._scene = e.scene, this._coordinator = e.coordinator, this._sceneDisposeObserver = this._scene.onDisposeObservable.add(() => this.dispose()); } /** * Create a context. A context represents one self contained execution for the graph, with its own variables. * @returns the context, where you can get and set variables */ createContext() { const e = new lO({ scene: this._scene, coordinator: this._coordinator }); return this._executionContexts.push(e), e; } /** * Add an event block. When the graph is started, it will start listening to events * from the block and execute the graph when they are triggered. * @param block */ addEventBlock(e) { this._eventBlocks.push(e); } /** * Starts the flow graph. Initializes the event blocks and starts listening to events. */ start() { if (this.state !== $x.Started) { this.state = $x.Started, this._executionContexts.length === 0 && this.createContext(); for (const e of this._executionContexts) { const t = this._getContextualOrder(e); for (const r of t) r._startPendingTasks(e); } } } _getContextualOrder(e) { const t = []; for (const r of this._eventBlocks) if (r.getClassName() === bm.ClassName) { const n = r._getReferencedMesh(e); let i = 0; for (; i < t.length; i++) { const a = t[i]._getReferencedMesh(e); if (n && a && eM(n, a)) break; } t.splice(i, 0, r); } else t.push(r); return t; } /** * Disposes of the flow graph. Cancels any pending tasks and removes all event listeners. */ dispose() { if (this.state !== $x.Stopped) { this.state = $x.Stopped; for (const e of this._executionContexts) e._clearPendingBlocks(); this._executionContexts.length = 0, this._eventBlocks.length = 0, this._scene.onDisposeObservable.remove(this._sceneDisposeObserver), this._sceneDisposeObserver = null; } } /** * Executes a function in all blocks of a flow graph, starting with the event blocks. * @param visitor the function to execute. */ visitAllBlocks(e) { const t = [], r = /* @__PURE__ */ new Set(); for (const n of this._eventBlocks) t.push(n), r.add(n.uniqueId); for (; t.length > 0; ) { const n = t.pop(); e(n); for (const i of n.dataInputs) for (const s of i._connectedPoint) r.has(s._ownerBlock.uniqueId) || (t.push(s._ownerBlock), r.add(s._ownerBlock.uniqueId)); if (n instanceof og) for (const i of n.signalOutputs) for (const s of i._connectedPoint) r.has(s._ownerBlock.uniqueId) || (t.push(s._ownerBlock), r.add(s._ownerBlock.uniqueId)); } } /** * Serializes a graph * @param serializationObject the object to write the values in * @param valueSerializeFunction a function to serialize complex values */ serialize(e = {}, t) { e.variableDefinitions = {}, e.allBlocks = [], this.visitAllBlocks((r) => { const n = {}; r.serialize(n), e.allBlocks.push(n); }), e.executionContexts = []; for (const r of this._executionContexts) { const n = {}; r.serialize(n, t), e.executionContexts.push(n); } } /** * Given a list of blocks, find an output data connection that has a specific unique id * @param blocks * @param uniqueId * @returns */ static GetDataOutConnectionByUniqueId(e, t) { for (const r of e) for (const n of r.dataOutputs) if (n.uniqueId === t) return n; throw new Error("Could not find data out connection with unique id " + t); } /** * Given a list of blocks, find an input signal connection that has a specific unique id * @param blocks * @param uniqueId * @returns */ static GetSignalInConnectionByUniqueId(e, t) { for (const r of e) if (r instanceof og) { for (const n of r.signalInputs) if (n.uniqueId === t) return n; } throw new Error("Could not find signal in connection with unique id " + t); } /** * Parses a graph from a given serialization object * @param serializationObject the object where the values are written * @param coordinator the flow graph coordinator * @param valueParseFunction a function to parse complex values in a scene * @returns */ static Parse(e, t, r) { const n = t.createGraph(), i = []; for (const s of e.allBlocks) { const a = CA.Parse(s); i.push(a), a instanceof HU && n.addEventBlock(a); } for (const s of i) { for (const a of s.dataInputs) for (const f of a.connectedPointIds) { const o = JS.GetDataOutConnectionByUniqueId(i, f); a.connectTo(o); } if (s instanceof og) for (const a of s.signalOutputs) for (const f of a.connectedPointIds) { const o = JS.GetSignalInConnectionByUniqueId(i, f); a.connectTo(o); } } for (const s of e.executionContexts) lO.Parse(s, n, r); return n; } } class Nge { } class TS { constructor(e) { var t; this._config = e, this._flowGraphs = [], this._customEventsMap = /* @__PURE__ */ new Map(), this._config.scene.onDisposeObservable.add(() => { this.dispose(); }), ((t = TS.SceneCoordinators.get(this._config.scene)) !== null && t !== void 0 ? t : []).push(this); } /** * Creates a new flow graph and adds it to the list of existing flow graphs * @returns a new flow graph */ createGraph() { const e = new JS({ scene: this._config.scene, coordinator: this }); return this._flowGraphs.push(e), e; } /** * Removes a flow graph from the list of existing flow graphs and disposes it * @param graph the graph to remove */ removeGraph(e) { const t = this._flowGraphs.indexOf(e); t !== -1 && (e.dispose(), this._flowGraphs.splice(t, 1)); } /** * Starts all graphs */ start() { this._flowGraphs.forEach((e) => e.start()); } /** * Disposes all graphs */ dispose() { var e; this._flowGraphs.forEach((n) => n.dispose()), this._flowGraphs.length = 0; const t = (e = TS.SceneCoordinators.get(this._config.scene)) !== null && e !== void 0 ? e : [], r = t.indexOf(this); r !== -1 && t.splice(r, 1); } serialize(e, t) { e._flowGraphs = [], this._flowGraphs.forEach((r) => { const n = {}; r.serialize(n, t), e._flowGraphs.push(n); }); } static Parse(e, t, r) { var n; const i = new TS({ scene: t }); return (n = e._flowGraphs) === null || n === void 0 || n.forEach((s) => { JS.Parse(s, i, r); }), i; } /** * Get an observable that will be notified when the event with the given id is fired. * @param id the id of the event * @returns the observable for the event */ getCustomEventObservable(e) { let t = this._customEventsMap.get(e); return t || (t = new Oe(), this._customEventsMap.set(e, t)), t; } /** * Notifies the observable for the given event id with the given data. * @param id the id of the event * @param data the data to send with the event */ notifyCustomEvent(e, t) { const r = this._customEventsMap.get(e); r && r.notifyObservers(t); } } TS.SceneCoordinators = /* @__PURE__ */ new Map(); class Qge { constructor(e) { this._context = e, this._context.onNodeExecutedObservable.add((t) => { ye.Log(`Node executed: ${t.getClassName()}`); }); } } const Yge = /([./])({?\w+}?)/g; class Cy { constructor(e) { this._templateSubstitutions = {}, this._pathParts = [], this._templateStrings = [], this.hasTemplateStrings = !1, this._path = e; const { pathParts: t, templateStrings: r } = this._getPathPartsAndTemplateStrings(e); this._pathParts = t, this._templateStrings = r, this.hasTemplateStrings = r.length > 0; } _getPathPartsAndTemplateStrings(e) { const t = e.matchAll(Yge), r = [], n = []; let i = t.next(); for (; !i.done; ) { const s = i.value, [, a, f] = s; let o = f, d = !1; f.startsWith("{") && f.endsWith("}") && (d = !0, o = f.slice(1, f.length - 1), n.indexOf(o) === -1 && n.push(o)), r.push({ value: f, isTemplate: d, valueWithoutBraces: o, separator: a }), i = t.next(); } return { pathParts: r, templateStrings: n }; } /** * Gets the template strings in this path. * @returns an array containing the template strings in this path. */ getTemplateStrings() { return this._templateStrings; } setTemplateSubstitution(e, t) { if (this._templateStrings.indexOf(e) === -1) throw new Error(`Template string ${e} does not exist in path ${this._path}`); this._templateSubstitutions[e] = t; } _evaluateTemplates() { for (const e of this._pathParts) if (e.isTemplate) { const t = this._templateSubstitutions[e.valueWithoutBraces]; if (t === void 0) throw new Error(`Template string ${e.value} was not substituted`); e.replacedValue = t.toString(); } } _getFinalPath() { let e = ""; for (const t of this._pathParts) e += t.separator, t.isTemplate ? e += t.replacedValue : e += t.value; return e; } /* * Breaks the path into a chain of entities, for example, * /x/y/z would be split into [context._userVariables.x, context._userVariables.x.y, context._userVariables.x.y.z], * and the path that was split, i.e. /x/y/z, would be split into ["x", "y", "z"]. */ _evaluatePath(e) { this._evaluateTemplates(); const t = [], r = []; let n = e._userVariables; for (const i of this._pathParts) { if (n === void 0) throw new Error(`Could not find path ${this._getFinalPath()} in target context`); const s = i.isTemplate ? i.replacedValue : i.value; if (!s) throw new Error(`Invalid path ${this._getFinalPath()}`); n = n[s], t.push(n), r.push(s); } return { entityChain: t, splitPath: r }; } getProperty(e) { const { entityChain: t } = this._evaluatePath(e); return t[t.length - 1]; } setProperty(e, t) { const { entityChain: r, splitPath: n } = this._evaluatePath(e), i = r[r.length - 2], s = n[n.length - 1]; i[s] = t; } getClassName() { return "FGPath"; } serialize(e = {}) { return e.path = this._path, e.className = this.getClassName(), e; } Parse(e) { return new Cy(e.path); } } Ue("FGPath", Cy); class _l extends og { constructor(e) { super(e), this.onDone = this._registerSignalOutput("onDone"); } } class Fie extends _l { constructor(e) { super(e), this.message = this.registerDataInput("message", w6); } /** * @internal */ _execute(e) { const t = this.message.getValue(e); console.log(t), this.onDone._activateSignal(e); } getClassName() { return "FGLogBlock"; } } Ue("FGLogBlock", Fie); class Nie extends _l { constructor(e) { super(e), this.variableName = this.registerDataInput("variableName", pV), this.input = this.registerDataInput("input", w6); } _execute(e) { const t = this.variableName.getValue(e), r = this.input.getValue(e); e.setVariable(t, r), this.onDone._activateSignal(e); } getClassName() { return "FGSetVariableBlock"; } } Ue("FGSetVariableBlock", Nie); class VF { constructor(e, t) { this.templateStringInputs = [], this.path = e, this.ownerBlock = t; for (const r of e.getTemplateStrings()) this.templateStringInputs.push(this.ownerBlock.registerDataInput(r, Yt)); } /** * Get the inputs of all of the numeric data inputs and use them to fill in the * template strings in the path. * @param context the context to use to get the values of the numeric data inputs * @returns the path with the template strings filled in */ substitutePath(e) { for (const t of this.templateStringInputs) { const r = t.getValue(e), n = t.name; this.path.setTemplateSubstitution(n, r); } return this.path; } /** * Substitutes the template strings in the path and gets the property on the target object. * @param context * @returns */ getProperty(e) { return this.substitutePath(e), this.path.getProperty(e); } /** * Substitutes the template strings in the path and sets the property on the target object. * @param context * @param value */ setProperty(e, t) { this.substitutePath(e), this.path.setProperty(e, t); } } class Qie extends _l { constructor(e) { super(e), this.config = e, this.value = this.registerDataInput("value", w6), this.templateComponent = new VF(e.path, this); } _execute(e) { const t = this.value.getValue(e); this.templateComponent.setProperty(e, t), this.onDone._activateSignal(e); } serialize(e = {}) { super.serialize(e), e.config.path = this.config.path.serialize(); } getClassName() { return "FGSetPropertyBlock"; } } Ue("FGSetPropertyBlock", Qie); class Yie extends _l { constructor(e) { super(e), this.eventId = this.registerDataInput("eventId", pV), this.eventData = this.registerDataInput("eventData", w6); } _execute(e) { const t = this.eventId.getValue(e), r = this.eventData.getValue(e); e.configuration.coordinator.notifyCustomEvent(t, r), this.onDone._activateSignal(e); } getClassName() { return "FGSendCustomEventBlock"; } } Ue("FGSendCustomEventBlock", Yie); class Mie extends og { constructor(e) { super(e), this.condition = this.registerDataInput("condition", m6), this.onTrue = this._registerSignalOutput("onTrue"), this.onFalse = this._registerSignalOutput("onFalse"); } _execute(e) { this.condition.getValue(e) ? this.onTrue._activateSignal(e) : this.onFalse._activateSignal(e); } getClassName() { return "FGBranchBlock"; } } Ue("FGBranchBlock", Mie); class Lie extends _l { constructor(e) { super(e), this.reset = this._registerSignalInput("reset"), this.maxNumberOfExecutions = this.registerDataInput("numberOfExecutions", Yt), this.currentCount = this.registerDataOutput("currentCount", Yt); } _execute(e, t) { if (t === this.reset) this.currentCount.setValue(0, e); else { const r = this.currentCount.getValue(e); r < this.maxNumberOfExecutions.getValue(e) && (this.currentCount.setValue(r + 1, e), this.onDone._activateSignal(e)); } } getClassName() { return "FGDoNBlock"; } } Ue("FGDoNBlock", Lie); class Kie extends _l { constructor(e) { super(e), this.startIndex = this.registerDataInput("startIndex", Yt), this.endIndex = this.registerDataInput("endIndex", Yt), this.step = this.registerDataInput("step", Yt), this.index = this.registerDataOutput("index", Yt), this.onLoop = this._registerSignalOutput("onLoop"), this.onDone = this._registerSignalOutput("onDone"); } _executeLoop(e) { let t = e._getExecutionVariable(this, "index"); const r = e._getExecutionVariable(this, "endIndex"); if (t < r) { this.index.setValue(t, e), this.onLoop._activateSignal(e); const n = e._getExecutionVariable(this, "step", 1); t += n, e._setExecutionVariable(this, "index", t), this._executeLoop(e); } else this.onDone._activateSignal(e); } /** * @internal */ _execute(e) { const t = this.startIndex.getValue(e), r = this.endIndex.getValue(e), n = this.step.getValue(e); e._setExecutionVariable(this, "index", t), e._setExecutionVariable(this, "endIndex", r), e._setExecutionVariable(this, "step", n), this._executeLoop(e); } getClassName() { return "FGForLoopBlock"; } } Ue("FGForLoopBlock", Kie); class Jie extends _l { constructor(e) { super(e), this.reset = this._registerSignalInput("reset"), this.duration = this.registerDataInput("duration", Yt), this.timeRemaining = this.registerDataOutput("timeRemaining", Yt); } _execute(e, t) { const r = e._getExecutionVariable(this, "lastExecutedTime"), n = this.duration.getValue(e), i = Date.now(); if (t === this.reset || r === void 0 || i - r > n) this.timeRemaining.setValue(0, e), this.onDone._activateSignal(e), e._setExecutionVariable(this, "lastExecutedTime", i); else { const s = n - (i - r); this.timeRemaining.setValue(s, e); } } getClassName() { return "FGThrottleBlock"; } } Ue("FGThrottleBlock", Jie); class zie extends $Y { constructor(e) { super(e), this.timeout = this.registerDataInput("timeout", Yt), this.onTimerDone = this._registerSignalOutput("onTimerDone"); } _preparePendingTasks(e) { const t = this.timeout.getValue(e); if (t !== void 0 && t >= 0) { const r = e._getExecutionVariable(this, "runningTimers") || [], n = e.configuration.scene, i = new kte({ timeout: t, contextObservable: n.onBeforeRenderObservable, onEnded: () => this._onEnded(i, e) }); i.start(), r.push(i), e._setExecutionVariable(this, "runningTimers", r); } } /** * @internal */ _execute(e) { this._startPendingTasks(e), this.onDone._activateSignal(e); } _onEnded(e, t) { const r = t._getExecutionVariable(this, "runningTimers") || [], n = r.indexOf(e); n !== -1 ? r.splice(n, 1) : ye.Warn("FlowGraphTimerBlock: Timer ended but was not found in the running timers list"), t._removePendingBlock(this), this.onTimerDone._activateSignal(t); } _cancelPendingTasks(e) { const t = e._getExecutionVariable(this, "runningTimers") || []; for (const r of t) r.dispose(); e._deleteExecutionVariable(this, "runningTimers"); } getClassName() { return "FGTimerBlock"; } } Ue("FGTimerBlock", zie); class Gie extends og { constructor(e) { super(e), this.config = e, this._cachedUnusedIndexes = [], this.reset = this._registerSignalInput("reset"), this.currentIndex = this.registerDataOutput("currentIndex", Yt); } configure() { super.configure(), this.config.startIndex = this.config.startIndex !== void 0 ? this.config.startIndex : 0, this.config.startIndex = Math.max(0, Math.min(this.config.startIndex, this.config.numberOutputFlows - 1)), this.outFlows = []; for (let e = 0; e < this.config.numberOutputFlows; e++) this.outFlows.push(this._registerSignalOutput(`out${e}`)); } _getUnusedIndexes(e) { const t = this._cachedUnusedIndexes; if (t.length = 0, e._hasExecutionVariable(this, "unusedIndexes")) { const r = e._getExecutionVariable(this, "unusedIndexes"); for (let n = 0; n < r.length; n++) t.push(r[n]); } else for (let r = 0; r < this.config.numberOutputFlows; r++) t.push(r); return t; } _getNextOutput(e, t) { if (this.config.isRandom) { const r = Math.floor(Math.random() * t.length); return t[r]; } else return e + 1; } _execute(e, t) { var r; const n = (r = e._getExecutionVariable(this, "currentIndex")) !== null && r !== void 0 ? r : this.config.startIndex - 1; let i = this._getUnusedIndexes(e); if (t === this.reset) { e._deleteExecutionVariable(this, "currentIndex"), e._deleteExecutionVariable(this, "unusedIndexes"); return; } let s = this._getNextOutput(n, i); if (s >= this.config.numberOutputFlows && this.config.loop) s = 0; else if (s >= this.config.numberOutputFlows && !this.config.loop) return; if (i = i.filter((a) => a !== s), i.length === 0) for (let a = 0; a < this.config.numberOutputFlows; a++) i.push(a); e._setExecutionVariable(this, "unusedIndexes", i), e._setExecutionVariable(this, "currentIndex", s), this.currentIndex.setValue(s, e), this.outFlows[s]._activateSignal(e); } getClassName() { return "FGMultiGateBlock"; } serialize(e) { super.serialize(e), e.config.numberOutputFlows = this.config.numberOutputFlows, e.config.isRandom = this.config.isRandom, e.config.loop = this.config.loop, e.config.startIndex = this.config.startIndex; } } Ue("FGMultiGateBlock", Gie); class Zie extends og { constructor(e) { super(e), this.config = e, this.selection = this.registerDataInput("selection", w6); } configure() { super.configure(), this.outputFlows = []; for (let e = 0; e <= this.config.cases.length; e++) this.outputFlows.push(this._registerSignalOutput(`out${e}`)); } _execute(e, t) { const r = this.selection.getValue(e); for (let n = 0; n < this.config.cases.length; n++) if (r === this.config.cases[n]) { this.outputFlows[n]._activateSignal(e); return; } this.outputFlows[this.outputFlows.length - 1]._activateSignal(e); } getClassName() { return "FGSwitchBlock"; } serialize(e) { super.serialize(e), e.cases = this.config.cases; } } Ue("FGSwitchBlock", Zie); class _ie extends _l { constructor(e) { super(e), this.config = e, this.inFlows = [], this._cachedActivationState = [], this.reset = this._registerSignalInput("reset"); } configure() { for (let e = 1; e < this.config.numberInputFlows; e++) this.inFlows.push(this._registerSignalInput(`in${e}`)); } _getCurrentActivationState(e) { const t = this._cachedActivationState; if (t.length = 0, e._hasExecutionVariable(this, "activationState")) { const r = e._getExecutionVariable(this, "activationState"); for (let n = 0; n < r.length; n++) t.push(r[n]); } else for (let r = 0; r < this.config.numberInputFlows; r++) t.push(!1); return t; } _execute(e, t) { const r = this._getCurrentActivationState(e); if (t === this.reset) for (let n = 0; n < this.config.numberInputFlows; n++) r[n] = !1; else if (t === this.onStart) r[0] = !0; else { const n = this.inFlows.indexOf(t); n >= 0 && (r[n + 1] = !0); } if (e._setExecutionVariable(this, "activationState", r.slice()), r.every((n) => n)) { this.onDone._activateSignal(e); for (let n = 0; n < this.config.numberInputFlows; n++) r[n] = !1; } } getClassName() { return "FGWaitAllBlock"; } serialize(e) { super.serialize(e), e.config.numberInputFlows = this.config.numberInputFlows; } } Ue("FGWaitAllBlock", _ie); class $ie extends _l { constructor(e) { super(e), this.count = this.registerDataOutput("count", Yt), this.reset = this._registerSignalInput("reset"); } _execute(e, t) { var r; if (t === this.reset) { e._setExecutionVariable(this, "count", 0), this.count.setValue(0, e); return; } const n = ((r = e._getExecutionVariable(this, "count")) !== null && r !== void 0 ? r : 0) + 1; e._setExecutionVariable(this, "count", n), this.count.setValue(n, e), this.onDone._activateSignal(e); } getClassName() { return "FGCounterBlock"; } } Ue("FGCounterBlock", $ie); class ese extends _l { constructor(e) { super(e), this.config = e, this.condition = this.registerDataInput("condition", m6), this.loopBody = this._registerSignalOutput("loopBody"); } _execute(e, t) { var r; let n = this.condition.getValue(e); for (!((r = this.config) === null || r === void 0) && r.isDo && !n && this.loopBody._activateSignal(e); n; ) this.loopBody._activateSignal(e), n = this.condition.getValue(e); this.onDone._activateSignal(e); } getClassName() { return "FGWhileLoopBlock"; } serialize(e) { var t; super.serialize(e), e.isDo = (t = this.config) === null || t === void 0 ? void 0 : t.isDo; } } Ue("FGWhileLoopBlock", ese); class tse extends _l { constructor(e) { super(e), this.count = this.registerDataInput("count", Yt), this.reset = this._registerSignalInput("reset"), this.currentCount = this.registerDataOutput("currentCount", Yt); } _execute(e, t) { if (t === this.reset) { e._setExecutionVariable(this, "debounceCount", 0); return; } const r = this.count.getValue(e), i = e._getExecutionVariable(this, "debounceCount", 0) + 1; this.currentCount.setValue(i, e), e._setExecutionVariable(this, "debounceCount", i), i >= r && (this.onDone._activateSignal(e), e._setExecutionVariable(this, "debounceCount", 0)); } getClassName() { return "FGDebounceBlock"; } } Ue("FGDebounceBlock", tse); class rse extends og { constructor(e) { super(e), this.onOn = this._registerSignalOutput("onOn"), this.onOff = this._registerSignalOutput("onOff"), this.isOn = this.registerDataOutput("isOn", m6); } _execute(e, t) { let r = e._getExecutionVariable(this, "value", !1); r = !r, e._setExecutionVariable(this, "value", r), this.isOn.setValue(r, e), r ? this.onOn._activateSignal(e) : this.onOff._activateSignal(e); } getClassName() { return "FGFlipFlopBlock"; } } Ue("FGFlipFlopBlock", rse); class nse extends $Y { constructor(e) { super(e), this.config = e, this.templateTargetComponent = new VF(e.targetPath, this), this.templateAnimationComponent = new VF(e.animationPath, this), this.speed = this.registerDataInput("speed", Yt), this.loop = this.registerDataInput("loop", m6), this.from = this.registerDataInput("from", Yt), this.to = this.registerDataInput("to", Yt), this.onAnimationEnd = this._registerSignalOutput("onAnimationEnd"), this.runningAnimatable = this.registerDataOutput("runningAnimatable", w6); } /** * @internal * @param context */ _preparePendingTasks(e) { var t; const r = this.templateTargetComponent.getProperty(e), n = this.templateAnimationComponent.getProperty(e); if (!r || !n) throw new Error("Cannot play animation without target or animation"); const i = (t = e._getExecutionVariable(this, "runningAnimatables")) !== null && t !== void 0 ? t : [], s = this.runningAnimatable.getValue(e); if (s && s.paused) s.restart(); else { const f = e.configuration.scene.beginDirectAnimation(r, [n], this.from.getValue(e), this.to.getValue(e), this.loop.getValue(e), this.speed.getValue(e), () => this._onAnimationEnd(f, e)); this.runningAnimatable.setValue(f, e), i.push(f); } e._setExecutionVariable(this, "runningAnimatables", i); } _execute(e) { this._startPendingTasks(e), this.onDone._activateSignal(e); } _onAnimationEnd(e, t) { var r; const n = (r = t._getExecutionVariable(this, "runningAnimatables")) !== null && r !== void 0 ? r : [], i = n.indexOf(e); i !== -1 && n.splice(i, 1), t._removePendingBlock(this), this.onAnimationEnd._activateSignal(t); } /** * @internal * Stop any currently running animations. */ _cancelPendingTasks(e) { var t; const r = (t = e._getExecutionVariable(this, "runningAnimatables")) !== null && t !== void 0 ? t : []; for (const n of r) n.stop(); e._deleteExecutionVariable(this, "runningAnimatables"); } getClassName() { return "FGPlayAnimationBlock"; } serialize(e = {}) { super.serialize(e), e.config.targetPath = this.config.targetPath.serialize(), e.config.animationPath = this.config.animationPath.serialize(); } } Ue("FGPlayAnimationBlock", nse); class ise extends _l { constructor(e) { super(e), this.animationToStop = this.registerDataInput("animationToStop", w6); } _execute(e) { this.animationToStop.getValue(e).stop(), this.onDone._activateSignal(e); } getClassName() { return "FGStopAnimationBlock"; } } Ue("FGStopAnimationBlock", ise); class sse extends _l { constructor(e) { super(e), this.animationToPause = this.registerDataInput("animationToPause", w6); } _execute(e) { this.animationToPause.getValue(e).pause(), this.onDone._activateSignal(e); } getClassName() { return "FGPauseAnimationBlock"; } } Ue("FGPauseAnimationBlock", sse); class ase extends _l { constructor(e) { super(e), this.audio = this.registerDataInput("audio", w6); } _execute(e, t) { const r = this.audio.getValue(e); r instanceof qc && r.play(), this.onDone._activateSignal(e); } getClassName() { return "FGPlayAudioBlock"; } } Ue("FGPlayAudioBlock", ase); class ose extends _l { constructor(e) { super(e), this.audio = this.registerDataInput("audio", w6); } _execute(e, t) { const r = this.audio.getValue(e); r instanceof qc && r.stop(); } getClassName() { return "FGStopAudioBlock"; } } Ue("FGStopAudioBlock", ose); class fse extends CA { constructor(e) { super(e), this.condition = this.registerDataInput("condition", m6), this.trueValue = this.registerDataInput("trueValue", w6), this.falseValue = this.registerDataInput("falseValue", w6), this.output = this.registerDataOutput("output", w6); } /** * @internal */ _updateOutputs(e) { this.output.setValue(this.condition.getValue(e) ? this.trueValue.getValue(e) : this.falseValue.getValue(e), e); } getClassName() { return "FGConditionalDataBlock"; } } Ue("FGConditionalDataBlock", fse); class Ase extends CA { /** * Construct a FlowGraphGetVariableBlock. * @param params optional construction parameters */ constructor(e) { super(e), this.variableName = this.registerDataInput("variableName", pV), this.output = this.registerDataOutput("output", w6); } /** * @internal */ _updateOutputs(e) { const t = this.variableName.getValue(e); e.hasVariable(t) && this.output.setValue(e.getVariable(t), e); } getClassName() { return "FGGetVariableBlock"; } } Ue("FGGetVariableBlock", Ase); class dse extends CA { /** * Creates a new FlowGraphCoordinateTransformBlock */ constructor(e) { super(e), this.sourceSystem = this.registerDataInput("sourceSystem", w6), this.destinationSystem = this.registerDataInput("destinationSystem", w6), this.inputCoordinates = this.registerDataInput("inputCoordinates", ma), this.outputCoordinates = this.registerDataOutput("outputCoordinates", ma); } _updateOutputs(e) { const t = this.sourceSystem.getValue(e), r = this.destinationSystem.getValue(e), n = this.inputCoordinates.getValue(e), i = t.getWorldMatrix(), s = r.getWorldMatrix(), a = ue.Matrix[0].copyFrom(s); a.invert(); const f = ue.Matrix[1]; a.multiplyToRef(i, f); const o = this.outputCoordinates.getValue(e); S.TransformCoordinatesToRef(n, f, o); } getClassName() { return "FGCoordinateTransformBlock"; } } Ue("FGCoordinateTransformBlock", dse); class vse extends CA { constructor(e) { super(e), this.config = e, this.output = this.registerDataOutput("output", yie(e.value)); } _updateOutputs(e) { this.output.setValue(this.config.value, e); } getClassName() { return "FGConstantBlock"; } } Ue("FGConstantBlock", vse); const rZ = "cachedOperationValue", nZ = "cachedExecutionId"; class tM extends CA { constructor(e, t) { super(t), this.output = this.registerDataOutput("output", e); } _updateOutputs(e) { const t = e._getExecutionVariable(this, nZ), r = e._getExecutionVariable(this, rZ); if (r !== void 0 && t === e.executionId) this.output.setValue(r, e); else { const n = this._doOperation(e); e._setExecutionVariable(this, rZ, n), e._setExecutionVariable(this, nZ, e.executionId), this.output.setValue(n, e); } } } class ns extends tM { constructor(e, t, r, n, i, s) { super(r, s), this._operation = n, this._className = i, this.leftInput = this.registerDataInput("leftInput", e), this.rightInput = this.registerDataInput("rightInput", t); } _doOperation(e) { return this._operation(this.leftInput.getValue(e), this.rightInput.getValue(e)); } getClassName() { return this._className; } } class p1 extends tM { constructor(e, t, r, n, i) { super(t, i), this._operation = r, this._className = n, this.input = this.registerDataInput("input", e); } _doOperation(e) { return this._operation(this.input.getValue(e)); } getClassName() { return this._className; } } const Fu = "FGBitwise", use = "AndBlock", lse = "OrBlock", Pse = "XorBlock", cse = "NotBlock", pse = "LeftShiftBlock", hse = "RightShiftBlock", Hse = "CountLeadingZerosBlock", gse = "CountTrailingZerosBlock"; class Xse extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => t & r, `${Fu}${use}`, e); } } Ue(`${Fu}${use}`, Xse); class Tse extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => t | r, `${Fu}${lse}`, e); } } Ue(`${Fu}${lse}`, Tse); class qse extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => t ^ r, `${Fu}${Pse}`, e); } } Ue(`${Fu}${Pse}`, qse); class bse extends p1 { constructor(e) { super(Yt, Yt, (t) => ~t, `${Fu}${cse}`, e); } } Ue(`${Fu}${cse}`, bse); class xse extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => t << r, `${Fu}${pse}`, e); } } Ue(`${Fu}${pse}`, xse); class Dse extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => t >> r, `${Fu}${hse}`, e); } } Ue(`${Fu}${hse}`, Dse); class jse extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.clz32(t), `${Fu}${Hse}`, e); } } Ue(`${Fu}${Hse}`, jse); class wse extends p1 { // from: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Math/clz32#implementing_count_leading_ones_and_beyond _ctrz(e) { return e >>>= 0, e === 0 ? 32 : (e &= -e, 31 - Math.clz32(e)); } constructor(e) { super(Yt, Yt, (t) => this._ctrz(t), `${Fu}${gse}`, e); } } Ue(`${Fu}${gse}`, wse); const gU = "FGLogic", mse = "AndBlock", Bse = "OrBlock", Wse = "NotBlock"; class Sse extends ns { constructor(e) { super(m6, m6, m6, (t, r) => t && r, `${gU}${mse}`, e); } } Ue(`${gU}${mse}`, Sse); class Use extends ns { constructor(e) { super(m6, m6, m6, (t, r) => t || r, `${gU}${Bse}`, e); } } Ue(`${gU}${Bse}`, Use); class Ise extends p1 { constructor(e) { super(m6, m6, (t) => !t, `${gU}${Wse}`, e); } } Ue(`${gU}${Wse}`, Ise); class Rse extends tM { constructor(e, t, r, n) { super(e, n), this._operation = t, this._className = r; } _doOperation(e) { return this._operation(); } getClassName() { return this._className; } } const Vse = "FGAddNumberBlock"; class Cse extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => t + r, Vse, e); } } Ue(Vse, Cse); const Ose = "FGSubtractNumberBlock"; class yse extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => t - r, Ose, e); } } Ue(Ose, yse); const kse = "FGMultiplyNumberBlock"; class Ese extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => t * r, kse, e); } } Ue(kse, Ese); const Fse = "FGDivideNumberBlock"; class Nse extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => t / r, Fse, e); } } Ue(Fse, Nse); const Qse = "FGModNumberBlock"; class Yse extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => t % r, Qse, e); } } Ue(Qse, Yse); const Mse = "FGPowNumberBlock"; class Lse extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => Math.pow(t, r), Mse, e); } } Ue(Mse, Lse); const Kse = "FGIsNaNNumberBlock"; class Jse extends p1 { constructor(e) { super(Yt, m6, (t) => isNaN(t), Kse, e); } } Ue(Kse, Jse); const zse = "FGIsInfinityNumberBlock"; class Gse extends p1 { constructor(e) { super(Yt, m6, (t) => !isFinite(t), zse, e); } } Ue(zse, Gse); const Zse = "FGSqrtNumberBlock"; class _se extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.sqrt(t), Zse, e); } } Ue(Zse, _se); const $se = "FGAbsNumberBlock"; class eae extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.abs(t), $se, e); } } Ue($se, eae); const tae = "FGNegateNumberBlock"; class rae extends p1 { constructor(e) { super(Yt, Yt, (t) => -t, tae, e); } } Ue(tae, rae); const nae = "FGFloorNumberBlock"; class iae extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.floor(t), nae, e); } } Ue(nae, iae); const sae = "FGCeilNumberBlock"; class aae extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.ceil(t), sae, e); } } Ue(sae, aae); const oae = "FGRoundNumberBlock"; class fae extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => Math.round(t / Math.pow(10, r)) / Math.pow(10, r), oae, e); } } Ue(oae, fae); const Aae = "FGTruncNumberBlock"; class dae extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.trunc(t), Aae, e); } } Ue(Aae, dae); const vae = "FGExpNumberBlock"; class uae extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.exp(t), vae, e); } } Ue(vae, uae); const lae = "FGLog10NumberBlock"; class Pae extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.log10(t), lae, e); } } Ue(lae, Pae); const cae = "FGLogNumberBlock"; class pae extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.log(t), cae, e); } } Ue(cae, pae); const hae = "FGLnNumberBlock"; class Hae extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.log(t) / Math.LN2, hae, e); } } Ue(hae, Hae); const gae = "FGSineNumberBlock"; class Xae extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.sin(t), gae, e); } } Ue(gae, Xae); const Tae = "FGCosNumberBlock"; class qae extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.cos(t), Tae, e); } } Ue(Tae, qae); const bae = "FGTanNumberBlock"; class xae extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.tan(t), bae, e); } } Ue(bae, xae); const Dae = "FGASineNumberBlock"; class jae extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.asin(t), Dae, e); } } Ue(Dae, jae); const wae = "FGACosNumberBlock"; class mae extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.acos(t), wae, e); } } Ue(wae, mae); const Bae = "FGATanNumberBlock"; class Wae extends p1 { constructor(e) { super(Yt, Yt, (t) => Math.atan(t), Bae, e); } } Ue(Bae, Wae); const Sae = "FGENumberBlock"; class Uae extends Rse { constructor(e) { super(Yt, () => Math.E, Sae, e); } } Ue(Sae, Uae); const Iae = "FGPiNumberBlock"; class Rae extends Rse { constructor(e) { super(Yt, () => Math.PI, Iae, e); } } Ue(Iae, Rae); const Vae = "FGATan2NumberBlock"; class Cae extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => Math.atan2(t, r), Vae, e); } } Ue(Vae, Cae); const Oae = "FGRandomNumberBlock"; class yae extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => t + Math.random() * (r - t), Oae, e); } } Ue(Oae, yae); const kae = "FGMinNumberBlock"; class Eae extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => Math.min(t, r), kae, e); } } Ue(kae, Eae); const Fae = "FGMaxNumberBlock"; class Nae extends ns { constructor(e) { super(Yt, Yt, Yt, (t, r) => Math.max(t, r), Fae, e); } } Ue(Fae, Nae); const Qae = "FGEqualsNumberBlock"; class Yae extends ns { constructor(e) { super(Yt, Yt, m6, (t, r) => t === r, Qae, e); } } Ue(Qae, Yae); const Mae = "FGGreaterThanNumberBlock"; class Lae extends ns { constructor(e) { super(Yt, Yt, m6, (t, r) => t > r, Mae, e); } } Ue(Mae, Lae); const Kae = "FGGreaterThanOrEqualsNumberBlock"; class Jae extends ns { constructor(e) { super(Yt, Yt, m6, (t, r) => t >= r, Kae, e); } } Ue(Kae, Jae); const zae = "FGLessThanNumberBlock"; class Gae extends ns { constructor(e) { super(Yt, Yt, m6, (t, r) => t < r, zae, e); } } Ue(zae, Gae); const Zae = "FGLessThanOrEqualsNumberBlock"; class _ae extends ns { constructor(e) { super(Yt, Yt, m6, (t, r) => t <= r, Zae, e); } } Ue(Zae, _ae); const $ae = "FGMixNumberBlock"; class eoe extends CA { constructor(e) { super(e), this.leftInput = this.registerDataInput("leftInput", Yt), this.rightInput = this.registerDataInput("rightInput", Yt), this.alphaInput = this.registerDataInput("alphaInput", Yt), this.resultOutput = this.registerDataOutput("resultOutput", Yt); } _updateOutputs(e) { const t = this.leftInput.getValue(e), r = this.rightInput.getValue(e), n = this.alphaInput.getValue(e), i = t + (r - t) * n; this.resultOutput.setValue(i, e); } getClassName() { return $ae; } } Ue($ae, eoe); const toe = "FGAddVector2Block"; class roe extends ns { constructor(e) { super(Y9, Y9, Y9, (t, r) => t.add(r), toe, e); } } Ue(toe, roe); const noe = "FGSubtractVector2Block"; class ioe extends ns { constructor(e) { super(Y9, Y9, Y9, (t, r) => t.subtract(r), noe, e); } } Ue(noe, ioe); const soe = "FGMultiplyVector2Block"; class aoe extends ns { constructor(e) { super(Y9, Y9, Y9, (t, r) => t.multiply(r), soe, e); } } Ue(soe, aoe); const ooe = "FGDivideVector2Block"; class foe extends ns { constructor(e) { super(Y9, Y9, Y9, (t, r) => t.divide(r), ooe, e); } } Ue(ooe, foe); const Aoe = "FGScaleVector2Block"; class doe extends ns { constructor(e) { super(Y9, Yt, Y9, (t, r) => t.scale(r), Aoe, e); } } Ue(Aoe, doe); const voe = "FGLengthVector2Block"; class uoe extends p1 { constructor(e) { super(Y9, Yt, (t) => t.length(), voe, e); } } Ue(voe, uoe); const loe = "FGNormalizeVector2Block"; class Poe extends p1 { constructor(e) { super(Y9, Y9, (t) => { const r = t.clone(); return r.normalize(), r; }, loe, e); } } Ue(loe, Poe); const Mge = "FGCreateVector2Block"; class coe extends CA { constructor(e) { super(e), this._cachedVector = at.Zero(), this.x = this.registerDataInput("x", Yt), this.y = this.registerDataInput("y", Yt), this.vector = this.registerDataOutput("vector", Y9); } _updateOutputs(e) { this._cachedVector.x = this.x.getValue(e), this._cachedVector.y = this.y.getValue(e), this.vector.setValue(this._cachedVector, e); } } Ue(Mge, coe); const Lge = "FGSplitVector2Block"; class poe extends CA { constructor(e) { super(e), this.vector = this.registerDataInput("vector", Y9), this.x = this.registerDataOutput("x", Yt), this.y = this.registerDataOutput("y", Yt); } _updateOutputs(e) { const t = this.vector.getValue(e); this.x.setValue(t.x, e), this.y.setValue(t.y, e); } } Ue(Lge, poe); const Kge = "FGRotate2dVector2Block"; class hoe extends CA { constructor(e) { super(e), this._cachedVector = at.Zero(), this.input = this.registerDataInput("input", Y9), this.angle = this.registerDataInput("angle", Yt), this.output = this.registerDataOutput("output", Y9); } _updateOutputs(e) { const t = this.input.getValue(e), r = this.angle.getValue(e); this._cachedVector.x = t.x * Math.cos(r) - t.y * Math.sin(r), this._cachedVector.y = t.x * Math.sin(r) + t.y * Math.cos(r), this.output.setValue(this._cachedVector, e); } } Ue(Kge, hoe); const Hoe = "FGAddVector3Block"; class goe extends ns { constructor(e) { super(ma, ma, ma, (t, r) => t.add(r), Hoe, e); } } Ue(Hoe, goe); const Xoe = "FGSubtractVector3Block"; class Toe extends ns { constructor(e) { super(ma, ma, ma, (t, r) => t.subtract(r), Xoe, e); } } Ue(Xoe, Toe); const qoe = "FGMultiplyVector3Block"; class boe extends ns { constructor(e) { super(ma, ma, ma, (t, r) => t.multiply(r), qoe, e); } } Ue(qoe, boe); const xoe = "FGDivideVector3Block"; class Doe extends ns { constructor(e) { super(ma, ma, ma, (t, r) => t.divide(r), xoe, e); } } Ue(xoe, Doe); const joe = "FGScaleVector3Block"; class woe extends ns { constructor(e) { super(ma, Yt, ma, (t, r) => t.scale(r), joe, e); } } Ue(joe, woe); const moe = "FGLengthVector3Block"; class Boe extends p1 { constructor(e) { super(ma, Yt, (t) => t.length(), moe, e); } } Ue(moe, Boe); const Woe = "FGNormalizeVector3Block"; class Soe extends p1 { constructor(e) { super(ma, ma, (t) => t.normalizeToNew(), Woe, e); } } Ue(Woe, Soe); const Uoe = "FGDotVector3Block"; class Ioe extends ns { constructor(e) { super(ma, ma, Yt, (t, r) => S.Dot(t, r), Uoe, e); } } Ue(Uoe, Ioe); const Roe = "FGCrossVector3Block"; class Voe extends ns { constructor(e) { super(ma, ma, ma, (t, r) => S.Cross(t, r), Roe, e); } } Ue(Roe, Voe); const Coe = "FGCreateVector3Block"; class Ooe extends CA { constructor(e) { super(e), this._cachedVector = S.Zero(), this.x = this.registerDataInput("x", Yt), this.y = this.registerDataInput("y", Yt), this.z = this.registerDataInput("y", Yt), this.vector = this.registerDataOutput("vector", ma); } _updateOutputs(e) { this._cachedVector.x = this.x.getValue(e), this._cachedVector.y = this.y.getValue(e), this._cachedVector.z = this.z.getValue(e), this.vector.setValue(this._cachedVector, e); } getClassName() { return Coe; } } Ue(Coe, Ooe); const yoe = "FGSplitVector3Block"; class koe extends CA { constructor(e) { super(e), this.vector = this.registerDataInput("vector", ma), this.x = this.registerDataOutput("x", Yt), this.y = this.registerDataOutput("y", Yt), this.z = this.registerDataOutput("z", Yt); } _updateOutputs(e) { const t = this.vector.getValue(e); this.x.setValue(t.x, e), this.y.setValue(t.y, e), this.z.setValue(t.z, e); } getClassName() { return yoe; } } Ue(yoe, koe); const Eoe = "FGRotateVector3Block"; class Foe extends CA { constructor(e) { super(e), this._cachedQuaternion = new Ze(), this.input = this.registerDataInput("input", ma), this.angle = this.registerDataInput("angle", Yt), this.output = this.registerDataOutput("output", ma); } _updateOutputs(e) { const t = Ze.RotationAxisToRef(this.axis.getValue(e), this.angle.getValue(e), this._cachedQuaternion), r = this.input.getValue(e), n = this.output.getValue(e); r.applyRotationQuaternionToRef(t, n); } getClassName() { return Eoe; } } Ue(Eoe, Foe); const Noe = "FGTransformVector3Block"; class Qoe extends ns { constructor(e) { super(D1, ma, ma, (t, r) => S.TransformCoordinatesToRef(r, t, this._cachedResult), Noe, e), this._cachedResult = S.Zero(); } } Ue(Noe, Qoe); const Yoe = "FGAddVector4Block"; class Moe extends ns { constructor(e) { super(iA, iA, iA, (t, r) => t.add(r), Yoe, e); } } Ue(Yoe, Moe); const Loe = "FGSubtractVector4Block"; class Koe extends ns { constructor(e) { super(iA, iA, iA, (t, r) => t.subtract(r), Loe, e); } } Ue(Loe, Koe); const Joe = "FGMultiplyVector4Block"; class zoe extends ns { constructor(e) { super(iA, iA, iA, (t, r) => t.multiply(r), Joe, e); } } Ue(Joe, zoe); const Goe = "FGDivideVector4Block"; class Zoe extends ns { constructor(e) { super(iA, iA, iA, (t, r) => t.divide(r), Goe, e); } } Ue(Goe, Zoe); const _oe = "FGScaleVector4Block"; class $oe extends ns { constructor(e) { super(iA, Yt, iA, (t, r) => t.scale(r), _oe, e); } } Ue(_oe, $oe); const e1e = "FGLengthVector4Block"; class t1e extends p1 { constructor(e) { super(iA, Yt, (t) => t.length(), e1e, e); } } Ue(e1e, t1e); const CF = "FGNormalizeVector4Block"; class r1e extends p1 { constructor(e) { super(iA, iA, (t) => { const r = t.clone(); return r.normalize(), r; }, CF, e); } getClassName() { return CF; } } Ue(CF, r1e); const n1e = "FGCreateVector4Block"; class i1e extends CA { constructor(e) { super(e), this._cachedVector = Ir.Zero(), this.x = this.registerDataInput("x", Yt), this.y = this.registerDataInput("y", Yt), this.z = this.registerDataInput("y", Yt), this.w = this.registerDataInput("w", Yt), this.vector = this.registerDataOutput("vector", iA); } _updateOutputs(e) { this._cachedVector.x = this.x.getValue(e), this._cachedVector.y = this.y.getValue(e), this._cachedVector.z = this.z.getValue(e), this._cachedVector.w = this.w.getValue(e), this.vector.setValue(this._cachedVector, e); } getClassName() { return n1e; } } Ue(n1e, i1e); const s1e = "FGSplitVector4Block"; class a1e extends CA { constructor(e) { super(e), this.vector = this.registerDataInput("vector", iA), this.x = this.registerDataOutput("x", Yt), this.y = this.registerDataOutput("y", Yt), this.z = this.registerDataOutput("z", Yt), this.w = this.registerDataOutput("w", Yt); } _updateOutputs(e) { const t = this.vector.getValue(e); this.x.setValue(t.x, e), this.y.setValue(t.y, e), this.z.setValue(t.z, e), this.w.setValue(t.w, e); } getClassName() { return s1e; } } Ue(s1e, a1e); const o1e = "FGAddMatrixBlock"; class f1e extends ns { constructor(e) { super(D1, D1, D1, (t, r) => t.addToRef(r, this._cachedMatrix), o1e, e), this._cachedMatrix = he.Zero(); } } Ue(o1e, f1e); const A1e = "FGAddMatrixAndNumberBlock"; class d1e extends ns { constructor(e) { super(D1, Yt, D1, (t, r) => { for (let n = 0; n < t.m.length; n++) this._cachedArray[n] = t.m[n] + r; return he.FromArrayToRef(this._cachedArray, 0, this._cachedMatrix); }, A1e, e), this._cachedArray = new Float32Array(16), this._cachedMatrix = he.Zero(); } } Ue(A1e, d1e); const v1e = "FGSubtractMatrixBlock"; class u1e extends ns { constructor(e) { super(D1, D1, D1, (t, r) => t.addToRef(r.scaleToRef(-1, ue.Matrix[0]), this._cachedMatrix), v1e, e), this._cachedMatrix = he.Zero(); } } Ue(v1e, u1e); const l1e = "FGSubtractMatrixAndNumberBlock"; class P1e extends ns { constructor(e) { super(D1, Yt, D1, (t, r) => { for (let n = 0; n < t.m.length; n++) this._cachedArray[n] = t.m[n] - r; return he.FromArrayToRef(this._cachedArray, 0, this._cachedMatrix); }, l1e, e), this._cachedArray = new Float32Array(16), this._cachedMatrix = he.Zero(); } } Ue(l1e, P1e); const c1e = "FGMultiplyMatrixBlock"; class p1e extends ns { constructor(e) { super(D1, D1, D1, (t, r) => t.multiplyToRef(r, this._cachedMatrix), c1e, e), this._cachedMatrix = he.Zero(); } } Ue(c1e, p1e); const h1e = "FGDivideMatrixBlock"; class H1e extends ns { constructor(e) { super(D1, D1, D1, (t, r) => t.multiplyToRef(r.invertToRef(ue.Matrix[0]), this._cachedResultMatrix), h1e, e), this._cachedResultMatrix = he.Zero(); } } Ue(h1e, H1e); const g1e = "FGDivideMatrixAndNumberBlock"; class X1e extends ns { constructor(e) { super(D1, Yt, D1, (t, r) => { for (let n = 0; n < t.m.length; n++) this._cachedArray[n] = t.m[n] / r; return he.FromArrayToRef(this._cachedArray, 0, this._cachedMatrix); }, g1e, e), this._cachedArray = new Float32Array(16), this._cachedMatrix = he.Zero(); } } Ue(g1e, X1e); const T1e = "FGScaleMatrixBlock"; class q1e extends ns { constructor(e) { super(D1, Yt, D1, (t, r) => t.scaleToRef(r, this._cachedMatrix), T1e, e), this._cachedMatrix = he.Zero(); } } Ue(T1e, q1e); const b1e = "FGClampMatrixBlock"; class x1e extends CA { constructor(e) { super(e), this._cachedArray = new Float32Array(16), this._cachedMatrix = he.Identity(), this.input = this.registerDataInput("input", D1), this.min = this.registerDataInput("min", Yt), this.max = this.registerDataInput("max", Yt), this.output = this.registerDataOutput("output", D1); } _updateOutputs(e) { const t = this.input.getValue(e), r = this.min.getValue(e), n = this.max.getValue(e); for (let i = 0; i < t.m.length; i++) this._cachedArray[i] = Math.min(Math.max(t.m[i], r), n); he.FromArrayToRef(this._cachedArray, 0, this._cachedMatrix), this.output.setValue(this._cachedMatrix, e); } getClassName() { return b1e; } } Ue(b1e, x1e); const D1e = "FGDecomposeMatrixBlock"; class j1e extends CA { constructor(e) { super(e), this._cachedTranslation = new S(), this._cachedRotation = new Ze(), this._cachedScale = new S(), this.input = this.registerDataInput("input", D1), this.translation = this.registerDataOutput("translation", ma), this.rotation = this.registerDataOutput("rotation", hV), this.scale = this.registerDataOutput("scale", ma); } _updateOutputs(e) { this.input.getValue(e).decompose(this._cachedScale, this._cachedRotation, this._cachedTranslation), this.translation.setValue(this._cachedTranslation, e), this.rotation.setValue(this._cachedRotation, e), this.scale.setValue(this._cachedScale, e); } getClassName() { return D1e; } } Ue(D1e, j1e); const w1e = "FGComposeMatrixBlock"; class m1e extends CA { constructor(e) { super(e), this._cachedMatrix = new he(), this.output = this.registerDataOutput("input", D1), this.translation = this.registerDataInput("translation", ma), this.rotation = this.registerDataInput("rotation", hV), this.scale = this.registerDataInput("scale", ma); } _updateOutputs(e) { const t = this.translation.getValue(e), r = this.rotation.getValue(e), n = this.scale.getValue(e); he.ComposeToRef(n, r, t, this._cachedMatrix), this.output.setValue(this._cachedMatrix, e); } getClassName() { return w1e; } } Ue(w1e, m1e); const B1e = "FGQuaternionToRotationMatrixBlock"; class W1e extends p1 { constructor(e) { super(hV, D1, (t) => he.FromQuaternionToRef(t, this._cachedMatrix), B1e, e), this._cachedMatrix = new he(); } } Ue(B1e, W1e); const S1e = "FGGetTransformationMatrixBlock"; class U1e extends ns { constructor(e) { super(w6, w6, D1, (t, r) => { const n = t.getWorldMatrix(); return r.getWorldMatrix().invertToRef(ue.Matrix[0]).multiplyToRef(n, this._cachedResult); }, S1e, e), this._cachedResult = he.Zero(); } } Ue(S1e, U1e); class I1e extends HU { /** * @internal */ _preparePendingTasks(e) { if (!e._getExecutionVariable(this, "sceneReadyObserver")) { const r = e.configuration.scene.onReadyObservable.add(() => { this._execute(e); }); e._setExecutionVariable(this, "sceneReadyObserver", r); } } /** * @internal */ _cancelPendingTasks(e) { const t = e._getExecutionVariable(this, "sceneReadyObserver"); e.configuration.scene.onReadyObservable.remove(t), e._deleteExecutionVariable(this, "sceneReadyObserver"); } getClassName() { return "FGSceneReadyEventBlock"; } } Ue("FGSceneReadyEventBlock", I1e); class R1e extends HU { constructor(e) { super(e), this.config = e, this.eventData = this.registerDataOutput("eventData", w6); } _preparePendingTasks(e) { const t = e.configuration.coordinator.getCustomEventObservable(this.config.eventId); this._eventObserver = t.add((r) => { this.eventData.setValue(r, e), this._execute(e); }); } _cancelPendingTasks(e) { const t = e.configuration.coordinator.getCustomEventObservable(this.config.eventId); t ? t.remove(this._eventObserver) : ye.Warn(`FlowGraphReceiveCustomEventBlock: Missing observable for event ${this.config.eventId}`); } getClassName() { return "FGReceiveCustomEventBlock"; } serialize(e) { super.serialize(e), e.eventId = this.config.eventId; } } Ue("FGReceiveCustomEventBlock", R1e); class V1e extends HU { /** * @internal */ _preparePendingTasks(e) { if (!e._getExecutionVariable(this, "sceneBeforeRender")) { const r = e.configuration.scene.onBeforeRenderObservable.add(() => { this._execute(e); }); e._setExecutionVariable(this, "sceneBeforeRender", r); } } /** * @internal */ _cancelPendingTasks(e) { const t = e._getExecutionVariable(this, "sceneBeforeRender"); e.configuration.scene.onBeforeRenderObservable.remove(t), e._deleteExecutionVariable(this, "sceneBeforeRender"); } getClassName() { return "FGSceneTickEventBlock"; } } Ue("FGSceneTickEventBlock", V1e); const Jge = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({ __proto__: null, AbstractActionManager: Ml, AbstractAssetTask: pg, AbstractMesh: jn, AbstractScene: J1, AcquireNativeObjectAsync: Hte, Action: wa, ActionEvent: mo, ActionManager: Rs, AddBlock: qre, get AddressMode() { return zw; }, AdvancedTimer: kte, AlignBlock: Une, AlphaState: k$, AmmoJSPlugin: eg, AnaglyphArcRotateCamera: Uee, AnaglyphFreeCamera: Iee, AnaglyphGamepadCamera: Ree, AnaglyphPostProcess: HQ, AnaglyphUniversalCamera: Vee, Analyser: H6e, AndOrNotEvaluator: Mx, Angle: JH, Animatable: ZN, get AnimatedInputBlockTypes() { return tq; }, Animation: st, AnimationAssetTask: sge, AnimationEvent: wO, AnimationGroup: w0, AnimationGroupMask: d6e, get AnimationGroupMaskMode() { return EI; }, get AnimationKeyInterpolation() { return CI; }, AnimationPropertiesOverride: t6e, AnimationRange: Am, AnisotropyBlock: fV, ApplyPostProcess: ty, Arc2: G$, ArcFollowCamera: wee, ArcRotateCamera: ps, ArcRotateCameraGamepadInput: VR, ArcRotateCameraInputsManager: CO, ArcRotateCameraKeyboardMoveInput: M0, ArcRotateCameraMouseWheelInput: AU, ArcRotateCameraPointersInput: ku, ArcRotateCameraVRDeviceOrientationInput: lQ, ArcTan2Block: Nre, AssetContainer: xR, get AssetTaskState() { return Y2; }, AssetsManager: age, AssetsProgressEvent: pie, AsyncLoop: up, AttachToBoxBehavior: T6e, AudioEngine: vee, AudioSceneComponent: yu, get AutoLayoutMode() { return SS; }, AutoReleaseWorkerPool: lU, AutoRotationBehavior: hee, AxesViewer: aq, Axis: bf, AxisDragGizmo: lp, AxisScaleGizmo: Pp, BRDFTextureTools: bPe, BabylonFileLoaderConfiguration: SI, BackEase: _$, BackgroundMaterial: Vi, BakedVertexAnimationManager: mm, BallAndSocketConstraint: $he, BaseCameraMouseWheelInput: IR, BaseCameraPointersInput: VO, BaseError: dm, BaseParticleSystem: N0, BaseSixDofDragBehavior: C2, BaseTexture: ls, BasisTools: py, BasisToolsOptions: qq, BasisTranscodeConfiguration: Npe, BezierCurve: z$, BezierCurveEase: f6e, BiPlanarBlock: one, BinaryFileAssetTask: Xie, BlackAndWhitePostProcess: AV, get BlendFactor() { return ad; }, get BlendOperation() { return Z8; }, BloomEffect: DF, BloomMergePostProcess: Wy, BlurPostProcess: c9, Bone: da, BoneAxesViewer: Bve, BoneIKController: hc, BoneLookController: wA, BonesBlock: nre, BooleanGeometryBlock: Dy, get BooleanGeometryOperations() { return ZH; }, BounceEase: r6e, BouncingBehavior: fD, BoundingBlock: Qne, BoundingBox: cp, BoundingBoxGizmo: zQ, BoundingBoxRenderer: sie, BoundingInfo: Md, BoundingSphere: rg, BoxBlock: Xy, BoxBuilder: Wve, BoxParticleEmitter: AD, Buffer: P9, get BufferBindingType() { return dq; }, get BufferMapState() { return aF; }, get BufferUsage() { return ga; }, CSG: YH, Camera: Tr, CameraGizmo: rS, CameraInputTypes: VA, CameraInputsManager: RR, CannonJSPlugin: sO, get CanvasAlphaMode() { return JI; }, CapsuleBlock: gY, CapsuleBuilder: Uve, CascadedShadowGenerator: X6, ChromaticAberrationPostProcess: Pg, CircleEase: Z$, CircleOfConfusionPostProcess: Om, ClampBlock: Hy, ClearCoatBlock: Xm, ClipPlanesBlock: Hre, ClipboardEventTypes: qp, ClipboardInfo: wC, CloudBlock: oY, CloudPoint: Zne, Collider: LR, Color3: Ne, Color3Gradient: Jne, Color4: xt, ColorCorrectionPostProcess: dV, ColorCurves: so, ColorGradient: yY, ColorGradingTexture: PD, ColorMergerBlock: mre, ColorSplitterBlock: vQ, get ColorWrite() { return dF; }, CombineAction: j$, get CompareFunction() { return q6; }, CompatibilityOptions: us, get CompilationMessageType() { return fF; }, CompleteGreasedLineColorTable: gne, CompleteGreasedLineWidthTable: Hne, get ComputeBindingType() { return io; }, ComputeEffect: Pm, ComputeNormalsBlock: Dne, get ComputePassTimestampLocation() { return vF; }, ComputeShader: vU, ComputeShaderParticleSystem: Kne, Condition: DS, ConditionBlock: jY, get ConditionBlockTests() { return y9; }, ConditionalBlock: ine, get ConditionalBlockConditions() { return lc; }, ConeParticleEmitter: wR, Constants: et, ContainerAssetTask: hie, get ConversionMode() { return dO; }, ConvolutionPostProcess: cg, get Coordinate() { return Lx; }, CopyTextureToTexture: lie, CopyTools: g6e, CreateBox: k0, CreateBoxVertexData: LO, CreateCapsule: KR, CreateCapsuleVertexData: JO, CreateCylinder: Ld, CreateCylinderVertexData: MO, CreateDashedLines: ZO, CreateDashedLinesVertexData: IQ, CreateDecal: ey, CreateDisc: uU, CreateDiscVertexData: zO, CreateEnvTextureAsync: cte, CreateGeodesic: rte, CreateGoldberg: ite, CreateGoldbergVertexData: nte, CreateGreasedLine: mhe, CreateGreasedLineMaterial: hne, CreateGround: Wm, CreateGroundFromHeightMap: NO, CreateGroundFromHeightMapVertexData: xQ, CreateGroundVertexData: y2, CreateHemisphere: zI, CreateIcoSphere: UR, CreateIcoSphereVertexData: RO, CreateImageDataArrayBufferViews: ny, CreateLathe: _O, CreateLineSystem: rm, CreateLineSystemVertexData: UQ, CreateLines: ka, CreatePlane: u4, CreatePlaneVertexData: UO, CreatePolygon: JR, CreatePolygonVertexData: RQ, CreatePolyhedron: hm, CreatePolyhedronVertexData: VQ, CreateResizedCopy: ute, CreateRibbon: n4, CreateRibbonVertexData: jQ, CreateScreenshot: PV, CreateScreenshotAsync: ZY, CreateScreenshotUsingRenderTarget: cV, CreateScreenshotUsingRenderTargetAsync: _Y, CreateScreenshotWithResizeAsync: mie, CreateSegmentedBoxVertexData: $ee, CreateSphere: UA, CreateSphereVertexData: KO, CreateText: ste, CreateTextShapePaths: OQ, CreateTiledBox: BQ, CreateTiledBoxVertexData: mQ, CreateTiledGround: FO, CreateTiledGroundVertexData: bQ, CreateTiledPlane: wQ, CreateTiledPlaneVertexData: $W, CreateTorus: Ag, CreateTorusKnot: GO, CreateTorusKnotVertexData: WQ, CreateTorusVertexData: QO, CreateTube: $O, CrossBlock: xre, CubeMapToSphericalPolynomialTools: Sm, CubeTexture: v1, CubeTextureAssetTask: bie, CubicEase: n6e, get CullMode() { return eS; }, CurrentScreenBlock: oQ, Curve3: j0, CurveBlock: dne, get CurveBlockTypes() { return no; }, CustomBlock: Dre, CustomOptimization: jie, CustomParticleEmitter: dD, CustomProceduralTexture: rre, CylinderBlock: HY, CylinderBuilder: mve, CylinderDirectedParticleEmitter: mR, CylinderParticleEmitter: aU, DDSTools: Aa, DataBuffer: tg, DataReader: gI, DataStorage: UF, Database: Cv, DebugBlock: yne, DebugLayer: pm, get DebugLayerTab() { return rF; }, DecalBuilder: rue, DecalMapConfiguration: cU, DecalMapDefines: vne, Decode: QN, DecodeBase64ToBinary: TR, DecodeBase64ToString: bO, DecodeBase64UrlToBinary: iU, DecodeBase64UrlToString: zN, DeepCopier: sA, DefaultCollisionCoordinator: _ee, DefaultKTX2DecoderOptions: Ite, DefaultLoadingScreen: oq, DefaultRenderingPipeline: K9, Deferred: MW, DepthCullingState: YN, DepthOfFieldBlurPostProcess: sR, DepthOfFieldEffect: jF, get DepthOfFieldEffectBlurLevel() { return am; }, DepthOfFieldMergePostProcess: eie, DepthPeelingRenderer: J2, DepthPeelingSceneComponent: oie, DepthReducer: Kte, DepthRenderer: jD, DepthRendererSceneComponent: aie, DepthSortedParticle: zne, DerivativeBlock: fre, DesaturateBlock: tne, DetailMapConfiguration: l4, get DeviceInputEventType() { return z5; }, get DeviceLostReason() { return lF; }, DeviceOrientationCamera: OO, DeviceSource: G5, DeviceSourceManager: J$, get DeviceType() { return vn; }, DirectionalLight: IA, DirectionalLightFrustumViewer: aue, DiscBlock: XY, DiscBuilder: Rve, DiscardBlock: are, DisplayPassPostProcess: Sy, DistanceBlock: Ore, DistanceConstraint: e0e, DistanceJoint: w6e, DivideBlock: Sre, DoNothingAction: RN, DomManagement: I9e, DotBlock: jre, DracoCompression: Yd, DrawWrapper: zo, get DualSenseInput() { return L5; }, get DualShockButton() { return dp; }, get DualShockDpad() { return Kw; }, get DualShockInput() { return M5; }, DualShockPad: Bee, DumpTools: eA, DynamicFloat32Array: M8, DynamicTexture: Xp, EasingFunction: u1, EdgesRenderer: uV, Effect: An, EffectFallbacks: c1, EffectLayer: mc, EffectLayerSceneComponent: Nte, EffectRenderer: fU, EffectWrapper: ng, ElasticEase: i6e, ElbowBlock: ane, EncodeArrayBufferToBase64: XR, EndsWith: C$, Engine: Ge, EngineFactory: Kle, get EngineFormat() { return nO; }, EngineInstrumentation: Ete, EngineStore: gr, EngineView: fue, EnvironmentHelper: gD, EnvironmentTextureTools: hue, Epsilon: Dn, EquiRectangularCubeTexture: VS, EquiRectangularCubeTextureAssetTask: Die, ErrorCodes: Z2, get ErrorFilter() { return PF; }, EventConstants: wm, EventState: IN, ExecuteCodeAction: w$, ExponentialEase: $$, ExternalTexture: FQ, ExtractHighlightsPostProcess: By, ExtrudePolygon: zR, ExtrudeShape: GR, ExtrudeShapeCustom: ZR, FactorGradient: kY, FadeInOutBehavior: q6e, get FeatureName() { return k2; }, get FileTools() { return kW; }, FileToolsOptions: A9, FilesInput: KY, FilesInputStore: em, get FilterMode() { return Ji; }, FilterPostProcess: vV, FlowGraph: JS, FlowGraphACosNumberBlock: mae, FlowGraphASinNumberBlock: jae, FlowGraphATan2NumberBlock: Cae, FlowGraphATanNumberBlock: Wae, FlowGraphAbsNumberBlock: eae, FlowGraphAddMatrixAndNumberBlock: d1e, FlowGraphAddMatrixBlock: f1e, FlowGraphAddNumberBlock: Cse, FlowGraphAddVector2Block: roe, FlowGraphAddVector3Block: goe, FlowGraphAddVector4Block: Moe, FlowGraphBitwiseAndBlock: Xse, FlowGraphBitwiseLeftShiftBlock: xse, FlowGraphBitwiseNotBlock: bse, FlowGraphBitwiseOrBlock: Tse, FlowGraphBitwiseRightShiftBlock: Dse, FlowGraphBitwiseXorBlock: qse, FlowGraphBlock: CA, FlowGraphBranchBlock: Mie, FlowGraphCeilNumberBlock: aae, FlowGraphClampMatrixBlock: x1e, FlowGraphComposeMatrixBlock: m1e, FlowGraphConditionalDataBlock: fse, FlowGraphConstantBlock: vse, FlowGraphContext: lO, FlowGraphContextLogger: Qge, FlowGraphCoordinateTransformBlock: dse, FlowGraphCoordinator: TS, FlowGraphCosNumberBlock: qae, FlowGraphCountLeadingZerosBlock: jse, FlowGraphCountTrailingZerosBlock: wse, FlowGraphCounterBlock: $ie, FlowGraphCreateVector2Block: coe, FlowGraphCreateVector3Block: Ooe, FlowGraphCreateVector4Block: i1e, FlowGraphCrossVector3Block: Voe, FlowGraphDebounceBlock: tse, FlowGraphDecomposeMatrixBlock: j1e, FlowGraphDivideMatrixAndNumberBlock: X1e, FlowGraphDivideMatrixBlock: H1e, FlowGraphDivideNumberBlock: Nse, FlowGraphDivideVector2Block: foe, FlowGraphDivideVector3Block: Doe, FlowGraphDivideVector4Block: Zoe, FlowGraphDoNBlock: Lie, FlowGraphDotVector3Block: Ioe, FlowGraphENumberBlock: Uae, FlowGraphEqualsNumberBlock: Yae, FlowGraphEventBlock: HU, FlowGraphExecutionBlock: og, FlowGraphExpNumberBlock: uae, FlowGraphFlipFlopBlock: rse, FlowGraphFloorNumberBlock: iae, FlowGraphForLoopBlock: Kie, FlowGraphGetTransformationMatrixBlock: U1e, FlowGraphGetVariableBlock: Ase, FlowGraphGreaterThanNumberBlock: Lae, FlowGraphGreaterThanOrEqualsNumberBlock: Jae, FlowGraphIsInfinityNumberBlock: Gse, FlowGraphIsNaNNumberBlock: Jse, FlowGraphLengthVector2Block: uoe, FlowGraphLengthVector3Block: Boe, FlowGraphLengthVector4Block: t1e, FlowGraphLessThanNumberBlock: Gae, FlowGraphLessThanOrEqualsNumberBlock: _ae, FlowGraphLnNumberBlock: Hae, FlowGraphLog10NumberBlock: Pae, FlowGraphLogBlock: Fie, FlowGraphLogNumberBlock: pae, FlowGraphLogicAndBlock: Sse, FlowGraphLogicNotBlock: Ise, FlowGraphLogicOrBlock: Use, FlowGraphMaxNumberBlock: Nae, FlowGraphMeshPickEventBlock: bm, FlowGraphMinNumberBlock: Eae, FlowGraphMixNumberBlock: eoe, FlowGraphModNumberBlock: Yse, FlowGraphMultiGateBlock: Gie, FlowGraphMultiplyMatrixBlock: p1e, FlowGraphMultiplyNumberBlock: Ese, FlowGraphMultiplyVector2Block: aoe, FlowGraphMultiplyVector3Block: boe, FlowGraphMultiplyVector4Block: zoe, FlowGraphNegateNumberBlock: rae, FlowGraphNormalizeVector2Block: Poe, FlowGraphNormalizeVector3Block: Soe, FlowGraphNormalizeVector4Block: r1e, FlowGraphPath: Cy, FlowGraphPauseAnimationBlock: sse, FlowGraphPiNumberBlock: Rae, FlowGraphPlayAnimationBlock: nse, FlowGraphPlayAudioBlock: ase, FlowGraphPowNumberBlock: Lse, FlowGraphQuaternionToRotationMatrixBlock: W1e, FlowGraphRandomNumberBlock: yae, FlowGraphReceiveCustomEventBlock: R1e, FlowGraphRotate2dVector2Block: hoe, FlowGraphRotate3dVector3Block: Foe, FlowGraphRoundNumberBlock: fae, FlowGraphScaleMatrixBlock: q1e, FlowGraphScaleVector2Block: doe, FlowGraphScaleVector3Block: woe, FlowGraphScaleVector4Block: $oe, FlowGraphSceneReadyEventBlock: I1e, FlowGraphSceneTickEventBlock: V1e, FlowGraphSendCustomEventBlock: Yie, FlowGraphSetPropertyBlock: Qie, FlowGraphSetVariableBlock: Nie, FlowGraphSignalConnection: uO, FlowGraphSinNumberBlock: Xae, FlowGraphSplitVector2Block: poe, FlowGraphSplitVector3Block: koe, FlowGraphSplitVector4Block: a1e, FlowGraphSqrtNumberBlock: _se, get FlowGraphState() { return $x; }, FlowGraphStopAnimationBlock: ise, FlowGraphStopAudioBlock: ose, FlowGraphSubtractMatrixAndNumberBlock: P1e, FlowGraphSubtractMatrixBlock: u1e, FlowGraphSubtractNumberBlock: yse, FlowGraphSubtractVector2Block: ioe, FlowGraphSubtractVector3Block: Toe, FlowGraphSubtractVector4Block: Koe, FlowGraphSwitchBlock: Zie, FlowGraphTanNumberBlock: xae, FlowGraphThrottleBlock: Jie, FlowGraphTimerBlock: zie, FlowGraphTransformVector3Block: Qoe, FlowGraphTruncNumberBlock: dae, FlowGraphWaitAllBlock: _ie, FlowGraphWhileLoopBlock: ese, FluidRenderer: Iy, FluidRendererSceneComponent: Pie, get FluidRenderingDebug() { return Pc; }, FluidRenderingObject: LY, FluidRenderingObjectCustomParticles: uie, FluidRenderingObjectParticleSystem: vie, FluidRenderingTargetRenderer: wF, FlyCamera: FR, FlyCameraInputsManager: Dee, FlyCameraKeyboardInput: jq, FlyCameraMouseInput: CR, FogBlock: pre, FollowBehavior: j6e, FollowCamera: Fv, FollowCameraInputsManager: jee, FollowCameraKeyboardMoveInput: aA, FollowCameraMouseWheelInput: qD, FollowCameraPointersInput: Eu, FragCoordBlock: Are, FragDepthBlock: lre, FragmentOutputBlock: iq, FramingBehavior: Vu, FreeCamera: SA, FreeCameraDeviceOrientationInput: PQ, FreeCameraGamepadInput: ER, FreeCameraInputsManager: kR, FreeCameraKeyboardMoveInput: wc, FreeCameraMouseInput: OR, FreeCameraMouseWheelInput: L0, FreeCameraTouchInput: yR, FreeCameraVirtualJoystickInput: cQ, FresnelBlock: Rre, FresnelParameters: GI, FromHalfFloat: QH, get FrontFace() { return MI; }, FrontFacingBlock: ore, Frustum: Xc, FxaaPostProcess: ym, GPUParticleSystem: bc, GUID: z9e, Gamepad: N9, GamepadCamera: yO, GamepadManager: Wee, GamepadSystemSceneComponent: See, GaussianSplatting: Fd, GenerateBase64StringFromPixelData: SO, GenerateBase64StringFromTexture: rQ, GenerateBase64StringFromTextureAsync: nQ, GenericPad: xee, GeodesicData: _R, Geometry: Tf, GeometryBufferRenderer: po, GeometryBufferRendererSceneComponent: rie, GeometryCollectionBlock: BY, GeometryElbowBlock: xne, GeometryInfoBlock: kne, GeometryInputBlock: xf, GeometryOptimizeBlock: Ty, GeometryOutputBlock: vY, GeometryTextureBlock: CY, GeometryTextureFetchBlock: OY, GeometryTransformBlock: UY, GeometryTrigonometryBlock: SY, get GeometryTrigonometryBlockOperations() { return Bs; }, GetClass: Jo, GetDOMTextContent: gR, GetEnvInfo: ry, GetEnvironmentBRDFTexture: nV, GetInternalFormatFromBasisFormat: ere, GetTGAHeader: sV, GetTextureDataAsync: lte, Gizmo: Lo, get GizmoAnchorPoint() { return QI; }, get GizmoCoordinatesMode() { return pD; }, GizmoManager: Jle, GlowLayer: zl, GoldbergMesh: $R, GradientBlock: Jre, GradientBlockColorStep: mC, GradientHelper: kl, GrainPostProcess: km, GreasedLineBaseMesh: dY, GreasedLineMaterialDefaults: D6, GreasedLineMesh: T1, get GreasedLineMeshColorDistribution() { return E2; }, get GreasedLineMeshColorDistributionType() { return K2; }, get GreasedLineMeshColorMode() { return Tc; }, get GreasedLineMeshMaterialType() { return OS; }, get GreasedLineMeshWidthDistribution() { return rq; }, GreasedLinePluginMaterial: o4, get GreasedLineRibbonAutoDirectionMode() { return cD; }, get GreasedLineRibbonFacesMode() { return kS; }, GreasedLineRibbonMesh: L1, get GreasedLineRibbonPointsMode() { return m0; }, GreasedLineSimpleMaterial: AY, GreasedLineTools: Qs, GridBlock: pY, GroundBuilder: xve, GroundMesh: dU, HDRCubeTexture: _2, HDRCubeTextureAssetTask: xie, HDRFiltering: _Q, HDRTools: iO, HandConstraintBehavior: E6e, get HandConstraintOrientation() { return z8; }, get HandConstraintVisibility() { return Jx; }, get HandConstraintZone() { return Kx; }, get HandPart() { return J8; }, HardwareScalingOptimization: mF, HavokPlugin: f0e, HeightToNormalBlock: aV, HemisphereBuilder: zle, HemisphericLight: y0, HemisphericParticleEmitter: BR, HighlightLayer: Ad, HighlightsPostProcess: E0e, Hinge2Joint: B6e, HingeConstraint: t0e, HingeJoint: m6e, HtmlElementTexture: cy, IFlowGraphCoordinatorConfiguration: Nge, IWebXRControllerPhysicsOptions: bge, IcoSphereBlock: PY, IcoSphereBuilder: y6e, ImageAssetTask: Tie, ImageProcessingBlock: nY, ImageProcessingConfiguration: Ui, ImageProcessingConfigurationDefines: M$, ImageProcessingPostProcess: Uy, ImageSourceBlock: E0, IncrementValueAction: b$, get IndexFormat() { return Tq; }, InputBlock: gi, get InspectableType() { return SF; }, InstancedLinesMesh: SQ, InstancedMesh: bp, InstancesBlock: ire, InstantiateBlock: Rne, InstantiateLinearBlock: Vne, InstantiateOnFacesBlock: IY, InstantiateOnVerticesBlock: by, InstantiateOnVolumeBlock: RY, InstantiateRadialBlock: Cne, InstantiatedEntries: dee, IntFloatConverterBlock: One, InternalTexture: As, get InternalTextureSource() { return ri; }, InterpolateValueAction: I$, IntersectionInfo: LC, IsBase64DataUrl: qR, IsDocumentAvailable: $w, IsFileURL: JN, IsNavigatorAvailable: Mw, IsWindowObjectExist: u9, get JoystickAxis() { return V9; }, KeepAssets: Aee, KeyboardEventTypes: t4, KeyboardInfo: MC, KeyboardInfoPre: Y5, KhronosTextureContainer: yv, KhronosTextureContainer2: H6, LatheBuilder: Kve, Layer: Yte, LayerSceneComponent: Qte, LengthBlock: yre, LensFlare: ly, LensFlareSystem: gm, LensFlareSystemSceneComponent: Mte, LensFlaresOptimization: RC, LensRenderingPipeline: d7e, LerpBlock: Wre, Light: ci, LightBlock: AO, LightGizmo: D0, LightInformationBlock: sre, LineEdgesRenderer: fie, LinesBuilder: Nve, LinesMesh: Tp, LoadFile: gq, LoadFileError: mS, LoadImage: nU, get LoadOp() { return f9; }, LoadTextureFromTranscodeResult: fO, LockConstraint: n0e, Logger: Se, get MapMode() { return vD; }, MapRangeBlock: qne, MappingBlock: VY, get MappingTypes() { return GH; }, Material: gt, MaterialAnisotropicDefines: wte, MaterialClearCoatDefines: Dte, MaterialDefines: na, MaterialDetailMapDefines: Lee, MaterialFlags: Dt, MaterialGreasedLineDefines: une, MaterialHelper: Ye, MaterialIridescenceDefines: jte, MaterialPluginBase: Gl, get MaterialPluginEvent() { return x6; }, MaterialPluginManager: sq, MaterialSheenDefines: mte, MaterialSubSurfaceDefines: Bte, MathBlock: DY, get MathBlockOperations() { return Ov; }, Matrix: he, MatrixBuilderBlock: nne, MatrixComposeBlock: Ene, MatrixDeterminantBlock: fne, MatrixTransposeBlock: Ane, MaxBlock: Vre, MergeGeometryBlock: mY, MergeMeshesOptimization: uq, Mesh: Ee, MeshAssetTask: Hie, MeshAttributeExistsBlock: fY, get MeshAttributeExistsBlockTypes() { return xa; }, MeshBlock: lY, MeshBuilder: Ef, get MeshDebugMode() { return im; }, MeshDebugPluginMaterial: Q9, MeshExploder: oge, MeshLODLevel: fee, MeshParticleEmitter: IO, MeshUVSpaceRenderer: Qw, MeshoptCompression: Q2, MinBlock: Cre, MinMaxReducer: Lte, get MipmapFilterMode() { return oF; }, MirrorTexture: uD, ModBlock: rne, ModelShape: xF, MorphTarget: lD, MorphTargetManager: U0, MorphTargetsBlock: rY, MotionBlurPostProcess: BD, MotorEnabledJoint: iQ, MultiMaterial: Dc, MultiObserver: JY, MultiPointerScaleBehavior: x6e, MultiRenderTarget: Zx, MultiplyBlock: zC, NLerpBlock: zre, NativeDataStream: hD, NativeEngine: Hm, get NativePointerInput() { return kI; }, NativeXRFrame: Vie, NativeXRLayerRenderTargetTextureProvider: Gee, NativeXRLayerWrapper: zee, NativeXRRenderTarget: Zee, NegateBlock: kre, Node: Cs, NodeGeometry: Ql, NodeGeometryBlock: yi, get NodeGeometryBlockConnectionPointTypes() { return Me; }, NodeGeometryBuildState: Xne, NodeGeometryConnectionPoint: qF, get NodeGeometryConnectionPointCompatibilityStates() { return $8; }, get NodeGeometryConnectionPointDirection() { return ES; }, get NodeGeometryContextualSources() { return Ws; }, NodeMaterial: ja, NodeMaterialBlock: Mr, get NodeMaterialBlockConnectionPointMode() { return jA; }, get NodeMaterialBlockConnectionPointTypes() { return de; }, get NodeMaterialBlockTargets() { return Ve; }, NodeMaterialConnectionPoint: um, get NodeMaterialConnectionPointCompatibilityStates() { return pc; }, NodeMaterialConnectionPointCustomObject: Ko, get NodeMaterialConnectionPointDirection() { return ao; }, NodeMaterialDefines: HI, get NodeMaterialModes() { return Vv; }, NodeMaterialOptimizer: _pe, get NodeMaterialSystemValues() { return Bi; }, NodeMaterialTeleportInBlock: Xre, NodeMaterialTeleportOutBlock: Tre, NoiseBlock: bne, NoiseProceduralTexture: ZI, NormalBlendBlock: Zre, NormalizeBlock: wre, NormalizeVectorBlock: wne, NullBlock: Tne, NullEngine: ote, NullEngineOptions: ate, Observable: Oe, Observer: c$, OcclusionMaterial: bpe, Octree: cm, OctreeBlock: NI, OctreeSceneComponent: DQ, OimoJSPlugin: gF, OnAfterEnteringVRObservableEvent: jve, OneMinusBlock: iY, get Orientation() { return tm; }, OutlineRenderer: MS, PBRAnisotropicConfiguration: DD, PBRBaseMaterial: fs, PBRBaseSimpleMaterial: Yv, PBRClearCoatConfiguration: h9, PBRIridescenceConfiguration: dd, PBRMaterial: mr, PBRMaterialDefines: cF, PBRMetallicRoughnessBlock: g9, PBRMetallicRoughnessMaterial: s4, PBRSheenConfiguration: K0, PBRSpecularGlossinessMaterial: a4, PBRSubSurfaceConfiguration: _o, PHI: kf, PadNumber: y$, PanoramaToCubeMapTools: wq, Particle: FS, ParticleBlendMultiplyBlock: dQ, ParticleHelper: Tm, ParticleRampGradientBlock: AQ, ParticleSystem: ti, ParticleSystemSet: bq, ParticleTextureBlock: fQ, ParticlesOptimization: VC, PassCubePostProcess: hQ, PassPostProcess: lg, Path2: sU, Path3D: vm, PathCursor: A6e, PerfCollectionStrategy: pge, PerfCounter: v9, PerformanceConfigurator: k9, PerformanceMonitor: iee, PerformanceViewerCollector: b0, PerturbNormalBlock: PU, PhotoDome: US, Physics6DoFConstraint: $ne, Physics6DoFLimit: _he, PhysicsAggregate: a0e, PhysicsBody: wy, PhysicsConstraint: mD, get PhysicsConstraintAxis() { return MH; }, get PhysicsConstraintAxisLimitMode() { return I2; }, get PhysicsConstraintMotorType() { return _x; }, get PhysicsConstraintType() { return kv; }, PhysicsEngine: $Q, PhysicsEngineV2: jy, get PhysicsEventType() { return q0; }, PhysicsHelper: d0e, PhysicsImpostor: tn, PhysicsJoint: ta, get PhysicsMaterialCombineMode() { return R2; }, get PhysicsMotionType() { return W0; }, PhysicsRadialExplosionEventOptions: sS, get PhysicsRadialImpulseFalloff() { return iR; }, PhysicsRaycastResult: iV, PhysicsShape: P4, PhysicsShapeBox: QY, PhysicsShapeCapsule: FY, PhysicsShapeContainer: Zhe, PhysicsShapeConvexHull: zhe, PhysicsShapeCylinder: NY, PhysicsShapeMesh: Ghe, PhysicsShapeSphere: EY, get PhysicsShapeType() { return Ff; }, PhysicsUpdraftEventOptions: YY, get PhysicsUpdraftMode() { return YS; }, PhysicsViewer: sue, PhysicsVortexEventOptions: MY, PickingInfo: F9, get PipelineErrorReason() { return AF; }, PivotTools: Vs, Plane: BA, PlaneBlock: uY, PlaneBuilder: b6e, PlaneDragGizmo: tS, PlaneRotationGizmo: S0, PlayAnimationAction: x$, PlaySoundAction: m$, get PointColor() { return Iv; }, PointLight: ag, PointParticleEmitter: WR, PointerDragBehavior: O9, PointerEventTypes: ir, PointerInfo: vp, PointerInfoBase: GN, PointerInfoPre: K$, get PointerInput() { return Yn; }, PointsCloudSystem: _ne, PointsGroup: WC, Polar: g0, Polygon: Yve, PolygonBuilder: Mve, PolygonMeshBuilder: ete, PolyhedronBuilder: zve, PolyhedronData: _C, PositionGizmo: oy, PositionNormalTextureVertex: aQ, PositionNormalVertex: sQ, PostProcess: kr, PostProcessManager: yI, PostProcessRenderEffect: gs, PostProcessRenderPipeline: SD, PostProcessRenderPipelineManager: nie, PostProcessRenderPipelineManagerSceneComponent: iie, PostProcessesOptimization: IC, PosterizeBlock: Lre, PowBlock: Ere, PowerEase: s6e, get PowerPreference() { return sF; }, PrePassOutputBlock: cre, PrePassRenderer: mA, PrePassRendererSceneComponent: Aie, PrePassTextureBlock: gre, PrecisionDate: Yi, PredicateCondition: H$, PressureObserverWrapper: Vy, get PrimitiveTopology() { return Fl; }, PrismaticConstraint: i0e, ProceduralTexture: Y0, ProceduralTextureSceneComponent: Tee, get PropertyTypeForEdition() { return Gr; }, PushMaterial: P1, QuadraticEase: _N, QuadraticErrorSimplification: cne, QuarticEase: a6e, Quaternion: Ze, get QueryType() { return KI; }, QuinticEase: o6e, RGBDTextureTools: $C, RandomBlock: wY, get RandomBlockLocks() { return B0; }, RandomGUID: v4, RandomNumberBlock: Fre, RawCubeTexture: hy, RawTexture: Bo, RawTexture2DArray: Py, RawTexture3D: zpe, Ray: Hi, RayHelper: yQ, ReadFile: BS, ReadFileError: xO, RecastJSCrowd: Yne, RecastJSPlugin: Bhe, ReciprocalBlock: Yre, ReflectBlock: $re, ReflectionBlock: Cm, ReflectionProbe: Im, ReflectionTextureBlock: hre, Reflector: fR, RefractBlock: ene, RefractionBlock: wD, RefractionPostProcess: WD, RefractionTexture: tY, RegisterClass: Ue, RegisterMaterialPlugin: Tve, RegisterNativeTypeAsync: gte, RemapBlock: jR, get RenderPassTimestampLocation() { return uF; }, RenderTargetTexture: Ta, RenderTargetWrapper: DR, RenderTargetsOptimization: BF, RenderingGroup: x0, RenderingGroupInfo: L$, RenderingManager: $6, ReplaceColorBlock: Mre, RequestFile: jO, RequestFileError: OI, RetryStrategy: V$, RibbonBuilder: Ive, RichType: Zl, RichTypeAny: w6, RichTypeBoolean: m6, RichTypeColor3: Cie, RichTypeColor4: Oie, RichTypeMatrix: D1, RichTypeNumber: Yt, RichTypeQuaternion: hV, RichTypeString: pV, RichTypeVector2: Y9, RichTypeVector3: ma, RichTypeVector4: iA, RollingAverage: see, Rotate2dBlock: _re, RotationGizmo: GQ, RotationXBlock: mne, RotationYBlock: Bne, RotationZBlock: Wne, RuntimeAnimation: R$, RuntimeError: O0, SSAO2RenderingPipeline: H9, SSAORenderingPipeline: pU, SSRRenderingPipeline: Wa, get SamplerBindingType() { return Xq; }, Scalar: Xt, ScaleBlock: bre, ScaleGizmo: ZQ, ScalingBlock: Sne, Scene: sr, SceneComponentConstants: Ot, SceneDepthBlock: oV, SceneInstrumentation: Fte, SceneLoader: Hn, get SceneLoaderAnimationGroupLoadingMode() { return O2; }, SceneLoaderFlags: l9, SceneOptimization: c4, SceneOptimizer: zY, SceneOptimizerOptions: om, get ScenePerformancePriority() { return $H; }, SceneRecorder: vge, SceneSerializer: G2, ScreenSizeBlock: dre, ScreenSpaceBlock: vre, ScreenSpaceCurvaturePostProcess: hU, ScreenSpaceReflectionPostProcess: jp, ScreenshotTools: Age, SerializationHelper: jt, SetColorsBlock: bY, SetCorsBehavior: DO, SetMaterialIDBlock: WY, SetNormalsBlock: qY, SetParentAction: VN, SetPositionsBlock: TY, SetStateAction: T$, SetTangentsBlock: xY, SetUVsBlock: qy, SetValueAction: q$, ShaderCodeInliner: Aq, get ShaderLanguage() { return za; }, ShaderMaterial: Zo, get ShaderStage() { return G8; }, ShaderStore: Le, ShadowDepthWrapper: ehe, ShadowGenerator: ln, ShadowGeneratorSceneComponent: Jte, ShadowLight: xD, ShadowMapBlock: Pre, ShadowsOptimization: UC, ShapeBuilder: Lve, SharpenPostProcess: Em, SheenBlock: Vm, SimplexPerlin3DBlock: Gre, SimplicationQueueSceneComponent: pne, SimplificationQueue: Pne, SimplificationSettings: xhe, get SimplificationType() { return nR; }, SineEase: $N, SixDofDragBehavior: Hee, Size: Qd, Skeleton: r4, SkeletonViewer: Ru, SliderConstraint: r0e, SmartArray: qf, SmartArrayNoDuplicate: K8, SmoothStepBlock: Qre, SolidParticle: bF, SolidParticleSystem: Jhe, SolidParticleVertex: Gne, Sound: qc, SoundTrack: uee, get SourceTextureFormat() { return pF; }, get Space() { return ai; }, SphereBlock: cY, SphereBuilder: Sve, SphereDirectedParticleEmitter: SR, SphereParticleEmitter: oU, Spherical: X0, SphericalHarmonics: HD, SphericalPolynomial: i4, SpotLight: nA, SpringConstraint: s0e, Sprite: Ry, SpriteManager: z2, SpriteMap: nge, SpritePackedManager: ige, SpriteSceneComponent: cie, Stage: Y1, StandardMaterial: Wt, StandardMaterialDefines: Kee, StandardRenderingPipeline: Ss, StartsWith: O$, StateCondition: g$, get StencilOperation() { return T0; }, StencilState: L2, StencilStateComposer: KN, StepBlock: Ire, StereoscopicArcRotateCamera: Oee, StereoscopicFreeCamera: yee, StereoscopicGamepadCamera: kee, StereoscopicInterlacePostProcess: z6e, StereoscopicInterlacePostProcessI: Cee, StereoscopicScreenUniversalCamera: G6e, StereoscopicUniversalCamera: Eee, StickValues: F6e, StopAnimationAction: D$, StopSoundAction: B$, StorageBuffer: bee, get StorageTextureAccess() { return eO; }, get StoreOp() { return Hc; }, StringDictionary: YC, StringTools: R9e, SubEmitter: vq, get SubEmitterType() { return NS; }, SubMesh: rA, SubSurfaceBlock: CS, SubSurfaceSceneComponent: die, SubtractBlock: Ure, SurfaceMagnetismBehavior: D6e, SwitchBooleanAction: X$, get SwitchInput() { return J5; }, TBNBlock: Rm, TGATools: Zte, Tags: Zi, TargetCamera: b1, TargetedAnimation: eee, TeleportInBlock: Fne, TeleportOutBlock: Nne, TestBase64DataUrl: Q$, TextFileAssetTask: gie, Texture: We, get TextureAspect() { return zH; }, TextureAssetTask: qie, TextureBlock: _I, get TextureDimension() { return Hp; }, get TextureFormat() { return we; }, TextureOptimization: SC, TexturePacker: I0, TexturePackerFrame: XF, get TextureSampleType() { return Ll; }, TextureSampler: MN, TextureTools: Pte, get TextureUsage() { return Po; }, get TextureViewDimension() { return Da; }, ThinEngine: hr, ThinRenderTargetTexture: Gpe, ThinTexture: nq, TiledBoxBuilder: Cve, TiledPlaneBuilder: Vve, get TimerState() { return Gx; }, TmpColors: Hs, TmpVectors: ue, ToGammaSpace: JW, ToHalfFloat: LH, ToLinearSpace: xI, TonemapPostProcess: C7e, get TonemappingOperator() { return _w; }, Tools: ye, TorusBlock: hY, TorusBuilder: Dve, TorusKnotBuilder: Ove, TouchCamera: pQ, TrailMesh: rR, Trajectory: fa, TrajectoryClassifier: vO, TranscodeAsync: oO, get TranscodeTarget() { return nS; }, TransformBlock: JC, TransformNode: Hr, TranslationBlock: Ine, TriPlanarBlock: gy, TrigonometryBlock: uQ, get TrigonometryBlockOperations() { return o9; }, TubeBuilder: Jve, TwirlBlock: ure, UniformBuffer: yr, UniversalCamera: bD, UnregisterAllMaterialPlugins: gQ, UnregisterMaterialPlugin: qve, UploadContent: eY, UploadEnvLevelsAsync: EQ, UploadEnvSpherical: iy, UploadLevelsAsync: YI, UtilityLayerRenderer: Ds, VRCameraMetrics: Bm, VRDeviceOrientationArcRotateCamera: Yee, VRDeviceOrientationFreeCamera: EO, VRDeviceOrientationGamepadCamera: Mee, VRDistortionCorrectionPostProcess: _5, VRExperienceHelper: lm, VRMultiviewToSingleviewPostProcess: Qee, ValidatedNativeDataStream: Xte, ValueCondition: C9, Vector2: at, Vector3: S, Vector4: Ir, VectorConverterBlock: jne, VectorMergerBlock: BI, VectorSplitterBlock: Bre, VertexAnimationBaker: X6e, VertexBuffer: J, VertexData: Ut, VertexDataMaterialInfo: wI, get VertexFormat() { return jo; }, VertexOutputBlock: mI, get VertexStepMode() { return LI; }, VideoDome: uy, VideoRecorder: aR, VideoTexture: xc, ViewDirectionBlock: sY, Viewport: WA, VirtualJoystick: $r, VirtualJoysticksCamera: Fee, VolumetricLightScatteringPostProcess: Bc, VoronoiNoiseBlock: sne, WaveBlock: Kre, get WaveBlockKind() { return Gw; }, WebGL2ParticleSystem: Lne, WebGL2ShaderProcessor: LN, WebGLDataBuffer: jS, WebGLHardwareTexture: jm, WebGLPipelineContext: E$, WebGPUCacheBindGroups: d1, WebGPUCacheRenderPipeline: co, WebGPUCacheRenderPipelineTree: N2, WebGPUCacheSampler: rD, WebGPUDataBuffer: qte, WebGPUDrawContext: tV, WebGPUEngine: wn, WebGPUTintWASM: Nl, WebRequest: ho, WebXRAbstractFeature: L9, WebXRAbstractMotionController: Um, WebXRAnchorSystem: aS, WebXRBackgroundRemover: fS, WebXRCamera: IS, WebXRCompositionLayerWrapper: Wie, WebXRControllerComponent: Jl, WebXRControllerMovement: Pq, WebXRControllerPhysics: AS, WebXRControllerPointerSelection: ig, WebXRDefaultExperience: vy, WebXRDefaultExperienceOptions: P3e, WebXRDepthSensing: HS, WebXRDomOverlay: PS, WebXREnterExitUI: dy, WebXREnterExitUIButton: yte, WebXREnterExitUIOptions: l3e, WebXRExperienceHelper: Ay, WebXREyeTracking: pS, WebXRFeatureName: Gi, WebXRFeaturePointSystem: vS, WebXRFeaturesManager: So, WebXRGenericHandController: Rie, WebXRGenericTriggerMotionController: RS, WebXRHTCViveMotionController: qm, WebXRHand: qee, get WebXRHandJoint() { return gn; }, WebXRHandTracking: va, WebXRHitTest: dS, WebXRHitTestLegacy: lq, WebXRImageTracking: lS, WebXRInput: Ote, WebXRInputSource: Cte, WebXRLayers: hS, WebXRLightEstimation: cS, WebXRManagedOutputCanvas: Jee, WebXRManagedOutputCanvasOptions: YR, WebXRMeshDetector: uS, WebXRMicrosoftMixedRealityController: cq, WebXRMotionControllerManager: E9, WebXRMotionControllerTeleportation: nm, get WebXRNearControllerMode() { return zx; }, WebXRNearInteraction: sg, WebXROculusTouchMotionController: R0, WebXRPlaneDetector: oS, WebXRProfiledMotionController: Vte, WebXRProjectionLayerWrapper: Sie, WebXRRawCameraAccess: XS, WebXRSessionManager: MR, WebXRSpaceWarp: gS, WebXRSpaceWarpRenderTargetTextureProvider: Iie, get WebXRState() { return d9; }, get WebXRTrackingState() { return Jw; }, WebXRWalkingLocomotion: CC, WeightedSound: lee, WorkerPool: Ute, WorleyNoise3DBlock: aY, XRSpaceWarpRenderTarget: Uie, get Xbox360Button() { return El; }, get Xbox360Dpad() { return Lw; }, Xbox360Pad: mee, get XboxInput() { return K5; }, _BabylonLoaderRegistered: Tpe, _BasisTextureLoader: tre, _CreationDataStorage: tQ, _DDSTextureLoader: Wte, _ENVTextureLoader: Ste, _HDRTextureLoader: $te, _InstancesBatch: Z5, _KTXTextureLoader: Rte, _MeshCollisionData: aee, _OcclusionDataStorage: Ate, _PrimaryIsoTriangle: CQ, _TGATextureLoader: _te, _TimeToken: fte, _UpdateRGBDAsync: pte, _forceSceneHelpersToBundle: c3e, _forceTransformFeedbackToBundle: oue, _injectLTSFileTools: Y$, _staticOffsetValueColor3: FN, _staticOffsetValueColor4: NN, _staticOffsetValueQuaternion: ON, _staticOffsetValueSize: EN, _staticOffsetValueVector2: kN, _staticOffsetValueVector3: yN, addClipPlaneUniforms: Mf, allocateAndCopyTypedBuffer: KC, bindClipPlane: Df, captureEquirectangularFromScene: Xge, className: G9e, createDetailMapPlugin: ohe, createPBRAnisotropicPlugin: the, createPBRBRDFPlugin: rhe, createPBRClearCoatPlugin: nhe, createPBRIridescencePlugin: ihe, createPBRSheenPlugin: she, createPBRSubSurfacePlugin: ahe, createYieldingScheduler: tee, editableInPropertyPage: rn, expandToProperty: At, extractMinAndMax: WS, extractMinAndMaxIndexed: nee, getRichTypeFromValue: yie, inlineScheduler: FI, makeAsyncFunction: v6e, makeSyncFunction: ree, nativeOverride: Hq, normalizeEnvInfo: eV, prepareDefinesForClipPlanes: oee, prepareStringDefinesForClipPlanes: xq, runCoroutine: eQ, runCoroutineAsync: BO, runCoroutineSync: mO, serialize: M, serializeAsCameraReference: U$, serializeAsColor3: Oi, serializeAsColor4: rU, serializeAsColorCurves: W$, serializeAsFresnelParameters: eU, serializeAsImageProcessingConfiguration: CN, serializeAsMatrix: qO, serializeAsMeshReference: tU, serializeAsQuaternion: S$, serializeAsTexture: en, serializeAsVector2: HR, serializeAsVector3: fo, setAndStartTimer: HF, setStereoscopicAnaglyphRigMode: NR, setStereoscopicRigMode: QR, setVRRigMode: kO }, Symbol.toStringTag, { value: "Module" })), iZ = Object.getOwnPropertyDescriptor(jn.prototype, "visibility"); Object.defineProperty(jn.prototype, "visibility", { get() { return iZ.get.call(this); }, set(A) { iZ.set.call(this, A), this.getChildren().forEach((e) => { e instanceof Hr && (e.visibility = A); }); } }); Object.defineProperties(Hr.prototype, { _visibility: { value: 1, writable: !0 }, visibility: { get() { return this._visibility; }, set(A) { this._visibility = A, this.getChildren().forEach((e) => { e instanceof Hr && (e.visibility = A); }); } } }); const zge = "data:application/javascript;base64,DQp2YXIgRHJhY29EZWNvZGVyTW9kdWxlID0gKGZ1bmN0aW9uKCkgew0KICB2YXIgX3NjcmlwdERpciA9IHR5cGVvZiBkb2N1bWVudCAhPT0gJ3VuZGVmaW5lZCcgJiYgZG9jdW1lbnQuY3VycmVudFNjcmlwdCA/IGRvY3VtZW50LmN1cnJlbnRTY3JpcHQuc3JjIDogdW5kZWZpbmVkOw0KICBpZiAodHlwZW9mIF9fZmlsZW5hbWUgIT09ICd1bmRlZmluZWQnKSBfc2NyaXB0RGlyID0gX3NjcmlwdERpciB8fCBfX2ZpbGVuYW1lOw0KICByZXR1cm4gKA0KZnVuY3Rpb24oRHJhY29EZWNvZGVyTW9kdWxlKSB7DQogIERyYWNvRGVjb2Rlck1vZHVsZSA9IERyYWNvRGVjb2Rlck1vZHVsZSB8fCB7fTsNCg0KdmFyIE1vZHVsZT10eXBlb2YgRHJhY29EZWNvZGVyTW9kdWxlIT09InVuZGVmaW5lZCI/RHJhY29EZWNvZGVyTW9kdWxlOnt9O3ZhciByZWFkeVByb21pc2VSZXNvbHZlLHJlYWR5UHJvbWlzZVJlamVjdDtNb2R1bGVbInJlYWR5Il09bmV3IFByb21pc2UoZnVuY3Rpb24ocmVzb2x2ZSxyZWplY3Qpe3JlYWR5UHJvbWlzZVJlc29sdmU9cmVzb2x2ZTtyZWFkeVByb21pc2VSZWplY3Q9cmVqZWN0fSk7dmFyIGlzUnVudGltZUluaXRpYWxpemVkPWZhbHNlO3ZhciBpc01vZHVsZVBhcnNlZD1mYWxzZTtNb2R1bGVbIm9uUnVudGltZUluaXRpYWxpemVkIl09ZnVuY3Rpb24oKXtpc1J1bnRpbWVJbml0aWFsaXplZD10cnVlO2lmKGlzTW9kdWxlUGFyc2VkKXtpZih0eXBlb2YgTW9kdWxlWyJvbk1vZHVsZUxvYWRlZCJdPT09ImZ1bmN0aW9uIil7TW9kdWxlWyJvbk1vZHVsZUxvYWRlZCJdKE1vZHVsZSl9fX07TW9kdWxlWyJvbk1vZHVsZVBhcnNlZCJdPWZ1bmN0aW9uKCl7aXNNb2R1bGVQYXJzZWQ9dHJ1ZTtpZihpc1J1bnRpbWVJbml0aWFsaXplZCl7aWYodHlwZW9mIE1vZHVsZVsib25Nb2R1bGVMb2FkZWQiXT09PSJmdW5jdGlvbiIpe01vZHVsZVsib25Nb2R1bGVMb2FkZWQiXShNb2R1bGUpfX19O2Z1bmN0aW9uIGlzVmVyc2lvblN1cHBvcnRlZCh2ZXJzaW9uU3RyaW5nKXtpZih0eXBlb2YgdmVyc2lvblN0cmluZyE9PSJzdHJpbmciKXJldHVybiBmYWxzZTtjb25zdCB2ZXJzaW9uPXZlcnNpb25TdHJpbmcuc3BsaXQoIi4iKTtpZih2ZXJzaW9uLmxlbmd0aDwyfHx2ZXJzaW9uLmxlbmd0aD4zKXJldHVybiBmYWxzZTtpZih2ZXJzaW9uWzBdPT0xJiZ2ZXJzaW9uWzFdPj0wJiZ2ZXJzaW9uWzFdPD01KXJldHVybiB0cnVlO2lmKHZlcnNpb25bMF0hPTB8fHZlcnNpb25bMV0+MTApcmV0dXJuIGZhbHNlO3JldHVybiB0cnVlfU1vZHVsZVsiaXNWZXJzaW9uU3VwcG9ydGVkIl09aXNWZXJzaW9uU3VwcG9ydGVkO3ZhciBtb2R1bGVPdmVycmlkZXM9e307dmFyIGtleTtmb3Ioa2V5IGluIE1vZHVsZSl7aWYoTW9kdWxlLmhhc093blByb3BlcnR5KGtleSkpe21vZHVsZU92ZXJyaWRlc1trZXldPU1vZHVsZVtrZXldfX12YXIgYXJndW1lbnRzXz1bXTt2YXIgdGhpc1Byb2dyYW09Ii4vdGhpcy5wcm9ncmFtIjt2YXIgcXVpdF89ZnVuY3Rpb24oc3RhdHVzLHRvVGhyb3cpe3Rocm93IHRvVGhyb3d9O3ZhciBFTlZJUk9OTUVOVF9JU19XRUI9dHlwZW9mIHdpbmRvdz09PSJvYmplY3QiO3ZhciBFTlZJUk9OTUVOVF9JU19XT1JLRVI9dHlwZW9mIGltcG9ydFNjcmlwdHM9PT0iZnVuY3Rpb24iO3ZhciBFTlZJUk9OTUVOVF9JU19OT0RFPXR5cGVvZiBwcm9jZXNzPT09Im9iamVjdCImJnR5cGVvZiBwcm9jZXNzLnZlcnNpb25zPT09Im9iamVjdCImJnR5cGVvZiBwcm9jZXNzLnZlcnNpb25zLm5vZGU9PT0ic3RyaW5nIjt2YXIgc2NyaXB0RGlyZWN0b3J5PSIiO2Z1bmN0aW9uIGxvY2F0ZUZpbGUocGF0aCl7aWYoTW9kdWxlWyJsb2NhdGVGaWxlIl0pe3JldHVybiBNb2R1bGVbImxvY2F0ZUZpbGUiXShwYXRoLHNjcmlwdERpcmVjdG9yeSl9cmV0dXJuIHNjcmlwdERpcmVjdG9yeStwYXRofXZhciByZWFkXyxyZWFkQXN5bmMscmVhZEJpbmFyeSxzZXRXaW5kb3dUaXRsZTtmdW5jdGlvbiBsb2dFeGNlcHRpb25PbkV4aXQoZSl7aWYoZSBpbnN0YW5jZW9mIEV4aXRTdGF0dXMpcmV0dXJuO3ZhciB0b0xvZz1lO2VycigiZXhpdGluZyBkdWUgdG8gZXhjZXB0aW9uOiAiK3RvTG9nKX12YXIgbm9kZUZTO3ZhciBub2RlUGF0aDtpZihFTlZJUk9OTUVOVF9JU19OT0RFKXtpZihFTlZJUk9OTUVOVF9JU19XT1JLRVIpe3NjcmlwdERpcmVjdG9yeT1yZXF1aXJlKCJwYXRoIikuZGlybmFtZShzY3JpcHREaXJlY3RvcnkpKyIvIn1lbHNle3NjcmlwdERpcmVjdG9yeT1fX2Rpcm5hbWUrIi8ifXJlYWRfPWZ1bmN0aW9uIHNoZWxsX3JlYWQoZmlsZW5hbWUsYmluYXJ5KXt2YXIgcmV0PXRyeVBhcnNlQXNEYXRhVVJJKGZpbGVuYW1lKTtpZihyZXQpe3JldHVybiBiaW5hcnk/cmV0OnJldC50b1N0cmluZygpfWlmKCFub2RlRlMpbm9kZUZTPXJlcXVpcmUoImZzIik7aWYoIW5vZGVQYXRoKW5vZGVQYXRoPXJlcXVpcmUoInBhdGgiKTtmaWxlbmFtZT1ub2RlUGF0aFsibm9ybWFsaXplIl0oZmlsZW5hbWUpO3JldHVybiBub2RlRlNbInJlYWRGaWxlU3luYyJdKGZpbGVuYW1lLGJpbmFyeT9udWxsOiJ1dGY4Iil9O3JlYWRCaW5hcnk9ZnVuY3Rpb24gcmVhZEJpbmFyeShmaWxlbmFtZSl7dmFyIHJldD1yZWFkXyhmaWxlbmFtZSx0cnVlKTtpZighcmV0LmJ1ZmZlcil7cmV0PW5ldyBVaW50OEFycmF5KHJldCl9YXNzZXJ0KHJldC5idWZmZXIpO3JldHVybiByZXR9O3JlYWRBc3luYz1mdW5jdGlvbiByZWFkQXN5bmMoZmlsZW5hbWUsb25sb2FkLG9uZXJyb3Ipe3ZhciByZXQ9dHJ5UGFyc2VBc0RhdGFVUkkoZmlsZW5hbWUpO2lmKHJldCl7b25sb2FkKHJldCl9aWYoIW5vZGVGUylub2RlRlM9cmVxdWlyZSgiZnMiKTtpZighbm9kZVBhdGgpbm9kZVBhdGg9cmVxdWlyZSgicGF0aCIpO2ZpbGVuYW1lPW5vZGVQYXRoWyJub3JtYWxpemUiXShmaWxlbmFtZSk7bm9kZUZTWyJyZWFkRmlsZSJdKGZpbGVuYW1lLGZ1bmN0aW9uKGVycixkYXRhKXtpZihlcnIpb25lcnJvcihlcnIpO2Vsc2Ugb25sb2FkKGRhdGEuYnVmZmVyKX0pfTtpZihwcm9jZXNzWyJhcmd2Il0ubGVuZ3RoPjEpe3RoaXNQcm9ncmFtPXByb2Nlc3NbImFyZ3YiXVsxXS5yZXBsYWNlKC9cXC9nLCIvIil9YXJndW1lbnRzXz1wcm9jZXNzWyJhcmd2Il0uc2xpY2UoMik7cXVpdF89ZnVuY3Rpb24oc3RhdHVzLHRvVGhyb3cpe2lmKGtlZXBSdW50aW1lQWxpdmUoKSl7cHJvY2Vzc1siZXhpdENvZGUiXT1zdGF0dXM7dGhyb3cgdG9UaHJvd31sb2dFeGNlcHRpb25PbkV4aXQodG9UaHJvdyk7cHJvY2Vzc1siZXhpdCJdKHN0YXR1cyl9O01vZHVsZVsiaW5zcGVjdCJdPWZ1bmN0aW9uKCl7cmV0dXJuIltFbXNjcmlwdGVuIE1vZHVsZSBvYmplY3RdIn19ZWxzZSBpZihFTlZJUk9OTUVOVF9JU19XRUJ8fEVOVklST05NRU5UX0lTX1dPUktFUil7aWYoRU5WSVJPTk1FTlRfSVNfV09SS0VSKXtzY3JpcHREaXJlY3Rvcnk9c2VsZi5sb2NhdGlvbi5ocmVmfWVsc2UgaWYodHlwZW9mIGRvY3VtZW50IT09InVuZGVmaW5lZCImJmRvY3VtZW50LmN1cnJlbnRTY3JpcHQpe3NjcmlwdERpcmVjdG9yeT1kb2N1bWVudC5jdXJyZW50U2NyaXB0LnNyY31pZihfc2NyaXB0RGlyKXtzY3JpcHREaXJlY3Rvcnk9X3NjcmlwdERpcn1pZihzY3JpcHREaXJlY3RvcnkuaW5kZXhPZigiYmxvYjoiKSE9PTApe3NjcmlwdERpcmVjdG9yeT1zY3JpcHREaXJlY3Rvcnkuc3Vic3RyKDAsc2NyaXB0RGlyZWN0b3J5LnJlcGxhY2UoL1s/I10uKi8sIiIpLmxhc3RJbmRleE9mKCIvIikrMSl9ZWxzZXtzY3JpcHREaXJlY3Rvcnk9IiJ9e3JlYWRfPWZ1bmN0aW9uKHVybCl7dHJ5e3ZhciB4aHI9bmV3IFhNTEh0dHBSZXF1ZXN0O3hoci5vcGVuKCJHRVQiLHVybCxmYWxzZSk7eGhyLnNlbmQobnVsbCk7cmV0dXJuIHhoci5yZXNwb25zZVRleHR9Y2F0Y2goZXJyKXt2YXIgZGF0YT10cnlQYXJzZUFzRGF0YVVSSSh1cmwpO2lmKGRhdGEpe3JldHVybiBpbnRBcnJheVRvU3RyaW5nKGRhdGEpfXRocm93IGVycn19O2lmKEVOVklST05NRU5UX0lTX1dPUktFUil7cmVhZEJpbmFyeT1mdW5jdGlvbih1cmwpe3RyeXt2YXIgeGhyPW5ldyBYTUxIdHRwUmVxdWVzdDt4aHIub3BlbigiR0VUIix1cmwsZmFsc2UpO3hoci5yZXNwb25zZVR5cGU9ImFycmF5YnVmZmVyIjt4aHIuc2VuZChudWxsKTtyZXR1cm4gbmV3IFVpbnQ4QXJyYXkoeGhyLnJlc3BvbnNlKX1jYXRjaChlcnIpe3ZhciBkYXRhPXRyeVBhcnNlQXNEYXRhVVJJKHVybCk7aWYoZGF0YSl7cmV0dXJuIGRhdGF9dGhyb3cgZXJyfX19cmVhZEFzeW5jPWZ1bmN0aW9uKHVybCxvbmxvYWQsb25lcnJvcil7dmFyIHhocj1uZXcgWE1MSHR0cFJlcXVlc3Q7eGhyLm9wZW4oIkdFVCIsdXJsLHRydWUpO3hoci5yZXNwb25zZVR5cGU9ImFycmF5YnVmZmVyIjt4aHIub25sb2FkPWZ1bmN0aW9uKCl7aWYoeGhyLnN0YXR1cz09MjAwfHx4aHIuc3RhdHVzPT0wJiZ4aHIucmVzcG9uc2Upe29ubG9hZCh4aHIucmVzcG9uc2UpO3JldHVybn12YXIgZGF0YT10cnlQYXJzZUFzRGF0YVVSSSh1cmwpO2lmKGRhdGEpe29ubG9hZChkYXRhLmJ1ZmZlcik7cmV0dXJufW9uZXJyb3IoKX07eGhyLm9uZXJyb3I9b25lcnJvcjt4aHIuc2VuZChudWxsKX19c2V0V2luZG93VGl0bGU9ZnVuY3Rpb24odGl0bGUpe2RvY3VtZW50LnRpdGxlPXRpdGxlfX1lbHNle312YXIgb3V0PU1vZHVsZVsicHJpbnQiXXx8Y29uc29sZS5sb2cuYmluZChjb25zb2xlKTt2YXIgZXJyPU1vZHVsZVsicHJpbnRFcnIiXXx8Y29uc29sZS53YXJuLmJpbmQoY29uc29sZSk7Zm9yKGtleSBpbiBtb2R1bGVPdmVycmlkZXMpe2lmKG1vZHVsZU92ZXJyaWRlcy5oYXNPd25Qcm9wZXJ0eShrZXkpKXtNb2R1bGVba2V5XT1tb2R1bGVPdmVycmlkZXNba2V5XX19bW9kdWxlT3ZlcnJpZGVzPW51bGw7aWYoTW9kdWxlWyJhcmd1bWVudHMiXSlhcmd1bWVudHNfPU1vZHVsZVsiYXJndW1lbnRzIl07aWYoTW9kdWxlWyJ0aGlzUHJvZ3JhbSJdKXRoaXNQcm9ncmFtPU1vZHVsZVsidGhpc1Byb2dyYW0iXTtpZihNb2R1bGVbInF1aXQiXSlxdWl0Xz1Nb2R1bGVbInF1aXQiXTt2YXIgd2FzbUJpbmFyeTtpZihNb2R1bGVbIndhc21CaW5hcnkiXSl3YXNtQmluYXJ5PU1vZHVsZVsid2FzbUJpbmFyeSJdO3ZhciBub0V4aXRSdW50aW1lPU1vZHVsZVsibm9FeGl0UnVudGltZSJdfHx0cnVlO3ZhciBXZWJBc3NlbWJseT17TWVtb3J5OmZ1bmN0aW9uKG9wdHMpe3RoaXMuYnVmZmVyPW5ldyBBcnJheUJ1ZmZlcihvcHRzWyJpbml0aWFsIl0qNjU1MzYpfSxNb2R1bGU6ZnVuY3Rpb24oYmluYXJ5KXt9LEluc3RhbmNlOmZ1bmN0aW9uKG1vZHVsZSxpbmZvKXt0aGlzLmV4cG9ydHM9KA0KLy8gRU1TQ1JJUFRFTl9TVEFSVF9BU00NCmZ1bmN0aW9uIGluc3RhbnRpYXRlKGxhKXtmdW5jdGlvbiBjKGQpe2Quc2V0PWZ1bmN0aW9uKGEsYil7dGhpc1thXT1ifTtkLmdldD1mdW5jdGlvbihhKXtyZXR1cm4gdGhpc1thXX07cmV0dXJuIGR9dmFyIGU7dmFyIGY9bmV3IFVpbnQ4QXJyYXkoMTIzKTtmb3IodmFyIGE9MjU7YT49MDstLWEpe2ZbNDgrYV09NTIrYTtmWzY1K2FdPWE7Zls5NythXT0yNithfWZbNDNdPTYyO2ZbNDddPTYzO2Z1bmN0aW9uIGwobSxuLG8pe3ZhciBnLGgsYT0wLGk9bixqPW8ubGVuZ3RoLGs9bisoaiozPj4yKS0ob1tqLTJdPT0iPSIpLShvW2otMV09PSI9Iik7Zm9yKDthPGo7YSs9NCl7Zz1mW28uY2hhckNvZGVBdChhKzEpXTtoPWZbby5jaGFyQ29kZUF0KGErMildO21baSsrXT1mW28uY2hhckNvZGVBdChhKV08PDJ8Zz4+NDtpZihpPGspbVtpKytdPWc8PDR8aD4+MjtpZihpPGspbVtpKytdPWg8PDZ8ZltvLmNoYXJDb2RlQXQoYSszKV19fWZ1bmN0aW9uIHAocSl7bChlLDEwMjgsIllBUUFBQUVBQUFBQ0FBQUFBd0FBQUFRQUFBQUZBQUFBQmdBQUFBY0FBQUFJQUFBQUNRQUFBQW9BQUFBTEFBQUFEQUFBQUU0MVpISmhZMjh5T0VGMGRISnBZblYwWlU5amRHRm9aV1J5YjI1VWNtRnVjMlp2Y20xRkFBQkFMQUFBT0FRQUFQd0dBQUFBQUFBQTBBUUFBQTRBQUFBUEFBQUFFQUFBQUJFQUFBQVNBQUFBRXdBQUFCUUFBQUFWQUFBQUZnQUFBQW9BQUFBWEFBQUFHQUFBQUU0MVpISmhZMjh6TUVGMGRISnBZblYwWlZGMVlXNTBhWHBoZEdsdmJsUnlZVzV6Wm05eWJVVUFBQUFBUUN3QUFLUUVBQUQ4QmdBQVgxOXVaWGgwWDNCeWFXMWxJRzkyWlhKbWJHOTNBSFpsWTNSdmNnQnpkR1E2T21WNFkyVndkR2x2YmdCemEybHdYMkYwZEhKcFluVjBaVjkwY21GdWMyWnZjbTBBWW1GemFXTmZjM1J5YVc1bkFHRnNiRzlqWVhSdmNqeFVQam82WVd4c2IyTmhkR1VvYzJsNlpWOTBJRzRwSUNkdUp5QmxlR05sWldSeklHMWhlR2x0ZFcwZ2MzVndjRzl5ZEdWa0lITnBlbVVBUkZKQlEwOEFWWE5wYm1jZ2FXNWpiMjF3WVhScFlteGxJR1JsWTI5a1pYSWdabTl5SUhSb1pTQnBibkIxZENCblpXOXRaWFJ5ZVM0QVJtRnBiR1ZrSUhSdklHUmxZMjlrWlNCd2IybHVkQ0JoZEhSeWFXSjFkR1Z6TGdCR1lXbHNaV1FnZEc4Z2FXNXBkR2xoYkdsNlpTQjBhR1VnWkdWamIyUmxjaTRBVlc1emRYQndiM0owWldRZ2JXbHViM0lnZG1WeWMybHZiaTRBVlc1emRYQndiM0owWldRZ2JXRnFiM0lnZG1WeWMybHZiaTRBU1c1d2RYUWdhWE1nYm05MElHRWdiV1Z6YUM0QVZXNXpkWEJ3YjNKMFpXUWdaMlZ2YldWMGNua2dkSGx3WlM0QVRtOTBJR0VnUkhKaFkyOGdabWxzWlM0QVZXNXpkWEJ3YjNKMFpXUWdaVzVqYjJScGJtY2diV1YwYUc5a0xnQkdZV2xzWldRZ2RHOGdaR1ZqYjJSbElHMWxkR0ZrWVhSaExnQkdZV2xzWldRZ2RHOGdaR1ZqYjJSbElHZGxiMjFsZEhKNUlHUmhkR0V1QUZCMWNtVWdkbWx5ZEhWaGJDQm1kVzVqZEdsdmJpQmpZV3hzWldRaEFFNDFaSEpoWTI4eE9FRjBkSEpwWW5WMFpWUnlZVzV6Wm05eWJVVUFBQmdzQUFEZUJnQUFBQUFBQUlnSEFBQVpBQUFBR2dBQUFCc0FBQUFjQUFBQUhRQUFBQjRBQUFBZkFBQUFJQUFBQUNFQUFBQWlBQUFBSXdBQUFDUUFBQUJPTldSeVlXTnZNVGRCZEhSeWFXSjFkR1Z6UkdWamIyUmxja1VBVGpWa2NtRmpiekkyUVhSMGNtbGlkWFJsYzBSbFkyOWtaWEpKYm5SbGNtWmhZMlZGQUFBQUFCZ3NBQUJZQndBQVFDd0FBRHdIQUFDQUJ3QUFBQUFBQU9nSEFBQWxBQUFBSmdBQUFDY0FBQUFvQUFBQUtRQUFBQ29BQUFBckFBQUFMQUFBQUMwQUFBQk9OV1J5WVdOdk1qWlRaWEYxWlc1MGFXRnNRWFIwY21saWRYUmxSR1ZqYjJSbGNrVUFBQUFBR0N3QUFNQUhBQUQvLy8vL0FBQUFBR0FJQUFBdUFBQUFMd0FBQUJzQUFBQXdBQUFBTVFBQUFCNEFBQUFmQUFBQUlBQUFBRElBQUFBekFBQUFOQUFBQURVQUFBQTJBQUFBVGpWa2NtRmpiek0zVTJWeGRXVnVkR2xoYkVGMGRISnBZblYwWlVSbFkyOWtaWEp6UTI5dWRISnZiR3hsY2tVQVFDd0FBREFJQUFDSUJ3QUFBQUFBQU5RSUFBQTNBQUFBT0FBQUFEa0FBQUFvQUFBQUtRQUFBQ29BQUFBNkFBQUFMQUFBQURzQUFBQThBQUFBUFFBQUFENEFBQUEvQUFBQVRqVmtjbUZqYnpNelUyVnhkV1Z1ZEdsaGJFbHVkR1ZuWlhKQmRIUnlhV0oxZEdWRVpXTnZaR1Z5UlFCQUxBQUFxQWdBQU9nSEFBQUFBQUFBWUFzQUFFQUFBQUJCQUFBQVFnQUFBRU1BQUFCRUFBQUFSUUFBQUVZQUFBQkhBQUFBU0FBQUFFa0FBQUJLQUFBQVN3QUFBRTQxWkhKaFkyODBNRTFsYzJoUWNtVmthV04wYVc5dVUyTm9aVzFsVUdGeVlXeHNaV3h2WjNKaGJVUmxZMjlrWlhKSmFVNVRYek0zVUhKbFpHbGpkR2x2YmxOamFHVnRaVmR5WVhCRVpXTnZaR2x1WjFSeVlXNXpabTl5YlVscGFVVkZUbE5mTWpSTlpYTm9VSEpsWkdsamRHbHZibE5qYUdWdFpVUmhkR0ZKVGxOZk1qUk5aWE5vUVhSMGNtbGlkWFJsUTI5eWJtVnlWR0ZpYkdWRlJVVkZSUUJPTldSeVlXTnZNamROWlhOb1VISmxaR2xqZEdsdmJsTmphR1Z0WlVSbFkyOWtaWEpKYVU1VFh6TTNVSEpsWkdsamRHbHZibE5qYUdWdFpWZHlZWEJFWldOdlpHbHVaMVJ5WVc1elptOXliVWxwYVVWRlRsTmZNalJOWlhOb1VISmxaR2xqZEdsdmJsTmphR1Z0WlVSaGRHRkpUbE5mTWpSTlpYTm9RWFIwY21saWRYUmxRMjl5Ym1WeVZHRmliR1ZGUlVWRlJRQk9OV1J5WVdOdk1qTlFjbVZrYVdOMGFXOXVVMk5vWlcxbFJHVmpiMlJsY2tscFRsTmZNemRRY21Wa2FXTjBhVzl1VTJOb1pXMWxWM0poY0VSbFkyOWthVzVuVkhKaGJuTm1iM0p0U1dscFJVVkZSUUJPTldSeVlXTnZNemRRY21Wa2FXTjBhVzl1VTJOb1pXMWxWSGx3WldSRVpXTnZaR1Z5U1c1MFpYSm1ZV05sU1dscFJVVUFUalZrY21GamJ6TXlVSEpsWkdsamRHbHZibE5qYUdWdFpVUmxZMjlrWlhKSmJuUmxjbVpoWTJWRkFFNDFaSEpoWTI4eU5WQnlaV1JwWTNScGIyNVRZMmhsYldWSmJuUmxjbVpoWTJWRkFCZ3NBQUFFQ3dBQVFDd0FBTmtLQUFBb0N3QUFRQ3dBQUtVS0FBQXdDd0FBUUN3QUFGRUtBQUE4Q3dBQVFDd0FBTHNKQUFCSUN3QUFRQ3dBQUJnSkFBQlVDd0FBQUFBQUFGUUxBQUJBQUFBQVRBQUFBQ0lBQUFCREFBQUFJZ0FBQUVVQUFBQkdBQUFBUndBQUFFZ0FBQUJKQUFBQVNnQUFBQ0lBQUFBQUFBQUFTQXNBQUVBQUFBQk5BQUFBSWdBQUFFTUFBQUFpQUFBQVJRQUFBRVlBQUFCSEFBQUFTQUFBQUVrQUFBQktBQUFBSWdBQUFBQUFBQURJREFBQVRnQUFBRThBQUFCUUFBQUFRd0FBQUZFQUFBQkZBQUFBUmdBQUFFY0FBQUJJQUFBQVNRQUFBRklBQUFCVEFBQUFUalZrY21GamJ6VTJUV1Z6YUZCeVpXUnBZM1JwYjI1VFkyaGxiV1ZEYjI1emRISmhhVzVsWkUxMWJIUnBVR0Z5WVd4c1pXeHZaM0poYlVSbFkyOWtaWEpKYVU1VFh6TTNVSEpsWkdsamRHbHZibE5qYUdWdFpWZHlZWEJFWldOdlpHbHVaMVJ5WVc1elptOXliVWxwYVVWRlRsTmZNalJOWlhOb1VISmxaR2xqZEdsdmJsTmphR1Z0WlVSaGRHRkpUbE5mTWpSTlpYTm9RWFIwY21saWRYUmxRMjl5Ym1WeVZHRmliR1ZGUlVWRlJRQUFRQ3dBQUJRTUFBQlVDd0FBQUFBQUFMUU5BQUJVQUFBQVZRQUFBRllBQUFCREFBQUFWd0FBQUZnQUFBQlpBQUFBV2dBQUFFZ0FBQUJKQUFBQVd3QUFBRndBQUFCT05XUnlZV052TkRSTlpYTm9VSEpsWkdsamRHbHZibE5qYUdWdFpWUmxlRU52YjNKa2MxQnZjblJoWW14bFJHVmpiMlJsY2tscFRsTmZNemRRY21Wa2FXTjBhVzl1VTJOb1pXMWxWM0poY0VSbFkyOWthVzVuVkhKaGJuTm1iM0p0U1dscFJVVk9VMTh5TkUxbGMyaFFjbVZrYVdOMGFXOXVVMk5vWlcxbFJHRjBZVWxPVTE4eU5FMWxjMmhCZEhSeWFXSjFkR1ZEYjNKdVpYSlVZV0pzWlVWRlJVVkZBQUJBTEFBQURBMEFBRlFMQUFBQUFBQUFvQTRBQUYwQUFBQmVBQUFBWHdBQUFFTUFBQUJnQUFBQVlRQUFBR0lBQUFCakFBQUFTQUFBQUVrQUFBQmtBQUFBWlFBQUFFNDFaSEpoWTI4ME1rMWxjMmhRY21Wa2FXTjBhVzl1VTJOb1pXMWxSMlZ2YldWMGNtbGpUbTl5YldGc1JHVmpiMlJsY2tscFRsTmZNemRRY21Wa2FXTjBhVzl1VTJOb1pXMWxWM0poY0VSbFkyOWthVzVuVkhKaGJuTm1iM0p0U1dscFJVVk9VMTh5TkUxbGMyaFFjbVZrYVdOMGFXOXVVMk5vWlcxbFJHRjBZVWxPVTE4eU5FMWxjMmhCZEhSeWFXSjFkR1ZEYjNKdVpYSlVZV0pzWlVWRlJVVkZBQUFBQUVBc0FBRDREUUFBVkFzQUFBQUFBQUFvRUFBQVpnQUFBR2NBQUFCb0FBQUFhUUFBQUdvQUFBQk9OV1J5WVdOdk5EaE5aWE5vVUhKbFpHbGpkR2x2YmxOamFHVnRaVWRsYjIxbGRISnBZMDV2Y20xaGJGQnlaV1JwWTNSdmNrRnlaV0ZKYVU1VFh6TTNVSEpsWkdsamRHbHZibE5qYUdWdFpWZHlZWEJFWldOdlpHbHVaMVJ5WVc1elptOXliVWxwYVVWRlRsTmZNalJOWlhOb1VISmxaR2xqZEdsdmJsTmphR1Z0WlVSaGRHRkpUbE5mTWpSTlpYTm9RWFIwY21saWRYUmxRMjl5Ym1WeVZHRmliR1ZGUlVWRlJRQk9OV1J5WVdOdk5EaE5aWE5vVUhKbFpHbGpkR2x2YmxOamFHVnRaVWRsYjIxbGRISnBZMDV2Y20xaGJGQnlaV1JwWTNSdmNrSmhjMlZKYVU1VFh6TTNVSEpsWkdsamRHbHZibE5qYUdWdFpWZHlZWEJFWldOdlpHbHVaMVJ5WVc1elptOXliVWxwYVVWRlRsTmZNalJOWlhOb1VISmxaR2xqZEdsdmJsTmphR1Z0WlVSaGRHRkpUbE5mTWpSTlpYTm9RWFIwY21saWRYUmxRMjl5Ym1WeVZHRmliR1ZGUlVWRlJRQUFBQmdzQUFCekR3QUFRQ3dBQU1nT0FBQWdFQUFBQUFBQUFDQVFBQUJtQUFBQWF3QUFBQ0lBQUFCcEFBQUFJZ0FBQUFBQUFBQzBFUUFBUUFBQUFHd0FBQUJ0QUFBQVF3QUFBRzRBQUFCRkFBQUFSZ0FBQUVjQUFBQklBQUFBU1FBQUFFb0FBQUJ2QUFBQVRqVmtjbUZqYnpRd1RXVnphRkJ5WldScFkzUnBiMjVUWTJobGJXVlFZWEpoYkd4bGJHOW5jbUZ0UkdWamIyUmxja2xwVGxOZk16ZFFjbVZrYVdOMGFXOXVVMk5vWlcxbFYzSmhjRVJsWTI5a2FXNW5WSEpoYm5ObWIzSnRTV2xwUlVWT1UxOHlORTFsYzJoUWNtVmthV04wYVc5dVUyTm9aVzFsUkdGMFlVbE9VMTh4TVVOdmNtNWxjbFJoWW14bFJVVkZSVVVBVGpWa2NtRmpiekkzVFdWemFGQnlaV1JwWTNScGIyNVRZMmhsYldWRVpXTnZaR1Z5U1dsT1UxOHpOMUJ5WldScFkzUnBiMjVUWTJobGJXVlhjbUZ3UkdWamIyUnBibWRVY21GdWMyWnZjbTFKYVdsRlJVNVRYekkwVFdWemFGQnlaV1JwWTNScGIyNVRZMmhsYldWRVlYUmhTVTVUWHpFeFEyOXlibVZ5VkdGaWJHVkZSVVZGUlFBQVFDd0FBQjRSQUFCSUN3QUFRQ3dBQUlnUUFBQ29FUUFBQUFBQUFLZ1JBQUJBQUFBQWNBQUFBQ0lBQUFCREFBQUFJZ0FBQUVVQUFBQkdBQUFBUndBQUFFZ0FBQUJKQUFBQVNnQUFBQ0lBQUFBQUFBQUEyQklBQUhFQUFBQnlBQUFBY3dBQUFFTUFBQUIwQUFBQVJRQUFBRVlBQUFCSEFBQUFTQUFBQUVrQUFBQjFBQUFBZGdBQUFFNDFaSEpoWTI4MU5rMWxjMmhRY21Wa2FXTjBhVzl1VTJOb1pXMWxRMjl1YzNSeVlXbHVaV1JOZFd4MGFWQmhjbUZzYkdWc2IyZHlZVzFFWldOdlpHVnlTV2xPVTE4ek4xQnlaV1JwWTNScGIyNVRZMmhsYldWWGNtRndSR1ZqYjJScGJtZFVjbUZ1YzJadmNtMUphV2xGUlU1VFh6STBUV1Z6YUZCeVpXUnBZM1JwYjI1VFkyaGxiV1ZFWVhSaFNVNVRYekV4UTI5eWJtVnlWR0ZpYkdWRlJVVkZSUUFBQUVBc0FBQXdFZ0FBcUJFQUFBQUFBQUM0RXdBQWR3QUFBSGdBQUFCNUFBQUFRd0FBQUhvQUFBQjdBQUFBZkFBQUFIMEFBQUJJQUFBQVNRQUFBSDRBQUFCL0FBQUFUalZrY21GamJ6UTBUV1Z6YUZCeVpXUnBZM1JwYjI1VFkyaGxiV1ZVWlhoRGIyOXlaSE5RYjNKMFlXSnNaVVJsWTI5a1pYSkphVTVUWHpNM1VISmxaR2xqZEdsdmJsTmphR1Z0WlZkeVlYQkVaV052WkdsdVoxUnlZVzV6Wm05eWJVbHBhVVZGVGxOZk1qUk5aWE5vVUhKbFpHbGpkR2x2YmxOamFHVnRaVVJoZEdGSlRsTmZNVEZEYjNKdVpYSlVZV0pzWlVWRlJVVkZBQUFBUUN3QUFCd1RBQUNvRVFBQUFBQUFBSlFVQUFDQUFBQUFnUUFBQUlJQUFBQkRBQUFBZ3dBQUFJUUFBQUNGQUFBQWhnQUFBRWdBQUFCSkFBQUFod0FBQUlnQUFBQk9OV1J5WVdOdk5ESk5aWE5vVUhKbFpHbGpkR2x2YmxOamFHVnRaVWRsYjIxbGRISnBZMDV2Y20xaGJFUmxZMjlrWlhKSmFVNVRYek0zVUhKbFpHbGpkR2x2YmxOamFHVnRaVmR5WVhCRVpXTnZaR2x1WjFSeVlXNXpabTl5YlVscGFVVkZUbE5mTWpSTlpYTm9VSEpsWkdsamRHbHZibE5qYUdWdFpVUmhkR0ZKVGxOZk1URkRiM0p1WlhKVVlXSnNaVVZGUlVWRkFFQXNBQUQ4RXdBQXFCRUFBQUFBQUFBQUZnQUFpUUFBQUlvQUFBQ0xBQUFBakFBQUFJMEFBQUJPTldSeVlXTnZORGhOWlhOb1VISmxaR2xqZEdsdmJsTmphR1Z0WlVkbGIyMWxkSEpwWTA1dmNtMWhiRkJ5WldScFkzUnZja0Z5WldGSmFVNVRYek0zVUhKbFpHbGpkR2x2YmxOamFHVnRaVmR5WVhCRVpXTnZaR2x1WjFSeVlXNXpabTl5YlVscGFVVkZUbE5mTWpSTlpYTm9VSEpsWkdsamRHbHZibE5qYUdWdFpVUmhkR0ZKVGxOZk1URkRiM0p1WlhKVVlXSnNaVVZGUlVWRkFFNDFaSEpoWTI4ME9FMWxjMmhRY21Wa2FXTjBhVzl1VTJOb1pXMWxSMlZ2YldWMGNtbGpUbTl5YldGc1VISmxaR2xqZEc5eVFtRnpaVWxwVGxOZk16ZFFjbVZrYVdOMGFXOXVVMk5vWlcxbFYzSmhjRVJsWTI5a2FXNW5WSEpoYm5ObWIzSnRTV2xwUlVWT1UxOHlORTFsYzJoUWNtVmthV04wYVc5dVUyTm9aVzFsUkdGMFlVbE9VMTh4TVVOdmNtNWxjbFJoWW14bFJVVkZSVVVBR0N3QUFGb1ZBQUJBTEFBQXZCUUFBUGdWQUFBQUFBQUErQlVBQUlrQUFBQ09BQUFBSWdBQUFJd0FBQUFpQUFBQUFBQUFBTHdXQUFCQUFBQUFqd0FBQUpBQUFBQkRBQUFBa1FBQUFFVUFBQUJHQUFBQVJ3QUFBRWdBQUFCSkFBQUFTZ0FBQUpJQUFBQk9OV1J5WVdOdk1qaFFjbVZrYVdOMGFXOXVVMk5vWlcxbFJHVnNkR0ZFWldOdlpHVnlTV2xPVTE4ek4xQnlaV1JwWTNScGIyNVRZMmhsYldWWGNtRndSR1ZqYjJScGJtZFVjbUZ1YzJadmNtMUphV2xGUlVWRkFBQUFBRUFzQUFCZ0ZnQUFTQXNBQUFBQUFBQXdGd0FBa3dBQUFKUUFBQUNWQUFBQUtBQUFBQ2tBQUFDV0FBQUFPZ0FBQUN3QUFBQTdBQUFBbHdBQUFKZ0FBQUNaQUFBQW1nQUFBRTQxWkhKaFkyOHpNbE5sY1hWbGJuUnBZV3hPYjNKdFlXeEJkSFJ5YVdKMWRHVkVaV052WkdWeVJRQUFRQ3dBQUFRWEFBRFVDQUFBQUFBQUFHUVpBQUNiQUFBQW5BQUFBSjBBQUFDZUFBQUFud0FBQUtBQUFBQ2hBQUFBb2dBQUFLTUFBQUNrQUFBQXBRQUFBS1lBQUFCT05XUnlZV052TkRKTlpYTm9VSEpsWkdsamRHbHZibE5qYUdWdFpVZGxiMjFsZEhKcFkwNXZjbTFoYkVSbFkyOWtaWEpKYVU1VFh6WXlVSEpsWkdsamRHbHZibE5qYUdWdFpVNXZjbTFoYkU5amRHRm9aV1J5YjI1RFlXNXZibWxqWVd4cGVtVmtSR1ZqYjJScGJtZFVjbUZ1YzJadmNtMUphVVZGVGxOZk1qUk5aWE5vVUhKbFpHbGpkR2x2YmxOamFHVnRaVVJoZEdGSlRsTmZNalJOWlhOb1FYUjBjbWxpZFhSbFEyOXlibVZ5VkdGaWJHVkZSVVZGUlFCT05XUnlZV052TWpkTlpYTm9VSEpsWkdsamRHbHZibE5qYUdWdFpVUmxZMjlrWlhKSmFVNVRYell5VUhKbFpHbGpkR2x2YmxOamFHVnRaVTV2Y20xaGJFOWpkR0ZvWldSeWIyNURZVzV2Ym1sallXeHBlbVZrUkdWamIyUnBibWRVY21GdWMyWnZjbTFKYVVWRlRsTmZNalJOWlhOb1VISmxaR2xqZEdsdmJsTmphR1Z0WlVSaGRHRkpUbE5mTWpSTlpYTm9RWFIwY21saWRYUmxRMjl5Ym1WeVZHRmliR1ZGUlVWRlJRQk9OV1J5WVdOdk1qTlFjbVZrYVdOMGFXOXVVMk5vWlcxbFJHVmpiMlJsY2tscFRsTmZOakpRY21Wa2FXTjBhVzl1VTJOb1pXMWxUbTl5YldGc1QyTjBZV2hsWkhKdmJrTmhibTl1YVdOaGJHbDZaV1JFWldOdlpHbHVaMVJ5WVc1elptOXliVWxwUlVWRlJRQUFRQ3dBQU44WUFBQThDd0FBUUN3QUFERVlBQUJNR1FBQVFDd0FBSFFYQUFCWUdRQUFBQUFBQUJ3YkFBQ25BQUFBcUFBQUFLa0FBQUNxQUFBQXF3QUFBRTQxWkhKaFkyODBPRTFsYzJoUWNtVmthV04wYVc5dVUyTm9aVzFsUjJWdmJXVjBjbWxqVG05eWJXRnNVSEpsWkdsamRHOXlRWEpsWVVscFRsTmZOakpRY21Wa2FXTjBhVzl1VTJOb1pXMWxUbTl5YldGc1QyTjBZV2hsWkhKdmJrTmhibTl1YVdOaGJHbDZaV1JFWldOdlpHbHVaMVJ5WVc1elptOXliVWxwUlVWT1UxOHlORTFsYzJoUWNtVmthV04wYVc5dVUyTm9aVzFsUkdGMFlVbE9VMTh5TkUxbGMyaEJkSFJ5YVdKMWRHVkRiM0p1WlhKVVlXSnNaVVZGUlVWRkFFNDFaSEpoWTI4ME9FMWxjMmhRY21Wa2FXTjBhVzl1VTJOb1pXMWxSMlZ2YldWMGNtbGpUbTl5YldGc1VISmxaR2xqZEc5eVFtRnpaVWxwVGxOZk5qSlFjbVZrYVdOMGFXOXVVMk5vWlcxbFRtOXliV0ZzVDJOMFlXaGxaSEp2YmtOaGJtOXVhV05oYkdsNlpXUkVaV052WkdsdVoxUnlZVzV6Wm05eWJVbHBSVVZPVTE4eU5FMWxjMmhRY21Wa2FXTjBhVzl1VTJOb1pXMWxSR0YwWVVsT1UxOHlORTFsYzJoQmRIUnlhV0oxZEdWRGIzSnVaWEpVWVdKc1pVVkZSVVZGQUFBQUdDd0FBRThhQUFCQUxBQUFqQmtBQUJRYkFBQUFBQUFBd0J3QUFLd0FBQUN0QUFBQXJnQUFBSjRBQUFDdkFBQUFzQUFBQUxFQUFBQ3lBQUFBb3dBQUFLUUFBQUN6QUFBQXRBQUFBRTQxWkhKaFkyODBNazFsYzJoUWNtVmthV04wYVc5dVUyTm9aVzFsUjJWdmJXVjBjbWxqVG05eWJXRnNSR1ZqYjJSbGNrbHBUbE5mTmpKUWNtVmthV04wYVc5dVUyTm9aVzFsVG05eWJXRnNUMk4wWVdobFpISnZia05oYm05dWFXTmhiR2w2WldSRVpXTnZaR2x1WjFSeVlXNXpabTl5YlVscFJVVk9VMTh5TkUxbGMyaFFjbVZrYVdOMGFXOXVVMk5vWlcxbFJHRjBZVWxPVTE4eE1VTnZjbTVsY2xSaFlteGxSVVZGUlVVQVRqVmtjbUZqYnpJM1RXVnphRkJ5WldScFkzUnBiMjVUWTJobGJXVkVaV052WkdWeVNXbE9VMTgyTWxCeVpXUnBZM1JwYjI1VFkyaGxiV1ZPYjNKdFlXeFBZM1JoYUdWa2NtOXVRMkZ1YjI1cFkyRnNhWHBsWkVSbFkyOWthVzVuVkhKaGJuTm1iM0p0U1dsRlJVNVRYekkwVFdWemFGQnlaV1JwWTNScGIyNVRZMmhsYldWRVlYUmhTVTVUWHpFeFEyOXlibVZ5VkdGaWJHVkZSVVZGUlFBQUFBQkFMQUFBRUJ3QUFFd1pBQUJBTEFBQVlCc0FBTFFjQUFBQUFBQUFYQjRBQUxVQUFBQzJBQUFBdHdBQUFMZ0FBQUM1QUFBQVRqVmtjbUZqYnpRNFRXVnphRkJ5WldScFkzUnBiMjVUWTJobGJXVkhaVzl0WlhSeWFXTk9iM0p0WVd4UWNtVmthV04wYjNKQmNtVmhTV2xPVTE4Mk1sQnlaV1JwWTNScGIyNVRZMmhsYldWT2IzSnRZV3hQWTNSaGFHVmtjbTl1UTJGdWIyNXBZMkZzYVhwbFpFUmxZMjlrYVc1blZISmhibk5tYjNKdFNXbEZSVTVUWHpJMFRXVnphRkJ5WldScFkzUnBiMjVUWTJobGJXVkVZWFJoU1U1VFh6RXhRMjl5Ym1WeVZHRmliR1ZGUlVWRlJRQk9OV1J5WVdOdk5EaE5aWE5vVUhKbFpHbGpkR2x2YmxOamFHVnRaVWRsYjIxbGRISnBZMDV2Y20xaGJGQnlaV1JwWTNSdmNrSmhjMlZKYVU1VFh6WXlVSEpsWkdsamRHbHZibE5qYUdWdFpVNXZjbTFoYkU5amRHRm9aV1J5YjI1RFlXNXZibWxqWVd4cGVtVmtSR1ZqYjJScGJtZFVjbUZ1YzJadmNtMUphVVZGVGxOZk1qUk5aWE5vVUhKbFpHbGpkR2x2YmxOamFHVnRaVVJoZEdGSlRsTmZNVEZEYjNKdVpYSlVZV0pzWlVWRlJVVkZBQmdzQUFDZUhRQUFRQ3dBQU9nY0FBQlVIZ0FBQUFBQUFCUWZBQUM2QUFBQXV3QUFBTHdBQUFDZUFBQUF2UUFBQUw0QUFBQy9BQUFBd0FBQUFLTUFBQUNrQUFBQXdRQUFBTUlBQUFCT05XUnlZV052TWpoUWNtVmthV04wYVc5dVUyTm9aVzFsUkdWc2RHRkVaV052WkdWeVNXbE9VMTgyTWxCeVpXUnBZM1JwYjI1VFkyaGxiV1ZPYjNKdFlXeFBZM1JoYUdWa2NtOXVRMkZ1YjI1cFkyRnNhWHBsWkVSbFkyOWthVzVuVkhKaGJuTm1iM0p0U1dsRlJVVkZBQUFBQUVBc0FBQ2dIZ0FBVEJrQUFBQUFBQUNZSHdBQXd3QUFBTVFBQUFERkFBQUFLQUFBQUNrQUFBREdBQUFBT2dBQUFDd0FBQUE3QUFBQXh3QUFBRDBBQUFBK0FBQUF5QUFBQU1rQUFBREtBQUFBVGpWa2NtRmpiek00VTJWeGRXVnVkR2xoYkZGMVlXNTBhWHBoZEdsdmJrRjBkSEpwWW5WMFpVUmxZMjlrWlhKRkFBQUFBRUFzQUFCa0h3QUExQWdBQUFBQUFBRDRId0FBeXdBQUFNd0FBQUROQUFBQXpnQUFBQ0lBQUFEUEFBQUEwQUFBQU5FQUFBRFNBQUFBMHdBQUFOUUFBQURWQUFBQUlnQUFBRTQxWkhKaFkyOHhNVTFsYzJoRVpXTnZaR1Z5UlFBQUFFQXNBQURnSHdBQXZDY0FBQUFBQUFCa0lBQUExZ0FBQU5jQUFBRE5BQUFBMkFBQUFOa0FBQURQQUFBQTBBQUFBTkVBQUFEYUFBQUEyd0FBQU53QUFBRGRBQUFBM2dBQUFFNDFaSEpoWTI4eU1rMWxjMmhGWkdkbFluSmxZV3RsY2tSbFkyOWtaWEpGQUFBQUFFQXNBQUJBSUFBQStCOEFBQUFBQUFCWUlRQUEzd0FBQU9BQUFBRGhBQUFBNGdBQUFPTUFBQURrQUFBQTVRQUFBT1lBQUFEbkFBQUE2QUFBQVAvLy8vOEFBQUFBdUNFQUFPa0FBQURxQUFBQTZ3QUFBT3dBQUFEdEFBQUE3Z0FBQU84QUFBRHdBQUFBOFFBQUFQSUFBQUJPTldSeVlXTnZNalpOWlhOb1JXUm5aV0p5WldGclpYSkVaV052WkdWeVNXMXdiRWxPVTE4ek1VMWxjMmhGWkdkbFluSmxZV3RsY2xSeVlYWmxjbk5oYkVSbFkyOWtaWEpGUlVVQVRqVmtjbUZqYnpNMVRXVnphRVZrWjJWaWNtVmhhMlZ5UkdWamIyUmxja2x0Y0d4SmJuUmxjbVpoWTJWRkFBQUFHQ3dBQUNBaEFBQkFMQUFBMUNBQUFGQWhBQUJPTldSeVlXTnZNalpOWlhOb1JXUm5aV0p5WldGclpYSkVaV052WkdWeVNXMXdiRWxPVTE4ek9FMWxjMmhGWkdkbFluSmxZV3RsY2xSeVlYWmxjbk5oYkZaaGJHVnVZMlZFWldOdlpHVnlSVVZGQUFCQUxBQUFaQ0VBQUZBaEFBQUFBQUFBdENJQUFQTUFBQUQwQUFBQTlRQUFBRTQxWkhKaFkyOHhPVVJsY0hSb1JtbHljM1JVY21GMlpYSnpaWEpKVGxOZk1qUk5aWE5vUVhSMGNtbGlkWFJsUTI5eWJtVnlWR0ZpYkdWRlRsTmZNelpOWlhOb1FYUjBjbWxpZFhSbFNXNWthV05sYzBWdVkyOWthVzVuVDJKelpYSjJaWEpKVXpGZlJVVkZSUUJPTldSeVlXTnZNVE5VY21GMlpYSnpaWEpDWVhObFNVNVRYekkwVFdWemFFRjBkSEpwWW5WMFpVTnZjbTVsY2xSaFlteGxSVTVUWHpNMlRXVnphRUYwZEhKcFluVjBaVWx1WkdsalpYTkZibU52WkdsdVowOWljMlZ5ZG1WeVNWTXhYMFZGUlVVQUdDd0FBRVVpQUFCQUxBQUEyQ0VBQUt3aUFBQUFBQUFBckNJQUFQWUFBQUQzQUFBQTlRPT0iKTtsKGUsODkzMiwiQVFBQUFBTUFBQUFGQUFBQUJ3QUFBQUFBQUFDNEl3QUErQUFBQVBrQUFBRDZBQUFBK3dBQUFFNDFaSEpoWTI4eU1rMWxjMmhVY21GMlpYSnpZV3hUWlhGMVpXNWpaWEpKVGxOZk1qaE5ZWGhRY21Wa2FXTjBhVzl1UkdWbmNtVmxWSEpoZG1WeWMyVnlTVTVUWHpFeFEyOXlibVZ5VkdGaWJHVkZUbE5mTXpaTlpYTm9RWFIwY21saWRYUmxTVzVrYVdObGMwVnVZMjlrYVc1blQySnpaWEoyWlhKSlV6SmZSVVZGUlVWRkFFNDFaSEpoWTI4eE5WQnZhVzUwYzFObGNYVmxibU5sY2tVQUFBQUFHQ3dBQUpNakFBQkFMQUFBRENNQUFMQWpBQUQvLy8vL0FBQUFBS2drQUFEOEFBQUEvUUFBQVA0QUFBQk9OV1J5WVdOdk1qaE5ZWGhRY21Wa2FXTjBhVzl1UkdWbmNtVmxWSEpoZG1WeWMyVnlTVTVUWHpFeFEyOXlibVZ5VkdGaWJHVkZUbE5mTXpaTlpYTm9RWFIwY21saWRYUmxTVzVrYVdObGMwVnVZMjlrYVc1blQySnpaWEoyWlhKSlV6RmZSVVZGUlFCT05XUnlZV052TVROVWNtRjJaWEp6WlhKQ1lYTmxTVTVUWHpFeFEyOXlibVZ5VkdGaWJHVkZUbE5mTXpaTlpYTm9RWFIwY21saWRYUmxTVzVrYVdObGMwVnVZMjlrYVc1blQySnpaWEoyWlhKSlV6RmZSVVZGUlFBQUdDd0FBRVVrQUFCQUxBQUEzQ01BQUtBa0FBQUFBQUFBb0NRQUFQOEFBQUFBQVFBQS9nQUFBQUFBQUFCZ0pRQUFBUUVBQUFJQkFBQURBUUFBQkFFQUFFNDFaSEpoWTI4eU1rMWxjMmhVY21GMlpYSnpZV3hUWlhGMVpXNWpaWEpKVGxOZk1UbEVaWEIwYUVacGNuTjBWSEpoZG1WeWMyVnlTVTVUWHpFeFEyOXlibVZ5VkdGaWJHVkZUbE5mTXpaTlpYTm9RWFIwY21saWRYUmxTVzVrYVdObGMwVnVZMjlrYVc1blQySnpaWEoyWlhKSlV6SmZSVVZGUlVWRkFBQUFRQ3dBQU9Ba0FBQ3dJd0FBQUFBQUFPQWxBQUFGQVFBQUJnRUFBUDRBQUFCT05XUnlZV052TVRsRVpYQjBhRVpwY25OMFZISmhkbVZ5YzJWeVNVNVRYekV4UTI5eWJtVnlWR0ZpYkdWRlRsTmZNelpOWlhOb1FYUjBjbWxpZFhSbFNXNWthV05sYzBWdVkyOWthVzVuVDJKelpYSjJaWEpKVXpGZlJVVkZSUUJBTEFBQWdDVUFBS0FrQUFBQUFBQUFrQ1lBQUFjQkFBQUlBUUFBQ1FFQUFBb0JBQUJPTldSeVlXTnZNakpOWlhOb1ZISmhkbVZ5YzJGc1UyVnhkV1Z1WTJWeVNVNVRYekU1UkdWd2RHaEdhWEp6ZEZSeVlYWmxjbk5sY2tsT1UxOHlORTFsYzJoQmRIUnlhV0oxZEdWRGIzSnVaWEpVWVdKc1pVVk9VMTh6TmsxbGMyaEJkSFJ5YVdKMWRHVkpibVJwWTJWelJXNWpiMlJwYm1kUFluTmxjblpsY2tsVE1sOUZSVVZGUlVVQUFFQXNBQUFFSmdBQXNDTT0iKTtsKGUsOTg5MiwiL0NZQUFNc0FBQUFMQVFBQXpRQUFBTTRBQUFBTUFRQUF6d0FBQU5BQUFBRFJBQUFBMGdBQUFOTUFBQURVQUFBQTFRQUFBQTBCQUFCT05XUnlZV052TWpGTlpYTm9VMlZ4ZFdWdWRHbGhiRVJsWTI5a1pYSkZBRUFzQUFEY0pnQUErQjhBQUFBQUFBQThKd0FBRGdFQUFBOEJBQUFRQVFBQUVRRUFBRTQxWkhKaFkyOHhOVXhwYm1WaGNsTmxjWFZsYm1ObGNrVUFBQUJBTEFBQUlDY0FBTEFqQUFBQUFBQUF2Q2NBQU1zQUFBQVNBUUFBRXdFQUFNNEFBQUFpQUFBQUZBRUFBTkFBQUFEUkFBQUEwZz09Iik7bChlLDEwMTEyLCJSbUZwYkdWa0lIUnZJSEJoY25ObElFUnlZV052SUdobFlXUmxjaTRBVGpWa2NtRmpiekUzVUc5cGJuUkRiRzkxWkVSbFkyOWtaWEpGQUFBQUdDd0FBSjRuQUFBQkFBQUFBUUFBQUFJQUFBQUNBQUFBQkFBQUFBUUFBQUFJQUFBQUNBQUFBQVFBQUFBSUFBQUFBUT09Iik7bChlLDEwMjQzLCJ3QUFBQU1BQUFBREFBQUFBd1AvLy8vLy8vLy8vQUFBQUFFQW9BQUFWQVFBQUZnRUFBQmNCQUFBWUFRQUFUalZrY21GamJ6Uk5aWE5vUlFBQUFFQXNBQUF3S0FBQWhDZ0FBUC8vLy84QUFBQUFBQUFBQUlRb0FBQVpBUUFBR2dFQUFCc0JBQUFjQVFBQVRqVmtjbUZqYnpFd1VHOXBiblJEYkc5MVpFVUFBQUFBR0N3QUFHd28iKTtsKGUsMTAzODgsIkFnQUFBQU1BQUFBRkFBQUFCd0FBQUFzQUFBQU5BQUFBRVFBQUFCTUFBQUFYQUFBQUhRQUFBQjhBQUFBbEFBQUFLUUFBQUNzQUFBQXZBQUFBTlFBQUFEc0FBQUE5QUFBQVF3QUFBRWNBQUFCSkFBQUFUd0FBQUZNQUFBQlpBQUFBWVFBQUFHVUFBQUJuQUFBQWF3QUFBRzBBQUFCeEFBQUFmd0FBQUlNQUFBQ0pBQUFBaXdBQUFKVUFBQUNYQUFBQW5RQUFBS01BQUFDbkFBQUFyUUFBQUxNQUFBQzFBQUFBdndBQUFNRUFBQURGQUFBQXh3QUFBTk1BQUFBQkFBQUFDd0FBQUEwQUFBQVJBQUFBRXdBQUFCY0FBQUFkQUFBQUh3QUFBQ1VBQUFBcEFBQUFLd0FBQUM4QUFBQTFBQUFBT3dBQUFEMEFBQUJEQUFBQVJ3QUFBRWtBQUFCUEFBQUFVd0FBQUZrQUFBQmhBQUFBWlFBQUFHY0FBQUJyQUFBQWJRQUFBSEVBQUFCNUFBQUFmd0FBQUlNQUFBQ0pBQUFBaXdBQUFJOEFBQUNWQUFBQWx3QUFBSjBBQUFDakFBQUFwd0FBQUtrQUFBQ3RBQUFBc3dBQUFMVUFBQUM3QUFBQXZ3QUFBTUVBQUFERkFBQUF4d0FBQU5FQUFBQXdNREF4TURJd016QTBNRFV3TmpBM01EZ3dPVEV3TVRFeE1qRXpNVFF4TlRFMk1UY3hPREU1TWpBeU1USXlNak15TkRJMU1qWXlOekk0TWprek1ETXhNekl6TXpNME16VXpOak0zTXpnek9UUXdOREUwTWpRek5EUTBOVFEyTkRjME9EUTVOVEExTVRVeU5UTTFORFUxTlRZMU56VTROVGsyTURZeE5qSTJNelkwTmpVMk5qWTNOamcyT1Rjd056RTNNamN6TnpRM05UYzJOemMzT0RjNU9EQTRNVGd5T0RNNE5EZzFPRFk0TnpnNE9EazVNRGt4T1RJNU16azBPVFU1TmprM09UZzVPUT09Iik7bChlLDEwOTgwLCJDZ0FBQUdRQUFBRG9Bd0FBRUNjQUFLQ0dBUUJBUWc4QWdKYVlBQURoOVFVQXlwbzdBQUFBQUN3ckFBQWRBUUFBSGdFQUFCOEJBQUJUZERsbGVHTmxjSFJwYjI0QUFBQUFHQ3dBQUJ3ckFBQUFBQUFBV0NzQUFBMEFBQUFnQVFBQUlRRUFBRk4wTVRGc2IyZHBZMTlsY25KdmNnQkFMQUFBU0NzQUFDd3JBQUFBQUFBQWpDc0FBQTBBQUFBaUFRQUFJUUVBQUZOME1USnNaVzVuZEdoZlpYSnliM0lBQUFBQVFDd0FBSGdyQUFCWUt3QUFVM1E1ZEhsd1pWOXBibVp2QUFBQUFCZ3NBQUNZS3dBQVRqRXdYMTlqZUhoaFltbDJNVEUyWDE5emFHbHRYM1I1Y0dWZmFXNW1iMFVBQUFBQVFDd0FBTEFyQUFDb0t3QUFUakV3WDE5amVIaGhZbWwyTVRFM1gxOWpiR0Z6YzE5MGVYQmxYMmx1Wm05RkFBQUFRQ3dBQU9BckFBRFVLd0FBQUFBQUFBUXNBQUFqQVFBQUpBRUFBQ1VCQUFBbUFRQUFKd0VBQUNnQkFBQXBBUUFBS2dFQUFBQUFBQUNJTEFBQUl3RUFBQ3NCQUFBbEFRQUFKZ0VBQUNjQkFBQXNBUUFBTFFFQUFDNEJBQUJPTVRCZlgyTjRlR0ZpYVhZeE1qQmZYM05wWDJOc1lYTnpYM1I1Y0dWZmFXNW1iMFVBQUFBQVFDd0FBR0FzQUFBRUxBPT0iKTtsKGUsMTE0MTMsIkwxQT0iKX12YXIgcj1uZXcgQXJyYXlCdWZmZXIoMTYpO3ZhciBzPW5ldyBJbnQzMkFycmF5KHIpO3ZhciB0PW5ldyBGbG9hdDMyQXJyYXkocik7dmFyIHU9bmV3IEZsb2F0NjRBcnJheShyKTtmdW5jdGlvbiB2KHcpe3RbMl09d31mdW5jdGlvbiB4KHkpe3JldHVybiBzW3ldfWZ1bmN0aW9uIGphKGthKXt2YXIgej1rYS5hO3ZhciBBPXouYnVmZmVyO3ouZ3Jvdz1oYTt2YXIgQj1uZXcgSW50OEFycmF5KEEpO3ZhciBDPW5ldyBJbnQxNkFycmF5KEEpO3ZhciBEPW5ldyBJbnQzMkFycmF5KEEpO3ZhciBFPW5ldyBVaW50OEFycmF5KEEpO3ZhciBGPW5ldyBVaW50MTZBcnJheShBKTt2YXIgRz1uZXcgVWludDMyQXJyYXkoQSk7dmFyIEg9bmV3IEZsb2F0MzJBcnJheShBKTt2YXIgST1uZXcgRmxvYXQ2NEFycmF5KEEpO3ZhciBKPU1hdGguaW11bDt2YXIgSz1NYXRoLmZyb3VuZDt2YXIgTD1NYXRoLmFiczt2YXIgTT1NYXRoLmNsejMyO3ZhciBOPU1hdGgubWluO3ZhciBPPU1hdGgubWF4O3ZhciBQPU1hdGguZmxvb3I7dmFyIFE9TWF0aC5jZWlsO3ZhciBSPU1hdGgudHJ1bmM7dmFyIFM9TWF0aC5zcXJ0O3ZhciBUPWthLmFib3J0O3ZhciBVPU5hTjt2YXIgVj1JbmZpbml0eTt2YXIgVz1rYS5iO3ZhciBYPWthLmM7dmFyIFk9a2EuZDt2YXIgWj1rYS5lO3ZhciBfPWthLmY7dmFyICQ9NTI1NDkxMjt2YXIgYWE9MDsNCi8vIEVNU0NSSVBURU5fU1RBUlRfRlVOQ1MNCmZ1bmN0aW9uIGpjKGEsYixjLGQpe3ZhciBlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTAsbz0wLHA9MCxxPTAscj0wLHM9MCx0PTAsdT0wLHY9MCx3PTAseD0wO2lmKCFhKXtyZXR1cm4gMX1nPURbYysyMD4+Ml07bD1EW2MrMTI+PjJdO2U9RFtjKzE2Pj4yXTthOntpZigoZ3wwKT49KGx8MCkmZT4+PjA+PUdbYys4Pj4yXXwoZ3wwKT4obHwwKSl7YnJlYWsgYX1sPUVbZStEW2M+PjJdfDBdO2U9ZSsxfDA7Zz1lP2c6ZysxfDA7RFtjKzE2Pj4yXT1lO0RbYysyMD4+Ml09ZztiOntzd2l0Y2gobHwwKXtjYXNlIDA6aj1hO2c9YjtuPWQ7YT0wO2Q9MDtsPSQrLTY0fDA7JD1sO0RbbCs1Nj4+Ml09MDtEW2wrNDg+PjJdPTA7RFtsKzUyPj4yXT0wO0RbbCs0MD4+Ml09MDtEW2wrNDQ+PjJdPTA7RFtsKzMyPj4yXT0wO0RbbCszNj4+Ml09MDtEW2wrMjQ+PjJdPTA7RFtsKzI4Pj4yXT0wO0RbbCsxNj4+Ml09MDtEW2wrMjA+PjJdPTA7RFtsKzg+PjJdPTA7RFtsKzEyPj4yXT0wO2M6e2lmKCFpYyhsKzh8MCxjKSl7YnJlYWsgY31pZighSGQobCs4fDAsYyl8KERbbCsyMD4+Ml0/MDpqKSl7YnJlYWsgY31jYyhjLDAsMCk7aWYoail7cj1nPDwyO3M9RFtsKzM2Pj4yXTt1PURbbCs0OD4+Ml07dj1EW2wrMjQ+PjJdO3doaWxlKDEpe2k9RFtsKzU2Pj4yXTtkOntpZihpPj4+MD4xNjM4Myl7YnJlYWsgZH1hPURbbCs1Mj4+Ml07d2hpbGUoMSl7aWYoKGF8MCk8PTApe2JyZWFrIGR9YT1hLTF8MDtEW2wrNTI+PjJdPWE7aT1FW2ErdXwwXXxpPDw4O0RbbCs1Nj4+Ml09aTtpZihpPj4+MDwxNjM4NCl7Y29udGludWV9YnJlYWt9fWE9aSY0MDk1O2s9RFt2KyhhPDwyKT4+Ml07Yj1zKyhrPDwzKXwwO0RbbCs1Nj4+Ml09KEooRFtiPj4yXSxpPj4+MTJ8MCkrYXwwKS1EW2IrND4+Ml07ZTp7aWYoKGd8MCk8PTApe2JyZWFrIGV9YT0wO2lmKCFFW2MrMzZ8MF18az4+PjA+MzIpe2JyZWFrIGN9aWYoayl7dz1rJi0yO3g9ayYxO2U9ZCtnfDA7Yj1EW2MrMzI+PjJdO3doaWxlKDEpe209RFtjKzI4Pj4yXTtxPURbYysyND4+Ml07aT0wO2E9YjtmPTA7cD0wO2lmKChrfDApIT0xKXt3aGlsZSgxKXtoPShhPj4+M3wwKStxfDA7Zjp7aWYoaD4+PjA+PW0+Pj4wKXtoPTA7YnJlYWsgZn1oPUVbaHwwXTtiPWErMXwwO0RbYyszMj4+Ml09YjtoPWg+Pj4oYSY3KSYxO2E9Yn1oPWg8PGl8ZjtmPTA7dD0oYT4+PjN8MCkrcXwwO2lmKHQ+Pj4wPG0+Pj4wKXtmPUVbdHwwXTtiPWErMXwwO0RbYyszMj4+Ml09YjtmPWY+Pj4oYSY3KSYxO2E9Yn10PWl8MTtpPWkrMnwwO2Y9aHxmPDx0O3A9cCsyfDA7aWYoKHd8MCkhPShwfDApKXtjb250aW51ZX1icmVha319aD1uKyhkPDwyKXwwO2lmKHgpe3A9bTttPShhPj4+M3wwKStxfDA7aWYocD4+PjA+bT4+PjApe209RVttfDBdO2I9YSsxfDA7RFtjKzMyPj4yXT1iO2E9bT4+PihhJjcpJjF9ZWxzZXthPTB9Zj1hPDxpfGZ9RFtoPj4yXT1mO2Q9ZCsxfDA7aWYoKGV8MCkhPShkfDApKXtjb250aW51ZX1icmVha31kPWU7YnJlYWsgZX1wYShuKyhkPDwyKXwwLDAscik7ZD1kK2d8MH1vPWcrb3wwO2lmKGo+Pj4wPm8+Pj4wKXtjb250aW51ZX1icmVha319QltjKzM2fDBdPTA7ZD1EW2MrMjA+PjJdO2E9MDtiPURbYyszMj4+Ml0rN3wwO2E9Yj4+PjA8Nz8xOmE7Zz1hPDwyOXxiPj4+MztiPWcrRFtjKzE2Pj4yXXwwO2E9KGE+Pj4zfDApK2R8MDtEW2MrMTY+PjJdPWI7RFtjKzIwPj4yXT1iPj4+MDxnPj4+MD9hKzF8MDphO2E9MX1iPURbbCszNj4+Ml07aWYoYil7RFtsKzQwPj4yXT1iO21hKGIpfWI9RFtsKzI0Pj4yXTtpZihiKXtEW2wrMjg+PjJdPWI7bWEoYil9Yj1EW2wrOD4+Ml07aWYoYil7RFtsKzEyPj4yXT1iO21hKGIpfSQ9bC0gLTY0fDA7cmV0dXJuIGE7Y2FzZSAxOmJyZWFrIGI7ZGVmYXVsdDpicmVhayBhfX1iPTA7Zz1EW2MrMjA+PjJdO2U9RFtjKzEyPj4yXTtmPURbYysxNj4+Ml07Zzp7aWYoKGd8MCk+PShlfDApJmY+Pj4wPj1HW2MrOD4+Ml18KGV8MCk8KGd8MCkpe2JyZWFrIGd9ZT1FW2YrRFtjPj4yXXwwXTtmPWYrMXwwO2c9Zj9nOmcrMXwwO0RbYysxNj4+Ml09ZjtEW2MrMjA+PjJdPWc7aDp7c3dpdGNoKGUtMXwwKXtjYXNlIDQ6Zj0kKy02NHwwOyQ9ZjtEW2YrNTY+PjJdPTA7RFtmKzQ4Pj4yXT0wO0RbZis1Mj4+Ml09MDtEW2YrNDA+PjJdPTA7RFtmKzQ0Pj4yXT0wO0RbZiszMj4+Ml09MDtEW2YrMzY+PjJdPTA7RFtmKzI0Pj4yXT0wO0RbZisyOD4+Ml09MDtEW2YrMTY+PjJdPTA7RFtmKzIwPj4yXT0wO0RbZis4Pj4yXT0wO0RbZisxMj4+Ml09MDtpOntpZighaWMoZis4fDAsYyl8KERbZisyMD4+Ml0/MDphKSl7YnJlYWsgaX1pZighSGQoZis4fDAsYykpe2JyZWFrIGl9aWYoIWEpe2I9MTticmVhayBpfWM9RFtmKzM2Pj4yXTtnPURbZis0OD4+Ml07aj1EW2YrMjQ+PjJdO2g9RFtmKzU2Pj4yXTt3aGlsZSgxKXtqOntpZihoPj4+MD4xNjM4Myl7YnJlYWsgan1iPURbZis1Mj4+Ml07d2hpbGUoMSl7aWYoKGJ8MCk8PTApe2JyZWFrIGp9Yj1iLTF8MDtEW2YrNTI+PjJdPWI7aD1FW2IrZ3wwXXxoPDw4O0RbZis1Nj4+Ml09aDtpZihoPj4+MDwxNjM4NCl7Y29udGludWV9YnJlYWt9fWI9aCY0MDk1O249RFtqKyhiPDwyKT4+Ml07ZT1jKyhuPDwzKXwwO2g9KEooRFtlPj4yXSxoPj4+MTJ8MCkrYnwwKS1EW2UrND4+Ml18MDtEW2YrNTY+PjJdPWg7RFsobzw8MikrZD4+Ml09bjtiPTE7bz1vKzF8MDtpZigob3wwKSE9KGF8MCkpe2NvbnRpbnVlfWJyZWFrfX1hPURbZiszNj4+Ml07aWYoYSl7RFtmKzQwPj4yXT1hO21hKGEpfWE9RFtmKzI0Pj4yXTtpZihhKXtEW2YrMjg+PjJdPWE7bWEoYSl9YT1EW2YrOD4+Ml07aWYoYSl7RFtmKzEyPj4yXT1hO21hKGEpfSQ9Zi0gLTY0fDA7YnJlYWsgZztjYXNlIDg6bD1hO2U9JCstNjR8MDskPWU7RFtlKzQ4Pj4yXT0wO0RbZSs0MD4+Ml09MDtEW2UrNDQ+PjJdPTA7RFtlKzMyPj4yXT0wO0RbZSszNj4+Ml09MDtEW2UrMjQ+PjJdPTA7RFtlKzI4Pj4yXT0wO0RbZSsxNj4+Ml09MDtEW2UrMjA+PjJdPTA7RFtlKzg+PjJdPTA7RFtlKzEyPj4yXT0wO0RbZT4+Ml09MDtEW2UrND4+Ml09MDtrOntsOntpZighRltjKzM4Pj4xXSl7YnJlYWsgbH1pZighU2EoMSxlKzEyfDAsYykpe2JyZWFrIGx9aT1EW2UrMTI+PjJdO2I9RFtlPj4yXTthPURbZSs0Pj4yXS1iPj4yO206e2lmKGk+Pj4wPmE+Pj4wKXtzYShlLGktYXwwKTtpPURbZSsxMj4+Ml07YnJlYWsgbX1pZihhPj4+MDw9aT4+PjApe2JyZWFrIG19RFtlKzQ+PjJdPWIrKGk8PDIpfWE9MTtpZighaSl7YnJlYWsga31xPURbYys4Pj4yXTtvPURbYysxMj4+Ml07cD1EW2U+PjJdO2Y9MDt3aGlsZSgxKXtnPURbYysyMD4+Ml07Yj1EW2MrMTY+PjJdO2E9MDtpZigob3wwKTw9KGd8MCkmcT4+PjA8PWI+Pj4wfChnfDApPihvfDApKXticmVhayBrfXI9RFtjPj4yXTttPUVbcitifDBdO2E9YisxfDA7Zz1hP2c6ZysxfDA7az1hO0RbYysxNj4+Ml09YTthPWc7RFtjKzIwPj4yXT1hO2I9bT4+PjJ8MDtoPTA7bjp7bzp7cDp7cTp7Zz1tJjM7c3dpdGNoKGd8MCl7Y2FzZSAwOmJyZWFrIG87Y2FzZSAzOmJyZWFrIHE7ZGVmYXVsdDpicmVhayBwfX1iPWIrZnwwO2E9MDtpZihiPj4+MD49aT4+PjApe2JyZWFrIGt9cGEocCsoZjw8Mil8MCwwLChtJjI1MikrNHwwKTtmPWI7YnJlYWsgbn13aGlsZSgxKXtpZigoYXwwKT49KG98MCkmaz4+PjA+PXE+Pj4wfChhfDApPihvfDApKXticmVhayBsfWk9RVtrK3J8MF07az1rKzF8MDthPWs/YTphKzF8MDtEW2MrMTY+PjJdPWs7RFtjKzIwPj4yXT1hO2I9aTw8KGg8PDN8Nil8YjtoPWgrMXwwO2lmKChnfDApIT0oaHwwKSl7Y29udGludWV9YnJlYWt9fURbcCsoZjw8Mik+PjJdPWJ9Zj1mKzF8MDtpPURbZSsxMj4+Ml07aWYoZj4+PjA8aT4+PjApe2NvbnRpbnVlfWJyZWFrfW89ZSsxNnwwO3E9RFtlPj4yXTthPURbZSsxNj4+Ml07Yj1EW2UrMjA+PjJdLWF8MDtmPWI+PjI7cjp7aWYoZj4+PjA8PTgxOTEpe3NhKG8sODE5Mi1mfDApO2JyZWFrIHJ9aWYoKGJ8MCk9PTMyNzY4KXticmVhayByfURbZSsyMD4+Ml09YSszMjc2OH1hPWUrMjh8MDtmPURbYT4+Ml07Yj1EW2UrMzI+PjJdLWY+PjM7czp7aWYoYj4+PjA8aT4+PjApe2JiKGEsaS1ifDApO2Y9RFthPj4yXTticmVhayBzfWlmKGI+Pj4wPmk+Pj4wKXtEW2UrMzI+PjJdPShpPDwzKStmfWlmKCFpKXticmVhayBsfX1oPTA7YT0wO3doaWxlKDEpe2c9cSsoaDw8Mil8MDtrPURbZz4+Ml07Yj1hO209KGg8PDMpK2Z8MDtEW20rND4+Ml09YTtEW20+PjJdPWs7Zz1EW2c+PjJdO2E9ZythfDA7aWYoYT4+PjA+ODE5Mil7YnJlYWsgbH10OntpZihhPj4+MDw9Yj4+PjApe2JyZWFrIHR9bT1EW28+PjJdO2s9MDtwPWcmNztpZihwKXt3aGlsZSgxKXtEW20rKGI8PDIpPj4yXT1oO2I9YisxfDA7az1rKzF8MDtpZigocHwwKSE9KGt8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZihnLTE+Pj4wPD02KXticmVhayB0fXdoaWxlKDEpe2c9bSsoYjw8Mil8MDtEW2c+PjJdPWg7RFtnKzI4Pj4yXT1oO0RbZysyND4+Ml09aDtEW2crMjA+PjJdPWg7RFtnKzE2Pj4yXT1oO0RbZysxMj4+Ml09aDtEW2crOD4+Ml09aDtEW2crND4+Ml09aDtiPWIrOHwwO2lmKChifDApIT0oYXwwKSl7Y29udGludWV9YnJlYWt9fWg9aCsxfDA7aWYoKGl8MCkhPShofDApKXtjb250aW51ZX1icmVha31uPShhfDApPT04MTkyfWE9bn11OntpZighYXwoRFtlKzEyPj4yXT8wOmwpKXticmVhayB1fWlmKCFPYSgxLGUrNTZ8MCxjKSl7YnJlYWsgdX1hPURbYys4Pj4yXTtiPURbYysxNj4+Ml07Zj1hLWJ8MDtnPURbZSs2MD4+Ml07bj1EW2MrMjA+PjJdO2g9RFtjKzEyPj4yXS0obisoYT4+PjA8Yj4+PjApfDApfDA7YT1EW2UrNTY+PjJdO2lmKChnfDApPT0oaHwwKSZmPj4+MDxhPj4+MHxnPj4+MD5oPj4+MCl7YnJlYWsgdX1nPWcrbnwwO2Y9YStifDA7Zz1mPj4+MDxhPj4+MD9nKzF8MDpnO0RbYysxNj4+Ml09ZjtEW2MrMjA+PjJdPWc7aWYoKGF8MCk8PTApe2JyZWFrIHV9Yz1iK0RbYz4+Ml18MDtEW2UrNDA+PjJdPWM7Yj1hLTF8MDtmPWMrYnwwO2c9RVtmfDBdO3Y6e2lmKGc+Pj4wPD02Myl7RFtlKzQ0Pj4yXT1iO2E9RVtmfDBdJjYzO2JyZWFrIHZ9dzp7c3dpdGNoKChnPj4+NnwwKS0xfDApe2Nhc2UgMDppZihhPj4+MDwyKXticmVhayB1fWI9YS0yfDA7RFtlKzQ0Pj4yXT1iO2E9KGErY3wwKS0yfDA7YT1FW2ErMXwwXTw8OCYxNjEyOHxFW2F8MF07YnJlYWsgdjtjYXNlIDE6aWYoYT4+PjA8Myl7YnJlYWsgdX1iPWEtM3wwO0RbZSs0ND4+Ml09YjthPShhK2N8MCktM3wwO2E9RVthKzJ8MF08PDE2JjQxMjg3Njh8RVthKzF8MF08PDh8RVthfDBdO2JyZWFrIHY7ZGVmYXVsdDpicmVhayB3fX1iPWEtNHwwO0RbZSs0ND4+Ml09YjthPShhK2N8MCktNHwwO2E9RVthKzJ8MF08PDE2fEVbYSszfDBdPDwyNCYxMDU2OTY0NjA4fEVbYSsxfDBdPDw4fEVbYXwwXX1mPWErMzI3Njh8MDtEW2UrNDg+PjJdPWY7aWYoZj4+PjA+ODM4ODYwNyl7YnJlYWsgdX1pZighbCl7aj0xO2JyZWFrIHV9Zz1EW2UrMjg+PjJdO2E9MDtuPURbZSsxNj4+Ml07d2hpbGUoMSl7eDp7aWYoZj4+PjA+MzI3Njcpe2JyZWFrIHh9d2hpbGUoMSl7aWYoKGJ8MCk8PTApe2JyZWFrIHh9Yj1iLTF8MDtEW2UrNDQ+PjJdPWI7Zj1FW2IrY3wwXXxmPDw4O0RbZSs0OD4+Ml09ZjtpZihmPj4+MDwzMjc2OCl7Y29udGludWV9YnJlYWt9fWo9ZiY4MTkxO2g9RFtuKyhqPDwyKT4+Ml07aT1nKyhoPDwzKXwwO2Y9KEooRFtpPj4yXSxmPj4+MTN8MCkranwwKS1EW2krND4+Ml18MDtEW2UrNDg+PjJdPWY7RFsoYTw8MikrZD4+Ml09aDtqPTE7YT1hKzF8MDtpZigobHwwKSE9KGF8MCkpe2NvbnRpbnVlfWJyZWFrfX1hPURbZSsyOD4+Ml07aWYoYSl7RFtlKzMyPj4yXT1hO21hKGEpfWE9RFtlKzE2Pj4yXTtpZihhKXtEW2UrMjA+PjJdPWE7bWEoYSl9YT1EW2U+PjJdO2lmKGEpe0RbZSs0Pj4yXT1hO21hKGEpfSQ9ZS0gLTY0fDA7Yj1qO2JyZWFrIGc7Y2FzZSA5Omw9YTtlPSQrLTY0fDA7JD1lO0RbZSs0OD4+Ml09MDtEW2UrNDA+PjJdPTA7RFtlKzQ0Pj4yXT0wO0RbZSszMj4+Ml09MDtEW2UrMzY+PjJdPTA7RFtlKzI0Pj4yXT0wO0RbZSsyOD4+Ml09MDtEW2UrMTY+PjJdPTA7RFtlKzIwPj4yXT0wO0RbZSs4Pj4yXT0wO0RbZSsxMj4+Ml09MDtEW2U+PjJdPTA7RFtlKzQ+PjJdPTA7eTp7ejp7aWYoIUZbYyszOD4+MV0pe2JyZWFrIHp9aWYoIVNhKDEsZSsxMnwwLGMpKXticmVhayB6fWk9RFtlKzEyPj4yXTtiPURbZT4+Ml07YT1EW2UrND4+Ml0tYj4+MjtBOntpZihpPj4+MD5hPj4+MCl7c2EoZSxpLWF8MCk7aT1EW2UrMTI+PjJdO2JyZWFrIEF9aWYoYT4+PjA8PWk+Pj4wKXticmVhayBBfURbZSs0Pj4yXT1iKyhpPDwyKX1hPTE7aWYoIWkpe2JyZWFrIHl9bT1EW2MrOD4+Ml07bz1EW2MrMTI+PjJdO3E9RFtlPj4yXTtmPTA7d2hpbGUoMSl7Yj1EW2MrMjA+PjJdO2g9RFtjKzE2Pj4yXTthPTA7aWYoKG98MCk8PShifDApJm0+Pj4wPD1oPj4+MHwoYnwwKT4ob3wwKSl7YnJlYWsgeX1wPURbYz4+Ml07Zz1FW3AraHwwXTthPWI7Yj1oKzF8MDthPWI/YTphKzF8MDtrPWI7RFtjKzE2Pj4yXT1iO0RbYysyMD4+Ml09YTtiPWc+Pj4yfDA7aD0wO0I6e0M6e0Q6e0U6e3I9ZyYzO3N3aXRjaChyfDApe2Nhc2UgMDpicmVhayBDO2Nhc2UgMzpicmVhayBFO2RlZmF1bHQ6YnJlYWsgRH19Yj1iK2Z8MDthPTA7aWYoYj4+PjA+PWk+Pj4wKXticmVhayB5fXBhKHErKGY8PDIpfDAsMCwoZyYyNTIpKzR8MCk7Zj1iO2JyZWFrIEJ9d2hpbGUoMSl7aWYoKGF8MCk+PShvfDApJms+Pj4wPj1tPj4+MHwoYXwwKT4ob3wwKSl7YnJlYWsgen1pPUVbaytwfDBdO2c9YTthPWsrMXwwO2c9YT9nOmcrMXwwO2s9YTtEW2MrMTY+PjJdPWE7YT1nO0RbYysyMD4+Ml09YTtiPWk8PChoPDwzfDYpfGI7aD1oKzF8MDtpZigocnwwKSE9KGh8MCkpe2NvbnRpbnVlfWJyZWFrfX1EW3ErKGY8PDIpPj4yXT1ifWY9ZisxfDA7aT1EW2UrMTI+PjJdO2lmKGY+Pj4wPGk+Pj4wKXtjb250aW51ZX1icmVha31vPWUrMTZ8MDtxPURbZT4+Ml07YT1EW2UrMTY+PjJdO2I9RFtlKzIwPj4yXS1hfDA7Zj1iPj4yO0Y6e2lmKGY+Pj4wPD0zMjc2Nyl7c2EobywzMjc2OC1mfDApO2JyZWFrIEZ9aWYoKGJ8MCk9PTEzMTA3Mil7YnJlYWsgRn1EW2UrMjA+PjJdPWErMTMxMDcyfWE9ZSsyOHwwO2Y9RFthPj4yXTtiPURbZSszMj4+Ml0tZj4+MztHOntpZihiPj4+MDxpPj4+MCl7YmIoYSxpLWJ8MCk7Zj1EW2E+PjJdO2JyZWFrIEd9aWYoYj4+PjA+aT4+PjApe0RbZSszMj4+Ml09KGk8PDMpK2Z9aWYoIWkpe2JyZWFrIHp9fWg9MDthPTA7d2hpbGUoMSl7Zz1xKyhoPDwyKXwwO2s9RFtnPj4yXTtiPWE7bT0oaDw8MykrZnwwO0RbbSs0Pj4yXT1hO0RbbT4+Ml09aztnPURbZz4+Ml07YT1nK2F8MDtpZihhPj4+MD4zMjc2OCl7YnJlYWsgen1IOntpZihhPj4+MDw9Yj4+PjApe2JyZWFrIEh9bT1EW28+PjJdO2s9MDtwPWcmNztpZihwKXt3aGlsZSgxKXtEW20rKGI8PDIpPj4yXT1oO2I9YisxfDA7az1rKzF8MDtpZigocHwwKSE9KGt8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZihnLTE+Pj4wPD02KXticmVhayBIfXdoaWxlKDEpe2c9bSsoYjw8Mil8MDtEW2c+PjJdPWg7RFtnKzI4Pj4yXT1oO0RbZysyND4+Ml09aDtEW2crMjA+PjJdPWg7RFtnKzE2Pj4yXT1oO0RbZysxMj4+Ml09aDtEW2crOD4+Ml09aDtEW2crND4+Ml09aDtiPWIrOHwwO2lmKChifDApIT0oYXwwKSl7Y29udGludWV9YnJlYWt9fWg9aCsxfDA7aWYoKGl8MCkhPShofDApKXtjb250aW51ZX1icmVha31uPShhfDApPT0zMjc2OH1hPW59STp7aWYoIWF8KERbZSsxMj4+Ml0/MDpsKSl7YnJlYWsgSX1pZighT2EoMSxlKzU2fDAsYykpe2JyZWFrIEl9YT1EW2MrOD4+Ml07Zj1EW2MrMTY+PjJdO2I9ZjtnPWEtYnwwO249RFtlKzYwPj4yXTtoPURbYysyMD4+Ml07YT1EW2MrMTI+PjJdLShoKyhhPj4+MDxiPj4+MCl8MCl8MDtiPURbZSs1Nj4+Ml07aWYoKG58MCk9PShhfDApJmc+Pj4wPGI+Pj4wfGE+Pj4wPG4+Pj4wKXticmVhayBJfWE9aCtufDA7Zz1iK2Z8MDthPWc+Pj4wPGI+Pj4wP2ErMXwwOmE7RFtjKzE2Pj4yXT1nO0RbYysyMD4+Ml09YTthPWI7aWYoKGF8MCk8PTApe2JyZWFrIEl9Yz1mK0RbYz4+Ml18MDtEW2UrNDA+PjJdPWM7Yj1hLTF8MDtmPWMrYnwwO2c9RVtmfDBdO0o6e2lmKGc+Pj4wPD02Myl7RFtlKzQ0Pj4yXT1iO2E9RVtmfDBdJjYzO2JyZWFrIEp9Szp7c3dpdGNoKChnPj4+NnwwKS0xfDApe2Nhc2UgMDppZihhPj4+MDwyKXticmVhayBJfWI9YS0yfDA7RFtlKzQ0Pj4yXT1iO2E9KGErY3wwKS0yfDA7YT1FW2ErMXwwXTw8OCYxNjEyOHxFW2F8MF07YnJlYWsgSjtjYXNlIDE6aWYoYT4+PjA8Myl7YnJlYWsgSX1iPWEtM3wwO0RbZSs0ND4+Ml09YjthPShhK2N8MCktM3wwO2E9RVthKzJ8MF08PDE2JjQxMjg3Njh8RVthKzF8MF08PDh8RVthfDBdO2JyZWFrIEo7ZGVmYXVsdDpicmVhayBLfX1iPWEtNHwwO0RbZSs0ND4+Ml09YjthPShhK2N8MCktNHwwO2E9RVthKzJ8MF08PDE2fEVbYSszfDBdPDwyNCYxMDU2OTY0NjA4fEVbYSsxfDBdPDw4fEVbYXwwXX1mPWErMTMxMDcyfDA7RFtlKzQ4Pj4yXT1mO2lmKGY+Pj4wPjMzNTU0NDMxKXticmVhayBJfWlmKCFsKXtqPTE7YnJlYWsgSX1nPURbZSsyOD4+Ml07YT0wO249RFtlKzE2Pj4yXTt3aGlsZSgxKXtMOntpZihmPj4+MD4xMzEwNzEpe2JyZWFrIEx9d2hpbGUoMSl7aWYoKGJ8MCk8PTApe2JyZWFrIEx9Yj1iLTF8MDtEW2UrNDQ+PjJdPWI7Zj1FW2IrY3wwXXxmPDw4O0RbZSs0OD4+Ml09ZjtpZihmPj4+MDwxMzEwNzIpe2NvbnRpbnVlfWJyZWFrfX1qPWYmMzI3Njc7aD1EW24rKGo8PDIpPj4yXTtpPWcrKGg8PDMpfDA7Zj0oSihEW2k+PjJdLGY+Pj4xNXwwKStqfDApLURbaSs0Pj4yXXwwO0RbZSs0OD4+Ml09ZjtEWyhhPDwyKStkPj4yXT1oO2o9MTthPWErMXwwO2lmKChsfDApIT0oYXwwKSl7Y29udGludWV9YnJlYWt9fWE9RFtlKzI4Pj4yXTtpZihhKXtEW2UrMzI+PjJdPWE7bWEoYSl9YT1EW2UrMTY+PjJdO2lmKGEpe0RbZSsyMD4+Ml09YTttYShhKX1hPURbZT4+Ml07aWYoYSl7RFtlKzQ+PjJdPWE7bWEoYSl9JD1lLSAtNjR8MDtiPWo7YnJlYWsgZztjYXNlIDEwOmw9YTtlPSQrLTY0fDA7JD1lO0RbZSs0OD4+Ml09MDtEW2UrNDA+PjJdPTA7RFtlKzQ0Pj4yXT0wO0RbZSszMj4+Ml09MDtEW2UrMzY+PjJdPTA7RFtlKzI0Pj4yXT0wO0RbZSsyOD4+Ml09MDtEW2UrMTY+PjJdPTA7RFtlKzIwPj4yXT0wO0RbZSs4Pj4yXT0wO0RbZSsxMj4+Ml09MDtEW2U+PjJdPTA7RFtlKzQ+PjJdPTA7TTp7Tjp7aWYoIUZbYyszOD4+MV0pe2JyZWFrIE59aWYoIVNhKDEsZSsxMnwwLGMpKXticmVhayBOfWk9RFtlKzEyPj4yXTtiPURbZT4+Ml07YT1EW2UrND4+Ml0tYj4+MjtPOntpZihpPj4+MD5hPj4+MCl7c2EoZSxpLWF8MCk7aT1EW2UrMTI+PjJdO2JyZWFrIE99aWYoYT4+PjA8PWk+Pj4wKXticmVhayBPfURbZSs0Pj4yXT1iKyhpPDwyKX1hPTE7aWYoIWkpe2JyZWFrIE19cT1EW2MrOD4+Ml07bz1EW2MrMTI+PjJdO3A9RFtlPj4yXTtmPTA7d2hpbGUoMSl7Zz1EW2MrMjA+PjJdO2I9RFtjKzE2Pj4yXTthPTA7aWYoKG98MCk8PShnfDApJnE+Pj4wPD1iPj4+MHwoZ3wwKT4ob3wwKSl7YnJlYWsgTX1yPURbYz4+Ml07bT1FW3IrYnwwXTthPWIrMXwwO2c9YT9nOmcrMXwwO2s9YTtEW2MrMTY+PjJdPWE7YT1nO0RbYysyMD4+Ml09YTtiPW0+Pj4yfDA7aD0wO1A6e1E6e1I6e1M6e3M9bSYzO3N3aXRjaChzfDApe2Nhc2UgMDpicmVhayBRO2Nhc2UgMzpicmVhayBTO2RlZmF1bHQ6YnJlYWsgUn19Yj1iK2Z8MDthPTA7aWYoYj4+PjA+PWk+Pj4wKXticmVhayBNfXBhKHArKGY8PDIpfDAsMCwobSYyNTIpKzR8MCk7Zj1iO2JyZWFrIFB9d2hpbGUoMSl7aWYoKGF8MCk+PShvfDApJms+Pj4wPj1xPj4+MHwoYXwwKT4ob3wwKSl7YnJlYWsgTn1pPUVbaytyfDBdO2c9YTthPWsrMXwwO2c9YT9nOmcrMXwwO2s9YTtEW2MrMTY+PjJdPWE7YT1nO0RbYysyMD4+Ml09YTtiPWk8PChoPDwzfDYpfGI7aD1oKzF8MDtpZigoc3wwKSE9KGh8MCkpe2NvbnRpbnVlfWJyZWFrfX1EW3ArKGY8PDIpPj4yXT1ifWY9ZisxfDA7aT1EW2UrMTI+PjJdO2lmKGY+Pj4wPGk+Pj4wKXtjb250aW51ZX1icmVha31vPWUrMTZ8MDtxPURbZT4+Ml07YT1EW2UrMTY+PjJdO2I9RFtlKzIwPj4yXS1hfDA7Zj1iPj4yO1Q6e2lmKGY+Pj4wPD02NTUzNSl7c2Eobyw2NTUzNi1mfDApO2JyZWFrIFR9aWYoKGJ8MCk9PTI2MjE0NCl7YnJlYWsgVH1EW2UrMjA+PjJdPWErMjYyMTQ0fWE9ZSsyOHwwO2Y9RFthPj4yXTtiPURbZSszMj4+Ml0tZj4+MztVOntpZihiPj4+MDxpPj4+MCl7YmIoYSxpLWJ8MCk7Zj1EW2E+PjJdO2JyZWFrIFV9aWYoYj4+PjA+aT4+PjApe0RbZSszMj4+Ml09KGk8PDMpK2Z9aWYoIWkpe2JyZWFrIE59fWg9MDthPTA7d2hpbGUoMSl7Zz1xKyhoPDwyKXwwO2s9RFtnPj4yXTtiPWE7bT0oaDw8MykrZnwwO0RbbSs0Pj4yXT1hO0RbbT4+Ml09aztnPURbZz4+Ml07YT1nK2F8MDtpZihhPj4+MD42NTUzNil7YnJlYWsgTn1WOntpZihhPj4+MDw9Yj4+PjApe2JyZWFrIFZ9bT1EW28+PjJdO2s9MDtwPWcmNztpZihwKXt3aGlsZSgxKXtEW20rKGI8PDIpPj4yXT1oO2I9YisxfDA7az1rKzF8MDtpZigocHwwKSE9KGt8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZihnLTE+Pj4wPD02KXticmVhayBWfXdoaWxlKDEpe2c9bSsoYjw8Mil8MDtEW2c+PjJdPWg7RFtnKzI4Pj4yXT1oO0RbZysyND4+Ml09aDtEW2crMjA+PjJdPWg7RFtnKzE2Pj4yXT1oO0RbZysxMj4+Ml09aDtEW2crOD4+Ml09aDtEW2crND4+Ml09aDtiPWIrOHwwO2lmKChifDApIT0oYXwwKSl7Y29udGludWV9YnJlYWt9fWg9aCsxfDA7aWYoKGl8MCkhPShofDApKXtjb250aW51ZX1icmVha31uPShhfDApPT02NTUzNn1hPW59Vzp7aWYoIWF8KERbZSsxMj4+Ml0/MDpsKSl7YnJlYWsgV31pZighT2EoMSxlKzU2fDAsYykpe2JyZWFrIFd9YT1EW2MrOD4+Ml07Yj1EW2MrMTY+PjJdO2Y9YS1ifDA7Zz1EW2UrNjA+PjJdO249RFtjKzIwPj4yXTtoPURbYysxMj4+Ml0tKG4rKGE+Pj4wPGI+Pj4wKXwwKXwwO2E9RFtlKzU2Pj4yXTtpZigoZ3wwKT09KGh8MCkmZj4+PjA8YT4+PjB8Zz4+PjA+aD4+PjApe2JyZWFrIFd9Zz1nK258MDtmPWErYnwwO2c9Zj4+PjA8YT4+PjA/ZysxfDA6ZztEW2MrMTY+PjJdPWY7RFtjKzIwPj4yXT1nO2lmKChhfDApPD0wKXticmVhayBXfWM9YitEW2M+PjJdfDA7RFtlKzQwPj4yXT1jO2I9YS0xfDA7Zj1jK2J8MDtnPUVbZnwwXTtYOntpZihnPj4+MDw9NjMpe0RbZSs0ND4+Ml09YjthPUVbZnwwXSY2MzticmVhayBYfVk6e3N3aXRjaCgoZz4+PjZ8MCktMXwwKXtjYXNlIDA6aWYoYT4+PjA8Mil7YnJlYWsgV31iPWEtMnwwO0RbZSs0ND4+Ml09YjthPShhK2N8MCktMnwwO2E9RVthKzF8MF08PDgmMTYxMjh8RVthfDBdO2JyZWFrIFg7Y2FzZSAxOmlmKGE+Pj4wPDMpe2JyZWFrIFd9Yj1hLTN8MDtEW2UrNDQ+PjJdPWI7YT0oYStjfDApLTN8MDthPUVbYSsyfDBdPDwxNiY0MTI4NzY4fEVbYSsxfDBdPDw4fEVbYXwwXTticmVhayBYO2RlZmF1bHQ6YnJlYWsgWX19Yj1hLTR8MDtEW2UrNDQ+PjJdPWI7YT0oYStjfDApLTR8MDthPUVbYSsyfDBdPDwxNnxFW2ErM3wwXTw8MjQmMTA1Njk2NDYwOHxFW2ErMXwwXTw8OHxFW2F8MF19Zj1hKzI2MjE0NHwwO0RbZSs0OD4+Ml09ZjtpZihmPj4+MD42NzEwODg2Myl7YnJlYWsgV31pZighbCl7aj0xO2JyZWFrIFd9Zz1EW2UrMjg+PjJdO2E9MDtuPURbZSsxNj4+Ml07d2hpbGUoMSl7Wjp7aWYoZj4+PjA+MjYyMTQzKXticmVhayBafXdoaWxlKDEpe2lmKChifDApPD0wKXticmVhayBafWI9Yi0xfDA7RFtlKzQ0Pj4yXT1iO2Y9RVtiK2N8MF18Zjw8ODtEW2UrNDg+PjJdPWY7aWYoZj4+PjA8MjYyMTQ0KXtjb250aW51ZX1icmVha319aj1mJjY1NTM1O2g9RFtuKyhqPDwyKT4+Ml07aT1nKyhoPDwzKXwwO2Y9KEooRFtpPj4yXSxmPj4+MTZ8MCkranwwKS1EW2krND4+Ml18MDtEW2UrNDg+PjJdPWY7RFsoYTw8MikrZD4+Ml09aDtqPTE7YT1hKzF8MDtpZigobHwwKSE9KGF8MCkpe2NvbnRpbnVlfWJyZWFrfX1hPURbZSsyOD4+Ml07aWYoYSl7RFtlKzMyPj4yXT1hO21hKGEpfWE9RFtlKzE2Pj4yXTtpZihhKXtEW2UrMjA+PjJdPWE7bWEoYSl9YT1EW2U+PjJdO2lmKGEpe0RbZSs0Pj4yXT1hO21hKGEpfSQ9ZS0gLTY0fDA7Yj1qO2JyZWFrIGc7Y2FzZSAxMTpsPWE7ZT0kKy02NHwwOyQ9ZTtEW2UrNDg+PjJdPTA7RFtlKzQwPj4yXT0wO0RbZSs0ND4+Ml09MDtEW2UrMzI+PjJdPTA7RFtlKzM2Pj4yXT0wO0RbZSsyND4+Ml09MDtEW2UrMjg+PjJdPTA7RFtlKzE2Pj4yXT0wO0RbZSsyMD4+Ml09MDtEW2UrOD4+Ml09MDtEW2UrMTI+PjJdPTA7RFtlPj4yXT0wO0RbZSs0Pj4yXT0wO186eyQ6e2lmKCFGW2MrMzg+PjFdKXticmVhayAkfWlmKCFTYSgxLGUrMTJ8MCxjKSl7YnJlYWsgJH1pPURbZSsxMj4+Ml07Yj1EW2U+PjJdO2E9RFtlKzQ+PjJdLWI+PjI7YWE6e2lmKGk+Pj4wPmE+Pj4wKXtzYShlLGktYXwwKTtpPURbZSsxMj4+Ml07YnJlYWsgYWF9aWYoYT4+PjA8PWk+Pj4wKXticmVhayBhYX1EW2UrND4+Ml09YisoaTw8Mil9YT0xO2lmKCFpKXticmVhayBffXE9RFtjKzg+PjJdO289RFtjKzEyPj4yXTtwPURbZT4+Ml07Zj0wO3doaWxlKDEpe2c9RFtjKzIwPj4yXTtiPURbYysxNj4+Ml07YT0wO2lmKChvfDApPD0oZ3wwKSZxPj4+MDw9Yj4+PjB8KGd8MCk+KG98MCkpe2JyZWFrIF99cj1EW2M+PjJdO209RVtyK2J8MF07YT1iKzF8MDtnPWE/ZzpnKzF8MDtrPWE7RFtjKzE2Pj4yXT1hO2E9ZztEW2MrMjA+PjJdPWE7Yj1tPj4+MnwwO2g9MDtiYTp7Y2E6e2RhOntlYTp7Zz1tJjM7c3dpdGNoKGd8MCl7Y2FzZSAwOmJyZWFrIGNhO2Nhc2UgMzpicmVhayBlYTtkZWZhdWx0OmJyZWFrIGRhfX1iPWIrZnwwO2E9MDtpZihiPj4+MD49aT4+PjApe2JyZWFrIF99cGEocCsoZjw8Mil8MCwwLChtJjI1MikrNHwwKTtmPWI7YnJlYWsgYmF9d2hpbGUoMSl7aWYoKGF8MCk+PShvfDApJms+Pj4wPj1xPj4+MHwoYXwwKT4ob3wwKSl7YnJlYWsgJH1pPUVbaytyfDBdO2s9aysxfDA7YT1rP2E6YSsxfDA7RFtjKzE2Pj4yXT1rO0RbYysyMD4+Ml09YTtiPWk8PChoPDwzfDYpfGI7aD1oKzF8MDtpZigoZ3wwKSE9KGh8MCkpe2NvbnRpbnVlfWJyZWFrfX1EW3ArKGY8PDIpPj4yXT1ifWY9ZisxfDA7aT1EW2UrMTI+PjJdO2lmKGY+Pj4wPGk+Pj4wKXtjb250aW51ZX1icmVha31vPWUrMTZ8MDtxPURbZT4+Ml07YT1EW2UrMTY+PjJdO2I9RFtlKzIwPj4yXS1hfDA7Zj1iPj4yO2ZhOntpZihmPj4+MDw9MjYyMTQzKXtzYShvLDI2MjE0NC1mfDApO2JyZWFrIGZhfWlmKChifDApPT0xMDQ4NTc2KXticmVhayBmYX1EW2UrMjA+PjJdPWEtIC0xMDQ4NTc2fWE9ZSsyOHwwO2Y9RFthPj4yXTtiPURbZSszMj4+Ml0tZj4+MztnYTp7aWYoYj4+PjA8aT4+PjApe2JiKGEsaS1ifDApO2Y9RFthPj4yXTticmVhayBnYX1pZihiPj4+MD5pPj4+MCl7RFtlKzMyPj4yXT0oaTw8MykrZn1pZighaSl7YnJlYWsgJH19aD0wO2E9MDt3aGlsZSgxKXtnPXErKGg8PDIpfDA7az1EW2c+PjJdO2I9YTttPShoPDwzKStmfDA7RFttKzQ+PjJdPWE7RFttPj4yXT1rO2c9RFtnPj4yXTthPWcrYXwwO2lmKGE+Pj4wPjI2MjE0NCl7YnJlYWsgJH1oYTp7aWYoYT4+PjA8PWI+Pj4wKXticmVhayBoYX1tPURbbz4+Ml07az0wO3A9ZyY3O2lmKHApe3doaWxlKDEpe0RbbSsoYjw8Mik+PjJdPWg7Yj1iKzF8MDtrPWsrMXwwO2lmKChwfDApIT0oa3wwKSl7Y29udGludWV9YnJlYWt9fWlmKGctMT4+PjA8PTYpe2JyZWFrIGhhfXdoaWxlKDEpe2c9bSsoYjw8Mil8MDtEW2c+PjJdPWg7RFtnKzI4Pj4yXT1oO0RbZysyND4+Ml09aDtEW2crMjA+PjJdPWg7RFtnKzE2Pj4yXT1oO0RbZysxMj4+Ml09aDtEW2crOD4+Ml09aDtEW2crND4+Ml09aDtiPWIrOHwwO2lmKChifDApIT0oYXwwKSl7Y29udGludWV9YnJlYWt9fWg9aCsxfDA7aWYoKGl8MCkhPShofDApKXtjb250aW51ZX1icmVha31uPShhfDApPT0yNjIxNDR9YT1ufWlhOntpZighYXwoRFtlKzEyPj4yXT8wOmwpKXticmVhayBpYX1pZighT2EoMSxlKzU2fDAsYykpe2JyZWFrIGlhfWE9RFtjKzg+PjJdO2I9RFtjKzE2Pj4yXTtmPWEtYnwwO2c9RFtlKzYwPj4yXTtuPURbYysyMD4+Ml07aD1EW2MrMTI+PjJdLShuKyhhPj4+MDxiPj4+MCl8MCl8MDthPURbZSs1Nj4+Ml07aWYoKGd8MCk9PShofDApJmY+Pj4wPGE+Pj4wfGc+Pj4wPmg+Pj4wKXticmVhayBpYX1nPWcrbnwwO2Y9YStifDA7Zz1mPj4+MDxhPj4+MD9nKzF8MDpnO0RbYysxNj4+Ml09ZjtEW2MrMjA+PjJdPWc7aWYoKGF8MCk8PTApe2JyZWFrIGlhfWM9YitEW2M+PjJdfDA7RFtlKzQwPj4yXT1jO2I9YS0xfDA7Zj1jK2J8MDtnPUVbZnwwXTtqYTp7aWYoZz4+PjA8PTYzKXtEW2UrNDQ+PjJdPWI7YT1FW2Z8MF0mNjM7YnJlYWsgamF9a2E6e3N3aXRjaCgoZz4+PjZ8MCktMXwwKXtjYXNlIDA6aWYoYT4+PjA8Mil7YnJlYWsgaWF9Yj1hLTJ8MDtEW2UrNDQ+PjJdPWI7YT0oYStjfDApLTJ8MDthPUVbYSsxfDBdPDw4JjE2MTI4fEVbYXwwXTticmVhayBqYTtjYXNlIDE6aWYoYT4+PjA8Myl7YnJlYWsgaWF9Yj1hLTN8MDtEW2UrNDQ+PjJdPWI7YT0oYStjfDApLTN8MDthPUVbYSsyfDBdPDwxNiY0MTI4NzY4fEVbYSsxfDBdPDw4fEVbYXwwXTticmVhayBqYTtkZWZhdWx0OmJyZWFrIGthfX1iPWEtNHwwO0RbZSs0ND4+Ml09YjthPShhK2N8MCktNHwwO2E9RVthKzJ8MF08PDE2fEVbYSszfDBdPDwyNCYxMDU2OTY0NjA4fEVbYSsxfDBdPDw4fEVbYXwwXX1mPWEtIC0xMDQ4NTc2fDA7RFtlKzQ4Pj4yXT1mO2lmKGY+Pj4wPjI2ODQzNTQ1NSl7YnJlYWsgaWF9aWYoIWwpe2o9MTticmVhayBpYX1nPURbZSsyOD4+Ml07YT0wO249RFtlKzE2Pj4yXTt3aGlsZSgxKXtsYTp7aWYoZj4+PjA+MTA0ODU3NSl7YnJlYWsgbGF9d2hpbGUoMSl7aWYoKGJ8MCk8PTApe2JyZWFrIGxhfWI9Yi0xfDA7RFtlKzQ0Pj4yXT1iO2Y9RVtiK2N8MF18Zjw8ODtEW2UrNDg+PjJdPWY7aWYoZj4+PjA8MTA0ODU3Nil7Y29udGludWV9YnJlYWt9fWo9ZiYyNjIxNDM7aD1EW24rKGo8PDIpPj4yXTtpPWcrKGg8PDMpfDA7Zj0oSihEW2k+PjJdLGY+Pj4xOHwwKStqfDApLURbaSs0Pj4yXXwwO0RbZSs0OD4+Ml09ZjtEWyhhPDwyKStkPj4yXT1oO2o9MTthPWErMXwwO2lmKChsfDApIT0oYXwwKSl7Y29udGludWV9YnJlYWt9fWE9RFtlKzI4Pj4yXTtpZihhKXtEW2UrMzI+PjJdPWE7bWEoYSl9YT1EW2UrMTY+PjJdO2lmKGEpe0RbZSsyMD4+Ml09YTttYShhKX1hPURbZT4+Ml07aWYoYSl7RFtlKzQ+PjJdPWE7bWEoYSl9JD1lLSAtNjR8MDtiPWo7YnJlYWsgZztjYXNlIDEyOmw9YTtlPSQrLTY0fDA7JD1lO0RbZSs0OD4+Ml09MDtEW2UrNDA+PjJdPTA7RFtlKzQ0Pj4yXT0wO0RbZSszMj4+Ml09MDtEW2UrMzY+PjJdPTA7RFtlKzI0Pj4yXT0wO0RbZSsyOD4+Ml09MDtEW2UrMTY+PjJdPTA7RFtlKzIwPj4yXT0wO0RbZSs4Pj4yXT0wO0RbZSsxMj4+Ml09MDtEW2U+PjJdPTA7RFtlKzQ+PjJdPTA7bWE6e25hOntpZighRltjKzM4Pj4xXSl7YnJlYWsgbmF9aWYoIVNhKDEsZSsxMnwwLGMpKXticmVhayBuYX1pPURbZSsxMj4+Ml07Yj1EW2U+PjJdO2E9RFtlKzQ+PjJdLWI+PjI7b2E6e2lmKGk+Pj4wPmE+Pj4wKXtzYShlLGktYXwwKTtpPURbZSsxMj4+Ml07YnJlYWsgb2F9aWYoYT4+PjA8PWk+Pj4wKXticmVhayBvYX1EW2UrND4+Ml09YisoaTw8Mil9YT0xO2lmKCFpKXticmVhayBtYX1tPURbYys4Pj4yXTtvPURbYysxMj4+Ml07cT1EW2U+PjJdO2Y9MDt3aGlsZSgxKXtiPURbYysyMD4+Ml07aD1EW2MrMTY+PjJdO2E9MDtpZigob3wwKTw9KGJ8MCkmbT4+PjA8PWg+Pj4wfChifDApPihvfDApKXticmVhayBtYX1wPURbYz4+Ml07Zz1FW3AraHwwXTthPWI7Yj1oKzF8MDthPWI/YTphKzF8MDtrPWI7RFtjKzE2Pj4yXT1iO0RbYysyMD4+Ml09YTtiPWc+Pj4yfDA7aD0wO3BhOntxYTp7cmE6e3NhOntyPWcmMztzd2l0Y2gocnwwKXtjYXNlIDA6YnJlYWsgcWE7Y2FzZSAzOmJyZWFrIHNhO2RlZmF1bHQ6YnJlYWsgcmF9fWI9YitmfDA7YT0wO2lmKGI+Pj4wPj1pPj4+MCl7YnJlYWsgbWF9cGEocSsoZjw8Mil8MCwwLChnJjI1MikrNHwwKTtmPWI7YnJlYWsgcGF9d2hpbGUoMSl7aWYoKGF8MCk+PShvfDApJms+Pj4wPj1tPj4+MHwoYXwwKT4ob3wwKSl7YnJlYWsgbmF9aT1FW2srcHwwXTtnPWE7YT1rKzF8MDtnPWE/ZzpnKzF8MDtrPWE7RFtjKzE2Pj4yXT1hO2E9ZztEW2MrMjA+PjJdPWE7Yj1pPDwoaDw8M3w2KXxiO2g9aCsxfDA7aWYoKHJ8MCkhPShofDApKXtjb250aW51ZX1icmVha319RFtxKyhmPDwyKT4+Ml09Yn1mPWYrMXwwO2k9RFtlKzEyPj4yXTtpZihmPj4+MDxpPj4+MCl7Y29udGludWV9YnJlYWt9bz1lKzE2fDA7cT1EW2U+PjJdO2E9RFtlKzE2Pj4yXTtiPURbZSsyMD4+Ml0tYXwwO2Y9Yj4+Mjt0YTp7aWYoZj4+PjA8PTUyNDI4Nyl7c2Eobyw1MjQyODgtZnwwKTticmVhayB0YX1pZigoYnwwKT09MjA5NzE1Mil7YnJlYWsgdGF9RFtlKzIwPj4yXT1hKzIwOTcxNTJ9YT1lKzI4fDA7Zj1EW2E+PjJdO2I9RFtlKzMyPj4yXS1mPj4zO3VhOntpZihiPj4+MDxpPj4+MCl7YmIoYSxpLWJ8MCk7Zj1EW2E+PjJdO2JyZWFrIHVhfWlmKGI+Pj4wPmk+Pj4wKXtEW2UrMzI+PjJdPShpPDwzKStmfWlmKCFpKXticmVhayBuYX19aD0wO2E9MDt3aGlsZSgxKXtnPXErKGg8PDIpfDA7az1EW2c+PjJdO2I9YTttPShoPDwzKStmfDA7RFttKzQ+PjJdPWE7RFttPj4yXT1rO2c9RFtnPj4yXTthPWcrYXwwO2lmKGE+Pj4wPjUyNDI4OCl7YnJlYWsgbmF9dmE6e2lmKGE+Pj4wPD1iPj4+MCl7YnJlYWsgdmF9bT1EW28+PjJdO2s9MDtwPWcmNztpZihwKXt3aGlsZSgxKXtEW20rKGI8PDIpPj4yXT1oO2I9YisxfDA7az1rKzF8MDtpZigocHwwKSE9KGt8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZihnLTE+Pj4wPD02KXticmVhayB2YX13aGlsZSgxKXtnPW0rKGI8PDIpfDA7RFtnPj4yXT1oO0RbZysyOD4+Ml09aDtEW2crMjQ+PjJdPWg7RFtnKzIwPj4yXT1oO0RbZysxNj4+Ml09aDtEW2crMTI+PjJdPWg7RFtnKzg+PjJdPWg7RFtnKzQ+PjJdPWg7Yj1iKzh8MDtpZigoYnwwKSE9KGF8MCkpe2NvbnRpbnVlfWJyZWFrfX1oPWgrMXwwO2lmKChpfDApIT0oaHwwKSl7Y29udGludWV9YnJlYWt9bj0oYXwwKT09NTI0Mjg4fWE9bn13YTp7aWYoIWF8KERbZSsxMj4+Ml0/MDpsKSl7YnJlYWsgd2F9aWYoIU9hKDEsZSs1NnwwLGMpKXticmVhayB3YX1hPURbYys4Pj4yXTtmPURbYysxNj4+Ml07Yj1mO2c9YS1ifDA7bj1EW2UrNjA+PjJdO2g9RFtjKzIwPj4yXTthPURbYysxMj4+Ml0tKGgrKGE+Pj4wPGI+Pj4wKXwwKXwwO2I9RFtlKzU2Pj4yXTtpZigobnwwKT09KGF8MCkmZz4+PjA8Yj4+PjB8YT4+PjA8bj4+PjApe2JyZWFrIHdhfWE9aCtufDA7Zz1iK2Z8MDthPWc+Pj4wPGI+Pj4wP2ErMXwwOmE7RFtjKzE2Pj4yXT1nO0RbYysyMD4+Ml09YTthPWI7aWYoKGF8MCk8PTApe2JyZWFrIHdhfWM9ZitEW2M+PjJdfDA7RFtlKzQwPj4yXT1jO2I9YS0xfDA7Zj1jK2J8MDtnPUVbZnwwXTt4YTp7aWYoZz4+PjA8PTYzKXtEW2UrNDQ+PjJdPWI7YT1FW2Z8MF0mNjM7YnJlYWsgeGF9eWE6e3N3aXRjaCgoZz4+PjZ8MCktMXwwKXtjYXNlIDA6aWYoYT4+PjA8Mil7YnJlYWsgd2F9Yj1hLTJ8MDtEW2UrNDQ+PjJdPWI7YT0oYStjfDApLTJ8MDthPUVbYSsxfDBdPDw4JjE2MTI4fEVbYXwwXTticmVhayB4YTtjYXNlIDE6aWYoYT4+PjA8Myl7YnJlYWsgd2F9Yj1hLTN8MDtEW2UrNDQ+PjJdPWI7YT0oYStjfDApLTN8MDthPUVbYSsyfDBdPDwxNiY0MTI4NzY4fEVbYSsxfDBdPDw4fEVbYXwwXTticmVhayB4YTtkZWZhdWx0OmJyZWFrIHlhfX1iPWEtNHwwO0RbZSs0ND4+Ml09YjthPShhK2N8MCktNHwwO2E9RVthKzJ8MF08PDE2fEVbYSszfDBdPDwyNCYxMDU2OTY0NjA4fEVbYSsxfDBdPDw4fEVbYXwwXX1mPWErMjA5NzE1MnwwO0RbZSs0OD4+Ml09ZjtpZihmPj4+MD41MzY4NzA5MTEpe2JyZWFrIHdhfWlmKCFsKXtqPTE7YnJlYWsgd2F9Zz1EW2UrMjg+PjJdO2E9MDtuPURbZSsxNj4+Ml07d2hpbGUoMSl7emE6e2lmKGY+Pj4wPjIwOTcxNTEpe2JyZWFrIHphfXdoaWxlKDEpe2lmKChifDApPD0wKXticmVhayB6YX1iPWItMXwwO0RbZSs0ND4+Ml09YjtmPUVbYitjfDBdfGY8PDg7RFtlKzQ4Pj4yXT1mO2lmKGY+Pj4wPDIwOTcxNTIpe2NvbnRpbnVlfWJyZWFrfX1qPWYmNTI0Mjg3O2g9RFtuKyhqPDwyKT4+Ml07aT1nKyhoPDwzKXwwO2Y9KEooRFtpPj4yXSxmPj4+MTl8MCkranwwKS1EW2krND4+Ml18MDtEW2UrNDg+PjJdPWY7RFsoYTw8MikrZD4+Ml09aDtqPTE7YT1hKzF8MDtpZigobHwwKSE9KGF8MCkpe2NvbnRpbnVlfWJyZWFrfX1hPURbZSsyOD4+Ml07aWYoYSl7RFtlKzMyPj4yXT1hO21hKGEpfWE9RFtlKzE2Pj4yXTtpZihhKXtEW2UrMjA+PjJdPWE7bWEoYSl9YT1EW2U+PjJdO2lmKGEpe0RbZSs0Pj4yXT1hO21hKGEpfSQ9ZS0gLTY0fDA7Yj1qO2JyZWFrIGc7Y2FzZSAxNzpiPUdkKGEsYyxkKTticmVhayBnO2Nhc2UgMDpjYXNlIDE6Y2FzZSAyOmNhc2UgMzpjYXNlIDU6Y2FzZSA2OmNhc2UgNzpmPTA7aj0kKy02NHwwOyQ9ajtEW2orNDg+PjJdPTA7RFtqKzQwPj4yXT0wO0Rbais0ND4+Ml09MDtEW2orMzI+PjJdPTA7RFtqKzM2Pj4yXT0wO0RbaisyND4+Ml09MDtEW2orMjg+PjJdPTA7RFtqKzE2Pj4yXT0wO0RbaisyMD4+Ml09MDtEW2orOD4+Ml09MDtEW2orMTI+PjJdPTA7RFtqPj4yXT0wO0Rbais0Pj4yXT0wO0FhOntpZighaWMoaixjKXwoRFtqKzEyPj4yXT8wOmEpKXticmVhayBBYX1pZighT2EoMSxqKzU2fDAsYykpe2JyZWFrIEFhfWI9RFtjKzg+PjJdO249RFtjKzE2Pj4yXTtnPW47ZT1iLWd8MDtsPURbais2MD4+Ml07aD1EW2MrMjA+PjJdO2c9RFtjKzEyPj4yXS0oaCsoYj4+PjA8Zz4+PjApfDApfDA7Yj1EW2orNTY+PjJdO2lmKChsfDApPT0oZ3wwKSZlPj4+MDxiPj4+MHxnPj4+MDxsPj4+MCl7YnJlYWsgQWF9Zz1oK2x8MDtlPWIrbnwwO2c9ZT4+PjA8Yj4+PjA/ZysxfDA6ZztEW2MrMTY+PjJdPWU7RFtjKzIwPj4yXT1nO2lmKChifDApPD0wKXticmVhayBBYX1nPW4rRFtjPj4yXXwwO0Rbais0MD4+Ml09ZztjPWItMXwwO249ZytjfDA7ZT1FW258MF07QmE6e2lmKGU+Pj4wPD02Myl7RFtqKzQ0Pj4yXT1jO2I9RVtufDBdJjYzO2JyZWFrIEJhfUNhOntzd2l0Y2goKGU+Pj42fDApLTF8MCl7Y2FzZSAwOmlmKGI+Pj4wPDIpe2JyZWFrIEFhfWM9Yi0yfDA7RFtqKzQ0Pj4yXT1jO2I9KGIrZ3wwKS0yfDA7Yj1FW2IrMXwwXTw8OCYxNjEyOHxFW2J8MF07YnJlYWsgQmE7Y2FzZSAxOmlmKGI+Pj4wPDMpe2JyZWFrIEFhfWM9Yi0zfDA7RFtqKzQ0Pj4yXT1jO2I9KGIrZ3wwKS0zfDA7Yj1FW2IrMnwwXTw8MTYmNDEyODc2OHxFW2IrMXwwXTw8OHxFW2J8MF07YnJlYWsgQmE7ZGVmYXVsdDpicmVhayBDYX19Yz1iLTR8MDtEW2orNDQ+PjJdPWM7Yj0oYitnfDApLTR8MDtiPUVbYisyfDBdPDwxNnxFW2IrM3wwXTw8MjQmMTA1Njk2NDYwOHxFW2IrMXwwXTw8OHxFW2J8MF19aD1iKzE2Mzg0fDA7RFtqKzQ4Pj4yXT1oO2lmKGg+Pj4wPjQxOTQzMDMpe2JyZWFrIEFhfWlmKCFhKXtmPTE7YnJlYWsgQWF9Yj1EW2orMjg+PjJdO249RFtqKzE2Pj4yXTt3aGlsZSgxKXtEYTp7aWYoaD4+PjA+MTYzODMpe2JyZWFrIERhfXdoaWxlKDEpe2lmKChjfDApPD0wKXticmVhayBEYX1jPWMtMXwwO0Rbais0ND4+Ml09YztoPUVbYytnfDBdfGg8PDg7RFtqKzQ4Pj4yXT1oO2lmKGg+Pj4wPDE2Mzg0KXtjb250aW51ZX1icmVha319Zj1oJjQwOTU7ZT1EW24rKGY8PDIpPj4yXTtsPWIrKGU8PDMpfDA7aD0oSihEW2w+PjJdLGg+Pj4xMnwwKStmfDApLURbbCs0Pj4yXXwwO0Rbais0OD4+Ml09aDtEWyhrPDwyKStkPj4yXT1lO2Y9MTtrPWsrMXwwO2lmKChrfDApIT0oYXwwKSl7Y29udGludWV9YnJlYWt9fWE9RFtqKzI4Pj4yXTtpZihhKXtEW2orMzI+PjJdPWE7bWEoYSl9YT1EW2orMTY+PjJdO2lmKGEpe0RbaisyMD4+Ml09YTttYShhKX1hPURbaj4+Ml07aWYoYSl7RFtqKzQ+PjJdPWE7bWEoYSl9JD1qLSAtNjR8MDtiPWY7YnJlYWsgZztjYXNlIDEzOmNhc2UgMTQ6Y2FzZSAxNTpjYXNlIDE2OmJyZWFrIGg7ZGVmYXVsdDpicmVhayBnfX1iPUdkKGEsYyxkKX1mPWJ9cmV0dXJuIGZ9ZnVuY3Rpb24gY2goYSl7YT1hfDA7dmFyIGI9MCxjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MCxvPTAscD0wLHE9MCxyPTAscz0wLHQ9MCx1PTAsdj0wLHc9MCx4PTAseT0wLHo9MCxBPTAsSD0wLEk9MCxLPTAsTD0wLE09MCxOPTA7dT0kKy02NHwwOyQ9dTtEW2ErMTMyPj4yXT0wO2lmKERbYSsxNDg+PjJdKXtiPURbYSsxNDQ+PjJdO2lmKGIpe3doaWxlKDEpe2g9RFtiPj4yXTttYShiKTtiPWg7aWYoYil7Y29udGludWV9YnJlYWt9fWI9MDtEW2ErMTQ0Pj4yXT0wO2g9RFthKzE0MD4+Ml07YTp7aWYoIWgpe2JyZWFrIGF9aWYoaC0xPj4+MD49Myl7bz1oJi00O3doaWxlKDEpe2c9Yjw8MjtEW2crRFthKzEzNj4+Ml0+PjJdPTA7RFtEW2ErMTM2Pj4yXSsoZ3w0KT4+Ml09MDtEW0RbYSsxMzY+PjJdKyhnfDgpPj4yXT0wO0RbRFthKzEzNj4+Ml0rKGd8MTIpPj4yXT0wO2I9Yis0fDA7az1rKzR8MDtpZigob3wwKSE9KGt8MCkpe2NvbnRpbnVlfWJyZWFrfX1oPWgmMztpZighaCl7YnJlYWsgYX13aGlsZSgxKXtEW0RbYSsxMzY+PjJdKyhiPDwyKT4+Ml09MDtiPWIrMXwwO2M9YysxfDA7aWYoKGh8MCkhPShjfDApKXtjb250aW51ZX1icmVha319RFthKzE0OD4+Ml09MH1iOntjOntpZighRGEoMSx1KzYwfDAsRFtEW2ErND4+Ml0rMzI+PjJdKSl7YnJlYWsgY31EW2ErMTU2Pj4yXT1EW3UrNjA+PjJdO2lmKCFEYSgxLHUrNTZ8MCxEW0RbYSs0Pj4yXSszMj4+Ml0pKXticmVhayBjfWM9RFt1KzU2Pj4yXTtpZihjPj4+MD4xNDMxNjU1NzY1fEdbYSsxNTY+PjJdPkooYywzKT4+PjApe2JyZWFrIGN9aD1EW0RbYSs0Pj4yXSszMj4+Ml07Zz1EW2grOD4+Ml07bz1EW2grMTI+PjJdO2I9RFtoKzIwPj4yXTtkPWc7Zz1EW2grMTY+PjJdO2lmKChvfDApPD0oYnwwKSZkPj4+MDw9Zz4+PjB8KGJ8MCk+KG98MCkpe2JyZWFrIGN9bz1FW2crRFtoPj4yXXwwXTtnPWcrMXwwO2I9Zz9iOmIrMXwwO0RbaCsxNj4+Ml09ZztEW2grMjA+PjJdPWI7aWYoIURhKDEsdSs1MnwwLGgpKXticmVhayBjfXM9RFt1KzUyPj4yXTtpZihzPj4+MD5jPj4+MHxjPj4+MD5zKygocz4+PjApLzN8MCk+Pj4wKXticmVhayBjfWlmKCFEYSgxLHUrNDh8MCxEW0RbYSs0Pj4yXSszMj4+Ml0pKXticmVhayBjfWI9RFt1KzQ4Pj4yXTtpZihiPj4+MD5zPj4+MCl7YnJlYWsgY31EW2ErMjg+PjJdPURbYSsyND4+Ml07Zz1uYSg4OCk7V2MoZyk7aD1EW2ErOD4+Ml07RFthKzg+PjJdPWc7aWYoaCl7YWIoaCk7aWYoIURbYSs4Pj4yXSl7YnJlYWsgY319RFthKzE2ND4+Ml09RFthKzE2MD4+Ml07SGIoYSsxNjB8MCxjKTtEW2ErMTc2Pj4yXT1EW2ErMTcyPj4yXTtIYihhKzE3MnwwLGMpO0RbYS0gLTY0Pj4yXT0wO0RbYSs5Mj4+Ml09LTE7RFthKzg0Pj4yXT0tMTtEW2ErODg+PjJdPS0xO0RbYSs0MD4+Ml09RFthKzM2Pj4yXTtEW2ErNTI+PjJdPURbYSs0OD4+Ml07RFthKzc2Pj4yXT1EW2ErNzI+PjJdO0k9YSsyMTZ8MDt4ZChJKTt3ZChJLG8pO2lmKCFWYyhEW2ErOD4+Ml0sYyxiK0RbYSsxNTY+PjJdfDApKXticmVhayBjfWg9RFthKzE1Nj4+Ml07Qlt1Kzh8MF09MTtFYShhKzEyMHwwLGIraHwwLHUrOHwwKTtpZigodmQoYSxEW0RbYSs0Pj4yXSszMj4+Ml0pfDApPT0tMSl7YnJlYWsgY31mPWErMjMyfDA7dWQoZixhKTtNPWEsTj1iYVtEW0RbYT4+Ml0rMzY+PjJdXShhKXwwLERbTSszODA+PjJdPU47RFthKzM3Mj4+Ml09bztEW2ErMzg0Pj4yXT1iK0RbYSsxNTY+PjJdO2I9dSs4fDA7Q1tiKzM4Pj4xXT0wO0RbYj4+Ml09MDtEW2IrOD4+Ml09MDtEW2IrMTI+PjJdPTA7RFtiKzE2Pj4yXT0wO0RbYisyMD4+Ml09MDtEW2IrMjQ+PjJdPTA7RFtiKzI4Pj4yXT0wO0JbYisyOXwwXT0wO0JbYiszMHwwXT0wO0JbYiszMXwwXT0wO0JbYiszMnwwXT0wO0JbYiszM3wwXT0wO0JbYiszNHwwXT0wO0JbYiszNXwwXT0wO0JbYiszNnwwXT0wO0s9YjtvPWI7Zz0wO2U9JC0xNnwwOyQ9ZTtkOntpZighS2EoZis4MHwwLGYpKXticmVhayBkfWlmKCFyZChmKSl7YnJlYWsgZH1iPURbZis0Pj4yXTtEW28+PjJdPURbZj4+Ml07RFtvKzQ+PjJdPWI7Yj1EW2YrMzY+PjJdO0RbbyszMj4+Ml09RFtmKzMyPj4yXTtEW28rMzY+PjJdPWI7Yj1EW2YrMjg+PjJdO0RbbysyND4+Ml09RFtmKzI0Pj4yXTtEW28rMjg+PjJdPWI7Yj1EW2YrMjA+PjJdO0RbbysxNj4+Ml09RFtmKzE2Pj4yXTtEW28rMjA+PjJdPWI7Yj1EW2YrMTI+PjJdO0Rbbys4Pj4yXT1EW2YrOD4+Ml07RFtvKzEyPj4yXT1iO0RbZisxNzY+PjJdPTI7RFtmKzE4MD4+Ml09NztiPURbZisxNTI+PjJdO2lmKChifDApPDApe2JyZWFrIGR9RFtlKzEyPj4yXT0wO2c9MjtrPURbZisxNTY+PjJdO2g9RFtmKzE2MD4+Ml0taz4+MjtlOntpZihoPj4+MDxiPj4+MCl7eGEoZisxNTZ8MCxiLWh8MCxlKzEyfDApO2c9RFtmKzE3Nj4+Ml07Yz1EW2YrMTgwPj4yXTticmVhayBlfWM9NztpZihiPj4+MD49aD4+PjApe2JyZWFrIGV9RFtmKzE2MD4+Ml09aysoYjw8Mil9aD1mKzE4NHwwO2M9KGMtZ3wwKSsxfDA7Yj1EW2YrMTg4Pj4yXTtnPURbZisxODQ+PjJdO2s9KGItZ3wwKS8xMnwwO2Y6e2lmKGM+Pj4wPms+Pj4wKXtiPWMta3wwO2Q9RFtoKzg+PjJdO2c9RFtoKzQ+PjJdO2c6e2lmKGI+Pj4wPD0oZC1nfDApLzEyPj4+MCl7aWYoYil7Yj1KKChKKGIsMTIpLTEyPj4+MCkvMTJ8MCwxMikrMTJ8MDtnPXBhKGcsMCxiKStifDB9RFtoKzQ+PjJdPWc7YnJlYWsgZ31oOntpOntqOntrPURbaD4+Ml07cD0oZy1rfDApLzEyfDA7Yz1wK2J8MDtpZihjPj4+MDwzNTc5MTM5NDIpe2Q9KGQta3wwKS8xMnwwO3E9ZDw8MTtkPWQ+Pj4wPDE3ODk1Njk3MD9jPj4+MD5xPj4+MD9jOnE6MzU3OTEzOTQxO2lmKGQpe2lmKGQ+Pj4wPj0zNTc5MTM5NDIpe2JyZWFrIGp9aT1uYShKKGQsMTIpKX1jPUoocCwxMikraXwwO3A9SigoSihiLDEyKS0xMj4+PjApLzEyfDAsMTIpKzEyfDA7Yj1wYShjLDAscCk7cD1iK3B8MDtpPUooZCwxMikraXwwO2lmKChnfDApPT0oa3wwKSl7YnJlYWsgaX13aGlsZSgxKXtjPWMtMTJ8MDtnPWctMTJ8MDtEW2M+PjJdPURbZz4+Ml07RFtjKzQ+PjJdPURbZys0Pj4yXTtEW2MrOD4+Ml09RFtnKzg+PjJdO0RbZys4Pj4yXT0wO0RbZz4+Ml09MDtEW2crND4+Ml09MDtpZigoZ3wwKSE9KGt8MCkpe2NvbnRpbnVlfWJyZWFrfURbaCs4Pj4yXT1pO2I9RFtoKzQ+PjJdO0RbaCs0Pj4yXT1wO2c9RFtoPj4yXTtEW2g+PjJdPWM7aWYoKGJ8MCk9PShnfDApKXticmVhayBofXdoaWxlKDEpe2M9Yi0xMnwwO2s9RFtjPj4yXTtpZihrKXtEW2ItOD4+Ml09azttYShrKX1iPWM7aWYoKGJ8MCkhPShnfDApKXtjb250aW51ZX1icmVha31icmVhayBofWJyZWFrIGJ9cmEoMTMyNik7VCgpfURbaCs4Pj4yXT1pO0RbaCs0Pj4yXT1wO0RbaD4+Ml09Yn1pZihnKXttYShnKX19Yz1EW2YrMTg4Pj4yXTticmVhayBmfWlmKGM+Pj4wPj1rPj4+MCl7Yz1iO2JyZWFrIGZ9Yz1nK0ooYywxMil8MDtpZigoYnwwKSE9KGN8MCkpe3doaWxlKDEpe2c9Yi0xMnwwO2s9RFtnPj4yXTtpZihrKXtEW2ItOD4+Ml09azttYShrKX1iPWc7aWYoKGJ8MCkhPShjfDApKXtjb250aW51ZX1icmVha319RFtmKzE4OD4+Ml09Y31pPWYrMTk2fDA7Zz1EW2YrMTg0Pj4yXTtiPShjLWd8MCkvMTJ8MDtkPURbZisxOTY+PjJdO2s9RFtmKzIwMD4+Ml0tZD4+MjtrOntpZihiPj4+MD5rPj4+MCl7c2EoaSxiLWt8MCk7Zz1EW2YrMTg0Pj4yXTtjPURbZisxODg+PjJdO2JyZWFrIGt9aWYoYj4+PjA+PWs+Pj4wKXticmVhayBrfURbZisyMDA+PjJdPWQrKGI8PDIpfWlmKChjfDApPT0oZ3wwKSl7Zz0xO2JyZWFrIGR9Yj0wO3doaWxlKDEpe2w6e2lmKCFEYSgxLGUrOHwwLG8pKXticmVhayBsfWc9RFtlKzg+PjJdO2M9RFtmKzE0OD4+Ml07aWYoZz4+PjA+KERbYys0Pj4yXS1EW2M+PjJdPj4yPj4+MCkvMz4+PjApe2JyZWFrIGx9aWYoZyl7Yz1EW2g+PjJdO3E9SihiLDEyKTtrPWMrcXwwO2Q9RFtrPj4yXTtwPURbays0Pj4yXS1kPj4yO206e2lmKHA+Pj4wPGc+Pj4wKXtzYShrLGctcHwwKTtjPURbaD4+Ml07YnJlYWsgbX1pZihnPj4+MD49cD4+PjApe2JyZWFrIG19RFtrKzQ+PjJdPWQrKGc8PDIpfWpjKGcsMSxvLERbYytxPj4yXSk7RFtEW2k+PjJdKyhiPDwyKT4+Ml09Z31nPTE7Yj1iKzF8MDtpZihiPj4+MDwoRFtmKzE4OD4+Ml0tRFtmKzE4ND4+Ml18MCkvMTI+Pj4wKXtjb250aW51ZX1icmVhayBkfWJyZWFrfWc9MH0kPWUrMTZ8MDtuOntpZighZyl7YnJlYWsgbn1jPTA7Yj0wO2c9MDtxPTA7aD0wO289MDtrPTA7cD0wO209JC05NnwwOyQ9bTtEW20rNzI+PjJdPTA7RFttKzY0Pj4yXT0wO0RbbSs2OD4+Ml09MDtEW20rNDg+PjJdPTA7RFttKzUyPj4yXT0wO0RbbSs0MD4+Ml09MDtEW20rNDQ+PjJdPTA7RFttKzU2Pj4yXT0xMDY1MzUzMjE2O0RbbSszMj4+Ml09MDtEW20rMjQ+PjJdPTA7RFttKzI4Pj4yXT0wO2k9MTtmPWE7eD1EW2ErMTI0Pj4yXTtvOntwOntxOntyOntpZigoc3wwKTw9MCl7YnJlYWsgcn16PWYrMjMyfDA7TD1EW2YrMjE2Pj4yXSE9RFtmKzIyMD4+Ml07czp7d2hpbGUoMSl7YT1rO2s9YSsxfDA7dDp7dTp7djp7ZD1EW2YrNDA0Pj4yXTtpZigoZHwwKT09LTEpe0RbZis0MDA+PjJdPTc7YnJlYWsgdn1lPS0xO2o9RFtmKzQyOD4+Ml0rKGQ8PDIpfDA7bD1EW2o+PjJdO2Q9bC0xfDA7RFtqPj4yXT1kO2lmKChsfDApPD0wKXticmVhayBxfWo9RFtEW0RbZis0MTY+PjJdK0ooRFtmKzQwND4+Ml0sMTIpPj4yXSsoZDw8Mik+PjJdO2Q9RFsoajw8MikrODkyOD4+Ml07RFtmKzQwMD4+Ml09ZDtpZighail7aWYoKGN8MCk9PShofDApKXticmVhayBxfWQ9LTE7aT1EW2YrOD4+Ml07cj1EW2krMjQ+PjJdO3Q9Yy00fDA7ZT1EW3Q+PjJdO2I9LTE7dzp7aWYoKGV8MCk9PS0xKXticmVhayB3fWw9ZSsxfDA7bD0obD4+PjApJTN8MD9sOmUtMnwwO2I9LTE7aWYoKGx8MCk9PS0xKXticmVhayB3fWI9RFtEW2k+PjJdKyhsPDwyKT4+Ml19aj1EW3IrKGI8PDIpPj4yXTtpZigoanwwKSE9LTEpe2Q9aisxfDA7ZD0oZD4+PjApJTN8MD9kOmotMnwwfWw9RFtpKzEyPj4yXTtqPUooYSwzKTthPWorMXwwO0RbbCsoZTw8Mik+PjJdPWE7dz1hPDwyO0RbdytsPj4yXT1lO249aisyfDA7RFtsKyhkPDwyKT4+Ml09bjt5PW48PDI7RFt5K2w+PjJdPWQ7bD0tMTthPS0xO3g6e2lmKChlfDApPT0tMSl7YnJlYWsgeH15OntpZigoZT4+PjApJTN8MCl7ZT1lLTF8MDticmVhayB5fWU9ZSsyfDA7YT0tMTtpZigoZXwwKT09LTEpe2JyZWFrIHh9fWE9RFtEW2k+PjJdKyhlPDwyKT4+Ml19ejp7aWYoKGR8MCk9PS0xKXticmVhayB6fWU9ZCsxfDA7ZT0oZT4+PjApJTN8MD9lOmQtMnwwO2lmKChlfDApPT0tMSl7YnJlYWsgen1sPURbRFtpPj4yXSsoZTw8Mik+PjJdfWU9LTE7aWYoKGF8MCk9PShifDApfChifDApPT0obHwwKSl7YnJlYWsgcX1lPURbaT4+Ml07RFtlKyhqPDwyKT4+Ml09YjtEW2Urdz4+Ml09bDtEW2UreT4+Ml09YTtpZigoYXwwKSE9LTEpe0RbcisoYTw8Mik+PjJdPW59YT1EW2YrMTIwPj4yXSsoYj4+PjMmNTM2ODcwOTA4KXwwO2U9RFthPj4yXTtNPWEsTj1kaShiKSZlLERbTT4+Ml09TjtEW3Q+PjJdPWo7Yj1oO2ZjKHosaik7YnJlYWsgdH1BOntzd2l0Y2goZC0xfDApe2Nhc2UgMjpjYXNlIDQ6aWYoKGN8MCk9PShofDApKXticmVhayBxfWQ9RFtmKzg+PjJdO2U9RFtkKzEyPj4yXTtuPShqfDApPT0zO2o9SihhLDMpO3I9KG4/MjoxKStqfDA7dD1yPDwyO3c9Yy00fDA7Yj1EW3c+PjJdO0RbZSt0Pj4yXT1iO0RbZSsoYjw8Mik+PjJdPXI7UmEoZCsyNHwwKTtlPS0xO2M9RFtmKzg+PjJdO2w9RFtjKzI0Pj4yXTtpZigoeHwwKTxEW2MrMjg+PjJdLWw+PjIpe2JyZWFrIHF9Yz1EW2M+PjJdO2U9RFtkKzI4Pj4yXS1EW2QrMjQ+PjJdfDA7ZD0oZT4+MiktMXwwO0RbYyt0Pj4yXT1kO2lmKGUpe0RbbCsoZDw8Mik+PjJdPXJ9ZT1uP2o6aisyfDA7cj1jKyhqK248PDIpfDA7Qjp7aWYoKGJ8MCk9PS0xKXtEW2MrKGU8PDIpPj4yXT0tMTtkPS0xO2JyZWFrIEJ9Qzp7RDp7RTp7aWYoKGI+Pj4wKSUzfDApe2Q9Yi0xfDA7YnJlYWsgRX1kPWIrMnwwO2lmKChkfDApPT0tMSl7YnJlYWsgRH19ZD1EW2MrKGQ8PDIpPj4yXTtEW2MrKGU8PDIpPj4yXT1kO2lmKChkfDApPT0tMSl7YnJlYWsgQ31EW2wrKGQ8PDIpPj4yXT1lO2JyZWFrIEN9RFtjKyhlPDwyKT4+Ml09LTF9ZD1iKzF8MDtiPShkPj4+MCklM3wwP2Q6Yi0yfDA7ZD0tMTtpZigoYnwwKT09LTEpe2JyZWFrIEJ9ZD1EW2MrKGI8PDIpPj4yXX1EW3I+PjJdPWQ7RFt3Pj4yXT1qO2I9aDticmVhayB1O2Nhc2UgNjpicmVhayB2O2Nhc2UgMDpicmVhayBBO2RlZmF1bHQ6YnJlYWsgcX19aWYoKGJ8MCk9PShjfDApKXticmVhayBxfWc9Yy00fDA7bD1EW2c+PjJdO0RbbSs2OD4+Ml09ZztuPURbbSs0ND4+Ml07Rjp7aWYoIW4pe2JyZWFrIEZ9ZT1EW20rNDA+PjJdO3I9Y2kobik+Pj4wPjE7ZD1hJm4rMjE0NzQ4MzY0NztHOntpZighcil7YnJlYWsgR31kPWE7aWYoYT4+PjA8bj4+PjApe2JyZWFrIEd9ZD0oYT4+PjApJShuPj4+MCl8MH1qPWQ7ZT1EW2UrKGo8PDIpPj4yXTtpZighZSl7YnJlYWsgRn1lPURbZT4+Ml07aWYoIWUpe2JyZWFrIEZ9SDp7aWYoIXIpe2Q9bi0xfDA7d2hpbGUoMSl7bj1EW2UrND4+Ml07STp7aWYoKG58MCkhPShhfDApKXtpZigoanwwKT09KGQmbikpe2JyZWFrIEl9YnJlYWsgRn1pZigoYXwwKT09RFtlKzg+PjJdKXticmVhayBIfX1lPURbZT4+Ml07aWYoZSl7Y29udGludWV9YnJlYWt9YnJlYWsgRn13aGlsZSgxKXtkPURbZSs0Pj4yXTtKOntpZigoZHwwKSE9KGF8MCkpe2lmKGQ+Pj4wPj1uPj4+MCl7ZD0oZD4+PjApJShuPj4+MCl8MH1pZigoZHwwKT09KGp8MCkpe2JyZWFrIEp9YnJlYWsgRn1pZigoYXwwKT09RFtlKzg+PjJdKXticmVhayBIfX1lPURbZT4+Ml07aWYoZSl7Y29udGludWV9YnJlYWt9YnJlYWsgRn1pZigoZ3wwKSE9KHZ8MCkpe0RbZz4+Ml09RFtlKzEyPj4yXTtEW20rNjg+PjJdPWM7Zz1jO2JyZWFrIEZ9Yz12LWJ8MDtkPWM+PjI7aD1kKzF8MDtpZihoPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgYn1nPWM+PjE7Zz1kPj4+MDw1MzY4NzA5MTE/Zz4+PjA8aD4+PjA/aDpnOjEwNzM3NDE4MjM7aWYoZyl7aWYoZz4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIHB9aD1uYShnPDwyKX1lbHNle2g9MH1kPWgrKGQ8PDIpfDA7RFtkPj4yXT1EW2UrMTI+PjJdO3Y9KGc8PDIpK2h8MDtnPWQrNHwwO2lmKChjfDApPjApe29hKGgsYixjKX1EW20rNzI+PjJdPXY7RFttKzY4Pj4yXT1nO0RbbSs2ND4+Ml09aDtpZihiKXttYShiKX1iPWh9aWYoKGJ8MCk9PShnfDApKXticmVhayBzfXc9Zy00fDA7Yz1EW3c+PjJdO2U9KGN8MCk9PS0xO2o9RFtmKzg+PjJdO2lmKCFlJkRbRFtqKzEyPj4yXSsoYzw8Mik+PjJdIT0tMSl7YnJlYWsgc31uPURbaisxMj4+Ml07aWYoKGx8MCkhPS0xJkRbbisobDw8Mik+PjJdIT0tMSl7YnJlYWsgc31yPUooYSwzKTt0PXIrMnwwO0RbbisoYzw8Mik+PjJdPXQ7YT10PDwyO0RbYStuPj4yXT1jO2k9cisxfDA7RFtuKyhsPDwyKT4+Ml09aTt5PWk8PDI7RFtuK3k+PjJdPWw7Szp7TDp7TTp7aWYoIWUpe2lmKChjPj4+MCklM3wwKXtkPWMtMXwwO2JyZWFrIE19ZD1jKzJ8MDtpZigoZHwwKSE9LTEpe2JyZWFrIE19aT1EW2o+PjJdO2Q9LTE7YnJlYWsgTH1kPS0xO2k9RFtqPj4yXTtEW2krKHI8PDIpPj4yXT0tMTtlPS0xO2JyZWFrIEt9aT1EW2o+PjJdO2Q9RFtpKyhkPDwyKT4+Ml19RFsocjw8MikraT4+Ml09ZDtlPWMrMXwwO2M9KGU+Pj4wKSUzfDA/ZTpjLTJ8MDtlPS0xO2lmKChjfDApPT0tMSl7YnJlYWsgS31lPURbKGM8PDIpK2k+PjJdfURbaSt5Pj4yXT1lO046e2lmKChsfDApPT0tMSl7RFthK2k+PjJdPS0xO2U9LTE7Yz0tMTticmVhayBOfU86e1A6e1E6e2lmKChsPj4+MCklM3wwKXtlPWwtMXwwO2JyZWFrIFF9ZT1sKzJ8MDtpZigoZXwwKT09LTEpe2JyZWFrIFB9fWM9YStpfDA7YT1EWyhlPDwyKStpPj4yXTtEW2M+PjJdPWE7aWYoKGF8MCk9PS0xKXticmVhayBPfURbRFtqKzI0Pj4yXSsoYTw8Mik+PjJdPXQ7YnJlYWsgT31EW2EraT4+Ml09LTF9ZT0tMTthPWwrMXwwO2E9KGE+Pj4wKSUzfDA/YTpsLTJ8MDtjPS0xO2lmKChhfDApPT0tMSl7YnJlYWsgTn1lPWE7Yz1EWyhhPDwyKStpPj4yXX1hPURbZiszODg+PjJdO2w9ZDw8Mjt0PWErbHwwO3k9YTthPWM8PDI7RFt0Pj4yXT1EW3Q+PjJdK0RbeSthPj4yXTtqPURbaisyND4+Ml07YT1qK2F8MDtpZigoZHwwKSE9LTEpe0RbaitsPj4yXT1EW2E+PjJdfVI6e2lmKChlfDApPT0tMSl7YnJlYWsgUn13aGlsZSgxKXtEWyhlPDwyKStpPj4yXT1kO2o9ZSsxfDA7ZT0oaj4+PjApJTN8MD9qOmUtMnwwO2lmKChlfDApPT0tMSl7YnJlYWsgUn1lPURbbisoZTw8Mik+PjJdO2lmKChlfDApPT0tMSl7YnJlYWsgUn1qPWUrMXwwO2U9KGo+Pj4wKSUzfDA/ajplLTJ8MDtpZigoZXwwKSE9LTEpe2NvbnRpbnVlfWJyZWFrfX1EW2E+PjJdPS0xO1M6e2lmKEwpe2JyZWFrIFN9aWYoKHB8MCkhPShBfDApKXtEW3A+PjJdPWM7cD1wKzR8MDtEW20rMjg+PjJdPXA7YnJlYWsgU31UOnthPUEtcXwwO2k9YT4+MjtvPWkrMXwwO2lmKG8+Pj4wPDEwNzM3NDE4MjQpe2U9YT4+MTtlPWk+Pj4wPDUzNjg3MDkxMT9lPj4+MDxvPj4+MD9vOmU6MTA3Mzc0MTgyMztpZihlKXtpZihlPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgVH1vPW5hKGU8PDIpfWVsc2V7bz0wfWk9bysoaTw8Mil8MDtEW2k+PjJdPWM7QT0oZTw8Mikrb3wwO3A9aSs0fDA7aWYoKGF8MCk+MCl7b2EobyxxLGEpfURbbSszMj4+Ml09QTtEW20rMjg+PjJdPXA7RFttKzI0Pj4yXT1vO2lmKHEpe21hKHEpfXE9bzticmVhayBTfWJyZWFrIGJ9YnJlYWsgcH1EW3c+PjJdPXI7Yz1nO2ZjKHoscik7YnJlYWsgdH1qPURbZis4Pj4yXTtSYShqKzI0fDApO2U9LTE7ZD1EW2YrOD4+Ml07Zz1KKGEsMyk7bD1EW2orMjg+PjJdLURbaisyND4+Ml18MDtqPWw+PjI7cj1qLTF8MDtEW0RbZD4+Ml0rKGc8PDIpPj4yXT1yO1JhKGQrMjR8MCk7bj1nKzF8MDtEW0RbZD4+Ml0rKG48PDIpPj4yXT0oRFtkKzI4Pj4yXS1EW2QrMjQ+PjJdPj4yKS0xO2Q9RFtmKzg+PjJdO1JhKGQrMjR8MCk7dD1nKzJ8MDtEW0RbZD4+Ml0rKHQ8PDIpPj4yXT0oRFtkKzI4Pj4yXS1EW2QrMjQ+PjJdPj4yKS0xO3c9RFtmKzg+PjJdO2Q9RFt3KzI0Pj4yXTtpZigoeHwwKTxEW3crMjg+PjJdLWQ+PjIpe2JyZWFrIHF9VTp7Vjp7aWYoIWwpe0RbZCsoajw8Mik+PjJdPW47ZT0xO2JyZWFrIFZ9RFtkKyhyPDwyKT4+Ml09ZztlPTA7aWYoKGx8MCk9PS00KXticmVhayBWfURbZCsoajw8Mik+PjJdPW47ZT1qKzF8MDtpZigoZXwwKT09LTEpe2JyZWFrIFV9fURbZCsoZTw8Mik+PjJdPXR9aWYoKGN8MCkhPSh2fDApKXtEW2M+PjJdPWc7Zz1jKzR8MDtEW20rNjg+PjJdPWc7YnJlYWsgdX1jPXYtaHwwO2Q9Yz4+MjtiPWQrMXwwO2lmKGI+Pj4wPj0xMDczNzQxODI0KXticmVhayBifWU9Yz4+MTtlPWQ+Pj4wPDUzNjg3MDkxMT9iPj4+MD5lPj4+MD9iOmU6MTA3Mzc0MTgyMztpZihlKXtpZihlPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgcH1iPW5hKGU8PDIpfWVsc2V7Yj0wfWQ9YisoZDw8Mil8MDtEW2Q+PjJdPWc7dj0oZTw8MikrYnwwO2c9ZCs0fDA7aWYoKGN8MCk+MCl7b2EoYixoLGMpfURbbSs3Mj4+Ml09djtEW20rNjg+PjJdPWc7RFttKzY0Pj4yXT1iO2lmKGgpe21hKGgpfWg9Yn1mYyh6LERbZy00Pj4yXSk7ZT1EW2YrNDA+PjJdO1c6e2lmKChlfDApPT1EW2YrMzY+PjJdKXticmVhayBXfWQ9cysoYV4tMSl8MDtqPWctNHwwO3doaWxlKDEpe2E9RFtlLTg+PjJdO2lmKGE+Pj4wPmQ+Pj4wKXticmVhayBzfWlmKChhfDApIT0oZHwwKSl7YnJlYWsgV31sPUVbZS00fDBdO2E9ZS0xMnwwO2M9RFthPj4yXTtEW2YrNDA+PjJdPWE7aWYoKGN8MCk8MCl7YnJlYWsgc31hPURbaj4+Ml07RFttKzIwPj4yXT1zKyhjXi0xKTtjPW0rMjB8MDtEW20rODg+PjJdPWM7cWQobSxtKzQwfDAsYyxtKzg4fDApO2U9RFttPj4yXTtYOntpZihsJjEpe2M9LTE7aWYoKGF8MCk9PS0xKXticmVhayBYfWM9YSsxfDA7Yz0oYz4+PjApJTN8MD9jOmEtMnwwO2JyZWFrIFh9Yz0tMTtpZigoYXwwKT09LTEpe2JyZWFrIFh9Yz1hLTF8MDtpZigoYT4+PjApJTN8MCl7YnJlYWsgWH1jPWErMnwwfURbZSsxMj4+Ml09YztlPURbZis0MD4+Ml07aWYoKGV8MCkhPURbZiszNj4+Ml0pe2NvbnRpbnVlfWJyZWFrfX1jPWd9aT0oa3wwKTwoc3wwKTtpZigoa3wwKSE9KHN8MCkpe2NvbnRpbnVlfWJyZWFrfWs9czticmVhayByfWU9LTE7aWYoaSYxKXticmVhayBxfX1lPS0xO2M9RFtmKzg+PjJdO2lmKCh4fDApPERbYysyOD4+Ml0tRFtjKzI0Pj4yXT4+Mil7YnJlYWsgcX1pZigoZ3wwKSE9KGh8MCkpe3M9Zis2MHwwO3I9ZiszMTJ8MDt3aGlsZSgxKXtnPWctNHwwO2o9RFtnPj4yXTtEW20rNjg+PjJdPWc7WTp7aWYoR2Eocikpe249RFtmKzg+PjJdO2w9RFtuPj4yXTtpZigoKERbbis0Pj4yXS1sPj4yPj4+MCkvM3wwKTw9KGt8MCkpe2JyZWFrIHF9Yj0tMTthPS0xO2Q9RFtuKzI0Pj4yXTtjPS0xO1o6e2lmKChqfDApPT0tMSl7YnJlYWsgWn1xPWorMXwwO3E9KHE+Pj4wKSUzfDA/cTpqLTJ8MDtjPS0xO2lmKChxfDApPT0tMSl7YnJlYWsgWn1jPURbbCsocTw8Mik+PjJdfXE9YztjPURbZCsocTw8Mik+PjJdO186e2lmKChjfDApPT0tMSl7YnJlYWsgX31pPWMrMXwwO2M9KGk+Pj4wKSUzfDA/aTpjLTJ8MDtpZigoY3wwKT09LTEpe2JyZWFrIF99Yj1jKzF8MDtiPShiPj4+MCklM3wwP2I6Yy0yfDA7aWYoKGJ8MCkhPS0xKXthPURbbCsoYjw8Mik+PjJdfWI9Y312PS0xO2k9LTE7ZD1EW2QrKGE8PDIpPj4yXTtjPS0xOyQ6e2lmKChkfDApPT0tMSl7YnJlYWsgJH14PWQrMXwwO2Q9KHg+Pj4wKSUzfDA/eDpkLTJ8MDtjPS0xO2lmKChkfDApPT0tMSl7YnJlYWsgJH1jPWQrMXwwO2k9ZDtjPShjPj4+MCklM3wwP2M6ZC0yfDA7aWYoKGN8MCkhPS0xKXtjPURbbCsoYzw8Mik+PjJdfWVsc2V7Yz0tMX19ZD1EW24rMTI+PjJdO249SihrLDMpO3g9bjw8MjtEW2QreD4+Ml09ajtEW2QrKGo8PDIpPj4yXT1uO2o9bisxfDA7ej1qPDwyO0RbeitkPj4yXT1iO0RbZCsoYjw8Mik+PjJdPWo7Yj1uKzJ8MDtBPWI8PDI7RFtBK2Q+PjJdPWk7RFtkKyhpPDwyKT4+Ml09YjtEW2wreD4+Ml09YTtpPWwrenwwO0RbaT4+Ml09YztjPWwrQXwwO0RbYz4+Ml09cTtkPWo+Pj4wPG4+Pj4wPy0xOmE7YT1EW2YrMTIwPj4yXTtxPShkPj4+MyY1MzY4NzA5MDgpK2F8MDtsPURbcT4+Ml07TT1xLE49ZGkoZCkmbCxEW00+PjJdPU47dj0oanwwKSE9LTE/RFtpPj4yXTp2O2k9YSsodj4+PjMmNTM2ODcwOTA4KXwwO2Q9RFtpPj4yXTtNPWksTj1kaSh2KSZkLERbTT4+Ml09TjtkPS0xO2Q9KGJ8MCkhPS0xP0RbYz4+Ml06ZDthPWErKGQ+Pj4zJjUzNjg3MDkwOCl8MDtiPURbYT4+Ml07TT1hLE49ZGkoZCkmYixEW00+PjJdPU47ZD1EW2YrNjQ+PjJdO2E9RFtmKzY4Pj4yXTtpZigoZHwwKT09YTw8NSl7aWYoKGQrMXwwKTwwKXticmVhayBifWlmKGQ+Pj4wPD0xMDczNzQxODIyKXtiPWQrMzImLTMyO2E9YTw8NjthPWE+Pj4wPGI+Pj4wP2I6YX1lbHNle2E9MjE0NzQ4MzY0N31UYShzLGEpO2Q9RFtmKzY0Pj4yXX1rPWsrMXwwO0RbZis2ND4+Ml09ZCsxO2E9RFtmKzYwPj4yXSsoZD4+PjMmNTM2ODcwOTA4KXwwO0RbYT4+Ml09RFthPj4yXXwxPDxkO2E9RFtmKzc2Pj4yXTtpZigoYXwwKSE9RFtmKzgwPj4yXSl7RFthPj4yXT1uO0RbZis3Nj4+Ml09YSs0O2JyZWFrIFl9Yj1EW2YrNzI+PjJdO2M9YS1ifDA7ZD1jPj4yO2E9ZCsxfDA7aWYoYT4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGJ9aT1jPj4xO2k9ZD4+PjA8NTM2ODcwOTExP2E+Pj4wPmk+Pj4wP2E6aToxMDczNzQxODIzO2lmKGkpe2lmKGk+Pj4wPj0xMDczNzQxODI0KXticmVhayBwfWE9bmEoaTw8Mil9ZWxzZXthPTB9ZD1hKyhkPDwyKXwwO0RbZD4+Ml09bjtpZigoY3wwKT4wKXtvYShhLGIsYyl9RFtmKzgwPj4yXT1hKyhpPDwyKTtEW2YrNzY+PjJdPWQrNDtEW2YrNzI+PjJdPWE7aWYoIWIpe2JyZWFrIFl9bWEoYik7YnJlYWsgWX1jPURbZis2ND4+Ml07YT1EW2YrNjg+PjJdO2lmKChjfDApPT1hPDw1KXtpZigoYysxfDApPDApe2JyZWFrIGJ9aWYoYz4+PjA8PTEwNzM3NDE4MjIpe2I9YyszMiYtMzI7YT1hPDw2O2E9YT4+PjA8Yj4+PjA/YjphfWVsc2V7YT0yMTQ3NDgzNjQ3fVRhKHMsYSk7Yz1EW2YrNjQ+PjJdfURbZis2ND4+Ml09YysxO2E9RFtmKzYwPj4yXSsoYz4+PjMmNTM2ODcwOTA4KXwwO2I9RFthPj4yXTtNPWEsTj1kaShjKSZiLERbTT4+Ml09TjthPURbZis3Nj4+Ml07aWYoKGF8MCkhPURbZis4MD4+Ml0pe0RbYT4+Ml09ajtEW2YrNzY+PjJdPWErNDticmVhayBZfWI9RFtmKzcyPj4yXTtjPWEtYnwwO2Q9Yz4+MjthPWQrMXwwO2lmKGE+Pj4wPj0xMDczNzQxODI0KXticmVhayBifWk9Yz4+MTtpPWQ+Pj4wPDUzNjg3MDkxMT9hPj4+MD5pPj4+MD9hOmk6MTA3Mzc0MTgyMztpZihpKXtpZihpPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgcH1hPW5hKGk8PDIpfWVsc2V7YT0wfWQ9YSsoZDw8Mil8MDtEW2Q+PjJdPWo7aWYoKGN8MCk+MCl7b2EoYSxiLGMpfURbZis4MD4+Ml09YSsoaTw8Mik7RFtmKzc2Pj4yXT1kKzQ7RFtmKzcyPj4yXT1hO2lmKCFiKXticmVhayBZfW1hKGIpfWlmKChnfDApIT0oaHwwKSl7Y29udGludWV9YnJlYWt9Yz1EW2YrOD4+Ml19aWYoKChEW2MrND4+Ml0tRFtjPj4yXT4+Mj4+PjApLzN8MCkhPShrfDApKXticmVhayBxfWQ9RFtjKzI0Pj4yXTtlPURbYysyOD4+Ml0tZD4+MjtpZigob3wwKT09KHB8MCkpe289cDticmVhayBxfWE9bzt3aGlsZSgxKXtiPURbYT4+Ml07Zz1lLTF8MDtpPShnPDwyKStkfDA7aWYoRFtpPj4yXT09LTEpe3doaWxlKDEpe2c9ZS0yfDA7ZT1lLTF8MDtpPShnPDwyKStkfDA7aWYoRFtpPj4yXT09LTEpe2NvbnRpbnVlfWJyZWFrfX1pZihiPj4+MDw9Zz4+PjApe0RbbT4+Ml09YztkPURbaT4+Ml07QlttKzEyfDBdPTE7RFttKzg+PjJdPWQ7RFttKzQ+PjJdPWQ7aWYoKGR8MCkhPS0xKXt3aGlsZSgxKXtEW0RbYz4+Ml0rKGQ8PDIpPj4yXT1iO2tjKG0pO2M9RFtmKzg+PjJdO2Q9RFttKzg+PjJdO2lmKChkfDApIT0tMSl7Y29udGludWV9YnJlYWt9fWQ9RFtjKzI0Pj4yXTtrPWQrKGc8PDIpfDA7aWYoKGJ8MCkhPS0xKXtEWyhiPDwyKStkPj4yXT1EW2s+PjJdfURbaz4+Ml09LTE7az0xPDxiO3M9RFtmKzEyMD4+Ml07Yj1zKyhiPj4+MyY1MzY4NzA5MDgpfDA7cz1zKyhnPj4+MyY1MzY4NzA5MDgpfDA7Zz0xPDxnO2lmKERbcz4+Ml0mZyl7az1rfERbYj4+Ml19ZWxzZXtrPURbYj4+Ml0mKGteLTEpfURbYj4+Ml09aztEW3M+PjJdPURbcz4+Ml0mKGdeLTEpO2U9ZS0xfDB9YT1hKzR8MDtpZigocHwwKSE9KGF8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZihvKXttYShvKX1hPURbbSs0OD4+Ml07aWYoYSl7d2hpbGUoMSl7Yj1EW2E+PjJdO21hKGEpO2E9YjtpZihhKXtjb250aW51ZX1icmVha319YT1EW20rNDA+PjJdO0RbbSs0MD4+Ml09MDtpZihhKXttYShhKX1pZihoKXtEW20rNjg+PjJdPWg7bWEoaCl9JD1tKzk2fDA7YnJlYWsgb31yYSgxMzI2KTtUKCl9aWYoKGV8MCk9PS0xKXticmVhayBufWE9RFtLKzE2Pj4yXTtiPWErRFtLPj4yXXwwO2g9RFtLKzg+PjJdO2g9aC1hfDA7YT1EW0RbZis0Pj4yXSszMj4+Ml07Q1thKzM4Pj4xXT1GW2ErMzg+PjFdO0RbYT4+Ml09YjtEW2ErMTY+PjJdPTA7RFthKzIwPj4yXT0wO0RbYSs4Pj4yXT1oO0RbYSsxMj4+Ml09MDthYTp7aWYoRFtmKzIxNj4+Ml09PURbZisyMjA+PjJdKXticmVhayBhYX1hPURbZis4Pj4yXTtpZihEW2ErND4+Ml09PURbYT4+Ml0pe2JyZWFrIGFhfWI9MDt3aGlsZSgxKXtpZih0ZChmLGIpKXtiPWIrM3wwO2E9RFtmKzg+PjJdO2lmKGI+Pj4wPERbYSs0Pj4yXS1EW2E+PjJdPj4yPj4+MCl7Y29udGludWV9YnJlYWsgYWF9YnJlYWt9YnJlYWsgbn1pZihFW2YrMzA4fDBdKXtCW2YrMzA4fDBdPTA7Yj1EW2YrMjkyPj4yXTthPTA7aD1EW2YrMzA0Pj4yXSs3fDA7YT1oPj4+MDw3PzE6YTtjPWE+Pj4zfDA7aD1hPDwyOXxoPj4+MzthPWgrRFtmKzI4OD4+Ml18MDtiPWIrY3wwO0RbZisyODg+PjJdPWE7RFtmKzI5Mj4+Ml09YT4+PjA8aD4+PjA/YisxfDA6Yn1iPURbZisyMTY+PjJdO2lmKChifDApIT1EW2YrMjIwPj4yXSl7d2hpbGUoMSl7YT1KKEgsMTQ0KTtUYygoYStifDApKzR8MCxEW2YrOD4+Ml0pO2M9RFtJPj4yXTtoPWErY3wwO2I9RFtoKzEzMj4+Ml07aD1EW2grMTM2Pj4yXTtpZigoYnwwKSE9KGh8MCkpe3doaWxlKDEpe1JjKChhK2N8MCkrNHwwLERbYj4+Ml0pO2M9RFtJPj4yXTtiPWIrNHwwO2lmKChofDApIT0oYnwwKSl7Y29udGludWV9YnJlYWt9fVNjKChhK2N8MCkrNHwwKTtIPUgrMXwwO2I9RFtmKzIxNj4+Ml07aWYoSD4+PjA8KERbZisyMjA+PjJdLWJ8MCkvMTQ0Pj4+MCl7Y29udGludWV9YnJlYWt9fWE9RFtmKzg+PjJdO0diKGYrMTg0fDAsRFthKzI4Pj4yXS1EW2ErMjQ+PjJdPj4yKTtjPURbZisyMTY+PjJdO2lmKChjfDApIT1EW2YrMjIwPj4yXSl7Yj0wO3doaWxlKDEpe2E9SihiLDE0NCkrY3wwO2g9RFthKzYwPj4yXS1EW2ErNTY+PjJdPj4yO2M9YSsxMDR8MDthPURbZis4Pj4yXTthPURbYSsyOD4+Ml0tRFthKzI0Pj4yXT4+MjtHYihjLChhfDApPihofDApP2E6aCk7Yj1iKzF8MDtjPURbZisyMTY+PjJdO2lmKGI+Pj4wPChEW2YrMjIwPj4yXS1jfDApLzE0ND4+PjApe2NvbnRpbnVlfWJyZWFrfX1IPXNkKGYsZSl9fSQ9dS0gLTY0fDA7cmV0dXJuIEh8MH1xYSgpO1QoKX1mdW5jdGlvbiBlaChhKXthPWF8MDt2YXIgYj0wLGM9MCxkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTAsaz0wLGw9MCxtPTAsbj0wLG89MCxwPTAscT0wLHI9MCxzPTAsdD0wLHU9MCx2PTAsdz0wLHg9MCx5PTAsej0wLEE9MCxIPTAsST0wLEs9MCxMPTA7dj0kKy02NHwwOyQ9djtEW2ErMTMyPj4yXT0wO2lmKERbYSsxNDg+PjJdKXtiPURbYSsxNDQ+PjJdO2lmKGIpe3doaWxlKDEpe2U9RFtiPj4yXTttYShiKTtiPWU7aWYoZSl7Y29udGludWV9YnJlYWt9fWI9MDtEW2ErMTQ0Pj4yXT0wO2U9RFthKzE0MD4+Ml07YTp7aWYoIWUpe2JyZWFrIGF9aWYoZS0xPj4+MD49Myl7aD1lJi00O3doaWxlKDEpe2M9Yjw8MjtEW2MrRFthKzEzNj4+Ml0+PjJdPTA7RFtEW2ErMTM2Pj4yXSsoY3w0KT4+Ml09MDtEW0RbYSsxMzY+PjJdKyhjfDgpPj4yXT0wO0RbRFthKzEzNj4+Ml0rKGN8MTIpPj4yXT0wO2I9Yis0fDA7ZD1kKzR8MDtpZigoaHwwKSE9KGR8MCkpe2NvbnRpbnVlfWJyZWFrfX1lPWUmMztpZighZSl7YnJlYWsgYX13aGlsZSgxKXtEW0RbYSsxMzY+PjJdKyhiPDwyKT4+Ml09MDtiPWIrMXwwO249bisxfDA7aWYoKGV8MCkhPShufDApKXtjb250aW51ZX1icmVha319RFthKzE0OD4+Ml09MH1iOntpZighRGEoMSx2KzYwfDAsRFtEW2ErND4+Ml0rMzI+PjJdKSl7YnJlYWsgYn1EW2ErMTU2Pj4yXT1EW3YrNjA+PjJdO2lmKCFEYSgxLHYrNTZ8MCxEW0RbYSs0Pj4yXSszMj4+Ml0pKXticmVhayBifWQ9RFt2KzU2Pj4yXTtpZihkPj4+MD4xNDMxNjU1NzY1fEdbYSsxNTY+PjJdPkooZCwzKT4+PjApe2JyZWFrIGJ9ZT1EW0RbYSs0Pj4yXSszMj4+Ml07Yz1EW2UrOD4+Ml07bj1EW2UrMTI+PjJdO2I9RFtlKzIwPj4yXTtmPWM7Yz1EW2UrMTY+PjJdO2lmKChufDApPD0oYnwwKSZmPj4+MDw9Yz4+PjB8KGJ8MCk+KG58MCkpe2JyZWFrIGJ9bj1FW2MrRFtlPj4yXXwwXTtjPWMrMXwwO2I9Yz9iOmIrMXwwO0RbZSsxNj4+Ml09YztEW2UrMjA+PjJdPWI7aWYoIURhKDEsdis1MnwwLGUpKXticmVhayBifW09RFt2KzUyPj4yXTtpZihtPj4+MD5kPj4+MHxkPj4+MD5tKygobT4+PjApLzN8MCk+Pj4wKXticmVhayBifWlmKCFEYSgxLHYrNDh8MCxEW0RbYSs0Pj4yXSszMj4+Ml0pKXticmVhayBifWU9RFt2KzQ4Pj4yXTtpZihlPj4+MD5tPj4+MCl7YnJlYWsgYn1EW2ErMjg+PjJdPURbYSsyND4+Ml07Yz1uYSg4OCk7V2MoYyk7Yj1EW2ErOD4+Ml07RFthKzg+PjJdPWM7aWYoYil7YWIoYik7aWYoIURbYSs4Pj4yXSl7YnJlYWsgYn19RFthKzE2ND4+Ml09RFthKzE2MD4+Ml07SGIoYSsxNjB8MCxkKTtEW2ErMTc2Pj4yXT1EW2ErMTcyPj4yXTtIYihhKzE3MnwwLGQpO0RbYS0gLTY0Pj4yXT0wO0RbYSs5Mj4+Ml09LTE7RFthKzg0Pj4yXT0tMTtEW2ErODg+PjJdPS0xO0RbYSs0MD4+Ml09RFthKzM2Pj4yXTtEW2ErNTI+PjJdPURbYSs0OD4+Ml07RFthKzc2Pj4yXT1EW2ErNzI+PjJdO0g9YSsyMTZ8MDt4ZChIKTt3ZChILG4pO2lmKCFWYyhEW2ErOD4+Ml0sZCxlK0RbYSsxNTY+PjJdfDApKXticmVhayBifWI9RFthKzE1Nj4+Ml07Qlt2Kzh8MF09MTtFYShhKzEyMHwwLGIrZXwwLHYrOHwwKTtpZigodmQoYSxEW0RbYSs0Pj4yXSszMj4+Ml0pfDApPT0tMSl7YnJlYWsgYn1lPWErMjMyfDA7dWQoZSxhKTtEW2ErMzcyPj4yXT1uO289dis4fDA7Q1tvKzM4Pj4xXT0wO0Rbbz4+Ml09MDtEW28rOD4+Ml09MDtEW28rMTI+PjJdPTA7RFtvKzE2Pj4yXT0wO0RbbysyMD4+Ml09MDtEW28rMjQ+PjJdPTA7RFtvKzI4Pj4yXT0wO0JbbysyOXwwXT0wO0JbbyszMHwwXT0wO0JbbyszMXwwXT0wO0JbbyszMnwwXT0wO0JbbyszM3wwXT0wO0JbbyszNHwwXT0wO0JbbyszNXwwXT0wO0JbbyszNnwwXT0wO2I9MDtuPSQtMTZ8MDskPW47ZD1EW2UrND4+Ml07RFtlKzQwPj4yXT1EW2U+PjJdO0RbZSs0ND4+Ml09ZDtkPURbZSszNj4+Ml07RFtlKzcyPj4yXT1EW2UrMzI+PjJdO0RbZSs3Nj4+Ml09ZDtjPURbZSsyOD4+Ml07ZD1lLSAtNjR8MDtEW2Q+PjJdPURbZSsyND4+Ml07RFtkKzQ+PjJdPWM7ZD1EW2UrMjA+PjJdO0RbZSs1Nj4+Ml09RFtlKzE2Pj4yXTtEW2UrNjA+PjJdPWQ7ZD1EW2UrMTI+PjJdO0RbZSs0OD4+Ml09RFtlKzg+PjJdO0RbZSs1Mj4+Ml09ZDtjOntkOntpZihjYyhlKzQwfDAsMSxuKzh8MCkpe2Q9RFtlKzQ0Pj4yXTtEW2U+PjJdPURbZSs0MD4+Ml07RFtlKzQ+PjJdPWQ7ZD1EW2UrNzY+PjJdO0RbZSszMj4+Ml09RFtlKzcyPj4yXTtEW2UrMzY+PjJdPWQ7ZD1EW2UrNjg+PjJdO0RbZSsyND4+Ml09RFtlKzY0Pj4yXTtEW2UrMjg+PjJdPWQ7Yz1EW2UrNjA+PjJdO2g9YztkPURbZSs1Nj4+Ml07RFtlKzE2Pj4yXT1kO0RbZSsyMD4+Ml09YztmPURbZSs1Mj4+Ml07Yz1EW2UrNDg+PjJdO0RbZSs4Pj4yXT1jO0RbZSsxMj4+Ml09ZjtnPWMtZHwwO2s9RFtuKzEyPj4yXTtmPWYtKChjPj4+MDxkPj4+MCkraHwwKXwwO2M9RFtuKzg+PjJdO2lmKChrfDApPT0oZnwwKSZnPj4+MD49Yz4+PjB8Zj4+PjA+az4+PjApe2JyZWFrIGR9fWJyZWFrIGN9Zj1jK2R8MDtkPWgra3wwO0RbZSsxNj4+Ml09ZjtEW2UrMjA+PjJdPWM+Pj4wPmY+Pj4wP2QrMXwwOmQ7aWYoIUthKGUrODB8MCxlKSl7YnJlYWsgY31pZighcmQoZSkpe2JyZWFrIGN9Yj1EW2UrND4+Ml07RFtvPj4yXT1EW2U+PjJdO0Rbbys0Pj4yXT1iO2I9RFtlKzM2Pj4yXTtEW28rMzI+PjJdPURbZSszMj4+Ml07RFtvKzM2Pj4yXT1iO2I9RFtlKzI4Pj4yXTtEW28rMjQ+PjJdPURbZSsyND4+Ml07RFtvKzI4Pj4yXT1iO2I9RFtlKzIwPj4yXTtEW28rMTY+PjJdPURbZSsxNj4+Ml07RFtvKzIwPj4yXT1iO2I9RFtlKzEyPj4yXTtEW28rOD4+Ml09RFtlKzg+PjJdO0RbbysxMj4+Ml09YjtiPTF9JD1uKzE2fDA7ZTp7aWYoIWIpe2JyZWFrIGV9ZD0wO2U9MDtiPTA7bj0wO2s9JC05NnwwOyQ9aztEW2srNzI+PjJdPTA7RFtrKzY0Pj4yXT0wO0Rbays2OD4+Ml09MDtEW2srNDg+PjJdPTA7RFtrKzUyPj4yXT0wO0Rbays0MD4+Ml09MDtEW2srNDQ+PjJdPTA7RFtrKzU2Pj4yXT0xMDY1MzUzMjE2O0RbayszMj4+Ml09MDtEW2srMjQ+PjJdPTA7RFtrKzI4Pj4yXT0wO2k9MTtoPWE7dD1EW2ErMTI0Pj4yXTtmOntnOntoOntpOntqOntrOntsOnttOntpZigobXwwKTw9MCl7YnJlYWsgbX1JPURbaCsyMTY+PjJdIT1EW2grMjIwPj4yXTt3aGlsZSgxKXtmPXI7cj1mKzF8MDtuOntvOntwOntxOntyOntzOnt0Ont1Ont2Ont3Ont4OntpZighRVtoKzMwOHwwXSl7YnJlYWsgeH15Ont6OntnPURbaCsyOTY+PjJdO2M9RFtoKzMwND4+Ml07YT1nKyhjPj4+M3wwKXwwO2w9RFtoKzMwMD4+Ml07aWYoYT4+PjA+PWw+Pj4wKXticmVhayB6fWo9RVthfDBdO2E9YysxfDA7RFtoKzMwND4+Ml09YTtpZighKGo+Pj4oYyY3KSYxKSl7YnJlYWsgen1qPWE+Pj4zfDA7cD1nK2p8MDtBOntpZihwPj4+MD49bD4+PjApe2M9YTthPTA7YnJlYWsgQX1wPUVbcHwwXTtjPWMrMnwwO0RbaCszMDQ+PjJdPWM7aj1jPj4+M3wwO2E9cD4+PihhJjcpJjF9Zz1nK2p8MDtpZihnPj4+MDxsPj4+MCl7Zz1FW2d8MF07RFtoKzMwND4+Ml09YysxO2M9Zz4+PihjJjcpPDwxJjJ9ZWxzZXtjPTB9YT0oYXxjKTw8MTtzd2l0Y2goYS0xfDApe2Nhc2UgMDpjYXNlIDI6Y2FzZSA0OmJyZWFrIGg7Y2FzZSA1OmJyZWFrIHc7Y2FzZSAxOmNhc2UgMzpicmVhayB5O2RlZmF1bHQ6YnJlYWsgeH19aWYoKGJ8MCk9PShlfDApKXtjPS0xO2JyZWFrIGx9aj0tMTtnPURbaCs4Pj4yXTtwPURbZysyND4+Ml07cz1lLTR8MDtjPURbcz4+Ml07YT0tMTtCOntpZigoY3wwKT09LTEpe2JyZWFrIEJ9aT1jKzF8MDtpPShpPj4+MCklM3wwP2k6Yy0yfDA7YT0tMTtpZigoaXwwKT09LTEpe2JyZWFrIEJ9YT1EW0RbZz4+Ml0rKGk8PDIpPj4yXX1kPURbcCsoYTw8Mik+PjJdO2lmKChkfDApIT0tMSl7aT1kKzF8MDtqPShpPj4+MCklM3wwP2k6ZC0yfDB9ZD1EW2crMTI+PjJdO2Y9SihmLDMpO2k9ZisxfDA7RFtkKyhjPDwyKT4+Ml09aTt3PWk8PDI7RFt3K2Q+PjJdPWM7bD1mKzJ8MDtEW2QrKGo8PDIpPj4yXT1sO3g9bDw8MjtEW3grZD4+Ml09ajtpPS0xO2Q9LTE7Qzp7aWYoKGN8MCk9PS0xKXticmVhayBDfUQ6e2lmKChjPj4+MCklM3wwKXtjPWMtMXwwO2JyZWFrIER9Yz1jKzJ8MDtkPS0xO2lmKChjfDApPT0tMSl7YnJlYWsgQ319ZD1EW0RbZz4+Ml0rKGM8PDIpPj4yXX1FOntpZigoanwwKT09LTEpe2JyZWFrIEV9Yz1qKzF8MDtjPShjPj4+MCklM3wwP2M6ai0yfDA7aWYoKGN8MCk9PS0xKXticmVhayBFfWk9RFtEW2c+PjJdKyhjPDwyKT4+Ml19Yz0tMTtpZigoYXwwKT09KGR8MCl8KGF8MCk9PShpfDApKXticmVhayBsfWM9RFtnPj4yXTtEW2MrKGY8PDIpPj4yXT1hO0RbYyt3Pj4yXT1pO0RbYyt4Pj4yXT1kO2lmKChkfDApIT0tMSl7RFtwKyhkPDwyKT4+Ml09bH1kPURbaCsxMjA+PjJdKyhhPj4+MyY1MzY4NzA5MDgpfDA7Yz1EW2Q+PjJdO0s9ZCxMPWRpKGEpJmMsRFtLPj4yXT1MO0Rbcz4+Ml09ZjtkPWI7YnJlYWsgbn1pZigoYnwwKT09KGV8MCkpe2M9LTE7YnJlYWsgbH1kPURbaCs4Pj4yXTtjPURbZCsxMj4+Ml07Zz1KKGYsMyk7cD0oYXwxKT09NTtqPWcrKHA/MjoxKXwwO3c9ajw8Mjt4PWUtNHwwO2E9RFt4Pj4yXTtEW2Mrdz4+Ml09YTtEW2MrKGE8PDIpPj4yXT1qO1JhKGQrMjR8MCk7Yz0tMTtzPURbaCs4Pj4yXTtsPURbcysyND4+Ml07aWYoKHR8MCk8RFtzKzI4Pj4yXS1sPj4yKXticmVhayBsfWM9RFtzPj4yXTtkPURbZCsyOD4+Ml0tRFtkKzI0Pj4yXXwwO3M9KGQ+PjIpLTF8MDtEW2Mrdz4+Ml09cztpZihkKXtEW2wrKHM8PDIpPj4yXT1qfWo9cD9nOmcrMnwwO3A9YysoZytwPDwyKXwwO0Y6e2lmKChhfDApPT0tMSl7RFtjKyhqPDwyKT4+Ml09LTE7ZD0tMTticmVhayBGfUc6e0g6e0k6e2lmKChhPj4+MCklM3wwKXtkPWEtMXwwO2JyZWFrIEl9ZD1hKzJ8MDtpZigoZHwwKT09LTEpe2JyZWFrIEh9fWQ9RFtjKyhkPDwyKT4+Ml07RFtjKyhqPDwyKT4+Ml09ZDtpZigoZHwwKT09LTEpe2JyZWFrIEd9RFtsKyhkPDwyKT4+Ml09ajticmVhayBHfURbYysoajw8Mik+PjJdPS0xfWo9YSsxfDA7YT0oaj4+PjApJTN8MD9qOmEtMnwwO2Q9LTE7aWYoKGF8MCk9PS0xKXticmVhayBGfWQ9RFtjKyhhPDwyKT4+Ml19RFtwPj4yXT1kO0RbeD4+Ml09ZztkPWI7YnJlYWsgdn1pZigoYnwwKT09KGV8MCkpe2M9LTE7YnJlYWsgbH1hPWUtNHwwO2w9RFthPj4yXTtEW2srNjg+PjJdPWE7aj1EW2srNDQ+PjJdO0o6e2lmKCFqKXtlPWE7YnJlYWsgSn1nPWNpKGopPj4+MD4xO0s6e2lmKCFnKXtkPWYmaisyMTQ3NDgzNjQ3O2JyZWFrIEt9ZD1mO2lmKGo+Pj4wPmQ+Pj4wKXticmVhayBLfWQ9KGY+Pj4wKSUoaj4+PjApfDB9Yz1EW0Rbays0MD4+Ml0rKGQ8PDIpPj4yXTtpZighYyl7ZT1hO2JyZWFrIEp9Yz1EW2M+PjJdO2lmKCFjKXtlPWE7YnJlYWsgSn1MOntpZighZyl7Zz1qLTF8MDt3aGlsZSgxKXtqPURbYys0Pj4yXTtNOntpZigoanwwKSE9KGZ8MCkpe2lmKChnJmopPT0oZHwwKSl7YnJlYWsgTX1lPWE7YnJlYWsgSn1pZigoZnwwKT09RFtjKzg+PjJdKXticmVhayBMfX1jPURbYz4+Ml07aWYoYyl7Y29udGludWV9YnJlYWt9ZT1hO2JyZWFrIEp9d2hpbGUoMSl7Zz1EW2MrND4+Ml07Tjp7aWYoKGd8MCkhPShmfDApKXtpZihnPj4+MD49aj4+PjApe2c9KGc+Pj4wKSUoaj4+PjApfDB9aWYoKGR8MCk9PShnfDApKXticmVhayBOfWU9YTticmVhayBKfWlmKChmfDApPT1EW2MrOD4+Ml0pe2JyZWFrIEx9fWM9RFtjPj4yXTtpZihjKXtjb250aW51ZX1icmVha31lPWE7YnJlYWsgSn1pZigoYXwwKSE9KHV8MCkpe0RbYT4+Ml09RFtjKzEyPj4yXTtEW2srNjg+PjJdPWU7YnJlYWsgSn1hPXUtYnwwO2Q9YT4+MjtlPWQrMXwwO2lmKGU+Pj4wPj0xMDczNzQxODI0KXticmVhayB0fXE9YT4+MTtlPWQ+Pj4wPDUzNjg3MDkxMT9lPj4+MD5xPj4+MD9lOnE6MTA3Mzc0MTgyMztpZihlKXtpZihlPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgZ31xPW5hKGU8PDIpfWVsc2V7cT0wfWQ9cSsoZDw8Mil8MDtEW2Q+PjJdPURbYysxMj4+Ml07dT0oZTw8MikrcXwwO2U9ZCs0fDA7aWYoKGF8MCk+MCl7b2EocSxiLGEpfURbays3Mj4+Ml09dTtEW2srNjg+PjJdPWU7RFtrKzY0Pj4yXT1xO2lmKCFiKXticmVhayBKfW1hKGIpfWlmKChlfDApPT0ocXwwKSl7YnJlYWsgdX13PWUtNHwwO2E9RFt3Pj4yXTtkPShhfDApPT0tMTtiPURbaCs4Pj4yXTtpZighZCZEW0RbYisxMj4+Ml0rKGE8PDIpPj4yXSE9LTEpe2JyZWFrIHV9Zz1EW2IrMTI+PjJdO2lmKChsfDApIT0tMSZEW2crKGw8PDIpPj4yXSE9LTEpe2JyZWFrIHV9Zj1KKGYsMyk7cz1mKzJ8MDtEW2crKGE8PDIpPj4yXT1zO3A9czw8MjtEW3ArZz4+Ml09YTtjPWYrMXwwO0RbZysobDw8Mik+PjJdPWM7eD1jPDwyO0RbeCtnPj4yXT1sO2lmKGQpe2JyZWFrIHN9aWYoKGE+Pj4wKSUzfDApe2k9YS0xfDA7YnJlYWsgcX1pPWErMnwwO2lmKChpfDApIT0tMSl7YnJlYWsgcX1qPURbYj4+Ml07ZD0tMTticmVhayBwfWo9RFtoKzg+PjJdO1JhKGorMjR8MCk7Yz0tMTtnPURbaCs4Pj4yXTthPUooZiwzKTtsPURbaisyOD4+Ml0tRFtqKzI0Pj4yXXwwO2o9bD4+MjtzPWotMXwwO0RbRFtnPj4yXSsoYTw8Mik+PjJdPXM7UmEoZysyNHwwKTtwPWErMXwwO0RbRFtnPj4yXSsocDw8Mik+PjJdPShEW2crMjg+PjJdLURbZysyND4+Ml0+PjIpLTE7Zz1EW2grOD4+Ml07UmEoZysyNHwwKTt3PWErMnwwO0RbRFtnPj4yXSsodzw8Mik+PjJdPShEW2crMjg+PjJdLURbZysyND4+Ml0+PjIpLTE7eD1EW2grOD4+Ml07Zz1EW3grMjQ+PjJdO2lmKCh0fDApPERbeCsyOD4+Ml0tZz4+Mil7YnJlYWsgbH1POntQOntpZighbCl7RFtnKyhqPDwyKT4+Ml09cDtjPTE7YnJlYWsgUH1EW2crKHM8PDIpPj4yXT1hO2M9MDtpZigobHwwKT09LTQpe2JyZWFrIFB9RFtnKyhqPDwyKT4+Ml09cDtjPWorMXwwO2lmKChjfDApPT0tMSl7YnJlYWsgT319RFtnKyhjPDwyKT4+Ml09d31pZigoZXwwKSE9KHV8MCkpe0RbZT4+Ml09YTtlPWUrNHwwO0Rbays2OD4+Ml09ZTticmVhayB2fWI9ZS1kfDA7Yz1iPj4yO2U9YysxfDA7aWYoZT4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIHJ9cT1iPj4xO2U9Yz4+PjA8NTM2ODcwOTExP2U+Pj4wPnE+Pj4wP2U6cToxMDczNzQxODIzO2lmKGUpe2lmKGU+Pj4wPj0xMDczNzQxODI0KXticmVhayBnfXE9bmEoZTw8Mil9ZWxzZXtxPTB9Yz1xKyhjPDwyKXwwO0RbYz4+Ml09YTt1PShlPDwyKStxfDA7ZT1jKzR8MDtpZigoYnwwKT4wKXtvYShxLGQsYil9RFtrKzcyPj4yXT11O0Rbays2OD4+Ml09ZTtEW2srNjQ+PjJdPXE7aWYoZCl7bWEoZCl9Yj1xO2Q9Yn1jPURbaCs0MD4+Ml07aWYoKGN8MCk9PURbaCszNj4+Ml0pe2JyZWFrIG59Zj1tKyhmXi0xKXwwO2c9ZS00fDA7d2hpbGUoMSl7YT1EW2MtOD4+Ml07aWYoYT4+PjA+Zj4+PjApe2JyZWFrIHV9aWYoKGF8MCkhPShmfDApKXticmVhayBufWo9RVtjLTR8MF07YT1jLTEyfDA7Yz1EW2E+PjJdO0RbaCs0MD4+Ml09YTtpZigoY3wwKTwwKXticmVhayB1fWE9RFtnPj4yXTtEW2srMjA+PjJdPW0rKGNeLTEpO2M9aysyMHwwO0Rbays4OD4+Ml09YztxZChrLGsrNDB8MCxjLGsrODh8MCk7cD1EW2s+PjJdO1E6e2lmKGomMSl7Yz0tMTtpZigoYXwwKT09LTEpe2JyZWFrIFF9Yz1hKzF8MDtjPShjPj4+MCklM3wwP2M6YS0yfDA7YnJlYWsgUX1jPS0xO2lmKChhfDApPT0tMSl7YnJlYWsgUX1jPWEtMXwwO2lmKChhPj4+MCklM3wwKXticmVhayBRfWM9YSsyfDB9RFtwKzEyPj4yXT1jO2M9RFtoKzQwPj4yXTtpZigoY3wwKSE9RFtoKzM2Pj4yXSl7Y29udGludWV9YnJlYWt9YnJlYWsgbn1jPS0xO2lmKGkmMSl7YnJlYWsgbH1icmVhayBtfXFhKCk7VCgpfWQ9LTE7aj1EW2I+PjJdO0RbaisoZjw8Mik+PjJdPS0xO2M9LTE7YnJlYWsgb31xYSgpO1QoKX1qPURbYj4+Ml07ZD1EW2orKGk8PDIpPj4yXX1EWyhmPDwyKStqPj4yXT1kO2k9YSsxfDA7YT0oaT4+PjApJTN8MD9pOmEtMnwwO2M9LTE7aWYoKGF8MCk9PS0xKXticmVhayBvfWM9RFsoYTw8Mikraj4+Ml19RFtqK3g+PjJdPWM7Ujp7aWYoKGx8MCk9PS0xKXtEW2orcD4+Ml09LTE7aT0tMTtjPS0xO2JyZWFrIFJ9Uzp7VDp7VTp7aWYoKGw+Pj4wKSUzfDApe2M9bC0xfDA7YnJlYWsgVX1jPWwrMnwwO2lmKChjfDApPT0tMSl7YnJlYWsgVH19YT1EWyhjPDwyKStqPj4yXTtEW2orcD4+Ml09YTtpZigoYXwwKT09LTEpe2JyZWFrIFN9RFtEW2IrMjQ+PjJdKyhhPDwyKT4+Ml09czticmVhayBTfURbaitwPj4yXT0tMX1pPS0xO2E9bCsxfDA7YT0oYT4+PjApJTN8MD9hOmwtMnwwO2M9LTE7aWYoKGF8MCk9PS0xKXticmVhayBSfWk9RFsoYTw8Mikraj4+Ml07Yz1hfWI9RFtiKzI0Pj4yXTthPWIrKGk8PDIpfDA7aWYoKGR8MCkhPS0xKXtEW2IrKGQ8PDIpPj4yXT1EW2E+PjJdfVY6e2lmKChjfDApPT0tMSl7YnJlYWsgVn13aGlsZSgxKXtEWyhjPDwyKStqPj4yXT1kO2I9YysxfDA7Yj0oYj4+PjApJTN8MD9iOmMtMnwwO2lmKChifDApPT0tMSl7YnJlYWsgVn1iPURbZysoYjw8Mik+PjJdO2lmKChifDApPT0tMSl7YnJlYWsgVn1jPWIrMXwwO2M9KGM+Pj4wKSUzfDA/YzpiLTJ8MDtpZigoY3wwKSE9LTEpe2NvbnRpbnVlfWJyZWFrfX1EW2E+PjJdPS0xO1c6e2lmKEkpe2JyZWFrIFd9aWYoKHl8MCkhPSh6fDApKXtEW3k+PjJdPWk7eT15KzR8MDtEW2srMjg+PjJdPXk7YnJlYWsgV31YOntiPXotbnwwO2M9Yj4+MjthPWMrMXwwO2lmKGE+Pj4wPDEwNzM3NDE4MjQpe2Q9Yj4+MTtkPWM+Pj4wPDUzNjg3MDkxMT9hPj4+MD5kPj4+MD9hOmQ6MTA3Mzc0MTgyMztpZihkKXtpZihkPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgWH1hPW5hKGQ8PDIpfWVsc2V7YT0wfWM9YSsoYzw8Mil8MDtEW2M+PjJdPWk7ej1hKyhkPDwyKXwwO3k9Yys0fDA7aWYoKGJ8MCk+MCl7b2EoYSxuLGIpfURbayszMj4+Ml09ejtEW2srMjg+PjJdPXk7RFtrKzI0Pj4yXT1hO2lmKG4pe21hKG4pfW49YTticmVhayBXfXFhKCk7VCgpfWJyZWFrIGd9RFt3Pj4yXT1mO2I9cTtkPWJ9aT0obXwwKT4ocnwwKTtpZigobXwwKSE9KHJ8MCkpe2NvbnRpbnVlfWJyZWFrfXI9bX1jPS0xO2k9RFtoKzg+PjJdO2lmKCh0fDApPERbaSsyOD4+Ml0tRFtpKzI0Pj4yXT4+Mil7YnJlYWsgbH1pZigoZXwwKSE9KHF8MCkpe2E9aCs2MHwwO3A9aCszMTJ8MDt3aGlsZSgxKXtlPWUtNHwwO2w9RFtlPj4yXTtEW2srNjg+PjJdPWU7WTp7aWYoR2EocCkpe3Q9RFtoKzg+PjJdO3U9RFt0Pj4yXTtpZigoKERbdCs0Pj4yXS11Pj4yPj4+MCkvM3wwKTw9KHJ8MCkpe2JyZWFrIGx9Yj0tMTtkPS0xO209RFt0KzI0Pj4yXTtmPS0xO1o6e2lmKChsfDApPT0tMSl7YnJlYWsgWn1nPWwrMXwwO2c9KGc+Pj4wKSUzfDA/ZzpsLTJ8MDtmPS0xO2lmKChnfDApPT0tMSl7YnJlYWsgWn1mPURbdSsoZzw8Mik+PjJdfWc9ZjtmPURbbSsoZzw8Mik+PjJdO186e2lmKChmfDApPT0tMSl7YnJlYWsgX31pPWYrMXwwO2Y9KGk+Pj4wKSUzfDA/aTpmLTJ8MDtpZigoZnwwKT09LTEpe2JyZWFrIF99Yj1mKzF8MDtiPShiPj4+MCklM3wwP2I6Zi0yfDA7aWYoKGJ8MCkhPS0xKXtkPURbdSsoYjw8Mik+PjJdfWI9Zn1pPS0xO2o9LTE7bT1EW20rKGQ8PDIpPj4yXTtmPS0xOyQ6e2lmKChtfDApPT0tMSl7YnJlYWsgJH16PW0rMXwwO209KHo+Pj4wKSUzfDA/ejptLTJ8MDtmPS0xO2lmKChtfDApPT0tMSl7YnJlYWsgJH1mPW0rMXwwO2Y9KGY+Pj4wKSUzfDA/ZjptLTJ8MDtpZigoZnwwKSE9LTEpe2o9RFt1KyhmPDwyKT4+Ml19Zj1tfXQ9RFt0KzEyPj4yXTttPUoociwzKTt6PW08PDI7RFt0K3o+PjJdPWw7RFt0KyhsPDwyKT4+Ml09bTtsPW0rMXwwO3M9bDw8MjtEW3MrdD4+Ml09YjtEW3QrKGI8PDIpPj4yXT1sO2I9bSsyfDA7ST1iPDwyO0RbSSt0Pj4yXT1mO0RbdCsoZjw8Mik+PjJdPWI7RFt1K3o+PjJdPWQ7Zj11K3N8MDtEW2Y+PjJdPWo7aj11K0l8MDtEW2o+PjJdPWc7Zz1sPj4+MDxtPj4+MD8tMTpkO2Q9RFtoKzEyMD4+Ml07dT0oZz4+PjMmNTM2ODcwOTA4KStkfDA7dD1EW3U+PjJdO0s9dSxMPWRpKGcpJnQsRFtLPj4yXT1MO2k9KGx8MCkhPS0xP0RbZj4+Ml06aTtmPWQrKGk+Pj4zJjUzNjg3MDkwOCl8MDtnPURbZj4+Ml07Sz1mLEw9ZGkoaSkmZyxEW0s+PjJdPUw7aT0tMTtpPShifDApIT0tMT9EW2o+PjJdOmk7Yj1kKyhpPj4+MyY1MzY4NzA5MDgpfDA7ZD1EW2I+PjJdO0s9YixMPWRpKGkpJmQsRFtLPj4yXT1MO2k9RFtoKzY0Pj4yXTtiPURbaCs2OD4+Ml07aWYoKGl8MCk9PWI8PDUpe2lmKChpKzF8MCk8MCl7YnJlYWsga31pZihpPj4+MDw9MTA3Mzc0MTgyMil7ZD1pKzMyJi0zMjtiPWI8PDY7Yj1iPj4+MDxkPj4+MD9kOmJ9ZWxzZXtiPTIxNDc0ODM2NDd9VGEoYSxiKTtpPURbaCs2ND4+Ml19cj1yKzF8MDtEW2grNjQ+PjJdPWkrMTtiPURbaCs2MD4+Ml0rKGk+Pj4zJjUzNjg3MDkwOCl8MDtEW2I+PjJdPURbYj4+Ml18MTw8aTtiPURbaCs3Nj4+Ml07aWYoKGJ8MCkhPURbaCs4MD4+Ml0pe0RbYj4+Ml09bTtEW2grNzY+PjJdPWIrNDticmVhayBZfWQ9RFtoKzcyPj4yXTtmPWItZHwwO2k9Zj4+MjtiPWkrMXwwO2lmKGI+Pj4wPj0xMDczNzQxODI0KXticmVhayBqfWc9Zj4+MTtnPWk+Pj4wPDUzNjg3MDkxMT9iPj4+MD5nPj4+MD9iOmc6MTA3Mzc0MTgyMztpZihnKXtpZihnPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgZ31iPW5hKGc8PDIpfWVsc2V7Yj0wfWk9YisoaTw8Mil8MDtEW2k+PjJdPW07aWYoKGZ8MCk+MCl7b2EoYixkLGYpfURbaCs4MD4+Ml09YisoZzw8Mik7RFtoKzc2Pj4yXT1pKzQ7RFtoKzcyPj4yXT1iO2lmKCFkKXticmVhayBZfW1hKGQpO2JyZWFrIFl9Yj1EW2grNjQ+PjJdO2Q9RFtoKzY4Pj4yXTtpZigoYnwwKT09ZDw8NSl7aWYoKGIrMXwwKTwwKXticmVhayBrfWlmKGI+Pj4wPD0xMDczNzQxODIyKXtiPWIrMzImLTMyO2Q9ZDw8NjtiPWI+Pj4wPmQ+Pj4wP2I6ZH1lbHNle2I9MjE0NzQ4MzY0N31UYShhLGIpO2I9RFtoKzY0Pj4yXX1EW2grNjQ+PjJdPWIrMTtkPURbaCs2MD4+Ml0rKGI+Pj4zJjUzNjg3MDkwOCl8MDtmPURbZD4+Ml07Sz1kLEw9ZGkoYikmZixEW0s+PjJdPUw7Yj1EW2grNzY+PjJdO2lmKChifDApIT1EW2grODA+PjJdKXtEW2I+PjJdPWw7RFtoKzc2Pj4yXT1iKzQ7YnJlYWsgWX1kPURbaCs3Mj4+Ml07Zj1iLWR8MDtnPWY+PjI7Yj1nKzF8MDtpZihiPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgan1tPWY+PjE7bT1nPj4+MDw1MzY4NzA5MTE/Yj4+PjA+bT4+PjA/YjptOjEwNzM3NDE4MjM7aWYobSl7aWYobT4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGl9Yj1uYShtPDwyKX1lbHNle2I9MH1nPWIrKGc8PDIpfDA7RFtnPj4yXT1sO2lmKChmfDApPjApe29hKGIsZCxmKX1EW2grODA+PjJdPWIrKG08PDIpO0RbaCs3Nj4+Ml09Zys0O0RbaCs3Mj4+Ml09YjtpZighZCl7YnJlYWsgWX1tYShkKX1pZigoZXwwKSE9KHF8MCkpe2NvbnRpbnVlfWJyZWFrfWk9RFtoKzg+PjJdfWlmKCgoRFtpKzQ+PjJdLURbaT4+Ml0+PjI+Pj4wKS8zfDApIT0ocnwwKSl7YnJlYWsgbH1yPURbaSsyND4+Ml07Yz1EW2krMjg+PjJdLXI+PjI7aWYoKG58MCk9PSh5fDApKXtuPXk7YnJlYWsgbH1kPW47d2hpbGUoMSl7YT1EW2Q+PjJdO2U9Yy0xfDA7aj0oZTw8MikrcnwwO2lmKERbaj4+Ml09PS0xKXt3aGlsZSgxKXtlPWMtMnwwO2M9Yy0xfDA7aj0oZTw8MikrcnwwO2lmKERbaj4+Ml09PS0xKXtjb250aW51ZX1icmVha319aWYoYT4+PjA8PWU+Pj4wKXtEW2s+PjJdPWk7cj1EW2o+PjJdO0JbaysxMnwwXT0xO0Rbays4Pj4yXT1yO0Rbays0Pj4yXT1yO2lmKChyfDApIT0tMSl7d2hpbGUoMSl7RFtEW2k+PjJdKyhyPDwyKT4+Ml09YTtrYyhrKTtpPURbaCs4Pj4yXTtyPURbays4Pj4yXTtpZigocnwwKSE9LTEpe2NvbnRpbnVlfWJyZWFrfX1yPURbaSsyND4+Ml07Yj1yKyhlPDwyKXwwO2lmKChhfDApIT0tMSl7RFsoYTw8Mikrcj4+Ml09RFtiPj4yXX1EW2I+PjJdPS0xO2I9MTw8YTtmPURbaCsxMjA+PjJdO2E9ZisoYT4+PjMmNTM2ODcwOTA4KXwwO2Y9ZisoZT4+PjMmNTM2ODcwOTA4KXwwO2U9MTw8ZTtpZihEW2Y+PjJdJmUpe2I9YnxEW2E+PjJdfWVsc2V7Yj1EW2E+PjJdJihiXi0xKX1EW2E+PjJdPWI7RFtmPj4yXT1EW2Y+PjJdJihlXi0xKTtjPWMtMXwwfWQ9ZCs0fDA7aWYoKHl8MCkhPShkfDApKXtjb250aW51ZX1icmVha319aWYobil7bWEobil9ZT1EW2srNDg+PjJdO2lmKGUpe3doaWxlKDEpe2E9RFtlPj4yXTttYShlKTtlPWE7aWYoYSl7Y29udGludWV9YnJlYWt9fWE9RFtrKzQwPj4yXTtEW2srNDA+PjJdPTA7aWYoYSl7bWEoYSl9aWYocSl7RFtrKzY4Pj4yXT1xO21hKHEpfSQ9ays5NnwwO2JyZWFrIGZ9cWEoKTtUKCl9cWEoKTtUKCl9cmEoMTMyNil9VCgpfXJhKDEzMjYpO1QoKX1pZigoY3wwKT09LTEpe2JyZWFrIGV9YT1EW28rMTY+PjJdO2U9YStEW28+PjJdfDA7Yj1EW28rOD4+Ml07Yj1iLWF8MDthPURbRFtoKzQ+PjJdKzMyPj4yXTtDW2ErMzg+PjFdPUZbYSszOD4+MV07RFthPj4yXT1lO0RbYSsxNj4+Ml09MDtEW2ErMjA+PjJdPTA7RFthKzg+PjJdPWI7RFthKzEyPj4yXT0wO2FhOntpZihEW2grMjE2Pj4yXT09RFtoKzIyMD4+Ml0pe2JyZWFrIGFhfWE9RFtoKzg+PjJdO2lmKERbYSs0Pj4yXT09RFthPj4yXSl7YnJlYWsgYWF9Yj0wO3doaWxlKDEpe2lmKHRkKGgsYikpe2I9YiszfDA7YT1EW2grOD4+Ml07aWYoYj4+PjA8RFthKzQ+PjJdLURbYT4+Ml0+PjI+Pj4wKXtjb250aW51ZX1icmVhayBhYX1icmVha31icmVhayBlfWlmKEVbaCszMDh8MF0pe0JbaCszMDh8MF09MDtlPURbaCsyOTI+PjJdO2E9MDtiPURbaCszMDQ+PjJdKzd8MDthPWI+Pj4wPDc/MTphO2Q9YT4+PjN8MDtiPWE8PDI5fGI+Pj4zO2E9YitEW2grMjg4Pj4yXXwwO2Q9ZStkfDA7RFtoKzI4OD4+Ml09YTtEW2grMjkyPj4yXT1hPj4+MDxiPj4+MD9kKzF8MDpkfWI9RFtoKzIxNj4+Ml07aWYoKGJ8MCkhPURbaCsyMjA+PjJdKXt3aGlsZSgxKXthPUooQSwxNDQpO1RjKChhK2J8MCkrNHwwLERbaCs4Pj4yXSk7bj1EW0g+PjJdO2U9YStufDA7Yj1EW2UrMTMyPj4yXTtlPURbZSsxMzY+PjJdO2lmKChifDApIT0oZXwwKSl7d2hpbGUoMSl7UmMoKGErbnwwKSs0fDAsRFtiPj4yXSk7bj1EW0g+PjJdO2I9Yis0fDA7aWYoKGV8MCkhPShifDApKXtjb250aW51ZX1icmVha319U2MoKGErbnwwKSs0fDApO0E9QSsxfDA7Yj1EW2grMjE2Pj4yXTtpZihBPj4+MDwoRFtoKzIyMD4+Ml0tYnwwKS8xNDQ+Pj4wKXtjb250aW51ZX1icmVha319YT1EW2grOD4+Ml07R2IoaCsxODR8MCxEW2ErMjg+PjJdLURbYSsyND4+Ml0+PjIpO249RFtoKzIxNj4+Ml07aWYoKG58MCkhPURbaCsyMjA+PjJdKXtiPTA7d2hpbGUoMSl7YT1KKGIsMTQ0KStufDA7ZT1EW2ErNjA+PjJdLURbYSs1Nj4+Ml0+PjI7ZD1hKzEwNHwwO2E9RFtoKzg+PjJdO2E9RFthKzI4Pj4yXS1EW2ErMjQ+PjJdPj4yO0diKGQsKGF8MCk+KGV8MCk/YTplKTtiPWIrMXwwO249RFtoKzIxNj4+Ml07aWYoYj4+PjA8KERbaCsyMjA+PjJdLW58MCkvMTQ0Pj4+MCl7Y29udGludWV9YnJlYWt9fUE9c2QoaCxjKX19JD12LSAtNjR8MDtyZXR1cm4gQXwwfWZ1bmN0aW9uIGxmKGEsYixjLGQsZSxmKXthPWF8MDtiPWJ8MDtjPWN8MDtkPWR8MDtlPWV8MDtmPWZ8MDt2YXIgZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MCxvPTAscD0wLHE9MCxyPTAscz0wLHQ9MCx1PTAsdj0wLHc9MDthOntiOntjOntkOntlOntzd2l0Y2goZC0xfDApe2Nhc2UgMDphPTA7bD0kLTE2fDA7JD1sO2o9RFtiKzgwPj4yXTtnPUJbYysyNHwwXTtmOntpZigoSihqLGcpfDApIT0oZXwwKSl7YnJlYWsgZn1iPURbYysyOD4+Ml0hPTE7YT1FW2MrODR8MF07aWYoIShifCFhKSl7b2EoZixEW0RbYz4+Ml0+PjJdK0RbYys0OD4+Ml18MCxlKTthPTE7YnJlYWsgZn1kPTA7RFtsKzg+PjJdPTA7RFtsPj4yXT0wO0RbbCs0Pj4yXT0wO2lmKGcpe2lmKChnfDApPDApe2JyZWFrIGR9aD1uYShnKTtEW2w+PjJdPWg7cGEoaCwwLGcpfWc6e2g6e2lmKCFqKXticmVhayBofWlmKCFiKXtpZihnKXtvPWcmLTQ7az1nJjM7Yj0wO3M9Zy0xPj4+MDwzO3doaWxlKDEpe2U9RFtEW2M+PjJdPj4yXTtxPURbYys0MD4+Ml07YT1EW2MrNDg+PjJdKyRoKHEsRFtjKzQ0Pj4yXSxhJjI1NT9kOkRbRFtjKzY4Pj4yXSsoZDw8Mik+PjJdLDApfDA7ZT1vYShoLGUrYXwwLHEpO209MDthPTA7aT0wO2lmKCFzKXt3aGlsZSgxKXtnPWIrZnwwO0JbZ3wwXT1FW2ErZXwwXTtCW2crMXwwXT1FW2UrKGF8MSl8MF07QltnKzJ8MF09RVtlKyhhfDIpfDBdO0JbZyszfDBdPUVbZSsoYXwzKXwwXTthPWErNHwwO2I9Yis0fDA7aT1pKzR8MDtpZigob3wwKSE9KGl8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZihrKXt3aGlsZSgxKXtCW2IrZnwwXT1FW2ErZXwwXTthPWErMXwwO2I9YisxfDA7bT1tKzF8MDtpZigoa3wwKSE9KG18MCkpe2NvbnRpbnVlfWJyZWFrfX1hPTE7ZD1kKzF8MDtpZigoanwwKT09KGR8MCkpe2JyZWFrIGd9YT1FW2MrODR8MF07Y29udGludWV9fW09RFtjPj4yXTtlPURbYys0OD4+Ml07cz1EW2MrNjg+PjJdO2Y9RFtjKzQ0Pj4yXTtjPURbYys0MD4+Ml07cT1jO2I9MDtpZigoanwwKSE9MSl7cD1qJi0yO3I9YSYyNTU7d2hpbGUoMSl7Zz1ifDE7bj1EW20+PjJdO2s9JGgoYyxmLHI/YjpEW3MrKGI8PDIpPj4yXSwwKStlfDA7az1vYShoLG4ra3wwLHEpO249RFttPj4yXTtpZighcil7Zz1EW3MrKGc8PDIpPj4yXX1nPSRoKGMsZixnLDApK2V8MDtvYShrLGcrbnwwLHEpO2I9YisyfDA7aT1pKzJ8MDtpZigocHwwKSE9KGl8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZighKGomMSkpe2JyZWFrIGh9Zz1EW20+PjJdO2lmKCEoYSYyNTUpKXtiPURbcysoYjw8Mik+PjJdfWE9JGgoYyxmLGIsMCkrZXwwO29hKGgsYStnfDAscSk7YnJlYWsgaH1pZighZyl7Yj0wO2Q9MTt3aGlsZSgxKXtpZighRGIoYyxhJjI1NT9iOkRbRFtjKzY4Pj4yXSsoYjw8Mik+PjJdLEJbYysyNHwwXSxoKSl7YnJlYWsgaH1iPWIrMXwwO2Q9aj4+PjA+Yj4+PjA7aWYoKGJ8MCk9PShqfDApKXticmVhayBofWE9RVtjKzg0fDBdO2NvbnRpbnVlfX1pPWcmLTQ7az1nJjM7Yj0wO289Zy0xPj4+MDwzO2Q9MTtlPTA7d2hpbGUoMSl7aWYoIURiKGMsYSYyNTU/ZTpEW0RbYys2OD4+Ml0rKGU8PDIpPj4yXSxCW2MrMjR8MF0saCkpe2JyZWFrIGh9ZD0wO2E9MDttPTA7aWYoIW8pe3doaWxlKDEpe2c9YitmfDA7QltnfDBdPUVbYStofDBdO0JbZysxfDBdPUVbKGF8MSkraHwwXTtCW2crMnwwXT1FWyhhfDIpK2h8MF07QltnKzN8MF09RVsoYXwzKStofDBdO2E9YSs0fDA7Yj1iKzR8MDttPW0rNHwwO2lmKChpfDApIT0obXwwKSl7Y29udGludWV9YnJlYWt9fWlmKGspe3doaWxlKDEpe0JbYitmfDBdPUVbYStofDBdO2E9YSsxfDA7Yj1iKzF8MDtkPWQrMXwwO2lmKChrfDApIT0oZHwwKSl7Y29udGludWV9YnJlYWt9fWU9ZSsxfDA7aWYoKGp8MCkhPShlfDApKXtkPWU+Pj4wPGo+Pj4wO2E9RVtjKzg0fDBdO2NvbnRpbnVlfWJyZWFrfWE9ZT4+PjA+PWo+Pj4wO2JyZWFrIGd9YT1kXjE7aWYoIWgpe2JyZWFrIGZ9fW1hKGgpfWJyZWFrIGM7Y2FzZSAyOmE9MDttPSQtMTZ8MDskPW07ZD1CW2MrMjR8MF07Zz1kPDwxO2o9RFtiKzgwPj4yXTtpOntpZigoSihnLGopfDApIT0oZXwwKSl7YnJlYWsgaX1iPURbYysyOD4+Ml0hPTM7YT1FW2MrODR8MF07aWYoIShifCFhKSl7b2EoZixEW0RbYz4+Ml0+PjJdK0RbYys0OD4+Ml18MCxlKTthPTE7YnJlYWsgaX1EW20rOD4+Ml09MDtEW20+PjJdPTA7RFttKzQ+PjJdPTA7aWYoZCl7aWYoKGR8MCk8MCl7YnJlYWsgZH1oPW5hKGcpO0RbbT4+Ml09aDtwYShoLDAsZyl9ajp7azp7aWYoIWope2JyZWFrIGt9aWYoIWIpe289RFtjPj4yXTtnPURbYys0OD4+Ml07cT1EW2MrNjg+PjJdO2s9RFtjKzQ0Pj4yXTtsPURbYys0MD4+Ml07cj1sO2lmKGQpe3U9ZCYtNDt0PWQmMztiPTA7dj1hJjI1NTt3PWQtMT4+PjA8MztkPTA7d2hpbGUoMSl7Yz1EW28+PjJdO2E9JGgobCxrLHY/ZDpEW3ErKGQ8PDIpPj4yXSwwKStnfDA7aT1vYShoLGMrYXwwLHIpO2U9MDthPTA7Yz0wO2lmKCF3KXt3aGlsZSgxKXtwPShiPDwxKStmfDA7bj1hPDwxO0NbcD4+MV09RltuK2k+PjFdO0NbcCsyPj4xXT1GW2krKG58Mik+PjFdO0NbcCs0Pj4xXT1GW2krKG58NCk+PjFdO0NbcCs2Pj4xXT1GW2krKG58Nik+PjFdO2E9YSs0fDA7Yj1iKzR8MDtjPWMrNHwwO2lmKCh1fDApIT0oY3wwKSl7Y29udGludWV9YnJlYWt9fWlmKHQpe3doaWxlKDEpe0NbKGI8PDEpK2Y+PjFdPUZbaSsoYTw8MSk+PjFdO2E9YSsxfDA7Yj1iKzF8MDtlPWUrMXwwO2lmKCh0fDApIT0oZXwwKSl7Y29udGludWV9YnJlYWt9fWE9MTtkPWQrMXwwO2lmKChqfDApIT0oZHwwKSl7Y29udGludWV9YnJlYWt9YnJlYWsgan1iPTA7aWYoKGp8MCkhPTEpe3A9aiYtMjtmPWEmMjU1O2Q9MDt3aGlsZSgxKXtjPWJ8MTtuPURbbz4+Ml07ZT0kaChsLGssZj9iOkRbcSsoYjw8Mik+PjJdLDApK2d8MDtlPW9hKGgsbitlfDAscik7bj1EW28+PjJdO2lmKCFmKXtjPURbcSsoYzw8Mik+PjJdfWM9JGgobCxrLGMsMCkrZ3wwO29hKGUsYytufDAscik7Yj1iKzJ8MDtkPWQrMnwwO2lmKChwfDApIT0oZHwwKSl7Y29udGludWV9YnJlYWt9fWlmKCEoaiYxKSl7YnJlYWsga31jPURbbz4+Ml07aWYoIShhJjI1NSkpe2I9RFtxKyhiPDwyKT4+Ml19YT0kaChsLGssYiwwKStnfDA7b2EoaCxhK2N8MCxyKTticmVhayBrfWlmKCFkKXtiPTA7aT0xO3doaWxlKDEpe2lmKCFCYihjLGEmMjU1P2I6RFtEW2MrNjg+PjJdKyhiPDwyKT4+Ml0sQltjKzI0fDBdLGgpKXticmVhayBrfWI9YisxfDA7aT1qPj4+MD5iPj4+MDtpZigoYnwwKT09KGp8MCkpe2JyZWFrIGt9YT1FW2MrODR8MF07Y29udGludWV9fW89ZCYtNDtsPWQmMztiPTA7cz1kLTE+Pj4wPDM7aT0xO2Q9MDt3aGlsZSgxKXtpZighQmIoYyxhJjI1NT9kOkRbRFtjKzY4Pj4yXSsoZDw8Mik+PjJdLEJbYysyNHwwXSxoKSl7YnJlYWsga31pPTA7YT0wO2U9MDtpZighcyl7d2hpbGUoMSl7Zz0oYjw8MSkrZnwwO2s9YTw8MTtDW2c+PjFdPUZbaytoPj4xXTtDW2crMj4+MV09Rlsoa3wyKStoPj4xXTtDW2crND4+MV09Rlsoa3w0KStoPj4xXTtDW2crNj4+MV09Rlsoa3w2KStoPj4xXTthPWErNHwwO2I9Yis0fDA7ZT1lKzR8MDtpZigob3wwKSE9KGV8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZihsKXt3aGlsZSgxKXtDWyhiPDwxKStmPj4xXT1GWyhhPDwxKStoPj4xXTthPWErMXwwO2I9YisxfDA7aT1pKzF8MDtpZigobHwwKSE9KGl8MCkpe2NvbnRpbnVlfWJyZWFrfX1kPWQrMXwwO2lmKChqfDApIT0oZHwwKSl7aT1kPj4+MDxqPj4+MDthPUVbYys4NHwwXTtjb250aW51ZX1icmVha31hPWQ+Pj4wPj1qPj4+MDticmVhayBqfWE9aV4xO2lmKCFoKXticmVhayBpfX1tYShoKX1icmVhayBiO2Nhc2UgNDphPTA7bT0kLTE2fDA7JD1tO2Q9QltjKzI0fDBdO2c9ZDw8MjtqPURbYis4MD4+Ml07bDp7aWYoKEooZyxqKXwwKSE9KGV8MCkpe2JyZWFrIGx9Yj1EW2MrMjg+PjJdIT01O2E9RVtjKzg0fDBdO2lmKCEoYnwhYSkpe29hKGYsRFtEW2M+PjJdPj4yXStEW2MrNDg+PjJdfDAsZSk7YT0xO2JyZWFrIGx9RFttKzg+PjJdPTA7RFttPj4yXT0wO0RbbSs0Pj4yXT0wO2lmKGQpe2lmKChkfDApPDApe2JyZWFrIGR9aD1uYShnKTtEW20+PjJdPWg7cGEoaCwwLGcpfW06e246e2lmKCFqKXticmVhayBufWlmKCFiKXtvPURbYz4+Ml07Zz1EW2MrNDg+PjJdO3E9RFtjKzY4Pj4yXTtrPURbYys0ND4+Ml07bD1EW2MrNDA+PjJdO3I9bDtpZihkKXt1PWQmLTQ7dD1kJjM7Yj0wO3Y9YSYyNTU7dz1kLTE+Pj4wPDM7ZD0wO3doaWxlKDEpe2M9RFtvPj4yXTthPSRoKGwsayx2P2Q6RFtxKyhkPDwyKT4+Ml0sMCkrZ3wwO2k9b2EoaCxjK2F8MCxyKTtlPTA7YT0wO2M9MDtpZighdyl7d2hpbGUoMSl7cD0oYjw8MikrZnwwO249YTw8MjtEW3A+PjJdPURbbitpPj4yXTtEW3ArND4+Ml09RFtpKyhufDQpPj4yXTtEW3ArOD4+Ml09RFtpKyhufDgpPj4yXTtEW3ArMTI+PjJdPURbaSsobnwxMik+PjJdO2E9YSs0fDA7Yj1iKzR8MDtjPWMrNHwwO2lmKCh1fDApIT0oY3wwKSl7Y29udGludWV9YnJlYWt9fWlmKHQpe3doaWxlKDEpe0RbKGI8PDIpK2Y+PjJdPURbaSsoYTw8Mik+PjJdO2E9YSsxfDA7Yj1iKzF8MDtlPWUrMXwwO2lmKCh0fDApIT0oZXwwKSl7Y29udGludWV9YnJlYWt9fWE9MTtkPWQrMXwwO2lmKChqfDApIT0oZHwwKSl7Y29udGludWV9YnJlYWt9YnJlYWsgbX1iPTA7aWYoKGp8MCkhPTEpe3A9aiYtMjtmPWEmMjU1O2Q9MDt3aGlsZSgxKXtjPWJ8MTtuPURbbz4+Ml07ZT0kaChsLGssZj9iOkRbcSsoYjw8Mik+PjJdLDApK2d8MDtlPW9hKGgsbitlfDAscik7bj1EW28+PjJdO2lmKCFmKXtjPURbcSsoYzw8Mik+PjJdfWM9JGgobCxrLGMsMCkrZ3wwO29hKGUsYytufDAscik7Yj1iKzJ8MDtkPWQrMnwwO2lmKChwfDApIT0oZHwwKSl7Y29udGludWV9YnJlYWt9fWlmKCEoaiYxKSl7YnJlYWsgbn1jPURbbz4+Ml07aWYoIShhJjI1NSkpe2I9RFtxKyhiPDwyKT4+Ml19YT0kaChsLGssYiwwKStnfDA7b2EoaCxhK2N8MCxyKTticmVhayBufWlmKCFkKXtiPTA7aT0xO3doaWxlKDEpe2lmKCF6YihjLGEmMjU1P2I6RFtEW2MrNjg+PjJdKyhiPDwyKT4+Ml0sQltjKzI0fDBdLGgpKXticmVhayBufWI9YisxfDA7aT1qPj4+MD5iPj4+MDtpZigoYnwwKT09KGp8MCkpe2JyZWFrIG59YT1FW2MrODR8MF07Y29udGludWV9fW89ZCYtNDtsPWQmMztiPTA7cz1kLTE+Pj4wPDM7aT0xO2Q9MDt3aGlsZSgxKXtpZighemIoYyxhJjI1NT9kOkRbRFtjKzY4Pj4yXSsoZDw8Mik+PjJdLEJbYysyNHwwXSxoKSl7YnJlYWsgbn1pPTA7YT0wO2U9MDtpZighcyl7d2hpbGUoMSl7Zz0oYjw8MikrZnwwO2s9YTw8MjtEW2c+PjJdPURbaytoPj4yXTtEW2crND4+Ml09RFsoa3w0KStoPj4yXTtEW2crOD4+Ml09RFsoa3w4KStoPj4yXTtEW2crMTI+PjJdPURbKGt8MTIpK2g+PjJdO2E9YSs0fDA7Yj1iKzR8MDtlPWUrNHwwO2lmKChvfDApIT0oZXwwKSl7Y29udGludWV9YnJlYWt9fWlmKGwpe3doaWxlKDEpe0RbKGI8PDIpK2Y+PjJdPURbKGE8PDIpK2g+PjJdO2E9YSsxfDA7Yj1iKzF8MDtpPWkrMXwwO2lmKChsfDApIT0oaXwwKSl7Y29udGludWV9YnJlYWt9fWQ9ZCsxfDA7aWYoKGp8MCkhPShkfDApKXtpPWQ+Pj4wPGo+Pj4wO2E9RVtjKzg0fDBdO2NvbnRpbnVlfWJyZWFrfWE9ZD4+PjA+PWo+Pj4wO2JyZWFrIG19YT1pXjE7aWYoIWgpe2JyZWFrIGx9fW1hKGgpfWJyZWFrIGI7Y2FzZSAxOmE9MDtsPSQtMTZ8MDskPWw7aj1EW2IrODA+PjJdO2c9QltjKzI0fDBdO286e2lmKChKKGosZyl8MCkhPShlfDApKXticmVhayBvfWI9RFtjKzI4Pj4yXSE9MjthPUVbYys4NHwwXTtpZighKGJ8IWEpKXtvYShmLERbRFtjPj4yXT4+Ml0rRFtjKzQ4Pj4yXXwwLGUpO2E9MTticmVhayBvfWQ9MDtEW2wrOD4+Ml09MDtEW2w+PjJdPTA7RFtsKzQ+PjJdPTA7aWYoZyl7aWYoKGd8MCk8MCl7YnJlYWsgZH1oPW5hKGcpO0RbbD4+Ml09aDtwYShoLDAsZyl9cDp7cTp7aWYoIWope2JyZWFrIHF9aWYoIWIpe2lmKGcpe289ZyYtNDtrPWcmMztiPTA7cz1nLTE+Pj4wPDM7d2hpbGUoMSl7ZT1EW0RbYz4+Ml0+PjJdO3E9RFtjKzQwPj4yXTthPURbYys0OD4+Ml0rJGgocSxEW2MrNDQ+PjJdLGEmMjU1P2Q6RFtEW2MrNjg+PjJdKyhkPDwyKT4+Ml0sMCl8MDtlPW9hKGgsZSthfDAscSk7bT0wO2E9MDtpPTA7aWYoIXMpe3doaWxlKDEpe2c9YitmfDA7QltnfDBdPUVbYStlfDBdO0JbZysxfDBdPUVbZSsoYXwxKXwwXTtCW2crMnwwXT1FW2UrKGF8Mil8MF07QltnKzN8MF09RVtlKyhhfDMpfDBdO2E9YSs0fDA7Yj1iKzR8MDtpPWkrNHwwO2lmKChvfDApIT0oaXwwKSl7Y29udGludWV9YnJlYWt9fWlmKGspe3doaWxlKDEpe0JbYitmfDBdPUVbYStlfDBdO2E9YSsxfDA7Yj1iKzF8MDttPW0rMXwwO2lmKChrfDApIT0obXwwKSl7Y29udGludWV9YnJlYWt9fWE9MTtkPWQrMXwwO2lmKChqfDApPT0oZHwwKSl7YnJlYWsgcH1hPUVbYys4NHwwXTtjb250aW51ZX19bT1EW2M+PjJdO2U9RFtjKzQ4Pj4yXTtzPURbYys2OD4+Ml07Zj1EW2MrNDQ+PjJdO2M9RFtjKzQwPj4yXTtxPWM7Yj0wO2lmKChqfDApIT0xKXtwPWomLTI7cj1hJjI1NTt3aGlsZSgxKXtnPWJ8MTtuPURbbT4+Ml07az0kaChjLGYscj9iOkRbcysoYjw8Mik+PjJdLDApK2V8MDtrPW9hKGgsbitrfDAscSk7bj1EW20+PjJdO2lmKCFyKXtnPURbcysoZzw8Mik+PjJdfWc9JGgoYyxmLGcsMCkrZXwwO29hKGssZytufDAscSk7Yj1iKzJ8MDtpPWkrMnwwO2lmKChwfDApIT0oaXwwKSl7Y29udGludWV9YnJlYWt9fWlmKCEoaiYxKSl7YnJlYWsgcX1nPURbbT4+Ml07aWYoIShhJjI1NSkpe2I9RFtzKyhiPDwyKT4+Ml19YT0kaChjLGYsYiwwKStlfDA7b2EoaCxhK2d8MCxxKTticmVhayBxfWlmKCFnKXtiPTA7ZD0xO3doaWxlKDEpe2lmKCFDYihjLGEmMjU1P2I6RFtEW2MrNjg+PjJdKyhiPDwyKT4+Ml0sQltjKzI0fDBdLGgpKXticmVhayBxfWI9YisxfDA7ZD1qPj4+MD5iPj4+MDtpZigoYnwwKT09KGp8MCkpe2JyZWFrIHF9YT1FW2MrODR8MF07Y29udGludWV9fWk9ZyYtNDtrPWcmMztiPTA7bz1nLTE+Pj4wPDM7ZD0xO2U9MDt3aGlsZSgxKXtpZighQ2IoYyxhJjI1NT9lOkRbRFtjKzY4Pj4yXSsoZTw8Mik+PjJdLEJbYysyNHwwXSxoKSl7YnJlYWsgcX1kPTA7YT0wO209MDtpZighbyl7d2hpbGUoMSl7Zz1iK2Z8MDtCW2d8MF09RVthK2h8MF07QltnKzF8MF09RVsoYXwxKStofDBdO0JbZysyfDBdPUVbKGF8MikraHwwXTtCW2crM3wwXT1FWyhhfDMpK2h8MF07YT1hKzR8MDtiPWIrNHwwO209bSs0fDA7aWYoKGl8MCkhPShtfDApKXtjb250aW51ZX1icmVha319aWYoayl7d2hpbGUoMSl7QltiK2Z8MF09RVthK2h8MF07YT1hKzF8MDtiPWIrMXwwO2Q9ZCsxfDA7aWYoKGt8MCkhPShkfDApKXtjb250aW51ZX1icmVha319ZT1lKzF8MDtpZigoanwwKSE9KGV8MCkpe2Q9ZT4+PjA8aj4+PjA7YT1FW2MrODR8MF07Y29udGludWV9YnJlYWt9YT1lPj4+MD49aj4+PjA7YnJlYWsgcH1hPWReMTtpZighaCl7YnJlYWsgb319bWEoaCl9YnJlYWsgYztjYXNlIDM6YT0wO209JC0xNnwwOyQ9bTtkPUJbYysyNHwwXTtnPWQ8PDE7aj1EW2IrODA+PjJdO3I6e2lmKChKKGcsail8MCkhPShlfDApKXticmVhayByfWI9RFtjKzI4Pj4yXSE9NDthPUVbYys4NHwwXTtpZighKGJ8IWEpKXtvYShmLERbRFtjPj4yXT4+Ml0rRFtjKzQ4Pj4yXXwwLGUpO2E9MTticmVhayByfURbbSs4Pj4yXT0wO0RbbT4+Ml09MDtEW20rND4+Ml09MDtpZihkKXtpZigoZHwwKTwwKXticmVhayBkfWg9bmEoZyk7RFttPj4yXT1oO3BhKGgsMCxnKX1zOnt0OntpZighail7YnJlYWsgdH1pZighYil7bz1EW2M+PjJdO2c9RFtjKzQ4Pj4yXTtxPURbYys2OD4+Ml07az1EW2MrNDQ+PjJdO2w9RFtjKzQwPj4yXTtyPWw7aWYoZCl7dT1kJi00O3Q9ZCYzO2I9MDt2PWEmMjU1O3c9ZC0xPj4+MDwzO2Q9MDt3aGlsZSgxKXtjPURbbz4+Ml07YT0kaChsLGssdj9kOkRbcSsoZDw8Mik+PjJdLDApK2d8MDtpPW9hKGgsYythfDAscik7ZT0wO2E9MDtjPTA7aWYoIXcpe3doaWxlKDEpe3A9KGI8PDEpK2Z8MDtuPWE8PDE7Q1twPj4xXT1GW24raT4+MV07Q1twKzI+PjFdPUZbaSsobnwyKT4+MV07Q1twKzQ+PjFdPUZbaSsobnw0KT4+MV07Q1twKzY+PjFdPUZbaSsobnw2KT4+MV07YT1hKzR8MDtiPWIrNHwwO2M9Yys0fDA7aWYoKHV8MCkhPShjfDApKXtjb250aW51ZX1icmVha319aWYodCl7d2hpbGUoMSl7Q1soYjw8MSkrZj4+MV09RltpKyhhPDwxKT4+MV07YT1hKzF8MDtiPWIrMXwwO2U9ZSsxfDA7aWYoKHR8MCkhPShlfDApKXtjb250aW51ZX1icmVha319YT0xO2Q9ZCsxfDA7aWYoKGp8MCkhPShkfDApKXtjb250aW51ZX1icmVha31icmVhayBzfWI9MDtpZigoanwwKSE9MSl7cD1qJi0yO2Y9YSYyNTU7ZD0wO3doaWxlKDEpe2M9YnwxO249RFtvPj4yXTtlPSRoKGwsayxmP2I6RFtxKyhiPDwyKT4+Ml0sMCkrZ3wwO2U9b2EoaCxuK2V8MCxyKTtuPURbbz4+Ml07aWYoIWYpe2M9RFtxKyhjPDwyKT4+Ml19Yz0kaChsLGssYywwKStnfDA7b2EoZSxjK258MCxyKTtiPWIrMnwwO2Q9ZCsyfDA7aWYoKHB8MCkhPShkfDApKXtjb250aW51ZX1icmVha319aWYoIShqJjEpKXticmVhayB0fWM9RFtvPj4yXTtpZighKGEmMjU1KSl7Yj1EW3ErKGI8PDIpPj4yXX1hPSRoKGwsayxiLDApK2d8MDtvYShoLGErY3wwLHIpO2JyZWFrIHR9aWYoIWQpe2I9MDtpPTE7d2hpbGUoMSl7aWYoIUFiKGMsYSYyNTU/YjpEW0RbYys2OD4+Ml0rKGI8PDIpPj4yXSxCW2MrMjR8MF0saCkpe2JyZWFrIHR9Yj1iKzF8MDtpPWo+Pj4wPmI+Pj4wO2lmKChifDApPT0oanwwKSl7YnJlYWsgdH1hPUVbYys4NHwwXTtjb250aW51ZX19bz1kJi00O2w9ZCYzO2I9MDtzPWQtMT4+PjA8MztpPTE7ZD0wO3doaWxlKDEpe2lmKCFBYihjLGEmMjU1P2Q6RFtEW2MrNjg+PjJdKyhkPDwyKT4+Ml0sQltjKzI0fDBdLGgpKXticmVhayB0fWk9MDthPTA7ZT0wO2lmKCFzKXt3aGlsZSgxKXtnPShiPDwxKStmfDA7az1hPDwxO0NbZz4+MV09RltrK2g+PjFdO0NbZysyPj4xXT1GWyhrfDIpK2g+PjFdO0NbZys0Pj4xXT1GWyhrfDQpK2g+PjFdO0NbZys2Pj4xXT1GWyhrfDYpK2g+PjFdO2E9YSs0fDA7Yj1iKzR8MDtlPWUrNHwwO2lmKChvfDApIT0oZXwwKSl7Y29udGludWV9YnJlYWt9fWlmKGwpe3doaWxlKDEpe0NbKGI8PDEpK2Y+PjFdPUZbKGE8PDEpK2g+PjFdO2E9YSsxfDA7Yj1iKzF8MDtpPWkrMXwwO2lmKChsfDApIT0oaXwwKSl7Y29udGludWV9YnJlYWt9fWQ9ZCsxfDA7aWYoKGp8MCkhPShkfDApKXtpPWQ+Pj4wPGo+Pj4wO2E9RVtjKzg0fDBdO2NvbnRpbnVlfWJyZWFrfWE9ZD4+PjA+PWo+Pj4wO2JyZWFrIHN9YT1pXjE7aWYoIWgpe2JyZWFrIHJ9fW1hKGgpfWJyZWFrIGI7Y2FzZSA1OmE9MDttPSQtMTZ8MDskPW07ZD1CW2MrMjR8MF07Zz1kPDwyO2o9RFtiKzgwPj4yXTt1OntpZigoSihnLGopfDApIT0oZXwwKSl7YnJlYWsgdX1iPURbYysyOD4+Ml0hPTY7YT1FW2MrODR8MF07aWYoIShifCFhKSl7b2EoZixEW0RbYz4+Ml0+PjJdK0RbYys0OD4+Ml18MCxlKTthPTE7YnJlYWsgdX1EW20rOD4+Ml09MDtEW20+PjJdPTA7RFttKzQ+PjJdPTA7aWYoZCl7aWYoKGR8MCk8MCl7YnJlYWsgZH1oPW5hKGcpO0RbbT4+Ml09aDtwYShoLDAsZyl9djp7dzp7aWYoIWope2JyZWFrIHd9aWYoIWIpe289RFtjPj4yXTtnPURbYys0OD4+Ml07cT1EW2MrNjg+PjJdO2s9RFtjKzQ0Pj4yXTtsPURbYys0MD4+Ml07cj1sO2lmKGQpe3U9ZCYtNDt0PWQmMztiPTA7dj1hJjI1NTt3PWQtMT4+PjA8MztkPTA7d2hpbGUoMSl7Yz1EW28+PjJdO2E9JGgobCxrLHY/ZDpEW3ErKGQ8PDIpPj4yXSwwKStnfDA7aT1vYShoLGMrYXwwLHIpO2U9MDthPTA7Yz0wO2lmKCF3KXt3aGlsZSgxKXtwPShiPDwyKStmfDA7bj1hPDwyO0RbcD4+Ml09RFtuK2k+PjJdO0RbcCs0Pj4yXT1EW2krKG58NCk+PjJdO0RbcCs4Pj4yXT1EW2krKG58OCk+PjJdO0RbcCsxMj4+Ml09RFtpKyhufDEyKT4+Ml07YT1hKzR8MDtiPWIrNHwwO2M9Yys0fDA7aWYoKHV8MCkhPShjfDApKXtjb250aW51ZX1icmVha319aWYodCl7d2hpbGUoMSl7RFsoYjw8MikrZj4+Ml09RFtpKyhhPDwyKT4+Ml07YT1hKzF8MDtiPWIrMXwwO2U9ZSsxfDA7aWYoKHR8MCkhPShlfDApKXtjb250aW51ZX1icmVha319YT0xO2Q9ZCsxfDA7aWYoKGp8MCkhPShkfDApKXtjb250aW51ZX1icmVha31icmVhayB2fWI9MDtpZigoanwwKSE9MSl7cD1qJi0yO2Y9YSYyNTU7ZD0wO3doaWxlKDEpe2M9YnwxO249RFtvPj4yXTtlPSRoKGwsayxmP2I6RFtxKyhiPDwyKT4+Ml0sMCkrZ3wwO2U9b2EoaCxuK2V8MCxyKTtuPURbbz4+Ml07aWYoIWYpe2M9RFtxKyhjPDwyKT4+Ml19Yz0kaChsLGssYywwKStnfDA7b2EoZSxjK258MCxyKTtiPWIrMnwwO2Q9ZCsyfDA7aWYoKHB8MCkhPShkfDApKXtjb250aW51ZX1icmVha319aWYoIShqJjEpKXticmVhayB3fWM9RFtvPj4yXTtpZighKGEmMjU1KSl7Yj1EW3ErKGI8PDIpPj4yXX1hPSRoKGwsayxiLDApK2d8MDtvYShoLGErY3wwLHIpO2JyZWFrIHd9aWYoIWQpe2I9MDtpPTE7d2hpbGUoMSl7aWYoIXliKGMsYSYyNTU/YjpEW0RbYys2OD4+Ml0rKGI8PDIpPj4yXSxCW2MrMjR8MF0saCkpe2JyZWFrIHd9Yj1iKzF8MDtpPWo+Pj4wPmI+Pj4wO2lmKChifDApPT0oanwwKSl7YnJlYWsgd31hPUVbYys4NHwwXTtjb250aW51ZX19bz1kJi00O2w9ZCYzO2I9MDtzPWQtMT4+PjA8MztpPTE7ZD0wO3doaWxlKDEpe2lmKCF5YihjLGEmMjU1P2Q6RFtEW2MrNjg+PjJdKyhkPDwyKT4+Ml0sQltjKzI0fDBdLGgpKXticmVhayB3fWk9MDthPTA7ZT0wO2lmKCFzKXt3aGlsZSgxKXtnPShiPDwyKStmfDA7az1hPDwyO0RbZz4+Ml09RFtrK2g+PjJdO0RbZys0Pj4yXT1EWyhrfDQpK2g+PjJdO0RbZys4Pj4yXT1EWyhrfDgpK2g+PjJdO0RbZysxMj4+Ml09RFsoa3wxMikraD4+Ml07YT1hKzR8MDtiPWIrNHwwO2U9ZSs0fDA7aWYoKG98MCkhPShlfDApKXtjb250aW51ZX1icmVha319aWYobCl7d2hpbGUoMSl7RFsoYjw8MikrZj4+Ml09RFsoYTw8MikraD4+Ml07YT1hKzF8MDtiPWIrMXwwO2k9aSsxfDA7aWYoKGx8MCkhPShpfDApKXtjb250aW51ZX1icmVha319ZD1kKzF8MDtpZigoanwwKSE9KGR8MCkpe2k9ZD4+PjA8aj4+PjA7YT1FW2MrODR8MF07Y29udGludWV9YnJlYWt9YT1kPj4+MD49aj4+PjA7YnJlYWsgdn1hPWleMTtpZighaCl7YnJlYWsgdX19bWEoaCl9YnJlYWsgYjtjYXNlIDg6YnJlYWsgZTtkZWZhdWx0OmJyZWFrIGF9fWE9MDtvPSQtMTZ8MDskPW87az1CW2MrMjR8MF07ZD1rPDwyO2o9RFtiKzgwPj4yXTt4OntpZigoSihkLGopfDApIT0oZXwwKSl7YnJlYWsgeH1lPURbYysyOD4+Ml07RFtvKzg+PjJdPTA7RFtvPj4yXT0wO0Rbbys0Pj4yXT0wO2I9MDt5Ont6OntBOntCOntpZighayl7YnJlYWsgQn1pZigoa3wwKTwwKXticmVhayBBfWI9bmEoZCk7RFtvPj4yXT1iO2c9KGs8PDIpK2J8MDtEW28rOD4+Ml09ZztkPWQtNHwwO2w9KGQ+Pj4yfDApKzEmNztDOntpZighbCl7YT1iO2JyZWFrIEN9YT1iO3doaWxlKDEpe0RbYT4+Ml09LTEwNzM3NDE4MjQ7YT1hKzR8MDtoPWgrMXwwO2lmKChsfDApIT0oaHwwKSl7Y29udGludWV9YnJlYWt9fWlmKGQ+Pj4wPDI4KXticmVhayBCfXdoaWxlKDEpe0RbYSsyND4+Ml09LTEwNzM3NDE4MjQ7RFthKzI4Pj4yXT0tMTA3Mzc0MTgyNDtEW2ErMTY+PjJdPS0xMDczNzQxODI0O0RbYSsyMD4+Ml09LTEwNzM3NDE4MjQ7RFthKzg+PjJdPS0xMDczNzQxODI0O0RbYSsxMj4+Ml09LTEwNzM3NDE4MjQ7RFthPj4yXT0tMTA3Mzc0MTgyNDtEW2ErND4+Ml09LTEwNzM3NDE4MjQ7YT1hKzMyfDA7aWYoKGd8MCkhPShhfDApKXtjb250aW51ZX1icmVha319aWYoIWope2JyZWFrIHp9aWYoKGV8MCk9PTkpe2E9MDtzPURbYz4+Ml07ZD1EW2MrNDg+PjJdO3I9RFtjKzY4Pj4yXTtwPUVbYys4NHwwXTtlPURbYys0ND4+Ml07Zz1EW2MrNDA+PjJdO249ZztpZigoa3wwKTw9MCl7aWYoKGp8MCkhPTEpe2s9aiYtMjtjPTA7d2hpbGUoMSl7Zj1hfDE7bD1EW3M+PjJdO2g9JGgoZyxlLHA/YTpEW3IrKGE8PDIpPj4yXSwwKStkfDA7aD1vYShiLGwraHwwLG4pO2w9RFtzPj4yXTtpZighcCl7Zj1EW3IrKGY8PDIpPj4yXX1mPSRoKGcsZSxmLDApK2R8MDtvYShoLGYrbHwwLG4pO2E9YSsyfDA7Yz1jKzJ8MDtpZigoa3wwKSE9KGN8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZighKGomMSkpe2JyZWFrIHp9Yz1EW3M+PjJdO2lmKCFwKXthPURbcisoYTw8Mik+PjJdfWE9JGgoZyxlLGEsMCkrZHwwO29hKGIsYStjfDAsbik7YnJlYWsgen12PWsmLTQ7dD1rJjM7aD0wO3c9ay0xPj4+MDwzO3doaWxlKDEpe2M9RFtzPj4yXTthPSRoKGcsZSxwP2k6RFtyKyhpPDwyKT4+Ml0sMCkrZHwwO2M9b2EoYixjK2F8MCxuKTtsPTA7YT0wO3U9MDtpZighdyl7d2hpbGUoMSl7az0oaDw8MikrZnwwO209YTw8MjtIW2s+PjJdPUhbbStjPj4yXTtIW2srND4+Ml09SFtjKyhtfDQpPj4yXTtIW2srOD4+Ml09SFtjKyhtfDgpPj4yXTtIW2srMTI+PjJdPUhbYysobXwxMik+PjJdO2E9YSs0fDA7aD1oKzR8MDt1PXUrNHwwO2lmKCh2fDApIT0odXwwKSl7Y29udGludWV9YnJlYWt9fWlmKHQpe3doaWxlKDEpe0hbKGg8PDIpK2Y+PjJdPUhbYysoYTw8Mik+PjJdO2E9YSsxfDA7aD1oKzF8MDtsPWwrMXwwO2lmKCh0fDApIT0obHwwKSl7Y29udGludWV9YnJlYWt9fWE9MTtpPWkrMXwwO2lmKChqfDApIT0oaXwwKSl7Y29udGludWV9YnJlYWt9YnJlYWsgeX1tPTE7aWYoKGt8MCk8PTApe2E9MDt3aGlsZSgxKXtpZighbWIoYyxFW2MrODR8MF0/YTpEW0RbYys2OD4+Ml0rKGE8PDIpPj4yXSxCW2MrMjR8MF0sYikpe2JyZWFrIHp9YT1hKzF8MDttPWo+Pj4wPmE+Pj4wO2lmKChhfDApIT0oanwwKSl7Y29udGludWV9YnJlYWt9YnJlYWsgen1zPWsmLTQ7Zz1rJjM7aD0wO2s9ay0xPj4+MDwzO3doaWxlKDEpe2lmKCFtYihjLEVbYys4NHwwXT9pOkRbRFtjKzY4Pj4yXSsoaTw8Mik+PjJdLEJbYysyNHwwXSxiKSl7YnJlYWsgen1tPTA7YT0wO2w9MDtpZighayl7d2hpbGUoMSl7ZD0oaDw8MikrZnwwO2U9YTw8MjtIW2Q+PjJdPUhbZStiPj4yXTtIW2QrND4+Ml09SFsoZXw0KStiPj4yXTtIW2QrOD4+Ml09SFsoZXw4KStiPj4yXTtIW2QrMTI+PjJdPUhbKGV8MTIpK2I+PjJdO2E9YSs0fDA7aD1oKzR8MDtsPWwrNHwwO2lmKChzfDApIT0obHwwKSl7Y29udGludWV9YnJlYWt9fWlmKGcpe3doaWxlKDEpe0hbKGg8PDIpK2Y+PjJdPUhbKGE8PDIpK2I+PjJdO2E9YSsxfDA7aD1oKzF8MDttPW0rMXwwO2lmKChnfDApIT0obXwwKSl7Y29udGludWV9YnJlYWt9fWk9aSsxfDA7bT1qPj4+MD5pPj4+MDtpZigoaXwwKSE9KGp8MCkpe2NvbnRpbnVlfWJyZWFrfWE9aT4+PjA+PWo+Pj4wO2JyZWFrIHl9cWEoKTtUKCl9YT1tXjE7aWYoIWIpe2JyZWFrIHh9fW1hKGIpfSQ9bysxNnwwO2g9YSYxO2JyZWFrIGF9cWEoKTtUKCl9JD1sKzE2fDA7aD1hJjE7YnJlYWsgYX0kPW0rMTZ8MDtoPWEmMX1yZXR1cm4gaHwwfWZ1bmN0aW9uIHdmKGEsYixjKXthPWF8MDtiPWJ8MDtjPWN8MDt2YXIgZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MCxvPTAscD0wLHE9MCxyPTAscz0wLHQ9MCx1PTAsdj0wLHc9MCx4PTAseT0wLHo9MCxBPTAsQz0wLEY9MCxHPTAsSD0wLEk9MCxLPTAsTD0wLE09MCxOPTA7ej1jO2M9MDttPSQtOTZ8MDskPW07az1tKzE2fDA7cGEoaywwLDc2KTtEW20rOTI+PjJdPS0xO0RbbSs4Pj4yXT0wO0RbbT4+Ml09MDtEW20rND4+Ml09MDtyPSQtMTZ8MDskPXI7RFtrKzY4Pj4yXT0wO0Rbays3Mj4+Ml09MDtEW2s+PjJdPWI7dD0kLTE2fDA7JD10O3g9YjthPURbYisyMD4+Ml07YTp7aWYoKERbYisyND4+Ml0tYXwwKTw9MCl7YnJlYWsgYX1hPURbYT4+Ml07aWYoKGF8MCk9PS0xKXticmVhayBhfWM9RFtEW2IrOD4+Ml0rKGE8PDIpPj4yXX1iOntjOntkOntpZighYyl7YT0wO2JyZWFrIGR9YT1EW3grMTAwPj4yXTtmPURbeCs5Nj4+Ml07RFt0Kzg+PjJdPTA7RFt0Pj4yXT0wO0RbdCs0Pj4yXT0wO2g9YS1mfDA7Yj0oaHwwKS8xMnwwO2U6e2lmKCFoKXticmVhayBlfWlmKGI+Pj4wPj0zNTc5MTM5NDIpe2JyZWFrIGN9ZD1uYShoKTtEW3Q+PjJdPWQ7RFt0Kzg+PjJdPWQrSihiLDEyKTthPTA7aT1kO2Q9SigoaC0xMj4+PjApLzEyfDAsMTIpKzEyfDA7aD1wYShpLDAsZCk7RFt0KzQ+PjJdPWQraDtpZihFW2MrODR8MF0pe2M9Yj4+PjA+MT9iOjE7ZD1jJjE7aWYoYj4+PjA+PTIpe2k9YyYtMjt3aGlsZSgxKXtjPUooYSwxMik7Yj1jK2Z8MDtsPURbYis0Pj4yXTtuPURbYj4+Ml07Yz1jK2h8MDtEW2MrOD4+Ml09RFtiKzg+PjJdO0RbYz4+Ml09bjtEW2MrND4+Ml09bDtjPUooYXwxLDEyKTtiPWMrZnwwO2w9RFtiKzg+PjJdO249RFtiKzQ+PjJdO2M9YytofDA7RFtjPj4yXT1EW2I+PjJdO0RbYys0Pj4yXT1uO0RbYys4Pj4yXT1sO2E9YSsyfDA7ZT1lKzJ8MDtpZigoaXwwKSE9KGV8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZighZCl7YnJlYWsgZX1iPUooYSwxMik7YT1iK2Z8MDtjPURbYSs0Pj4yXTtlPURbYT4+Ml07Yj1iK2h8MDtEW2IrOD4+Ml09RFthKzg+PjJdO0RbYj4+Ml09ZTtEW2IrND4+Ml09YzticmVhayBlfWQ9Yj4+PjA+MT9iOjE7YT1EW2MrNjg+PjJdO3doaWxlKDEpe2M9SihlLDEyKTtiPWMrZnwwO2k9RFthKyhEW2I+PjJdPDwyKT4+Ml07bD1EW2ErKERbYis0Pj4yXTw8Mik+PjJdO2M9YytofDA7RFtjKzg+PjJdPURbYSsoRFtiKzg+PjJdPDwyKT4+Ml07RFtjKzQ+PjJdPWw7RFtjPj4yXT1pO2U9ZSsxfDA7aWYoKGR8MCkhPShlfDApKXtjb250aW51ZX1icmVha319Rj0kLTE2fDA7JD1GO2Q9bmEoODgpO0RbZD4+Ml09MDtEW2QrND4+Ml09MDtEW2QrNTY+PjJdPTA7RFtkKzQ4Pj4yXT0wO0RbZCs1Mj4+Ml09MDtEW2QrNDA+PjJdPTA7RFtkKzQ0Pj4yXT0wO0RbZCszMj4+Ml09MDtEW2QrMzY+PjJdPTA7RFtkKzI0Pj4yXT0wO0RbZCsyOD4+Ml09MDtEW2QrMTY+PjJdPTA7RFtkKzIwPj4yXT0wO0RbZCs4Pj4yXT0wO0RbZCsxMj4+Ml09MDthPWQtIC02NHwwO0RbYT4+Ml09MDtEW2ErND4+Ml09MDtEW2QrNzI+PjJdPTA7RFtkKzc2Pj4yXT0wO0RbZCs4MD4+Ml09MDtEW2QrODQ+PjJdPTA7RFtkKzYwPj4yXT1kO0E9JC0xNnwwOyQ9QTtEW2QrODA+PjJdPTA7RFtkKzg0Pj4yXT0wO2E9RFtkKzc2Pj4yXTtEW2QrNzY+PjJdPTA7aWYoYSl7bWEoYSl9RFtkKzY4Pj4yXT0wO0RbZCs3Mj4+Ml09MDtiPWQtIC02NHwwO2E9RFtiPj4yXTtEW2I+PjJdPTA7aWYoYSl7bWEoYSl9ZT1EW3Q+PjJdO2I9RFt0KzQ+PjJdLWV8MDtsPShifDApLzEyfDA7YT1KKGwsMyk7Yz1EW2Q+PjJdO2Y9RFtkKzQ+PjJdLWM+PjI7Zjp7aWYoYT4+PjA+Zj4+PjApe3NhKGQsYS1mfDApO2U9RFt0Pj4yXTtiPURbdCs0Pj4yXS1lfDA7bD0oYnwwKS8xMnwwO2M9RFtkPj4yXTticmVhayBmfWlmKGE+Pj4wPj1mPj4+MCl7YnJlYWsgZn1EW2QrND4+Ml09KGE8PDIpK2N9aWYoYil7aD1sPj4+MD4xP2w6MTthPTA7d2hpbGUoMSl7Zj1KKGEsMTIpO2I9ZitjfDA7Zj1mK2V8MDtEW2I+PjJdPURbZj4+Ml07RFtiKzQ+PjJdPURbZis0Pj4yXTtEW2IrOD4+Ml09RFtmKzg+PjJdO2E9YSsxfDA7aWYoKGh8MCkhPShhfDApKXtjb250aW51ZX1icmVha319RFtBKzEyPj4yXT0tMTtmPSQtNDh8MDskPWY7Zzp7aDp7aTp7dz1BKzEyfDA7ajp7aWYoIXcpe2JyZWFrIGp9aT1EW2QrND4+Ml07bj1EW2Q+PjJdO2I9aS1ufDA7cD1iPj4yO2M9RFtkKzEyPj4yXTthPURbZCsxNj4+Ml0tYz4+MjtrOntpZihwPj4+MD5hPj4+MCl7eGEoZCsxMnwwLHAtYXwwLDEwMjYwKTtpPURbZCs0Pj4yXTtuPURbZD4+Ml07Yj1pLW58MDtwPWI+PjI7YnJlYWsga31pZihhPj4+MDw9cD4+PjApe2JyZWFrIGt9RFtkKzE2Pj4yXT1jKyhwPDwyKX1jPTA7RFtmKzQwPj4yXT0wO0RbZiszMj4+Ml09MDtEW2YrMzY+PjJdPTA7bDp7aWYoIWIpe0RbZisyND4+Ml09MDtEW2YrMTY+PjJdPTA7RFtmKzIwPj4yXT0wO2JyZWFrIGx9bTp7aWYoKGJ8MCk+PTApe2M9bmEoYik7RFtmKzM2Pj4yXT1jO0RbZiszMj4+Ml09YztEW2YrNDA+PjJdPShwPDwyKStjO2E9YztiPTA7d2hpbGUoMSl7aD1EWyhiPDwyKStuPj4yXTthPWEtYz4+MjtuOntpZihoPj4+MDxhPj4+MCl7YnJlYWsgbn1EW2YrMTY+PjJdPTA7ZT1oKzF8MDtpZihlPj4+MD5hPj4+MCl7eGEoZiszMnwwLGUtYXwwLGYrMTZ8MCk7bj1EW2Q+PjJdO2k9RFtkKzQ+PjJdO2M9RFtmKzMyPj4yXTticmVhayBufWlmKGE+Pj4wPD1lPj4+MCl7YnJlYWsgbn1EW2YrMzY+PjJdPShlPDwyKStjfWE9KGg8PDIpK2N8MDtEW2E+PjJdPURbYT4+Ml0rMTtiPWIrMXwwO2E9aS1ufDA7cD1hPj4yO2lmKGI+Pj4wPj1wPj4+MCl7YnJlYWsgbX1hPURbZiszNj4+Ml07Y29udGludWV9fWJyZWFrIGl9RFtmKzI0Pj4yXT0wO0RbZisxNj4+Ml09MDtEW2YrMjA+PjJdPTA7aWYoIWEpe2JyZWFrIGx9aWYocD4+PjA+PTUzNjg3MDkxMil7YnJlYWsgaH1iPWE8PDE7YT1uYShiKTtEW2YrMTY+PjJdPWE7ZT1hKyhwPDwzKXwwO0RbZisyND4+Ml09ZTtwYShhLDI1NSxiKTtEW2YrMjA+PjJdPWV9aT0wO0RbZis4Pj4yXT0wO0RbZj4+Ml09MDtEW2YrND4+Ml09MDthPURbZiszNj4+Ml0tY3wwO2g9YT4+MjtvOntpZighYSl7YnJlYWsgb31pZigoYXwwKTwwKXticmVhayBofXM9bmEoYSk7RFtmPj4yXT1zO0RbZis4Pj4yXT0oaDw8MikrcztiPWE7YT1wYShzLDAsYSk7RFtmKzQ+PjJdPWIrYTtlPWg+Pj4wPjE/aDoxO2w9ZSYzO2I9MDtpZihlLTE+Pj4wPj0zKXtxPWUmLTQ7d2hpbGUoMSl7ZT1nPDwyO0RbZSthPj4yXT1iO3k9ZXw0O2I9RFtjK2U+PjJdK2J8MDtEW3krYT4+Ml09Yjt2PWV8ODtiPWIrRFtjK3k+PjJdfDA7RFt2K2E+PjJdPWI7ZT1lfDEyO2I9YitEW2Mrdj4+Ml18MDtEW2UrYT4+Ml09YjtiPWIrRFtjK2U+PjJdfDA7Zz1nKzR8MDtqPWorNHwwO2lmKChxfDApIT0oanwwKSl7Y29udGludWV9YnJlYWt9fWlmKCFsKXticmVhayBvfXdoaWxlKDEpe2U9Zzw8MjtEW2UrYT4+Ml09YjtnPWcrMXwwO2I9RFtjK2U+PjJdK2J8MDtvPW8rMXwwO2lmKChsfDApIT0ob3wwKSl7Y29udGludWV9YnJlYWt9fWlmKHApe3k9RFtkKzEyPj4yXTt3aGlsZSgxKXtHPWk8PDI7Yj1HK258MDtqPS0xO2U9aSsxfDA7YT0oZT4+PjApJTN8MD9lOmktMnwwO2lmKChhfDApIT0tMSl7aj1EWyhhPDwyKStuPj4yXX1hPURbYj4+Ml07cDp7cTp7aWYoISgoaT4+PjApJTN8MCkpe289LTE7Yj1pKzJ8MDtpZigoYnwwKSE9LTEpe289RFsoYjw8Mikrbj4+Ml19aWYoISgoYXwwKT09KGp8MCl8KGF8MCk9PShvfDApKSYoanwwKSE9KG98MCkpe2JyZWFrIHF9RFtkKzQwPj4yXT1EW2QrNDA+PjJdKzE7ZT1pKzN8MDticmVhayBwfW89RFtiLTQ+PjJdfWI9bzw8Mjt2PURbYitjPj4yXTtyOntzOntpZigodnwwKTw9MCl7YnJlYWsgc31sPURbZisxNj4+Ml07Yj1EW2Ircz4+Ml07Zz0wO3doaWxlKDEpe3E9bCsoYjw8Myl8MDt1PURbcT4+Ml07aWYoKHV8MCk9PS0xKXticmVhayBzfXQ6e2lmKChqfDApIT0odXwwKSl7YnJlYWsgdH1xPURbcSs0Pj4yXTtpZigocXwwKSE9LTEpe3U9RFsocTw8Mikrbj4+Ml19ZWxzZXt1PS0xfWlmKCh1fDApPT0oYXwwKSl7YnJlYWsgdH13aGlsZSgxKXt1OnthPWI7Zz1nKzF8MDtpZigodnwwKTw9KGd8MCkpe2JyZWFrIHV9dT1sKyhhPDwzKXwwO2I9YSsxfDA7SD1sKyhiPDwzKXwwO0k9RFtIPj4yXTtEW3U+PjJdPUk7RFt1KzQ+PjJdPURbSCs0Pj4yXTtpZigoSXwwKSE9LTEpe2NvbnRpbnVlfX1icmVha31EW2wrKGE8PDMpPj4yXT0tMTtpZigocXwwKT09LTEpe2JyZWFrIHN9RFt5K0c+PjJdPXE7RFt5KyhxPDwyKT4+Ml09aTticmVhayByfWI9YisxfDA7Zz1nKzF8MDtpZigodnwwKSE9KGd8MCkpe2NvbnRpbnVlfWJyZWFrfX1hPWo8PDI7bD1EW2ErYz4+Ml07aWYoKGx8MCk8PTApe2JyZWFrIHJ9aj1EW2YrMTY+PjJdO2I9RFthK3M+PjJdO2c9MDt3aGlsZSgxKXthPWorKGI8PDMpfDA7aWYoRFthPj4yXT09LTEpe0RbYT4+Ml09bztEW2ErND4+Ml09aTticmVhayByfWI9YisxfDA7Zz1nKzF8MDtpZigobHwwKSE9KGd8MCkpe2NvbnRpbnVlfWJyZWFrfX19aT1lO2lmKHA+Pj4wPmk+Pj4wKXtjb250aW51ZX1icmVha319RFt3Pj4yXT1oO2lmKHMpe21hKHMpfWE9RFtmKzE2Pj4yXTtpZihhKXtEW2YrMjA+PjJdPWE7bWEoYSl9YT1EW2YrMzI+PjJdO2lmKCFhKXticmVhayBqfURbZiszNj4+Ml09YTttYShhKX0kPWYrNDh8MDt5PSh3fDApIT0wO2lmKHkpe2o9JC0zMnwwOyQ9ajtvPURbZD4+Ml07YT1EW2QrND4+Ml07RFtqKzI0Pj4yXT0wO0RbaisxNj4+Ml09MDtEW2orMjA+PjJdPTA7Yj1hLW98MDt2OntpZighYil7YnJlYWsgdn1pZigoYnwwKTwwKXticmVhayBofWI9Yj4+MjtnPWItMT4+PjV8MDtlPWcrMXwwO2M9bmEoZTw8Mik7RFtqKzI0Pj4yXT1lO0RbaisxNj4+Ml09YztEW2orMjA+PjJdPWI7RFtjKygoYj4+PjA8MzM/MDpnKTw8Mik+PjJdPTA7Zz1jO2M9Yj4+PjU8PDI7Zz1wYShnLDAsYyk7Yj1iJjMxO2lmKCFiKXticmVhayB2fWM9YytnfDA7RFtjPj4yXT1EW2M+PjJdJigtMT4+PjMyLWJeLTEpfURbais4Pj4yXT0wO0Rbaj4+Ml09MDt3aGlsZSgxKXt3OntwPTA7ZT0wO2lmKChhfDApPT0ob3wwKSl7YnJlYWsgd313aGlsZSgxKXtjPURbaisxNj4+Ml07eDp7aWYoRFtjKyhlPj4+MyY1MzY4NzA5MDgpPj4yXT4+PmUmMSl7YnJlYWsgeH1nPURbaj4+Ml07RFtqKzQ+PjJdPWc7Yj1EW2QrMTI+PjJdO2E9ZTt3aGlsZSgxKXt5OntmPWErMXwwO2g9YTthPShmPj4+MCklM3wwP2Y6YS0yfDA7aWYoKGF8MCk9PS0xKXticmVhayB5fWE9RFtiKyhhPDwyKT4+Ml07aWYoKGF8MCk9PS0xKXticmVhayB5fWY9YSsxfDA7YT0oZj4+PjApJTN8MD9mOmEtMnwwO2lmKChlfDApPT0oYXwwKXwoYXwwKT09LTEpe2JyZWFrIHl9aWYoIShEWyhhPj4+MyY1MzY4NzA5MDgpK2M+PjJdPj4+YSYxKSl7Y29udGludWV9fWJyZWFrfWI9ZztsPWg7ejp7QTp7d2hpbGUoMSl7YT0obD4+PjMmNTM2ODcwOTA4KStjfDA7RFthPj4yXT1EW2E+PjJdfDE8PGw7YT1sKzF8MDtmPShhPj4+MCklM3wwP2E6bC0yfDA7dj0obD4+PjApJTN8MDtuPSh2Py0xOjIpK2x8MDtxPW48PDI7Qjp7aWYoKGJ8MCk9PShnfDApKXticmVhayBCfXc9RFsoZjw8Mikrbz4+Ml07cz1EW2QrMTI+PjJdO2E9YjtpZigobnwwKSE9LTEpe3U9cytxfDA7d2hpbGUoMSl7Qzp7aWYoKHd8MCkhPURbYT4+Ml0pe2JyZWFrIEN9Yz1EW2ErND4+Ml07aT1EW3U+PjJdO2lmKChjfDApPT0oaXwwKSl7YnJlYWsgQ31nPS0xO2E9LTE7aWYoKGN8MCk9PS0xKXticmVhayB6fWJyZWFrIEF9YT1hKzh8MDtpZigoZ3wwKSE9KGF8MCkpe2NvbnRpbnVlfWJyZWFrfWJyZWFrIEJ9d2hpbGUoMSl7aWYoKHd8MCk9PURbYT4+Ml0pe2k9LTE7bj0tMTtjPURbYSs0Pj4yXTtpZigoY3wwKSE9LTEpe2JyZWFrIEF9fWE9YSs4fDA7aWYoKGd8MCkhPShhfDApKXtjb250aW51ZX1icmVha319bj1EW28rcT4+Ml07RDp7aWYoRFtqKzg+PjJdIT0oZ3wwKSl7RFtnPj4yXT1uO0RbZys0Pj4yXT1mO2c9Zys4fDA7RFtqKzQ+PjJdPWc7YnJlYWsgRH1jPWctYnwwO2c9Yz4+MzthPWcrMXwwO2lmKGE+Pj4wPj01MzY4NzA5MTIpe2JyZWFrIGh9aT1jPj4yO2k9Zz4+PjA8MjY4NDM1NDU1P2E+Pj4wPmk+Pj4wP2E6aTo1MzY4NzA5MTE7aWYoaSl7aWYoaT4+PjA+PTUzNjg3MDkxMil7YnJlYWsgaX1hPW5hKGk8PDMpfWVsc2V7YT0wfWc9YSsoZzw8Myl8MDtEW2c+PjJdPW47RFtnKzQ+PjJdPWY7Zz1nKzh8MDtpZigoY3wwKT4wKXtvYShhLGIsYyl9RFtqKzg+PjJdPWErKGk8PDMpO0Rbais0Pj4yXT1nO0Rbaj4+Ml09YTtpZighYil7YnJlYWsgRH1tYShiKX1FOntGOntpZih2KXthPWwtMXwwO2JyZWFrIEZ9YT1sKzJ8MDtpZigoYXwwKT09LTEpe2JyZWFrIEV9fWE9RFtEW2QrMTI+PjJdKyhhPDwyKT4+Ml07aWYoKGF8MCk9PS0xKXticmVhayBFfWw9YSsoKGE+Pj4wKSUzfDA/LTE6Mil8MDtpZigoaHwwKT09KGx8MCl8KGx8MCk9PS0xKXticmVhayBFfW89RFtkPj4yXTtiPURbaj4+Ml07Yz1EW2orMTY+PjJdO2NvbnRpbnVlfWJyZWFrfW89RFtkPj4yXTticmVhayB4fWc9YzthPURbcysoYzw8Mik+PjJdfWlmKChpfDApIT0tMSl7RFtzKyhpPDwyKT4+Ml09LTF9aWYoKGF8MCkhPS0xKXtEW0RbZCsxMj4+Ml0rKGE8PDIpPj4yXT0tMX1hPURbZCsxMj4+Ml07RFthKyhuPDwyKT4+Ml09LTE7RFthKyhnPDwyKT4+Ml09LTE7cD0xfWU9ZSsxfDA7YT1EW2QrND4+Ml07aWYoZT4+PjA8YS1vPj4yPj4+MCl7Y29udGludWV9YnJlYWt9aWYocCl7Y29udGludWV9fWJyZWFrfWE9RFtqPj4yXTtpZihhKXttYShhKX1hPURbaisxNj4+Ml07aWYoYSl7bWEoYSl9JD1qKzMyfDA7ZT0wO249MDtwPTA7aD0kLTMyfDA7JD1oO2E9RFtBKzEyPj4yXTtEW2QrMzY+PjJdPWE7bz1kKzI0fDA7Yz1EW2QrMjQ+PjJdO2I9RFtkKzI4Pj4yXS1jPj4yO0c6e0g6e2lmKGI+Pj4wPGE+Pj4wKXt4YShvLGEtYnwwLDEwMjYwKTtEW2grMjQ+PjJdPTA7RFtoKzE2Pj4yXT0wO0RbaCsyMD4+Ml09MDticmVhayBIfWlmKGE+Pj4wPGI+Pj4wKXtEW2QrMjg+PjJdPWMrKGE8PDIpfURbaCsyND4+Ml09MDtEW2grMTY+PjJdPTA7RFtoKzIwPj4yXT0wO2lmKCFhKXticmVhayBHfX1pZigoYXwwKTwwKXticmVhayBofWI9YS0xPj4+NXwwO2M9YisxfDA7ZT1uYShjPDwyKTtEW2grMjQ+PjJdPWM7RFtoKzE2Pj4yXT1lO0RbaCsyMD4+Ml09YTtEWygoYT4+PjA8MzM/MDpiKTw8MikrZT4+Ml09MDtiPWE+Pj41PDwyO2M9cGEoZSwwLGIpO2c9YSYzMTtpZighZyl7YnJlYWsgR31iPWIrY3wwO0RbYj4+Ml09RFtiPj4yXSYoLTE+Pj4zMi1nXi0xKX1pPURbZD4+Ml07bD1EW2QrND4+Ml07RFtoKzg+PjJdPTA7RFtoPj4yXT0wO0RbaCs0Pj4yXT0wO2I9bC1pfDA7STp7aWYoIWIpe2JyZWFrIEl9aWYoKGJ8MCk8MCl7YnJlYWsgaH1iPWI+PjI7Yz1iLTE+Pj41fDA7Zz1jKzF8MDtqPW5hKGc8PDIpO0RbaCs4Pj4yXT1nO0RbaD4+Ml09ajtEW2grND4+Ml09YjtEWygoYj4+PjA8MzM/MDpjKTw8Mikraj4+Ml09MDtjPWI+Pj41PDwyO2c9cGEoaiwwLGMpO2Y9YiYzMTtpZihmKXtjPWMrZ3wwO0RbYz4+Ml09RFtjPj4yXSYoLTE+Pj4zMi1mXi0xKX1pZihiPj4+MDwzKXticmVhayBJfXdoaWxlKDEpe3E9SihuLDMpO2c9KHE8PDIpK2l8MDtiPURbZz4+Ml07Yz0tMTtmPXErMXwwO2lmKChmfDApIT0tMSl7Yz1EWyhmPDwyKStpPj4yXX1KOntpZigoYnwwKT09KGN8MCkpe2JyZWFrIEp9Zj1iO2I9RFtnKzg+PjJdO2lmKChmfDApPT0oYnwwKSl7YnJlYWsgSn1zPTA7aWYoKGJ8MCk9PShjfDApKXticmVhayBKfXdoaWxlKDEpe2c9cytxfDA7aWYoIShEWyhnPj4+MyY1MzY4NzA5MDgpK2o+PjJdPj4+ZyYxKSl7Yj1EWyhnPDwyKStpPj4yXTtjPTE8PGI7aj1iPj4+NXwwO2w9YyZEWyhqPDwyKStlPj4yXTtpZihsKXtjPURbZCsyOD4+Ml07Szp7aWYoKGN8MCkhPURbZCszMj4+Ml0pe0RbYz4+Ml09LTE7RFtkKzI4Pj4yXT1jKzQ7YnJlYWsgS31lPURbbz4+Ml07Zj1jLWV8MDtqPWY+PjI7Yz1qKzF8MDtpZihjPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgaH1pPWY+PjE7aT1qPj4+MDw1MzY4NzA5MTE/Yz4+PjA+aT4+PjA/YzppOjEwNzM3NDE4MjM7aWYoaSl7aWYoaT4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGl9Yz1uYShpPDwyKX1lbHNle2M9MH1qPWMrKGo8PDIpfDA7RFtqPj4yXT0tMTtpZigoZnwwKT4wKXtvYShjLGUsZil9RFtkKzMyPj4yXT1jKyhpPDwyKTtEW2QrMjg+PjJdPWorNDtEW2QrMjQ+PjJdPWM7aWYoIWUpe2JyZWFrIEt9bWEoZSl9Yz1EW2QrNTI+PjJdO0w6e2lmKChjfDApIT1EW2QrNTY+PjJdKXtEW2M+PjJdPWI7RFtkKzUyPj4yXT1jKzQ7YnJlYWsgTH1lPURbZCs0OD4+Ml07Zj1jLWV8MDtqPWY+PjI7Yz1qKzF8MDtpZihjPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgaH1pPWY+PjE7aT1qPj4+MDw1MzY4NzA5MTE/Yz4+PjA+aT4+PjA/YzppOjEwNzM3NDE4MjM7aWYoaSl7aWYoaT4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGl9Yz1uYShpPDwyKX1lbHNle2M9MH1qPWMrKGo8PDIpfDA7RFtqPj4yXT1iO2lmKChmfDApPjApe29hKGMsZSxmKX1EW2QrNTY+PjJdPWMrKGk8PDIpO0RbZCs1Mj4+Ml09ais0O0RbZCs0OD4+Ml09YztpZighZSl7YnJlYWsgTH1tYShlKX1jPURbaCsyMD4+Ml07Yj1EW2grMjQ+PjJdO2lmKChjfDApPT1iPDw1KXtpZigoYysxfDApPDApe2JyZWFrIGh9ZT1oKzE2fDA7aWYoYz4+PjA8PTEwNzM3NDE4MjIpe2M9YyszMiYtMzI7Yj1iPDw2O2I9Yj4+PjA8Yz4+PjA/YzpifWVsc2V7Yj0yMTQ3NDgzNjQ3fVRhKGUsYik7Yz1EW2grMjA+PjJdfURbaCsyMD4+Ml09YysxO2I9RFtoKzE2Pj4yXSsoYz4+PjMmNTM2ODcwOTA4KXwwO2U9RFtiPj4yXTtNPWIsTj1kaShjKSZlLERbTT4+Ml09TjtjPTE8PGE7aj1hPj4+NXwwO2I9YTthPWErMXwwfWY9YTtlPURbaCsxNj4+Ml07YT1lKyhqPDwyKXwwO0RbYT4+Ml09RFthPj4yXXxjO3Y9RFtkKzI0Pj4yXSsoYjw8Mil8MDt3PURbZCsxMj4+Ml07aT1EW2Q+PjJdO2o9RFtoPj4yXTthPWc7TTp7Tjp7Tzp7UDp7UTp7Ujp7d2hpbGUoMSl7aWYoKGF8MCk9PS0xKXticmVhayBSfWM9KGE+Pj4zJjUzNjg3MDkwOCkranwwO0RbYz4+Ml09RFtjPj4yXXwxPDxhO0Rbdj4+Ml09YTtpZihsKXtEWyhhPDwyKStpPj4yXT1ifXU9YSsxfDA7YT0odT4+PjApJTN8MD91OmEtMnwwO2M9LTE7Uzp7aWYoKGF8MCk9PS0xKXticmVhayBTfWE9RFt3KyhhPDwyKT4+Ml07Yz0tMTtpZigoYXwwKT09LTEpe2JyZWFrIFN9Yz1hKzF8MDtjPShjPj4+MCklM3wwP2M6YS0yfDB9YT1jO2lmKChnfDApIT0oYXwwKSl7Y29udGludWV9YnJlYWt9aWYoKGd8MCkhPS0xKXticmVhayBNfWE9MTticmVhayBRfWlmKChnPj4+MCklM3wwKXthPWctMXwwO2JyZWFrIFF9YT1nKzJ8MDtpZigoYXwwKT09LTEpe2JyZWFrIFB9fWE9RFt3KyhhPDwyKT4+Ml07aWYoKGF8MCk9PS0xKXticmVhayBQfWlmKCEoKGE+Pj4wKSUzfDApKXticmVhayBPfWE9YS0xfDA7aT1EW2Q+PjJdO2o9RFtoPj4yXTticmVhayBOfWk9RFtkPj4yXTtqPURbaD4+Ml07YnJlYWsgTX1pPURbZD4+Ml07aj1EW2g+PjJdO2E9YSsyfDA7aWYoKGF8MCk9PS0xKXticmVhayBNfX1jPURbZCsxMj4+Ml07d2hpbGUoMSl7Zz0oYT4+PjMmNTM2ODcwOTA4KStqfDA7RFtnPj4yXT1EW2c+PjJdfDE8PGE7aWYobCl7RFsoYTw8MikraT4+Ml09Yn1UOntpZigoYT4+PjApJTN8MCl7YT1hLTF8MDticmVhayBUfWE9YSsyfDA7aWYoKGF8MCk9PS0xKXticmVhayBNfX1hPURbYysoYTw8Mik+PjJdO2lmKChhfDApPT0tMSl7YnJlYWsgTX1hPWErKChhPj4+MCklM3wwPy0xOjIpfDA7aWYoKGF8MCkhPS0xKXtjb250aW51ZX1icmVha319YT1mfXM9cysxfDA7aWYoKHN8MCkhPTMpe2NvbnRpbnVlfWJyZWFrfWk9RFtkPj4yXTtsPURbZCs0Pj4yXX1uPW4rMXwwO2lmKG4+Pj4wPChsLWk+PjI+Pj4wKS8zPj4+MCl7Y29udGludWV9YnJlYWt9ZT1EW2grMTY+PjJdfURbZCs0ND4+Ml09MDthPURbaCsyMD4+Ml07aWYoYSl7Yj1hJjMxO2c9KGE+Pj4zJjUzNjg3MDkwOCkrZXwwO2E9MDtjPWU7d2hpbGUoMSl7aWYoIShEW2M+PjJdPj4+YSYxKSl7cD1wKzF8MDtEW2QrNDQ+PjJdPXB9Zj0oYXwwKT09MzE7YT1mPzA6YSsxfDA7Yz0oZjw8MikrY3wwO2lmKChnfDApIT0oY3wwKXwoYXwwKSE9KGJ8MCkpe2NvbnRpbnVlfWJyZWFrfX1hPURbaD4+Ml07aWYoYSl7bWEoYSk7ZT1EW2grMTY+PjJdfWlmKGUpe21hKGUpfSQ9aCszMnwwfSQ9QSsxNnwwO2lmKCF5KXtEW0YrOD4+Ml09MDthYihkKTtkPTB9JD1GKzE2fDA7YT1kO2JyZWFrIGd9cmEoMTMyNik7VCgpfXFhKCk7VCgpfWI9RFt0Pj4yXTtpZighYil7YnJlYWsgZH1EW3QrND4+Ml09YjttYShiKX0kPXQrMTZ8MDticmVhayBifXFhKCk7VCgpfWI9RFtrKzQ+PjJdO0Rbays0Pj4yXT1hO2lmKGIpe2FiKGIpO2E9RFtrKzQ+PjJdfVU6e2lmKCFhKXticmVhayBVfWE9RFt4KzEwMD4+Ml07Yj1EW3grOTY+PjJdO0JbcisxMnwwXT0wO0VhKGsrNTZ8MCwoYS1ifDApLzEyfDAscisxMnwwKTthPURbeCsxMDA+PjJdO2I9RFt4Kzk2Pj4yXTtpZigoYXwwKT09KGJ8MCkpe0s9MTticmVhayBVfXdoaWxlKDEpe2lmKCEoRFtEW2srNTY+PjJdKyhDPj4+MyY1MzY4NzA5MDgpPj4yXT4+PkMmMSkpe2E9SihDLDMpO1RiKGssMCxhKTtiPURbays4Pj4yXTtjPURbaysxMj4+Ml07VGIoaywxLGErMXwwKTtnPURbaysyMD4+Ml07ZT1EW2srMjQ+PjJdO1RiKGssMixhKzJ8MCk7Zz1lLWc+PjI7Yj1jLWJ8MDtjPWI+PjI7YT1nPj4+MD5jPj4+MDtjPURbayszNj4+Ml0tRFtrKzMyPj4yXT4+Mj4+PjA+KGE/ZzpjKT4+PjA/MjphPzE6Yj8wOi0xO1Y6e2lmKERbays2OD4+Ml08PTApe2JyZWFrIFZ9RFtyKzEyPj4yXT1EW2srNzY+PjJdO0Rbcis4Pj4yXT1tO1FhKHIrOHwwLHIrMTJ8MCk7YT1EWygoYzw8Mikra3wwKSs0ND4+Ml07aWYoKGF8MCk8MCl7YT0tMX1lbHNle2I9KGE+Pj4wKS8zfDA7YT1EWyhEW0Rbaz4+Ml0rOTY+PjJdK0ooYiwxMil8MCkrKGEtSihiLDMpPDwyKT4+Ml19RFtyKzEyPj4yXT1hO0Rbcis4Pj4yXT1tO1FhKHIrOHwwLHIrMTJ8MCk7Yj1EW2srNzI+PjJdO0Rbays3Mj4+Ml09YisyO2lmKCEoYiYxKSl7YnJlYWsgVn1EW3IrMTI+PjJdPWE7RFtyKzg+PjJdPW07UWEocis4fDAscisxMnwwKTtEW2srNzI+PjJdPURbays3Mj4+Ml0rMX1iPSQtMTZ8MDskPWI7RFtrKzY4Pj4yXT1EW2srNjg+PjJdKzE7YT1KKGMsMTIpK2t8MDtnPURbYSsxMj4+Ml0tRFthKzg+PjJdfDA7Vzp7aWYoKGd8MCk8PTApe2JyZWFrIFd9YT0tMTtjPURbKChjPDwyKStrfDApKzQ0Pj4yXTtlPShjPj4+MCkvM3wwO2g9KGN8MCk9PS0xO2Y9aD8tMTplO2Q9RFtrKzU2Pj4yXSsoZj4+PjMmNTM2ODcwOTA4KXwwO0RbZD4+Ml09RFtkPj4yXXwxPDxmO0Rbays3Mj4+Ml09RFtrKzcyPj4yXSsxO0RbYisxMj4+Ml09KGN8MCk+PTA/RFsoRFtEW2s+PjJdKzk2Pj4yXStKKGUsMTIpfDApKygoYz4+PjApJTM8PDIpPj4yXTotMTtEW2IrOD4+Ml09bTtRYShiKzh8MCxiKzEyfDApO1g6e2lmKCFoKXtmPWMrMXwwO2Y9KGY+Pj4wKSUzfDA/ZjpjLTJ8MDtpZigoZnwwKT49MCl7aD0oZj4+PjApLzN8MDtkPURbKERbRFtrPj4yXSs5Nj4+Ml0rSihoLDEyKXwwKSsoZi1KKGgsMyk8PDIpPj4yXX1lbHNle2Q9LTF9RFtiKzEyPj4yXT1kO0RbYis4Pj4yXT1tO1FhKGIrOHwwLGIrMTJ8MCk7ZT1jKyhjLUooZSwzKXwwPy0xOjIpfDA7aWYoKGV8MCk8MCl7YnJlYWsgWH1hPShlPj4+MCkvM3wwO2E9RFsoRFtEW2s+PjJdKzk2Pj4yXStKKGEsMTIpfDApKyhlLUooYSwzKTw8Mik+PjJdO2JyZWFrIFh9RFtiKzEyPj4yXT0tMTtEW2IrOD4+Ml09bTtRYShiKzh8MCxiKzEyfDApfURbays3Nj4+Ml09YTtEW2IrMTI+PjJdPWE7RFtiKzg+PjJdPW07YT0tMTtRYShiKzh8MCxiKzEyfDApO2E9KGN8MCkhPS0xP0RbRFtEW2srND4+Ml0rMTI+PjJdKyhjPDwyKT4+Ml06YTtpZihnPj4+MDw9Nyl7YnJlYWsgV31jPWc+Pj4yfDA7aD1jPj4+MD4xP2M6MTtjPTE7d2hpbGUoMSl7Zz1hO2Y9KGE+Pj4wKS8zfDA7YT0oYXwwKT09LTE/LTE6ZjtlPURbays1Nj4+Ml0rKGE+Pj4zJjUzNjg3MDkwOCl8MDtEW2U+PjJdPURbZT4+Ml18MTw8YTtEW2srNzI+PjJdPURbays3Mj4+Ml0rMTthPS0xO2E9KGd8MCk+PTA/RFsoRFtEW2s+PjJdKzk2Pj4yXStKKGYsMTIpfDApKygoZz4+PjApJTM8PDIpPj4yXTphO0Rbays3Nj4+Ml09YTtEW2IrMTI+PjJdPWE7RFtiKzg+PjJdPW07UWEoYis4fDAsYisxMnwwKTtZOntaOntfOntpZihjJjEpe2U9LTE7aWYoKGd8MCk9PS0xKXticmVhayBZfWlmKChnfDApIT0oSihmLDMpfDApKXthPWctMXwwO2JyZWFrIFp9YT1nKzJ8MDticmVhayBffWU9LTE7aWYoKGd8MCk9PS0xKXticmVhayBZfWE9ZysxfDA7YT0oYT4+PjApJTN8MD9hOmctMnwwfWU9LTE7aWYoKGF8MCk9PS0xKXticmVhayBZfX1lPURbRFtEW2srND4+Ml0rMTI+PjJdKyhhPDwyKT4+Ml19YT1lO2M9YysxfDA7aWYoKGh8MCkhPShjfDApKXtjb250aW51ZX1icmVha319JD1iKzE2fDA7Yj1EW3grOTY+PjJdO2E9RFt4KzEwMD4+Ml19Sz0xO0M9QysxfDA7aWYoQz4+PjA8KGEtYnwwKS8xMj4+PjApe2NvbnRpbnVlfWJyZWFrfX0kPXIrMTZ8MDskOntpZihLKXthPURbej4+Ml07aWYoYSl7RFt6KzQ+PjJdPWE7bWEoYSl9RFt6Pj4yXT1EW20+PjJdO0Rbeis0Pj4yXT1EW20rND4+Ml07RFt6Kzg+PjJdPURbbSs4Pj4yXTtMPURbbSs4ND4+Ml07YnJlYWsgJH1hPURbbT4+Ml07aWYoIWEpe2JyZWFrICR9RFttKzQ+PjJdPWE7bWEoYSl9YT1EW20rNzI+PjJdO2lmKGEpe21hKGEpfWE9RFttKzQ4Pj4yXTtpZihhKXtEW20rNTI+PjJdPWE7bWEoYSl9YT1EW20rMzY+PjJdO2lmKGEpe0RbbSs0MD4+Ml09YTttYShhKX1hPURbbSsyND4+Ml07aWYoYSl7RFttKzI4Pj4yXT1hO21hKGEpfWE9RFttKzIwPj4yXTtEW20rMjA+PjJdPTA7aWYoYSl7YWIoYSl9JD1tKzk2fDA7cmV0dXJuIEx8MH1mdW5jdGlvbiBBYyhhKXthPWF8MDt2YXIgYj0wLGM9MCxkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTAsaz0wLGw9MCxtPTAsbj0wO2w9JC0xNnwwOyQ9bDthOntiOntjOntkOntlOntmOntnOntoOntpOntqOntrOntpZihhPj4+MDw9MjQ0KXtlPURbMjg4MV07aD1hPj4+MDwxMT8xNjphKzExJi04O2M9aD4+PjN8MDtiPWU+Pj5jfDA7aWYoYiYzKXtkPWMrKChiXi0xKSYxKXwwO2I9ZDw8MztmPURbYisxMTU3Mj4+Ml07YT1mKzh8MDtjPURbZis4Pj4yXTtiPWIrMTE1NjR8MDtsOntpZigoY3wwKT09KGJ8MCkpe209MTE1MjQsbj1kaShkKSZlLERbbT4+Ml09bjticmVhayBsfURbYysxMj4+Ml09YjtEW2IrOD4+Ml09Y31iPWQ8PDM7RFtmKzQ+PjJdPWJ8MztiPWIrZnwwO0RbYis0Pj4yXT1EW2IrND4+Ml18MTticmVhayBhfWs9RFsyODgzXTtpZihrPj4+MD49aD4+PjApe2JyZWFrIGt9aWYoYil7YT0yPDxjO2E9KDAtYXxhKSZiPDxjO2I9KDAtYSZhKS0xfDA7YT1iPj4+MTImMTY7Yz1hO2I9Yj4+PmF8MDthPWI+Pj41Jjg7Yz1jfGE7Yj1iPj4+YXwwO2E9Yj4+PjImNDtjPWN8YTtiPWI+Pj5hfDA7YT1iPj4+MSYyO2M9Y3xhO2I9Yj4+PmF8MDthPWI+Pj4xJjE7Yz0oY3xhKSsoYj4+PmF8MCl8MDthPWM8PDM7Zz1EW2ErMTE1NzI+PjJdO2I9RFtnKzg+PjJdO2E9YSsxMTU2NHwwO206e2lmKChifDApPT0oYXwwKSl7ZT1kaShjKSZlO0RbMjg4MV09ZTticmVhayBtfURbYisxMj4+Ml09YTtEW2ErOD4+Ml09Yn1hPWcrOHwwO0RbZys0Pj4yXT1ofDM7ZD1nK2h8MDtiPWM8PDM7Zj1iLWh8MDtEW2QrND4+Ml09ZnwxO0RbYitnPj4yXT1mO2lmKGspe2I9az4+PjN8MDtjPShiPDwzKSsxMTU2NHwwO2c9RFsyODg2XTtiPTE8PGI7bjp7aWYoIShiJmUpKXtEWzI4ODFdPWJ8ZTtiPWM7YnJlYWsgbn1iPURbYys4Pj4yXX1EW2MrOD4+Ml09ZztEW2IrMTI+PjJdPWc7RFtnKzEyPj4yXT1jO0RbZys4Pj4yXT1ifURbMjg4Nl09ZDtEWzI4ODNdPWY7YnJlYWsgYX1qPURbMjg4Ml07aWYoIWope2JyZWFrIGt9Yj0oaiYwLWopLTF8MDthPWI+Pj4xMiYxNjtjPWE7Yj1iPj4+YXwwO2E9Yj4+PjUmODtjPWN8YTtiPWI+Pj5hfDA7YT1iPj4+MiY0O2M9Y3xhO2I9Yj4+PmF8MDthPWI+Pj4xJjI7Yz1jfGE7Yj1iPj4+YXwwO2E9Yj4+PjEmMTtiPURbKChjfGEpKyhiPj4+YXwwKTw8MikrMTE4Mjg+PjJdO2Q9KERbYis0Pj4yXSYtOCktaHwwO2M9Yjt3aGlsZSgxKXtvOnthPURbYysxNj4+Ml07aWYoIWEpe2E9RFtjKzIwPj4yXTtpZighYSl7YnJlYWsgb319Yz0oRFthKzQ+PjJdJi04KS1ofDA7Zj1jPj4+MDxkPj4+MDtkPWY/YzpkO2I9Zj9hOmI7Yz1hO2NvbnRpbnVlfWJyZWFrfWk9RFtiKzI0Pj4yXTtmPURbYisxMj4+Ml07aWYoKGZ8MCkhPShifDApKXthPURbYis4Pj4yXTtEW2ErMTI+PjJdPWY7RFtmKzg+PjJdPWE7YnJlYWsgYn1jPWIrMjB8MDthPURbYz4+Ml07aWYoIWEpe2E9RFtiKzE2Pj4yXTtpZighYSl7YnJlYWsgan1jPWIrMTZ8MH13aGlsZSgxKXtnPWM7Zj1hO2M9YSsyMHwwO2E9RFtjPj4yXTtpZihhKXtjb250aW51ZX1jPWYrMTZ8MDthPURbZisxNj4+Ml07aWYoYSl7Y29udGludWV9YnJlYWt9RFtnPj4yXT0wO2JyZWFrIGJ9aD0tMTtpZihhPj4+MD40Mjk0OTY3MjMxKXticmVhayBrfWE9YSsxMXwwO2g9YSYtODtqPURbMjg4Ml07aWYoIWope2JyZWFrIGt9ZD0wLWh8MDtlPTA7cDp7aWYoaD4+PjA8MjU2KXticmVhayBwfWU9MzE7aWYoaD4+PjA+MTY3NzcyMTUpe2JyZWFrIHB9YT1hPj4+OHwwO2c9YSsxMDQ4MzIwPj4+MTYmODthPWE8PGc7Yz1hKzUyMDE5Mj4+PjE2JjQ7YT1hPDxjO2I9YSsyNDU3NjA+Pj4xNiYyO2E9KGE8PGI+Pj4xNXwwKS0oYnwoY3xnKSl8MDtlPShhPDwxfGg+Pj5hKzIxJjEpKzI4fDB9Yz1EWyhlPDwyKSsxMTgyOD4+Ml07cTp7cjp7czp7aWYoIWMpe2E9MDticmVhayBzfWE9MDtiPWg8PCgoZXwwKT09MzE/MDoyNS0oZT4+PjF8MCl8MCk7d2hpbGUoMSl7dDp7Zz0oRFtjKzQ+PjJdJi04KS1ofDA7aWYoZz4+PjA+PWQ+Pj4wKXticmVhayB0fWY9YztkPWc7aWYoZCl7YnJlYWsgdH1kPTA7YT1jO2JyZWFrIHJ9Zz1EW2MrMjA+PjJdO2M9RFsoKGI+Pj4yOSY0KStjfDApKzE2Pj4yXTthPWc/KGd8MCk9PShjfDApP2E6ZzphO2I9Yjw8MTtpZihjKXtjb250aW51ZX1icmVha319aWYoIShhfGYpKXtmPTA7YT0yPDxlO2E9KDAtYXxhKSZqO2lmKCFhKXticmVhayBrfWI9KGEmMC1hKS0xfDA7YT1iPj4+MTImMTY7Yz1hO2I9Yj4+PmF8MDthPWI+Pj41Jjg7Yz1jfGE7Yj1iPj4+YXwwO2E9Yj4+PjImNDtjPWN8YTtiPWI+Pj5hfDA7YT1iPj4+MSYyO2M9Y3xhO2I9Yj4+PmF8MDthPWI+Pj4xJjE7YT1EWygoY3xhKSsoYj4+PmF8MCk8PDIpKzExODI4Pj4yXX1pZighYSl7YnJlYWsgcX19d2hpbGUoMSl7Yj0oRFthKzQ+PjJdJi04KS1ofDA7Yz1iPj4+MDxkPj4+MDtkPWM/YjpkO2Y9Yz9hOmY7Yj1EW2ErMTY+PjJdO2lmKGIpe2E9Yn1lbHNle2E9RFthKzIwPj4yXX1pZihhKXtjb250aW51ZX1icmVha319aWYoIWZ8RFsyODgzXS1oPj4+MDw9ZD4+PjApe2JyZWFrIGt9ZT1EW2YrMjQ+PjJdO2I9RFtmKzEyPj4yXTtpZigoZnwwKSE9KGJ8MCkpe2E9RFtmKzg+PjJdO0RbYSsxMj4+Ml09YjtEW2IrOD4+Ml09YTticmVhayBjfWM9ZisyMHwwO2E9RFtjPj4yXTtpZighYSl7YT1EW2YrMTY+PjJdO2lmKCFhKXticmVhayBpfWM9ZisxNnwwfXdoaWxlKDEpe2c9YztiPWE7Yz1hKzIwfDA7YT1EW2M+PjJdO2lmKGEpe2NvbnRpbnVlfWM9YisxNnwwO2E9RFtiKzE2Pj4yXTtpZihhKXtjb250aW51ZX1icmVha31EW2c+PjJdPTA7YnJlYWsgY31jPURbMjg4M107aWYoYz4+PjA+PWg+Pj4wKXtkPURbMjg4Nl07Yj1jLWh8MDt1OntpZihiPj4+MD49MTYpe0RbMjg4M109YjthPWQraHwwO0RbMjg4Nl09YTtEW2ErND4+Ml09YnwxO0RbYytkPj4yXT1iO0RbZCs0Pj4yXT1ofDM7YnJlYWsgdX1EWzI4ODZdPTA7RFsyODgzXT0wO0RbZCs0Pj4yXT1jfDM7YT1jK2R8MDtEW2ErND4+Ml09RFthKzQ+PjJdfDF9YT1kKzh8MDticmVhayBhfWk9RFsyODg0XTtpZihpPj4+MD5oPj4+MCl7Yj1pLWh8MDtEWzI4ODRdPWI7Yz1EWzI4ODddO2E9YytofDA7RFsyODg3XT1hO0RbYSs0Pj4yXT1ifDE7RFtjKzQ+PjJdPWh8MzthPWMrOHwwO2JyZWFrIGF9YT0wO2o9aCs0N3wwO2lmKERbMjk5OV0pe2M9RFszMDAxXX1lbHNle0RbMzAwMl09LTE7RFszMDAzXT0tMTtEWzNlM109NDA5NjtEWzMwMDFdPTQwOTY7RFsyOTk5XT1sKzEyJi0xNl4xNDMxNjU1NzY4O0RbMzAwNF09MDtEWzI5OTJdPTA7Yz00MDk2fWc9aitjfDA7Zj0wLWN8MDtjPWcmZjtpZihjPj4+MDw9aD4+PjApe2JyZWFrIGF9ZD1EWzI5OTFdO2lmKGQpe2I9RFsyOTg5XTtlPWIrY3wwO2lmKGQ+Pj4wPGU+Pj4wfGI+Pj4wPj1lPj4+MCl7YnJlYWsgYX19aWYoRVsxMTk2OF0mNCl7YnJlYWsgZn12Ont3OntkPURbMjg4N107aWYoZCl7YT0xMTk3Mjt3aGlsZSgxKXtiPURbYT4+Ml07aWYoYj4+PjA8PWQ+Pj4wJmQ+Pj4wPGIrRFthKzQ+PjJdPj4+MCl7YnJlYWsgd31hPURbYSs4Pj4yXTtpZihhKXtjb250aW51ZX1icmVha319Yj1XYSgwKTtpZigoYnwwKT09LTEpe2JyZWFrIGd9ZT1jO2Q9RFszZTNdO2E9ZC0xfDA7aWYoYSZiKXtlPShjLWJ8MCkrKGErYiYwLWQpfDB9aWYoZT4+PjA8PWg+Pj4wfGU+Pj4wPjIxNDc0ODM2NDYpe2JyZWFrIGd9ZD1EWzI5OTFdO2lmKGQpe2E9RFsyOTg5XTtmPWErZXwwO2lmKGQ+Pj4wPGY+Pj4wfGE+Pj4wPj1mPj4+MCl7YnJlYWsgZ319YT1XYShlKTtpZigoYnwwKSE9KGF8MCkpe2JyZWFrIHZ9YnJlYWsgZX1lPWYmZy1pO2lmKGU+Pj4wPjIxNDc0ODM2NDYpe2JyZWFrIGd9Yj1XYShlKTtpZigoYnwwKT09KERbYT4+Ml0rRFthKzQ+PjJdfDApKXticmVhayBofWE9Yn1pZighKChhfDApPT0tMXxoKzQ4Pj4+MDw9ZT4+PjApKXtiPURbMzAwMV07Yj1iKyhqLWV8MCkmMC1iO2lmKGI+Pj4wPjIxNDc0ODM2NDYpe2I9YTticmVhayBlfWlmKChXYShiKXwwKSE9LTEpe2U9YitlfDA7Yj1hO2JyZWFrIGV9V2EoMC1lfDApO2JyZWFrIGd9Yj1hO2lmKChhfDApIT0tMSl7YnJlYWsgZX1icmVhayBnfWY9MDticmVhayBifWI9MDticmVhayBjfWlmKChifDApIT0tMSl7YnJlYWsgZX19RFsyOTkyXT1EWzI5OTJdfDR9aWYoYz4+PjA+MjE0NzQ4MzY0Nil7YnJlYWsgZH1iPVdhKGMpO2E9V2EoMCk7aWYoKGJ8MCk9PS0xfChhfDApPT0tMXxhPj4+MDw9Yj4+PjApe2JyZWFrIGR9ZT1hLWJ8MDtpZihlPj4+MDw9aCs0MD4+PjApe2JyZWFrIGR9fWE9RFsyOTg5XStlfDA7RFsyOTg5XT1hO2lmKGE+Pj4wPkdbMjk5MF0pe0RbMjk5MF09YX14Ont5Ont6OntnPURbMjg4N107aWYoZyl7YT0xMTk3Mjt3aGlsZSgxKXtkPURbYT4+Ml07Yz1EW2ErND4+Ml07aWYoKGQrY3wwKT09KGJ8MCkpe2JyZWFrIHp9YT1EW2ErOD4+Ml07aWYoYSl7Y29udGludWV9YnJlYWt9YnJlYWsgeX1hPURbMjg4NV07aWYoIShhPj4+MDw9Yj4+PjA/YTowKSl7RFsyODg1XT1ifWE9MDtEWzI5OTRdPWU7RFsyOTkzXT1iO0RbMjg4OV09LTE7RFsyODkwXT1EWzI5OTldO0RbMjk5Nl09MDt3aGlsZSgxKXtkPWE8PDM7Yz1kKzExNTY0fDA7RFtkKzExNTcyPj4yXT1jO0RbZCsxMTU3Nj4+Ml09YzthPWErMXwwO2lmKChhfDApIT0zMil7Y29udGludWV9YnJlYWt9ZD1lLTQwfDA7YT1iKzgmNz8tOC1iJjc6MDtjPWQtYXwwO0RbMjg4NF09YzthPWErYnwwO0RbMjg4N109YTtEW2ErND4+Ml09Y3wxO0RbKGIrZHwwKSs0Pj4yXT00MDtEWzI4ODhdPURbMzAwM107YnJlYWsgeH1pZihFW2ErMTJ8MF0mOHxkPj4+MD5nPj4+MHxiPj4+MDw9Zz4+PjApe2JyZWFrIHl9RFthKzQ+PjJdPWMrZTthPWcrOCY3Py04LWcmNzowO2M9YStnfDA7RFsyODg3XT1jO2I9RFsyODg0XStlfDA7YT1iLWF8MDtEWzI4ODRdPWE7RFtjKzQ+PjJdPWF8MTtEWyhiK2d8MCkrND4+Ml09NDA7RFsyODg4XT1EWzMwMDNdO2JyZWFrIHh9aWYoR1syODg1XT5iPj4+MCl7RFsyODg1XT1ifWM9YitlfDA7YT0xMTk3MjtBOntCOntDOntEOntFOntGOnt3aGlsZSgxKXtpZigoY3wwKSE9RFthPj4yXSl7YT1EW2ErOD4+Ml07aWYoYSl7Y29udGludWV9YnJlYWsgRn1icmVha31pZighKEVbYSsxMnwwXSY4KSl7YnJlYWsgRX19YT0xMTk3Mjt3aGlsZSgxKXtjPURbYT4+Ml07aWYoYz4+PjA8PWc+Pj4wKXtmPWMrRFthKzQ+PjJdfDA7aWYoZj4+PjA+Zz4+PjApe2JyZWFrIER9fWE9RFthKzg+PjJdO2NvbnRpbnVlfX1EW2E+PjJdPWI7RFthKzQ+PjJdPURbYSs0Pj4yXStlO2o9KGIrOCY3Py04LWImNzowKStifDA7RFtqKzQ+PjJdPWh8MztlPWMrKGMrOCY3Py04LWMmNzowKXwwO2k9aCtqfDA7Yz1lLWl8MDtpZigoZXwwKT09KGd8MCkpe0RbMjg4N109aTthPURbMjg4NF0rY3wwO0RbMjg4NF09YTtEW2krND4+Ml09YXwxO2JyZWFrIEJ9aWYoRFsyODg2XT09KGV8MCkpe0RbMjg4Nl09aTthPURbMjg4M10rY3wwO0RbMjg4M109YTtEW2krND4+Ml09YXwxO0RbYStpPj4yXT1hO2JyZWFrIEJ9YT1EW2UrND4+Ml07aWYoKGEmMyk9PTEpe2c9YSYtODtHOntpZihhPj4+MDw9MjU1KXtkPURbZSs4Pj4yXTthPWE+Pj4zfDA7Yj1EW2UrMTI+PjJdO2lmKChifDApPT0oZHwwKSl7bT0xMTUyNCxuPURbMjg4MV0mZGkoYSksRFttPj4yXT1uO2JyZWFrIEd9RFtkKzEyPj4yXT1iO0RbYis4Pj4yXT1kO2JyZWFrIEd9aD1EW2UrMjQ+PjJdO2I9RFtlKzEyPj4yXTtIOntpZigoZXwwKSE9KGJ8MCkpe2E9RFtlKzg+PjJdO0RbYSsxMj4+Ml09YjtEW2IrOD4+Ml09YTticmVhayBIfUk6e2E9ZSsyMHwwO2Q9RFthPj4yXTtpZihkKXticmVhayBJfWE9ZSsxNnwwO2Q9RFthPj4yXTtpZihkKXticmVhayBJfWI9MDticmVhayBIfXdoaWxlKDEpe2Y9YTtiPWQ7YT1iKzIwfDA7ZD1EW2E+PjJdO2lmKGQpe2NvbnRpbnVlfWE9YisxNnwwO2Q9RFtiKzE2Pj4yXTtpZihkKXtjb250aW51ZX1icmVha31EW2Y+PjJdPTB9aWYoIWgpe2JyZWFrIEd9ZD1EW2UrMjg+PjJdO2E9KGQ8PDIpKzExODI4fDA7Sjp7aWYoRFthPj4yXT09KGV8MCkpe0RbYT4+Ml09YjtpZihiKXticmVhayBKfW09MTE1Mjgsbj1EWzI4ODJdJmRpKGQpLERbbT4+Ml09bjticmVhayBHfURbaCsoRFtoKzE2Pj4yXT09KGV8MCk/MTY6MjApPj4yXT1iO2lmKCFiKXticmVhayBHfX1EW2IrMjQ+PjJdPWg7YT1EW2UrMTY+PjJdO2lmKGEpe0RbYisxNj4+Ml09YTtEW2ErMjQ+PjJdPWJ9YT1EW2UrMjA+PjJdO2lmKCFhKXticmVhayBHfURbYisyMD4+Ml09YTtEW2ErMjQ+PjJdPWJ9ZT1lK2d8MDtjPWMrZ3wwfURbZSs0Pj4yXT1EW2UrND4+Ml0mLTI7RFtpKzQ+PjJdPWN8MTtEW2MraT4+Ml09YztpZihjPj4+MDw9MjU1KXthPWM+Pj4zfDA7Yj0oYTw8MykrMTE1NjR8MDtjPURbMjg4MV07YT0xPDxhO0s6e2lmKCEoYyZhKSl7RFsyODgxXT1hfGM7YT1iO2JyZWFrIEt9YT1EW2IrOD4+Ml19RFtiKzg+PjJdPWk7RFthKzEyPj4yXT1pO0RbaSsxMj4+Ml09YjtEW2krOD4+Ml09YTticmVhayBCfWE9MzE7aWYoYz4+PjA8PTE2Nzc3MjE1KXthPWM+Pj44fDA7Zj1hKzEwNDgzMjA+Pj4xNiY4O2E9YTw8ZjtkPWErNTIwMTkyPj4+MTYmNDthPWE8PGQ7Yj1hKzI0NTc2MD4+PjE2JjI7YT0oYTw8Yj4+PjE1fDApLShifChkfGYpKXwwO2E9KGE8PDF8Yz4+PmErMjEmMSkrMjh8MH1EW2krMjg+PjJdPWE7RFtpKzE2Pj4yXT0wO0RbaSsyMD4+Ml09MDtmPShhPDwyKSsxMTgyOHwwO2Q9RFsyODgyXTtiPTE8PGE7TDp7aWYoIShkJmIpKXtEWzI4ODJdPWJ8ZDtEW2Y+PjJdPWk7RFtpKzI0Pj4yXT1mO2JyZWFrIEx9YT1jPDwoKGF8MCk9PTMxPzA6MjUtKGE+Pj4xfDApfDApO2I9RFtmPj4yXTt3aGlsZSgxKXtkPWI7aWYoKERbYis0Pj4yXSYtOCk9PShjfDApKXticmVhayBDfWI9YT4+PjI5fDA7YT1hPDwxO2Y9ZCsoYiY0KXwwO2I9RFtmKzE2Pj4yXTtpZihiKXtjb250aW51ZX1icmVha31EW2YrMTY+PjJdPWk7RFtpKzI0Pj4yXT1kfURbaSsxMj4+Ml09aTtEW2krOD4+Ml09aTticmVhayBCfWQ9ZS00MHwwO2E9Yis4Jjc/LTgtYiY3OjA7Yz1kLWF8MDtEWzI4ODRdPWM7YT1hK2J8MDtEWzI4ODddPWE7RFthKzQ+PjJdPWN8MTtEWyhiK2R8MCkrND4+Ml09NDA7RFsyODg4XT1EWzMwMDNdO2E9KGYrKGYtMzkmNz8zOS1mJjc6MCl8MCktNDd8MDtjPWE+Pj4wPGcrMTY+Pj4wP2c6YTtEW2MrND4+Ml09Mjc7YT1EWzI5OTZdO0RbYysxNj4+Ml09RFsyOTk1XTtEW2MrMjA+PjJdPWE7YT1EWzI5OTRdO0RbYys4Pj4yXT1EWzI5OTNdO0RbYysxMj4+Ml09YTtEWzI5OTVdPWMrODtEWzI5OTRdPWU7RFsyOTkzXT1iO0RbMjk5Nl09MDthPWMrMjR8MDt3aGlsZSgxKXtEW2ErND4+Ml09NztiPWErOHwwO2E9YSs0fDA7aWYoYj4+PjA8Zj4+PjApe2NvbnRpbnVlfWJyZWFrfWlmKChjfDApPT0oZ3wwKSl7YnJlYWsgeH1EW2MrND4+Ml09RFtjKzQ+PjJdJi0yO2Y9Yy1nfDA7RFtnKzQ+PjJdPWZ8MTtEW2M+PjJdPWY7aWYoZj4+PjA8PTI1NSl7YT1mPj4+M3wwO2I9KGE8PDMpKzExNTY0fDA7Yz1EWzI4ODFdO2E9MTw8YTtNOntpZighKGMmYSkpe0RbMjg4MV09YXxjO2E9YjticmVhayBNfWE9RFtiKzg+PjJdfURbYis4Pj4yXT1nO0RbYSsxMj4+Ml09ZztEW2crMTI+PjJdPWI7RFtnKzg+PjJdPWE7YnJlYWsgeH1hPTMxO0RbZysxNj4+Ml09MDtEW2crMjA+PjJdPTA7aWYoZj4+PjA8PTE2Nzc3MjE1KXthPWY+Pj44fDA7ZD1hKzEwNDgzMjA+Pj4xNiY4O2E9YTw8ZDtjPWErNTIwMTkyPj4+MTYmNDthPWE8PGM7Yj1hKzI0NTc2MD4+PjE2JjI7YT0oYTw8Yj4+PjE1fDApLShifChjfGQpKXwwO2E9KGE8PDF8Zj4+PmErMjEmMSkrMjh8MH1EW2crMjg+PjJdPWE7ZD0oYTw8MikrMTE4Mjh8MDtjPURbMjg4Ml07Yj0xPDxhO046e2lmKCEoYyZiKSl7RFsyODgyXT1ifGM7RFtkPj4yXT1nO0RbZysyND4+Ml09ZDticmVhayBOfWE9Zjw8KChhfDApPT0zMT8wOjI1LShhPj4+MXwwKXwwKTtiPURbZD4+Ml07d2hpbGUoMSl7Yz1iO2lmKChmfDApPT0oRFtiKzQ+PjJdJi04KSl7YnJlYWsgQX1iPWE+Pj4yOXwwO2E9YTw8MTtkPWMrKGImNCl8MDtiPURbZCsxNj4+Ml07aWYoYil7Y29udGludWV9YnJlYWt9RFtkKzE2Pj4yXT1nO0RbZysyND4+Ml09Y31EW2crMTI+PjJdPWc7RFtnKzg+PjJdPWc7YnJlYWsgeH1hPURbZCs4Pj4yXTtEW2ErMTI+PjJdPWk7RFtkKzg+PjJdPWk7RFtpKzI0Pj4yXT0wO0RbaSsxMj4+Ml09ZDtEW2krOD4+Ml09YX1hPWorOHwwO2JyZWFrIGF9YT1EW2MrOD4+Ml07RFthKzEyPj4yXT1nO0RbYys4Pj4yXT1nO0RbZysyND4+Ml09MDtEW2crMTI+PjJdPWM7RFtnKzg+PjJdPWF9YT1EWzI4ODRdO2lmKGE+Pj4wPD1oPj4+MCl7YnJlYWsgZH1iPWEtaHwwO0RbMjg4NF09YjtjPURbMjg4N107YT1jK2h8MDtEWzI4ODddPWE7RFthKzQ+PjJdPWJ8MTtEW2MrND4+Ml09aHwzO2E9Yys4fDA7YnJlYWsgYX1EWzI4NzldPTQ4O2E9MDticmVhayBhfU86e2lmKCFlKXticmVhayBPfWM9RFtmKzI4Pj4yXTthPShjPDwyKSsxMTgyOHwwO1A6e2lmKERbYT4+Ml09PShmfDApKXtEW2E+PjJdPWI7aWYoYil7YnJlYWsgUH1qPWRpKGMpJmo7RFsyODgyXT1qO2JyZWFrIE99RFtlKyhEW2UrMTY+PjJdPT0oZnwwKT8xNjoyMCk+PjJdPWI7aWYoIWIpe2JyZWFrIE99fURbYisyND4+Ml09ZTthPURbZisxNj4+Ml07aWYoYSl7RFtiKzE2Pj4yXT1hO0RbYSsyND4+Ml09Yn1hPURbZisyMD4+Ml07aWYoIWEpe2JyZWFrIE99RFtiKzIwPj4yXT1hO0RbYSsyND4+Ml09Yn1ROntpZihkPj4+MDw9MTUpe2E9ZCtofDA7RFtmKzQ+PjJdPWF8MzthPWErZnwwO0RbYSs0Pj4yXT1EW2ErND4+Ml18MTticmVhayBRfURbZis0Pj4yXT1ofDM7ZT1mK2h8MDtEW2UrND4+Ml09ZHwxO0RbZCtlPj4yXT1kO2lmKGQ+Pj4wPD0yNTUpe2E9ZD4+PjN8MDtiPShhPDwzKSsxMTU2NHwwO2M9RFsyODgxXTthPTE8PGE7Ujp7aWYoIShjJmEpKXtEWzI4ODFdPWF8YzthPWI7YnJlYWsgUn1hPURbYis4Pj4yXX1EW2IrOD4+Ml09ZTtEW2ErMTI+PjJdPWU7RFtlKzEyPj4yXT1iO0RbZSs4Pj4yXT1hO2JyZWFrIFF9YT0zMTtpZihkPj4+MDw9MTY3NzcyMTUpe2E9ZD4+Pjh8MDtnPWErMTA0ODMyMD4+PjE2Jjg7YT1hPDxnO2M9YSs1MjAxOTI+Pj4xNiY0O2E9YTw8YztiPWErMjQ1NzYwPj4+MTYmMjthPShhPDxiPj4+MTV8MCktKGJ8KGN8ZykpfDA7YT0oYTw8MXxkPj4+YSsyMSYxKSsyOHwwfURbZSsyOD4+Ml09YTtEW2UrMTY+PjJdPTA7RFtlKzIwPj4yXT0wO2I9KGE8PDIpKzExODI4fDA7Uzp7Yz0xPDxhO1Q6e2lmKCEoYyZqKSl7RFsyODgyXT1jfGo7RFtiPj4yXT1lO2JyZWFrIFR9YT1kPDwoKGF8MCk9PTMxPzA6MjUtKGE+Pj4xfDApfDApO2g9RFtiPj4yXTt3aGlsZSgxKXtiPWg7aWYoKERbYis0Pj4yXSYtOCk9PShkfDApKXticmVhayBTfWM9YT4+PjI5fDA7YT1hPDwxO2M9KGMmNCkrYnwwO2g9RFtjKzE2Pj4yXTtpZihoKXtjb250aW51ZX1icmVha31EW2MrMTY+PjJdPWV9RFtlKzI0Pj4yXT1iO0RbZSsxMj4+Ml09ZTtEW2UrOD4+Ml09ZTticmVhayBRfWE9RFtiKzg+PjJdO0RbYSsxMj4+Ml09ZTtEW2IrOD4+Ml09ZTtEW2UrMjQ+PjJdPTA7RFtlKzEyPj4yXT1iO0RbZSs4Pj4yXT1hfWE9Zis4fDA7YnJlYWsgYX1VOntpZighaSl7YnJlYWsgVX1jPURbYisyOD4+Ml07YT0oYzw8MikrMTE4Mjh8MDtWOntpZihEW2E+PjJdPT0oYnwwKSl7RFthPj4yXT1mO2lmKGYpe2JyZWFrIFZ9bT0xMTUyOCxuPWRpKGMpJmosRFttPj4yXT1uO2JyZWFrIFV9RFtpKyhEW2krMTY+PjJdPT0oYnwwKT8xNjoyMCk+PjJdPWY7aWYoIWYpe2JyZWFrIFV9fURbZisyND4+Ml09aTthPURbYisxNj4+Ml07aWYoYSl7RFtmKzE2Pj4yXT1hO0RbYSsyND4+Ml09Zn1hPURbYisyMD4+Ml07aWYoIWEpe2JyZWFrIFV9RFtmKzIwPj4yXT1hO0RbYSsyND4+Ml09Zn1XOntpZihkPj4+MDw9MTUpe2E9ZCtofDA7RFtiKzQ+PjJdPWF8MzthPWErYnwwO0RbYSs0Pj4yXT1EW2ErND4+Ml18MTticmVhayBXfURbYis0Pj4yXT1ofDM7Zj1iK2h8MDtEW2YrND4+Ml09ZHwxO0RbZCtmPj4yXT1kO2lmKGspe2E9az4+PjN8MDtjPShhPDwzKSsxMTU2NHwwO2c9RFsyODg2XTthPTE8PGE7WDp7aWYoIShhJmUpKXtEWzI4ODFdPWF8ZTthPWM7YnJlYWsgWH1hPURbYys4Pj4yXX1EW2MrOD4+Ml09ZztEW2ErMTI+PjJdPWc7RFtnKzEyPj4yXT1jO0RbZys4Pj4yXT1hfURbMjg4Nl09ZjtEWzI4ODNdPWR9YT1iKzh8MH0kPWwrMTZ8MDtyZXR1cm4gYXwwfWZ1bmN0aW9uIEtkKGEsYixjLGQpe3ZhciBlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTAsbz0wLHA9MCxxPTAscj0wO2g9JC04MHwwOyQ9aDtlPURbYyszNj4+Ml07RFtoKzcyPj4yXT1EW2MrMzI+PjJdO0RbaCs3Nj4+Ml09ZTtmPURbYysyOD4+Ml07ZT1oLSAtNjR8MDtEW2U+PjJdPURbYysyND4+Ml07RFtlKzQ+PjJdPWY7ZT1EW2MrMjA+PjJdO0RbaCs1Nj4+Ml09RFtjKzE2Pj4yXTtEW2grNjA+PjJdPWU7ZT1EW2MrMTI+PjJdO0RbaCs0OD4+Ml09RFtjKzg+PjJdO0RbaCs1Mj4+Ml09ZTtlPURbYys0Pj4yXTtEW2grNDA+PjJdPURbYz4+Ml07RFtoKzQ0Pj4yXT1lO2VjKGEsaCs0MHwwLGgrMjR8MCk7YTp7aWYoRFthPj4yXSl7YnJlYWsgYX1sPWErNHwwO2lmKEJbYSsxNXwwXTwwKXttYShEW2w+PjJdKX1pZihFW2grMzF8MF0hPTEpe2I9bmEoMzIpO2M9RVsxNTkwXXxFWzE1OTFdPDw4fChFWzE1OTJdPDwxNnxFWzE1OTNdPDwyNCk7QltiKzE2fDBdPWM7QltiKzE3fDBdPWM+Pj44O0JbYisxOHwwXT1jPj4+MTY7QltiKzE5fDBdPWM+Pj4yNDtjPUVbMTU4Nl18RVsxNTg3XTw8OHwoRVsxNTg4XTw8MTZ8RVsxNTg5XTw8MjQpO2Q9RVsxNTgyXXxFWzE1ODNdPDw4fChFWzE1ODRdPDwxNnxFWzE1ODVdPDwyNCk7QltiKzh8MF09ZDtCW2IrOXwwXT1kPj4+ODtCW2IrMTB8MF09ZD4+PjE2O0JbYisxMXwwXT1kPj4+MjQ7QltiKzEyfDBdPWM7QltiKzEzfDBdPWM+Pj44O0JbYisxNHwwXT1jPj4+MTY7QltiKzE1fDBdPWM+Pj4yNDtjPUVbMTU3OF18RVsxNTc5XTw8OHwoRVsxNTgwXTw8MTZ8RVsxNTgxXTw8MjQpO2Q9RVsxNTc0XXxFWzE1NzVdPDw4fChFWzE1NzZdPDwxNnxFWzE1NzddPDwyNCk7QltifDBdPWQ7QltiKzF8MF09ZD4+Pjg7QltiKzJ8MF09ZD4+PjE2O0JbYiszfDBdPWQ+Pj4yNDtCW2IrNHwwXT1jO0JbYis1fDBdPWM+Pj44O0JbYis2fDBdPWM+Pj4xNjtCW2IrN3wwXT1jPj4+MjQ7QltiKzIwfDBdPTA7RFthPj4yXT0tMTt0YShsLGIsMjApO21hKGIpO2JyZWFrIGF9aT0kLTE2fDA7JD1pO2I6e2M6e3N3aXRjaChFW2grMzJ8MF0pe2Nhc2UgMDplPW5hKDQ4KTtGZChlKTtEW2U+PjJdPTk4OTY7RFtoKzg+PjJdPTA7RFtoKzEyPj4yXT0wO0RbaD4+Ml09MDtEW2grND4+Ml09MDtEW2grMTY+PjJdPWU7YnJlYWsgYjtjYXNlIDE6ZT1uYSg1Mik7RmQoZSk7RFtlKzQ4Pj4yXT0wO0RbZT4+Ml09ODIwNDtEW2grOD4+Ml09MDtEW2grMTI+PjJdPTA7RFtoPj4yXT0wO0RbaCs0Pj4yXT0wO0RbaCsxNj4+Ml09ZTticmVhayBiO2RlZmF1bHQ6YnJlYWsgY319Zj1uYSgzMik7ZT1FWzE2NjRdfEVbMTY2NV08PDh8KEVbMTY2Nl08PDE2fEVbMTY2N108PDI0KTtCW2YrMjR8MF09ZTtCW2YrMjV8MF09ZT4+Pjg7QltmKzI2fDBdPWU+Pj4xNjtCW2YrMjd8MF09ZT4+PjI0O2U9RVsxNjYwXXxFWzE2NjFdPDw4fChFWzE2NjJdPDwxNnxFWzE2NjNdPDwyNCk7Zz1FWzE2NTZdfEVbMTY1N108PDh8KEVbMTY1OF08PDE2fEVbMTY1OV08PDI0KTtCW2YrMTZ8MF09ZztCW2YrMTd8MF09Zz4+Pjg7QltmKzE4fDBdPWc+Pj4xNjtCW2YrMTl8MF09Zz4+PjI0O0JbZisyMHwwXT1lO0JbZisyMXwwXT1lPj4+ODtCW2YrMjJ8MF09ZT4+PjE2O0JbZisyM3wwXT1lPj4+MjQ7ZT1FWzE2NTJdfEVbMTY1M108PDh8KEVbMTY1NF08PDE2fEVbMTY1NV08PDI0KTtnPUVbMTY0OF18RVsxNjQ5XTw8OHwoRVsxNjUwXTw8MTZ8RVsxNjUxXTw8MjQpO0JbZis4fDBdPWc7QltmKzl8MF09Zz4+Pjg7QltmKzEwfDBdPWc+Pj4xNjtCW2YrMTF8MF09Zz4+PjI0O0JbZisxMnwwXT1lO0JbZisxM3wwXT1lPj4+ODtCW2YrMTR8MF09ZT4+PjE2O0JbZisxNXwwXT1lPj4+MjQ7ZT1FWzE2NDRdfEVbMTY0NV08PDh8KEVbMTY0Nl08PDE2fEVbMTY0N108PDI0KTtnPUVbMTY0MF18RVsxNjQxXTw8OHwoRVsxNjQyXTw8MTZ8RVsxNjQzXTw8MjQpO0JbZnwwXT1nO0JbZisxfDBdPWc+Pj44O0JbZisyfDBdPWc+Pj4xNjtCW2YrM3wwXT1nPj4+MjQ7QltmKzR8MF09ZTtCW2YrNXwwXT1lPj4+ODtCW2YrNnwwXT1lPj4+MTY7QltmKzd8MF09ZT4+PjI0O0JbZisyOHwwXT0wO0RbaT4+Ml09LTE7ZT1pfDQ7dGEoZSxmLDI4KTtrPUJbaSsxNXwwXTtEW2g+PjJdPURbaT4+Ml07Zz1oKzR8MDtkOntpZigoa3wwKT49MCl7az1EW2UrND4+Ml07RFtnPj4yXT1EW2U+PjJdO0RbZys0Pj4yXT1rO0RbZys4Pj4yXT1EW2UrOD4+Ml07YnJlYWsgZH10YShnLERbaSs0Pj4yXSxEW2krOD4+Ml0pfURbaCsxNj4+Ml09MDtpZihCW2krMTV8MF08MCl7bWEoRFtpKzQ+PjJdKX1tYShmKX0kPWkrMTZ8MDtlPURbaD4+Ml07ZTp7aWYoZSl7RFthPj4yXT1lO2lmKEJbaCsxNXwwXT49MCl7YT1ofDQ7Yj1EW2ErND4+Ml07RFtsPj4yXT1EW2E+PjJdO0RbbCs0Pj4yXT1iO0RbbCs4Pj4yXT1EW2ErOD4+Ml07YnJlYWsgZX10YShsLERbaCs0Pj4yXSxEW2grOD4+Ml0pO2JyZWFrIGV9ZT1EW2grMTY+PjJdO0RbaCsxNj4+Ml09MDtEW2UrNDQ+PjJdPWQ7aT0kLTMyfDA7JD1pO0RbZSszMj4+Ml09YztEW2UrNDA+PjJdPWI7RFtlKzQ+PjJdPWQ7ZWMoYSxjLGkrMTZ8MCk7Zjp7aWYoRFthPj4yXSl7YnJlYWsgZn1mPWErNHwwO2lmKEJbYSsxNXwwXTwwKXttYShEW2Y+PjJdKX1iPUVbaSsyM3wwXTtpZigoYmFbRFtEW2U+PjJdKzg+PjJdXShlKXwwKSE9KGJ8MCkpe2I9bmEoNjQpO2M9RVsxNDQ4XXxFWzE0NDldPDw4O0JbYis0OHwwXT1jO0JbYis0OXwwXT1jPj4+ODtjPUVbMTQ0NF18RVsxNDQ1XTw8OHwoRVsxNDQ2XTw8MTZ8RVsxNDQ3XTw8MjQpO2Q9RVsxNDQwXXxFWzE0NDFdPDw4fChFWzE0NDJdPDwxNnxFWzE0NDNdPDwyNCk7QltiKzQwfDBdPWQ7QltiKzQxfDBdPWQ+Pj44O0JbYis0MnwwXT1kPj4+MTY7QltiKzQzfDBdPWQ+Pj4yNDtCW2IrNDR8MF09YztCW2IrNDV8MF09Yz4+Pjg7QltiKzQ2fDBdPWM+Pj4xNjtCW2IrNDd8MF09Yz4+PjI0O2M9RVsxNDM2XXxFWzE0MzddPDw4fChFWzE0MzhdPDwxNnxFWzE0MzldPDwyNCk7ZD1FWzE0MzJdfEVbMTQzM108PDh8KEVbMTQzNF08PDE2fEVbMTQzNV08PDI0KTtCW2IrMzJ8MF09ZDtCW2IrMzN8MF09ZD4+Pjg7QltiKzM0fDBdPWQ+Pj4xNjtCW2IrMzV8MF09ZD4+PjI0O0JbYiszNnwwXT1jO0JbYiszN3wwXT1jPj4+ODtCW2IrMzh8MF09Yz4+PjE2O0JbYiszOXwwXT1jPj4+MjQ7Yz1FWzE0MjhdfEVbMTQyOV08PDh8KEVbMTQzMF08PDE2fEVbMTQzMV08PDI0KTtkPUVbMTQyNF18RVsxNDI1XTw8OHwoRVsxNDI2XTw8MTZ8RVsxNDI3XTw8MjQpO0JbYisyNHwwXT1kO0JbYisyNXwwXT1kPj4+ODtCW2IrMjZ8MF09ZD4+PjE2O0JbYisyN3wwXT1kPj4+MjQ7QltiKzI4fDBdPWM7QltiKzI5fDBdPWM+Pj44O0JbYiszMHwwXT1jPj4+MTY7QltiKzMxfDBdPWM+Pj4yNDtjPUVbMTQyMF18RVsxNDIxXTw8OHwoRVsxNDIyXTw8MTZ8RVsxNDIzXTw8MjQpO2Q9RVsxNDE2XXxFWzE0MTddPDw4fChFWzE0MThdPDwxNnxFWzE0MTldPDwyNCk7QltiKzE2fDBdPWQ7QltiKzE3fDBdPWQ+Pj44O0JbYisxOHwwXT1kPj4+MTY7QltiKzE5fDBdPWQ+Pj4yNDtCW2IrMjB8MF09YztCW2IrMjF8MF09Yz4+Pjg7QltiKzIyfDBdPWM+Pj4xNjtCW2IrMjN8MF09Yz4+PjI0O2M9RVsxNDEyXXxFWzE0MTNdPDw4fChFWzE0MTRdPDwxNnxFWzE0MTVdPDwyNCk7ZD1FWzE0MDhdfEVbMTQwOV08PDh8KEVbMTQxMF08PDE2fEVbMTQxMV08PDI0KTtCW2IrOHwwXT1kO0JbYis5fDBdPWQ+Pj44O0JbYisxMHwwXT1kPj4+MTY7QltiKzExfDBdPWQ+Pj4yNDtCW2IrMTJ8MF09YztCW2IrMTN8MF09Yz4+Pjg7QltiKzE0fDBdPWM+Pj4xNjtCW2IrMTV8MF09Yz4+PjI0O2M9RVsxNDA0XXxFWzE0MDVdPDw4fChFWzE0MDZdPDwxNnxFWzE0MDddPDwyNCk7ZD1FWzE0MDBdfEVbMTQwMV08PDh8KEVbMTQwMl08PDE2fEVbMTQwM108PDI0KTtCW2J8MF09ZDtCW2IrMXwwXT1kPj4+ODtCW2IrMnwwXT1kPj4+MTY7QltiKzN8MF09ZD4+PjI0O0JbYis0fDBdPWM7QltiKzV8MF09Yz4+Pjg7QltiKzZ8MF09Yz4+PjE2O0JbYis3fDBdPWM+Pj4yNDtCW2IrNTB8MF09MDtEW2E+PjJdPS0xO3RhKGYsYiw1MCk7bWEoYik7YnJlYWsgZn1jPUVbaSsyMXwwXTtCW2UrMzZ8MF09YztkPUVbaSsyMnwwXTtCW2UrMzd8MF09ZDtpZigoY3wwKSE9Mil7Yj1uYSgzMik7Yz1FWzE1NzFdfEVbMTU3Ml08PDg7QltiKzI0fDBdPWM7QltiKzI1fDBdPWM+Pj44O2M9RVsxNTY3XXxFWzE1NjhdPDw4fChFWzE1NjldPDwxNnxFWzE1NzBdPDwyNCk7ZD1FWzE1NjNdfEVbMTU2NF08PDh8KEVbMTU2NV08PDE2fEVbMTU2Nl08PDI0KTtCW2IrMTZ8MF09ZDtCW2IrMTd8MF09ZD4+Pjg7QltiKzE4fDBdPWQ+Pj4xNjtCW2IrMTl8MF09ZD4+PjI0O0JbYisyMHwwXT1jO0JbYisyMXwwXT1jPj4+ODtCW2IrMjJ8MF09Yz4+PjE2O0JbYisyM3wwXT1jPj4+MjQ7Yz1FWzE1NTldfEVbMTU2MF08PDh8KEVbMTU2MV08PDE2fEVbMTU2Ml08PDI0KTtkPUVbMTU1NV18RVsxNTU2XTw8OHwoRVsxNTU3XTw8MTZ8RVsxNTU4XTw8MjQpO0JbYis4fDBdPWQ7QltiKzl8MF09ZD4+Pjg7QltiKzEwfDBdPWQ+Pj4xNjtCW2IrMTF8MF09ZD4+PjI0O0JbYisxMnwwXT1jO0JbYisxM3wwXT1jPj4+ODtCW2IrMTR8MF09Yz4+PjE2O0JbYisxNXwwXT1jPj4+MjQ7Yz1FWzE1NTFdfEVbMTU1Ml08PDh8KEVbMTU1M108PDE2fEVbMTU1NF08PDI0KTtkPUVbMTU0N118RVsxNTQ4XTw8OHwoRVsxNTQ5XTw8MTZ8RVsxNTUwXTw8MjQpO0JbYnwwXT1kO0JbYisxfDBdPWQ+Pj44O0JbYisyfDBdPWQ+Pj4xNjtCW2IrM3wwXT1kPj4+MjQ7QltiKzR8MF09YztCW2IrNXwwXT1jPj4+ODtCW2IrNnwwXT1jPj4+MTY7QltiKzd8MF09Yz4+PjI0O0JbYisyNnwwXT0wO0RbYT4+Ml09LTU7dGEoZixiLDI2KTttYShiKTticmVhayBmfWI9Yj8yOjM7aWYoKGJ8MCkhPShkfDApKXtiPW5hKDMyKTtjPUVbMTU0NF18RVsxNTQ1XTw8ODtCW2IrMjR8MF09YztCW2IrMjV8MF09Yz4+Pjg7Yz1FWzE1NDBdfEVbMTU0MV08PDh8KEVbMTU0Ml08PDE2fEVbMTU0M108PDI0KTtkPUVbMTUzNl18RVsxNTM3XTw8OHwoRVsxNTM4XTw8MTZ8RVsxNTM5XTw8MjQpO0JbYisxNnwwXT1kO0JbYisxN3wwXT1kPj4+ODtCW2IrMTh8MF09ZD4+PjE2O0JbYisxOXwwXT1kPj4+MjQ7QltiKzIwfDBdPWM7QltiKzIxfDBdPWM+Pj44O0JbYisyMnwwXT1jPj4+MTY7QltiKzIzfDBdPWM+Pj4yNDtjPUVbMTUzMl18RVsxNTMzXTw8OHwoRVsxNTM0XTw8MTZ8RVsxNTM1XTw8MjQpO2Q9RVsxNTI4XXxFWzE1MjldPDw4fChFWzE1MzBdPDwxNnxFWzE1MzFdPDwyNCk7QltiKzh8MF09ZDtCW2IrOXwwXT1kPj4+ODtCW2IrMTB8MF09ZD4+PjE2O0JbYisxMXwwXT1kPj4+MjQ7QltiKzEyfDBdPWM7QltiKzEzfDBdPWM+Pj44O0JbYisxNHwwXT1jPj4+MTY7QltiKzE1fDBdPWM+Pj4yNDtjPUVbMTUyNF18RVsxNTI1XTw8OHwoRVsxNTI2XTw8MTZ8RVsxNTI3XTw8MjQpO2Q9RVsxNTIwXXxFWzE1MjFdPDw4fChFWzE1MjJdPDwxNnxFWzE1MjNdPDwyNCk7QltifDBdPWQ7QltiKzF8MF09ZD4+Pjg7QltiKzJ8MF09ZD4+PjE2O0JbYiszfDBdPWQ+Pj4yNDtCW2IrNHwwXT1jO0JbYis1fDBdPWM+Pj44O0JbYis2fDBdPWM+Pj4xNjtCW2IrN3wwXT1jPj4+MjQ7QltiKzI2fDBdPTA7RFthPj4yXT0tNTt0YShmLGIsMjYpO21hKGIpO2JyZWFrIGZ9Q1tEW2UrMzI+PjJdKzM4Pj4xXT1ifDUxMjtnOntpZihDW2krMjY+PjFdPj0wKXticmVhayBnfWs9JC0xNnwwOyQ9aztkPW5hKDM2KTtiPWQ7RFtiKzQ+PjJdPTA7RFtiKzg+PjJdPTA7RFtiKzI0Pj4yXT0wO0RbYisyOD4+Ml09MDtiPWIrMTZ8MDtEW2I+PjJdPTA7RFtiKzQ+PjJdPTA7RFtkPj4yXT1kKzQ7RFtkKzMyPj4yXT0wO0RbZCsxMj4+Ml09YjtEW2s+PjJdPTA7Yz1EW2UrMzI+PjJdO209JC0xNnwwOyQ9bTtiPTA7aDp7aWYoIWQpe2JyZWFrIGh9RFtrPj4yXT1jO0RbbSsxMj4+Ml09MDtiPTA7aWYoIWViKDEsbSsxMnwwLGMpKXticmVhayBofXA9RFttKzEyPj4yXTtpZihwKXt3aGlsZSgxKXtpOntpZihlYigxLG0rOHwwLERbaz4+Ml0pKXtiPW5hKDI4KTtEW2IrND4+Ml09MDtEW2IrOD4+Ml09MDtjPWIrMTZ8MDtEW2M+PjJdPTA7RFtjKzQ+PjJdPTA7RFtiPj4yXT1iKzQ7RFtiKzEyPj4yXT1jO0RbYisyND4+Ml09RFttKzg+PjJdO2lmKFBjKGssYikpe2JyZWFrIGl9Q2EoYisxMnwwLERbYisxNj4+Ml0pO0JhKGIsRFtiKzQ+PjJdKTttYShiKX1iPTA7YnJlYWsgaH1nPSQtMTZ8MDskPWc7RFtnKzg+PjJdPWI7ajp7aWYoIWIpe0RbZys4Pj4yXT0wO2JyZWFrIGp9Yz1EW2QrMjg+PjJdO2s6e2lmKGM+Pj4wPEdbZCszMj4+Ml0pe0RbZys4Pj4yXT0wO0RbYz4+Ml09YjtEW2QrMjg+PjJdPWMrNDticmVhayBrfWM9MDtsOnttOntuOntqPURbZCsyND4+Ml07bz1EW2QrMjg+PjJdLWo+PjI7Yj1vKzF8MDtpZihiPj4+MDwxMDczNzQxODI0KXtqPURbZCszMj4+Ml0tanwwO249aj4+MTtiPWo+PjI+Pj4wPDUzNjg3MDkxMT9iPj4+MD5uPj4+MD9iOm46MTA3Mzc0MTgyMztpZihiKXtpZihiPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgbn1jPW5hKGI8PDIpfW49RFtnKzg+PjJdO0RbZys4Pj4yXT0wO2o9KG88PDIpK2N8MDtEW2o+PjJdPW47bz0oYjw8MikrY3wwO249ais0fDA7Yz1EW2QrMjg+PjJdO2I9RFtkKzI0Pj4yXTtpZigoY3wwKT09KGJ8MCkpe2JyZWFrIG19d2hpbGUoMSl7Yz1jLTR8MDtyPURbYz4+Ml07RFtjPj4yXT0wO2o9ai00fDA7RFtqPj4yXT1yO2lmKChifDApIT0oY3wwKSl7Y29udGludWV9YnJlYWt9RFtkKzMyPj4yXT1vO2M9RFtkKzI4Pj4yXTtEW2QrMjg+PjJdPW47Yj1EW2QrMjQ+PjJdO0RbZCsyND4+Ml09ajtpZigoYnwwKT09KGN8MCkpe2JyZWFrIGx9d2hpbGUoMSl7Yz1jLTR8MDtqPURbYz4+Ml07RFtjPj4yXT0wO2lmKGope0NhKGorMTJ8MCxEW2orMTY+PjJdKTtCYShqLERbais0Pj4yXSk7bWEoail9aWYoKGJ8MCkhPShjfDApKXtjb250aW51ZX1icmVha31icmVhayBsfXFhKCk7VCgpfXJhKDEzMjYpO1QoKX1EW2QrMzI+PjJdPW87RFtkKzI4Pj4yXT1uO0RbZCsyND4+Ml09an1pZihiKXttYShiKX19Yj1EW2crOD4+Ml07RFtnKzg+PjJdPTA7aWYoIWIpe2JyZWFrIGp9Q2EoYisxMnwwLERbYisxNj4+Ml0pO0JhKGIsRFtiKzQ+PjJdKTttYShiKX0kPWcrMTZ8MDtxPXErMXwwO2lmKChxfDApIT0ocHwwKSl7Y29udGludWV9YnJlYWt9fWI9UGMoayxkKX0kPW0rMTZ8MDtvOntpZihiKXtjPURbZSs0Pj4yXTtiPURbYys0Pj4yXTtEW2MrND4+Ml09ZDtpZihiKXtkYyhiKX1EW2E+PjJdPTA7RFthKzQ+PjJdPTA7RFthKzg+PjJdPTA7RFthKzEyPj4yXT0wO2JyZWFrIG99Yj1uYSgzMik7Yz1FWzE2OTNdfEVbMTY5NF08PDg7QltiKzI0fDBdPWM7QltiKzI1fDBdPWM+Pj44O2M9RVsxNjg5XXxFWzE2OTBdPDw4fChFWzE2OTFdPDwxNnxFWzE2OTJdPDwyNCk7Zz1FWzE2ODVdfEVbMTY4Nl08PDh8KEVbMTY4N108PDE2fEVbMTY4OF08PDI0KTtCW2IrMTZ8MF09ZztCW2IrMTd8MF09Zz4+Pjg7QltiKzE4fDBdPWc+Pj4xNjtCW2IrMTl8MF09Zz4+PjI0O0JbYisyMHwwXT1jO0JbYisyMXwwXT1jPj4+ODtCW2IrMjJ8MF09Yz4+PjE2O0JbYisyM3wwXT1jPj4+MjQ7Yz1FWzE2ODFdfEVbMTY4Ml08PDh8KEVbMTY4M108PDE2fEVbMTY4NF08PDI0KTtnPUVbMTY3N118RVsxNjc4XTw8OHwoRVsxNjc5XTw8MTZ8RVsxNjgwXTw8MjQpO0JbYis4fDBdPWc7QltiKzl8MF09Zz4+Pjg7QltiKzEwfDBdPWc+Pj4xNjtCW2IrMTF8MF09Zz4+PjI0O0JbYisxMnwwXT1jO0JbYisxM3wwXT1jPj4+ODtCW2IrMTR8MF09Yz4+PjE2O0JbYisxNXwwXT1jPj4+MjQ7Yz1FWzE2NzNdfEVbMTY3NF08PDh8KEVbMTY3NV08PDE2fEVbMTY3Nl08PDI0KTtnPUVbMTY2OV18RVsxNjcwXTw8OHwoRVsxNjcxXTw8MTZ8RVsxNjcyXTw8MjQpO0JbYnwwXT1nO0JbYisxfDBdPWc+Pj44O0JbYisyfDBdPWc+Pj4xNjtCW2IrM3wwXT1nPj4+MjQ7QltiKzR8MF09YztCW2IrNXwwXT1jPj4+ODtCW2IrNnwwXT1jPj4+MTY7QltiKzd8MF09Yz4+PjI0O0JbYisyNnwwXT0wO0RbYT4+Ml09LTE7dGEoYSs0fDAsYiwyNik7bWEoYik7RFtrKzg+PjJdPTA7ZGMoZCl9JD1rKzE2fDA7aWYoRFthPj4yXSl7YnJlYWsgZn1pZihCW2YrMTF8MF0+PTApe2JyZWFrIGd9bWEoRFtmPj4yXSl9aWYoIShiYVtEW0RbZT4+Ml0rMTI+PjJdXShlKXwwKSl7Yj1uYSg0OCk7QltiKzMyfDBdPUVbMTUxOF07Yz1FWzE1MTRdfEVbMTUxNV08PDh8KEVbMTUxNl08PDE2fEVbMTUxN108PDI0KTtkPUVbMTUxMF18RVsxNTExXTw8OHwoRVsxNTEyXTw8MTZ8RVsxNTEzXTw8MjQpO0JbYisyNHwwXT1kO0JbYisyNXwwXT1kPj4+ODtCW2IrMjZ8MF09ZD4+PjE2O0JbYisyN3wwXT1kPj4+MjQ7QltiKzI4fDBdPWM7QltiKzI5fDBdPWM+Pj44O0JbYiszMHwwXT1jPj4+MTY7QltiKzMxfDBdPWM+Pj4yNDtjPUVbMTUwNl18RVsxNTA3XTw8OHwoRVsxNTA4XTw8MTZ8RVsxNTA5XTw8MjQpO2Q9RVsxNTAyXXxFWzE1MDNdPDw4fChFWzE1MDRdPDwxNnxFWzE1MDVdPDwyNCk7QltiKzE2fDBdPWQ7QltiKzE3fDBdPWQ+Pj44O0JbYisxOHwwXT1kPj4+MTY7QltiKzE5fDBdPWQ+Pj4yNDtCW2IrMjB8MF09YztCW2IrMjF8MF09Yz4+Pjg7QltiKzIyfDBdPWM+Pj4xNjtCW2IrMjN8MF09Yz4+PjI0O2M9RVsxNDk4XXxFWzE0OTldPDw4fChFWzE1MDBdPDwxNnxFWzE1MDFdPDwyNCk7ZD1FWzE0OTRdfEVbMTQ5NV08PDh8KEVbMTQ5Nl08PDE2fEVbMTQ5N108PDI0KTtCW2IrOHwwXT1kO0JbYis5fDBdPWQ+Pj44O0JbYisxMHwwXT1kPj4+MTY7QltiKzExfDBdPWQ+Pj4yNDtCW2IrMTJ8MF09YztCW2IrMTN8MF09Yz4+Pjg7QltiKzE0fDBdPWM+Pj4xNjtCW2IrMTV8MF09Yz4+PjI0O2M9RVsxNDkwXXxFWzE0OTFdPDw4fChFWzE0OTJdPDwxNnxFWzE0OTNdPDwyNCk7ZD1FWzE0ODZdfEVbMTQ4N108PDh8KEVbMTQ4OF08PDE2fEVbMTQ4OV08PDI0KTtCW2J8MF09ZDtCW2IrMXwwXT1kPj4+ODtCW2IrMnwwXT1kPj4+MTY7QltiKzN8MF09ZD4+PjI0O0JbYis0fDBdPWM7QltiKzV8MF09Yz4+Pjg7QltiKzZ8MF09Yz4+PjE2O0JbYis3fDBdPWM+Pj4yNDtCW2IrMzN8MF09MDtEW2E+PjJdPS0xO3RhKGYsYiwzMyk7bWEoYik7YnJlYWsgZn1pZighKGJhW0RbRFtlPj4yXSsyMD4+Ml1dKGUpfDApKXtiPUZiKGksMTY5Nik7RFthPj4yXT0tMTtpZihCW2IrMTF8MF0+PTApe2I9RFtpKzQ+PjJdO0RbZj4+Ml09RFtpPj4yXTtEW2YrND4+Ml09YjtEW2YrOD4+Ml09RFtpKzg+PjJdO2JyZWFrIGZ9dGEoZixEW2I+PjJdLERbYis0Pj4yXSk7aWYoQltiKzExfDBdPj0wKXticmVhayBmfW1hKERbYj4+Ml0pO2JyZWFrIGZ9aWYoIShiYVtEW0RbZT4+Ml0rMjQ+PjJdXShlKXwwKSl7Yj1GYihpLDE0NTEpO0RbYT4+Ml09LTE7aWYoQltiKzExfDBdPj0wKXtiPURbaSs0Pj4yXTtEW2Y+PjJdPURbaT4+Ml07RFtmKzQ+PjJdPWI7RFtmKzg+PjJdPURbaSs4Pj4yXTticmVhayBmfXRhKGYsRFtiPj4yXSxEW2IrND4+Ml0pO2lmKEJbYisxMXwwXT49MCl7YnJlYWsgZn1tYShEW2I+PjJdKTticmVhayBmfURbYT4+Ml09MDtEW2ErND4+Ml09MDtEW2ErOD4+Ml09MDtEW2ErMTI+PjJdPTB9JD1pKzMyfDA7aWYoIURbYT4+Ml0pe2lmKEJbbCsxMXwwXTwwKXttYShEW2w+PjJdKX1EW2E+PjJdPTA7RFthKzQ+PjJdPTA7RFthKzg+PjJdPTA7RFthKzEyPj4yXT0wfWJhW0RbRFtlPj4yXSs0Pj4yXV0oZSl9YT1EW2grMTY+PjJdO0RbaCsxNj4+Ml09MDtpZihhKXtiYVtEW0RbYT4+Ml0rND4+Ml1dKGEpfWlmKEJbaCsxNXwwXT49MCl7YnJlYWsgYX1tYShEW2grND4+Ml0pfSQ9aCs4MHwwfWZ1bmN0aW9uIFBjKGEsYil7dmFyIGM9MCxkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTAsaz0wLGw9MCxtPTAsbj0wLG89MCxwPTAscT0wLHI9MDtrPSQtNDh8MDskPWs7Yz1uYSg4KTtEW2MrND4+Ml09YjtEW2M+PjJdPTA7Yj1jKzh8MDtEW2srNDA+PjJdPWI7RFtrKzM2Pj4yXT1iO0RbayszMj4+Ml09YzthOntiOntjOntkOnt3aGlsZSgxKXtiPWItOHwwO2o9RFtiKzQ+PjJdO2c9RFtiPj4yXTtEW2srMzY+PjJdPWI7aWYoZyl7RFtrKzI0Pj4yXT0wO0RbaysxNj4+Ml09MDtEW2srMjA+PjJdPTA7Yz0xO2I9RFthPj4yXTtlPURbYis4Pj4yXTtoPURbYisxMj4+Ml07ZD1EW2IrMjA+PjJdO2Y9RFtiKzE2Pj4yXTtlOntpZigoaHwwKTw9KGR8MCkmZj4+PjA+PWU+Pj4wfChkfDApPihofDApKXticmVhayBlfWU9RVtmK0RbYj4+Ml18MF07Zj1mKzF8MDtkPWY/ZDpkKzF8MDtEW2IrMTY+PjJdPWY7RFtiKzIwPj4yXT1kO1FiKGsrMTZ8MCxlKTtpZihlKXtkPURbYT4+Ml07bT1SYihrKzE2fDApO2Y9RFtkKzIwPj4yXTtpPURbZCsxNj4+Ml07Yj1lO249aStlfDA7aD1EW2QrMTI+PjJdO2Y9ZT4+PjA+bj4+PjA/ZisxfDA6ZjtpZihuPj4+MD5HW2QrOD4+Ml0mKGh8MCk8PShmfDApfChmfDApPihofDApKXticmVhayBlfW9hKG0saStEW2Q+PjJdfDAsZSk7Yz1EW2QrMjA+PjJdO2U9YitEW2QrMTY+PjJdfDA7Yz1lPj4+MDxiPj4+MD9jKzF8MDpjO0RbZCsxNj4+Ml09ZTtEW2QrMjA+PjJdPWN9aj1uYSgyNCk7Yj1qO0RbYis0Pj4yXT0wO0RbYis4Pj4yXT0wO2I9YisxNnwwO0RbYj4+Ml09MDtEW2IrND4+Ml09MDtEW2o+PjJdPWorNDtEW2orMTI+PjJdPWI7ZT0kLTMyfDA7JD1lO2Y9ZysxMnwwO2M9aysxNnwwO2w9X2EoZixjKTtwPWcrMTZ8MDtmOntpZigobHwwKT09KHB8MCkpe0RbZSsxNj4+Ml09YztiPURbZis0Pj4yXTtnOntoOntpZihiKXtkPUVbYysxMXwwXTtnPWQ8PDI0Pj4yNDwwO2g9Zz9EW2M+PjJdOmM7Zz1nP0RbYys0Pj4yXTpkO2M9Zis0fDA7d2hpbGUoMSl7ZD1FW2IrMjd8MF07bj1kPDwyND4+MjQ8MDtkPW4/RFtiKzIwPj4yXTpkO209ZD4+PjA8Zz4+PjA7aTp7ajp7azp7bDp7bTp7aT1tP2Q6ZztuOntpZihpKXtvPWIrMTZ8MDtuPW4/RFtvPj4yXTpvO289dmEoaCxuLGkpO2lmKCFvKXtpZihkPj4+MD5nPj4+MCl7YnJlYWsgbn1icmVhayBtfWlmKChvfDApPj0wKXticmVhayBtfWJyZWFrIG59aWYoZD4+PjA8PWc+Pj4wKXticmVhayBsfX1kPURbYj4+Ml07aWYoZCl7YnJlYWsgaX1icmVhayBofWQ9dmEobixoLGkpO2lmKGQpe2JyZWFrIGt9fWlmKG0pe2JyZWFrIGp9YnJlYWsgZ31pZigoZHwwKT49MCl7YnJlYWsgZ319Yz1iKzR8MDtkPURbYis0Pj4yXTtpZighZCl7YnJlYWsgZ31iPWN9Yz1iO2I9ZDtjb250aW51ZX19Yj1mKzR8MH1jPWJ9ZD1EW2M+PjJdO2lmKGQpe2I9MH1lbHNle2Q9bmEoMzIpO2g9ZCsxNnwwO2c9RFtlKzE2Pj4yXTtvOntpZihCW2crMTF8MF0+PTApe249RFtnKzQ+PjJdO0RbaD4+Ml09RFtnPj4yXTtEW2grND4+Ml09bjtEW2grOD4+Ml09RFtnKzg+PjJdO2JyZWFrIG99dGEoaCxEW2c+PjJdLERbZys0Pj4yXSl9RFtkKzg+PjJdPWI7RFtkPj4yXT0wO0RbZCs0Pj4yXT0wO0RbZCsyOD4+Ml09MDtEW2M+PjJdPWQ7Yj1EW0RbZj4+Ml0+PjJdO2lmKGIpe0RbZj4+Ml09YjtiPURbYz4+Ml19ZWxzZXtiPWR9cGIoRFtmKzQ+PjJdLGIpO0RbZis4Pj4yXT1EW2YrOD4+Ml0rMTtiPTF9QltlKzI4fDBdPWI7RFtlKzI0Pj4yXT1kO2M9RFtlKzI0Pj4yXTtiPURbYysyOD4+Ml07RFtjKzI4Pj4yXT1qO2lmKCFiKXticmVhayBmfUNhKGIrMTJ8MCxEW2IrMTY+PjJdKTtCYShiLERbYis0Pj4yXSk7bWEoYik7YnJlYWsgZn1pZighail7YnJlYWsgZn1DYShqKzEyfDAsRFtqKzE2Pj4yXSk7QmEoaixEW2orND4+Ml0pO21hKGopfSQ9ZSszMnwwO2M9KGx8MCkhPShwfDApfWlmKEJbaysyN3wwXTwwKXttYShEW2srMTY+PjJdKX1pZihjKXticmVhayBifX1pZighail7YnJlYWsgYn1EW2srMTY+PjJdPTA7aWYoIWViKDEsaysxNnwwLERbYT4+Ml0pKXticmVhayBifWI9MDtyPURbaysxNj4+Ml07aWYocil7d2hpbGUoMSl7bj0wO2k9JC0zMnwwOyQ9aTtEW2krMjQ+PjJdPTA7RFtpKzE2Pj4yXT0wO0RbaSsyMD4+Ml09MDtjPURbYT4+Ml07ZT1EW2MrOD4+Ml07cDp7cTp7Zj1EW2MrMTI+PjJdO2Q9RFtjKzIwPj4yXTtnPURbYysxNj4+Ml07cjp7aWYoKGZ8MCk8PShkfDApJmc+Pj4wPj1lPj4+MHwoZHwwKT4oZnwwKSl7YnJlYWsgcn1mPUVbZytEW2M+PjJdfDBdO2U9YztjPWQ7ZD1nKzF8MDtjPWQ/YzpjKzF8MDtEW2UrMTY+PjJdPWQ7RFtlKzIwPj4yXT1jO1FiKGkrMTZ8MCxmKTtpZihmKXtlPURbYT4+Ml07bT1SYihpKzE2fDApO2Q9RFtlKzIwPj4yXTtsPURbZSsxNj4+Ml07Yz1mO2g9bCtjfDA7Zz1EW2UrMTI+PjJdO2Q9Yz4+PjA+aD4+PjA/ZCsxfDA6ZDtpZihoPj4+MD5HW2UrOD4+Ml0mKGd8MCk8PShkfDApfChkfDApPihnfDApKXticmVhayByfW9hKG0sbCtEW2U+PjJdfDAsZik7ZD1EW2UrMjA+PjJdO2Y9YytEW2UrMTY+PjJdfDA7ZD1mPj4+MDxjPj4+MD9kKzF8MDpkO0RbZSsxNj4+Ml09ZjtEW2UrMjA+PjJdPWR9RFtpKzEyPj4yXT0wO2lmKCFlYigxLGkrMTJ8MCxEW2E+PjJdKSl7YnJlYWsgcn1jPURbaSsxMj4+Ml07aWYoIWMpe2JyZWFrIHJ9RFtpKzg+PjJdPTA7RFtpPj4yXT0wO0RbaSs0Pj4yXT0wO2lmKChjfDApPDApe2JyZWFrIHF9ZD1uYShjKTtEW2k+PjJdPWQ7ZT1jK2R8MDtEW2krOD4+Ml09ZTttPXBhKGQsMCxjKTtEW2krND4+Ml09ZTtlPURbYT4+Ml07bj1EW2UrOD4+Ml07Zz1EW2UrMTI+PjJdO2Y9RFtlKzIwPj4yXTtsPURbZSsxNj4+Ml07aD1jK2x8MDtmPWg+Pj4wPGM+Pj4wP2YrMXwwOmY7bj0oZnwwKTw9KGd8MCkmaD4+PjA8PW4+Pj4wfChmfDApPChnfDApO2lmKG4pe29hKG0sbCtEW2U+PjJdfDAsYyk7ZD1jO2Y9YytEW2UrMTY+PjJdfDA7Yz1EW2UrMjA+PjJdO0RbZSsxNj4+Ml09ZjtEW2UrMjA+PjJdPWQ+Pj4wPmY+Pj4wP2MrMXwwOmM7Zz0kLTQ4fDA7JD1nO2Q9X2EoaixpKzE2fDApO2lmKChkfDApIT0oais0fDApKXtlPURbZCs0Pj4yXTtzOntpZighZSl7Zj1EW2QrOD4+Ml07aWYoRFtmPj4yXT09KGR8MCkpe2JyZWFrIHN9Yz1kKzh8MDt3aGlsZSgxKXtlPURbYz4+Ml07Yz1lKzh8MDtmPURbZSs4Pj4yXTtpZigoZXwwKSE9RFtmPj4yXSl7Y29udGludWV9YnJlYWt9YnJlYWsgc313aGlsZSgxKXtmPWU7ZT1EW2U+PjJdO2lmKGUpe2NvbnRpbnVlfWJyZWFrfX1pZihEW2o+PjJdPT0oZHwwKSl7RFtqPj4yXT1mfURbais4Pj4yXT1EW2orOD4+Ml0tMTtmPURbais0Pj4yXTtoPWQ7dDp7dTp7ZT1EW2Q+PjJdO2lmKGUpe2M9RFtoKzQ+PjJdO2lmKCFjKXticmVhayB1fXdoaWxlKDEpe2Q9YztjPURbYz4+Ml07aWYoYyl7Y29udGludWV9YnJlYWt9fWU9RFtkKzQ+PjJdO2lmKGUpe2JyZWFrIHV9ZT0wO209MTticmVhayB0fURbZSs4Pj4yXT1EW2QrOD4+Ml07bT0wfWw9RFtkKzg+PjJdO2M9RFtsPj4yXTt2OntpZigoZHwwKT09KGN8MCkpe0RbbD4+Ml09ZTtpZigoZHwwKT09KGZ8MCkpe2M9MDtmPWU7YnJlYWsgdn1jPURbbCs0Pj4yXTticmVhayB2fURbbCs0Pj4yXT1lfW89IUVbZCsxMnwwXTtpZigoZHwwKSE9KGh8MCkpe2w9RFtoKzg+PjJdO0RbZCs4Pj4yXT1sO0RbbCsoKChofDApIT1EW0RbaCs4Pj4yXT4+Ml0pPDwyKT4+Ml09ZDtsPURbaD4+Ml07RFtkPj4yXT1sO0RbbCs4Pj4yXT1kO2w9RFtoKzQ+PjJdO0RbZCs0Pj4yXT1sO2lmKGwpe0RbbCs4Pj4yXT1kfUJbZCsxMnwwXT1FW2grMTJ8MF07Zj0oZnwwKT09KGh8MCk/ZDpmfXc6e2lmKG98IWYpe2JyZWFrIHd9aWYobSl7d2hpbGUoMSl7ZT1FW2MrMTJ8MF07eDp7ZD1EW2MrOD4+Ml07aWYoRFtkPj4yXSE9KGN8MCkpe2lmKCFlKXtCW2MrMTJ8MF09MTtCW2QrMTJ8MF09MDtlPURbZCs0Pj4yXTttPURbZT4+Ml07RFtkKzQ+PjJdPW07aWYobSl7RFttKzg+PjJdPWR9RFtlKzg+PjJdPURbZCs4Pj4yXTttPURbZCs4Pj4yXTtEWygoKGR8MCkhPURbbT4+Ml0pPDwyKSttPj4yXT1lO0RbZT4+Ml09ZDtEW2QrOD4+Ml09ZTtkPWM7Yz1EW2M+PjJdO2Y9KGN8MCk9PShmfDApP2Q6ZjtjPURbYys0Pj4yXX15Ont6OntkPURbYz4+Ml07QTp7aWYoIShFW2QrMTJ8MF0/MDpkKSl7ZT1EW2MrND4+Ml07aWYoRVtlKzEyfDBdPzA6ZSl7YnJlYWsgQX1CW2MrMTJ8MF09MDtjPURbYys4Pj4yXTtCOntpZigoZnwwKT09KGN8MCkpe2M9ZjticmVhayBCfWlmKEVbYysxMnwwXSl7YnJlYWsgeH19QltjKzEyfDBdPTE7YnJlYWsgd31lPURbYys0Pj4yXTtpZighZSl7YnJlYWsgen19aWYoRVtlKzEyfDBdKXticmVhayB6fWQ9YzticmVhayB5fUJbZCsxMnwwXT0xO0JbYysxMnwwXT0wO2U9RFtkKzQ+PjJdO0RbYz4+Ml09ZTtpZihlKXtEW2UrOD4+Ml09Y31EW2QrOD4+Ml09RFtjKzg+PjJdO2U9RFtjKzg+PjJdO0RbKChEW2U+PjJdIT0oY3wwKSk8PDIpK2U+PjJdPWQ7RFtkKzQ+PjJdPWM7RFtjKzg+PjJdPWQ7ZT1jfWM9RFtkKzg+PjJdO0JbZCsxMnwwXT1FW2MrMTJ8MF07QltjKzEyfDBdPTE7QltlKzEyfDBdPTE7ZD1EW2MrND4+Ml07ZT1EW2Q+PjJdO0RbYys0Pj4yXT1lO2lmKGUpe0RbZSs4Pj4yXT1jfURbZCs4Pj4yXT1EW2MrOD4+Ml07ZT1EW2MrOD4+Ml07RFsoKChjfDApIT1EW2U+PjJdKTw8MikrZT4+Ml09ZDtEW2Q+PjJdPWM7RFtjKzg+PjJdPWQ7YnJlYWsgd31pZighZSl7QltjKzEyfDBdPTE7QltkKzEyfDBdPTA7ZT1EW2MrND4+Ml07RFtkPj4yXT1lO2lmKGUpe0RbZSs4Pj4yXT1kfURbYys4Pj4yXT1EW2QrOD4+Ml07ZT1EW2QrOD4+Ml07RFsoKChkfDApIT1EW2U+PjJdKTw8MikrZT4+Ml09YztEW2MrND4+Ml09ZDtEW2QrOD4+Ml09YztmPShkfDApPT0oZnwwKT9jOmY7Yz1EW2Q+PjJdfWU9RFtjPj4yXTtDOntpZighKCFlfEVbZSsxMnwwXSkpe2Q9YzticmVhayBDfWQ9RFtjKzQ+PjJdO2lmKCEoRVtkKzEyfDBdPzA6ZCkpe0JbYysxMnwwXT0wO2M9RFtjKzg+PjJdO2lmKChjfDApIT0oZnwwKT9FW2MrMTJ8MF06MCl7YnJlYWsgeH1CW2MrMTJ8MF09MTticmVhayB3fWlmKGUpe2lmKCFFW2UrMTJ8MF0pe2Q9YzticmVhayBDfWQ9RFtjKzQ+PjJdfUJbZCsxMnwwXT0xO0JbYysxMnwwXT0wO2U9RFtkPj4yXTtEW2MrND4+Ml09ZTtpZihlKXtEW2UrOD4+Ml09Y31EW2QrOD4+Ml09RFtjKzg+PjJdO2U9RFtjKzg+PjJdO0RbKChEW2U+PjJdIT0oY3wwKSk8PDIpK2U+PjJdPWQ7RFtkPj4yXT1jO0RbYys4Pj4yXT1kO2U9Y31jPURbZCs4Pj4yXTtCW2QrMTJ8MF09RVtjKzEyfDBdO0JbYysxMnwwXT0xO0JbZSsxMnwwXT0xO2Q9RFtjPj4yXTtlPURbZCs0Pj4yXTtEW2M+PjJdPWU7aWYoZSl7RFtlKzg+PjJdPWN9RFtkKzg+PjJdPURbYys4Pj4yXTtlPURbYys4Pj4yXTtEWygoKGN8MCkhPURbZT4+Ml0pPDwyKStlPj4yXT1kO0RbZCs0Pj4yXT1jO0RbYys4Pj4yXT1kO2JyZWFrIHd9ZD1jO2M9RFtjKzg+PjJdO2M9RFsoKChkfDApPT1EW2M+PjJdKTw8MikrYz4+Ml07Y29udGludWV9fUJbZSsxMnwwXT0xfWM9RFtoKzI4Pj4yXTtpZihjKXtEW2grMzI+PjJdPWM7bWEoYyl9aWYoQltoKzI3fDBdPDApe21hKERbaCsxNj4+Ml0pfW1hKGgpfURbZz4+Ml09MDtEW2crND4+Ml09MDtEW2crOD4+Ml09MDtjPURbaSs0Pj4yXS1EW2k+PjJdfDA7Z2IoZyxjKTtvYShEW2c+PjJdLERbaT4+Ml0sYyk7RDp7aWYoQltpKzI3fDBdPj0wKXtEW2crMjQ+PjJdPURbaSsyND4+Ml07Yz1EW2krMjA+PjJdO0RbZysxNj4+Ml09RFtpKzE2Pj4yXTtEW2crMjA+PjJdPWM7YnJlYWsgRH10YShnKzE2fDAsRFtpKzE2Pj4yXSxEW2krMjA+PjJdKX1EW2crMzY+PjJdPTA7RFtnKzI4Pj4yXT0wO0RbZyszMj4+Ml09MDtnYihnKzI4fDAsRFtnKzQ+PjJdLURbZz4+Ml18MCk7Yz1EW2c+PjJdO29hKERbZysyOD4+Ml0sYyxEW2crND4+Ml0tY3wwKTtmPWcrMTZ8MDtkPWY7Yz1EW2orND4+Ml07RTp7Rjp7aWYoYyl7ZT1FW2QrMTF8MF07aD1lPDwyND4+MjQ8MDttPWg/RFtkPj4yXTpkO2g9aD9EW2QrND4+Ml06ZTtkPWorNHwwO3doaWxlKDEpe2U9RVtjKzI3fDBdO2w9ZTw8MjQ+PjI0PDA7ZT1sP0RbYysyMD4+Ml06ZTtwPWU+Pj4wPGg+Pj4wO0c6e0g6e0k6e0o6e0s6e289cD9lOmg7TDp7aWYobyl7cT1jKzE2fDA7bD1sP0RbcT4+Ml06cTtxPXZhKG0sbCxvKTtpZighcSl7aWYoZT4+PjA+aD4+PjApe2JyZWFrIEx9YnJlYWsgS31pZigocXwwKT49MCl7YnJlYWsgS31icmVhayBMfWlmKGU+Pj4wPD1oPj4+MCl7YnJlYWsgSn19ZT1EW2M+PjJdO2lmKGUpe2JyZWFrIEd9YnJlYWsgRn1lPXZhKGwsbSxvKTtpZihlKXticmVhayBJfX1pZihwKXticmVhayBIfWJyZWFrIEV9aWYoKGV8MCk+PTApe2JyZWFrIEV9fWQ9Yys0fDA7ZT1EW2MrND4+Ml07aWYoIWUpe2JyZWFrIEV9Yz1kfWQ9YztjPWU7Y29udGludWV9fWM9ais0fDB9ZD1jfWU9RFtkPj4yXTtpZihlKXtjPTB9ZWxzZXtlPW5hKDQwKTtEW2UrMjQ+PjJdPURbZis4Pj4yXTtoPURbZis0Pj4yXTtEW2UrMTY+PjJdPURbZj4+Ml07RFtlKzIwPj4yXT1oO0RbZj4+Ml09MDtEW2YrND4+Ml09MDtEW2YrOD4+Ml09MDtEW2UrMzY+PjJdPTA7aD1lKzI4fDA7RFtoPj4yXT0wO0RbaCs0Pj4yXT0wO2diKGgsRFtmKzE2Pj4yXS1EW2YrMTI+PjJdfDApO209RFtoPj4yXTtoPURbZisxMj4+Ml07b2EobSxoLERbZisxNj4+Ml0taHwwKTtEW2UrOD4+Ml09YztEW2U+PjJdPTA7RFtlKzQ+PjJdPTA7RFtkPj4yXT1lO2M9RFtEW2o+PjJdPj4yXTtpZihjKXtEW2o+PjJdPWM7Yz1EW2Q+PjJdfWVsc2V7Yz1lfXBiKERbais0Pj4yXSxjKTtEW2orOD4+Ml09RFtqKzg+PjJdKzE7Yz0xfUJbZys0NHwwXT1jO0RbZys0MD4+Ml09ZTtjPURbZysyOD4+Ml07aWYoYyl7RFtnKzMyPj4yXT1jO21hKGMpfWlmKEJbZysyN3wwXTwwKXttYShEW2crMTY+PjJdKX1jPURbZz4+Ml07aWYoYyl7RFtnKzQ+PjJdPWM7bWEoYyl9JD1nKzQ4fDA7ZD1EW2k+PjJdO2lmKCFkKXticmVhayByfX1EW2krND4+Ml09ZDttYShkKX1pZihCW2krMjd8MF08MCl7bWEoRFtpKzE2Pj4yXSl9JD1pKzMyfDA7YnJlYWsgcH1xYSgpO1QoKX1pZighbil7YnJlYWsgYn1iPWIrMXwwO2lmKChyfDApIT0oYnwwKSl7Y29udGludWV9YnJlYWt9fURbaysxMj4+Ml09MDtpZighZWIoMSxrKzEyfDAsRFthPj4yXSkpe2JyZWFrIGJ9Yj1EW2E+PjJdO2M9RFtiKzg+PjJdO2Q9RFtiKzE2Pj4yXTtlPWMtZHwwO2Y9RFtrKzEyPj4yXTtiPURbYisxMj4+Ml0tKERbYisyMD4+Ml0rKGM+Pj4wPGQ+Pj4wKXwwKXwwO2lmKGU+Pj4wPGY+Pj4wJihifDApPD0wfChifDApPDApe2JyZWFrIGJ9Yz0wO2I9RFtrKzM2Pj4yXTtpZihmKXt3aGlsZSgxKXtlPURbays0MD4+Ml07TTp7aWYoZT4+PjA+Yj4+PjApe0RbYis0Pj4yXT0wO0RbYj4+Ml09ajtiPWIrOHwwO0RbayszNj4+Ml09YjticmVhayBNfWQ9RFtrKzMyPj4yXTtnPWItZHwwO2g9Zz4+MztiPWgrMXwwO2lmKGI+Pj4wPj01MzY4NzA5MTIpe2JyZWFrIGR9ZT1lLWR8MDtuPWU+PjI7Yj1lPj4zPj4+MDwyNjg0MzU0NTU/Yj4+PjA+bj4+PjA/YjpuOjUzNjg3MDkxMTtpZihiPj4+MD49NTM2ODcwOTEyKXticmVhayBjfW09aDw8MztoPWI8PDM7ZT1uYShoKTtiPW0rZXwwO0RbYis0Pj4yXT0wO0RbYj4+Ml09ajtiPWIrOHwwO2lmKChnfDApPjApe29hKGUsZCxnKX1EW2srNDA+PjJdPWUraDtEW2srMzY+PjJdPWI7RFtrKzMyPj4yXT1lO2lmKCFkKXticmVhayBNfW1hKGQpfWM9YysxfDA7aWYoKGZ8MCkhPShjfDApKXtjb250aW51ZX1icmVha319aWYoRFtrKzMyPj4yXSE9KGJ8MCkpe2NvbnRpbnVlfWJyZWFrfWE9MTticmVhayBhfXFhKCk7VCgpfXJhKDEzMjYpO1QoKX1iPURbayszMj4+Ml07YT0wfWlmKGIpe21hKGIpfSQ9ays0OHwwO3JldHVybiBhfWZ1bmN0aW9uIGhlKGEsYixjKXthPWF8MDtiPWJ8MDtjPWN8MDt2YXIgZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MDtqPSQtNDh8MDskPWo7YTp7aWYoKGN8MCkhPTEpe2JyZWFrIGF9Zj1EW2ErND4+Ml07YT1EW2ErMTI+PjJdO0Rbais0MD4+Ml09MDtEW2orMzI+PjJdPTA7RFtqKzM2Pj4yXT0wO0RbaisyND4+Ml09MDtEW2orMjg+PjJdPTA7RFtqKzE2Pj4yXT0wO0RbaisyMD4+Ml09MDtEW2orOD4+Ml09MDtEW2orMTI+PjJdPTA7ZT1qKzh8MDtiOntpZigoYnwwKT09LTIpe2JyZWFrIGJ9bD1EW0RbRFtmKzQ+PjJdKzg+PjJdKyhhPDwyKT4+Ml07aWYoKGJhW0RbRFtmPj4yXSs4Pj4yXV0oZil8MCk9PTEpe2s9JC0zMnwwOyQ9aztnPURbRFtEW2YrND4+Ml0rOD4+Ml0rKGE8PDIpPj4yXTtjOntkOntlOntpZigoYmFbRFtEW2Y+PjJdKzg+PjJdXShmKXwwKSE9MXxiLTE+Pj4wPjUpe2JyZWFrIGV9aT1iYVtEW0RbZj4+Ml0rMzY+PjJdXShmKXwwO2g9YmFbRFtEW2Y+PjJdKzQ0Pj4yXV0oZixhKXwwO2lmKCFpfCFoKXticmVhayBlfWM9YmFbRFtEW2Y+PjJdKzQwPj4yXV0oZixhKXwwO2lmKGMpe2E9RFtmKzQ0Pj4yXTtEW2srMTI+PjJdPWM7RFtrKzg+PjJdPWE7RFtrKzIwPj4yXT1oO0RbaysxNj4+Ml09aCsxMjtmPWsrOHwwO2E9MDtmOntnOntzd2l0Y2goYi0xfDApe2Nhc2UgMDphPW5hKDYwKTtEW2ErND4+Ml09ZztEW2E+PjJdPTI5ODg7Zz1EW2UrOD4+Ml07aD1EW2UrMTI+PjJdO2k9RFtlKzE2Pj4yXTtkPURbZSsyMD4+Ml07Yz1EW2U+PjJdO2I9RFtlKzQ+PjJdO0RbYSs0MD4+Ml09MDtEW2ErMzI+PjJdPTA7RFthKzM2Pj4yXT0wO0RbYSsyND4+Ml09aTtEW2ErMjg+PjJdPWQ7RFthKzE2Pj4yXT1nO0RbYSsyMD4+Ml09aDtEW2ErOD4+Ml09YztEW2ErMTI+PjJdPWI7Yj1EW2UrMjQ+PjJdO2Q9RFtlKzI4Pj4yXS1ifDA7aWYoZCl7aWYoKGR8MCk8MCl7YnJlYWsgZH1jPW5hKGQpO0RbYSszMj4+Ml09YztEW2ErNDA+PjJdPWMrKGQ+PjI8PDIpO209YSxuPW9hKGMsYixkKStkfDAsRFttKzM2Pj4yXT1ufWI9RFtmKzQ+PjJdO0RbYSs0ND4+Ml09RFtmPj4yXTtEW2ErNDg+PjJdPWI7Yj1EW2YrMTI+PjJdO0RbYSs1Mj4+Ml09RFtmKzg+PjJdO0RbYSs1Nj4+Ml09YjtEW2E+PjJdPTIyODA7YnJlYWsgZjtjYXNlIDM6YT1uYSgxMTIpO0RbYSs0Pj4yXT1nO0RbYT4+Ml09Mjk4ODtnPURbZSs4Pj4yXTtoPURbZSsxMj4+Ml07aT1EW2UrMTY+PjJdO2Q9RFtlKzIwPj4yXTtjPURbZT4+Ml07Yj1EW2UrND4+Ml07RFthKzQwPj4yXT0wO0RbYSszMj4+Ml09MDtEW2ErMzY+PjJdPTA7RFthKzI0Pj4yXT1pO0RbYSsyOD4+Ml09ZDtEW2ErMTY+PjJdPWc7RFthKzIwPj4yXT1oO0RbYSs4Pj4yXT1jO0RbYSsxMj4+Ml09YjtiPURbZSsyND4+Ml07ZD1EW2UrMjg+PjJdLWJ8MDtpZihkKXtpZigoZHwwKTwwKXticmVhayBkfWM9bmEoZCk7RFthKzMyPj4yXT1jO0RbYSs0MD4+Ml09YysoZD4+Mjw8Mik7bT1hLG49b2EoYyxiLGQpK2R8MCxEW20rMzY+PjJdPW59Yj1EW2YrND4+Ml07RFthKzQ0Pj4yXT1EW2Y+PjJdO0RbYSs0OD4+Ml09YjtiPURbZisxMj4+Ml07RFthKzUyPj4yXT1EW2YrOD4+Ml07RFthKzU2Pj4yXT1iO0RbYSs2MD4+Ml09MDtEW2ErNjQ+PjJdPTA7RFthPj4yXT0zMDQ0O0RbYSs2OD4+Ml09MDtEW2ErNzI+PjJdPTA7RFthKzc2Pj4yXT0wO0RbYSs4MD4+Ml09MDtEW2ErODQ+PjJdPTA7RFthKzg4Pj4yXT0wO0RbYSs5Mj4+Ml09MDtEW2ErOTY+PjJdPTA7RFthKzEwMD4+Ml09MDtEW2ErMTA0Pj4yXT0wO0RbYSsxMDg+PjJdPTA7YnJlYWsgZjtjYXNlIDQ6YT1uYSgxMDQpO0RbYSs0Pj4yXT1nO0RbYT4+Ml09Mjk4ODtnPURbZSs4Pj4yXTtoPURbZSsxMj4+Ml07aT1EW2UrMTY+PjJdO2Q9RFtlKzIwPj4yXTtjPURbZT4+Ml07Yj1EW2UrND4+Ml07RFthKzQwPj4yXT0wO0RbYSszMj4+Ml09MDtEW2ErMzY+PjJdPTA7RFthKzI0Pj4yXT1pO0RbYSsyOD4+Ml09ZDtEW2ErMTY+PjJdPWc7RFthKzIwPj4yXT1oO0RbYSs4Pj4yXT1jO0RbYSsxMj4+Ml09YjtiPURbZSsyND4+Ml07ZD1EW2UrMjg+PjJdLWJ8MDtpZihkKXtpZigoZHwwKTwwKXticmVhayBkfWM9bmEoZCk7RFthKzMyPj4yXT1jO0RbYSs0MD4+Ml09YysoZD4+Mjw8Mik7bT1hLG49b2EoYyxiLGQpK2R8MCxEW20rMzY+PjJdPW59Yj1EW2YrND4+Ml07RFthKzQ0Pj4yXT1EW2Y+PjJdO0RbYSs0OD4+Ml09YjtjPURbZis4Pj4yXTtiPURbZisxMj4+Ml07RFthKzg0Pj4yXT0wO0RbYSs3Nj4+Ml09MDtEW2ErODA+PjJdPTA7RFthKzYwPj4yXT0wO0RbYSs2ND4+Ml09MDtEW2E+PjJdPTMyOTI7RFthKzUyPj4yXT1jO0RbYSs1Nj4+Ml09YjtiPURbZis0Pj4yXTtEW2ErODg+PjJdPURbZj4+Ml07RFthKzkyPj4yXT1iO2I9RFtmKzEyPj4yXTtEW2ErOTY+PjJdPURbZis4Pj4yXTtEW2ErMTAwPj4yXT1iO2JyZWFrIGY7Y2FzZSA1OmJyZWFrIGc7ZGVmYXVsdDpicmVhayBmfX1hPW5hKDEyOCk7RFthKzQ+PjJdPWc7RFthPj4yXT0yOTg4O2c9RFtlKzg+PjJdO2g9RFtlKzEyPj4yXTtpPURbZSsxNj4+Ml07ZD1EW2UrMjA+PjJdO2M9RFtlPj4yXTtiPURbZSs0Pj4yXTtEW2ErNDA+PjJdPTA7RFthKzMyPj4yXT0wO0RbYSszNj4+Ml09MDtEW2ErMjQ+PjJdPWk7RFthKzI4Pj4yXT1kO0RbYSsxNj4+Ml09ZztEW2ErMjA+PjJdPWg7RFthKzg+PjJdPWM7RFthKzEyPj4yXT1iO2g6e2k6e2M9RFtlKzI4Pj4yXS1EW2UrMjQ+PjJdfDA7aWYoYyl7aWYoKGN8MCk8MCl7YnJlYWsgaX1iPW5hKGMpO0RbYSszMj4+Ml09YjtEW2ErMzY+PjJdPWI7RFthKzQwPj4yXT1iKyhjPj4yPDwyKTtjPURbZSsyND4+Ml07ZD1EW2UrMjg+PjJdLWN8MDtpZigoZHwwKT4wKXtiPW9hKGIsYyxkKStkfDB9RFthKzM2Pj4yXT1ifURbYT4+Ml09MjkzMjtiPURbZis0Pj4yXTtEW2ErNDQ+PjJdPURbZj4+Ml07RFthKzQ4Pj4yXT1iO2I9RFtmKzEyPj4yXTtEW2ErNTI+PjJdPURbZis4Pj4yXTtEW2ErNTY+PjJdPWI7Yj1hLSAtNjR8MDtEW2I+PjJdPTA7RFtiKzQ+PjJdPTA7RFthKzYwPj4yXT00MTU2O0RbYT4+Ml09MzUyODtiPURbZis0Pj4yXTtEW2ErNzI+PjJdPURbZj4+Ml07RFthKzc2Pj4yXT1iO2I9RFtmKzEyPj4yXTtEW2ErODA+PjJdPURbZis4Pj4yXTtEW2ErODQ+PjJdPWI7RFthKzEwND4+Ml09MTA2NTM1MzIxNjtEW2ErMTA4Pj4yXT0tMTtEW2ErOTY+PjJdPS0xO0RbYSsxMDA+PjJdPS0xO0RbYSs4OD4+Ml09MTtEW2ErOTI+PjJdPS0xO0RbYSs2MD4+Ml09Mzc2NDtEW2ErMTEyPj4yXT0wO0RbYSsxMTY+PjJdPTA7QlthKzExN3wwXT0wO0JbYSsxMTh8MF09MDtCW2ErMTE5fDBdPTA7QlthKzEyMHwwXT0wO0JbYSsxMjF8MF09MDtCW2ErMTIyfDBdPTA7QlthKzEyM3wwXT0wO0JbYSsxMjR8MF09MDticmVhayBofXFhKCk7VCgpfWJyZWFrIGZ9ZD1hO2JyZWFrIGV9YT1EW2YrNDQ+PjJdO0RbaysxMj4+Ml09aTtEW2srOD4+Ml09YTtEW2srMjA+PjJdPWg7RFtrKzE2Pj4yXT1oKzEyO2Y9ays4fDA7YT0wO2o6e2s6e3N3aXRjaChiLTF8MCl7Y2FzZSAwOmE9bmEoNjApO0RbYSs0Pj4yXT1nO0RbYT4+Ml09Mjk4ODtnPURbZSs4Pj4yXTtoPURbZSsxMj4+Ml07aT1EW2UrMTY+PjJdO2Q9RFtlKzIwPj4yXTtjPURbZT4+Ml07Yj1EW2UrND4+Ml07RFthKzQwPj4yXT0wO0RbYSszMj4+Ml09MDtEW2ErMzY+PjJdPTA7RFthKzI0Pj4yXT1pO0RbYSsyOD4+Ml09ZDtEW2ErMTY+PjJdPWc7RFthKzIwPj4yXT1oO0RbYSs4Pj4yXT1jO0RbYSsxMj4+Ml09YjtiPURbZSsyND4+Ml07ZD1EW2UrMjg+PjJdLWJ8MDtpZihkKXtpZigoZHwwKTwwKXticmVhayBkfWM9bmEoZCk7RFthKzMyPj4yXT1jO0RbYSs0MD4+Ml09YysoZD4+Mjw8Mik7bT1hLG49b2EoYyxiLGQpK2R8MCxEW20rMzY+PjJdPW59Yj1EW2YrND4+Ml07RFthKzQ0Pj4yXT1EW2Y+PjJdO0RbYSs0OD4+Ml09YjtiPURbZisxMj4+Ml07RFthKzUyPj4yXT1EW2YrOD4+Ml07RFthKzU2Pj4yXT1iO0RbYT4+Ml09NDE4NDticmVhayBqO2Nhc2UgMzphPW5hKDExMik7RFthKzQ+PjJdPWc7RFthPj4yXT0yOTg4O2c9RFtlKzg+PjJdO2g9RFtlKzEyPj4yXTtpPURbZSsxNj4+Ml07ZD1EW2UrMjA+PjJdO2M9RFtlPj4yXTtiPURbZSs0Pj4yXTtEW2ErNDA+PjJdPTA7RFthKzMyPj4yXT0wO0RbYSszNj4+Ml09MDtEW2ErMjQ+PjJdPWk7RFthKzI4Pj4yXT1kO0RbYSsxNj4+Ml09ZztEW2ErMjA+PjJdPWg7RFthKzg+PjJdPWM7RFthKzEyPj4yXT1iO2I9RFtlKzI0Pj4yXTtkPURbZSsyOD4+Ml0tYnwwO2lmKGQpe2lmKChkfDApPDApe2JyZWFrIGR9Yz1uYShkKTtEW2ErMzI+PjJdPWM7RFthKzQwPj4yXT1jKyhkPj4yPDwyKTttPWEsbj1vYShjLGIsZCkrZHwwLERbbSszNj4+Ml09bn1iPURbZis0Pj4yXTtEW2ErNDQ+PjJdPURbZj4+Ml07RFthKzQ4Pj4yXT1iO2I9RFtmKzEyPj4yXTtEW2ErNTI+PjJdPURbZis4Pj4yXTtEW2ErNTY+PjJdPWI7RFthKzYwPj4yXT0wO0RbYSs2ND4+Ml09MDtEW2E+PjJdPTQ2MDg7RFthKzY4Pj4yXT0wO0RbYSs3Mj4+Ml09MDtEW2ErNzY+PjJdPTA7RFthKzgwPj4yXT0wO0RbYSs4ND4+Ml09MDtEW2ErODg+PjJdPTA7RFthKzkyPj4yXT0wO0RbYSs5Nj4+Ml09MDtEW2ErMTAwPj4yXT0wO0RbYSsxMDQ+PjJdPTA7RFthKzEwOD4+Ml09MDticmVhayBqO2Nhc2UgNDphPW5hKDEwNCk7RFthKzQ+PjJdPWc7RFthPj4yXT0yOTg4O2c9RFtlKzg+PjJdO2g9RFtlKzEyPj4yXTtpPURbZSsxNj4+Ml07ZD1EW2UrMjA+PjJdO2M9RFtlPj4yXTtiPURbZSs0Pj4yXTtEW2ErNDA+PjJdPTA7RFthKzMyPj4yXT0wO0RbYSszNj4+Ml09MDtEW2ErMjQ+PjJdPWk7RFthKzI4Pj4yXT1kO0RbYSsxNj4+Ml09ZztEW2ErMjA+PjJdPWg7RFthKzg+PjJdPWM7RFthKzEyPj4yXT1iO2I9RFtlKzI0Pj4yXTtkPURbZSsyOD4+Ml0tYnwwO2lmKGQpe2lmKChkfDApPDApe2JyZWFrIGR9Yz1uYShkKTtEW2ErMzI+PjJdPWM7RFthKzQwPj4yXT1jKyhkPj4yPDwyKTttPWEsbj1vYShjLGIsZCkrZHwwLERbbSszNj4+Ml09bn1iPURbZis0Pj4yXTtEW2ErNDQ+PjJdPURbZj4+Ml07RFthKzQ4Pj4yXT1iO2M9RFtmKzg+PjJdO2I9RFtmKzEyPj4yXTtEW2ErODQ+PjJdPTA7RFthKzc2Pj4yXT0wO0RbYSs4MD4+Ml09MDtEW2ErNjA+PjJdPTA7RFthKzY0Pj4yXT0wO0RbYT4+Ml09NDg0NDtEW2ErNTI+PjJdPWM7RFthKzU2Pj4yXT1iO2I9RFtmKzQ+PjJdO0RbYSs4OD4+Ml09RFtmPj4yXTtEW2ErOTI+PjJdPWI7Yj1EW2YrMTI+PjJdO0RbYSs5Nj4+Ml09RFtmKzg+PjJdO0RbYSsxMDA+PjJdPWI7YnJlYWsgajtjYXNlIDU6YnJlYWsgaztkZWZhdWx0OmJyZWFrIGp9fWE9bmEoMTI4KTtEW2ErND4+Ml09ZztEW2E+PjJdPTI5ODg7Zz1EW2UrOD4+Ml07aD1EW2UrMTI+PjJdO2k9RFtlKzE2Pj4yXTtkPURbZSsyMD4+Ml07Yz1EW2U+PjJdO2I9RFtlKzQ+PjJdO0RbYSs0MD4+Ml09MDtEW2ErMzI+PjJdPTA7RFthKzM2Pj4yXT0wO0RbYSsyND4+Ml09aTtEW2ErMjg+PjJdPWQ7RFthKzE2Pj4yXT1nO0RbYSsyMD4+Ml09aDtEW2ErOD4+Ml09YztEW2ErMTI+PjJdPWI7bDp7bTp7Yz1EW2UrMjg+PjJdLURbZSsyND4+Ml18MDtpZihjKXtpZigoY3wwKTwwKXticmVhayBtfWI9bmEoYyk7RFthKzMyPj4yXT1iO0RbYSszNj4+Ml09YjtEW2ErNDA+PjJdPWIrKGM+PjI8PDIpO2M9RFtlKzI0Pj4yXTtkPURbZSsyOD4+Ml0tY3wwO2lmKChkfDApPjApe2I9b2EoYixjLGQpK2R8MH1EW2ErMzY+PjJdPWJ9RFthPj4yXT00NTUyO2I9RFtmKzQ+PjJdO0RbYSs0ND4+Ml09RFtmPj4yXTtEW2ErNDg+PjJdPWI7Yj1EW2YrMTI+PjJdO0RbYSs1Mj4+Ml09RFtmKzg+PjJdO0RbYSs1Nj4+Ml09YjtiPWEtIC02NHwwO0RbYj4+Ml09MDtEW2IrND4+Ml09MDtEW2ErNjA+PjJdPTU2NTI7RFthPj4yXT01MDY4O2I9RFtmKzQ+PjJdO0RbYSs3Mj4+Ml09RFtmPj4yXTtEW2ErNzY+PjJdPWI7Yj1EW2YrMTI+PjJdO0RbYSs4MD4+Ml09RFtmKzg+PjJdO0RbYSs4ND4+Ml09YjtEW2ErMTA0Pj4yXT0xMDY1MzUzMjE2O0RbYSsxMDg+PjJdPS0xO0RbYSs5Nj4+Ml09LTE7RFthKzEwMD4+Ml09LTE7RFthKzg4Pj4yXT0xO0RbYSs5Mj4+Ml09LTE7RFthKzYwPj4yXT01Mjg4O0RbYSsxMTI+PjJdPTA7RFthKzExNj4+Ml09MDtCW2ErMTE3fDBdPTA7QlthKzExOHwwXT0wO0JbYSsxMTl8MF09MDtCW2ErMTIwfDBdPTA7QlthKzEyMXwwXT0wO0JbYSsxMjJ8MF09MDtCW2ErMTIzfDBdPTA7QlthKzEyNHwwXT0wO2JyZWFrIGx9cWEoKTtUKCl9YnJlYWsgan1kPWF9JD1rKzMyfDA7YnJlYWsgY31xYSgpO1QoKX1pZihkKXticmVhayBifX1kPW5hKDQ0KTtEW2QrND4+Ml09bDtEW2Q+PjJdPTI5ODg7Zz1EW2UrOD4+Ml07aD1EW2UrMTI+PjJdO2k9RFtlKzE2Pj4yXTtjPURbZSsyMD4+Ml07Yj1EW2U+PjJdO2E9RFtlKzQ+PjJdO0RbZCs0MD4+Ml09MDtEW2QrMzI+PjJdPTA7RFtkKzM2Pj4yXT0wO0RbZCsyND4+Ml09aTtEW2QrMjg+PjJdPWM7RFtkKzE2Pj4yXT1nO0RbZCsyMD4+Ml09aDtEW2QrOD4+Ml09YjtEW2QrMTI+PjJdPWE7bjp7YT1EW2UrMjQ+PjJdO2M9RFtlKzI4Pj4yXS1hfDA7aWYoYyl7aWYoKGN8MCk8MCl7YnJlYWsgbn1iPW5hKGMpO0RbZCszMj4+Ml09YjtEW2QrNDA+PjJdPWIrKGM+PjI8PDIpO209ZCxuPW9hKGIsYSxjKStjfDAsRFttKzM2Pj4yXT1ufURbZD4+Ml09NTY4MDticmVhayBifXFhKCk7VCgpfWE9RFtqKzMyPj4yXTtpZighYSl7YnJlYWsgYX1EW2orMzY+PjJdPWE7bWEoYSl9JD1qKzQ4fDA7cmV0dXJuIGR8MH1mdW5jdGlvbiAkZChhLGIsYyxkLGUsZil7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7ZT1lfDA7Zj1mfDA7dmFyIGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTAsbz0wLHA9MCxxPTAscj0wLHM9MCx0PTAsdT0wLHY9MCx3PTAseD0wLHk9MCx6PTAsQT0wLEI9MCxDPTAsRT0wLEY9MCxHPTAsSD0wLEk9MCxLPTA7aD0kKy02NHwwOyQ9aDtEW2ErOD4+Ml09ZTtmPWErMzJ8MDtnPURbZj4+Ml07ZD1EW2ErMzY+PjJdLWc+PjI7YTp7Yjp7aWYoZD4+PjA8ZT4+PjApe3NhKGYsZS1kfDApO0RbaCs1Nj4+Ml09MDtEW2grNjA+PjJdPTA7RFtoKzQ4Pj4yXT0wO0RbaCs1Mj4+Ml09MDtEW2grNDA+PjJdPTA7RFtoKzQ0Pj4yXT0wO0RbaCszMj4+Ml09MDtEW2grMzY+PjJdPTA7RFtoKzI0Pj4yXT0wO0RbaCsyOD4+Ml09MDtEW2grMTY+PjJdPTA7RFtoKzIwPj4yXT0wO0RbaD4+Ml09MDticmVhayBifWlmKGQ+Pj4wPmU+Pj4wKXtEW2ErMzY+PjJdPWcrKGU8PDIpfURbaCs1Nj4+Ml09MDtEW2grNjA+PjJdPTA7RFtoKzQ4Pj4yXT0wO0RbaCs1Mj4+Ml09MDtEW2grNDA+PjJdPTA7RFtoKzQ0Pj4yXT0wO0RbaCszMj4+Ml09MDtEW2grMzY+PjJdPTA7RFtoKzI0Pj4yXT0wO0RbaCsyOD4+Ml09MDtEW2grMTY+PjJdPTA7RFtoKzIwPj4yXT0wO0RbaD4+Ml09MDtkPTA7aWYoIWUpe2JyZWFrIGF9fXhhKGgrMTZ8MCxlLGgpO2k9RFtoKzI4Pj4yXTtkPURbaCszMj4+Ml19RFtoPj4yXT0wO2Q9ZC1pPj4yO2M6e2lmKGQ+Pj4wPj1lPj4+MCl7aWYoZD4+PjA8PWU+Pj4wKXticmVhayBjfURbaCszMj4+Ml09KGU8PDIpK2k7YnJlYWsgY314YShoKzE2fDEyLGUtZHwwLGgpfURbaD4+Ml09MDtmPURbaCs0MD4+Ml07ZD1EW2grNDQ+PjJdLWY+PjI7ZDp7aWYoZD4+PjA+PWU+Pj4wKXtpZihkPj4+MDw9ZT4+PjApe2JyZWFrIGR9RFtoKzQ0Pj4yXT1mKyhlPDwyKTticmVhayBkfXhhKGgrNDB8MCxlLWR8MCxoKX1EW2g+PjJdPTA7Zj1EW2grNTI+PjJdO2Q9RFtoKzU2Pj4yXS1mPj4yO2U6e2lmKGQ+Pj4wPj1lPj4+MCl7aWYoZD4+PjA8PWU+Pj4wKXticmVhayBlfURbaCs1Nj4+Ml09ZisoZTw8Mik7YnJlYWsgZX14YShoKzUyfDAsZS1kfDAsaCl9aT0wO2Y6e2lmKERbYSs4Pj4yXTw9MCl7YnJlYWsgZn1qPURbYSszMj4+Ml07Zz1EW2grMTY+PjJdO3doaWxlKDEpe2Q9aTw8MjtmPURbZCtnPj4yXTttPURbYSsxNj4+Ml07Zzp7aWYoKGZ8MCk+KG18MCkpe0RbZCtqPj4yXT1tO2JyZWFrIGd9ZD1kK2p8MDttPURbYSsxMj4+Ml07aWYoKG18MCk+KGZ8MCkpe0RbZD4+Ml09bTticmVhayBnfURbZD4+Ml09Zn1pPWkrMXwwO2Q9RFthKzg+PjJdO2lmKChpfDApPChkfDApKXtjb250aW51ZX1icmVha31pZigoZHwwKTw9MCl7YnJlYWsgZn1kPTA7d2hpbGUoMSl7Zz1kPDwyO2Y9ZytjfDA7Zz1EW2IrZz4+Ml0rRFtnK2o+PjJdfDA7RFtmPj4yXT1nO2g6e2lmKChnfDApPkRbYSsxNj4+Ml0pe2c9Zy1EW2ErMjA+PjJdfDB9ZWxzZXtpZigoZ3wwKT49RFthKzEyPj4yXSl7YnJlYWsgaH1nPWcrRFthKzIwPj4yXXwwfURbZj4+Ml09Z31kPWQrMXwwO2lmKChkfDApPERbYSs4Pj4yXSl7Y29udGludWV9YnJlYWt9fUU9RFthKzUyPj4yXTtxPURbYSs0OD4+Ml07dz1uYSgxNik7ZD13O0RbZD4+Ml09MDtEW2QrND4+Ml09MDtEW2QrOD4+Ml09MDtEW2QrMTI+PjJdPTA7RFtoKzg+PjJdPTA7RFtoPj4yXT0wO0RbaCs0Pj4yXT0wO2k6e2lmKGUpe2lmKGU+Pj4wPj0xMDczNzQxODI0KXticmVhayBpfWQ9ZTw8Mjt2PW5hKGQpO0RbaD4+Ml09djtEW2grOD4+Ml09ZCt2O3BhKHYsMCxkKX1wPTE7ZD1EW2ErNTY+PjJdO3o9RFtkPj4yXTtkPURbZCs0Pj4yXS16fDA7ajp7aWYoKGR8MCk8NSl7YnJlYWsgan1kPWQ+PjI7Rj0oZHwwKT4yP2Q6MjtHPWQ+Pj4wPjE/ZDoxO0E9ZSYtMjtCPWUmMTtIPWUmLTQ7Qz1lJjM7eD1lLTF8MDtJPWU8PDI7bT0xO3doaWxlKDEpe2s6e2w6e206e246e2lmKChtfDApIT0oR3wwKSl7Zj1EWyhtPDwyKSt6Pj4yXTtkPShmPj4+MCklM3wwO286e3A6e2lmKChmfDApPT0tMSl7YnJlYWsgcH1qPTA7Zz1mKzJ8MDt5PShkfDApIT0wfChnfDApIT0tMTtuPTE7cj1kP2YtMXwwOmc7bz0xPDxyO3M9cj4+PjV8MDtLPURbcT4+Ml07ZD1mO3E6e3doaWxlKDEpe3I6e2lmKERbKGQ+Pj4zJjUzNjg3MDkwOCkrSz4+Ml0+Pj5kJjEpe2JyZWFrIHJ9Zz1EW0RbRFtxKzY0Pj4yXSsxMj4+Ml0rKGQ8PDIpPj4yXTtpZigoZ3wwKT09LTEpe2JyZWFrIHJ9az1EW0U+PjJdO2k9RFtxKzI4Pj4yXTtwPURbaysoRFtpKyhnPDwyKT4+Ml08PDIpPj4yXTtpZigocHwwKT49KG18MCkpe2JyZWFrIHJ9bD1nKzF8MDtsPURbaysoRFtpKygoKGw+Pj4wKSUzfDA/bDpnLTJ8MCk8PDIpPj4yXTw8Mik+PjJdO2lmKChsfDApPj0obXwwKSl7YnJlYWsgcn1pPURbaysoRFtpKyhnKygoZz4+PjApJTN8MD8tMToyKTw8Mik+PjJdPDwyKT4+Ml07aWYoKGl8MCk+PShtfDApKXticmVhayByfXM6e2lmKCFlKXticmVhayBzfWc9RFsoaCsxNnwwKStKKGosMTIpPj4yXTtrPUooZSxpKTtsPUooZSxsKTtwPUooZSxwKTtpPTA7dT0wO2lmKHgpe3doaWxlKDEpe0RbZysoaTw8Mik+PjJdPShEWyhpK2s8PDIpK2M+PjJdK0RbKGkrbDw8MikrYz4+Ml18MCktRFsoaStwPDwyKStjPj4yXTt0PWl8MTtEW2crKHQ8PDIpPj4yXT0oRFsoayt0PDwyKStjPj4yXStEWyhsK3Q8PDIpK2M+PjJdfDApLURbKHArdDw8MikrYz4+Ml07aT1pKzJ8MDt1PXUrMnwwO2lmKChBfDApIT0odXwwKSl7Y29udGludWV9YnJlYWt9fWlmKCFCKXticmVhayBzfURbZysoaTw8Mik+PjJdPShEWyhpK2s8PDIpK2M+PjJdK0RbKGkrbDw8MikrYz4+Ml18MCktRFsoaStwPDwyKStjPj4yXX1nPTQ7aj1qKzF8MDtpZigoanwwKT09NCl7YnJlYWsgcX19dDp7aWYobiYxKXtpPWQtMnwwO2c9ZCsxfDA7ZD0tMTtnPShnPj4+MCklM3wwP2c6aTtpZigoZ3wwKT09LTF8RFtEW3E+PjJdKyhnPj4+MyY1MzY4NzA5MDgpPj4yXT4+PmcmMSl7YnJlYWsgdH1nPURbRFtEW3ErNjQ+PjJdKzEyPj4yXSsoZzw8Mik+PjJdO2lmKChnfDApPT0tMSl7YnJlYWsgdH1kPWcrMXwwO2Q9KGQ+Pj4wKSUzfDA/ZDpnLTJ8MDticmVhayB0fXU6e2lmKChkPj4+MCklM3wwKXtpPWQtMXwwO2JyZWFrIHV9aT1kKzJ8MDtkPS0xO2lmKChpfDApPT0tMSl7YnJlYWsgdH19ZD0tMTtpZihEW0RbcT4+Ml0rKGk+Pj4zJjUzNjg3MDkwOCk+PjJdPj4+aSYxKXticmVhayB0fWc9RFtEW0RbcSs2ND4+Ml0rMTI+PjJdKyhpPDwyKT4+Ml07aWYoKGd8MCk9PS0xKXticmVhayB0fWlmKChnPj4+MCklM3wwKXtkPWctMXwwO2JyZWFrIHR9ZD1nKzJ8MH12OntpZigoZHwwKT09KGZ8MCkpe2JyZWFrIHZ9aWYoISgobl4xKSYxfChkfDApIT0tMSkpe2lmKCF5fG8mRFtEW3E+PjJdKyhzPDwyKT4+Ml0pe2JyZWFrIHZ9ZD1EW0RbRFtxKzY0Pj4yXSsxMj4+Ml0rKHI8PDIpPj4yXTtpZigoZHwwKT09LTEpe2JyZWFrIHZ9bj0wO2Q9KGQ+Pj4wKSUzfDA/ZC0xfDA6ZCsyfDB9aWYoKGR8MCkhPS0xKXtjb250aW51ZX19YnJlYWt9Zz1qO2lmKChnfDApPD0wKXticmVhayBwfX1pZihlKXtwYShEW2g+PjJdLDAsSSl9ZD1nLTF8MDt0PShkPDwyKSt3fDA7ZD1KKGQsMTIpK2F8MDt1PWQ7eT1EW2QtIC02ND4+Ml07cD0wO249MDtkPTA7d2hpbGUoMSl7Zj1EW3Q+PjJdO0RbdD4+Ml09ZisxO2lmKGY+Pj4wPj15Pj4+MCl7YnJlYWsgan13OntpZihEW0RbdSs2MD4+Ml0rKGY+Pj4zJjUzNjg3MDkwOCk+PjJdPj4+ZiYxKXticmVhayB3fWQ9ZCsxfDA7aWYoIWUpe2JyZWFrIHd9aj1EW2g+PjJdO2s9RFsoaCsxNnwwKStKKG4sMTIpPj4yXTtyPTA7aT0wO2Y9MDtpZih4Pj4+MD49Myl7d2hpbGUoMSl7bD1pPDwyO289bCtqfDA7RFtvPj4yXT1EW28+PjJdK0RbaytsPj4yXTtvPWx8NDtzPW8ranwwO0Rbcz4+Ml09RFtzPj4yXStEW2srbz4+Ml07bz1sfDg7cz1vK2p8MDtEW3M+PjJdPURbcz4+Ml0rRFtrK28+PjJdO2w9bHwxMjtvPWwranwwO0Rbbz4+Ml09RFtvPj4yXStEW2srbD4+Ml07aT1pKzR8MDtmPWYrNHwwO2lmKChIfDApIT0oZnwwKSl7Y29udGludWV9YnJlYWt9fWlmKCFDKXticmVhayB3fXdoaWxlKDEpe2Y9aTw8MjtsPWYranwwO0RbbD4+Ml09RFtsPj4yXStEW2Yraz4+Ml07aT1pKzF8MDtyPXIrMXwwO2lmKChDfDApIT0ocnwwKSl7Y29udGludWV9YnJlYWt9fW49bisxfDA7aWYoKG58MCkhPShnfDApKXtjb250aW51ZX1icmVha31uPUooZSxtKTtmPW47aWYoIWQpe2JyZWFrIG99aWYoIWUpe2JyZWFrIGx9Zj1EW2g+PjJdO2k9MDtnPTA7aWYoeCl7YnJlYWsgbn1icmVhayBtfWY9SihlLG0pfWlmKERbYSs4Pj4yXTw9MCl7YnJlYWsga31uPShKKG0tMXwwLGUpPDwyKStjfDA7aj1EW2ErMzI+PjJdO2k9MDt3aGlsZSgxKXtkPWk8PDI7Zz1EW2Qrbj4+Ml07az1EW2ErMTY+PjJdO3g6e2lmKChnfDApPihrfDApKXtEW2Qraj4+Ml09azticmVhayB4fWQ9ZCtqfDA7az1EW2ErMTI+PjJdO2lmKChrfDApPihnfDApKXtEW2Q+PjJdPWs7YnJlYWsgeH1EW2Q+PjJdPWd9aT1pKzF8MDtnPURbYSs4Pj4yXTtpZigoaXwwKTwoZ3wwKSl7Y29udGludWV9YnJlYWt9ZD0wO2lmKChnfDApPD0wKXticmVhayBrfWY9Zjw8MjtpPWYrY3wwO249YitmfDA7d2hpbGUoMSl7Zz1kPDwyO2Y9ZytpfDA7Zz1EW2crbj4+Ml0rRFtnK2o+PjJdfDA7RFtmPj4yXT1nO3k6e2lmKChnfDApPkRbYSsxNj4+Ml0pe2c9Zy1EW2ErMjA+PjJdfDB9ZWxzZXtpZigoZ3wwKT49RFthKzEyPj4yXSl7YnJlYWsgeX1nPWcrRFthKzIwPj4yXXwwfURbZj4+Ml09Z31kPWQrMXwwO2lmKChkfDApPERbYSs4Pj4yXSl7Y29udGludWV9YnJlYWt9YnJlYWsga311YSgpO1QoKX13aGlsZSgxKXtqPWk8PDI7az1qK2Z8MDtEW2s+PjJdPURbaz4+Ml0vKGR8MCk7aj1mKyhqfDQpfDA7RFtqPj4yXT1EW2o+PjJdLyhkfDApO2k9aSsyfDA7Zz1nKzJ8MDtpZigoQXwwKSE9KGd8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZighQil7YnJlYWsgbH1mPWYrKGk8PDIpfDA7RFtmPj4yXT1EW2Y+PjJdLyhkfDApfWlmKERbYSs4Pj4yXTw9MCl7YnJlYWsga31qPURbYSszMj4+Ml07aT0wO3doaWxlKDEpe2Q9aTw8MjtmPURbZCt2Pj4yXTtnPURbYSsxNj4+Ml07ejp7aWYoKGZ8MCk+KGd8MCkpe0RbZCtqPj4yXT1nO2JyZWFrIHp9ZD1kK2p8MDtnPURbYSsxMj4+Ml07aWYoKGd8MCk+KGZ8MCkpe0RbZD4+Ml09ZzticmVhayB6fURbZD4+Ml09Zn1pPWkrMXwwO2Y9RFthKzg+PjJdO2lmKChpfDApPChmfDApKXtjb250aW51ZX1icmVha31kPTA7aWYoKGZ8MCk8PTApe2JyZWFrIGt9Zj1uPDwyO2k9ZitjfDA7bj1iK2Z8MDt3aGlsZSgxKXtnPWQ8PDI7Zj1nK2l8MDtnPURbZytuPj4yXStEW2craj4+Ml18MDtEW2Y+PjJdPWc7QTp7aWYoKGd8MCk+RFthKzE2Pj4yXSl7Zz1nLURbYSsyMD4+Ml18MH1lbHNle2lmKChnfDApPj1EW2ErMTI+PjJdKXticmVhayBBfWc9ZytEW2ErMjA+PjJdfDB9RFtmPj4yXT1nfWQ9ZCsxfDA7aWYoKGR8MCk8RFthKzg+PjJdKXtjb250aW51ZX1icmVha319cD0xO209bSsxfDA7aWYoKEZ8MCkhPShtfDApKXtjb250aW51ZX1icmVha319YT1EW2g+PjJdO2lmKGEpe21hKGEpfW1hKHcpO2E9RFtoKzUyPj4yXTtpZihhKXtEW2grNTY+PjJdPWE7bWEoYSl9YT1EW2grNDA+PjJdO2lmKGEpe0RbaCs0ND4+Ml09YTttYShhKX1hPURbaCsyOD4+Ml07aWYoYSl7RFtoKzMyPj4yXT1hO21hKGEpfWE9RFtoKzE2Pj4yXTtpZihhKXtEW2grMjA+PjJdPWE7bWEoYSl9JD1oLSAtNjR8MDtyZXR1cm4gcHwwfXFhKCk7VCgpfWZ1bmN0aW9uIG1iKGEsYixjLGQpe3ZhciBlPTAsZj0wLGc9MCxoPTAsaT0wO2E6e2I6e2lmKCFkKXticmVhayBifWM6e2Q6e3N3aXRjaChEW2ErMjg+PjJdLTF8MCl7Y2FzZSAwOmk9MTtlOntlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk8PTApe2JyZWFrIGV9ZT1EW2E+PjJdO2g9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtmPWI7Yj1oK2J8MDtnPURbZSs0Pj4yXTtlPWctaHwwO2lmKCFFW2ErMzJ8MF0pe2lmKChlfDApPD0oZnwwKSl7YnJlYWsgYX1mPTA7d2hpbGUoMSl7SFsoZjw8MikrZD4+Ml09QltifDBdO2Y9ZisxfDA7ZT1CW2ErMjR8MF07aWYoKGZ8MCk+PSgoKGN8MCk8KGV8MCk/YzplKXwwKSl7YnJlYWsgZX1iPWIrMXwwO2lmKGc+Pj4wPmI+Pj4wKXtjb250aW51ZX1icmVha31icmVhayBhfWlmKChlfDApPD0oZnwwKSl7YnJlYWsgYX1mPTA7d2hpbGUoMSl7SFsoZjw8MikrZD4+Ml09SyhCW2J8MF0pL0soMTI3KTtmPWYrMXwwO2U9QlthKzI0fDBdO2lmKChmfDApPj0oKChjfDApPChlfDApP2M6ZSl8MCkpe2JyZWFrIGV9Yj1iKzF8MDtpZihnPj4+MD5iPj4+MCl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgY31icmVhayBiO2Nhc2UgMTppPTE7Zjp7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPD0wKXticmVhayBmfWU9RFthPj4yXTtoPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Zj1iO2I9aCtifDA7Zz1EW2UrND4+Ml07ZT1nLWh8MDtpZighRVthKzMyfDBdKXtpZigoZXwwKTw9KGZ8MCkpe2JyZWFrIGF9Zj0wO3doaWxlKDEpe0hbKGY8PDIpK2Q+PjJdPUVbYnwwXTtmPWYrMXwwO2U9QlthKzI0fDBdO2lmKChmfDApPj0oKChjfDApPChlfDApP2M6ZSl8MCkpe2JyZWFrIGZ9Yj1iKzF8MDtpZihnPj4+MD5iPj4+MCl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1pZigoZXwwKTw9KGZ8MCkpe2JyZWFrIGF9Zj0wO3doaWxlKDEpe0hbKGY8PDIpK2Q+PjJdPUsoRVtifDBdKS9LKDI1NSk7Zj1mKzF8MDtlPUJbYSsyNHwwXTtpZigoZnwwKT49KCgoY3wwKTwoZXwwKT9jOmUpfDApKXticmVhayBmfWI9YisxfDA7aWYoZz4+PjA+Yj4+PjApe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGF9aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDI6aT0xO2c6e2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKTw9MCl7YnJlYWsgZ31lPURbYT4+Ml07aD1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2Y9YjtiPWgrYnwwO2c9RFtlKzQ+PjJdO2U9Zy1ofDA7aWYoIUVbYSszMnwwXSl7aWYoKGV8MCk8PShmfDApKXticmVhayBhfWY9MDt3aGlsZSgxKXtIWyhmPDwyKStkPj4yXT1DW2I+PjFdO2Y9ZisxfDA7ZT1CW2ErMjR8MF07aWYoKGZ8MCk+PSgoKGN8MCk8KGV8MCk/YzplKXwwKSl7YnJlYWsgZ31iPWIrMnwwO2lmKGc+Pj4wPmI+Pj4wKXtjb250aW51ZX1icmVha31icmVhayBhfWlmKChlfDApPD0oZnwwKSl7YnJlYWsgYX1mPTA7d2hpbGUoMSl7SFsoZjw8MikrZD4+Ml09SyhDW2I+PjFdKS9LKDMyNzY3KTtmPWYrMXwwO2U9QlthKzI0fDBdO2lmKChmfDApPj0oKChjfDApPChlfDApP2M6ZSl8MCkpe2JyZWFrIGd9Yj1iKzJ8MDtpZihnPj4+MD5iPj4+MCl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgY31icmVhayBiO2Nhc2UgMzppPTE7aDp7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPD0wKXticmVhayBofWU9RFthPj4yXTtoPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Zj1iO2I9aCtifDA7Zz1EW2UrND4+Ml07ZT1nLWh8MDtpZighRVthKzMyfDBdKXtpZigoZXwwKTw9KGZ8MCkpe2JyZWFrIGF9Zj0wO3doaWxlKDEpe0hbKGY8PDIpK2Q+PjJdPUZbYj4+MV07Zj1mKzF8MDtlPUJbYSsyNHwwXTtpZigoZnwwKT49KCgoY3wwKTwoZXwwKT9jOmUpfDApKXticmVhayBofWI9YisyfDA7aWYoZz4+PjA+Yj4+PjApe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGF9aWYoKGV8MCk8PShmfDApKXticmVhayBhfWY9MDt3aGlsZSgxKXtIWyhmPDwyKStkPj4yXT1LKEZbYj4+MV0pL0soNjU1MzUpO2Y9ZisxfDA7ZT1CW2ErMjR8MF07aWYoKGZ8MCk+PSgoKGN8MCk8KGV8MCk/YzplKXwwKSl7YnJlYWsgaH1iPWIrMnwwO2lmKGc+Pj4wPmI+Pj4wKXtjb250aW51ZX1icmVha31icmVhayBhfWlmKChjfDApPihlfDApKXticmVhayBjfWJyZWFrIGI7Y2FzZSA0Omk9MTtpOntlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk8PTApe2JyZWFrIGl9ZT1EW2E+PjJdO2g9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtmPWI7Yj1oK2J8MDtnPURbZSs0Pj4yXTtlPWctaHwwO2lmKCFFW2ErMzJ8MF0pe2lmKChlfDApPD0oZnwwKSl7YnJlYWsgYX1mPTA7d2hpbGUoMSl7SFsoZjw8MikrZD4+Ml09RFtiPj4yXTtmPWYrMXwwO2U9QlthKzI0fDBdO2lmKChmfDApPj0oKChjfDApPChlfDApP2M6ZSl8MCkpe2JyZWFrIGl9Yj1iKzR8MDtpZihnPj4+MD5iPj4+MCl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1pZigoZXwwKTw9KGZ8MCkpe2JyZWFrIGF9Zj0wO3doaWxlKDEpe0hbKGY8PDIpK2Q+PjJdPUsoRFtiPj4yXSkqSyg0LjY1NjYxMjg3MzA3NzM5M2UtMTApO2Y9ZisxfDA7ZT1CW2ErMjR8MF07aWYoKGZ8MCk+PSgoKGN8MCk8KGV8MCk/YzplKXwwKSl7YnJlYWsgaX1iPWIrNHwwO2lmKGc+Pj4wPmI+Pj4wKXtjb250aW51ZX1icmVha31icmVhayBhfWlmKChjfDApPihlfDApKXticmVhayBjfWJyZWFrIGI7Y2FzZSA1Omk9MTtqOntlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk8PTApe2JyZWFrIGp9ZT1EW2E+PjJdO2g9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtmPWI7Yj1oK2J8MDtnPURbZSs0Pj4yXTtlPWctaHwwO2lmKCFFW2ErMzJ8MF0pe2lmKChlfDApPD0oZnwwKSl7YnJlYWsgYX1mPTA7d2hpbGUoMSl7SFsoZjw8MikrZD4+Ml09R1tiPj4yXTtmPWYrMXwwO2U9QlthKzI0fDBdO2lmKChmfDApPj0oKChjfDApPChlfDApP2M6ZSl8MCkpe2JyZWFrIGp9Yj1iKzR8MDtpZihnPj4+MD5iPj4+MCl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1pZigoZXwwKTw9KGZ8MCkpe2JyZWFrIGF9Zj0wO3doaWxlKDEpe0hbKGY8PDIpK2Q+PjJdPUsoR1tiPj4yXSkqSygyLjMyODMwNjQzNjUzODY5NjNlLTEwKTtmPWYrMXwwO2U9QlthKzI0fDBdO2lmKChmfDApPj0oKChjfDApPChlfDApP2M6ZSl8MCkpe2JyZWFrIGp9Yj1iKzR8MDtpZihnPj4+MD5iPj4+MCl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgY31icmVhayBiO2Nhc2UgNjppPTE7azp7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPD0wKXticmVhayBrfWU9RFthPj4yXTtoPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Zj1iO2I9aCtifDA7Zz1EW2UrND4+Ml07ZT1nLWh8MDtpZighRVthKzMyfDBdKXtpZigoZXwwKTw9KGZ8MCkpe2JyZWFrIGF9Zj0wO3doaWxlKDEpe0hbKGY8PDIpK2Q+PjJdPStHW2I+PjJdKyArRFtiKzQ+PjJdKjQyOTQ5NjcyOTY7Zj1mKzF8MDtlPUJbYSsyNHwwXTtpZigoZnwwKT49KCgoY3wwKTwoZXwwKT9jOmUpfDApKXticmVhayBrfWI9Yis4fDA7aWYoZz4+PjA+Yj4+PjApe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGF9aWYoKGV8MCk8PShmfDApKXticmVhayBhfWY9MDt3aGlsZSgxKXtIWyhmPDwyKStkPj4yXT1LKCtHW2I+PjJdKyArRFtiKzQ+PjJdKjQyOTQ5NjcyOTYpKksoMS4wODQyMDIxNzI0ODU1MDQ0ZS0xOSk7Zj1mKzF8MDtlPUJbYSsyNHwwXTtpZigoZnwwKT49KCgoY3wwKTwoZXwwKT9jOmUpfDApKXticmVhayBrfWI9Yis4fDA7aWYoZz4+PjA+Yj4+PjApe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGF9aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDc6aT0xO2w6e2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKTw9MCl7YnJlYWsgbH1lPURbYT4+Ml07aD1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2Y9YjtiPWgrYnwwO2c9RFtlKzQ+PjJdO2U9Zy1ofDA7aWYoIUVbYSszMnwwXSl7aWYoKGV8MCk8PShmfDApKXticmVhayBhfWY9MDt3aGlsZSgxKXtIWyhmPDwyKStkPj4yXT0rR1tiPj4yXSsgK0dbYis0Pj4yXSo0Mjk0OTY3Mjk2O2Y9ZisxfDA7ZT1CW2ErMjR8MF07aWYoKGZ8MCk+PSgoKGN8MCk8KGV8MCk/YzplKXwwKSl7YnJlYWsgbH1iPWIrOHwwO2lmKGc+Pj4wPmI+Pj4wKXtjb250aW51ZX1icmVha31icmVhayBhfWlmKChlfDApPD0oZnwwKSl7YnJlYWsgYX1mPTA7d2hpbGUoMSl7SFsoZjw8MikrZD4+Ml09SygrR1tiPj4yXSsgK0dbYis0Pj4yXSo0Mjk0OTY3Mjk2KSpLKDUuNDIxMDEwODYyNDI3NTIyZS0yMCk7Zj1mKzF8MDtlPUJbYSsyNHwwXTtpZigoZnwwKT49KCgoY3wwKTwoZXwwKT9jOmUpfDApKXticmVhayBsfWI9Yis4fDA7aWYoZz4+PjA+Yj4+PjApe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGF9aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDg6aT0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zz1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZytifDA7Zz1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWc+Pj4wKXticmVhayBhfUhbKGY8PDIpK2Q+PjJdPUhbYj4+Ml07Yj1iKzR8MDtmPWYrMXwwO2U9QlthKzI0fDBdO2lmKChmfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPihlfDApKXticmVhayBjfWJyZWFrIGI7Y2FzZSA5Omk9MTtlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2c9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWcrYnwwO2c9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1nPj4+MCl7YnJlYWsgYX1IWyhmPDwyKStkPj4yXT1JW2I+PjNdO2I9Yis4fDA7Zj1mKzF8MDtlPUJbYSsyNHwwXTtpZigoZnwwKTwoKChjfDApPChlfDApP2M6ZSl8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgY31icmVhayBiO2Nhc2UgMTA6YnJlYWsgZDtkZWZhdWx0OmJyZWFrIGJ9fWk9MTtlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7Zz1EW2E+PjJdO2U9RFtnPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWUrYnwwO3doaWxlKDEpe2lmKEdbZys0Pj4yXTw9Yj4+PjApe2JyZWFrIGF9SFsoZjw8MikrZD4+Ml09RVtifDBdP0soMSk6SygwKTtiPWIrMXwwO2Y9ZisxfDA7ZT1CW2ErMjR8MF07aWYoKGZ8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk8PShlfDApKXticmVhayBifX1wYSgoZTw8MikrZHwwLDAsYy1lPDwyKX1yZXR1cm4gaX1yZXR1cm4gMH1mdW5jdGlvbiBWaChhLGIsYyxkLGUsZil7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7ZT1lfDA7Zj1mfDA7dmFyIGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTAsbz0wLHA9MCxxPTAscj0wLHM9MCx0PTAsdT0wLHY9MCx3PTAseD0wLHk9MCx6PTAsQT0wLEI9MCxDPTAsRT0wLEY9MCxHPTAsSD0wLEk9MDtoPSQrLTY0fDA7JD1oO0RbYSs4Pj4yXT1lO2Y9YSszMnwwO2c9RFtmPj4yXTtkPURbYSszNj4+Ml0tZz4+MjthOntiOntpZihkPj4+MDxlPj4+MCl7c2EoZixlLWR8MCk7RFtoKzU2Pj4yXT0wO0RbaCs2MD4+Ml09MDtEW2grNDg+PjJdPTA7RFtoKzUyPj4yXT0wO0RbaCs0MD4+Ml09MDtEW2grNDQ+PjJdPTA7RFtoKzMyPj4yXT0wO0RbaCszNj4+Ml09MDtEW2grMjQ+PjJdPTA7RFtoKzI4Pj4yXT0wO0RbaCsxNj4+Ml09MDtEW2grMjA+PjJdPTA7RFtoPj4yXT0wO2JyZWFrIGJ9aWYoZD4+PjA+ZT4+PjApe0RbYSszNj4+Ml09ZysoZTw8Mil9RFtoKzU2Pj4yXT0wO0RbaCs2MD4+Ml09MDtEW2grNDg+PjJdPTA7RFtoKzUyPj4yXT0wO0RbaCs0MD4+Ml09MDtEW2grNDQ+PjJdPTA7RFtoKzMyPj4yXT0wO0RbaCszNj4+Ml09MDtEW2grMjQ+PjJdPTA7RFtoKzI4Pj4yXT0wO0RbaCsxNj4+Ml09MDtEW2grMjA+PjJdPTA7RFtoPj4yXT0wO2Q9MDtpZighZSl7YnJlYWsgYX19eGEoaCsxNnwwLGUsaCk7aT1EW2grMjg+PjJdO2Q9RFtoKzMyPj4yXX1EW2g+PjJdPTA7ZD1kLWk+PjI7Yzp7aWYoZD4+PjA+PWU+Pj4wKXtpZihkPj4+MDw9ZT4+PjApe2JyZWFrIGN9RFtoKzMyPj4yXT0oZTw8MikraTticmVhayBjfXhhKGgrMTZ8MTIsZS1kfDAsaCl9RFtoPj4yXT0wO2Y9RFtoKzQwPj4yXTtkPURbaCs0ND4+Ml0tZj4+MjtkOntpZihkPj4+MD49ZT4+PjApe2lmKGQ+Pj4wPD1lPj4+MCl7YnJlYWsgZH1EW2grNDQ+PjJdPWYrKGU8PDIpO2JyZWFrIGR9eGEoaCs0MHwwLGUtZHwwLGgpfURbaD4+Ml09MDtmPURbaCs1Mj4+Ml07ZD1EW2grNTY+PjJdLWY+PjI7ZTp7aWYoZD4+PjA+PWU+Pj4wKXtpZihkPj4+MDw9ZT4+PjApe2JyZWFrIGV9RFtoKzU2Pj4yXT1mKyhlPDwyKTticmVhayBlfXhhKGgrNTJ8MCxlLWR8MCxoKX1pPTA7Zjp7aWYoRFthKzg+PjJdPD0wKXticmVhayBmfWs9RFthKzMyPj4yXTtnPURbaCsxNj4+Ml07d2hpbGUoMSl7ZD1pPDwyO2Y9RFtkK2c+PjJdO249RFthKzE2Pj4yXTtnOntpZigoZnwwKT4obnwwKSl7RFtkK2s+PjJdPW47YnJlYWsgZ31kPWQra3wwO249RFthKzEyPj4yXTtpZigobnwwKT4oZnwwKSl7RFtkPj4yXT1uO2JyZWFrIGd9RFtkPj4yXT1mfWk9aSsxfDA7ZD1EW2ErOD4+Ml07aWYoKGl8MCk8KGR8MCkpe2NvbnRpbnVlfWJyZWFrfWlmKChkfDApPD0wKXticmVhayBmfWQ9MDt3aGlsZSgxKXtnPWQ8PDI7Zj1nK2N8MDtnPURbYitnPj4yXStEW2craz4+Ml18MDtEW2Y+PjJdPWc7aDp7aWYoKGd8MCk+RFthKzE2Pj4yXSl7aj1nLURbYSsyMD4+Ml18MH1lbHNle2lmKChnfDApPj1EW2ErMTI+PjJdKXticmVhayBofWo9ZytEW2ErMjA+PjJdfDB9RFtmPj4yXT1qfWQ9ZCsxfDA7aWYoKGR8MCk8RFthKzg+PjJdKXtjb250aW51ZX1icmVha319RT1EW2ErNTI+PjJdO3k9RFthKzQ4Pj4yXTt2PW5hKDE2KTtkPXY7RFtkPj4yXT0wO0RbZCs0Pj4yXT0wO0RbZCs4Pj4yXT0wO0RbZCsxMj4+Ml09MDtEW2grOD4+Ml09MDtEW2g+PjJdPTA7RFtoKzQ+PjJdPTA7aTp7aWYoZSl7aWYoZT4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGl9ZD1lPDwyO3U9bmEoZCk7RFtoPj4yXT11O0RbaCs4Pj4yXT1kK3U7cGEodSwwLGQpfXI9MTtkPURbYSs1Nj4+Ml07ej1EW2Q+PjJdO2Q9RFtkKzQ+PjJdLXp8MDtqOntpZigoZHwwKTw1KXticmVhayBqfWQ9ZD4+MjtGPShkfDApPjI/ZDoyO0c9ZD4+PjA+MT9kOjE7QT1lJi0yO0I9ZSYxO0g9ZSYtNDtDPWUmMzt3PWUtMXwwO0k9ZTw8MjtuPTE7d2hpbGUoMSl7azp7bDp7bTp7bjp7aWYoKG58MCkhPShHfDApKXtmPURbKG48PDIpK3o+PjJdO2Q9KGY+Pj4wKSUzfDA7bzp7cDp7aWYoKGZ8MCk9PS0xKXticmVhayBwfWs9MDtnPWYrMnwwO3g9KGR8MCkhPTB8KGd8MCkhPS0xO3M9RFt5KzEyPj4yXTtxPXMrKChkP2YtMXwwOmcpPDwyKXwwO289MTtkPWY7cTp7d2hpbGUoMSl7Zz1EW3MrKGQ8PDIpPj4yXTtyOntpZigoZ3wwKT09LTEpe2JyZWFrIHJ9bD0tMTtyPURbRT4+Ml07bT1EW3k+PjJdO2k9cisoRFttKyhnPDwyKT4+Ml08PDIpfDA7cD1nKzF8MDtwPShwPj4+MCklM3wwP3A6Zy0yfDA7aWYoKHB8MCkhPS0xKXtsPURbbSsocDw8Mik+PjJdfXA9RFtpPj4yXTtzOnt0OntpZigoZz4+PjApJTN8MCl7aT1nLTF8MDticmVhayB0fWk9ZysyfDA7aj0tMTtpZigoaXwwKT09LTEpe2JyZWFrIHN9fWo9RFttKyhpPDwyKT4+Ml19aWYoKG58MCk8PShwfDApKXticmVhayByfWk9RFtyKyhsPDwyKT4+Ml07aWYoKGl8MCk+PShufDApKXticmVhayByfWw9RFtyKyhqPDwyKT4+Ml07aWYoKGx8MCk+PShufDApKXticmVhayByfWc9RFsoaCsxNnwwKStKKGssMTIpPj4yXTt1OntpZighZSl7YnJlYWsgdX1sPUooZSxsKTttPUooZSxpKTtyPUooZSxwKTtpPTA7aj0wO2lmKHcpe3doaWxlKDEpe0RbZysoaTw8Mik+PjJdPShEWyhpK2w8PDIpK2M+PjJdK0RbKGkrbTw8MikrYz4+Ml18MCktRFsoaStyPDwyKStjPj4yXTtwPWl8MTtEW2crKHA8PDIpPj4yXT0oRFsobCtwPDwyKStjPj4yXStEWyhtK3A8PDIpK2M+PjJdfDApLURbKHArcjw8MikrYz4+Ml07aT1pKzJ8MDtqPWorMnwwO2lmKChBfDApIT0oanwwKSl7Y29udGludWV9YnJlYWt9fWlmKCFCKXticmVhayB1fURbZysoaTw8Mik+PjJdPShEWyhpK2w8PDIpK2M+PjJdK0RbKGkrbTw8MikrYz4+Ml18MCktRFsoaStyPDwyKStjPj4yXX1nPTQ7az1rKzF8MDtpZigoa3wwKT09NCl7YnJlYWsgcX19djp7aWYobyYxKXtpPWQrMXwwO2Q9KGk+Pj4wKSUzfDA/aTpkLTJ8MDtqPS0xO2lmKChkfDApPT0tMSl7YnJlYWsgdn1kPURbcysoZDw8Mik+PjJdO2o9LTE7aWYoKGR8MCk9PS0xKXticmVhayB2fWc9ZCsxfDA7aj0oZz4+PjApJTN8MD9nOmQtMnwwO2JyZWFrIHZ9dzp7aWYoKGQ+Pj4wKSUzfDApe2k9ZC0xfDA7YnJlYWsgd31pPWQrMnwwO2o9LTE7aWYoKGl8MCk9PS0xKXticmVhayB2fX1kPURbcysoaTw8Mik+PjJdO2o9LTE7aWYoKGR8MCk9PS0xKXticmVhayB2fWo9ZC0xfDA7aWYoKGQ+Pj4wKSUzfDApe2JyZWFrIHZ9aj1kKzJ8MH1kPWo7eDp7aWYoKGZ8MCk9PShkfDApKXticmVhayB4fWlmKCEoKG9eMSkmMXwoZHwwKSE9LTEpKXtpZigheCl7YnJlYWsgeH1kPURbcT4+Ml07aWYoKGR8MCk9PS0xKXticmVhayB4fW89MDtkPShkPj4+MCklM3wwP2QtMXwwOmQrMnwwfWlmKChkfDApIT0tMSl7Y29udGludWV9fWJyZWFrfWc9aztpZigoZ3wwKTw9MCl7YnJlYWsgcH19aWYoZSl7cGEoRFtoPj4yXSwwLEkpfWQ9Zy0xfDA7cD0oZDw8MikrdnwwO2Q9SihkLDEyKSthfDA7aj1kO3g9RFtkLSAtNjQ+PjJdO3I9MDtvPTA7ZD0wO3doaWxlKDEpe2Y9RFtwPj4yXTtEW3A+PjJdPWYrMTtpZihmPj4+MD49eD4+PjApe2JyZWFrIGp9eTp7aWYoRFtEW2orNjA+PjJdKyhmPj4+MyY1MzY4NzA5MDgpPj4yXT4+PmYmMSl7YnJlYWsgeX1kPWQrMXwwO2lmKCFlKXticmVhayB5fWs9RFtoPj4yXTtzPURbKGgrMTZ8MCkrSihvLDEyKT4+Ml07bD0wO2k9MDtmPTA7aWYodz4+PjA+PTMpe3doaWxlKDEpe209aTw8MjtxPW0ra3wwO0RbcT4+Ml09RFtxPj4yXStEW20rcz4+Ml07cT1tfDQ7dD1xK2t8MDtEW3Q+PjJdPURbdD4+Ml0rRFtzK3E+PjJdO3E9bXw4O3Q9cStrfDA7RFt0Pj4yXT1EW3Q+PjJdK0RbcytxPj4yXTttPW18MTI7cT1tK2t8MDtEW3E+PjJdPURbcT4+Ml0rRFttK3M+PjJdO2k9aSs0fDA7Zj1mKzR8MDtpZigoSHwwKSE9KGZ8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZighQyl7YnJlYWsgeX13aGlsZSgxKXtmPWk8PDI7bT1mK2t8MDtEW20+PjJdPURbbT4+Ml0rRFtmK3M+PjJdO2k9aSsxfDA7bD1sKzF8MDtpZigoQ3wwKSE9KGx8MCkpe2NvbnRpbnVlfWJyZWFrfX1vPW8rMXwwO2lmKChvfDApIT0oZ3wwKSl7Y29udGludWV9YnJlYWt9bz1KKGUsbik7Zj1vO2lmKCFkKXticmVhayBvfWlmKCFlKXticmVhayBsfWY9RFtoPj4yXTtpPTA7Zz0wO2lmKHcpe2JyZWFrIG59YnJlYWsgbX1mPUooZSxuKX1pZihEW2ErOD4+Ml08PTApe2JyZWFrIGt9bz0oSihuLTF8MCxlKTw8MikrY3wwO2s9RFthKzMyPj4yXTtpPTA7d2hpbGUoMSl7ZD1pPDwyO2c9RFtkK28+PjJdO2w9RFthKzE2Pj4yXTt6OntpZigoZ3wwKT4obHwwKSl7RFtkK2s+PjJdPWw7YnJlYWsgen1kPWQra3wwO2w9RFthKzEyPj4yXTtpZigobHwwKT4oZ3wwKSl7RFtkPj4yXT1sO2JyZWFrIHp9RFtkPj4yXT1nfWk9aSsxfDA7Zz1EW2ErOD4+Ml07aWYoKGl8MCk8KGd8MCkpe2NvbnRpbnVlfWJyZWFrfWQ9MDtpZigoZ3wwKTw9MCl7YnJlYWsga31mPWY8PDI7aT1mK2N8MDtvPWIrZnwwO3doaWxlKDEpe2c9ZDw8MjtmPWcraXwwO2c9RFtnK28+PjJdK0RbZytrPj4yXXwwO0RbZj4+Ml09ZztBOntpZigoZ3wwKT5EW2ErMTY+PjJdKXtqPWctRFthKzIwPj4yXXwwfWVsc2V7aWYoKGd8MCk+PURbYSsxMj4+Ml0pe2JyZWFrIEF9aj1nK0RbYSsyMD4+Ml18MH1EW2Y+PjJdPWp9ZD1kKzF8MDtpZigoZHwwKTxEW2ErOD4+Ml0pe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGt9dWEoKTtUKCl9d2hpbGUoMSl7az1pPDwyO2w9aytmfDA7RFtsPj4yXT1EW2w+PjJdLyhkfDApO2s9Zisoa3w0KXwwO0Rbaz4+Ml09RFtrPj4yXS8oZHwwKTtpPWkrMnwwO2c9ZysyfDA7aWYoKEF8MCkhPShnfDApKXtjb250aW51ZX1icmVha319aWYoIUIpe2JyZWFrIGx9Zj1mKyhpPDwyKXwwO0RbZj4+Ml09RFtmPj4yXS8oZHwwKX1pZihEW2ErOD4+Ml08PTApe2JyZWFrIGt9az1EW2ErMzI+PjJdO2k9MDt3aGlsZSgxKXtkPWk8PDI7Zj1EW2QrdT4+Ml07Zz1EW2ErMTY+PjJdO0I6e2lmKChmfDApPihnfDApKXtEW2Qraz4+Ml09ZzticmVhayBCfWQ9ZCtrfDA7Zz1EW2ErMTI+PjJdO2lmKChnfDApPihmfDApKXtEW2Q+PjJdPWc7YnJlYWsgQn1EW2Q+PjJdPWZ9aT1pKzF8MDtmPURbYSs4Pj4yXTtpZigoaXwwKTwoZnwwKSl7Y29udGludWV9YnJlYWt9ZD0wO2lmKChmfDApPD0wKXticmVhayBrfWY9bzw8MjtpPWYrY3wwO289YitmfDA7d2hpbGUoMSl7Zz1kPDwyO2Y9ZytpfDA7Zz1EW2crbz4+Ml0rRFtnK2s+PjJdfDA7RFtmPj4yXT1nO0M6e2lmKChnfDApPkRbYSsxNj4+Ml0pe2o9Zy1EW2ErMjA+PjJdfDB9ZWxzZXtpZigoZ3wwKT49RFthKzEyPj4yXSl7YnJlYWsgQ31qPWcrRFthKzIwPj4yXXwwfURbZj4+Ml09an1kPWQrMXwwO2lmKChkfDApPERbYSs4Pj4yXSl7Y29udGludWV9YnJlYWt9fXI9MTtuPW4rMXwwO2lmKChGfDApIT0obnwwKSl7Y29udGludWV9YnJlYWt9fWE9RFtoPj4yXTtpZihhKXttYShhKX1tYSh2KTthPURbaCs1Mj4+Ml07aWYoYSl7RFtoKzU2Pj4yXT1hO21hKGEpfWE9RFtoKzQwPj4yXTtpZihhKXtEW2grNDQ+PjJdPWE7bWEoYSl9YT1EW2grMjg+PjJdO2lmKGEpe0RbaCszMj4+Ml09YTttYShhKX1hPURbaCsxNj4+Ml07aWYoYSl7RFtoKzIwPj4yXT1hO21hKGEpfSQ9aC0gLTY0fDA7cmV0dXJuIHJ8MH1xYSgpO1QoKX1mdW5jdGlvbiBrZihhLGIpe2E9YXwwO2I9YnwwO3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MCxvPTAscD0wLHE9MCxyPTAscz0wLHQ9MCx1PTAsdj0wO209JC0xNnwwOyQ9bTtEW20rMTI+PjJdPWI7Yj1uYSgzMik7RFttPj4yXT1iO0RbbSs0Pj4yXT0yNDtEW20rOD4+Ml09LTIxNDc0ODM2MTY7QltiKzI0fDBdPTA7ZD1FWzEzMDhdfEVbMTMwOV08PDh8KEVbMTMxMF08PDE2fEVbMTMxMV08PDI0KTtjPUVbMTMwNF18RVsxMzA1XTw8OHwoRVsxMzA2XTw8MTZ8RVsxMzA3XTw8MjQpO0JbYisxNnwwXT1jO0JbYisxN3wwXT1jPj4+ODtCW2IrMTh8MF09Yz4+PjE2O0JbYisxOXwwXT1jPj4+MjQ7QltiKzIwfDBdPWQ7QltiKzIxfDBdPWQ+Pj44O0JbYisyMnwwXT1kPj4+MTY7QltiKzIzfDBdPWQ+Pj4yNDtkPUVbMTMwMF18RVsxMzAxXTw8OHwoRVsxMzAyXTw8MTZ8RVsxMzAzXTw8MjQpO2M9RVsxMjk2XXxFWzEyOTddPDw4fChFWzEyOThdPDwxNnxFWzEyOTldPDwyNCk7QltiKzh8MF09YztCW2IrOXwwXT1jPj4+ODtCW2IrMTB8MF09Yz4+PjE2O0JbYisxMXwwXT1jPj4+MjQ7QltiKzEyfDBdPWQ7QltiKzEzfDBdPWQ+Pj44O0JbYisxNHwwXT1kPj4+MTY7QltiKzE1fDBdPWQ+Pj4yNDtkPUVbMTI5Ml18RVsxMjkzXTw8OHwoRVsxMjk0XTw8MTZ8RVsxMjk1XTw8MjQpO2M9RVsxMjg4XXxFWzEyODldPDw4fChFWzEyOTBdPDwxNnxFWzEyOTFdPDwyNCk7QltifDBdPWM7QltiKzF8MF09Yz4+Pjg7QltiKzJ8MF09Yz4+PjE2O0JbYiszfDBdPWM+Pj4yNDtCW2IrNHwwXT1kO0JbYis1fDBdPWQ+Pj44O0JbYis2fDBdPWQ+Pj4xNjtCW2IrN3wwXT1kPj4+MjQ7aD0kLTQ4fDA7JD1oO2s9YTtkPWErMTZ8MDthPURbZD4+Ml07YTp7Yjp7aWYoIWEpe2JyZWFrIGJ9Yz1EW20rMTI+PjJdO2I9ZDt3aGlsZSgxKXtmPShjfDApPkRbYSsxNj4+Ml07Yj1mP2I6YTthPURbKGY8PDIpK2E+PjJdO2lmKGEpe2NvbnRpbnVlfWJyZWFrfWlmKChifDApPT0oZHwwKSl7YnJlYWsgYn1pZigoY3wwKT49RFtiKzE2Pj4yXSl7YnJlYWsgYX19dD1oKzI0fDA7YT10O0RbYSs0Pj4yXT0wO0RbYSs4Pj4yXT0wO3U9YSs0fDA7RFthPj4yXT11O2E9RFttKzEyPj4yXTtjPWgrMTZ8MDtEW2M+PjJdPTA7RFtjKzQ+PjJdPTA7RFtoKzg+PjJdPWE7RFtoKzEyPj4yXT1jO2I9RFt0Pj4yXTtpZigodXwwKSE9KGJ8MCkpe2w9aCs4fDQ7d2hpbGUoMSl7Zj1iO2c9YisxNnwwO3E9JC0xNnwwOyQ9cTtpPXErMTJ8MDthPXErOHwwO2M6e2Q6e2U6e2Y6e2c6e2g6e3I9bCs0fDA7aTp7aWYoKHJ8MCk9PShjfDApKXticmVhayBpfWI9RVtjKzI3fDBdO2U9Yjw8MjQ+PjI0PDA7bj1FW2crMTF8MF07dj1uPDwyND4+MjQ7aj0odnwwKTwwO2I9ZT9EW2MrMjA+PjJdOmI7bj1qP0RbZys0Pj4yXTpuO289Yj4+PjA8bj4+PjA7cD1vP2I6bjtpZihwKXtqPWo/RFtnPj4yXTpnO3M9YysxNnwwO2U9ZT9EW3M+PjJdOnM7cz12YShqLGUscCk7aWYoIXMpe2lmKGI+Pj4wPm4+Pj4wKXticmVhayBpfWJyZWFrIGh9aWYoKHN8MCk+PTApe2JyZWFrIGh9YnJlYWsgaX1pZihiPj4+MDw9bj4+PjApe2JyZWFrIGd9fWU9RFtjPj4yXTtqOnthPWM7azp7aWYoKGF8MCk9PURbbD4+Ml0pe2JyZWFrIGt9bDp7aWYoZSl7Yj1lO3doaWxlKDEpe2E9YjtiPURbYis0Pj4yXTtpZihiKXtjb250aW51ZX1icmVha31icmVhayBsfWE9Yys4fDA7aWYoKGN8MCk9PURbRFtjKzg+PjJdPj4yXSl7d2hpbGUoMSl7Yj1EW2E+PjJdO2E9Yis4fDA7aWYoKGJ8MCk9PURbRFtiKzg+PjJdPj4yXSl7Y29udGludWV9YnJlYWt9fWE9RFthPj4yXX1qPUVbZysxMXwwXTtiPWo8PDI0Pj4yNDwwO289RVthKzI3fDBdO249bzw8MjQ+PjI0PDA7bTp7aj1iP0RbZys0Pj4yXTpqO289bj9EW2ErMjA+PjJdOm87cD1qPj4+MDxvPj4+MD9qOm87aWYocCl7cj1hKzE2fDA7Yj12YShuP0Rbcj4+Ml06cixiP0RbZz4+Ml06ZyxwKTtpZihiKXticmVhayBtfX1pZihqPj4+MD5vPj4+MCl7YnJlYWsga31icmVhayBqfWlmKChifDApPj0wKXticmVhayBqfX1pZighZSl7RFtpPj4yXT1jO2E9YzticmVhayBjfURbaT4+Ml09YTthPWErNHwwO2JyZWFrIGN9YT1JZChsLGksZyk7YnJlYWsgY31iPXZhKGUsaixwKTtpZihiKXticmVhayBmfX1pZihvKXticmVhayBlfWJyZWFrIGR9aWYoKGJ8MCk+PTApe2JyZWFrIGR9fWU9RFtjKzQ+PjJdO246e2lmKGUpe2I9ZTt3aGlsZSgxKXthPWI7Yj1EW2I+PjJdO2lmKGIpe2NvbnRpbnVlfWJyZWFrfWJyZWFrIG59YT1EW2MrOD4+Ml07aWYoKGN8MCk9PURbYT4+Ml0pe2JyZWFrIG59Yj1jKzh8MDt3aGlsZSgxKXtqPURbYj4+Ml07Yj1qKzh8MDthPURbais4Pj4yXTtpZigoanwwKSE9RFthPj4yXSl7Y29udGludWV9YnJlYWt9fW86e3A6e2lmKChhfDApPT0ocnwwKSl7YnJlYWsgcH1qPUVbYSsyN3wwXTtiPWo8PDI0Pj4yNDwwO3E6e2o9Yj9EW2ErMjA+PjJdOmo7bz1qPj4+MDxuPj4+MD9qOm47aWYobyl7cD1hKzE2fDA7Yj12YSgodnwwKTwwP0RbZz4+Ml06ZyxiP0RbcD4+Ml06cCxvKTtpZihiKXticmVhayBxfX1pZihqPj4+MD5uPj4+MCl7YnJlYWsgcH1icmVhayBvfWlmKChifDApPj0wKXticmVhayBvfX1pZighZSl7RFtpPj4yXT1jO2E9Yys0fDA7YnJlYWsgY31EW2k+PjJdPWE7YnJlYWsgY31hPUlkKGwsaSxnKTticmVhayBjfURbaT4+Ml09YztEW2E+PjJdPWN9Yj1hO2E9RFtiPj4yXTtpZihhKXtiPTB9ZWxzZXthPW5hKDQwKTtlPWErMTZ8MDtyOntpZihCW2crMTF8MF0+PTApe2k9RFtnKzQ+PjJdO0RbZT4+Ml09RFtnPj4yXTtEW2UrND4+Ml09aTtEW2UrOD4+Ml09RFtnKzg+PjJdO2JyZWFrIHJ9dGEoZSxEW2c+PjJdLERbZys0Pj4yXSl9ZT1hKzI4fDA7czp7aWYoQltnKzIzfDBdPj0wKXtpPURbZysxNj4+Ml07RFtlPj4yXT1EW2crMTI+PjJdO0RbZSs0Pj4yXT1pO0RbZSs4Pj4yXT1EW2crMjA+PjJdO2JyZWFrIHN9dGEoZSxEW2crMTI+PjJdLERbZysxNj4+Ml0pfURbYSs4Pj4yXT1EW3ErMTI+PjJdO0RbYT4+Ml09MDtEW2ErND4+Ml09MDtEW2I+PjJdPWE7ZT1EW0RbbD4+Ml0+PjJdO2lmKGUpe0RbbD4+Ml09ZTtiPURbYj4+Ml19ZWxzZXtiPWF9cGIoRFtsKzQ+PjJdLGIpO0RbbCs4Pj4yXT1EW2wrOD4+Ml0rMTtiPTF9QltoKzQ0fDBdPWI7RFtoKzQwPj4yXT1hOyQ9cSsxNnwwO2E9RFtmKzQ+PjJdO3Q6e2lmKCFhKXtiPURbZis4Pj4yXTtpZigoZnwwKT09RFtiPj4yXSl7YnJlYWsgdH1hPWYrOHwwO3doaWxlKDEpe2Y9RFthPj4yXTthPWYrOHwwO2I9RFtmKzg+PjJdO2lmKChmfDApIT1EW2I+PjJdKXtjb250aW51ZX1icmVha31icmVhayB0fXdoaWxlKDEpe2I9YTthPURbYj4+Ml07aWYoYSl7Y29udGludWV9YnJlYWt9fWlmKChifDApIT0odXwwKSl7Y29udGludWV9YnJlYWt9fWE9RFtkPj4yXTt1OntpZihhKXtkPWsrMTZ8MDtmPURbaCs4Pj4yXTt3aGlsZSgxKXtiPURbYSsxNj4+Ml07djp7aWYoKGJ8MCk+KGZ8MCkpe2I9RFthPj4yXTtpZihiKXticmVhayB2fWQ9YTticmVhayB1fWlmKChifDApPj0oZnwwKSl7YnJlYWsgdX1kPWErNHwwO2I9RFthKzQ+PjJdO2lmKCFiKXticmVhayB1fWE9ZH1kPWE7YT1iO2NvbnRpbnVlfX1hPWR9Yj1EW2Q+PjJdO2lmKCFiKXtiPW5hKDMyKTtEW2IrMTY+PjJdPURbaCs4Pj4yXTtEW2IrMjA+PjJdPURbaCsxMj4+Ml07Zj1iKzI0fDA7ZT1EW2grMTY+PjJdO0RbZj4+Ml09ZTtnPURbaCsyMD4+Ml07RFtiKzI4Pj4yXT1nO3c6e2lmKCFnKXtEW2IrMjA+PjJdPWY7YnJlYWsgd31EW2UrOD4+Ml09ZjtEW2grMTY+PjJdPTA7RFtoKzIwPj4yXT0wO0RbaCsxMj4+Ml09Y31EW2IrOD4+Ml09YTtEW2I+PjJdPTA7RFtiKzQ+PjJdPTA7RFtkPj4yXT1iO2E9RFtEW2srMTI+PjJdPj4yXTtpZihhKXtEW2srMTI+PjJdPWE7YT1EW2Q+PjJdfWVsc2V7YT1ifXBiKERbaysxNj4+Ml0sYSk7RFtrKzIwPj4yXT1EW2srMjA+PjJdKzF9aGIoaCs4fDQsRFtoKzE2Pj4yXSk7aGIodCxEW3QrND4+Ml0pfSQ9aCs0OHwwO2s9JC00OHwwOyQ9aztjPSQtMzJ8MDskPWM7Zz1jKzMyfDA7ZD1jKzIxfDA7YT1kO2Y9Zy1hfDA7eDp7aWYoISgoZnwwKTw9OSYoZnwwKTwoMS0oR1syNzQ0XT4xKXwwKSkpe0JbYXwwXT00OTtEW2MrOD4+Ml09YSsxO2E9MDticmVhayB4fURbYys4Pj4yXT1nO2E9NjF9RFtjKzEyPj4yXT1hO2g9JC0xNnwwOyQ9aDthPWsrOHwwO2U9JC0xNnwwOyQ9ZTt5OntsPURbYys4Pj4yXTtmPWwtZHwwO2lmKGY+Pj4wPD00Mjk0OTY3Mjc5KXt6OntpZihmPj4+MDw9MTApe0JbYSsxMXwwXT1mO2M9YTticmVhayB6fWlmKGY+Pj4wPj0xMSl7aT1mKzE2Ji0xNjtjPWktMXwwO2M9KGN8MCk9PTExP2k6Y31lbHNle2M9MTB9aT1jKzF8MDtjPW5hKGkpO0RbYT4+Ml09YztEW2ErOD4+Ml09aXwtMjE0NzQ4MzY0ODtEW2ErND4+Ml09Zn13aGlsZSgxKXtpZigoZHwwKSE9KGx8MCkpe0JbY3wwXT1FW2R8MF07Yz1jKzF8MDtkPWQrMXwwO2NvbnRpbnVlfWJyZWFrfUJbZSsxNXwwXT0wO0JbY3wwXT1FW2UrMTV8MF07JD1lKzE2fDA7YnJlYWsgeX1BYSgpO1QoKX0kPWgrMTZ8MDskPWc7RFtrKzMyPj4yXT1tO2M9YisyMHwwO2E9RFtjKzQ+PjJdO0E6e0I6e2lmKGEpe2I9RVttKzExfDBdO2Q9Yjw8MjQ+PjI0PDA7ZT1kP0RbbT4+Ml06bTtmPWQ/RFttKzQ+PjJdOmI7Yj1jKzR8MDt3aGlsZSgxKXtkPUVbYSsyN3wwXTtnPWQ8PDI0Pj4yNDwwO2Q9Zz9EW2ErMjA+PjJdOmQ7aD1kPj4+MDxmPj4+MDtDOntEOntFOntGOntHOntsPWg/ZDpmO0g6e2lmKGwpe2k9YSsxNnwwO2c9Zz9EW2k+PjJdOmk7aT12YShlLGcsbCk7aWYoIWkpe2lmKGQ+Pj4wPmY+Pj4wKXticmVhayBIfWJyZWFrIEd9aWYoKGl8MCk+PTApe2JyZWFrIEd9YnJlYWsgSH1pZihkPj4+MDw9Zj4+PjApe2JyZWFrIEZ9fWQ9RFthPj4yXTtpZihkKXticmVhayBDfWJyZWFrIEJ9ZD12YShnLGUsbCk7aWYoZCl7YnJlYWsgRX19aWYoaCl7YnJlYWsgRH1icmVhayBBfWlmKChkfDApPj0wKXticmVhayBBfX1iPWErNHwwO2Q9RFthKzQ+PjJdO2lmKCFkKXticmVhayBBfWE9Yn1iPWE7YT1kO2NvbnRpbnVlfX1hPWMrNHwwfWI9YX1kPURbYj4+Ml07aWYoZCl7YT0wfWVsc2V7ZD1uYSg0MCk7ZT1kKzE2fDA7Zj1EW2srMzI+PjJdO0k6e2lmKEJbZisxMXwwXT49MCl7Zz1EW2YrND4+Ml07RFtlPj4yXT1EW2Y+PjJdO0RbZSs0Pj4yXT1nO0RbZSs4Pj4yXT1EW2YrOD4+Ml07YnJlYWsgSX10YShlLERbZj4+Ml0sRFtmKzQ+PjJdKX1EW2QrOD4+Ml09YTtEW2Q+PjJdPTA7RFtkKzQ+PjJdPTA7RFtkKzM2Pj4yXT0wO0RbZCsyOD4+Ml09MDtEW2QrMzI+PjJdPTA7RFtiPj4yXT1kO2E9RFtEW2M+PjJdPj4yXTtpZihhKXtEW2M+PjJdPWE7YT1EW2I+PjJdfWVsc2V7YT1kfXBiKERbYys0Pj4yXSxhKTtEW2MrOD4+Ml09RFtjKzg+PjJdKzE7YT0xfUJbays0NHwwXT1hO0Rbays0MD4+Ml09ZDthPURbays0MD4+Ml07aWYoQlthKzM5fDBdPDApe21hKERbYSsyOD4+Ml0pfWI9RFtrKzEyPj4yXTtEW2ErMjg+PjJdPURbays4Pj4yXTtEW2ErMzI+PjJdPWI7RFthKzM2Pj4yXT1EW2srMTY+PjJdOyQ9ays0OHwwO2lmKEJbbSsxMXwwXTwwKXttYShEW20+PjJdKX0kPW0rMTZ8MH1mdW5jdGlvbiBnZShhLGIpe2E9YXwwO2I9YnwwO3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MCxvPTA7aT1hO2E6e2I6e2M6e2Q6e2U6e2Y6e2c6e2g6e2s9RFthKzg+PjJdO3N3aXRjaChEW2srMjg+PjJdLTF8MCl7Y2FzZSA0OmJyZWFrIGM7Y2FzZSA1OmJyZWFrIGQ7Y2FzZSAyOmJyZWFrIGU7Y2FzZSAzOmJyZWFrIGY7Y2FzZSAwOmJyZWFrIGc7Y2FzZSAxOmJyZWFrIGg7ZGVmYXVsdDpicmVhayBhfX1jPUJbaysyNHwwXTtkPW5hKChjfDApPj0wP2M6LTEpO2E9RFtpKzE2Pj4yXTtpZihEW2ErODA+PjJdKXtoPURbRFthPj4yXT4+Ml0rRFthKzQ4Pj4yXXwwfWVsc2V7aD0wfWlmKCFiKXticmVhayBifWlmKChjfDApPjApe2s9YyYtNDttPWMmMztlPWMtMT4+PjA8Mzt3aGlsZSgxKXthPTA7Zz0wO2lmKCFlKXt3aGlsZSgxKXtqPWgrKGY8PDIpfDA7QlthK2R8MF09RFtqPj4yXTtCWyhhfDEpK2R8MF09RFtqKzQ+PjJdO0JbKGF8MikrZHwwXT1EW2orOD4+Ml07QlsoYXwzKStkfDBdPURbaisxMj4+Ml07YT1hKzR8MDtmPWYrNHwwO2c9Zys0fDA7aWYoKGt8MCkhPShnfDApKXtjb250aW51ZX1icmVha319Zz0wO2lmKG0pe3doaWxlKDEpe0JbYStkfDBdPURbaCsoZjw8Mik+PjJdO2E9YSsxfDA7Zj1mKzF8MDtnPWcrMXwwO2lmKChtfDApIT0oZ3wwKSl7Y29udGludWV9YnJlYWt9fW9hKERbRFtEW2krOD4+Ml0rNjQ+PjJdPj4yXStufDAsZCxjKTtuPWMrbnwwO2w9bCsxfDA7aWYoKGx8MCkhPShifDApKXtjb250aW51ZX1icmVha31icmVhayBifW9hKERbRFtrKzY0Pj4yXT4+Ml0sZCxjKTtpZigoYnwwKT09MSl7YnJlYWsgYn1lPWItMXwwO2g9ZSYxO2E9MDtpZigoYnwwKSE9Mil7Yj1lJi0yO3doaWxlKDEpe2E9YStjfDA7b2EoYStEW0RbRFtpKzg+PjJdKzY0Pj4yXT4+Ml18MCxkLGMpO2E9YStjfDA7b2EoYStEW0RbRFtpKzg+PjJdKzY0Pj4yXT4+Ml18MCxkLGMpO2Y9ZisyfDA7aWYoKGJ8MCkhPShmfDApKXtjb250aW51ZX1icmVha319aWYoIWgpe2JyZWFrIGJ9b2EoRFtEW0RbaSs4Pj4yXSs2ND4+Ml0+PjJdKyhhK2N8MCl8MCxkLGMpO2JyZWFrIGJ9Yz1CW2srMjR8MF07ZD1uYSgoY3wwKT49MD9jOi0xKTthPURbaSsxNj4+Ml07aWYoRFthKzgwPj4yXSl7aD1EW0RbYT4+Ml0+PjJdK0RbYSs0OD4+Ml18MH1lbHNle2g9MH1pZighYil7YnJlYWsgYn1pZigoY3wwKT4wKXtrPWMmLTQ7bT1jJjM7ZT1jLTE+Pj4wPDM7d2hpbGUoMSl7YT0wO2c9MDtpZighZSl7d2hpbGUoMSl7aj1oKyhmPDwyKXwwO0JbYStkfDBdPURbaj4+Ml07QlsoYXwxKStkfDBdPURbais0Pj4yXTtCWyhhfDIpK2R8MF09RFtqKzg+PjJdO0JbKGF8MykrZHwwXT1EW2orMTI+PjJdO2E9YSs0fDA7Zj1mKzR8MDtnPWcrNHwwO2lmKChrfDApIT0oZ3wwKSl7Y29udGludWV9YnJlYWt9fWc9MDtpZihtKXt3aGlsZSgxKXtCW2ErZHwwXT1EW2grKGY8PDIpPj4yXTthPWErMXwwO2Y9ZisxfDA7Zz1nKzF8MDtpZigobXwwKSE9KGd8MCkpe2NvbnRpbnVlfWJyZWFrfX1vYShEW0RbRFtpKzg+PjJdKzY0Pj4yXT4+Ml0rbnwwLGQsYyk7bj1jK258MDtsPWwrMXwwO2lmKChsfDApIT0oYnwwKSl7Y29udGludWV9YnJlYWt9YnJlYWsgYn1vYShEW0Rbays2ND4+Ml0+PjJdLGQsYyk7aWYoKGJ8MCk9PTEpe2JyZWFrIGJ9ZT1iLTF8MDtoPWUmMTthPTA7aWYoKGJ8MCkhPTIpe2I9ZSYtMjt3aGlsZSgxKXthPWErY3wwO29hKGErRFtEW0RbaSs4Pj4yXSs2ND4+Ml0+PjJdfDAsZCxjKTthPWErY3wwO29hKGErRFtEW0RbaSs4Pj4yXSs2ND4+Ml0+PjJdfDAsZCxjKTtmPWYrMnwwO2lmKChifDApIT0oZnwwKSl7Y29udGludWV9YnJlYWt9fWlmKCFoKXticmVhayBifW9hKERbRFtEW2krOD4+Ml0rNjQ+PjJdPj4yXSsoYStjfDApfDAsZCxjKTticmVhayBifWU9QltrKzI0fDBdO2E9ZStlfDA7ZD1uYShhPj4+MDxlPj4+MD8tMTphKTthPURbaSsxNj4+Ml07aWYoRFthKzgwPj4yXSl7aD1EW0RbYT4+Ml0+PjJdK0RbYSs0OD4+Ml18MH1lbHNle2g9MH1pZighYil7YnJlYWsgYn1jPWU8PDE7aWYoKGV8MCk+MCl7az1lJi00O209ZSYzO2U9ZS0xPj4+MDwzO3doaWxlKDEpe2E9MDtnPTA7aWYoIWUpe3doaWxlKDEpe2w9YTw8MTtqPWgrKGY8PDIpfDA7Q1tsK2Q+PjFdPURbaj4+Ml07Q1sobHwyKStkPj4xXT1EW2orND4+Ml07Q1sobHw0KStkPj4xXT1EW2orOD4+Ml07Q1sobHw2KStkPj4xXT1EW2orMTI+PjJdO2E9YSs0fDA7Zj1mKzR8MDtnPWcrNHwwO2lmKChrfDApIT0oZ3wwKSl7Y29udGludWV9YnJlYWt9fWc9MDtpZihtKXt3aGlsZSgxKXtDWyhhPDwxKStkPj4xXT1EW2grKGY8PDIpPj4yXTthPWErMXwwO2Y9ZisxfDA7Zz1nKzF8MDtpZigobXwwKSE9KGd8MCkpe2NvbnRpbnVlfWJyZWFrfX1vYShEW0RbRFtpKzg+PjJdKzY0Pj4yXT4+Ml0rb3wwLGQsYyk7bz1jK298MDtuPW4rMXwwO2lmKChufDApIT0oYnwwKSl7Y29udGludWV9YnJlYWt9YnJlYWsgYn1vYShEW0Rbays2ND4+Ml0+PjJdLGQsYyk7aWYoKGJ8MCk9PTEpe2JyZWFrIGJ9ZT1iLTF8MDtoPWUmMTthPTA7aWYoKGJ8MCkhPTIpe2I9ZSYtMjt3aGlsZSgxKXthPWErY3wwO29hKGErRFtEW0RbaSs4Pj4yXSs2ND4+Ml0+PjJdfDAsZCxjKTthPWErY3wwO29hKGErRFtEW0RbaSs4Pj4yXSs2ND4+Ml0+PjJdfDAsZCxjKTtmPWYrMnwwO2lmKChifDApIT0oZnwwKSl7Y29udGludWV9YnJlYWt9fWlmKCFoKXticmVhayBifW9hKERbRFtEW2krOD4+Ml0rNjQ+PjJdPj4yXSsoYStjfDApfDAsZCxjKTticmVhayBifWU9QltrKzI0fDBdO2E9ZStlfDA7ZD1uYShhPj4+MDxlPj4+MD8tMTphKTthPURbaSsxNj4+Ml07aWYoRFthKzgwPj4yXSl7aD1EW0RbYT4+Ml0+PjJdK0RbYSs0OD4+Ml18MH1lbHNle2g9MH1pZighYil7YnJlYWsgYn1jPWU8PDE7aWYoKGV8MCk+MCl7az1lJi00O209ZSYzO2U9ZS0xPj4+MDwzO3doaWxlKDEpe2E9MDtnPTA7aWYoIWUpe3doaWxlKDEpe2w9YTw8MTtqPWgrKGY8PDIpfDA7Q1tsK2Q+PjFdPURbaj4+Ml07Q1sobHwyKStkPj4xXT1EW2orND4+Ml07Q1sobHw0KStkPj4xXT1EW2orOD4+Ml07Q1sobHw2KStkPj4xXT1EW2orMTI+PjJdO2E9YSs0fDA7Zj1mKzR8MDtnPWcrNHwwO2lmKChrfDApIT0oZ3wwKSl7Y29udGludWV9YnJlYWt9fWc9MDtpZihtKXt3aGlsZSgxKXtDWyhhPDwxKStkPj4xXT1EW2grKGY8PDIpPj4yXTthPWErMXwwO2Y9ZisxfDA7Zz1nKzF8MDtpZigobXwwKSE9KGd8MCkpe2NvbnRpbnVlfWJyZWFrfX1vYShEW0RbRFtpKzg+PjJdKzY0Pj4yXT4+Ml0rb3wwLGQsYyk7bz1jK298MDtuPW4rMXwwO2lmKChufDApIT0oYnwwKSl7Y29udGludWV9YnJlYWt9YnJlYWsgYn1vYShEW0Rbays2ND4+Ml0+PjJdLGQsYyk7aWYoKGJ8MCk9PTEpe2JyZWFrIGJ9ZT1iLTF8MDtoPWUmMTthPTA7aWYoKGJ8MCkhPTIpe2I9ZSYtMjt3aGlsZSgxKXthPWErY3wwO29hKGErRFtEW0RbaSs4Pj4yXSs2ND4+Ml0+PjJdfDAsZCxjKTthPWErY3wwO29hKGErRFtEW0RbaSs4Pj4yXSs2ND4+Ml0+PjJdfDAsZCxjKTtmPWYrMnwwO2lmKChifDApIT0oZnwwKSl7Y29udGludWV9YnJlYWt9fWlmKCFoKXticmVhayBifW9hKERbRFtEW2krOD4+Ml0rNjQ+PjJdPj4yXSsoYStjfDApfDAsZCxjKTticmVhayBifWU9QltrKzI0fDBdO2M9ZTw8MjtkPW5hKChlfDApIT0oZSYxMDczNzQxODIzKT8tMTpjKTthPURbaSsxNj4+Ml07aWYoRFthKzgwPj4yXSl7aD1EW0RbYT4+Ml0+PjJdK0RbYSs0OD4+Ml18MH1lbHNle2g9MH1pZighYil7YnJlYWsgYn1pZigoZXwwKT4wKXtrPWUmLTQ7bT1lJjM7ZT1lLTE+Pj4wPDM7d2hpbGUoMSl7YT0wO2c9MDtpZighZSl7d2hpbGUoMSl7bD1hPDwyO2o9aCsoZjw8Mil8MDtEW2wrZD4+Ml09RFtqPj4yXTtEWyhsfDQpK2Q+PjJdPURbais0Pj4yXTtEWyhsfDgpK2Q+PjJdPURbais4Pj4yXTtEWyhsfDEyKStkPj4yXT1EW2orMTI+PjJdO2E9YSs0fDA7Zj1mKzR8MDtnPWcrNHwwO2lmKChrfDApIT0oZ3wwKSl7Y29udGludWV9YnJlYWt9fWc9MDtpZihtKXt3aGlsZSgxKXtEWyhhPDwyKStkPj4yXT1EW2grKGY8PDIpPj4yXTthPWErMXwwO2Y9ZisxfDA7Zz1nKzF8MDtpZigobXwwKSE9KGd8MCkpe2NvbnRpbnVlfWJyZWFrfX1vYShEW0RbRFtpKzg+PjJdKzY0Pj4yXT4+Ml0rb3wwLGQsYyk7bz1jK298MDtuPW4rMXwwO2lmKChufDApIT0oYnwwKSl7Y29udGludWV9YnJlYWt9YnJlYWsgYn1vYShEW0Rbays2ND4+Ml0+PjJdLGQsYyk7aWYoKGJ8MCk9PTEpe2JyZWFrIGJ9ZT1iLTF8MDtoPWUmMTthPTA7aWYoKGJ8MCkhPTIpe2I9ZSYtMjt3aGlsZSgxKXthPWErY3wwO29hKGErRFtEW0RbaSs4Pj4yXSs2ND4+Ml0+PjJdfDAsZCxjKTthPWErY3wwO29hKGErRFtEW0RbaSs4Pj4yXSs2ND4+Ml0+PjJdfDAsZCxjKTtmPWYrMnwwO2lmKChifDApIT0oZnwwKSl7Y29udGludWV9YnJlYWt9fWlmKCFoKXticmVhayBifW9hKERbRFtEW2krOD4+Ml0rNjQ+PjJdPj4yXSsoYStjfDApfDAsZCxjKTticmVhayBifWU9QltrKzI0fDBdO2M9ZTw8MjtkPW5hKChlfDApIT0oZSYxMDczNzQxODIzKT8tMTpjKTthPURbaSsxNj4+Ml07aWYoRFthKzgwPj4yXSl7aD1EW0RbYT4+Ml0+PjJdK0RbYSs0OD4+Ml18MH1lbHNle2g9MH1pZighYil7YnJlYWsgYn1pZigoZXwwKT4wKXtrPWUmLTQ7bT1lJjM7ZT1lLTE+Pj4wPDM7d2hpbGUoMSl7YT0wO2c9MDtpZighZSl7d2hpbGUoMSl7bD1hPDwyO2o9aCsoZjw8Mil8MDtEW2wrZD4+Ml09RFtqPj4yXTtEWyhsfDQpK2Q+PjJdPURbais0Pj4yXTtEWyhsfDgpK2Q+PjJdPURbais4Pj4yXTtEWyhsfDEyKStkPj4yXT1EW2orMTI+PjJdO2E9YSs0fDA7Zj1mKzR8MDtnPWcrNHwwO2lmKChrfDApIT0oZ3wwKSl7Y29udGludWV9YnJlYWt9fWc9MDtpZihtKXt3aGlsZSgxKXtEWyhhPDwyKStkPj4yXT1EW2grKGY8PDIpPj4yXTthPWErMXwwO2Y9ZisxfDA7Zz1nKzF8MDtpZigobXwwKSE9KGd8MCkpe2NvbnRpbnVlfWJyZWFrfX1vYShEW0RbRFtpKzg+PjJdKzY0Pj4yXT4+Ml0rb3wwLGQsYyk7bz1jK298MDtuPW4rMXwwO2lmKChufDApIT0oYnwwKSl7Y29udGludWV9YnJlYWt9YnJlYWsgYn1vYShEW0Rbays2ND4+Ml0+PjJdLGQsYyk7aWYoKGJ8MCk9PTEpe2JyZWFrIGJ9ZT1iLTF8MDtoPWUmMTthPTA7aWYoKGJ8MCkhPTIpe2I9ZSYtMjt3aGlsZSgxKXthPWErY3wwO29hKGErRFtEW0RbaSs4Pj4yXSs2ND4+Ml0+PjJdfDAsZCxjKTthPWErY3wwO29hKGErRFtEW0RbaSs4Pj4yXSs2ND4+Ml0+PjJdfDAsZCxjKTtmPWYrMnwwO2lmKChifDApIT0oZnwwKSl7Y29udGludWV9YnJlYWt9fWlmKCFoKXticmVhayBifW9hKERbRFtEW2krOD4+Ml0rNjQ+PjJdPj4yXSsoYStjfDApfDAsZCxjKX1tYShkKTtkPTF9cmV0dXJuIGR8MH1mdW5jdGlvbiBCZChhLGIpe2E9YXwwO2I9YnwwO3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MCxvPTAscD0wLHE9MCxyPTA7aD0kLTk2fDA7JD1oO3I9RFthKzQ+PjJdO2Q9RFtyKzMyPj4yXTtlPURbZCs4Pj4yXTtuPURbZCsxMj4+Ml07Zj1uO2M9RFtkKzIwPj4yXTtwPURbZCsxNj4+Ml07YTp7aWYoKGZ8MCk8PShjfDApJnA+Pj4wPj1lPj4+MHwoY3wwKT4oZnwwKSl7YnJlYWsgYX1rPURbZD4+Ml07cT1FW2srcHwwXTtnPXArMXwwO2Y9Zz9jOmMrMXwwO0RbZCsxNj4+Ml09ZztEW2QrMjA+PjJdPWY7aWYoZT4+PjA8PWc+Pj4wJihmfDApPj0obnwwKXwoZnwwKT4obnwwKSl7YnJlYWsgYX1sPUVbZytrfDBdO2c9cCsyfDA7Zj1nPj4+MDwyP2MrMXwwOmM7bT1nO0RbZCsxNj4+Ml09ZztEW2QrMjA+PjJdPWY7aT1xPDwyND4+MjQ7Yjp7aWYoKGl8MCk+PTApe2c9RFthKzIxNj4+Ml07aWYocT4+PjA+PShEW2ErMjIwPj4yXS1nfDApLzE0ND4+PjApe2JyZWFrIGF9bz1nK0oocSwxNDQpfDA7aWYoRFtvPj4yXTwwKXticmVhayBifWJyZWFrIGF9aWYoRFthKzIxMj4+Ml0+PTApe2JyZWFrIGF9bz1hKzIxMnwwfURbbz4+Ml09YjtjOntkOntnPUZbciszNj4+MV07ZTp7aWYoKChnPDw4fGc+Pj44KSY2NTUzNSk+Pj4wPj0yNTgpe2lmKGU+Pj4wPD1tPj4+MCYoZnwwKT49KG58MCl8KGZ8MCk+KG58MCkpe2JyZWFrIGF9Zz1FW2srbXwwXTtmPXArM3wwO2M9Zj4+PjA8Mz9jKzF8MDpjO0RbZCsxNj4+Ml09ZjtEW2QrMjA+PjJdPWM7Yz1nPj4+MD4xO2lmKGMpe2JyZWFrIGF9Yz1jPzA6ZztpZighbCl7YnJlYWsgZX1pZihjKXticmVhayBhfWJyZWFrIGR9aWYobCl7YnJlYWsgZH1jPTB9aWYoKGl8MCk8MCl7Zj1hKzE4NHwwfWVsc2V7ZD1EW2ErMjE2Pj4yXStKKHEsMTQ0KXwwO0JbZCsxMDB8MF09MDtmPWQrMTA0fDB9ZD1mO2Y6e2lmKChjfDApPT0xKXtlPSQtMTEyfDA7JD1lO209RFtEW2ErND4+Ml0rNDQ+PjJdO2M9bmEoMTIwKTtEW2M+PjJdPTg5NTY7RFtjKzQ+PjJdPTA7RFtjKzExNj4+Ml09MDtEW2MrMTEyPj4yXT1kO0RbYysxMDg+PjJdPW07RFtjKzEyPj4yXT0wO0RbYysxNj4+Ml09MDtEW2MrMjA+PjJdPTA7RFtjKzI0Pj4yXT0wO0RbYysyOD4+Ml09MDtEW2MrMzI+PjJdPTA7RFtjKzM2Pj4yXT0wO0RbYys0MD4+Ml09MDtEW2MrNDQ+PjJdPTA7RFtjKzQ4Pj4yXT0wO0RbYys1Mj4+Ml09MDtEW2MrNTY+PjJdPTA7RFtjKzYwPj4yXT0wO0RbYys4Pj4yXT05MTY4O2Y9Yy0gLTY0fDA7RFtmPj4yXT0wO0RbZis0Pj4yXT0wO0RbYys3Mj4+Ml09MDtEW2MrNzY+PjJdPTA7RFtjKzgwPj4yXT0wO0RbYys4ND4+Ml09MDtEW2MrODg+PjJdPTA7RFtjKzEwND4+Ml09MDtEW2MrOTY+PjJdPTA7RFtjKzEwMD4+Ml09MDtsPURbYSs4Pj4yXTtEW2UrNDg+PjJdPTA7RFtlKzUyPj4yXT0wO0RbZSs0MD4+Ml09MDtEW2UrNDQ+PjJdPTA7aT1lKzMyfDA7Zj1pO0RbZj4+Ml09MDtEW2YrND4+Ml09MDtEW2UrMjQ+PjJdPTA7RFtlKzI4Pj4yXT0wO0RbZSsxNj4+Ml09MDtEW2UrMjA+PjJdPTA7Zj1lLSAtNjR8MDtEW2Y+PjJdPTA7RFtmKzQ+PjJdPTA7RFtlKzcyPj4yXT0wO0RbZSs3Nj4+Ml09MDtEW2UrODA+PjJdPTA7RFtlKzg0Pj4yXT0wO0RbZSs4OD4+Ml09MDtEW2UrMTA0Pj4yXT0wO0RbZSs1Nj4+Ml09MDtEW2UrNjA+PjJdPTA7RFtlKzg+PjJdPTkxNjg7RFtlKzk2Pj4yXT0wO0RbZSsxMDA+PjJdPTA7RFtlKzEyPj4yXT1sO2c9RFtsPj4yXTtmPURbbCs0Pj4yXTtCW2UrMTExfDBdPTA7bj1pO2k9ZSsxMTF8MDtFYShuLChmLWc+PjI+Pj4wKS8zfDAsaSk7Zj1EW2UrMTI+PjJdO2c9RFtmKzI4Pj4yXTtmPURbZisyND4+Ml07QltlKzExMXwwXT0wO0VhKGUrNDR8MCxnLWY+PjIsaSk7RFtlKzI4Pj4yXT1jO0RbZSsyND4+Ml09bTtEW2UrMjA+PjJdPWQ7RFtlKzE2Pj4yXT1sO2Q9Yys4fDA7Zj1lKzh8MDtoYyhkLGYpO2c6e2lmKChkfDApPT0oZnwwKSl7RFtjKzkyPj4yXT1EW2YrODQ+PjJdO2JyZWFrIGd9ZmIoYys1NnwwLERbZis0OD4+Ml0sRFtmKzUyPj4yXSk7ZmIoYys2OHwwLERbZis2MD4+Ml0sRFtmLSAtNjQ+PjJdKTtmYihjKzgwfDAsRFtmKzcyPj4yXSxEW2YrNzY+PjJdKTtEW2MrOTI+PjJdPURbZis4ND4+Ml07aDp7aT1EW2YrOTI+PjJdO2w9RFtmKzg4Pj4yXTtnPWktbHwwO209Zz4+MjtvPURbYysxMDQ+PjJdO2s9RFtjKzk2Pj4yXTtpZihtPj4+MDw9by1rPj4yPj4+MCl7ZD1EW2MrMTAwPj4yXS1rfDA7Zj1kPj4yO2c9Zj4+PjA8bT4+PjA/ZCtsfDA6aTtkPWctbHwwO2lmKGQpe05hKGssbCxkKX1pZihmPj4+MDxtPj4+MCl7ZD1EW2MrMTAwPj4yXTtmPWktZ3wwO2lmKChmfDApPjApe2Q9b2EoZCxnLGYpK2Z8MH1EW2MrMTAwPj4yXT1kO2JyZWFrIGh9RFtjKzEwMD4+Ml09ZCtrO2JyZWFrIGh9aWYoayl7RFtjKzEwMD4+Ml09azttYShrKTtEW2MrMTA0Pj4yXT0wO0RbYys5Nj4+Ml09MDtEW2MrMTAwPj4yXT0wO289MH1pOntpZigoZ3wwKTwwKXticmVhayBpfWQ9bz4+MTtkPW8+PjI+Pj4wPDUzNjg3MDkxMT9kPj4+MDxtPj4+MD9tOmQ6MTA3Mzc0MTgyMztpZihkPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgaX1mPWQ8PDI7ZD1uYShmKTtEW2MrOTY+PjJdPWQ7RFtjKzEwMD4+Ml09ZDtEW2MrMTA0Pj4yXT1kK2Y7aWYoZyl7ZD1vYShkLGwsZykrZ3wwfURbYysxMDA+PjJdPWQ7YnJlYWsgaH1xYSgpO1QoKX19RFtlKzg+PjJdPTkxNjg7ZD1EW2UrOTY+PjJdO2lmKGQpe0RbZSsxMDA+PjJdPWQ7bWEoZCl9ZD1EW2UrODA+PjJdO2lmKGQpe0RbZSs4ND4+Ml09ZDttYShkKX1kPURbZSs2OD4+Ml07aWYoZCl7RFtlKzcyPj4yXT1kO21hKGQpfWQ9RFtlKzU2Pj4yXTtpZihkKXtEW2UrNjA+PjJdPWQ7bWEoZCl9RFtlKzg+PjJdPTk0MDQ7ZD1EW2UrNDQ+PjJdO2lmKGQpe21hKGQpfWQ9RFtlKzMyPj4yXTtpZihkKXttYShkKX0kPWUrMTEyfDA7YnJlYWsgZn1lPSQrLTY0fDA7JD1lO2w9RFtEW2ErND4+Ml0rNDQ+PjJdO2M9bmEoODApO0RbYz4+Ml09OTQyNDtEW2MrND4+Ml09MDtEW2MrNzY+PjJdPTA7RFtjKzcyPj4yXT1kO0RbYys2OD4+Ml09bDtEW2MrOD4+Ml09OTU4ODtEW2MrMTI+PjJdPTA7RFtjKzE2Pj4yXT0wO0RbYysyMD4+Ml09MDtEW2MrMjQ+PjJdPTA7RFtjKzI4Pj4yXT0wO0RbYyszMj4+Ml09MDtEW2MrMzY+PjJdPTA7RFtjKzQwPj4yXT0wO0RbYys0ND4+Ml09MDtEW2MrNDg+PjJdPTA7RFtjKzUyPj4yXT0wO0RbYy0gLTY0Pj4yXT0wO209Yys1NnwwO2Y9bTtEW2Y+PjJdPTA7RFtmKzQ+PjJdPTA7az1EW2ErOD4+Ml07RFtlKzQwPj4yXT0wO0RbZSs0ND4+Ml09MDtEW2UrMzI+PjJdPTA7RFtlKzM2Pj4yXT0wO2k9ZSsyNHwwO2Y9aTtEW2Y+PjJdPTA7RFtmKzQ+PjJdPTA7RFtlKzE2Pj4yXT0wO0RbZSsyMD4+Ml09MDtEW2UrOD4+Ml09MDtEW2UrMTI+PjJdPTA7RFtlKzU2Pj4yXT0wO0RbZSs0OD4+Ml09MDtEW2UrNTI+PjJdPTA7RFtlPj4yXT05NTg4O0RbZSs0Pj4yXT1rO2c9RFtrPj4yXTtmPURbays0Pj4yXTtCW2UrNjN8MF09MDtuPWk7aT1lKzYzfDA7RWEobiwoZi1nPj4yPj4+MCkvM3wwLGkpO2Y9RFtlKzQ+PjJdO2c9RFtmKzI4Pj4yXTtmPURbZisyND4+Ml07QltlKzYzfDBdPTA7RWEoZSszNnwwLGctZj4+MixpKTtEW2UrMjA+PjJdPWM7RFtlKzE2Pj4yXT1sO0RbZSsxMj4+Ml09ZDtEW2UrOD4+Ml09aztoYyhjKzh8MCxlKTtmYihtLERbZSs0OD4+Ml0sRFtlKzUyPj4yXSk7RFtlPj4yXT05NTg4O2Q9RFtlKzQ4Pj4yXTtpZihkKXtEW2UrNTI+PjJdPWQ7bWEoZCl9RFtlPj4yXT05NDA0O2Q9RFtlKzM2Pj4yXTtpZihkKXttYShkKX1kPURbZSsyND4+Ml07aWYoZCl7bWEoZCl9JD1lLSAtNjR8MH1pZighYyl7YnJlYWsgYX1icmVhayBjfWlmKChpfDApPDApe2JyZWFrIGF9Zz1EW3IrNDQ+PjJdO2Q9RFthKzIxNj4+Ml07Yz1uYSg4MCk7RFtjKzc2Pj4yXT0wO0RbYys2OD4+Ml09ZztEW2MrOD4+Ml09ODY1MjtEW2M+PjJdPTk3MTY7RFtjKzQ+PjJdPTA7Zj1kK0oocSwxNDQpfDA7aj1mKzEwNHwwO0RbYys3Mj4+Ml09ajtEW2MtIC02ND4+Ml09MDtEW2MrNTY+PjJdPTA7RFtjKzYwPj4yXT0wO0RbYys1Mj4+Ml09MDtEW2MrNDQ+PjJdPTA7RFtjKzQ4Pj4yXT0wO0RbYyszNj4+Ml09MDtEW2MrNDA+PjJdPTA7RFtjKzI4Pj4yXT0wO0RbYyszMj4+Ml09MDtEW2MrMjA+PjJdPTA7RFtjKzI0Pj4yXT0wO0RbYysxMj4+Ml09MDtEW2MrMTY+PjJdPTA7RFtoKzI0Pj4yXT1nO0RbaCs2OD4+Ml09MDtEW2grNzI+PjJdPTA7RFtoKzYwPj4yXT0wO0RbaCs2ND4+Ml09MDtEW2grNTI+PjJdPTA7RFtoKzU2Pj4yXT0wO0RbaCs0ND4+Ml09MDtEW2grNDg+PjJdPTA7RFtoKzg0Pj4yXT0wO0RbaCs4OD4+Ml09MDtEW2grNzY+PjJdPTA7RFtoKzgwPj4yXT0wO0RbaCsyOD4+Ml09YztkPURbaCsyOD4+Ml07RFtoKzg+PjJdPURbaCsyND4+Ml07RFtoKzEyPj4yXT1kO2Y9Zis0fDA7RFtoKzE2Pj4yXT1mO0RbaCsyMD4+Ml09ajtEW2grMzY+PjJdPTA7RFtoKzQwPj4yXT0wO0RbaCszMj4+Ml09ODY1MjtkPURbaCsyMD4+Ml07RFtoPj4yXT1EW2grMTY+PjJdO0RbaCs0Pj4yXT1kO2o9aCszMnwwO0FkKGosZixoKTtkPWMrOHwwO2hjKGQsaik7aWYoKGR8MCkhPShqfDApKXtmYihjKzU2fDAsRFtqKzQ4Pj4yXSxEW2orNTI+PjJdKX16ZChqKX1jPXdjKG5hKDY0KSxjKTtpPURbYSs0Pj4yXTthPWM7Yz1iO2o6e2s6e2lmKChjfDApPj0wKXtmPWkrOHwwO2I9RFtpKzEyPj4yXTtkPURbaSs4Pj4yXTtnPWItZD4+MjtsOntpZigoZ3wwKT4oY3wwKSl7YnJlYWsgbH1qPWMrMXwwO2lmKGM+Pj4wPj1nPj4+MCl7T2IoZixqLWd8MCk7YnJlYWsgbH1pZihnPj4+MDw9aj4+PjApe2JyZWFrIGx9ZD1kKyhqPDwyKXwwO2lmKChkfDApIT0oYnwwKSl7d2hpbGUoMSl7Yj1iLTR8MDtqPURbYj4+Ml07RFtiPj4yXT0wO2lmKGope2JhW0RbRFtqPj4yXSs0Pj4yXV0oail9aWYoKGJ8MCkhPShkfDApKXtjb250aW51ZX1icmVha319RFtpKzEyPj4yXT1kfWQ9RFtmPj4yXSsoYzw8Mil8MDtiPURbZD4+Ml07RFtkPj4yXT1hO2lmKGIpe2JyZWFrIGt9YnJlYWsgan1iPWE7aWYoIWEpe2JyZWFrIGp9fWJhW0RbRFtiPj4yXSs0Pj4yXV0oYil9aj0oY14tMSk+Pj4zMXwwfSQ9aCs5NnwwO3JldHVybiBqfDB9ZnVuY3Rpb24gWWQoYSxiLGMsZCxlLGYpe2E9YXwwO2I9YnwwO2M9Y3wwO2Q9ZHwwO2U9ZXwwO2Y9ZnwwO3ZhciBnPTAsaD0wLGk9MCxqPTAsaz0wLGw9MCxtPTAsbj0wLG89MCxwPTAscT0wLHI9MCxzPTAsdD0wLHU9MCx2PTAsdz0wLHg9MCx5PTAsej0wLEE9MCxDPTAsRj0wLEc9MCxIPTAsST0wLEo9MCxLPTAsTD0wLE09MCxOPTAsTz0wLFA9MCxRPTAsUj0wLFM9MCxVPTAsVj0wLFc9MCxYPTA7aWYoKGV8MCk9PTIpe0RbYSs4Pj4yXT0yO0RbYS0gLTY0Pj4yXT1mO2Q9YSszMnwwO2U9RFtkPj4yXTtmPURbYSszNj4+Ml0tZXwwO2c9Zj4+MjthOntpZihnPj4+MDw9MSl7c2EoZCwyLWd8MCk7YnJlYWsgYX1pZigoZnwwKT09OCl7YnJlYWsgYX1EW2ErMzY+PjJdPWUrOH1iOntkPURbYSs1Nj4+Ml07ZT1EW2QrND4+Ml07ZD1EW2Q+PjJdO2Y9ZS1kfDA7aWYoKGZ8MCk8PTApe2Y9MDticmVhayBifWlmKChkfDApIT0oZXwwKSl7bT1hKzYwfDA7Sj1mPj4yO1U9KEp8MCk+MT9KOjE7Zj0xO3doaWxlKDEpe2o9JC04MHwwOyQ9ajtlPS0xO2Q9RFsocTw8MikrZD4+Ml07Zz0tMTtjOntpZigoZHwwKT09LTEpe2JyZWFrIGN9ZT1kKzF8MDtlPShlPj4+MCklM3wwP2U6ZC0yfDA7Zz1kLTF8MDtpZigoZD4+PjApJTN8MCl7YnJlYWsgY31nPWQrMnwwfWk9RFttKzM2Pj4yXTtkPURbaT4+Ml07ZDp7ZTp7Zjp7Zzp7aDp7aT1EW2krND4+Ml0tZD4+MjtoPWU8PDI7ZT1EW0RbbSszMj4+Ml0rMjg+PjJdO2s9RFtoK2U+PjJdO2lmKGk+Pj4wPD1rPj4+MCl7YnJlYWsgaH1lPURbZSsoZzw8Mik+PjJdO2lmKGU+Pj4wPj1pPj4+MCl7YnJlYWsgaH1pOntqOntnPURbZCsoZTw8Mik+PjJdO2k9RFtkKyhrPDwyKT4+Ml07aWYoKGd8MCk+PShxfDApfChpfDApPj0ocXwwKSl7YnJlYWsgan1kPShnPDwzKStjfDA7dj1EW2QrND4+Ml07ZT0oaTw8MykrY3wwO3M9RFtlKzQ+PjJdO3g9RFtkPj4yXTtGPURbZT4+Ml07aWYoISgoeHwwKSE9KEZ8MCl8KHN8MCkhPSh2fDApKSl7RFttKzg+PjJdPUY7RFttKzEyPj4yXT1zO2JyZWFrIGl9ZD1EW0RbbSs0Pj4yXSsocTw8Mik+PjJdO0Rbais3Mj4+Ml09MDtEW2orNzY+PjJdPTA7ZT1qLSAtNjR8MDtEW2U+PjJdPTA7RFtlKzQ+PjJdPTA7RFtqKzU2Pj4yXT0wO0Rbais2MD4+Ml09MDtlPURbbT4+Ml07aWYoIUVbZSs4NHwwXSl7ZD1EW0RbZSs2OD4+Ml0rKGQ8PDIpPj4yXX1GYShlLGQsQltlKzI0fDBdLGorNTZ8MCk7ZD1EW0RbbSs0Pj4yXSsoaTw8Mik+PjJdO0Rbais0OD4+Ml09MDtEW2orNTI+PjJdPTA7RFtqKzQwPj4yXT0wO0Rbais0ND4+Ml09MDtEW2orMzI+PjJdPTA7RFtqKzM2Pj4yXT0wO2U9RFttPj4yXTtpZighRVtlKzg0fDBdKXtkPURbRFtlKzY4Pj4yXSsoZDw8Mik+PjJdfUZhKGUsZCxCW2UrMjR8MF0saiszMnwwKTtkPURbRFttKzQ+PjJdKyhnPDwyKT4+Ml07RFtqKzI0Pj4yXT0wO0RbaisyOD4+Ml09MDtEW2orMTY+PjJdPTA7RFtqKzIwPj4yXT0wO0Rbais4Pj4yXT0wO0RbaisxMj4+Ml09MDtlPURbbT4+Ml07aWYoIUVbZSs4NHwwXSl7ZD1EW0RbZSs2OD4+Ml0rKGQ8PDIpPj4yXX1GYShlLGQsQltlKzI0fDBdLGorOHwwKTtLPURbais0ND4+Ml07ZD1EW2orMTY+PjJdO0c9RFtqKzQwPj4yXTtlPUc7az1EW2orMjA+PjJdLShLKyhkPj4+MDxlPj4+MCl8MCl8MDtvPWQtZXwwO2Q9JGgobyxrLG8sayk7ZT1hYTtuPWQ7TD1EW2orMzY+PjJdO2Q9RFtqKzg+PjJdO0g9RFtqKzMyPj4yXTtnPUg7aD1EW2orMTI+PjJdLShMKyhkPj4+MDxnPj4+MCl8MCl8MDt0PWQtZ3wwO2c9JGgodCxoLHQsaCk7ZD1uK2d8MDtlPWFhK2V8MDtlPWQ+Pj4wPGc+Pj4wP2UrMXwwOmU7bj1kO009RFtqKzUyPj4yXTtkPURbaisyND4+Ml07ST1EW2orNDg+PjJdO2c9STtsPURbaisyOD4+Ml0tKE0rKGQ+Pj4wPGc+Pj4wKXwwKXwwO3A9ZC1nfDA7cj0kaChwLGwscCxsKTtkPW4rcnwwO2c9YWErZXwwO3U9ZDtyPWQ+Pj4wPHI+Pj4wP2crMXwwOmc7aWYoIShkfHIpKXticmVhayBqfW49RFtqKzY0Pj4yXTtkPW47Tz1EW2orNjg+PjJdO2Q9JGgoZC1HfDAsTy0oKGQ+Pj4wPEc+Pj4wKStLfDApfDAsbyxrKTtlPWFhO2c9ZDtQPURbais1Nj4+Ml07ZD1QO1E9RFtqKzYwPj4yXTtpPSRoKGQtSHwwLFEtKChkPj4+MDxIPj4+MCkrTHwwKXwwLHQsaCk7ZD1nK2l8MDtnPWFhK2V8MDtnPWQ+Pj4wPGk+Pj4wP2crMXwwOmc7ZT1kO1I9RFtqKzcyPj4yXTtkPVI7Uz1EW2orNzY+PjJdO2k9JGgoZC1JfDAsUy0oKGQ+Pj4wPEk+Pj4wKStNfDApfDAscCxsKTtkPWUraXwwO2U9YWErZ3wwO3k9ZDt3PWQ+Pj4wPGk+Pj4wP2UrMXwwOmU7ZD1sPj4zMTtlPWQrcHwwO2c9ZCtsfDA7Zz1lPj4+MDxkPj4+MD9nKzF8MDpnO2U9ZV5kO2Q9ZF5nO2k9ZDtkPWs+PjMxO3o9ZCtvfDA7Zz1kK2t8MDtBPWReejtkPWReKGQ+Pj4wPno+Pj4wP2crMXwwOmcpO3o9ZDtOPTA7ZD1oPj4zMTtDPWQrdHwwO2c9ZCtofDA7Zz1DPj4+MDxkPj4+MD9nKzF8MDpnO1Y9ZTtDPUNeZDtkPWReZztnPSh6fDApPT0oZHwwKSZBPj4+MD5DPj4+MHxkPj4+MDx6Pj4+MDtBPWc/QTpDO2Q9Zz96OmQ7ZT0oaXwwKT09KGR8MCkmZT4+PjA+QT4+PjB8ZD4+PjA8aT4+PjA7ZT1iaSgtMSwyMTQ3NDgzNjQ3LGU/VjpBLGU/aTpkKT4+PjA8eT4+PjA7ZD1hYTtpZihlJihkfDApPD0od3wwKXwoZHwwKTwod3wwKSl7YnJlYWsgZH1pPTE7ZD0wO2U9bjtvPWFpKCRoKG8sayx5LHcpLGFhLHUscik7az1vK0d8MDtnPWFhK0t8MDtnPWs+Pj4wPG8+Pj4wP2crMXwwOmc7Zz1PLSgoZT4+PjA8az4+PjApK2d8MCl8MDtlPWUta3wwO2c9JGgoZSxnLGUsZyk7bz1hYTtlPVA7bj1nO2g9YWkoJGgodCxoLHksdyksYWEsdSxyKTtrPWgrSHwwO2c9YWErTHwwO2c9aD4+PjA+az4+PjA/ZysxfDA6ZztnPVEtKChlPj4+MDxrPj4+MCkrZ3wwKXwwO2U9ZS1rfDA7az0kaChlLGcsZSxnKTtnPW4ra3wwO2U9YWErb3wwO2U9Zz4+PjA8az4+PjA/ZSsxfDA6ZTtrPWU7ZT1SO249ZztsPWFpKCRoKHAsbCx5LHcpLGFhLHUscik7aD1sK0l8MDtnPWFhK018MDtnPWg+Pj4wPGw+Pj4wP2crMXwwOmc7Zz1TLSgoZT4+PjA8aD4+PjApK2d8MCl8MDtlPWUtaHwwO2g9JGgoZSxnLGUsZyk7ZT1uK2h8MDtnPWFhK2t8MDtrPSRoKGUsZT4+PjA8aD4+PjA/ZysxfDA6Zyx1LHIpO2U9YWE7aD1lO2lmKCFlJms+Pj4wPD0xKXticmVhayBnfWw9aztlPWg7d2hpbGUoMSl7Zz1kPDwxfGk+Pj4zMTtpPWk8PDE7ZD1nO289IWUmbD4+PjA+N3woZXwwKSE9MDtsPShlJjMpPDwzMHxsPj4+MjtlPWU+Pj4yfDA7aWYobyl7Y29udGludWV9YnJlYWt9YnJlYWsgZn1pZigoaXwwKTwocXwwKSl7ZD1pPDwxfWVsc2V7aWYoKHF8MCk8PTApe0RbbSs4Pj4yXT0wO0RbbSsxMj4+Ml09MDticmVhayBpfWQ9KHE8PDEpLTJ8MH1kPShkPDwyKStjfDA7RFttKzg+PjJdPURbZD4+Ml07RFttKzEyPj4yXT1EW2QrND4+Ml19Tj0xO2JyZWFrIGR9dWEoKTtUKCl9ZD1oO2k9aztpZihpLTF8MCl7YnJlYWsgZX19d2hpbGUoMSl7ZT1iaShrLGgsaSxkKStpfDA7Zz1kK2FhfDA7Zz1lPj4+MDxpPj4+MD9nKzF8MDpnO2k9KGcmMSk8PDMxfGU+Pj4xO2Q9Zz4+PjF8MDtlPSRoKGksZCxpLGQpO2c9YWE7aWYoKGh8MCk9PShnfDApJmU+Pj4wPms+Pj4wfGc+Pj4wPmg+Pj4wKXtjb250aW51ZX1icmVha319az1EW20rMjA+PjJdO2lmKGspe2U9ay0xfDA7bD1EW0RbbSsxNj4+Ml0rKGU+Pj4zJjUzNjg3MDkwOCk+PjJdO0RbbSsyMD4+Ml09ZTtnPXM7bz12LWd8MDtoPWc+PjMxO3Q9KHY+PjMxKS0oaCsoZz4+PjA+dj4+PjApfDApfDA7Zz0kaCh5LHcsbyx0KTtwPWFhO3M9JGgocyxoLHUscik7aD1zK2d8MDtnPWFhK3B8MDtnPWg+Pj4wPHM+Pj4wP2crMXwwOmc7bj1oO2g9RjtwPXgtaHwwO3M9aD4+MzE7dj0oeD4+MzEpLShzKyhoPj4+MD54Pj4+MCl8MCl8MDtoPSRoKGksZCxwLHYpO3g9aDtlPWw+Pj5lJjE7bD1lPzAtaHwwOmg7aD1uK2x8MDtuPWc7Zz1hYTtnPW4rKGU/MC0oZysoKHh8MCkhPTApfDApfDA6Zyl8MDtXPW0sWD1haShoLGg+Pj4wPGw+Pj4wP2crMXwwOmcsdSxyKSxEW1crMTI+PjJdPVg7Zz0kaChwLHYseSx3KTtsPWFhO3A9JGgodSxyLEYscyk7aD1wK2d8MDtnPWFhK2x8MDtnPWg+Pj4wPHA+Pj4wP2crMXwwOmc7bj1oO2Q9JGgoaSxkLG8sdCk7aD1lP2Q6MC1kfDA7aT1uK2h8MDtuPWc7Zz1hYTtlPW4rKGU/ZzowLSgoKGR8MCkhPTApK2d8MCl8MCl8MDtXPW0sWD1haShpLGk+Pj4wPGg+Pj4wP2UrMXwwOmUsdSxyKSxEW1crOD4+Ml09WH1OPShrfDApIT0wfSQ9ais4MHwwO2lmKCFOKXticmVhayBifWs6e2lmKERbYSs4Pj4yXTw9MCl7YnJlYWsga31nPURbYSszMj4+Ml07ZD0wO3doaWxlKDEpe2U9ZDw8MjtmPURbKGUrYXwwKSs2OD4+Ml07aT1EW2ErMTY+PjJdO2w6e2lmKChmfDApPihpfDApKXtEW2UrZz4+Ml09aTticmVhayBsfWU9ZStnfDA7aT1EW2ErMTI+PjJdO2lmKChpfDApPihmfDApKXtEW2U+PjJdPWk7YnJlYWsgbH1EW2U+PjJdPWZ9ZD1kKzF8MDtmPURbYSs4Pj4yXTtpZigoZHwwKTwoZnwwKSl7Y29udGludWV9YnJlYWt9ZT0wO2lmKChmfDApPD0wKXticmVhayBrfWQ9cTw8MztpPWQrY3wwO2s9YitkfDA7d2hpbGUoMSl7Zj1lPDwyO2Q9ZitpfDA7Zj1EW2Yraz4+Ml0rRFtmK2c+PjJdfDA7RFtkPj4yXT1mO206e2lmKChmfDApPkRbYSsxNj4+Ml0pe2Y9Zi1EW2ErMjA+PjJdfDB9ZWxzZXtpZigoZnwwKT49RFthKzEyPj4yXSl7YnJlYWsgbX1mPWYrRFthKzIwPj4yXXwwfURbZD4+Ml09Zn1lPWUrMXwwO2lmKChlfDApPERbYSs4Pj4yXSl7Y29udGludWV9YnJlYWt9fXE9cSsxfDA7Zj0oSnwwKT4ocXwwKTtpZigocXwwKT09KFV8MCkpe2JyZWFrIGJ9ZT1EW2ErNTY+PjJdO2Q9RFtlPj4yXTtpZihEW2UrND4+Ml0tZD4+Mj4+PjA+cT4+PjApe2NvbnRpbnVlfWJyZWFrfX11YSgpO1QoKX1hPWZeMX1lbHNle2E9MH1yZXR1cm4gYSYxfWZ1bmN0aW9uIFNoKGEsYixjLGQsZSxmKXthPWF8MDtiPWJ8MDtjPWN8MDtkPWR8MDtlPWV8MDtmPWZ8MDt2YXIgZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MCxvPTAscD0wLHE9MCxyPTAscz0wLHQ9MCx1PTAsdj0wLHc9MCx4PTAseT0wLHo9MCxBPTAsQz0wLEY9MCxHPTAsSD0wLEk9MCxKPTAsSz0wLEw9MCxNPTAsTj0wLE89MCxQPTAsUT0wLFI9MCxTPTAsVT0wLFY9MCxXPTA7aWYoKGV8MCk9PTIpe0RbYSs4Pj4yXT0yO0RbYS0gLTY0Pj4yXT1mO2Q9YSszMnwwO2U9RFtkPj4yXTtmPURbYSszNj4+Ml0tZXwwO2c9Zj4+MjthOntpZihnPj4+MDw9MSl7c2EoZCwyLWd8MCk7YnJlYWsgYX1pZigoZnwwKT09OCl7YnJlYWsgYX1EW2ErMzY+PjJdPWUrOH1iOntkPURbYSs1Nj4+Ml07ZT1EW2QrND4+Ml07ZD1EW2Q+PjJdO2Y9ZS1kfDA7aWYoKGZ8MCk8PTApe2Y9MDticmVhayBifWlmKChkfDApIT0oZXwwKSl7bT1hKzYwfDA7ST1mPj4yO1U9KEl8MCk+MT9JOjE7Zj0xO3doaWxlKDEpe2k9JC04MHwwOyQ9aTtlPS0xO2M6e2Q6e2c9RFsocDw8MikrZD4+Ml07aWYoKGd8MCk9PS0xKXticmVhayBkfWg9RFttKzMyPj4yXTtkPWcrMXwwO2Q9KGQ+Pj4wKSUzfDA/ZDpnLTJ8MDtpZigoZHwwKSE9LTEpe2U9RFtEW2g+PjJdKyhkPDwyKT4+Ml19ZD0tMTtnPWcrKChnPj4+MCklM3wwPy0xOjIpfDA7aWYoKGd8MCkhPS0xKXtkPURbRFtoPj4yXSsoZzw8Mik+PjJdfWg9RFttKzM2Pj4yXTtnPURbaD4+Ml07aD1EW2grND4+Ml0tZz4+MjtpZihoPj4+MDw9ZT4+PjB8ZD4+PjA+PWg+Pj4wKXticmVhayBkfWg9RFtnKyhlPDwyKT4+Ml07ZTp7Zjp7Zzp7aDp7aTp7ajp7Zz1EW2crKGQ8PDIpPj4yXTtpZigoZ3wwKT49KHB8MCl8KGh8MCk+PShwfDApKXticmVhayBqfWQ9KGc8PDMpK2N8MDt3PURbZCs0Pj4yXTtlPShoPDwzKStjfDA7cj1EW2UrND4+Ml07dD1EW2Q+PjJdO0Y9RFtlPj4yXTtpZighKCh0fDApIT0oRnwwKXwocnwwKSE9KHd8MCkpKXtEW20rOD4+Ml09RjtEW20rMTI+PjJdPXI7YnJlYWsgaX1kPURbRFttKzQ+PjJdKyhwPDwyKT4+Ml07RFtpKzcyPj4yXT0wO0RbaSs3Nj4+Ml09MDtlPWktIC02NHwwO0RbZT4+Ml09MDtEW2UrND4+Ml09MDtEW2krNTY+PjJdPTA7RFtpKzYwPj4yXT0wO2U9RFttPj4yXTtpZighRVtlKzg0fDBdKXtkPURbRFtlKzY4Pj4yXSsoZDw8Mik+PjJdfUZhKGUsZCxCW2UrMjR8MF0saSs1NnwwKTtkPURbRFttKzQ+PjJdKyhoPDwyKT4+Ml07RFtpKzQ4Pj4yXT0wO0RbaSs1Mj4+Ml09MDtEW2krNDA+PjJdPTA7RFtpKzQ0Pj4yXT0wO0RbaSszMj4+Ml09MDtEW2krMzY+PjJdPTA7ZT1EW20+PjJdO2lmKCFFW2UrODR8MF0pe2Q9RFtEW2UrNjg+PjJdKyhkPDwyKT4+Ml19RmEoZSxkLEJbZSsyNHwwXSxpKzMyfDApO2U9RFtEW20rND4+Ml0rKGc8PDIpPj4yXTtEW2krMjQ+PjJdPTA7RFtpKzI4Pj4yXT0wO0RbaSsxNj4+Ml09MDtEW2krMjA+PjJdPTA7RFtpKzg+PjJdPTA7RFtpKzEyPj4yXT0wO2Q9RFttPj4yXTtpZighRVtkKzg0fDBdKXtlPURbRFtkKzY4Pj4yXSsoZTw8Mik+PjJdfUZhKGQsZSxCW2QrMjR8MF0saSs4fDApO0o9RFtpKzQ0Pj4yXTtkPURbaSsxNj4+Ml07eD1EW2krNDA+PjJdO2U9eDtrPURbaSsyMD4+Ml0tKEorKGQ+Pj4wPGU+Pj4wKXwwKXwwO249ZC1lfDA7ZD0kaChuLGssbixrKTtlPWFhO289ZDtLPURbaSszNj4+Ml07ZD1EW2krOD4+Ml07Rz1EW2krMzI+PjJdO2c9RztqPURbaSsxMj4+Ml0tKEsrKGQ+Pj4wPGc+Pj4wKXwwKXwwO3U9ZC1nfDA7Zz0kaCh1LGosdSxqKTtkPW8rZ3wwO2U9YWErZXwwO2U9ZD4+PjA8Zz4+PjA/ZSsxfDA6ZTtvPWQ7TD1EW2krNTI+PjJdO2Q9RFtpKzI0Pj4yXTtIPURbaSs0OD4+Ml07Zz1IO2w9RFtpKzI4Pj4yXS0oTCsoZD4+PjA8Zz4+PjApfDApfDA7cz1kLWd8MDtxPSRoKHMsbCxzLGwpO2Q9bytxfDA7Zz1hYStlfDA7dj1kO3E9ZD4+PjA8cT4+PjA/ZysxfDA6ZztpZighKGR8cSkpe2JyZWFrIGp9Tj1EW2krNjQ+PjJdO2Q9TjtPPURbaSs2OD4+Ml07ZD0kaChkLXh8MCxPLSgoZD4+PjA8eD4+PjApK0p8MCl8MCxuLGspO2U9YWE7Zz1kO1A9RFtpKzU2Pj4yXTtkPVA7UT1EW2krNjA+PjJdO2g9JGgoZC1HfDAsUS0oKGQ+Pj4wPEc+Pj4wKStLfDApfDAsdSxqKTtkPWcraHwwO2c9YWErZXwwO2c9ZD4+PjA8aD4+PjA/ZysxfDA6ZztlPWQ7Uj1EW2krNzI+PjJdO2Q9UjtTPURbaSs3Nj4+Ml07aD0kaChkLUh8MCxTLSgoZD4+PjA8SD4+PjApK0x8MCl8MCxzLGwpO2Q9ZStofDA7ZT1hYStnfDA7ej1kO3k9ZD4+PjA8aD4+PjA/ZSsxfDA6ZTtkPWw+PjMxO2U9ZCtzfDA7Zz1kK2x8MDtnPWU+Pj4wPGQ+Pj4wP2crMXwwOmc7aD1lXmQ7ZD1kXmc7TT1kO2Q9az4+MzE7Zz1kK258MDtlPWQra3wwO0E9Z15kO2Q9ZF4oZz4+PjA8ZD4+PjA/ZSsxfDA6ZSk7bz1kO2U9MDtkPWo+PjMxO0M9ZCt1fDA7Zz1kK2p8MDtnPUM+Pj4wPGQ+Pj4wP2crMXwwOmc7Qz1DXmQ7ZD1kXmc7Zz0ob3wwKT09KGR8MCkmQT4+PjA+Qz4+PjB8ZD4+PjA8bz4+PjA7QT1nP0E6QztkPWc/bzpkO2c9KE18MCk9PShkfDApJmg+Pj4wPkE+Pj4wfGQ+Pj4wPE0+Pj4wO2c9YmkoLTEsMjE0NzQ4MzY0NyxnP2g6QSxnP006ZCk+Pj4wPHo+Pj4wO2Q9YWE7aWYoZyYoZHwwKTw9KHl8MCl8KGR8MCk8KHl8MCkpe2JyZWFrIGV9aD0xO2Q9MDtlPU47bj1haSgkaChuLGsseix5KSxhYSx2LHEpO2s9bit4fDA7Zz1hYStKfDA7Zz1rPj4+MDxuPj4+MD9nKzF8MDpnO2c9Ty0oKGU+Pj4wPGs+Pj4wKStnfDApfDA7ZT1lLWt8MDtnPSRoKGUsZyxlLGcpO249YWE7ZT1QO289ZztqPWFpKCRoKHUsaix6LHkpLGFhLHYscSk7az1qK0d8MDtnPWFhK0t8MDtnPWo+Pj4wPms+Pj4wP2crMXwwOmc7Zz1RLSgoZT4+PjA8az4+PjApK2d8MCl8MDtlPWUta3wwO2s9JGgoZSxnLGUsZyk7Zz1vK2t8MDtlPWFhK258MDtlPWc+Pj4wPGs+Pj4wP2UrMXwwOmU7az1lO2U9UjtvPWc7bD1haSgkaChzLGwseix5KSxhYSx2LHEpO2o9bCtIfDA7Zz1hYStMfDA7Zz1qPj4+MDxsPj4+MD9nKzF8MDpnO2c9Uy0oKGU+Pj4wPGo+Pj4wKStnfDApfDA7ZT1lLWp8MDtqPSRoKGUsZyxlLGcpO2U9bytqfDA7Zz1hYStrfDA7az0kaChlLGU+Pj4wPGo+Pj4wP2crMXwwOmcsdixxKTtlPWFhO2o9ZTtpZighZSZrPj4+MDw9MSl7YnJlYWsgaH1sPWs7ZT1qO3doaWxlKDEpe2c9ZDw8MXxoPj4+MzE7aD1oPDwxO2Q9ZztuPSFlJmw+Pj4wPjd8KGV8MCkhPTA7bD0oZSYzKTw8MzB8bD4+PjI7ZT1lPj4+MnwwO2lmKG4pe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGd9aWYoKGh8MCk8KHB8MCkpe2Q9aDw8MX1lbHNle2lmKChwfDApPD0wKXtEW20rOD4+Ml09MDtEW20rMTI+PjJdPTA7YnJlYWsgaX1kPShwPDwxKS0yfDB9ZD0oZDw8MikrY3wwO0RbbSs4Pj4yXT1EW2Q+PjJdO0RbbSsxMj4+Ml09RFtkKzQ+PjJdfWU9MTticmVhayBlfWQ9ajtoPWs7aWYoaC0xfDApe2JyZWFrIGZ9fXdoaWxlKDEpe2c9YmkoayxqLGgsZCkraHwwO2U9ZCthYXwwO2U9Zz4+PjA8aD4+PjA/ZSsxfDA6ZTtoPShlJjEpPDwzMXxnPj4+MTtkPWU+Pj4xfDA7ZT0kaChoLGQsaCxkKTtnPWFhO2lmKChqfDApPT0oZ3wwKSZlPj4+MD5rPj4+MHxnPj4+MD5qPj4+MCl7Y29udGludWV9YnJlYWt9fWs9RFttKzIwPj4yXTtpZihrKXtlPWstMXwwO2w9RFtEW20rMTY+PjJdKyhlPj4+MyY1MzY4NzA5MDgpPj4yXTtEW20rMjA+PjJdPWU7Zz1yO249dy1nfDA7aj1nPj4zMTt1PSh3Pj4zMSktKGorKGc+Pj4wPnc+Pj4wKXwwKXwwO2c9JGgoeix5LG4sdSk7cz1hYTtyPSRoKHIsaix2LHEpO2o9citnfDA7Zz1hYStzfDA7Zz1qPj4+MDxyPj4+MD9nKzF8MDpnO289ajtqPUY7cz10LWp8MDtyPWo+PjMxO3c9KHQ+PjMxKS0ocisoaj4+PjA+dD4+PjApfDApfDA7aj0kaChoLGQscyx3KTt0PWo7aj1sPj4+ZSYxO2U9ajt4PWU/MC10fDA6dDtsPW8reHwwO289ZztnPWFhO2U9bysoZT8wLShnKygodHwwKSE9MCl8MCl8MDpnKXwwO1Y9bSxXPWFpKGwsbD4+PjA8eD4+PjA/ZSsxfDA6ZSx2LHEpLERbVisxMj4+Ml09VztlPSRoKHMsdyx6LHkpO2c9YWE7bD0kaCh2LHEsRixyKTtlPWwrZXwwO2c9YWErZ3wwO2c9ZT4+PjA8bD4+PjA/ZysxfDA6ZztkPSRoKGgsZCxuLHUpO2w9aj9kOjAtZHwwO2g9bCtlfDA7ZT1hYTtlPShqP2U6MC0oKChkfDApIT0wKStlfDApfDApK2d8MDtWPW0sVz1haShoLGg+Pj4wPGw+Pj4wP2UrMXwwOmUsdixxKSxEW1YrOD4+Ml09V31lPShrfDApIT0wfSQ9aSs4MHwwO2JyZWFrIGN9dWEoKTtUKCl9aWYoIWUpe2JyZWFrIGJ9azp7aWYoRFthKzg+PjJdPD0wKXticmVhayBrfWc9RFthKzMyPj4yXTtkPTA7d2hpbGUoMSl7ZT1kPDwyO2Y9RFsoZSthfDApKzY4Pj4yXTtoPURbYSsxNj4+Ml07bDp7aWYoKGZ8MCk+KGh8MCkpe0RbZStnPj4yXT1oO2JyZWFrIGx9ZT1lK2d8MDtoPURbYSsxMj4+Ml07aWYoKGh8MCk+KGZ8MCkpe0RbZT4+Ml09aDticmVhayBsfURbZT4+Ml09Zn1kPWQrMXwwO2Y9RFthKzg+PjJdO2lmKChkfDApPChmfDApKXtjb250aW51ZX1icmVha31lPTA7aWYoKGZ8MCk8PTApe2JyZWFrIGt9ZD1wPDwzO2g9ZCtjfDA7az1iK2R8MDt3aGlsZSgxKXtmPWU8PDI7ZD1mK2h8MDtmPURbZitrPj4yXStEW2YrZz4+Ml18MDtEW2Q+PjJdPWY7bTp7aWYoKGZ8MCk+RFthKzE2Pj4yXSl7Zj1mLURbYSsyMD4+Ml18MH1lbHNle2lmKChmfDApPj1EW2ErMTI+PjJdKXticmVhayBtfWY9ZitEW2ErMjA+PjJdfDB9RFtkPj4yXT1mfWU9ZSsxfDA7aWYoKGV8MCk8RFthKzg+PjJdKXtjb250aW51ZX1icmVha319cD1wKzF8MDtmPShJfDApPihwfDApO2lmKChwfDApPT0oVXwwKSl7YnJlYWsgYn1lPURbYSs1Nj4+Ml07ZD1EW2U+PjJdO2lmKERbZSs0Pj4yXS1kPj4yPj4+MD5wPj4+MCl7Y29udGludWV9YnJlYWt9fXVhKCk7VCgpfWE9Zl4xfWVsc2V7YT0wfXJldHVybiBhJjF9ZnVuY3Rpb24gSGMoYSl7dmFyIGI9MCxjPTAsZD0wLGU9MCxmPTAsZz0wO2U9JC0xNnwwOyQ9ZTtEW2UrMTI+PjJdPWE7YTp7aWYoYT4+PjA8PTIxMSl7ZD1EW0djKDEwMzg0LDEwNTc2LGUrMTJ8MCk+PjJdO2JyZWFrIGF9aWYoYT4+PjA+PTQyOTQ5NjcyOTIpe1NiKCk7VCgpfWY9KGE+Pj4wKS8yMTB8MDtkPUooZiwyMTApO0RbZSs4Pj4yXT1hLWQ7Zz1HYygxMDU3NiwxMDc2OCxlKzh8MCktMTA1NzY+PjI7d2hpbGUoMSl7ZD1EWyhnPDwyKSsxMDU3Nj4+Ml0rZHwwO2E9NTtiOnt3aGlsZSgxKXtjOntpZigoYXwwKT09NDcpe2E9MjExO3doaWxlKDEpe2I9KGQ+Pj4wKS8oYT4+PjApfDA7aWYoYj4+PjA8YT4+PjApe2JyZWFrIGJ9aWYoKEooYSxiKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzEwfDA7Yz0oZD4+PjApLyhiPj4+MCl8MDtpZihiPj4+MD5jPj4+MCl7YnJlYWsgYn1pZigoSihiLGMpfDApPT0oZHwwKSl7YnJlYWsgY31iPWErMTJ8MDtjPShkPj4+MCkvKGI+Pj4wKXwwO2lmKGI+Pj4wPmM+Pj4wKXticmVhayBifWlmKChKKGIsYyl8MCk9PShkfDApKXticmVhayBjfWI9YSsxNnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE4fDA7Yz0oZD4+PjApLyhiPj4+MCl8MDtpZihiPj4+MD5jPj4+MCl7YnJlYWsgYn1pZigoSihiLGMpfDApPT0oZHwwKSl7YnJlYWsgY31iPWErMjJ8MDtjPShkPj4+MCkvKGI+Pj4wKXwwO2lmKGI+Pj4wPmM+Pj4wKXticmVhayBifWlmKChKKGIsYyl8MCk9PShkfDApKXticmVhayBjfWI9YSsyOHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzMwfDA7Yz0oZD4+PjApLyhiPj4+MCl8MDtpZihiPj4+MD5jPj4+MCl7YnJlYWsgYn1pZigoSihiLGMpfDApPT0oZHwwKSl7YnJlYWsgY31iPWErMzZ8MDtjPShkPj4+MCkvKGI+Pj4wKXwwO2lmKGI+Pj4wPmM+Pj4wKXticmVhayBifWlmKChKKGIsYyl8MCk9PShkfDApKXticmVhayBjfWI9YSs0MHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzQyfDA7Yz0oZD4+PjApLyhiPj4+MCl8MDtpZihiPj4+MD5jPj4+MCl7YnJlYWsgYn1pZigoSihiLGMpfDApPT0oZHwwKSl7YnJlYWsgY31iPWErNDZ8MDtjPShkPj4+MCkvKGI+Pj4wKXwwO2lmKGI+Pj4wPmM+Pj4wKXticmVhayBifWlmKChKKGIsYyl8MCk9PShkfDApKXticmVhayBjfWI9YSs1MnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzU4fDA7Yz0oZD4+PjApLyhiPj4+MCl8MDtpZihiPj4+MD5jPj4+MCl7YnJlYWsgYn1pZigoSihiLGMpfDApPT0oZHwwKSl7YnJlYWsgY31iPWErNjB8MDtjPShkPj4+MCkvKGI+Pj4wKXwwO2lmKGI+Pj4wPmM+Pj4wKXticmVhayBifWlmKChKKGIsYyl8MCk9PShkfDApKXticmVhayBjfWI9YSs2NnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzcwfDA7Yz0oZD4+PjApLyhiPj4+MCl8MDtpZihiPj4+MD5jPj4+MCl7YnJlYWsgYn1pZigoSihiLGMpfDApPT0oZHwwKSl7YnJlYWsgY31iPWErNzJ8MDtjPShkPj4+MCkvKGI+Pj4wKXwwO2lmKGI+Pj4wPmM+Pj4wKXticmVhayBifWlmKChKKGIsYyl8MCk9PShkfDApKXticmVhayBjfWI9YSs3OHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzgyfDA7Yz0oZD4+PjApLyhiPj4+MCl8MDtpZihiPj4+MD5jPj4+MCl7YnJlYWsgYn1pZigoSihiLGMpfDApPT0oZHwwKSl7YnJlYWsgY31iPWErODh8MDtjPShkPj4+MCkvKGI+Pj4wKXwwO2lmKGI+Pj4wPmM+Pj4wKXticmVhayBifWlmKChKKGIsYyl8MCk9PShkfDApKXticmVhayBjfWI9YSs5NnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzEwMHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzEwMnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzEwNnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzEwOHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzExMnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzEyMHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzEyNnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzEzMHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzEzNnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzEzOHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE0MnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE0OHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE1MHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE1NnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE2MnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE2NnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE2OHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE3MnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE3OHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE4MHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE4NnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE5MHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE5MnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE5NnwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzE5OHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9aWYoKEooYixjKXwwKT09KGR8MCkpe2JyZWFrIGN9Yj1hKzIwOHwwO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9YT1hKzIxMHwwO2lmKChKKGIsYyl8MCkhPShkfDApKXtjb250aW51ZX1icmVha31icmVhayBjfWI9RFsoYTw8MikrMTAzODQ+PjJdO2M9KGQ+Pj4wKS8oYj4+PjApfDA7aWYoYj4+PjA+Yz4+PjApe2JyZWFrIGJ9YT1hKzF8MDtpZigoSihiLGMpfDApIT0oZHwwKSl7Y29udGludWV9fWJyZWFrfWQ9ZysxfDA7YT0oZHwwKT09NDg7Zz1hPzA6ZDtmPWErZnwwO2Q9SihmLDIxMCk7Y29udGludWV9YnJlYWt9RFtlKzEyPj4yXT1kfSQ9ZSsxNnwwO3JldHVybiBkfWZ1bmN0aW9uIERiKGEsYixjLGQpe3ZhciBlPTAsZj0wLGc9MCxoPTAsaT1LKDApLGo9MDthOntpZighZCl7YnJlYWsgYX1iOntjOntzd2l0Y2goRFthKzI4Pj4yXS0xfDApe2Nhc2UgMDpkOntlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk8PTApe2JyZWFrIGR9Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Zj1EW2E+PjJdO2U9RFtmPj4yXTtpZigoYnwwKT49KERbZis0Pj4yXS1lfDApKXticmVhayBhfWY9YitlfDA7Yj0wO3doaWxlKDEpe0JbYitkfDBdPUVbZnwwXTtiPWIrMXwwO2U9QlthKzI0fDBdO2lmKChifDApPj0oKChjfDApPChlfDApP2M6ZSl8MCkpe2JyZWFrIGR9Zj1mKzF8MDtpZihmPj4+MDxHW0RbYT4+Ml0rND4+Ml0pe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGF9Zz0xO2lmKChjfDApPihlfDApKXticmVhayBifWJyZWFrIGE7Y2FzZSAxOmU6e2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKTw9MCl7YnJlYWsgZX1iPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtmPURbYT4+Ml07ZT1EW2Y+PjJdO2lmKChifDApPj0oRFtmKzQ+PjJdLWV8MCkpe2JyZWFrIGF9Zj1iK2V8MDtiPTA7d2hpbGUoMSl7ZT1CW2Z8MF07aWYoKGV8MCk8MCl7YnJlYWsgYX1CW2IrZHwwXT1lO2I9YisxfDA7ZT1CW2ErMjR8MF07aWYoKGJ8MCk+PSgoKGN8MCk8KGV8MCk/YzplKXwwKSl7YnJlYWsgZX1mPWYrMXwwO2lmKGY+Pj4wPEdbRFthPj4yXSs0Pj4yXSl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1nPTE7aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGJ9YnJlYWsgYTtjYXNlIDI6Zjp7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPD0wKXticmVhayBmfWI9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2Y9RFthPj4yXTtlPURbZj4+Ml07aWYoKGJ8MCk+PShEW2YrND4+Ml0tZXwwKSl7YnJlYWsgYX1mPWIrZXwwO2I9MDt3aGlsZSgxKXtlPUZbZj4+MV07aWYoKGUtMTI4JjY1NTM1KT4+PjA8NjUyODApe2JyZWFrIGF9QltiK2R8MF09ZTtiPWIrMXwwO2U9QlthKzI0fDBdO2lmKChifDApPj0oKChjfDApPChlfDApP2M6ZSl8MCkpe2JyZWFrIGZ9Zj1mKzJ8MDtpZihmPj4+MDxHW0RbYT4+Ml0rND4+Ml0pe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGF9Zz0xO2lmKChjfDApPihlfDApKXticmVhayBifWJyZWFrIGE7Y2FzZSAzOmc6e2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKTw9MCl7YnJlYWsgZ31iPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtmPURbYT4+Ml07ZT1EW2Y+PjJdO2lmKChifDApPj0oRFtmKzQ+PjJdLWV8MCkpe2JyZWFrIGF9Zj1iK2V8MDtiPTA7d2hpbGUoMSl7ZT1GW2Y+PjFdO2lmKGU+Pj4wPjEyNyl7YnJlYWsgYX1CW2IrZHwwXT1lO2I9YisxfDA7ZT1CW2ErMjR8MF07aWYoKGJ8MCk+PSgoKGN8MCk8KGV8MCk/YzplKXwwKSl7YnJlYWsgZ31mPWYrMnwwO2lmKGY+Pj4wPEdbRFthPj4yXSs0Pj4yXSl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1nPTE7aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGJ9YnJlYWsgYTtjYXNlIDQ6aDp7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPD0wKXticmVhayBofWI9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2Y9RFthPj4yXTtlPURbZj4+Ml07aWYoKGJ8MCk+PShEW2YrND4+Ml0tZXwwKSl7YnJlYWsgYX1mPWIrZXwwO2I9MDt3aGlsZSgxKXtlPURbZj4+Ml07aWYoZS0xMjg+Pj4wPDQyOTQ5NjcwNDApe2JyZWFrIGF9QltiK2R8MF09ZTtiPWIrMXwwO2U9QlthKzI0fDBdO2lmKChifDApPj0oKChjfDApPChlfDApP2M6ZSl8MCkpe2JyZWFrIGh9Zj1mKzR8MDtpZihmPj4+MDxHW0RbYT4+Ml0rND4+Ml0pe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGF9Zz0xO2lmKChjfDApPihlfDApKXticmVhayBifWJyZWFrIGE7Y2FzZSA1Omk6e2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKTw9MCl7YnJlYWsgaX1iPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtmPURbYT4+Ml07ZT1EW2Y+PjJdO2lmKChifDApPj0oRFtmKzQ+PjJdLWV8MCkpe2JyZWFrIGF9Zj1iK2V8MDtiPTA7d2hpbGUoMSl7ZT1EW2Y+PjJdO2lmKGU+Pj4wPjEyNyl7YnJlYWsgYX1CW2IrZHwwXT1lO2I9YisxfDA7ZT1CW2ErMjR8MF07aWYoKGJ8MCk+PSgoKGN8MCk8KGV8MCk/YzplKXwwKSl7YnJlYWsgaX1mPWYrNHwwO2lmKGY+Pj4wPEdbRFthPj4yXSs0Pj4yXSl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1nPTE7aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGJ9YnJlYWsgYTtjYXNlIDY6ajp7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPD0wKXticmVhayBqfWI9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2Y9RFthPj4yXTtlPURbZj4+Ml07aWYoKGJ8MCk+PShEW2YrND4+Ml0tZXwwKSl7YnJlYWsgYX1mPWIrZXwwO2I9MDt3aGlsZSgxKXtoPURbZj4+Ml07ZT1EW2YrND4+Ml0tKGg+Pj4wPDEyOCl8MDtpZigoZXwwKT09LTEmaC0xMjg+Pj4wPDQyOTQ5NjcwNDB8KGV8MCkhPS0xKXticmVhayBhfUJbYitkfDBdPWg7Yj1iKzF8MDtlPUJbYSsyNHwwXTtpZigoYnwwKT49KCgoY3wwKTwoZXwwKT9jOmUpfDApKXticmVhayBqfWY9Zis4fDA7aWYoZj4+PjA8R1tEW2E+PjJdKzQ+PjJdKXtjb250aW51ZX1icmVha31icmVhayBhfWc9MTtpZigoY3wwKT4oZXwwKSl7YnJlYWsgYn1icmVhayBhO2Nhc2UgNzprOntlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk8PTApe2JyZWFrIGt9Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Zj1EW2E+PjJdO2U9RFtmPj4yXTtpZigoYnwwKT49KERbZis0Pj4yXS1lfDApKXticmVhayBhfWY9YitlfDA7Yj0wO3doaWxlKDEpe2U9RFtmKzQ+PjJdO2g9RFtmPj4yXTtpZighZSZoPj4+MD4xMjd8ZSl7YnJlYWsgYX1CW2IrZHwwXT1oO2I9YisxfDA7ZT1CW2ErMjR8MF07aWYoKGJ8MCk+PSgoKGN8MCk8KGV8MCk/YzplKXwwKSl7YnJlYWsga31mPWYrOHwwO2lmKGY+Pj4wPEdbRFthPj4yXSs0Pj4yXSl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1nPTE7aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGJ9YnJlYWsgYTtjYXNlIDg6bDp7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPD0wKXticmVhayBsfWI9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2Y9RFthPj4yXTtlPURbZj4+Ml07aWYoKGJ8MCk+PShEW2YrND4+Ml0tZXwwKSl7YnJlYWsgYX1mPWIrZXwwO2I9MDt3aGlsZSgxKXtoPWIrZHwwO2k9SFtmPj4yXTttOntpZihLKEwoaSkpPEsoMjE0NzQ4MzY0OCkpe2U9fn5pO2JyZWFrIG19ZT0tMjE0NzQ4MzY0OH1CW2h8MF09ZTtiPWIrMXwwO2U9QlthKzI0fDBdO2lmKChifDApPj0oKChjfDApPChlfDApP2M6ZSl8MCkpe2JyZWFrIGx9Zj1mKzR8MDtpZihmPj4+MDxHW0RbYT4+Ml0rND4+Ml0pe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGF9Zz0xO2lmKChjfDApPihlfDApKXticmVhayBifWJyZWFrIGE7Y2FzZSA5Om46e2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKTw9MCl7YnJlYWsgbn1iPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtmPURbYT4+Ml07ZT1EW2Y+PjJdO2lmKChifDApPj0oRFtmKzQ+PjJdLWV8MCkpe2JyZWFrIGF9Zj1iK2V8MDtiPTA7d2hpbGUoMSl7aD1iK2R8MDtqPUlbZj4+M107bzp7aWYoTChqKTwyMTQ3NDgzNjQ4KXtlPX5+ajticmVhayBvfWU9LTIxNDc0ODM2NDh9QltofDBdPWU7Yj1iKzF8MDtlPUJbYSsyNHwwXTtpZigoYnwwKT49KCgoY3wwKTwoZXwwKT9jOmUpfDApKXticmVhayBufWY9Zis4fDA7aWYoZj4+PjA8R1tEW2E+PjJdKzQ+PjJdKXtjb250aW51ZX1icmVha31icmVhayBhfWc9MTtpZigoY3wwKT4oZXwwKSl7YnJlYWsgYn1icmVhayBhO2Nhc2UgMTA6YnJlYWsgYztkZWZhdWx0OmJyZWFrIGF9fXA6e2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKTw9MCl7YnJlYWsgcH1iPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtmPURbYT4+Ml07ZT1EW2Y+PjJdO2lmKChifDApPj0oRFtmKzQ+PjJdLWV8MCkpe2JyZWFrIGF9Zj1iK2V8MDtiPTA7d2hpbGUoMSl7QltiK2R8MF09RVtmfDBdO2I9YisxfDA7ZT1CW2ErMjR8MF07aWYoKGJ8MCk+PSgoKGN8MCk8KGV8MCk/YzplKXwwKSl7YnJlYWsgcH1mPWYrMXwwO2lmKGY+Pj4wPEdbRFthPj4yXSs0Pj4yXSl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1nPTE7aWYoKGN8MCk8PShlfDApKXticmVhayBhfX1wYShkK2V8MCwwLGMtZXwwKX1yZXR1cm4gZ31mdW5jdGlvbiBDYihhLGIsYyxkKXt2YXIgZT0wLGY9MCxnPTAsaD0wLGk9SygwKSxqPTA7YTp7aWYoIWQpe2JyZWFrIGF9Yjp7Yzp7c3dpdGNoKERbYSsyOD4+Ml0tMXwwKXtjYXNlIDA6ZDp7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPD0wKXticmVhayBkfWI9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2Y9RFthPj4yXTtlPURbZj4+Ml07aWYoKGJ8MCk+PShEW2YrND4+Ml0tZXwwKSl7YnJlYWsgYX1mPWIrZXwwO2I9MDt3aGlsZSgxKXtlPUJbZnwwXTtpZigoZXwwKTwwKXticmVhayBhfUJbYitkfDBdPWU7Yj1iKzF8MDtlPUJbYSsyNHwwXTtpZigoYnwwKT49KCgoY3wwKTwoZXwwKT9jOmUpfDApKXticmVhayBkfWY9ZisxfDA7aWYoZj4+PjA8R1tEW2E+PjJdKzQ+PjJdKXtjb250aW51ZX1icmVha31icmVhayBhfWc9MTtpZigoY3wwKT4oZXwwKSl7YnJlYWsgYn1icmVhayBhO2Nhc2UgMTplOntlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk8PTApe2JyZWFrIGV9Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Zj1EW2E+PjJdO2U9RFtmPj4yXTtpZigoYnwwKT49KERbZis0Pj4yXS1lfDApKXticmVhayBhfWY9YitlfDA7Yj0wO3doaWxlKDEpe0JbYitkfDBdPUVbZnwwXTtiPWIrMXwwO2U9QlthKzI0fDBdO2lmKChifDApPj0oKChjfDApPChlfDApP2M6ZSl8MCkpe2JyZWFrIGV9Zj1mKzF8MDtpZihmPj4+MDxHW0RbYT4+Ml0rND4+Ml0pe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGF9Zz0xO2lmKChjfDApPihlfDApKXticmVhayBifWJyZWFrIGE7Y2FzZSAyOmY6e2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKTw9MCl7YnJlYWsgZn1iPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtmPURbYT4+Ml07ZT1EW2Y+PjJdO2lmKChifDApPj0oRFtmKzQ+PjJdLWV8MCkpe2JyZWFrIGF9Zj1iK2V8MDtiPTA7d2hpbGUoMSl7ZT1GW2Y+PjFdO2lmKGU+Pj4wPjI1NSl7YnJlYWsgYX1CW2IrZHwwXT1lO2I9YisxfDA7ZT1CW2ErMjR8MF07aWYoKGJ8MCk+PSgoKGN8MCk8KGV8MCk/YzplKXwwKSl7YnJlYWsgZn1mPWYrMnwwO2lmKGY+Pj4wPEdbRFthPj4yXSs0Pj4yXSl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1nPTE7aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGJ9YnJlYWsgYTtjYXNlIDM6Zzp7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPD0wKXticmVhayBnfWI9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2Y9RFthPj4yXTtlPURbZj4+Ml07aWYoKGJ8MCk+PShEW2YrND4+Ml0tZXwwKSl7YnJlYWsgYX1mPWIrZXwwO2I9MDt3aGlsZSgxKXtlPUZbZj4+MV07aWYoZT4+PjA+MjU1KXticmVhayBhfUJbYitkfDBdPWU7Yj1iKzF8MDtlPUJbYSsyNHwwXTtpZigoYnwwKT49KCgoY3wwKTwoZXwwKT9jOmUpfDApKXticmVhayBnfWY9ZisyfDA7aWYoZj4+PjA8R1tEW2E+PjJdKzQ+PjJdKXtjb250aW51ZX1icmVha31icmVhayBhfWc9MTtpZigoY3wwKT4oZXwwKSl7YnJlYWsgYn1icmVhayBhO2Nhc2UgNDpoOntlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk8PTApe2JyZWFrIGh9Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Zj1EW2E+PjJdO2U9RFtmPj4yXTtpZigoYnwwKT49KERbZis0Pj4yXS1lfDApKXticmVhayBhfWY9YitlfDA7Yj0wO3doaWxlKDEpe2U9RFtmPj4yXTtpZihlPj4+MD4yNTUpe2JyZWFrIGF9QltiK2R8MF09ZTtiPWIrMXwwO2U9QlthKzI0fDBdO2lmKChifDApPj0oKChjfDApPChlfDApP2M6ZSl8MCkpe2JyZWFrIGh9Zj1mKzR8MDtpZihmPj4+MDxHW0RbYT4+Ml0rND4+Ml0pe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGF9Zz0xO2lmKChjfDApPihlfDApKXticmVhayBifWJyZWFrIGE7Y2FzZSA1Omk6e2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKTw9MCl7YnJlYWsgaX1iPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtmPURbYT4+Ml07ZT1EW2Y+PjJdO2lmKChifDApPj0oRFtmKzQ+PjJdLWV8MCkpe2JyZWFrIGF9Zj1iK2V8MDtiPTA7d2hpbGUoMSl7ZT1EW2Y+PjJdO2lmKGU+Pj4wPjI1NSl7YnJlYWsgYX1CW2IrZHwwXT1lO2I9YisxfDA7ZT1CW2ErMjR8MF07aWYoKGJ8MCk+PSgoKGN8MCk8KGV8MCk/YzplKXwwKSl7YnJlYWsgaX1mPWYrNHwwO2lmKGY+Pj4wPEdbRFthPj4yXSs0Pj4yXSl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1nPTE7aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGJ9YnJlYWsgYTtjYXNlIDY6ajp7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPD0wKXticmVhayBqfWI9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2Y9RFthPj4yXTtlPURbZj4+Ml07aWYoKGJ8MCk+PShEW2YrND4+Ml0tZXwwKSl7YnJlYWsgYX1mPWIrZXwwO2I9MDt3aGlsZSgxKXtlPURbZis0Pj4yXTtoPURbZj4+Ml07aWYoIWUmaD4+PjA+MjU1fGUpe2JyZWFrIGF9QltiK2R8MF09aDtiPWIrMXwwO2U9QlthKzI0fDBdO2lmKChifDApPj0oKChjfDApPChlfDApP2M6ZSl8MCkpe2JyZWFrIGp9Zj1mKzh8MDtpZihmPj4+MDxHW0RbYT4+Ml0rND4+Ml0pe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGF9Zz0xO2lmKChjfDApPihlfDApKXticmVhayBifWJyZWFrIGE7Y2FzZSA3Oms6e2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKTw9MCl7YnJlYWsga31iPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtmPURbYT4+Ml07ZT1EW2Y+PjJdO2lmKChifDApPj0oRFtmKzQ+PjJdLWV8MCkpe2JyZWFrIGF9Zj1iK2V8MDtiPTA7d2hpbGUoMSl7ZT1EW2YrND4+Ml07aD1EW2Y+PjJdO2lmKCFlJmg+Pj4wPjI1NXxlKXticmVhayBhfUJbYitkfDBdPWg7Yj1iKzF8MDtlPUJbYSsyNHwwXTtpZigoYnwwKT49KCgoY3wwKTwoZXwwKT9jOmUpfDApKXticmVhayBrfWY9Zis4fDA7aWYoZj4+PjA8R1tEW2E+PjJdKzQ+PjJdKXtjb250aW51ZX1icmVha31icmVhayBhfWc9MTtpZigoY3wwKT4oZXwwKSl7YnJlYWsgYn1icmVhayBhO2Nhc2UgODpsOntlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk8PTApe2JyZWFrIGx9Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Zj1EW2E+PjJdO2U9RFtmPj4yXTtpZigoYnwwKT49KERbZis0Pj4yXS1lfDApKXticmVhayBhfWY9YitlfDA7Yj0wO3doaWxlKDEpe2g9YitkfDA7aT1IW2Y+PjJdO206e2lmKGk8Syg0Mjk0OTY3Mjk2KSZpPj1LKDApKXtlPX5+aT4+PjA7YnJlYWsgbX1lPTB9QltofDBdPWU7Yj1iKzF8MDtlPUJbYSsyNHwwXTtpZigoYnwwKT49KCgoY3wwKTwoZXwwKT9jOmUpfDApKXticmVhayBsfWY9Zis0fDA7aWYoZj4+PjA8R1tEW2E+PjJdKzQ+PjJdKXtjb250aW51ZX1icmVha31icmVhayBhfWc9MTtpZigoY3wwKT4oZXwwKSl7YnJlYWsgYn1icmVhayBhO2Nhc2UgOTpuOntlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk8PTApe2JyZWFrIG59Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Zj1EW2E+PjJdO2U9RFtmPj4yXTtpZigoYnwwKT49KERbZis0Pj4yXS1lfDApKXticmVhayBhfWY9YitlfDA7Yj0wO3doaWxlKDEpe2g9YitkfDA7aj1JW2Y+PjNdO286e2lmKGo8NDI5NDk2NzI5NiZqPj0wKXtlPX5+aj4+PjA7YnJlYWsgb31lPTB9QltofDBdPWU7Yj1iKzF8MDtlPUJbYSsyNHwwXTtpZigoYnwwKT49KCgoY3wwKTwoZXwwKT9jOmUpfDApKXticmVhayBufWY9Zis4fDA7aWYoZj4+PjA8R1tEW2E+PjJdKzQ+PjJdKXtjb250aW51ZX1icmVha31icmVhayBhfWc9MTtpZigoY3wwKT4oZXwwKSl7YnJlYWsgYn1icmVhayBhO2Nhc2UgMTA6YnJlYWsgYztkZWZhdWx0OmJyZWFrIGF9fXA6e2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKTw9MCl7YnJlYWsgcH1iPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtmPURbYT4+Ml07ZT1EW2Y+PjJdO2lmKChifDApPj0oRFtmKzQ+PjJdLWV8MCkpe2JyZWFrIGF9Zj1iK2V8MDtiPTA7d2hpbGUoMSl7QltiK2R8MF09RVtmfDBdO2I9YisxfDA7ZT1CW2ErMjR8MF07aWYoKGJ8MCk+PSgoKGN8MCk8KGV8MCk/YzplKXwwKSl7YnJlYWsgcH1mPWYrMXwwO2lmKGY+Pj4wPEdbRFthPj4yXSs0Pj4yXSl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1nPTE7aWYoKGN8MCk8PShlfDApKXticmVhayBhfX1wYShkK2V8MCwwLGMtZXwwKX1yZXR1cm4gZ31mdW5jdGlvbiBsZChhLGIpe3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTA7YTp7Yjp7Yzp7ZDp7ZTp7aWYoRFthKzkyPj4yXT09RFthKzg4Pj4yXSl7YnJlYWsgZX1jPURbYSs1Mj4+Ml07Zjp7aWYoKGN8MCkhPURbYSs1Nj4+Ml0pe0RbYz4+Ml09YjtEW2ErNTI+PjJdPWMrNDticmVhayBmfWc9RFthKzQ4Pj4yXTtoPWMtZ3wwO2Q9aD4+MjtlPWQrMXwwO2lmKGU+Pj4wPj0xMDczNzQxODI0KXticmVhayBifWM9aD4+MTtmPWQ+Pj4wPDUzNjg3MDkxMT9jPj4+MDxlPj4+MD9lOmM6MTA3Mzc0MTgyMztpZihmKXtpZihmPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgYX1jPW5hKGY8PDIpfWVsc2V7Yz0wfWU9YysoZDw8Mil8MDtEW2U+PjJdPWI7aWYoKGh8MCk+MCl7b2EoYyxnLGgpfURbYSs1Nj4+Ml09YysoZjw8Mik7RFthKzUyPj4yXT1lKzQ7RFthKzQ4Pj4yXT1jO2lmKCFnKXticmVhayBmfW1hKGcpfURbYSs4ND4+Ml09MDtlPS0xO2M9LTE7Zzp7aWYoKGJ8MCk9PS0xKXticmVhayBnfWY9RFthKzQ+PjJdO2M9YisxfDA7Yz0oYz4+PjApJTN8MD9jOmItMnwwO2lmKChjfDApIT0tMSl7ZT1EW0RbZj4+Ml0rKGM8PDIpPj4yXX1oOntpZigoYj4+PjApJTN8MCl7ZD1iLTF8MDticmVhayBofWQ9YisyfDA7Yz0tMTtpZigoZHwwKT09LTEpe2JyZWFrIGd9fWM9RFtEW2Y+PjJdKyhkPDwyKT4+Ml19aj1jPj4+MyY1MzY4NzA5MDg7ZD1EW2ErMzY+PjJdO2c9ZCsoZT4+PjMmNTM2ODcwOTA4KXwwO2g9RFtnPj4yXTtmPTE8PGU7aWYoIShoJmYpKXtEW2c+PjJdPWZ8aDtmPWErOHwwO2lmKChifDApIT0tMSl7ZD1iKzF8MDtkPShkPj4+MCklM3wwP2Q6Yi0yfDB9ZWxzZXtkPS0xfUlhKGYsZSxkKTtkPURbYSszNj4+Ml19Zj1kK2p8MDtkPURbZj4+Ml07ZT0xPDxjO2lmKCEoZCZlKSl7RFtmPj4yXT1kfGU7ZD1hKzh8MDtlPS0xO2k6e2lmKChifDApPT0tMSl7YnJlYWsgaX1lPWItMXwwO2lmKChiPj4+MCklM3wwKXticmVhayBpfWU9YisyfDB9SWEoZCxjLGUpfWU9LTE7ZT0oYnwwKSE9LTE/RFtEW0RbYSs0Pj4yXT4+Ml0rKGI8PDIpPj4yXTplO2Y9RFthKzM2Pj4yXSsoZT4+PjMmNTM2ODcwOTA4KXwwO2Q9RFtmPj4yXTtjPTE8PGU7aWYoIShkJmMpKXtEW2Y+PjJdPWN8ZDtJYShhKzh8MCxlLGIpfWQ9RFthKzg0Pj4yXTtpZigoZHwwKT4yKXticmVhayBlfXdoaWxlKDEpe2U9SihkLDEyKSthfDA7Yj1EW2UrNTI+PjJdO2lmKChifDApPT1EW2UrNDg+PjJdKXtkPWQrMXwwO2lmKChkfDApIT0zKXtjb250aW51ZX1icmVhayBlfWM9Yi00fDA7Yj1EW2M+PjJdO0RbZSs1Mj4+Ml09YztEW2ErODQ+PjJdPWQ7aWYoKGJ8MCk9PS0xKXticmVhayBlfWU9RFthKzI0Pj4yXTtjPShiPj4+MCkvM3wwO2o6e2lmKERbZSsoYz4+PjMmMjY4NDM1NDUyKT4+Ml0+Pj5jJjEpe2JyZWFrIGp9azp7d2hpbGUoMSl7bD0oYj4+PjApLzN8MDtjPShsPj4+MyYyNjg0MzU0NTIpK2V8MDtEW2M+PjJdPURbYz4+Ml18MTw8bDtlPS0xO2w6e206e246e286e3A6e3E6e3I6e3M6e2U9KGJ8MCkhPS0xP0RbRFtEW2ErND4+Ml0+PjJdKyhiPDwyKT4+Ml06ZTtmPURbYSszNj4+Ml0rKGU+Pj4zJjUzNjg3MDkwOCl8MDtkPURbZj4+Ml07Yz0xPDxlO2lmKCEoZCZjKSl7RFtmPj4yXT1jfGQ7aD1EWyhEW0RbYSsxNj4+Ml0rOTY+PjJdK0oobCwxMil8MCkrKChiPj4+MCklMzw8Mik+PjJdO2s9RFtEW2ErMjA+PjJdKzQ+PjJdO2M9RFtrKzQ+PjJdO3Q6e2lmKChjfDApIT1EW2srOD4+Ml0pe0RbYz4+Ml09aDtEW2srND4+Ml09Yys0O2JyZWFrIHR9aT1EW2s+PjJdO2o9Yy1pfDA7Zj1qPj4yO2Q9ZisxfDA7aWYoZD4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIHN9Yz1qPj4xO2c9Zj4+PjA8NTM2ODcwOTExP2M+Pj4wPGQ+Pj4wP2Q6YzoxMDczNzQxODIzO2lmKGcpe2lmKGc+Pj4wPj0xMDczNzQxODI0KXticmVhayBhfWM9bmEoZzw8Mil9ZWxzZXtjPTB9ZD1jKyhmPDwyKXwwO0RbZD4+Ml09aDtpZigoanwwKT4wKXtvYShjLGksail9RFtrKzg+PjJdPWMrKGc8PDIpO0Rbays0Pj4yXT1kKzQ7RFtrPj4yXT1jO2lmKCFpKXticmVhayB0fW1hKGkpfWk9RFthKzEyPj4yXTtjPURbaSs0Pj4yXTt1OntpZigoY3wwKSE9RFtpKzg+PjJdKXtEW2M+PjJdPWI7RFtpKzQ+PjJdPWMrNDticmVhayB1fWo9RFtpPj4yXTtnPWMtanwwO2Y9Zz4+MjtkPWYrMXwwO2lmKGQ+Pj4wPj0xMDczNzQxODI0KXticmVhayByfWM9Zz4+MTtoPWY+Pj4wPDUzNjg3MDkxMT9jPj4+MDxkPj4+MD9kOmM6MTA3Mzc0MTgyMztpZihoKXtpZihoPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgYX1jPW5hKGg8PDIpfWVsc2V7Yz0wfWQ9YysoZjw8Mil8MDtEW2Q+PjJdPWI7aWYoKGd8MCk+MCl7b2EoYyxqLGcpfURbaSs4Pj4yXT1jKyhoPDwyKTtEW2krND4+Ml09ZCs0O0RbaT4+Ml09YztpZighail7YnJlYWsgdX1tYShqKX1jPURbYSsxMj4+Ml07RFtEW2MrMTI+PjJdKyhlPDwyKT4+Ml09RFtjKzI0Pj4yXTtEW2MrMjQ+PjJdPURbYysyND4+Ml0rMX1pZigoYnwwKT09LTEpe2JyZWFrIGt9Zz1EW2ErND4+Ml07ZT0tMTtjPWIrMXwwO2M9KGM+Pj4wKSUzfDA/YzpiLTJ8MDtpZigoY3wwKSE9LTEpe2U9RFtEW2crMTI+PjJdKyhjPDwyKT4+Ml19djp7dzp7aWYoKEoobCwzKXwwKSE9KGJ8MCkpe2Y9Yi0xfDA7YnJlYWsgd31mPWIrMnwwO2I9LTE7aWYoKGZ8MCk9PS0xKXticmVhayB2fX1iPURbRFtnKzEyPj4yXSsoZjw8Mik+PjJdfWg9KGJ8MCk9PS0xO2Y9KGI+Pj4wKS8zfDA7ZD0oZT4+PjApLzN8MDtjPShlfDApPT0tMTtpZighYyl7Yz1jPy0xOmQ7Yz1EW0RbYSsyND4+Ml0rKGM+Pj4zJjUzNjg3MDkwOCk+PjJdJjE8PGM7aWYoaCl7YnJlYWsgcX1sPShjfDApIT0wO2JyZWFrIHB9bD0xO2lmKCFoKXticmVhayBwfWJyZWFrIGt9cWEoKTtUKCl9cWEoKTtUKCl9aWYoIWMpe2JyZWFrIG99YnJlYWsga31jPWg/LTE6Zjt4OntpZihEW0RbYSsyND4+Ml0rKGM+Pj4zJjUzNjg3MDkwOCk+PjJdPj4+YyYxKXticmVhayB4fWY9MDtjPURbRFtnPj4yXSsoYjw8Mik+PjJdO2lmKCEoRFtEW2ErMzY+PjJdKyhjPj4+MyY1MzY4NzA5MDgpPj4yXT4+PmMmMSkpe2M9RFthKzg4Pj4yXSsoYzw8Mil8MDtkPURbYz4+Ml07RFtjPj4yXT1kKzE7Zj0oZHwwKTw9MD8yOjF9aWYoRFthKzg0Pj4yXT49KGZ8MCk/bDowKXticmVhayBtfWs9SihmLDEyKSthfDA7Yz1EW2srNTI+PjJdO3k6e2lmKChjfDApIT1EW2srNTY+PjJdKXtEW2M+PjJdPWI7RFtrKzUyPj4yXT1jKzQ7YnJlYWsgeX1pPURbays0OD4+Ml07aj1jLWl8MDtoPWo+PjI7ZD1oKzF8MDtpZihkPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgZH1jPWo+PjE7Zz1oPj4+MDw1MzY4NzA5MTE/Yz4+PjA8ZD4+PjA/ZDpjOjEwNzM3NDE4MjM7aWYoZyl7aWYoZz4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGF9Yz1uYShnPDwyKX1lbHNle2M9MH1kPWMrKGg8PDIpfDA7RFtkPj4yXT1iO2lmKChqfDApPjApe29hKGMsaSxqKX1EW2srNDg+PjJdPWM7RFtrKzUyPj4yXT1kKzQ7RFtrKzU2Pj4yXT1jKyhnPDwyKTtpZighaSl7YnJlYWsgeX1tYShpKX1pZihEW2ErODQ+PjJdPD0oZnwwKSl7YnJlYWsgeH1EW2ErODQ+PjJdPWZ9aWYobCl7YnJlYWsga31iPS0xO2lmKChlfDApPT0tMSl7YnJlYWsgbn19Yj1EW0RbRFthKzQ+PjJdPj4yXSsoZTw8Mik+PjJdfWY9MDtpZighKERbRFthKzM2Pj4yXSsoYj4+PjMmNTM2ODcwOTA4KT4+Ml0+Pj5iJjEpKXtiPURbYSs4OD4+Ml0rKGI8PDIpfDA7Yz1EW2I+PjJdO0RbYj4+Ml09YysxO2Y9KGN8MCk8PTA/MjoxfWlmKERbYSs4ND4+Ml08KGZ8MCkpe2JyZWFrIGx9Yj1lfWU9RFthKzI0Pj4yXTtjb250aW51ZX1icmVha31pPUooZiwxMikrYXwwO2I9RFtpKzUyPj4yXTt6OntpZigoYnwwKSE9RFtpKzU2Pj4yXSl7RFtiPj4yXT1lO0RbaSs1Mj4+Ml09Yis0O2JyZWFrIHp9aj1EW2krNDg+PjJdO2c9Yi1qfDA7ZD1nPj4yO2M9ZCsxfDA7aWYoYz4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGN9Yj1nPj4xO2g9ZD4+PjA8NTM2ODcwOTExP2I+Pj4wPGM+Pj4wP2M6YjoxMDczNzQxODIzO2lmKGgpe2lmKGg+Pj4wPj0xMDczNzQxODI0KXticmVhayBhfWI9bmEoaDw8Mil9ZWxzZXtiPTB9Yz1iKyhkPDwyKXwwO0RbYz4+Ml09ZTtpZigoZ3wwKT4wKXtvYShiLGosZyl9RFtpKzQ4Pj4yXT1iO0RbaSs1Mj4+Ml09Yys0O0RbaSs1Nj4+Ml09YisoaDw8Mik7aWYoIWope2JyZWFrIHp9bWEoail9ZD1EW2ErODQ+PjJdO2lmKChkfDApPD0oZnwwKSl7YnJlYWsgan1EW2ErODQ+PjJdPWY7ZD1mO2JyZWFrIGp9ZD1EW2ErODQ+PjJdfWlmKChkfDApPDMpe2NvbnRpbnVlfWJyZWFrfX1yZXR1cm4gMX1xYSgpO1QoKX1xYSgpO1QoKX1xYSgpO1QoKX1yYSgxMzI2KTtUKCl9ZnVuY3Rpb24gc2QoYSxiKXt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTAsbz0wLHA9MCxxPTAscj0wLHM9MCx0PTAsdT0wLHY9MDtoPSQtNDh8MDskPWg7Yz1EW0RbYSs0Pj4yXSs0ND4+Ml07ZD1EW2ErOD4+Ml07Zj1EW2Q+PjJdO2Q9RFtkKzQ+PjJdO0RbaCs0MD4+Ml09MDtEW2grMzI+PjJdPTA7RFtoKzM2Pj4yXT0wO2Q9KGQtZj4+Mj4+PjApLzN8MDtqPURbYys5Nj4+Ml07Zj0oRFtjKzEwMD4+Ml0tanwwKS8xMnwwO2E6e2lmKGQ+Pj4wPmY+Pj4wKXtnYyhjKzk2fDAsZC1mfDAsaCszMnwwKTticmVhayBhfWlmKGQ+Pj4wPj1mPj4+MCl7YnJlYWsgYX1EW2MrMTAwPj4yXT1qK0ooZCwxMil9Yjp7aWYoRFthKzIxNj4+Ml09PURbYSsyMjA+PjJdKXtrPURbYSs0Pj4yXTtpPURbays0ND4+Ml07ZD1EW2krMTAwPj4yXTtpZigoZHwwKSE9RFtpKzk2Pj4yXSl7Yz0wO3doaWxlKDEpe2c9RFthKzg+PjJdO2U9SihjLDMpO2M6e2Q6e2lmKChlfDApPT0tMSl7bD1EWyhEW2c+PjJdKyhlPDwyKXwwKSs0Pj4yXTtqPS0xO2U9MTticmVhayBkfWw9LTE7aj1EW0RbZz4+Ml0rKGU8PDIpPj4yXTtmPWUrMXwwO2lmKChmfDApPT0tMSl7ZT0wO2JyZWFrIGR9bD1EW0RbZz4+Ml0rKGY8PDIpPj4yXTtlPWUrMnwwO2Y9LTE7aWYoKGV8MCk9PS0xKXticmVhayBjfX1mPURbRFtnPj4yXSsoZTw8Mik+PjJdfWc9ZjtmPWMrMXwwO2U9ZDtkPURbaSs5Nj4+Ml07ZT0oZS1kfDApLzEyfDA7aWYoZT4+PjA8PWM+Pj4wKXtEW2grNDA+PjJdPTA7RFtoKzMyPj4yXT0wO0RbaCszNj4+Ml09MDtnYyhpKzk2fDAsZi1lfDAsaCszMnwwKTtrPURbYSs0Pj4yXTtkPURbaSs5Nj4+Ml19Yz1KKGMsMTIpK2R8MDtEW2MrOD4+Ml09ZztEW2MrND4+Ml09bDtEW2M+PjJdPWo7Yz1mO2k9RFtrKzQ0Pj4yXTtkPURbaSsxMDA+PjJdO2lmKGM+Pj4wPChkLURbaSs5Nj4+Ml18MCkvMTI+Pj4wKXtjb250aW51ZX1icmVha319RFtEW2srND4+Ml0rODA+PjJdPWI7Yz0xO2JyZWFrIGJ9RFtoKzI0Pj4yXT0wO0RbaCsxNj4+Ml09MDtEW2grMjA+PjJdPTA7az1EW2ErOD4+Ml07Yj1EW2s+PjJdO2M9RFtrKzQ+PjJdO0RbaCs4Pj4yXT0wO0RbaD4+Ml09MDtEW2grND4+Ml09MDtlOntmOntnOntoOntpOntqOntrOntiPWMtYnwwO2lmKGIpe2lmKChifDApPDApe2JyZWFrIGt9bj1uYShiKTtEW2g+PjJdPW47RFtoKzg+PjJdPShiPj4yPDwyKStuO3U9aCx2PXBhKG4sMCxiKStifDAsRFt1KzQ+PjJdPXZ9Yz1EW2srMjQ+PjJdO2lmKChEW2srMjg+PjJdLWN8MCk8PTApe2Q9MDtiPTA7YnJlYWsgZn1kPTA7Zj0wO2I9MDt3aGlsZSgxKXtqPURbKG88PDIpK2M+PjJdO2w6e2lmKChqfDApPT0tMSl7YnJlYWsgbH1tOntpZihEW0RbYSsxMjA+PjJdKyhvPj4+MyY1MzY4NzA5MDgpPj4yXT4+Pm8mMSl7YnJlYWsgbX1xPURbYSsyMTY+PjJdO2M9RFthKzIyMD4+Ml0tcXwwO2lmKCFjKXticmVhayBtfWM9KGN8MCkvMTQ0fDA7cj1jPj4+MD4xP2M6MTtsPTA7Yz0oaj4+PjApJTN8MDtnPWorMnwwO3M9KGN8MCkhPTB8KGd8MCkhPS0xO3Q9Yz9qLTF8MDpnO3doaWxlKDEpe2k9ajw8MjtlPXErSihsLDE0NCl8MDtjPURbaStEW0RbZSs2OD4+Ml0+PjJdPj4yXTtuOntpZighKERbRFtlKzE2Pj4yXSsoYz4+PjMmNTM2ODcwOTA4KT4+Ml0+Pj5jJjEpKXticmVhayBufWM9LTE7bzp7aWYoIXMpe2JyZWFrIG99Zz1EW0RbaysxMj4+Ml0rKHQ8PDIpPj4yXTtjPS0xO2lmKChnfDApPT0tMSl7YnJlYWsgb31jPWctMXwwO2lmKChnPj4+MCklM3wwKXticmVhayBvfWM9ZysyfDB9aWYoKGp8MCk9PShjfDApKXticmVhayBufWU9RFtlKzMyPj4yXTtnPURbZStpPj4yXTt3aGlsZSgxKXttPTA7aWYoKGN8MCk9PS0xKXticmVhayBlfWlmKERbZSsoYzw8Mik+PjJdIT0oZ3wwKSl7aj1jO2JyZWFrIG19cDp7cTp7aWYoKGM+Pj4wKSUzfDApe2k9Yy0xfDA7YnJlYWsgcX1pPWMrMnwwO209LTE7aWYoKGl8MCk9PS0xKXticmVhayBwfX1jPURbRFtrKzEyPj4yXSsoaTw8Mik+PjJdO209LTE7aWYoKGN8MCk9PS0xKXticmVhayBwfW09Yy0xfDA7aWYoKGM+Pj4wKSUzfDApe2JyZWFrIHB9bT1jKzJ8MH1jPW07aWYoKGp8MCkhPShjfDApKXtjb250aW51ZX1icmVha319bD1sKzF8MDtpZigocnwwKSE9KGx8MCkpe2NvbnRpbnVlfWJyZWFrfX1jPWItZnwwO2c9Yz4+MjtEWyhqPDwyKStuPj4yXT1nO3I6e2lmKGI+Pj4wPHA+Pj4wKXtEW2I+PjJdPWo7Yj1iKzR8MDtEW2grMjA+PjJdPWI7YnJlYWsgcn1iPWcrMXwwO2lmKGI+Pj4wPj0xMDczNzQxODI0KXticmVhayBqfWQ9cC1mfDA7ZT1kPj4xO2I9ZD4+Mj4+PjA8NTM2ODcwOTExP2I+Pj4wPmU+Pj4wP2I6ZToxMDczNzQxODIzO2lmKGIpe2lmKGI+Pj4wPj0xMDczNzQxODI0KXticmVhayBpfWQ9bmEoYjw8Mil9ZWxzZXtkPTB9Zz1kKyhnPDwyKXwwO0RbZz4+Ml09ajtwPShiPDwyKStkfDA7Yj1nKzR8MDtpZigoY3wwKT4wKXtvYShkLGYsYyl9RFtoKzI0Pj4yXT1wO0RbaCsyMD4+Ml09YjtEW2grMTY+PjJdPWQ7aWYoZil7bWEoZik7az1EW2ErOD4+Ml19Zj1kfWlmKChqfDApPT0tMSl7YnJlYWsgbH1zOntpZigoaj4+PjApJTN8MCl7Yz1qLTF8MDticmVhayBzfWM9aisyfDA7aWYoKGN8MCk9PS0xKXticmVhayBsfX1jPURbRFtrKzEyPj4yXSsoYzw8Mik+PjJdO2lmKChjfDApPT0tMSl7YnJlYWsgbH1jPWMrKChjPj4+MCklM3wwPy0xOjIpfDA7aWYoKGN8MCk9PS0xKXticmVhayBsfWU9ajtpZigoY3wwKT09KGV8MCkpe2JyZWFrIGx9d2hpbGUoMSl7Zz1jO3Q6e3U6e2k9RFthKzIxNj4+Ml07Yz1EW2ErMjIwPj4yXS1pfDA7aWYoIWMpe2JyZWFrIHV9Yz0oY3wwKS8xNDR8MDtsPWM+Pj4wPjE/YzoxO2M9MDt3aGlsZSgxKXtxPURbKGkrSihjLDE0NCl8MCkrMzI+PjJdO3I9Zzw8MjtpZihEW3Ercj4+Ml09PURbcSsoZTw8Mik+PjJdKXtjPWMrMXwwO2lmKChsfDApIT0oY3wwKSl7Y29udGludWV9YnJlYWsgdX1icmVha31jPWItZHwwO2U9Yz4+MjtEW24rcj4+Ml09ZTtpZihiPj4+MDxwPj4+MCl7RFtiPj4yXT1nO2I9Yis0fDA7RFtoKzIwPj4yXT1iO2Y9ZDticmVhayB0fWI9ZSsxfDA7aWYoYj4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGh9Zj1wLWR8MDtpPWY+PjE7Yj1mPj4yPj4+MDw1MzY4NzA5MTE/Yj4+PjA+aT4+PjA/YjppOjEwNzM3NDE4MjM7aWYoYil7aWYoYj4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGd9Zj1uYShiPDwyKX1lbHNle2Y9MH1lPWYrKGU8PDIpfDA7RFtlPj4yXT1nO3A9KGI8PDIpK2Z8MDtiPWUrNHwwO2lmKChjfDApPjApe29hKGYsZCxjKX1EW2grMjQ+PjJdPXA7RFtoKzIwPj4yXT1iO0RbaCsxNj4+Ml09ZjtpZighZCl7ZD1mO2JyZWFrIHR9bWEoZCk7az1EW2ErOD4+Ml07ZD1mO2JyZWFrIHR9RFsoZzw8Mikrbj4+Ml09RFsoZTw8Mikrbj4+Ml19aWYoKGd8MCk9PS0xKXticmVhayBsfXY6e2lmKChnPj4+MCklM3wwKXtjPWctMXwwO2JyZWFrIHZ9Yz1nKzJ8MDtpZigoY3wwKT09LTEpe2JyZWFrIGx9fWM9RFtEW2srMTI+PjJdKyhjPDwyKT4+Ml07aWYoKGN8MCk9PS0xKXticmVhayBsfWM9YysoKGM+Pj4wKSUzfDA/LTE6Mil8MDtpZigoY3wwKT09LTEpe2JyZWFrIGx9ZT1nO2lmKChjfDApIT0oanwwKSl7Y29udGludWV9YnJlYWt9fW89bysxfDA7Yz1EW2srMjQ+PjJdO2lmKChvfDApPERbaysyOD4+Ml0tYz4+Mil7Y29udGludWV9YnJlYWt9YnJlYWsgZn1xYSgpO1QoKX1xYSgpO1QoKX1yYSgxMzI2KTtUKCl9cWEoKTtUKCl9cmEoMTMyNik7VCgpfW89RFthKzQ+PjJdO2k9RFtvKzQ0Pj4yXTtsPURbaSsxMDA+PjJdO2lmKChsfDApIT1EW2krOTY+PjJdKXtjPTA7d2hpbGUoMSl7Zj1jKzF8MDtnPUooYywxMik7aj1nK258MDtlPURbais4Pj4yXTtrPURbais0Pj4yXTtqPURbaj4+Ml07bT1jO2M9RFtpKzk2Pj4yXTtsPShsLWN8MCkvMTJ8MDtpZihtPj4+MD49bD4+PjApe0RbaCs0MD4+Ml09MDtEW2grMzI+PjJdPTA7RFtoKzM2Pj4yXT0wO2djKGkrOTZ8MCxmLWx8MCxoKzMyfDApO289RFthKzQ+PjJdO2M9RFtpKzk2Pj4yXX1jPWMrZ3wwO0RbYys4Pj4yXT1lO0RbYys0Pj4yXT1rO0RbYz4+Ml09ajtjPWY7aT1EW28rNDQ+PjJdO2w9RFtpKzEwMD4+Ml07aWYoYz4+PjA8KGwtRFtpKzk2Pj4yXXwwKS8xMj4+PjApe2NvbnRpbnVlfWJyZWFrfX1EW0Rbbys0Pj4yXSs4MD4+Ml09Yi1kPj4yO209MX1jPW07aWYobil7bWEobil9aWYoIWQpe2JyZWFrIGJ9RFtoKzIwPj4yXT1kO21hKGQpfSQ9aCs0OHwwO3JldHVybiBjfQ0KZnVuY3Rpb24gRmEoYSxiLGMsZCl7dmFyIGU9MCxmPTAsZz0wLGg9MCxpPUsoMCksaj0wLGs9MDthOntpZighZCl7YnJlYWsgYX1iOntjOntzd2l0Y2goRFthKzI4Pj4yXS0xfDApe2Nhc2UgMDplPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2c9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWcrYnwwO2g9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1oPj4+MCl7YnJlYWsgYX1lPShmPDwzKStkfDA7Zz1CW2J8MF07RFtlPj4yXT1nO0RbZSs0Pj4yXT1nPj4zMTtiPWIrMXwwO2Y9ZisxfDA7ZT1CW2ErMjR8MF07aWYoKGZ8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGJ9YnJlYWsgYTtjYXNlIDE6ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtnPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1nK2J8MDtoPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49aD4+PjApe2JyZWFrIGF9ZT0oZjw8MykrZHwwO0RbZT4+Ml09RVtifDBdO0RbZSs0Pj4yXT0wO2I9YisxfDA7Zj1mKzF8MDtlPUJbYSsyNHwwXTtpZigoZnwwKTwoKChjfDApPChlfDApP2M6ZSl8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgYn1icmVhayBhO2Nhc2UgMjplPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2c9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWcrYnwwO2g9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1oPj4+MCl7YnJlYWsgYX1lPShmPDwzKStkfDA7Zz1DW2I+PjFdO0RbZT4+Ml09ZztEW2UrND4+Ml09Zz4+MzE7Yj1iKzJ8MDtmPWYrMXwwO2U9QlthKzI0fDBdO2lmKChmfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPihlfDApKXticmVhayBifWJyZWFrIGE7Y2FzZSAzOmU9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zz1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZytifDA7aD1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWg+Pj4wKXticmVhayBhfWU9KGY8PDMpK2R8MDtEW2U+PjJdPUZbYj4+MV07RFtlKzQ+PjJdPTA7Yj1iKzJ8MDtmPWYrMXwwO2U9QlthKzI0fDBdO2lmKChmfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPihlfDApKXticmVhayBifWJyZWFrIGE7Y2FzZSA0OmU9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zz1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZytifDA7aD1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWg+Pj4wKXticmVhayBhfWU9KGY8PDMpK2R8MDtnPURbYj4+Ml07RFtlPj4yXT1nO0RbZSs0Pj4yXT1nPj4zMTtiPWIrNHwwO2Y9ZisxfDA7ZT1CW2ErMjR8MF07aWYoKGZ8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGJ9YnJlYWsgYTtjYXNlIDU6ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtnPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1nK2J8MDtoPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49aD4+PjApe2JyZWFrIGF9ZT0oZjw8MykrZHwwO0RbZT4+Ml09RFtiPj4yXTtEW2UrND4+Ml09MDtiPWIrNHwwO2Y9ZisxfDA7ZT1CW2ErMjR8MF07aWYoKGZ8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGJ9YnJlYWsgYTtjYXNlIDY6ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtnPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1nK2J8MDtoPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49aD4+PjApe2JyZWFrIGF9Zz1EW2IrND4+Ml07ZT0oZjw8MykrZHwwO0RbZT4+Ml09RFtiPj4yXTtEW2UrND4+Ml09ZztiPWIrOHwwO2Y9ZisxfDA7ZT1CW2ErMjR8MF07aWYoKGZ8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGJ9YnJlYWsgYTtjYXNlIDc6ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtnPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1nK2J8MDtoPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49aD4+PjApe2JyZWFrIGF9ZT1EW2I+PjJdO2c9RFtiKzQ+PjJdO2lmKChnfDApPDApe2JyZWFrIGF9az0oZjw8MykrZHwwO0Rbaz4+Ml09ZTtEW2srND4+Ml09ZztiPWIrOHwwO2Y9ZisxfDA7ZT1CW2ErMjR8MF07aWYoKGZ8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGJ9YnJlYWsgYTtjYXNlIDg6ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtnPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1nK2J8MDtrPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49az4+PjApe2JyZWFrIGF9ZT0oZjw8MykrZHwwO2k9SFtiPj4yXTtkOntpZihLKEwoaSkpPEsoMHg4MDAwMDAwMDAwMDAwMDAwKSl7Zz1LKEwoaSkpPj1LKDEpP35+KGk+SygwKT9LKE4oSyhQKEsoaSpLKDIuMzI4MzA2NDM2NTM4Njk2M2UtMTApKSkpLEsoNDI5NDk2NzI5NikpKTpLKFEoSyhLKGktSyh+fmk+Pj4wPj4+MCkpKksoMi4zMjgzMDY0MzY1Mzg2OTYzZS0xMCkpKSkpPj4+MDowO2g9fn5pPj4+MDticmVhayBkfWc9LTIxNDc0ODM2NDg7aD0wfURbZT4+Ml09aDtEW2UrND4+Ml09ZztiPWIrNHwwO2Y9ZisxfDA7ZT1CW2ErMjR8MF07aWYoKGZ8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGJ9YnJlYWsgYTtjYXNlIDk6ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtnPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1nK2J8MDtrPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49az4+PjApe2JyZWFrIGF9ZT0oZjw8MykrZHwwO2o9SVtiPj4zXTtlOntpZihMKGopPDB4ODAwMDAwMDAwMDAwMDAwMCl7Zz1MKGopPj0xP35+KGo+MD9OKFAoaioyLjMyODMwNjQzNjUzODY5NjNlLTEwKSw0Mjk0OTY3Mjk1KTpRKChqLSsofn5qPj4+MD4+PjApKSoyLjMyODMwNjQzNjUzODY5NjNlLTEwKSk+Pj4wOjA7aD1+fmo+Pj4wO2JyZWFrIGV9Zz0tMjE0NzQ4MzY0ODtoPTB9RFtlPj4yXT1oO0RbZSs0Pj4yXT1nO2I9Yis4fDA7Zj1mKzF8MDtlPUJbYSsyNHwwXTtpZigoZnwwKTwoKChjfDApPChlfDApP2M6ZSl8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgYn1icmVhayBhO2Nhc2UgMTA6YnJlYWsgYztkZWZhdWx0OmJyZWFrIGF9fWU9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zz1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZytifDA7aD1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWg+Pj4wKXticmVhayBhfWU9KGY8PDMpK2R8MDtEW2U+PjJdPUVbYnwwXTtEW2UrND4+Ml09MDtiPWIrMXwwO2Y9ZisxfDA7ZT1CW2ErMjR8MF07aWYoKGZ8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk8PShlfDApKXticmVhayBhfX1wYSgoZTw8MykrZHwwLDAsYy1lPDwzKX19ZnVuY3Rpb24gQ2coYSl7YT1hfDA7dmFyIGI9MCxjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MCxvPTA7ZD0kLTMyfDA7JD1kO2E6e2lmKCEkYSgxLGQrMjh8MCxEW2ErMzI+PjJdKSl7YnJlYWsgYX1pZighJGEoMSxkKzI0fDAsRFthKzMyPj4yXSkpe2JyZWFrIGF9az1EW2QrMjg+PjJdO2lmKGs+Pj4wPjE0MzE2NTU3NjUpe2JyZWFrIGF9ZT1EW2ErMzI+PjJdO2o9RFtlKzg+PjJdO2I9ajtjPURbZSsxNj4+Ml07Zj1iLWN8MDtnPURbZSsxMj4+Ml07bj1iPj4+MDxjPj4+MDtiPURbZSsyMD4+Ml07Zj1haShmLGctKG4rYnwwKXwwLDMsMCk7aWYoIWFhJmY+Pj4wPGs+Pj4wKXticmVhayBhfW89RFtkKzI0Pj4yXTtmPSRoKGssMCwzLDApO2lmKCFhYSZmPj4+MDxvPj4+MHwoKGJ8MCk+PShnfDApJmM+Pj4wPj1qPj4+MHwoYnwwKT4oZ3wwKSkpe2JyZWFrIGF9Zz1FW2MrRFtlPj4yXXwwXTtqPWMrMXwwO2Y9aj9iOmIrMXwwO0RbZSsxNj4+Ml09ajtEW2UrMjA+PjJdPWY7Yjp7aWYoIWcpe2U9MDtiPSQtMzJ8MDskPWI7RFtiKzI0Pj4yXT0wO0RbYisxNj4+Ml09MDtEW2IrMjA+PjJdPTA7Yzp7ZDp7Zj1KKGssMyk7aWYoZil7aWYoZj4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGR9Yz1KKGssMTIpO2U9bmEoYyk7RFtiKzE2Pj4yXT1lO3BhKGUsMCxjKX1mPWpjKGYsMSxEW2ErMzI+PjJdLGUpO2U6e2Y6e2lmKCEoIWZ8IWspKXtnPTA7Yz0wO3doaWxlKDEpe249aDtqPShjPDwyKStlfDA7aD1EW2o+PjJdO2k9aD4+PjF8MDtoPW4rKGgmMT8wLWl8MDppKXwwO0RbYj4+Ml09aDtpPURbais0Pj4yXTtsPWk+Pj4xfDA7aD1oKyhpJjE/MC1sfDA6bCl8MDtEW2IrND4+Ml09aDtqPURbais4Pj4yXTtpPWo+Pj4xfDA7aD1oKyhqJjE/MC1pfDA6aSl8MDtEW2IrOD4+Ml09aDtuYihEW2ErNDQ+PjJdKzk2fDAsYik7Yz1jKzN8MDtnPWcrMXwwO2lmKChnfDApIT0oa3wwKSl7Y29udGludWV9YnJlYWt9YnJlYWsgZn1pZighZSl7YnJlYWsgZX19bWEoZSl9JD1iKzMyfDA7YnJlYWsgY31xYSgpO1QoKX1pZighZil7YnJlYWsgYX1icmVhayBifWc6e2lmKG8+Pj4wPD0yNTUpe2lmKCFrKXticmVhayBifURbZCsxNj4+Ml09MDtEW2QrOD4+Ml09MDtEW2QrMTI+PjJdPTA7Yz1EW2UrMTI+PjJdO2I9YztoPURbZSs4Pj4yXTtpZigoZnwwKT49KGJ8MCkmaj4+PjA+PWg+Pj4wfChifDApPChmfDApKXticmVhayBnfXdoaWxlKDEpe2c9RFtlPj4yXTtsPUVbZytqfDBdO2I9ZjtpPWorMXwwO2I9aT9iOmIrMXwwO0RbZSsxNj4+Ml09aTtEW2UrMjA+PjJdPWI7RFtkKzg+PjJdPWw7aWYoaD4+PjA8PWk+Pj4wJihifDApPj0oY3wwKXwoYnwwKT4oY3wwKSl7YnJlYWsgZ31sPUVbZytpfDBdO2I9ZjtpPWorMnwwO2I9aT4+PjA8Mj9iKzF8MDpiO0RbZSsxNj4+Ml09aTtEW2UrMjA+PjJdPWI7RFtkKzEyPj4yXT1sO2lmKGg+Pj4wPD1pPj4+MCYoYnwwKT49KGN8MCl8KGJ8MCk+KGN8MCkpe2JyZWFrIGd9Yj1FW2craXwwXTtjPWorM3wwO2Y9Yz4+PjA8Mz9mKzF8MDpmO0RbZSsxNj4+Ml09YztEW2UrMjA+PjJdPWY7RFtkKzE2Pj4yXT1iO25iKERbYSs0ND4+Ml0rOTZ8MCxkKzh8MCk7bT1tKzF8MDtpZigoa3wwKT09KG18MCkpe2JyZWFrIGJ9ZT1EW2ErMzI+PjJdO2I9ZTtqPURbYisxNj4+Ml07Zj1EW2IrMjA+PjJdO0RbZCsxNj4+Ml09MDtEW2QrOD4+Ml09MDtEW2QrMTI+PjJdPTA7aD1EW2IrOD4+Ml07Yz1EW2IrMTI+PjJdO2I9YztpZihqPj4+MDxoPj4+MCYoZnwwKTw9KGJ8MCl8KGJ8MCk+KGZ8MCkpe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGd9aWYobz4+PjA8PTY1NTM1KXtpZighayl7YnJlYWsgYn1EW2QrMTY+PjJdPTA7RFtkKzg+PjJdPTA7RFtkKzEyPj4yXT0wO2c9RFtlKzEyPj4yXTtjPWMrM3wwO2I9Yz4+PjA8Mz9iKzF8MDpiO2k9RFtlKzg+PjJdO2g9YztjPWI7aWYoaT4+PjA8aD4+PjAmKGJ8MCk+PShnfDApfChifDApPihnfDApKXticmVhayBnfXdoaWxlKDEpe2w9RFtlPj4yXTtiPWwranwwO2I9RVtifDBdfEVbYisxfDBdPDw4O0RbZSsxNj4+Ml09aDtEW2UrMjA+PjJdPWM7RFtkKzg+PjJdPWI7Yj1mO2M9ais0fDA7Yj1jPj4+MDw0P2IrMXwwOmI7aWYoYz4+PjA+aT4+PjAmKGJ8MCk+PShnfDApfChifDApPihnfDApKXticmVhayBnfWg9aCtsfDA7aD1FW2h8MF18RVtoKzF8MF08PDg7RFtlKzE2Pj4yXT1jO0RbZSsyMD4+Ml09YjtEW2QrMTI+PjJdPWg7Yj1mO2Y9ais2fDA7Yj1mPj4+MDw2P2IrMXwwOmI7aWYoZj4+PjA+aT4+PjAmKGJ8MCk+PShnfDApfChifDApPihnfDApKXticmVhayBnfWM9YytsfDA7Yz1FW2N8MF18RVtjKzF8MF08PDg7RFtlKzE2Pj4yXT1mO0RbZSsyMD4+Ml09YjtEW2QrMTY+PjJdPWM7bmIoRFthKzQ0Pj4yXSs5NnwwLGQrOHwwKTttPW0rMXwwO2lmKChrfDApPT0obXwwKSl7YnJlYWsgYn1lPURbYSszMj4+Ml07Yj1lO2o9RFtiKzE2Pj4yXTtmPURbYisyMD4+Ml07RFtkKzE2Pj4yXT0wO0RbZCs4Pj4yXT0wO0RbZCsxMj4+Ml09MDtnPURbYisxMj4+Ml07aT1EW2IrOD4+Ml07Yj1mO2M9aisyfDA7Yj1jPj4+MDwyP2IrMXwwOmI7aD1jO2M9YjtpZigoYnwwKTw9KGd8MCkmaD4+PjA8PWk+Pj4wfChifDApPChnfDApKXtjb250aW51ZX1icmVha31icmVhayBnfWg6e2lmKEdbRFthKzQ0Pj4yXSs4MD4+Ml0+MjA5NzE1MSl7YnJlYWsgaH1nPUZbYSszNj4+MV07aWYoKChnPDw4fGc+Pj44KSY2NTUzNSk+Pj4wPDUxNCl7YnJlYWsgaH1pZighayl7YnJlYWsgYn1EW2QrMTY+PjJdPTA7RFtkKzg+PjJdPTA7RFtkKzEyPj4yXT0wO2lmKCEkYSgxLGQrNHwwLGUpKXticmVhayBnfXdoaWxlKDEpe0RbZCs4Pj4yXT1EW2QrND4+Ml07aWYoISRhKDEsZCs0fDAsRFthKzMyPj4yXSkpe2JyZWFrIGd9RFtkKzEyPj4yXT1EW2QrND4+Ml07aWYoISRhKDEsZCs0fDAsRFthKzMyPj4yXSkpe2JyZWFrIGd9RFtkKzE2Pj4yXT1EW2QrND4+Ml07bmIoRFthKzQ0Pj4yXSs5NnwwLGQrOHwwKTttPW0rMXwwO2lmKChrfDApPT0obXwwKSl7YnJlYWsgYn1mPURbYSszMj4+Ml07RFtkKzE2Pj4yXT0wO0RbZCs4Pj4yXT0wO0RbZCsxMj4+Ml09MDtpZigkYSgxLGQrNHwwLGYpKXtjb250aW51ZX1icmVha31icmVhayBnfWlmKCFrKXticmVhayBifURbZCsxNj4+Ml09MDtEW2QrOD4+Ml09MDtEW2QrMTI+PjJdPTA7Zz1EW2UrMTI+PjJdO2M9Yys1fDA7Yj1jPj4+MDw1P2IrMXwwOmI7aT1EW2UrOD4+Ml07aD1jO2M9YjtpZihpPj4+MDxoPj4+MCYoYnwwKT49KGd8MCl8KGJ8MCk+KGd8MCkpe2JyZWFrIGd9d2hpbGUoMSl7bD1EW2U+PjJdO2I9bCtqfDA7Yj1FW2J8MF18RVtiKzF8MF08PDh8KEVbYisyfDBdPDwxNnxFW2IrM3wwXTw8MjQpO0RbZSsxNj4+Ml09aDtEW2UrMjA+PjJdPWM7RFtkKzg+PjJdPWI7Yj1mO2M9ais4fDA7Yj1jPj4+MDw4P2IrMXwwOmI7bj1jO2lmKGM+Pj4wPmk+Pj4wJihifDApPj0oZ3wwKXwoYnwwKT4oZ3wwKSl7YnJlYWsgZ31jPWgrbHwwO2M9RVtjfDBdfEVbYysxfDBdPDw4fChFW2MrMnwwXTw8MTZ8RVtjKzN8MF08PDI0KTtEW2UrMTY+PjJdPW47RFtlKzIwPj4yXT1iO0RbZCsxMj4+Ml09YztiPWorMTJ8MDtmPWI+Pj4wPDEyP2YrMXwwOmY7Yz1iO2lmKGI+Pj4wPmk+Pj4wJihmfDApPj0oZ3wwKXwoZnwwKT4oZ3wwKSl7YnJlYWsgZ31iPWwrbnwwO2I9RVtifDBdfEVbYisxfDBdPDw4fChFW2IrMnwwXTw8MTZ8RVtiKzN8MF08PDI0KTtEW2UrMTY+PjJdPWM7RFtlKzIwPj4yXT1mO0RbZCsxNj4+Ml09YjtuYihEW2ErNDQ+PjJdKzk2fDAsZCs4fDApO209bSsxfDA7aWYoKGt8MCk9PShtfDApKXticmVhayBifWU9RFthKzMyPj4yXTtiPWU7aj1EW2IrMTY+PjJdO2Y9RFtiKzIwPj4yXTtEW2QrMTY+PjJdPTA7RFtkKzg+PjJdPTA7RFtkKzEyPj4yXT0wO2c9RFtiKzEyPj4yXTtpPURbYis4Pj4yXTtiPWY7Yz1qKzR8MDtiPWM+Pj4wPDQ/YisxfDA6YjtoPWM7Yz1iO2lmKChifDApPD0oZ3wwKSZoPj4+MDw9aT4+PjB8KGJ8MCk8KGd8MCkpe2NvbnRpbnVlfWJyZWFrfX1tPTA7YnJlYWsgYX1EW0RbYSs0Pj4yXSs4MD4+Ml09bzttPTF9JD1kKzMyfDA7cmV0dXJuIG18MH1mdW5jdGlvbiBMYihhLGIsYyl7YT1hfDA7Yj1ifDA7Yz1jfDA7dmFyIGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTAsbz0wLHA9MCxxPTAscj0wLHM9MCx0PTAsdT0wLHY9MCx3PTAseD0wO2Y9JC05NnwwOyQ9ZjtlPURbYSsxNj4+Ml07QltmKzkyfDBdPTE7RFtmKzg4Pj4yXT1iO0RbZis4ND4+Ml09YjtEW2YrODA+PjJdPWU7bT1EW2ErMjA+PjJdO2Q9RFttPj4yXTthOntiOntlPURbRFtlKzI4Pj4yXSsoYjw8Mik+PjJdO2lmKGU+Pj4wPERbbSs0Pj4yXS1kPj4yPj4+MCl7ZD1EW0RbYSs4Pj4yXSsoRFtkKyhlPDwyKT4+Ml08PDIpPj4yXTtlPURbYSs0Pj4yXTtpZighRVtlKzg0fDBdKXtkPURbRFtlKzY4Pj4yXSsoZDw8Mik+PjJdfURbZis3Mj4+Ml09MDtEW2YrNzY+PjJdPTA7bT1mLSAtNjR8MDtEW20+PjJdPTA7RFttKzQ+PjJdPTA7RFtmKzU2Pj4yXT0wO0RbZis2MD4+Ml09MDtGYShlLGQsQltlKzI0fDBdLGYrNTZ8MCk7ZT1iKzF8MDtkPShlPj4+MCklM3wwO2lmKGI+Pj4wPD1lPj4+MCl7bT1kP2U6Yi0yfDA7aD0oKGI+Pj4wKSUzfDA/LTE6MikrYnwwO3doaWxlKDEpe2Q9bTtlPWg7Yzp7aWYoIURbYSsyOD4+Ml0pe2JyZWFrIGN9ZT1iKzF8MDtkPShlPj4+MCklM3wwP2U6Yi0yfDA7ZT1iLTF8MDtpZigoYj4+PjApJTN8MCl7YnJlYWsgY31lPWIrMnwwfWs9RFthKzIwPj4yXTtiPURbaz4+Ml07ZD1EW0RbRFthKzE2Pj4yXSsyOD4+Ml0rKGQ8PDIpPj4yXTtpZihkPj4+MD49RFtrKzQ+PjJdLWI+PjI+Pj4wKXticmVhayBifWQ9RFtEW2ErOD4+Ml0rKERbYisoZDw8Mik+PjJdPDwyKT4+Ml07Yj1EW2ErND4+Ml07aWYoIUVbYis4NHwwXSl7ZD1EW0RbYis2OD4+Ml0rKGQ8PDIpPj4yXX1EW2YrNDg+PjJdPTA7RFtmKzUyPj4yXT0wO0RbZis0MD4+Ml09MDtEW2YrNDQ+PjJdPTA7RFtmKzMyPj4yXT0wO0RbZiszNj4+Ml09MDtGYShiLGQsQltiKzI0fDBdLGYrMzJ8MCk7ZD1EW2ErMjA+PjJdO2I9RFtkPj4yXTtlPURbRFtEW2ErMTY+PjJdKzI4Pj4yXSsoZTw8Mik+PjJdO2lmKGU+Pj4wPj1EW2QrND4+Ml0tYj4+Mj4+PjApe2JyZWFrIGF9ZD1EW0RbYSs4Pj4yXSsoRFtiKyhlPDwyKT4+Ml08PDIpPj4yXTtiPURbYSs0Pj4yXTtpZighRVtiKzg0fDBdKXtkPURbRFtiKzY4Pj4yXSsoZDw8Mik+PjJdfURbZisyND4+Ml09MDtEW2YrMjg+PjJdPTA7RFtmKzE2Pj4yXT0wO0RbZisyMD4+Ml09MDtEW2YrOD4+Ml09MDtEW2YrMTI+PjJdPTA7RmEoYixkLEJbYisyNHwwXSxmKzh8MCk7ZT1EW2YrOD4+Ml07Yj1EW2YrNTY+PjJdO2Q9ZS1ifDA7az1EW2YrNjA+PjJdO249RFtmKzEyPj4yXS0oaysoYj4+PjA+ZT4+PjApfDApfDA7aT1EW2YrNDA+PjJdO2U9RFtmKzY0Pj4yXTtzPWktZXwwO3Q9RFtmKzY4Pj4yXTtpPURbZis0ND4+Ml0tKHQrKGU+Pj4wPmk+Pj4wKXwwKXwwO3U9JGgoZCxuLHMsaSk7dj1qLXV8MDtnPWctKGFhKyhqPj4+MDx1Pj4+MCl8MCl8MDt3PXY7aj1EW2YrMTY+PjJdO3U9ai1lfDA7dD1EW2YrMjA+PjJdLSgoZT4+PjA+aj4+PjApK3R8MCl8MDtqPURbZiszMj4+Ml07dj1qLWJ8MDtrPURbZiszNj4+Ml0tKChiPj4+MD5qPj4+MCkra3wwKXwwO2U9JGgodSx0LHYsayk7aj13K2V8MDtiPWFhK2d8MDtiPWU+Pj4wPmo+Pj4wP2IrMXwwOmI7Zz1iO3c9bztwPW47Yj1EW2YrNDg+PjJdO2U9RFtmKzcyPj4yXTtuPWItZXwwO289RFtmKzc2Pj4yXTt4PURbZis1Mj4+Ml0tKG8rKGI+Pj4wPGU+Pj4wKXwwKXwwO3A9JGgoZCxwLG4seCk7ZD13K3B8MDtiPWFhK2x8MDtiPWQ+Pj4wPHA+Pj4wP2IrMXwwOmI7bD1EW2YrMjQ+PjJdO3A9bC1lfDA7ZT1EW2YrMjg+PjJdLSgoZT4+PjA+bD4+PjApK298MCl8MDtsPSRoKHAsZSx2LGspO289ZC1sfDA7bD1iLShhYSsoZD4+PjA8bD4+PjApfDApfDA7Yj0kaCh1LHQsbix4KTtkPXEtYnwwO2I9ci0oYWErKGI+Pj4wPnE+Pj4wKXwwKXwwO3I9JGgocCxlLHMsaSk7cT1yK2R8MDtiPWFhK2J8MDtiPXE+Pj4wPHI+Pj4wP2IrMXwwOmI7cj1iO2I9RFtmKzg4Pj4yXTtlPURbZis4MD4+Ml07ZDp7aWYoRVtmKzkyfDBdKXtlOntmOntnOntoOntpZigoYnwwKT09LTEpe2JyZWFrIGh9ZD1iKzF8MDtiPShkPj4+MCklM3wwP2Q6Yi0yfDA7aWYoKGJ8MCk9PS0xfERbRFtlPj4yXSsoYj4+PjMmNTM2ODcwOTA4KT4+Ml0+Pj5iJjEpe2JyZWFrIGh9Yj1EW0RbRFtlKzY0Pj4yXSsxMj4+Ml0rKGI8PDIpPj4yXTtpZigoYnwwKSE9LTEpe2JyZWFrIGd9fURbZis4OD4+Ml09LTE7YnJlYWsgZn1kPWIrMXwwO2I9KGQ+Pj4wKSUzfDA/ZDpiLTJ8MDtEW2YrODg+PjJdPWI7aWYoKGJ8MCkhPS0xKXticmVhayBlfX1iPURbZis4ND4+Ml07ZD0tMTtpOntpZigoYnwwKT09LTEpe2JyZWFrIGl9ajp7aWYoKGI+Pj4wKSUzfDApe2I9Yi0xfDA7YnJlYWsgan1iPWIrMnwwO2Q9LTE7aWYoKGJ8MCk9PS0xKXticmVhayBpfX1kPS0xO2lmKERbRFtlPj4yXSsoYj4+PjMmNTM2ODcwOTA4KT4+Ml0+Pj5iJjEpe2JyZWFrIGl9Yj1EW0RbRFtlKzY0Pj4yXSsxMj4+Ml0rKGI8PDIpPj4yXTtkPS0xO2lmKChifDApPT0tMSl7YnJlYWsgaX1kPWItMXwwO2lmKChiPj4+MCklM3wwKXticmVhayBpfWQ9YisyfDB9QltmKzkyfDBdPTA7RFtmKzg4Pj4yXT1kO2JyZWFrIGR9aWYoKGJ8MCkhPURbZis4ND4+Ml0pe2JyZWFrIGR9RFtmKzg4Pj4yXT0tMTticmVhayBkfWQ9LTE7azp7aWYoKGJ8MCk9PS0xKXticmVhayBrfWw6e2lmKChiPj4+MCklM3wwKXtiPWItMXwwO2JyZWFrIGx9Yj1iKzJ8MDtkPS0xO2lmKChifDApPT0tMSl7YnJlYWsga319ZD0tMTtpZihEW0RbZT4+Ml0rKGI+Pj4zJjUzNjg3MDkwOCk+PjJdPj4+YiYxKXticmVhayBrfWI9RFtEW0RbZSs2ND4+Ml0rMTI+PjJdKyhiPDwyKT4+Ml07ZD0tMTtpZigoYnwwKT09LTEpe2JyZWFrIGt9ZD1iLTF8MDtpZigoYj4+PjApJTN8MCl7YnJlYWsga31kPWIrMnwwfURbZis4OD4+Ml09ZH1iPURbZis4OD4+Ml07aWYoKGJ8MCkhPS0xKXtjb250aW51ZX1icmVha319Yj1yPj4zMTtkPWIrcXwwO2U9YjtiPWIrcnwwO2s9ZF5lO2g9ZV4oZD4+PjA8ZT4+PjA/YisxfDA6Yik7bj0tMTtkPTIxNDc0ODM2NDc7Yj1sPj4zMTtpPWI7ZT1iK298MDtiPWIrbHwwO2I9ZT4+PjA8aT4+PjA/YisxfDA6YjtlPWVeaTtiPWJeaTtpPWI7cz1lXi0xO2I9Yl4yMTQ3NDgzNjQ3O209ZzttOntuOntpZighRFthKzI4Pj4yXSl7aWYoKGJ8MCk9PShofDApJms+Pj4wPnM+Pj4wfGI+Pj4wPGg+Pj4wKXticmVhayBtfWI9aCtpfDA7YT1lK2t8MDtiPWE+Pj4wPGU+Pj4wP2IrMXwwOmI7ZT1hO2E9YjtiPWc+PjMxO2Q9YitqfDA7aD1nO2c9YjtiPWgrYnwwO2I9ZD4+PjA8Zz4+PjA/YisxfDA6YjtoPWReZztkPWgrZXwwO2c9Yl5nO2I9ZDtnPWdeMjE0NzQ4MzY0NzthPShnfDApPT0oYXwwKSYoaF4tMSk+Pj4wPGU+Pj4wfGE+Pj4wPmc+Pj4wO2c9IShhJjApO2E9YT8tMTpiO2lmKGcmKGF8MCk8PTUzNjg3MDkxMnwoYXwwKTw1MzY4NzA5MTIpe2JyZWFrIG19Yj0wO2E9YT4+PjI5fDA7YnJlYWsgbn1vOntpZigoYnwwKT09KGh8MCkmaz4+PjA+cz4+PjB8Yj4+PjA8aD4+PjApe2JyZWFrIG99Yj1oK2l8MDthPWUra3wwO2I9YT4+PjA8ZT4+PjA/YisxfDA6YjtlPWI7aD1nO2I9Zz4+MzE7Zz1iK2p8MDtpPWg7aD1iO2I9aStifDA7Yj1nPj4+MDxoPj4+MD9iKzF8MDpiO2c9Z15oO2I9Yl5oO2g9Yl4yMTQ3NDgzNjQ3O2lmKChofDApPT0oZXwwKSYoZ14tMSk+Pj4wPGE+Pj4wfGU+Pj4wPmg+Pj4wKXticmVhayBvfWI9YitlfDA7YT1hK2d8MDtiPWE+Pj4wPGc+Pj4wP2IrMXwwOmI7bj1hO2Q9YjtpZighYiZhPj4+MDw1MzY4NzA5MTMpe2JyZWFrIG19fWI9ZD4+PjI5fDA7YT0oZCY1MzY4NzA5MTEpPDwzfG4+Pj4yOX1qPWFpKGosbSxhLGIpO289YWkobyxsLGEsYik7cT1haShxLHIsYSxiKX1EW2MrOD4+Ml09ajtEW2MrND4+Ml09bztEW2M+PjJdPXE7JD1mKzk2fDA7cmV0dXJufXVhKCk7VCgpfXVhKCk7VCgpfXVhKCk7VCgpfWZ1bmN0aW9uIEJiKGEsYixjLGQpe3ZhciBlPTAsZj0wLGc9MCxoPTAsaT0wLGo9SygwKSxrPTA7YTp7Yjp7aWYoIWQpe2JyZWFrIGJ9Yzp7ZDp7c3dpdGNoKERbYSsyOD4+Ml0tMXwwKXtjYXNlIDA6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfUNbKGc8PDEpK2Q+PjFdPUJbYnwwXTtiPWIrMXwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDE6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfUNbKGc8PDEpK2Q+PjFdPUVbYnwwXTtiPWIrMXwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDI6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfUNbKGc8PDEpK2Q+PjFdPUZbYj4+MV07Yj1iKzJ8MDtnPWcrMXwwO2U9QlthKzI0fDBdO2lmKChnfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPihlfDApKXticmVhayBjfWJyZWFrIGI7Y2FzZSAzOmU9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfWU9Q1tiPj4xXTtpZigoZXwwKTwwKXticmVhayBifUNbKGc8PDEpK2Q+PjFdPWU7Yj1iKzJ8MDtnPWcrMXwwO2U9QlthKzI0fDBdO2lmKChnfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWg9MTtpZigoY3wwKT4oZXwwKSl7YnJlYWsgY31icmVhayBiO2Nhc2UgNDpoPTE7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtmPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1mK2J8MDtmPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49Zj4+PjApe2JyZWFrIGF9ZT1EW2I+PjJdO2lmKGUtMzI3Njg+Pj4wPDQyOTQ5MDE3NjApe2JyZWFrIGF9Q1soZzw8MSkrZD4+MV09ZTtiPWIrNHwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDU6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfWU9RFtiPj4yXTtpZihlPj4+MD4zMjc2Nyl7YnJlYWsgYX1DWyhnPDwxKStkPj4xXT1lO2I9Yis0fDA7Zz1nKzF8MDtlPUJbYSsyNHwwXTtpZigoZ3wwKTwoKChjfDApPChlfDApP2M6ZSl8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgY31icmVhayBiO2Nhc2UgNjpoPTE7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtmPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1mK2J8MDtmPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49Zj4+PjApe2JyZWFrIGF9aT1EW2I+PjJdO2U9RFtiKzQ+PjJdLShpPj4+MDwzMjc2OCl8MDtpZigoZXwwKT09LTEmaS0zMjc2OD4+PjA8NDI5NDkwMTc2MHwoZXwwKSE9LTEpe2JyZWFrIGF9Q1soZzw8MSkrZD4+MV09aTtiPWIrOHwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDc6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfWU9RFtiKzQ+PjJdO2k9RFtiPj4yXTtpZighZSZpPj4+MD4zMjc2N3xlKXticmVhayBhfUNbKGc8PDEpK2Q+PjFdPWk7Yj1iKzh8MDtnPWcrMXwwO2U9QlthKzI0fDBdO2lmKChnfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPihlfDApKXticmVhayBjfWJyZWFrIGI7Y2FzZSA4Omg9MTtlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2Y9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWYrYnwwO2Y9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1mPj4+MCl7YnJlYWsgYX1pPShnPDwxKStkfDA7aj1IW2I+PjJdO2U6e2lmKEsoTChqKSk8SygyMTQ3NDgzNjQ4KSl7ZT1+fmo7YnJlYWsgZX1lPS0yMTQ3NDgzNjQ4fUNbaT4+MV09ZTtiPWIrNHwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDk6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfWk9KGc8PDEpK2R8MDtrPUlbYj4+M107Zjp7aWYoTChrKTwyMTQ3NDgzNjQ4KXtlPX5+azticmVhayBmfWU9LTIxNDc0ODM2NDh9Q1tpPj4xXT1lO2I9Yis4fDA7Zz1nKzF8MDtlPUJbYSsyNHwwXTtpZigoZ3wwKTwoKChjfDApPChlfDApP2M6ZSl8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgY31icmVhayBiO2Nhc2UgMTA6YnJlYWsgZDtkZWZhdWx0OmJyZWFrIGJ9fWg9MTtlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2Y9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWYrYnwwO2Y9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1mPj4+MCl7YnJlYWsgYX1DWyhnPDwxKStkPj4xXT1FW2J8MF07Yj1iKzF8MDtnPWcrMXwwO2U9QlthKzI0fDBdO2lmKChnfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPD0oZXwwKSl7YnJlYWsgYn19cGEoKGU8PDEpK2R8MCwwLGMtZTw8MSl9cmV0dXJuIGh9cmV0dXJuIDB9ZnVuY3Rpb24gQWIoYSxiLGMsZCl7dmFyIGU9MCxmPTAsZz0wLGg9MCxpPTAsaj1LKDApLGs9MDthOntiOntpZighZCl7YnJlYWsgYn1jOntkOntzd2l0Y2goRFthKzI4Pj4yXS0xfDApe2Nhc2UgMDplPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2Y9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWYrYnwwO2Y9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1mPj4+MCl7YnJlYWsgYX1lPUJbYnwwXTtpZigoZXwwKTwwKXticmVhayBifUNbKGc8PDEpK2Q+PjFdPWUmMjU1O2I9YisxfDA7Zz1nKzF8MDtlPUJbYSsyNHwwXTtpZigoZ3wwKTwoKChjfDApPChlfDApP2M6ZSl8MCkpe2NvbnRpbnVlfWJyZWFrfX1oPTE7aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDE6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfUNbKGc8PDEpK2Q+PjFdPUVbYnwwXTtiPWIrMXwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDI6ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtmPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1mK2J8MDtmPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49Zj4+PjApe2JyZWFrIGF9ZT1DW2I+PjFdO2lmKChlfDApPDApe2JyZWFrIGJ9Q1soZzw8MSkrZD4+MV09ZTtiPWIrMnwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aD0xO2lmKChjfDApPihlfDApKXticmVhayBjfWJyZWFrIGI7Y2FzZSAzOmg9MTtlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2Y9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWYrYnwwO2Y9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1mPj4+MCl7YnJlYWsgYX1DWyhnPDwxKStkPj4xXT1GW2I+PjFdO2I9YisyfDA7Zz1nKzF8MDtlPUJbYSsyNHwwXTtpZigoZ3wwKTwoKChjfDApPChlfDApP2M6ZSl8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgY31icmVhayBiO2Nhc2UgNDpoPTE7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtmPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1mK2J8MDtmPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49Zj4+PjApe2JyZWFrIGF9ZT1EW2I+PjJdO2lmKGU+Pj4wPjY1NTM1KXticmVhayBhfUNbKGc8PDEpK2Q+PjFdPWU7Yj1iKzR8MDtnPWcrMXwwO2U9QlthKzI0fDBdO2lmKChnfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPihlfDApKXticmVhayBjfWJyZWFrIGI7Y2FzZSA1Omg9MTtlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2Y9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWYrYnwwO2Y9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1mPj4+MCl7YnJlYWsgYX1lPURbYj4+Ml07aWYoZT4+PjA+NjU1MzUpe2JyZWFrIGF9Q1soZzw8MSkrZD4+MV09ZTtiPWIrNHwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDY6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfWU9RFtiKzQ+PjJdO2k9RFtiPj4yXTtpZighZSZpPj4+MD42NTUzNXxlKXticmVhayBhfUNbKGc8PDEpK2Q+PjFdPWk7Yj1iKzh8MDtnPWcrMXwwO2U9QlthKzI0fDBdO2lmKChnfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPihlfDApKXticmVhayBjfWJyZWFrIGI7Y2FzZSA3Omg9MTtlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2Y9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWYrYnwwO2Y9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1mPj4+MCl7YnJlYWsgYX1lPURbYis0Pj4yXTtpPURbYj4+Ml07aWYoIWUmaT4+PjA+NjU1MzV8ZSl7YnJlYWsgYX1DWyhnPDwxKStkPj4xXT1pO2I9Yis4fDA7Zz1nKzF8MDtlPUJbYSsyNHwwXTtpZigoZ3wwKTwoKChjfDApPChlfDApP2M6ZSl8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgY31icmVhayBiO2Nhc2UgODpoPTE7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtmPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1mK2J8MDtmPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49Zj4+PjApe2JyZWFrIGF9aT0oZzw8MSkrZHwwO2o9SFtiPj4yXTtlOntpZihqPEsoNDI5NDk2NzI5Nikmaj49SygwKSl7ZT1+fmo+Pj4wO2JyZWFrIGV9ZT0wfUNbaT4+MV09ZTtiPWIrNHwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDk6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfWk9KGc8PDEpK2R8MDtrPUlbYj4+M107Zjp7aWYoazw0Mjk0OTY3Mjk2Jms+PTApe2U9fn5rPj4+MDticmVhayBmfWU9MH1DW2k+PjFdPWU7Yj1iKzh8MDtnPWcrMXwwO2U9QlthKzI0fDBdO2lmKChnfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPihlfDApKXticmVhayBjfWJyZWFrIGI7Y2FzZSAxMDpicmVhayBkO2RlZmF1bHQ6YnJlYWsgYn19aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfUNbKGc8PDEpK2Q+PjFdPUVbYnwwXTtiPWIrMXwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk8PShlfDApKXticmVhayBifX1wYSgoZTw8MSkrZHwwLDAsYy1lPDwxKX1yZXR1cm4gaH1yZXR1cm4gMH1mdW5jdGlvbiBlYyhhLGIsYyl7dmFyIGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wO2U9JC00OHwwOyQ9ZTtkPUZbNTA2OV18Rls1MDcwXTw8MTY7Zj1GWzUwNjddfEZbNTA2OF08PDE2O0NbZSszOD4+MV09ZjtDW2UrNDA+PjFdPWY+Pj4xNjtDW2UrNDI+PjFdPWQ7Q1tlKzQ0Pj4xXT1kPj4+MTY7ZD1EWzI1MzNdO0RbZSszMj4+Ml09RFsyNTMyXTtEW2UrMzY+PjJdPWQ7ZD1EWzI1MzFdO0RbZSsyND4+Ml09RFsyNTMwXTtEW2UrMjg+PjJdPWQ7ZD1EWzI1MjldO0RbZSsxNj4+Ml09RFsyNTI4XTtEW2UrMjA+PjJdPWQ7Zj1EW2IrMTI+PjJdO2Q9RFtiKzIwPj4yXTtnPURbYisxNj4+Ml07aD1nKzV8MDtkPWg+Pj4wPDU/ZCsxfDA6ZDthOntiOntpZihoPj4+MD5HW2IrOD4+Ml0mKGR8MCk+PShmfDApfChkfDApPihmfDApKXtiPXphKGUrMTZ8MCk7aWYoYj4+PjA+PTQyOTQ5NjcyODApe2JyZWFrIGF9Yzp7ZDp7aWYoYj4+PjA+PTExKXtkPWIrMTYmLTE2O2M9bmEoZCk7RFtlKzg+PjJdPWR8LTIxNDc0ODM2NDg7RFtlPj4yXT1jO0RbZSs0Pj4yXT1iO2JyZWFrIGR9QltlKzExfDBdPWI7Yz1lO2lmKCFiKXticmVhayBjfX1vYShjLGUrMTZ8MCxiKX1CW2IrY3wwXT0wO0RbYT4+Ml09LTI7YT1hKzR8MDtpZihCW2UrMTF8MF0+PTApe2I9RFtlKzQ+PjJdO0RbYT4+Ml09RFtlPj4yXTtEW2ErND4+Ml09YjtEW2ErOD4+Ml09RFtlKzg+PjJdO2JyZWFrIGJ9Yj1hO2E9RFtlPj4yXTt0YShiLGEsRFtlKzQ+PjJdKTttYShhKTticmVhayBifWQ9ZytEW2I+PjJdfDA7Zj1FW2R8MF18RVtkKzF8MF08PDh8KEVbZCsyfDBdPDwxNnxFW2QrM3wwXTw8MjQpO0JbY3wwXT1mO0JbYysxfDBdPWY+Pj44O0JbYysyfDBdPWY+Pj4xNjtCW2MrM3wwXT1mPj4+MjQ7QltjKzR8MF09RVtkKzR8MF07ZD1EW2IrMjA+PjJdO2Y9RFtiKzE2Pj4yXSs1fDA7ZD1mPj4+MDw1P2QrMXwwOmQ7RFtiKzE2Pj4yXT1mO0RbYisyMD4+Ml09ZDtpZih2YShjLDEzOTQsNSkpe2I9bmEoMzIpO0JbYisxNnwwXT1FWzE2MzhdO2M9RVsxNjM0XXxFWzE2MzVdPDw4fChFWzE2MzZdPDwxNnxFWzE2MzddPDwyNCk7ZD1FWzE2MzBdfEVbMTYzMV08PDh8KEVbMTYzMl08PDE2fEVbMTYzM108PDI0KTtCW2IrOHwwXT1kO0JbYis5fDBdPWQ+Pj44O0JbYisxMHwwXT1kPj4+MTY7QltiKzExfDBdPWQ+Pj4yNDtCW2IrMTJ8MF09YztCW2IrMTN8MF09Yz4+Pjg7QltiKzE0fDBdPWM+Pj4xNjtCW2IrMTV8MF09Yz4+PjI0O2M9RVsxNjI2XXxFWzE2MjddPDw4fChFWzE2MjhdPDwxNnxFWzE2MjldPDwyNCk7ZD1FWzE2MjJdfEVbMTYyM108PDh8KEVbMTYyNF08PDE2fEVbMTYyNV08PDI0KTtCW2J8MF09ZDtCW2IrMXwwXT1kPj4+ODtCW2IrMnwwXT1kPj4+MTY7QltiKzN8MF09ZD4+PjI0O0JbYis0fDBdPWM7QltiKzV8MF09Yz4+Pjg7QltiKzZ8MF09Yz4+PjE2O0JbYis3fDBdPWM+Pj4yNDtCW2IrMTd8MF09MDtEW2E+PjJdPS0xO3RhKGErNHwwLGIsMTcpO21hKGIpO2JyZWFrIGJ9Zz1EW2IrMTI+PjJdO2lmKChnfDApPD0oZHwwKSZHW2IrOD4+Ml08PWY+Pj4wfChkfDApPihnfDApKXtiPXphKGUrMTZ8MCk7aWYoYj4+PjA+PTQyOTQ5NjcyODApe2JyZWFrIGF9ZTp7Zjp7aWYoYj4+PjA+PTExKXtkPWIrMTYmLTE2O2M9bmEoZCk7RFtlKzg+PjJdPWR8LTIxNDc0ODM2NDg7RFtlPj4yXT1jO0RbZSs0Pj4yXT1iO2JyZWFrIGZ9QltlKzExfDBdPWI7Yz1lO2lmKCFiKXticmVhayBlfX1vYShjLGUrMTZ8MCxiKX1CW2IrY3wwXT0wO0RbYT4+Ml09LTI7YT1hKzR8MDtpZihCW2UrMTF8MF0+PTApe2I9RFtlKzQ+PjJdO0RbYT4+Ml09RFtlPj4yXTtEW2ErND4+Ml09YjtEW2ErOD4+Ml09RFtlKzg+PjJdO2JyZWFrIGJ9Yj1hO2E9RFtlPj4yXTt0YShiLGEsRFtlKzQ+PjJdKTttYShhKTticmVhayBifUJbYys1fDBdPUVbZitEW2I+PjJdfDBdO2Q9RFtiKzIwPj4yXTtmPURbYisxNj4+Ml0rMXwwO2Q9Zj9kOmQrMXwwO0RbYisxNj4+Ml09ZjtEW2IrMjA+PjJdPWQ7Zz1EW2IrMTI+PjJdO2lmKChnfDApPD0oZHwwKSZHW2IrOD4+Ml08PWY+Pj4wfChkfDApPihnfDApKXtiPXphKGUrMTZ8MCk7aWYoYj4+PjA+PTQyOTQ5NjcyODApe2JyZWFrIGF9Zzp7aDp7aWYoYj4+PjA+PTExKXtkPWIrMTYmLTE2O2M9bmEoZCk7RFtlKzg+PjJdPWR8LTIxNDc0ODM2NDg7RFtlPj4yXT1jO0RbZSs0Pj4yXT1iO2JyZWFrIGh9QltlKzExfDBdPWI7Yz1lO2lmKCFiKXticmVhayBnfX1vYShjLGUrMTZ8MCxiKX1CW2IrY3wwXT0wO0RbYT4+Ml09LTI7YT1hKzR8MDtpZihCW2UrMTF8MF0+PTApe2I9RFtlKzQ+PjJdO0RbYT4+Ml09RFtlPj4yXTtEW2ErND4+Ml09YjtEW2ErOD4+Ml09RFtlKzg+PjJdO2JyZWFrIGJ9Yj1hO2E9RFtlPj4yXTt0YShiLGEsRFtlKzQ+PjJdKTttYShhKTticmVhayBifUJbYys2fDBdPUVbZitEW2I+PjJdfDBdO2Q9RFtiKzIwPj4yXTtmPURbYisxNj4+Ml0rMXwwO2Q9Zj9kOmQrMXwwO0RbYisxNj4+Ml09ZjtEW2IrMjA+PjJdPWQ7Zz1EW2IrMTI+PjJdO2lmKChnfDApPD0oZHwwKSZHW2IrOD4+Ml08PWY+Pj4wfChkfDApPihnfDApKXtiPXphKGUrMTZ8MCk7aWYoYj4+PjA+PTQyOTQ5NjcyODApe2JyZWFrIGF9aTp7ajp7aWYoYj4+PjA+PTExKXtkPWIrMTYmLTE2O2M9bmEoZCk7RFtlKzg+PjJdPWR8LTIxNDc0ODM2NDg7RFtlPj4yXT1jO0RbZSs0Pj4yXT1iO2JyZWFrIGp9QltlKzExfDBdPWI7Yz1lO2lmKCFiKXticmVhayBpfX1vYShjLGUrMTZ8MCxiKX1CW2IrY3wwXT0wO0RbYT4+Ml09LTI7YT1hKzR8MDtpZihCW2UrMTF8MF0+PTApe2I9RFtlKzQ+PjJdO0RbYT4+Ml09RFtlPj4yXTtEW2ErND4+Ml09YjtEW2ErOD4+Ml09RFtlKzg+PjJdO2JyZWFrIGJ9Yj1hO2E9RFtlPj4yXTt0YShiLGEsRFtlKzQ+PjJdKTttYShhKTticmVhayBifUJbYys3fDBdPUVbZitEW2I+PjJdfDBdO2Q9RFtiKzIwPj4yXTtmPURbYisxNj4+Ml0rMXwwO2Q9Zj9kOmQrMXwwO0RbYisxNj4+Ml09ZjtEW2IrMjA+PjJdPWQ7Zz1EW2IrMTI+PjJdO2lmKChnfDApPD0oZHwwKSZHW2IrOD4+Ml08PWY+Pj4wfChkfDApPihnfDApKXtiPUZiKGUsZSsxNnwwKTtEW2E+PjJdPS0yO2E9YSs0fDA7aWYoQltiKzExfDBdPj0wKXtiPURbZSs0Pj4yXTtEW2E+PjJdPURbZT4+Ml07RFthKzQ+PjJdPWI7RFthKzg+PjJdPURbZSs4Pj4yXTticmVhayBifXRhKGEsRFtiPj4yXSxEW2IrND4+Ml0pO2lmKEJbYisxMXwwXT49MCl7YnJlYWsgYn1tYShEW2I+PjJdKTticmVhayBifUJbYys4fDBdPUVbZitEW2I+PjJdfDBdO2Q9RFtiKzIwPj4yXTtmPWQ7aT1EW2IrMTY+PjJdO2c9aSsxfDA7ZD1nP2Q6ZCsxfDA7RFtiKzE2Pj4yXT1nO0RbYisyMD4+Ml09ZDtoPURbYisxMj4+Ml07ZD1mO2Y9aSszfDA7ZD1mPj4+MDwzP2QrMXwwOmQ7aWYoZj4+PjA+R1tiKzg+PjJdJihkfDApPj0oaHwwKXwoZHwwKT4oaHwwKSl7Yj1GYihlLGUrMTZ8MCk7RFthPj4yXT0tMjthPWErNHwwO2lmKEJbYisxMXwwXT49MCl7Yj1EW2UrND4+Ml07RFthPj4yXT1EW2U+PjJdO0RbYSs0Pj4yXT1iO0RbYSs4Pj4yXT1EW2UrOD4+Ml07YnJlYWsgYn10YShhLERbYj4+Ml0sRFtiKzQ+PjJdKTtpZihCW2IrMTF8MF0+PTApe2JyZWFrIGJ9bWEoRFtiPj4yXSk7YnJlYWsgYn1mPWM7Yz1nK0RbYj4+Ml18MDtDW2YrMTA+PjFdPUVbY3wwXXxFW2MrMXwwXTw8ODtkPURbYisyMD4+Ml07Yz1EW2IrMTY+PjJdKzJ8MDtkPWM+Pj4wPDI/ZCsxfDA6ZDtEW2IrMTY+PjJdPWM7RFtiKzIwPj4yXT1kO0RbYSs4Pj4yXT0wO0RbYSsxMj4+Ml09MDtEW2E+PjJdPTA7RFthKzQ+PjJdPTB9JD1lKzQ4fDA7cmV0dXJufUFhKCk7VCgpfWZ1bmN0aW9uIHpiKGEsYixjLGQpe3ZhciBlPTAsZj0wLGc9MCxoPTAsaT0wLGo9SygwKSxrPTA7YTp7Yjp7aWYoIWQpe2JyZWFrIGJ9Yzp7ZDp7c3dpdGNoKERbYSsyOD4+Ml0tMXwwKXtjYXNlIDA6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfURbKGc8PDIpK2Q+PjJdPUJbYnwwXTtiPWIrMXwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDE6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfURbKGc8PDIpK2Q+PjJdPUVbYnwwXTtiPWIrMXwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDI6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfURbKGc8PDIpK2Q+PjJdPUNbYj4+MV07Yj1iKzJ8MDtnPWcrMXwwO2U9QlthKzI0fDBdO2lmKChnfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPihlfDApKXticmVhayBjfWJyZWFrIGI7Y2FzZSAzOmg9MTtlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2Y9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWYrYnwwO2Y9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1mPj4+MCl7YnJlYWsgYX1EWyhnPDwyKStkPj4yXT1GW2I+PjFdO2I9YisyfDA7Zz1nKzF8MDtlPUJbYSsyNHwwXTtpZigoZ3wwKTwoKChjfDApPChlfDApP2M6ZSl8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgY31icmVhayBiO2Nhc2UgNDpoPTE7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtmPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1mK2J8MDtmPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49Zj4+PjApe2JyZWFrIGF9RFsoZzw8MikrZD4+Ml09RFtiPj4yXTtiPWIrNHwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDU6ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtmPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1mK2J8MDtmPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49Zj4+PjApe2JyZWFrIGF9ZT1EW2I+PjJdO2lmKChlfDApPDApe2JyZWFrIGJ9RFsoZzw8MikrZD4+Ml09ZTtiPWIrNHwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aD0xO2lmKChjfDApPihlfDApKXticmVhayBjfWJyZWFrIGI7Y2FzZSA2Omg9MTtlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2Y9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWYrYnwwO2Y9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1mPj4+MCl7YnJlYWsgYX1lPURbYj4+Ml07aWYoKERbYis0Pj4yXS0oZT4+PjA8MjE0NzQ4MzY0OCl8MCkhPS0xKXticmVhayBhfURbKGc8PDIpK2Q+PjJdPWU7Yj1iKzh8MDtnPWcrMXwwO2U9QlthKzI0fDBdO2lmKChnfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPihlfDApKXticmVhayBjfWJyZWFrIGI7Y2FzZSA3Omg9MTtlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2Y9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWYrYnwwO2Y9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1mPj4+MCl7YnJlYWsgYX1lPURbYis0Pj4yXTtpPURbYj4+Ml07aWYoIWUmaT4+PjA+MjE0NzQ4MzY0N3xlKXticmVhayBhfURbKGc8PDIpK2Q+PjJdPWk7Yj1iKzh8MDtnPWcrMXwwO2U9QlthKzI0fDBdO2lmKChnfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPihlfDApKXticmVhayBjfWJyZWFrIGI7Y2FzZSA4Omg9MTtlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2Y9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWYrYnwwO2Y9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1mPj4+MCl7YnJlYWsgYX1pPShnPDwyKStkfDA7aj1IW2I+PjJdO2U6e2lmKEsoTChqKSk8SygyMTQ3NDgzNjQ4KSl7ZT1+fmo7YnJlYWsgZX1lPS0yMTQ3NDgzNjQ4fURbaT4+Ml09ZTtiPWIrNHwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDk6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfWk9KGc8PDIpK2R8MDtrPUlbYj4+M107Zjp7aWYoTChrKTwyMTQ3NDgzNjQ4KXtlPX5+azticmVhayBmfWU9LTIxNDc0ODM2NDh9RFtpPj4yXT1lO2I9Yis4fDA7Zz1nKzF8MDtlPUJbYSsyNHwwXTtpZigoZ3wwKTwoKChjfDApPChlfDApP2M6ZSl8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgY31icmVhayBiO2Nhc2UgMTA6YnJlYWsgZDtkZWZhdWx0OmJyZWFrIGJ9fWg9MTtlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2Y9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWYrYnwwO2Y9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1mPj4+MCl7YnJlYWsgYX1EWyhnPDwyKStkPj4yXT1FW2J8MF07Yj1iKzF8MDtnPWcrMXwwO2U9QlthKzI0fDBdO2lmKChnfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPD0oZXwwKSl7YnJlYWsgYn19cGEoKGU8PDIpK2R8MCwwLGMtZTw8Mil9cmV0dXJuIGh9cmV0dXJuIDB9ZnVuY3Rpb24geWIoYSxiLGMsZCl7dmFyIGU9MCxmPTAsZz0wLGg9MCxpPTAsaj1LKDApLGs9MDthOntiOntpZighZCl7YnJlYWsgYn1jOntkOntzd2l0Y2goRFthKzI4Pj4yXS0xfDApe2Nhc2UgMDpoPTE7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtmPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1mK2J8MDtmPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49Zj4+PjApe2JyZWFrIGF9RFsoZzw8MikrZD4+Ml09QltifDBdO2I9YisxfDA7Zz1nKzF8MDtlPUJbYSsyNHwwXTtpZigoZ3wwKTwoKChjfDApPChlfDApP2M6ZSl8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgY31icmVhayBiO2Nhc2UgMTpoPTE7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtmPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1mK2J8MDtmPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49Zj4+PjApe2JyZWFrIGF9RFsoZzw8MikrZD4+Ml09RVtifDBdO2I9YisxfDA7Zz1nKzF8MDtlPUJbYSsyNHwwXTtpZigoZ3wwKTwoKChjfDApPChlfDApP2M6ZSl8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgY31icmVhayBiO2Nhc2UgMjpoPTE7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtmPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1mK2J8MDtmPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49Zj4+PjApe2JyZWFrIGF9RFsoZzw8MikrZD4+Ml09Q1tiPj4xXTtiPWIrMnwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDM6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfURbKGc8PDIpK2Q+PjJdPUZbYj4+MV07Yj1iKzJ8MDtnPWcrMXwwO2U9QlthKzI0fDBdO2lmKChnfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPihlfDApKXticmVhayBjfWJyZWFrIGI7Y2FzZSA0Omg9MTtlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2Y9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWYrYnwwO2Y9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1mPj4+MCl7YnJlYWsgYX1EWyhnPDwyKStkPj4yXT1EW2I+PjJdO2I9Yis0fDA7Zz1nKzF8MDtlPUJbYSsyNHwwXTtpZigoZ3wwKTwoKChjfDApPChlfDApP2M6ZSl8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgY31icmVhayBiO2Nhc2UgNTpoPTE7ZT1CW2ErMjR8MF07aWYoKCgoY3wwKTwoZXwwKT9jOmUpfDApPjApe2U9RFthPj4yXTtmPURbZT4+Ml07Yj1EW2ErNDg+PjJdKyRoKERbYSs0MD4+Ml0sRFthKzQ0Pj4yXSxiLDApfDA7Yj1mK2J8MDtmPURbZSs0Pj4yXTt3aGlsZSgxKXtpZihiPj4+MD49Zj4+PjApe2JyZWFrIGF9RFsoZzw8MikrZD4+Ml09RFtiPj4yXTtiPWIrNHwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDY6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfWU9RFtiPj4yXTtpZihEW2IrND4+Ml0pe2JyZWFrIGF9RFsoZzw8MikrZD4+Ml09ZTtiPWIrOHwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDc6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfWU9RFtiPj4yXTtpZihEW2IrND4+Ml0pe2JyZWFrIGF9RFsoZzw8MikrZD4+Ml09ZTtiPWIrOHwwO2c9ZysxfDA7ZT1CW2ErMjR8MF07aWYoKGd8MCk8KCgoY3wwKTwoZXwwKT9jOmUpfDApKXtjb250aW51ZX1icmVha319aWYoKGN8MCk+KGV8MCkpe2JyZWFrIGN9YnJlYWsgYjtjYXNlIDg6aD0xO2U9QlthKzI0fDBdO2lmKCgoKGN8MCk8KGV8MCk/YzplKXwwKT4wKXtlPURbYT4+Ml07Zj1EW2U+PjJdO2I9RFthKzQ4Pj4yXSskaChEW2ErNDA+PjJdLERbYSs0ND4+Ml0sYiwwKXwwO2I9ZitifDA7Zj1EW2UrND4+Ml07d2hpbGUoMSl7aWYoYj4+PjA+PWY+Pj4wKXticmVhayBhfWk9KGc8PDIpK2R8MDtqPUhbYj4+Ml07ZTp7aWYoajxLKDQyOTQ5NjcyOTYpJmo+PUsoMCkpe2U9fn5qPj4+MDticmVhayBlfWU9MH1EW2k+PjJdPWU7Yj1iKzR8MDtnPWcrMXwwO2U9QlthKzI0fDBdO2lmKChnfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPihlfDApKXticmVhayBjfWJyZWFrIGI7Y2FzZSA5Omg9MTtlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2Y9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWYrYnwwO2Y9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1mPj4+MCl7YnJlYWsgYX1pPShnPDwyKStkfDA7az1JW2I+PjNdO2Y6e2lmKGs8NDI5NDk2NzI5NiZrPj0wKXtlPX5+az4+PjA7YnJlYWsgZn1lPTB9RFtpPj4yXT1lO2I9Yis4fDA7Zz1nKzF8MDtlPUJbYSsyNHwwXTtpZigoZ3wwKTwoKChjfDApPChlfDApP2M6ZSl8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZigoY3wwKT4oZXwwKSl7YnJlYWsgY31icmVhayBiO2Nhc2UgMTA6YnJlYWsgZDtkZWZhdWx0OmJyZWFrIGJ9fWg9MTtlPUJbYSsyNHwwXTtpZigoKChjfDApPChlfDApP2M6ZSl8MCk+MCl7ZT1EW2E+PjJdO2Y9RFtlPj4yXTtiPURbYSs0OD4+Ml0rJGgoRFthKzQwPj4yXSxEW2ErNDQ+PjJdLGIsMCl8MDtiPWYrYnwwO2Y9RFtlKzQ+PjJdO3doaWxlKDEpe2lmKGI+Pj4wPj1mPj4+MCl7YnJlYWsgYX1EWyhnPDwyKStkPj4yXT1FW2J8MF07Yj1iKzF8MDtnPWcrMXwwO2U9QlthKzI0fDBdO2lmKChnfDApPCgoKGN8MCk8KGV8MCk/YzplKXwwKSl7Y29udGludWV9YnJlYWt9fWlmKChjfDApPD0oZXwwKSl7YnJlYWsgYn19cGEoKGU8PDIpK2R8MCwwLGMtZTw8Mil9cmV0dXJuIGh9cmV0dXJuIDB9ZnVuY3Rpb24gb2UoYSl7YT1hfDA7dmFyIGI9MCxjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MCxvPTAscD0wLHE9MDtrPSQtMTZ8MDskPWs7bj0xO289YmFbRFtEW2E+PjJdKzI0Pj4yXV0oYSl8MDthOntpZigob3wwKTw9MCl7bj0wO2JyZWFrIGF9cT1hKzQ4fDA7d2hpbGUoMSl7Yjp7Yzp7aWYoIURbKGJhW0RbRFthPj4yXSsyOD4+Ml1dKGEpfDApKzQwPj4yXSl7YnJlYWsgY31lPWw8PDI7Yj1EW2UrRFthKzM2Pj4yXT4+Ml07ZD1EW2IrOD4+Ml07Zz1jYihiKTtpZighZyl7YnJlYWsgY31iPURbKGJhW0RbRFthPj4yXSsyOD4+Ml1dKGEpfDApKzQwPj4yXTtoPURbZCs1Nj4+Ml07Yz1uYSgzMik7RFtrPj4yXT1jO0Rbays0Pj4yXT0yNDtEW2srOD4+Ml09LTIxNDc0ODM2MTY7QltjKzI0fDBdPTA7Zj1FWzEzMDhdfEVbMTMwOV08PDh8KEVbMTMxMF08PDE2fEVbMTMxMV08PDI0KTtkPUVbMTMwNF18RVsxMzA1XTw8OHwoRVsxMzA2XTw8MTZ8RVsxMzA3XTw8MjQpO0JbYysxNnwwXT1kO0JbYysxN3wwXT1kPj4+ODtCW2MrMTh8MF09ZD4+PjE2O0JbYysxOXwwXT1kPj4+MjQ7QltjKzIwfDBdPWY7QltjKzIxfDBdPWY+Pj44O0JbYysyMnwwXT1mPj4+MTY7QltjKzIzfDBdPWY+Pj4yNDtmPUVbMTMwMF18RVsxMzAxXTw8OHwoRVsxMzAyXTw8MTZ8RVsxMzAzXTw8MjQpO2Q9RVsxMjk2XXxFWzEyOTddPDw4fChFWzEyOThdPDwxNnxFWzEyOTldPDwyNCk7QltjKzh8MF09ZDtCW2MrOXwwXT1kPj4+ODtCW2MrMTB8MF09ZD4+PjE2O0JbYysxMXwwXT1kPj4+MjQ7QltjKzEyfDBdPWY7QltjKzEzfDBdPWY+Pj44O0JbYysxNHwwXT1mPj4+MTY7QltjKzE1fDBdPWY+Pj4yNDtmPUVbMTI5Ml18RVsxMjkzXTw8OHwoRVsxMjk0XTw8MTZ8RVsxMjk1XTw8MjQpO2Q9RVsxMjg4XXxFWzEyODldPDw4fChFWzEyOTBdPDwxNnxFWzEyOTFdPDwyNCk7QltjfDBdPWQ7QltjKzF8MF09ZD4+Pjg7QltjKzJ8MF09ZD4+PjE2O0JbYyszfDBdPWQ+Pj4yNDtCW2MrNHwwXT1mO0JbYys1fDBdPWY+Pj44O0JbYys2fDBdPWY+Pj4xNjtCW2MrN3wwXT1mPj4+MjQ7Zj1iKzE2fDA7aj1mO2k9RFtmPj4yXTtkOntpZighaSl7YnJlYWsgZH13aGlsZSgxKXtkPShofDApPkRbaSsxNj4+Ml07aj1kP2o6aTtpPURbKGQ8PDIpK2k+PjJdO2lmKGkpe2NvbnRpbnVlfWJyZWFrfWlmKChmfDApPT0oanwwKXwoaHwwKTxEW2orMTY+PjJdKXticmVhayBkfWk9RFtqKzI0Pj4yXTtpZighaSl7YnJlYWsgZH1kPWorMjB8MDt3aGlsZSgxKXtmPUVbaSsyN3wwXTtoPWY8PDI0Pj4yNDwwO209aD9EW2krMjA+PjJdOmY7cD1tPj4+MDwyNDtlOntmOntqPXA/bToyNDtnOntpZihqKXtmPWkrMTZ8MDtoPWg/RFtmPj4yXTpmO2Y9dmEoYyxoLGopO2g6e2lmKCFmKXtpZihtPj4+MDw9MjQpe2JyZWFrIGh9YnJlYWsgZX1pZigoZnwwKTwwKXticmVhayBlfX1mPXZhKGgsYyxqKTtpZighZil7YnJlYWsgZ31pZigoZnwwKTwwKXticmVhayBmfWI9ZDticmVhayBkfWlmKG0+Pj4wPjI0KXticmVhayBlfX1pZihwKXticmVhayBmfWI9ZDticmVhayBkfWk9aSs0fDB9aT1EW2k+PjJdO2lmKGkpe2NvbnRpbnVlfWJyZWFrfX1pPTA7ZD1iKzR8MDtiPV9hKGIsayk7aTp7aWYoKGR8MCk9PShifDApKXticmVhayBpfWQ9QltiKzM5fDBdPDA/RFtiKzI4Pj4yXTpiKzI4fDA7aj0wO2Y9MDt3aGlsZSgxKXtiPWQ7ZD1iKzF8MDtjPUJbYnwwXTtpZigoY3wwKT09MzJ8Yy05Pj4+MDw1KXtjb250aW51ZX1icmVha31qOntrOntsOntjPUJbYnwwXTtzd2l0Y2goYy00M3wwKXtjYXNlIDA6YnJlYWsgaztjYXNlIDI6YnJlYWsgbDtkZWZhdWx0OmJyZWFrIGp9fWY9MX1jPUJbZHwwXTtiPWR9aWYoYy00OD4+PjA8MTApe3doaWxlKDEpe2o9KEooaiwxMCktQltifDBdfDApKzQ4fDA7ZD1CW2IrMXwwXTtiPWIrMXwwO2lmKGQtNDg+Pj4wPDEwKXtjb250aW51ZX1icmVha319Yj1mP2o6MC1qfDA7aWYoKGJ8MCk9PS0xKXticmVhayBpfWk9KGJ8MCkhPTB9aWYoQltrKzExfDBdPDApe21hKERbaz4+Ml0pfWlmKCFpKXticmVhayBjfWU9RFtEW0RbYSszNj4+Ml0rZT4+Ml0rOD4+Ml07aWYoIURbZSs2ND4+Ml0pe2I9bmEoMzIpO2M9YjtEW2IrMTY+PjJdPTA7RFtiKzIwPj4yXT0wO0RbYis4Pj4yXT0wO0RbYj4+Ml09MDtEW2IrND4+Ml09MDtEW2IrMjQ+PjJdPTA7RFtiKzI4Pj4yXT0wO2Q9RFtlKzY0Pj4yXTtEW2UrNjQ+PjJdPWI7aWYoZCl7Yj1EW2Q+PjJdO2lmKGIpe0RbZCs0Pj4yXT1iO21hKGIpfW1hKGQpO2M9RFtlKzY0Pj4yXX1EW2U+PjJdPWM7Yj1EW2MrMjA+PjJdO0RbZSs4Pj4yXT1EW2MrMTY+PjJdO0RbZSsxMj4+Ml09YjtkPURbYysyND4+Ml07Yj1EW2MrMjg+PjJdO0RbZSs0OD4+Ml09MDtEW2UrNTI+PjJdPTA7RFtlKzQwPj4yXT0wO0RbZSs0ND4+Ml09MDtEW2UrMTY+PjJdPWQ7RFtlKzIwPj4yXT1ifW06e0JbZSsyNHwwXT1FW2crMjR8MF07RFtlKzI4Pj4yXT1EW2crMjg+PjJdO0JbZSszMnwwXT1FW2crMzJ8MF07Yj1EW2crNDQ+PjJdO0RbZSs0MD4+Ml09RFtnKzQwPj4yXTtEW2UrNDQ+PjJdPWI7Yj1EW2crNTI+PjJdO0RbZSs0OD4+Ml09RFtnKzQ4Pj4yXTtEW2UrNTI+PjJdPWI7RFtlKzU2Pj4yXT1EW2crNTY+PjJdO2I9RFtnKzEyPj4yXTtEW2UrOD4+Ml09RFtnKzg+PjJdO0RbZSsxMj4+Ml09YjtiPURbZysyMD4+Ml07RFtlKzE2Pj4yXT1EW2crMTY+PjJdO0RbZSsyMD4+Ml09YjtEW2UrNjA+PjJdPURbZys2MD4+Ml07ZD1EW2c+PjJdO246e2lmKCFkKXtEW2U+PjJdPTA7Yz0xO2JyZWFrIG59Yj1EW2U+PjJdO2M9MDtpZighYil7YnJlYWsgbn1jPWI7Yj1EW2Q+PjJdO2lkKGMsYixEW2QrND4+Ml0tYnwwLDApO2M9MX1pZighYyl7YnJlYWsgbX1CW2UrODR8MF09RVtnKzg0fDBdO0RbZSs4MD4+Ml09RFtnKzgwPj4yXTtpZigoZXwwKSE9KGd8MCkpe2ZiKGUrNjh8MCxEW2crNjg+PjJdLERbZys3Mj4+Ml0pfW86e2g9RFtnKzg4Pj4yXTtwOntpZihoKXtiPW5hKDQwKTtkPURbaD4+Ml07RFtiKzE2Pj4yXT0wO0RbYis4Pj4yXT0wO0RbYisxMj4+Ml09MDtEW2I+PjJdPWQ7Yz1EW2grMTI+PjJdLURbaCs4Pj4yXXwwO2lmKGMpe2lmKChjfDApPDApe2JyZWFrIG99ZD1uYShjKTtEW2IrOD4+Ml09ZDtEW2IrMTI+PjJdPWQ7RFtiKzE2Pj4yXT1jK2Q7Yz1EW2grOD4+Ml07Zj1EW2grMTI+PjJdLWN8MDtpZigoZnwwKT4wKXtkPW9hKGQsYyxmKStmfDB9RFtiKzEyPj4yXT1kfWQ9RFtoKzM2Pj4yXTtEW2IrMzI+PjJdPURbaCszMj4+Ml07RFtiKzM2Pj4yXT1kO2Q9RFtoKzI4Pj4yXTtEW2IrMjQ+PjJdPURbaCsyND4+Ml07RFtiKzI4Pj4yXT1kO2M9RFtlKzg4Pj4yXTtEW2UrODg+PjJdPWI7aWYoYyl7YnJlYWsgcH1icmVhayBtfWM9RFtlKzg4Pj4yXTtEW2UrODg+PjJdPTA7aWYoIWMpe2JyZWFrIG19fWI9RFtjKzg+PjJdO2lmKGIpe0RbYysxMj4+Ml09YjttYShiKX1tYShjKTticmVhayBtfXFhKCk7VCgpfWJyZWFrIGJ9Yj1EW0RbYSszNj4+Ml0rKGw8PDIpPj4yXTtpZighKGJhW0RbRFtiPj4yXSsyND4+Ml1dKGIscSl8MCkpe2JyZWFrIGF9fWw9bCsxfDA7bj0ob3wwKT4obHwwKTtpZigobHwwKSE9KG98MCkpe2NvbnRpbnVlfWJyZWFrfX0kPWsrMTZ8MDtyZXR1cm4obl4tMSkmMX1mdW5jdGlvbiBpaChhLGIsYyxkKXthPWF8MDtiPWJ8MDtjPWN8MDtkPWR8MDt2YXIgZT1LKDApLGY9MCxnPTAsaD0wLGk9MCxqPTAsaz1LKDApLGw9MCxtPTAsbj0wLG89MCxwPTAscT0wLHI9MCxzPTAsdD0wLHU9MCx2PTAsdz0wLHg9MCx5PTAsej0wO2lmKERbYz4+Ml09PURbYys0Pj4yXSl7Zz1EW2QrODA+PjJdO3Y9JC0xNnwwOyQ9djtqPURbYSs0Pj4yXTtoPURbZCs0OD4+Ml07ZD1EW0RbZD4+Ml0+PjJdO2k9QltiKzI0fDBdO2M9dis4fDA7RFtjPj4yXT0xMDY1MzUzMjE2O3U9YztIW2M+PjJdPUsoLTE8PGpeLTEpL0hbYSsyMD4+Ml07dz1uYSgoaXwwKSE9KGkmMTA3Mzc0MTgyMyk/LTE6aTw8Mik7YTp7aWYoIWd8KGl8MCk8PTApe2JyZWFrIGF9cz1kK2h8MDt5PURbYj4+Ml07Yz1EW2IrNDg+PjJdO3o9RFtiKzQ0Pj4yXTt4PURbYis0MD4+Ml07aWYoIUVbYis4NHwwXSl7bj1EW2IrNjg+PjJdO3Q9aSYtMjtqPWkmMTtiPTA7d2hpbGUoMSl7Zj1EW3k+PjJdO2Q9JGgoeCx6LERbbisobzw8Mik+PjJdLDApK2N8MDtwPW9hKHcsZitkfDAseCk7az1IW3U+PjJdO2w9RFthKzg+PjJdO2Q9MDttPTA7aWYoKGl8MCkhPTEpe3doaWxlKDEpe2Y9cysoYjw8Mil8MDtxPWQ8PDI7ZT1LKFAoSyhLKGsqSyhIW3ErcD4+Ml0tSFtsK3E+PjJdKSkrSyguNSkpKSk7Yjp7aWYoSyhMKGUpKTxLKDIxNDc0ODM2NDgpKXtoPX5+ZTticmVhayBifWg9LTIxNDc0ODM2NDh9RFtmPj4yXT1oO2g9cXw0O2U9SyhQKEsoSyhrKksoSFtoK3A+PjJdLUhbbCtoPj4yXSkpK0soLjUpKSkpO2M6e2lmKEsoTChlKSk8SygyMTQ3NDgzNjQ4KSl7aD1+fmU7YnJlYWsgY31oPS0yMTQ3NDgzNjQ4fURbZis0Pj4yXT1oO2Q9ZCsyfDA7Yj1iKzJ8MDttPW0rMnwwO2lmKCh0fDApIT0obXwwKSl7Y29udGludWV9YnJlYWt9fWlmKGope2Y9cysoYjw8Mil8MDtkPWQ8PDI7ZT1LKFAoSyhLKGsqSyhIW2QrcD4+Ml0tSFtkK2w+PjJdKSkrSyguNSkpKSk7ZDp7aWYoSyhMKGUpKTxLKDIxNDc0ODM2NDgpKXtkPX5+ZTticmVhayBkfWQ9LTIxNDc0ODM2NDh9RFtmPj4yXT1kO2I9YisxfDB9bz1vKzF8MDtpZigoZ3wwKSE9KG98MCkpe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGF9bj1pJi0yO3Q9aSYxO2I9MDt3aGlsZSgxKXtoPURbeT4+Ml07ZD0kaCh4LHosbyxmKStjfDA7cD1vYSh3LGgrZHwwLHgpO2s9SFt1Pj4yXTtsPURbYSs4Pj4yXTtkPTA7bT0wO2lmKChpfDApIT0xKXt3aGlsZSgxKXtoPXMrKGI8PDIpfDA7cT1kPDwyO2U9SyhQKEsoSyhrKksoSFtxK3A+PjJdLUhbbCtxPj4yXSkpK0soLjUpKSkpO2U6e2lmKEsoTChlKSk8SygyMTQ3NDgzNjQ4KSl7aj1+fmU7YnJlYWsgZX1qPS0yMTQ3NDgzNjQ4fURbaD4+Ml09ajtqPXF8NDtlPUsoUChLKEsoaypLKEhbaitwPj4yXS1IW2wraj4+Ml0pKStLKC41KSkpKTtmOntpZihLKEwoZSkpPEsoMjE0NzQ4MzY0OCkpe2o9fn5lO2JyZWFrIGZ9aj0tMjE0NzQ4MzY0OH1EW2grND4+Ml09ajtkPWQrMnwwO2I9YisyfDA7bT1tKzJ8MDtpZigobnwwKSE9KG18MCkpe2NvbnRpbnVlfWJyZWFrfX1pZih0KXtoPXMrKGI8PDIpfDA7ZD1kPDwyO2U9SyhQKEsoSyhrKksoSFtkK3A+PjJdLUhbZCtsPj4yXSkpK0soLjUpKSkpO2c6e2lmKEsoTChlKSk8SygyMTQ3NDgzNjQ4KSl7ZD1+fmU7YnJlYWsgZ31kPS0yMTQ3NDgzNjQ4fURbaD4+Ml09ZDtiPWIrMXwwfWQ9bysxfDA7Zj1kP2Y6ZisxfDA7bz1kO2lmKChnfDApIT0oZHwwKXxmKXtjb250aW51ZX1icmVha319bWEodyk7JD12KzE2fDA7cmV0dXJuIDF9dj0kLTE2fDA7JD12O3U9RFthKzQ+PjJdO2c9RFtkKzQ4Pj4yXTtmPURbRFtkPj4yXT4+Ml07cj1CW2IrMjR8MF07ZD12Kzh8MDtEW2Q+PjJdPTEwNjUzNTMyMTY7aD1kO0hbZD4+Ml09SygtMTw8dV4tMSkvSFthKzIwPj4yXTt3PW5hKChyfDApIT0ociYxMDczNzQxODIzKT8tMTpyPDwyKTt5PURbYz4+Ml07Yz1EW2MrND4+Ml0teXwwO2g6e2lmKCFjfChyfDApPD0wKXticmVhayBofXM9ZitnfDA7ej1EW2I+PjJdO2Q9RFtiKzQ4Pj4yXTt4PURbYis0ND4+Ml07bz1EW2IrNDA+PjJdO2M9Yz4+MjtxPWM+Pj4wPjE/YzoxO2lmKEVbYis4NHwwXSl7aj1yJi0yO3U9ciYxO2I9MDt3aGlsZSgxKXtmPURbej4+Ml07Yz0kaChvLHgsRFt5KyhtPDwyKT4+Ml0sMCkrZHwwO2w9b2EodyxmK2N8MCxvKTtrPUhbaD4+Ml07bj1EW2ErOD4+Ml07Yz0wO2k9MDtpZigocnwwKSE9MSl7d2hpbGUoMSl7Zj1zKyhiPDwyKXwwO3Q9Yzw8MjtlPUsoUChLKEsoaypLKEhbdCtsPj4yXS1IW24rdD4+Ml0pKStLKC41KSkpKTtpOntpZihLKEwoZSkpPEsoMjE0NzQ4MzY0OCkpe2c9fn5lO2JyZWFrIGl9Zz0tMjE0NzQ4MzY0OH1EW2Y+PjJdPWc7Zz10fDQ7ZT1LKFAoSyhLKGsqSyhIW2crbD4+Ml0tSFtuK2c+PjJdKSkrSyguNSkpKSk7ajp7aWYoSyhMKGUpKTxLKDIxNDc0ODM2NDgpKXtnPX5+ZTticmVhayBqfWc9LTIxNDc0ODM2NDh9RFtmKzQ+PjJdPWc7Yz1jKzJ8MDtiPWIrMnwwO2k9aSsyfDA7aWYoKGp8MCkhPShpfDApKXtjb250aW51ZX1icmVha319aWYodSl7Zj1zKyhiPDwyKXwwO2M9Yzw8MjtlPUsoUChLKEsoaypLKEhbYytsPj4yXS1IW2Mrbj4+Ml0pKStLKC41KSkpKTtrOntpZihLKEwoZSkpPEsoMjE0NzQ4MzY0OCkpe2M9fn5lO2JyZWFrIGt9Yz0tMjE0NzQ4MzY0OH1EW2Y+PjJdPWM7Yj1iKzF8MH1tPW0rMXwwO2lmKChxfDApIT0obXwwKSl7Y29udGludWV9YnJlYWt9YnJlYWsgaH10PURbYis2OD4+Ml07aj1yJi0yO3U9ciYxO2I9MDt3aGlsZSgxKXtmPURbej4+Ml07Yz0kaChvLHgsRFt0KyhEW3krKG08PDIpPj4yXTw8Mik+PjJdLDApK2R8MDtwPW9hKHcsZitjfDAsbyk7az1IW2g+PjJdO2w9RFthKzg+PjJdO2M9MDtpPTA7aWYoKHJ8MCkhPTEpe3doaWxlKDEpe2Y9cysoYjw8Mil8MDtuPWM8PDI7ZT1LKFAoSyhLKGsqSyhIW24rcD4+Ml0tSFtsK24+PjJdKSkrSyguNSkpKSk7bDp7aWYoSyhMKGUpKTxLKDIxNDc0ODM2NDgpKXtnPX5+ZTticmVhayBsfWc9LTIxNDc0ODM2NDh9RFtmPj4yXT1nO2c9bnw0O2U9SyhQKEsoSyhrKksoSFtnK3A+PjJdLUhbbCtnPj4yXSkpK0soLjUpKSkpO206e2lmKEsoTChlKSk8SygyMTQ3NDgzNjQ4KSl7Zz1+fmU7YnJlYWsgbX1nPS0yMTQ3NDgzNjQ4fURbZis0Pj4yXT1nO2M9YysyfDA7Yj1iKzJ8MDtpPWkrMnwwO2lmKChqfDApIT0oaXwwKSl7Y29udGludWV9YnJlYWt9fWlmKHUpe2Y9cysoYjw8Mil8MDtjPWM8PDI7ZT1LKFAoSyhLKGsqSyhIW2MrcD4+Ml0tSFtjK2w+PjJdKSkrSyguNSkpKSk7bjp7aWYoSyhMKGUpKTxLKDIxNDc0ODM2NDgpKXtjPX5+ZTticmVhayBufWM9LTIxNDc0ODM2NDh9RFtmPj4yXT1jO2I9YisxfDB9bT1tKzF8MDtpZigocXwwKSE9KG18MCkpe2NvbnRpbnVlfWJyZWFrfX1tYSh3KTskPXYrMTZ8MDtyZXR1cm4gMX1mdW5jdGlvbiBuaChhKXthPWF8MDt2YXIgYj0wLGM9MCxkPTAsZT0wLGY9MDtjPURbYSszMj4+Ml07ZT1EW2MrMTY+PjJdO2Q9RFtjKzEyPj4yXTtiPURbYysyMD4+Ml07aWYoR1tjKzg+PjJdPmU+Pj4wJihkfDApPj0oYnwwKXwoYnwwKTwoZHwwKSl7Zj1FW2UrRFtjPj4yXXwwXTtkPWUrMXwwO2I9ZD9iOmIrMXwwO0RbYysxNj4+Ml09ZDtEW2MrMjA+PjJdPWI7Yz1EW2ErNDg+PjJdO0RbYSs0OD4+Ml09MDtpZihjKXtiYVtEW0RbYz4+Ml0rND4+Ml1dKGMpfWE6e2I6e2M6e2Q6e3N3aXRjaChmfDApe2Nhc2UgMDpjPW5hKDM4NCk7RFtjPj4yXT04MzEyO3BhKGMrNHwwLDAsODApO0RbYys5Nj4+Ml09MDtEW2MrMTAwPj4yXT0wO0RbYys5Mj4+Ml09LTE7RFtjKzg0Pj4yXT0tMTtEW2MrODg+PjJdPS0xO0RbYysxMDQ+PjJdPTA7RFtjKzEwOD4+Ml09MDtEW2MrMTEyPj4yXT0wO0RbYysxMTY+PjJdPTA7RFtjKzEyMD4+Ml09MDtEW2MrMTI0Pj4yXT0wO0RbYysxMjg+PjJdPTA7RFtjKzEzMj4+Ml09MDtEW2MrMTM2Pj4yXT0wO0RbYysxNDA+PjJdPTA7RFtjKzE0ND4+Ml09MDtEW2MrMTQ4Pj4yXT0wO0RbYysxNTY+PjJdPTA7RFtjKzE2MD4+Ml09MDtEW2MrMTUyPj4yXT0xMDY1MzUzMjE2O0RbYysxNjQ+PjJdPTA7RFtjKzE2OD4+Ml09MDtEW2MrMTcyPj4yXT0wO0RbYysxNzY+PjJdPTA7RFtjKzE4MD4+Ml09MDtEW2MrMTg0Pj4yXT0wO0RbYysxODg+PjJdPTA7RFtjKzE5Mj4+Ml09MDtEW2MrMTk2Pj4yXT0wO0RbYysyMDA+PjJdPTA7RFtjKzIwND4+Ml09MDtEW2MrMjA4Pj4yXT0wO0RbYysyMTI+PjJdPS0xO0RbYysyMTY+PjJdPTA7RFtjKzIyMD4+Ml09MDtEW2MrMjI0Pj4yXT0wO2I9YysyMzJ8MDtDW2IrMzg+PjFdPTA7RFtiPj4yXT0wO0RbYis4Pj4yXT0wO0RbYisxMj4+Ml09MDtEW2IrMTY+PjJdPTA7RFtiKzIwPj4yXT0wO0RbYisyND4+Ml09MDtEW2IrMjg+PjJdPTA7QltiKzI5fDBdPTA7QltiKzMwfDBdPTA7QltiKzMxfDBdPTA7QltiKzMyfDBdPTA7QltiKzMzfDBdPTA7QltiKzM0fDBdPTA7QltiKzM1fDBdPTA7QltiKzM2fDBdPTA7Yj1jKzI3MnwwO0NbYiszOD4+MV09MDtEW2I+PjJdPTA7RFtiKzg+PjJdPTA7RFtiKzEyPj4yXT0wO0RbYisxNj4+Ml09MDtEW2IrMjA+PjJdPTA7RFtiKzI0Pj4yXT0wO0RbYisyOD4+Ml09MDtCW2IrMjl8MF09MDtCW2IrMzB8MF09MDtCW2IrMzF8MF09MDtCW2IrMzJ8MF09MDtCW2IrMzN8MF09MDtCW2IrMzR8MF09MDtCW2IrMzV8MF09MDtCW2IrMzZ8MF09MDtiPWMrMzEyfDA7RFtiPj4yXT0wO0RbYis0Pj4yXT0wO0JbYis1fDBdPTA7QltiKzZ8MF09MDtCW2IrN3wwXT0wO0JbYis4fDBdPTA7QltiKzl8MF09MDtCW2IrMTB8MF09MDtCW2IrMTF8MF09MDtCW2IrMTJ8MF09MDtiPWMrMzI4fDA7Q1tiKzM4Pj4xXT0wO0RbYj4+Ml09MDtEW2IrOD4+Ml09MDtEW2IrMTI+PjJdPTA7RFtiKzE2Pj4yXT0wO0RbYisyMD4+Ml09MDtEW2IrMjQ+PjJdPTA7RFtiKzI4Pj4yXT0wO0JbYisyOXwwXT0wO0JbYiszMHwwXT0wO0JbYiszMXwwXT0wO0JbYiszMnwwXT0wO0JbYiszM3wwXT0wO0JbYiszNHwwXT0wO0JbYiszNXwwXT0wO0JbYiszNnwwXT0wO0RbYyszNzY+PjJdPTA7RFtjKzM2OD4+Ml09MDtEW2MrMzcyPj4yXT0wO2JyZWFrIGM7Y2FzZSAyOmJyZWFrIGQ7ZGVmYXVsdDpicmVhayBifX1jPW5hKDQ0MCk7RFtjPj4yXT04MzY0O3BhKGMrNHwwLDAsODApO0RbYys5Nj4+Ml09MDtEW2MrMTAwPj4yXT0wO0RbYys5Mj4+Ml09LTE7RFtjKzg0Pj4yXT0tMTtEW2MrODg+PjJdPS0xO0RbYysxMDQ+PjJdPTA7RFtjKzEwOD4+Ml09MDtEW2MrMTEyPj4yXT0wO0RbYysxMTY+PjJdPTA7RFtjKzEyMD4+Ml09MDtEW2MrMTI0Pj4yXT0wO0RbYysxMjg+PjJdPTA7RFtjKzEzMj4+Ml09MDtEW2MrMTM2Pj4yXT0wO0RbYysxNDA+PjJdPTA7RFtjKzE0ND4+Ml09MDtEW2MrMTQ4Pj4yXT0wO0RbYysxNTY+PjJdPTA7RFtjKzE2MD4+Ml09MDtEW2MrMTUyPj4yXT0xMDY1MzUzMjE2O0RbYysxNjQ+PjJdPTA7RFtjKzE2OD4+Ml09MDtEW2MrMTcyPj4yXT0wO0RbYysxNzY+PjJdPTA7RFtjKzE4MD4+Ml09MDtEW2MrMTg0Pj4yXT0wO0RbYysxODg+PjJdPTA7RFtjKzE5Mj4+Ml09MDtEW2MrMTk2Pj4yXT0wO0RbYysyMDA+PjJdPTA7RFtjKzIwND4+Ml09MDtEW2MrMjA4Pj4yXT0wO0RbYysyMTI+PjJdPS0xO0RbYysyMTY+PjJdPTA7RFtjKzIyMD4+Ml09MDtEW2MrMjI0Pj4yXT0wO2I9YysyMzJ8MDtDW2IrMzg+PjFdPTA7RFtiPj4yXT0wO0RbYis4Pj4yXT0wO0RbYisxMj4+Ml09MDtEW2IrMTY+PjJdPTA7RFtiKzIwPj4yXT0wO0RbYisyND4+Ml09MDtEW2IrMjg+PjJdPTA7QltiKzI5fDBdPTA7QltiKzMwfDBdPTA7QltiKzMxfDBdPTA7QltiKzMyfDBdPTA7QltiKzMzfDBdPTA7QltiKzM0fDBdPTA7QltiKzM1fDBdPTA7QltiKzM2fDBdPTA7Yj1jKzI3MnwwO0NbYiszOD4+MV09MDtEW2I+PjJdPTA7RFtiKzg+PjJdPTA7RFtiKzEyPj4yXT0wO0RbYisxNj4+Ml09MDtEW2IrMjA+PjJdPTA7RFtiKzI0Pj4yXT0wO0RbYisyOD4+Ml09MDtCW2IrMjl8MF09MDtCW2IrMzB8MF09MDtCW2IrMzF8MF09MDtCW2IrMzJ8MF09MDtCW2IrMzN8MF09MDtCW2IrMzR8MF09MDtCW2IrMzV8MF09MDtCW2IrMzZ8MF09MDtiPWMrMzEyfDA7RFtiPj4yXT0wO0RbYis0Pj4yXT0wO0JbYis1fDBdPTA7QltiKzZ8MF09MDtCW2IrN3wwXT0wO0JbYis4fDBdPTA7QltiKzl8MF09MDtCW2IrMTB8MF09MDtCW2IrMTF8MF09MDtCW2IrMTJ8MF09MDtiPWMrMzI4fDA7Q1tiKzM4Pj4xXT0wO0RbYj4+Ml09MDtEW2IrOD4+Ml09MDtEW2IrMTI+PjJdPTA7RFtiKzE2Pj4yXT0wO0RbYisyMD4+Ml09MDtEW2IrMjQ+PjJdPTA7RFtiKzI4Pj4yXT0wO0JbYisyOXwwXT0wO0JbYiszMHwwXT0wO0JbYiszMXwwXT0wO0JbYiszMnwwXT0wO0JbYiszM3wwXT0wO0JbYiszNHwwXT0wO0JbYiszNXwwXT0wO0JbYiszNnwwXT0wO0RbYyszOTI+PjJdPTA7RFtjKzM5Nj4+Ml09MDtEW2MrMzg0Pj4yXT0wO0RbYyszODg+PjJdPTA7RFtjKzM3Nj4+Ml09MDtEW2MrMzgwPj4yXT0wO0RbYyszNjg+PjJdPTA7RFtjKzM3Mj4+Ml09MDtEW2MrNDE2Pj4yXT0wO0RbYys0MjA+PjJdPTA7RFtjKzQwOD4+Ml09MjtEW2MrNDEyPj4yXT03O0RbYys0MDA+PjJdPS0xO0RbYys0MDQ+PjJdPS0xO0RbYys0MjQ+PjJdPTA7RFtjKzQyOD4+Ml09MDtEW2MrNDMyPj4yXT0wO0RbYys0MzY+PjJdPTB9Yj1EW2ErNDg+PjJdO0RbYSs0OD4+Ml09YztpZighYil7YnJlYWsgYX1iYVtEW0RbYj4+Ml0rND4+Ml1dKGIpfWM9RFthKzQ4Pj4yXTtpZihjKXticmVhayBhfXJldHVybiAwfWE9YmFbRFtEW2M+PjJdKzg+PjJdXShjLGEpfDB9ZWxzZXthPTB9cmV0dXJuIGF8MH1mdW5jdGlvbiBHZChhLGIsYyl7dmFyIGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTAsbz0wLHA9MCxxPTAscj0wLHM9MDtlPSQrLTY0fDA7JD1lO0RbZSs0OD4+Ml09MDtEW2UrNDA+PjJdPTA7RFtlKzQ0Pj4yXT0wO0RbZSszMj4+Ml09MDtEW2UrMzY+PjJdPTA7RFtlKzI0Pj4yXT0wO0RbZSsyOD4+Ml09MDtEW2UrMTY+PjJdPTA7RFtlKzIwPj4yXT0wO0RbZSs4Pj4yXT0wO0RbZSsxMj4+Ml09MDtEW2U+PjJdPTA7RFtlKzQ+PjJdPTA7aD1iO2E6e2I6e2lmKCFGW2IrMzg+PjFdKXticmVhayBifWlmKCFTYSgxLGUrMTJ8MCxoKSl7YnJlYWsgYn1sPURbZSsxMj4+Ml07Yj1EW2U+PjJdO2Q9RFtlKzQ+PjJdLWI+PjI7Yzp7aWYobD4+PjA+ZD4+PjApe3NhKGUsbC1kfDApO2w9RFtlKzEyPj4yXTticmVhayBjfWlmKGQ+Pj4wPD1sPj4+MCl7YnJlYWsgY31EW2UrND4+Ml09YisobDw8Mil9Zj0xO2lmKCFsKXticmVhayBhfXA9RFtoKzg+PjJdO3E9RFtoKzEyPj4yXTtvPURbZT4+Ml07d2hpbGUoMSl7ZD1EW2grMjA+PjJdO2o9RFtoKzE2Pj4yXTtmPTA7aWYoKHF8MCk8PShkfDApJnA+Pj4wPD1qPj4+MHwoZHwwKT4ocXwwKSl7YnJlYWsgYX1tPURbaD4+Ml07cj1FW20ranwwXTtiPWorMXwwO2Q9Yj9kOmQrMXwwO2Y9YjtEW2grMTY+PjJdPWI7RFtoKzIwPj4yXT1kO2I9cj4+PjJ8MDtpPTA7ZDp7ZTp7Zjp7Zzp7bj1yJjM7c3dpdGNoKG58MCl7Y2FzZSAzOmJyZWFrIGc7Y2FzZSAwOmJyZWFrIGU7ZGVmYXVsdDpicmVhayBmfX1iPWIrZ3wwO2Y9MDtpZihiPj4+MD49bD4+PjApe2JyZWFrIGF9cGEobysoZzw8Mil8MCwwLChyJjI1MikrNHwwKTtnPWI7YnJlYWsgZH13aGlsZSgxKXtpZigoZHwwKT49KHF8MCkmZj4+PjA+PXA+Pj4wfChkfDApPihxfDApKXticmVhayBifWo9RVtmK218MF07Zj1mKzF8MDtkPWY/ZDpkKzF8MDtEW2grMTY+PjJdPWY7RFtoKzIwPj4yXT1kO2I9ajw8KGk8PDN8Nil8YjtpPWkrMXwwO2lmKChufDApIT0oaXwwKSl7Y29udGludWV9YnJlYWt9fURbbysoZzw8Mik+PjJdPWJ9Zz1nKzF8MDtsPURbZSsxMj4+Ml07aWYoZz4+PjA8bD4+PjApe2NvbnRpbnVlfWJyZWFrfXA9ZSsxNnwwO209RFtlPj4yXTtmPURbZSsxNj4+Ml07ZD1EW2UrMjA+PjJdLWZ8MDtiPWQ+PjI7aDp7aWYoYj4+PjA8PTEwNDg1NzUpe3NhKHAsMTA0ODU3Ni1ifDApO2JyZWFrIGh9aWYoKGR8MCk9PTQxOTQzMDQpe2JyZWFrIGh9RFtlKzIwPj4yXT1mKzQxOTQzMDR9ZD1lKzI4fDA7Zz1EW2Q+PjJdO2I9RFtlKzMyPj4yXS1nPj4zO2k6e2lmKGI+Pj4wPGw+Pj4wKXtiYihkLGwtYnwwKTtnPURbZD4+Ml07YnJlYWsgaX1pZihiPj4+MD5sPj4+MCl7RFtlKzMyPj4yXT0obDw8MykrZ31pZighbCl7YnJlYWsgYn19aT0wO2Q9MDt3aGlsZSgxKXtuPShpPDwyKSttfDA7aj1EW24+PjJdO2Y9KGk8PDMpK2d8MDtiPWQ7RFtmKzQ+PjJdPWQ7RFtmPj4yXT1qO289RFtuPj4yXTtkPW8rZHwwO2lmKGQ+Pj4wPjEwNDg1NzYpe2JyZWFrIGJ9ajp7aWYoYj4+PjA+PWQ+Pj4wKXticmVhayBqfW49RFtwPj4yXTtqPTA7Zj1vJjc7aWYoZil7d2hpbGUoMSl7RFtuKyhiPDwyKT4+Ml09aTtiPWIrMXwwO2o9aisxfDA7aWYoKGZ8MCkhPShqfDApKXtjb250aW51ZX1icmVha319aWYoby0xPj4+MDw9Nil7YnJlYWsgan13aGlsZSgxKXtmPW4rKGI8PDIpfDA7RFtmPj4yXT1pO0RbZisyOD4+Ml09aTtEW2YrMjQ+PjJdPWk7RFtmKzIwPj4yXT1pO0RbZisxNj4+Ml09aTtEW2YrMTI+PjJdPWk7RFtmKzg+PjJdPWk7RFtmKzQ+PjJdPWk7Yj1iKzh8MDtpZigoZHwwKSE9KGJ8MCkpe2NvbnRpbnVlfWJyZWFrfX1pPWkrMXwwO2lmKChsfDApIT0oaXwwKSl7Y29udGludWV9YnJlYWt9az0oZHwwKT09MTA0ODU3Nn1mPWt9azp7aWYoIWZ8KERbZSsxMj4+Ml0/MDphKSl7YnJlYWsga31pZighT2EoMSxlKzU2fDAsaCkpe2JyZWFrIGt9ZD1EW2grOD4+Ml07Yj1EW2grMTY+PjJdO2o9ZC1ifDA7Zz1EW2UrNjA+PjJdO2Y9RFtoKzIwPj4yXTtkPURbaCsxMj4+Ml0tKGYrKGI+Pj4wPmQ+Pj4wKXwwKXwwO2s9RFtlKzU2Pj4yXTtpZigoZ3wwKT09KGR8MCkmaj4+PjA8az4+PjB8ZD4+PjA8Zz4+PjApe2JyZWFrIGt9ZD1mK2d8MDtqPWIra3wwO2Q9aj4+PjA8Yj4+PjA/ZCsxfDA6ZDtEW2grMTY+PjJdPWo7RFtoKzIwPj4yXT1kO2lmKChrfDApPD0wKXticmVhayBrfW09YitEW2g+PjJdfDA7RFtlKzQwPj4yXT1tO2I9ay0xfDA7Zj1tK2J8MDtkPUVbZnwwXTtsOntpZihkPj4+MDw9NjMpe0RbZSs0ND4+Ml09YjtkPUVbZnwwXSY2MzticmVhayBsfW06e3N3aXRjaCgoZD4+PjZ8MCktMXwwKXtjYXNlIDA6aWYoaz4+PjA8Mil7YnJlYWsga31iPWstMnwwO0RbZSs0ND4+Ml09YjtkPShrK218MCktMnwwO2Q9RVtkKzF8MF08PDgmMTYxMjh8RVtkfDBdO2JyZWFrIGw7Y2FzZSAxOmlmKGs+Pj4wPDMpe2JyZWFrIGt9Yj1rLTN8MDtEW2UrNDQ+PjJdPWI7ZD0oayttfDApLTN8MDtkPUVbZCsyfDBdPDwxNiY0MTI4NzY4fEVbZCsxfDBdPDw4fEVbZHwwXTticmVhayBsO2RlZmF1bHQ6YnJlYWsgbX19Yj1rLTR8MDtEW2UrNDQ+PjJdPWI7ZD0oayttfDApLTR8MDtkPUVbZCsyfDBdPDwxNnxFW2QrM3wwXTw8MjQmMTA1Njk2NDYwOHxFW2QrMXwwXTw8OHxFW2R8MF19Zz1kKzQxOTQzMDR8MDtEW2UrNDg+PjJdPWc7aWYoZz4+PjA+MTA3Mzc0MTgyMyl7YnJlYWsga31pZighYSl7cz0xO2JyZWFrIGt9bj1EW2UrMjg+PjJdO2Q9MDtqPURbZSsxNj4+Ml07d2hpbGUoMSl7bjp7aWYoZz4+PjA+NDE5NDMwMyl7YnJlYWsgbn13aGlsZSgxKXtpZigoYnwwKTw9MCl7YnJlYWsgbn1iPWItMXwwO0RbZSs0ND4+Ml09YjtnPUVbYittfDBdfGc8PDg7RFtlKzQ4Pj4yXT1nO2lmKGc+Pj4wPDQxOTQzMDQpe2NvbnRpbnVlfWJyZWFrfX1mPWcmMTA0ODU3NTtrPURbaisoZjw8Mik+PjJdO2g9bisoazw8Myl8MDtnPShKKERbaD4+Ml0sZz4+PjIwfDApK2Z8MCktRFtoKzQ+PjJdfDA7RFtlKzQ4Pj4yXT1nO0RbKGQ8PDIpK2M+PjJdPWs7cz0xO2Q9ZCsxfDA7aWYoKGR8MCkhPShhfDApKXtjb250aW51ZX1icmVha319YT1EW2UrMjg+PjJdO2lmKGEpe0RbZSszMj4+Ml09YTttYShhKX1hPURbZSsxNj4+Ml07aWYoYSl7RFtlKzIwPj4yXT1hO21hKGEpfWE9RFtlPj4yXTtpZihhKXtEW2UrND4+Ml09YTttYShhKX0kPWUtIC02NHwwO3JldHVybiBzfWZ1bmN0aW9uIHdkKGEsYil7dmFyIGM9MCxkPTAsZT0wLGY9MCxnPTAsaD0wO2M9RFthKzQ+PjJdO2Q9RFthPj4yXTtmPShjLWR8MCkvMTQ0fDA7aWYoZj4+PjA8Yj4+PjApe2Q9YTtnPWItZnwwO2M9RFthKzg+PjJdO2E9RFthKzQ+PjJdO2E6e2lmKGc+Pj4wPD0oYy1hfDApLzE0ND4+PjApe2lmKGcpe2I9SihnLDE0NCkrYXwwO3doaWxlKDEpe0RbYT4+Ml09LTE7VWMoYSs0fDApO0RbYSsxMDQ+PjJdPTA7RFthKzEwOD4+Ml09MDtCW2ErMTAwfDBdPTE7RFthKzExMj4+Ml09MDtEW2ErMTE2Pj4yXT0wO0RbYSsxMjA+PjJdPTA7RFthKzEyND4+Ml09MDtEW2ErMTI4Pj4yXT0wO0RbYSsxMzI+PjJdPTA7RFthKzEzNj4+Ml09MDtEW2ErMTQwPj4yXT0wO2E9YSsxNDR8MDtpZigoYnwwKSE9KGF8MCkpe2NvbnRpbnVlfWJyZWFrfWE9Yn1EW2QrND4+Ml09YTticmVhayBhfWI6e2M6e2Q6e2I9YTthPURbZD4+Ml07Zj0oYi1hfDApLzE0NHwwO2g9ZitnfDA7aWYoaD4+PjA8Mjk4MjYxNjIpe2I9KGMtYXwwKS8xNDR8MDthPWI8PDE7ZT1iPj4+MDwxNDkxMzA4MD9hPj4+MDxoPj4+MD9oOmE6Mjk4MjYxNjE7aWYoZSl7aWYoZT4+PjA+PTI5ODI2MTYyKXticmVhayBkfWM9bmEoSihlLDE0NCkpfWVsc2V7Yz0wfWI9YytKKGYsMTQ0KXwwO2g9YitKKGcsMTQ0KXwwO2E9Yjt3aGlsZSgxKXtEW2E+PjJdPS0xO1VjKGErNHwwKTtEW2ErMTA0Pj4yXT0wO0RbYSsxMDg+PjJdPTA7QlthKzEwMHwwXT0xO0RbYSsxMTI+PjJdPTA7RFthKzExNj4+Ml09MDtEW2ErMTIwPj4yXT0wO0RbYSsxMjQ+PjJdPTA7RFthKzEyOD4+Ml09MDtEW2ErMTMyPj4yXT0wO0RbYSsxMzY+PjJdPTA7RFthKzE0MD4+Ml09MDthPWErMTQ0fDA7aWYoKGh8MCkhPShhfDApKXtjb250aW51ZX1icmVha31mPWMrSihlLDE0NCl8MDthPURbZCs0Pj4yXTtlPURbZD4+Ml07aWYoKGF8MCk9PShlfDApKXticmVhayBjfXdoaWxlKDEpe2I9Yi0xNDR8MDthPWEtMTQ0fDA7RFtiPj4yXT1EW2E+PjJdO0RbYis0Pj4yXT1EW2ErND4+Ml07RFtiKzg+PjJdPURbYSs4Pj4yXTtEW2IrMTI+PjJdPURbYSsxMj4+Ml07RFthKzEyPj4yXT0wO0RbYSs0Pj4yXT0wO0RbYSs4Pj4yXT0wO0RbYisxNj4+Ml09RFthKzE2Pj4yXTtEW2IrMjA+PjJdPURbYSsyMD4+Ml07RFtiKzI0Pj4yXT1EW2ErMjQ+PjJdO0RbYSsyND4+Ml09MDtEW2ErMTY+PjJdPTA7RFthKzIwPj4yXT0wO2M9RVthKzI4fDBdO0RbYis0MD4+Ml09MDtEW2IrMzI+PjJdPTA7RFtiKzM2Pj4yXT0wO0JbYisyOHwwXT1jO0RbYiszMj4+Ml09RFthKzMyPj4yXTtEW2IrMzY+PjJdPURbYSszNj4+Ml07RFtiKzQwPj4yXT1EW2ErNDA+PjJdO0RbYSs0MD4+Ml09MDtEW2ErMzI+PjJdPTA7RFthKzM2Pj4yXT0wO0RbYis1Mj4+Ml09MDtEW2IrNDQ+PjJdPTA7RFtiKzQ4Pj4yXT0wO0RbYis0ND4+Ml09RFthKzQ0Pj4yXTtEW2IrNDg+PjJdPURbYSs0OD4+Ml07RFtiKzUyPj4yXT1EW2ErNTI+PjJdO0RbYSs1Mj4+Ml09MDtEW2ErNDQ+PjJdPTA7RFthKzQ4Pj4yXT0wO2M9Yi0gLTY0fDA7RFtjPj4yXT0wO0RbYis1Nj4+Ml09MDtEW2IrNjA+PjJdPTA7RFtiKzU2Pj4yXT1EW2ErNTY+PjJdO0RbYis2MD4+Ml09RFthKzYwPj4yXTtnPWM7Yz1hLSAtNjR8MDtEW2c+PjJdPURbYz4+Ml07RFtjPj4yXT0wO0RbYSs1Nj4+Ml09MDtEW2ErNjA+PjJdPTA7RFtiKzY4Pj4yXT1EW2ErNjg+PjJdO2M9RFthKzcyPj4yXTtEW2IrODQ+PjJdPTA7RFtiKzc2Pj4yXT0wO0RbYis4MD4+Ml09MDtEW2IrNzI+PjJdPWM7RFtiKzc2Pj4yXT1EW2ErNzY+PjJdO0RbYis4MD4+Ml09RFthKzgwPj4yXTtEW2IrODQ+PjJdPURbYSs4ND4+Ml07RFthKzg0Pj4yXT0wO0RbYSs3Nj4+Ml09MDtEW2ErODA+PjJdPTA7RFtiKzk2Pj4yXT0wO0RbYis4OD4+Ml09MDtEW2IrOTI+PjJdPTA7RFtiKzg4Pj4yXT1EW2ErODg+PjJdO0RbYis5Mj4+Ml09RFthKzkyPj4yXTtEW2IrOTY+PjJdPURbYSs5Nj4+Ml07RFthKzk2Pj4yXT0wO0RbYSs4OD4+Ml09MDtEW2ErOTI+PjJdPTA7Yz1FW2ErMTAwfDBdO0RbYisxMTI+PjJdPTA7RFtiKzEwND4+Ml09MDtEW2IrMTA4Pj4yXT0wO0JbYisxMDB8MF09YztEW2IrMTA0Pj4yXT1EW2ErMTA0Pj4yXTtEW2IrMTA4Pj4yXT1EW2ErMTA4Pj4yXTtEW2IrMTEyPj4yXT1EW2ErMTEyPj4yXTtEW2ErMTEyPj4yXT0wO0RbYSsxMDQ+PjJdPTA7RFthKzEwOD4+Ml09MDtEW2IrMTI0Pj4yXT0wO0RbYisxMTY+PjJdPTA7RFtiKzEyMD4+Ml09MDtEW2IrMTE2Pj4yXT1EW2ErMTE2Pj4yXTtEW2IrMTIwPj4yXT1EW2ErMTIwPj4yXTtEW2IrMTI0Pj4yXT1EW2ErMTI0Pj4yXTtEW2ErMTI0Pj4yXT0wO0RbYSsxMTY+PjJdPTA7RFthKzEyMD4+Ml09MDtjPURbYSsxMjg+PjJdO0RbYisxNDA+PjJdPTA7RFtiKzEzMj4+Ml09MDtEW2IrMTM2Pj4yXT0wO0RbYisxMjg+PjJdPWM7RFtiKzEzMj4+Ml09RFthKzEzMj4+Ml07RFtiKzEzNj4+Ml09RFthKzEzNj4+Ml07RFtiKzE0MD4+Ml09RFthKzE0MD4+Ml07RFthKzE0MD4+Ml09MDtEW2ErMTMyPj4yXT0wO0RbYSsxMzY+PjJdPTA7aWYoKGF8MCkhPShlfDApKXtjb250aW51ZX1icmVha31EW2QrOD4+Ml09ZjthPURbZCs0Pj4yXTtEW2QrND4+Ml09aDtlPURbZD4+Ml07RFtkPj4yXT1iO2lmKChhfDApPT0oZXwwKSl7YnJlYWsgYn13aGlsZSgxKXtiPURbYS0xMj4+Ml07aWYoYil7RFthLTg+PjJdPWI7bWEoYil9Yj1EW2EtMjg+PjJdO2lmKGIpe0RbYS0yND4+Ml09YjttYShiKX1iPURbYS00MD4+Ml07aWYoYil7RFthLTM2Pj4yXT1iO21hKGIpfW9iKGEtMTQwfDApO2E9YS0xNDR8MDtpZigoZXwwKSE9KGF8MCkpe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGJ9cWEoKTtUKCl9cmEoMTMyNik7VCgpfURbZCs4Pj4yXT1mO0RbZCs0Pj4yXT1oO0RbZD4+Ml09Yn1pZihlKXttYShlKX19cmV0dXJufWlmKGI+Pj4wPGY+Pj4wKXtkPWQrSihiLDE0NCl8MDtpZigoZHwwKSE9KGN8MCkpe3doaWxlKDEpe2I9RFtjLTEyPj4yXTtpZihiKXtEW2MtOD4+Ml09YjttYShiKX1iPURbYy0yOD4+Ml07aWYoYil7RFtjLTI0Pj4yXT1iO21hKGIpfWI9RFtjLTQwPj4yXTtpZihiKXtEW2MtMzY+PjJdPWI7bWEoYil9b2IoYy0xNDB8MCk7Yz1jLTE0NHwwO2lmKChkfDApIT0oY3wwKSl7Y29udGludWV9YnJlYWt9fURbYSs0Pj4yXT1kfX1mdW5jdGlvbiBKYihhLGIsYyl7YT1hfDA7Yj1ifDA7Yz1jfDA7dmFyIGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTAsbz0wLHA9MCxxPTAscj0wLHM9MCx0PTAsdT0wLHY9MCx3PTAseD0wO2U9JC05NnwwOyQ9ZTtkPURbYSsxNj4+Ml07QltlKzkyfDBdPTE7RFtlKzg4Pj4yXT1iO0RbZSs4ND4+Ml09YjtEW2UrODA+PjJdPWQ7YTp7aWYoKGJ8MCk9PS0xKXticmVhayBhfW09RFthKzIwPj4yXTtmPURbbT4+Ml07ZD1EW0RbZD4+Ml0rKGI8PDIpPj4yXTtpZihkPj4+MD49RFttKzQ+PjJdLWY+PjI+Pj4wKXticmVhayBhfWQ9RFtEW2ErOD4+Ml0rKERbZisoZDw8Mik+PjJdPDwyKT4+Ml07Zj1EW2ErND4+Ml07aWYoIUVbZis4NHwwXSl7ZD1EW0RbZis2OD4+Ml0rKGQ8PDIpPj4yXX1EW2UrNzI+PjJdPTA7RFtlKzc2Pj4yXT0wO209ZS0gLTY0fDA7RFttPj4yXT0wO0RbbSs0Pj4yXT0wO0RbZSs1Nj4+Ml09MDtEW2UrNjA+PjJdPTA7RmEoZixkLEJbZisyNHwwXSxlKzU2fDApO2Q9YisxfDA7bT0oZD4+PjApJTN8MD9kOmItMnwwO2g9KChiPj4+MCklM3wwPy0xOjIpK2J8MDtiOntjOnt3aGlsZSgxKXtmPW07ZD1oO2Q6e2lmKCFEW2ErMjg+PjJdKXticmVhayBkfWQ9YisxfDA7Zj0oZD4+PjApJTN8MD9kOmItMnwwO2Q9Yi0xfDA7aWYoKGI+Pj4wKSUzfDApe2JyZWFrIGR9ZD1iKzJ8MH1pZigoZnwwKT09LTEpe2JyZWFrIGJ9az1EW2ErMjA+PjJdO2I9RFtrPj4yXTtmPURbRFtEW2ErMTY+PjJdPj4yXSsoZjw8Mik+PjJdO2lmKGY+Pj4wPj1EW2srND4+Ml0tYj4+Mj4+PjApe2JyZWFrIGJ9Zj1EW0RbYSs4Pj4yXSsoRFsoZjw8MikrYj4+Ml08PDIpPj4yXTtiPURbYSs0Pj4yXTtpZighRVtiKzg0fDBdKXtmPURbRFtiKzY4Pj4yXSsoZjw8Mik+PjJdfURbZSs0OD4+Ml09MDtEW2UrNTI+PjJdPTA7RFtlKzQwPj4yXT0wO0RbZSs0ND4+Ml09MDtEW2UrMzI+PjJdPTA7RFtlKzM2Pj4yXT0wO0ZhKGIsZixCW2IrMjR8MF0sZSszMnwwKTtpZigoZHwwKT09LTEpe2JyZWFrIGN9Zj1EW2ErMjA+PjJdO2I9RFtmPj4yXTtkPURbRFtEW2ErMTY+PjJdPj4yXSsoZDw8Mik+PjJdO2lmKGQ+Pj4wPj1EW2YrND4+Ml0tYj4+Mj4+PjApe2JyZWFrIGN9Zj1EW0RbYSs4Pj4yXSsoRFtiKyhkPDwyKT4+Ml08PDIpPj4yXTtiPURbYSs0Pj4yXTtpZighRVtiKzg0fDBdKXtmPURbRFtiKzY4Pj4yXSsoZjw8Mik+PjJdfURbZSsyND4+Ml09MDtEW2UrMjg+PjJdPTA7RFtlKzE2Pj4yXT0wO0RbZSsyMD4+Ml09MDtEW2UrOD4+Ml09MDtEW2UrMTI+PjJdPTA7RmEoYixmLEJbYisyNHwwXSxlKzh8MCk7ZD1EW2UrOD4+Ml07Yj1EW2UrNTY+PjJdO2Y9ZC1ifDA7az1EW2UrNjA+PjJdO249RFtlKzEyPj4yXS0oaysoYj4+PjA+ZD4+PjApfDApfDA7aT1EW2UrNDA+PjJdO2Q9RFtlKzY0Pj4yXTtzPWktZHwwO3Q9RFtlKzY4Pj4yXTtpPURbZSs0ND4+Ml0tKHQrKGQ+Pj4wPmk+Pj4wKXwwKXwwO3U9JGgoZixuLHMsaSk7dj1qLXV8MDtnPWctKGFhKyhqPj4+MDx1Pj4+MCl8MCl8MDt3PXY7aj1EW2UrMTY+PjJdO3U9ai1kfDA7dD1EW2UrMjA+PjJdLSgoZD4+PjA+aj4+PjApK3R8MCl8MDtqPURbZSszMj4+Ml07dj1qLWJ8MDtrPURbZSszNj4+Ml0tKChiPj4+MD5qPj4+MCkra3wwKXwwO2Q9JGgodSx0LHYsayk7aj13K2R8MDtiPWFhK2d8MDtiPWQ+Pj4wPmo+Pj4wP2IrMXwwOmI7Zz1iO3c9bztwPW47Yj1EW2UrNDg+PjJdO2Q9RFtlKzcyPj4yXTtuPWItZHwwO289RFtlKzc2Pj4yXTt4PURbZSs1Mj4+Ml0tKG8rKGI+Pj4wPGQ+Pj4wKXwwKXwwO3A9JGgoZixwLG4seCk7Zj13K3B8MDtiPWFhK2x8MDtiPWY+Pj4wPHA+Pj4wP2IrMXwwOmI7bD1EW2UrMjQ+PjJdO3A9bC1kfDA7ZD1EW2UrMjg+PjJdLSgoZD4+PjA+bD4+PjApK298MCl8MDtsPSRoKHAsZCx2LGspO289Zi1sfDA7bD1iLShhYSsoZj4+PjA8bD4+PjApfDApfDA7Yj0kaCh1LHQsbix4KTtmPXEtYnwwO2I9ci0oYWErKGI+Pj4wPnE+Pj4wKXwwKXwwO3I9JGgocCxkLHMsaSk7cT1yK2Z8MDtiPWFhK2J8MDtiPXE+Pj4wPHI+Pj4wP2IrMXwwOmI7cj1iO2tjKGUrODB8MCk7Yj1EW2UrODg+PjJdO2lmKChifDApIT0tMSl7Y29udGludWV9YnJlYWt9Yj1yPj4zMTtmPWIrcXwwO2Q9YjtiPWIrcnwwO2s9Zl5kO2g9ZF4oZj4+PjA8ZD4+PjA/YisxfDA6Yik7bj0tMTtmPTIxNDc0ODM2NDc7Yj1sPj4zMTtpPWI7ZD1iK298MDtiPWIrbHwwO2I9ZD4+PjA8aT4+PjA/YisxfDA6YjtkPWReaTtiPWJeaTtpPWI7cz1kXi0xO2I9Yl4yMTQ3NDgzNjQ3O209ZztlOntmOntpZighRFthKzI4Pj4yXSl7aWYoKGJ8MCk9PShofDApJms+Pj4wPnM+Pj4wfGI+Pj4wPGg+Pj4wKXticmVhayBlfWI9aCtpfDA7YT1kK2t8MDtiPWE+Pj4wPGQ+Pj4wP2IrMXwwOmI7ZD1hO2E9YjtiPWc+PjMxO2Y9YitqfDA7aD1nO2c9YjtiPWgrYnwwO2I9Zj4+PjA8Zz4+PjA/YisxfDA6YjtoPWZeZztmPWgrZHwwO2c9Yl5nO2I9ZjtnPWdeMjE0NzQ4MzY0NzthPShnfDApPT0oYXwwKSYoaF4tMSk+Pj4wPGQ+Pj4wfGE+Pj4wPmc+Pj4wO2c9IShhJjApO2E9YT8tMTpiO2lmKGcmKGF8MCk8PTUzNjg3MDkxMnwoYXwwKTw1MzY4NzA5MTIpe2JyZWFrIGV9Yj0wO2E9YT4+PjI5fDA7YnJlYWsgZn1nOntpZigoYnwwKT09KGh8MCkmaz4+PjA+cz4+PjB8Yj4+PjA8aD4+PjApe2JyZWFrIGd9Yj1oK2l8MDthPWQra3wwO2I9YT4+PjA8ZD4+PjA/YisxfDA6YjtkPWI7aD1nO2I9Zz4+MzE7Zz1iK2p8MDtpPWg7aD1iO2I9aStifDA7Yj1nPj4+MDxoPj4+MD9iKzF8MDpiO2c9Z15oO2I9Yl5oO2g9Yl4yMTQ3NDgzNjQ3O2lmKChofDApPT0oZHwwKSYoZ14tMSk+Pj4wPGE+Pj4wfGQ+Pj4wPmg+Pj4wKXticmVhayBnfWI9YitkfDA7YT1hK2d8MDtiPWE+Pj4wPGc+Pj4wP2IrMXwwOmI7bj1hO2Y9YjtpZighYiZhPj4+MDw1MzY4NzA5MTMpe2JyZWFrIGV9fWI9Zj4+PjI5fDA7YT0oZiY1MzY4NzA5MTEpPDwzfG4+Pj4yOX1qPWFpKGosbSxhLGIpO289YWkobyxsLGEsYik7cT1haShxLHIsYSxiKX1EW2MrOD4+Ml09ajtEW2MrND4+Ml09bztEW2M+PjJdPXE7JD1lKzk2fDA7cmV0dXJufXVhKCk7VCgpfXVhKCk7VCgpfXVhKCk7VCgpfWZ1bmN0aW9uIEljKGEsYil7YT1hfDA7Yj1ifDA7dmFyIGM9MCxkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTA7YTp7aWYoKGJ8MCk8MCl7YnJlYWsgYX1kPURbYSsxMj4+Ml07Yz1EW2ErOD4+Ml07aWYoZC1jPj4yPj4+MDw9Yj4+PjApe2JyZWFrIGF9ZT1jKyhiPDwyKXwwO2Y9RFtlPj4yXTtpPURbZis2MD4+Ml07Zz1EW2YrNTY+PjJdO2M9ZSs0fDA7Yjp7aWYoKGN8MCkhPShkfDApKXt3aGlsZSgxKXtoPURbYz4+Ml07RFtjPj4yXT0wO0RbZT4+Ml09aDtpZihmKXt5YShmKX1lPWUrNHwwO2M9Yys0fDA7aWYoKGN8MCkhPShkfDApKXtmPURbZT4+Ml07Y29udGludWV9YnJlYWt9ZD1EW2ErMTI+PjJdO2lmKChlfDApPT0oZHwwKSl7YnJlYWsgYn19d2hpbGUoMSl7ZD1kLTR8MDtjPURbZD4+Ml07RFtkPj4yXT0wO2lmKGMpe3lhKGMpfWlmKChkfDApIT0oZXwwKSl7Y29udGludWV9YnJlYWt9fURbYSsxMj4+Ml09ZTtmPURbYSs0Pj4yXTtjOntpZighZnwoaXwwKTwwKXticmVhayBjfWQ9RFtmKzI0Pj4yXTtjPURbZisyOD4+Ml07aWYoKGR8MCk9PShjfDApKXticmVhayBjfXdoaWxlKDEpe2lmKChpfDApPT1EW0RbZD4+Ml0rMjQ+PjJdKXtlPWQrNHwwO2k9RFtmKzI4Pj4yXTtkOntpZigoZXwwKSE9KGl8MCkpe3doaWxlKDEpe2g9RFtlPj4yXTtEW2U+PjJdPTA7Yz1EW2Q+PjJdO0RbZD4+Ml09aDtpZihjKXtDYShjKzEyfDAsRFtjKzE2Pj4yXSk7QmEoYyxEW2MrND4+Ml0pO21hKGMpfWQ9ZCs0fDA7ZT1lKzR8MDtpZigoaXwwKSE9KGV8MCkpe2NvbnRpbnVlfWJyZWFrfWU9RFtmKzI4Pj4yXTtpZigoZXwwKT09KGR8MCkpe2JyZWFrIGR9fXdoaWxlKDEpe2U9ZS00fDA7Yz1EW2U+PjJdO0RbZT4+Ml09MDtpZihjKXtDYShjKzEyfDAsRFtjKzE2Pj4yXSk7QmEoYyxEW2MrND4+Ml0pO21hKGMpfWlmKChkfDApIT0oZXwwKSl7Y29udGludWV9YnJlYWt9fURbZisyOD4+Ml09ZDticmVhayBjfWQ9ZCs0fDA7aWYoKGN8MCkhPShkfDApKXtjb250aW51ZX1icmVha319ZTp7aWYoKGd8MCk+NCl7YnJlYWsgZX1mOntlPUooZywxMikrYXwwO2Q9RFtlKzIwPj4yXTtjPURbZSsyND4+Ml07aWYoKGR8MCk9PShjfDApKXticmVhayBmfXdoaWxlKDEpe2lmKERbZD4+Ml09PShifDApKXticmVhayBmfWQ9ZCs0fDA7aWYoKGN8MCkhPShkfDApKXtjb250aW51ZX1icmVha31icmVhayBlfWlmKChkfDApPT0oY3wwKSl7YnJlYWsgZX1mPWQrNHwwO2M9Yy1mfDA7aWYoYyl7TmEoZCxmLGMpfURbZSsyND4+Ml09ZCtjfWU9RFthKzIwPj4yXTtkPURbYSsyND4+Ml0tZXwwO2c6e2lmKCFkKXticmVhayBnfWM9ZD4+MjtmPWM+Pj4wPjE/YzoxO2k9ZiYxO2Q9MDtpZihjPj4+MD49Mil7Zj1mJi0yO2M9MDt3aGlsZSgxKXtnPWQ8PDI7aD1nK2V8MDtqPURbaD4+Ml07aWYoKGp8MCk+KGJ8MCkpe0RbaD4+Ml09ai0xfWc9ZSsoZ3w0KXwwO2g9RFtnPj4yXTtpZigoaHwwKT4oYnwwKSl7RFtnPj4yXT1oLTF9ZD1kKzJ8MDtjPWMrMnwwO2lmKChmfDApIT0oY3wwKSl7Y29udGludWV9YnJlYWt9fWlmKCFpKXticmVhayBnfWQ9ZSsoZDw8Mil8MDtjPURbZD4+Ml07aWYoKGN8MCk8PShifDApKXticmVhayBnfURbZD4+Ml09Yy0xfWU9RFthKzMyPj4yXTtkPURbYSszNj4+Ml0tZXwwO2g6e2lmKCFkKXticmVhayBofWM9ZD4+MjtmPWM+Pj4wPjE/YzoxO2k9ZiYxO2Q9MDtpZihjPj4+MD49Mil7Zj1mJi0yO2M9MDt3aGlsZSgxKXtnPWQ8PDI7aD1nK2V8MDtqPURbaD4+Ml07aWYoKGp8MCk+KGJ8MCkpe0RbaD4+Ml09ai0xfWc9ZSsoZ3w0KXwwO2g9RFtnPj4yXTtpZigoaHwwKT4oYnwwKSl7RFtnPj4yXT1oLTF9ZD1kKzJ8MDtjPWMrMnwwO2lmKChmfDApIT0oY3wwKSl7Y29udGludWV9YnJlYWt9fWlmKCFpKXticmVhayBofWQ9ZSsoZDw8Mil8MDtjPURbZD4+Ml07aWYoKGN8MCk8PShifDApKXticmVhayBofURbZD4+Ml09Yy0xfWU9RFthKzQ0Pj4yXTtkPURbYSs0OD4+Ml0tZXwwO2k6e2lmKCFkKXticmVhayBpfWM9ZD4+MjtmPWM+Pj4wPjE/YzoxO2k9ZiYxO2Q9MDtpZihjPj4+MD49Mil7Zj1mJi0yO2M9MDt3aGlsZSgxKXtnPWQ8PDI7aD1nK2V8MDtqPURbaD4+Ml07aWYoKGp8MCk+KGJ8MCkpe0RbaD4+Ml09ai0xfWc9ZSsoZ3w0KXwwO2g9RFtnPj4yXTtpZigoaHwwKT4oYnwwKSl7RFtnPj4yXT1oLTF9ZD1kKzJ8MDtjPWMrMnwwO2lmKChmfDApIT0oY3wwKSl7Y29udGludWV9YnJlYWt9fWlmKCFpKXticmVhayBpfWQ9ZSsoZDw8Mil8MDtjPURbZD4+Ml07aWYoKGN8MCk8PShifDApKXticmVhayBpfURbZD4+Ml09Yy0xfWU9RFthKzU2Pj4yXTtkPURbYSs2MD4+Ml0tZXwwO2o6e2lmKCFkKXticmVhayBqfWM9ZD4+MjtmPWM+Pj4wPjE/YzoxO2k9ZiYxO2Q9MDtpZihjPj4+MD49Mil7Zj1mJi0yO2M9MDt3aGlsZSgxKXtnPWQ8PDI7aD1nK2V8MDtqPURbaD4+Ml07aWYoKGp8MCk+KGJ8MCkpe0RbaD4+Ml09ai0xfWc9ZSsoZ3w0KXwwO2g9RFtnPj4yXTtpZigoaHwwKT4oYnwwKSl7RFtnPj4yXT1oLTF9ZD1kKzJ8MDtjPWMrMnwwO2lmKChmfDApIT0oY3wwKSl7Y29udGludWV9YnJlYWt9fWlmKCFpKXticmVhayBqfWQ9ZSsoZDw8Mil8MDtjPURbZD4+Ml07aWYoKGN8MCk8PShifDApKXticmVhayBqfURbZD4+Ml09Yy0xfWM9RFthKzcyPj4yXTthPURbYSs2OD4+Ml07ZD1jLWF8MDtpZighZCl7YnJlYWsgYX1jPWQ+PjI7ZT1jPj4+MD4xP2M6MTtmPWUmMTtkPTA7aWYoYz4+PjA+PTIpe2U9ZSYtMjtjPTA7d2hpbGUoMSl7aT1kPDwyO2c9aSthfDA7aD1EW2c+PjJdO2lmKChofDApPihifDApKXtEW2c+PjJdPWgtMX1pPWErKGl8NCl8MDtnPURbaT4+Ml07aWYoKGd8MCk+KGJ8MCkpe0RbaT4+Ml09Zy0xfWQ9ZCsyfDA7Yz1jKzJ8MDtpZigoZXwwKSE9KGN8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZighZil7YnJlYWsgYX1jPWI7YT1hKyhkPDwyKXwwO2I9RFthPj4yXTtpZigoY3wwKT49KGJ8MCkpe2JyZWFrIGF9RFthPj4yXT1iLTF9fWZ1bmN0aW9uIG1hKGEpe2E9YXwwO3ZhciBiPTAsYz0wLGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MDthOntpZighYSl7YnJlYWsgYX1kPWEtOHwwO2I9RFthLTQ+PjJdO2E9YiYtODtmPWQrYXwwO2I6e2lmKGImMSl7YnJlYWsgYn1pZighKGImMykpe2JyZWFrIGF9Yj1EW2Q+PjJdO2Q9ZC1ifDA7aWYoZD4+PjA8R1syODg1XSl7YnJlYWsgYX1hPWErYnwwO2lmKERbMjg4Nl0hPShkfDApKXtpZihiPj4+MDw9MjU1KXtlPURbZCs4Pj4yXTtiPWI+Pj4zfDA7Yz1EW2QrMTI+PjJdO2lmKChjfDApPT0oZXwwKSl7aT0xMTUyNCxqPURbMjg4MV0mZGkoYiksRFtpPj4yXT1qO2JyZWFrIGJ9RFtlKzEyPj4yXT1jO0RbYys4Pj4yXT1lO2JyZWFrIGJ9aD1EW2QrMjQ+PjJdO2I9RFtkKzEyPj4yXTtjOntpZigoZHwwKSE9KGJ8MCkpe2M9RFtkKzg+PjJdO0RbYysxMj4+Ml09YjtEW2IrOD4+Ml09YzticmVhayBjfWQ6e2U9ZCsyMHwwO2M9RFtlPj4yXTtpZihjKXticmVhayBkfWU9ZCsxNnwwO2M9RFtlPj4yXTtpZihjKXticmVhayBkfWI9MDticmVhayBjfXdoaWxlKDEpe2c9ZTtiPWM7ZT1iKzIwfDA7Yz1EW2U+PjJdO2lmKGMpe2NvbnRpbnVlfWU9YisxNnwwO2M9RFtiKzE2Pj4yXTtpZihjKXtjb250aW51ZX1icmVha31EW2c+PjJdPTB9aWYoIWgpe2JyZWFrIGJ9ZT1EW2QrMjg+PjJdO2M9KGU8PDIpKzExODI4fDA7ZTp7aWYoRFtjPj4yXT09KGR8MCkpe0RbYz4+Ml09YjtpZihiKXticmVhayBlfWk9MTE1Mjgsaj1EWzI4ODJdJmRpKGUpLERbaT4+Ml09ajticmVhayBifURbaCsoRFtoKzE2Pj4yXT09KGR8MCk/MTY6MjApPj4yXT1iO2lmKCFiKXticmVhayBifX1EW2IrMjQ+PjJdPWg7Yz1EW2QrMTY+PjJdO2lmKGMpe0RbYisxNj4+Ml09YztEW2MrMjQ+PjJdPWJ9Yz1EW2QrMjA+PjJdO2lmKCFjKXticmVhayBifURbYisyMD4+Ml09YztEW2MrMjQ+PjJdPWI7YnJlYWsgYn1iPURbZis0Pj4yXTtpZigoYiYzKSE9Myl7YnJlYWsgYn1EWzI4ODNdPWE7RFtmKzQ+PjJdPWImLTI7RFtkKzQ+PjJdPWF8MTtEW2ErZD4+Ml09YTtyZXR1cm59aWYoZD4+PjA+PWY+Pj4wKXticmVhayBhfWI9RFtmKzQ+PjJdO2lmKCEoYiYxKSl7YnJlYWsgYX1mOntpZighKGImMikpe2lmKERbMjg4N109PShmfDApKXtEWzI4ODddPWQ7YT1EWzI4ODRdK2F8MDtEWzI4ODRdPWE7RFtkKzQ+PjJdPWF8MTtpZihEWzI4ODZdIT0oZHwwKSl7YnJlYWsgYX1EWzI4ODNdPTA7RFsyODg2XT0wO3JldHVybn1pZihEWzI4ODZdPT0oZnwwKSl7RFsyODg2XT1kO2E9RFsyODgzXSthfDA7RFsyODgzXT1hO0RbZCs0Pj4yXT1hfDE7RFthK2Q+PjJdPWE7cmV0dXJufWE9KGImLTgpK2F8MDtnOntpZihiPj4+MDw9MjU1KXtlPURbZis4Pj4yXTtiPWI+Pj4zfDA7Yz1EW2YrMTI+PjJdO2lmKChjfDApPT0oZXwwKSl7aT0xMTUyNCxqPURbMjg4MV0mZGkoYiksRFtpPj4yXT1qO2JyZWFrIGd9RFtlKzEyPj4yXT1jO0RbYys4Pj4yXT1lO2JyZWFrIGd9aD1EW2YrMjQ+PjJdO2I9RFtmKzEyPj4yXTtoOntpZigoZnwwKSE9KGJ8MCkpe2M9RFtmKzg+PjJdO0RbYysxMj4+Ml09YjtEW2IrOD4+Ml09YzticmVhayBofWk6e2U9ZisyMHwwO2M9RFtlPj4yXTtpZihjKXticmVhayBpfWU9ZisxNnwwO2M9RFtlPj4yXTtpZihjKXticmVhayBpfWI9MDticmVhayBofXdoaWxlKDEpe2c9ZTtiPWM7ZT1iKzIwfDA7Yz1EW2U+PjJdO2lmKGMpe2NvbnRpbnVlfWU9YisxNnwwO2M9RFtiKzE2Pj4yXTtpZihjKXtjb250aW51ZX1icmVha31EW2c+PjJdPTB9aWYoIWgpe2JyZWFrIGd9ZT1EW2YrMjg+PjJdO2M9KGU8PDIpKzExODI4fDA7ajp7aWYoRFtjPj4yXT09KGZ8MCkpe0RbYz4+Ml09YjtpZihiKXticmVhayBqfWk9MTE1Mjgsaj1EWzI4ODJdJmRpKGUpLERbaT4+Ml09ajticmVhayBnfURbaCsoRFtoKzE2Pj4yXT09KGZ8MCk/MTY6MjApPj4yXT1iO2lmKCFiKXticmVhayBnfX1EW2IrMjQ+PjJdPWg7Yz1EW2YrMTY+PjJdO2lmKGMpe0RbYisxNj4+Ml09YztEW2MrMjQ+PjJdPWJ9Yz1EW2YrMjA+PjJdO2lmKCFjKXticmVhayBnfURbYisyMD4+Ml09YztEW2MrMjQ+PjJdPWJ9RFtkKzQ+PjJdPWF8MTtEW2ErZD4+Ml09YTtpZihEWzI4ODZdIT0oZHwwKSl7YnJlYWsgZn1EWzI4ODNdPWE7cmV0dXJufURbZis0Pj4yXT1iJi0yO0RbZCs0Pj4yXT1hfDE7RFthK2Q+PjJdPWF9aWYoYT4+PjA8PTI1NSl7YT1hPj4+M3wwO2I9KGE8PDMpKzExNTY0fDA7Yz1EWzI4ODFdO2E9MTw8YTtrOntpZighKGMmYSkpe0RbMjg4MV09YXxjO2E9YjticmVhayBrfWE9RFtiKzg+PjJdfURbYis4Pj4yXT1kO0RbYSsxMj4+Ml09ZDtEW2QrMTI+PjJdPWI7RFtkKzg+PjJdPWE7cmV0dXJufWU9MzE7RFtkKzE2Pj4yXT0wO0RbZCsyMD4+Ml09MDtpZihhPj4+MDw9MTY3NzcyMTUpe2I9YT4+Pjh8MDtnPWIrMTA0ODMyMD4+PjE2Jjg7Yj1iPDxnO2U9Yis1MjAxOTI+Pj4xNiY0O2I9Yjw8ZTtjPWIrMjQ1NzYwPj4+MTYmMjtiPShiPDxjPj4+MTV8MCktKGN8KGV8ZykpfDA7ZT0oYjw8MXxhPj4+YisyMSYxKSsyOHwwfURbZCsyOD4+Ml09ZTtnPShlPDwyKSsxMTgyOHwwO2w6e206e2M9RFsyODgyXTtiPTE8PGU7bjp7aWYoIShjJmIpKXtEWzI4ODJdPWJ8YztEW2c+PjJdPWQ7RFtkKzI0Pj4yXT1nO2JyZWFrIG59ZT1hPDwoKGV8MCk9PTMxPzA6MjUtKGU+Pj4xfDApfDApO2I9RFtnPj4yXTt3aGlsZSgxKXtjPWI7aWYoKERbYis0Pj4yXSYtOCk9PShhfDApKXticmVhayBtfWI9ZT4+PjI5fDA7ZT1lPDwxO2c9YysoYiY0KXwwO2I9RFtnKzE2Pj4yXTtpZihiKXtjb250aW51ZX1icmVha31EW2crMTY+PjJdPWQ7RFtkKzI0Pj4yXT1jfURbZCsxMj4+Ml09ZDtEW2QrOD4+Ml09ZDticmVhayBsfWE9RFtjKzg+PjJdO0RbYSsxMj4+Ml09ZDtEW2MrOD4+Ml09ZDtEW2QrMjQ+PjJdPTA7RFtkKzEyPj4yXT1jO0RbZCs4Pj4yXT1hfWE9RFsyODg5XS0xfDA7RFsyODg5XT1hP2E6LTF9fWZ1bmN0aW9uIHFkKGEsYixjLGQpe3ZhciBlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPUsoMCksbD0wLG09MCxuPUsoMCk7aj1EW2M+PjJdO2E6e2I6e2Y9RFtiKzQ+PjJdO2lmKCFmKXticmVhayBifWc9Y2koZik7Yzp7aWYoZz4+PjA+PTIpe2U9ajtpZihlPj4+MD49Zj4+PjApe2U9KGo+Pj4wKSUoZj4+PjApfDB9Yz1EW0RbYj4+Ml0rKGU8PDIpPj4yXTtpZighYyl7YnJlYWsgYn1pZihnPj4+MDw9MSl7YnJlYWsgY313aGlsZSgxKXtjPURbYz4+Ml07aWYoIWMpe2JyZWFrIGJ9Zz1EW2MrND4+Ml07aWYoKGd8MCkhPShqfDApKXtpZihmPj4+MDw9Zz4+PjApe2c9KGc+Pj4wKSUoZj4+PjApfDB9aWYoKGV8MCkhPShnfDApKXticmVhayBifX1pZihEW2MrOD4+Ml0hPShqfDApKXtjb250aW51ZX1icmVha31iPTA7YnJlYWsgYX1lPWYtMSZqO2M9RFtEW2I+PjJdKyhlPDwyKT4+Ml07aWYoIWMpe2JyZWFrIGJ9fWc9Zi0xfDA7d2hpbGUoMSl7Yz1EW2M+PjJdO2lmKCFjKXticmVhayBifWg9RFtjKzQ+PjJdO2lmKChofDApIT0oanwwKSYoZyZoKSE9KGV8MCkpe2JyZWFrIGJ9aWYoRFtjKzg+PjJdIT0oanwwKSl7Y29udGludWV9YnJlYWt9Yj0wO2JyZWFrIGF9Yz1uYSgxNik7ZD1EW0RbZD4+Ml0+PjJdO0RbYysxMj4+Ml09MDtEW2MrOD4+Ml09ZDtEW2MrND4+Ml09ajtEW2M+PjJdPTA7bj1LKERbYisxMj4+Ml0rMT4+PjApO2s9SFtiKzE2Pj4yXTtkOntpZighKGY/bj5LKGsqSyhmPj4+MCkpOjEpKXticmVhayBkfWc9KGYtMSZmKSE9MHxmPj4+MDwzfGY8PDE7ZT0yO2s9SyhRKEsobi9rKSkpO2U6e2lmKGs8Syg0Mjk0OTY3Mjk2KSZrPj1LKDApKXtkPX5+az4+PjA7YnJlYWsgZX1kPTB9ZD1kPj4+MD5nPj4+MD9kOmc7Zjp7aWYoKGR8MCk9PTEpe2JyZWFrIGZ9aWYoIShkJmQtMSkpe2U9ZDticmVhayBmfWU9SGMoZCk7Zj1EW2IrND4+Ml19Zzp7aWYoZT4+PjA8PWY+Pj4wKXtpZihlPj4+MD49Zj4+PjApe2JyZWFrIGd9Zz1mPj4+MDwzO2s9SyhRKEsoSyhHW2IrMTI+PjJdKS9IW2IrMTY+PjJdKSkpO2g6e2lmKGs8Syg0Mjk0OTY3Mjk2KSZrPj1LKDApKXtkPX5+az4+PjA7YnJlYWsgaH1kPTB9aTp7ajp7aWYoZyl7YnJlYWsgan1pZihjaShmKT4+PjA+MSl7YnJlYWsgan1kPWQ+Pj4wPDI/ZDoxPDwzMi1NKGQtMXwwKTticmVhayBpfWQ9SGMoZCl9ZT1kPj4+MD5lPj4+MD9kOmU7aWYoZT4+PjA+PWY+Pj4wKXticmVhayBnfX1mPTA7aD1lO2s6e2w6e206e246e2lmKGUpe2lmKGg+Pj4wPj0xMDczNzQxODI0KXticmVhayBufWU9bmEoaDw8Mik7ZD1EW2I+PjJdO0RbYj4+Ml09ZTtpZihkKXttYShkKX1EW2IrND4+Ml09aDtlPTA7aWYoaC0xPj4+MD49Myl7Zz1oJi00O3doaWxlKDEpe2Q9ZTw8MjtEW2QrRFtiPj4yXT4+Ml09MDtEW0RbYj4+Ml0rKGR8NCk+PjJdPTA7RFtEW2I+PjJdKyhkfDgpPj4yXT0wO0RbRFtiPj4yXSsoZHwxMik+PjJdPTA7ZT1lKzR8MDtmPWYrNHwwO2lmKChnfDApIT0oZnwwKSl7Y29udGludWV9YnJlYWt9fWQ9aCYzO2lmKGQpe3doaWxlKDEpe0RbRFtiPj4yXSsoZTw8Mik+PjJdPTA7ZT1lKzF8MDtpPWkrMXwwO2lmKChkfDApIT0oaXwwKSl7Y29udGludWV9YnJlYWt9fWY9RFtiKzg+PjJdO2lmKCFmKXticmVhayBrfWQ9Yis4fDA7Zz1EW2YrND4+Ml07ZT1jaShoKTtpZihlPj4+MDwyKXticmVhayBtfWc9Zz4+PjA+PWg+Pj4wPyhnPj4+MCklKGg+Pj4wKXwwOmc7RFtEW2I+PjJdKyhnPDwyKT4+Ml09ZDtkPURbZj4+Ml07aWYoIWQpe2JyZWFrIGt9aWYoZT4+PjA8PTEpe2JyZWFrIGx9d2hpbGUoMSl7aT1EW2QrND4+Ml07aWYoaD4+PjA8PWk+Pj4wKXtpPShpPj4+MCklKGg+Pj4wKXwwfW86e2lmKChnfDApPT0oaXwwKSl7Zj1kO2JyZWFrIG99ZT1kO2w9aTw8MjttPWwrRFtiPj4yXXwwO2lmKCFEW20+PjJdKXtEW20+PjJdPWY7Zj1kO2c9aTticmVhayBvfXdoaWxlKDEpe2k9ZTtlPURbZT4+Ml07aWYoRFtkKzg+PjJdPT1EW2UrOD4+Ml0/ZTowKXtjb250aW51ZX1icmVha31EW2Y+PjJdPWU7RFtpPj4yXT1EW0RbbCtEW2I+PjJdPj4yXT4+Ml07RFtEW2wrRFtiPj4yXT4+Ml0+PjJdPWR9ZD1EW2Y+PjJdO2lmKGQpe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGt9ZD1EW2I+PjJdO0RbYj4+Ml09MDtpZihkKXttYShkKX1EW2IrND4+Ml09MDticmVhayBrfXJhKDEzMjYpO1QoKX1nPWgtMSZnO0RbRFtiPj4yXSsoZzw8Mik+PjJdPWQ7ZD1EW2Y+PjJdO2lmKCFkKXticmVhayBrfX1sPWgtMXwwO3doaWxlKDEpe2g9bCZEW2QrND4+Ml07cDp7aWYoKGh8MCk9PShnfDApKXtmPWQ7YnJlYWsgcH1lPWQ7aT1oPDwyO209aStEW2I+PjJdfDA7aWYoRFttPj4yXSl7d2hpbGUoMSl7aD1lO2U9RFtlPj4yXTtpZihEW2QrOD4+Ml09PURbZSs4Pj4yXT9lOjApe2NvbnRpbnVlfWJyZWFrfURbZj4+Ml09ZTtEW2g+PjJdPURbRFtpK0RbYj4+Ml0+PjJdPj4yXTtEW0RbaStEW2I+PjJdPj4yXT4+Ml09ZDticmVhayBwfURbbT4+Ml09ZjtmPWQ7Zz1ofWQ9RFtmPj4yXTtpZihkKXtjb250aW51ZX1icmVha319fWY9RFtiKzQ+PjJdO2Q9Zi0xfDA7aWYoIShkJmYpKXtlPWQmajticmVhayBkfWlmKGY+Pj4wPmo+Pj4wKXtlPWo7YnJlYWsgZH1lPShqPj4+MCklKGY+Pj4wKXwwfWU9RFtiPj4yXSsoZTw8Mil8MDtkPURbZT4+Ml07cTp7aWYoIWQpe0RbYz4+Ml09RFtiKzg+PjJdO0RbYis4Pj4yXT1jO0RbZT4+Ml09Yis4O2Q9RFtjPj4yXTtpZighZCl7YnJlYWsgcX1kPURbZCs0Pj4yXTtlPWYtMXwwO3I6e2lmKCEoZSZmKSl7ZD1kJmU7YnJlYWsgcn1pZihkPj4+MDxmPj4+MCl7YnJlYWsgcn1kPShkPj4+MCklKGY+Pj4wKXwwfURbRFtiPj4yXSsoZDw8Mik+PjJdPWM7YnJlYWsgcX1EW2M+PjJdPURbZD4+Ml07RFtkPj4yXT1jfURbYisxMj4+Ml09RFtiKzEyPj4yXSsxO2I9MX1CW2ErNHwwXT1iO0RbYT4+Ml09Y31mdW5jdGlvbiBZaChhLGIsYyxkLGUsZil7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7ZT1lfDA7Zj1mfDA7dmFyIGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTAsbz0wLHA9MCxxPTAscj0wLHM9MCx0PTAsdT0wO0RbYSs4Pj4yXT1lO2Q9YSszMnwwO2s9RFtkPj4yXTtmPURbYSszNj4+Ml0taz4+MjthOntpZihmPj4+MDxlPj4+MCl7c2EoZCxlLWZ8MCk7az1EW2Q+PjJdO2Q9RFthKzg+PjJdO2JyZWFrIGF9aWYoZT4+PjA8Zj4+PjApe0RbYSszNj4+Ml09KGU8PDIpK2t9ZD1lfXE9RFthKzUyPj4yXTtuPURbYSs0OD4+Ml07Zj0wO209KGUmMTA3Mzc0MTgyMykhPShlfDApPy0xOmU8PDI7bT1wYShuYShtKSwwLG0pO2I6e2lmKChkfDApPD0wKXticmVhayBifXdoaWxlKDEpe2Q9Zjw8MjtnPURbZCttPj4yXTtpPURbYSsxNj4+Ml07Yzp7aWYoKGd8MCk+KGl8MCkpe0RbZCtrPj4yXT1pO2JyZWFrIGN9ZD1kK2t8MDtpPURbYSsxMj4+Ml07aWYoKGl8MCk+KGd8MCkpe0RbZD4+Ml09aTticmVhayBjfURbZD4+Ml09Z31kPURbYSs4Pj4yXTtmPWYrMXwwO2lmKChkfDApPihmfDApKXtjb250aW51ZX1icmVha31pZigoZHwwKTw9MCl7YnJlYWsgYn1mPTA7d2hpbGUoMSl7Zz1mPDwyO2Q9ZytjfDA7Zz1EW2IrZz4+Ml0rRFtnK2s+PjJdfDA7RFtkPj4yXT1nO2Q6e2lmKChnfDApPkRbYSsxNj4+Ml0pe2g9Zy1EW2ErMjA+PjJdfDB9ZWxzZXtpZigoZ3wwKT49RFthKzEyPj4yXSl7YnJlYWsgZH1oPWcrRFthKzIwPj4yXXwwfURbZD4+Ml09aH1kPURbYSs4Pj4yXTtmPWYrMXwwO2lmKChkfDApPihmfDApKXtjb250aW51ZX1icmVha319Zj1EW2ErNTY+PjJdO289RFtmPj4yXTtmPURbZis0Pj4yXS1vfDA7aWYoKGZ8MCk+PTUpe2Y9Zj4+MjtyPShmfDApPjI/ZjoyO3M9Zj4+PjA+MT9mOjE7dD1lJi0yO3U9ZSYxO2s9MTt3aGlsZSgxKXtlOntmOntpZigoa3wwKSE9KHN8MCkpe3A9SihlLGspO2Y9RFsoazw8Mikrbz4+Ml07aWYoKGZ8MCk9PS0xKXticmVhayBmfWY9RFtEW24rMTI+PjJdKyhmPDwyKT4+Ml07aWYoKGZ8MCk9PS0xKXticmVhayBmfWk9RFtxPj4yXTtnPURbbj4+Ml07aj1EW2krKERbZysoZjw8Mik+PjJdPDwyKT4+Ml07aD1mKzF8MDtoPShoPj4+MCklM3wwP2g6Zi0yfDA7aWYoKGh8MCkhPS0xKXtoPURbZysoaDw8Mik+PjJdfWVsc2V7aD0tMX1nOntoOntpZigoZj4+PjApJTN8MCl7Zj1mLTF8MDticmVhayBofWY9ZisyfDA7bD0tMTtpZigoZnwwKT09LTEpe2JyZWFrIGd9fWw9RFtnKyhmPDwyKT4+Ml19aWYoKGp8MCk+PShrfDApKXticmVhayBmfWY9RFsoaDw8MikraT4+Ml07aWYoKGZ8MCk+PShrfDApKXticmVhayBmfWc9RFtpKyhsPDwyKT4+Ml07aWYoKGd8MCk+PShrfDApKXticmVhayBmfWk6e2lmKChlfDApPD0wKXticmVhayBpfWc9SihlLGcpO2k9SihlLGYpO2o9SihlLGopO2Y9MDtsPTA7aWYoKGV8MCkhPTEpe3doaWxlKDEpe0RbbSsoZjw8Mik+PjJdPShEWyhmK2c8PDIpK2M+PjJdK0RbKGYraTw8MikrYz4+Ml18MCktRFsoZitqPDwyKStjPj4yXTtoPWZ8MTtEW20rKGg8PDIpPj4yXT0oRFsoZytoPDwyKStjPj4yXStEWyhpK2g8PDIpK2M+PjJdfDApLURbKGgrajw8MikrYz4+Ml07Zj1mKzJ8MDtsPWwrMnwwO2lmKCh0fDApIT0obHwwKSl7Y29udGludWV9YnJlYWt9fWlmKCF1KXticmVhayBpfURbbSsoZjw8Mik+PjJdPShEWyhmK2c8PDIpK2M+PjJdK0RbKGYraTw8MikrYz4+Ml18MCktRFsoZitqPDwyKStjPj4yXX1pZigoZHwwKTw9MCl7YnJlYWsgZX1pPURbYSszMj4+Ml07Zj0wO3doaWxlKDEpe2Q9Zjw8MjtnPURbZCttPj4yXTtqPURbYSsxNj4+Ml07ajp7aWYoKGd8MCk+KGp8MCkpe0RbZCtpPj4yXT1qO2JyZWFrIGp9ZD1kK2l8MDtqPURbYSsxMj4+Ml07aWYoKGp8MCk+KGd8MCkpe0RbZD4+Ml09ajticmVhayBqfURbZD4+Ml09Z31kPURbYSs4Pj4yXTtmPWYrMXwwO2lmKChkfDApPihmfDApKXtjb250aW51ZX1icmVha31mPTA7aWYoKGR8MCk8PTApe2JyZWFrIGV9ZD1wPDwyO2o9ZCtjfDA7aD1iK2R8MDt3aGlsZSgxKXtnPWY8PDI7ZD1nK2p8MDtnPURbZytoPj4yXStEW2craT4+Ml18MDtEW2Q+PjJdPWc7azp7aWYoKGd8MCk+RFthKzE2Pj4yXSl7bD1nLURbYSsyMD4+Ml18MH1lbHNle2lmKChnfDApPj1EW2ErMTI+PjJdKXticmVhayBrfWw9ZytEW2ErMjA+PjJdfDB9RFtkPj4yXT1sfWQ9RFthKzg+PjJdO2Y9ZisxfDA7aWYoKGR8MCk+KGZ8MCkpe2NvbnRpbnVlfWJyZWFrfWJyZWFrIGV9dWEoKTtUKCl9aWYoKGR8MCk8PTApe2JyZWFrIGV9aj0oSihrLTF8MCxlKTw8MikrY3wwO2k9RFthKzMyPj4yXTtmPTA7d2hpbGUoMSl7ZD1mPDwyO2c9RFtkK2o+PjJdO2g9RFthKzE2Pj4yXTtsOntpZigoZ3wwKT4oaHwwKSl7RFtkK2k+PjJdPWg7YnJlYWsgbH1kPWQraXwwO2g9RFthKzEyPj4yXTtpZigoaHwwKT4oZ3wwKSl7RFtkPj4yXT1oO2JyZWFrIGx9RFtkPj4yXT1nfWQ9RFthKzg+PjJdO2Y9ZisxfDA7aWYoKGR8MCk+KGZ8MCkpe2NvbnRpbnVlfWJyZWFrfWY9MDtpZigoZHwwKTw9MCl7YnJlYWsgZX1kPXA8PDI7aj1kK2N8MDtoPWIrZHwwO3doaWxlKDEpe2c9Zjw8MjtkPWcranwwO2c9RFtnK2g+PjJdK0RbZytpPj4yXXwwO0RbZD4+Ml09ZzttOntpZigoZ3wwKT5EW2ErMTY+PjJdKXtsPWctRFthKzIwPj4yXXwwfWVsc2V7aWYoKGd8MCk+PURbYSsxMj4+Ml0pe2JyZWFrIG19bD1nK0RbYSsyMD4+Ml18MH1EW2Q+PjJdPWx9ZD1EW2ErOD4+Ml07Zj1mKzF8MDtpZigoZHwwKT4oZnwwKSl7Y29udGludWV9YnJlYWt9fWs9aysxfDA7aWYoKHJ8MCkhPShrfDApKXtjb250aW51ZX1icmVha319bWEobSk7cmV0dXJuIDF9ZnVuY3Rpb24gU2MoYSl7dmFyIGI9MCxjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MDtEW2ErNTY+PjJdPURbYSs1Mj4+Ml07RFthKzQ0Pj4yXT1EW2ErNDA+PjJdO2E6e2I6e2M6e2U9RFthKzY0Pj4yXTtjPURbZSsyND4+Ml07aWYoKGN8MCkhPURbZSsyOD4+Ml0pe3doaWxlKDEpe2Q9YjtpPURbKGs8PDIpK2M+PjJdO2Q6e2lmKChpfDApPT0tMSl7YnJlYWsgZH1iPURbYSs1Nj4+Ml07ZTp7aWYoKGJ8MCkhPURbYSs2MD4+Ml0pe0RbYj4+Ml09ZDtEW2ErNTY+PjJdPWIrNDticmVhayBlfWM9RFthKzUyPj4yXTtlPWItY3wwO2c9ZT4+MjtiPWcrMXwwO2lmKGI+Pj4wPj0xMDczNzQxODI0KXticmVhayBjfWY9ZT4+MTtmPWc+Pj4wPDUzNjg3MDkxMT9iPj4+MD5mPj4+MD9iOmY6MTA3Mzc0MTgyMztpZihmKXtpZihmPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgYX1iPW5hKGY8PDIpfWVsc2V7Yj0wfWc9YisoZzw8Mil8MDtEW2c+PjJdPWQ7aWYoKGV8MCk+MCl7b2EoYixjLGUpfURbYSs2MD4+Ml09YisoZjw8Mik7RFthKzU2Pj4yXT1nKzQ7RFthKzUyPj4yXT1iO2lmKCFjKXticmVhayBlfW1hKGMpfWY6e2lmKCEoRFtEW2ErMTI+PjJdKyhrPj4+MyY1MzY4NzA5MDgpPj4yXT4+PmsmMSkpe2JyZWFrIGZ9Yj1pKzF8MDtiPShiPj4+MCklM3wwP2I6aS0yfDA7aWYoKGJ8MCk9PS0xKXticmVhayBmfWU9RFthPj4yXTtpZihEW2UrKGI+Pj4zJjUzNjg3MDkwOCk+PjJdPj4+YiYxKXticmVhayBmfWI9RFtEW0RbYSs2ND4+Ml0rMTI+PjJdKyhiPDwyKT4+Ml07aWYoKGJ8MCk9PS0xKXticmVhayBmfWM9YisxfDA7Yz0oYz4+PjApJTN8MD9jOmItMnwwO2lmKChjfDApPT0tMSl7YnJlYWsgZn1mPURbYSs2ND4+Ml07d2hpbGUoMSl7aT1jO2I9YysxfDA7Yj0oYj4+PjApJTN8MD9iOmMtMnwwO2lmKChifDApPT0tMXxEW2UrKGI+Pj4zJjUzNjg3MDkwOCk+PjJdPj4+YiYxKXticmVhayBmfWI9RFtEW2YrMTI+PjJdKyhiPDwyKT4+Ml07aWYoKGJ8MCk9PS0xKXticmVhayBmfWM9YisxfDA7Yz0oYz4+PjApJTN8MD9jOmItMnwwO2lmKChjfDApIT0tMSl7Y29udGludWV9YnJlYWt9fURbRFthKzI4Pj4yXSsoaTw8Mik+PjJdPWQ7Yj1EW2ErNDQ+PjJdO2c6e2lmKChifDApIT1EW2ErNDg+PjJdKXtEW2I+PjJdPWk7RFthKzQ0Pj4yXT1iKzQ7YnJlYWsgZ31jPURbYSs0MD4+Ml07ZT1iLWN8MDtnPWU+PjI7Yj1nKzF8MDtpZihiPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgYn1mPWU+PjE7Zj1nPj4+MDw1MzY4NzA5MTE/Yj4+PjA+Zj4+PjA/YjpmOjEwNzM3NDE4MjM7aWYoZil7aWYoZj4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGF9Yj1uYShmPDwyKX1lbHNle2I9MH1nPWIrKGc8PDIpfDA7RFtnPj4yXT1pO2lmKChlfDApPjApe29hKGIsYyxlKX1EW2ErNDg+PjJdPWIrKGY8PDIpO0RbYSs0ND4+Ml09Zys0O0RbYSs0MD4+Ml09YjtpZighYyl7YnJlYWsgZ31tYShjKX1iPWQrMXwwO2U9RFthKzY0Pj4yXTtoOntpZigoaT4+PjApJTN8MCl7Yz1pLTF8MDticmVhayBofWM9aSsyfDA7aWYoKGN8MCk9PS0xKXticmVhayBkfX1jPURbRFtlKzEyPj4yXSsoYzw8Mik+PjJdO2lmKChjfDApPT0tMSl7YnJlYWsgZH1jPWMrKChjPj4+MCklM3wwPy0xOjIpfDA7aWYoKGN8MCk9PS0xfChjfDApPT0oaXwwKSl7YnJlYWsgZH13aGlsZSgxKXtlPWMrMXwwO2U9KGU+Pj4wKSUzfDA/ZTpjLTJ8MDtpZihEW0RbYT4+Ml0rKGU+Pj4zJjUzNjg3MDkwOCk+PjJdPj4+ZSYxKXtkPURbYSs1Nj4+Ml07aTp7aWYoKGR8MCkhPURbYSs2MD4+Ml0pe0RbZD4+Ml09YjtEW2ErNTY+PjJdPWQrNDticmVhayBpfWU9RFthKzUyPj4yXTtmPWQtZXwwO2g9Zj4+MjtkPWgrMXwwO2lmKGQ+Pj4wPj0xMDczNzQxODI0KXticmVhayBjfWc9Zj4+MTtnPWg+Pj4wPDUzNjg3MDkxMT9kPj4+MD5nPj4+MD9kOmc6MTA3Mzc0MTgyMztpZihnKXtpZihnPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgYX1kPW5hKGc8PDIpfWVsc2V7ZD0wfWg9ZCsoaDw8Mil8MDtEW2g+PjJdPWI7aWYoKGZ8MCk+MCl7b2EoZCxlLGYpfURbYSs2MD4+Ml09ZCsoZzw8Mik7RFthKzU2Pj4yXT1oKzQ7RFthKzUyPj4yXT1kO2lmKCFlKXticmVhayBpfW1hKGUpfWU9YisxfDA7ZD1EW2ErNDQ+PjJdO2o6e2lmKChkfDApIT1EW2ErNDg+PjJdKXtEW2Q+PjJdPWM7RFthKzQ0Pj4yXT1kKzQ7YnJlYWsgan1mPURbYSs0MD4+Ml07Zz1kLWZ8MDtqPWc+PjI7ZD1qKzF8MDtpZihkPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgYn1oPWc+PjE7aD1qPj4+MDw1MzY4NzA5MTE/ZD4+PjA+aD4+PjA/ZDpoOjEwNzM3NDE4MjM7aWYoaCl7aWYoaD4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGF9ZD1uYShoPDwyKX1lbHNle2Q9MH1qPWQrKGo8PDIpfDA7RFtqPj4yXT1jO2lmKChnfDApPjApe29hKGQsZixnKX1EW2ErNDg+PjJdPWQrKGg8PDIpO0RbYSs0ND4+Ml09ais0O0RbYSs0MD4+Ml09ZDtpZighZil7YnJlYWsgan1tYShmKX1kPWI7Yj1lfURbRFthKzI4Pj4yXSsoYzw8Mik+PjJdPWQ7ZT1EW2ErNjQ+PjJdO2s6e2lmKChjPj4+MCklM3wwKXtjPWMtMXwwO2JyZWFrIGt9Yz1jKzJ8MDtpZigoY3wwKT09LTEpe2JyZWFrIGR9fWM9RFtEW2UrMTI+PjJdKyhjPDwyKT4+Ml07aWYoKGN8MCk9PS0xKXticmVhayBkfWM9YysoKGM+Pj4wKSUzfDA/LTE6Mil8MDtpZigoY3wwKT09LTEpe2JyZWFrIGR9aWYoKGN8MCkhPShpfDApKXtjb250aW51ZX1icmVha319az1rKzF8MDtjPURbZSsyND4+Ml07aWYoaz4+PjA8RFtlKzI4Pj4yXS1jPj4yPj4+MCl7Y29udGludWV9YnJlYWt9fXJldHVybn1xYSgpO1QoKX1xYSgpO1QoKX1yYSgxMzI2KTtUKCl9ZnVuY3Rpb24gY2UoYSxiLGMsZCxlLGYpe2E9YXwwO2I9YnwwO2M9Y3wwO2Q9ZHwwO2U9ZXwwO2Y9ZnwwO3ZhciBnPTAsaD0wLGk9MCxqPTAsaz0wLGw9MCxtPTAsbj0wLG89MCxwPTAscT0wLHI9MCxzPTAsdD0wLHU9MDtEW2ErOD4+Ml09ZTtkPWErMzJ8MDtrPURbZD4+Ml07Zj1EW2ErMzY+PjJdLWs+PjI7YTp7aWYoZj4+PjA8ZT4+PjApe3NhKGQsZS1mfDApO2s9RFtkPj4yXTtkPURbYSs4Pj4yXTticmVhayBhfWlmKGU+Pj4wPGY+Pj4wKXtEW2ErMzY+PjJdPShlPDwyKStrfWQ9ZX1xPURbYSs1Mj4+Ml07bT1EW2ErNDg+PjJdO2Y9MDtsPShlJjEwNzM3NDE4MjMpIT0oZXwwKT8tMTplPDwyO2w9cGEobmEobCksMCxsKTtiOntpZigoZHwwKTw9MCl7YnJlYWsgYn13aGlsZSgxKXtkPWY8PDI7Zz1EW2QrbD4+Ml07aD1EW2ErMTY+PjJdO2M6e2lmKChnfDApPihofDApKXtEW2Qraz4+Ml09aDticmVhayBjfWQ9ZCtrfDA7aD1EW2ErMTI+PjJdO2lmKChofDApPihnfDApKXtEW2Q+PjJdPWg7YnJlYWsgY31EW2Q+PjJdPWd9ZD1EW2ErOD4+Ml07Zj1mKzF8MDtpZigoZHwwKT4oZnwwKSl7Y29udGludWV9YnJlYWt9aWYoKGR8MCk8PTApe2JyZWFrIGJ9Zj0wO3doaWxlKDEpe2c9Zjw8MjtkPWcrY3wwO2c9RFtiK2c+PjJdK0RbZytrPj4yXXwwO0RbZD4+Ml09ZztkOntpZigoZ3wwKT5EW2ErMTY+PjJdKXtnPWctRFthKzIwPj4yXXwwfWVsc2V7aWYoKGd8MCk+PURbYSsxMj4+Ml0pe2JyZWFrIGR9Zz1nK0RbYSsyMD4+Ml18MH1EW2Q+PjJdPWd9ZD1EW2ErOD4+Ml07Zj1mKzF8MDtpZigoZHwwKT4oZnwwKSl7Y29udGludWV9YnJlYWt9fWY9RFthKzU2Pj4yXTtvPURbZj4+Ml07Zj1EW2YrND4+Ml0tb3wwO2lmKChmfDApPj01KXtmPWY+PjI7cj0oZnwwKT4yP2Y6MjtzPWY+Pj4wPjE/ZjoxO3Q9ZSYtMjt1PWUmMTtrPTE7d2hpbGUoMSl7ZTp7Zjp7aWYoKGt8MCkhPShzfDApKXtwPUooZSxrKTtmPURbKGs8PDIpK28+PjJdO2lmKChmfDApPT0tMXxEW0RbbT4+Ml0rKGY+Pj4zJjUzNjg3MDkwOCk+PjJdPj4+ZiYxKXticmVhayBmfWY9RFtEW0RbbSs2ND4+Ml0rMTI+PjJdKyhmPDwyKT4+Ml07aWYoKGZ8MCk9PS0xKXticmVhayBmfWg9RFtxPj4yXTtnPURbbSsyOD4+Ml07aj1EW2grKERbZysoZjw8Mik+PjJdPDwyKT4+Ml07aWYoKGp8MCk+PShrfDApKXticmVhayBmfWk9ZisxfDA7aT1EW2grKERbZysoKChpPj4+MCklM3wwP2k6Zi0yfDApPDwyKT4+Ml08PDIpPj4yXTtpZigoaXwwKT49KGt8MCkpe2JyZWFrIGZ9Zj1EW2grKERbZysoZisoKGY+Pj4wKSUzfDA/LTE6Mik8PDIpPj4yXTw8Mik+PjJdO2lmKChmfDApPj0oa3wwKSl7YnJlYWsgZn1nOntpZigoZXwwKTw9MCl7YnJlYWsgZ31nPUooZSxmKTtoPUooZSxpKTtqPUooZSxqKTtmPTA7bj0wO2lmKChlfDApIT0xKXt3aGlsZSgxKXtEW2wrKGY8PDIpPj4yXT0oRFsoZitnPDwyKStjPj4yXStEWyhmK2g8PDIpK2M+PjJdfDApLURbKGYrajw8MikrYz4+Ml07aT1mfDE7RFtsKyhpPDwyKT4+Ml09KERbKGcraTw8MikrYz4+Ml0rRFsoaCtpPDwyKStjPj4yXXwwKS1EWyhqK2k8PDIpK2M+PjJdO2Y9ZisyfDA7bj1uKzJ8MDtpZigodHwwKSE9KG58MCkpe2NvbnRpbnVlfWJyZWFrfX1pZighdSl7YnJlYWsgZ31EW2wrKGY8PDIpPj4yXT0oRFsoZitnPDwyKStjPj4yXStEWyhmK2g8PDIpK2M+PjJdfDApLURbKGYrajw8MikrYz4+Ml19aWYoKGR8MCk8PTApe2JyZWFrIGV9aD1EW2ErMzI+PjJdO2Y9MDt3aGlsZSgxKXtkPWY8PDI7Zz1EW2QrbD4+Ml07aj1EW2ErMTY+PjJdO2g6e2lmKChnfDApPihqfDApKXtEW2QraD4+Ml09ajticmVhayBofWQ9ZCtofDA7aj1EW2ErMTI+PjJdO2lmKChqfDApPihnfDApKXtEW2Q+PjJdPWo7YnJlYWsgaH1EW2Q+PjJdPWd9ZD1EW2ErOD4+Ml07Zj1mKzF8MDtpZigoZHwwKT4oZnwwKSl7Y29udGludWV9YnJlYWt9Zj0wO2lmKChkfDApPD0wKXticmVhayBlfWQ9cDw8MjtqPWQrY3wwO2k9YitkfDA7d2hpbGUoMSl7Zz1mPDwyO2Q9ZytqfDA7Zz1EW2craT4+Ml0rRFtnK2g+PjJdfDA7RFtkPj4yXT1nO2k6e2lmKChnfDApPkRbYSsxNj4+Ml0pe2c9Zy1EW2ErMjA+PjJdfDB9ZWxzZXtpZigoZ3wwKT49RFthKzEyPj4yXSl7YnJlYWsgaX1nPWcrRFthKzIwPj4yXXwwfURbZD4+Ml09Z31kPURbYSs4Pj4yXTtmPWYrMXwwO2lmKChkfDApPihmfDApKXtjb250aW51ZX1icmVha31icmVhayBlfXVhKCk7VCgpfWlmKChkfDApPD0wKXticmVhayBlfWo9KEooay0xfDAsZSk8PDIpK2N8MDtoPURbYSszMj4+Ml07Zj0wO3doaWxlKDEpe2Q9Zjw8MjtnPURbZCtqPj4yXTtpPURbYSsxNj4+Ml07ajp7aWYoKGd8MCk+KGl8MCkpe0RbZCtoPj4yXT1pO2JyZWFrIGp9ZD1kK2h8MDtpPURbYSsxMj4+Ml07aWYoKGl8MCk+KGd8MCkpe0RbZD4+Ml09aTticmVhayBqfURbZD4+Ml09Z31kPURbYSs4Pj4yXTtmPWYrMXwwO2lmKChkfDApPihmfDApKXtjb250aW51ZX1icmVha31mPTA7aWYoKGR8MCk8PTApe2JyZWFrIGV9ZD1wPDwyO2o9ZCtjfDA7aT1iK2R8MDt3aGlsZSgxKXtnPWY8PDI7ZD1nK2p8MDtnPURbZytpPj4yXStEW2craD4+Ml18MDtEW2Q+PjJdPWc7azp7aWYoKGd8MCk+RFthKzE2Pj4yXSl7Zz1nLURbYSsyMD4+Ml18MH1lbHNle2lmKChnfDApPj1EW2ErMTI+PjJdKXticmVhayBrfWc9ZytEW2ErMjA+PjJdfDB9RFtkPj4yXT1nfWQ9RFthKzg+PjJdO2Y9ZisxfDA7aWYoKGR8MCk+KGZ8MCkpe2NvbnRpbnVlfWJyZWFrfX1rPWsrMXwwO2lmKChyfDApIT0oa3wwKSl7Y29udGludWV9YnJlYWt9fW1hKGwpO3JldHVybiAxfWZ1bmN0aW9uIGpkKGEsYil7dmFyIGM9MCxkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTAsaz0wO2lmKChifDApPT0tMSl7cmV0dXJuIDF9Yz0oYj4+PjApLzN8MDtpZighKERbRFthKzI0Pj4yXSsoYz4+PjMmMjY4NDM1NDUyKT4+Ml0+Pj5jJjEpKXtkPURbYSs0OD4+Ml07RFthKzUyPj4yXT1kO2E6e2lmKChkfDApIT1EW2ErNTY+PjJdKXtEW2Q+PjJdPWI7RFthKzUyPj4yXT1kKzQ7YnJlYWsgYX1lPW5hKDQpO0RbZT4+Ml09YjtmPWUrNHwwO0RbYSs1Nj4+Ml09ZjtEW2ErNTI+PjJdPWY7RFthKzQ4Pj4yXT1lO2lmKCFkKXticmVhayBhfW1hKGQpfWY9RFtEW2ErND4+Ml0rMjg+PjJdO2Q9YisxfDA7ZT0oZD4+PjApJTN8MD9kOmItMnwwO2Q9RFtmKyhlPDwyKT4+Ml07aWYoKGR8MCk9PS0xKXtyZXR1cm4gMH1nPShiLUooYywzKXwwPy0xOjIpK2J8MDtjPURbZisoZzw8Mik+PjJdO2lmKChjfDApPT0tMSl7cmV0dXJuIDB9Yj1EW2ErMzY+PjJdO2Y9YisoZD4+PjMmNTM2ODcwOTA4KXwwO2g9RFtmPj4yXTtpPTE8PGQ7aWYoIShoJmkpKXtEW2Y+PjJdPWh8aTtJYShhKzh8MCxkLGUpO2I9RFthKzM2Pj4yXX1iPShjPj4+MyY1MzY4NzA5MDgpK2J8MDtkPURbYj4+Ml07ZT0xPDxjO2lmKCEoZCZlKSl7RFtiPj4yXT1kfGU7SWEoYSs4fDAsYyxnKX1jPURbYSs1Mj4+Ml07aWYoKGN8MCk9PURbYSs0OD4+Ml0pe3JldHVybiAxfWs9YSs4fDA7d2hpbGUoMSl7Yz1jLTR8MDtiPURbYz4+Ml07ZD0oYj4+PjApLzN8MDtiOntjOntpZigoYnwwKT09LTEpe2JyZWFrIGN9ZT1EW2ErMjQ+PjJdKyhkPj4+MyYyNjg0MzU0NTIpfDA7Zj1EW2U+PjJdO2Q9MTw8ZDtpZihmJmQpe2JyZWFrIGN9RFtlPj4yXT1kfGY7ZT1EW2ErND4+Ml07ZD1EW0RbZSsyOD4+Ml0rKGI8PDIpPj4yXTtpZigoZHwwKT09LTEpe3JldHVybiAwfXdoaWxlKDEpe2M9YjtkOntlOntmPURbYSszNj4+Ml0rKGQ+Pj4zJjUzNjg3MDkwOCl8MDtnPURbZj4+Ml07aD0xPDxkO2lmKGcmaCl7YnJlYWsgZX1mOntiPURbRFtlKzQwPj4yXSsoZDw8Mik+PjJdO2c6e2lmKChifDApPT0tMSl7YnJlYWsgZ31pPWIrMXwwO2I9KGk+Pj4wKSUzfDA/aTpiLTJ8MDtpZigoYnwwKT09LTF8RFtEW2U+PjJdKyhiPj4+MyY1MzY4NzA5MDgpPj4yXT4+PmImMSl7YnJlYWsgZ31iPURbRFtEW2UrNjQ+PjJdKzEyPj4yXSsoYjw8Mik+PjJdO2lmKChifDApIT0tMSl7YnJlYWsgZn19RFtmPj4yXT1nfGg7SWEoayxkLGMpO2JyZWFrIGV9RFtmPj4yXT1nfGg7SWEoayxkLGMpO2Q9YisxfDA7aWYoKCgoZD4+PjApJTN8MD9kOmItMnwwKXwwKT09LTEpe2JyZWFrIGV9Yj0tMTtlPURbYSs0Pj4yXTtoOntpZigoY3wwKT09LTEpe2JyZWFrIGh9ZD1jKzF8MDtkPShkPj4+MCklM3wwP2Q6Yy0yfDA7aWYoKGR8MCk9PS0xfERbRFtlPj4yXSsoZD4+PjMmNTM2ODcwOTA4KT4+Ml0+Pj5kJjEpe2JyZWFrIGh9Yj1EW0RbRFtlKzY0Pj4yXSsxMj4+Ml0rKGQ8PDIpPj4yXX1kPShiPj4+MCkvM3wwO2g9MTw8ZDtjPURbYSsyND4+Ml07aT1kPj4+NXwwO2Y9RFtjKyhpPDwyKT4+Ml07YnJlYWsgZH1pOntqOntpZigoY3wwKT09LTEpe2JyZWFrIGp9ZD0tMTtiPWMrMXwwO2I9KGI+Pj4wKSUzfDA/YjpjLTJ8MDtlPURbYSs0Pj4yXTtpZighKChifDApPT0tMXxEW0RbZT4+Ml0rKGI+Pj4zJjUzNjg3MDkwOCk+PjJdPj4+YiYxKSl7ZD1EW0RbRFtlKzY0Pj4yXSsxMj4+Ml0rKGI8PDIpPj4yXX1rOntsOntpZigoYz4+PjApJTN8MCl7Yz1jLTF8MDticmVhayBsfWM9YysyfDA7Yj0tMTtpZigoY3wwKT09LTEpe2JyZWFrIGt9fWI9LTE7aWYoRFtEW2U+PjJdKyhjPj4+MyY1MzY4NzA5MDgpPj4yXT4+PmMmMSl7YnJlYWsga31iPURbRFtEW2UrNjQ+PjJdKzEyPj4yXSsoYzw8Mik+PjJdfWo9KGJ8MCk9PS0xO2c9aj8tMTooYj4+PjApLzN8MDtmPShkPj4+MCkvM3wwO2g9KGR8MCk9PS0xO2lmKCFoKXtjPURbYSsyND4+Ml07aD1oPy0xOmY7aT1oPj4+NXwwO2Y9RFtjKyhpPDwyKT4+Ml07aD0xPDxoO2lmKCEoZiZoKSl7YnJlYWsgaX19aWYoail7YnJlYWsgan1oPTE8PGc7Yz1EW2ErMjQ+PjJdO2k9Zz4+PjV8MDtmPURbYysoaTw8Mik+PjJdO2lmKCEoaCZmKSl7YnJlYWsgZH19Yz1EW2ErNTI+PjJdLTR8MDtEW2ErNTI+PjJdPWM7YnJlYWsgYn1pZihqKXtiPWQ7YnJlYWsgZH1pZihEWyhnPj4+MyY1MzY4NzA5MDgpK2M+PjJdPj4+ZyYxKXtiPWQ7YnJlYWsgZH1jPURbYSs1Mj4+Ml07RFtjLTQ+PjJdPWI7aWYoKGN8MCkhPURbYSs1Nj4+Ml0pe0RbYz4+Ml09ZDtjPWMrNHwwO2JyZWFrIGN9bTp7ZT1EW2ErNDg+PjJdO2Y9Yy1lfDA7Yz1mPj4yO2I9YysxfDA7aWYoYj4+PjA8MTA3Mzc0MTgyNCl7Zz1mPj4xO2c9Yz4+PjA8NTM2ODcwOTExP2I+Pj4wPmc+Pj4wP2I6ZzoxMDczNzQxODIzO2lmKGcpe2lmKGc+Pj4wPj0xMDczNzQxODI0KXticmVhayBtfWI9bmEoZzw8Mil9ZWxzZXtiPTB9Yz1iKyhjPDwyKXwwO0RbYz4+Ml09ZDtjPWMrNHwwO2lmKChmfDApPjApe29hKGIsZSxmKX1EW2ErNTY+PjJdPWIrKGc8PDIpO0RbYSs1Mj4+Ml09YztEW2ErNDg+PjJdPWI7aWYoIWUpe2JyZWFrIGJ9bWEoZSk7Yz1EW2ErNTI+PjJdO2JyZWFrIGJ9cWEoKTtUKCl9cmEoMTMyNik7VCgpfURbKGk8PDIpK2M+PjJdPWZ8aDtkPURbRFtlKzI4Pj4yXSsoYjw8Mik+PjJdO2lmKChkfDApIT0tMSl7Y29udGludWV9YnJlYWt9cmV0dXJuIDB9RFthKzUyPj4yXT1jfWlmKERbYSs0OD4+Ml0hPShjfDApKXtjb250aW51ZX1icmVha319cmV0dXJuIDF9ZnVuY3Rpb24gVGIoYSxiLGMpe3ZhciBkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTAsaz0wLGw9MCxtPTAsbj0wLG89MCxwPTAscT0wO2k9SihiLDEyKSthfDA7RFtpKzEyPj4yXT1EW2krOD4+Ml07bD0oY3wwKT09LTE/LTE6KGM+Pj4wKS8zfDA7bj0xO2o9YzthOntiOntjOnt3aGlsZSgxKXtkOntvPWUmMTtpZihvKXtpZigoanwwKT09LTEpe2JyZWFrIGR9aWYoKFFjKGEsKChqPj4+MCklM3wwPy0xOjIpK2p8MCl8MCk9PS0xKXticmVhayBhfWM9aisxfDA7ZD0oYz4+PjApJTN8MD9jOmotMnwwO2lmKChkfDApPT0tMSl7YnJlYWsgYX1jPWQrMXwwO2M9KGM+Pj4wKSUzfDA/YzpkLTJ8MDtpZigoY3wwKT09LTEpe2JyZWFrIGF9ZD1EW0RbRFthKzQ+PjJdKzEyPj4yXSsoYzw8Mik+PjJdO2lmKChkfDApPT0tMSl7YnJlYWsgYX1jPWQrMXwwO2M9KGM+Pj4wKSUzfDA/YzpkLTJ8MDtpZigoY3wwKT09LTEpe2JyZWFrIGF9bD0oYz4+PjApLzN8MH1rPTE8PGw7Zz1EW2ErNTY+PjJdKyhsPj4+MyY1MzY4NzA5MDgpfDA7bT1EW2c+PjJdO2U6e2lmKGsmbSl7YnJlYWsgZX1lPTA7d2hpbGUoMSl7RFtnPj4yXT1tfGs7ZD1EW2krMTI+PjJdO2Y6e2lmKChkfDApIT1EW2krMTY+PjJdKXtEW2Q+PjJdPWw7RFtpKzEyPj4yXT1kKzQ7YnJlYWsgZn1nPURbaSs4Pj4yXTtrPWQtZ3wwO2Y9az4+MjtoPWYrMXwwO2lmKGg+Pj4wPj0xMDczNzQxODI0KXticmVhayBjfWQ9az4+MTttPWY+Pj4wPDUzNjg3MDkxMT9kPj4+MDxoPj4+MD9oOmQ6MTA3Mzc0MTgyMztpZihtKXtpZihtPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgYn1kPW5hKG08PDIpfWVsc2V7ZD0wfWg9ZCsoZjw8Mil8MDtEW2g+PjJdPWw7aWYoKGt8MCk+MCl7b2EoZCxnLGspfURbaSs4Pj4yXT1kO0RbaSsxMj4+Ml09aCs0O0RbaSsxNj4+Ml09ZCsobTw8Mik7aWYoIWcpe2JyZWFrIGZ9bWEoZyl9aD1lKzF8MDtnOntoOntpOntpZighZSl7YnJlYWsgaX1pZihoJjEpe2lmKChjfDApPT0tMSl7Yz0tMTticmVhayBnfWQ9YysxfDA7Yz0oZD4+PjApJTN8MD9kOmMtMnwwO2JyZWFrIGl9aj1vP2M6ajtpZigoY3wwKT09LTEpe2M9LTE7YnJlYWsgZ31pZigoYz4+PjApJTN8MCl7ZT1jLTF8MDticmVhayBofWM9YysyfDB9ZD1jO2M9LTE7ZT1kO2lmKChkfDApPT0tMSl7YnJlYWsgZ319Yz1EW0RbRFthKzQ+PjJdKzEyPj4yXSsoZTw8Mik+PjJdO2c9LTE7bT0tMTtkPWUrMXwwO2Y9KGQ+Pj4wKSUzfDA/ZDplLTJ8MDtpZigoZnwwKT49MCl7ZD0oZj4+PjApLzN8MDttPURbKERbRFthPj4yXSs5Nj4+Ml0rSihkLDEyKXwwKSsoZi1KKGQsMyk8PDIpPj4yXX1qOntpZigoY3wwKT09LTEpe2s9MTticmVhayBqfWs9MDtmPSgoYz4+PjApJTN8MD8tMToyKStjfDA7aWYoKGZ8MCk8MCl7YnJlYWsgan1kPShmPj4+MCkvM3wwO2c9RFsoRFtEW2E+PjJdKzk2Pj4yXStKKGQsMTIpfDApKyhmLUooZCwzKTw8Mik+PjJdfWlmKChnfDApIT0obXwwKSl7Yz0tMTticmVhayBnfWs6e2w6e2U9KChlPj4+MCklM3wwPy0xOjIpK2V8MDttOntuOntpZigoZXwwKT49MCl7ZD0oZT4+PjApLzN8MDtnPURbKERbRFthPj4yXSs5Nj4+Ml0rSihkLDEyKXwwKSsoZS1KKGQsMyk8PDIpPj4yXTtlPS0xO2lmKCFrKXticmVhayBufWJyZWFrIG19Zz0tMTtpZihrKXticmVhayBsfX1kPWMrMXwwO2U9KGQ+Pj4wKSUzfDA/ZDpjLTJ8MDtpZigoZXwwKTwwKXtlPS0xO2JyZWFrIG19ZD0oZT4+PjApLzN8MDtlPURbKERbRFthPj4yXSs5Nj4+Ml0rSihkLDEyKXwwKSsoZS1KKGQsMyk8PDIpPj4yXX1pZigoZXwwKSE9KGd8MCkpe2M9LTE7YnJlYWsgZ31pZigoY3wwKSE9LTEpe2JyZWFrIGt9Yz0tMTticmVhayBnfWlmKChjfDApIT0tMSl7YnJlYWsga31jPS0xO2JyZWFrIGd9ZT1oO2w9KGM+Pj4wKS8zfDA7Zz1EW2ErNTY+PjJdKyhsPj4+MyYyNjg0MzU0NTIpfDA7bT1EW2c+PjJdO2s9MTw8bDtpZighKG0maykpe2NvbnRpbnVlfX1icmVha31pZighb3whKGgmMSkpe2JyZWFrIGV9Zj1EW2krMTI+PjJdLTR8MDtoPURbZj4+Ml07ZT1EW2ErNTY+PjJdKyhoPj4+MyY1MzY4NzA5MDgpfDA7ZD1EW2U+PjJdO3A9ZSxxPWRpKGgpJmQsRFtwPj4yXT1xO0RbaSsxMj4+Ml09Zn1lPTE7ZD1uO249MDtpZihkKXtjb250aW51ZX1icmVhayBhfWJyZWFrfWo9LTE7UWMoYSwtMSk7YnJlYWsgYX1xYSgpO1QoKX1yYSgxMzI2KTtUKCl9RFsoKGI8PDIpK2F8MCkrNDQ+PjJdPWo7Zj1EW2krOD4+Ml07Yj1EW2krMTI+PjJdLWZ8MDtvOntpZighYil7YnJlYWsgb31qPURbYSs1Nj4+Ml07Yj1iPj4yO2E9Yj4+PjA+MT9iOjE7aD1hJjE7Yz0wO2lmKGI+Pj4wPj0yKXtuPWEmLTI7bD0wO3doaWxlKDEpe2U9Yzw8MjtkPURbZStmPj4yXTtiPWorKGQ+Pj4zJjUzNjg3MDkwOCl8MDthPURbYj4+Ml07cD1iLHE9ZGkoZCkmYSxEW3A+PjJdPXE7ZD1EW2YrKGV8NCk+PjJdO2I9aisoZD4+PjMmNTM2ODcwOTA4KXwwO2E9RFtiPj4yXTtwPWIscT1kaShkKSZhLERbcD4+Ml09cTtjPWMrMnwwO2w9bCsyfDA7aWYoKG58MCkhPShsfDApKXtjb250aW51ZX1icmVha319aWYoIWgpe2JyZWFrIG99Yz1EW2YrKGM8PDIpPj4yXTtiPWorKGM+Pj4zJjUzNjg3MDkwOCl8MDthPURbYj4+Ml07cD1iLHE9ZGkoYykmYSxEW3A+PjJdPXF9fWZ1bmN0aW9uIGtkKGEsYil7dmFyIGM9MCxkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTA7aWYoKGJ8MCk9PS0xKXtyZXR1cm4gMX1kPShiPj4+MCkvM3wwO2lmKCEoRFtEW2ErMjQ+PjJdKyhkPj4+MyYyNjg0MzU0NTIpPj4yXT4+PmQmMSkpe2M9RFthKzQ4Pj4yXTtEW2ErNTI+PjJdPWM7YTp7aWYoKGN8MCkhPURbYSs1Nj4+Ml0pe0RbYz4+Ml09YjtEW2ErNTI+PjJdPWMrNDticmVhayBhfWU9bmEoNCk7RFtlPj4yXT1iO2Y9ZSs0fDA7RFthKzU2Pj4yXT1mO0RbYSs1Mj4+Ml09ZjtEW2ErNDg+PjJdPWU7aWYoIWMpe2JyZWFrIGF9bWEoYyl9ZT0tMTtmPURbYSs0Pj4yXTtjPWIrMXwwO2c9KGM+Pj4wKSUzfDA/YzpiLTJ8MDtpZigoZ3wwKSE9LTEpe2U9RFtEW2Y+PjJdKyhnPDwyKT4+Ml19Yjp7aD1iLUooZCwzKXwwO2lmKGgpe2M9Yi0xfDA7YnJlYWsgYn1jPWIrMnwwO2lmKChjfDApIT0tMSl7YnJlYWsgYn1yZXR1cm4gMH1pZigoZXwwKT09LTEpe3JldHVybiAwfWQ9RFtEW2Y+PjJdKyhjPDwyKT4+Ml07aWYoKGR8MCk9PS0xKXtyZXR1cm4gMH1jPURbYSszNj4+Ml07Zj1jKyhlPj4+MyY1MzY4NzA5MDgpfDA7aT1EW2Y+PjJdO2o9MTw8ZTtpZighKGkmaikpe0RbZj4+Ml09aXxqO0lhKGErOHwwLGUsZyk7Yz1EW2ErMzY+PjJdfWM9KGQ+Pj4zJjUzNjg3MDkwOCkrY3wwO2U9RFtjPj4yXTtmPTE8PGQ7aWYoIShlJmYpKXtEW2M+PjJdPWV8ZjtJYShhKzh8MCxkLChoPy0xOjIpK2J8MCl9Yz1EW2ErNTI+PjJdO2lmKChjfDApPT1EW2ErNDg+PjJdKXtyZXR1cm4gMX1qPWErOHwwO3doaWxlKDEpe2M9Yy00fDA7Yj1EW2M+PjJdO2Q9KGI+Pj4wKS8zfDA7Yzp7ZDp7aWYoKGJ8MCk9PS0xKXticmVhayBkfWU9RFthKzI0Pj4yXSsoZD4+PjMmMjY4NDM1NDUyKXwwO2Y9RFtlPj4yXTtkPTE8PGQ7aWYoZiZkKXticmVhayBkfURbZT4+Ml09ZHxmO3doaWxlKDEpe2Q9RFthKzQ+PjJdO2M9RFtEW2Q+PjJdKyhiPDwyKT4+Ml07aWYoKGN8MCk9PS0xKXtyZXR1cm4gMH1lOntmOntlPURbYSszNj4+Ml0rKGM+Pj4zJjUzNjg3MDkwOCl8MDtmPURbZT4+Ml07Zz0xPDxjO2lmKGYmZyl7YnJlYWsgZn1nOntoPURbRFtkKzI0Pj4yXSsoYzw8Mik+PjJdO2g6e2lmKChofDApPT0tMSl7YnJlYWsgaH1pPWgrMXwwO2g9KGk+Pj4wKSUzfDA/aTpoLTJ8MDtpZigoaHwwKT09LTEpe2JyZWFrIGh9ZD1EW0RbZCsxMj4+Ml0rKGg8PDIpPj4yXTtpZigoZHwwKSE9LTEpe2JyZWFrIGd9fURbZT4+Ml09ZnxnO0lhKGosYyxiKTticmVhayBmfURbZT4+Ml09ZnxnO0lhKGosYyxiKTtjPWQrMXwwO2lmKCgoKGM+Pj4wKSUzfDA/YzpkLTJ8MCl8MCk9PS0xKXticmVhayBmfWQ9Yi0yfDA7Yz1iKzF8MDtiPS0xO2M9KGM+Pj4wKSUzfDA/YzpkO2lmKChjfDApIT0tMSl7Yj1EW0RbRFthKzQ+PjJdKzEyPj4yXSsoYzw8Mik+PjJdfWM9KGI+Pj4wKS8zfDA7Zz0xPDxjO2U9RFthKzI0Pj4yXTtoPWM+Pj41fDA7ZD1EW2UrKGg8PDIpPj4yXTticmVhayBlfWM9LTE7Zj1EW2ErND4+Ml07ZD1iKzF8MDtkPShkPj4+MCklM3wwP2Q6Yi0yfDA7aWYoKGR8MCkhPS0xKXtjPURbRFtmKzEyPj4yXSsoZDw8Mik+PjJdfWk6e2o6e2lmKChiPj4+MCklM3wwKXtlPWItMXwwO2JyZWFrIGp9ZT1iKzJ8MDtiPS0xO2lmKChlfDApPT0tMSl7YnJlYWsgaX19Yj1EW0RbZisxMj4+Ml0rKGU8PDIpPj4yXX1pPShifDApPT0tMTtmPWk/LTE6KGI+Pj4wKS8zfDA7ZD0oYz4+PjApLzN8MDtrOntnPShjfDApPT0tMTtpZighZyl7ZT1EW2ErMjQ+PjJdO2c9Zz8tMTpkO2g9Zz4+PjV8MDtkPURbZSsoaDw8Mik+PjJdO2c9MTw8ZztpZighKGQmZykpe2JyZWFrIGt9fWlmKCFpKXtnPTE8PGY7ZT1EW2ErMjQ+PjJdO2g9Zj4+PjV8MDtkPURbZSsoaDw8Mik+PjJdO2lmKCEoZyZkKSl7YnJlYWsgZX19Yz1EW2ErNTI+PjJdLTR8MDtEW2ErNTI+PjJdPWM7YnJlYWsgY31pZihpKXtiPWM7YnJlYWsgZX1pZihEWyhmPj4+MyY1MzY4NzA5MDgpK2U+PjJdPj4+ZiYxKXtiPWM7YnJlYWsgZX1kPURbYSs1Mj4+Ml07RFtkLTQ+PjJdPWI7aWYoKGR8MCkhPURbYSs1Nj4+Ml0pe0RbZD4+Ml09YztjPWQrNHwwO2JyZWFrIGR9bDp7Yj1kO2Q9RFthKzQ4Pj4yXTtlPWItZHwwO2c9ZT4+MjtiPWcrMXwwO2lmKGI+Pj4wPDEwNzM3NDE4MjQpe2Y9ZT4+MTtmPWc+Pj4wPDUzNjg3MDkxMT9iPj4+MD5mPj4+MD9iOmY6MTA3Mzc0MTgyMztpZihmKXtpZihmPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgbH1iPW5hKGY8PDIpfWVsc2V7Yj0wfWc9YisoZzw8Mil8MDtEW2c+PjJdPWM7Yz1nKzR8MDtpZigoZXwwKT4wKXtvYShiLGQsZSl9RFthKzU2Pj4yXT1iKyhmPDwyKTtEW2ErNTI+PjJdPWM7RFthKzQ4Pj4yXT1iO2lmKCFkKXticmVhayBjfW1hKGQpO2M9RFthKzUyPj4yXTticmVhayBjfXFhKCk7VCgpfXJhKDEzMjYpO1QoKX1EWyhoPDwyKStlPj4yXT1kfGc7aWYoKGJ8MCkhPS0xKXtjb250aW51ZX1icmVha31yZXR1cm4gMH1EW2ErNTI+PjJdPWN9aWYoRFthKzQ4Pj4yXSE9KGN8MCkpe2NvbnRpbnVlfWJyZWFrfX1yZXR1cm4gMX1mdW5jdGlvbiB2YyhhLGIsYyl7YT1hfDA7Yj1ifDA7Yz1jfDA7dmFyIGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTAsbz0wLHA9MDthOntuPWJhW0RbRFthPj4yXSs0ND4+Ml1dKGEpfDA7aWYoKG58MCk8PTApe2JyZWFrIGF9ZT1EW2IrND4+Ml0tRFtiPj4yXT4+MjtnPSQrLTY0fDA7JD1nO2Q9bGIoZyk7Zj1KKERbMjU0OV0sbik7JGIoZCxEW0RbYSs4Pj4yXSs1Nj4+Ml0sbjw8MjQ+PjI0LDUsMCxmLGY+PjMxKTtmPW5hKDk2KTtkPVpiKGYsZCk7QltkKzg0fDBdPTE7RFtkKzcyPj4yXT1EW2QrNjg+PjJdO1liKGQsZSk7RFtkKzYwPj4yXT1EW0RbYSs4Pj4yXSs2MD4+Ml07ZD1EW2ErMTY+PjJdO0RbYSsxNj4+Ml09ZjtpZihkKXt5YShkKX0kPWctIC02NHwwO2c9RFthKzE2Pj4yXTtpZighRFtnKzgwPj4yXSl7YnJlYWsgYX1pPURbRFtnPj4yXT4+Ml07aWYoIWkpe2JyZWFrIGF9aD1EW2MrMTI+PjJdO2Q9RFtjKzIwPj4yXTtqPURbYys4Pj4yXTtmPURbYysxNj4+Ml07aWYoKGh8MCk8PShkfDApJmo+Pj4wPD1mPj4+MHwoZHwwKT4oaHwwKSl7YnJlYWsgYX1rPUooZSxuKTtpPWkrRFtnKzQ4Pj4yXXwwO2w9RFtjPj4yXTttPUVbbCtmfDBdO2c9ZisxfDA7ZT1nP2Q6ZCsxfDA7RFtjKzE2Pj4yXT1nO0RbYysyMD4+Ml09ZTtiOntjOntkOntpZihtKXtpZihqYyhrLG4sYyxpKSl7YnJlYWsgZH1icmVhayBhfWlmKChlfDApPj0oaHwwKSZnPj4+MD49aj4+PjB8KGV8MCk+KGh8MCkpe2JyZWFrIGF9Zz1FW2crbHwwXTtlPWYrMnwwO2Q9ZT4+PjA8Mj9kKzF8MDpkO0RbYysxNj4+Ml09ZTtEW2MrMjA+PjJdPWQ7ZD1EW0RbYSsxNj4+Ml0rNjQ+PjJdO2Q9RFtkKzQ+PjJdLURbZD4+Ml18MDtlOntpZigoZ3wwKT09RFsyNTQ5XSl7ZT1kO2Q9azw8MjtpZihlPj4+MDxkPj4+MCl7YnJlYWsgYX1oPURbYys4Pj4yXTtnPURbYysxMj4+Ml07ZT1EW2MrMjA+PjJdO2o9RFtjKzE2Pj4yXTtmPWQranwwO2U9Zj4+PjA8ZD4+PjA/ZSsxfDA6ZTtpZigoZXwwKTw9KGd8MCkmZj4+PjA8PWg+Pj4wfChlfDApPChnfDApKXticmVhayBlfWJyZWFrIGF9aWYoZD4+PjA8SihnLGspPj4+MCl7YnJlYWsgYX1oPURbYysxMj4+Ml07ZT1EW2MrMjA+PjJdO2w9RFtjKzg+PjJdO2Y9bDtqPURbYysxNj4+Ml07ZD1nO3A9Zi1qPj4+MDwkaChkLDAsaywwKT4+PjA7Zj1oLShlKyhmPj4+MDxqPj4+MCl8MCl8MDttPWFhO2lmKHAmKGZ8MCk8PShtfDApfChmfDApPChtfDApKXticmVhayBhfWY9MTtpZighayl7YnJlYWsgY31mPTA7bT1sO2w9ZCtqfDA7ZT1sPj4+MDxkPj4+MD9lKzF8MDplO2lmKG0+Pj4wPGw+Pj4wJihlfDApPj0oaHwwKXwoZXwwKT4oaHwwKSl7YnJlYWsgYn13aGlsZSgxKXtvYShpKyhvPDwyKXwwLGorRFtjPj4yXXwwLGcpO2U9RFtjKzIwPj4yXTtoPWQrRFtjKzE2Pj4yXXwwO2U9aD4+PjA8ZD4+PjA/ZSsxfDA6ZTtqPWg7RFtjKzE2Pj4yXT1oO0RbYysyMD4+Ml09ZTtvPW8rMXwwO2lmKChrfDApPT0ob3wwKSl7YnJlYWsgZH1tPURbYys4Pj4yXTtoPURbYysxMj4+Ml07bD1kK2p8MDtlPWw+Pj4wPGQ+Pj4wP2UrMXwwOmU7aWYoKGV8MCk8PShofDApJmw+Pj4wPD1tPj4+MHwoZXwwKTwoaHwwKSl7Y29udGludWV9YnJlYWt9YnJlYWsgYn1vYShpLGorRFtjPj4yXXwwLGQpO2U9ZDtnPWQrRFtjKzE2Pj4yXXwwO2Q9RFtjKzIwPj4yXTtEW2MrMTY+PjJdPWc7RFtjKzIwPj4yXT1lPj4+MD5nPj4+MD9kKzF8MDpkfWY9MTtpZighayl7YnJlYWsgY31kPURbYSsyMD4+Ml07aWYoZCl7Zj0wO2lmKGJhW0RbRFtkPj4yXSszMj4+Ml1dKGQpfDApe2JyZWFrIGN9fWQ9MDtmPTA7Zjp7aWYoKGt8MCk8PTApe2JyZWFrIGZ9aWYoKGt8MCkhPTEpe2g9ayYtMjt3aGlsZSgxKXtlPWQ8PDI7Zz1EW2UraT4+Ml07RFtlK2k+PjJdPTAtKGcmMSleZz4+PjE7Zz1lfDQ7ZT1EW2craT4+Ml07RFtnK2k+PjJdPTAtKGUmMSleZT4+PjE7ZD1kKzJ8MDtmPWYrMnwwO2lmKChofDApIT0oZnwwKSl7Y29udGludWV9YnJlYWt9fWlmKCEoayYxKSl7YnJlYWsgZn1lPWQ8PDI7ZD1EW2UraT4+Ml07RFtlK2k+PjJdPTAtKGQmMSleZD4+PjF9Zj0wfWQ9RFthKzIwPj4yXTtnOntpZighZCl7YnJlYWsgZ31pZighKGJhW0RbRFtkPj4yXSs0MD4+Ml1dKGQsYyl8MCkpe2JyZWFrIGF9aWYoZil7YnJlYWsgZ31mPTA7YT1EW2ErMjA+PjJdO2lmKCEoYmFbRFtEW2E+PjJdKzQ0Pj4yXV0oYSxpLGksayxuLERbYj4+Ml0pfDApKXticmVhayBifX1mPTF9cmV0dXJuIGZ8MH1yZXR1cm4gMH1mdW5jdGlvbiBwYyhhLGIpe2E9YXwwO2I9YnwwO3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTA7aD0kLTMyfDA7JD1oO2E6e2I6e2lmKCFqYigxLGgrMjh8MCxiKSl7YnJlYWsgYn1jOntkPURbaCsyOD4+Ml07aWYoZCl7aWIoYSs2MHwwLGQpO2M9aCs4fDA7RFtjPj4yXT0wO0RbYys0Pj4yXT0wO0JbYys1fDBdPTA7QltjKzZ8MF09MDtCW2MrN3wwXT0wO0JbYys4fDBdPTA7QltjKzl8MF09MDtCW2MrMTB8MF09MDtCW2MrMTF8MF09MDtCW2MrMTJ8MF09MDtpZighS2EoYyxiKSl7YnJlYWsgY313aGlsZSgxKXtmPTE8PGU7aT1HYShjKTtnPURbYSs2MD4+Ml0rKGU+Pj4zJjUzNjg3MDkwOCl8MDtpZihpKXtmPWZ8RFtnPj4yXX1lbHNle2Y9RFtnPj4yXSYoZl4tMSl9RFtnPj4yXT1mO2U9ZSsxfDA7aWYoKGR8MCkhPShlfDApKXtjb250aW51ZX1icmVha319aWYoIWpiKDEsaCsyOHwwLGIpKXticmVhayBifWQ9RFtoKzI4Pj4yXTtpZihkKXtlPTA7aWIoYSs3MnwwLGQpO2M9aCs4fDA7RFtjPj4yXT0wO0RbYys0Pj4yXT0wO0JbYys1fDBdPTA7QltjKzZ8MF09MDtCW2MrN3wwXT0wO0JbYys4fDBdPTA7QltjKzl8MF09MDtCW2MrMTB8MF09MDtCW2MrMTF8MF09MDtCW2MrMTJ8MF09MDtpZighS2EoYyxiKSl7YnJlYWsgY313aGlsZSgxKXtmPTE8PGU7aT1HYShjKTtnPURbYSs3Mj4+Ml0rKGU+Pj4zJjUzNjg3MDkwOCl8MDtpZihpKXtmPWZ8RFtnPj4yXX1lbHNle2Y9RFtnPj4yXSYoZl4tMSl9RFtnPj4yXT1mO2U9ZSsxfDA7aWYoKGR8MCkhPShlfDApKXtjb250aW51ZX1icmVha319aWYoIWpiKDEsaCsyOHwwLGIpKXticmVhayBifWQ9RFtoKzI4Pj4yXTtpZihkKXtlPTA7aWIoYSs4NHwwLGQpO2M9aCs4fDA7RFtjPj4yXT0wO0RbYys0Pj4yXT0wO0JbYys1fDBdPTA7QltjKzZ8MF09MDtCW2MrN3wwXT0wO0JbYys4fDBdPTA7QltjKzl8MF09MDtCW2MrMTB8MF09MDtCW2MrMTF8MF09MDtCW2MrMTJ8MF09MDtpZighS2EoYyxiKSl7YnJlYWsgY313aGlsZSgxKXtmPTE8PGU7aT1HYShjKTtnPURbYSs4ND4+Ml0rKGU+Pj4zJjUzNjg3MDkwOCl8MDtpZihpKXtmPWZ8RFtnPj4yXX1lbHNle2Y9RFtnPj4yXSYoZl4tMSl9RFtnPj4yXT1mO2U9ZSsxfDA7aWYoKGR8MCkhPShlfDApKXtjb250aW51ZX1icmVha319aWYoIWpiKDEsaCsyOHwwLGIpKXticmVhayBifWQ9RFtoKzI4Pj4yXTtpZihkKXtlPTA7aWIoYSs5NnwwLGQpO2M9aCs4fDA7RFtjPj4yXT0wO0RbYys0Pj4yXT0wO0JbYys1fDBdPTA7QltjKzZ8MF09MDtCW2MrN3wwXT0wO0JbYys4fDBdPTA7QltjKzl8MF09MDtCW2MrMTB8MF09MDtCW2MrMTF8MF09MDtCW2MrMTJ8MF09MDtpZighS2EoYyxiKSl7YnJlYWsgY313aGlsZSgxKXtmPTE8PGU7aT1HYShjKTtnPURbYSs5Nj4+Ml0rKGU+Pj4zJjUzNjg3MDkwOCl8MDtpZihpKXtmPWZ8RFtnPj4yXX1lbHNle2Y9RFtnPj4yXSYoZl4tMSl9RFtnPj4yXT1mO2U9ZSsxfDA7aWYoKGR8MCkhPShlfDApKXtjb250aW51ZX1icmVha319ZT0wO2Q9RFtiKzEyPj4yXTtmPWQ7Yz1EW2IrMjA+PjJdO2c9YztpPURbYisxNj4+Ml07aj1pKzR8MDtjPWo+Pj4wPDQ/YysxfDA6YztrPURbYis4Pj4yXTtpZihrPj4+MDxqPj4+MCYoY3wwKT49KGR8MCl8KGN8MCk+KGR8MCkpe2JyZWFrIGF9bD1EW2I+PjJdO2Q9bCtpfDA7ZD1FW2R8MF18RVtkKzF8MF08PDh8KEVbZCsyfDBdPDwxNnxFW2QrM3wwXTw8MjQpO0RbYisxNj4+Ml09ajtEW2IrMjA+PjJdPWM7Yz1nO2c9aSs4fDA7Yz1nPj4+MDw4P2MrMXwwOmM7aT1nO2c9YztpZihpPj4+MD5rPj4+MCYoY3wwKT49KGZ8MCl8KGN8MCk+KGZ8MCkpe2JyZWFrIGF9Yz1qK2x8MDtjPUVbY3wwXXxFW2MrMXwwXTw8OHwoRVtjKzJ8MF08PDE2fEVbYyszfDBdPDwyNCk7RFtiKzE2Pj4yXT1pO0RbYisyMD4+Ml09ZztpZigoY3wwKTwoZHwwKSl7YnJlYWsgYX1EW2ErMTY+PjJdPWM7RFthKzEyPj4yXT1kO2I9KGM+PjMxKS0oKGQ+PjMxKSsoYz4+PjA8ZD4+PjApfDApfDA7Yz1jLWR8MDtpZighYiZjPj4+MD4yMTQ3NDgzNjQ2fGIpe2JyZWFrIGF9ZT0xO2I9YysxfDA7RFthKzIwPj4yXT1iO2M9Yj4+PjF8MDtEW2ErMjQ+PjJdPWM7RFthKzI4Pj4yXT0wLWM7aWYoYiYxKXticmVhayBhfURbYSsyND4+Ml09Yy0xO2JyZWFrIGF9fWU9MH0kPWgrMzJ8MDtyZXR1cm4gZXwwfWZ1bmN0aW9uIFpoKGEsYixjLGQsZSxmKXthPWF8MDtiPWJ8MDtjPWN8MDtkPWR8MDtlPWV8MDtmPWZ8MDt2YXIgZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MCxvPTA7Zz0kLTMyfDA7JD1nO0RbYSs2OD4+Ml09ZjtlPURbYSs1Nj4+Ml07ZD1EW2U+PjJdO2Y9RFtlKzQ+PjJdO0RbZysyND4+Ml09MDtEW2crMTY+PjJdPTA7RFtnKzIwPj4yXT0wO2E6e2Y9Zi1kfDA7aWYoKGZ8MCk8PTApe2JyZWFrIGF9ZD1EW2U+PjJdO2lmKChkfDApIT1EW2UrND4+Ml0pe209YSsxMTJ8MDtuPWErNjB8MDtlPWY+Pj4yfDA7bz1lPj4+MD4xP2U6MTt3aGlsZSgxKXtMYihuLERbKGs8PDIpK2Q+PjJdLGcrMTZ8MCk7ZT1EW2crMjA+PjJdO2Q9ZT4+MzE7Zj1EW2crMTY+PjJdO2g9Zj4+MzE7aj1EW2crMjQ+PjJdO2k9aj4+MzE7aT1pXmkrajtoPWkrKChkXmQrZSkrKGheZitoKXwwKXwwO2Q9MDtkPWg+Pj4wPGk+Pj4wPzE6ZDtiOntpZighKGR8aCkpe0RbZysxNj4+Ml09RFthKzEwOD4+Ml07YnJlYWsgYn1pPURbYSsxMDg+PjJdO2w9aT4+MzE7Zj1haSgkaChpLGwsZixmPj4zMSksYWEsaCxkKTtEW2crMTY+PjJdPWY7ZD1haSgkaChpLGwsZSxlPj4zMSksYWEsaCxkKTtEW2crMjA+PjJdPWQ7ZT1kO2Q9ZD4+MzE7ZT1lK2ReZDtkPWY+PjMxO2Q9ZSsoZCtmXmQpfDA7aWYoKGp8MCk+PTApe0RbZysyND4+Ml09aS1kO2JyZWFrIGJ9RFtnKzI0Pj4yXT1kLWl9ZD1HYShtKTtmPURbZysxNj4+Ml07Yzp7aWYoZCl7RFtnKzI0Pj4yXT0wLURbZysyND4+Ml07ZT0wLURbZysyMD4+Ml18MDtEW2crMjA+PjJdPWU7Zj0wLWZ8MDtEW2crMTY+PjJdPWY7YnJlYWsgY31lPURbZysyMD4+Ml19ZDp7aWYoKGZ8MCk+PTApe2Q9RFthKzEwOD4+Ml07Zj1kK0RbZysyND4+Ml18MDtkPWQrZXwwO2JyZWFrIGR9ZTp7aWYoKGV8MCk8MCl7Zj1EW2crMjQ+PjJdO2Q9Zj4+MzE7ZD1kXmQrZjticmVhayBlfWY9RFtnKzI0Pj4yXTtkPWY+PjMxO2Q9RFthKzEwMD4+Ml0tKGReZCtmKXwwfWlmKChmfDApPDApe2Y9ZTtlPWU+PjMxO2Y9ZitlXmU7YnJlYWsgZH1mPWU7ZT1lPj4zMTtmPURbYSsxMDA+PjJdLShmK2VeZSl8MH1lPURbYSsxMDA+PjJdO2Y6e2lmKCEoZHxmKSl7Zj1lO2Q9ZjticmVhayBmfWlmKCEoKGV8MCkhPShmfDApfGQpKXtkPWY7YnJlYWsgZn1oPShkfDApIT0oZXwwKTtpZighKGZ8aCkpe2Y9ZDticmVhayBmfWc6e2lmKGQpe2JyZWFrIGd9aj1EW2ErMTA4Pj4yXTtpZigoanwwKT49KGZ8MCkpe2JyZWFrIGd9Zj0oajw8MSktZnwwO2Q9MDticmVhayBmfWg6e2lmKGgpe2JyZWFrIGh9aD1EW2ErMTA4Pj4yXTtpZigoaHwwKTw9KGZ8MCkpe2JyZWFrIGh9Zj0oaDw8MSktZnwwO2JyZWFrIGZ9aTp7aWYoKGV8MCkhPShmfDApKXticmVhayBpfWU9RFthKzEwOD4+Ml07aWYoKGV8MCk8PShkfDApKXticmVhayBpfWQ9KGU8PDEpLWR8MDticmVhayBmfWlmKGYpe2JyZWFrIGZ9Zj0wO2U9RFthKzEwOD4+Ml07aWYoKGV8MCk+PShkfDApKXticmVhayBmfWQ9KGU8PDEpLWR8MH1EW2crMTI+PjJdPWY7RFtnKzg+PjJdPWQ7ajp7aWYoRFthKzg+PjJdPD0wKXticmVhayBqfWg9RFthKzMyPj4yXTtmPTA7d2hpbGUoMSl7ZT1EW2ErMTY+PjJdO2s6e2lmKChlfDApPChkfDApKXtEW2grKGY8PDIpPj4yXT1lO2JyZWFrIGt9ZT1oKyhmPDwyKXwwO2o9RFthKzEyPj4yXTtpZigoanwwKT4oZHwwKSl7RFtlPj4yXT1qO2JyZWFrIGt9RFtlPj4yXT1kfWY9ZisxfDA7ZT1EW2ErOD4+Ml07aWYoKGZ8MCk8KGV8MCkpe2Q9RFsoZys4fDApKyhmPDwyKT4+Ml07Y29udGludWV9YnJlYWt9ZD0wO2lmKChlfDApPD0wKXticmVhayBqfWU9azw8MztqPWUrY3wwO2k9YitlfDA7d2hpbGUoMSl7Zj1kPDwyO2U9ZitqfDA7Zj1EW2YraT4+Ml0rRFtmK2g+PjJdfDA7RFtlPj4yXT1mO2w6e2lmKChmfDApPkRbYSsxNj4+Ml0pe2Y9Zi1EW2ErMjA+PjJdfDB9ZWxzZXtpZigoZnwwKT49RFthKzEyPj4yXSl7YnJlYWsgbH1mPWYrRFthKzIwPj4yXXwwfURbZT4+Ml09Zn1kPWQrMXwwO2lmKChkfDApPERbYSs4Pj4yXSl7Y29udGludWV9YnJlYWt9fWs9aysxfDA7aWYoKG98MCk9PShrfDApKXticmVhayBhfWU9RFthKzU2Pj4yXTtkPURbZT4+Ml07aWYoRFtlKzQ+PjJdLWQ+PjI+Pj4wPms+Pj4wKXtjb250aW51ZX1icmVha319dWEoKTtUKCl9JD1nKzMyfDA7cmV0dXJuIDF9ZnVuY3Rpb24gUGgoYSxiLGMsZCxlLGYpe2E9YXwwO2I9YnwwO2M9Y3wwO2Q9ZHwwO2U9ZXwwO2Y9ZnwwO3ZhciBnPTAsaD0wLGk9MCxqPTAsaz0wLGw9MCxtPTAsbj0wLG89MDtnPSQtMzJ8MDskPWc7RFthKzY4Pj4yXT1mO2U9RFthKzU2Pj4yXTtkPURbZT4+Ml07Zj1EW2UrND4+Ml07RFtnKzI0Pj4yXT0wO0RbZysxNj4+Ml09MDtEW2crMjA+PjJdPTA7YTp7Zj1mLWR8MDtpZigoZnwwKTw9MCl7YnJlYWsgYX1kPURbZT4+Ml07aWYoKGR8MCkhPURbZSs0Pj4yXSl7bT1hKzExMnwwO249YSs2MHwwO2U9Zj4+PjJ8MDtvPWU+Pj4wPjE/ZToxO3doaWxlKDEpe0piKG4sRFsoazw8MikrZD4+Ml0sZysxNnwwKTtlPURbZysyMD4+Ml07ZD1lPj4zMTtmPURbZysxNj4+Ml07aD1mPj4zMTtqPURbZysyND4+Ml07aT1qPj4zMTtpPWleaStqO2g9aSsoKGReZCtlKSsoaF5mK2gpfDApfDA7ZD0wO2Q9aD4+PjA8aT4+PjA/MTpkO2I6e2lmKCEoZHxoKSl7RFtnKzE2Pj4yXT1EW2ErMTA4Pj4yXTticmVhayBifWk9RFthKzEwOD4+Ml07bD1pPj4zMTtmPWFpKCRoKGksbCxmLGY+PjMxKSxhYSxoLGQpO0RbZysxNj4+Ml09ZjtkPWFpKCRoKGksbCxlLGU+PjMxKSxhYSxoLGQpO0RbZysyMD4+Ml09ZDtlPWQ7ZD1kPj4zMTtlPWUrZF5kO2Q9Zj4+MzE7ZD1lKyhkK2ZeZCl8MDtpZigoanwwKT49MCl7RFtnKzI0Pj4yXT1pLWQ7YnJlYWsgYn1EW2crMjQ+PjJdPWQtaX1kPUdhKG0pO2Y9RFtnKzE2Pj4yXTtjOntpZihkKXtEW2crMjQ+PjJdPTAtRFtnKzI0Pj4yXTtlPTAtRFtnKzIwPj4yXXwwO0RbZysyMD4+Ml09ZTtmPTAtZnwwO0RbZysxNj4+Ml09ZjticmVhayBjfWU9RFtnKzIwPj4yXX1kOntpZigoZnwwKT49MCl7ZD1EW2ErMTA4Pj4yXTtmPWQrRFtnKzI0Pj4yXXwwO2Q9ZCtlfDA7YnJlYWsgZH1lOntpZigoZXwwKTwwKXtmPURbZysyND4+Ml07ZD1mPj4zMTtkPWReZCtmO2JyZWFrIGV9Zj1EW2crMjQ+PjJdO2Q9Zj4+MzE7ZD1EW2ErMTAwPj4yXS0oZF5kK2YpfDB9aWYoKGZ8MCk8MCl7Zj1lO2U9ZT4+MzE7Zj1mK2VeZTticmVhayBkfWY9ZTtlPWU+PjMxO2Y9RFthKzEwMD4+Ml0tKGYrZV5lKXwwfWU9RFthKzEwMD4+Ml07Zjp7aWYoIShkfGYpKXtmPWU7ZD1mO2JyZWFrIGZ9aWYoISgoZXwwKSE9KGZ8MCl8ZCkpe2Q9ZjticmVhayBmfWg9KGR8MCkhPShlfDApO2lmKCEoZnxoKSl7Zj1kO2JyZWFrIGZ9Zzp7aWYoZCl7YnJlYWsgZ31qPURbYSsxMDg+PjJdO2lmKChqfDApPj0oZnwwKSl7YnJlYWsgZ31mPShqPDwxKS1mfDA7ZD0wO2JyZWFrIGZ9aDp7aWYoaCl7YnJlYWsgaH1oPURbYSsxMDg+PjJdO2lmKChofDApPD0oZnwwKSl7YnJlYWsgaH1mPShoPDwxKS1mfDA7YnJlYWsgZn1pOntpZigoZXwwKSE9KGZ8MCkpe2JyZWFrIGl9ZT1EW2ErMTA4Pj4yXTtpZigoZXwwKTw9KGR8MCkpe2JyZWFrIGl9ZD0oZTw8MSktZHwwO2JyZWFrIGZ9aWYoZil7YnJlYWsgZn1mPTA7ZT1EW2ErMTA4Pj4yXTtpZigoZXwwKT49KGR8MCkpe2JyZWFrIGZ9ZD0oZTw8MSktZHwwfURbZysxMj4+Ml09ZjtEW2crOD4+Ml09ZDtqOntpZihEW2ErOD4+Ml08PTApe2JyZWFrIGp9aD1EW2ErMzI+PjJdO2Y9MDt3aGlsZSgxKXtlPURbYSsxNj4+Ml07azp7aWYoKGV8MCk8KGR8MCkpe0RbaCsoZjw8Mik+PjJdPWU7YnJlYWsga31lPWgrKGY8PDIpfDA7aj1EW2ErMTI+PjJdO2lmKChqfDApPihkfDApKXtEW2U+PjJdPWo7YnJlYWsga31EW2U+PjJdPWR9Zj1mKzF8MDtlPURbYSs4Pj4yXTtpZigoZnwwKTwoZXwwKSl7ZD1EWyhnKzh8MCkrKGY8PDIpPj4yXTtjb250aW51ZX1icmVha31kPTA7aWYoKGV8MCk8PTApe2JyZWFrIGp9ZT1rPDwzO2o9ZStjfDA7aT1iK2V8MDt3aGlsZSgxKXtmPWQ8PDI7ZT1mK2p8MDtmPURbZitpPj4yXStEW2YraD4+Ml18MDtEW2U+PjJdPWY7bDp7aWYoKGZ8MCk+RFthKzE2Pj4yXSl7Zj1mLURbYSsyMD4+Ml18MH1lbHNle2lmKChmfDApPj1EW2ErMTI+PjJdKXticmVhayBsfWY9ZitEW2ErMjA+PjJdfDB9RFtlPj4yXT1mfWQ9ZCsxfDA7aWYoKGR8MCk8RFthKzg+PjJdKXtjb250aW51ZX1icmVha319az1rKzF8MDtpZigob3wwKT09KGt8MCkpe2JyZWFrIGF9ZT1EW2ErNTY+PjJdO2Q9RFtlPj4yXTtpZihEW2UrND4+Ml0tZD4+Mj4+PjA+az4+PjApe2NvbnRpbnVlfWJyZWFrfX11YSgpO1QoKX0kPWcrMzJ8MDtyZXR1cm4gMX1mdW5jdGlvbiBGaChhLGIsYyxkLGUsZil7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7ZT1lfDA7Zj1mfDA7dmFyIGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTAsbz0wLHA9MDtoPSQtNDh8MDskPWg7ZD1EW2ErOD4+Ml07aWYoZC0zMT4+PjA+PTQyOTQ5NjcyNjcpe0RbYSs3Nj4+Ml09ZDtlPS0xPDxkO2Q9LTItZXwwO0RbYSs4ND4+Ml09ZDtEW2ErODA+PjJdPWVeLTE7RFthKzkyPj4yXT0oZHwwKS8yO0hbYSs4OD4+Ml09SygyKS9LKGR8MCl9RFthKzUyPj4yXT1mO2U9RFthKzQwPj4yXTtkPURbZT4+Ml07Zz1EW2UrND4+Ml07Zj0wO0RbaCsxNj4+Ml09MDtEW2grOD4+Ml09MDtEW2grMTI+PjJdPTA7YTp7Zz1nLWR8MDtpZigoZ3wwKTw9MCl7YnJlYWsgYX1kPURbZT4+Ml07aWYoKGR8MCkhPURbZSs0Pj4yXSl7bT1hKzh8MDtuPWErOTZ8MDtvPWErNDR8MDtlPWc+Pj4yfDA7cD1lPj4+MD4xP2U6MTt3aGlsZSgxKXtMYihvLERbKGY8PDIpK2Q+PjJdLGgrOHwwKTtlPURbaCsxMj4+Ml07ZD1lPj4zMTtnPURbaCs4Pj4yXTtpPWc+PjMxO2s9RFtoKzE2Pj4yXTtqPWs+PjMxO2o9al5qK2s7aT1qKygoZF5kK2UpKyhpXmcraSl8MCl8MDtkPTA7ZD1pPj4+MDxqPj4+MD8xOmQ7Yjp7aWYoIShkfGkpKXtEW2grOD4+Ml09RFthKzkyPj4yXTticmVhayBifWo9RFthKzkyPj4yXTtsPWo+PjMxO2c9YWkoJGgoaixsLGcsZz4+MzEpLGFhLGksZCk7RFtoKzg+PjJdPWc7ZD1haSgkaChqLGwsZSxlPj4zMSksYWEsaSxkKTtEW2grMTI+PjJdPWQ7ZT1kO2Q9ZD4+MzE7ZT1lK2ReZDtkPWc+PjMxO2Q9ZSsoZCtnXmQpfDA7aWYoKGt8MCk+PTApe0RbaCsxNj4+Ml09ai1kO2JyZWFrIGJ9RFtoKzE2Pj4yXT1kLWp9ZD1HYShuKTtlPURbaCs4Pj4yXTtjOntpZihkKXtEW2grMTY+PjJdPTAtRFtoKzE2Pj4yXTtnPTAtRFtoKzEyPj4yXXwwO0RbaCsxMj4+Ml09ZztlPTAtZXwwO0RbaCs4Pj4yXT1lO2JyZWFrIGN9Zz1EW2grMTI+PjJdfWQ6e2lmKChlfDApPj0wKXtlPURbYSs5Mj4+Ml07ZD1lK0RbaCsxNj4+Ml18MDtlPWUrZ3wwO2JyZWFrIGR9ZTp7aWYoKGd8MCk8MCl7ZD1EW2grMTY+PjJdO2U9ZD4+MzE7ZT1lXmQrZTticmVhayBlfWQ9RFtoKzE2Pj4yXTtlPWQ+PjMxO2U9RFthKzg0Pj4yXS0oZV5kK2UpfDB9aWYoKGR8MCk8MCl7ZD1nPj4zMTtkPWQrZ15kO2JyZWFrIGR9ZD1nPj4zMTtkPURbYSs4ND4+Ml0tKGQrZ15kKXwwfWc9RFthKzg0Pj4yXTtmOntpZighKGR8ZSkpe2Q9ZztlPWQ7YnJlYWsgZn1pZighKChkfDApIT0oZ3wwKXxlKSl7ZT1kO2JyZWFrIGZ9aT0oZXwwKSE9KGd8MCk7aWYoIShkfGkpKXtkPWU7YnJlYWsgZn1nOntpZihlKXticmVhayBnfWs9RFthKzkyPj4yXTtpZigoa3wwKT49KGR8MCkpe2JyZWFrIGd9ZD0oazw8MSktZHwwO2U9MDticmVhayBmfWg6e2lmKGkpe2JyZWFrIGh9aT1EW2ErOTI+PjJdO2lmKChpfDApPD0oZHwwKSl7YnJlYWsgaH1kPShpPDwxKS1kfDA7YnJlYWsgZn1pOntpZigoZHwwKSE9KGd8MCkpe2JyZWFrIGl9Zz1EW2ErOTI+PjJdO2lmKChnfDApPD0oZXwwKSl7YnJlYWsgaX1lPShnPDwxKS1lfDA7YnJlYWsgZn1pZihkKXticmVhayBmfWQ9MDtnPURbYSs5Mj4+Ml07aWYoKGd8MCk+PShlfDApKXticmVhayBmfWU9KGc8PDEpLWV8MH1nPWY8PDM7aT1nK2J8MDtrPURbaSs0Pj4yXTtpPURbaT4+Ml07RFtoKzM2Pj4yXT1kO0RbaCszMj4+Ml09ZTtEW2grMjQ+PjJdPWk7RFtoKzI4Pj4yXT1rO0liKGgrNDB8MCxtLGgrMzJ8MCxoKzI0fDApO2Q9YytnfDA7RFtkPj4yXT1EW2grNDA+PjJdO0RbZCs0Pj4yXT1EW2grNDQ+PjJdO2Y9ZisxfDA7aWYoKHB8MCk9PShmfDApKXticmVhayBhfWU9RFthKzQwPj4yXTtkPURbZT4+Ml07aWYoRFtlKzQ+PjJdLWQ+PjI+Pj4wPmY+Pj4wKXtjb250aW51ZX1icmVha319dWEoKTtUKCl9JD1oKzQ4fDA7cmV0dXJuIDF9ZnVuY3Rpb24gQmgoYSxiLGMsZCxlLGYpe2E9YXwwO2I9YnwwO2M9Y3wwO2Q9ZHwwO2U9ZXwwO2Y9ZnwwO3ZhciBnPTAsaD0wLGk9MCxqPTAsaz0wLGw9MCxtPTAsbj0wLG89MCxwPTA7aD0kLTQ4fDA7JD1oO2Q9RFthKzg+PjJdO2lmKGQtMzE+Pj4wPj00Mjk0OTY3MjY3KXtEW2ErNzY+PjJdPWQ7ZT0tMTw8ZDtkPS0yLWV8MDtEW2ErODQ+PjJdPWQ7RFthKzgwPj4yXT1lXi0xO0RbYSs5Mj4+Ml09KGR8MCkvMjtIW2ErODg+PjJdPUsoMikvSyhkfDApfURbYSs1Mj4+Ml09ZjtlPURbYSs0MD4+Ml07ZD1EW2U+PjJdO2c9RFtlKzQ+PjJdO2Y9MDtEW2grMTY+PjJdPTA7RFtoKzg+PjJdPTA7RFtoKzEyPj4yXT0wO2E6e2c9Zy1kfDA7aWYoKGd8MCk8PTApe2JyZWFrIGF9ZD1EW2U+PjJdO2lmKChkfDApIT1EW2UrND4+Ml0pe209YSs4fDA7bj1hKzk2fDA7bz1hKzQ0fDA7ZT1nPj4+MnwwO3A9ZT4+PjA+MT9lOjE7d2hpbGUoMSl7SmIobyxEWyhmPDwyKStkPj4yXSxoKzh8MCk7ZT1EW2grMTI+PjJdO2Q9ZT4+MzE7Zz1EW2grOD4+Ml07aT1nPj4zMTtrPURbaCsxNj4+Ml07aj1rPj4zMTtqPWpeaitrO2k9aisoKGReZCtlKSsoaV5nK2kpfDApfDA7ZD0wO2Q9aT4+PjA8aj4+PjA/MTpkO2I6e2lmKCEoZHxpKSl7RFtoKzg+PjJdPURbYSs5Mj4+Ml07YnJlYWsgYn1qPURbYSs5Mj4+Ml07bD1qPj4zMTtnPWFpKCRoKGosbCxnLGc+PjMxKSxhYSxpLGQpO0RbaCs4Pj4yXT1nO2Q9YWkoJGgoaixsLGUsZT4+MzEpLGFhLGksZCk7RFtoKzEyPj4yXT1kO2U9ZDtkPWQ+PjMxO2U9ZStkXmQ7ZD1nPj4zMTtkPWUrKGQrZ15kKXwwO2lmKChrfDApPj0wKXtEW2grMTY+PjJdPWotZDticmVhayBifURbaCsxNj4+Ml09ZC1qfWQ9R2Eobik7ZT1EW2grOD4+Ml07Yzp7aWYoZCl7RFtoKzE2Pj4yXT0wLURbaCsxNj4+Ml07Zz0wLURbaCsxMj4+Ml18MDtEW2grMTI+PjJdPWc7ZT0wLWV8MDtEW2grOD4+Ml09ZTticmVhayBjfWc9RFtoKzEyPj4yXX1kOntpZigoZXwwKT49MCl7ZT1EW2ErOTI+PjJdO2Q9ZStEW2grMTY+PjJdfDA7ZT1lK2d8MDticmVhayBkfWU6e2lmKChnfDApPDApe2Q9RFtoKzE2Pj4yXTtlPWQ+PjMxO2U9ZV5kK2U7YnJlYWsgZX1kPURbaCsxNj4+Ml07ZT1kPj4zMTtlPURbYSs4ND4+Ml0tKGVeZCtlKXwwfWlmKChkfDApPDApe2Q9Zz4+MzE7ZD1kK2deZDticmVhayBkfWQ9Zz4+MzE7ZD1EW2ErODQ+PjJdLShkK2deZCl8MH1nPURbYSs4ND4+Ml07Zjp7aWYoIShkfGUpKXtkPWc7ZT1kO2JyZWFrIGZ9aWYoISgoZHwwKSE9KGd8MCl8ZSkpe2U9ZDticmVhayBmfWk9KGV8MCkhPShnfDApO2lmKCEoZHxpKSl7ZD1lO2JyZWFrIGZ9Zzp7aWYoZSl7YnJlYWsgZ31rPURbYSs5Mj4+Ml07aWYoKGt8MCk+PShkfDApKXticmVhayBnfWQ9KGs8PDEpLWR8MDtlPTA7YnJlYWsgZn1oOntpZihpKXticmVhayBofWk9RFthKzkyPj4yXTtpZigoaXwwKTw9KGR8MCkpe2JyZWFrIGh9ZD0oaTw8MSktZHwwO2JyZWFrIGZ9aTp7aWYoKGR8MCkhPShnfDApKXticmVhayBpfWc9RFthKzkyPj4yXTtpZigoZ3wwKTw9KGV8MCkpe2JyZWFrIGl9ZT0oZzw8MSktZXwwO2JyZWFrIGZ9aWYoZCl7YnJlYWsgZn1kPTA7Zz1EW2ErOTI+PjJdO2lmKChnfDApPj0oZXwwKSl7YnJlYWsgZn1lPShnPDwxKS1lfDB9Zz1mPDwzO2k9ZytifDA7az1EW2krND4+Ml07aT1EW2k+PjJdO0RbaCszNj4+Ml09ZDtEW2grMzI+PjJdPWU7RFtoKzI0Pj4yXT1pO0RbaCsyOD4+Ml09aztJYihoKzQwfDAsbSxoKzMyfDAsaCsyNHwwKTtkPWMrZ3wwO0RbZD4+Ml09RFtoKzQwPj4yXTtEW2QrND4+Ml09RFtoKzQ0Pj4yXTtmPWYrMXwwO2lmKChwfDApPT0oZnwwKSl7YnJlYWsgYX1lPURbYSs0MD4+Ml07ZD1EW2U+PjJdO2lmKERbZSs0Pj4yXS1kPj4yPj4+MD5mPj4+MCl7Y29udGludWV9YnJlYWt9fXVhKCk7VCgpfSQ9aCs0OHwwO3JldHVybiAxfWZ1bmN0aW9uIEtoKGEsYixjKXthPWF8MDtiPWJ8MDtjPWN8MDt2YXIgZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MDtlPSQtMzJ8MDskPWU7YTp7aWYoKGN8MCkhPTMpe2JyZWFrIGF9Yz1EW2ErND4+Ml07YT1EW2ErMTI+PjJdO0RbZSsyND4+Ml09LTE7RFtlKzE2Pj4yXT0tMTtEW2UrMjA+PjJdPTEwNjUzNTMyMTY7RFtlKzg+PjJdPS0xO0RbZSsxMj4+Ml09LTE7aWYoKGJ8MCk9PS0yKXticmVhayBhfWs9RFtEW0RbYys0Pj4yXSs4Pj4yXSsoYTw8Mik+PjJdO2lmKChiYVtEW0RbYz4+Ml0rOD4+Ml1dKGMpfDApPT0xKXtqPURbRFtEW2MrND4+Ml0rOD4+Ml0rKGE8PDIpPj4yXTtiOntjOntpZigoYmFbRFtEW2M+PjJdKzg+PjJdXShjKXwwKSE9MXxiLTE+Pj4wPjUpe2JyZWFrIGN9aD1iYVtEW0RbYz4+Ml0rMzY+PjJdXShjKXwwO2Y9YmFbRFtEW2M+PjJdKzQ0Pj4yXV0oYyxhKXwwO2lmKCFofCFmKXticmVhayBjfWc9ZisxMnwwO2k9YmFbRFtEW2M+PjJdKzQwPj4yXV0oYyxhKXwwO2M9RFtjKzQ0Pj4yXTtpZihpKXtpZigoYnwwKSE9Nil7YnJlYWsgY31hPW5hKDExMik7RFthKzQ+PjJdPWo7Yj1EW2UrMTI+PjJdO0RbYSs4Pj4yXT1EW2UrOD4+Ml07RFthKzEyPj4yXT1iO2I9RFtlKzIwPj4yXTtEW2ErMTY+PjJdPURbZSsxNj4+Ml07RFthKzIwPj4yXT1iO0RbYSsyND4+Ml09RFtlKzI0Pj4yXTtEW2ErNDA+PjJdPWY7RFthKzM2Pj4yXT1nO0RbYSszMj4+Ml09aTtEW2ErMjg+PjJdPWM7RFthKzY4Pj4yXT1mO0RbYS0gLTY0Pj4yXT1nO0RbYSs2MD4+Ml09aTtEW2ErNTY+PjJdPWM7RFthKzQ4Pj4yXT0wO0RbYSs1Mj4+Ml09MDtEW2E+PjJdPTU5NTY7RFthKzg4Pj4yXT0xMDY1MzUzMjE2O0RbYSs5Mj4+Ml09LTE7RFthKzgwPj4yXT0tMTtEW2ErODQ+PjJdPS0xO0RbYSs3Mj4+Ml09MTtEW2ErNzY+PjJdPS0xO0RbYSs0ND4+Ml09NjUyMDtiPWErOTZ8MDtEW2I+PjJdPTA7RFtiKzQ+PjJdPTA7QltiKzV8MF09MDtCW2IrNnwwXT0wO0JbYis3fDBdPTA7QltiKzh8MF09MDtCW2IrOXwwXT0wO0JbYisxMHwwXT0wO0JbYisxMXwwXT0wO0JbYisxMnwwXT0wO2JyZWFrIGJ9aWYoKGJ8MCkhPTYpe2JyZWFrIGN9ZD1uYSgxMTIpO0RbZCs0Pj4yXT1qO2E9RFtlKzEyPj4yXTtEW2QrOD4+Ml09RFtlKzg+PjJdO0RbZCsxMj4+Ml09YTthPURbZSsyMD4+Ml07RFtkKzE2Pj4yXT1EW2UrMTY+PjJdO0RbZCsyMD4+Ml09YTtEW2QrMjQ+PjJdPURbZSsyND4+Ml07RFtkKzQwPj4yXT1mO0RbZCszNj4+Ml09ZztEW2QrMzI+PjJdPWg7RFtkKzI4Pj4yXT1jO0RbZCs2OD4+Ml09ZjtEW2QtIC02ND4+Ml09ZztEW2QrNjA+PjJdPWg7RFtkKzU2Pj4yXT1jO0RbZCs0OD4+Ml09MDtEW2QrNTI+PjJdPTA7RFtkPj4yXT02OTYwO0RbZCs4OD4+Ml09MTA2NTM1MzIxNjtEW2QrOTI+PjJdPS0xO0RbZCs4MD4+Ml09LTE7RFtkKzg0Pj4yXT0tMTtEW2QrNzI+PjJdPTE7RFtkKzc2Pj4yXT0tMTtEW2QrNDQ+PjJdPTczODA7YT1kKzk2fDA7RFthPj4yXT0wO0RbYSs0Pj4yXT0wO0JbYSs1fDBdPTA7QlthKzZ8MF09MDtCW2ErN3wwXT0wO0JbYSs4fDBdPTA7QlthKzl8MF09MDtCW2ErMTB8MF09MDtCW2ErMTF8MF09MDtCW2ErMTJ8MF09MH1hPWR9ZD1hO2lmKGEpe2JyZWFrIGF9fWQ9bmEoMjgpO0RbZCs0Pj4yXT1rO2E9RFtlKzEyPj4yXTtEW2QrOD4+Ml09RFtlKzg+PjJdO0RbZCsxMj4+Ml09YTthPURbZSsyMD4+Ml07RFtkKzE2Pj4yXT1EW2UrMTY+PjJdO0RbZCsyMD4+Ml09YTtEW2QrMjQ+PjJdPURbZSsyND4+Ml07RFtkPj4yXT03NzkyfSQ9ZSszMnwwO3JldHVybiBkfDB9ZnVuY3Rpb24gamYoYSxiKXthPWF8MDtiPWJ8MDt2YXIgYz0wLGQ9MCxlPTAsZj0wO2Y9JC0zMnwwOyQ9ZjtlPWYrOHwwO2M9JC04MHwwOyQ9YzthPURbYiszNj4+Ml07RFtjKzcyPj4yXT1EW2IrMzI+PjJdO0RbYys3Nj4+Ml09YTtkPURbYisyOD4+Ml07YT1jLSAtNjR8MDtEW2E+PjJdPURbYisyND4+Ml07RFthKzQ+PjJdPWQ7YT1EW2IrMjA+PjJdO0RbYys1Nj4+Ml09RFtiKzE2Pj4yXTtEW2MrNjA+PjJdPWE7YT1EW2IrMTI+PjJdO0RbYys0OD4+Ml09RFtiKzg+PjJdO0RbYys1Mj4+Ml09YTthPURbYis0Pj4yXTtEW2MrNDA+PjJdPURbYj4+Ml07RFtjKzQ0Pj4yXT1hO2VjKGMrOHwwLGMrNDB8MCxjKzI0fDApO2E9RFtjKzg+PjJdO2E6e2lmKGEpe0RbZT4+Ml09YTthPWUrNHwwO2lmKEJbYysyM3wwXT49MCl7Yj1jKzh8NDtlPURbYis0Pj4yXTtEW2E+PjJdPURbYj4+Ml07RFthKzQ+PjJdPWU7RFthKzg+PjJdPURbYis4Pj4yXTticmVhayBhfXRhKGEsRFtjKzEyPj4yXSxEW2MrMTY+PjJdKTtpZihCW2MrMjN8MF0+PTApe2JyZWFrIGF9bWEoRFtjKzEyPj4yXSk7YnJlYWsgYX1pZihCW2MrMjN8MF08MCl7bWEoRFtjKzEyPj4yXSl9YT1FW2MrMzF8MF07aWYoYT4+PjA+PTIpe2E9bmEoMzIpO2I9RVsxNjE5XXxFWzE2MjBdPDw4O0JbYSsyNHwwXT1iO0JbYSsyNXwwXT1iPj4+ODtiPUVbMTYxNV18RVsxNjE2XTw8OHwoRVsxNjE3XTw8MTZ8RVsxNjE4XTw8MjQpO2Q9RVsxNjExXXxFWzE2MTJdPDw4fChFWzE2MTNdPDwxNnxFWzE2MTRdPDwyNCk7QlthKzE2fDBdPWQ7QlthKzE3fDBdPWQ+Pj44O0JbYSsxOHwwXT1kPj4+MTY7QlthKzE5fDBdPWQ+Pj4yNDtCW2ErMjB8MF09YjtCW2ErMjF8MF09Yj4+Pjg7QlthKzIyfDBdPWI+Pj4xNjtCW2ErMjN8MF09Yj4+PjI0O2I9RVsxNjA3XXxFWzE2MDhdPDw4fChFWzE2MDldPDwxNnxFWzE2MTBdPDwyNCk7ZD1FWzE2MDNdfEVbMTYwNF08PDh8KEVbMTYwNV08PDE2fEVbMTYwNl08PDI0KTtCW2ErOHwwXT1kO0JbYSs5fDBdPWQ+Pj44O0JbYSsxMHwwXT1kPj4+MTY7QlthKzExfDBdPWQ+Pj4yNDtCW2ErMTJ8MF09YjtCW2ErMTN8MF09Yj4+Pjg7QlthKzE0fDBdPWI+Pj4xNjtCW2ErMTV8MF09Yj4+PjI0O2I9RVsxNTk5XXxFWzE2MDBdPDw4fChFWzE2MDFdPDwxNnxFWzE2MDJdPDwyNCk7ZD1FWzE1OTVdfEVbMTU5Nl08PDh8KEVbMTU5N108PDE2fEVbMTU5OF08PDI0KTtCW2F8MF09ZDtCW2ErMXwwXT1kPj4+ODtCW2ErMnwwXT1kPj4+MTY7QlthKzN8MF09ZD4+PjI0O0JbYSs0fDBdPWI7QlthKzV8MF09Yj4+Pjg7QlthKzZ8MF09Yj4+PjE2O0JbYSs3fDBdPWI+Pj4yNDtCW2ErMjZ8MF09MDtEW2MrOD4+Ml09LTE7Yj1jKzh8NDt0YShiLGEsMjYpO2Q9QltjKzIzfDBdO0RbZT4+Ml09RFtjKzg+PjJdO2U9ZSs0fDA7Yjp7aWYoKGR8MCk+PTApe2Q9RFtiKzQ+PjJdO0RbZT4+Ml09RFtiPj4yXTtEW2UrND4+Ml09ZDtEW2UrOD4+Ml09RFtiKzg+PjJdO2JyZWFrIGJ9dGEoZSxEW2MrMTI+PjJdLERbYysxNj4+Ml0pfWlmKEJbYysyM3wwXTwwKXttYShEW2MrMTI+PjJdKX1tYShhKTticmVhayBhfURbZT4+Ml09MDtEW2UrND4+Ml09MDtEW2UrMTY+PjJdPWE7RFtlKzg+PjJdPTA7RFtlKzEyPj4yXT0wfSQ9Yys4MHwwO2E9RFtmKzI0Pj4yXTtpZihCW2YrMjN8MF08MCl7bWEoRFtmKzEyPj4yXSl9JD1mKzMyfDA7cmV0dXJuIGF8MH1mdW5jdGlvbiBKZihhLGIsYyl7YT1hfDA7Yj1ifDA7Yz1jfDA7dmFyIGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTA7aT1jO2E6e2Q9YTtpZihEW2QrMTI+PjJdPT0oYnwwKSl7YnJlYWsgYX1hPWI7Yj1EW2QrND4+Ml07Yz1EW2Q+PjJdO2lmKChifDApIT0oY3wwKSl7d2hpbGUoMSl7ZT1iLTEyfDA7aWYoQltiLTF8MF08MCl7bWEoRFtlPj4yXSl9Yj1lO2lmKChjfDApIT0oYnwwKSl7Y29udGludWV9YnJlYWt9fURbZCsxMj4+Ml09YTtEW2QrND4+Ml09YztiPURbYT4+Ml07aj1hKzR8MDtpZigoYnwwKT09KGp8MCkpe2JyZWFrIGF9d2hpbGUoMSl7Yjp7aWYoRFtkKzg+PjJdIT0oY3wwKSl7Yzp7aWYoQltiKzI3fDBdPj0wKXthPURbYisyMD4+Ml07RFtjPj4yXT1EW2IrMTY+PjJdO0RbYys0Pj4yXT1hO0RbYys4Pj4yXT1EW2IrMjQ+PjJdO2JyZWFrIGN9dGEoYyxEW2IrMTY+PjJdLERbYisyMD4+Ml0pfURbZCs0Pj4yXT1jKzEyO2JyZWFrIGJ9Zz0wO2Q6e2U6e2Y6e2U9RFtkKzQ+PjJdO2E9RFtkPj4yXTtmPShlLWF8MCkvMTJ8MDtjPWYrMXwwO2lmKGM+Pj4wPDM1NzkxMzk0Mil7aD0oRFtkKzg+PjJdLWF8MCkvMTJ8MDtrPWg8PDE7Yz1oPj4+MDwxNzg5NTY5NzA/Yz4+PjA+az4+PjA/YzprOjM1NzkxMzk0MTtpZihjKXtpZihjPj4+MD49MzU3OTEzOTQyKXticmVhayBmfWc9bmEoSihjLDEyKSl9aD1KKGMsMTIpO2M9SihmLDEyKStnfDA7Zzp7aWYoQltiKzI3fDBdPj0wKXtmPURbYisyMD4+Ml07RFtjPj4yXT1EW2IrMTY+PjJdO0RbYys0Pj4yXT1mO0RbYys4Pj4yXT1EW2IrMjQ+PjJdO2JyZWFrIGd9dGEoYyxEW2IrMTY+PjJdLERbYisyMD4+Ml0pO2U9RFtkKzQ+PjJdO2E9RFtkPj4yXX1nPWcraHwwO2Y9YysxMnwwO2lmKChhfDApPT0oZXwwKSl7YnJlYWsgZX13aGlsZSgxKXtlPWUtMTJ8MDtoPURbZSs0Pj4yXTtjPWMtMTJ8MDtEW2M+PjJdPURbZT4+Ml07RFtjKzQ+PjJdPWg7RFtjKzg+PjJdPURbZSs4Pj4yXTtEW2U+PjJdPTA7RFtlKzQ+PjJdPTA7RFtlKzg+PjJdPTA7aWYoKGF8MCkhPShlfDApKXtjb250aW51ZX1icmVha31EW2QrOD4+Ml09ZzthPURbZCs0Pj4yXTtEW2QrND4+Ml09ZjtlPURbZD4+Ml07RFtkPj4yXT1jO2lmKChhfDApPT0oZXwwKSl7YnJlYWsgZH13aGlsZSgxKXtjPWEtMTJ8MDtpZihCW2EtMXwwXTwwKXttYShEW2M+PjJdKX1hPWM7aWYoKGN8MCkhPShlfDApKXtjb250aW51ZX1icmVha31icmVhayBkfXFhKCk7VCgpfXJhKDEzMjYpO1QoKX1EW2QrOD4+Ml09ZztEW2QrND4+Ml09ZjtEW2Q+PjJdPWN9aWYoZSl7bWEoZSl9fWM9RFtiKzQ+PjJdO2g6e2lmKCFjKXthPURbYis4Pj4yXTtpZihEW2E+PjJdPT0oYnwwKSl7YnJlYWsgaH1iPWIrOHwwO3doaWxlKDEpe2M9RFtiPj4yXTtiPWMrOHwwO2E9RFtjKzg+PjJdO2lmKChjfDApIT1EW2E+PjJdKXtjb250aW51ZX1icmVha31icmVhayBofXdoaWxlKDEpe2E9YztjPURbYz4+Ml07aWYoYyl7Y29udGludWV9YnJlYWt9fWlmKChhfDApPT0oanwwKSl7YnJlYWsgYX1jPURbZCs0Pj4yXTtiPWE7Y29udGludWV9fWM9MDtpOntpZigoaXwwKTwwKXticmVhayBpfWE9RFtkPj4yXTtpZigoRFtkKzQ+PjJdLWF8MCkvMTI+Pj4wPD1pPj4+MCl7YnJlYWsgaX1hPWErSihpLDEyKXwwO2M9QlthKzExfDBdPDA/RFthPj4yXTphfXJldHVybiBjfDB9ZnVuY3Rpb24gaWMoYSxiKXt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTAsbz0wO2E6e2lmKCFGW2IrMzg+PjFdKXticmVhayBhfWlmKCFTYSgxLGErMTJ8MCxiKSl7YnJlYWsgYX1mPURbYSsxMj4+Ml07Yz1EW2E+PjJdO2Q9RFthKzQ+PjJdLWM+PjI7Yjp7aWYoZj4+PjA+ZD4+PjApe3NhKGEsZi1kfDApO2Y9RFthKzEyPj4yXTticmVhayBifWlmKGQ+Pj4wPD1mPj4+MCl7YnJlYWsgYn1EW2ErND4+Ml09YysoZjw8Mil9aWYoIWYpe3JldHVybiAxfWo9RFtiKzg+PjJdO2k9RFtiKzEyPj4yXTtsPURbYT4+Ml07d2hpbGUoMSl7ZD1EW2IrMjA+PjJdO2M9RFtiKzE2Pj4yXTtpZigoaXwwKTw9KGR8MCkmaj4+PjA8PWM+Pj4wfChkfDApPihpfDApKXtyZXR1cm4gMH1tPURbYj4+Ml07az1FW20rY3wwXTtjPWMrMXwwO2Q9Yz9kOmQrMXwwO2g9YztEW2IrMTY+PjJdPWM7RFtiKzIwPj4yXT1kO2M9az4+PjJ8MDtlPTA7Yzp7ZDp7ZTp7Zjp7bj1rJjM7c3dpdGNoKG58MCl7Y2FzZSAzOmJyZWFrIGY7Y2FzZSAwOmJyZWFrIGQ7ZGVmYXVsdDpicmVhayBlfX1kPWMrZ3wwO2lmKGQ+Pj4wPj1mPj4+MCl7cmV0dXJuIDB9cGEobCsoZzw8Mil8MCwwLChrJjI1MikrNHwwKTtnPWQ7YnJlYWsgY313aGlsZSgxKXtpZigoZHwwKT49KGl8MCkmaD4+PjA+PWo+Pj4wfChkfDApPihpfDApKXticmVhayBhfWY9RVtoK218MF07aD1oKzF8MDtkPWg/ZDpkKzF8MDtEW2IrMTY+PjJdPWg7RFtiKzIwPj4yXT1kO2M9Zjw8KGU8PDN8Nil8YztlPWUrMXwwO2lmKChufDApIT0oZXwwKSl7Y29udGludWV9YnJlYWt9fURbbCsoZzw8Mik+PjJdPWN9Zj1EW2ErMTI+PjJdO2c9ZysxfDA7aWYoZj4+PjA+Zz4+PjApe2NvbnRpbnVlfWJyZWFrfWQ9YSsxNnwwO2s9RFthPj4yXTtiPURbYSsxNj4+Ml07Yz1EW2ErMjA+PjJdLWJ8MDtnPWM+PjI7Zzp7aWYoZz4+PjA8PTQwOTUpe3NhKGQsNDA5Ni1nfDApO2JyZWFrIGd9aWYoKGN8MCk9PTE2Mzg0KXticmVhayBnfURbYSsyMD4+Ml09YisxNjM4NH1iPWErMjh8MDtnPURbYj4+Ml07Yz1EW2ErMzI+PjJdLWc+PjM7aDp7aWYoYz4+PjA8Zj4+PjApe2JiKGIsZi1jfDApO2c9RFtiPj4yXTticmVhayBofWlmKGM+Pj4wPmY+Pj4wKXtEW2ErMzI+PjJdPShmPDwzKStnfWlmKCFmKXticmVhayBhfX1lPTA7YT0wO3doaWxlKDEpe2I9aysoZTw8Mil8MDtpPURbYj4+Ml07Yz1hO2g9KGU8PDMpK2d8MDtEW2grND4+Ml09YTtEW2g+PjJdPWk7aT1EW2I+PjJdO2E9aSthfDA7aWYoYT4+PjA+NDA5Nil7YnJlYWsgYX1pOntpZihhPj4+MDw9Yz4+PjApe2JyZWFrIGl9aD1EW2Q+PjJdO2I9MDtqPWkmNztpZihqKXt3aGlsZSgxKXtEW2grKGM8PDIpPj4yXT1lO2M9YysxfDA7Yj1iKzF8MDtpZigoanwwKSE9KGJ8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZihpLTE+Pj4wPD02KXticmVhayBpfXdoaWxlKDEpe2I9aCsoYzw8Mil8MDtEW2I+PjJdPWU7RFtiKzI4Pj4yXT1lO0RbYisyND4+Ml09ZTtEW2IrMjA+PjJdPWU7RFtiKzE2Pj4yXT1lO0RbYisxMj4+Ml09ZTtEW2IrOD4+Ml09ZTtEW2IrND4+Ml09ZTtjPWMrOHwwO2lmKChjfDApIT0oYXwwKSl7Y29udGludWV9YnJlYWt9fWU9ZSsxfDA7aWYoKGZ8MCkhPShlfDApKXtjb250aW51ZX1icmVha31vPShhfDApPT00MDk2fXJldHVybiBvfWZ1bmN0aW9uIEpjKGEsYixjKXthPWF8MDtiPWJ8MDtjPWN8MDt2YXIgZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTA7az0kLTE2fDA7JD1rO0Rbays4Pj4yXT1jO2U9RFthKzEyPj4yXTtkPURbYSs4Pj4yXTtmPWUtZD4+MjthOntpZigoZnwwKT4oYnwwKSl7YnJlYWsgYX1oPWIrMXwwO2lmKGg+Pj4wPmY+Pj4wKXtmPWgtZnwwO2c9RFthKzE2Pj4yXTtkPURbYSsxMj4+Ml07aWYoZj4+PjA8PWctZD4+Mj4+PjApe2lmKGYpe2U9ZDtkPWY8PDI7ZD1wYShlLDAsZCkrZHwwfURbYSsxMj4+Ml09ZDticmVhayBhfWI6e2M6e2Q6e2g9RFthKzg+PjJdO2o9ZC1oPj4yO2U9aitmfDA7aWYoZT4+PjA8MTA3Mzc0MTgyNCl7Zz1nLWh8MDtsPWc+PjE7Zz1nPj4yPj4+MDw1MzY4NzA5MTE/ZT4+PjA+bD4+PjA/ZTpsOjEwNzM3NDE4MjM7aWYoZyl7aWYoZz4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGR9aT1uYShnPDwyKX1lPShqPDwyKStpfDA7aj1mPDwyO2Y9cGEoZSwwLGopO2o9ZitqfDA7Zz0oZzw8MikraXwwO2lmKChkfDApPT0oaHwwKSl7YnJlYWsgY313aGlsZSgxKXtkPWQtNHwwO2Y9RFtkPj4yXTtEW2Q+PjJdPTA7ZT1lLTR8MDtEW2U+PjJdPWY7aWYoKGR8MCkhPShofDApKXtjb250aW51ZX1icmVha31EW2ErMTY+PjJdPWc7Zj1EW2ErMTI+PjJdO0RbYSsxMj4+Ml09ajtkPURbYSs4Pj4yXTtEW2ErOD4+Ml09ZTtpZigoZHwwKT09KGZ8MCkpe2JyZWFrIGJ9d2hpbGUoMSl7Zj1mLTR8MDtlPURbZj4+Ml07RFtmPj4yXT0wO2lmKGUpe3lhKGUpfWlmKChkfDApIT0oZnwwKSl7Y29udGludWV9YnJlYWt9YnJlYWsgYn1xYSgpO1QoKX1yYSgxMzI2KTtUKCl9RFthKzE2Pj4yXT1nO0RbYSsxMj4+Ml09ajtEW2ErOD4+Ml09Zn1pZihkKXttYShkKX1icmVhayBhfWlmKGY+Pj4wPD1oPj4+MCl7YnJlYWsgYX1kPWQrKGg8PDIpfDA7aWYoKGR8MCkhPShlfDApKXt3aGlsZSgxKXtlPWUtNHwwO2M9RFtlPj4yXTtEW2U+PjJdPTA7aWYoYyl7eWEoYyl9aWYoKGR8MCkhPShlfDApKXtjb250aW51ZX1icmVha31jPURbays4Pj4yXX1EW2ErMTI+PjJdPWR9ZTp7Zjp7ZD1EW2MrNTY+PjJdO2c6e2lmKChkfDApPjQpe2JyZWFrIGd9ZT1KKGQsMTIpK2F8MDtkPURbZSsyND4+Ml07aWYoKGR8MCkhPURbZSsyOD4+Ml0pe0RbZD4+Ml09YjtEW2UrMjQ+PjJdPWQrNDticmVhayBnfWY9RFtlKzIwPj4yXTtoPWQtZnwwO2k9aD4+MjtkPWkrMXwwO2lmKGQ+Pj4wPj0xMDczNzQxODI0KXticmVhayBmfWc9aD4+MTtnPWk+Pj4wPDUzNjg3MDkxMT9kPj4+MD5nPj4+MD9kOmc6MTA3Mzc0MTgyMztpZihnKXtpZihnPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgZX1kPW5hKGc8PDIpfWVsc2V7ZD0wfWk9ZCsoaTw8Mil8MDtEW2k+PjJdPWI7aWYoKGh8MCk+MCl7b2EoZCxmLGgpfURbZSsyMD4+Ml09ZDtEW2UrMjQ+PjJdPWkrNDtEW2UrMjg+PjJdPWQrKGc8PDIpO2lmKCFmKXticmVhayBnfW1hKGYpfURbYys2MD4+Ml09YjthPURbYSs4Pj4yXTtEW2srOD4+Ml09MDtiPWErKGI8PDIpfDA7YT1EW2I+PjJdO0RbYj4+Ml09YztpZihhKXt5YShhKX1hPURbays4Pj4yXTtEW2srOD4+Ml09MDtpZihhKXt5YShhKX0kPWsrMTZ8MDtyZXR1cm59cWEoKTtUKCl9cmEoMTMyNik7VCgpfWZ1bmN0aW9uIHJmKGEsYixjLGQpe2E9YXwwO2I9YnwwO2M9Y3wwO2Q9ZHwwO3ZhciBlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTA7ZT0kLTE2fDA7JD1lO2s9RFtiKzgwPj4yXTtuPUVbYysyNHwwXTtiPW48PDI0Pj4yNDtmPUooayxiKTthOntiOntjOnthPURbYysyOD4+Ml07ZDp7aWYoISghKChhfDApPT0xfChhfDApPT0yKXwhRVtjKzg0fDBdKSl7YT1EW2MrNDg+PjJdO2M9RFtEW2M+PjJdPj4yXTtiPTA7RFtlKzg+PjJdPTA7RFtlPj4yXT0wO0RbZSs0Pj4yXT0wO2lmKGYpe2lmKChmfDApPDApe2JyZWFrIGN9Zz1uYShmKTtiPW9hKGcsYStjfDAsZikrZnwwfWE9RFtkPj4yXTtpZihhKXtEW2QrND4+Ml09YTttYShhKX1EW2QrOD4+Ml09YjtEW2QrND4+Ml09YjtEW2Q+PjJdPWc7YT0xO2JyZWFrIGR9RFtlKzg+PjJdPTA7RFtlPj4yXT0wO0RbZSs0Pj4yXT0wO2lmKGIpe2lmKChifDApPDApe2JyZWFrIGN9Zz1uYShiKTtEW2U+PjJdPWc7YT1iK2d8MDtEW2UrOD4+Ml09YTtwYShnLDAsYik7RFtlKzQ+PjJdPWF9YT1EW2QrND4+Ml07aT1EW2Q+PjJdO2o9YS1pfDA7ZTp7aWYoaj4+PjA8Zj4+PjApe2w9Zi1qfDA7bT1EW2QrOD4+Ml07aWYobD4+PjA8PW0tYT4+PjApe2lmKGwpe2E9cGEoYSwwLGwpK2x8MH1EW2QrND4+Ml09YTticmVhayBlfWlmKChmfDApPDApe2JyZWFrIGJ9YT1tLWl8MDttPWE8PDE7YT1hPj4+MDwxMDczNzQxODIzP2Y+Pj4wPm0+Pj4wP2Y6bToyMTQ3NDgzNjQ3O2lmKGEpe2g9bmEoYSl9cGEoaCtqfDAsMCxsKTtpZigoanwwKT4wKXtvYShoLGksail9RFtkKzg+PjJdPWEraDtEW2QrND4+Ml09ZitoO0RbZD4+Ml09aDtpZighaSl7YnJlYWsgZX1tYShpKTticmVhayBlfWlmKGY+Pj4wPj1qPj4+MCl7YnJlYWsgZX1EW2QrND4+Ml09ZitpfWY6e2lmKCFrKXtiPTA7YnJlYWsgZn1pZighYil7YT0wO2I9MTt3aGlsZSgxKXtpZighRGIoYyxFW2MrODR8MF0/YTpEW0RbYys2OD4+Ml0rKGE8PDIpPj4yXSxCW2MrMjR8MF0sZykpe2JyZWFrIGZ9YT1hKzF8MDtiPWs+Pj4wPmE+Pj4wO2lmKChhfDApIT0oa3wwKSl7Y29udGludWV9YnJlYWt9YnJlYWsgZn1hPWItMXwwO2k9YSYtMjtqPWEmMTthPTA7Yj0xO2g9MDt3aGlsZSgxKXtmPURbZT4+Ml07aWYoRGIoYyxFW2MrODR8MF0/aDpEW0RbYys2OD4+Ml0rKGg8PDIpPj4yXSxCW2MrMjR8MF0sZikpe0JbRFtkPj4yXSthfDBdPUVbZnwwXTtiPTE7YT1hKzF8MDtnPTA7Zzp7aDp7c3dpdGNoKG4tMXwwKXtkZWZhdWx0OndoaWxlKDEpe0JbRFtkPj4yXSthfDBdPUVbRFtlPj4yXStifDBdO0JbKERbZD4+Ml0rYXwwKSsxfDBdPUVbKERbZT4+Ml0rYnwwKSsxfDBdO2I9YisyfDA7YT1hKzJ8MDtnPWcrMnwwO2lmKChpfDApIT0oZ3wwKSl7Y29udGludWV9YnJlYWt9O2JyZWFrO2Nhc2UgMDpicmVhayBnO2Nhc2UgMTpicmVhayBofX1pZighail7YnJlYWsgZ31CW0RbZD4+Ml0rYXwwXT1FW0RbZT4+Ml0rYnwwXTthPWErMXwwfWg9aCsxfDA7Yj1rPj4+MD5oPj4+MDtpZigoaHwwKSE9KGt8MCkpe2NvbnRpbnVlfX1icmVha31nPURbZT4+Ml19aWYoZyl7bWEoZyl9YT0hYn0kPWUrMTZ8MDthPWEmMTticmVhayBhfXFhKCk7VCgpfXFhKCk7VCgpfXJldHVybiBhfDB9ZnVuY3Rpb24gcWYoYSxiLGMsZCl7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7dmFyIGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MDtlPSQtMTZ8MDskPWU7az1EW2IrODA+PjJdO249RVtjKzI0fDBdO2I9bjw8MjQ+PjI0O2Y9SihrLGIpO2E6e2I6e2M6e2E9RFtjKzI4Pj4yXTtkOntpZighKCEoKGF8MCk9PTF8KGF8MCk9PTIpfCFFW2MrODR8MF0pKXthPURbYys0OD4+Ml07Yz1EW0RbYz4+Ml0+PjJdO2I9MDtEW2UrOD4+Ml09MDtEW2U+PjJdPTA7RFtlKzQ+PjJdPTA7aWYoZil7aWYoKGZ8MCk8MCl7YnJlYWsgY31nPW5hKGYpO2I9b2EoZyxhK2N8MCxmKStmfDB9YT1EW2Q+PjJdO2lmKGEpe0RbZCs0Pj4yXT1hO21hKGEpfURbZCs4Pj4yXT1iO0RbZCs0Pj4yXT1iO0RbZD4+Ml09ZzthPTE7YnJlYWsgZH1EW2UrOD4+Ml09MDtEW2U+PjJdPTA7RFtlKzQ+PjJdPTA7aWYoYil7aWYoKGJ8MCk8MCl7YnJlYWsgY31nPW5hKGIpO0RbZT4+Ml09ZzthPWIrZ3wwO0RbZSs4Pj4yXT1hO3BhKGcsMCxiKTtEW2UrND4+Ml09YX1hPURbZCs0Pj4yXTtpPURbZD4+Ml07aj1hLWl8MDtlOntpZihqPj4+MDxmPj4+MCl7bD1mLWp8MDttPURbZCs4Pj4yXTtpZihsPj4+MDw9bS1hPj4+MCl7aWYobCl7YT1wYShhLDAsbCkrbHwwfURbZCs0Pj4yXT1hO2JyZWFrIGV9aWYoKGZ8MCk8MCl7YnJlYWsgYn1hPW0taXwwO209YTw8MTthPWE+Pj4wPDEwNzM3NDE4MjM/Zj4+PjA+bT4+PjA/ZjptOjIxNDc0ODM2NDc7aWYoYSl7aD1uYShhKX1wYShoK2p8MCwwLGwpO2lmKChqfDApPjApe29hKGgsaSxqKX1EW2QrOD4+Ml09YStoO0RbZCs0Pj4yXT1mK2g7RFtkPj4yXT1oO2lmKCFpKXticmVhayBlfW1hKGkpO2JyZWFrIGV9aWYoZj4+PjA+PWo+Pj4wKXticmVhayBlfURbZCs0Pj4yXT1mK2l9Zjp7aWYoIWspe2I9MDticmVhayBmfWlmKCFiKXthPTA7Yj0xO3doaWxlKDEpe2lmKCFDYihjLEVbYys4NHwwXT9hOkRbRFtjKzY4Pj4yXSsoYTw8Mik+PjJdLEJbYysyNHwwXSxnKSl7YnJlYWsgZn1hPWErMXwwO2I9az4+PjA+YT4+PjA7aWYoKGF8MCkhPShrfDApKXtjb250aW51ZX1icmVha31icmVhayBmfWE9Yi0xfDA7aT1hJi0yO2o9YSYxO2E9MDtiPTE7aD0wO3doaWxlKDEpe2Y9RFtlPj4yXTtpZihDYihjLEVbYys4NHwwXT9oOkRbRFtjKzY4Pj4yXSsoaDw8Mik+PjJdLEJbYysyNHwwXSxmKSl7QltEW2Q+PjJdK2F8MF09RVtmfDBdO2I9MTthPWErMXwwO2c9MDtnOntoOntzd2l0Y2gobi0xfDApe2RlZmF1bHQ6d2hpbGUoMSl7QltEW2Q+PjJdK2F8MF09RVtEW2U+PjJdK2J8MF07QlsoRFtkPj4yXSthfDApKzF8MF09RVsoRFtlPj4yXStifDApKzF8MF07Yj1iKzJ8MDthPWErMnwwO2c9ZysyfDA7aWYoKGl8MCkhPShnfDApKXtjb250aW51ZX1icmVha307YnJlYWs7Y2FzZSAwOmJyZWFrIGc7Y2FzZSAxOmJyZWFrIGh9fWlmKCFqKXticmVhayBnfUJbRFtkPj4yXSthfDBdPUVbRFtlPj4yXStifDBdO2E9YSsxfDB9aD1oKzF8MDtiPWs+Pj4wPmg+Pj4wO2lmKChofDApIT0oa3wwKSl7Y29udGludWV9fWJyZWFrfWc9RFtlPj4yXX1pZihnKXttYShnKX1hPSFifSQ9ZSsxNnwwO2E9YSYxO2JyZWFrIGF9cWEoKTtUKCl9cWEoKTtUKCl9cmV0dXJuIGF8MH1mdW5jdGlvbiBaYyhhLGIpe2E9YXwwO2I9YnwwO3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MCxvPTAscD0wLHE9MCxyPTA7Zz0kLTgwfDA7JD1nO2E6e2lmKCFWYigxLGcrNzZ8MCxiKSl7YnJlYWsgYX1oPURbZys3Nj4+Ml07aWYoIWgpe2JyZWFrIGF9ZD1EW2IrOD4+Ml07Yz1EW2IrMTY+PjJdO2Q9JGgoZC1jfDAsRFtiKzEyPj4yXS0oRFtiKzIwPj4yXSsoYz4+PjA+ZD4+PjApfDApfDAsNSwwKTtjPWFhO2lmKGQ+Pj4wPGg+Pj4wJihjfDApPD0wfChjfDApPDApe2JyZWFrIGF9Yz1EW2ErND4+Ml07ZD1EW2ErOD4+Ml0tYz4+MjtiOntpZihkPj4+MDxoPj4+MCl7c2EoYSs0fDAsaC1kfDApO2JyZWFrIGJ9aWYoZD4+PjA8PWg+Pj4wKXticmVhayBifURbYSs4Pj4yXT1jKyhoPDwyKX1yPWErMTZ8MDtqPURbYSszMj4+Ml07az0xO3doaWxlKDEpe2M6e2U9RFtiKzEyPj4yXTtjPWU7ZD1EW2IrMjA+PjJdO3A9RFtiKzg+PjJdO209RFtiKzE2Pj4yXTtpZigoY3wwKTw9KGR8MCkmcD4+PjA8PW0+Pj4wfChjfDApPChkfDApKXticmVhayBjfXE9RFtiPj4yXTtvPUVbcSttfDBdO2M9ZDtmPW0rMXwwO2M9Zj9jOmMrMXwwO0RbYisxNj4+Ml09ZjtEW2IrMjA+PjJdPWM7aWYoKGN8MCk+PShlfDApJmY+Pj4wPj1wPj4+MHwoY3wwKT4oZXwwKSl7YnJlYWsgY31mPUVbZitxfDBdO2M9ZDtpPW0rMnwwO2M9aT4+PjA8Mj9jKzF8MDpjO0RbYisxNj4+Ml09aTtEW2IrMjA+PjJdPWM7aWYoKGN8MCk+PShlfDApJmk+Pj4wPj1wPj4+MHwoY3wwKT4oZXwwKSl7YnJlYWsgY31pPUVbaStxfDBdO2M9ZDtuPW0rM3wwO2M9bj4+PjA8Mz9jKzF8MDpjO0RbYisxNj4+Ml09bjtEW2IrMjA+PjJdPWM7aWYoKGN8MCk+PShlfDApJm4+Pj4wPj1wPj4+MHwoY3wwKT4oZXwwKSl7YnJlYWsgY31lPUVbbitxfDBdO2M9ZDtkPW0rNHwwO2M9ZD4+PjA8ND9jKzF8MDpjO0RbYisxNj4+Ml09ZDtEW2IrMjA+PjJdPWM7aWYoIWl8KChmLTEyJjI1NSk+Pj4wPDI0NXxvPj4+MD40KSl7YnJlYWsgY31jPWxiKGcrOHwwKTtuPWk8PDI0Pj4yNDtlPShlfDApIT0wO2Q9Zi0xfDA7aWYoZD4+PjA8PTEwKXtkPURbKGQ8PDIpKzEwMTgwPj4yXX1lbHNle2Q9LTF9ZD1KKGQsaSk7JGIoYyxvLG4sZixlLGQsZD4+MzEpO2lmKCFWYigxLGcrNHwwLGIpKXticmVhayBjfWY9RFtnKzQ+PjJdO0RbZys2OD4+Ml09ZjtkPVpiKG5hKDk2KSxjKTtiYVtEW0Rbaj4+Ml0rOD4+Ml1dKGosRFtqKzEyPj4yXS1EW2orOD4+Ml0+PjIsZCk7ZD0oRFtqKzEyPj4yXS1EW2orOD4+Ml0+PjIpLTF8MDtvPWQ8PDI7RFtEW28rRFtqKzg+PjJdPj4yXSs2MD4+Ml09ZjtEW0RbYSs0Pj4yXSsobDw8Mik+PjJdPWQ7az1EW2ErMTY+PjJdO2M9RFthKzIwPj4yXS1rPj4yO2Q6e2lmKChjfDApPihkfDApKXticmVhayBkfURbZz4+Ml09LTE7ZD1kKzF8MDtpZihkPj4+MD5jPj4+MCl7eGEocixkLWN8MCxnKTtrPURbcj4+Ml07YnJlYWsgZH1pZihjPj4+MDw9ZD4+PjApe2JyZWFrIGR9RFthKzIwPj4yXT0oZDw8Mikra31EW2srbz4+Ml09bDtsPWwrMXwwO2s9bD4+PjA8aD4+PjA7aWYoKGh8MCkhPShsfDApKXtjb250aW51ZX19YnJlYWt9bD0ha30kPWcrODB8MDtyZXR1cm4gbCYxfWZ1bmN0aW9uIHRkKGEsYil7dmFyIGM9MCxkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTAsaz0wLGw9MCxtPTA7aT0kLTE2fDA7JD1pO0RbaT4+Ml09YjtnPS0xO2E6e2lmKChifDApPT0tMSl7RFtpKzQ+PjJdPS0xO2JyZWFrIGF9Yz1iKzF8MDtEW2krND4+Ml09KGM+Pj4wKSUzfDA/YzpiLTJ8MDtpZigoYj4+PjApJTN8MCl7Zz1iLTF8MDticmVhayBhfWc9YisyfDB9RFtpKzg+PjJdPWc7bT0oYnwwKT09LTE/LTE6KGI+Pj4wKS8zfDA7Yjp7Yzp7ZDp7ZTp7d2hpbGUoMSl7Zjp7Zzp7aWYoKGJ8MCkhPS0xKXtjPURbRFtEW2ErOD4+Ml0rMTI+PjJdKyhiPDwyKT4+Ml07aWYoKGN8MCkhPS0xKXticmVhayBnfX1nPTA7Yz1EW2ErMjE2Pj4yXTtpZigoY3wwKT09RFthKzIyMD4+Ml0pe2JyZWFrIGZ9d2hpbGUoMSl7Zj1KKGcsMTQ0KStjfDA7Yz1EW2YrMTM2Pj4yXTtkPURbZisxNDA+PjJdO2g6e2lmKGM+Pj4wPGQ+Pj4wKXtEW2M+PjJdPWI7RFtmKzEzNj4+Ml09Yys0O2JyZWFrIGh9aD1EW2YrMTMyPj4yXTtqPWMtaHwwO2U9aj4+MjtjPWUrMXwwO2lmKGM+Pj4wPj0xMDczNzQxODI0KXticmVhayBlfWs9ZTw8MjtkPWQtaHwwO2U9ZD4+MTtkPWQ+PjI+Pj4wPDUzNjg3MDkxMT9jPj4+MD5lPj4+MD9jOmU6MTA3Mzc0MTgyMztpZihkKXtpZihkPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgZH1jPW5hKGQ8PDIpfWVsc2V7Yz0wfWU9aytjfDA7RFtlPj4yXT1iO2lmKChqfDApPjApe29hKGMsaCxqKX1EW2YrMTMyPj4yXT1jO0RbZisxMzY+PjJdPWUrNDtEW2YrMTQwPj4yXT1jKyhkPDwyKTtpZighaCl7YnJlYWsgaH1tYShoKX1nPWcrMXwwO2M9RFthKzIxNj4+Ml07aWYoZz4+PjA8KERbYSsyMjA+PjJdLWN8MCkvMTQ0Pj4+MCl7Y29udGludWV9YnJlYWt9YnJlYWsgZn1pZigoYz4+PjApLzM+Pj4wPG0+Pj4wKXticmVhayBmfWc9MDtpZihEW2ErMjIwPj4yXT09RFthKzIxNj4+Ml0pe2JyZWFrIGZ9d2hpbGUoMSl7aTp7aWYoIUdhKERbYSszNjg+PjJdKyhnPDw0KXwwKSl7YnJlYWsgaX1mPURbYSsyMTY+PjJdK0ooZywxNDQpfDA7Yz1EW2YrMTM2Pj4yXTtkPURbZisxNDA+PjJdO2lmKGM+Pj4wPGQ+Pj4wKXtEW2M+PjJdPWI7RFtmKzEzNj4+Ml09Yys0O2JyZWFrIGl9aD1EW2YrMTMyPj4yXTtqPWMtaHwwO2U9aj4+MjtjPWUrMXwwO2lmKGM+Pj4wPj0xMDczNzQxODI0KXticmVhayBjfWs9ZTw8MjtkPWQtaHwwO2U9ZD4+MTtkPWQ+PjI+Pj4wPDUzNjg3MDkxMT9jPj4+MD5lPj4+MD9jOmU6MTA3Mzc0MTgyMztpZihkKXtpZihkPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgYn1jPW5hKGQ8PDIpfWVsc2V7Yz0wfWU9aytjfDA7RFtlPj4yXT1iO2lmKChqfDApPjApe29hKGMsaCxqKX1EW2YrMTMyPj4yXT1jO0RbZisxMzY+PjJdPWUrNDtEW2YrMTQwPj4yXT1jKyhkPDwyKTtpZighaCl7YnJlYWsgaX1tYShoKX1nPWcrMXwwO2lmKGc+Pj4wPChEW2ErMjIwPj4yXS1EW2ErMjE2Pj4yXXwwKS8xNDQ+Pj4wKXtjb250aW51ZX1icmVha319bD1sKzF8MDtpZigobHwwKSE9Myl7Yj1EWyhsPDwyKStpPj4yXTtjb250aW51ZX1icmVha30kPWkrMTZ8MDtyZXR1cm4gMX1xYSgpO1QoKX1yYSgxMzI2KTtUKCl9cWEoKTtUKCl9cmEoMTMyNik7VCgpfWZ1bmN0aW9uIHZkKGEsYil7dmFyIGM9MCxkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTAsaz0wLGw9MCxtPTA7aT0kLTE2fDA7JD1pO2s9LTE7YTp7Yjp7Yzp7aWYoIURhKDEsaSsxMnwwLGIpKXticmVhayBjfWQ9RFtpKzEyPj4yXTtpZihkKXtjPURbYSs4Pj4yXTtpZihkPj4+MD4oRFtjKzQ+PjJdLURbYz4+Ml0+PjI+Pj4wKS8zPj4+MCl7YnJlYWsgY313aGlsZSgxKXtpZighRGEoMSxpKzh8MCxiKSl7YnJlYWsgY31jPURbaSs4Pj4yXTtpZighRGEoMSxpKzh8MCxiKSl7YnJlYWsgY31mPWMrZnwwO2M9RFtpKzg+PjJdO2lmKGY+Pj4wPGM+Pj4wKXticmVhayBjfWc9Zi1jfDA7Yz1EW2ErNDA+PjJdO2Q6e2lmKChjfDApIT1EW2ErNDQ+PjJdKXtEW2MrND4+Ml09ZjtEW2M+PjJdPWc7RFthKzQwPj4yXT1jKzEyO2JyZWFrIGR9ZT1jO2M9RFthKzM2Pj4yXTtqPWUtY3wwO2g9KGp8MCkvMTJ8MDtlPWgrMXwwO2lmKGU+Pj4wPj0zNTc5MTM5NDIpe2JyZWFrIGJ9bD1oPDwxO2U9aD4+PjA8MTc4OTU2OTcwP2U+Pj4wPmw+Pj4wP2U6bDozNTc5MTM5NDE7aWYoZT4+PjA+PTM1NzkxMzk0Mil7YnJlYWsgYX1lPUooZSwxMik7bD1uYShlKTtoPWwrSihoLDEyKXwwO0RbaCs0Pj4yXT1mO0RbaD4+Ml09ZztnPWgrSigoanwwKS8tMTJ8MCwxMil8MDtpZigoanwwKT4wKXtvYShnLGMsail9RFthKzQ0Pj4yXT1lK2w7RFthKzQwPj4yXT1oKzEyO0RbYSszNj4+Ml09ZztpZighYyl7YnJlYWsgZH1tYShjKX1tPW0rMXwwO2lmKChkfDApIT0obXwwKSl7Y29udGludWV9YnJlYWt9Zj0wO2NjKGIsMCwwKTtoPWQ+Pj4wPjE/ZDoxO3doaWxlKDEpe2Q9RVtiKzM2fDBdO2M9RltEW2ErND4+Ml0rMzY+PjFdO2U6e2Y6e2lmKCgoYzw8OHxjPj4+OCkmNjU1MzUpPj4+MDw9NTEzKXtpZighZCl7YnJlYWsgZX1nPTA7Yz1EW2IrMzI+PjJdO2o9Yz4+PjN8MDtrPURbYisyND4+Ml07ZD1qK2t8MDtlPURbYisyOD4+Ml07Zzp7aWYoZD4+PjA+PWU+Pj4wKXtkPWM7YnJlYWsgZ31nPUVbZHwwXTtkPWMrMXwwO0RbYiszMj4+Ml09ZDtqPWQ+Pj4zfDA7Zz1nPj4+KGMmNykmMX1pZihlPj4+MD5qK2s+Pj4wKXticmVhayBmfWJyZWFrIGV9aWYoIWQpe2JyZWFrIGV9Zz0wO2Q9RFtiKzMyPj4yXTtjPURbYisyND4+Ml0rKGQ+Pj4zfDApfDA7aWYoYz4+PjA+PUdbYisyOD4+Ml0pe2JyZWFrIGV9Zz1FW2N8MF0+Pj4oZCY3KSYxfURbYiszMj4+Ml09ZCsxfWQ9RFthKzM2Pj4yXStKKGYsMTIpfDA7QltkKzh8MF09RVtkKzh8MF0mMjU0fGcmMTtmPWYrMXwwO2lmKChofDApIT0oZnwwKSl7Y29udGludWV9YnJlYWt9QltiKzM2fDBdPTA7Yz1EW2IrMjA+PjJdO2E9MDtkPURbYiszMj4+Ml0rN3wwO2E9ZD4+PjA8Nz8xOmE7Zj1hPDwyOXxkPj4+MztkPWYrRFtiKzE2Pj4yXXwwO2E9KGE+Pj4zfDApK2N8MDtEW2IrMTY+PjJdPWQ7RFtiKzIwPj4yXT1kPj4+MDxmPj4+MD9hKzF8MDphfWs9RFtiKzE2Pj4yXX0kPWkrMTZ8MDtyZXR1cm4ga31xYSgpO1QoKX1yYSgxMzI2KTtUKCl9ZnVuY3Rpb24gZ2MoYSxiLGMpe3ZhciBkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTAsaz0wLGw9MDtlPURbYSs4Pj4yXTtnPURbYSs0Pj4yXTtpZigoZS1nfDApLzEyPj4+MD49Yj4+PjApe2E6e2lmKCFiKXticmVhayBhfWQ9ZztlPUooYiwxMiktMTJ8MDtmPSgoZT4+PjApLzEyfDApKzEmMztpZihmKXt3aGlsZSgxKXtqPURbYys0Pj4yXTtEW2Q+PjJdPURbYz4+Ml07RFtkKzQ+PjJdPWo7RFtkKzg+PjJdPURbYys4Pj4yXTtkPWQrMTJ8MDtoPWgrMXwwO2lmKChmfDApIT0oaHwwKSl7Y29udGludWV9YnJlYWt9fWc9SihiLDEyKStnfDA7aWYoZT4+PjA8MzYpe2JyZWFrIGF9d2hpbGUoMSl7Yj1EW2MrND4+Ml07RFtkPj4yXT1EW2M+PjJdO0RbZCs0Pj4yXT1iO0RbZCs4Pj4yXT1EW2MrOD4+Ml07RFtkKzIwPj4yXT1EW2MrOD4+Ml07Yj1EW2MrND4+Ml07RFtkKzEyPj4yXT1EW2M+PjJdO0RbZCsxNj4+Ml09YjtEW2QrMzI+PjJdPURbYys4Pj4yXTtiPURbYys0Pj4yXTtEW2QrMjQ+PjJdPURbYz4+Ml07RFtkKzI4Pj4yXT1iO2I9RFtjKzQ+PjJdO0RbZCszNj4+Ml09RFtjPj4yXTtEW2QrNDA+PjJdPWI7RFtkKzQ0Pj4yXT1EW2MrOD4+Ml07ZD1kKzQ4fDA7aWYoKGd8MCkhPShkfDApKXtjb250aW51ZX1icmVha319RFthKzQ+PjJdPWc7cmV0dXJufWI6e2Y9RFthPj4yXTtpPShnLWZ8MCkvMTJ8MDtkPWkrYnwwO2lmKGQ+Pj4wPDM1NzkxMzk0Mil7ZT0oZS1mfDApLzEyfDA7Zj1lPDwxO2Y9ZT4+PjA8MTc4OTU2OTcwP2Q+Pj4wPmY+Pj4wP2Q6ZjozNTc5MTM5NDE7aWYoZil7aWYoZj4+PjA+PTM1NzkxMzk0Mil7YnJlYWsgYn1qPW5hKEooZiwxMikpfWU9SihpLDEyKStqfDA7ZD1lO2I9SihiLDEyKTtpPWItMTJ8MDtrPSgoaT4+PjApLzEyfDApKzEmMztpZihrKXtkPWU7d2hpbGUoMSl7bD1EW2MrND4+Ml07RFtkPj4yXT1EW2M+PjJdO0RbZCs0Pj4yXT1sO0RbZCs4Pj4yXT1EW2MrOD4+Ml07ZD1kKzEyfDA7aD1oKzF8MDtpZigoa3wwKSE9KGh8MCkpe2NvbnRpbnVlfWJyZWFrfX1oPWIrZXwwO2lmKGk+Pj4wPj0zNil7d2hpbGUoMSl7Yj1EW2MrND4+Ml07RFtkPj4yXT1EW2M+PjJdO0RbZCs0Pj4yXT1iO0RbZCs4Pj4yXT1EW2MrOD4+Ml07RFtkKzIwPj4yXT1EW2MrOD4+Ml07Yj1EW2MrND4+Ml07RFtkKzEyPj4yXT1EW2M+PjJdO0RbZCsxNj4+Ml09YjtEW2QrMzI+PjJdPURbYys4Pj4yXTtiPURbYys0Pj4yXTtEW2QrMjQ+PjJdPURbYz4+Ml07RFtkKzI4Pj4yXT1iO2I9RFtjKzQ+PjJdO0RbZCszNj4+Ml09RFtjPj4yXTtEW2QrNDA+PjJdPWI7RFtkKzQ0Pj4yXT1EW2MrOD4+Ml07ZD1kKzQ4fDA7aWYoKGh8MCkhPShkfDApKXtjb250aW51ZX1icmVha319Yj1EW2E+PjJdO2M9Zy1ifDA7ZD1lK0ooKGN8MCkvLTEyfDAsMTIpfDA7aWYoKGN8MCk+MCl7b2EoZCxiLGMpfURbYSs4Pj4yXT1KKGYsMTIpK2o7RFthKzQ+PjJdPWg7RFthPj4yXT1kO2lmKGIpe21hKGIpfXJldHVybn1xYSgpO1QoKX1yYSgxMzI2KTtUKCl9ZnVuY3Rpb24gc2YoYSxiLGMsZCl7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7dmFyIGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MCxvPTAscD0wLHE9MDthPTA7aD0kLTE2fDA7JD1oO2s9RFtiKzgwPj4yXTtlPUJbYysyNHwwXTtEW2grOD4+Ml09MDtEW2g+PjJdPTA7RFtoKzQ+PjJdPTA7YTp7Yjp7aWYoZSl7aWYoKGV8MCk8MCl7YnJlYWsgYn1iPWU8PDI7YT1uYShiKTtEW2g+PjJdPWE7Zz1hK2J8MDtEW2grOD4+Ml09ZztpPWItNHwwO2w9KGk+Pj4yfDApKzEmNztjOntpZighbCl7Yj1hO2JyZWFrIGN9Yj1hO3doaWxlKDEpe0RbYj4+Ml09LTEwNzM3NDE4MjQ7Yj1iKzR8MDtmPWYrMXwwO2lmKChsfDApIT0oZnwwKSl7Y29udGludWV9YnJlYWt9fWlmKGk+Pj4wPj0yOCl7d2hpbGUoMSl7RFtiKzI0Pj4yXT0tMTA3Mzc0MTgyNDtEW2IrMjg+PjJdPS0xMDczNzQxODI0O0RbYisxNj4+Ml09LTEwNzM3NDE4MjQ7RFtiKzIwPj4yXT0tMTA3Mzc0MTgyNDtEW2IrOD4+Ml09LTEwNzM3NDE4MjQ7RFtiKzEyPj4yXT0tMTA3Mzc0MTgyNDtEW2I+PjJdPS0xMDczNzQxODI0O0RbYis0Pj4yXT0tMTA3Mzc0MTgyNDtiPWIrMzJ8MDtpZigoZ3wwKSE9KGJ8MCkpe2NvbnRpbnVlfWJyZWFrfX1EW2grND4+Ml09Z31iPUooZSxrKTtnPURbZD4+Ml07Zj1EW2QrND4+Ml0tZz4+MjtkOntpZihiPj4+MD5mPj4+MCl7c2EoZCxiLWZ8MCk7YnJlYWsgZH1pZihiPj4+MD49Zj4+PjApe2JyZWFrIGR9RFtkKzQ+PjJdPWcrKGI8PDIpfWU6e2lmKCFrKXticmVhayBlfWo9MTtpZigoZXwwKTw9MCl7Yj0wO3doaWxlKDEpe2lmKCFtYihjLEVbYys4NHwwXT9iOkRbRFtjKzY4Pj4yXSsoYjw8Mik+PjJdLEJbYysyNHwwXSxhKSl7YnJlYWsgZX1iPWIrMXwwO2o9az4+PjA+Yj4+PjA7aWYoKGJ8MCkhPShrfDApKXtjb250aW51ZX1icmVha31icmVhayBlfXA9ZSYtNDtsPWUmMztmPTA7cT1lLTE+Pj4wPDM7d2hpbGUoMSl7aWYobWIoYyxFW2MrODR8MF0/bTpEW0RbYys2OD4+Ml0rKG08PDIpPj4yXSxCW2MrMjR8MF0sYSkpe289RFtkPj4yXTtuPTA7ZT1EW2g+PjJdO2I9MDtqPTA7aWYoIXEpe3doaWxlKDEpe2c9bysoZjw8Mil8MDtpPWI8PDI7SFtnPj4yXT1IW2krZT4+Ml07SFtnKzQ+PjJdPUhbKGl8NCkrZT4+Ml07SFtnKzg+PjJdPUhbKGl8OCkrZT4+Ml07SFtnKzEyPj4yXT1IWyhpfDEyKStlPj4yXTtiPWIrNHwwO2Y9Zis0fDA7aj1qKzR8MDtpZigocHwwKSE9KGp8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZihsKXt3aGlsZSgxKXtIW28rKGY8PDIpPj4yXT1IWyhiPDwyKStlPj4yXTtiPWIrMXwwO2Y9ZisxfDA7bj1uKzF8MDtpZigobHwwKSE9KG58MCkpe2NvbnRpbnVlfWJyZWFrfX1tPW0rMXwwO2o9bT4+PjA8az4+PjA7aWYoKGt8MCkhPShtfDApKXtjb250aW51ZX19YnJlYWt9YT1EW2g+PjJdfWlmKGEpe21hKGEpfSQ9aCsxNnwwO2E9KGpeLTEpJjE7YnJlYWsgYX1xYSgpO1QoKX1yZXR1cm4gYXwwfWZ1bmN0aW9uIEliKGEsYixjLGQpe3ZhciBlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wO2s9RFtiKzE2Pj4yXTtnPURbYys0Pj4yXS1rfDA7ZT1nO2Y9RFtjPj4yXS1rfDA7RFtjPj4yXT1mO0RbYys0Pj4yXT1lO2o9RFtiKzE2Pj4yXTtlPWU+PjMxO2g9ZStnXmU7ZT1mPj4zMTtsPShqfDApPj0oaCsoZStmXmUpfDApO2E6e2lmKGwpe2U9ZzticmVhayBhfWI6e2M6e2lmKChmfDApPj0wKXtpPTE7aD0xO2lmKChnfDApPj0wKXticmVhayBifWU9MTtpPS0xO2g9LTE7aWYoZil7YnJlYWsgY31icmVhayBifWU9LTE7aT0tMTtoPS0xO2lmKChnfDApPD0wKXticmVhayBifX1pPShnfDApPD0wPy0xOjE7aD1lfWU9Zjw8MTtmPUooaCxqKTtlPWUtZnwwO2g9KEooaSxoKXwwKT49MDtpPUooaSxqKTtlPSgoaD8wLWV8MDplKStpfDApLzJ8MDtEW2MrND4+Ml09ZTtqPWY7Zj0oZzw8MSktaXwwO2Y9KGorKGg/MC1mfDA6Zil8MCkvMnwwO0RbYz4+Ml09Zn1kOntlOntmOntnOntoOntpOntqOntpZihmKXtpZigoZnwwKTwwKXticmVhayBqfWlmKChlfDApPj0wKXticmVhayBpfWJyZWFrIGZ9aWYoZSl7YnJlYWsgaH1pPTE7Zz0wO2U9MDtoPTA7YnJlYWsgZH1pPTE7aWYoKGV8MCk+MCl7YnJlYWsgZ31oPShlfDApPjA/MjUzOjA7Zz1lO2U9ZjticmVhayBkfWc9MC1lfDA7ZT0wLWZ8MDtoPTI1NDticmVhayBlfWlmKChlfDApPD0wKXticmVhayBmfX1lPTAtZXwwO2c9ZjtoPTI1MzticmVhayBlfWc9MC1mfDA7aD0yNTV9RFtjPj4yXT1lO0RbYys0Pj4yXT1nO2k9MH1jPURbZCs0Pj4yXStnfDA7Zj1EW2Q+PjJdK2V8MDtnPURbYisxNj4+Ml07azp7aWYoKGZ8MCk+KGd8MCkpe2Y9Zi1EW2IrND4+Ml18MDticmVhayBrfWlmKCgwLWd8MCk8PShmfDApKXticmVhayBrfWY9RFtiKzQ+PjJdK2Z8MH1sOntpZigoY3wwKT4oZ3wwKSl7Yz1jLURbYis0Pj4yXXwwO2JyZWFrIGx9aWYoKDAtZ3wwKTw9KGN8MCkpe2JyZWFrIGx9Yz1EW2IrND4+Ml0rY3wwfW06e2lmKGkpe2I9YzticmVhayBtfWI9YztuOntzd2l0Y2goKGgmMyktMXwwKXtjYXNlIDA6Yj0wLWZ8MDtmPWM7YnJlYWsgbTtjYXNlIDE6Yj0wLWN8MDtmPTAtZnwwO2JyZWFrIG07Y2FzZSAyOmJyZWFrIG47ZGVmYXVsdDpicmVhayBtfX1iPWY7Zj0wLWN8MH1vOntpZihsKXtjPWI7YnJlYWsgb31wOntxOntpZigoZnwwKT49MCl7Yz0xO2U9MTtpZigoYnwwKT49MCl7YnJlYWsgcH1kPTE7Yz0tMTtlPS0xO2lmKGYpe2JyZWFrIHF9YnJlYWsgcH1kPS0xO2M9LTE7ZT0tMTtpZigoYnwwKTw9MCl7YnJlYWsgcH19Yz0oYnwwKTw9MD8tMToxO2U9ZH1kPWY8PDE7Zj1KKGUsZyk7ZD1kLWZ8MDtEW2E+PjJdPWQ7aj0wLWR8MDtoPWQ7ZD0oSihjLGUpfDApPj0wO2U9SihjLGcpO2M9KChkP2o6aCkrZXwwKS8yfDA7Yj0oYjw8MSktZXwwO2Y9KGYrKGQ/MC1ifDA6Yil8MCkvMnwwfURbYT4+Ml09ZitrO0RbYSs0Pj4yXT1jK2t9ZnVuY3Rpb24gcmIoYSxiLGMsZCl7dmFyIGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wO2g9ZC1jfDA7aWYoKGh8MCk8PTApe3JldHVybn1hOntmPURbYSs4Pj4yXTtpPURbYSs0Pj4yXTtpZigoaHwwKTw9KGYtaXwwKSl7aj1pLWJ8MDtpZigoanwwKT49KGh8MCkpe2c9aTtmPWQ7YnJlYWsgYX1nPWk7Zj1jK2p8MDtpZigoZnwwKSE9KGR8MCkpe2U9Zjt3aGlsZSgxKXtCW2d8MF09RVtlfDBdO2c9ZysxfDA7ZT1lKzF8MDtpZigoZXwwKSE9KGR8MCkpe2NvbnRpbnVlfWJyZWFrfX1EW2ErND4+Ml09ZztpZigoanwwKT4wKXticmVhayBhfXJldHVybn1lPURbYT4+Ml07ZD1oKyhpLWV8MCl8MDtpZigoZHwwKT49MCl7Zz1iLWV8MDtmPWYtZXwwO2o9Zjw8MTtmPWY+Pj4wPDEwNzM3NDE4MjM/ZD4+PjA+aj4+PjA/ZDpqOjIxNDc0ODM2NDc7aWYoZil7ZD1uYShmKX1lbHNle2Q9MH1jPW9hKGcrZHwwLGMsaCk7aWYoKGd8MCk+MCl7b2EoZCxlLGcpfWM9YytofDA7aWYoKGJ8MCkhPShpfDApKXtnPShiXi0xKStpfDA7aD1pLWImNztpZihoKXtlPTA7d2hpbGUoMSl7QltjfDBdPUVbYnwwXTtjPWMrMXwwO2I9YisxfDA7ZT1lKzF8MDtpZigoaHwwKSE9KGV8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZihnPj4+MD49Nyl7d2hpbGUoMSl7QltjfDBdPUVbYnwwXTtCW2MrMXwwXT1FW2IrMXwwXTtCW2MrMnwwXT1FW2IrMnwwXTtCW2MrM3wwXT1FW2IrM3wwXTtCW2MrNHwwXT1FW2IrNHwwXTtCW2MrNXwwXT1FW2IrNXwwXTtCW2MrNnwwXT1FW2IrNnwwXTtCW2MrN3wwXT1FW2IrN3wwXTtjPWMrOHwwO2I9Yis4fDA7aWYoKGl8MCkhPShifDApKXtjb250aW51ZX1icmVha319ZT1EW2E+PjJdfURbYSs4Pj4yXT1kK2Y7RFthKzQ+PjJdPWM7RFthPj4yXT1kO2lmKGUpe21hKGUpfXJldHVybn1xYSgpO1QoKX1lPWc7ZD1lLWh8MDtpZihpPj4+MD5kPj4+MCl7d2hpbGUoMSl7QltlfDBdPUVbZHwwXTtlPWUrMXwwO2Q9ZCsxfDA7aWYoaT4+PjA+ZD4+PjApe2NvbnRpbnVlfWJyZWFrfX1EW2ErND4+Ml09ZTthPWctKGIraHwwKXwwO2lmKGEpe05hKGctYXwwLGIsYSl9aWYoKGN8MCk9PShmfDApKXtyZXR1cm59YT0oY14tMSkrZnwwO2c9Zi1jJjc7Yjp7aWYoIWcpe2U9YjticmVhayBifWQ9MDtlPWI7d2hpbGUoMSl7QltlfDBdPUVbY3wwXTtlPWUrMXwwO2M9YysxfDA7ZD1kKzF8MDtpZigoZ3wwKSE9KGR8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZihhPj4+MDw3KXtyZXR1cm59d2hpbGUoMSl7QltlfDBdPUVbY3wwXTtCW2UrMXwwXT1FW2MrMXwwXTtCW2UrMnwwXT1FW2MrMnwwXTtCW2UrM3wwXT1FW2MrM3wwXTtCW2UrNHwwXT1FW2MrNHwwXTtCW2UrNXwwXT1FW2MrNXwwXTtCW2UrNnwwXT1FW2MrNnwwXTtCW2UrN3wwXT1FW2MrN3wwXTtlPWUrOHwwO2M9Yys4fDA7aWYoKGZ8MCkhPShjfDApKXtjb250aW51ZX1icmVha319ZnVuY3Rpb24gbWYoYSxiLGMsZCl7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7dmFyIGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MCxvPTAscD0wLHE9MDtlPSQtMTZ8MDskPWU7aT1EW2IrODA+PjJdO2E9QltjKzI0fDBdO2Y9SihpLGEpO2E6e2I6e2I9RFtjKzI4Pj4yXTtjOntpZighKCEoKGJ8MCk9PTV8KGJ8MCk9PTYpfCFFW2MrODR8MF0pKXtoPURbYys0OD4+Ml07aT1EW0RbYz4+Ml0+PjJdO2I9MDtEW2UrOD4+Ml09MDtEW2U+PjJdPTA7RFtlKzQ+PjJdPTA7Yz0wO2E9Zjw8MjtpZihhKXtpZigoYXwwKTwwKXticmVhayBifWM9bmEoYSk7Yj1vYShjLGgraXwwLGEpO2o9YithfDA7Yj1iKyhhPj4yPDwyKXwwfWE9RFtkPj4yXTtpZihhKXtEW2QrND4+Ml09YTttYShhKX1EW2QrOD4+Ml09YjtEW2QrND4+Ml09ajtEW2Q+PjJdPWM7YT0xO2JyZWFrIGN9RFtlKzg+PjJdPTA7RFtlPj4yXT0wO0RbZSs0Pj4yXT0wO2lmKGEpe2lmKChhfDApPDApe2JyZWFrIGJ9Yj1hPDwyO2g9bmEoYik7RFtlPj4yXT1oO2c9YitofDA7RFtlKzg+PjJdPWc7cGEoaCwwLGIpO0RbZSs0Pj4yXT1nfWc9RFtkPj4yXTtiPURbZCs0Pj4yXS1nPj4yO2Q6e2lmKGI+Pj4wPGY+Pj4wKXtzYShkLGYtYnwwKTticmVhayBkfWlmKGI+Pj4wPD1mPj4+MCl7YnJlYWsgZH1EW2QrND4+Ml09ZysoZjw8Mil9ZTp7aWYoIWkpe2I9MDticmVhayBlfWlmKCFhKXthPTA7Yj0xO3doaWxlKDEpe2lmKCF5YihjLEVbYys4NHwwXT9hOkRbRFtjKzY4Pj4yXSsoYTw8Mik+PjJdLEJbYysyNHwwXSxoKSl7YnJlYWsgZX1hPWErMXwwO2I9aT4+PjA+YT4+PjA7aWYoKGF8MCkhPShpfDApKXtjb250aW51ZX1icmVha31icmVhayBlfXA9YSYtNDtuPWEmMztxPWEtMT4+PjA8MztiPTE7Zj0wO3doaWxlKDEpe2lmKHliKGMsRVtjKzg0fDBdP2Y6RFtEW2MrNjg+PjJdKyhmPDwyKT4+Ml0sQltjKzI0fDBdLGgpKXtvPURbZD4+Ml07bT0wO2c9RFtlPj4yXTthPTA7Yj0wO2lmKCFxKXt3aGlsZSgxKXtrPShqPDwyKStvfDA7bD1hPDwyO0Rbaz4+Ml09RFtnK2w+PjJdO0Rbays0Pj4yXT1EW2crKGx8NCk+PjJdO0Rbays4Pj4yXT1EW2crKGx8OCk+PjJdO0RbaysxMj4+Ml09RFtnKyhsfDEyKT4+Ml07YT1hKzR8MDtqPWorNHwwO2I9Yis0fDA7aWYoKHB8MCkhPShifDApKXtjb250aW51ZX1icmVha319aWYobil7d2hpbGUoMSl7RFsoajw8Mikrbz4+Ml09RFtnKyhhPDwyKT4+Ml07YT1hKzF8MDtqPWorMXwwO209bSsxfDA7aWYoKG18MCkhPShufDApKXtjb250aW51ZX1icmVha319Zj1mKzF8MDtiPWk+Pj4wPmY+Pj4wO2lmKChmfDApIT0oaXwwKSl7Y29udGludWV9fWJyZWFrfWg9RFtlPj4yXX1pZihoKXttYShoKX1hPWJeMX0kPWUrMTZ8MDthPWEmMTticmVhayBhfXFhKCk7VCgpfXJldHVybiBhfDB9ZnVuY3Rpb24gX2MoYSxiLGMsZCl7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7dmFyIGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MCxvPTAscD0wLHE9MDtlPSQtMTZ8MDskPWU7aT1EW2IrODA+PjJdO2E9QltjKzI0fDBdO2Y9SihpLGEpO2E6e2I6e2I9RFtjKzI4Pj4yXTtjOntpZighKCEoKGJ8MCk9PTV8KGJ8MCk9PTYpfCFFW2MrODR8MF0pKXtoPURbYys0OD4+Ml07aT1EW0RbYz4+Ml0+PjJdO2I9MDtEW2UrOD4+Ml09MDtEW2U+PjJdPTA7RFtlKzQ+PjJdPTA7Yz0wO2E9Zjw8MjtpZihhKXtpZigoYXwwKTwwKXticmVhayBifWM9bmEoYSk7Yj1vYShjLGgraXwwLGEpO2o9YithfDA7Yj1iKyhhPj4yPDwyKXwwfWE9RFtkPj4yXTtpZihhKXtEW2QrND4+Ml09YTttYShhKX1EW2QrOD4+Ml09YjtEW2QrND4+Ml09ajtEW2Q+PjJdPWM7YT0xO2JyZWFrIGN9RFtlKzg+PjJdPTA7RFtlPj4yXT0wO0RbZSs0Pj4yXT0wO2lmKGEpe2lmKChhfDApPDApe2JyZWFrIGJ9Yj1hPDwyO2g9bmEoYik7RFtlPj4yXT1oO2c9YitofDA7RFtlKzg+PjJdPWc7cGEoaCwwLGIpO0RbZSs0Pj4yXT1nfWc9RFtkPj4yXTtiPURbZCs0Pj4yXS1nPj4yO2Q6e2lmKGI+Pj4wPGY+Pj4wKXtzYShkLGYtYnwwKTticmVhayBkfWlmKGI+Pj4wPD1mPj4+MCl7YnJlYWsgZH1EW2QrND4+Ml09ZysoZjw8Mil9ZTp7aWYoIWkpe2I9MDticmVhayBlfWlmKCFhKXthPTA7Yj0xO3doaWxlKDEpe2lmKCF6YihjLEVbYys4NHwwXT9hOkRbRFtjKzY4Pj4yXSsoYTw8Mik+PjJdLEJbYysyNHwwXSxoKSl7YnJlYWsgZX1hPWErMXwwO2I9aT4+PjA+YT4+PjA7aWYoKGF8MCkhPShpfDApKXtjb250aW51ZX1icmVha31icmVhayBlfXA9YSYtNDtuPWEmMztxPWEtMT4+PjA8MztiPTE7Zj0wO3doaWxlKDEpe2lmKHpiKGMsRVtjKzg0fDBdP2Y6RFtEW2MrNjg+PjJdKyhmPDwyKT4+Ml0sQltjKzI0fDBdLGgpKXtvPURbZD4+Ml07bT0wO2c9RFtlPj4yXTthPTA7Yj0wO2lmKCFxKXt3aGlsZSgxKXtrPShqPDwyKStvfDA7bD1hPDwyO0Rbaz4+Ml09RFtnK2w+PjJdO0Rbays0Pj4yXT1EW2crKGx8NCk+PjJdO0Rbays4Pj4yXT1EW2crKGx8OCk+PjJdO0RbaysxMj4+Ml09RFtnKyhsfDEyKT4+Ml07YT1hKzR8MDtqPWorNHwwO2I9Yis0fDA7aWYoKHB8MCkhPShifDApKXtjb250aW51ZX1icmVha319aWYobil7d2hpbGUoMSl7RFsoajw8Mikrbz4+Ml09RFtnKyhhPDwyKT4+Ml07YT1hKzF8MDtqPWorMXwwO209bSsxfDA7aWYoKG18MCkhPShufDApKXtjb250aW51ZX1icmVha319Zj1mKzF8MDtiPWk+Pj4wPmY+Pj4wO2lmKChmfDApIT0oaXwwKSl7Y29udGludWV9fWJyZWFrfWg9RFtlPj4yXX1pZihoKXttYShoKX1hPWJeMX0kPWUrMTZ8MDthPWEmMTticmVhayBhfXFhKCk7VCgpfXJldHVybiBhfDB9ZnVuY3Rpb24gcGYoYSxiLGMsZCl7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7dmFyIGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MCxvPTAscD0wLHE9MDtlPSQtMTZ8MDskPWU7aT1EW2IrODA+PjJdO2E9QltjKzI0fDBdO2Y9SihpLGEpO2E6e2I6e2I9RFtjKzI4Pj4yXTtjOntpZighKCEoKGJ8MCk9PTN8KGJ8MCk9PTQpfCFFW2MrODR8MF0pKXtoPURbYys0OD4+Ml07aT1EW0RbYz4+Ml0+PjJdO2I9MDtEW2UrOD4+Ml09MDtEW2U+PjJdPTA7RFtlKzQ+PjJdPTA7Yz0wO2E9Zjw8MTtpZihhKXtpZigoYXwwKTwwKXticmVhayBifWM9bmEoYSk7Yj1vYShjLGgraXwwLGEpO2o9YithfDA7Yj1iKyhhPj4xPDwxKXwwfWE9RFtkPj4yXTtpZihhKXtEW2QrND4+Ml09YTttYShhKX1EW2QrOD4+Ml09YjtEW2QrND4+Ml09ajtEW2Q+PjJdPWM7YT0xO2JyZWFrIGN9RFtlKzg+PjJdPTA7RFtlPj4yXT0wO0RbZSs0Pj4yXT0wO2lmKGEpe2lmKChhfDApPDApe2JyZWFrIGJ9Yj1hPDwxO2g9bmEoYik7RFtlPj4yXT1oO2c9YitofDA7RFtlKzg+PjJdPWc7cGEoaCwwLGIpO0RbZSs0Pj4yXT1nfWc9RFtkPj4yXTtiPURbZCs0Pj4yXS1nPj4xO2Q6e2lmKGI+Pj4wPGY+Pj4wKXtnZChkLGYtYnwwKTticmVhayBkfWlmKGI+Pj4wPD1mPj4+MCl7YnJlYWsgZH1EW2QrND4+Ml09ZysoZjw8MSl9ZTp7aWYoIWkpe2I9MDticmVhayBlfWlmKCFhKXthPTA7Yj0xO3doaWxlKDEpe2lmKCFCYihjLEVbYys4NHwwXT9hOkRbRFtjKzY4Pj4yXSsoYTw8Mik+PjJdLEJbYysyNHwwXSxoKSl7YnJlYWsgZX1hPWErMXwwO2I9aT4+PjA+YT4+PjA7aWYoKGF8MCkhPShpfDApKXtjb250aW51ZX1icmVha31icmVhayBlfXA9YSYtNDtuPWEmMztxPWEtMT4+PjA8MztiPTE7Zj0wO3doaWxlKDEpe2lmKEJiKGMsRVtjKzg0fDBdP2Y6RFtEW2MrNjg+PjJdKyhmPDwyKT4+Ml0sQltjKzI0fDBdLGgpKXtvPURbZD4+Ml07bT0wO2c9RFtlPj4yXTthPTA7Yj0wO2lmKCFxKXt3aGlsZSgxKXtrPShqPDwxKStvfDA7bD1hPDwxO0Nbaz4+MV09RltnK2w+PjFdO0NbaysyPj4xXT1GW2crKGx8Mik+PjFdO0Nbays0Pj4xXT1GW2crKGx8NCk+PjFdO0Nbays2Pj4xXT1GW2crKGx8Nik+PjFdO2E9YSs0fDA7aj1qKzR8MDtiPWIrNHwwO2lmKChwfDApIT0oYnwwKSl7Y29udGludWV9YnJlYWt9fWlmKG4pe3doaWxlKDEpe0NbKGo8PDEpK28+PjFdPUZbZysoYTw8MSk+PjFdO2E9YSsxfDA7aj1qKzF8MDttPW0rMXwwO2lmKChtfDApIT0obnwwKSl7Y29udGludWV9YnJlYWt9fWY9ZisxfDA7Yj1pPj4+MD5mPj4+MDtpZigoZnwwKSE9KGl8MCkpe2NvbnRpbnVlfX1icmVha31oPURbZT4+Ml19aWYoaCl7bWEoaCl9YT1iXjF9JD1lKzE2fDA7YT1hJjE7YnJlYWsgYX1xYSgpO1QoKX1yZXR1cm4gYXwwfWZ1bmN0aW9uIG5mKGEsYixjLGQpe2E9YXwwO2I9YnwwO2M9Y3wwO2Q9ZHwwO3ZhciBlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTAsbz0wLHA9MCxxPTA7ZT0kLTE2fDA7JD1lO2k9RFtiKzgwPj4yXTthPUJbYysyNHwwXTtmPUooaSxhKTthOntiOntiPURbYysyOD4+Ml07Yzp7aWYoISghKChifDApPT0zfChifDApPT00KXwhRVtjKzg0fDBdKSl7aD1EW2MrNDg+PjJdO2k9RFtEW2M+PjJdPj4yXTtiPTA7RFtlKzg+PjJdPTA7RFtlPj4yXT0wO0RbZSs0Pj4yXT0wO2M9MDthPWY8PDE7aWYoYSl7aWYoKGF8MCk8MCl7YnJlYWsgYn1jPW5hKGEpO2I9b2EoYyxoK2l8MCxhKTtqPWIrYXwwO2I9YisoYT4+MTw8MSl8MH1hPURbZD4+Ml07aWYoYSl7RFtkKzQ+PjJdPWE7bWEoYSl9RFtkKzg+PjJdPWI7RFtkKzQ+PjJdPWo7RFtkPj4yXT1jO2E9MTticmVhayBjfURbZSs4Pj4yXT0wO0RbZT4+Ml09MDtEW2UrND4+Ml09MDtpZihhKXtpZigoYXwwKTwwKXticmVhayBifWI9YTw8MTtoPW5hKGIpO0RbZT4+Ml09aDtnPWIraHwwO0RbZSs4Pj4yXT1nO3BhKGgsMCxiKTtEW2UrND4+Ml09Z31nPURbZD4+Ml07Yj1EW2QrND4+Ml0tZz4+MTtkOntpZihiPj4+MDxmPj4+MCl7Z2QoZCxmLWJ8MCk7YnJlYWsgZH1pZihiPj4+MDw9Zj4+PjApe2JyZWFrIGR9RFtkKzQ+PjJdPWcrKGY8PDEpfWU6e2lmKCFpKXtiPTA7YnJlYWsgZX1pZighYSl7YT0wO2I9MTt3aGlsZSgxKXtpZighQWIoYyxFW2MrODR8MF0/YTpEW0RbYys2OD4+Ml0rKGE8PDIpPj4yXSxCW2MrMjR8MF0saCkpe2JyZWFrIGV9YT1hKzF8MDtiPWk+Pj4wPmE+Pj4wO2lmKChhfDApIT0oaXwwKSl7Y29udGludWV9YnJlYWt9YnJlYWsgZX1wPWEmLTQ7bj1hJjM7cT1hLTE+Pj4wPDM7Yj0xO2Y9MDt3aGlsZSgxKXtpZihBYihjLEVbYys4NHwwXT9mOkRbRFtjKzY4Pj4yXSsoZjw8Mik+PjJdLEJbYysyNHwwXSxoKSl7bz1EW2Q+PjJdO209MDtnPURbZT4+Ml07YT0wO2I9MDtpZighcSl7d2hpbGUoMSl7az0oajw8MSkrb3wwO2w9YTw8MTtDW2s+PjFdPUZbZytsPj4xXTtDW2srMj4+MV09RltnKyhsfDIpPj4xXTtDW2srND4+MV09RltnKyhsfDQpPj4xXTtDW2srNj4+MV09RltnKyhsfDYpPj4xXTthPWErNHwwO2o9ais0fDA7Yj1iKzR8MDtpZigocHwwKSE9KGJ8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZihuKXt3aGlsZSgxKXtDWyhqPDwxKStvPj4xXT1GW2crKGE8PDEpPj4xXTthPWErMXwwO2o9aisxfDA7bT1tKzF8MDtpZigobXwwKSE9KG58MCkpe2NvbnRpbnVlfWJyZWFrfX1mPWYrMXwwO2I9aT4+PjA+Zj4+PjA7aWYoKGZ8MCkhPShpfDApKXtjb250aW51ZX19YnJlYWt9aD1EW2U+PjJdfWlmKGgpe21hKGgpfWE9Yl4xfSQ9ZSsxNnwwO2E9YSYxO2JyZWFrIGF9cWEoKTtUKCl9cmV0dXJuIGF8MH1mdW5jdGlvbiBPaChhLGIsYyxkLGUsZil7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7ZT1lfDA7Zj1mfDA7dmFyIGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTAsbz0wLHA9MDtEW2ErOD4+Ml09ZTtmPWErMzJ8MDtpPURbZj4+Ml07aD1EW2ErMzY+PjJdLWk+PjI7YTp7aWYoaD4+PjA8ZT4+PjApe3NhKGYsZS1ofDApO2k9RFtmPj4yXTtmPURbYSs4Pj4yXTticmVhayBhfWlmKGU+Pj4wPGg+Pj4wKXtEW2ErMzY+PjJdPShlPDwyKStpfWY9ZX1oPTA7Zz0oZSYxMDczNzQxODIzKSE9KGV8MCk/LTE6ZTw8MjttPXBhKG5hKGcpLDAsZyk7Yjp7aWYoKGZ8MCk8PTApe2JyZWFrIGJ9d2hpbGUoMSl7Zj1oPDwyO2c9RFtmK20+PjJdO2o9RFthKzE2Pj4yXTtjOntpZigoZ3wwKT4oanwwKSl7RFtmK2k+PjJdPWo7YnJlYWsgY31mPWYraXwwO2o9RFthKzEyPj4yXTtpZigoanwwKT4oZ3wwKSl7RFtmPj4yXT1qO2JyZWFrIGN9RFtmPj4yXT1nfWY9RFthKzg+PjJdO2g9aCsxfDA7aWYoKGZ8MCk+KGh8MCkpe2NvbnRpbnVlfWJyZWFrfWlmKChmfDApPD0wKXticmVhayBifWg9MDt3aGlsZSgxKXtnPWg8PDI7Zj1nK2N8MDtnPURbYitnPj4yXStEW2craT4+Ml18MDtEW2Y+PjJdPWc7ZDp7aWYoKGd8MCk+RFthKzE2Pj4yXSl7Zz1nLURbYSsyMD4+Ml18MH1lbHNle2lmKChnfDApPj1EW2ErMTI+PjJdKXticmVhayBkfWc9ZytEW2ErMjA+PjJdfDB9RFtmPj4yXT1nfWY9RFthKzg+PjJdO2g9aCsxfDA7aWYoKGZ8MCk+KGh8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZighKChkfDApPD0oZXwwKXwoZnwwKTw9MCkpe289MC1lPDwyO2k9ZTt3aGlsZSgxKXtlOntpZigoZnwwKTw9MCl7YnJlYWsgZX1sPWk8PDI7bj1sK2N8MDtwPW4rb3wwO2o9RFthKzMyPj4yXTtoPTA7d2hpbGUoMSl7Zj1oPDwyO2c9RFtmK3A+PjJdO2s9RFthKzE2Pj4yXTtmOntpZigoZ3wwKT4oa3wwKSl7RFtmK2o+PjJdPWs7YnJlYWsgZn1mPWYranwwO2s9RFthKzEyPj4yXTtpZigoa3wwKT4oZ3wwKSl7RFtmPj4yXT1rO2JyZWFrIGZ9RFtmPj4yXT1nfWY9RFthKzg+PjJdO2g9aCsxfDA7aWYoKGZ8MCk+KGh8MCkpe2NvbnRpbnVlfWJyZWFrfWg9MDtpZigoZnwwKTw9MCl7YnJlYWsgZX1sPWIrbHwwO3doaWxlKDEpe2c9aDw8MjtmPWcrbnwwO2c9RFtnK2w+PjJdK0RbZytqPj4yXXwwO0RbZj4+Ml09ZztnOntpZigoZ3wwKT5EW2ErMTY+PjJdKXtnPWctRFthKzIwPj4yXXwwfWVsc2V7aWYoKGd8MCk+PURbYSsxMj4+Ml0pe2JyZWFrIGd9Zz1nK0RbYSsyMD4+Ml18MH1EW2Y+PjJdPWd9Zj1EW2ErOD4+Ml07aD1oKzF8MDtpZigoZnwwKT4oaHwwKSl7Y29udGludWV9YnJlYWt9fWk9ZStpfDA7aWYoKGl8MCk8KGR8MCkpe2NvbnRpbnVlfWJyZWFrfX1tYShtKTtyZXR1cm4gMX1mdW5jdGlvbiBtZChhKXthPWF8MDt2YXIgYj0wLGM9MCxkPTAsZT0wLGY9MDtEW2E+PjJdPTgzNjQ7ZT1hKzIzMnwwO2I9RFtlKzE5Nj4+Ml07aWYoYil7RFtlKzIwMD4+Ml09YjttYShiKX1jPURbZSsxODQ+PjJdO2lmKGMpe2I9RFtlKzE4OD4+Ml07aWYoKGN8MCk9PShifDApKXtiPWN9ZWxzZXt3aGlsZSgxKXtkPWItMTJ8MDtmPURbZD4+Ml07aWYoZil7RFtiLTg+PjJdPWY7bWEoZil9Yj1kO2lmKChjfDApIT0oYnwwKSl7Y29udGludWV9YnJlYWt9Yj1EW2UrMTg0Pj4yXX1EW2UrMTg4Pj4yXT1jO21hKGIpfWI9RFtlKzE1Nj4+Ml07aWYoYil7RFtlKzE2MD4+Ml09YjttYShiKX1jPURbZSsxMzY+PjJdO0RbZSsxMzY+PjJdPTA7aWYoYyl7ZD1jLTR8MDtiPURbZD4+Ml07aWYoYil7Yj1jKyhiPDw0KXwwO3doaWxlKDEpe2I9Yi0xNnwwO2lmKChjfDApIT0oYnwwKSl7Y29udGludWV9YnJlYWt9fW1hKGQpfWM9RFthKzIxNj4+Ml07aWYoYyl7Yj1EW2ErMjIwPj4yXTtpZigoY3wwKT09KGJ8MCkpe2I9Y31lbHNle3doaWxlKDEpe2Q9RFtiLTEyPj4yXTtpZihkKXtEW2ItOD4+Ml09ZDttYShkKX1kPURbYi0yOD4+Ml07aWYoZCl7RFtiLTI0Pj4yXT1kO21hKGQpfWQ9RFtiLTQwPj4yXTtpZihkKXtEW2ItMzY+PjJdPWQ7bWEoZCl9b2IoYi0xNDB8MCk7Yj1iLTE0NHwwO2lmKChjfDApIT0oYnwwKSl7Y29udGludWV9YnJlYWt9Yj1EW2ErMjE2Pj4yXX1EW2ErMjIwPj4yXT1jO21hKGIpfWI9RFthKzE5Nj4+Ml07aWYoYil7RFthKzIwMD4+Ml09YjttYShiKX1iPURbYSsxODQ+PjJdO2lmKGIpe0RbYSsxODg+PjJdPWI7bWEoYil9Yj1EW2ErMTcyPj4yXTtpZihiKXtEW2ErMTc2Pj4yXT1iO21hKGIpfWI9RFthKzE2MD4+Ml07aWYoYil7RFthKzE2ND4+Ml09YjttYShiKX1iPURbYSsxNDQ+PjJdO2lmKGIpe3doaWxlKDEpe2M9RFtiPj4yXTttYShiKTtiPWM7aWYoYil7Y29udGludWV9YnJlYWt9fWI9RFthKzEzNj4+Ml07RFthKzEzNj4+Ml09MDtpZihiKXttYShiKX1iPURbYSsxMjA+PjJdO2lmKGIpe21hKGIpfWI9RFthKzEwOD4+Ml07aWYoYil7bWEoYil9Yj1EW2ErOTY+PjJdO2lmKGIpe21hKGIpfWI9RFthKzcyPj4yXTtpZihiKXtEW2ErNzY+PjJdPWI7bWEoYil9Yj1EW2ErNjA+PjJdO2lmKGIpe21hKGIpfWI9RFthKzQ4Pj4yXTtpZihiKXtEW2ErNTI+PjJdPWI7bWEoYil9Yj1EW2ErMzY+PjJdO2lmKGIpe0RbYSs0MD4+Ml09YjttYShiKX1iPURbYSsyND4+Ml07aWYoYil7RFthKzI4Pj4yXT1iO21hKGIpfWI9RFthKzEyPj4yXTtpZihiKXtEW2ErMTY+PjJdPWI7bWEoYil9Yj1EW2ErOD4+Ml07RFthKzg+PjJdPTA7aWYoYil7YWIoYil9cmV0dXJuIGF8MH1mdW5jdGlvbiBsYyhhLGIpe2E9YXwwO2I9YnwwO3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTA7aj0kLTE2fDA7JD1qO2M9RFtiKzIwPj4yXTtkPURbYisxNj4+Ml07Zz1kKzR8MDtjPWc+Pj4wPDQ/YysxfDA6YztlPURbYisxMj4+Ml07YTp7aWYoKGV8MCk8PShjfDApJmc+Pj4wPkdbYis4Pj4yXXwoY3wwKT4oZXwwKSl7YnJlYWsgYX1kPWQrRFtiPj4yXXwwO2Q9RVtkfDBdfEVbZCsxfDBdPDw4fChFW2QrMnwwXTw8MTZ8RVtkKzN8MF08PDI0KTtEW2IrMTY+PjJdPWc7RFtiKzIwPj4yXT1jO2lmKChkfDApPDApe2JyZWFrIGF9aWIoYSs3NnwwLGQpO2M9ajtEW2M+PjJdPTA7RFtjKzQ+PjJdPTA7QltjKzV8MF09MDtCW2MrNnwwXT0wO0JbYys3fDBdPTA7QltjKzh8MF09MDtCW2MrOXwwXT0wO0JbYysxMHwwXT0wO0JbYysxMXwwXT0wO0JbYysxMnwwXT0wO2I6e2lmKCFLYShjLGIpKXticmVhayBifWlmKGQpe2U9MTt3aGlsZSgxKXtmPTE8PGg7aT1HYShjKTtnPURbYSs3Nj4+Ml0rKGg+Pj4zJjUzNjg3MDkwOCl8MDtlPWVeaTtpZihlJjEpe2Y9RFtnPj4yXSYoZl4tMSl9ZWxzZXtmPWZ8RFtnPj4yXX1lPWVeMTtEW2c+PjJdPWY7aD1oKzF8MDtpZigoZHwwKSE9KGh8MCkpe2NvbnRpbnVlfWJyZWFrfX1oPTA7ZD1EW2IrMTI+PjJdO2c9ZDtjPURbYisyMD4+Ml07ZT1jO2Y9RFtiKzE2Pj4yXTtpPWYrNHwwO2M9aT4+PjA8ND9jKzF8MDpjO2s9RFtiKzg+PjJdO2lmKGs+Pj4wPGk+Pj4wJihjfDApPj0oZHwwKXwoY3wwKT4oZHwwKSl7YnJlYWsgYn1sPURbYj4+Ml07ZD1sK2Z8MDtkPUVbZHwwXXxFW2QrMXwwXTw8OHwoRVtkKzJ8MF08PDE2fEVbZCszfDBdPDwyNCk7RFtiKzE2Pj4yXT1pO0RbYisyMD4+Ml09YztjPWU7ZT1mKzh8MDtjPWU+Pj4wPDg/YysxfDA6YztmPWU7ZT1jO2lmKGY+Pj4wPms+Pj4wJihjfDApPj0oZ3wwKXwoY3wwKT4oZ3wwKSl7YnJlYWsgYn1jPWkrbHwwO2M9RVtjfDBdfEVbYysxfDBdPDw4fChFW2MrMnwwXTw8MTZ8RVtjKzN8MF08PDI0KTtEW2IrMTY+PjJdPWY7RFtiKzIwPj4yXT1lO2lmKChjfDApPChkfDApKXticmVhayBifURbYSsxNj4+Ml09YztEW2ErMTI+PjJdPWQ7Yj0oYz4+MzEpLSgoZD4+MzEpKyhjPj4+MDxkPj4+MCl8MCl8MDtjPWMtZHwwO2lmKCFiJmM+Pj4wPjIxNDc0ODM2NDZ8Yil7YnJlYWsgYn1oPTE7Yj1jKzF8MDtEW2ErMjA+PjJdPWI7Yz1iPj4+MXwwO0RbYSsyND4+Ml09YztEW2ErMjg+PjJdPTAtYztpZihiJjEpe2JyZWFrIGJ9RFthKzI0Pj4yXT1jLTF9fSQ9aisxNnwwO3JldHVybiBofDB9ZnVuY3Rpb24gZmMoYSxiKXt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MDtmPS0xO2Q9LTE7YTp7aWYoKGJ8MCk9PS0xKXticmVhayBhfWQ9YisxfDA7Zj0oZD4+PjApJTN8MD9kOmItMnwwO2Q9Yi0xfDA7aWYoKGI+Pj4wKSUzfDApe2JyZWFrIGF9ZD1iKzJ8MH1iOntjOntkOntzd2l0Y2goRFthKzE2OD4+Ml0pe2Nhc2UgMDpjYXNlIDE6ZT1EW2ErMTQ4Pj4yXTtjPTE7Yj1EW2ErMTU2Pj4yXTtnPWIrKCgoZnwwKSE9LTE/RFtEW2U+PjJdKyhmPDwyKT4+Ml06LTEpPDwyKXwwO0RbZz4+Ml09RFtnPj4yXSsxO2I9KCgoZHwwKSE9LTE/RFtEW2U+PjJdKyhkPDwyKT4+Ml06LTEpPDwyKStifDA7YnJlYWsgYztjYXNlIDU6ZT1EW2ErMTQ4Pj4yXTtjPS0xO2M9KChifDApIT0tMT9EW0RbZT4+Ml0rKGI8PDIpPj4yXTpjKTw8MjtiPURbYSsxNTY+PjJdO2M9YytifDA7RFtjPj4yXT1EW2M+PjJdKzE7Yz0oKChmfDApIT0tMT9EW0RbZT4+Ml0rKGY8PDIpPj4yXTotMSk8PDIpK2J8MDtEW2M+PjJdPURbYz4+Ml0rMTtjPTI7Yj0oKChkfDApIT0tMT9EW0RbZT4+Ml0rKGQ8PDIpPj4yXTotMSk8PDIpK2J8MDticmVhayBjO2Nhc2UgMzplPURbYSsxNDg+PjJdO2M9LTE7Yz0oKGJ8MCkhPS0xP0RbRFtlPj4yXSsoYjw8Mik+PjJdOmMpPDwyO2I9RFthKzE1Nj4+Ml07Yz1jK2J8MDtEW2M+PjJdPURbYz4+Ml0rMTtjPSgoKGZ8MCkhPS0xP0RbRFtlPj4yXSsoZjw8Mik+PjJdOi0xKTw8MikrYnwwO0RbYz4+Ml09RFtjPj4yXSsyO2M9MTtiPSgoKGR8MCkhPS0xP0RbRFtlPj4yXSsoZDw8Mik+PjJdOi0xKTw8MikrYnwwO2JyZWFrIGM7Y2FzZSA3OmJyZWFrIGQ7ZGVmYXVsdDpicmVhayBifX1lPURbYSsxNDg+PjJdO2M9LTE7Yz0oKGJ8MCkhPS0xP0RbRFtlPj4yXSsoYjw8Mik+PjJdOmMpPDwyO2I9RFthKzE1Nj4+Ml07Yz1jK2J8MDtEW2M+PjJdPURbYz4+Ml0rMjtjPSgoKGZ8MCkhPS0xP0RbRFtlPj4yXSsoZjw8Mik+PjJdOi0xKTw8MikrYnwwO0RbYz4+Ml09RFtjPj4yXSsyO2M9MjtiPSgoKGR8MCkhPS0xP0RbRFtlPj4yXSsoZDw8Mik+PjJdOi0xKTw8MikrYnwwfURbYj4+Ml09RFtiPj4yXStjfWM9YTtkPURbYSsxODA+PjJdO2I9RFtEW2ErMTU2Pj4yXSsoKChmfDApIT0tMT9EW0RbRFthKzE0OD4+Ml0+PjJdKyhmPDwyKT4+Ml06LTEpPDwyKT4+Ml07YT1EW2ErMTc2Pj4yXTtEW2MrMTcyPj4yXT0oYXwwKT4oYnwwKT8wOigoYnwwKT4oZHwwKT9kOmIpLWF8MH1mdW5jdGlvbiBVYihhLGIsYyl7dmFyIGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTA7ZD1EW2ErOD4+Ml07ZT1EW2E+PjJdO2lmKGQtZT4+Mj4+PjA+PWI+Pj4wKXtnPURbYSs0Pj4yXTtoPWctZT4+MjtmPWI+Pj4wPmg+Pj4wP2g6YjthOntpZighZil7YnJlYWsgYX1rPWYtMXwwO2k9ZiY3O2I6e2lmKCFpKXtkPWU7YnJlYWsgYn1kPWU7d2hpbGUoMSl7RFtkPj4yXT1EW2M+PjJdO2Y9Zi0xfDA7ZD1kKzR8MDtqPWorMXwwO2lmKChqfDApIT0oaXwwKSl7Y29udGludWV9YnJlYWt9fWlmKGs+Pj4wPDcpe2JyZWFrIGF9d2hpbGUoMSl7RFtkPj4yXT1EW2M+PjJdO0RbZCs0Pj4yXT1EW2M+PjJdO0RbZCs4Pj4yXT1EW2M+PjJdO0RbZCsxMj4+Ml09RFtjPj4yXTtEW2QrMTY+PjJdPURbYz4+Ml07RFtkKzIwPj4yXT1EW2M+PjJdO0RbZCsyND4+Ml09RFtjPj4yXTtEW2QrMjg+PjJdPURbYz4+Ml07ZD1kKzMyfDA7Zj1mLTh8MDtpZihmKXtjb250aW51ZX1icmVha319aWYoYj4+PjA+aD4+PjApe2Q9YTthPWItaHwwO2lmKGEpe2E9KGE8PDIpK2d8MDt3aGlsZSgxKXtEW2c+PjJdPURbYz4+Ml07Zz1nKzR8MDtpZigoYXwwKSE9KGd8MCkpe2NvbnRpbnVlfWJyZWFrfX1lbHNle2E9Z31EW2QrND4+Ml09YTtyZXR1cm59RFthKzQ+PjJdPWUrKGI8PDIpO3JldHVybn1pZihlKXtEW2ErND4+Ml09ZTttYShlKTtEW2ErOD4+Ml09MDtEW2E+PjJdPTA7RFthKzQ+PjJdPTA7ZD0wfWM6e2lmKGI+Pj4wPj0xMDczNzQxODI0KXticmVhayBjfWU9ZD4+MTtkPWQ+PjI+Pj4wPDUzNjg3MDkxMT9iPj4+MD5lPj4+MD9iOmU6MTA3Mzc0MTgyMztpZihkPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgY31kPWQ8PDI7ZT1uYShkKTtEW2E+PjJdPWU7RFthKzg+PjJdPWQrZTtjPURbYz4+Ml07ZD1lO2I9Yjw8MjtnPWItNHwwO2g9KGc+Pj4yfDApKzEmNztpZihoKXt3aGlsZSgxKXtEW2Q+PjJdPWM7ZD1kKzR8MDtmPWYrMXwwO2lmKChofDApIT0oZnwwKSl7Y29udGludWV9YnJlYWt9fWI9YitlfDA7aWYoZz4+PjA+PTI4KXt3aGlsZSgxKXtEW2QrMjg+PjJdPWM7RFtkKzI0Pj4yXT1jO0RbZCsyMD4+Ml09YztEW2QrMTY+PjJdPWM7RFtkKzEyPj4yXT1jO0RbZCs4Pj4yXT1jO0RbZCs0Pj4yXT1jO0RbZD4+Ml09YztkPWQrMzJ8MDtpZigoYnwwKSE9KGR8MCkpe2NvbnRpbnVlfWJyZWFrfX1EW2ErND4+Ml09YjtyZXR1cm59cWEoKTtUKCl9ZnVuY3Rpb24gd2coYSl7YT1hfDA7dmFyIGI9MCxjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MDthOntiPURbYSszMj4+Ml07Zj1EW2IrOD4+Ml07Zz1EW2IrMTI+PjJdO2Q9RFtiKzIwPj4yXTtoPWY7Zj1EW2IrMTY+PjJdO2U9MDtiOntpZigoZ3wwKTw9KGR8MCkmaD4+PjA8PWY+Pj4wfChkfDApPihnfDApKXticmVhayBifWg9RVtmK0RbYj4+Ml18MF07Zz1mKzF8MDtkPWc/ZDpkKzF8MDtEW2IrMTY+PjJdPWc7RFtiKzIwPj4yXT1kO2M6e2lmKCFoKXticmVhayBjfXdoaWxlKDEpe2lmKGJhW0RbRFthPj4yXSsxNj4+Ml1dKGEsYyl8MCl7Yz1jKzF8MDtpZigoaHwwKSE9KGN8MCkpe2NvbnRpbnVlfWJyZWFrIGN9YnJlYWt9cmV0dXJuIDB9Yz1EW2ErOD4+Ml07ZD1EW2ErMTI+PjJdO2lmKChjfDApIT0oZHwwKSl7d2hpbGUoMSl7Yj1EW2M+PjJdO2lmKCEoYmFbRFtEW2I+PjJdKzg+PjJdXShiLGEsRFthKzQ+PjJdKXwwKSl7YnJlYWsgYX1jPWMrNHwwO2lmKChkfDApIT0oY3wwKSl7Y29udGludWV9YnJlYWt9fWQ6e2lmKCFoKXticmVhayBkfWM9MDt3aGlsZSgxKXtiPURbRFthKzg+PjJdKyhjPDwyKT4+Ml07aWYoIShiYVtEW0RbYj4+Ml0rMTI+PjJdXShiLERbYSszMj4+Ml0pfDApKXticmVhayBhfWM9YysxfDA7aWYoKGh8MCkhPShjfDApKXtjb250aW51ZX1icmVha31pZighaCl7YnJlYWsgZH1mPWErMjB8MDt3aGlsZSgxKXtjPTA7Zz1pPDwyO2I9RFtnK0RbYSs4Pj4yXT4+Ml07ZD1iYVtEW0RbYj4+Ml0rMjQ+PjJdXShiKXwwO2lmKChkfDApPjApe3doaWxlKDEpe2I9RFtEW2ErOD4+Ml0rZz4+Ml07az1iYVtEW0RbYj4+Ml0rMjA+PjJdXShiLGMpfDA7ZT1EW2ErMjA+PjJdO2o9RFthKzI0Pj4yXS1lPj4yO2U6e2lmKGs+Pj4wPGo+Pj4wKXticmVhayBlfWI9aysxfDA7aWYoYj4+PjA+aj4+PjApe3NhKGYsYi1qfDApO2U9RFtmPj4yXTticmVhayBlfWlmKGI+Pj4wPj1qPj4+MCl7YnJlYWsgZX1EW2ErMjQ+PjJdPShiPDwyKStlfURbKGs8PDIpK2U+PjJdPWk7Yz1jKzF8MDtpZigoZHwwKSE9KGN8MCkpe2NvbnRpbnVlfWJyZWFrfX1pPWkrMXwwO2lmKChofDApIT0oaXwwKSl7Y29udGludWV9YnJlYWt9fWU9MDtpZighKGJhW0RbRFthPj4yXSsyOD4+Ml1dKGEpfDApKXticmVhayBifWU9YmFbRFtEW2E+PjJdKzMyPj4yXV0oYSl8MH1yZXR1cm4gZXwwfXJldHVybiAwfWZ1bmN0aW9uIFNlKGEsYixjKXthPWF8MDtiPWJ8MDtjPWN8MDt2YXIgZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MDtKYyhhLGIsYyk7Yz1EW2ErODQ+PjJdO2Q9RFthKzg4Pj4yXS1jPj4yO2E6e2lmKChkfDApPihifDApKXticmVhayBhfWI9YisxfDA7aWYoYj4+PjA+ZD4+PjApe2I6e2Q9Yi1kfDA7ZT1EW2ErOTI+PjJdO2M9RFthKzg4Pj4yXTtpZihkPj4+MDw9ZS1jPj4yPj4+MCl7Yzp7aWYoIWQpe2JyZWFrIGN9Yj1jO2Y9KGQ8PDIpLTR8MDtnPShmPj4+MnwwKSsxJjc7aWYoZyl7d2hpbGUoMSl7RFtiPj4yXT0xO2I9Yis0fDA7aD1oKzF8MDtpZigoZ3wwKSE9KGh8MCkpe2NvbnRpbnVlfWJyZWFrfX1jPShkPDwyKStjfDA7aWYoZj4+PjA8Mjgpe2JyZWFrIGN9d2hpbGUoMSl7RFtiKzI0Pj4yXT0xO0RbYisyOD4+Ml09MTtEW2IrMTY+PjJdPTE7RFtiKzIwPj4yXT0xO0RbYis4Pj4yXT0xO0RbYisxMj4+Ml09MTtEW2I+PjJdPTE7RFtiKzQ+PjJdPTE7Yj1iKzMyfDA7aWYoKGN8MCkhPShifDApKXtjb250aW51ZX1icmVha319RFthKzg4Pj4yXT1jO2JyZWFrIGJ9ZDp7Zj1EW2ErODQ+PjJdO2o9Yy1mfDA7Yz1qPj4yO2I9YytkfDA7aWYoYj4+PjA8MTA3Mzc0MTgyNCl7ZT1lLWZ8MDtpPWU+PjE7ZT1lPj4yPj4+MDw1MzY4NzA5MTE/Yj4+PjA+aT4+PjA/YjppOjEwNzM3NDE4MjM7aWYoZSl7aWYoZT4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGR9Zz1uYShlPDwyKX1jPShjPDwyKStnfDA7Yj1jO2Q9ZDw8MjtpPWQtNHwwO2s9KGk+Pj4yfDApKzEmNztpZihrKXtiPWM7d2hpbGUoMSl7RFtiPj4yXT0xO2I9Yis0fDA7aD1oKzF8MDtpZigoa3wwKSE9KGh8MCkpe2NvbnRpbnVlfWJyZWFrfX1jPWMrZHwwO2lmKGk+Pj4wPj0yOCl7d2hpbGUoMSl7RFtiKzI0Pj4yXT0xO0RbYisyOD4+Ml09MTtEW2IrMTY+PjJdPTE7RFtiKzIwPj4yXT0xO0RbYis4Pj4yXT0xO0RbYisxMj4+Ml09MTtEW2I+PjJdPTE7RFtiKzQ+PjJdPTE7Yj1iKzMyfDA7aWYoKGN8MCkhPShifDApKXtjb250aW51ZX1icmVha319aWYoKGp8MCk+MCl7b2EoZyxmLGopfURbYSs5Mj4+Ml09KGU8PDIpK2c7RFthKzg4Pj4yXT1jO0RbYSs4ND4+Ml09ZztpZihmKXttYShmKX1icmVhayBifXFhKCk7VCgpfXJhKDEzMjYpO1QoKX1yZXR1cm59aWYoYj4+PjA+PWQ+Pj4wKXticmVhayBhfURbYSs4OD4+Ml09YysoYjw8Mil9fWZ1bmN0aW9uIHhhKGEsYixjKXt2YXIgZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTA7Zj1EW2ErOD4+Ml07ZT1EW2ErND4+Ml07aWYoZi1lPj4yPj4+MD49Yj4+PjApe2E6e2lmKCFiKXticmVhayBhfWQ9ZTtnPShiPDwyKS00fDA7aD0oZz4+PjJ8MCkrMSY3O2lmKGgpe3doaWxlKDEpe0RbZD4+Ml09RFtjPj4yXTtkPWQrNHwwO2k9aSsxfDA7aWYoKGh8MCkhPShpfDApKXtjb250aW51ZX1icmVha319ZT0oYjw8MikrZXwwO2lmKGc+Pj4wPDI4KXticmVhayBhfXdoaWxlKDEpe0RbZD4+Ml09RFtjPj4yXTtEW2QrND4+Ml09RFtjPj4yXTtEW2QrOD4+Ml09RFtjPj4yXTtEW2QrMTI+PjJdPURbYz4+Ml07RFtkKzE2Pj4yXT1EW2M+PjJdO0RbZCsyMD4+Ml09RFtjPj4yXTtEW2QrMjQ+PjJdPURbYz4+Ml07RFtkKzI4Pj4yXT1EW2M+PjJdO2Q9ZCszMnwwO2lmKChlfDApIT0oZHwwKSl7Y29udGludWV9YnJlYWt9fURbYSs0Pj4yXT1lO3JldHVybn1iOntnPURbYT4+Ml07az1lLWd8MDtlPWs+PjI7ZD1lK2J8MDtpZihkPj4+MDwxMDczNzQxODI0KXtmPWYtZ3wwO2o9Zj4+MTtmPWY+PjI+Pj4wPDUzNjg3MDkxMT9kPj4+MD5qPj4+MD9kOmo6MTA3Mzc0MTgyMztpZihmKXtpZihmPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgYn1oPW5hKGY8PDIpfWU9KGU8PDIpK2h8MDtkPWU7Yj1iPDwyO2o9Yi00fDA7bD0oaj4+PjJ8MCkrMSY3O2lmKGwpe2Q9ZTt3aGlsZSgxKXtEW2Q+PjJdPURbYz4+Ml07ZD1kKzR8MDtpPWkrMXwwO2lmKChsfDApIT0oaXwwKSl7Y29udGludWV9YnJlYWt9fWI9YitlfDA7aWYoaj4+PjA+PTI4KXt3aGlsZSgxKXtEW2Q+PjJdPURbYz4+Ml07RFtkKzQ+PjJdPURbYz4+Ml07RFtkKzg+PjJdPURbYz4+Ml07RFtkKzEyPj4yXT1EW2M+PjJdO0RbZCsxNj4+Ml09RFtjPj4yXTtEW2QrMjA+PjJdPURbYz4+Ml07RFtkKzI0Pj4yXT1EW2M+PjJdO0RbZCsyOD4+Ml09RFtjPj4yXTtkPWQrMzJ8MDtpZigoYnwwKSE9KGR8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZigoa3wwKT4wKXtvYShoLGcsayl9RFthKzg+PjJdPShmPDwyKStoO0RbYSs0Pj4yXT1iO0RbYT4+Ml09aDtpZihnKXttYShnKX1yZXR1cm59cWEoKTtUKCl9cmEoMTMyNik7VCgpfWZ1bmN0aW9uIG9hKGEsYixjKXt2YXIgZD0wLGU9MCxmPTA7aWYoYz4+PjA+PTUxMil7WChhfDAsYnwwLGN8MCl8MDtyZXR1cm4gYX1lPWErY3wwO2E6e2lmKCEoKGFeYikmMykpe2I6e2lmKCEoYSYzKSl7Yz1hO2JyZWFrIGJ9aWYoIWMpe2M9YTticmVhayBifWM9YTt3aGlsZSgxKXtCW2N8MF09RVtifDBdO2I9YisxfDA7Yz1jKzF8MDtpZighKGMmMykpe2JyZWFrIGJ9aWYoYz4+PjA8ZT4+PjApe2NvbnRpbnVlfWJyZWFrfX1kPWUmLTQ7Yzp7aWYoZD4+PjA8NjQpe2JyZWFrIGN9Zj1kKy02NHwwO2lmKGY+Pj4wPGM+Pj4wKXticmVhayBjfXdoaWxlKDEpe0RbYz4+Ml09RFtiPj4yXTtEW2MrND4+Ml09RFtiKzQ+PjJdO0RbYys4Pj4yXT1EW2IrOD4+Ml07RFtjKzEyPj4yXT1EW2IrMTI+PjJdO0RbYysxNj4+Ml09RFtiKzE2Pj4yXTtEW2MrMjA+PjJdPURbYisyMD4+Ml07RFtjKzI0Pj4yXT1EW2IrMjQ+PjJdO0RbYysyOD4+Ml09RFtiKzI4Pj4yXTtEW2MrMzI+PjJdPURbYiszMj4+Ml07RFtjKzM2Pj4yXT1EW2IrMzY+PjJdO0RbYys0MD4+Ml09RFtiKzQwPj4yXTtEW2MrNDQ+PjJdPURbYis0ND4+Ml07RFtjKzQ4Pj4yXT1EW2IrNDg+PjJdO0RbYys1Mj4+Ml09RFtiKzUyPj4yXTtEW2MrNTY+PjJdPURbYis1Nj4+Ml07RFtjKzYwPj4yXT1EW2IrNjA+PjJdO2I9Yi0gLTY0fDA7Yz1jLSAtNjR8MDtpZihmPj4+MD49Yz4+PjApe2NvbnRpbnVlfWJyZWFrfX1pZihjPj4+MD49ZD4+PjApe2JyZWFrIGF9d2hpbGUoMSl7RFtjPj4yXT1EW2I+PjJdO2I9Yis0fDA7Yz1jKzR8MDtpZihkPj4+MD5jPj4+MCl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1pZihlPj4+MDw0KXtjPWE7YnJlYWsgYX1kPWUtNHwwO2lmKGQ+Pj4wPGE+Pj4wKXtjPWE7YnJlYWsgYX1jPWE7d2hpbGUoMSl7QltjfDBdPUVbYnwwXTtCW2MrMXwwXT1FW2IrMXwwXTtCW2MrMnwwXT1FW2IrMnwwXTtCW2MrM3wwXT1FW2IrM3wwXTtiPWIrNHwwO2M9Yys0fDA7aWYoZD4+PjA+PWM+Pj4wKXtjb250aW51ZX1icmVha319aWYoYz4+PjA8ZT4+PjApe3doaWxlKDEpe0JbY3wwXT1FW2J8MF07Yj1iKzF8MDtjPWMrMXwwO2lmKChlfDApIT0oY3wwKSl7Y29udGludWV9YnJlYWt9fXJldHVybiBhfWZ1bmN0aW9uIG5kKGEpe2E9YXwwO3ZhciBiPTAsYz0wLGQ9MCxlPTA7RFthPj4yXT04MzEyO2U9RFthKzM2OD4+Ml07RFthKzM2OD4+Ml09MDtpZihlKXtkPWUtNHwwO2I9RFtkPj4yXTtpZihiKXtjPShiPDw0KStlfDA7d2hpbGUoMSl7Yz1jLTE2fDA7aWYoKGV8MCkhPShjfDApKXtjb250aW51ZX1icmVha319bWEoZCl9ZD1EW2ErMjE2Pj4yXTtpZihkKXtjPURbYSsyMjA+PjJdO2lmKChkfDApPT0oY3wwKSl7Yj1kfWVsc2V7d2hpbGUoMSl7Yj1EW2MtMTI+PjJdO2lmKGIpe0RbYy04Pj4yXT1iO21hKGIpfWI9RFtjLTI4Pj4yXTtpZihiKXtEW2MtMjQ+PjJdPWI7bWEoYil9Yj1EW2MtNDA+PjJdO2lmKGIpe0RbYy0zNj4+Ml09YjttYShiKX1vYihjLTE0MHwwKTtjPWMtMTQ0fDA7aWYoKGR8MCkhPShjfDApKXtjb250aW51ZX1icmVha31iPURbYSsyMTY+PjJdfURbYSsyMjA+PjJdPWQ7bWEoYil9Yj1EW2ErMTk2Pj4yXTtpZihiKXtEW2ErMjAwPj4yXT1iO21hKGIpfWI9RFthKzE4ND4+Ml07aWYoYil7RFthKzE4OD4+Ml09YjttYShiKX1iPURbYSsxNzI+PjJdO2lmKGIpe0RbYSsxNzY+PjJdPWI7bWEoYil9Yj1EW2ErMTYwPj4yXTtpZihiKXtEW2ErMTY0Pj4yXT1iO21hKGIpfWM9RFthKzE0ND4+Ml07aWYoYyl7d2hpbGUoMSl7Yj1EW2M+PjJdO21hKGMpO2M9YjtpZihiKXtjb250aW51ZX1icmVha319Yj1EW2ErMTM2Pj4yXTtEW2ErMTM2Pj4yXT0wO2lmKGIpe21hKGIpfWI9RFthKzEyMD4+Ml07aWYoYil7bWEoYil9Yj1EW2ErMTA4Pj4yXTtpZihiKXttYShiKX1iPURbYSs5Nj4+Ml07aWYoYil7bWEoYil9Yj1EW2ErNzI+PjJdO2lmKGIpe0RbYSs3Nj4+Ml09YjttYShiKX1iPURbYSs2MD4+Ml07aWYoYil7bWEoYil9Yj1EW2ErNDg+PjJdO2lmKGIpe0RbYSs1Mj4+Ml09YjttYShiKX1iPURbYSszNj4+Ml07aWYoYil7RFthKzQwPj4yXT1iO21hKGIpfWI9RFthKzI0Pj4yXTtpZihiKXtEW2ErMjg+PjJdPWI7bWEoYil9Yj1EW2ErMTI+PjJdO2lmKGIpe0RbYSsxNj4+Ml09YjttYShiKX1iPURbYSs4Pj4yXTtEW2ErOD4+Ml09MDtpZihiKXthYihiKX1yZXR1cm4gYXwwfWZ1bmN0aW9uIGJpKGEsYixjLGQpe3ZhciBlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MDtnPWM7YTp7Yjp7Yzp7ZDp7ZTp7Zjp7Zzp7aDp7aTp7ajp7azp7aWYoYil7aWYoIWcpe2JyZWFrIGt9aWYoIWQpe2JyZWFrIGp9Yz1NKGQpLU0oYil8MDtpZihjPj4+MDw9MzEpe2JyZWFrIGl9YnJlYWsgY31pZigoZHwwKT09MXxkPj4+MD4xKXticmVhayBjfWE9KGE+Pj4wKS8oZz4+PjApfDA7YWE9MDticmVhayBhfWlmKCFhKXticmVhayBofWlmKCFkKXticmVhayBnfWlmKGQtMSZkKXticmVhayBnfWE9Yj4+PmVpKGQpfDA7YWE9MDticmVhayBhfWlmKCEoZy0xJmcpKXticmVhayBmfWg9KE0oZykrMzN8MCktTShiKXwwO2U9MC1ofDA7YnJlYWsgZH1oPWMrMXwwO2U9NjMtY3wwO2JyZWFrIGR9YT0oYj4+PjApLyhkPj4+MCl8MDthYT0wO2JyZWFrIGF9Yz1NKGQpLU0oYil8MDtpZihjPj4+MDwzMSl7YnJlYWsgZX1icmVhayBjfWlmKChnfDApPT0xKXticmVhayBifWM9ZWkoZyk7ZD1jJjMxO2lmKChjJjYzKT4+PjA+PTMyKXtjPTA7YT1iPj4+ZHwwfWVsc2V7Yz1iPj4+ZHwwO2E9KCgxPDxkKS0xJmIpPDwzMi1kfGE+Pj5kfWFhPWM7YnJlYWsgYX1oPWMrMXwwO2U9NjMtY3wwfWM9aCY2MztmPWMmMzE7aWYoYz4+PjA+PTMyKXtjPTA7aj1iPj4+ZnwwfWVsc2V7Yz1iPj4+ZnwwO2o9KCgxPDxmKS0xJmIpPDwzMi1mfGE+Pj5mfWY9YztjPWUmNjM7ZT1jJjMxO2lmKGM+Pj4wPj0zMil7Yz1hPDxlO2E9MH1lbHNle2M9KDE8PGUpLTEmYT4+PjMyLWV8Yjw8ZTthPWE8PGV9Yj1jO2lmKGgpe2M9ZC0xfDA7ZT1nLTF8MDtsPShlfDApIT0tMT9jKzF8MDpjO3doaWxlKDEpe2M9ajw8MXxiPj4+MzE7Zj1mPDwxfGo+Pj4zMTtpPWwtKGYrKGM+Pj4wPmU+Pj4wKXwwKT4+MzE7az1nJmk7aj1jLWt8MDtmPWYtKChkJmkpKyhjPj4+MDxrPj4+MCl8MCl8MDtiPWI8PDF8YT4+PjMxO2E9bXxhPDwxO2k9aSYxO209aTtoPWgtMXwwO2lmKGgpe2NvbnRpbnVlfWJyZWFrfX1hYT1iPDwxfGE+Pj4zMTthPWl8YTw8MTticmVhayBhfWE9MDtiPTB9YWE9Yn1yZXR1cm4gYX1mdW5jdGlvbiBwYihhLGIpe3ZhciBjPTAsZD0wLGU9MDtjPShhfDApPT0oYnwwKTtCW2IrMTJ8MF09YzthOntpZihjKXticmVhayBhfXdoaWxlKDEpe2Q9RFtiKzg+PjJdO2lmKEVbZCsxMnwwXSl7YnJlYWsgYX1iOntjPURbZCs4Pj4yXTtlPURbYz4+Ml07Yzp7aWYoKGR8MCk9PShlfDApKXtlPURbYys0Pj4yXTtpZighKCFlfEVbZSsxMnwwXSkpe2JyZWFrIGJ9ZDp7aWYoRFtkPj4yXT09KGJ8MCkpe2I9ZDticmVhayBkfWI9RFtkKzQ+PjJdO2E9RFtiPj4yXTtEW2QrND4+Ml09YTtpZihhKXtEW2ErOD4+Ml09ZDtjPURbZCs4Pj4yXX1EW2IrOD4+Ml09YzthPURbZCs4Pj4yXTtEWygoKGR8MCkhPURbYT4+Ml0pPDwyKSthPj4yXT1iO0RbYj4+Ml09ZDtEW2QrOD4+Ml09YjtjPURbYis4Pj4yXX1CW2IrMTJ8MF09MTtCW2MrMTJ8MF09MDthPURbYz4+Ml07Yj1EW2ErND4+Ml07RFtjPj4yXT1iO2lmKGIpe0RbYis4Pj4yXT1jfURbYSs4Pj4yXT1EW2MrOD4+Ml07Yj1EW2MrOD4+Ml07RFsoKERbYj4+Ml0hPShjfDApKTw8MikrYj4+Ml09YTtEW2ErND4+Ml09YztiPWMrOHwwO2JyZWFrIGN9aWYoIShFW2UrMTJ8MF18IWUpKXticmVhayBifWU6e2lmKERbZD4+Ml0hPShifDApKXtiPWQ7YnJlYWsgZX1hPURbYis0Pj4yXTtEW2Q+PjJdPWE7aWYoYSl7RFthKzg+PjJdPWQ7Yz1EW2QrOD4+Ml19RFtiKzg+PjJdPWM7YT1EW2QrOD4+Ml07RFsoKChkfDApIT1EW2E+PjJdKTw8MikrYT4+Ml09YjtEW2IrND4+Ml09ZDtEW2QrOD4+Ml09YjtjPURbYis4Pj4yXX1CW2IrMTJ8MF09MTtCW2MrMTJ8MF09MDthPURbYys0Pj4yXTtiPURbYT4+Ml07RFtjKzQ+PjJdPWI7aWYoYil7RFtiKzg+PjJdPWN9RFthKzg+PjJdPURbYys4Pj4yXTtiPURbYys4Pj4yXTtEWygoRFtiPj4yXSE9KGN8MCkpPDwyKStiPj4yXT1hO0RbYT4+Ml09YztiPWMrOHwwfURbYj4+Ml09YTticmVhayBhfUJbZCsxMnwwXT0xO2Q9KGF8MCk9PShjfDApO0JbYysxMnwwXT1kO0JbZSsxMnwwXT0xO2I9YztpZighZCl7Y29udGludWV9YnJlYWt9fX1mdW5jdGlvbiB5ZChhLGIsYyl7YT1hfDA7Yj1ifDA7Yz1jfDA7dmFyIGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT1LKDApLGo9MCxrPTAsbD0wLG09MCxuPTAsbz0wLHA9MCxxPTAscj0wLHM9MCx0PTAsdT0wO2s9JC0xNnwwOyQ9aztpZihEW2MrMjg+PjJdPT05KXtkPURbYSs0Pj4yXTtnPUJbYysyNHwwXTtlPWc8PDI7Zj1uYSgoZyYxMDczNzQxODIzKSE9KGd8MCk/LTE6ZSk7bD1rKzh8MDtEW2w+PjJdPTEwNjUzNTMyMTY7aT1IW2ErMjA+PjJdO2Q9LTE8PGReLTE7aWYoKGR8MCk+MCl7SFtsPj4yXT1pL0soZHwwKX1vPShkfDApPjA7YTp7aWYoIW8pe2JyZWFrIGF9aj1EW2MrODA+PjJdO2lmKCFqKXticmVhayBhfWQ9MDtpZigoZ3wwKTw9MCl7aWYoKGp8MCkhPTEpe2E9aiYtMjtiPTA7d2hpbGUoMSl7b2EoRFtEW2MrNjQ+PjJdPj4yXStkfDAsZixlKTtkPWQrZXwwO29hKGQrRFtEW2MrNjQ+PjJdPj4yXXwwLGYsZSk7ZD1kK2V8MDtiPWIrMnwwO2lmKChhfDApIT0oYnwwKSl7Y29udGludWV9YnJlYWt9fWlmKCEoaiYxKSl7YnJlYWsgYX1vYShEW0RbYys2ND4+Ml0+PjJdK2R8MCxmLGUpO2JyZWFrIGF9cD1EW0RbYj4+Ml0+PjJdK0RbYis0OD4+Ml18MDt0PWcmLTI7dT1nJjE7d2hpbGUoMSl7bT1EW2ErOD4+Ml07aT1IW2w+PjJdO2I9MDtuPTA7aWYoKGd8MCkhPTEpe3doaWxlKDEpe2g9Yjw8MjtxPShkPDwyKStwfDA7SFtoK2Y+PjJdPUsoaSpLKERbcT4+Ml0pKStIW2grbT4+Ml07aD1ofDQ7SFtoK2Y+PjJdPUsoaSpLKERbcSs0Pj4yXSkpK0hbaCttPj4yXTtiPWIrMnwwO2Q9ZCsyfDA7bj1uKzJ8MDtpZigodHwwKSE9KG58MCkpe2NvbnRpbnVlfWJyZWFrfX1pZih1KXtiPWI8PDI7SFtiK2Y+PjJdPUsoaSpLKERbKGQ8PDIpK3A+PjJdKSkrSFtiK20+PjJdO2Q9ZCsxfDB9b2EoRFtEW2MrNjQ+PjJdPj4yXStyfDAsZixlKTtyPWUrcnwwO3M9cysxfDA7aWYoKHN8MCkhPShqfDApKXtjb250aW51ZX1icmVha319bWEoZil9JD1rKzE2fDA7cmV0dXJuIG98MH1mdW5jdGlvbiB0ZShhLGIpe2E9YXwwO2I9YnwwO3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MDtpZihaYyhhLGIpKXtpPWErMzZ8MDtnPWJhW0RbRFthPj4yXSsyND4+Ml1dKGEpfDA7ZD1EW2ErNDA+PjJdO2U9RFthKzM2Pj4yXTtjPWQtZT4+MjthOntpZihnPj4+MD5jPj4+MCl7T2IoaSxnLWN8MCk7YnJlYWsgYX1pZihjPj4+MDw9Zz4+PjApe2JyZWFrIGF9ZT1lKyhnPDwyKXwwO2lmKChlfDApIT0oZHwwKSl7d2hpbGUoMSl7ZD1kLTR8MDtjPURbZD4+Ml07RFtkPj4yXT0wO2lmKGMpe2JhW0RbRFtjPj4yXSs0Pj4yXV0oYyl9aWYoKGR8MCkhPShlfDApKXtjb250aW51ZX1icmVha319RFthKzQwPj4yXT1lfWI6e2lmKChnfDApPD0wKXtlPTA7YnJlYWsgYn1lPTE7Yz1EW2IrMjA+PjJdO2Q9RFtiKzEyPj4yXTtmPURbYisxNj4+Ml07aWYoKGN8MCk+PShkfDApJmY+Pj4wPj1HW2IrOD4+Ml18KGN8MCk+KGR8MCkpe2JyZWFrIGJ9ZD0wO3doaWxlKDEpe2g9RVtmK0RbYj4+Ml18MF07Zj1mKzF8MDtjPWY/YzpjKzF8MDtEW2IrMTY+PjJdPWY7RFtiKzIwPj4yXT1jO2Y9YmFbRFtEW2E+PjJdKzQ4Pj4yXV0oYSxoKXwwO2g9ZDw8MjtqPWgrRFthKzM2Pj4yXXwwO2M9RFtqPj4yXTtEW2o+PjJdPWY7aWYoYyl7YmFbRFtEW2M+PjJdKzQ+PjJdXShjKX1jPURbRFtpPj4yXStoPj4yXTtpZighYyl7YnJlYWsgYn1pZighKGw9YyxtPWJhW0RbRFthPj4yXSsyOD4+Ml1dKGEpfDAsbj1iYVtEW0RbYT4+Ml0rMjA+PjJdXShhLGQpfDAsaz1EW0RbYz4+Ml0rOD4+Ml0sYmFba10obHwwLG18MCxufDApfDApKXticmVhayBifWQ9ZCsxfDA7ZT0oZ3wwKT4oZHwwKTtpZigoZHwwKT09KGd8MCkpe2JyZWFrIGJ9Zj1EW2IrMTY+PjJdO2M9RFtiKzIwPj4yXTtoPURbYisxMj4+Ml07aWYoZj4+PjA8R1tiKzg+PjJdJihjfDApPD0oaHwwKXwoY3wwKTwoaHwwKSl7Y29udGludWV9YnJlYWt9fWE9IWV9ZWxzZXthPTB9cmV0dXJuIGF8MH1mdW5jdGlvbiB4aChhLGIpe2E9YXwwO2I9YnwwO3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTA7Yz1EW2IrODg+PjJdO2lmKCEoIWN8RFtjPj4yXSE9MSkpe2Q9RFtjKzg+PjJdO0RbYSs0Pj4yXT1FW2R8MF18RVtkKzF8MF08PDh8KEVbZCsyfDBdPDwxNnxFW2QrM3wwXTw8MjQpO2Q9QltiKzI0fDBdO2U9RFthKzg+PjJdO2Y9RFthKzEyPj4yXS1lPj4yO2E6e2lmKGQ+Pj4wPmY+Pj4wKXtzYShhKzh8MCxkLWZ8MCk7ZD1CW2IrMjR8MF07ZT1EW2ErOD4+Ml07YnJlYWsgYX1pZihkPj4+MD49Zj4+PjApe2JyZWFrIGF9RFthKzEyPj4yXT0oZDw8MikrZX1rPTE7Zj1EW2MrOD4+Ml07Yjp7aWYoKGR8MCk8PTApe2I9NDticmVhayBifWg9ZCYzO2M6e2lmKGQtMT4+PjA8Myl7Yj00O2Q9MDticmVhayBjfWw9ZCYtNDtkPTA7Yj00O3doaWxlKDEpe2c9ZDw8MjtjPWIrZnwwO0RbZytlPj4yXT1FW2N8MF18RVtjKzF8MF08PDh8KEVbYysyfDBdPDwxNnxFW2MrM3wwXTw8MjQpO0RbKGd8NCkrZT4+Ml09RVtjKzR8MF18RVtjKzV8MF08PDh8KEVbYys2fDBdPDwxNnxFW2MrN3wwXTw8MjQpO0RbKGd8OCkrZT4+Ml09RVtjKzh8MF18RVtjKzl8MF08PDh8KEVbYysxMHwwXTw8MTZ8RVtjKzExfDBdPDwyNCk7RFsoZ3wxMikrZT4+Ml09RVtjKzEyfDBdfEVbYysxM3wwXTw8OHwoRVtjKzE0fDBdPDwxNnxFW2MrMTV8MF08PDI0KTtkPWQrNHwwO2I9YisxNnwwO2k9aSs0fDA7aWYoKGx8MCkhPShpfDApKXtjb250aW51ZX1icmVha319aWYoIWgpe2JyZWFrIGJ9d2hpbGUoMSl7Yz1iK2Z8MDtEWyhkPDwyKStlPj4yXT1FW2N8MF18RVtjKzF8MF08PDh8KEVbYysyfDBdPDwxNnxFW2MrM3wwXTw8MjQpO2Q9ZCsxfDA7Yj1iKzR8MDtqPWorMXwwO2lmKChqfDApIT0oaHwwKSl7Y29udGludWV9YnJlYWt9fWM9YTthPWIrZnwwO0RbYysyMD4+Ml09RVthfDBdfEVbYSsxfDBdPDw4fChFW2ErMnwwXTw8MTZ8RVthKzN8MF08PDI0KX1yZXR1cm4ga3wwfWZ1bmN0aW9uIElhKGEsYixjKXt2YXIgZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wO2Q9KGM+Pj4wKS8zfDA7Zz1EWyhEW0RbYSs4Pj4yXSs5Nj4+Ml0rSihkLDEyKXwwKSsoYy1KKGQsMyk8PDIpPj4yXTtlPURbRFthKzEyPj4yXSs0Pj4yXTtkPURbZSs0Pj4yXTthOntpZigoZHwwKSE9RFtlKzg+PjJdKXtEW2Q+PjJdPWc7RFtlKzQ+PjJdPWQrNDticmVhayBhfWI6e2g9RFtlPj4yXTtpPWQtaHwwO2o9aT4+MjtkPWorMXwwO2lmKGQ+Pj4wPDEwNzM3NDE4MjQpe2Y9aT4+MTtmPWo+Pj4wPDUzNjg3MDkxMT9kPj4+MD5mPj4+MD9kOmY6MTA3Mzc0MTgyMztpZihmKXtpZihmPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgYn1kPW5hKGY8PDIpfWVsc2V7ZD0wfWo9ZCsoajw8Mil8MDtEW2o+PjJdPWc7aWYoKGl8MCk+MCl7b2EoZCxoLGkpfURbZSs4Pj4yXT1kKyhmPDwyKTtEW2UrND4+Ml09ais0O0RbZT4+Ml09ZDtpZihoKXttYShoKX1icmVhayBhfXFhKCk7VCgpfXJhKDEzMjYpO1QoKX1lPURbYSs0Pj4yXTtkPURbZSs0Pj4yXTtjOntkOntlOntpZigoZHwwKSE9RFtlKzg+PjJdKXtEW2Q+PjJdPWM7RFtlKzQ+PjJdPWQrNDticmVhayBlfWg9RFtlPj4yXTtpPWQtaHwwO2c9aT4+MjtkPWcrMXwwO2lmKGQ+Pj4wPj0xMDczNzQxODI0KXticmVhayBkfWY9aT4+MTtmPWc+Pj4wPDUzNjg3MDkxMT9kPj4+MD5mPj4+MD9kOmY6MTA3Mzc0MTgyMztpZihmKXtpZihmPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgY31kPW5hKGY8PDIpfWVsc2V7ZD0wfWc9ZCsoZzw8Mil8MDtEW2c+PjJdPWM7aWYoKGl8MCk+MCl7b2EoZCxoLGkpfURbZSs4Pj4yXT1kKyhmPDwyKTtEW2UrND4+Ml09Zys0O0RbZT4+Ml09ZDtpZighaCl7YnJlYWsgZX1tYShoKX1hPURbYSs0Pj4yXTtEW0RbYSsxMj4+Ml0rKGI8PDIpPj4yXT1EW2ErMjQ+PjJdO0RbYSsyND4+Ml09RFthKzI0Pj4yXSsxO3JldHVybn1xYSgpO1QoKX1yYSgxMzI2KTtUKCl9ZnVuY3Rpb24gUmYoYSxiLGMsZCl7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7dmFyIGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTA7aD1EW2QrODA+PjJdO2U9JC00OHwwOyQ9ZTthPURbYSs0Pj4yXTthOntpZihhLTMxPj4+MDw0Mjk0OTY3MjY3KXticmVhayBhfWk9RFtEW2Q+PjJdPj4yXStEW2QrNDg+PjJdfDA7RFtlKzE2Pj4yXT1hO2E9LTE8PGE7RFtlKzIwPj4yXT1hXi0xO2E9LTItYXwwO0RbZSsyND4+Ml09YTtEW2UrMzI+PjJdPShhfDApLzI7SFtlKzI4Pj4yXT1LKDIpL0soYXwwKTtnPURbYz4+Ml07aWYoKGd8MCkhPURbYys0Pj4yXSl7YT0wO2Q9MDt3aGlsZSgxKXtmPURbKGQ8PDIpK2c+PjJdO2g9ZSszNnwwO2o9RFtEW2I+PjJdPj4yXTtsPURbYis0OD4+Ml07Zz1EW2IrNDQ+PjJdO2s9RFtiKzQwPj4yXTtpZighRVtiKzg0fDBdKXtmPURbRFtiKzY4Pj4yXSsoZjw8Mik+PjJdfWY9JGgoayxnLGYsMCkrbHwwO29hKGgsZitqfDAsayk7RGMoZSsxNnwwLGgsZSsxMnwwLGUrOHwwKTtnPWE8PDI7RFtnK2k+PjJdPURbZSsxMj4+Ml07RFsoZ3w0KStpPj4yXT1EW2UrOD4+Ml07Zj0xO2E9YSsyfDA7ZD1kKzF8MDtnPURbYz4+Ml07aWYoZD4+PjA8RFtjKzQ+PjJdLWc+PjI+Pj4wKXtjb250aW51ZX1icmVha31icmVhayBhfWlmKCFoKXtmPTE7YnJlYWsgYX1kPTA7YT0wO3doaWxlKDEpe2o9ZSszNnwwO2M9RFtEW2I+PjJdPj4yXTtmPURbYis0MD4+Ml07Zz1EW2IrNDg+PjJdKyRoKGYsRFtiKzQ0Pj4yXSxFW2IrODR8MF0/YTpEW0RbYis2OD4+Ml0rKGE8PDIpPj4yXSwwKXwwO29hKGosYytnfDAsZik7RGMoZSsxNnwwLGosZSsxMnwwLGUrOHwwKTtjPWQ8PDI7RFtjK2k+PjJdPURbZSsxMj4+Ml07RFsoY3w0KStpPj4yXT1EW2UrOD4+Ml07ZD1kKzJ8MDtmPTE7YT1hKzF8MDtpZigoaHwwKSE9KGF8MCkpe2NvbnRpbnVlfWJyZWFrfX0kPWUrNDh8MDtyZXR1cm4gZnwwfWZ1bmN0aW9uIGliKGEsYil7dmFyIGM9MCxkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MDtkPSQtMTZ8MDskPWQ7YTp7ZT1EW2ErND4+Ml07Yjp7aWYoZT4+PjA8Yj4+PjApe2Y9Yi1lfDA7aD1EW2ErOD4+Ml07Yz1oPDw1O2M6e2lmKCEoZj4+PjA+Yz4+PjB8ZT4+PjA+Yy1mPj4+MCkpe0RbYSs0Pj4yXT1iO2c9ZSYzMTtiPURbYT4+Ml0rKGU+Pj4zJjUzNjg3MDkwOCl8MDticmVhayBjfURbZCs4Pj4yXT0wO0RbZD4+Ml09MDtEW2QrND4+Ml09MDtpZigoYnwwKTwwKXticmVhayBhfWlmKGM+Pj4wPD0xMDczNzQxODIyKXtjPWIrMzEmLTMyO2I9aDw8NjtiPWI+Pj4wPGM+Pj4wP2M6Yn1lbHNle2I9MjE0NzQ4MzY0N31UYShkLGIpO2U9RFthKzQ+PjJdO0RbZCs0Pj4yXT1lK2Y7aT1EW2E+PjJdO2I9RFtkPj4yXTtkOntpZigoZXwwKTw9MCl7YnJlYWsgZH1jPWU+Pj41fDA7aD1jPDwyO2I9TmEoYixpLGgpK2h8MDtnPWUtKGM8PDUpfDA7ZTp7aWYoKGd8MCk8PTApe2c9MDticmVhayBlfWM9LTE+Pj4zMi1nfDA7RFtiPj4yXT1EW2I+PjJdJihjXi0xKXxjJkRbaStoPj4yXX1pPURbYT4+Ml19RFthPj4yXT1EW2Q+PjJdO0RbZD4+Ml09aTtjPURbYSs0Pj4yXTtEW2ErND4+Ml09RFtkKzQ+PjJdO0RbZCs0Pj4yXT1jO2M9RFthKzg+PjJdO0RbYSs4Pj4yXT1EW2QrOD4+Ml07RFtkKzg+PjJdPWM7aWYoIWkpe2JyZWFrIGN9bWEoaSl9aWYoIWYpe2JyZWFrIGJ9aWYoZyl7Yz0zMi1nfDA7YT1jPj4+MD5mPj4+MD9mOmM7RFtiPj4yXT1EW2I+PjJdJigtMTw8ZyYtMT4+PmMtYV4tMSk7Zj1mLWF8MDtiPWIrNHwwfWM9Zj4+PjU8PDI7YT1wYShiLDAsYyk7Yj1mJjMxO2lmKCFiKXticmVhayBifWE9YStjfDA7RFthPj4yXT1EW2E+PjJdJigtMT4+PjMyLWJeLTEpO2JyZWFrIGJ9RFthKzQ+PjJdPWJ9JD1kKzE2fDA7cmV0dXJufXFhKCk7VCgpfWZ1bmN0aW9uIFFiKGEsYil7dmFyIGM9MCxkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTA7ZD1FW2ErMTF8MF0+Pj43fDA/RFthKzQ+PjJdOkVbYSsxMXwwXTtpZihkPj4+MDxiPj4+MCl7aD0kLTE2fDA7JD1oO2Y9Yi1kfDA7aWYoZil7Yj1FW2ErMTF8MF0+Pj43fDA7Zz1iP0RbYSs0Pj4yXTpFW2ErMTF8MF07aT1nK2Z8MDtiPWI/KERbYSs4Pj4yXSYyMTQ3NDgzNjQ3KS0xfDA6MTA7aWYoYi1nPj4+MDxmPj4+MCl7YTp7ZD0kLTE2fDA7JD1kO2M9aS1ifDA7aWYoYz4+PjA8PS0xNy1iPj4+MCl7aj1FW2ErMTF8MF0+Pj43fDA/RFthPj4yXTphO2I6e2lmKGI+Pj4wPDIxNDc0ODM2MjMpe0RbZCs4Pj4yXT1iPDwxO0RbZCsxMj4+Ml09YitjO2M9JC0xNnwwOyQ9YzskPWMrMTZ8MDtjPWQrOHwwO2U9ZCsxMnwwO2M9RFsoR1tlPj4yXTxHW2M+PjJdP2M6ZSk+PjJdO2lmKGM+Pj4wPj0xMSl7ZT1jKzE2Ji0xNjtjPWUtMXwwO2M9KGN8MCk9PTExP2U6Y31lbHNle2M9MTB9YnJlYWsgYn1jPS0xOH1lPWMrMXwwO2M9bmEoZSk7aWYoZyl7WGEoYyxqLGcpfWlmKChifDApIT0xMCl7bWEoail9RFthPj4yXT1jO0RbYSs4Pj4yXT1lfC0yMTQ3NDgzNjQ4OyQ9ZCsxNnwwO2JyZWFrIGF9QWEoKTtUKCl9fWI9RVthKzExfDBdPj4+N3wwP0RbYT4+Ml06YTtkPWcrYnwwO2lmKGYpe3BhKGQsMCxmKX1jOntpZihFW2ErMTF8MF0+Pj43fDApe0RbYSs0Pj4yXT1pO2JyZWFrIGN9QlthKzExfDBdPWl9QltoKzE1fDBdPTA7QltiK2l8MF09RVtoKzE1fDBdfSQ9aCsxNnwwO3JldHVybn1kPSQtMTZ8MDskPWQ7ZDp7aWYoRVthKzExfDBdPj4+N3wwKXtmPURbYT4+Ml07QltkKzE1fDBdPTA7QltiK2Z8MF09RVtkKzE1fDBdO0RbYSs0Pj4yXT1iO2JyZWFrIGR9QltkKzE0fDBdPTA7QlthK2J8MF09RVtkKzE0fDBdO0JbYSsxMXwwXT1ifSQ9ZCsxNnwwfWZ1bmN0aW9uIFNnKGEsYil7YT1hfDA7Yj1ifDA7dmFyIGM9MCxkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTAsaz0wLGw9MCxtPTA7bD1EW2ErMTI+PjJdO2M9RFthKzEwOD4+Ml07ZT1EW2MrODA+PjJdO0JbYis4NHwwXT0wO2Y9RFtiKzY4Pj4yXTtkPURbYis3Mj4+Ml0tZj4+MjthOntpZihkPj4+MDxlPj4+MCl7eGEoYis2OHwwLGUtZHwwLDkxNTYpO2M9RFthKzEwOD4+Ml07ZT1EW2MrODA+PjJdO2JyZWFrIGF9aWYoZT4+PjA+PWQ+Pj4wKXticmVhayBhfURbYis3Mj4+Ml09ZisoZTw8Mil9az1EW2MrOTY+PjJdO2M9RFtjKzEwMD4+Ml0ta3wwO2lmKCFjKXtyZXR1cm4gMX1jPShjfDApLzEyfDA7bT1jPj4+MD4xP2M6MTtjPTA7Yjp7d2hpbGUoMSl7aWYoKGN8MCk9PTE0MzE2NTU3NjUpe2JyZWFrIGJ9ZD1EW2w+PjJdKyhKKGMsMyk8PDIpfDA7aD1EW2Q+PjJdO2lmKChofDApPT0tMSl7YnJlYWsgYn1mPUooYywxMikra3wwO2c9RFtmPj4yXTtpZihnPj4+MD49ZT4+PjApe2JyZWFrIGJ9aj1EW0RbYSsxMTI+PjJdKzEyPj4yXTtpPURbaisoaDw8Mik+PjJdO2lmKGk+Pj4wPj1lPj4+MCl7YnJlYWsgYn1oPURbYis2OD4+Ml07RFtoKyhnPDwyKT4+Ml09aTtnPURbZCs0Pj4yXTtpZigoZ3wwKT09LTEpe2JyZWFrIGJ9aT1EW2YrND4+Ml07aWYoaT4+PjA+PWU+Pj4wKXticmVhayBifWc9RFsoZzw8Mikraj4+Ml07aWYoZz4+PjA+PWU+Pj4wKXticmVhayBifURbaCsoaTw8Mik+PjJdPWc7ZD1EW2QrOD4+Ml07aWYoKGR8MCk9PS0xKXticmVhayBifWY9RFtmKzg+PjJdO2lmKGY+Pj4wPj1lPj4+MCl7YnJlYWsgYn1kPURbKGQ8PDIpK2o+PjJdO2lmKGQ+Pj4wPj1lPj4+MCl7YnJlYWsgYn1EW2grKGY8PDIpPj4yXT1kO2M9YysxfDA7aWYoKG18MCkhPShjfDApKXtjb250aW51ZX1icmVha31yZXR1cm4gMX1yZXR1cm4gMH1mdW5jdGlvbiBKZyhhLGIpe2E9YXwwO2I9YnwwO3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wO2w9RFthKzEyPj4yXTtjPURbYSs2OD4+Ml07ZT1EW2MrODA+PjJdO0JbYis4NHwwXT0wO2Y9RFtiKzY4Pj4yXTtkPURbYis3Mj4+Ml0tZj4+MjthOntpZihkPj4+MDxlPj4+MCl7eGEoYis2OHwwLGUtZHwwLDkxNTYpO2M9RFthKzY4Pj4yXTtlPURbYys4MD4+Ml07YnJlYWsgYX1pZihlPj4+MD49ZD4+PjApe2JyZWFrIGF9RFtiKzcyPj4yXT1mKyhlPDwyKX1rPURbYys5Nj4+Ml07Yz1EW2MrMTAwPj4yXS1rfDA7aWYoIWMpe3JldHVybiAxfWM9KGN8MCkvMTJ8MDttPWM+Pj4wPjE/YzoxO2M9MDtiOnt3aGlsZSgxKXtpZigoY3wwKT09MTQzMTY1NTc2NSl7YnJlYWsgYn1kPURbbD4+Ml0rKEooYywzKTw8Mil8MDtoPURbZD4+Ml07aWYoKGh8MCk9PS0xKXticmVhayBifWY9SihjLDEyKStrfDA7Zz1EW2Y+PjJdO2lmKGc+Pj4wPj1lPj4+MCl7YnJlYWsgYn1qPURbRFthKzcyPj4yXSsxMj4+Ml07aT1EW2orKGg8PDIpPj4yXTtpZihpPj4+MD49ZT4+PjApe2JyZWFrIGJ9aD1EW2IrNjg+PjJdO0RbaCsoZzw8Mik+PjJdPWk7Zz1EW2QrND4+Ml07aWYoKGd8MCk9PS0xKXticmVhayBifWk9RFtmKzQ+PjJdO2lmKGk+Pj4wPj1lPj4+MCl7YnJlYWsgYn1nPURbKGc8PDIpK2o+PjJdO2lmKGc+Pj4wPj1lPj4+MCl7YnJlYWsgYn1EW2grKGk8PDIpPj4yXT1nO2Q9RFtkKzg+PjJdO2lmKChkfDApPT0tMSl7YnJlYWsgYn1mPURbZis4Pj4yXTtpZihmPj4+MD49ZT4+PjApe2JyZWFrIGJ9ZD1EWyhkPDwyKStqPj4yXTtpZihkPj4+MD49ZT4+PjApe2JyZWFrIGJ9RFtoKyhmPDwyKT4+Ml09ZDtjPWMrMXwwO2lmKChtfDApIT0oY3wwKSl7Y29udGludWV9YnJlYWt9cmV0dXJuIDF9cmV0dXJuIDB9ZnVuY3Rpb24gUmcoYSl7YT1hfDA7dmFyIGI9MCxjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MDtnPSQtMTZ8MDskPWc7Yj1EW2ErND4+Ml07ZD1EW2I+PjJdO2E6e2M9RFthKzEyPj4yXTtjPURbYysyOD4+Ml0tRFtjKzI0Pj4yXXwwO2U9Yz4+MjtiOntpZihlPj4+MDw9RFtiKzg+PjJdLWQ+PjI+Pj4wKXticmVhayBifWlmKChjfDApPDApe2JyZWFrIGF9Zj1EW2IrND4+Ml07Yz1uYShjKTtoPWMrKGU8PDIpfDA7ZT1mLWR8MDtmPWUrY3wwO2lmKChlfDApPjApe29hKGMsZCxlKX1EW2IrOD4+Ml09aDtEW2IrND4+Ml09ZjtEW2I+PjJdPWM7aWYoIWQpe2JyZWFrIGJ9bWEoZCl9Yj1EW2ErMTI+PjJdO2Q9RFtiKzI4Pj4yXTtiPURbYisyND4+Ml07RFtnKzEyPj4yXT0wO2I9ZC1iPj4yO2M9YSs5NnwwO2U9RFtjPj4yXTtkPURbYSsxMDA+PjJdLWU+PjI7Yzp7aWYoYj4+PjA+ZD4+PjApe3hhKGMsYi1kfDAsZysxMnwwKTticmVhayBjfWlmKGI+Pj4wPj1kPj4+MCl7YnJlYWsgY31EW2ErMTAwPj4yXT1lKyhiPDwyKX1lPWErOHwwO2I9RFthKzExNj4+Ml07ZDp7aWYoYil7Yz1EW2I+PjJdO2lmKChjfDApPT1EW2IrND4+Ml0pe2Q9MTticmVhayBkfWI9MDt3aGlsZSgxKXtkPWxkKGUsRFsoYjw8MikrYz4+Ml0pO2lmKCFkKXticmVhayBkfWY9RFthKzExNj4+Ml07Yz1EW2Y+PjJdO2I9YisxfDA7aWYoYj4+PjA8RFtmKzQ+PjJdLWM+PjI+Pj4wKXtjb250aW51ZX1icmVha31icmVhayBkfWQ9MTthPURbYSsxMj4+Ml07YT1EW2ErND4+Ml0tRFthPj4yXT4+MjtpZihhPj4+MDwzKXticmVhayBkfWE9KGE+Pj4wKS8zfDA7Yj0wO3doaWxlKDEpe2Q9bGQoZSxKKGIsMykpO2lmKCFkKXticmVhayBkfWI9YisxfDA7aWYoKGF8MCkhPShifDApKXtjb250aW51ZX1icmVha319JD1nKzE2fDA7cmV0dXJuIGR8MH1yYSgxMzI2KTtUKCl9ZnVuY3Rpb24gS2EoYSxiKXt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wO2c9JC0xNnwwOyQ9ZztlPURbYisyMD4+Ml07Yz1EW2IrMTI+PjJdO2Q9RFtiKzE2Pj4yXTthOntpZigoZXwwKT49KGN8MCkmZD4+PjA+PUdbYis4Pj4yXXwoY3wwKTwoZXwwKSl7YnJlYWsgYX1CW2ErMTJ8MF09RVtkK0RbYj4+Ml18MF07ZT1EW2IrMjA+PjJdO2M9RFtiKzE2Pj4yXSsxfDA7ZT1jP2U6ZSsxfDA7RFtiKzE2Pj4yXT1jO0RbYisyMD4+Ml09ZTtpZighTGQoMSxnKzEyfDAsYikpe2JyZWFrIGF9ZT1EW2IrOD4+Ml07aD1EW2IrMTY+PjJdO2M9aDtkPWUtY3wwO2M9Yz4+PjA+ZT4+PjA7ZT1EW2IrMjA+PjJdO2Y9RFtiKzEyPj4yXS0oYytlfDApfDA7Yz1EW2crMTI+PjJdO2lmKChmfDApPD0wJmQ+Pj4wPGM+Pj4wfChmfDApPDB8KGN8MCk8PTApe2JyZWFrIGF9ZD1oK0RbYj4+Ml18MDtEW2E+PjJdPWQ7az1hO2Y9Yy0xfDA7aT1mK2R8MDtqPUVbaXwwXTtiOntpZihqPj4+MDw9NjMpe0RbYSs0Pj4yXT1mO2E9RVtpfDBdJjYzO2JyZWFrIGJ9Yzp7c3dpdGNoKChqPj4+NnwwKS0xfDApe2Nhc2UgMDppZihjPj4+MDwyKXticmVhayBhfURbYSs0Pj4yXT1jLTI7ZD0oYytkfDApLTJ8MDthPUVbZCsxfDBdPDw4JjE2MTI4fEVbZHwwXTticmVhayBiO2Nhc2UgMTpicmVhayBjO2RlZmF1bHQ6YnJlYWsgYX19aWYoYz4+PjA8Myl7YnJlYWsgYX1EW2ErND4+Ml09Yy0zO2Q9KGMrZHwwKS0zfDA7YT1FW2QrMnwwXTw8MTYmNDEyODc2OHxFW2QrMXwwXTw8OHxFW2R8MF19YT1hKzQwOTZ8MDtEW2srOD4+Ml09YTtpZihhPj4+MD4xMDQ4NTc1KXticmVhayBhfWE9ZTtkPWMraHwwO2E9ZD4+PjA8Yz4+PjA/YSsxfDA6YTtEW2IrMTY+PjJdPWQ7RFtiKzIwPj4yXT1hO2w9MX0kPWcrMTZ8MDtyZXR1cm4gbH1mdW5jdGlvbiBvZChhLGIsYyl7YT1hfDA7Yj1ifDA7Yz1jfDA7dmFyIGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MDtmPURbYSsxMj4+Ml07aD1EW2ErOD4+Ml07ZD1mLWg+PjI7Yj1CW2IrMjR8MF07YTp7aWYoZD4+PjA8Yj4+PjApe3NhKGErOHwwLGItZHwwKTtoPURbYSs4Pj4yXTtmPURbYSsxMj4+Ml07YnJlYWsgYX1pZihiPj4+MD49ZD4+PjApe2JyZWFrIGF9Zj0oYjw8MikraHwwO0RbYSsxMj4+Ml09Zn1iPTA7aT1EW2MrMjA+PjJdO2U9RFtjKzE2Pj4yXTtkPWYtaHwwO2Y9ZDtnPWUrZHwwO2o9RFtjKzEyPj4yXTtpPWQ+Pj4wPmc+Pj4wP2krMXwwOmk7Yjp7aWYoZz4+PjA+R1tjKzg+PjJdJihqfDApPD0oaXwwKXwoaXwwKT4oanwwKSl7YnJlYWsgYn1vYShoLGUrRFtjPj4yXXwwLGQpO2Q9RFtjKzIwPj4yXTtlPWYrRFtjKzE2Pj4yXXwwO2Q9ZT4+PjA8Zj4+PjA/ZCsxfDA6ZDtnPWU7RFtjKzE2Pj4yXT1lO0RbYysyMD4+Ml09ZDtlPURbYysxMj4+Ml07Zj1nKzR8MDtkPWY+Pj4wPDQ/ZCsxfDA6ZDtpZihmPj4+MD5HW2MrOD4+Ml0mKGR8MCk+PShlfDApfChkfDApPihlfDApKXticmVhayBifWQ9ZytEW2M+PjJdfDA7RFthKzIwPj4yXT1FW2R8MF18RVtkKzF8MF08PDh8KEVbZCsyfDBdPDwxNnxFW2QrM3wwXTw8MjQpO2Q9RFtjKzIwPj4yXTtnPURbYysxNj4+Ml07ZT1nKzR8MDtmPWU+Pj4wPDQ/ZCsxfDA6ZDtqPWU7RFtjKzE2Pj4yXT1lO0RbYysyMD4+Ml09ZjtlPURbYysxMj4+Ml07aWYoKGZ8MCk+PShlfDApJmo+Pj4wPj1HW2MrOD4+Ml18KGZ8MCk+KGV8MCkpe2JyZWFrIGJ9ZT1FW2orRFtjPj4yXXwwXTtmPWcrNXwwO2Q9Zj4+PjA8NT9kKzF8MDpkO0RbYysxNj4+Ml09ZjtEW2MrMjA+PjJdPWQ7aWYoZS0xPj4+MD4yOSl7YnJlYWsgYn1EW2ErND4+Ml09ZTtiPTF9cmV0dXJuIGJ8MH1mdW5jdGlvbiBUYyhhLGIpe3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MDtnPSQtMTZ8MDskPWc7YTp7Yjp7aWYoYil7RFthKzg4Pj4yXT0wO0RbYSs5Mj4+Ml09MDtjPURbYSs4ND4+Ml07RFthKzg0Pj4yXT0wO2lmKGMpe21hKGMpfURbYSs3Nj4+Ml09MDtEW2ErODA+PjJdPTA7Yz1EW2ErNzI+PjJdO0RbYSs3Mj4+Ml09MDtpZihjKXttYShjKX1jPURbYj4+Ml07ZD1EW2IrND4+Ml07QltnKzE1fDBdPTA7RWEoYSxkLWM+PjIsZysxNXwwKTtjPURbYisyOD4+Ml07ZD1EW2IrMjQ+PjJdO0JbZysxNHwwXT0wO0VhKGErMTJ8MCxjLWQ+PjIsZysxNHwwKTtVYihhKzI4fDAsRFtiKzQ+PjJdLURbYj4+Ml0+PjIsMTAzMTYpO2Q9RFtiKzI4Pj4yXS1EW2IrMjQ+PjJdfDA7ZT1kPj4yO2M9RFthKzUyPj4yXTtjOntpZihlPj4+MDw9RFthKzYwPj4yXS1jPj4yPj4+MCl7YnJlYWsgY31pZigoZHwwKTwwKXticmVhayBifWY9RFthKzU2Pj4yXTtkPW5hKGQpO2g9ZCsoZTw8Mil8MDtlPWYtY3wwO2Y9ZStkfDA7aWYoKGV8MCk+MCl7b2EoZCxjLGUpfURbYSs2MD4+Ml09aDtEW2ErNTY+PjJdPWY7RFthKzUyPj4yXT1kO2lmKCFjKXticmVhayBjfW1hKGMpfWQ9RFtiKzI4Pj4yXS1EW2IrMjQ+PjJdfDA7ZT1kPj4yO2M9RFthKzQwPj4yXTtkOntpZihlPj4+MDw9RFthKzQ4Pj4yXS1jPj4yPj4+MCl7YnJlYWsgZH1pZigoZHwwKTwwKXticmVhayBhfWY9RFthKzQ0Pj4yXTtkPW5hKGQpO2g9ZCsoZTw8Mil8MDtlPWYtY3wwO2Y9ZStkfDA7aWYoKGV8MCk+MCl7b2EoZCxjLGUpfURbYSs0OD4+Ml09aDtEW2ErNDQ+PjJdPWY7RFthKzQwPj4yXT1kO2lmKCFjKXticmVhayBkfW1hKGMpfUJbYSsyNHwwXT0xO0RbYSs2ND4+Ml09Yn0kPWcrMTZ8MDtyZXR1cm59cmEoMTMyNik7VCgpfXJhKDEzMjYpO1QoKX1mdW5jdGlvbiBFZyhhLGIpe2E9YXwwO2I9YnwwO3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wO2g9RFthKzEyPj4yXTtjPURbYSs2OD4+Ml07ZT1EW2MrODA+PjJdO0JbYis4NHwwXT0wO2c9RFtiKzY4Pj4yXTtkPURbYis3Mj4+Ml0tZz4+MjthOntpZihkPj4+MDxlPj4+MCl7eGEoYis2OHwwLGUtZHwwLDkxNTYpO2M9RFthKzY4Pj4yXTtlPURbYys4MD4+Ml07YnJlYWsgYX1pZihlPj4+MD49ZD4+PjApe2JyZWFrIGF9RFtiKzcyPj4yXT1nKyhlPDwyKX1rPURbYys5Nj4+Ml07Yz1EW2MrMTAwPj4yXS1rfDA7aWYoIWMpe3JldHVybiAxfWM9KGN8MCkvMTJ8MDtsPWM+Pj4wPjE/YzoxO209RFtoKzI4Pj4yXTtjPTA7Yjp7d2hpbGUoMSl7ZD0oSihjLDMpPDwyKSttfDA7Zz1EW2Q+PjJdO2lmKChnfDApPT0tMSl7YnJlYWsgYn1oPUooYywxMikra3wwO2k9RFtoPj4yXTtpZihpPj4+MD49ZT4+PjApe2JyZWFrIGJ9Zj1nPDwyO2c9RFtEW2ErNzI+PjJdKzEyPj4yXTtmPURbZitnPj4yXTtpZihmPj4+MD49ZT4+PjApe2JyZWFrIGJ9aj1pPDwyO2k9RFtiKzY4Pj4yXTtEW2oraT4+Ml09ZjtmPURbZCs0Pj4yXTtpZigoZnwwKT09LTEpe2JyZWFrIGJ9aj1EW2grND4+Ml07aWYoaj4+PjA+PWU+Pj4wKXticmVhayBifWY9RFtnKyhmPDwyKT4+Ml07aWYoZj4+PjA+PWU+Pj4wKXticmVhayBifURbaSsoajw8Mik+PjJdPWY7ZD1EW2QrOD4+Ml07aWYoKGR8MCk9PS0xKXticmVhayBifWg9RFtoKzg+PjJdO2lmKGg+Pj4wPj1lPj4+MCl7YnJlYWsgYn1kPURbZysoZDw8Mik+PjJdO2lmKGQ+Pj4wPj1lPj4+MCl7YnJlYWsgYn1EW2krKGg8PDIpPj4yXT1kO2M9YysxfDA7aWYoKGx8MCkhPShjfDApKXtjb250aW51ZX1icmVha31yZXR1cm4gMX1yZXR1cm4gMH1mdW5jdGlvbiB4YyhhLGIsYyl7YT1hfDA7Yj1ifDA7Yz1jfDA7dmFyIGQ9MCxlPUsoMCksZj1LKDApLGc9SygwKSxoPUsoMCksaT1LKDApLGo9MCxrPUsoMCksbD1LKDApLG09SygwKSxuPUsoMCksbz0wO2E6e2lmKERbYysyOD4+Ml0hPTl8RVtjKzI0fDBdIT0zKXticmVhayBhfWE9RFthKzQ+PjJdO2lmKGEtMzE+Pj4wPDQyOTQ5NjcyNjcpe2JyZWFrIGF9bz0xO2o9RFtjKzgwPj4yXTtpZighail7YnJlYWsgYX1rPUsoSygyKS9LKCgxPDxhKS0yfDApKTtjPURbRFtjPj4yXT4+Ml0rRFtjKzQ4Pj4yXXwwO2E9RFtEW2I+PjJdPj4yXStEW2IrNDg+PjJdfDA7Yj0wO3doaWxlKDEpe2c9SygwKTtsPUsoMCk7bT1LKDApO2U9SyhLKEsoRFthPj4yXSkqaykrSygtMSkpO2Y9SyhLKEsoRFthKzQ+PjJdKSprKStLKC0xKSk7aT1LKEsoSygxKS1LKEwoZSkpKS1LKEwoZikpKTtoPUsoTyhLKC1pKSxLKDApKSk7bj1LKC1oKTtmPUsoZisoZjxLKDApP2g6bikpO2U9SyhlKyhlPEsoMCk/aDpuKSk7aD1LKEsoZipmKStLKEsoaSppKStLKGUqZSkpKTtpZighKCtoPDFlLTYpKXtnPUsoSygxKS9LKFMoaCkpKTttPUsoZipnKTtsPUsoZSpnKTtnPUsoaSpnKX1hPWErOHwwO2Q9KHYobSkseCgyKSk7QltjKzh8MF09ZDtCW2MrOXwwXT1kPj4+ODtCW2MrMTB8MF09ZD4+PjE2O0JbYysxMXwwXT1kPj4+MjQ7ZD0odihsKSx4KDIpKTtCW2MrNHwwXT1kO0JbYys1fDBdPWQ+Pj44O0JbYys2fDBdPWQ+Pj4xNjtCW2MrN3wwXT1kPj4+MjQ7ZD0odihnKSx4KDIpKTtCW2N8MF09ZDtCW2MrMXwwXT1kPj4+ODtCW2MrMnwwXT1kPj4+MTY7QltjKzN8MF09ZD4+PjI0O2M9YysxMnwwO2I9YisxfDA7aWYoKGp8MCkhPShifDApKXtjb250aW51ZX1icmVha319cmV0dXJuIG98MH1mdW5jdGlvbiBEYyhhLGIsYyxkKXt2YXIgZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTAsaz0wLGw9MCxtPTA7aj0rSFtiPj4yXTtrPStIW2IrND4+Ml07bD0rSFtiKzg+PjJdO2c9TChqKStMKGspK0wobCk7YTp7aWYoIShnPjFlLTYpKXtqPTE7az0wO2U9MDticmVhayBhfWc9MS9nO2s9ZyprO2o9ZypqO2U9ZypsPDB9aD1EW2ErMTY+PjJdO2w9KyhofDApO2c9UChqKmwrLjUpO2I6e2lmKEwoZyk8MjE0NzQ4MzY0OCl7bT1+fmc7YnJlYWsgYn1tPS0yMTQ3NDgzNjQ4fWY9bT4+MzE7aT1mK21eZjtnPVAoaypsKy41KTtjOntpZihMKGcpPDIxNDc0ODM2NDgpe2Y9fn5nO2JyZWFrIGN9Zj0tMjE0NzQ4MzY0OH1iPWY+PjMxO2I9aC0oaSsoZitiXmIpfDApfDA7aT0oYnwwKTwwPzA6YjtlPWU/MC1pfDA6aTtmPWYrKGI+PjMxJigoZnwwKT4wP2I6MC1ifDApKXwwO2Q6e2lmKChtfDApPj0wKXtiPWUraHwwO2E9RFthKzg+PjJdO2U9ZitofDA7YnJlYWsgZH1iPWY+PjMxO2I9YitmXmI7YT1EW2ErOD4+Ml07Yj0oZXwwKTwwP2I6YS1ifDA7ZT0oZnwwKTwwP2k6YS1pfDB9ZTp7aWYoIShifGUpKXtiPWE7YnJlYWsgZX1pZighKChhfDApIT0oYnwwKXxlKSl7Yj1hO2JyZWFrIGV9Zj0oYXwwKSE9KGV8MCk7aWYoIShifGYpKXtiPWE7YnJlYWsgZX1pZighKChifDApPD0oaHwwKXxlKSl7Yj0oaDw8MSktYnwwO2E9MDticmVhayBlfWlmKCEoKGJ8MCk+PShofDApfGYpKXtiPShoPDwxKS1ifDA7YnJlYWsgZX1pZighKChhfDApIT0oYnwwKXwoZXwwKT49KGh8MCkpKXtiPWE7YT0oaDw8MSktZXwwO2JyZWFrIGV9aWYoYil7YT1lO2JyZWFrIGV9Yj0wO2lmKChlfDApPD0oaHwwKSl7YT1lO2JyZWFrIGV9YT0oaDw8MSktZXwwfURbYz4+Ml09YTtEW2Q+PjJdPWJ9ZnVuY3Rpb24gUmMoYSxiKXt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MCxoPTA7Zz1EW2E+PjJdO2M9ZysoYj4+PjMmNTM2ODcwOTA4KXwwO0RbYz4+Ml09RFtjPj4yXXwxPDxiO2Y9RFthKzY0Pj4yXTtlPShifDApPT0tMTtkPS0xO2E6e2lmKGUpe2JyZWFrIGF9Yz1iKzF8MDtjPShjPj4+MCklM3wwP2M6Yi0yfDA7ZD0tMTtpZigoY3wwKT09LTEpe2JyZWFrIGF9ZD1EW0RbZj4+Ml0rKGM8PDIpPj4yXX1jPURbYSsxMj4+Ml07aD0oZD4+PjMmNTM2ODcwOTA4KStjfDA7RFtoPj4yXT1EW2g+PjJdfDE8PGQ7Yjp7Yzp7aWYoIWUpe2Q6e2U6e2lmKChiPj4+MCklM3wwKXtlPWItMXwwO2JyZWFrIGV9ZT1iKzJ8MDtkPS0xO2lmKChlfDApPT0tMSl7YnJlYWsgZH19ZD1EW0RbZj4+Ml0rKGU8PDIpPj4yXX1lPShkPj4+MyY1MzY4NzA5MDgpK2N8MDtEW2U+PjJdPURbZT4+Ml18MTw8ZDtkPS0xO2I9RFtEW2YrMTI+PjJdKyhiPDwyKT4+Ml07aWYoKGJ8MCk9PS0xKXticmVhayBifUJbYSsyNHwwXT0wO2E9KGI+Pj4zJjUzNjg3MDkwOCkrZ3wwO0RbYT4+Ml09RFthPj4yXXwxPDxiO2E9YisxfDA7YT0oYT4+PjApJTN8MD9hOmItMnwwO2lmKChhfDApIT0tMSl7ZD1EW0RbZj4+Ml0rKGE8PDIpPj4yXX1hPWMrKGQ+Pj4zJjUzNjg3MDkwOCl8MDtEW2E+PjJdPURbYT4+Ml18MTw8ZDtmOntnOntpZigoYj4+PjApJTN8MCl7Yj1iLTF8MDticmVhayBnfWI9YisyfDA7YT0tMTtpZigoYnwwKT09LTEpe2JyZWFrIGZ9fWE9RFtEW2Y+PjJdKyhiPDwyKT4+Ml19Yj0xPDxhO2E9YysoYT4+PjMmNTM2ODcwOTA4KXwwO2M9RFthPj4yXTticmVhayBjfWE9Yys1MzY4NzA5MDh8MDtiPURbYys1MzY4NzA5MDg+PjJdO2M9LTIxNDc0ODM2NDh9RFthPj4yXT1ifGN9fWZ1bmN0aW9uIGhjKGEsYil7dmFyIGM9MCxkPTA7Yz1EW2IrOD4+Ml07RFthKzQ+PjJdPURbYis0Pj4yXTtEW2ErOD4+Ml09YztEW2ErMjA+PjJdPURbYisyMD4+Ml07Yz1EW2IrMTY+PjJdO0RbYSsxMj4+Ml09RFtiKzEyPj4yXTtEW2ErMTY+PjJdPWM7YTp7Yjp7aWYoKGF8MCkhPShifDApKXtjPURbYisyOD4+Ml07aWYoYyl7Yzp7aWYoRFthKzMyPj4yXTw8NT4+PjA+PWM+Pj4wKXtkPURbYSsyND4+Ml07YnJlYWsgY31kPURbYSsyND4+Ml07aWYoZCl7bWEoZCk7RFthKzMyPj4yXT0wO0RbYSsyND4+Ml09MDtEW2ErMjg+PjJdPTA7Yz1EW2IrMjg+PjJdfWlmKChjfDApPDApe2JyZWFrIGJ9Yz0oYy0xPj4+NXwwKSsxfDA7ZD1uYShjPDwyKTtEW2ErMzI+PjJdPWM7RFthKzI4Pj4yXT0wO0RbYSsyND4+Ml09ZDtjPURbYisyOD4+Ml19TmEoZCxEW2IrMjQ+PjJdLChjLTE+Pj4zJjUzNjg3MDkwOCkrNHwwKTtjPURbYisyOD4+Ml19ZWxzZXtjPTB9RFthKzI4Pj4yXT1jO2M9RFtiKzQwPj4yXTtpZihjKXtkOntpZihEW2ErNDQ+PjJdPDw1Pj4+MD49Yz4+PjApe2Q9RFthKzM2Pj4yXTticmVhayBkfWQ9RFthKzM2Pj4yXTtpZihkKXttYShkKTtEW2ErNDQ+PjJdPTA7RFthKzM2Pj4yXT0wO0RbYSs0MD4+Ml09MDtjPURbYis0MD4+Ml19aWYoKGN8MCk8MCl7YnJlYWsgYX1jPShjLTE+Pj41fDApKzF8MDtkPW5hKGM8PDIpO0RbYSs0ND4+Ml09YztEW2ErNDA+PjJdPTA7RFthKzM2Pj4yXT1kO2M9RFtiKzQwPj4yXX1OYShkLERbYiszNj4+Ml0sKGMtMT4+PjMmNTM2ODcwOTA4KSs0fDApO2I9RFtiKzQwPj4yXX1lbHNle2I9MH1EW2ErNDA+PjJdPWJ9cmV0dXJufXFhKCk7VCgpfXFhKCk7VCgpfWZ1bmN0aW9uIEhkKGEsYil7dmFyIGM9MCxkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTA7Zz0kLTE2fDA7JD1nO2E6e2lmKCFPYSgxLGcrOHwwLGIpKXticmVhayBhfWM9RFtiKzg+PjJdO2U9RFtiKzE2Pj4yXTtoPURbZysxMj4+Ml07ZD1EW2IrMjA+PjJdO2Y9RFtiKzEyPj4yXS0oZCsoYz4+PjA8ZT4+PjApfDApfDA7aT1jLWV8MDtjPURbZys4Pj4yXTtpZigoaHwwKT09KGZ8MCkmaT4+PjA8Yz4+PjB8Zj4+PjA8aD4+PjApe2JyZWFrIGF9ZD1kK2h8MDtmPWMrZXwwO2Q9Zj4+PjA8ZT4+PjA/ZCsxfDA6ZDtEW2IrMTY+PjJdPWY7RFtiKzIwPj4yXT1kO2lmKChjfDApPD0wKXticmVhayBhfWQ9ZStEW2I+PjJdfDA7RFthKzQwPj4yXT1kO2g9YTtmPWMtMXwwO2U9ZCtmfDA7Yj1FW2V8MF07Yjp7aWYoYj4+PjA8PTYzKXtEW2ErNDQ+PjJdPWY7YT1FW2V8MF0mNjM7YnJlYWsgYn1jOntzd2l0Y2goKGI+Pj42fDApLTF8MCl7Y2FzZSAwOmlmKGM+Pj4wPDIpe2JyZWFrIGF9RFthKzQ0Pj4yXT1jLTI7Yj0oYytkfDApLTJ8MDthPUVbYisxfDBdPDw4JjE2MTI4fEVbYnwwXTticmVhayBiO2Nhc2UgMTppZihjPj4+MDwzKXticmVhayBhfURbYSs0ND4+Ml09Yy0zO2I9KGMrZHwwKS0zfDA7YT1FW2IrMnwwXTw8MTYmNDEyODc2OHxFW2IrMXwwXTw8OHxFW2J8MF07YnJlYWsgYjtkZWZhdWx0OmJyZWFrIGN9fURbYSs0ND4+Ml09Yy00O2I9KGMrZHwwKS00fDA7YT1FW2IrMnwwXTw8MTZ8RVtiKzN8MF08PDI0JjEwNTY5NjQ2MDh8RVtiKzF8MF08PDh8RVtifDBdfWE9YSsxNjM4NHwwO0RbaCs0OD4+Ml09YTtqPWE+Pj4wPDQxOTQzMDR9JD1nKzE2fDA7cmV0dXJuIGp9ZnVuY3Rpb24gSWUoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO3ZhciBkPTAsZT0wLGY9MCxnPTAsaD0wO2Y9JCstNjR8MDskPWY7ZD0xO2E6e2lmKEphKGEsYiwwKSl7YnJlYWsgYX1kPTA7aWYoIWIpe2JyZWFrIGF9ZD0kKy02NHwwOyQ9ZDtlPURbYj4+Ml07Zz1EW2UtND4+Ml07aD1EW2UtOD4+Ml07RFtkKzIwPj4yXT0wO0RbZCsxNj4+Ml09MTEyMjA7RFtkKzEyPj4yXT1iO0RbZCs4Pj4yXT0xMTI2ODtlPTA7cGEoZCsyNHwwLDAsMzkpO2I9YitofDA7Yjp7aWYoSmEoZywxMTI2OCwwKSl7RFtkKzU2Pj4yXT0xO2JhW0RbRFtnPj4yXSsyMD4+Ml1dKGcsZCs4fDAsYixiLDEsMCk7ZT1EW2QrMzI+PjJdPT0xP2I6MDticmVhayBifWJhW0RbRFtnPj4yXSsyND4+Ml1dKGcsZCs4fDAsYiwxLDApO2M6e3N3aXRjaChEW2QrNDQ+PjJdKXtjYXNlIDA6ZT1EW2QrNDg+PjJdPT0xP0RbZCszNj4+Ml09PTE/RFtkKzQwPj4yXT09MT9EW2QrMjg+PjJdOjA6MDowO2JyZWFrIGI7Y2FzZSAxOmJyZWFrIGM7ZGVmYXVsdDpicmVhayBifX1pZihEW2QrMzI+PjJdIT0xKXtpZihEW2QrNDg+PjJdfERbZCszNj4+Ml0hPTF8RFtkKzQwPj4yXSE9MSl7YnJlYWsgYn19ZT1EW2QrMjQ+PjJdfSQ9ZC0gLTY0fDA7ZD0wO2lmKCFlKXticmVhayBhfWI9Zis4fDA7cGEoYnw0LDAsNTIpO0RbZis1Nj4+Ml09MTtEW2YrMjA+PjJdPS0xO0RbZisxNj4+Ml09YTtEW2YrOD4+Ml09ZTtiYVtEW0RbZT4+Ml0rMjg+PjJdXShlLGIsRFtjPj4yXSwxKTthPURbZiszMj4+Ml07aWYoKGF8MCk9PTEpe0RbYz4+Ml09RFtmKzI0Pj4yXX1kPShhfDApPT0xfSQ9Zi0gLTY0fDA7cmV0dXJuIGR8MH1mdW5jdGlvbiBOZihhLGIsYyxkKXthPWF8MDtiPWJ8MDtjPWN8MDtkPWR8MDt2YXIgZT0wLGY9MCxnPTA7YTp7YT0kLTMyfDA7JD1hO2U9emEoYyk7aWYoZT4+PjA8NDI5NDk2NzI4MCl7Yjp7Yzp7aWYoZT4+PjA+PTExKXtnPWUrMTYmLTE2O2Y9bmEoZyk7RFthKzI0Pj4yXT1nfC0yMTQ3NDgzNjQ4O0RbYSsxNj4+Ml09ZjtEW2ErMjA+PjJdPWU7YnJlYWsgY31CW2ErMjd8MF09ZTtmPWErMTZ8MDtpZighZSl7YnJlYWsgYn19b2EoZixjLGUpfUJbZStmfDBdPTA7RFthKzg+PjJdPTA7RFthPj4yXT0wO0RbYSs0Pj4yXT0wO2Q6e2M9X2EoYixhKzE2fDApO2lmKChjfDApPT0oYis0fDApKXticmVhayBkfWI9RFtjKzI4Pj4yXTtmPURbYyszMj4+Ml07aWYoKGJ8MCk9PShmfDApKXticmVhayBkfWI9Zi1ifDA7aWYoYiYzKXticmVhayBkfWU9Yj4+PjJ8MDtmPURbYSs0Pj4yXTtiPURbYT4+Ml07Zz1mLWI+PjI7ZTp7aWYoZT4+PjA+Zz4+PjApe3NhKGEsZS1nfDApO2I9RFthPj4yXTtmPURbYSs0Pj4yXTticmVhayBlfWlmKGU+Pj4wPj1nPj4+MCl7YnJlYWsgZX1mPShlPDwyKStifDA7RFthKzQ+PjJdPWZ9aWYoKGJ8MCkhPShmfDApKXtlPWI7Yj1EW2MrMjg+PjJdO29hKGUsYixEW2MrMzI+PjJdLWJ8MCk7YnJlYWsgZH11YSgpO1QoKX1iPURbZD4+Ml07aWYoYil7RFtkKzQ+PjJdPWI7bWEoYil9RFtkPj4yXT1EW2E+PjJdO0RbZCs0Pj4yXT1EW2ErND4+Ml07RFtkKzg+PjJdPURbYSs4Pj4yXTtpZihCW2ErMjd8MF08MCl7bWEoRFthKzE2Pj4yXSl9JD1hKzMyfDA7YnJlYWsgYX1BYSgpO1QoKX19ZnVuY3Rpb24ga2MoYSl7dmFyIGI9MCxjPTAsZD0wO2I9RFthKzg+PjJdO2Q9RFthPj4yXTthOntpZihFW2ErMTJ8MF0pe2I6e2M6e2Q6e2U6e2lmKChifDApPT0tMSl7YnJlYWsgZX1jPWIrMXwwO2I9KGM+Pj4wKSUzfDA/YzpiLTJ8MDtpZigoYnwwKT09LTEpe2JyZWFrIGV9Yj1EW0RbZCsxMj4+Ml0rKGI8PDIpPj4yXTtpZigoYnwwKSE9LTEpe2JyZWFrIGR9fURbYSs4Pj4yXT0tMTticmVhayBjfWM9YisxfDA7Yj0oYz4+PjApJTN8MD9jOmItMnwwO0RbYSs4Pj4yXT1iO2lmKChifDApIT0tMSl7YnJlYWsgYn19Yz1EW2ErND4+Ml07Yj0tMTtmOntpZigoY3wwKT09LTEpe2JyZWFrIGZ9Zzp7aWYoKGM+Pj4wKSUzfDApe2M9Yy0xfDA7YnJlYWsgZ31jPWMrMnwwO2I9LTE7aWYoKGN8MCk9PS0xKXticmVhayBmfX1jPURbRFtkKzEyPj4yXSsoYzw8Mik+PjJdO2I9LTE7aWYoKGN8MCk9PS0xKXticmVhayBmfWI9Yy0xfDA7aWYoKGM+Pj4wKSUzfDApe2JyZWFrIGZ9Yj1jKzJ8MH1CW2ErMTJ8MF09MDtEW2ErOD4+Ml09YjtyZXR1cm59aWYoKGJ8MCkhPURbYSs0Pj4yXSl7YnJlYWsgYX1EW2ErOD4+Ml09LTE7cmV0dXJufWM9LTE7aDp7aWYoKGJ8MCk9PS0xKXticmVhayBofWk6e2lmKChiPj4+MCklM3wwKXtiPWItMXwwO2JyZWFrIGl9Yj1iKzJ8MDtjPS0xO2lmKChifDApPT0tMSl7YnJlYWsgaH19Yj1EW0RbZCsxMj4+Ml0rKGI8PDIpPj4yXTtjPS0xO2lmKChifDApPT0tMSl7YnJlYWsgaH1jPWItMXwwO2lmKChiPj4+MCklM3wwKXticmVhayBofWM9YisyfDB9RFthKzg+PjJdPWN9fWZ1bmN0aW9uIEpkKGEpe3ZhciBiPTAsYz0wLGQ9MDtiPW5hKDMyKTtjPUVbMTYxOV18RVsxNjIwXTw8ODtCW2IrMjR8MF09YztCW2IrMjV8MF09Yz4+Pjg7Yz1FWzE2MTVdfEVbMTYxNl08PDh8KEVbMTYxN108PDE2fEVbMTYxOF08PDI0KTtkPUVbMTYxMV18RVsxNjEyXTw8OHwoRVsxNjEzXTw8MTZ8RVsxNjE0XTw8MjQpO0JbYisxNnwwXT1kO0JbYisxN3wwXT1kPj4+ODtCW2IrMTh8MF09ZD4+PjE2O0JbYisxOXwwXT1kPj4+MjQ7QltiKzIwfDBdPWM7QltiKzIxfDBdPWM+Pj44O0JbYisyMnwwXT1jPj4+MTY7QltiKzIzfDBdPWM+Pj4yNDtjPUVbMTYwN118RVsxNjA4XTw8OHwoRVsxNjA5XTw8MTZ8RVsxNjEwXTw8MjQpO2Q9RVsxNjAzXXxFWzE2MDRdPDw4fChFWzE2MDVdPDwxNnxFWzE2MDZdPDwyNCk7QltiKzh8MF09ZDtCW2IrOXwwXT1kPj4+ODtCW2IrMTB8MF09ZD4+PjE2O0JbYisxMXwwXT1kPj4+MjQ7QltiKzEyfDBdPWM7QltiKzEzfDBdPWM+Pj44O0JbYisxNHwwXT1jPj4+MTY7QltiKzE1fDBdPWM+Pj4yNDtjPUVbMTU5OV18RVsxNjAwXTw8OHwoRVsxNjAxXTw8MTZ8RVsxNjAyXTw8MjQpO2Q9RVsxNTk1XXxFWzE1OTZdPDw4fChFWzE1OTddPDwxNnxFWzE1OThdPDwyNCk7QltifDBdPWQ7QltiKzF8MF09ZD4+Pjg7QltiKzJ8MF09ZD4+PjE2O0JbYiszfDBdPWQ+Pj4yNDtCW2IrNHwwXT1jO0JbYis1fDBdPWM+Pj44O0JbYis2fDBdPWM+Pj4xNjtCW2IrN3wwXT1jPj4+MjQ7QltiKzI2fDBdPTA7RFthPj4yXT0tMTt0YShhKzR8MCxiLDI2KTttYShiKX1mdW5jdGlvbiBQZihhLGIsYyl7YT1hfDA7Yj1ifDA7Yz1jfDA7dmFyIGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MCxrPTA7ZD0kLTE2fDA7JD1kO2E6e2U9emEoYyk7aWYoZT4+PjA8NDI5NDk2NzI4MCl7Yjp7Yzp7aWYoZT4+PjA+PTExKXtmPWUrMTYmLTE2O2E9bmEoZik7RFtkKzg+PjJdPWZ8LTIxNDc0ODM2NDg7RFtkPj4yXT1hO0RbZCs0Pj4yXT1lO2JyZWFrIGN9QltkKzExfDBdPWU7YT1kO2lmKCFlKXticmVhayBifX1vYShhLGMsZSl9QlthK2V8MF09MDtjPUVbZCsxMXwwXTtlPWM8PDI0Pj4yNDtmPURbZD4+Ml07Yj1EW2IrND4+Ml07YT0wO2Q6e2lmKCFiKXticmVhayBkfWE9YztjPShlfDApPDA7YT1jP0RbZCs0Pj4yXTphO2o9Yz9mOmQ7d2hpbGUoMSl7Yz1FW2IrMjd8MF07Zz1jPDwyND4+MjQ8MDtjPWc/RFtiKzIwPj4yXTpjO2s9Yz4+PjA8YT4+PjA7ZTp7Zjp7aT1rP2M6YTtnOntpZihpKXtoPWIrMTZ8MDtnPWc/RFtoPj4yXTpoO2g9dmEoaixnLGkpO2g6e2lmKCFoKXtpZihhPj4+MD49Yz4+PjApe2JyZWFrIGh9YnJlYWsgZX1pZigoaHwwKTwwKXticmVhayBlfX1jPXZhKGcsaixpKTtpZighYyl7YnJlYWsgZ31pZigoY3wwKTwwKXticmVhayBmfWE9MTticmVhayBkfWlmKGE+Pj4wPGM+Pj4wKXticmVhayBlfX1pZihrKXticmVhayBmfWE9MTticmVhayBkfWI9Yis0fDB9Yj1EW2I+PjJdO2lmKGIpe2NvbnRpbnVlfWJyZWFrfWE9MH1pZigoZXwwKTwwKXttYShmKX0kPWQrMTZ8MDticmVhayBhfUFhKCk7VCgpfXJldHVybiBhfDB9ZnVuY3Rpb24gTmEoYSxiLGMpe3ZhciBkPTAsZT0wO2E6e2lmKChhfDApPT0oYnwwKSl7YnJlYWsgYX1lPWErY3wwO2lmKGItZT4+PjA8PTAtKGM8PDEpPj4+MCl7cmV0dXJuIG9hKGEsYixjKX1kPShhXmIpJjM7Yjp7Yzp7aWYoYT4+PjA8Yj4+PjApe2lmKGQpe2Q9YTticmVhayBifWlmKCEoYSYzKSl7ZD1hO2JyZWFrIGN9ZD1hO3doaWxlKDEpe2lmKCFjKXticmVhayBhfUJbZHwwXT1FW2J8MF07Yj1iKzF8MDtjPWMtMXwwO2Q9ZCsxfDA7aWYoZCYzKXtjb250aW51ZX1icmVha31icmVhayBjfWQ6e2lmKGQpe2JyZWFrIGR9aWYoZSYzKXt3aGlsZSgxKXtpZighYyl7YnJlYWsgYX1jPWMtMXwwO2Q9YythfDA7QltkfDBdPUVbYitjfDBdO2lmKGQmMyl7Y29udGludWV9YnJlYWt9fWlmKGM+Pj4wPD0zKXticmVhayBkfXdoaWxlKDEpe2M9Yy00fDA7RFtjK2E+PjJdPURbYitjPj4yXTtpZihjPj4+MD4zKXtjb250aW51ZX1icmVha319aWYoIWMpe2JyZWFrIGF9d2hpbGUoMSl7Yz1jLTF8MDtCW2MrYXwwXT1FW2IrY3wwXTtpZihjKXtjb250aW51ZX1icmVha31icmVhayBhfWlmKGM+Pj4wPD0zKXticmVhayBifXdoaWxlKDEpe0RbZD4+Ml09RFtiPj4yXTtiPWIrNHwwO2Q9ZCs0fDA7Yz1jLTR8MDtpZihjPj4+MD4zKXtjb250aW51ZX1icmVha319aWYoIWMpe2JyZWFrIGF9d2hpbGUoMSl7QltkfDBdPUVbYnwwXTtkPWQrMXwwO2I9YisxfDA7Yz1jLTF8MDtpZihjKXtjb250aW51ZX1icmVha319cmV0dXJuIGF9ZnVuY3Rpb24gT2MoYSxiLGMpe3ZhciBkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTAsaz0wLGw9MCxtPTA7ZD0kLTE2fDA7JD1kO2c9RFthKzI0Pj4yXTtsPURbYSsyOD4+Ml07YTp7aWYoKGd8MCkhPShsfDApKXt3aGlsZSgxKXtEW2QrOD4+Ml09MDtEW2Q+PjJdPTA7RFtkKzQ+PjJdPTA7ZT1OYyhEW2c+PjJdLGIsZCk7YT1FW2QrMTF8MF07aT1hPDwyND4+MjQ7aj0zO2I6e2M6e2Q6e2U6e2lmKCFlKXticmVhayBlfWo9MDtlPUVbYysxMXwwXTtmPWU8PDI0Pj4yNDtrPShpfDApPDA/RFtkKzQ+PjJdOmE7aWYoKGt8MCkhPSgoKGZ8MCk8MD9EW2MrND4+Ml06ZSl8MCkpe2JyZWFrIGV9aD0oZnwwKTwwP0RbYz4+Ml06YztmPURbZD4+Ml07ZT0oaXwwKTwwO2Y6e2lmKCFlKXtpZighaSl7YnJlYWsgZn1lPWQ7aWYoRVtofDBdIT0oZiYyNTUpKXticmVhayBifXdoaWxlKDEpe2E9YS0xfDA7aWYoIWEpe2JyZWFrIGZ9Zj1FW2grMXwwXTtoPWgrMXwwO2U9ZSsxfDA7aWYoKGZ8MCk9PUVbZXwwXSl7Y29udGludWV9YnJlYWt9YnJlYWsgZX1pZighayl7YnJlYWsgZn1pZih2YShlP2Y6ZCxoLGspKXticmVhayBkfX1tPURbZz4+Ml07aj0xfWlmKChpfDApPj0wKXticmVhayBjfX1tYShEW2Q+PjJdKX1zd2l0Y2goanwwKXtjYXNlIDA6Y2FzZSAzOmJyZWFrIGI7ZGVmYXVsdDpicmVhayBhfX1nPWcrNHwwO2lmKChsfDApIT0oZ3wwKSl7Y29udGludWV9YnJlYWt9fW09MH0kPWQrMTZ8MDtyZXR1cm4gbX1mdW5jdGlvbiBPYihhLGIpe3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTA7ZD1EW2ErOD4+Ml07Yz1EW2ErND4+Ml07aWYoZC1jPj4yPj4+MD49Yj4+PjApe2lmKGIpe2I9Yjw8MjtjPXBhKGMsMCxiKStifDB9RFthKzQ+PjJdPWM7cmV0dXJufWE6e2I6e2M6e2c9RFthPj4yXTtmPWMtZz4+MjtlPWYrYnwwO2lmKGU+Pj4wPDEwNzM3NDE4MjQpe2Q9ZC1nfDA7aD1kPj4xO2U9ZD4+Mj4+PjA8NTM2ODcwOTExP2U+Pj4wPmg+Pj4wP2U6aDoxMDczNzQxODIzO2lmKGUpe2lmKGU+Pj4wPj0xMDczNzQxODI0KXticmVhayBjfWk9bmEoZTw8Mil9ZD0oZjw8MikraXwwO2Y9Yjw8MjtiPXBhKGQsMCxmKTtmPWIrZnwwO2U9KGU8PDIpK2l8MDtpZigoY3wwKT09KGd8MCkpe2JyZWFrIGJ9d2hpbGUoMSl7Yz1jLTR8MDtiPURbYz4+Ml07RFtjPj4yXT0wO2Q9ZC00fDA7RFtkPj4yXT1iO2lmKChjfDApIT0oZ3wwKSl7Y29udGludWV9YnJlYWt9RFthKzg+PjJdPWU7Yj1EW2ErND4+Ml07RFthKzQ+PjJdPWY7Yz1EW2E+PjJdO0RbYT4+Ml09ZDtpZigoYnwwKT09KGN8MCkpe2JyZWFrIGF9d2hpbGUoMSl7Yj1iLTR8MDthPURbYj4+Ml07RFtiPj4yXT0wO2lmKGEpe2JhW0RbRFthPj4yXSs0Pj4yXV0oYSl9aWYoKGJ8MCkhPShjfDApKXtjb250aW51ZX1icmVha31icmVhayBhfXFhKCk7VCgpfXJhKDEzMjYpO1QoKX1EW2ErOD4+Ml09ZTtEW2ErND4+Ml09ZjtEW2E+PjJdPWJ9aWYoYyl7bWEoYyl9fWZ1bmN0aW9uIFNkKGEsYil7YT1hfDA7Yj1ifDA7dmFyIGM9MCxkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTAsaz0wO2Q9RFtiKzEyPj4yXTtnPWQ7Yz1EW2IrMjA+PjJdO2U9YztmPURbYisxNj4+Ml07aD1mKzR8MDtjPWg+Pj4wPDQ/YysxfDA6YztpPURbYis4Pj4yXTthOntpZihpPj4+MDxoPj4+MCYoY3wwKT49KGR8MCl8KGN8MCk+KGR8MCkpe2JyZWFrIGF9aj1EW2I+PjJdO2Q9aitmfDA7ZD1FW2R8MF18RVtkKzF8MF08PDh8KEVbZCsyfDBdPDwxNnxFW2QrM3wwXTw8MjQpO0RbYisxNj4+Ml09aDtEW2IrMjA+PjJdPWM7Yz1lO2U9Zis4fDA7Yz1lPj4+MDw4P2MrMXwwOmM7Zj1lO2U9YztpZihmPj4+MD5pPj4+MCYoY3wwKT49KGd8MCl8KGN8MCk+KGd8MCkpe2JyZWFrIGF9Yz1oK2p8MDtjPUVbY3wwXXxFW2MrMXwwXTw8OHwoRVtjKzJ8MF08PDE2fEVbYyszfDBdPDwyNCk7RFtiKzE2Pj4yXT1mO0RbYisyMD4+Ml09ZTtpZigoY3wwKTwoZHwwKSl7YnJlYWsgYX1EW2ErMTY+PjJdPWM7RFthKzEyPj4yXT1kO2c9KGM+PjMxKS0oKGQ+PjMxKSsoYz4+PjA8ZD4+PjApfDApfDA7Yz1jLWR8MDtpZighZyZjPj4+MD4yMTQ3NDgzNjQ2fGcpe2JyZWFrIGF9Yz1jKzF8MDtEW2ErMjA+PjJdPWM7ZD1jPj4+MXwwO0RbYSsyND4+Ml09ZDtEW2ErMjg+PjJdPTAtZDtpZighKGMmMSkpe0RbYSsyND4+Ml09ZC0xfWs9S2EoYSsxMTJ8MCxiKX1yZXR1cm4ga3wwfWZ1bmN0aW9uIHBhKGEsYixjKXt2YXIgZD0wLGU9MCxmPTA7YTp7aWYoIWMpe2JyZWFrIGF9QlthfDBdPWI7ZT1hK2N8MDtCW2UtMXwwXT1iO2lmKGM+Pj4wPDMpe2JyZWFrIGF9QlthKzJ8MF09YjtCW2ErMXwwXT1iO0JbZS0zfDBdPWI7QltlLTJ8MF09YjtpZihjPj4+MDw3KXticmVhayBhfUJbYSszfDBdPWI7QltlLTR8MF09YjtpZihjPj4+MDw5KXticmVhayBhfWU9MC1hJjM7Zj1lK2F8MDtkPUooYiYyNTUsMTY4NDMwMDkpO0RbZj4+Ml09ZDtiPWMtZSYtNDtjPWIrZnwwO0RbYy00Pj4yXT1kO2lmKGI+Pj4wPDkpe2JyZWFrIGF9RFtmKzg+PjJdPWQ7RFtmKzQ+PjJdPWQ7RFtjLTg+PjJdPWQ7RFtjLTEyPj4yXT1kO2lmKGI+Pj4wPDI1KXticmVhayBhfURbZisyND4+Ml09ZDtEW2YrMjA+PjJdPWQ7RFtmKzE2Pj4yXT1kO0RbZisxMj4+Ml09ZDtEW2MtMTY+PjJdPWQ7RFtjLTIwPj4yXT1kO0RbYy0yND4+Ml09ZDtEW2MtMjg+PjJdPWQ7Yz1iO2I9ZiY0fDI0O2M9Yy1ifDA7aWYoYz4+PjA8MzIpe2JyZWFrIGF9ZD0kaChkLDAsMSwxKTtlPWFhO2I9YitmfDA7d2hpbGUoMSl7RFtiKzI0Pj4yXT1kO0RbYisyOD4+Ml09ZTtEW2IrMTY+PjJdPWQ7RFtiKzIwPj4yXT1lO0RbYis4Pj4yXT1kO0RbYisxMj4+Ml09ZTtEW2I+PjJdPWQ7RFtiKzQ+PjJdPWU7Yj1iKzMyfDA7Yz1jLTMyfDA7aWYoYz4+PjA+MzEpe2NvbnRpbnVlfWJyZWFrfX1yZXR1cm4gYX1mdW5jdGlvbiBRYyhhLGIpe3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wO2Q9LTE7Zj0tMTtlPS0xO2E6e2I6e2lmKChifDApPT0tMSl7YnJlYWsgYn1nPTE7Zj1EW0RbRFthKzQ+PjJdKzEyPj4yXSsoYjw8Mik+PjJdO2M9YisxfDA7Yz0oYz4+PjApJTN8MD9jOmItMnwwO2lmKChjfDApPj0wKXtlPShjPj4+MCkvM3wwO2U9RFsoRFtEW2E+PjJdKzk2Pj4yXStKKGUsMTIpfDApKyhjLUooZSwzKTw8Mik+PjJdfWM6e2lmKChmfDApPT0tMSl7YnJlYWsgY31nPTA7Yz0oKGY+Pj4wKSUzfDA/LTE6MikrZnwwO2lmKChjfDApPDApe2JyZWFrIGN9ZD0oYz4+PjApLzN8MDtkPURbKERbRFthPj4yXSs5Nj4+Ml0rSihkLDEyKXwwKSsoYy1KKGQsMyk8PDIpPj4yXX1jPS0xO2lmKChkfDApIT0oZXwwKSl7YnJlYWsgYX1lPS0xO2Q6e2I9KChiPj4+MCklM3wwPy0xOjIpK2J8MDtpZigoYnwwKT49MCl7ZD0oYj4+PjApLzN8MDtkPURbKERbRFthPj4yXSs5Nj4+Ml0rSihkLDEyKXwwKSsoYi1KKGQsMyk8PDIpPj4yXTtpZihnKXticmVhayBifWJyZWFrIGR9ZD0tMTtpZighZyl7YnJlYWsgZH1icmVhayBifWI9ZisxfDA7Yj0oYj4+PjApJTN8MD9iOmYtMnwwO2lmKChifDApPDApe2JyZWFrIGJ9Yz1EW0RbYT4+Ml0rOTY+PjJdO2E9KGI+Pj4wKS8zfDA7ZT1EWyhjK0ooYSwxMil8MCkrKGItSihhLDMpPDwyKT4+Ml19Yz0oZHwwKT09KGV8MCk/ZjotMX1yZXR1cm4gY31mdW5jdGlvbiBkZShhLGIpe2E9YXwwO2I9YnwwO3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MDtkPURbYisxMj4+Ml07aD1kO2M9RFtiKzIwPj4yXTtlPWM7Zj1EW2IrMTY+PjJdO2c9Zis0fDA7Yz1nPj4+MDw0P2MrMXwwOmM7aT1EW2IrOD4+Ml07YTp7aWYoaT4+PjA8Zz4+PjAmKGN8MCk+PShkfDApfChjfDApPihkfDApKXticmVhayBhfWo9RFtiPj4yXTtkPWorZnwwO2Q9RVtkfDBdfEVbZCsxfDBdPDw4fChFW2QrMnwwXTw8MTZ8RVtkKzN8MF08PDI0KTtEW2IrMTY+PjJdPWc7RFtiKzIwPj4yXT1jO2M9ZTtlPWYrOHwwO2M9ZT4+PjA8OD9jKzF8MDpjO2Y9ZTtlPWM7aWYoZj4+PjA+aT4+PjAmKGN8MCk+PShofDApfChjfDApPihofDApKXticmVhayBhfWM9ZytqfDA7Yz1FW2N8MF18RVtjKzF8MF08PDh8KEVbYysyfDBdPDwxNnxFW2MrM3wwXTw8MjQpO0RbYisxNj4+Ml09ZjtEW2IrMjA+PjJdPWU7aWYoKGN8MCk8KGR8MCkpe2JyZWFrIGF9RFthKzE2Pj4yXT1jO0RbYSsxMj4+Ml09ZDtiPShjPj4zMSktKChkPj4zMSkrKGM+Pj4wPGQ+Pj4wKXwwKXwwO2M9Yy1kfDA7aWYoIWImYz4+PjA+MjE0NzQ4MzY0NnxiKXticmVhayBhfWs9MTtiPWMrMXwwO0RbYSsyMD4+Ml09YjtjPWI+Pj4xfDA7RFthKzI0Pj4yXT1jO0RbYSsyOD4+Ml09MC1jO2lmKGImMSl7YnJlYWsgYX1EW2ErMjQ+PjJdPWMtMX1yZXR1cm4ga3wwfWZ1bmN0aW9uIERnKGEpe2E9YXwwO3ZhciBiPTAsYz0wLGQ9MCxlPTAsZj0wLGc9MDtiPURbYSs0Pj4yXTtkPURbYj4+Ml07YTp7Yz1EW2ErMTI+PjJdO2M9RFtjKzU2Pj4yXS1EW2MrNTI+PjJdfDA7ZT1jPj4yO2I6e2lmKGU+Pj4wPD1EW2IrOD4+Ml0tZD4+Mj4+PjApe2JyZWFrIGJ9aWYoKGN8MCk8MCl7YnJlYWsgYX1mPURbYis0Pj4yXTtjPW5hKGMpO2c9YysoZTw8Mil8MDtlPWYtZHwwO2Y9ZStjfDA7aWYoKGV8MCk+MCl7b2EoYyxkLGUpfURbYis4Pj4yXT1nO0RbYis0Pj4yXT1mO0RbYj4+Ml09YztpZighZCl7YnJlYWsgYn1tYShkKX1lPWErOHwwO2I9RFthKzc2Pj4yXTtjOntpZihiKXtkPURbYj4+Ml07aWYoKGR8MCk9PURbYis0Pj4yXSl7cmV0dXJuIDF9Yj0wO3doaWxlKDEpe2M9amQoZSxEWyhiPDwyKStkPj4yXSk7aWYoIWMpe2JyZWFrIGN9Zj1EW2ErNzY+PjJdO2Q9RFtmPj4yXTtiPWIrMXwwO2lmKGI+Pj4wPERbZis0Pj4yXS1kPj4yPj4+MCl7Y29udGludWV9YnJlYWt9YnJlYWsgY31jPTE7YT1EW0RbYSsxMj4+Ml0rNjQ+PjJdO2E9RFthKzQ+PjJdLURbYT4+Ml0+PjI7aWYoYT4+PjA8Myl7YnJlYWsgY31hPShhPj4+MCkvM3wwO2I9MDt3aGlsZSgxKXtjPWpkKGUsSihiLDMpKTtpZighYyl7YnJlYWsgY31iPWIrMXwwO2lmKChhfDApIT0oYnwwKSl7Y29udGludWV9YnJlYWt9fXJldHVybiBjfDB9cmEoMTMyNik7VCgpfWZ1bmN0aW9uIEZmKGEsYixjLGQpe2E9YXwwO2I9YnwwO2M9Y3wwO2Q9ZHwwO3ZhciBlPTAsZj0wLGc9MDtmPSQrLTY0fDA7JD1mO2U9Zis4fDA7Q1tlKzM4Pj4xXT0wO0RbZT4+Ml09MDtEW2UrOD4+Ml09MDtEW2UrMTI+PjJdPTA7RFtlKzE2Pj4yXT0wO0RbZSsyMD4+Ml09MDtEW2UrMjQ+PjJdPTA7RFtlKzI4Pj4yXT0wO0JbZSsyOXwwXT0wO0JbZSszMHwwXT0wO0JbZSszMXwwXT0wO0JbZSszMnwwXT0wO0JbZSszM3wwXT0wO0JbZSszNHwwXT0wO0JbZSszNXwwXT0wO0JbZSszNnwwXT0wO0RbZSsxNj4+Ml09MDtEW2UrMjA+PjJdPTA7RFtlPj4yXT1iO0RbZSs4Pj4yXT1jO0RbZSsxMj4+Ml09MDtiPWYrNDh8MDtLZChiLGEsZSxkKTtEW2ErMjQ+PjJdPURbZis0OD4+Ml07Yz1hKzI0fDA7YTp7aWYoKGN8MCk9PShifDApKXticmVhayBhfWU9Zis0OHw0O2I9RVtmKzYzfDBdO2Q9Yjw8MjQ+PjI0O2c9YSsyOHwwO2lmKEJbZysxMXwwXT49MCl7aWYoKGR8MCk+PTApe2E9RFtlKzQ+PjJdO0RbZz4+Ml09RFtlPj4yXTtEW2crND4+Ml09YTtEW2crOD4+Ml09RFtlKzg+PjJdO2JyZWFrIGF9c2IoZyxEW2YrNTI+PjJdLERbZis1Nj4+Ml0pO2JyZWFrIGF9YT0oZHwwKTwwO3RiKGcsYT9EW2YrNTI+PjJdOmUsYT9EW2YrNTY+PjJdOmIpfWlmKEJbZis2M3wwXTwwKXttYShEW2YrNTI+PjJdKX0kPWYtIC02NHwwO3JldHVybiBjfDB9ZnVuY3Rpb24gSWcoYSl7YT1hfDA7dmFyIGI9MCxjPTAsZD0wLGU9MCxmPTAsZz0wO2I9RFthKzQ+PjJdO2Q9RFtiPj4yXTthOntjPURbYSsxMj4+Ml07Yz1EW2MrMjg+PjJdLURbYysyND4+Ml18MDtlPWM+PjI7Yjp7aWYoZT4+PjA8PURbYis4Pj4yXS1kPj4yPj4+MCl7YnJlYWsgYn1pZigoY3wwKTwwKXticmVhayBhfWY9RFtiKzQ+PjJdO2M9bmEoYyk7Zz1jKyhlPDwyKXwwO2U9Zi1kfDA7Zj1lK2N8MDtpZigoZXwwKT4wKXtvYShjLGQsZSl9RFtiKzg+PjJdPWc7RFtiKzQ+PjJdPWY7RFtiPj4yXT1jO2lmKCFkKXticmVhayBifW1hKGQpfWU9YSs4fDA7Yj1EW2ErNzY+PjJdO2M6e2lmKGIpe2Q9RFtiPj4yXTtpZigoZHwwKT09RFtiKzQ+PjJdKXtyZXR1cm4gMX1iPTA7d2hpbGUoMSl7Yz1rZChlLERbKGI8PDIpK2Q+PjJdKTtpZighYyl7YnJlYWsgY31mPURbYSs3Nj4+Ml07ZD1EW2Y+PjJdO2I9YisxfDA7aWYoYj4+PjA8RFtmKzQ+PjJdLWQ+PjI+Pj4wKXtjb250aW51ZX1icmVha31icmVhayBjfWM9MTthPURbYSsxMj4+Ml07YT1EW2ErND4+Ml0tRFthPj4yXT4+MjtpZihhPj4+MDwzKXticmVhayBjfWE9KGE+Pj4wKS8zfDA7Yj0wO3doaWxlKDEpe2M9a2QoZSxKKGIsMykpO2lmKCFjKXticmVhayBjfWI9YisxfDA7aWYoKGF8MCkhPShifDApKXtjb250aW51ZX1icmVha319cmV0dXJuIGN8MH1yYSgxMzI2KTtUKCl9ZnVuY3Rpb24gR2YoYSxiLGMsZCl7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7dmFyIGU9MCxmPTAsZz0wO2Q9JCstNjR8MDskPWQ7Q1tkKzQ2Pj4xXT0wO0RbZCs4Pj4yXT0wO0RbZCsxNj4+Ml09MDtEW2QrMjA+PjJdPTA7RFtkKzI0Pj4yXT0wO0RbZCsyOD4+Ml09MDtEW2QrMzI+PjJdPTA7RFtkKzM2Pj4yXT0wO0JbZCszN3wwXT0wO0JbZCszOHwwXT0wO0JbZCszOXwwXT0wO0JbZCs0MHwwXT0wO0JbZCs0MXwwXT0wO0JbZCs0MnwwXT0wO0JbZCs0M3wwXT0wO0JbZCs0NHwwXT0wO0RbZCsyND4+Ml09MDtEW2QrMjg+PjJdPTA7RFtkKzg+PjJdPWI7RFtkKzE2Pj4yXT1jO0RbZCsyMD4+Ml09MDtiPWQrNDh8MDtKZChiKTtEW2ErMjQ+PjJdPURbZCs0OD4+Ml07ZT1hKzI0fDA7YTp7aWYoKGJ8MCk9PShlfDApKXticmVhayBhfWI9ZCs0OHw0O2Y9RVtkKzYzfDBdO2M9Zjw8MjQ+PjI0O2E9YSsyOHwwO2lmKEJbYSsxMXwwXT49MCl7aWYoKGN8MCk+PTApe2M9RFtiKzQ+PjJdO0RbYT4+Ml09RFtiPj4yXTtEW2ErND4+Ml09YztEW2ErOD4+Ml09RFtiKzg+PjJdO2JyZWFrIGF9c2IoYSxEW2QrNTI+PjJdLERbZCs1Nj4+Ml0pO2JyZWFrIGF9Zz1hO2E9KGN8MCk8MDt0YihnLGE/RFtkKzUyPj4yXTpiLGE/RFtkKzU2Pj4yXTpmKX1pZihCW2QrNjN8MF08MCl7bWEoRFtkKzUyPj4yXSl9JD1kLSAtNjR8MDtyZXR1cm4gZXwwfWZ1bmN0aW9uIHdlKGEsYixjKXthPWF8MDtiPWJ8MDtjPWN8MDt2YXIgZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wLGs9MCxsPTAsbT0wLG49MDtkPURbYj4+Ml07Yj1EW2IrND4+Ml07Zj1EW0RbYSs4Pj4yXSs0MD4+Ml07bT1uYSgoZnwwKT49MD9mOi0xKTtnPTE7aD1iLWR8MDthOntpZigoaHwwKTw9MCl7YnJlYWsgYX1iPTA7Zz0wO2Q9MCtEW2MrMjA+PjJdfDA7aT1EW2MrMTY+PjJdO2U9ZitpfDA7ZD1lPj4+MDxmPj4+MD9kKzF8MDpkO2o9ZTtlPURbYysxMj4+Ml07aWYoaj4+PjA+R1tjKzg+PjJdJihlfDApPD0oZHwwKXwoZHwwKT4oZXwwKSl7YnJlYWsgYX1rPWg+PjI7Zz0oa3wwKT4xP2s6MTt3aGlsZSgxKXtiOntlPW9hKG0saStEW2M+PjJdfDAsZik7RFtjKzE2Pj4yXT1qO0RbYysyMD4+Ml09ZDtvYShEW0RbRFthKzg+PjJdKzY0Pj4yXT4+Ml0rYnwwLGUsZik7bD1sKzF8MDtpZigoZ3wwKT09KGx8MCkpe2JyZWFrIGJ9Yj1iK2Z8MDtkPW4rRFtjKzIwPj4yXXwwO2k9RFtjKzE2Pj4yXTtlPWYraXwwO2Q9ZT4+PjA8Zj4+PjA/ZCsxfDA6ZDtqPWU7aD1lO2U9RFtjKzEyPj4yXTtpZigoZXwwKT49KGR8MCkmR1tjKzg+PjJdPj1oPj4+MHwoZHwwKTwoZXwwKSl7Y29udGludWV9fWJyZWFrfWc9KGx8MCk+PShrfDApfW1hKG0pO3JldHVybiBnfDB9ZnVuY3Rpb24gbmUoYSxiKXthPWF8MDtiPWJ8MDthPTA7YTp7c3dpdGNoKGJ8MCl7Y2FzZSAwOmE9bmEoMjApO0RbYSsxMj4+Ml09LTE7RFthKzE2Pj4yXT0wO0RbYSs0Pj4yXT0wO0RbYSs4Pj4yXT0wO0RbYT4+Ml09MTk0ODtyZXR1cm4gYXwwO2Nhc2UgMTphPW5hKDI0KTtEW2ErMTI+PjJdPS0xO0RbYSsxNj4+Ml09MDtEW2ErND4+Ml09MDtEW2ErOD4+Ml09MDtEW2E+PjJdPTE5NDg7RFthKzIwPj4yXT0wO0RbYT4+Ml09MjE2NDtyZXR1cm4gYXwwO2Nhc2UgMjphPW5hKDQ4KTtEW2ErMTI+PjJdPS0xO0RbYSsxNj4+Ml09MDtEW2ErND4+Ml09MDtEW2ErOD4+Ml09MDtEW2E+PjJdPTE5NDg7RFthKzIwPj4yXT0wO0RbYT4+Ml09MjE2NDtEW2ErMzI+PjJdPTA7RFthKzM2Pj4yXT0wO0RbYSsyOD4+Ml09LTE7RFthKzI0Pj4yXT0xMTQwO0RbYT4+Ml09Nzk3NjtEW2ErNDA+PjJdPTA7RFthKzQ0Pj4yXT0wO3JldHVybiBhfDA7Y2FzZSAzOmE9bmEoMzIpO0RbYSsxMj4+Ml09LTE7RFthKzE2Pj4yXT0wO0RbYSs0Pj4yXT0wO0RbYSs4Pj4yXT0wO0RbYT4+Ml09MTk0ODtEW2ErMjA+PjJdPTA7RFthPj4yXT0yMTY0O0RbYSsyOD4+Ml09LTE7RFthKzI0Pj4yXT0xMDMyO0RbYT4+Ml09NTg0MDticmVhaztkZWZhdWx0OmJyZWFrIGF9fXJldHVybiBhfDB9ZnVuY3Rpb24gcmgoYSxiKXthPWF8MDtiPWJ8MDt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MDtEW2I+PjJdPTE7Zj1iKzh8MDtjPURbYis4Pj4yXTtkPURbYisxMj4+Ml0tY3wwO2lmKGQ+Pj4wPD00Mjk0OTY3MjkxKXtFYihmLGQrNHwwKTtjPURbZj4+Ml19Yz1jK2R8MDtkPURbYSs0Pj4yXTtCW2N8MF09ZDtCW2MrMXwwXT1kPj4+ODtCW2MrMnwwXT1kPj4+MTY7QltjKzN8MF09ZD4+PjI0O2M9RFthKzg+PjJdO2lmKChjfDApIT1EW2ErMTI+PjJdKXtkPTA7d2hpbGUoMSl7Zz0oZDw8MikrY3wwO2M9RFtiKzg+PjJdO2U9RFtiKzEyPj4yXS1jfDA7aWYoZT4+PjA8PTQyOTQ5NjcyOTEpe0ViKGYsZSs0fDApO2M9RFtmPj4yXX1jPWMrZXwwO2U9RFtnPj4yXTtCW2N8MF09ZTtCW2MrMXwwXT1lPj4+ODtCW2MrMnwwXT1lPj4+MTY7QltjKzN8MF09ZT4+PjI0O2Q9ZCsxfDA7Yz1EW2ErOD4+Ml07aWYoZD4+PjA8RFthKzEyPj4yXS1jPj4yPj4+MCl7Y29udGludWV9YnJlYWt9fWM9RFtiKzEyPj4yXTtiPURbYis4Pj4yXTtjPWMtYnwwO2lmKGM+Pj4wPD00Mjk0OTY3MjkxKXtFYihmLGMrNHwwKTtiPURbZj4+Ml19Yj1iK2N8MDthPURbYSsyMD4+Ml07QltifDBdPWE7QltiKzF8MF09YT4+Pjg7QltiKzJ8MF09YT4+PjE2O0JbYiszfDBdPWE+Pj4yNH1mdW5jdGlvbiBDZihhLGIsYyxkKXthPWF8MDtiPWJ8MDtjPWN8MDtkPWR8MDt2YXIgZT0wLGY9MCxnPTA7ZT0kLTMyfDA7JD1lO2E6e2I6e2Y9emEoYyk7aWYoZj4+PjA8NDI5NDk2NzI4MCl7Yzp7ZDp7aWYoZj4+PjA+PTExKXtnPWYrMTYmLTE2O2E9bmEoZyk7RFtlKzI0Pj4yXT1nfC0yMTQ3NDgzNjQ4O0RbZSsxNj4+Ml09YTtEW2UrMjA+PjJdPWY7YnJlYWsgZH1CW2UrMjd8MF09ZjthPWUrMTZ8MDtpZighZil7YnJlYWsgY319b2EoYSxjLGYpfUJbYStmfDBdPTA7Yz16YShkKTtpZihjPj4+MD49NDI5NDk2NzI4MCl7YnJlYWsgYn1lOntmOntpZihjPj4+MD49MTEpe2Y9YysxNiYtMTY7YT1uYShmKTtEW2UrOD4+Ml09ZnwtMjE0NzQ4MzY0ODtEW2U+PjJdPWE7RFtlKzQ+PjJdPWM7YnJlYWsgZn1CW2UrMTF8MF09YzthPWU7aWYoIWMpe2JyZWFrIGV9fW9hKGEsZCxjKX1CW2ErY3wwXT0wO2M9RFtiKzQ+PjJdO2E9LTE7Zzp7aWYoIWMpe2JyZWFrIGd9Yz1PYyhjLGUrMTZ8MCxlKTthPS0xO2lmKCFjKXticmVhayBnfWE9S2MoYixEW2MrMjQ+PjJdKX1pZihCW2UrMTF8MF08MCl7bWEoRFtlPj4yXSl9aWYoQltlKzI3fDBdPDApe21hKERbZSsxNj4+Ml0pfSQ9ZSszMnwwO2JyZWFrIGF9QWEoKTtUKCl9QWEoKTtUKCl9cmV0dXJuIGF8MH1mdW5jdGlvbiBFYShhLGIsYyl7dmFyIGQ9MCxlPTAsZj0wLGc9MDtlPSQtMTZ8MDskPWU7RFthKzQ+PjJdPTA7YTp7Yjp7aWYoIWIpe2JyZWFrIGJ9Zz1EW2ErOD4+Ml07ZD1nPDw1O2M6e2lmKGQ+Pj4wPj1iPj4+MCl7RFthKzQ+PjJdPWI7YnJlYWsgY31EW2UrOD4+Ml09MDtEW2U+PjJdPTA7RFtlKzQ+PjJdPTA7aWYoKGJ8MCk8MCl7YnJlYWsgYX1pZihkPj4+MDw9MTA3Mzc0MTgyMil7Zj1iKzMxJi0zMjtkPWc8PDY7ZD1kPj4+MDxmPj4+MD9mOmR9ZWxzZXtkPTIxNDc0ODM2NDd9VGEoZSxkKTtmPURbYT4+Ml07RFthPj4yXT1EW2U+PjJdO0RbZT4+Ml09ZjtkPURbYSs0Pj4yXTtEW2ErND4+Ml09YjtEW2UrND4+Ml09ZDtkPURbYSs4Pj4yXTtEW2ErOD4+Ml09RFtlKzg+PjJdO0RbZSs4Pj4yXT1kO2lmKCFmKXticmVhayBjfW1hKGYpfWY9Yj4+PjV8MDtkPWY8PDI7YT1EW2E+PjJdO2lmKEVbY3wwXSl7YT1wYShhLDI1NSxkKTtiPWImMzE7aWYoIWIpe2JyZWFrIGJ9YT1hKyhmPDwyKXwwO0RbYT4+Ml09RFthPj4yXXwtMT4+PjMyLWI7YnJlYWsgYn1hPXBhKGEsMCxkKTtiPWImMzE7aWYoIWIpe2JyZWFrIGJ9YT1hKyhmPDwyKXwwO0RbYT4+Ml09RFthPj4yXSYoLTE+Pj4zMi1iXi0xKX0kPWUrMTZ8MDtyZXR1cm59cWEoKTtUKCl9ZnVuY3Rpb24gaWUoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO3ZhciBkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MCxqPTAsaz0wLGw9MDtoPURbYysxMj4+Ml07ZD1oO2U9RFtjKzIwPj4yXTtpPURbYys4Pj4yXTtmPURbYysxNj4+Ml07YTp7aWYoKGR8MCk8PShlfDApJmk+Pj4wPD1mPj4+MHwoZHwwKTwoZXwwKSl7YnJlYWsgYX1qPURbYz4+Ml07az1CW2orZnwwXTtkPWU7Zz1mKzF8MDtkPWc/ZDpkKzF8MDtEW2MrMTY+PjJdPWc7RFtjKzIwPj4yXT1kO2I6e2lmKChrfDApPT0tMil7YnJlYWsgYn1pZigoZHwwKT49KGh8MCkmZz4+PjA+PWk+Pj4wfChkfDApPihofDApKXticmVhayBhfWQ9QltnK2p8MF07Zj1mKzJ8MDtlPWY+Pj4wPDI/ZSsxfDA6ZTtEW2MrMTY+PjJdPWY7RFtjKzIwPj4yXT1lO2lmKChkLTQmMjU1KT4+PjA8MjUxKXticmVhayBhfWU9YmFbRFtEW2E+PjJdKzQwPj4yXV0oYSxrLGQpfDA7ZD1EW2ErMjA+PjJdO0RbYSsyMD4+Ml09ZTtpZighZCl7YnJlYWsgYn1iYVtEW0RbZD4+Ml0rND4+Ml1dKGQpfWQ9RFthKzIwPj4yXTtpZihkKXtpZighKGJhW0RbRFthPj4yXSsyOD4+Ml1dKGEsZCl8MCkpe2JyZWFrIGF9fWw9YmFbRFtEW2E+PjJdKzM2Pj4yXV0oYSxiLGMpfDB9cmV0dXJuIGx8MH1mdW5jdGlvbiBBaChhLGIsYyxkLGUsZil7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7ZT1lfDA7Zj1mfDA7dmFyIGc9MCxoPTAsaT0wLGo9MCxrPTAsbD0wLG09MCxuPTA7Zj0kLTMyfDA7JD1mO2g9KGUmMTA3Mzc0MTgyMykhPShlfDApPy0xOmU8PDI7aD1wYShuYShoKSwwLGgpO2c9RFtiPj4yXTtpPURbYis0Pj4yXTtrPURbaCs0Pj4yXTtEW2YrMTY+PjJdPURbaD4+Ml07RFtmKzIwPj4yXT1rO0RbZis4Pj4yXT1nO0RbZisxMj4+Ml09aTtpPWErOHwwO0liKGYrMjR8MCxpLGYrMTZ8MCxmKzh8MCk7RFtjPj4yXT1EW2YrMjQ+PjJdO0RbYys0Pj4yXT1EW2YrMjg+PjJdO2lmKChkfDApPihlfDApKXtrPTAtZTw8MjthPWU7d2hpbGUoMSl7Zz1hPDwyO2o9ZytifDA7bT1EW2o+PjJdO2o9RFtqKzQ+PjJdO2c9YytnfDA7bD1nK2t8MDtuPURbbCs0Pj4yXTtEW2YrMTY+PjJdPURbbD4+Ml07RFtmKzIwPj4yXT1uO0RbZis4Pj4yXT1tO0RbZisxMj4+Ml09ajtJYihmKzI0fDAsaSxmKzE2fDAsZis4fDApO0RbZz4+Ml09RFtmKzI0Pj4yXTtEW2crND4+Ml09RFtmKzI4Pj4yXTthPWErZXwwO2lmKChkfDApPihhfDApKXtjb250aW51ZX1icmVha319bWEoaCk7JD1mKzMyfDA7cmV0dXJuIDF9ZnVuY3Rpb24gZmIoYSxiLGMpe3ZhciBkPTAsZT0wLGY9MCxnPTAsaD0wO2Y9Yy1ifDA7Zz1mPj4yO2Q9RFthKzg+PjJdO2U9RFthPj4yXTtpZihnPj4+MDw9ZC1lPj4yPj4+MCl7Zj1EW2ErND4+Ml07ZD1mLWV8MDtoPWQ+PjI7ZD1nPj4+MD5oPj4+MD9iK2R8MDpjO2lmKChkfDApIT0oYnwwKSl7d2hpbGUoMSl7RFtlPj4yXT1EW2I+PjJdO2U9ZSs0fDA7Yj1iKzR8MDtpZigoZHwwKSE9KGJ8MCkpe2NvbnRpbnVlfWJyZWFrfX1pZihnPj4+MD5oPj4+MCl7Yj1jLWR8MDtpZigoYnwwKT4wKXtmPW9hKGYsZCxiKStifDB9RFthKzQ+PjJdPWY7cmV0dXJufURbYSs0Pj4yXT1lO3JldHVybn1pZihlKXtEW2ErND4+Ml09ZTttYShlKTtEW2ErOD4+Ml09MDtEW2E+PjJdPTA7RFthKzQ+PjJdPTA7ZD0wfWE6e2lmKChmfDApPDApe2JyZWFrIGF9Yz1kPj4xO2M9ZD4+Mj4+PjA8NTM2ODcwOTExP2M+Pj4wPGc+Pj4wP2c6YzoxMDczNzQxODIzO2lmKGM+Pj4wPj0xMDczNzQxODI0KXticmVhayBhfWU9Yzw8MjtjPW5hKGUpO0RbYT4+Ml09YztEW2ErND4+Ml09YztEW2ErOD4+Ml09YytlO2lmKGYpe2M9b2EoYyxiLGYpK2Z8MH1EW2ErND4+Ml09YztyZXR1cm59cWEoKTtUKCl9ZnVuY3Rpb24gTWQoYSxiKXthPWF8MDtiPWJ8MDt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MDtlPURbYisxMj4+Ml07Yz1EW2IrMjA+PjJdO2Y9YztnPURbYisxNj4+Ml07ZD1nKzR8MDtjPWQ+Pj4wPDQ/YysxfDA6YztoPURbYis4Pj4yXTtpPWQ7YTp7aWYoaD4+PjA8ZD4+PjAmKGN8MCk+PShlfDApfChjfDApPihlfDApKXticmVhayBhfWQ9ZytEW2I+PjJdfDA7ZD1FW2R8MF18RVtkKzF8MF08PDh8KEVbZCsyfDBdPDwxNnxFW2QrM3wwXTw8MjQpO0RbYisxNj4+Ml09aTtEW2IrMjA+PjJdPWM7Yz1mO2Y9Zys4fDA7Yz1mPj4+MDw4P2MrMXwwOmM7aWYoZj4+PjA+aD4+PjAmKGN8MCk+PShlfDApfChjfDApPihlfDApKXticmVhayBhfURbYisxNj4+Ml09ZjtEW2IrMjA+PjJdPWM7aWYoIShkJjEpKXticmVhayBhfWM9TShkKV4zMTtpZihjLTMwPj4+MDw0Mjk0OTY3MjY3KXticmVhayBhfURbYSs4Pj4yXT1jKzE7ZT0tMjw8YztjPS0yLWV8MDtEW2ErMTY+PjJdPWM7RFthKzEyPj4yXT1lXi0xO0RbYSsyND4+Ml09KGN8MCkvMjtIW2ErMjA+PjJdPUsoMikvSyhjfDApO2o9S2EoYSs5NnwwLGIpfXJldHVybiBqfDB9ZnVuY3Rpb24gcmQoYSl7dmFyIGI9MCxjPTAsZD0wLGU9MDtjPTE7ZD1EW2ErMTQwPj4yXTthOntpZigoZHwwKTw9MCl7YnJlYWsgYX1iPWQ8PDQ7Yz1uYSgoZHwwKSE9KGQmMjY4NDM1NDU1KT8tMTpifDQpO0RbYz4+Ml09ZDtjPWMrNHwwO2Q9YytifDA7Yj1jO3doaWxlKDEpe0RbYj4+Ml09MDtEW2IrND4+Ml09MDtCW2IrNXwwXT0wO0JbYis2fDBdPTA7QltiKzd8MF09MDtCW2IrOHwwXT0wO0JbYis5fDBdPTA7QltiKzEwfDBdPTA7QltiKzExfDBdPTA7QltiKzEyfDBdPTA7Yj1iKzE2fDA7aWYoKGR8MCkhPShifDApKXtjb250aW51ZX1icmVha31lPURbYSsxMzY+PjJdO0RbYSsxMzY+PjJdPWM7aWYoZSl7ZD1lLTR8MDtjPURbZD4+Ml07aWYoYyl7Yj0oYzw8NCkrZXwwO3doaWxlKDEpe2I9Yi0xNnwwO2lmKChlfDApIT0oYnwwKSl7Y29udGludWV9YnJlYWt9fW1hKGQpfWM9MTtpZihEW2ErMTQwPj4yXTw9MCl7YnJlYWsgYX1iPTA7d2hpbGUoMSl7Yz1LYShEW2ErMTM2Pj4yXSsoYjw8NCl8MCxhKTtpZighYyl7YnJlYWsgYX1iPWIrMXwwO2lmKChifDApPERbYSsxNDA+PjJdKXtjb250aW51ZX1icmVha319cmV0dXJuIGN9ZnVuY3Rpb24gTWYoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO3ZhciBkPTAsZT0wLGY9MCxnPTA7YT0kLTMyfDA7JD1hO0RbYSsyND4+Ml09MDtEW2ErMjg+PjJdPTA7YTp7ZD16YShjKTtpZihkPj4+MDw0Mjk0OTY3MjgwKXtiOntjOntpZihkPj4+MD49MTEpe2Y9ZCsxNiYtMTY7ZT1uYShmKTtEW2ErMTY+PjJdPWZ8LTIxNDc0ODM2NDg7RFthKzg+PjJdPWU7RFthKzEyPj4yXT1kO2JyZWFrIGN9QlthKzE5fDBdPWQ7ZT1hKzh8MDtpZighZCl7YnJlYWsgYn19b2EoZSxjLGQpfUJbZCtlfDBdPTA7Yz1iKzR8MDtiPV9hKGIsYSs4fDApO2Q6e2lmKChjfDApPT0oYnwwKSl7YnJlYWsgZH1jPURbYiszMj4+Ml07Yj1EW2IrMjg+PjJdO2lmKChjLWJ8MCkhPTgpe2JyZWFrIGR9Yz1FW2IrNHwwXXxFW2IrNXwwXTw8OHwoRVtiKzZ8MF08PDE2fEVbYis3fDBdPDwyNCk7RFthKzI0Pj4yXT1FW2J8MF18RVtiKzF8MF08PDh8KEVbYisyfDBdPDwxNnxFW2IrM3wwXTw8MjQpO0RbYSsyOD4+Ml09Y31nPUlbYSsyND4+M107aWYoQlthKzE5fDBdPDApe21hKERbYSs4Pj4yXSl9JD1hKzMyfDA7YnJlYWsgYX1BYSgpO1QoKX1yZXR1cm4rZ31mdW5jdGlvbiBZYihhLGIpe3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wO2E6e2lmKERbYSs2ND4+Ml0pe2JyZWFrIGF9Yz1uYSgzMik7RFtjKzE2Pj4yXT0wO0RbYysyMD4+Ml09MDtEW2MrOD4+Ml09MDtEW2M+PjJdPTA7RFtjKzQ+PjJdPTA7RFtjKzI0Pj4yXT0wO0RbYysyOD4+Ml09MDtkPURbYSs2ND4+Ml07RFthKzY0Pj4yXT1jO2lmKCFkKXticmVhayBhfWM9RFtkPj4yXTtpZihjKXtEW2QrND4+Ml09YzttYShjKX1tYShkKX1lPURbYSs2ND4+Ml07Yz1EW2ErMjg+PjJdLTF8MDtpZihjPj4+MDw9MTApe2Q9RFsoYzw8MikrMTAxODA+PjJdfWVsc2V7ZD0tMX1jPUooZCxCW2ErMjR8MF0pO2Q9YztnPWM+PjMxO2U9aWQoZSwwLCRoKGMsZyxiLDApLGFhKTtpZihlKXtjPURbYSs2ND4+Ml07RFthPj4yXT1jO2Y9RFtjKzIwPj4yXTtEW2ErOD4+Ml09RFtjKzE2Pj4yXTtEW2ErMTI+PjJdPWY7Zj1EW2MrMjQ+PjJdO2M9RFtjKzI4Pj4yXTtEW2ErNDg+PjJdPTA7RFthKzUyPj4yXT0wO0RbYSs0MD4+Ml09ZDtEW2ErNDQ+PjJdPWc7RFthKzE2Pj4yXT1mO0RbYSsyMD4+Ml09YztEW2ErODA+PjJdPWJ9cmV0dXJuIGV9ZnVuY3Rpb24gRWgoYSxiKXthPWF8MDtiPWJ8MDt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wLGo9MDtmPURbYisxMj4+Ml07Yz1EW2IrMjA+PjJdO2U9YztnPURbYisxNj4+Ml07ZD1nKzR8MDtjPWQ+Pj4wPDQ/YysxfDA6YztoPURbYis4Pj4yXTtpPWQ7YTp7aWYoaD4+PjA8ZD4+PjAmKGN8MCk+PShmfDApfChjfDApPihmfDApKXticmVhayBhfWQ9ZytEW2I+PjJdfDA7ZD1FW2R8MF18RVtkKzF8MF08PDh8KEVbZCsyfDBdPDwxNnxFW2QrM3wwXTw8MjQpO0RbYisxNj4+Ml09aTtEW2IrMjA+PjJdPWM7Yz1lO2U9Zys4fDA7Yz1lPj4+MDw4P2MrMXwwOmM7aWYoZT4+PjA+aD4+PjAmKGN8MCk+PShmfDApfChjfDApPihmfDApKXticmVhayBhfURbYisxNj4+Ml09ZTtEW2IrMjA+PjJdPWM7aWYoIShkJjEpKXticmVhayBhfWI9TShkKV4zMTtpZihiLTMwPj4+MDw0Mjk0OTY3MjY3KXticmVhayBhfWo9MTtEW2ErOD4+Ml09YisxO2M9LTI8PGI7Yj0tMi1jfDA7RFthKzE2Pj4yXT1iO0RbYSsxMj4+Ml09Y14tMTtEW2ErMjQ+PjJdPShifDApLzI7SFthKzIwPj4yXT1LKDIpL0soYnwwKX1yZXR1cm4ganwwfWZ1bmN0aW9uIElkKGEsYixjKXt2YXIgZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wO2Y9YSs0fDA7YTp7YT1EW2ErND4+Ml07aWYoYSl7ZT1FW2MrMTF8MF07ZD1lPDwyND4+MjQ8MDtpPWQ/RFtjPj4yXTpjO2U9ZD9EW2MrND4+Ml06ZTt3aGlsZSgxKXtjPUVbYSsyN3wwXTtkPWM8PDI0Pj4yNDwwO2M9ZD9EW2ErMjA+PjJdOmM7aj1jPj4+MDxlPj4+MDtiOntjOntkOntlOntmOntoPWo/YzplO2c6e2lmKGgpe2c9YSsxNnwwO2Q9ZD9EW2c+PjJdOmc7Zz12YShpLGQsaCk7aWYoIWcpe2lmKGM+Pj4wPmU+Pj4wKXticmVhayBnfWJyZWFrIGZ9aWYoKGd8MCk+PTApe2JyZWFrIGZ9YnJlYWsgZ31pZihjPj4+MDw9ZT4+PjApe2JyZWFrIGV9fWM9RFthPj4yXTtpZihjKXticmVhayBifURbYj4+Ml09YTtyZXR1cm4gYX1jPXZhKGQsaSxoKTtpZihjKXticmVhayBkfX1pZihqKXticmVhayBjfWJyZWFrIGF9aWYoKGN8MCk+PTApe2JyZWFrIGF9fWY9YSs0fDA7Yz1EW2ErND4+Ml07aWYoIWMpe2JyZWFrIGF9YT1mfWY9YTthPWM7Y29udGludWV9fURbYj4+Ml09ZjtyZXR1cm4gZn1EW2I+PjJdPWE7cmV0dXJuIGZ9ZnVuY3Rpb24gWmIoYSxiKXt2YXIgYz0wO2M9RFtiKzQ+PjJdO0RbYT4+Ml09RFtiPj4yXTtEW2ErND4+Ml09YztjPURbYis2MD4+Ml07RFthKzU2Pj4yXT1EW2IrNTY+PjJdO0RbYSs2MD4+Ml09YztjPURbYis1Mj4+Ml07RFthKzQ4Pj4yXT1EW2IrNDg+PjJdO0RbYSs1Mj4+Ml09YztjPURbYis0ND4+Ml07RFthKzQwPj4yXT1EW2IrNDA+PjJdO0RbYSs0ND4+Ml09YztjPURbYiszNj4+Ml07RFthKzMyPj4yXT1EW2IrMzI+PjJdO0RbYSszNj4+Ml09YztjPURbYisyOD4+Ml07RFthKzI0Pj4yXT1EW2IrMjQ+PjJdO0RbYSsyOD4+Ml09YztjPURbYisyMD4+Ml07RFthKzE2Pj4yXT1EW2IrMTY+PjJdO0RbYSsyMD4+Ml09YztjPURbYisxMj4+Ml07RFthKzg+PjJdPURbYis4Pj4yXTtEW2ErMTI+PjJdPWM7RFthKzg4Pj4yXT0wO0RbYSs2ND4+Ml09MDtEW2ErNjg+PjJdPTA7RFthKzcyPj4yXT0wO0RbYSs3Nj4+Ml09MDtCW2ErNzd8MF09MDtCW2ErNzh8MF09MDtCW2ErNzl8MF09MDtCW2ErODB8MF09MDtCW2ErODF8MF09MDtCW2ErODJ8MF09MDtCW2ErODN8MF09MDtCW2ErODR8MF09MDtyZXR1cm4gYX1mdW5jdGlvbiBfYShhLGIpe3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wO2c9YSs0fDA7YT1EW2ErND4+Ml07YTp7Yjp7aWYoIWEpe2JyZWFrIGJ9ZD1FW2IrMTF8MF07Yz1kPDwyND4+MjQ8MDtpPWM/RFtiPj4yXTpiO2U9Yz9EW2IrND4+Ml06ZDtjPWc7d2hpbGUoMSl7Yj1FW2ErMjd8MF07aj1iPDwyND4+MjQ8MDtoPWo/RFthKzIwPj4yXTpiO2Y9aD4+PjA+ZT4+PjA7ZD1mP2U6aDtjOntpZihkKXtiPWErMTZ8MDtiPXZhKGo/RFtiPj4yXTpiLGksZCk7aWYoYil7YnJlYWsgY319Yj1lPj4+MD5oPj4+MD8tMTpmfWM9KGJ8MCk8MD9jOmE7YT1EWyhiPj4+MjkmNCkrYT4+Ml07aWYoYSl7Y29udGludWV9YnJlYWt9aWYoKGN8MCk9PShnfDApKXticmVhayBifWE9RVtjKzI3fDBdO2Y9YTw8MjQ+PjI0PDA7ZDp7ZD1mP0RbYysyMD4+Ml06YTtiPWQ+Pj4wPGU+Pj4wP2Q6ZTtpZihiKXthPWMrMTZ8MDthPXZhKGksZj9EW2E+PjJdOmEsYik7aWYoYSl7YnJlYWsgZH19aWYoZD4+PjA+ZT4+PjApe2JyZWFrIGJ9YnJlYWsgYX1pZigoYXwwKT49MCl7YnJlYWsgYX19Yz1nfXJldHVybiBjfWZ1bmN0aW9uIEVlKGEsYixjLGQsZSl7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7ZT1lfDA7aWYoSmEoYSxEW2IrOD4+Ml0sZSkpe2lmKCEoRFtiKzI4Pj4yXT09MXxEW2IrND4+Ml0hPShjfDApKSl7RFtiKzI4Pj4yXT1kfXJldHVybn1hOntpZihKYShhLERbYj4+Ml0sZSkpe2lmKCEoRFtiKzE2Pj4yXSE9KGN8MCkmRFtiKzIwPj4yXSE9KGN8MCkpKXtpZigoZHwwKSE9MSl7YnJlYWsgYX1EW2IrMzI+PjJdPTE7cmV0dXJufURbYiszMj4+Ml09ZDtiOntpZihEW2IrNDQ+PjJdPT00KXticmVhayBifUNbYis1Mj4+MV09MDthPURbYSs4Pj4yXTtiYVtEW0RbYT4+Ml0rMjA+PjJdXShhLGIsYyxjLDEsZSk7aWYoRVtiKzUzfDBdKXtEW2IrNDQ+PjJdPTM7aWYoIUVbYis1MnwwXSl7YnJlYWsgYn1icmVhayBhfURbYis0ND4+Ml09NH1EW2IrMjA+PjJdPWM7RFtiKzQwPj4yXT1EW2IrNDA+PjJdKzE7aWYoRFtiKzM2Pj4yXSE9MXxEW2IrMjQ+PjJdIT0yKXticmVhayBhfUJbYis1NHwwXT0xO3JldHVybn1hPURbYSs4Pj4yXTtiYVtEW0RbYT4+Ml0rMjQ+PjJdXShhLGIsYyxkLGUpfX1mdW5jdGlvbiBCZyhhLGIpe2E9YXwwO2I9YnwwO3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wLGg9MDtmPW5hKDY0KTtjPW5hKDEyKTtEW2MrOD4+Ml09RFtEW2ErND4+Ml0rODA+PjJdO0RbYz4+Ml09MWU0O0RbYys0Pj4yXT0wO2Y9d2MoZixjKTthOntiOntpZigoYnwwKTwwKXtjPWY7YnJlYWsgYn1oPWErOHwwO2M9RFthKzEyPj4yXTtlPURbYSs4Pj4yXTtnPWMtZT4+MjtjOntpZigoZ3wwKT4oYnwwKSl7YnJlYWsgY31kPWIrMXwwO2lmKGI+Pj4wPj1nPj4+MCl7T2IoaCxkLWd8MCk7YnJlYWsgY31pZihkPj4+MD49Zz4+PjApe2JyZWFrIGN9ZT1lKyhkPDwyKXwwO2lmKChlfDApIT0oY3wwKSl7d2hpbGUoMSl7Yz1jLTR8MDtkPURbYz4+Ml07RFtjPj4yXT0wO2lmKGQpe2JhW0RbRFtkPj4yXSs0Pj4yXV0oZCl9aWYoKGN8MCkhPShlfDApKXtjb250aW51ZX1icmVha319RFthKzEyPj4yXT1lfWE9RFtoPj4yXSsoYjw8Mil8MDtjPURbYT4+Ml07RFthPj4yXT1mO2lmKCFjKXticmVhayBhfX1iYVtEW0RbYz4+Ml0rND4+Ml1dKGMpfXJldHVybihiXi0xKT4+PjMxfDB9ZnVuY3Rpb24gVmMoYSxiLGMpe3ZhciBkPTAsZT0wLGY9MCxnPTA7YTp7aWYoISgoYnwwKTwwfChjfDApPDApKXtiOntpZihiPj4+MD4xNDMxNjU1NzY1KXticmVhayBifWQ9SihiLDMpO1ViKGEsZCwxMDI1Nik7VWIoYSsxMnwwLGQsMTAyNjApO2Q9RFthKzI0Pj4yXTtjOntpZihEW2ErMzI+PjJdLWQ+PjI+Pj4wPj1jPj4+MCl7YnJlYWsgY31pZihjPj4+MD49MTA3Mzc0MTgyNCl7YnJlYWsgYX1lPURbYSsyOD4+Ml07Zj1jPDwyO2M9bmEoZik7Zj1jK2Z8MDtlPWUtZHwwO2c9ZStjfDA7aWYoKGV8MCk+MCl7b2EoYyxkLGUpfURbYSszMj4+Ml09ZjtEW2ErMjg+PjJdPWc7RFthKzI0Pj4yXT1jO2lmKCFkKXticmVhayBjfW1hKGQpfURbYSs4MD4+Ml09MDtEW2ErODQ+PjJdPTA7Yz1EW2ErNzY+PjJdO0RbYSs3Nj4+Ml09MDtpZihjKXttYShjKX1EW2ErNjg+PjJdPTA7RFthKzcyPj4yXT0wO2M9YS0gLTY0fDA7YT1EW2M+PjJdO0RbYz4+Ml09MDtpZighYSl7YnJlYWsgYn1tYShhKX1kPWI+Pj4wPDE0MzE2NTU3NjZ9cmV0dXJuIGR9cmEoMTMyNik7VCgpfWZ1bmN0aW9uIGNiKGEpe3ZhciBiPTAsYz0wLGQ9MCxlPTAsZj0wO2Q9RFthKzg+PjJdO2E6e2lmKEVbZCs4NHwwXSl7YnJlYWsgYX1iPURbYSsxNj4+Ml07aWYoIWJ8IUVbYis4NHwwXSl7YnJlYWsgYX1jPURbZCs3Mj4+Ml07ZT1EW2QrNjg+PjJdO0JbYis4NHwwXT0wO2M9Yy1lPj4yO2Y9RFtiKzY4Pj4yXTtlPURbYis3Mj4+Ml0tZj4+MjtiOntpZihjPj4+MD5lPj4+MCl7eGEoYis2OHwwLGMtZXwwLDIwMzIpO2Q9RFthKzg+PjJdO2JyZWFrIGJ9aWYoYz4+PjA+PWU+Pj4wKXticmVhayBifURbYis3Mj4+Ml09ZisoYzw8Mil9Yj1FW2QrODR8MF07aWYoYil7YnJlYWsgYX1jPURbZCs2OD4+Ml07aWYoKGN8MCk9PURbZCs3Mj4+Ml0pe2JyZWFrIGF9ZT1EW0RbYSsxNj4+Ml0rNjg+PjJdO2lmKCFiKXtiPTA7d2hpbGUoMSl7Zj1iPDwyO0RbZitlPj4yXT1EW2MrZj4+Ml07Yj1iKzF8MDtjPURbZCs2OD4+Ml07aWYoYj4+PjA8RFtkKzcyPj4yXS1jPj4yPj4+MCl7Y29udGludWV9YnJlYWt9YnJlYWsgYX1EW2U+PjJdPTB9cmV0dXJuIERbYSsxNj4+Ml19ZnVuY3Rpb24gbmIoYSxiKXt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MCxoPTA7Yz1EW2ErND4+Ml07aWYoKGN8MCkhPURbYSs4Pj4yXSl7ZD1EW2IrND4+Ml07RFtjPj4yXT1EW2I+PjJdO0RbYys0Pj4yXT1kO0RbYys4Pj4yXT1EW2IrOD4+Ml07RFthKzQ+PjJdPWMrMTI7cmV0dXJufWE6e2Y9RFthPj4yXTtnPWMtZnwwO2Q9KGd8MCkvMTJ8MDtjPWQrMXwwO2lmKGM+Pj4wPDM1NzkxMzk0Mil7ZT1kPDwxO2U9ZD4+PjA8MTc4OTU2OTcwP2M+Pj4wPmU+Pj4wP2M6ZTozNTc5MTM5NDE7aWYoZSl7aWYoZT4+PjA+PTM1NzkxMzk0Mil7YnJlYWsgYX1jPW5hKEooZSwxMikpfWVsc2V7Yz0wfWQ9YytKKGQsMTIpfDA7aD1EW2IrND4+Ml07RFtkPj4yXT1EW2I+PjJdO0RbZCs0Pj4yXT1oO0RbZCs4Pj4yXT1EW2IrOD4+Ml07Yj1kK0ooKGd8MCkvLTEyfDAsMTIpfDA7aWYoKGd8MCk+MCl7b2EoYixmLGcpfURbYSs4Pj4yXT1jK0ooZSwxMik7RFthKzQ+PjJdPWQrMTI7RFthPj4yXT1iO2lmKGYpe21hKGYpfXJldHVybn1xYSgpO1QoKX1yYSgxMzI2KTtUKCl9ZnVuY3Rpb24gRGQoYSxiKXthPWF8MDtiPWJ8MDt2YXIgYz0wLGQ9MCxlPTAsZj0wO2E6e2M9RFthKzIxNj4+Ml07aWYoKGN8MCk9PURbYSsyMjA+PjJdKXticmVhayBhfXdoaWxlKDEpe2I6e2M9RFtKKGUsMTQ0KStjPj4yXTtpZigoY3wwKTwwKXticmVhayBifWQ9RFthKzQ+PjJdO2Y9RFtkKzg+PjJdO2lmKChjfDApPj1EW2QrMTI+PjJdLWY+PjIpe2JyZWFrIGJ9ZD0wO2M9RFsoYzw8MikrZj4+Ml07aWYoKGJhW0RbRFtjPj4yXSsyND4+Ml1dKGMpfDApPD0wKXticmVhayBifXdoaWxlKDEpe2lmKChiYVtEW0RbYz4+Ml0rMjA+PjJdXShjLGQpfDApIT0oYnwwKSl7ZD1kKzF8MDtpZigoYmFbRFtEW2M+PjJdKzI0Pj4yXV0oYyl8MCk+KGR8MCkpe2NvbnRpbnVlfWJyZWFrIGJ9YnJlYWt9YT1EW2ErMjE2Pj4yXStKKGUsMTQ0KXwwO2U9RVthKzEwMHwwXT9hKzR8MDowO2JyZWFrIGF9ZT1lKzF8MDtjPURbYSsyMTY+PjJdO2lmKGU+Pj4wPChEW2ErMjIwPj4yXS1jfDApLzE0ND4+PjApe2NvbnRpbnVlfWJyZWFrfXJldHVybiAwfXJldHVybiBlfDB9ZnVuY3Rpb24gc2UoYSxiKXthPWF8MDtiPWJ8MDt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MDtjPURbYSs2MD4+Ml07YTp7aWYoIWMpe2JyZWFrIGF9RFtjKzQ+PjJdPWErNDg7aWYoIShiYVtEW0RbYz4+Ml0rMTI+PjJdXShjKXwwKSl7YnJlYWsgYX1iOntjPWJhW0RbRFthPj4yXSsyND4+Ml1dKGEpfDA7aWYoKGN8MCk8PTApe2JyZWFrIGJ9d2hpbGUoMSl7Yzp7Zj1EWyhiYVtEW0RbYT4+Ml0rMjg+PjJdXShhKXwwKSs0Pj4yXTtnPWJhW0RbRFthPj4yXSsyMD4+Ml1dKGEsZCl8MDtlPURbYSs2MD4+Ml07aWYoIShiYVtEW0RbZT4+Ml0rOD4+Ml1dKGUsRFtEW2YrOD4+Ml0rKGc8PDIpPj4yXSl8MCkpe2JyZWFrIGN9ZD1kKzF8MDtpZigoY3wwKSE9KGR8MCkpe2NvbnRpbnVlfWJyZWFrIGJ9YnJlYWt9cmV0dXJuIDB9ZD0wO2lmKCEoYmFbRFtEW2E+PjJdKzM2Pj4yXV0oYSxiKXwwKSl7YnJlYWsgYX1pZighKGJhW0RbRFthPj4yXSs0MD4+Ml1dKGEsYil8MCkpe2JyZWFrIGF9ZD1iYVtEW0RbYT4+Ml0rNDQ+PjJdXShhKXwwfXJldHVybiBkfDB9ZnVuY3Rpb24gRmMoYSxiLGMsZCxlLGYsZyl7dmFyIGg9MCxpPTAsaj0wO2g9JC0xNnwwOyQ9aDtpZigoYl4tMSktMTc+Pj4wPj1jPj4+MCl7aWYoRVthKzExfDBdPj4+N3wwKXtqPURbYT4+Ml19ZWxzZXtqPWF9YTp7aWYoYj4+PjA8MjE0NzQ4MzYyMyl7RFtoKzg+PjJdPWI8PDE7RFtoKzEyPj4yXT1iK2M7Yz0kLTE2fDA7JD1jOyQ9YysxNnwwO2M9aCs4fDA7aT1oKzEyfDA7Yz1EWyhHW2k+PjJdPEdbYz4+Ml0/YzppKT4+Ml07aWYoYz4+PjA+PTExKXtpPWMrMTYmLTE2O2M9aS0xfDA7Yz0oY3wwKT09MTE/aTpjfWVsc2V7Yz0xMH1icmVhayBhfWM9LTE4fWk9YysxfDA7Yz1uYShpKTtpZihmKXtYYShjLGcsZil9ZD1kLWV8MDtpZihkKXtYYShjK2Z8MCxlK2p8MCxkKX1pZigoYnwwKSE9MTApe21hKGopfURbYT4+Ml09YztEW2ErOD4+Ml09aXwtMjE0NzQ4MzY0ODtiPWE7YT1kK2Z8MDtEW2IrND4+Ml09YTtCW2grN3wwXT0wO0JbYStjfDBdPUVbaCs3fDBdOyQ9aCsxNnwwO3JldHVybn1BYSgpO1QoKX1mdW5jdGlvbiBDZChhLGIpe2E9YXwwO2I9YnwwO3ZhciBjPTAsZD0wLGU9MCxmPTA7Yz1EW2ErMjE2Pj4yXTtpZigoY3wwKSE9RFthKzIyMD4+Ml0pe3doaWxlKDEpe2E6e2M9RFtKKGUsMTQ0KStjPj4yXTtpZigoY3wwKTwwKXticmVhayBhfWQ9RFthKzQ+PjJdO2Y9RFtkKzg+PjJdO2lmKChjfDApPj1EW2QrMTI+PjJdLWY+PjIpe2JyZWFrIGF9ZD0wO2M9RFsoYzw8MikrZj4+Ml07aWYoKGJhW0RbRFtjPj4yXSsyND4+Ml1dKGMpfDApPD0wKXticmVhayBhfXdoaWxlKDEpe2lmKChiYVtEW0RbYz4+Ml0rMjA+PjJdXShjLGQpfDApIT0oYnwwKSl7ZD1kKzF8MDtpZigoYmFbRFtEW2M+PjJdKzI0Pj4yXV0oYyl8MCk+KGR8MCkpe2NvbnRpbnVlfWJyZWFrIGF9YnJlYWt9cmV0dXJuKERbYSsyMTY+PjJdK0ooZSwxNDQpfDApKzEwNHwwfWU9ZSsxfDA7Yz1EW2ErMjE2Pj4yXTtpZihlPj4+MDwoRFthKzIyMD4+Ml0tY3wwKS8xNDQ+Pj4wKXtjb250aW51ZX1icmVha319cmV0dXJuIGErMTg0fDB9ZnVuY3Rpb24gVGEoYSxiKXt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wO2M9JC0xNnwwOyQ9YzthOntiOntpZihEW2ErOD4+Ml08PDU+Pj4wPj1iPj4+MCl7YnJlYWsgYn1EW2MrOD4+Ml09MDtEW2M+PjJdPTA7RFtjKzQ+PjJdPTA7aWYoKGJ8MCk8MCl7YnJlYWsgYX1nPShiLTE+Pj41fDApKzF8MDtlPW5hKGc8PDIpO0RbYys4Pj4yXT1nO0RbYz4+Ml09ZTtmPURbYT4+Ml07Yj1EW2ErND4+Ml07RFtjKzQ+PjJdPWI7RFsoKGI+Pj4wPDMzPzA6Yi0xPj4+NXwwKTw8MikrZT4+Ml09MDtjOntpZigoYnwwKTw9MCl7YnJlYWsgY31oPWI+Pj41fDA7ZD1oPDwyO2k9TmEoZSxmLGQpO2I9Yi0oaDw8NSl8MDtpZigoYnwwKTw9MCl7YnJlYWsgY31kPWQraXwwO2I9LTE+Pj4zMi1ifDA7RFtkPj4yXT1EW2Q+PjJdJihiXi0xKXxiJkRbKGg8PDIpK2Y+PjJdfURbYSs4Pj4yXT1nO0RbYT4+Ml09ZTtpZighZil7YnJlYWsgYn1tYShmKX0kPWMrMTZ8MDtyZXR1cm59cWEoKTtUKCl9ZnVuY3Rpb24geGUoYSxiKXthPWF8MDtiPWJ8MDt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MDtpZigoYmFbRFtEW2I+PjJdKzIwPj4yXV0oYil8MCk8PTApe3JldHVybiAxfWE6e3doaWxlKDEpe2Q9TGMoRFtEW2ErND4+Ml0rND4+Ml0sYmFbRFtEW2I+PjJdKzI0Pj4yXV0oYixlKXwwKTtpZigoZHwwKT09LTEpe2JyZWFrIGF9Zj1EW2ErND4+Ml07Yz0wO2I6e2lmKChkfDApPDApe2JyZWFrIGJ9Zz1EW2YrND4+Ml07aWYoKGR8MCk+PURbZysxMj4+Ml0tRFtnKzg+PjJdPj4yKXticmVhayBifWM9RFtEW2YrOD4+Ml0rKERbRFtmKzIwPj4yXSsoZDw8Mik+PjJdPDwyKT4+Ml07Yz1iYVtEW0RbYz4+Ml0rMzI+PjJdXShjLGQpfDB9aWYoIWMpe2JyZWFrIGF9aWYoIShiYVtEW0RbYj4+Ml0rMjg+PjJdXShiLGMpfDApKXticmVhayBhfWU9ZSsxfDA7aWYoKGJhW0RbRFtiPj4yXSsyMD4+Ml1dKGIpfDApPihlfDApKXtjb250aW51ZX1icmVha31yZXR1cm4gMX1yZXR1cm4gMH1mdW5jdGlvbiBEZihhLGIsYyl7YT1hfDA7Yj1ifDA7Yz1jfDA7dmFyIGQ9MCxlPTAsZj0wO2E9JC0zMnwwOyQ9YTthOntkPXphKGMpO2lmKGQ+Pj4wPDQyOTQ5NjcyODApe2I6e2M6e2lmKGQ+Pj4wPj0xMSl7Zj1kKzE2Ji0xNjtlPW5hKGYpO0RbYSsyND4+Ml09ZnwtMjE0NzQ4MzY0ODtEW2ErMTY+PjJdPWU7RFthKzIwPj4yXT1kO2JyZWFrIGN9QlthKzI3fDBdPWQ7ZT1hKzE2fDA7aWYoIWQpe2JyZWFrIGJ9fW9hKGUsYyxkKX1CW2QrZXwwXT0wO0JbYSs0fDBdPTA7RFthPj4yXT0xNzAxNjY3MTgyO0JbYSsxMXwwXT00O2Q9RFtiKzQ+PjJdO2M9LTE7ZDp7aWYoIWQpe2JyZWFrIGR9ZD1PYyhkLGEsYSsxNnwwKTtjPS0xO2lmKCFkKXticmVhayBkfWM9S2MoYixEW2QrMjQ+PjJdKX1iPWM7aWYoQlthKzExfDBdPDApe21hKERbYT4+Ml0pfWlmKEJbYSsyN3wwXTwwKXttYShEW2ErMTY+PjJdKX0kPWErMzJ8MDticmVhayBhfUFhKCk7VCgpfXJldHVybiBifDB9ZnVuY3Rpb24gT2YoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO3ZhciBkPTAsZT0wLGY9MDtkPSQtMTZ8MDskPWQ7RFtkKzEyPj4yXT0wO2E6e2U9emEoYyk7aWYoZT4+PjA8NDI5NDk2NzI4MCl7Yjp7Yzp7aWYoZT4+PjA+PTExKXtmPWUrMTYmLTE2O2E9bmEoZik7RFtkKzg+PjJdPWZ8LTIxNDc0ODM2NDg7RFtkPj4yXT1hO0RbZCs0Pj4yXT1lO2JyZWFrIGN9QltkKzExfDBdPWU7YT1kO2lmKCFlKXticmVhayBifX1vYShhLGMsZSl9QlthK2V8MF09MDthPV9hKGIsZCk7ZDp7aWYoKGF8MCk9PShiKzR8MCkpe2JyZWFrIGR9Yj1EW2ErMzI+PjJdO2E9RFthKzI4Pj4yXTtpZigoYi1hfDApIT00KXticmVhayBkfURbZCsxMj4+Ml09RVthfDBdfEVbYSsxfDBdPDw4fChFW2ErMnwwXTw8MTZ8RVthKzN8MF08PDI0KX1hPURbZCsxMj4+Ml07aWYoQltkKzExfDBdPDApe21hKERbZD4+Ml0pfSQ9ZCsxNnwwO2JyZWFrIGF9QWEoKTtUKCl9cmV0dXJuIGF8MH1mdW5jdGlvbiB1YihhKXthPWF8MDt2YXIgYj0wLGM9MCxkPTA7RFthPj4yXT0xMDMzMjtiPURbYSs2OD4+Ml07aWYoYil7RFthKzcyPj4yXT1iO21hKGIpfWI9RFthKzU2Pj4yXTtpZihiKXtEW2ErNjA+PjJdPWI7bWEoYil9Yj1EW2ErNDQ+PjJdO2lmKGIpe0RbYSs0OD4+Ml09YjttYShiKX1iPURbYSszMj4+Ml07aWYoYil7RFthKzM2Pj4yXT1iO21hKGIpfWI9RFthKzIwPj4yXTtpZihiKXtEW2ErMjQ+PjJdPWI7bWEoYil9ZD1EW2ErOD4+Ml07aWYoZCl7Yz1EW2ErMTI+PjJdO2lmKChjfDApPT0oZHwwKSl7Yj1kfWVsc2V7d2hpbGUoMSl7Yz1jLTR8MDtiPURbYz4+Ml07RFtjPj4yXT0wO2lmKGIpe3lhKGIpfWlmKChkfDApIT0oY3wwKSl7Y29udGludWV9YnJlYWt9Yj1EW2ErOD4+Ml19RFthKzEyPj4yXT1kO21hKGIpfWI9RFthKzQ+PjJdO0RbYSs0Pj4yXT0wO2lmKGIpe2RjKGIpfXJldHVybiBhfDB9ZnVuY3Rpb24gc2EoYSxiKXt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wO2U9RFthKzg+PjJdO2M9RFthKzQ+PjJdO2lmKGUtYz4+Mj4+PjA+PWI+Pj4wKXtpZihiKXtiPWI8PDI7Yz1wYShjLDAsYikrYnwwfURbYSs0Pj4yXT1jO3JldHVybn1hOntmPURbYT4+Ml07Zz1jLWZ8MDtoPWc+PjI7ZD1oK2J8MDtpZihkPj4+MDwxMDczNzQxODI0KXtjPTA7ZT1lLWZ8MDtpPWU+PjE7ZD1lPj4yPj4+MDw1MzY4NzA5MTE/ZD4+PjA+aT4+PjA/ZDppOjEwNzM3NDE4MjM7aWYoZCl7aWYoZD4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGF9Yz1uYShkPDwyKX1iPWI8PDI7Yj1wYSgoaDw8MikrY3wwLDAsYikrYnwwO2lmKChnfDApPjApe29hKGMsZixnKX1EW2ErOD4+Ml09KGQ8PDIpK2M7RFthKzQ+PjJdPWI7RFthPj4yXT1jO2lmKGYpe21hKGYpfXJldHVybn1xYSgpO1QoKX1yYSgxMzI2KTtUKCl9ZnVuY3Rpb24gYmIoYSxiKXt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MCxoPTAsaT0wO2U9RFthKzg+PjJdO2M9RFthKzQ+PjJdO2lmKGUtYz4+Mz4+PjA+PWI+Pj4wKXtpZihiKXtiPWI8PDM7Yz1wYShjLDAsYikrYnwwfURbYSs0Pj4yXT1jO3JldHVybn1hOntmPURbYT4+Ml07Zz1jLWZ8MDtoPWc+PjM7ZD1oK2J8MDtpZihkPj4+MDw1MzY4NzA5MTIpe2M9MDtlPWUtZnwwO2k9ZT4+MjtkPWU+PjM+Pj4wPDI2ODQzNTQ1NT9kPj4+MD5pPj4+MD9kOmk6NTM2ODcwOTExO2lmKGQpe2lmKGQ+Pj4wPj01MzY4NzA5MTIpe2JyZWFrIGF9Yz1uYShkPDwzKX1iPWI8PDM7Yj1wYSgoaDw8MykrY3wwLDAsYikrYnwwO2lmKChnfDApPjApe29hKGMsZixnKX1EW2ErOD4+Ml09KGQ8PDMpK2M7RFthKzQ+PjJdPWI7RFthPj4yXT1jO2lmKGYpe21hKGYpfXJldHVybn1xYSgpO1QoKX1yYSgxMzI2KTtUKCl9ZnVuY3Rpb24gUWEoYSxiKXt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MCxoPTA7ZT1EW2E+PjJdO2E9RFtlKzQ+PjJdO2M9RFtlKzg+PjJdO2lmKGE+Pj4wPGM+Pj4wKXtEW2E+PjJdPURbYj4+Ml07RFtlKzQ+PjJdPWErNDtyZXR1cm59YTp7Zj1EW2U+PjJdO2c9YS1mfDA7ZD1nPj4yO2E9ZCsxfDA7aWYoYT4+PjA8MTA3Mzc0MTgyNCl7aD1kPDwyO2M9Yy1mfDA7ZD1jPj4xO2M9Yz4+Mj4+PjA8NTM2ODcwOTExP2E+Pj4wPmQ+Pj4wP2E6ZDoxMDczNzQxODIzO2lmKGMpe2lmKGM+Pj4wPj0xMDczNzQxODI0KXticmVhayBhfWE9bmEoYzw8Mil9ZWxzZXthPTB9ZD1oK2F8MDtEW2Q+PjJdPURbYj4+Ml07aWYoKGd8MCk+MCl7b2EoYSxmLGcpfURbZSs4Pj4yXT1hKyhjPDwyKTtEW2UrND4+Ml09ZCs0O0RbZT4+Ml09YTtpZihmKXttYShmKX1yZXR1cm59cWEoKTtUKCl9cmEoMTMyNik7VCgpfWZ1bmN0aW9uIGdiKGEsYil7dmFyIGM9MCxkPTAsZT0wLGY9MCxnPTAsaD0wO2E6e2M9RFthKzQ+PjJdO2U9RFthPj4yXTtmPWMtZXwwO2I6e2lmKGY+Pj4wPGI+Pj4wKXtnPWItZnwwO2Q9RFthKzg+PjJdO2lmKGc+Pj4wPD1kLWM+Pj4wKXtpZihnKXtjPXBhKGMsMCxnKStnfDB9RFthKzQ+PjJdPWM7cmV0dXJufWlmKChifDApPDApe2JyZWFrIGF9Yz0wO2Q9ZC1lfDA7aD1kPDwxO2Q9ZD4+PjA8MTA3Mzc0MTgyMz9iPj4+MD5oPj4+MD9iOmg6MjE0NzQ4MzY0NztpZihkKXtjPW5hKGQpfXBhKGMrZnwwLDAsZyk7aWYoKGZ8MCk+MCl7b2EoYyxlLGYpfURbYSs4Pj4yXT1jK2Q7RFthKzQ+PjJdPWIrYztEW2E+PjJdPWM7aWYoIWUpe2JyZWFrIGJ9bWEoZSk7cmV0dXJufWlmKGI+Pj4wPj1mPj4+MCl7YnJlYWsgYn1EW2ErND4+Ml09YitlfXJldHVybn1xYSgpO1QoKX1mdW5jdGlvbiBtZShhKXthPWF8MDt2YXIgYj0wLGM9MCxkPTA7RFthPj4yXT0yMDQ0O2I9RFthKzYwPj4yXTtEW2ErNjA+PjJdPTA7aWYoYil7YmFbRFtEW2I+PjJdKzQ+PjJdXShiKX1iPURbYSs0OD4+Ml07aWYoYil7RFthKzUyPj4yXT1iO21hKGIpfWQ9RFthKzM2Pj4yXTtpZihkKXtjPURbYSs0MD4+Ml07aWYoKGN8MCk9PShkfDApKXtiPWR9ZWxzZXt3aGlsZSgxKXtjPWMtNHwwO2I9RFtjPj4yXTtEW2M+PjJdPTA7aWYoYil7YmFbRFtEW2I+PjJdKzQ+PjJdXShiKX1pZigoZHwwKSE9KGN8MCkpe2NvbnRpbnVlfWJyZWFrfWI9RFthKzM2Pj4yXX1EW2ErNDA+PjJdPWQ7bWEoYil9RFthPj4yXT0xODA0O2I9RFthKzE2Pj4yXTtpZihiKXtEW2ErMjA+PjJdPWI7bWEoYil9Yj1EW2ErND4+Ml07aWYoYil7RFthKzg+PjJdPWI7bWEoYil9cmV0dXJuIGF8MH1mdW5jdGlvbiBsZShhKXthPWF8MDt2YXIgYj0wLGM9MCxkPTA7RFthPj4yXT0yMDQ0O2I9RFthKzYwPj4yXTtEW2ErNjA+PjJdPTA7aWYoYil7YmFbRFtEW2I+PjJdKzQ+PjJdXShiKX1iPURbYSs0OD4+Ml07aWYoYil7RFthKzUyPj4yXT1iO21hKGIpfWQ9RFthKzM2Pj4yXTtpZihkKXtjPURbYSs0MD4+Ml07aWYoKGN8MCk9PShkfDApKXtiPWR9ZWxzZXt3aGlsZSgxKXtjPWMtNHwwO2I9RFtjPj4yXTtEW2M+PjJdPTA7aWYoYil7YmFbRFtEW2I+PjJdKzQ+PjJdXShiKX1pZigoZHwwKSE9KGN8MCkpe2NvbnRpbnVlfWJyZWFrfWI9RFthKzM2Pj4yXX1EW2ErNDA+PjJdPWQ7bWEoYil9RFthPj4yXT0xODA0O2I9RFthKzE2Pj4yXTtpZihiKXtEW2ErMjA+PjJdPWI7bWEoYil9Yj1EW2ErND4+Ml07aWYoYil7RFthKzg+PjJdPWI7bWEoYil9bWEoYSl9ZnVuY3Rpb24gZ2QoYSxiKXt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MCxoPTA7ZT1EW2ErOD4+Ml07Yz1EW2ErND4+Ml07aWYoZS1jPj4xPj4+MD49Yj4+PjApe2lmKGIpe2I9Yjw8MTtjPXBhKGMsMCxiKStifDB9RFthKzQ+PjJdPWM7cmV0dXJufWE6e2Y9RFthPj4yXTtnPWMtZnwwO2g9Zz4+MTtkPWgrYnwwO2lmKChkfDApPj0wKXtjPTA7ZT1lLWZ8MDtkPWU+PjE+Pj4wPDEwNzM3NDE4MjM/ZD4+PjA+ZT4+PjA/ZDplOjIxNDc0ODM2NDc7aWYoZCl7aWYoKGR8MCk8MCl7YnJlYWsgYX1jPW5hKGQ8PDEpfWI9Yjw8MTtiPXBhKChoPDwxKStjfDAsMCxiKStifDA7aWYoKGd8MCk+MCl7b2EoYyxmLGcpfURbYSs4Pj4yXT0oZDw8MSkrYztEW2ErND4+Ml09YjtEW2E+PjJdPWM7aWYoZil7bWEoZil9cmV0dXJufXFhKCk7VCgpfXJhKDEzMjYpO1QoKX1mdW5jdGlvbiBkaChhLGIpe2E9YXwwO2I9YnwwO3ZhciBjPTAsZD0wLGU9MCxmPTA7ZD0kLTE2fDA7JD1kO2U9RFthKzQ+PjJdO2E6e2lmKChlfDApPT0tMSl7YnJlYWsgYX1jPURbYisyMD4+Ml07aWYoISFEW2IrMTY+PjJdJihjfDApPj0wfChjfDApPjApe2JyZWFrIGF9cmIoYixEW2IrND4+Ml0sRFthKzg+PjJdLERbYSsxMj4+Ml0pO2M9RFtiKzIwPj4yXTtpZighIURbYisxNj4+Ml0mKGN8MCk+PTB8KGN8MCk+MCl7YnJlYWsgYX1jPWErMjB8MDtyYihiLERbYis0Pj4yXSxjLGMrNHwwKTtjPURbYisyMD4+Ml07Zj1EW2IrMTY+PjJdO0JbZCsxNXwwXT1EW2ErND4+Ml07aWYoISFmJihjfDApPj0wfChjfDApPjApe2JyZWFrIGF9cmIoYixEW2IrND4+Ml0sZCsxNXwwLGQrMTZ8MCl9JD1kKzE2fDA7cmV0dXJuKGV8MCkhPS0xfDB9DQpmdW5jdGlvbiB5ZyhhKXthPWF8MDt2YXIgYj0wLGM9MCxkPTAsZT0wLGY9MDthOntiPURbYSs4Pj4yXTtiOntpZigoYnwwKTwwKXticmVhayBifWM9RFthKzQ+PjJdO2U9RFtjPj4yXTtkPURbYys0Pj4yXS1lPj4yO2M6e2lmKGQ+Pj4wPGI+Pj4wKXtzYShjLGItZHwwKTtmPURbYSs4Pj4yXTticmVhayBjfWY9YjtpZihiPj4+MD49ZD4+PjApe2JyZWFrIGN9RFtjKzQ+PjJdPWUrKGI8PDIpO2Y9Yn1kPWY7aWYoKGR8MCk8PTApe2JyZWFrIGJ9YT1EW2ErND4+Ml07Yz1EW2E+PjJdO2U9RFthKzQ+PjJdLWM+PjI7YT0wO3doaWxlKDEpe2lmKChhfDApPT0oZXwwKSl7YnJlYWsgYX1EW2MrKGE8PDIpPj4yXT1hO2E9YSsxfDA7aWYoKGR8MCkhPShhfDApKXtjb250aW51ZX1icmVha319cmV0dXJuKGJeLTEpPj4+MzF8MH11YSgpO1QoKX1mdW5jdGlvbiBSYShhKXt2YXIgYj0wLGM9MCxkPTAsZT0wLGY9MDtiPURbYSs0Pj4yXTtpZigoYnwwKSE9RFthKzg+PjJdKXtEW2I+PjJdPURbMjA4OF07RFthKzQ+PjJdPWIrNDtyZXR1cm59YTp7ZT1EW2E+PjJdO2Y9Yi1lfDA7ZD1mPj4yO2I9ZCsxfDA7aWYoYj4+PjA8MTA3Mzc0MTgyNCl7Yz1mPj4xO2M9ZD4+PjA8NTM2ODcwOTExP2I+Pj4wPmM+Pj4wP2I6YzoxMDczNzQxODIzO2lmKGMpe2lmKGM+Pj4wPj0xMDczNzQxODI0KXticmVhayBhfWI9bmEoYzw8Mil9ZWxzZXtiPTB9ZD1iKyhkPDwyKXwwO0RbZD4+Ml09RFsyMDg4XTtpZigoZnwwKT4wKXtvYShiLGUsZil9RFthKzg+PjJdPWIrKGM8PDIpO0RbYSs0Pj4yXT1kKzQ7RFthPj4yXT1iO2lmKGUpe21hKGUpfXJldHVybn1xYSgpO1QoKX1yYSgxMzI2KTtUKCl9ZnVuY3Rpb24gZ2YoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO3ZhciBkPTAsZT0wLGY9MCxnPTA7ZD0kLTE2fDA7JD1kO0tkKGQsYSxiLGMpO0RbYSsyND4+Ml09RFtkPj4yXTtlPWErMjR8MDthOntpZigoZXwwKT09KGR8MCkpe2JyZWFrIGF9Yj1kfDQ7Zj1FW2QrMTV8MF07Yz1mPDwyND4+MjQ7YT1hKzI4fDA7aWYoQlthKzExfDBdPj0wKXtpZigoY3wwKT49MCl7Yz1EW2IrND4+Ml07RFthPj4yXT1EW2I+PjJdO0RbYSs0Pj4yXT1jO0RbYSs4Pj4yXT1EW2IrOD4+Ml07YnJlYWsgYX1zYihhLERbZCs0Pj4yXSxEW2QrOD4+Ml0pO2JyZWFrIGF9Zz1hO2E9KGN8MCk8MDt0YihnLGE/RFtkKzQ+PjJdOmIsYT9EW2QrOD4+Ml06Zil9aWYoQltkKzE1fDBdPDApe21hKERbZCs0Pj4yXSl9JD1kKzE2fDA7cmV0dXJuIGV8MH1mdW5jdGlvbiB0ZihhLGIsYyxkKXthPWF8MDtiPWJ8MDtjPWN8MDtkPWR8MDt2YXIgZT0wLGY9MCxnPTA7YT0kLTMyfDA7JD1hO2U9QltiKzI0fDBdO2Y9RFsyNTYzXTtEW2ErMjQ+PjJdPURbMjU2Ml07RFthKzI4Pj4yXT1mO2Y9RFsyNTYxXTtEW2ErMTY+PjJdPURbMjU2MF07RFthKzIwPj4yXT1mO2E6e2I6e2M9bWIoYixjLGUsYSsxNnwwKTtpZihjKXtEW2ErOD4+Ml09MDtEW2E+PjJdPTA7RFthKzQ+PjJdPTA7Yj0wO2lmKGUpe2lmKChlfDApPDApe2JyZWFrIGJ9ZT1lPDwyO2I9bmEoZSk7Zz1vYShiLGErMTZ8MCxlKStlfDB9ZT1EW2Q+PjJdO2lmKGUpe0RbZCs0Pj4yXT1lO21hKGUpfURbZCs4Pj4yXT1nO0RbZCs0Pj4yXT1nO0RbZD4+Ml09Yn0kPWErMzJ8MDticmVhayBhfXFhKCk7VCgpfXJldHVybiBjfDB9ZnVuY3Rpb24gaGYoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO3ZhciBkPTAsZT0wLGY9MCxnPTA7Yj0kLTE2fDA7JD1iO0pkKGIpO0RbYSsyND4+Ml09RFtiPj4yXTtlPWErMjR8MDthOntpZigoZXwwKT09KGJ8MCkpe2JyZWFrIGF9Yz1ifDQ7Zj1FW2IrMTV8MF07ZD1mPDwyND4+MjQ7YT1hKzI4fDA7aWYoQlthKzExfDBdPj0wKXtpZigoZHwwKT49MCl7ZD1EW2MrND4+Ml07RFthPj4yXT1EW2M+PjJdO0RbYSs0Pj4yXT1kO0RbYSs4Pj4yXT1EW2MrOD4+Ml07YnJlYWsgYX1zYihhLERbYis0Pj4yXSxEW2IrOD4+Ml0pO2JyZWFrIGF9Zz1hO2E9KGR8MCk8MDt0YihnLGE/RFtiKzQ+PjJdOmMsYT9EW2IrOD4+Ml06Zil9aWYoQltiKzE1fDBdPDApe21hKERbYis0Pj4yXSl9JD1iKzE2fDA7cmV0dXJuIGV8MH1mdW5jdGlvbiBHYihhLGIpe3ZhciBjPTAsZD0wLGU9MCxmPTA7ZD1EW2ErMTI+PjJdO2M9RFthKzE2Pj4yXS1kPj4yO2E6e2lmKGM+Pj4wPGI+Pj4wKXtzYShhKzEyfDAsYi1jfDApO2JyZWFrIGF9aWYoYj4+PjA+PWM+Pj4wKXticmVhayBhfURbYSsxNj4+Ml09ZCsoYjw8Mil9Yjp7Yz1EW2E+PjJdO2M6e2lmKERbYSs4Pj4yXS1jPj4yPj4+MD49Yj4+PjApe2JyZWFrIGN9aWYoYj4+PjA+PTEwNzM3NDE4MjQpe2JyZWFrIGJ9ZD1EW2ErND4+Ml07ZT1iPDwyO2I9bmEoZSk7ZT1iK2V8MDtkPWQtY3wwO2Y9ZCtifDA7aWYoKGR8MCk+MCl7b2EoYixjLGQpfURbYSs4Pj4yXT1lO0RbYSs0Pj4yXT1mO0RbYT4+Ml09YjtpZighYyl7YnJlYWsgY31tYShjKX1yZXR1cm59cmEoMTMyNik7VCgpfWZ1bmN0aW9uIExmKGEsYixjKXthPWF8MDtiPWJ8MDtjPWN8MDt2YXIgZD0wLGU9MCxmPTAsZz0wO2Q9JC0xNnwwOyQ9ZDthOntlPXphKGMpO2lmKGU+Pj4wPDQyOTQ5NjcyODApe2I6e2M6e2lmKGU+Pj4wPj0xMSl7Zz1lKzE2Ji0xNjtmPW5hKGcpO0RbZCs4Pj4yXT1nfC0yMTQ3NDgzNjQ4O0RbZD4+Ml09ZjtEW2QrND4+Ml09ZTticmVhayBjfUJbZCsxMXwwXT1lO2Y9ZDtpZighZSl7YnJlYWsgYn19b2EoZixjLGUpfUJbZStmfDBdPTA7Zj1hKzE2fDA7Yz1OYyhiLGQsZik7Yj1CW2ErMjd8MF07YT1EW2ErMTY+PjJdO2lmKEJbZCsxMXwwXTwwKXttYShEW2Q+PjJdKX0kPWQrMTZ8MDthPWM/KGJ8MCk8MD9hOmY6MDticmVhayBhfUFhKCk7VCgpfXJldHVybiBhfDB9ZnVuY3Rpb24gbWcoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO3ZhciBkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MDtkPSQrLTY0fDA7JD1kO2U9YmFbRFtEW2E+PjJdKzQ0Pj4yXV0oYSxiKXwwO2E9YmFbRFtEW2E+PjJdKzQwPj4yXV0oYSxiKXwwO2Y9bGIoZCk7Zz1EW2IrNTY+PjJdO2g9ZTw8MjQ+PjI0O2k9YTthPWEtMXwwO2lmKGE+Pj4wPD0xMCl7YT1EWyhhPDwyKSsxMDE4MD4+Ml19ZWxzZXthPS0xfWE9SihhLGUpOyRiKGYsZyxoLGksMCxhLGE+PjMxKTthPVpiKG5hKDk2KSxmKTtZYihhLGMpO0JbYSs4NHwwXT0xO0RbYSs3Mj4+Ml09RFthKzY4Pj4yXTtEW2ErNjA+PjJdPURbYis2MD4+Ml07JD1kLSAtNjR8MDtyZXR1cm4gYXwwfWZ1bmN0aW9uIGhkKGEsYixjKXt2YXIgZD0wLGU9MCxmPTAsZz0wO2E6e2lmKGE+Pj4wPjEwKXticmVhayBhfWQ9RFtjKzIwPj4yXTtlPURbYysxMj4+Ml07Zj1EW2MrMTY+PjJdO2lmKChkfDApPj0oZXwwKSZmPj4+MD49R1tjKzg+PjJdfChkfDApPihlfDApKXticmVhayBhfWU9QltmK0RbYz4+Ml18MF07Zj1mKzF8MDtkPWY/ZDpkKzF8MDtEW2MrMTY+PjJdPWY7RFtjKzIwPj4yXT1kO2I6e2lmKChlfDApPDApe2lmKCFoZChhKzF8MCxiLGMpKXticmVhayBhfWM9RFtiPj4yXTthPURbYis0Pj4yXTw8N3xjPj4+MjU7Yz1lJjEyN3xjPDw3O2JyZWFrIGJ9YT0wO2M9ZSYyNTV9RFtiPj4yXT1jO0RbYis0Pj4yXT1hO2c9MX1yZXR1cm4gZ31mdW5jdGlvbiBPYShhLGIsYyl7dmFyIGQ9MCxlPTAsZj0wLGc9MDthOntpZihhPj4+MD4xMCl7YnJlYWsgYX1kPURbYysyMD4+Ml07ZT1EW2MrMTI+PjJdO2Y9RFtjKzE2Pj4yXTtpZigoZHwwKT49KGV8MCkmZj4+PjA+PUdbYys4Pj4yXXwoZHwwKT4oZXwwKSl7YnJlYWsgYX1lPUJbZitEW2M+PjJdfDBdO2Y9ZisxfDA7ZD1mP2Q6ZCsxfDA7RFtjKzE2Pj4yXT1mO0RbYysyMD4+Ml09ZDtiOntpZigoZXwwKTwwKXtpZighT2EoYSsxfDAsYixjKSl7YnJlYWsgYX1jPURbYj4+Ml07YT1EW2IrND4+Ml08PDd8Yz4+PjI1O2M9ZSYxMjd8Yzw8NzticmVhayBifWE9MDtjPWUmMjU1fURbYj4+Ml09YztEW2IrND4+Ml09YTtnPTF9cmV0dXJuIGd9ZnVuY3Rpb24gRGUoYSxiLGMsZCxlKXthPWF8MDtiPWJ8MDtjPWN8MDtkPWR8MDtlPWV8MDtpZihKYShhLERbYis4Pj4yXSxlKSl7aWYoIShEW2IrMjg+PjJdPT0xfERbYis0Pj4yXSE9KGN8MCkpKXtEW2IrMjg+PjJdPWR9cmV0dXJufWE6e2lmKCFKYShhLERbYj4+Ml0sZSkpe2JyZWFrIGF9aWYoIShEW2IrMTY+PjJdIT0oY3wwKSZEW2IrMjA+PjJdIT0oY3wwKSkpe2lmKChkfDApIT0xKXticmVhayBhfURbYiszMj4+Ml09MTtyZXR1cm59RFtiKzIwPj4yXT1jO0RbYiszMj4+Ml09ZDtEW2IrNDA+PjJdPURbYis0MD4+Ml0rMTtpZighKERbYiszNj4+Ml0hPTF8RFtiKzI0Pj4yXSE9Mikpe0JbYis1NHwwXT0xfURbYis0ND4+Ml09NH19ZnVuY3Rpb24gdmYoYSxiLGMsZCl7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7dmFyIGU9MCxmPTAsZz0wO2E6e2lmKEdbYis4MD4+Ml0+NjU1MzUpe2JyZWFrIGF9Zj1EW2IrOTY+PjJdO2I9RFtiKzEwMD4+Ml0tZnwwO2E9KGJ8MCkvMTJ8MDtlPUooYSw2KTtnPShlfDApPT0oY3wwKTtpZighYnwoY3wwKSE9KGV8MCkpe2JyZWFrIGF9Zz0xO2U9YT4+PjA+MT9hOjE7YT0wO3doaWxlKDEpe2I9SihhLDYpK2R8MDtjPUooYSwxMikrZnwwO0NbYj4+MV09RFtjPj4yXTtDW2IrMj4+MV09RFtjKzQ+PjJdO0NbYis0Pj4xXT1EW2MrOD4+Ml07YT1hKzF8MDtpZigoZXwwKSE9KGF8MCkpe2NvbnRpbnVlfWJyZWFrfX1yZXR1cm4gZ3wwfWZ1bmN0aW9uIGtoKGEpe2E9YXwwO3ZhciBiPTAsYz0wLGQ9MDtEW2E+PjJdPTgyMDQ7Yj1EW2ErNDg+PjJdO0RbYSs0OD4+Ml09MDtpZihiKXtiYVtEW0RbYj4+Ml0rND4+Ml1dKGIpfURbYT4+Ml09MTAwNjQ7Yj1EW2ErMjA+PjJdO2lmKGIpe0RbYSsyND4+Ml09YjttYShiKX1kPURbYSs4Pj4yXTtpZihkKXtjPURbYSsxMj4+Ml07aWYoKGN8MCk9PShkfDApKXtiPWR9ZWxzZXt3aGlsZSgxKXtjPWMtNHwwO2I9RFtjPj4yXTtEW2M+PjJdPTA7aWYoYil7YmFbRFtEW2I+PjJdKzQ+PjJdXShiKX1pZigoZHwwKSE9KGN8MCkpe2NvbnRpbnVlfWJyZWFrfWI9RFthKzg+PjJdfURbYSsxMj4+Ml09ZDttYShiKX1yZXR1cm4gYXwwfWZ1bmN0aW9uIGpoKGEpe2E9YXwwO3ZhciBiPTAsYz0wLGQ9MDtEW2E+PjJdPTgyMDQ7Yj1EW2ErNDg+PjJdO0RbYSs0OD4+Ml09MDtpZihiKXtiYVtEW0RbYj4+Ml0rND4+Ml1dKGIpfURbYT4+Ml09MTAwNjQ7Yj1EW2ErMjA+PjJdO2lmKGIpe0RbYSsyND4+Ml09YjttYShiKX1kPURbYSs4Pj4yXTtpZihkKXtjPURbYSsxMj4+Ml07aWYoKGN8MCk9PShkfDApKXtiPWR9ZWxzZXt3aGlsZSgxKXtjPWMtNHwwO2I9RFtjPj4yXTtEW2M+PjJdPTA7aWYoYil7YmFbRFtEW2I+PjJdKzQ+PjJdXShiKX1pZigoZHwwKSE9KGN8MCkpe2NvbnRpbnVlfWJyZWFrfWI9RFthKzg+PjJdfURbYSsxMj4+Ml09ZDttYShiKX1tYShhKX1mdW5jdGlvbiBCYyhhLGIsYyxkKXtCW2ErNTN8MF09MTthOntpZihEW2ErND4+Ml0hPShjfDApKXticmVhayBhfUJbYSs1MnwwXT0xO2M9RFthKzE2Pj4yXTtiOntpZighYyl7RFthKzM2Pj4yXT0xO0RbYSsyND4+Ml09ZDtEW2ErMTY+PjJdPWI7aWYoRFthKzQ4Pj4yXSE9MSl7YnJlYWsgYX1pZigoZHwwKT09MSl7YnJlYWsgYn1icmVhayBhfWlmKChifDApPT0oY3wwKSl7Yz1EW2ErMjQ+PjJdO2lmKChjfDApPT0yKXtEW2ErMjQ+PjJdPWQ7Yz1kfWlmKERbYSs0OD4+Ml0hPTEpe2JyZWFrIGF9aWYoKGN8MCk9PTEpe2JyZWFrIGJ9YnJlYWsgYX1EW2ErMzY+PjJdPURbYSszNj4+Ml0rMX1CW2ErNTR8MF09MX19ZnVuY3Rpb24gcmUoYSxiKXthPWF8MDtiPWJ8MDt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MDtlPWJhW0RbRFthPj4yXSsyND4+Ml1dKGEpfDA7Yz0xO2E6e2lmKChlfDApPD0wKXticmVhayBhfWQ9RFtEW2ErMzY+PjJdPj4yXTtmPWErNDh8MDtjPTA7aWYoIShiYVtEW0RbZD4+Ml0rMTY+PjJdXShkLGYsYil8MCkpe2JyZWFrIGF9ZD0xO3doaWxlKDEpe2M9ZDtpZigoZXwwKSE9KGN8MCkpe2Q9YysxfDA7Zz1EW0RbYSszNj4+Ml0rKGM8PDIpPj4yXTtpZihiYVtEW0RbZz4+Ml0rMTY+PjJdXShnLGYsYil8MCl7Y29udGludWV9fWJyZWFrfWM9KGN8MCk+PShlfDApfXJldHVybiBjfDB9ZnVuY3Rpb24gcGUoYSxiKXthPWF8MDtiPWJ8MDt2YXIgYz0wLGQ9MCxlPTAsZj0wLGc9MDtlPWJhW0RbRFthPj4yXSsyND4+Ml1dKGEpfDA7Yz0xO2E6e2lmKChlfDApPD0wKXticmVhayBhfWQ9RFtEW2ErMzY+PjJdPj4yXTtmPWErNDh8MDtjPTA7aWYoIShiYVtEW0RbZD4+Ml0rMjA+PjJdXShkLGYsYil8MCkpe2JyZWFrIGF9ZD0xO3doaWxlKDEpe2M9ZDtpZigoZXwwKSE9KGN8MCkpe2Q9YysxfDA7Zz1EW0RbYSszNj4+Ml0rKGM8PDIpPj4yXTtpZihiYVtEW0RbZz4+Ml0rMjA+PjJdXShnLGYsYil8MCl7Y29udGludWV9fWJyZWFrfWM9KGN8MCk+PShlfDApfXJldHVybiBjfDB9ZnVuY3Rpb24gQWQoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO3ZhciBkPTAsZT0wO2Q9JC0xNnwwOyQ9ZDtEW2ErND4+Ml09YjtiPURbYis2ND4+Ml07ZT1EW2I+PjJdO2I9RFtiKzQ+PjJdO0JbZCsxNXwwXT0wO0VhKGErMjR8MCwoYi1lPj4yPj4+MCkvM3wwLGQrMTV8MCk7Yj1EW2ErND4+Ml07ZT1EW2IrNTY+PjJdO2I9RFtiKzUyPj4yXTtCW2QrMTR8MF09MDtFYShhKzM2fDAsZS1iPj4yLGQrMTR8MCk7Yj1EW2MrMTI+PjJdO0RbYSsxNj4+Ml09RFtjKzg+PjJdO0RbYSsyMD4+Ml09YjtiPURbYys0Pj4yXTtEW2ErOD4+Ml09RFtjPj4yXTtEW2ErMTI+PjJdPWI7JD1kKzE2fDB9ZnVuY3Rpb24gVWMoYSl7RFthPj4yXT0wO0RbYSs0Pj4yXT0wO0RbYSsyOD4+Ml09MDtEW2ErMzI+PjJdPTA7QlthKzI0fDBdPTE7RFthKzE2Pj4yXT0wO0RbYSsyMD4+Ml09MDtEW2ErOD4+Ml09MDtEW2ErMTI+PjJdPTA7RFthKzM2Pj4yXT0wO0RbYSs0MD4+Ml09MDtEW2ErNDQ+PjJdPTA7RFthKzQ4Pj4yXT0wO0RbYSs1Mj4+Ml09MDtEW2ErNTY+PjJdPTA7RFthKzYwPj4yXT0wO0RbYSs2ND4+Ml09MDtEW2ErNzI+PjJdPTA7RFthKzc2Pj4yXT0wO0RbYSs4MD4+Ml09MDtEW2ErODQ+PjJdPTA7RFthKzg4Pj4yXT0wO0RbYSs5Mj4+Ml09MDtEW2ErNjg+PjJdPWF9ZnVuY3Rpb24gWGcoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO3ZhciBkPTAsZT0wO2Q9JC0xNnwwOyQ9ZDtEW2ErND4+Ml09YjtlPURbYj4+Ml07Yj1EW2IrND4+Ml07QltkKzE1fDBdPTA7RWEoYSsyNHwwLChiLWU+PjI+Pj4wKS8zfDAsZCsxNXwwKTtiPURbYSs0Pj4yXTtlPURbYisyOD4+Ml07Yj1EW2IrMjQ+PjJdO0JbZCsxNHwwXT0wO0VhKGErMzZ8MCxlLWI+PjIsZCsxNHwwKTtiPURbYysxMj4+Ml07RFthKzE2Pj4yXT1EW2MrOD4+Ml07RFthKzIwPj4yXT1iO2I9RFtjKzQ+PjJdO0RbYSs4Pj4yXT1EW2M+PjJdO0RbYSsxMj4+Ml09YjskPWQrMTZ8MH1mdW5jdGlvbiB1ZChhLGIpe3ZhciBjPTAsZD0wLGU9MCxmPTAsZz0wO0RbYSsxNDQ+PjJdPWI7Yz1EWyhiYVtEW0RbYj4+Ml0rMzI+PjJdXShiKXwwKSszMj4+Ml07ZT1EW2M+PjJdK0RbYysxNj4+Ml18MDtkPURbKGJhW0RbRFtiPj4yXSszMj4+Ml1dKGIpfDApKzMyPj4yXTtjPURbZCs4Pj4yXTtkPURbZCsxNj4+Ml07Yz1jLWR8MDtmPWEsZz1GW0RbKGJhW0RbRFtiPj4yXSszMj4+Ml1dKGIpfDApKzMyPj4yXSszOD4+MV0sQ1tmKzM4Pj4xXT1nO0RbYT4+Ml09ZTtEW2ErMTY+PjJdPTA7RFthKzIwPj4yXT0wO0RbYSs4Pj4yXT1jO0RbYSsxMj4+Ml09MH1mdW5jdGlvbiBXYyhhKXt2YXIgYj0wO0RbYT4+Ml09MDtEW2ErND4+Ml09MDtEW2ErNTY+PjJdPTA7RFthKzQ4Pj4yXT0wO0RbYSs1Mj4+Ml09MDtEW2ErNDA+PjJdPTA7RFthKzQ0Pj4yXT0wO0RbYSszMj4+Ml09MDtEW2ErMzY+PjJdPTA7RFthKzI0Pj4yXT0wO0RbYSsyOD4+Ml09MDtEW2ErMTY+PjJdPTA7RFthKzIwPj4yXT0wO0RbYSs4Pj4yXT0wO0RbYSsxMj4+Ml09MDtiPWEtIC02NHwwO0RbYj4+Ml09MDtEW2IrND4+Ml09MDtEW2ErNzI+PjJdPTA7RFthKzc2Pj4yXT0wO0RbYSs4MD4+Ml09MDtEW2ErODQ+PjJdPTA7RFthKzYwPj4yXT1hfWZ1bmN0aW9uIGpiKGEsYixjKXt2YXIgZD0wLGU9MCxmPTAsZz0wLGg9MDthOntpZihhPj4+MD41KXticmVhayBhfWY9RFtjKzIwPj4yXTtkPWY7Zz1EW2MrMTI+PjJdO2U9RFtjKzE2Pj4yXTtpZigoZHwwKT49KGd8MCkmZT4+PjA+PUdbYys4Pj4yXXwoZHwwKT4oZ3wwKSl7YnJlYWsgYX1kPUVbZStEW2M+PjJdfDBdO2U9ZSsxfDA7Zj1lP2Y6ZisxfDA7RFtjKzE2Pj4yXT1lO0RbYysyMD4+Ml09ZjtpZihkJjEyOCl7aWYoIWpiKGErMXwwLGIsYykpe2JyZWFrIGF9ZD1kJjEyN3xEW2I+PjJdPDw3fURbYj4+Ml09ZDtoPTF9cmV0dXJuIGh9ZnVuY3Rpb24gZWIoYSxiLGMpe3ZhciBkPTAsZT0wLGY9MCxnPTAsaD0wO2E6e2lmKGE+Pj4wPjUpe2JyZWFrIGF9Zj1EW2MrMjA+PjJdO2Q9ZjtnPURbYysxMj4+Ml07ZT1EW2MrMTY+PjJdO2lmKChkfDApPj0oZ3wwKSZlPj4+MD49R1tjKzg+PjJdfChkfDApPihnfDApKXticmVhayBhfWQ9RVtlK0RbYz4+Ml18MF07ZT1lKzF8MDtmPWU/ZjpmKzF8MDtEW2MrMTY+PjJdPWU7RFtjKzIwPj4yXT1mO2lmKGQmMTI4KXtpZighZWIoYSsxfDAsYixjKSl7YnJlYWsgYX1kPWQmMTI3fERbYj4+Ml08PDd9RFtiPj4yXT1kO2g9MX1yZXR1cm4gaH1mdW5jdGlvbiBWYihhLGIsYyl7dmFyIGQ9MCxlPTAsZj0wLGc9MCxoPTA7YTp7aWYoYT4+PjA+NSl7YnJlYWsgYX1mPURbYysyMD4+Ml07ZD1mO2c9RFtjKzEyPj4yXTtlPURbYysxNj4+Ml07aWYoKGR8MCk+PShnfDApJmU+Pj4wPj1HW2MrOD4+Ml18KGR8MCk+KGd8MCkpe2JyZWFrIGF9ZD1FW2UrRFtjPj4yXXwwXTtlPWUrMXwwO2Y9ZT9mOmYrMXwwO0RbYysxNj4+Ml09ZTtEW2MrMjA+PjJdPWY7aWYoZCYxMjgpe2lmKCFWYihhKzF8MCxiLGMpKXticmVhayBhfWQ9ZCYxMjd8RFtiPj4yXTw8N31EW2I+PjJdPWQ7aD0xfXJldHVybiBofWZ1bmN0aW9uIFNhKGEsYixjKXt2YXIgZD0wLGU9MCxmPTAsZz0wLGg9MDthOntpZihhPj4+MD41KXticmVhayBhfWY9RFtjKzIwPj4yXTtkPWY7Zz1EW2MrMTI+PjJdO2U9RFtjKzE2Pj4yXTtpZigoZHwwKT49KGd8MCkmZT4+PjA+PUdbYys4Pj4yXXwoZHwwKT4oZ3wwKSl7YnJlYWsgYX1kPUVbZStEW2M+PjJdfDBdO2U9ZSsxfDA7Zj1lP2Y6ZisxfDA7RFtjKzE2Pj4yXT1lO0RbYysyMD4+Ml09ZjtpZihkJjEyOCl7aWYoIVNhKGErMXwwLGIsYykpe2JyZWFrIGF9ZD1kJjEyN3xEW2I+PjJdPDw3fURbYj4+Ml09ZDtoPTF9cmV0dXJuIGh9ZnVuY3Rpb24gTGQoYSxiLGMpe3ZhciBkPTAsZT0wLGY9MCxnPTAsaD0wO2E6e2lmKGE+Pj4wPjUpe2JyZWFrIGF9Zj1EW2MrMjA+PjJdO2Q9ZjtnPURbYysxMj4+Ml07ZT1EW2MrMTY+PjJdO2lmKChkfDApPj0oZ3wwKSZlPj4+MD49R1tjKzg+PjJdfChkfDApPihnfDApKXticmVhayBhfWQ9RVtlK0RbYz4+Ml18MF07ZT1lKzF8MDtmPWU/ZjpmKzF8MDtEW2MrMTY+PjJdPWU7RFtjKzIwPj4yXT1mO2lmKGQmMTI4KXtpZighTGQoYSsxfDAsYixjKSl7YnJlYWsgYX1kPWQmMTI3fERbYj4+Ml08PDd9RFtiPj4yXT1kO2g9MX1yZXR1cm4gaH1mdW5jdGlvbiBEYShhLGIsYyl7dmFyIGQ9MCxlPTAsZj0wLGc9MCxoPTA7YTp7aWYoYT4+PjA+NSl7YnJlYWsgYX1mPURbYysyMD4+Ml07ZD1mO2c9RFtjKzEyPj4yXTtlPURbYysxNj4+Ml07aWYoKGR8MCk+PShnfDApJmU+Pj4wPj1HW2MrOD4+Ml18KGR8MCk+KGd8MCkpe2JyZWFrIGF9ZD1FW2UrRFtjPj4yXXwwXTtlPWUrMXwwO2Y9ZT9mOmYrMXwwO0RbYysxNj4+Ml09ZTtEW2MrMjA+PjJdPWY7aWYoZCYxMjgpe2lmKCFEYShhKzF8MCxiLGMpKXticmVhayBhfWQ9ZCYxMjd8RFtiPj4yXTw8N31EW2I+PjJdPWQ7aD0xfXJldHVybiBofWZ1bmN0aW9uICRhKGEsYixjKXt2YXIgZD0wLGU9MCxmPTAsZz0wLGg9MDthOntpZihhPj4+MD41KXticmVhayBhfWY9RFtjKzIwPj4yXTtkPWY7Zz1EW2MrMTI+PjJdO2U9RFtjKzE2Pj4yXTtpZigoZHwwKT49KGd8MCkmZT4+PjA+PUdbYys4Pj4yXXwoZHwwKT4oZ3wwKSl7YnJlYWsgYX1kPUVbZStEW2M+PjJdfDBdO2U9ZSsxfDA7Zj1lP2Y6ZisxfDA7RFtjKzE2Pj4yXT1lO0RbYysyMD4+Ml09ZjtpZihkJjEyOCl7aWYoISRhKGErMXwwLGIsYykpe2JyZWFrIGF9ZD1kJjEyN3xEW2I+PjJdPDw3fURbYj4+Ml09ZDtoPTF9cmV0dXJuIGh9ZnVuY3Rpb24gdWYoYSxiLGMsZCl7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7dmFyIGU9MCxmPTAsZz0wLGg9MDtnPURbYis5Nj4+Ml07Yj1EW2IrMTAwPj4yXS1nfDA7aWYoISgoYnwwKSE9KGN8MCl8IWIpKXthPShjfDApLzEyfDA7aD1hPj4+MD4xP2E6MTthPTA7d2hpbGUoMSl7ZT1KKGEsMTIpO2Y9ZStkfDA7ZT1lK2d8MDtEW2Y+PjJdPURbZT4+Ml07RFtmKzQ+PjJdPURbZSs0Pj4yXTtEW2YrOD4+Ml09RFtlKzg+PjJdO2E9YSsxfDA7aWYoKGh8MCkhPShhfDApKXtjb250aW51ZX1icmVha319cmV0dXJuKGJ8MCk9PShjfDApfDB9ZnVuY3Rpb24gdmEoYSxiLGMpe3ZhciBkPTAsZT0wO2E6e2I6e2lmKGM+Pj4wPj00KXtpZigoYXxiKSYzKXticmVhayBifXdoaWxlKDEpe2lmKERbYT4+Ml0hPURbYj4+Ml0pe2JyZWFrIGJ9Yj1iKzR8MDthPWErNHwwO2M9Yy00fDA7aWYoYz4+PjA+Myl7Y29udGludWV9YnJlYWt9fWlmKCFjKXticmVhayBhfX13aGlsZSgxKXtkPUVbYXwwXTtlPUVbYnwwXTtpZigoZHwwKT09KGV8MCkpe2I9YisxfDA7YT1hKzF8MDtjPWMtMXwwO2lmKGMpe2NvbnRpbnVlfWJyZWFrIGF9YnJlYWt9cmV0dXJuIGQtZXwwfXJldHVybiAwfWZ1bmN0aW9uIHlmKGEsYixjKXthPWF8MDtiPWJ8MDtjPWN8MDt2YXIgZD0wLGU9MDtkPURbYis0Pj4yXTthOntpZighZCl7YnJlYWsgYX1iPURbRFtEW2IrOD4+Ml0rKGM8PDIpPj4yXSs2MD4+Ml07aWYoKGJ8MCk8MCl7YnJlYWsgYX1hPURbZCsyND4+Ml07Yz1EW2QrMjg+PjJdO2lmKChhfDApPT0oY3wwKSl7YnJlYWsgYX1iOnt3aGlsZSgxKXtlPURbYT4+Ml07aWYoKGJ8MCk9PURbZSsyND4+Ml0pe2JyZWFrIGJ9YT1hKzR8MDtpZigoY3wwKSE9KGF8MCkpe2NvbnRpbnVlfWJyZWFrfWU9MH19cmV0dXJuIGV8MH1mdW5jdGlvbiBkYyhhKXt2YXIgYj0wLGM9MCxkPTA7aWYoYSl7ZD1EW2ErMjQ+PjJdO2lmKGQpe2M9RFthKzI4Pj4yXTtpZigoY3wwKT09KGR8MCkpe2I9ZH1lbHNle3doaWxlKDEpe2M9Yy00fDA7Yj1EW2M+PjJdO0RbYz4+Ml09MDtpZihiKXtDYShiKzEyfDAsRFtiKzE2Pj4yXSk7QmEoYixEW2IrND4+Ml0pO21hKGIpfWlmKChkfDApIT0oY3wwKSl7Y29udGludWV9YnJlYWt9Yj1EW2ErMjQ+PjJdfURbYSsyOD4+Ml09ZDttYShiKX1DYShhKzEyfDAsRFthKzE2Pj4yXSk7QmEoYSxEW2ErND4+Ml0pO21hKGEpfX1mdW5jdGlvbiBwaChhKXthPWF8MDt2YXIgYj0wLGM9MCxkPTA7RFthPj4yXT0xMDA2NDtiPURbYSsyMD4+Ml07aWYoYil7RFthKzI0Pj4yXT1iO21hKGIpfWQ9RFthKzg+PjJdO2lmKGQpe2M9RFthKzEyPj4yXTtpZigoY3wwKT09KGR8MCkpe2I9ZH1lbHNle3doaWxlKDEpe2M9Yy00fDA7Yj1EW2M+PjJdO0RbYz4+Ml09MDtpZihiKXtiYVtEW0RbYj4+Ml0rND4+Ml1dKGIpfWlmKChkfDApIT0oY3wwKSl7Y29udGludWV9YnJlYWt9Yj1EW2ErOD4+Ml19RFthKzEyPj4yXT1kO21hKGIpfXJldHVybiBhfDB9ZnVuY3Rpb24gVmcoYSl7YT1hfDA7dmFyIGI9MDtEW2ErOD4+Ml09OTE2ODtEW2E+PjJdPTg5NTY7Yj1EW2ErOTY+PjJdO2lmKGIpe0RbYSsxMDA+PjJdPWI7bWEoYil9Yj1EW2ErODA+PjJdO2lmKGIpe0RbYSs4ND4+Ml09YjttYShiKX1iPURbYSs2OD4+Ml07aWYoYil7RFthKzcyPj4yXT1iO21hKGIpfWI9RFthKzU2Pj4yXTtpZihiKXtEW2ErNjA+PjJdPWI7bWEoYil9RFthKzg+PjJdPTk0MDQ7Yj1EW2ErNDQ+PjJdO2lmKGIpe21hKGIpfWI9RFthKzMyPj4yXTtpZihiKXttYShiKX1yZXR1cm4gYXwwfWZ1bmN0aW9uIHphKGEpe3ZhciBiPTAsYz0wLGQ9MDtiPWE7YTp7aWYoYiYzKXt3aGlsZSgxKXtpZighRVtifDBdKXticmVhayBhfWI9YisxfDA7aWYoYiYzKXtjb250aW51ZX1icmVha319d2hpbGUoMSl7Yz1iO2I9Yis0fDA7ZD1EW2M+PjJdO2lmKCEoKGReLTEpJmQtMTY4NDMwMDkmLTIxMzkwNjIxNDQpKXtjb250aW51ZX1icmVha31pZighKGQmMjU1KSl7cmV0dXJuIGMtYXwwfXdoaWxlKDEpe2Q9RVtjKzF8MF07Yj1jKzF8MDtjPWI7aWYoZCl7Y29udGludWV9YnJlYWt9fXJldHVybiBiLWF8MH1mdW5jdGlvbiBUZyhhKXthPWF8MDt2YXIgYj0wO0RbYSs4Pj4yXT05MTY4O0RbYT4+Ml09ODk1NjtiPURbYSs5Nj4+Ml07aWYoYil7RFthKzEwMD4+Ml09YjttYShiKX1iPURbYSs4MD4+Ml07aWYoYil7RFthKzg0Pj4yXT1iO21hKGIpfWI9RFthKzY4Pj4yXTtpZihiKXtEW2ErNzI+PjJdPWI7bWEoYil9Yj1EW2ErNTY+PjJdO2lmKGIpe0RbYSs2MD4+Ml09YjttYShiKX1EW2ErOD4+Ml09OTQwNDtiPURbYSs0ND4+Ml07aWYoYil7bWEoYil9Yj1EW2ErMzI+PjJdO2lmKGIpe21hKGIpfW1hKGEpfWZ1bmN0aW9uIEFnKGEpe2E9YXwwO3ZhciBiPTAsYz0wLGQ9MDtEW2E+PjJdPTEwMDY0O2I9RFthKzIwPj4yXTtpZihiKXtEW2ErMjQ+PjJdPWI7bWEoYil9ZD1EW2ErOD4+Ml07aWYoZCl7Yz1EW2ErMTI+PjJdO2lmKChjfDApPT0oZHwwKSl7Yj1kfWVsc2V7d2hpbGUoMSl7Yz1jLTR8MDtiPURbYz4+Ml07RFtjPj4yXT0wO2lmKGIpe2JhW0RbRFtiPj4yXSs0Pj4yXV0oYil9aWYoKGR8MCkhPShjfDApKXtjb250aW51ZX1icmVha31iPURbYSs4Pj4yXX1EW2ErMTI+PjJdPWQ7bWEoYil9bWEoYSl9ZnVuY3Rpb24gcmMoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO3ZhciBkPTAsZT0wLGY9MCxnPTAsaD0wLGk9MDtoPURbYys4Pj4yXTtlPURbYysxNj4+Ml07Zz1EW2MrMTI+PjJdO2Y9ZztkPURbYysyMD4+Ml07aWYoaD4+PjA+ZT4+PjAmKGZ8MCk+PShkfDApfChkfDApPChmfDApKXtiPUVbZStEW2M+PjJdfDBdO2k9ZSsxfDA7Zj1pP2Q6ZCsxfDA7RFtjKzE2Pj4yXT1pO0RbYysyMD4+Ml09ZjtEW2ErND4+Ml09Yn1yZXR1cm4gZT4+PjA8aD4+PjAmKGR8MCk8PShnfDApfChkfDApPChnfDApfWZ1bmN0aW9uIEphKGEsYixjKXt2YXIgZD0wO2lmKCFjKXtyZXR1cm4gRFthKzQ+PjJdPT1EW2IrND4+Ml19aWYoKGF8MCk9PShifDApKXtyZXR1cm4gMX1kPURbYSs0Pj4yXTthPUVbZHwwXTtjPURbYis0Pj4yXTtiPUVbY3wwXTthOntpZighYXwoYnwwKSE9KGF8MCkpe2JyZWFrIGF9d2hpbGUoMSl7Yj1FW2MrMXwwXTthPUVbZCsxfDBdO2lmKCFhKXticmVhayBhfWM9YysxfDA7ZD1kKzF8MDtpZigoYXwwKT09KGJ8MCkpe2NvbnRpbnVlfWJyZWFrfX1yZXR1cm4oYXwwKT09KGJ8MCl9ZnVuY3Rpb24gV2coYSl7YT1hfDA7dmFyIGI9MDtEW2E+PjJdPTkxNjg7Yj1EW2ErODg+PjJdO2lmKGIpe0RbYSs5Mj4+Ml09YjttYShiKX1iPURbYSs3Mj4+Ml07aWYoYil7RFthKzc2Pj4yXT1iO21hKGIpfWI9RFthKzYwPj4yXTtpZihiKXtEW2EtIC02ND4+Ml09YjttYShiKX1iPURbYSs0OD4+Ml07aWYoYil7RFthKzUyPj4yXT1iO21hKGIpfURbYT4+Ml09OTQwNDtiPURbYSszNj4+Ml07aWYoYil7bWEoYil9Yj1EW2ErMjQ+PjJdO2lmKGIpe21hKGIpfXJldHVybiBhfDB9ZnVuY3Rpb24gSGIoYSxiKXt2YXIgYz0wLGQ9MCxlPTAsZj0wO2E6e2M9RFthPj4yXTtiOntpZihEW2ErOD4+Ml0tYz4+Mj4+PjA+PWI+Pj4wKXticmVhayBifWlmKGI+Pj4wPj0xMDczNzQxODI0KXticmVhayBhfWQ9RFthKzQ+PjJdO2U9Yjw8MjtiPW5hKGUpO2U9YitlfDA7ZD1kLWN8MDtmPWQrYnwwO2lmKChkfDApPjApe29hKGIsYyxkKX1EW2ErOD4+Ml09ZTtEW2ErND4+Ml09ZjtEW2E+PjJdPWI7aWYoIWMpe2JyZWFrIGJ9bWEoYyl9cmV0dXJufXJhKDEzMjYpO1QoKX1mdW5jdGlvbiBBZihhLGIsYyl7YT1hfDA7Yj1ifDA7Yz1jfDA7dmFyIGQ9MCxlPTA7YT1EW2IrMTI+PjJdO2I9RFtiKzg+PjJdO2Q9YS1ifDA7YT0wO2E6e2lmKCFkKXticmVhayBhfWE9ZD4+MjtkPWE+Pj4wPjE/YToxO2E9MDtiOnt3aGlsZSgxKXtlPURbYisoYTw8Mik+PjJdO2lmKERbZSs2MD4+Ml09PShjfDApKXticmVhayBifWE9YSsxfDA7aWYoKGR8MCkhPShhfDApKXtjb250aW51ZX1icmVha31hPTA7YnJlYWsgYX1hPShhfDApPT0tMT8wOmV9cmV0dXJuIGF8MH1mdW5jdGlvbiBRZyhhKXthPWF8MDt2YXIgYj0wO0RbYT4+Ml09OTE2ODtiPURbYSs4OD4+Ml07aWYoYil7RFthKzkyPj4yXT1iO21hKGIpfWI9RFthKzcyPj4yXTtpZihiKXtEW2ErNzY+PjJdPWI7bWEoYil9Yj1EW2ErNjA+PjJdO2lmKGIpe0RbYS0gLTY0Pj4yXT1iO21hKGIpfWI9RFthKzQ4Pj4yXTtpZihiKXtEW2ErNTI+PjJdPWI7bWEoYil9RFthPj4yXT05NDA0O2I9RFthKzM2Pj4yXTtpZihiKXttYShiKX1iPURbYSsyND4+Ml07aWYoYil7bWEoYil9bWEoYSl9ZnVuY3Rpb24gYWIoYSl7dmFyIGI9MDtpZihhKXtiPURbYSs3Nj4+Ml07aWYoYil7RFthKzgwPj4yXT1iO21hKGIpfWI9RFthLSAtNjQ+PjJdO2lmKGIpe0RbYSs2OD4+Ml09YjttYShiKX1iPURbYSs0OD4+Ml07aWYoYil7RFthKzUyPj4yXT1iO21hKGIpfWI9RFthKzI0Pj4yXTtpZihiKXtEW2ErMjg+PjJdPWI7bWEoYil9Yj1EW2ErMTI+PjJdO2lmKGIpe0RbYSsxNj4+Ml09YjttYShiKX1iPURbYT4+Ml07aWYoYil7RFthKzQ+PjJdPWI7bWEoYil9bWEoYSl9fWZ1bmN0aW9uIG9iKGEpe3ZhciBiPTA7Yj1EW2ErODQ+PjJdO2lmKGIpe0RbYSs4OD4+Ml09YjttYShiKX1iPURbYSs3Mj4+Ml07aWYoYil7RFthKzc2Pj4yXT1iO21hKGIpfWI9RFthKzUyPj4yXTtpZihiKXtEW2ErNTY+PjJdPWI7bWEoYil9Yj1EW2ErNDA+PjJdO2lmKGIpe0RbYSs0ND4+Ml09YjttYShiKX1iPURbYSsyOD4+Ml07aWYoYil7RFthKzMyPj4yXT1iO21hKGIpfWI9RFthKzEyPj4yXTtpZihiKXttYShiKX1hPURbYT4+Ml07aWYoYSl7bWEoYSl9fWZ1bmN0aW9uIHhkKGEpe3ZhciBiPTAsYz0wLGQ9MDtiPURbYSs0Pj4yXTtkPURbYT4+Ml07aWYoKGJ8MCkhPShkfDApKXt3aGlsZSgxKXtjPURbYi0xMj4+Ml07aWYoYyl7RFtiLTg+PjJdPWM7bWEoYyl9Yz1EW2ItMjg+PjJdO2lmKGMpe0RbYi0yND4+Ml09YzttYShjKX1jPURbYi00MD4+Ml07aWYoYyl7RFtiLTM2Pj4yXT1jO21hKGMpfW9iKGItMTQwfDApO2I9Yi0xNDR8MDtpZigoZHwwKSE9KGJ8MCkpe2NvbnRpbnVlfWJyZWFrfX1EW2ErND4+Ml09ZH1mdW5jdGlvbiB0YShhLGIsYyl7dmFyIGQ9MCxlPTA7YTp7Yjp7aWYoYz4+PjA8PTEwKXtkPWE7QltkKzExfDBdPWM7YnJlYWsgYn1pZihjPj4+MD40Mjk0OTY3Mjc5KXticmVhayBhfWlmKGM+Pj4wPj0xMSl7ZT1jKzE2Ji0xNjtkPWUtMXwwO2Q9KGR8MCk9PTExP2U6ZH1lbHNle2Q9MTB9ZT1kKzF8MDtkPW5hKGUpO0RbYT4+Ml09ZDtEW2ErOD4+Ml09ZXwtMjE0NzQ4MzY0ODtEW2ErND4+Ml09Y31YYShkLGIsYysxfDApO3JldHVybn1BYSgpO1QoKX1mdW5jdGlvbiBHYyhhLGIsYyl7dmFyIGQ9MCxlPTAsZj0wLGc9MCxoPTA7Zj0kLTE2fDA7JD1mO2Q9JC0xNnwwOyQ9ZDtiPWItYT4+Mjt3aGlsZSgxKXtpZihiKXtEW2QrMTI+PjJdPWE7ZT1iPj4+MXwwO0RbZCsxMj4+Ml09RFtkKzEyPj4yXSsoZTw8Mik7aD0oZV4tMSkrYnwwO2I9ZTtlPURbZCsxMj4+Ml07Zz1HW2U+PjJdPEdbYz4+Ml07Yj1nP2g6YjthPWc/ZSs0fDA6YTtjb250aW51ZX1icmVha30kPWQrMTZ8MDskPWYrMTZ8MDtyZXR1cm4gYX1mdW5jdGlvbiBmZChhLGIpe3ZhciBjPTAsZD0wO2Q9bmEoNDApO0RbZD4+Ml09LTE7Yz1kKzh8MDtEW2MrMTY+PjJdPTA7RFtjKzIwPj4yXT0wO0RbYys4Pj4yXT0wO0RbYz4+Ml09MDtEW2MrND4+Ml09MDtEW2MrMjQ+PjJdPTA7RFtjKzI4Pj4yXT0wO2JhW0RbRFthPj4yXSsxNj4+Ml1dKGEsZCk7YT1EW2IrODg+PjJdO0RbYis4OD4+Ml09ZDtpZihhKXtiPURbYSs4Pj4yXTtpZihiKXtEW2ErMTI+PjJdPWI7bWEoYil9bWEoYSl9cmV0dXJuIDF9ZnVuY3Rpb24gR2EoYSl7dmFyIGI9MCxjPTAsZD0wLGU9MCxmPTA7ZD1FW2ErMTJ8MF07Yz1EW2ErOD4+Ml07YTp7aWYoYz4+PjA+NDA5NSl7YnJlYWsgYX1iPURbYSs0Pj4yXTtpZigoYnwwKTw9MCl7YnJlYWsgYX1iPWItMXwwO0RbYSs0Pj4yXT1iO2M9RVtiK0RbYT4+Ml18MF18Yzw8OH1kPTAtZCYyNTU7Yj1KKGQsYz4+Pjh8MCk7ZT1jJjI1NTtmPWU+Pj4wPGQ+Pj4wO0RbYSs4Pj4yXT1mP2IrZXwwOmMtKGIrZHwwKXwwO3JldHVybiBmfWZ1bmN0aW9uIHhmKGEsYixjLGQpe2E9YXwwO2I9YnwwO2M9Y3wwO2Q9ZHwwO3ZhciBlPTAsZj0wO2E9JC0xNnwwOyQ9YTtiPURbYis5Nj4+Ml07RFthKzg+PjJdPTA7RFthPj4yXT0wO0RbYSs0Pj4yXT0wO2U9bmEoMTIpO2M9b2EoZSxiK0ooYywxMil8MCwxMik7Zj1jKzEyfDA7Yj1EW2Q+PjJdO2lmKGIpe0RbZCs0Pj4yXT1iO21hKGIpfURbZCs4Pj4yXT1jKzEyO0RbZCs0Pj4yXT1mO0RbZD4+Ml09ZTskPWErMTZ8MDtyZXR1cm4gMX1mdW5jdGlvbiB3YyhhLGIpe0RbYSs0Pj4yXT0wO0RbYSs4Pj4yXT0wO0RbYT4+Ml09MTgwNDtEW2ErMTI+PjJdPTA7RFthKzE2Pj4yXT0wO0RbYSsyMD4+Ml09MDtEW2ErMjQ+PjJdPTA7RFthKzI4Pj4yXT0wO0RbYSszMj4+Ml09MDtEW2ErMzY+PjJdPTA7RFthKzQwPj4yXT0wO0RbYT4+Ml09MjA0NDtEW2ErNDQ+PjJdPTA7RFthKzQ4Pj4yXT0wO0RbYSs1Mj4+Ml09MDtEW2ErNTY+PjJdPTA7RFthKzYwPj4yXT1iO3JldHVybiBhfWZ1bmN0aW9uIGlkKGEsYixjLGQpe2E6e2lmKCFiKXtpZigoZHwwKTwwKXtyZXR1cm4gMH1nYihhLGMpO2JyZWFrIGF9aWYoKGR8MCk8MCl7cmV0dXJuIDB9aWYoISghZCZEW2ErND4+Ml0tRFthPj4yXT4+PjA+PWM+Pj4wKSl7Z2IoYSxjKX1pZighYyl7YnJlYWsgYX1OYShEW2E+PjJdLGIsYyl9Yj1EW2ErMjg+PjJdO2M9RFthKzI0Pj4yXSsxfDA7Yj1jP2I6YisxfDA7RFthKzI0Pj4yXT1jO0RbYSsyOD4+Ml09YjtyZXR1cm4gMX1mdW5jdGlvbiBJZihhKXthPWF8MDt2YXIgYj0wLGM9MCxkPTA7aWYoYSl7aWYoQlthKzI3fDBdPDApe21hKERbYSsxNj4+Ml0pfWI9RFthPj4yXTtpZihiKXtjPURbYSs0Pj4yXTtpZigoY3wwKT09KGJ8MCkpe2Q9Yn1lbHNle3doaWxlKDEpe2Q9Yy0xMnwwO2lmKEJbYy0xfDBdPDApe21hKERbZD4+Ml0pfWM9ZDtpZigoY3wwKSE9KGJ8MCkpe2NvbnRpbnVlfWJyZWFrfWQ9RFthPj4yXX1EW2ErND4+Ml09YjttYShkKX1tYShhKX19ZnVuY3Rpb24gdWcoKXt2YXIgYT0wO2E9bmEoNDApO0NbYSszOD4+MV09MDtEW2E+PjJdPTA7RFthKzg+PjJdPTA7RFthKzEyPj4yXT0wO0RbYSsxNj4+Ml09MDtEW2ErMjA+PjJdPTA7RFthKzI0Pj4yXT0wO0RbYSsyOD4+Ml09MDtCW2ErMjl8MF09MDtCW2ErMzB8MF09MDtCW2ErMzF8MF09MDtCW2ErMzJ8MF09MDtCW2ErMzN8MF09MDtCW2ErMzR8MF09MDtCW2ErMzV8MF09MDtCW2ErMzZ8MF09MDtyZXR1cm4gYXwwfWZ1bmN0aW9uIHlhKGEpe2E9YXwwO3ZhciBiPTAsYz0wO2lmKGEpe2I9RFthKzg4Pj4yXTtEW2ErODg+PjJdPTA7aWYoYil7Yz1EW2IrOD4+Ml07aWYoYyl7RFtiKzEyPj4yXT1jO21hKGMpfW1hKGIpfWI9RFthKzY4Pj4yXTtpZihiKXtEW2ErNzI+PjJdPWI7bWEoYil9Yj1EW2ErNjQ+PjJdO0RbYSs2ND4+Ml09MDtpZihiKXtjPURbYj4+Ml07aWYoYyl7RFtiKzQ+PjJdPWM7bWEoYyl9bWEoYil9bWEoYSl9fWZ1bmN0aW9uIHRoKGEpe2E9YXwwO3ZhciBiPTA7RFthKzI0Pj4yXT0xMTQwO0RbYT4+Ml09Nzk3NjtiPURbYSszMj4+Ml07aWYoYil7RFthKzM2Pj4yXT1iO21hKGIpfURbYT4+Ml09MjE2NDtiPURbYSsyMD4+Ml07RFthKzIwPj4yXT0wO2lmKGIpe2JhW0RbRFtiPj4yXSs0Pj4yXV0oYil9RFthPj4yXT0xOTQ4O2I9RFthKzE2Pj4yXTtEW2ErMTY+PjJdPTA7aWYoYil7eWEoYil9cmV0dXJuIGF8MH1mdW5jdGlvbiBGYihhLGIpe3ZhciBjPTAsZD0wLGU9MDtjPXphKGIpO2lmKGM+Pj4wPDQyOTQ5NjcyODApe2E6e2I6e2lmKGM+Pj4wPj0xMSl7ZT1jKzE2Ji0xNjtkPW5hKGUpO0RbYSs4Pj4yXT1lfC0yMTQ3NDgzNjQ4O0RbYT4+Ml09ZDtEW2ErND4+Ml09YzticmVhayBifUJbYSsxMXwwXT1jO2Q9YTtpZighYyl7YnJlYWsgYX19b2EoZCxiLGMpfUJbYytkfDBdPTA7cmV0dXJuIGF9QWEoKTtUKCl9ZnVuY3Rpb24gYWkoYSxiLGMsZCl7dmFyIGU9MCxmPTAsZz0wLGg9MDtmPWJeZDtnPWY+PjMxO2U9Yj4+MzE7YT1hXmU7aD1hLWV8MDtlPShiXmUpLSgoYT4+PjA8ZT4+PjApK2V8MCl8MDthPWQ+PjMxO2I9Y15hO2Y9Zj4+MzE7YT1iaShoLGUsYi1hfDAsKGFeZCktKChhPj4+MD5iPj4+MCkrYXwwKXwwKV5mO2I9YS1mfDA7YWE9KGdeYWEpLSgoYT4+PjA8Zj4+PjApK2d8MCl8MDtyZXR1cm4gYn1mdW5jdGlvbiBzaChhKXthPWF8MDt2YXIgYj0wO0RbYSsyND4+Ml09MTE0MDtEW2E+PjJdPTc5NzY7Yj1EW2ErMzI+PjJdO2lmKGIpe0RbYSszNj4+Ml09YjttYShiKX1EW2E+PjJdPTIxNjQ7Yj1EW2ErMjA+PjJdO0RbYSsyMD4+Ml09MDtpZihiKXtiYVtEW0RbYj4+Ml0rND4+Ml1dKGIpfURbYT4+Ml09MTk0ODtiPURbYSsxNj4+Ml07RFthKzE2Pj4yXT0wO2lmKGIpe3lhKGIpfW1hKGEpfWZ1bmN0aW9uIHRiKGEsYixjKXt2YXIgZD0wLGU9MCxmPTA7ZT0kLTE2fDA7JD1lO2Q9RFthKzg+PjJdJjIxNDc0ODM2NDc7YTp7aWYoZD4+PjA+Yz4+PjApe2Q9RFthPj4yXTtEW2ErND4+Ml09YztYYShkLGIsYyk7QltlKzE1fDBdPTA7QltjK2R8MF09RVtlKzE1fDBdO2JyZWFrIGF9Zj1hO2E9RFthKzQ+PjJdO0ZjKGYsZC0xfDAsKGMtZHwwKSsxfDAsYSxhLGMsYil9JD1lKzE2fDB9ZnVuY3Rpb24gJGgoYSxiLGMsZCl7dmFyIGU9MCxmPTAsZz0wLGg9MCxpPTAsaj0wO2U9Yz4+PjE2fDA7Zj1hPj4+MTZ8MDtqPUooZSxmKTtnPWMmNjU1MzU7aD1hJjY1NTM1O2k9SihnLGgpO2Y9KGk+Pj4xNnwwKStKKGYsZyl8MDtlPShmJjY1NTM1KStKKGUsaCl8MDthYT0oSihiLGMpK2p8MCkrSihhLGQpKyhmPj4+MTYpKyhlPj4+MTYpfDA7cmV0dXJuIGkmNjU1MzV8ZTw8MTZ9ZnVuY3Rpb24gcWUoYSxiKXthPWF8MDtiPWJ8MDt2YXIgYz0wLGQ9MDtjPSQtMTZ8MDskPWM7YT1EW2ErND4+Ml07YTp7aWYoKGF8MCk9PS0xKXticmVhayBhfUJbYysxNXwwXT1hO2Q9RFtiKzIwPj4yXTtpZighIURbYisxNj4+Ml0mKGR8MCk+PTB8KGR8MCk+MCl7YnJlYWsgYX1yYihiLERbYis0Pj4yXSxjKzE1fDAsYysxNnwwKX0kPWMrMTZ8MDtyZXR1cm4oYXwwKSE9LTF8MH1mdW5jdGlvbiBwZygpe3ZhciBhPTA7YT1uYSg5Nik7bGIoYSk7RFthKzY0Pj4yXT0wO0RbYSs2OD4+Ml09MDtEW2ErODg+PjJdPTA7RFthKzcyPj4yXT0wO0RbYSs3Nj4+Ml09MDtCW2ErNzd8MF09MDtCW2ErNzh8MF09MDtCW2ErNzl8MF09MDtCW2ErODB8MF09MDtCW2ErODF8MF09MDtCW2ErODJ8MF09MDtCW2ErODN8MF09MDtCW2ErODR8MF09MDtyZXR1cm4gYXwwfWZ1bmN0aW9uIENjKGEsYixjKXt2YXIgZD0wO2Q9RFthKzE2Pj4yXTtpZighZCl7RFthKzM2Pj4yXT0xO0RbYSsyND4+Ml09YztEW2ErMTY+PjJdPWI7cmV0dXJufWE6e2lmKChifDApPT0oZHwwKSl7aWYoRFthKzI0Pj4yXSE9Mil7YnJlYWsgYX1EW2ErMjQ+PjJdPWM7cmV0dXJufUJbYSs1NHwwXT0xO0RbYSsyND4+Ml09MjtEW2ErMzY+PjJdPURbYSszNj4+Ml0rMX19ZnVuY3Rpb24gYmgoYSxiKXthPWF8MDtiPWJ8MDt2YXIgYz0wLGQ9MDtEW2I+PjJdPTI7Yz1EW2IrOD4+Ml07ZD1EW2IrMTI+PjJdLWN8MDtpZihkPj4+MDw9NDI5NDk2NzI5MSl7RWIoYis4fDAsZCs0fDApO2M9RFtiKzg+PjJdfWI9YytkfDA7YT1EW2ErND4+Ml07QltifDBdPWE7QltiKzF8MF09YT4+Pjg7QltiKzJ8MF09YT4+PjE2O0JbYiszfDBdPWE+Pj4yNH1mdW5jdGlvbiB2ZyhhKXthPWF8MDt2YXIgYj0wLGM9MCxkPTA7Yj1EW2ErOD4+Ml07ZD1EW2ErMTI+PjJdO2lmKChifDApPT0oZHwwKSl7cmV0dXJuIDF9d2hpbGUoMSl7Yz1EW2I+PjJdO2M9YmFbRFtEW2M+PjJdKzE2Pj4yXV0oYyxEW2ErMzI+PjJdKXwwO2lmKGMpe2I9Yis0fDA7aWYoKGR8MCkhPShifDApKXtjb250aW51ZX19YnJlYWt9cmV0dXJuIGN8MH1mdW5jdGlvbiBiZShhKXthPWF8MDt2YXIgYj0wO0RbYT4+Ml09MzA0NDtiPURbYSs5Nj4+Ml07aWYoYil7bWEoYil9Yj1EW2ErODQ+PjJdO2lmKGIpe21hKGIpfWI9RFthKzcyPj4yXTtpZihiKXttYShiKX1iPURbYSs2MD4+Ml07aWYoYil7bWEoYil9RFthPj4yXT0yOTg4O2I9RFthKzMyPj4yXTtpZihiKXtEW2ErMzY+PjJdPWI7bWEoYil9cmV0dXJuIGF8MH1mdW5jdGlvbiBYaChhKXthPWF8MDt2YXIgYj0wO0RbYT4+Ml09NDYwODtiPURbYSs5Nj4+Ml07aWYoYil7bWEoYil9Yj1EW2ErODQ+PjJdO2lmKGIpe21hKGIpfWI9RFthKzcyPj4yXTtpZihiKXttYShiKX1iPURbYSs2MD4+Ml07aWYoYil7bWEoYil9RFthPj4yXT0yOTg4O2I9RFthKzMyPj4yXTtpZihiKXtEW2ErMzY+PjJdPWI7bWEoYil9cmV0dXJuIGF8MH1mdW5jdGlvbiByYShhKXt2YXIgYj0wLGM9MCxkPTAsZT0wLGY9MDtiPV8oOCl8MDtEW2I+PjJdPTExMDI0O0RbYj4+Ml09MTEwNjg7Yz16YShhKTtkPW5hKGMrMTN8MCk7RFtkKzg+PjJdPTA7RFtkKzQ+PjJdPWM7RFtkPj4yXT1jO2U9YixmPW9hKGQrMTJ8MCxhLGMrMXwwKSxEW2UrND4+Ml09ZjtEW2I+PjJdPTExMTE2O1ooYnwwLDExMTQ4LDEzKTtUKCl9ZnVuY3Rpb24gYWUoYSl7YT1hfDA7dmFyIGI9MDtEW2E+PjJdPTMwNDQ7Yj1EW2ErOTY+PjJdO2lmKGIpe21hKGIpfWI9RFthKzg0Pj4yXTtpZihiKXttYShiKX1iPURbYSs3Mj4+Ml07aWYoYil7bWEoYil9Yj1EW2ErNjA+PjJdO2lmKGIpe21hKGIpfURbYT4+Ml09Mjk4ODtiPURbYSszMj4+Ml07aWYoYil7RFthKzM2Pj4yXT1iO21hKGIpfW1hKGEpfWZ1bmN0aW9uIFdoKGEpe2E9YXwwO3ZhciBiPTA7RFthPj4yXT00NjA4O2I9RFthKzk2Pj4yXTtpZihiKXttYShiKX1iPURbYSs4ND4+Ml07aWYoYil7bWEoYil9Yj1EW2ErNzI+PjJdO2lmKGIpe21hKGIpfWI9RFthKzYwPj4yXTtpZihiKXttYShiKX1EW2E+PjJdPTI5ODg7Yj1EW2ErMzI+PjJdO2lmKGIpe0RbYSszNj4+Ml09YjttYShiKX1tYShhKX1mdW5jdGlvbiBOYyhhLGIsYyl7dmFyIGQ9MCxlPTA7ZD1hKzR8MDthPV9hKGEsYik7aWYoKGR8MCk9PShhfDApKXtyZXR1cm4gMH1iPURbYSszMj4+Ml07ZD1EW2ErMjg+PjJdO2lmKChifDApIT0oZHwwKSl7UWIoYyxiLWR8MCk7ZT1SYihjKTtjPURbYSsyOD4+Ml07b2EoZSxjLERbYSszMj4+Ml0tY3wwKX1yZXR1cm4oYnwwKSE9KGR8MCl9ZnVuY3Rpb24gS2MoYSxiKXt2YXIgYz0wLGQ9MDtjPURbYSs4Pj4yXTthPURbYSsxMj4+Ml0tY3wwO2lmKGEpe2E9YT4+MjtkPWE+Pj4wPjE/YToxO2E9MDt3aGlsZSgxKXtpZihEW0RbKGE8PDIpK2M+PjJdKzYwPj4yXT09KGJ8MCkpe3JldHVybiBhfWE9YSsxfDA7aWYoKGR8MCkhPShhfDApKXtjb250aW51ZX1icmVha319cmV0dXJuLTF9ZnVuY3Rpb24gbGIoYSl7RFthKzg+PjJdPTA7RFthKzEyPj4yXT0wO0RbYT4+Ml09MDtEW2ErNDA+PjJdPTA7RFthKzQ0Pj4yXT0wO0RbYSsyOD4+Ml09OTtCW2ErMjR8MF09MTtEW2ErNTY+PjJdPS0xO0RbYSs2MD4+Ml09MDtEW2ErMTY+PjJdPTA7RFthKzIwPj4yXT0wO0RbYSs0OD4+Ml09MDtEW2ErNTI+PjJdPTA7cmV0dXJuIGF9ZnVuY3Rpb24gUmUoYSxiKXthPWF8MDtiPWJ8MDt2YXIgYz0wLGQ9MDtJYyhhLGIpO2E6e2lmKChifDApPDApe2JyZWFrIGF9Yz1EW2ErODg+PjJdO2Q9RFthKzg0Pj4yXTtpZihjLWQ+PjI8PShifDApKXticmVhayBhfWI9ZCsoYjw8Mil8MDtkPWIrNHwwO2M9Yy1kfDA7aWYoYyl7TmEoYixkLGMpfURbYSs4OD4+Ml09YitjfX1mdW5jdGlvbiBGZChhKXtEW2ErNDA+PjJdPTA7RFthKzQ+PjJdPTA7RFthKzg+PjJdPTA7RFthPj4yXT0xMDA2NDtEW2ErMTI+PjJdPTA7RFthKzE2Pj4yXT0wO0RbYSsyMD4+Ml09MDtEW2ErMjQ+PjJdPTA7RFthKzI4Pj4yXT0wO0RbYSszMj4+Ml09MDtDW2ErMzY+PjFdPTA7RFthKzQ0Pj4yXT0wO0RbYT4+Ml09ODEwOH1mdW5jdGlvbiBjYyhhLGIsYyl7dmFyIGQ9MDthOntpZihiKXtiPTA7aWYoIWhkKDEsYyxhKSl7YnJlYWsgYX19QlthKzM2fDBdPTE7RFthKzMyPj4yXT0wO2I9RFthKzE2Pj4yXTtjPWIrRFthPj4yXXwwO0RbYSsyND4+Ml09YztkPWE7YT1EW2ErOD4+Ml07RFtkKzI4Pj4yXT1jKyhhLWJ8MCk7Yj0xfXJldHVybiBifWZ1bmN0aW9uIGtlKGEsYil7YT1hfDA7Yj1ifDA7dmFyIGM9MCxkPTA7ZD1EW2ErMTY+PjJdO2M9MDthOntpZihEW2ErMjA+PjJdLWQ+PjI8PShifDApKXticmVhayBhfWI9RFsoYjw8MikrZD4+Ml07Yz0wO2lmKChifDApPDApe2JyZWFrIGF9Yz1jYihEW0RbYSszNj4+Ml0rKGI8PDIpPj4yXSl9cmV0dXJuIGN8MH1mdW5jdGlvbiBIZigpe3ZhciBhPTAsYj0wO2E9bmEoNDApO0RbYSs0Pj4yXT0wO0RbYSs4Pj4yXT0wO0RbYT4+Ml09YSs0O2I9YSsxNnwwO0RbYj4+Ml09MDtEW2IrND4+Ml09MDtEW2ErMjQ+PjJdPTA7RFthKzI4Pj4yXT0wO0RbYSsxMj4+Ml09YjtEW2ErMzI+PjJdPTA7RFthKzM2Pj4yXT0wO3JldHVybiBhfDB9ZnVuY3Rpb24gV2EoYSl7dmFyIGI9MCxjPTA7Yj1EWzI4NTNdO2M9YSszJi00O2E9YitjfDA7YTp7aWYoYT4+PjA8PWI+Pj4wP2M6MCl7YnJlYWsgYX1pZihhPj4+MD5jYSgpPDwxNj4+PjApe2lmKCEoWShhfDApfDApKXticmVhayBhfX1EWzI4NTNdPWE7cmV0dXJuIGJ9RFsyODc5XT00ODtyZXR1cm4tMX1mdW5jdGlvbiB6ZShhLGIsYyl7YT1hfDA7Yj1ifDA7Yz1jfDA7dmFyIGQ9MCxlPTA7ZD1EW2ErOD4+Ml07YTp7aWYoQltkKzI0fDBdPD0wKXticmVhayBhfWlmKCFZYihkLERbYis0Pj4yXS1EW2I+PjJdPj4yKSl7YnJlYWsgYX1lPWJhW0RbRFthPj4yXSszMj4+Ml1dKGEsYixjKXwwfXJldHVybiBlfDB9ZnVuY3Rpb24gc2IoYSxiLGMpe3ZhciBkPTAsZT0wO2Q9JC0xNnwwOyQ9ZDthOntpZihjPj4+MDw9MTApe0JbYSsxMXwwXT1jO1hhKGEsYixjKTtCW2QrMTV8MF09MDtCW2ErY3wwXT1FW2QrMTV8MF07YnJlYWsgYX1lPWE7YT1FW2ErMTF8MF07RmMoZSwxMCxjLTEwfDAsYSxhLGMsYil9JD1kKzE2fDB9ZnVuY3Rpb24gTmgoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO3ZhciBkPTA7RFthKzQ+PjJdPWI7Yj1EW0RbRFtiKzQ+PjJdKzg+PjJdKyhjPDwyKT4+Ml07RFthKzEyPj4yXT1jO0RbYSs4Pj4yXT1iO2E9RFthKzg+PjJdO2lmKEVbYSsyNHwwXT09Myl7ZD1EW2ErMjg+PjJdPT05fXJldHVybiBkfDB9ZnVuY3Rpb24gTWcoYSl7YT1hfDA7dmFyIGI9MDtEW2ErOD4+Ml09OTU4ODtEW2E+PjJdPTk0MjQ7Yj1EW2ErNTY+PjJdO2lmKGIpe0RbYSs2MD4+Ml09YjttYShiKX1EW2ErOD4+Ml09OTQwNDtiPURbYSs0ND4+Ml07aWYoYil7bWEoYil9Yj1EW2ErMzI+PjJdO2lmKGIpe21hKGIpfXJldHVybiBhfDB9ZnVuY3Rpb24gR2coYSl7YT1hfDA7dmFyIGI9MDtEW2ErOD4+Ml09ODY1MjtEW2E+PjJdPTk3MTY7Yj1EW2ErNTY+PjJdO2lmKGIpe0RbYSs2MD4+Ml09YjttYShiKX1EW2ErOD4+Ml09ODkwNDtiPURbYSs0ND4+Ml07aWYoYil7bWEoYil9Yj1EW2ErMzI+PjJdO2lmKGIpe21hKGIpfXJldHVybiBhfDB9ZnVuY3Rpb24gemgoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO3ZhciBkPTA7RFthKzQ+PjJdPWI7ZD1EW0RbRFtiKzQ+PjJdKzg+PjJdKyhjPDwyKT4+Ml07RFthKzEyPj4yXT1jO0RbYSs4Pj4yXT1kO3JldHVybiBEW0RbRFtEW2IrND4+Ml0rOD4+Ml0rKGM8PDIpPj4yXSsyOD4+Ml09PTl8MH1mdW5jdGlvbiBDYShhLGIpe2lmKGIpe0NhKGEsRFtiPj4yXSk7Q2EoYSxEW2IrND4+Ml0pO2E9RFtiKzI4Pj4yXTtEW2IrMjg+PjJdPTA7aWYoYSl7Q2EoYSsxMnwwLERbYSsxNj4+Ml0pO0JhKGEsRFthKzQ+PjJdKTttYShhKX1pZihCW2IrMjd8MF08MCl7bWEoRFtiKzE2Pj4yXSl9bWEoYil9fWZ1bmN0aW9uIEtnKGEpe2E9YXwwO3ZhciBiPTA7RFthKzg+PjJdPTk1ODg7RFthPj4yXT05NDI0O2I9RFthKzU2Pj4yXTtpZihiKXtEW2ErNjA+PjJdPWI7bWEoYil9RFthKzg+PjJdPTk0MDQ7Yj1EW2ErNDQ+PjJdO2lmKGIpe21hKGIpfWI9RFthKzMyPj4yXTtpZihiKXttYShiKX1tYShhKX1mdW5jdGlvbiBGZyhhKXthPWF8MDt2YXIgYj0wO0RbYSs4Pj4yXT04NjUyO0RbYT4+Ml09OTcxNjtiPURbYSs1Nj4+Ml07aWYoYil7RFthKzYwPj4yXT1iO21hKGIpfURbYSs4Pj4yXT04OTA0O2I9RFthKzQ0Pj4yXTtpZihiKXttYShiKX1iPURbYSszMj4+Ml07aWYoYil7bWEoYil9bWEoYSl9ZnVuY3Rpb24gT2UoYSxiKXthPWF8MDtiPWJ8MDt2YXIgYz0wO2E6e2lmKCEoYmFbRFtEW2E+PjJdKzM2Pj4yXV0oYSxiKXwwKSl7YnJlYWsgYX1pZighKGJhW0RbRFthPj4yXSs0MD4+Ml1dKGEsYil8MCkpe2JyZWFrIGF9Yz1iYVtEW0RbYT4+Ml0rNDQ+PjJdXShhKXwwfXJldHVybiBjfDB9ZnVuY3Rpb24gVWQoYSl7YT1hfDA7dmFyIGI9MDthOntpZighRFthLSAtNjQ+PjJdfCFEW2ErNjg+PjJdfCghRFthKzQ0Pj4yXXwhRFthKzQ4Pj4yXSkpe2JyZWFrIGF9aWYoIURbYSs1Mj4+Ml18IURbYSs1Nj4+Ml0pe2JyZWFrIGF9Yj1EW2ErOTI+PjJdIT0tMX1yZXR1cm4gYnwwfWZ1bmN0aW9uIHVjKGEpe2E9YXwwO3ZhciBiPTA7RFthPj4yXT0yMTY0O2I9RFthKzIwPj4yXTtEW2ErMjA+PjJdPTA7aWYoYil7YmFbRFtEW2I+PjJdKzQ+PjJdXShiKX1EW2E+PjJdPTE5NDg7Yj1EW2ErMTY+PjJdO0RbYSsxNj4+Ml09MDtpZihiKXt5YShiKX1yZXR1cm4gYXwwfWZ1bmN0aW9uIF9oKGEsYil7YT1hfDA7Yj1ifDA7dmFyIGM9MDtiPURbYis4OD4+Ml07aWYoISghYnxEW2I+PjJdIT0yKSl7Yz1hO2E9RFtiKzg+PjJdO0RbYys0Pj4yXT1FW2F8MF18RVthKzF8MF08PDh8KEVbYSsyfDBdPDwxNnxFW2ErM3wwXTw8MjQpO2M9MX1yZXR1cm4gY3wwfWZ1bmN0aW9uIE9kKGEpe2E9YXwwO3ZhciBiPTA7YTp7aWYoIURbYSs0OD4+Ml18IURbYSs1Mj4+Ml18KCFEW2ErMjg+PjJdfCFEW2ErMzI+PjJdKSl7YnJlYWsgYX1pZighRFthKzM2Pj4yXXwhRFthKzQwPj4yXSl7YnJlYWsgYX1iPURbYSs3Nj4+Ml0hPS0xfXJldHVybiBifDB9ZnVuY3Rpb24gemQoYSl7YT1hfDA7dmFyIGI9MDtEW2E+PjJdPTg2NTI7Yj1EW2ErNDg+PjJdO2lmKGIpe0RbYSs1Mj4+Ml09YjttYShiKX1EW2E+PjJdPTg5MDQ7Yj1EW2ErMzY+PjJdO2lmKGIpe21hKGIpfWI9RFthKzI0Pj4yXTtpZihiKXttYShiKX1yZXR1cm4gYXwwfWZ1bmN0aW9uIHRjKGEpe2E9YXwwO3ZhciBiPTA7RFthPj4yXT0yMTY0O2I9RFthKzIwPj4yXTtEW2ErMjA+PjJdPTA7aWYoYil7YmFbRFtEW2I+PjJdKzQ+PjJdXShiKX1EW2E+PjJdPTE5NDg7Yj1EW2ErMTY+PjJdO0RbYSsxNj4+Ml09MDtpZihiKXt5YShiKX1tYShhKX1mdW5jdGlvbiBOZyhhKXthPWF8MDt2YXIgYj0wO0RbYT4+Ml09OTU4ODtiPURbYSs0OD4+Ml07aWYoYil7RFthKzUyPj4yXT1iO21hKGIpfURbYT4+Ml09OTQwNDtiPURbYSszNj4+Ml07aWYoYil7bWEoYil9Yj1EW2ErMjQ+PjJdO2lmKGIpe21hKGIpfXJldHVybiBhfDB9ZnVuY3Rpb24gc2coKXt2YXIgYT0wLGI9MDtiPW5hKDQwKTtEW2I+PjJdPS0xO2E9Yis4fDA7RFthKzE2Pj4yXT0wO0RbYSsyMD4+Ml09MDtEW2ErOD4+Ml09MDtEW2E+PjJdPTA7RFthKzQ+PjJdPTA7RFthKzI0Pj4yXT0wO0RbYSsyOD4+Ml09MDtyZXR1cm4gYnwwfWZ1bmN0aW9uIF9nKGEpe2E9YXwwO3ZhciBiPTA7RFthPj4yXT04NjUyO2I9RFthKzQ4Pj4yXTtpZihiKXtEW2ErNTI+PjJdPWI7bWEoYil9RFthPj4yXT04OTA0O2I9RFthKzM2Pj4yXTtpZihiKXttYShiKX1iPURbYSsyND4+Ml07aWYoYil7bWEoYil9bWEoYSl9ZnVuY3Rpb24gSGcoYSl7YT1hfDA7dmFyIGI9MDtEW2E+PjJdPTk1ODg7Yj1EW2ErNDg+PjJdO2lmKGIpe0RbYSs1Mj4+Ml09YjttYShiKX1EW2E+PjJdPTk0MDQ7Yj1EW2ErMzY+PjJdO2lmKGIpe21hKGIpfWI9RFthKzI0Pj4yXTtpZihiKXttYShiKX1tYShhKX1mdW5jdGlvbiBDZShhLGIsYyxkLGUsZil7YT1hfDA7Yj1ifDA7Yz1jfDA7ZD1kfDA7ZT1lfDA7Zj1mfDA7aWYoSmEoYSxEW2IrOD4+Ml0sZikpe0JjKGIsYyxkLGUpO3JldHVybn1hPURbYSs4Pj4yXTtiYVtEW0RbYT4+Ml0rMjA+PjJdXShhLGIsYyxkLGUsZil9ZnVuY3Rpb24geWgoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO2E6e2lmKEVbRFthKzQ+PjJdKzM2fDBdPj0yKXtiPTA7aWYoIShiYVtEW0RbYT4+Ml0rNTI+PjJdXShhKXwwKSl7YnJlYWsgYX19Yj1mZChhKzI0fDAsRFthKzE2Pj4yXSl9cmV0dXJuIGJ8MH1mdW5jdGlvbiBiZygpe3ZhciBhPTA7YT1uYSgxMDgpO01jKGEpO0RbYSs4ND4+Ml09MDtEW2ErODg+PjJdPTA7RFthPj4yXT0xMDI3MjtEW2ErOTI+PjJdPTA7RFthKzk2Pj4yXT0wO0RbYSsxMDA+PjJdPTA7RFthKzEwND4+Ml09MDtyZXR1cm4gYXwwfWZ1bmN0aW9uIExjKGEsYil7dmFyIGM9MDtjPS0xO2E6e2lmKChifDApPT0tMXwoYnwwKT40KXticmVhayBhfWI9SihiLDEyKSthfDA7YT1EW2IrMjA+PjJdO2lmKChEW2IrMjQ+PjJdLWF8MCk8PTApe2JyZWFrIGF9Yz1EW2E+PjJdfXJldHVybiBjfWZ1bmN0aW9uICRiKGEsYixjLGQsZSxmLGcpe0RbYT4+Ml09MDtEW2ErNTY+PjJdPWI7RFthKzQ4Pj4yXT0wO0RbYSs1Mj4+Ml09MDtEW2ErNDA+PjJdPWY7RFthKzQ0Pj4yXT1nO0JbYSszMnwwXT1lO0RbYSsyOD4+Ml09ZDtCW2ErMjR8MF09Y31mdW5jdGlvbiBVZShhKXthPWF8MDt2YXIgYj0wO0RbYT4+Ml09MTAyNzI7Yj1EW2ErOTY+PjJdO2lmKGIpe0RbYSsxMDA+PjJdPWI7bWEoYil9Yj1EW2ErODQ+PjJdO2lmKGIpe0RbYSs4OD4+Ml09YjttYShiKX11YihhKTtyZXR1cm4gYXwwfWZ1bmN0aW9uIE1oKGEsYixjKXthPWF8MDtiPWJ8MDtjPWN8MDthOntpZihFW0RbYSs0Pj4yXSszNnwwXT49Mil7Yj0wO2lmKCFyYyhhKzI0fDAsY2IoYSksYykpe2JyZWFrIGF9fWI9ZmQoYSsyNHwwLERbYSsxNj4+Ml0pfXJldHVybiBifDB9ZnVuY3Rpb24gVGUoYSl7YT1hfDA7dmFyIGI9MDtEW2E+PjJdPTEwMjcyO2I9RFthKzk2Pj4yXTtpZihiKXtEW2ErMTAwPj4yXT1iO21hKGIpfWI9RFthKzg0Pj4yXTtpZihiKXtEW2ErODg+PjJdPWI7bWEoYil9dWIoYSk7bWEoYSl9ZnVuY3Rpb24gX2QoYSl7YT1hfDA7dmFyIGI9MDtEW2E+PjJdPTMyOTI7Yj1EW2ErNzY+PjJdO2lmKGIpe21hKGIpfURbYT4+Ml09Mjk4ODtiPURbYSszMj4+Ml07aWYoYil7RFthKzM2Pj4yXT1iO21hKGIpfXJldHVybiBhfDB9ZnVuY3Rpb24gVWgoYSl7YT1hfDA7dmFyIGI9MDtEW2E+PjJdPTQ4NDQ7Yj1EW2ErNzY+PjJdO2lmKGIpe21hKGIpfURbYT4+Ml09Mjk4ODtiPURbYSszMj4+Ml07aWYoYil7RFthKzM2Pj4yXT1iO21hKGIpfXJldHVybiBhfDB9ZnVuY3Rpb24gQmEoYSxiKXtpZihiKXtCYShhLERbYj4+Ml0pO0JhKGEsRFtiKzQ+PjJdKTthPURbYisyOD4+Ml07aWYoYSl7RFtiKzMyPj4yXT1hO21hKGEpfWlmKEJbYisyN3wwXTwwKXttYShEW2IrMTY+PjJdKX1tYShiKX19ZnVuY3Rpb24gUWYoKXt2YXIgYT0wO2E9bmEoMjgpO0RbYT4+Ml09MDtEW2ErND4+Ml09MDtEW2ErMjQ+PjJdPTA7RFthKzE2Pj4yXT0wO0RbYSsyMD4+Ml09MDtEW2ErOD4+Ml09MDtEW2ErMTI+PjJdPTA7cmV0dXJuIGF8MH1mdW5jdGlvbiBSYihhKXt2YXIgYj0wO2lmKEVbYSsxMXwwXT4+Pjd8MCl7Yj1EW2ErND4+Ml19ZWxzZXtiPUVbYSsxMXwwXX1pZighYil7U2IoKTtUKCl9aWYoRVthKzExfDBdPj4+N3wwKXthPURbYT4+Ml19cmV0dXJuIGF9ZnVuY3Rpb24gUWUoYSl7YT1hfDA7dmFyIGI9MDtEW2E+PjJdPTE4MDQ7Yj1EW2ErMTY+PjJdO2lmKGIpe0RbYSsyMD4+Ml09YjttYShiKX1iPURbYSs0Pj4yXTtpZihiKXtEW2ErOD4+Ml09YjttYShiKX1yZXR1cm4gYXwwfWZ1bmN0aW9uIEdlKGEsYixjLGQpe2E9YXwwO2I9YnwwO2M9Y3wwO2Q9ZHwwO2lmKEphKGEsRFtiKzg+PjJdLDApKXtDYyhiLGMsZCk7cmV0dXJufWE9RFthKzg+PjJdO2JhW0RbRFthPj4yXSsyOD4+Ml1dKGEsYixjLGQpfWZ1bmN0aW9uICRmKCl7dmFyIGE9MCxiPTA7YT1uYSgyNCk7RFthKzQ+PjJdPTA7RFthKzg+PjJdPTA7Yj1hKzE2fDA7RFtiPj4yXT0wO0RbYis0Pj4yXT0wO0RbYT4+Ml09YSs0O0RbYSsxMj4+Ml09YjtyZXR1cm4gYXwwfWZ1bmN0aW9uIFpkKGEpe2E9YXwwO3ZhciBiPTA7RFthPj4yXT0zMjkyO2I9RFthKzc2Pj4yXTtpZihiKXttYShiKX1EW2E+PjJdPTI5ODg7Yj1EW2ErMzI+PjJdO2lmKGIpe0RbYSszNj4+Ml09YjttYShiKX1tYShhKX1mdW5jdGlvbiBUaChhKXthPWF8MDt2YXIgYj0wO0RbYT4+Ml09NDg0NDtiPURbYSs3Nj4+Ml07aWYoYil7bWEoYil9RFthPj4yXT0yOTg4O2I9RFthKzMyPj4yXTtpZihiKXtEW2ErMzY+PjJdPWI7bWEoYil9bWEoYSl9ZnVuY3Rpb24gbmEoYSl7dmFyIGI9MDthPWE/YToxO2E6e3doaWxlKDEpe2I9QWMoYSk7aWYoYil7YnJlYWsgYX1iPURbMjg4MF07aWYoYil7YmFbYnwwXSgpO2NvbnRpbnVlfWJyZWFrfVcoKTtUKCl9cmV0dXJuIGJ9ZnVuY3Rpb24gaGIoYSxiKXtpZihiKXtoYihhLERbYj4+Ml0pO2hiKGEsRFtiKzQ+PjJdKTtpZihCW2IrMzl8MF08MCl7bWEoRFtiKzI4Pj4yXSl9aWYoQltiKzI3fDBdPDApe21hKERbYisxNj4+Ml0pfW1hKGIpfX1mdW5jdGlvbiBQYihhKXthPWF8MDt2YXIgYj0wLGM9MDtEW2E+PjJdPTExMDY4O2I9RFthKzQ+PjJdLTEyfDA7Yz1EW2IrOD4+Ml0tMXwwO0RbYis4Pj4yXT1jO2lmKChjfDApPDApe21hKGIpfXJldHVybiBhfDB9ZnVuY3Rpb24gS2IoYSxiKXthPWF8MDtiPWJ8MDt2YXIgYz0wO2E6e3N3aXRjaChifDApe2Nhc2UgMTpiPTE7Y2FzZSAwOkRbYSsyOD4+Ml09YjtjPTE7YnJlYWs7ZGVmYXVsdDpicmVhayBhfX1yZXR1cm4gY3wwfWZ1bmN0aW9uIGdnKCl7dmFyIGE9MDthPW5hKDI0KTtEW2ErOD4+Ml09MDtEW2ErMTI+PjJdPTA7RFthKzQ+PjJdPS0xO0RbYT4+Ml09MTE0MDtEW2ErMTY+PjJdPTA7RFthKzIwPj4yXT0wO3JldHVybiBhfDB9ZnVuY3Rpb24geWMoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO0RbYSs0Pj4yXT1iO2I9RFtEW0RbYis0Pj4yXSs4Pj4yXSsoYzw8Mik+PjJdO0RbYSsxMj4+Ml09YztEW2ErOD4+Ml09YjtyZXR1cm4gMX1mdW5jdGlvbiBuYyhhKXthPWF8MDt2YXIgYj0wO2lmKCEoIURbYSs2MD4+Ml18IURbYSs0ND4+Ml18KCFEW2ErNDg+PjJdfCFEW2ErNTI+PjJdKSkpe2I9RFthKzU2Pj4yXSE9MH1yZXR1cm4gYnwwfWZ1bmN0aW9uIEViKGEsYil7dmFyIGM9MCxkPTA7Z2IoYSxiKTtiPURbYSsyOD4+Ml07Yz1iO2Q9YisxfDA7Yj1EW2ErMjQ+PjJdKzF8MDtEW2ErMjQ+PjJdPWI7RFthKzI4Pj4yXT1iP2M6ZH1mdW5jdGlvbiBYZChhKXthPWF8MDt2YXIgYj0wO0RbYT4+Ml09MzUyODtEW2E+PjJdPTI5ODg7Yj1EW2ErMzI+PjJdO2lmKGIpe0RbYSszNj4+Ml09YjttYShiKX1yZXR1cm4gYXwwfWZ1bmN0aW9uIFJoKGEpe2E9YXwwO3ZhciBiPTA7RFthPj4yXT01MDY4O0RbYT4+Ml09Mjk4ODtiPURbYSszMj4+Ml07aWYoYil7RFthKzM2Pj4yXT1iO21hKGIpfXJldHVybiBhfDB9ZnVuY3Rpb24gZmYoYSl7YT1hfDA7aWYoYSl7aWYoQlthKzM5fDBdPDApe21hKERbYSsyOD4+Ml0pfVhiKGErMTJ8MCxEW2ErMTY+PjJdKTtoYihhLERbYSs0Pj4yXSk7bWEoYSl9fWZ1bmN0aW9uIFpnKGEpe2E9YXwwO3ZhciBiPTA7RFthPj4yXT04OTA0O2I9RFthKzM2Pj4yXTtpZihiKXttYShiKX1iPURbYSsyND4+Ml07aWYoYil7bWEoYil9cmV0dXJuIGF8MH1mdW5jdGlvbiBQZyhhKXthPWF8MDt2YXIgYj0wO0RbYT4+Ml09OTQwNDtiPURbYSszNj4+Ml07aWYoYil7bWEoYil9Yj1EW2ErMjQ+PjJdO2lmKGIpe21hKGIpfXJldHVybiBhfDB9ZnVuY3Rpb24gcWIoYSl7YT1hfDA7dmFyIGI9MDtpZighKCFEW2ErNTI+PjJdfCghRFthKzQ0Pj4yXXwhRFthKzQ4Pj4yXSkpKXtiPURbYSs1Nj4+Ml0hPTB9cmV0dXJuIGJ8MH1mdW5jdGlvbiBtYyhhLGIpe2E9YXwwO2I9YnwwO3ZhciBjPTA7aWYoIShEW2IrNTY+PjJdfCFifEVbYisyNHwwXSE9Mykpe0RbYSs2MD4+Ml09YjtjPTF9cmV0dXJuIGN8MH1mdW5jdGlvbiBXZChhKXthPWF8MDt2YXIgYj0wO0RbYT4+Ml09MzUyODtEW2E+PjJdPTI5ODg7Yj1EW2ErMzI+PjJdO2lmKGIpe0RbYSszNj4+Ml09YjttYShiKX1tYShhKX1mdW5jdGlvbiBRaChhKXthPWF8MDt2YXIgYj0wO0RbYT4+Ml09NTA2ODtEW2E+PjJdPTI5ODg7Yj1EW2ErMzI+PjJdO2lmKGIpe0RbYSszNj4+Ml09YjttYShiKX1tYShhKX1mdW5jdGlvbiB0ZyhhLGIsYyl7YT1hfDA7Yj1ifDA7Yz1jfDA7RFthKzE2Pj4yXT0wO0RbYSsyMD4+Ml09MDtEW2E+PjJdPWI7RFthKzg+PjJdPWM7RFthKzEyPj4yXT0wfWZ1bmN0aW9uIFlnKGEpe2E9YXwwO3ZhciBiPTA7RFthPj4yXT04OTA0O2I9RFthKzM2Pj4yXTtpZihiKXttYShiKX1iPURbYSsyND4+Ml07aWYoYil7bWEoYil9bWEoYSl9ZnVuY3Rpb24gVGQoYSxiKXthPWF8MDtiPWJ8MDt2YXIgYz0wO2lmKCEoRFtiKzU2Pj4yXXxFW2IrMjR8MF0hPTMpKXtEW2EtIC02ND4+Ml09YjtjPTF9cmV0dXJuIGN8MH1mdW5jdGlvbiBPZyhhKXthPWF8MDt2YXIgYj0wO0RbYT4+Ml09OTQwNDtiPURbYSszNj4+Ml07aWYoYil7bWEoYil9Yj1EW2ErMjQ+PjJdO2lmKGIpe21hKGIpfW1hKGEpfWZ1bmN0aW9uIE5kKGEsYil7YT1hfDA7Yj1ifDA7dmFyIGM9MDtpZighKERbYis1Nj4+Ml18RVtiKzI0fDBdIT0zKSl7RFthKzQ4Pj4yXT1iO2M9MX1yZXR1cm4gY3wwfWZ1bmN0aW9uIEJlKGEsYixjLGQsZSxmKXthPWF8MDtiPWJ8MDtjPWN8MDtkPWR8MDtlPWV8MDtmPWZ8MDtpZihKYShhLERbYis4Pj4yXSxmKSl7QmMoYixjLGQsZSl9fWZ1bmN0aW9uIHZlKGEpe2E9YXwwO3ZhciBiPTA7RFthPj4yXT0xOTQ4O2I9RFthKzE2Pj4yXTtEW2ErMTY+PjJdPTA7aWYoYil7eWEoYil9cmV0dXJuIGF8MH1mdW5jdGlvbiBlZShhKXthPWF8MDt2YXIgYj0wO0RbYT4+Ml09Mjk4ODtiPURbYSszMj4+Ml07aWYoYil7RFthKzM2Pj4yXT1iO21hKGIpfXJldHVybiBhfDB9ZnVuY3Rpb24gdmgoYSl7YT1hfDA7dmFyIGI9MDtiPWNiKGEpO3JldHVybiBvZChhKzI0fDAsYj9iOkRbYSs4Pj4yXSxEW0RbYSs0Pj4yXSszMj4+Ml0pfDB9ZnVuY3Rpb24gVWcoYSl7YT1hfDA7dmFyIGI9MDtEW2E+PjJdPTExNDA7Yj1EW2ErOD4+Ml07aWYoYil7RFthKzEyPj4yXT1iO21hKGIpfXJldHVybiBhfDB9ZnVuY3Rpb24gdWUoYSl7YT1hfDA7dmFyIGI9MDtEW2E+PjJdPTE5NDg7Yj1EW2ErMTY+PjJdO0RbYSsxNj4+Ml09MDtpZihiKXt5YShiKX1tYShhKX1mdW5jdGlvbiBOYihhKXthPWF8MDt2YXIgYj0wO0RbYT4+Ml09Mjk4ODtiPURbYSszMj4+Ml07aWYoYil7RFthKzM2Pj4yXT1iO21hKGIpfW1hKGEpfWZ1bmN0aW9uIExnKGEpe2E9YXwwO3ZhciBiPTA7RFthPj4yXT0xMTQwO2I9RFthKzg+PjJdO2lmKGIpe0RbYSsxMj4+Ml09YjttYShiKX1tYShhKX1mdW5jdGlvbiBqZShhLGIpe2E9YXwwO2I9YnwwO3JldHVybiBiYVtEW0RbYT4+Ml0rNDg+PjJdXShhLERbYis0Pj4yXS1EW2I+PjJdPj4yKXwwfWZ1bmN0aW9uIFhiKGEsYil7aWYoYil7WGIoYSxEW2I+PjJdKTtYYihhLERbYis0Pj4yXSk7aGIoYisyMHwwLERbYisyND4+Ml0pO21hKGIpfX1mdW5jdGlvbiByZyhhKXthPWF8MDt2YXIgYj0wO2lmKGEpe2I9RFthKzg+PjJdO2lmKGIpe0RbYSsxMj4+Ml09YjttYShiKX1tYShhKX19ZnVuY3Rpb24gcWgoYSl7YT1hfDA7aWYoIURbYSs0ND4+Ml0pe3JldHVybiAwfXJldHVybiBiYVtEW0RbYT4+Ml0rNDg+PjJdXShhKXwwfWZ1bmN0aW9uIGNpKGEpe3ZhciBiPTA7d2hpbGUoMSl7aWYoYSl7YT1hLTEmYTtiPWIrMXwwO2NvbnRpbnVlfWJyZWFrfXJldHVybiBifWZ1bmN0aW9uIEhlKGEsYixjLGQpe2E9YXwwO2I9YnwwO2M9Y3wwO2Q9ZHwwO2lmKEphKGEsRFtiKzg+PjJdLDApKXtDYyhiLGMsZCl9fWZ1bmN0aW9uIG9oKGEsYil7YT1hfDA7Yj1ifDA7YT1EW2ErNDg+PjJdO3JldHVybiBiYVtEW0RbYT4+Ml0rMjA+PjJdXShhLGIpfDB9ZnVuY3Rpb24gZ2goYSxiKXthPWF8MDtiPWJ8MDthPURbYSs0OD4+Ml07cmV0dXJuIGJhW0RbRFthPj4yXSsxMj4+Ml1dKGEsYil8MH1mdW5jdGlvbiBmaChhLGIpe2E9YXwwO2I9YnwwO2E9RFthKzQ4Pj4yXTtyZXR1cm4gYmFbRFtEW2E+PjJdKzE2Pj4yXV0oYSxiKXwwfWZ1bmN0aW9uIFphKCl7dmFyIGE9MDthPW5hKDEyKTtEW2E+PjJdPTA7RFthKzQ+PjJdPTA7RFthKzg+PjJdPTA7cmV0dXJuIGF8MH1mdW5jdGlvbiBZYShhKXthPWF8MDt2YXIgYj0wO2lmKGEpe2I9RFthPj4yXTtpZihiKXtEW2ErND4+Ml09YjttYShiKX1tYShhKX19ZnVuY3Rpb24gZGkoYSl7dmFyIGI9MDtiPWEmMzE7YT0wLWEmMzE7cmV0dXJuKC0xPj4+YiYtMik8PGJ8KC0xPDxhJi0yKT4+PmF9ZnVuY3Rpb24gemcoYSxiKXthPWF8MDtiPWJ8MDtCW2IrODR8MF09MTtEW2IrNzI+PjJdPURbYis2OD4+Ml07cmV0dXJuIDF9ZnVuY3Rpb24gb2YoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO0RbYSszMj4+Ml09YztEW2ErMjg+PjJdPWI7cmV0dXJuIDF9ZnVuY3Rpb24gX2YoYSl7YT1hfDA7aWYoYSl7Q2EoYSsxMnwwLERbYSsxNj4+Ml0pO0JhKGEsRFthKzQ+PjJdKTttYShhKX19ZnVuY3Rpb24gbWgoYSl7YT1hfDA7YT1EW2ErNDg+PjJdO3JldHVybiBiYVtEW0RbYT4+Ml0rMjQ+PjJdXShhKXwwfWZ1bmN0aW9uIGxoKGEpe2E9YXwwO2E9RFthKzQ4Pj4yXTtyZXR1cm4gYmFbRFtEW2E+PjJdKzI4Pj4yXV0oYSl8MH1mdW5jdGlvbiBoaChhKXthPWF8MDthPURbYSs0OD4+Ml07cmV0dXJuIGJhW0RbRFthPj4yXSszNj4+Ml1dKGEpfDB9ZnVuY3Rpb24gZGcoKXt2YXIgYT0wO2E9bmEoOCk7RFthKzQ+PjJdPS0xO0RbYT4+Ml09MTAzMjtyZXR1cm4gYXwwfWZ1bmN0aW9uIEJmKGEsYixjKXthPWF8MDtiPWJ8MDtjPWN8MDtyZXR1cm4gRFtEW2IrOD4+Ml0rKGM8PDIpPj4yXX1mdW5jdGlvbiB1aChhLGIpe2E9YXwwO2I9YnwwO3JldHVybiB5ZChhKzI0fDAsY2IoYSksRFthKzg+PjJdKXwwfWZ1bmN0aW9uIExoKGEsYil7YT1hfDA7Yj1ifDA7cmV0dXJuIHhjKGErMjR8MCxjYihhKSxEW2ErOD4+Ml0pfDB9ZnVuY3Rpb24gd2goYSxiKXthPWF8MDtiPWJ8MDtyZXR1cm4gYmFbRFtEW2E+PjJdKzU2Pj4yXV0oYSxiKXwwfWZ1bmN0aW9uIGRkKGEsYil7YT1hfDA7Yj1ifDA7cmV0dXJuIGJhW0RbRFthPj4yXSsxMj4+Ml1dKGEsYil8MH1mdW5jdGlvbiBYZihhKXthPWF8MDtpZihhKXtpZihCW2ErMTV8MF08MCl7bWEoRFthKzQ+PjJdKX1tYShhKX19ZnVuY3Rpb24gQWUoYSxiKXthPWF8MDtiPWJ8MDtEW2ErMTI+PjJdPS0xO0RbYSs4Pj4yXT1iO3JldHVybiAxfWZ1bmN0aW9uIGZnKGEsYil7YT1hfDA7Yj1ifDA7cmV0dXJuIEsoSFtEW2ErOD4+Ml0rKGI8PDIpPj4yXSl9ZnVuY3Rpb24gV2YoYSxiKXthPWF8MDtiPWJ8MDtyZXR1cm4gSyhIW0RbYT4+Ml0rKGI8PDIpPj4yXSl9ZnVuY3Rpb24gYWcoYSl7YT1hfDA7cmV0dXJuKERbYSsxMDA+PjJdLURbYSs5Nj4+Ml18MCkvMTJ8MH1mdW5jdGlvbiBZZihhKXthPWF8MDtyZXR1cm4oQlthKzE1fDBdPDA/RFthKzQ+PjJdOmErNHwwKXwwfWZ1bmN0aW9uIE5lKGEsYil7YT1hfDA7Yj1ifDA7cmV0dXJuIERbRFthKzQ+PjJdKyhiPDwyKT4+Ml19ZnVuY3Rpb24gVGYoYSxiKXthPWF8MDtiPWJ8MDtyZXR1cm4gQ1tEW2E+PjJdKyhiPDwxKT4+MV19ZnVuY3Rpb24gU2YoYSxiKXthPWF8MDtiPWJ8MDtyZXR1cm4gRltEW2E+PjJdKyhiPDwxKT4+MV19ZnVuY3Rpb24gJGMoYSxiKXthPWF8MDtiPWJ8MDtyZXR1cm4gRFtEW2E+PjJdKyhiPDwyKT4+Ml19ZnVuY3Rpb24gUGQoYSxiLGMpe2E9YXwwO2I9YnwwO2M9Y3wwO3JldHVybiB2YyhhLGIsYyl8MH1mdW5jdGlvbiBFZihhLGIsYyl7YT1hfDA7Yj1ifDA7Yz1jfDA7cmV0dXJuIExjKGIsYyl8MH1mdW5jdGlvbiBlaShhKXtpZihhKXtyZXR1cm4gMzEtTShhLTFeYSl8MH1yZXR1cm4gMzJ9DQpmdW5jdGlvbiBNYyhhKXtEW2E+PjJdPTEwMzMyO3BhKGErNHwwLDAsODApO3JldHVybiBhfWZ1bmN0aW9uIGNkKGEpe2E9YXwwO3JldHVybiBEW2ErMTI+PjJdLURbYSs4Pj4yXT4+Mn1mdW5jdGlvbiB4YihhKXthPWF8MDtpZihhKXtiYVtEW0RbYT4+Ml0rND4+Ml1dKGEpfX1mdW5jdGlvbiBWZihhLGIpe2E9YXwwO2I9YnwwO3JldHVybiBCW0RbYT4+Ml0rYnwwXX1mdW5jdGlvbiBVZihhLGIpe2E9YXwwO2I9YnwwO3JldHVybiBFW0RbYT4+Ml0rYnwwXX1mdW5jdGlvbiBGZShhKXthPWF8MDtyZXR1cm4gRFthKzg+PjJdLURbYSs0Pj4yXT4+Mn1mdW5jdGlvbiBFZChhLGIpe2E9YXwwO2I9YnwwO0RbYSs0Pj4yXT1iO3JldHVybiAxfWZ1bmN0aW9uIGFkKGEpe2E9YXwwO3JldHVybiBEW2ErND4+Ml0tRFthPj4yXT4+MX1mdW5jdGlvbiBfYihhKXthPWF8MDtyZXR1cm4gRFthKzQ+PjJdLURbYT4+Ml0+PjJ9ZnVuY3Rpb24gYmQoYSl7YT1hfDA7cmV0dXJuIERbYSs0Pj4yXS1EW2E+PjJdfDB9ZnVuY3Rpb24gemYoYSxiKXthPWF8MDtiPWJ8MDtyZXR1cm4gRFtiKzQ+PjJdfWZ1bmN0aW9uIHllKGEsYixjKXthPWF8MDtiPWJ8MDtjPWN8MDtyZXR1cm4gMX1mdW5jdGlvbiB4ZyhhLGIpe2E9YXwwO2I9YnwwO3JldHVybiBCW2IrMjR8MF19ZnVuY3Rpb24gZmUoYSl7YT1hfDA7cmV0dXJuIEJbRFthKzg+PjJdKzI0fDBdfWZ1bmN0aW9uIEtmKGEsYil7YT1hfDA7Yj1ifDA7cmV0dXJuIERbYis4Pj4yXX1mdW5jdGlvbiBKaChhKXthPWF8MDtEW2E+PjJdPTU5NTY7cmV0dXJuIGF8MH1mdW5jdGlvbiBEaChhKXthPWF8MDtEW2E+PjJdPTY5NjA7cmV0dXJuIGF8MH1mdW5jdGlvbiBlZyhhKXthPWF8MDtyZXR1cm4gSyhIW2ErMjA+PjJdKX1mdW5jdGlvbiBJaChhKXthPWF8MDtEW2E+PjJdPTU5NTY7bWEoYSl9ZnVuY3Rpb24gQ2goYSl7YT1hfDA7RFthPj4yXT02OTYwO21hKGEpfWZ1bmN0aW9uIG9nKGEpe2E9YXwwO3JldHVybiBEW2ErODg+PjJdfWZ1bmN0aW9uIG5nKGEpe2E9YXwwO3JldHVybiBEW2ErNTY+PjJdfWZ1bmN0aW9uIGpnKGEpe2E9YXwwO3JldHVybiBEW2ErNDA+PjJdfWZ1bmN0aW9uIGlnKGEpe2E9YXwwO3JldHVybiBEW2ErNDg+PjJdfWZ1bmN0aW9uIGhnKGEpe2E9YXwwO3JldHVybiBEW2ErNjA+PjJdfWZ1bmN0aW9uIGRiKGEpe2E9YXwwO3JldHVybiBEW2ErMjg+PjJdfWZ1bmN0aW9uIGFjKGEpe2E9YXwwO3JldHVybiBEW2ErODA+PjJdfWZ1bmN0aW9uIHpjKGEsYil7YT1hfDA7Yj1ifDA7cmV0dXJuIDF9ZnVuY3Rpb24gc2MoYSxiKXthPWF8MDtiPWJ8MDtyZXR1cm4tMX1mdW5jdGlvbiBwZChhKXthPWF8MDtyZXR1cm4gRFthKzg+PjJdfWZ1bmN0aW9uIGxnKGEpe2E9YXwwO3JldHVybiBCW2ErMjR8MF19ZnVuY3Rpb24ga2coYSl7YT1hfDA7cmV0dXJuIEVbYSszMnwwXX1mdW5jdGlvbiBaZihhKXthPWF8MDtyZXR1cm4hRFthPj4yXXwwfWZ1bmN0aW9uIFZhKGEpe2E9YXwwO3JldHVybiBEW2ErND4+Ml19ZnVuY3Rpb24gUWQoYSxiKXthPWF8MDtiPWJ8MDtyZXR1cm4gNn1mdW5jdGlvbiBIYShhLGIpe2E9YXwwO2I9YnwwO3JldHVybiAwfWZ1bmN0aW9uIEdoKGEsYil7YT1hfDA7Yj1ifDA7cmV0dXJuIDJ9ZnVuY3Rpb24gZWQoYSl7YT1hfDA7cmV0dXJuIERbYT4+Ml19ZnVuY3Rpb24gWGEoYSxiLGMpe2lmKGMpe29hKGEsYixjKX19ZnVuY3Rpb24gcWcoKXtyZXR1cm4gbGIobmEoNjQpKXwwfWZ1bmN0aW9uIGNnKCl7cmV0dXJuIE1jKG5hKDg0KSl8MH1mdW5jdGlvbiBiYyhhKXthPWF8MDtpZihhKXttYShhKX19ZnVuY3Rpb24gTGUoYSl7YT1hfDA7cmV0dXJuIDEyNzN9ZnVuY3Rpb24gSmUoYSl7YT1hfDA7UGIoYSk7bWEoYSl9ZnVuY3Rpb24gTWEoYSl7YT1hfDA7cmV0dXJuIGF8MH1mdW5jdGlvbiBhaChhKXthPWF8MDttYShuZChhKSl9ZnVuY3Rpb24gUGUoYSl7YT1hfDA7bWEodWIoYSkpfWZ1bmN0aW9uIEtlKGEpe2E9YXwwO21hKFBiKGEpKX1mdW5jdGlvbiAkZyhhKXthPWF8MDttYShtZChhKSl9ZnVuY3Rpb24gd2EoYSl7YT1hfDA7cmV0dXJuIDF9ZnVuY3Rpb24gcWMoYSl7YT1hfDA7cmV0dXJuIDR9ZnVuY3Rpb24gb2MoYSl7YT1hfDA7cmV0dXJuIDV9ZnVuY3Rpb24gVWEoYSl7YT1hfDA7cmV0dXJuIDB9ZnVuY3Rpb24gUmQoYSl7YT1hfDA7cmV0dXJuIDJ9ZnVuY3Rpb24gTWIoYSl7YT1hfDA7cmV0dXJuIDZ9ZnVuY3Rpb24gSGgoYSl7YT1hfDA7cmV0dXJuIDN9ZnVuY3Rpb24gcWEoKXtyYSgxMjY2KTtUKCl9ZnVuY3Rpb24gTGEoYSl7YT1hfDA7bWEoYSl9ZnVuY3Rpb24gQWEoKXtyYSgxMzEzKTtUKCl9ZnVuY3Rpb24gUGEoYSl7YT1hfDA7VCgpfWZ1bmN0aW9uIF9lKCl7cmV0dXJuIDExfWZ1bmN0aW9uIFplKCl7cmV0dXJuIDEyfWZ1bmN0aW9uICRlKCl7cmV0dXJuIDEwfWZ1bmN0aW9uIHdiKCl7cmV0dXJuLTF9ZnVuY3Rpb24gdmIoKXtyZXR1cm4gMX1mdW5jdGlvbiB1YSgpe1NiKCk7VCgpfWZ1bmN0aW9uIGtiKCl7cmV0dXJuIDB9ZnVuY3Rpb24gZWYoKXtyZXR1cm4gNX1mdW5jdGlvbiBkZigpe3JldHVybiA2fWZ1bmN0aW9uIGNmKCl7cmV0dXJuIDd9ZnVuY3Rpb24gYmYoKXtyZXR1cm4gOH1mdW5jdGlvbiBhZigpe3JldHVybiA5fWZ1bmN0aW9uIFllKCl7cmV0dXJuLTJ9ZnVuY3Rpb24gWWMoKXtyZXR1cm4gM31mdW5jdGlvbiBYZSgpe3JldHVybi0zfWZ1bmN0aW9uIFhjKCl7cmV0dXJuIDR9ZnVuY3Rpb24gV2UoKXtyZXR1cm4tNH1mdW5jdGlvbiBXYigpe3JldHVybiAyfWZ1bmN0aW9uIFZlKCl7cmV0dXJuLTV9ZnVuY3Rpb24gU2IoKXtXKCk7VCgpfWZ1bmN0aW9uIEVjKGEpe2E9YXwwfWZ1bmN0aW9uIE1lKCl7VCgpfWZ1bmN0aW9uIFZkKCl7fQ0KLy8gRU1TQ1JJUFRFTl9FTkRfRlVOQ1MNCmU9RTtwKGthKTt2YXIgYmE9YyhbbnVsbCxNYSxMYSxSZCxfaCxiaCxSZix4YyxxZSxyYyxtZyxRZCxHaCxQYixVZyxMZyx3YSx4aCxyaCxpaCx5ZCxkaCxvZCxRZCx4ZyxRZSxQYSxvZixaYyxPZSxOZSxGZSxkYixIYSxNZSx6Yyx3YSx2ZSx1ZSx5YyxBZSx6ZSx5ZSx6Yyx4ZSx3ZSxtZSxsZSx0ZSxzZSxrZSxyZSxwZSxvZSxuZSx1Yyx0Yyx5YyxqZSxpZSx2YyxoZSxmZSxnZSxlZSxOYix3YSxWYSxxYixVYSxzYyxIYSxVYSx3YSxkZSxjZSxQYSxQYSxiZSxhZSxxYyxxYixwYywkZCxfZCxaZCxvYyxuYyx3YSxIYSxtYyxsYyxZZCxYZCxXZCxNYixVZCx3YSxIYSxUZCxTZCxaaCxNYSxMYSxLYixkYixMYixQYSxOYix3YSxxYixZaCxQYSxYaCxXaCxxYyxxYixwYyxWaCxVaCxUaCxvYyxuYyx3YSxIYSxtYyxsYyxTaCxSaCxRaCxNYixVZCx3YSxIYSxUZCxTZCxQaCxNYSxMYSxLYixkYixKYixQYSxOYixVYSx3YSxPaCx1Yyx0YyxOaCxNaCxQZCxLaCxSZCxMaCxKaCxJaCxNYixWYSxPZCx3YSxIYSxOZCx3YSxIaCxNZCxGaCxNYSxMYSxLYixkYixMYixEaCxDaCxNYixPZCx3YSxIYSxOZCxNZCxCaCxNYSxMYSxLYixkYixKYixNYSxMYSxVYSx3YSxVYSxzYyxIYSxFaCxBaCx0aCxzaCx6aCx5aCxQZCx3aCx2aCx1aCxwaCxQYSx3YSx3YSxxaCx3Zyx2Zyx3YSxVYSxIYSxIYSxraCxqaCxuaCxvaCxsaCxoaCxnaCxmaCxtaCxuZCxhaCxFZCxEZCxDZCxCZCxlaCx3YSxWYSxwZCxtZCwkZyxFZCxEZCxDZCxCZCxjaCx3YSxWYSxwZCx6ZCxfZyxBZCxaZyxZZyxWZyxUZyxTZyxSZyxXZyxRZyxYZyxQZyxPZyxNZyxLZyxKZyxJZyxOZyxIZyxHZyxGZyxFZyxEZyxBZyxCZyxDZyxNYSxMYSx6Zyx5ZyxQYSxVYSx3YSxVZSxUZSxTZSxSZSx1YixQZSxKYyxJYyxNYSxMYSxMZSxLZSxWYSxKZSxNYSxMYSxFYyxFYyxJZSxCZSxEZSxIZSxMYSxDZSxFZSxHZV0pO2Z1bmN0aW9uIGNhKCl7cmV0dXJuIEEuYnl0ZUxlbmd0aC82NTUzNnwwfWZ1bmN0aW9uIGhhKGlhKXtpYT1pYXwwO3ZhciBkYT1jYSgpfDA7dmFyIGVhPWRhK2lhfDA7aWYoZGE8ZWEmJmVhPDY1NTM2KXt2YXIgZmE9bmV3IEFycmF5QnVmZmVyKEooZWEsNjU1MzYpKTt2YXIgZ2E9bmV3IEludDhBcnJheShmYSk7Z2Euc2V0KEIpO0I9bmV3IEludDhBcnJheShmYSk7Qz1uZXcgSW50MTZBcnJheShmYSk7RD1uZXcgSW50MzJBcnJheShmYSk7RT1uZXcgVWludDhBcnJheShmYSk7Rj1uZXcgVWludDE2QXJyYXkoZmEpO0c9bmV3IFVpbnQzMkFycmF5KGZhKTtIPW5ldyBGbG9hdDMyQXJyYXkoZmEpO0k9bmV3IEZsb2F0NjRBcnJheShmYSk7QT1mYTt6LmJ1ZmZlcj1BO2U9RX1yZXR1cm4gZGF9cmV0dXJueyJnIjpWZCwiaCI6YmEsImkiOmJjLCJqIjp1ZywiayI6dGcsImwiOmJjLCJtIjpzZywibiI6ZWQsIm8iOnJnLCJwIjpxZywicSI6YmMsInIiOnBnLCJzIjphYywidCI6b2csInUiOm5nLCJ2IjpkYiwidyI6bGcsIngiOmtnLCJ5IjpqZywieiI6aWcsIkEiOmhnLCJCIjp5YSwiQyI6Z2csIkQiOmRkLCJFIjpWYSwiRiI6ZmcsIkciOmVnLCJIIjp4YiwiSSI6ZGcsIkoiOmRkLCJLIjpWYSwiTCI6eGIsIk0iOmNnLCJOIjpjZCwiTyI6YWMsIlAiOnhiLCJRIjpiZywiUiI6YWcsIlMiOmNkLCJUIjphYywiVSI6eGIsIlYiOiRmLCJXIjpfZiwiWCI6ZWQsIlkiOlpmLCJaIjpZZiwiXyI6WGYsIiQiOlphLCJhYSI6V2YsImJhIjpfYiwiY2EiOllhLCJkYSI6WmEsImVhIjpWZiwiZmEiOmJkLCJnYSI6WWEsImhhIjpaYSwiaWEiOlVmLCJqYSI6YmQsImthIjpZYSwibGEiOlphLCJtYSI6VGYsIm5hIjphZCwib2EiOllhLCJwYSI6WmEsInFhIjpTZiwicmEiOmFkLCJzYSI6WWEsInRhIjpaYSwidWEiOiRjLCJ2YSI6X2IsIndhIjpZYSwieGEiOlphLCJ5YSI6JGMsInphIjpfYiwiQWEiOllhLCJCYSI6UWYsIkNhIjpQZiwiRGEiOk9mLCJFYSI6TmYsIkZhIjpNZiwiR2EiOkxmLCJIYSI6S2YsIklhIjpKZiwiSmEiOklmLCJLYSI6SGYsIkxhIjpHZiwiTWEiOkZmLCJOYSI6RWYsIk9hIjpEZiwiUGEiOkNmLCJRYSI6QmYsIlJhIjpBZiwiU2EiOnpmLCJUYSI6eWYsIlVhIjp4ZiwiVmEiOndmLCJXYSI6dmYsIlhhIjp1ZiwiWWEiOnRmLCJaYSI6c2YsIl9hIjpfYywiJGEiOnJmLCJhYiI6cWYsImJiIjpwZiwiY2IiOm5mLCJkYiI6X2MsImViIjptZiwiZmIiOmxmLCJnYiI6a2YsImhiIjpqZiwiaWIiOmhmLCJqYiI6Z2YsImtiIjpmZiwibGIiOndiLCJtYiI6a2IsIm5iIjp2Yiwib2IiOldiLCJwYiI6d2IsInFiIjprYiwicmIiOnZiLCJzYiI6V2IsInRiIjpZYywidWIiOlhjLCJ2YiI6d2IsIndiIjprYiwieGIiOnZiLCJ5YiI6a2IsInpiIjp2YiwiQWIiOldiLCJCYiI6WWMsIkNiIjpYYywiRGIiOmVmLCJFYiI6ZGYsIkZiIjpjZiwiR2IiOmJmLCJIYiI6YWYsIkliIjokZSwiSmIiOl9lLCJLYiI6WmUsIkxiIjprYiwiTWIiOndiLCJOYiI6WWUsIk9iIjpYZSwiUGIiOldlLCJRYiI6VmUsIlJiIjpBYywiU2IiOm1hfX1yZXR1cm4gamEobGEpfQ0KLy8gRU1TQ1JJUFRFTl9FTkRfQVNNDQoNCg0KDQoNCikoYXNtTGlicmFyeUFyZyl9LGluc3RhbnRpYXRlOmZ1bmN0aW9uKGJpbmFyeSxpbmZvKXtyZXR1cm57dGhlbjpmdW5jdGlvbihvayl7dmFyIG1vZHVsZT1uZXcgV2ViQXNzZW1ibHkuTW9kdWxlKGJpbmFyeSk7b2soeyJpbnN0YW5jZSI6bmV3IFdlYkFzc2VtYmx5Lkluc3RhbmNlKG1vZHVsZSl9KX19fSxSdW50aW1lRXJyb3I6RXJyb3J9O3dhc21CaW5hcnk9W107aWYodHlwZW9mIFdlYkFzc2VtYmx5IT09Im9iamVjdCIpe2Fib3J0KCJubyBuYXRpdmUgd2FzbSBzdXBwb3J0IGRldGVjdGVkIil9dmFyIHdhc21NZW1vcnk7dmFyIEFCT1JUPWZhbHNlO3ZhciBFWElUU1RBVFVTO2Z1bmN0aW9uIGFzc2VydChjb25kaXRpb24sdGV4dCl7aWYoIWNvbmRpdGlvbil7YWJvcnQoIkFzc2VydGlvbiBmYWlsZWQ6ICIrdGV4dCl9fXZhciBVVEY4RGVjb2Rlcj10eXBlb2YgVGV4dERlY29kZXIhPT0idW5kZWZpbmVkIj9uZXcgVGV4dERlY29kZXIoInV0ZjgiKTp1bmRlZmluZWQ7ZnVuY3Rpb24gVVRGOEFycmF5VG9TdHJpbmcoaGVhcCxpZHgsbWF4Qnl0ZXNUb1JlYWQpe3ZhciBlbmRJZHg9aWR4K21heEJ5dGVzVG9SZWFkO3ZhciBlbmRQdHI9aWR4O3doaWxlKGhlYXBbZW5kUHRyXSYmIShlbmRQdHI+PWVuZElkeCkpKytlbmRQdHI7aWYoZW5kUHRyLWlkeD4xNiYmaGVhcC5zdWJhcnJheSYmVVRGOERlY29kZXIpe3JldHVybiBVVEY4RGVjb2Rlci5kZWNvZGUoaGVhcC5zdWJhcnJheShpZHgsZW5kUHRyKSl9ZWxzZXt2YXIgc3RyPSIiO3doaWxlKGlkeDxlbmRQdHIpe3ZhciB1MD1oZWFwW2lkeCsrXTtpZighKHUwJjEyOCkpe3N0cis9U3RyaW5nLmZyb21DaGFyQ29kZSh1MCk7Y29udGludWV9dmFyIHUxPWhlYXBbaWR4KytdJjYzO2lmKCh1MCYyMjQpPT0xOTIpe3N0cis9U3RyaW5nLmZyb21DaGFyQ29kZSgodTAmMzEpPDw2fHUxKTtjb250aW51ZX12YXIgdTI9aGVhcFtpZHgrK10mNjM7aWYoKHUwJjI0MCk9PTIyNCl7dTA9KHUwJjE1KTw8MTJ8dTE8PDZ8dTJ9ZWxzZXt1MD0odTAmNyk8PDE4fHUxPDwxMnx1Mjw8NnxoZWFwW2lkeCsrXSY2M31pZih1MDw2NTUzNil7c3RyKz1TdHJpbmcuZnJvbUNoYXJDb2RlKHUwKX1lbHNle3ZhciBjaD11MC02NTUzNjtzdHIrPVN0cmluZy5mcm9tQ2hhckNvZGUoNTUyOTZ8Y2g+PjEwLDU2MzIwfGNoJjEwMjMpfX19cmV0dXJuIHN0cn1mdW5jdGlvbiBVVEY4VG9TdHJpbmcocHRyLG1heEJ5dGVzVG9SZWFkKXtyZXR1cm4gcHRyP1VURjhBcnJheVRvU3RyaW5nKEhFQVBVOCxwdHIsbWF4Qnl0ZXNUb1JlYWQpOiIifWZ1bmN0aW9uIHN0cmluZ1RvVVRGOEFycmF5KHN0cixoZWFwLG91dElkeCxtYXhCeXRlc1RvV3JpdGUpe2lmKCEobWF4Qnl0ZXNUb1dyaXRlPjApKXJldHVybiAwO3ZhciBzdGFydElkeD1vdXRJZHg7dmFyIGVuZElkeD1vdXRJZHgrbWF4Qnl0ZXNUb1dyaXRlLTE7Zm9yKHZhciBpPTA7aTxzdHIubGVuZ3RoOysraSl7dmFyIHU9c3RyLmNoYXJDb2RlQXQoaSk7aWYodT49NTUyOTYmJnU8PTU3MzQzKXt2YXIgdTE9c3RyLmNoYXJDb2RlQXQoKytpKTt1PTY1NTM2KygodSYxMDIzKTw8MTApfHUxJjEwMjN9aWYodTw9MTI3KXtpZihvdXRJZHg+PWVuZElkeClicmVhaztoZWFwW291dElkeCsrXT11fWVsc2UgaWYodTw9MjA0Nyl7aWYob3V0SWR4KzE+PWVuZElkeClicmVhaztoZWFwW291dElkeCsrXT0xOTJ8dT4+NjtoZWFwW291dElkeCsrXT0xMjh8dSY2M31lbHNlIGlmKHU8PTY1NTM1KXtpZihvdXRJZHgrMj49ZW5kSWR4KWJyZWFrO2hlYXBbb3V0SWR4KytdPTIyNHx1Pj4xMjtoZWFwW291dElkeCsrXT0xMjh8dT4+NiY2MztoZWFwW291dElkeCsrXT0xMjh8dSY2M31lbHNle2lmKG91dElkeCszPj1lbmRJZHgpYnJlYWs7aGVhcFtvdXRJZHgrK109MjQwfHU+PjE4O2hlYXBbb3V0SWR4KytdPTEyOHx1Pj4xMiY2MztoZWFwW291dElkeCsrXT0xMjh8dT4+NiY2MztoZWFwW291dElkeCsrXT0xMjh8dSY2M319aGVhcFtvdXRJZHhdPTA7cmV0dXJuIG91dElkeC1zdGFydElkeH1mdW5jdGlvbiBsZW5ndGhCeXRlc1VURjgoc3RyKXt2YXIgbGVuPTA7Zm9yKHZhciBpPTA7aTxzdHIubGVuZ3RoOysraSl7dmFyIHU9c3RyLmNoYXJDb2RlQXQoaSk7aWYodT49NTUyOTYmJnU8PTU3MzQzKXU9NjU1MzYrKCh1JjEwMjMpPDwxMCl8c3RyLmNoYXJDb2RlQXQoKytpKSYxMDIzO2lmKHU8PTEyNykrK2xlbjtlbHNlIGlmKHU8PTIwNDcpbGVuKz0yO2Vsc2UgaWYodTw9NjU1MzUpbGVuKz0zO2Vsc2UgbGVuKz00fXJldHVybiBsZW59ZnVuY3Rpb24gYWxpZ25VcCh4LG11bHRpcGxlKXtpZih4JW11bHRpcGxlPjApe3grPW11bHRpcGxlLXglbXVsdGlwbGV9cmV0dXJuIHh9dmFyIGJ1ZmZlcixIRUFQOCxIRUFQVTgsSEVBUDE2LEhFQVBVMTYsSEVBUDMyLEhFQVBVMzIsSEVBUEYzMixIRUFQRjY0O2Z1bmN0aW9uIHVwZGF0ZUdsb2JhbEJ1ZmZlckFuZFZpZXdzKGJ1Zil7YnVmZmVyPWJ1ZjtNb2R1bGVbIkhFQVA4Il09SEVBUDg9bmV3IEludDhBcnJheShidWYpO01vZHVsZVsiSEVBUDE2Il09SEVBUDE2PW5ldyBJbnQxNkFycmF5KGJ1Zik7TW9kdWxlWyJIRUFQMzIiXT1IRUFQMzI9bmV3IEludDMyQXJyYXkoYnVmKTtNb2R1bGVbIkhFQVBVOCJdPUhFQVBVOD1uZXcgVWludDhBcnJheShidWYpO01vZHVsZVsiSEVBUFUxNiJdPUhFQVBVMTY9bmV3IFVpbnQxNkFycmF5KGJ1Zik7TW9kdWxlWyJIRUFQVTMyIl09SEVBUFUzMj1uZXcgVWludDMyQXJyYXkoYnVmKTtNb2R1bGVbIkhFQVBGMzIiXT1IRUFQRjMyPW5ldyBGbG9hdDMyQXJyYXkoYnVmKTtNb2R1bGVbIkhFQVBGNjQiXT1IRUFQRjY0PW5ldyBGbG9hdDY0QXJyYXkoYnVmKX12YXIgSU5JVElBTF9NRU1PUlk9TW9kdWxlWyJJTklUSUFMX01FTU9SWSJdfHwxNjc3NzIxNjtpZihNb2R1bGVbIndhc21NZW1vcnkiXSl7d2FzbU1lbW9yeT1Nb2R1bGVbIndhc21NZW1vcnkiXX1lbHNle3dhc21NZW1vcnk9bmV3IFdlYkFzc2VtYmx5Lk1lbW9yeSh7ImluaXRpYWwiOklOSVRJQUxfTUVNT1JZLzY1NTM2LCJtYXhpbXVtIjoyMTQ3NDgzNjQ4LzY1NTM2fSl9aWYod2FzbU1lbW9yeSl7YnVmZmVyPXdhc21NZW1vcnkuYnVmZmVyfUlOSVRJQUxfTUVNT1JZPWJ1ZmZlci5ieXRlTGVuZ3RoO3VwZGF0ZUdsb2JhbEJ1ZmZlckFuZFZpZXdzKGJ1ZmZlcik7dmFyIHdhc21UYWJsZTt2YXIgX19BVFBSRVJVTl9fPVtdO3ZhciBfX0FUSU5JVF9fPVtdO3ZhciBfX0FUUE9TVFJVTl9fPVtdO3ZhciBydW50aW1lSW5pdGlhbGl6ZWQ9ZmFsc2U7dmFyIHJ1bnRpbWVLZWVwYWxpdmVDb3VudGVyPTA7ZnVuY3Rpb24ga2VlcFJ1bnRpbWVBbGl2ZSgpe3JldHVybiBub0V4aXRSdW50aW1lfHxydW50aW1lS2VlcGFsaXZlQ291bnRlcj4wfWZ1bmN0aW9uIHByZVJ1bigpe2lmKE1vZHVsZVsicHJlUnVuIl0pe2lmKHR5cGVvZiBNb2R1bGVbInByZVJ1biJdPT0iZnVuY3Rpb24iKU1vZHVsZVsicHJlUnVuIl09W01vZHVsZVsicHJlUnVuIl1dO3doaWxlKE1vZHVsZVsicHJlUnVuIl0ubGVuZ3RoKXthZGRPblByZVJ1bihNb2R1bGVbInByZVJ1biJdLnNoaWZ0KCkpfX1jYWxsUnVudGltZUNhbGxiYWNrcyhfX0FUUFJFUlVOX18pfWZ1bmN0aW9uIGluaXRSdW50aW1lKCl7cnVudGltZUluaXRpYWxpemVkPXRydWU7Y2FsbFJ1bnRpbWVDYWxsYmFja3MoX19BVElOSVRfXyl9ZnVuY3Rpb24gcG9zdFJ1bigpe2lmKE1vZHVsZVsicG9zdFJ1biJdKXtpZih0eXBlb2YgTW9kdWxlWyJwb3N0UnVuIl09PSJmdW5jdGlvbiIpTW9kdWxlWyJwb3N0UnVuIl09W01vZHVsZVsicG9zdFJ1biJdXTt3aGlsZShNb2R1bGVbInBvc3RSdW4iXS5sZW5ndGgpe2FkZE9uUG9zdFJ1bihNb2R1bGVbInBvc3RSdW4iXS5zaGlmdCgpKX19Y2FsbFJ1bnRpbWVDYWxsYmFja3MoX19BVFBPU1RSVU5fXyl9ZnVuY3Rpb24gYWRkT25QcmVSdW4oY2Ipe19fQVRQUkVSVU5fXy51bnNoaWZ0KGNiKX1mdW5jdGlvbiBhZGRPbkluaXQoY2Ipe19fQVRJTklUX18udW5zaGlmdChjYil9ZnVuY3Rpb24gYWRkT25Qb3N0UnVuKGNiKXtfX0FUUE9TVFJVTl9fLnVuc2hpZnQoY2IpfXZhciBydW5EZXBlbmRlbmNpZXM9MDt2YXIgcnVuRGVwZW5kZW5jeVdhdGNoZXI9bnVsbDt2YXIgZGVwZW5kZW5jaWVzRnVsZmlsbGVkPW51bGw7ZnVuY3Rpb24gYWRkUnVuRGVwZW5kZW5jeShpZCl7cnVuRGVwZW5kZW5jaWVzKys7aWYoTW9kdWxlWyJtb25pdG9yUnVuRGVwZW5kZW5jaWVzIl0pe01vZHVsZVsibW9uaXRvclJ1bkRlcGVuZGVuY2llcyJdKHJ1bkRlcGVuZGVuY2llcyl9fWZ1bmN0aW9uIHJlbW92ZVJ1bkRlcGVuZGVuY3koaWQpe3J1bkRlcGVuZGVuY2llcy0tO2lmKE1vZHVsZVsibW9uaXRvclJ1bkRlcGVuZGVuY2llcyJdKXtNb2R1bGVbIm1vbml0b3JSdW5EZXBlbmRlbmNpZXMiXShydW5EZXBlbmRlbmNpZXMpfWlmKHJ1bkRlcGVuZGVuY2llcz09MCl7aWYocnVuRGVwZW5kZW5jeVdhdGNoZXIhPT1udWxsKXtjbGVhckludGVydmFsKHJ1bkRlcGVuZGVuY3lXYXRjaGVyKTtydW5EZXBlbmRlbmN5V2F0Y2hlcj1udWxsfWlmKGRlcGVuZGVuY2llc0Z1bGZpbGxlZCl7dmFyIGNhbGxiYWNrPWRlcGVuZGVuY2llc0Z1bGZpbGxlZDtkZXBlbmRlbmNpZXNGdWxmaWxsZWQ9bnVsbDtjYWxsYmFjaygpfX19TW9kdWxlWyJwcmVsb2FkZWRJbWFnZXMiXT17fTtNb2R1bGVbInByZWxvYWRlZEF1ZGlvcyJdPXt9O2Z1bmN0aW9uIGFib3J0KHdoYXQpe3tpZihNb2R1bGVbIm9uQWJvcnQiXSl7TW9kdWxlWyJvbkFib3J0Il0od2hhdCl9fXdoYXQ9IkFib3J0ZWQoIit3aGF0KyIpIjtlcnIod2hhdCk7QUJPUlQ9dHJ1ZTtFWElUU1RBVFVTPTE7d2hhdCs9Ii4gQnVpbGQgd2l0aCAtcyBBU1NFUlRJT05TPTEgZm9yIG1vcmUgaW5mby4iO3ZhciBlPW5ldyBXZWJBc3NlbWJseS5SdW50aW1lRXJyb3Iod2hhdCk7cmVhZHlQcm9taXNlUmVqZWN0KGUpO3Rocm93IGV9dmFyIGRhdGFVUklQcmVmaXg9ImRhdGE6YXBwbGljYXRpb24vb2N0ZXQtc3RyZWFtO2Jhc2U2NCwiO2Z1bmN0aW9uIGlzRGF0YVVSSShmaWxlbmFtZSl7cmV0dXJuIGZpbGVuYW1lLnN0YXJ0c1dpdGgoZGF0YVVSSVByZWZpeCl9ZnVuY3Rpb24gaXNGaWxlVVJJKGZpbGVuYW1lKXtyZXR1cm4gZmlsZW5hbWUuc3RhcnRzV2l0aCgiZmlsZTovLyIpfXZhciB3YXNtQmluYXJ5RmlsZTt3YXNtQmluYXJ5RmlsZT0iZHJhY29fZGVjb2Rlcl9nbHRmLndhc20iO2lmKCFpc0RhdGFVUkkod2FzbUJpbmFyeUZpbGUpKXt3YXNtQmluYXJ5RmlsZT1sb2NhdGVGaWxlKHdhc21CaW5hcnlGaWxlKX1mdW5jdGlvbiBnZXRCaW5hcnkoZmlsZSl7dHJ5e2lmKGZpbGU9PXdhc21CaW5hcnlGaWxlJiZ3YXNtQmluYXJ5KXtyZXR1cm4gbmV3IFVpbnQ4QXJyYXkod2FzbUJpbmFyeSl9dmFyIGJpbmFyeT10cnlQYXJzZUFzRGF0YVVSSShmaWxlKTtpZihiaW5hcnkpe3JldHVybiBiaW5hcnl9aWYocmVhZEJpbmFyeSl7cmV0dXJuIHJlYWRCaW5hcnkoZmlsZSl9ZWxzZXt0aHJvdyJib3RoIGFzeW5jIGFuZCBzeW5jIGZldGNoaW5nIG9mIHRoZSB3YXNtIGZhaWxlZCJ9fWNhdGNoKGVycil7YWJvcnQoZXJyKX19ZnVuY3Rpb24gZ2V0QmluYXJ5UHJvbWlzZSgpe2lmKCF3YXNtQmluYXJ5JiYoRU5WSVJPTk1FTlRfSVNfV0VCfHxFTlZJUk9OTUVOVF9JU19XT1JLRVIpKXtpZih0eXBlb2YgZmV0Y2g9PT0iZnVuY3Rpb24iJiYhaXNGaWxlVVJJKHdhc21CaW5hcnlGaWxlKSl7cmV0dXJuIGZldGNoKHdhc21CaW5hcnlGaWxlLHtjcmVkZW50aWFsczoic2FtZS1vcmlnaW4ifSkudGhlbihmdW5jdGlvbihyZXNwb25zZSl7aWYoIXJlc3BvbnNlWyJvayJdKXt0aHJvdyJmYWlsZWQgdG8gbG9hZCB3YXNtIGJpbmFyeSBmaWxlIGF0ICciK3dhc21CaW5hcnlGaWxlKyInIn1yZXR1cm4gcmVzcG9uc2VbImFycmF5QnVmZmVyIl0oKX0pLmNhdGNoKGZ1bmN0aW9uKCl7cmV0dXJuIGdldEJpbmFyeSh3YXNtQmluYXJ5RmlsZSl9KX1lbHNle2lmKHJlYWRBc3luYyl7cmV0dXJuIG5ldyBQcm9taXNlKGZ1bmN0aW9uKHJlc29sdmUscmVqZWN0KXtyZWFkQXN5bmMod2FzbUJpbmFyeUZpbGUsZnVuY3Rpb24ocmVzcG9uc2Upe3Jlc29sdmUobmV3IFVpbnQ4QXJyYXkocmVzcG9uc2UpKX0scmVqZWN0KX0pfX19cmV0dXJuIFByb21pc2UucmVzb2x2ZSgpLnRoZW4oZnVuY3Rpb24oKXtyZXR1cm4gZ2V0QmluYXJ5KHdhc21CaW5hcnlGaWxlKX0pfWZ1bmN0aW9uIGNyZWF0ZVdhc20oKXt2YXIgaW5mbz17ImEiOmFzbUxpYnJhcnlBcmd9O2Z1bmN0aW9uIHJlY2VpdmVJbnN0YW5jZShpbnN0YW5jZSxtb2R1bGUpe3ZhciBleHBvcnRzPWluc3RhbmNlLmV4cG9ydHM7TW9kdWxlWyJhc20iXT1leHBvcnRzO3dhc21UYWJsZT1Nb2R1bGVbImFzbSJdWyJoIl07YWRkT25Jbml0KE1vZHVsZVsiYXNtIl1bImciXSk7cmVtb3ZlUnVuRGVwZW5kZW5jeSgid2FzbS1pbnN0YW50aWF0ZSIpfWFkZFJ1bkRlcGVuZGVuY3koIndhc20taW5zdGFudGlhdGUiKTtmdW5jdGlvbiByZWNlaXZlSW5zdGFudGlhdGlvblJlc3VsdChyZXN1bHQpe3JlY2VpdmVJbnN0YW5jZShyZXN1bHRbImluc3RhbmNlIl0pfWZ1bmN0aW9uIGluc3RhbnRpYXRlQXJyYXlCdWZmZXIocmVjZWl2ZXIpe3JldHVybiBnZXRCaW5hcnlQcm9taXNlKCkudGhlbihmdW5jdGlvbihiaW5hcnkpe3JldHVybiBXZWJBc3NlbWJseS5pbnN0YW50aWF0ZShiaW5hcnksaW5mbyl9KS50aGVuKGZ1bmN0aW9uKGluc3RhbmNlKXtyZXR1cm4gaW5zdGFuY2V9KS50aGVuKHJlY2VpdmVyLGZ1bmN0aW9uKHJlYXNvbil7ZXJyKCJmYWlsZWQgdG8gYXN5bmNocm9ub3VzbHkgcHJlcGFyZSB3YXNtOiAiK3JlYXNvbik7YWJvcnQocmVhc29uKX0pfWZ1bmN0aW9uIGluc3RhbnRpYXRlQXN5bmMoKXtpZighd2FzbUJpbmFyeSYmdHlwZW9mIFdlYkFzc2VtYmx5Lmluc3RhbnRpYXRlU3RyZWFtaW5nPT09ImZ1bmN0aW9uIiYmIWlzRGF0YVVSSSh3YXNtQmluYXJ5RmlsZSkmJiFpc0ZpbGVVUkkod2FzbUJpbmFyeUZpbGUpJiZ0eXBlb2YgZmV0Y2g9PT0iZnVuY3Rpb24iKXtyZXR1cm4gZmV0Y2god2FzbUJpbmFyeUZpbGUse2NyZWRlbnRpYWxzOiJzYW1lLW9yaWdpbiJ9KS50aGVuKGZ1bmN0aW9uKHJlc3BvbnNlKXt2YXIgcmVzdWx0PVdlYkFzc2VtYmx5Lmluc3RhbnRpYXRlU3RyZWFtaW5nKHJlc3BvbnNlLGluZm8pO3JldHVybiByZXN1bHQudGhlbihyZWNlaXZlSW5zdGFudGlhdGlvblJlc3VsdCxmdW5jdGlvbihyZWFzb24pe2Vycigid2FzbSBzdHJlYW1pbmcgY29tcGlsZSBmYWlsZWQ6ICIrcmVhc29uKTtlcnIoImZhbGxpbmcgYmFjayB0byBBcnJheUJ1ZmZlciBpbnN0YW50aWF0aW9uIik7cmV0dXJuIGluc3RhbnRpYXRlQXJyYXlCdWZmZXIocmVjZWl2ZUluc3RhbnRpYXRpb25SZXN1bHQpfSl9KX1lbHNle3JldHVybiBpbnN0YW50aWF0ZUFycmF5QnVmZmVyKHJlY2VpdmVJbnN0YW50aWF0aW9uUmVzdWx0KX19aWYoTW9kdWxlWyJpbnN0YW50aWF0ZVdhc20iXSl7dHJ5e3ZhciBleHBvcnRzPU1vZHVsZVsiaW5zdGFudGlhdGVXYXNtIl0oaW5mbyxyZWNlaXZlSW5zdGFuY2UpO3JldHVybiBleHBvcnRzfWNhdGNoKGUpe2VycigiTW9kdWxlLmluc3RhbnRpYXRlV2FzbSBjYWxsYmFjayBmYWlsZWQgd2l0aCBlcnJvcjogIitlKTtyZXR1cm4gZmFsc2V9fWluc3RhbnRpYXRlQXN5bmMoKS5jYXRjaChyZWFkeVByb21pc2VSZWplY3QpO3JldHVybnt9fWZ1bmN0aW9uIGNhbGxSdW50aW1lQ2FsbGJhY2tzKGNhbGxiYWNrcyl7d2hpbGUoY2FsbGJhY2tzLmxlbmd0aD4wKXt2YXIgY2FsbGJhY2s9Y2FsbGJhY2tzLnNoaWZ0KCk7aWYodHlwZW9mIGNhbGxiYWNrPT0iZnVuY3Rpb24iKXtjYWxsYmFjayhNb2R1bGUpO2NvbnRpbnVlfXZhciBmdW5jPWNhbGxiYWNrLmZ1bmM7aWYodHlwZW9mIGZ1bmM9PT0ibnVtYmVyIil7aWYoY2FsbGJhY2suYXJnPT09dW5kZWZpbmVkKXtnZXRXYXNtVGFibGVFbnRyeShmdW5jKSgpfWVsc2V7Z2V0V2FzbVRhYmxlRW50cnkoZnVuYykoY2FsbGJhY2suYXJnKX19ZWxzZXtmdW5jKGNhbGxiYWNrLmFyZz09PXVuZGVmaW5lZD9udWxsOmNhbGxiYWNrLmFyZyl9fX12YXIgd2FzbVRhYmxlTWlycm9yPVtdO2Z1bmN0aW9uIGdldFdhc21UYWJsZUVudHJ5KGZ1bmNQdHIpe3ZhciBmdW5jPXdhc21UYWJsZU1pcnJvcltmdW5jUHRyXTtpZighZnVuYyl7aWYoZnVuY1B0cj49d2FzbVRhYmxlTWlycm9yLmxlbmd0aCl3YXNtVGFibGVNaXJyb3IubGVuZ3RoPWZ1bmNQdHIrMTt3YXNtVGFibGVNaXJyb3JbZnVuY1B0cl09ZnVuYz13YXNtVGFibGUuZ2V0KGZ1bmNQdHIpfXJldHVybiBmdW5jfWZ1bmN0aW9uIF9fX2N4YV9hbGxvY2F0ZV9leGNlcHRpb24oc2l6ZSl7cmV0dXJuIF9tYWxsb2Moc2l6ZSsxNikrMTZ9ZnVuY3Rpb24gRXhjZXB0aW9uSW5mbyhleGNQdHIpe3RoaXMuZXhjUHRyPWV4Y1B0cjt0aGlzLnB0cj1leGNQdHItMTY7dGhpcy5zZXRfdHlwZT1mdW5jdGlvbih0eXBlKXtIRUFQMzJbdGhpcy5wdHIrND4+Ml09dHlwZX07dGhpcy5nZXRfdHlwZT1mdW5jdGlvbigpe3JldHVybiBIRUFQMzJbdGhpcy5wdHIrND4+Ml19O3RoaXMuc2V0X2Rlc3RydWN0b3I9ZnVuY3Rpb24oZGVzdHJ1Y3Rvcil7SEVBUDMyW3RoaXMucHRyKzg+PjJdPWRlc3RydWN0b3J9O3RoaXMuZ2V0X2Rlc3RydWN0b3I9ZnVuY3Rpb24oKXtyZXR1cm4gSEVBUDMyW3RoaXMucHRyKzg+PjJdfTt0aGlzLnNldF9yZWZjb3VudD1mdW5jdGlvbihyZWZjb3VudCl7SEVBUDMyW3RoaXMucHRyPj4yXT1yZWZjb3VudH07dGhpcy5zZXRfY2F1Z2h0PWZ1bmN0aW9uKGNhdWdodCl7Y2F1Z2h0PWNhdWdodD8xOjA7SEVBUDhbdGhpcy5wdHIrMTI+PjBdPWNhdWdodH07dGhpcy5nZXRfY2F1Z2h0PWZ1bmN0aW9uKCl7cmV0dXJuIEhFQVA4W3RoaXMucHRyKzEyPj4wXSE9MH07dGhpcy5zZXRfcmV0aHJvd249ZnVuY3Rpb24ocmV0aHJvd24pe3JldGhyb3duPXJldGhyb3duPzE6MDtIRUFQOFt0aGlzLnB0cisxMz4+MF09cmV0aHJvd259O3RoaXMuZ2V0X3JldGhyb3duPWZ1bmN0aW9uKCl7cmV0dXJuIEhFQVA4W3RoaXMucHRyKzEzPj4wXSE9MH07dGhpcy5pbml0PWZ1bmN0aW9uKHR5cGUsZGVzdHJ1Y3Rvcil7dGhpcy5zZXRfdHlwZSh0eXBlKTt0aGlzLnNldF9kZXN0cnVjdG9yKGRlc3RydWN0b3IpO3RoaXMuc2V0X3JlZmNvdW50KDApO3RoaXMuc2V0X2NhdWdodChmYWxzZSk7dGhpcy5zZXRfcmV0aHJvd24oZmFsc2UpfTt0aGlzLmFkZF9yZWY9ZnVuY3Rpb24oKXt2YXIgdmFsdWU9SEVBUDMyW3RoaXMucHRyPj4yXTtIRUFQMzJbdGhpcy5wdHI+PjJdPXZhbHVlKzF9O3RoaXMucmVsZWFzZV9yZWY9ZnVuY3Rpb24oKXt2YXIgcHJldj1IRUFQMzJbdGhpcy5wdHI+PjJdO0hFQVAzMlt0aGlzLnB0cj4+Ml09cHJldi0xO3JldHVybiBwcmV2PT09MX19dmFyIGV4Y2VwdGlvbkxhc3Q9MDt2YXIgdW5jYXVnaHRFeGNlcHRpb25Db3VudD0wO2Z1bmN0aW9uIF9fX2N4YV90aHJvdyhwdHIsdHlwZSxkZXN0cnVjdG9yKXt2YXIgaW5mbz1uZXcgRXhjZXB0aW9uSW5mbyhwdHIpO2luZm8uaW5pdCh0eXBlLGRlc3RydWN0b3IpO2V4Y2VwdGlvbkxhc3Q9cHRyO3VuY2F1Z2h0RXhjZXB0aW9uQ291bnQrKzt0aHJvdyBwdHJ9ZnVuY3Rpb24gX2Fib3J0KCl7YWJvcnQoIiIpfWZ1bmN0aW9uIF9lbXNjcmlwdGVuX21lbWNweV9iaWcoZGVzdCxzcmMsbnVtKXtIRUFQVTguY29weVdpdGhpbihkZXN0LHNyYyxzcmMrbnVtKX1mdW5jdGlvbiBlbXNjcmlwdGVuX3JlYWxsb2NfYnVmZmVyKHNpemUpe3RyeXt3YXNtTWVtb3J5Lmdyb3coc2l6ZS1idWZmZXIuYnl0ZUxlbmd0aCs2NTUzNT4+PjE2KTt1cGRhdGVHbG9iYWxCdWZmZXJBbmRWaWV3cyh3YXNtTWVtb3J5LmJ1ZmZlcik7cmV0dXJuIDF9Y2F0Y2goZSl7fX1mdW5jdGlvbiBfZW1zY3JpcHRlbl9yZXNpemVfaGVhcChyZXF1ZXN0ZWRTaXplKXt2YXIgb2xkU2l6ZT1IRUFQVTgubGVuZ3RoO3JlcXVlc3RlZFNpemU9cmVxdWVzdGVkU2l6ZT4+PjA7dmFyIG1heEhlYXBTaXplPTIxNDc0ODM2NDg7aWYocmVxdWVzdGVkU2l6ZT5tYXhIZWFwU2l6ZSl7cmV0dXJuIGZhbHNlfWZvcih2YXIgY3V0RG93bj0xO2N1dERvd248PTQ7Y3V0RG93bio9Mil7dmFyIG92ZXJHcm93bkhlYXBTaXplPW9sZFNpemUqKDErLjIvY3V0RG93bik7b3Zlckdyb3duSGVhcFNpemU9TWF0aC5taW4ob3Zlckdyb3duSGVhcFNpemUscmVxdWVzdGVkU2l6ZSsxMDA2NjMyOTYpO3ZhciBuZXdTaXplPU1hdGgubWluKG1heEhlYXBTaXplLGFsaWduVXAoTWF0aC5tYXgocmVxdWVzdGVkU2l6ZSxvdmVyR3Jvd25IZWFwU2l6ZSksNjU1MzYpKTt2YXIgcmVwbGFjZW1lbnQ9ZW1zY3JpcHRlbl9yZWFsbG9jX2J1ZmZlcihuZXdTaXplKTtpZihyZXBsYWNlbWVudCl7cmV0dXJuIHRydWV9fXJldHVybiBmYWxzZX12YXIgQVNTRVJUSU9OUz1mYWxzZTtmdW5jdGlvbiBpbnRBcnJheUZyb21TdHJpbmcoc3RyaW5neSxkb250QWRkTnVsbCxsZW5ndGgpe3ZhciBsZW49bGVuZ3RoPjA/bGVuZ3RoOmxlbmd0aEJ5dGVzVVRGOChzdHJpbmd5KSsxO3ZhciB1OGFycmF5PW5ldyBBcnJheShsZW4pO3ZhciBudW1CeXRlc1dyaXR0ZW49c3RyaW5nVG9VVEY4QXJyYXkoc3RyaW5neSx1OGFycmF5LDAsdThhcnJheS5sZW5ndGgpO2lmKGRvbnRBZGROdWxsKXU4YXJyYXkubGVuZ3RoPW51bUJ5dGVzV3JpdHRlbjtyZXR1cm4gdThhcnJheX1mdW5jdGlvbiBpbnRBcnJheVRvU3RyaW5nKGFycmF5KXt2YXIgcmV0PVtdO2Zvcih2YXIgaT0wO2k8YXJyYXkubGVuZ3RoO2krKyl7dmFyIGNocj1hcnJheVtpXTtpZihjaHI+MjU1KXtpZihBU1NFUlRJT05TKXthc3NlcnQoZmFsc2UsIkNoYXJhY3RlciBjb2RlICIrY2hyKyIgKCIrU3RyaW5nLmZyb21DaGFyQ29kZShjaHIpKyIpICBhdCBvZmZzZXQgIitpKyIgbm90IGluIDB4MDAtMHhGRi4iKX1jaHImPTI1NX1yZXQucHVzaChTdHJpbmcuZnJvbUNoYXJDb2RlKGNocikpfXJldHVybiByZXQuam9pbigiIil9dmFyIGRlY29kZUJhc2U2ND10eXBlb2YgYXRvYj09PSJmdW5jdGlvbiI/YXRvYjpmdW5jdGlvbihpbnB1dCl7dmFyIGtleVN0cj0iQUJDREVGR0hJSktMTU5PUFFSU1RVVldYWVphYmNkZWZnaGlqa2xtbm9wcXJzdHV2d3h5ejAxMjM0NTY3ODkrLz0iO3ZhciBvdXRwdXQ9IiI7dmFyIGNocjEsY2hyMixjaHIzO3ZhciBlbmMxLGVuYzIsZW5jMyxlbmM0O3ZhciBpPTA7aW5wdXQ9aW5wdXQucmVwbGFjZSgvW15BLVphLXowLTlcK1wvXD1dL2csIiIpO2Rve2VuYzE9a2V5U3RyLmluZGV4T2YoaW5wdXQuY2hhckF0KGkrKykpO2VuYzI9a2V5U3RyLmluZGV4T2YoaW5wdXQuY2hhckF0KGkrKykpO2VuYzM9a2V5U3RyLmluZGV4T2YoaW5wdXQuY2hhckF0KGkrKykpO2VuYzQ9a2V5U3RyLmluZGV4T2YoaW5wdXQuY2hhckF0KGkrKykpO2NocjE9ZW5jMTw8MnxlbmMyPj40O2NocjI9KGVuYzImMTUpPDw0fGVuYzM+PjI7Y2hyMz0oZW5jMyYzKTw8NnxlbmM0O291dHB1dD1vdXRwdXQrU3RyaW5nLmZyb21DaGFyQ29kZShjaHIxKTtpZihlbmMzIT09NjQpe291dHB1dD1vdXRwdXQrU3RyaW5nLmZyb21DaGFyQ29kZShjaHIyKX1pZihlbmM0IT09NjQpe291dHB1dD1vdXRwdXQrU3RyaW5nLmZyb21DaGFyQ29kZShjaHIzKX19d2hpbGUoaTxpbnB1dC5sZW5ndGgpO3JldHVybiBvdXRwdXR9O2Z1bmN0aW9uIGludEFycmF5RnJvbUJhc2U2NChzKXtpZih0eXBlb2YgRU5WSVJPTk1FTlRfSVNfTk9ERT09PSJib29sZWFuIiYmRU5WSVJPTk1FTlRfSVNfTk9ERSl7dmFyIGJ1Zj1CdWZmZXIuZnJvbShzLCJiYXNlNjQiKTtyZXR1cm4gbmV3IFVpbnQ4QXJyYXkoYnVmWyJidWZmZXIiXSxidWZbImJ5dGVPZmZzZXQiXSxidWZbImJ5dGVMZW5ndGgiXSl9dHJ5e3ZhciBkZWNvZGVkPWRlY29kZUJhc2U2NChzKTt2YXIgYnl0ZXM9bmV3IFVpbnQ4QXJyYXkoZGVjb2RlZC5sZW5ndGgpO2Zvcih2YXIgaT0wO2k8ZGVjb2RlZC5sZW5ndGg7KytpKXtieXRlc1tpXT1kZWNvZGVkLmNoYXJDb2RlQXQoaSl9cmV0dXJuIGJ5dGVzfWNhdGNoKF8pe3Rocm93IG5ldyBFcnJvcigiQ29udmVydGluZyBiYXNlNjQgc3RyaW5nIHRvIGJ5dGVzIGZhaWxlZC4iKX19ZnVuY3Rpb24gdHJ5UGFyc2VBc0RhdGFVUkkoZmlsZW5hbWUpe2lmKCFpc0RhdGFVUkkoZmlsZW5hbWUpKXtyZXR1cm59cmV0dXJuIGludEFycmF5RnJvbUJhc2U2NChmaWxlbmFtZS5zbGljZShkYXRhVVJJUHJlZml4Lmxlbmd0aCkpfXZhciBhc21MaWJyYXJ5QXJnPXsiZiI6X19fY3hhX2FsbG9jYXRlX2V4Y2VwdGlvbiwiZSI6X19fY3hhX3Rocm93LCJiIjpfYWJvcnQsImMiOl9lbXNjcmlwdGVuX21lbWNweV9iaWcsImQiOl9lbXNjcmlwdGVuX3Jlc2l6ZV9oZWFwLCJhIjp3YXNtTWVtb3J5fTt2YXIgYXNtPWNyZWF0ZVdhc20oKTt2YXIgX19fd2FzbV9jYWxsX2N0b3JzPU1vZHVsZVsiX19fd2FzbV9jYWxsX2N0b3JzIl09ZnVuY3Rpb24oKXtyZXR1cm4oX19fd2FzbV9jYWxsX2N0b3JzPU1vZHVsZVsiX19fd2FzbV9jYWxsX2N0b3JzIl09TW9kdWxlWyJhc20iXVsiZyJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX1ZvaWRQdHJfX19kZXN0cm95X19fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfVm9pZFB0cl9fX2Rlc3Ryb3lfX18wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9Wb2lkUHRyX19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX1ZvaWRQdHJfX19kZXN0cm95X19fMCJdPU1vZHVsZVsiYXNtIl1bImkiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyQnVmZmVyX0RlY29kZXJCdWZmZXJfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2RlckJ1ZmZlcl9EZWNvZGVyQnVmZmVyXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJCdWZmZXJfRGVjb2RlckJ1ZmZlcl8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyQnVmZmVyX0RlY29kZXJCdWZmZXJfMCJdPU1vZHVsZVsiYXNtIl1bImoiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyQnVmZmVyX0luaXRfMj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2RlckJ1ZmZlcl9Jbml0XzIiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJCdWZmZXJfSW5pdF8yPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyQnVmZmVyX0luaXRfMiJdPU1vZHVsZVsiYXNtIl1bImsiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyQnVmZmVyX19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJCdWZmZXJfX19kZXN0cm95X19fMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRGVjb2RlckJ1ZmZlcl9fX2Rlc3Ryb3lfX18wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyQnVmZmVyX19fZGVzdHJveV9fXzAiXT1Nb2R1bGVbImFzbSJdWyJsIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlVHJhbnNmb3JtRGF0YV9BdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVRyYW5zZm9ybURhdGFfQXR0cmlidXRlVHJhbnNmb3JtRGF0YV8wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhX0F0dHJpYnV0ZVRyYW5zZm9ybURhdGFfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlVHJhbnNmb3JtRGF0YV9BdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhXzAiXT1Nb2R1bGVbImFzbSJdWyJtIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlVHJhbnNmb3JtRGF0YV90cmFuc2Zvcm1fdHlwZV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhX3RyYW5zZm9ybV90eXBlXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVRyYW5zZm9ybURhdGFfdHJhbnNmb3JtX3R5cGVfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlVHJhbnNmb3JtRGF0YV90cmFuc2Zvcm1fdHlwZV8wIl09TW9kdWxlWyJhc20iXVsibiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVRyYW5zZm9ybURhdGFfX19kZXN0cm95X19fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlVHJhbnNmb3JtRGF0YV9fX2Rlc3Ryb3lfX18wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhX19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVRyYW5zZm9ybURhdGFfX19kZXN0cm95X19fMCJdPU1vZHVsZVsiYXNtIl1bIm8iXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9HZW9tZXRyeUF0dHJpYnV0ZV9HZW9tZXRyeUF0dHJpYnV0ZV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9HZW9tZXRyeUF0dHJpYnV0ZV9HZW9tZXRyeUF0dHJpYnV0ZV8wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9HZW9tZXRyeUF0dHJpYnV0ZV9HZW9tZXRyeUF0dHJpYnV0ZV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9HZW9tZXRyeUF0dHJpYnV0ZV9HZW9tZXRyeUF0dHJpYnV0ZV8wIl09TW9kdWxlWyJhc20iXVsicCJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0dlb21ldHJ5QXR0cmlidXRlX19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0dlb21ldHJ5QXR0cmlidXRlX19fZGVzdHJveV9fXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0dlb21ldHJ5QXR0cmlidXRlX19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0dlb21ldHJ5QXR0cmlidXRlX19fZGVzdHJveV9fXzAiXT1Nb2R1bGVbImFzbSJdWyJxIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfUG9pbnRBdHRyaWJ1dGVfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfUG9pbnRBdHRyaWJ1dGVfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfUG9pbnRBdHRyaWJ1dGVfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfUG9pbnRBdHRyaWJ1dGVfMCJdPU1vZHVsZVsiYXNtIl1bInIiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9zaXplXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX3NpemVfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfc2l6ZV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9zaXplXzAiXT1Nb2R1bGVbImFzbSJdWyJzIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfR2V0QXR0cmlidXRlVHJhbnNmb3JtRGF0YV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9HZXRBdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX0dldEF0dHJpYnV0ZVRyYW5zZm9ybURhdGFfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfR2V0QXR0cmlidXRlVHJhbnNmb3JtRGF0YV8wIl09TW9kdWxlWyJhc20iXVsidCJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX2F0dHJpYnV0ZV90eXBlXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX2F0dHJpYnV0ZV90eXBlXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX2F0dHJpYnV0ZV90eXBlXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX2F0dHJpYnV0ZV90eXBlXzAiXT1Nb2R1bGVbImFzbSJdWyJ1Il0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfZGF0YV90eXBlXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX2RhdGFfdHlwZV8wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9kYXRhX3R5cGVfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfZGF0YV90eXBlXzAiXT1Nb2R1bGVbImFzbSJdWyJ2Il0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfbnVtX2NvbXBvbmVudHNfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfbnVtX2NvbXBvbmVudHNfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfbnVtX2NvbXBvbmVudHNfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfbnVtX2NvbXBvbmVudHNfMCJdPU1vZHVsZVsiYXNtIl1bInciXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9ub3JtYWxpemVkXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX25vcm1hbGl6ZWRfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfbm9ybWFsaXplZF8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9ub3JtYWxpemVkXzAiXT1Nb2R1bGVbImFzbSJdWyJ4Il0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfYnl0ZV9zdHJpZGVfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfYnl0ZV9zdHJpZGVfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfYnl0ZV9zdHJpZGVfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfYnl0ZV9zdHJpZGVfMCJdPU1vZHVsZVsiYXNtIl1bInkiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9ieXRlX29mZnNldF8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9ieXRlX29mZnNldF8wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9ieXRlX29mZnNldF8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9ieXRlX29mZnNldF8wIl09TW9kdWxlWyJhc20iXVsieiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX3VuaXF1ZV9pZF8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV91bmlxdWVfaWRfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfdW5pcXVlX2lkXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX3VuaXF1ZV9pZF8wIl09TW9kdWxlWyJhc20iXVsiQSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX19fZGVzdHJveV9fXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX19fZGVzdHJveV9fXzAiXT1Nb2R1bGVbImFzbSJdWyJCIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm1fQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm1fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV8wIl09TW9kdWxlWyJhc20iXVsiQyJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9Jbml0RnJvbUF0dHJpYnV0ZV8xPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm1fSW5pdEZyb21BdHRyaWJ1dGVfMSJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtX0luaXRGcm9tQXR0cmlidXRlXzE9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9Jbml0RnJvbUF0dHJpYnV0ZV8xIl09TW9kdWxlWyJhc20iXVsiRCJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9xdWFudGl6YXRpb25fYml0c18wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm1fcXVhbnRpemF0aW9uX2JpdHNfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtX3F1YW50aXphdGlvbl9iaXRzXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9xdWFudGl6YXRpb25fYml0c18wIl09TW9kdWxlWyJhc20iXVsiRSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9taW5fdmFsdWVfMT1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtX21pbl92YWx1ZV8xIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm1fbWluX3ZhbHVlXzE9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9taW5fdmFsdWVfMSJdPU1vZHVsZVsiYXNtIl1bIkYiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm1fcmFuZ2VfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtX3JhbmdlXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9yYW5nZV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm1fcmFuZ2VfMCJdPU1vZHVsZVsiYXNtIl1bIkciXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm1fX19kZXN0cm95X19fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtX19fZGVzdHJveV9fXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9fX2Rlc3Ryb3lfX18wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm1fX19kZXN0cm95X19fMCJdPU1vZHVsZVsiYXNtIl1bIkgiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtX0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm1fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlT2N0YWhlZHJvblRyYW5zZm9ybV9BdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm1fQXR0cmlidXRlT2N0YWhlZHJvblRyYW5zZm9ybV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtX0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm1fMCJdPU1vZHVsZVsiYXNtIl1bIkkiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtX0luaXRGcm9tQXR0cmlidXRlXzE9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm1fSW5pdEZyb21BdHRyaWJ1dGVfMSJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlT2N0YWhlZHJvblRyYW5zZm9ybV9Jbml0RnJvbUF0dHJpYnV0ZV8xPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtX0luaXRGcm9tQXR0cmlidXRlXzEiXT1Nb2R1bGVbImFzbSJdWyJKIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlT2N0YWhlZHJvblRyYW5zZm9ybV9xdWFudGl6YXRpb25fYml0c18wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtX3F1YW50aXphdGlvbl9iaXRzXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm1fcXVhbnRpemF0aW9uX2JpdHNfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlT2N0YWhlZHJvblRyYW5zZm9ybV9xdWFudGl6YXRpb25fYml0c18wIl09TW9kdWxlWyJhc20iXVsiSyJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm1fX19kZXN0cm95X19fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlT2N0YWhlZHJvblRyYW5zZm9ybV9fX2Rlc3Ryb3lfX18wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtX19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm1fX19kZXN0cm95X19fMCJdPU1vZHVsZVsiYXNtIl1bIkwiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludENsb3VkX1BvaW50Q2xvdWRfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRDbG91ZF9Qb2ludENsb3VkXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX1BvaW50Q2xvdWRfUG9pbnRDbG91ZF8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludENsb3VkX1BvaW50Q2xvdWRfMCJdPU1vZHVsZVsiYXNtIl1bIk0iXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludENsb3VkX251bV9hdHRyaWJ1dGVzXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX1BvaW50Q2xvdWRfbnVtX2F0dHJpYnV0ZXNfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRDbG91ZF9udW1fYXR0cmlidXRlc18wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludENsb3VkX251bV9hdHRyaWJ1dGVzXzAiXT1Nb2R1bGVbImFzbSJdWyJOIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRDbG91ZF9udW1fcG9pbnRzXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX1BvaW50Q2xvdWRfbnVtX3BvaW50c18wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludENsb3VkX251bV9wb2ludHNfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRDbG91ZF9udW1fcG9pbnRzXzAiXT1Nb2R1bGVbImFzbSJdWyJPIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRDbG91ZF9fX2Rlc3Ryb3lfX18wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludENsb3VkX19fZGVzdHJveV9fXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX1BvaW50Q2xvdWRfX19kZXN0cm95X19fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRDbG91ZF9fX2Rlc3Ryb3lfX18wIl09TW9kdWxlWyJhc20iXVsiUCJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX01lc2hfTWVzaF8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9NZXNoX01lc2hfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfTWVzaF9NZXNoXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX01lc2hfTWVzaF8wIl09TW9kdWxlWyJhc20iXVsiUSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX01lc2hfbnVtX2ZhY2VzXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX01lc2hfbnVtX2ZhY2VzXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX01lc2hfbnVtX2ZhY2VzXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX01lc2hfbnVtX2ZhY2VzXzAiXT1Nb2R1bGVbImFzbSJdWyJSIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfTWVzaF9udW1fYXR0cmlidXRlc18wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9NZXNoX251bV9hdHRyaWJ1dGVzXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX01lc2hfbnVtX2F0dHJpYnV0ZXNfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfTWVzaF9udW1fYXR0cmlidXRlc18wIl09TW9kdWxlWyJhc20iXVsiUyJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX01lc2hfbnVtX3BvaW50c18wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9NZXNoX251bV9wb2ludHNfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfTWVzaF9udW1fcG9pbnRzXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX01lc2hfbnVtX3BvaW50c18wIl09TW9kdWxlWyJhc20iXVsiVCJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX01lc2hfX19kZXN0cm95X19fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfTWVzaF9fX2Rlc3Ryb3lfX18wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9NZXNoX19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX01lc2hfX19kZXN0cm95X19fMCJdPU1vZHVsZVsiYXNtIl1bIlUiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YV9NZXRhZGF0YV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YV9NZXRhZGF0YV8wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YV9NZXRhZGF0YV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YV9NZXRhZGF0YV8wIl09TW9kdWxlWyJhc20iXVsiViJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhX19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhX19fZGVzdHJveV9fXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhX19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhX19fZGVzdHJveV9fXzAiXT1Nb2R1bGVbImFzbSJdWyJXIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfU3RhdHVzX2NvZGVfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfU3RhdHVzX2NvZGVfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfU3RhdHVzX2NvZGVfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfU3RhdHVzX2NvZGVfMCJdPU1vZHVsZVsiYXNtIl1bIlgiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9TdGF0dXNfb2tfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfU3RhdHVzX29rXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX1N0YXR1c19va18wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9TdGF0dXNfb2tfMCJdPU1vZHVsZVsiYXNtIl1bIlkiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9TdGF0dXNfZXJyb3JfbXNnXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX1N0YXR1c19lcnJvcl9tc2dfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfU3RhdHVzX2Vycm9yX21zZ18wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9TdGF0dXNfZXJyb3JfbXNnXzAiXT1Nb2R1bGVbImFzbSJdWyJaIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfU3RhdHVzX19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX1N0YXR1c19fX2Rlc3Ryb3lfX18wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9TdGF0dXNfX19kZXN0cm95X19fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfU3RhdHVzX19fZGVzdHJveV9fXzAiXT1Nb2R1bGVbImFzbSJdWyJfIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRHJhY29GbG9hdDMyQXJyYXlfRHJhY29GbG9hdDMyQXJyYXlfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29GbG9hdDMyQXJyYXlfRHJhY29GbG9hdDMyQXJyYXlfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRHJhY29GbG9hdDMyQXJyYXlfRHJhY29GbG9hdDMyQXJyYXlfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29GbG9hdDMyQXJyYXlfRHJhY29GbG9hdDMyQXJyYXlfMCJdPU1vZHVsZVsiYXNtIl1bIiQiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0Zsb2F0MzJBcnJheV9HZXRWYWx1ZV8xPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0Zsb2F0MzJBcnJheV9HZXRWYWx1ZV8xIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0Zsb2F0MzJBcnJheV9HZXRWYWx1ZV8xPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0Zsb2F0MzJBcnJheV9HZXRWYWx1ZV8xIl09TW9kdWxlWyJhc20iXVsiYWEiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0Zsb2F0MzJBcnJheV9zaXplXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvRmxvYXQzMkFycmF5X3NpemVfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRHJhY29GbG9hdDMyQXJyYXlfc2l6ZV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0Zsb2F0MzJBcnJheV9zaXplXzAiXT1Nb2R1bGVbImFzbSJdWyJiYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RyYWNvRmxvYXQzMkFycmF5X19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvRmxvYXQzMkFycmF5X19fZGVzdHJveV9fXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RyYWNvRmxvYXQzMkFycmF5X19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvRmxvYXQzMkFycmF5X19fZGVzdHJveV9fXzAiXT1Nb2R1bGVbImFzbSJdWyJjYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50OEFycmF5X0RyYWNvSW50OEFycmF5XzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50OEFycmF5X0RyYWNvSW50OEFycmF5XzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50OEFycmF5X0RyYWNvSW50OEFycmF5XzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50OEFycmF5X0RyYWNvSW50OEFycmF5XzAiXT1Nb2R1bGVbImFzbSJdWyJkYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50OEFycmF5X0dldFZhbHVlXzE9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50OEFycmF5X0dldFZhbHVlXzEiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50OEFycmF5X0dldFZhbHVlXzE9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50OEFycmF5X0dldFZhbHVlXzEiXT1Nb2R1bGVbImFzbSJdWyJlYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50OEFycmF5X3NpemVfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQ4QXJyYXlfc2l6ZV8wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDhBcnJheV9zaXplXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50OEFycmF5X3NpemVfMCJdPU1vZHVsZVsiYXNtIl1bImZhIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQ4QXJyYXlfX19kZXN0cm95X19fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQ4QXJyYXlfX19kZXN0cm95X19fMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQ4QXJyYXlfX19kZXN0cm95X19fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQ4QXJyYXlfX19kZXN0cm95X19fMCJdPU1vZHVsZVsiYXNtIl1bImdhIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50OEFycmF5X0RyYWNvVUludDhBcnJheV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQ4QXJyYXlfRHJhY29VSW50OEFycmF5XzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDhBcnJheV9EcmFjb1VJbnQ4QXJyYXlfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50OEFycmF5X0RyYWNvVUludDhBcnJheV8wIl09TW9kdWxlWyJhc20iXVsiaGEiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQ4QXJyYXlfR2V0VmFsdWVfMT1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50OEFycmF5X0dldFZhbHVlXzEiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDhBcnJheV9HZXRWYWx1ZV8xPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQ4QXJyYXlfR2V0VmFsdWVfMSJdPU1vZHVsZVsiYXNtIl1bImlhIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50OEFycmF5X3NpemVfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50OEFycmF5X3NpemVfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50OEFycmF5X3NpemVfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50OEFycmF5X3NpemVfMCJdPU1vZHVsZVsiYXNtIl1bImphIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50OEFycmF5X19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDhBcnJheV9fX2Rlc3Ryb3lfX18wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQ4QXJyYXlfX19kZXN0cm95X19fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50OEFycmF5X19fZGVzdHJveV9fXzAiXT1Nb2R1bGVbImFzbSJdWyJrYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MTZBcnJheV9EcmFjb0ludDE2QXJyYXlfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQxNkFycmF5X0RyYWNvSW50MTZBcnJheV8wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDE2QXJyYXlfRHJhY29JbnQxNkFycmF5XzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MTZBcnJheV9EcmFjb0ludDE2QXJyYXlfMCJdPU1vZHVsZVsiYXNtIl1bImxhIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQxNkFycmF5X0dldFZhbHVlXzE9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MTZBcnJheV9HZXRWYWx1ZV8xIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDE2QXJyYXlfR2V0VmFsdWVfMT1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQxNkFycmF5X0dldFZhbHVlXzEiXT1Nb2R1bGVbImFzbSJdWyJtYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MTZBcnJheV9zaXplXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MTZBcnJheV9zaXplXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MTZBcnJheV9zaXplXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MTZBcnJheV9zaXplXzAiXT1Nb2R1bGVbImFzbSJdWyJuYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MTZBcnJheV9fX2Rlc3Ryb3lfX18wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDE2QXJyYXlfX19kZXN0cm95X19fMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQxNkFycmF5X19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MTZBcnJheV9fX2Rlc3Ryb3lfX18wIl09TW9kdWxlWyJhc20iXVsib2EiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQxNkFycmF5X0RyYWNvVUludDE2QXJyYXlfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MTZBcnJheV9EcmFjb1VJbnQxNkFycmF5XzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDE2QXJyYXlfRHJhY29VSW50MTZBcnJheV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQxNkFycmF5X0RyYWNvVUludDE2QXJyYXlfMCJdPU1vZHVsZVsiYXNtIl1bInBhIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MTZBcnJheV9HZXRWYWx1ZV8xPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQxNkFycmF5X0dldFZhbHVlXzEiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDE2QXJyYXlfR2V0VmFsdWVfMT1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MTZBcnJheV9HZXRWYWx1ZV8xIl09TW9kdWxlWyJhc20iXVsicWEiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQxNkFycmF5X3NpemVfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MTZBcnJheV9zaXplXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDE2QXJyYXlfc2l6ZV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQxNkFycmF5X3NpemVfMCJdPU1vZHVsZVsiYXNtIl1bInJhIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MTZBcnJheV9fX2Rlc3Ryb3lfX18wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQxNkFycmF5X19fZGVzdHJveV9fXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDE2QXJyYXlfX19kZXN0cm95X19fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MTZBcnJheV9fX2Rlc3Ryb3lfX18wIl09TW9kdWxlWyJhc20iXVsic2EiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDMyQXJyYXlfRHJhY29JbnQzMkFycmF5XzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MzJBcnJheV9EcmFjb0ludDMyQXJyYXlfMCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQzMkFycmF5X0RyYWNvSW50MzJBcnJheV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDMyQXJyYXlfRHJhY29JbnQzMkFycmF5XzAiXT1Nb2R1bGVbImFzbSJdWyJ0YSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MzJBcnJheV9HZXRWYWx1ZV8xPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDMyQXJyYXlfR2V0VmFsdWVfMSJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQzMkFycmF5X0dldFZhbHVlXzE9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MzJBcnJheV9HZXRWYWx1ZV8xIl09TW9kdWxlWyJhc20iXVsidWEiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDMyQXJyYXlfc2l6ZV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDMyQXJyYXlfc2l6ZV8wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDMyQXJyYXlfc2l6ZV8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDMyQXJyYXlfc2l6ZV8wIl09TW9kdWxlWyJhc20iXVsidmEiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDMyQXJyYXlfX19kZXN0cm95X19fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQzMkFycmF5X19fZGVzdHJveV9fXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MzJBcnJheV9fX2Rlc3Ryb3lfX18wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDMyQXJyYXlfX19kZXN0cm95X19fMCJdPU1vZHVsZVsiYXNtIl1bIndhIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MzJBcnJheV9EcmFjb1VJbnQzMkFycmF5XzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDMyQXJyYXlfRHJhY29VSW50MzJBcnJheV8wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQzMkFycmF5X0RyYWNvVUludDMyQXJyYXlfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MzJBcnJheV9EcmFjb1VJbnQzMkFycmF5XzAiXT1Nb2R1bGVbImFzbSJdWyJ4YSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDMyQXJyYXlfR2V0VmFsdWVfMT1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MzJBcnJheV9HZXRWYWx1ZV8xIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQzMkFycmF5X0dldFZhbHVlXzE9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDMyQXJyYXlfR2V0VmFsdWVfMSJdPU1vZHVsZVsiYXNtIl1bInlhIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MzJBcnJheV9zaXplXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDMyQXJyYXlfc2l6ZV8wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQzMkFycmF5X3NpemVfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MzJBcnJheV9zaXplXzAiXT1Nb2R1bGVbImFzbSJdWyJ6YSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDMyQXJyYXlfX19kZXN0cm95X19fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MzJBcnJheV9fX2Rlc3Ryb3lfX18wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQzMkFycmF5X19fZGVzdHJveV9fXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDMyQXJyYXlfX19kZXN0cm95X19fMCJdPU1vZHVsZVsiYXNtIl1bIkFhIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX01ldGFkYXRhUXVlcmllcl8wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfTWV0YWRhdGFRdWVyaWVyXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9NZXRhZGF0YVF1ZXJpZXJfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX01ldGFkYXRhUXVlcmllcl8wIl09TW9kdWxlWyJhc20iXVsiQmEiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfSGFzRW50cnlfMj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX0hhc0VudHJ5XzIiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9IYXNFbnRyeV8yPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfSGFzRW50cnlfMiJdPU1vZHVsZVsiYXNtIl1bIkNhIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX0dldEludEVudHJ5XzI9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9HZXRJbnRFbnRyeV8yIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfR2V0SW50RW50cnlfMj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX0dldEludEVudHJ5XzIiXT1Nb2R1bGVbImFzbSJdWyJEYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9HZXRJbnRFbnRyeUFycmF5XzM9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9HZXRJbnRFbnRyeUFycmF5XzMiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9HZXRJbnRFbnRyeUFycmF5XzM9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9HZXRJbnRFbnRyeUFycmF5XzMiXT1Nb2R1bGVbImFzbSJdWyJFYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9HZXREb3VibGVFbnRyeV8yPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfR2V0RG91YmxlRW50cnlfMiJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX0dldERvdWJsZUVudHJ5XzI9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9HZXREb3VibGVFbnRyeV8yIl09TW9kdWxlWyJhc20iXVsiRmEiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfR2V0U3RyaW5nRW50cnlfMj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX0dldFN0cmluZ0VudHJ5XzIiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9HZXRTdHJpbmdFbnRyeV8yPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfR2V0U3RyaW5nRW50cnlfMiJdPU1vZHVsZVsiYXNtIl1bIkdhIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX051bUVudHJpZXNfMT1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX051bUVudHJpZXNfMSJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX051bUVudHJpZXNfMT1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX051bUVudHJpZXNfMSJdPU1vZHVsZVsiYXNtIl1bIkhhIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX0dldEVudHJ5TmFtZV8yPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfR2V0RW50cnlOYW1lXzIiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9HZXRFbnRyeU5hbWVfMj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX0dldEVudHJ5TmFtZV8yIl09TW9kdWxlWyJhc20iXVsiSWEiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfX19kZXN0cm95X19fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX19fZGVzdHJveV9fXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9fX2Rlc3Ryb3lfX18wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfX19kZXN0cm95X19fMCJdPU1vZHVsZVsiYXNtIl1bIkphIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVyXzA9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfRGVjb2Rlcl8wIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0RlY29kZXJfMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVyXzAiXT1Nb2R1bGVbImFzbSJdWyJLYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfRGVjb2RlQXJyYXlUb1BvaW50Q2xvdWRfMz1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVBcnJheVRvUG9pbnRDbG91ZF8zIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0RlY29kZUFycmF5VG9Qb2ludENsb3VkXzM9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfRGVjb2RlQXJyYXlUb1BvaW50Q2xvdWRfMyJdPU1vZHVsZVsiYXNtIl1bIkxhIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVBcnJheVRvTWVzaF8zPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0RlY29kZUFycmF5VG9NZXNoXzMiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfRGVjb2RlQXJyYXlUb01lc2hfMz1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVBcnJheVRvTWVzaF8zIl09TW9kdWxlWyJhc20iXVsiTWEiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUlkXzI9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSWRfMiJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVJZF8yPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUlkXzIiXT1Nb2R1bGVbImFzbSJdWyJOYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSWRCeU5hbWVfMj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVJZEJ5TmFtZV8yIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUlkQnlOYW1lXzI9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSWRCeU5hbWVfMiJdPU1vZHVsZVsiYXNtIl1bIk9hIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVJZEJ5TWV0YWRhdGFFbnRyeV8zPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUlkQnlNZXRhZGF0YUVudHJ5XzMiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSWRCeU1ldGFkYXRhRW50cnlfMz1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVJZEJ5TWV0YWRhdGFFbnRyeV8zIl09TW9kdWxlWyJhc20iXVsiUGEiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZV8yPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZV8yIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZV8yPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZV8yIl09TW9kdWxlWyJhc20iXVsiUWEiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUJ5VW5pcXVlSWRfMj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVCeVVuaXF1ZUlkXzIiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlQnlVbmlxdWVJZF8yPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUJ5VW5pcXVlSWRfMiJdPU1vZHVsZVsiYXNtIl1bIlJhIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRNZXRhZGF0YV8xPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldE1ldGFkYXRhXzEiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0TWV0YWRhdGFfMT1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRNZXRhZGF0YV8xIl09TW9kdWxlWyJhc20iXVsiU2EiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZU1ldGFkYXRhXzI9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlTWV0YWRhdGFfMiJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVNZXRhZGF0YV8yPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZU1ldGFkYXRhXzIiXT1Nb2R1bGVbImFzbSJdWyJUYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0RmFjZUZyb21NZXNoXzM9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0RmFjZUZyb21NZXNoXzMiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0RmFjZUZyb21NZXNoXzM9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0RmFjZUZyb21NZXNoXzMiXT1Nb2R1bGVbImFzbSJdWyJVYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0VHJpYW5nbGVTdHJpcHNGcm9tTWVzaF8yPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldFRyaWFuZ2xlU3RyaXBzRnJvbU1lc2hfMiJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRUcmlhbmdsZVN0cmlwc0Zyb21NZXNoXzI9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0VHJpYW5nbGVTdHJpcHNGcm9tTWVzaF8yIl09TW9kdWxlWyJhc20iXVsiVmEiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldFRyaWFuZ2xlc1VJbnQxNkFycmF5XzM9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0VHJpYW5nbGVzVUludDE2QXJyYXlfMyJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRUcmlhbmdsZXNVSW50MTZBcnJheV8zPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldFRyaWFuZ2xlc1VJbnQxNkFycmF5XzMiXT1Nb2R1bGVbImFzbSJdWyJXYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0VHJpYW5nbGVzVUludDMyQXJyYXlfMz1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRUcmlhbmdsZXNVSW50MzJBcnJheV8zIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldFRyaWFuZ2xlc1VJbnQzMkFycmF5XzM9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0VHJpYW5nbGVzVUludDMyQXJyYXlfMyJdPU1vZHVsZVsiYXNtIl1bIlhhIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVGbG9hdF8zPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUZsb2F0XzMiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlRmxvYXRfMz1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVGbG9hdF8zIl09TW9kdWxlWyJhc20iXVsiWWEiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUZsb2F0Rm9yQWxsUG9pbnRzXzM9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlRmxvYXRGb3JBbGxQb2ludHNfMyJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVGbG9hdEZvckFsbFBvaW50c18zPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUZsb2F0Rm9yQWxsUG9pbnRzXzMiXT1Nb2R1bGVbImFzbSJdWyJaYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSW50Rm9yQWxsUG9pbnRzXzM9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSW50Rm9yQWxsUG9pbnRzXzMiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSW50Rm9yQWxsUG9pbnRzXzM9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSW50Rm9yQWxsUG9pbnRzXzMiXT1Nb2R1bGVbImFzbSJdWyJfYSJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSW50OEZvckFsbFBvaW50c18zPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUludDhGb3JBbGxQb2ludHNfMyJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVJbnQ4Rm9yQWxsUG9pbnRzXzM9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSW50OEZvckFsbFBvaW50c18zIl09TW9kdWxlWyJhc20iXVsiJGEiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZVVJbnQ4Rm9yQWxsUG9pbnRzXzM9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlVUludDhGb3JBbGxQb2ludHNfMyJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVVSW50OEZvckFsbFBvaW50c18zPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZVVJbnQ4Rm9yQWxsUG9pbnRzXzMiXT1Nb2R1bGVbImFzbSJdWyJhYiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSW50MTZGb3JBbGxQb2ludHNfMz1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVJbnQxNkZvckFsbFBvaW50c18zIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUludDE2Rm9yQWxsUG9pbnRzXzM9TW9kdWxlWyJfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSW50MTZGb3JBbGxQb2ludHNfMyJdPU1vZHVsZVsiYXNtIl1bImJiIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVVSW50MTZGb3JBbGxQb2ludHNfMz1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVVSW50MTZGb3JBbGxQb2ludHNfMyJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVVSW50MTZGb3JBbGxQb2ludHNfMz1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVVSW50MTZGb3JBbGxQb2ludHNfMyJdPU1vZHVsZVsiYXNtIl1bImNiIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVJbnQzMkZvckFsbFBvaW50c18zPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUludDMyRm9yQWxsUG9pbnRzXzMiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSW50MzJGb3JBbGxQb2ludHNfMz1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVJbnQzMkZvckFsbFBvaW50c18zIl09TW9kdWxlWyJhc20iXVsiZGIiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZVVJbnQzMkZvckFsbFBvaW50c18zPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZVVJbnQzMkZvckFsbFBvaW50c18zIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZVVJbnQzMkZvckFsbFBvaW50c18zPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZVVJbnQzMkZvckFsbFBvaW50c18zIl09TW9kdWxlWyJhc20iXVsiZWIiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZURhdGFBcnJheUZvckFsbFBvaW50c181PU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZURhdGFBcnJheUZvckFsbFBvaW50c181Il09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZURhdGFBcnJheUZvckFsbFBvaW50c181PU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZURhdGFBcnJheUZvckFsbFBvaW50c181Il09TW9kdWxlWyJhc20iXVsiZmIiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX1NraXBBdHRyaWJ1dGVUcmFuc2Zvcm1fMT1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9Ta2lwQXR0cmlidXRlVHJhbnNmb3JtXzEiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfU2tpcEF0dHJpYnV0ZVRyYW5zZm9ybV8xPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX1NraXBBdHRyaWJ1dGVUcmFuc2Zvcm1fMSJdPU1vZHVsZVsiYXNtIl1bImdiIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRFbmNvZGVkR2VvbWV0cnlUeXBlX0RlcHJlY2F0ZWRfMT1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRFbmNvZGVkR2VvbWV0cnlUeXBlX0RlcHJlY2F0ZWRfMSJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRFbmNvZGVkR2VvbWV0cnlUeXBlX0RlcHJlY2F0ZWRfMT1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRFbmNvZGVkR2VvbWV0cnlUeXBlX0RlcHJlY2F0ZWRfMSJdPU1vZHVsZVsiYXNtIl1bImhiIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVCdWZmZXJUb1BvaW50Q2xvdWRfMj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVCdWZmZXJUb1BvaW50Q2xvdWRfMiJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVCdWZmZXJUb1BvaW50Q2xvdWRfMj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVCdWZmZXJUb1BvaW50Q2xvdWRfMiJdPU1vZHVsZVsiYXNtIl1bImliIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVCdWZmZXJUb01lc2hfMj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVCdWZmZXJUb01lc2hfMiJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVCdWZmZXJUb01lc2hfMj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVCdWZmZXJUb01lc2hfMiJdPU1vZHVsZVsiYXNtIl1bImpiIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9fX2Rlc3Ryb3lfX18wPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX19fZGVzdHJveV9fXzAiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfX19kZXN0cm95X19fMD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9fX2Rlc3Ryb3lfX18wIl09TW9kdWxlWyJhc20iXVsia2IiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19BdHRyaWJ1dGVUcmFuc2Zvcm1UeXBlX0FUVFJJQlVURV9JTlZBTElEX1RSQU5TRk9STT1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fQXR0cmlidXRlVHJhbnNmb3JtVHlwZV9BVFRSSUJVVEVfSU5WQUxJRF9UUkFOU0ZPUk0iXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0F0dHJpYnV0ZVRyYW5zZm9ybVR5cGVfQVRUUklCVVRFX0lOVkFMSURfVFJBTlNGT1JNPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19BdHRyaWJ1dGVUcmFuc2Zvcm1UeXBlX0FUVFJJQlVURV9JTlZBTElEX1RSQU5TRk9STSJdPU1vZHVsZVsiYXNtIl1bImxiIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2VudW1fZHJhY29fQXR0cmlidXRlVHJhbnNmb3JtVHlwZV9BVFRSSUJVVEVfTk9fVFJBTlNGT1JNPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19BdHRyaWJ1dGVUcmFuc2Zvcm1UeXBlX0FUVFJJQlVURV9OT19UUkFOU0ZPUk0iXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0F0dHJpYnV0ZVRyYW5zZm9ybVR5cGVfQVRUUklCVVRFX05PX1RSQU5TRk9STT1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fQXR0cmlidXRlVHJhbnNmb3JtVHlwZV9BVFRSSUJVVEVfTk9fVFJBTlNGT1JNIl09TW9kdWxlWyJhc20iXVsibWIiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19BdHRyaWJ1dGVUcmFuc2Zvcm1UeXBlX0FUVFJJQlVURV9RVUFOVElaQVRJT05fVFJBTlNGT1JNPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19BdHRyaWJ1dGVUcmFuc2Zvcm1UeXBlX0FUVFJJQlVURV9RVUFOVElaQVRJT05fVFJBTlNGT1JNIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19BdHRyaWJ1dGVUcmFuc2Zvcm1UeXBlX0FUVFJJQlVURV9RVUFOVElaQVRJT05fVFJBTlNGT1JNPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19BdHRyaWJ1dGVUcmFuc2Zvcm1UeXBlX0FUVFJJQlVURV9RVUFOVElaQVRJT05fVFJBTlNGT1JNIl09TW9kdWxlWyJhc20iXVsibmIiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19BdHRyaWJ1dGVUcmFuc2Zvcm1UeXBlX0FUVFJJQlVURV9PQ1RBSEVEUk9OX1RSQU5TRk9STT1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fQXR0cmlidXRlVHJhbnNmb3JtVHlwZV9BVFRSSUJVVEVfT0NUQUhFRFJPTl9UUkFOU0ZPUk0iXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0F0dHJpYnV0ZVRyYW5zZm9ybVR5cGVfQVRUUklCVVRFX09DVEFIRURST05fVFJBTlNGT1JNPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19BdHRyaWJ1dGVUcmFuc2Zvcm1UeXBlX0FUVFJJQlVURV9PQ1RBSEVEUk9OX1RSQU5TRk9STSJdPU1vZHVsZVsiYXNtIl1bIm9iIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9JTlZBTElEPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19HZW9tZXRyeUF0dHJpYnV0ZV9UeXBlX0lOVkFMSUQiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0dlb21ldHJ5QXR0cmlidXRlX1R5cGVfSU5WQUxJRD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9JTlZBTElEIl09TW9kdWxlWyJhc20iXVsicGIiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19HZW9tZXRyeUF0dHJpYnV0ZV9UeXBlX1BPU0lUSU9OPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19HZW9tZXRyeUF0dHJpYnV0ZV9UeXBlX1BPU0lUSU9OIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19HZW9tZXRyeUF0dHJpYnV0ZV9UeXBlX1BPU0lUSU9OPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19HZW9tZXRyeUF0dHJpYnV0ZV9UeXBlX1BPU0lUSU9OIl09TW9kdWxlWyJhc20iXVsicWIiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19HZW9tZXRyeUF0dHJpYnV0ZV9UeXBlX05PUk1BTD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9OT1JNQUwiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0dlb21ldHJ5QXR0cmlidXRlX1R5cGVfTk9STUFMPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19HZW9tZXRyeUF0dHJpYnV0ZV9UeXBlX05PUk1BTCJdPU1vZHVsZVsiYXNtIl1bInJiIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9DT0xPUj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9DT0xPUiJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9DT0xPUj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9DT0xPUiJdPU1vZHVsZVsiYXNtIl1bInNiIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9URVhfQ09PUkQ9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0dlb21ldHJ5QXR0cmlidXRlX1R5cGVfVEVYX0NPT1JEIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19HZW9tZXRyeUF0dHJpYnV0ZV9UeXBlX1RFWF9DT09SRD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9URVhfQ09PUkQiXT1Nb2R1bGVbImFzbSJdWyJ0YiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0dlb21ldHJ5QXR0cmlidXRlX1R5cGVfR0VORVJJQz1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9HRU5FUklDIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19HZW9tZXRyeUF0dHJpYnV0ZV9UeXBlX0dFTkVSSUM9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0dlb21ldHJ5QXR0cmlidXRlX1R5cGVfR0VORVJJQyJdPU1vZHVsZVsiYXNtIl1bInViIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2VudW1fZHJhY29fRW5jb2RlZEdlb21ldHJ5VHlwZV9JTlZBTElEX0dFT01FVFJZX1RZUEU9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0VuY29kZWRHZW9tZXRyeVR5cGVfSU5WQUxJRF9HRU9NRVRSWV9UWVBFIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19FbmNvZGVkR2VvbWV0cnlUeXBlX0lOVkFMSURfR0VPTUVUUllfVFlQRT1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRW5jb2RlZEdlb21ldHJ5VHlwZV9JTlZBTElEX0dFT01FVFJZX1RZUEUiXT1Nb2R1bGVbImFzbSJdWyJ2YiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0VuY29kZWRHZW9tZXRyeVR5cGVfUE9JTlRfQ0xPVUQ9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0VuY29kZWRHZW9tZXRyeVR5cGVfUE9JTlRfQ0xPVUQiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0VuY29kZWRHZW9tZXRyeVR5cGVfUE9JTlRfQ0xPVUQ9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0VuY29kZWRHZW9tZXRyeVR5cGVfUE9JTlRfQ0xPVUQiXT1Nb2R1bGVbImFzbSJdWyJ3YiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0VuY29kZWRHZW9tZXRyeVR5cGVfVFJJQU5HVUxBUl9NRVNIPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19FbmNvZGVkR2VvbWV0cnlUeXBlX1RSSUFOR1VMQVJfTUVTSCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2VudW1fZHJhY29fRW5jb2RlZEdlb21ldHJ5VHlwZV9UUklBTkdVTEFSX01FU0g9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0VuY29kZWRHZW9tZXRyeVR5cGVfVFJJQU5HVUxBUl9NRVNIIl09TW9kdWxlWyJhc20iXVsieGIiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9JTlZBTElEPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9JTlZBTElEIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9JTlZBTElEPU1vZHVsZVsiX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9JTlZBTElEIl09TW9kdWxlWyJhc20iXVsieWIiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9JTlQ4PU1vZHVsZVsiX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9JTlQ4Il09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9JTlQ4PU1vZHVsZVsiX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9JTlQ4Il09TW9kdWxlWyJhc20iXVsiemIiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9VSU5UOD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfVUlOVDgiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX1VJTlQ4PU1vZHVsZVsiX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9VSU5UOCJdPU1vZHVsZVsiYXNtIl1bIkFiIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfSU5UMTY9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX0lOVDE2Il09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9JTlQxNj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfSU5UMTYiXT1Nb2R1bGVbImFzbSJdWyJCYiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX1VJTlQxNj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfVUlOVDE2Il09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9VSU5UMTY9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX1VJTlQxNiJdPU1vZHVsZVsiYXNtIl1bIkNiIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfSU5UMzI9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX0lOVDMyIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9JTlQzMj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfSU5UMzIiXT1Nb2R1bGVbImFzbSJdWyJEYiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX1VJTlQzMj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfVUlOVDMyIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9VSU5UMzI9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX1VJTlQzMiJdPU1vZHVsZVsiYXNtIl1bIkViIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfSU5UNjQ9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX0lOVDY0Il09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9JTlQ2ND1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfSU5UNjQiXT1Nb2R1bGVbImFzbSJdWyJGYiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX1VJTlQ2ND1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfVUlOVDY0Il09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9VSU5UNjQ9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX1VJTlQ2NCJdPU1vZHVsZVsiYXNtIl1bIkdiIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfRkxPQVQzMj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfRkxPQVQzMiJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfRkxPQVQzMj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfRkxPQVQzMiJdPU1vZHVsZVsiYXNtIl1bIkhiIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfRkxPQVQ2ND1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfRkxPQVQ2NCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfRkxPQVQ2ND1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfRkxPQVQ2NCJdPU1vZHVsZVsiYXNtIl1bIkliIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfQk9PTD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfQk9PTCJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfQk9PTD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfQk9PTCJdPU1vZHVsZVsiYXNtIl1bIkpiIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfVFlQRVNfQ09VTlQ9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX1RZUEVTX0NPVU5UIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9UWVBFU19DT1VOVD1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfVFlQRVNfQ09VTlQiXT1Nb2R1bGVbImFzbSJdWyJLYiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfT0s9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfT0siXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfT0s9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfT0siXT1Nb2R1bGVbImFzbSJdWyJMYiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfRFJBQ09fRVJST1I9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfRFJBQ09fRVJST1IiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfRFJBQ09fRVJST1I9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfRFJBQ09fRVJST1IiXT1Nb2R1bGVbImFzbSJdWyJNYiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfSU9fRVJST1I9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfSU9fRVJST1IiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfSU9fRVJST1I9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfSU9fRVJST1IiXT1Nb2R1bGVbImFzbSJdWyJOYiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfSU5WQUxJRF9QQVJBTUVURVI9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfSU5WQUxJRF9QQVJBTUVURVIiXT1mdW5jdGlvbigpe3JldHVybihfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfSU5WQUxJRF9QQVJBTUVURVI9TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfSU5WQUxJRF9QQVJBTUVURVIiXT1Nb2R1bGVbImFzbSJdWyJPYiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfVU5TVVBQT1JURURfVkVSU0lPTj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fU3RhdHVzQ29kZV9VTlNVUFBPUlRFRF9WRVJTSU9OIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19TdGF0dXNDb2RlX1VOU1VQUE9SVEVEX1ZFUlNJT049TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfVU5TVVBQT1JURURfVkVSU0lPTiJdPU1vZHVsZVsiYXNtIl1bIlBiIl0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIF9lbXNjcmlwdGVuX2VudW1fZHJhY29fU3RhdHVzQ29kZV9VTktOT1dOX1ZFUlNJT049TW9kdWxlWyJfZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfVU5LTk9XTl9WRVJTSU9OIl09ZnVuY3Rpb24oKXtyZXR1cm4oX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19TdGF0dXNDb2RlX1VOS05PV05fVkVSU0lPTj1Nb2R1bGVbIl9lbXNjcmlwdGVuX2VudW1fZHJhY29fU3RhdHVzQ29kZV9VTktOT1dOX1ZFUlNJT04iXT1Nb2R1bGVbImFzbSJdWyJRYiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBfbWFsbG9jPU1vZHVsZVsiX21hbGxvYyJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9tYWxsb2M9TW9kdWxlWyJfbWFsbG9jIl09TW9kdWxlWyJhc20iXVsiUmIiXSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgX2ZyZWU9TW9kdWxlWyJfZnJlZSJdPWZ1bmN0aW9uKCl7cmV0dXJuKF9mcmVlPU1vZHVsZVsiX2ZyZWUiXT1Nb2R1bGVbImFzbSJdWyJTYiJdKS5hcHBseShudWxsLGFyZ3VtZW50cyl9O3ZhciBjYWxsZWRSdW47ZnVuY3Rpb24gRXhpdFN0YXR1cyhzdGF0dXMpe3RoaXMubmFtZT0iRXhpdFN0YXR1cyI7dGhpcy5tZXNzYWdlPSJQcm9ncmFtIHRlcm1pbmF0ZWQgd2l0aCBleGl0KCIrc3RhdHVzKyIpIjt0aGlzLnN0YXR1cz1zdGF0dXN9ZGVwZW5kZW5jaWVzRnVsZmlsbGVkPWZ1bmN0aW9uIHJ1bkNhbGxlcigpe2lmKCFjYWxsZWRSdW4pcnVuKCk7aWYoIWNhbGxlZFJ1bilkZXBlbmRlbmNpZXNGdWxmaWxsZWQ9cnVuQ2FsbGVyfTtmdW5jdGlvbiBydW4oYXJncyl7YXJncz1hcmdzfHxhcmd1bWVudHNfO2lmKHJ1bkRlcGVuZGVuY2llcz4wKXtyZXR1cm59cHJlUnVuKCk7aWYocnVuRGVwZW5kZW5jaWVzPjApe3JldHVybn1mdW5jdGlvbiBkb1J1bigpe2lmKGNhbGxlZFJ1bilyZXR1cm47Y2FsbGVkUnVuPXRydWU7TW9kdWxlWyJjYWxsZWRSdW4iXT10cnVlO2lmKEFCT1JUKXJldHVybjtpbml0UnVudGltZSgpO3JlYWR5UHJvbWlzZVJlc29sdmUoTW9kdWxlKTtpZihNb2R1bGVbIm9uUnVudGltZUluaXRpYWxpemVkIl0pTW9kdWxlWyJvblJ1bnRpbWVJbml0aWFsaXplZCJdKCk7cG9zdFJ1bigpfWlmKE1vZHVsZVsic2V0U3RhdHVzIl0pe01vZHVsZVsic2V0U3RhdHVzIl0oIlJ1bm5pbmcuLi4iKTtzZXRUaW1lb3V0KGZ1bmN0aW9uKCl7c2V0VGltZW91dChmdW5jdGlvbigpe01vZHVsZVsic2V0U3RhdHVzIl0oIiIpfSwxKTtkb1J1bigpfSwxKX1lbHNle2RvUnVuKCl9fU1vZHVsZVsicnVuIl09cnVuO2lmKE1vZHVsZVsicHJlSW5pdCJdKXtpZih0eXBlb2YgTW9kdWxlWyJwcmVJbml0Il09PSJmdW5jdGlvbiIpTW9kdWxlWyJwcmVJbml0Il09W01vZHVsZVsicHJlSW5pdCJdXTt3aGlsZShNb2R1bGVbInByZUluaXQiXS5sZW5ndGg+MCl7TW9kdWxlWyJwcmVJbml0Il0ucG9wKCkoKX19cnVuKCk7ZnVuY3Rpb24gV3JhcHBlck9iamVjdCgpe31XcmFwcGVyT2JqZWN0LnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKFdyYXBwZXJPYmplY3QucHJvdG90eXBlKTtXcmFwcGVyT2JqZWN0LnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1XcmFwcGVyT2JqZWN0O1dyYXBwZXJPYmplY3QucHJvdG90eXBlLl9fY2xhc3NfXz1XcmFwcGVyT2JqZWN0O1dyYXBwZXJPYmplY3QuX19jYWNoZV9fPXt9O01vZHVsZVsiV3JhcHBlck9iamVjdCJdPVdyYXBwZXJPYmplY3Q7ZnVuY3Rpb24gZ2V0Q2FjaGUoX19jbGFzc19fKXtyZXR1cm4oX19jbGFzc19ffHxXcmFwcGVyT2JqZWN0KS5fX2NhY2hlX199TW9kdWxlWyJnZXRDYWNoZSJdPWdldENhY2hlO2Z1bmN0aW9uIHdyYXBQb2ludGVyKHB0cixfX2NsYXNzX18pe3ZhciBjYWNoZT1nZXRDYWNoZShfX2NsYXNzX18pO3ZhciByZXQ9Y2FjaGVbcHRyXTtpZihyZXQpcmV0dXJuIHJldDtyZXQ9T2JqZWN0LmNyZWF0ZSgoX19jbGFzc19ffHxXcmFwcGVyT2JqZWN0KS5wcm90b3R5cGUpO3JldC5wdHI9cHRyO3JldHVybiBjYWNoZVtwdHJdPXJldH1Nb2R1bGVbIndyYXBQb2ludGVyIl09d3JhcFBvaW50ZXI7ZnVuY3Rpb24gY2FzdE9iamVjdChvYmosX19jbGFzc19fKXtyZXR1cm4gd3JhcFBvaW50ZXIob2JqLnB0cixfX2NsYXNzX18pfU1vZHVsZVsiY2FzdE9iamVjdCJdPWNhc3RPYmplY3Q7TW9kdWxlWyJOVUxMIl09d3JhcFBvaW50ZXIoMCk7ZnVuY3Rpb24gZGVzdHJveShvYmope2lmKCFvYmpbIl9fZGVzdHJveV9fIl0pdGhyb3ciRXJyb3I6IENhbm5vdCBkZXN0cm95IG9iamVjdC4gKERpZCB5b3UgY3JlYXRlIGl0IHlvdXJzZWxmPykiO29ialsiX19kZXN0cm95X18iXSgpO2RlbGV0ZSBnZXRDYWNoZShvYmouX19jbGFzc19fKVtvYmoucHRyXX1Nb2R1bGVbImRlc3Ryb3kiXT1kZXN0cm95O2Z1bmN0aW9uIGNvbXBhcmUob2JqMSxvYmoyKXtyZXR1cm4gb2JqMS5wdHI9PT1vYmoyLnB0cn1Nb2R1bGVbImNvbXBhcmUiXT1jb21wYXJlO2Z1bmN0aW9uIGdldFBvaW50ZXIob2JqKXtyZXR1cm4gb2JqLnB0cn1Nb2R1bGVbImdldFBvaW50ZXIiXT1nZXRQb2ludGVyO2Z1bmN0aW9uIGdldENsYXNzKG9iail7cmV0dXJuIG9iai5fX2NsYXNzX199TW9kdWxlWyJnZXRDbGFzcyJdPWdldENsYXNzO3ZhciBlbnN1cmVDYWNoZT17YnVmZmVyOjAsc2l6ZTowLHBvczowLHRlbXBzOltdLG5lZWRlZDowLHByZXBhcmU6ZnVuY3Rpb24oKXtpZihlbnN1cmVDYWNoZS5uZWVkZWQpe2Zvcih2YXIgaT0wO2k8ZW5zdXJlQ2FjaGUudGVtcHMubGVuZ3RoO2krKyl7TW9kdWxlWyJfZnJlZSJdKGVuc3VyZUNhY2hlLnRlbXBzW2ldKX1lbnN1cmVDYWNoZS50ZW1wcy5sZW5ndGg9MDtNb2R1bGVbIl9mcmVlIl0oZW5zdXJlQ2FjaGUuYnVmZmVyKTtlbnN1cmVDYWNoZS5idWZmZXI9MDtlbnN1cmVDYWNoZS5zaXplKz1lbnN1cmVDYWNoZS5uZWVkZWQ7ZW5zdXJlQ2FjaGUubmVlZGVkPTB9aWYoIWVuc3VyZUNhY2hlLmJ1ZmZlcil7ZW5zdXJlQ2FjaGUuc2l6ZSs9MTI4O2Vuc3VyZUNhY2hlLmJ1ZmZlcj1Nb2R1bGVbIl9tYWxsb2MiXShlbnN1cmVDYWNoZS5zaXplKTthc3NlcnQoZW5zdXJlQ2FjaGUuYnVmZmVyKX1lbnN1cmVDYWNoZS5wb3M9MH0sYWxsb2M6ZnVuY3Rpb24oYXJyYXksdmlldyl7YXNzZXJ0KGVuc3VyZUNhY2hlLmJ1ZmZlcik7dmFyIGJ5dGVzPXZpZXcuQllURVNfUEVSX0VMRU1FTlQ7dmFyIGxlbj1hcnJheS5sZW5ndGgqYnl0ZXM7bGVuPWxlbis3Ji04O3ZhciByZXQ7aWYoZW5zdXJlQ2FjaGUucG9zK2xlbj49ZW5zdXJlQ2FjaGUuc2l6ZSl7YXNzZXJ0KGxlbj4wKTtlbnN1cmVDYWNoZS5uZWVkZWQrPWxlbjtyZXQ9TW9kdWxlWyJfbWFsbG9jIl0obGVuKTtlbnN1cmVDYWNoZS50ZW1wcy5wdXNoKHJldCl9ZWxzZXtyZXQ9ZW5zdXJlQ2FjaGUuYnVmZmVyK2Vuc3VyZUNhY2hlLnBvcztlbnN1cmVDYWNoZS5wb3MrPWxlbn1yZXR1cm4gcmV0fSxjb3B5OmZ1bmN0aW9uKGFycmF5LHZpZXcsb2Zmc2V0KXtvZmZzZXQ+Pj49MDt2YXIgYnl0ZXM9dmlldy5CWVRFU19QRVJfRUxFTUVOVDtzd2l0Y2goYnl0ZXMpe2Nhc2UgMjpvZmZzZXQ+Pj49MTticmVhaztjYXNlIDQ6b2Zmc2V0Pj4+PTI7YnJlYWs7Y2FzZSA4Om9mZnNldD4+Pj0zO2JyZWFrfWZvcih2YXIgaT0wO2k8YXJyYXkubGVuZ3RoO2krKyl7dmlld1tvZmZzZXQraV09YXJyYXlbaV19fX07ZnVuY3Rpb24gZW5zdXJlU3RyaW5nKHZhbHVlKXtpZih0eXBlb2YgdmFsdWU9PT0ic3RyaW5nIil7dmFyIGludEFycmF5PWludEFycmF5RnJvbVN0cmluZyh2YWx1ZSk7dmFyIG9mZnNldD1lbnN1cmVDYWNoZS5hbGxvYyhpbnRBcnJheSxIRUFQOCk7ZW5zdXJlQ2FjaGUuY29weShpbnRBcnJheSxIRUFQOCxvZmZzZXQpO3JldHVybiBvZmZzZXR9cmV0dXJuIHZhbHVlfWZ1bmN0aW9uIGVuc3VyZUludDgodmFsdWUpe2lmKHR5cGVvZiB2YWx1ZT09PSJvYmplY3QiKXt2YXIgb2Zmc2V0PWVuc3VyZUNhY2hlLmFsbG9jKHZhbHVlLEhFQVA4KTtlbnN1cmVDYWNoZS5jb3B5KHZhbHVlLEhFQVA4LG9mZnNldCk7cmV0dXJuIG9mZnNldH1yZXR1cm4gdmFsdWV9ZnVuY3Rpb24gVm9pZFB0cigpe3Rocm93ImNhbm5vdCBjb25zdHJ1Y3QgYSBWb2lkUHRyLCBubyBjb25zdHJ1Y3RvciBpbiBJREwifVZvaWRQdHIucHJvdG90eXBlPU9iamVjdC5jcmVhdGUoV3JhcHBlck9iamVjdC5wcm90b3R5cGUpO1ZvaWRQdHIucHJvdG90eXBlLmNvbnN0cnVjdG9yPVZvaWRQdHI7Vm9pZFB0ci5wcm90b3R5cGUuX19jbGFzc19fPVZvaWRQdHI7Vm9pZFB0ci5fX2NhY2hlX189e307TW9kdWxlWyJWb2lkUHRyIl09Vm9pZFB0cjtWb2lkUHRyLnByb3RvdHlwZVsiX19kZXN0cm95X18iXT1Wb2lkUHRyLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO19lbXNjcmlwdGVuX2JpbmRfVm9pZFB0cl9fX2Rlc3Ryb3lfX18wKHNlbGYpfTtmdW5jdGlvbiBEZWNvZGVyQnVmZmVyKCl7dGhpcy5wdHI9X2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyQnVmZmVyX0RlY29kZXJCdWZmZXJfMCgpO2dldENhY2hlKERlY29kZXJCdWZmZXIpW3RoaXMucHRyXT10aGlzfURlY29kZXJCdWZmZXIucHJvdG90eXBlPU9iamVjdC5jcmVhdGUoV3JhcHBlck9iamVjdC5wcm90b3R5cGUpO0RlY29kZXJCdWZmZXIucHJvdG90eXBlLmNvbnN0cnVjdG9yPURlY29kZXJCdWZmZXI7RGVjb2RlckJ1ZmZlci5wcm90b3R5cGUuX19jbGFzc19fPURlY29kZXJCdWZmZXI7RGVjb2RlckJ1ZmZlci5fX2NhY2hlX189e307TW9kdWxlWyJEZWNvZGVyQnVmZmVyIl09RGVjb2RlckJ1ZmZlcjtEZWNvZGVyQnVmZmVyLnByb3RvdHlwZVsiSW5pdCJdPURlY29kZXJCdWZmZXIucHJvdG90eXBlLkluaXQ9ZnVuY3Rpb24oZGF0YSxkYXRhX3NpemUpe3ZhciBzZWxmPXRoaXMucHRyO2Vuc3VyZUNhY2hlLnByZXBhcmUoKTtpZih0eXBlb2YgZGF0YT09Im9iamVjdCIpe2RhdGE9ZW5zdXJlSW50OChkYXRhKX1pZihkYXRhX3NpemUmJnR5cGVvZiBkYXRhX3NpemU9PT0ib2JqZWN0IilkYXRhX3NpemU9ZGF0YV9zaXplLnB0cjtfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJCdWZmZXJfSW5pdF8yKHNlbGYsZGF0YSxkYXRhX3NpemUpfTtEZWNvZGVyQnVmZmVyLnByb3RvdHlwZVsiX19kZXN0cm95X18iXT1EZWNvZGVyQnVmZmVyLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO19lbXNjcmlwdGVuX2JpbmRfRGVjb2RlckJ1ZmZlcl9fX2Rlc3Ryb3lfX18wKHNlbGYpfTtmdW5jdGlvbiBBdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhKCl7dGhpcy5wdHI9X2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhX0F0dHJpYnV0ZVRyYW5zZm9ybURhdGFfMCgpO2dldENhY2hlKEF0dHJpYnV0ZVRyYW5zZm9ybURhdGEpW3RoaXMucHRyXT10aGlzfUF0dHJpYnV0ZVRyYW5zZm9ybURhdGEucHJvdG90eXBlPU9iamVjdC5jcmVhdGUoV3JhcHBlck9iamVjdC5wcm90b3R5cGUpO0F0dHJpYnV0ZVRyYW5zZm9ybURhdGEucHJvdG90eXBlLmNvbnN0cnVjdG9yPUF0dHJpYnV0ZVRyYW5zZm9ybURhdGE7QXR0cmlidXRlVHJhbnNmb3JtRGF0YS5wcm90b3R5cGUuX19jbGFzc19fPUF0dHJpYnV0ZVRyYW5zZm9ybURhdGE7QXR0cmlidXRlVHJhbnNmb3JtRGF0YS5fX2NhY2hlX189e307TW9kdWxlWyJBdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhIl09QXR0cmlidXRlVHJhbnNmb3JtRGF0YTtBdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhLnByb3RvdHlwZVsidHJhbnNmb3JtX3R5cGUiXT1BdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhLnByb3RvdHlwZS50cmFuc2Zvcm1fdHlwZT1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO3JldHVybiBfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVRyYW5zZm9ybURhdGFfdHJhbnNmb3JtX3R5cGVfMChzZWxmKX07QXR0cmlidXRlVHJhbnNmb3JtRGF0YS5wcm90b3R5cGVbIl9fZGVzdHJveV9fIl09QXR0cmlidXRlVHJhbnNmb3JtRGF0YS5wcm90b3R5cGUuX19kZXN0cm95X189ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVRyYW5zZm9ybURhdGFfX19kZXN0cm95X19fMChzZWxmKX07ZnVuY3Rpb24gR2VvbWV0cnlBdHRyaWJ1dGUoKXt0aGlzLnB0cj1fZW1zY3JpcHRlbl9iaW5kX0dlb21ldHJ5QXR0cmlidXRlX0dlb21ldHJ5QXR0cmlidXRlXzAoKTtnZXRDYWNoZShHZW9tZXRyeUF0dHJpYnV0ZSlbdGhpcy5wdHJdPXRoaXN9R2VvbWV0cnlBdHRyaWJ1dGUucHJvdG90eXBlPU9iamVjdC5jcmVhdGUoV3JhcHBlck9iamVjdC5wcm90b3R5cGUpO0dlb21ldHJ5QXR0cmlidXRlLnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1HZW9tZXRyeUF0dHJpYnV0ZTtHZW9tZXRyeUF0dHJpYnV0ZS5wcm90b3R5cGUuX19jbGFzc19fPUdlb21ldHJ5QXR0cmlidXRlO0dlb21ldHJ5QXR0cmlidXRlLl9fY2FjaGVfXz17fTtNb2R1bGVbIkdlb21ldHJ5QXR0cmlidXRlIl09R2VvbWV0cnlBdHRyaWJ1dGU7R2VvbWV0cnlBdHRyaWJ1dGUucHJvdG90eXBlWyJfX2Rlc3Ryb3lfXyJdPUdlb21ldHJ5QXR0cmlidXRlLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO19lbXNjcmlwdGVuX2JpbmRfR2VvbWV0cnlBdHRyaWJ1dGVfX19kZXN0cm95X19fMChzZWxmKX07ZnVuY3Rpb24gUG9pbnRBdHRyaWJ1dGUoKXt0aGlzLnB0cj1fZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX1BvaW50QXR0cmlidXRlXzAoKTtnZXRDYWNoZShQb2ludEF0dHJpYnV0ZSlbdGhpcy5wdHJdPXRoaXN9UG9pbnRBdHRyaWJ1dGUucHJvdG90eXBlPU9iamVjdC5jcmVhdGUoV3JhcHBlck9iamVjdC5wcm90b3R5cGUpO1BvaW50QXR0cmlidXRlLnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1Qb2ludEF0dHJpYnV0ZTtQb2ludEF0dHJpYnV0ZS5wcm90b3R5cGUuX19jbGFzc19fPVBvaW50QXR0cmlidXRlO1BvaW50QXR0cmlidXRlLl9fY2FjaGVfXz17fTtNb2R1bGVbIlBvaW50QXR0cmlidXRlIl09UG9pbnRBdHRyaWJ1dGU7UG9pbnRBdHRyaWJ1dGUucHJvdG90eXBlWyJzaXplIl09UG9pbnRBdHRyaWJ1dGUucHJvdG90eXBlLnNpemU9ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtyZXR1cm4gX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9zaXplXzAoc2VsZil9O1BvaW50QXR0cmlidXRlLnByb3RvdHlwZVsiR2V0QXR0cmlidXRlVHJhbnNmb3JtRGF0YSJdPVBvaW50QXR0cmlidXRlLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhPWZ1bmN0aW9uKCl7dmFyIHNlbGY9dGhpcy5wdHI7cmV0dXJuIHdyYXBQb2ludGVyKF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfR2V0QXR0cmlidXRlVHJhbnNmb3JtRGF0YV8wKHNlbGYpLEF0dHJpYnV0ZVRyYW5zZm9ybURhdGEpfTtQb2ludEF0dHJpYnV0ZS5wcm90b3R5cGVbImF0dHJpYnV0ZV90eXBlIl09UG9pbnRBdHRyaWJ1dGUucHJvdG90eXBlLmF0dHJpYnV0ZV90eXBlPWZ1bmN0aW9uKCl7dmFyIHNlbGY9dGhpcy5wdHI7cmV0dXJuIF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfYXR0cmlidXRlX3R5cGVfMChzZWxmKX07UG9pbnRBdHRyaWJ1dGUucHJvdG90eXBlWyJkYXRhX3R5cGUiXT1Qb2ludEF0dHJpYnV0ZS5wcm90b3R5cGUuZGF0YV90eXBlPWZ1bmN0aW9uKCl7dmFyIHNlbGY9dGhpcy5wdHI7cmV0dXJuIF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfZGF0YV90eXBlXzAoc2VsZil9O1BvaW50QXR0cmlidXRlLnByb3RvdHlwZVsibnVtX2NvbXBvbmVudHMiXT1Qb2ludEF0dHJpYnV0ZS5wcm90b3R5cGUubnVtX2NvbXBvbmVudHM9ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtyZXR1cm4gX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9udW1fY29tcG9uZW50c18wKHNlbGYpfTtQb2ludEF0dHJpYnV0ZS5wcm90b3R5cGVbIm5vcm1hbGl6ZWQiXT1Qb2ludEF0dHJpYnV0ZS5wcm90b3R5cGUubm9ybWFsaXplZD1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO3JldHVybiEhX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9ub3JtYWxpemVkXzAoc2VsZil9O1BvaW50QXR0cmlidXRlLnByb3RvdHlwZVsiYnl0ZV9zdHJpZGUiXT1Qb2ludEF0dHJpYnV0ZS5wcm90b3R5cGUuYnl0ZV9zdHJpZGU9ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtyZXR1cm4gX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9ieXRlX3N0cmlkZV8wKHNlbGYpfTtQb2ludEF0dHJpYnV0ZS5wcm90b3R5cGVbImJ5dGVfb2Zmc2V0Il09UG9pbnRBdHRyaWJ1dGUucHJvdG90eXBlLmJ5dGVfb2Zmc2V0PWZ1bmN0aW9uKCl7dmFyIHNlbGY9dGhpcy5wdHI7cmV0dXJuIF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfYnl0ZV9vZmZzZXRfMChzZWxmKX07UG9pbnRBdHRyaWJ1dGUucHJvdG90eXBlWyJ1bmlxdWVfaWQiXT1Qb2ludEF0dHJpYnV0ZS5wcm90b3R5cGUudW5pcXVlX2lkPWZ1bmN0aW9uKCl7dmFyIHNlbGY9dGhpcy5wdHI7cmV0dXJuIF9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfdW5pcXVlX2lkXzAoc2VsZil9O1BvaW50QXR0cmlidXRlLnByb3RvdHlwZVsiX19kZXN0cm95X18iXT1Qb2ludEF0dHJpYnV0ZS5wcm90b3R5cGUuX19kZXN0cm95X189ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtfZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX19fZGVzdHJveV9fXzAoc2VsZil9O2Z1bmN0aW9uIEF0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybSgpe3RoaXMucHRyPV9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV8wKCk7Z2V0Q2FjaGUoQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtKVt0aGlzLnB0cl09dGhpc31BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm0ucHJvdG90eXBlPU9iamVjdC5jcmVhdGUoV3JhcHBlck9iamVjdC5wcm90b3R5cGUpO0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybS5wcm90b3R5cGUuY29uc3RydWN0b3I9QXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtO0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybS5wcm90b3R5cGUuX19jbGFzc19fPUF0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybTtBdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm0uX19jYWNoZV9fPXt9O01vZHVsZVsiQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtIl09QXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtO0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybS5wcm90b3R5cGVbIkluaXRGcm9tQXR0cmlidXRlIl09QXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtLnByb3RvdHlwZS5Jbml0RnJvbUF0dHJpYnV0ZT1mdW5jdGlvbihhdHQpe3ZhciBzZWxmPXRoaXMucHRyO2lmKGF0dCYmdHlwZW9mIGF0dD09PSJvYmplY3QiKWF0dD1hdHQucHRyO3JldHVybiEhX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm1fSW5pdEZyb21BdHRyaWJ1dGVfMShzZWxmLGF0dCl9O0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybS5wcm90b3R5cGVbInF1YW50aXphdGlvbl9iaXRzIl09QXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtLnByb3RvdHlwZS5xdWFudGl6YXRpb25fYml0cz1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO3JldHVybiBfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9xdWFudGl6YXRpb25fYml0c18wKHNlbGYpfTtBdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm0ucHJvdG90eXBlWyJtaW5fdmFsdWUiXT1BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm0ucHJvdG90eXBlLm1pbl92YWx1ZT1mdW5jdGlvbihheGlzKXt2YXIgc2VsZj10aGlzLnB0cjtpZihheGlzJiZ0eXBlb2YgYXhpcz09PSJvYmplY3QiKWF4aXM9YXhpcy5wdHI7cmV0dXJuIF9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtX21pbl92YWx1ZV8xKHNlbGYsYXhpcyl9O0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybS5wcm90b3R5cGVbInJhbmdlIl09QXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtLnByb3RvdHlwZS5yYW5nZT1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO3JldHVybiBfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9yYW5nZV8wKHNlbGYpfTtBdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm0ucHJvdG90eXBlWyJfX2Rlc3Ryb3lfXyJdPUF0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybS5wcm90b3R5cGUuX19kZXN0cm95X189ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9fX2Rlc3Ryb3lfX18wKHNlbGYpfTtmdW5jdGlvbiBBdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtKCl7dGhpcy5wdHI9X2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtX0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm1fMCgpO2dldENhY2hlKEF0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm0pW3RoaXMucHRyXT10aGlzfUF0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm0ucHJvdG90eXBlPU9iamVjdC5jcmVhdGUoV3JhcHBlck9iamVjdC5wcm90b3R5cGUpO0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm0ucHJvdG90eXBlLmNvbnN0cnVjdG9yPUF0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm07QXR0cmlidXRlT2N0YWhlZHJvblRyYW5zZm9ybS5wcm90b3R5cGUuX19jbGFzc19fPUF0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm07QXR0cmlidXRlT2N0YWhlZHJvblRyYW5zZm9ybS5fX2NhY2hlX189e307TW9kdWxlWyJBdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtIl09QXR0cmlidXRlT2N0YWhlZHJvblRyYW5zZm9ybTtBdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtLnByb3RvdHlwZVsiSW5pdEZyb21BdHRyaWJ1dGUiXT1BdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtLnByb3RvdHlwZS5Jbml0RnJvbUF0dHJpYnV0ZT1mdW5jdGlvbihhdHQpe3ZhciBzZWxmPXRoaXMucHRyO2lmKGF0dCYmdHlwZW9mIGF0dD09PSJvYmplY3QiKWF0dD1hdHQucHRyO3JldHVybiEhX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtX0luaXRGcm9tQXR0cmlidXRlXzEoc2VsZixhdHQpfTtBdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtLnByb3RvdHlwZVsicXVhbnRpemF0aW9uX2JpdHMiXT1BdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtLnByb3RvdHlwZS5xdWFudGl6YXRpb25fYml0cz1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO3JldHVybiBfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm1fcXVhbnRpemF0aW9uX2JpdHNfMChzZWxmKX07QXR0cmlidXRlT2N0YWhlZHJvblRyYW5zZm9ybS5wcm90b3R5cGVbIl9fZGVzdHJveV9fIl09QXR0cmlidXRlT2N0YWhlZHJvblRyYW5zZm9ybS5wcm90b3R5cGUuX19kZXN0cm95X189ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtfZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm1fX19kZXN0cm95X19fMChzZWxmKX07ZnVuY3Rpb24gUG9pbnRDbG91ZCgpe3RoaXMucHRyPV9lbXNjcmlwdGVuX2JpbmRfUG9pbnRDbG91ZF9Qb2ludENsb3VkXzAoKTtnZXRDYWNoZShQb2ludENsb3VkKVt0aGlzLnB0cl09dGhpc31Qb2ludENsb3VkLnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKFdyYXBwZXJPYmplY3QucHJvdG90eXBlKTtQb2ludENsb3VkLnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1Qb2ludENsb3VkO1BvaW50Q2xvdWQucHJvdG90eXBlLl9fY2xhc3NfXz1Qb2ludENsb3VkO1BvaW50Q2xvdWQuX19jYWNoZV9fPXt9O01vZHVsZVsiUG9pbnRDbG91ZCJdPVBvaW50Q2xvdWQ7UG9pbnRDbG91ZC5wcm90b3R5cGVbIm51bV9hdHRyaWJ1dGVzIl09UG9pbnRDbG91ZC5wcm90b3R5cGUubnVtX2F0dHJpYnV0ZXM9ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtyZXR1cm4gX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludENsb3VkX251bV9hdHRyaWJ1dGVzXzAoc2VsZil9O1BvaW50Q2xvdWQucHJvdG90eXBlWyJudW1fcG9pbnRzIl09UG9pbnRDbG91ZC5wcm90b3R5cGUubnVtX3BvaW50cz1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO3JldHVybiBfZW1zY3JpcHRlbl9iaW5kX1BvaW50Q2xvdWRfbnVtX3BvaW50c18wKHNlbGYpfTtQb2ludENsb3VkLnByb3RvdHlwZVsiX19kZXN0cm95X18iXT1Qb2ludENsb3VkLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO19lbXNjcmlwdGVuX2JpbmRfUG9pbnRDbG91ZF9fX2Rlc3Ryb3lfX18wKHNlbGYpfTtmdW5jdGlvbiBNZXNoKCl7dGhpcy5wdHI9X2Vtc2NyaXB0ZW5fYmluZF9NZXNoX01lc2hfMCgpO2dldENhY2hlKE1lc2gpW3RoaXMucHRyXT10aGlzfU1lc2gucHJvdG90eXBlPU9iamVjdC5jcmVhdGUoV3JhcHBlck9iamVjdC5wcm90b3R5cGUpO01lc2gucHJvdG90eXBlLmNvbnN0cnVjdG9yPU1lc2g7TWVzaC5wcm90b3R5cGUuX19jbGFzc19fPU1lc2g7TWVzaC5fX2NhY2hlX189e307TW9kdWxlWyJNZXNoIl09TWVzaDtNZXNoLnByb3RvdHlwZVsibnVtX2ZhY2VzIl09TWVzaC5wcm90b3R5cGUubnVtX2ZhY2VzPWZ1bmN0aW9uKCl7dmFyIHNlbGY9dGhpcy5wdHI7cmV0dXJuIF9lbXNjcmlwdGVuX2JpbmRfTWVzaF9udW1fZmFjZXNfMChzZWxmKX07TWVzaC5wcm90b3R5cGVbIm51bV9hdHRyaWJ1dGVzIl09TWVzaC5wcm90b3R5cGUubnVtX2F0dHJpYnV0ZXM9ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtyZXR1cm4gX2Vtc2NyaXB0ZW5fYmluZF9NZXNoX251bV9hdHRyaWJ1dGVzXzAoc2VsZil9O01lc2gucHJvdG90eXBlWyJudW1fcG9pbnRzIl09TWVzaC5wcm90b3R5cGUubnVtX3BvaW50cz1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO3JldHVybiBfZW1zY3JpcHRlbl9iaW5kX01lc2hfbnVtX3BvaW50c18wKHNlbGYpfTtNZXNoLnByb3RvdHlwZVsiX19kZXN0cm95X18iXT1NZXNoLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO19lbXNjcmlwdGVuX2JpbmRfTWVzaF9fX2Rlc3Ryb3lfX18wKHNlbGYpfTtmdW5jdGlvbiBNZXRhZGF0YSgpe3RoaXMucHRyPV9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFfTWV0YWRhdGFfMCgpO2dldENhY2hlKE1ldGFkYXRhKVt0aGlzLnB0cl09dGhpc31NZXRhZGF0YS5wcm90b3R5cGU9T2JqZWN0LmNyZWF0ZShXcmFwcGVyT2JqZWN0LnByb3RvdHlwZSk7TWV0YWRhdGEucHJvdG90eXBlLmNvbnN0cnVjdG9yPU1ldGFkYXRhO01ldGFkYXRhLnByb3RvdHlwZS5fX2NsYXNzX189TWV0YWRhdGE7TWV0YWRhdGEuX19jYWNoZV9fPXt9O01vZHVsZVsiTWV0YWRhdGEiXT1NZXRhZGF0YTtNZXRhZGF0YS5wcm90b3R5cGVbIl9fZGVzdHJveV9fIl09TWV0YWRhdGEucHJvdG90eXBlLl9fZGVzdHJveV9fPWZ1bmN0aW9uKCl7dmFyIHNlbGY9dGhpcy5wdHI7X2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YV9fX2Rlc3Ryb3lfX18wKHNlbGYpfTtmdW5jdGlvbiBTdGF0dXMoKXt0aHJvdyJjYW5ub3QgY29uc3RydWN0IGEgU3RhdHVzLCBubyBjb25zdHJ1Y3RvciBpbiBJREwifVN0YXR1cy5wcm90b3R5cGU9T2JqZWN0LmNyZWF0ZShXcmFwcGVyT2JqZWN0LnByb3RvdHlwZSk7U3RhdHVzLnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1TdGF0dXM7U3RhdHVzLnByb3RvdHlwZS5fX2NsYXNzX189U3RhdHVzO1N0YXR1cy5fX2NhY2hlX189e307TW9kdWxlWyJTdGF0dXMiXT1TdGF0dXM7U3RhdHVzLnByb3RvdHlwZVsiY29kZSJdPVN0YXR1cy5wcm90b3R5cGUuY29kZT1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO3JldHVybiBfZW1zY3JpcHRlbl9iaW5kX1N0YXR1c19jb2RlXzAoc2VsZil9O1N0YXR1cy5wcm90b3R5cGVbIm9rIl09U3RhdHVzLnByb3RvdHlwZS5vaz1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO3JldHVybiEhX2Vtc2NyaXB0ZW5fYmluZF9TdGF0dXNfb2tfMChzZWxmKX07U3RhdHVzLnByb3RvdHlwZVsiZXJyb3JfbXNnIl09U3RhdHVzLnByb3RvdHlwZS5lcnJvcl9tc2c9ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtyZXR1cm4gVVRGOFRvU3RyaW5nKF9lbXNjcmlwdGVuX2JpbmRfU3RhdHVzX2Vycm9yX21zZ18wKHNlbGYpKX07U3RhdHVzLnByb3RvdHlwZVsiX19kZXN0cm95X18iXT1TdGF0dXMucHJvdG90eXBlLl9fZGVzdHJveV9fPWZ1bmN0aW9uKCl7dmFyIHNlbGY9dGhpcy5wdHI7X2Vtc2NyaXB0ZW5fYmluZF9TdGF0dXNfX19kZXN0cm95X19fMChzZWxmKX07ZnVuY3Rpb24gRHJhY29GbG9hdDMyQXJyYXkoKXt0aGlzLnB0cj1fZW1zY3JpcHRlbl9iaW5kX0RyYWNvRmxvYXQzMkFycmF5X0RyYWNvRmxvYXQzMkFycmF5XzAoKTtnZXRDYWNoZShEcmFjb0Zsb2F0MzJBcnJheSlbdGhpcy5wdHJdPXRoaXN9RHJhY29GbG9hdDMyQXJyYXkucHJvdG90eXBlPU9iamVjdC5jcmVhdGUoV3JhcHBlck9iamVjdC5wcm90b3R5cGUpO0RyYWNvRmxvYXQzMkFycmF5LnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1EcmFjb0Zsb2F0MzJBcnJheTtEcmFjb0Zsb2F0MzJBcnJheS5wcm90b3R5cGUuX19jbGFzc19fPURyYWNvRmxvYXQzMkFycmF5O0RyYWNvRmxvYXQzMkFycmF5Ll9fY2FjaGVfXz17fTtNb2R1bGVbIkRyYWNvRmxvYXQzMkFycmF5Il09RHJhY29GbG9hdDMyQXJyYXk7RHJhY29GbG9hdDMyQXJyYXkucHJvdG90eXBlWyJHZXRWYWx1ZSJdPURyYWNvRmxvYXQzMkFycmF5LnByb3RvdHlwZS5HZXRWYWx1ZT1mdW5jdGlvbihpbmRleCl7dmFyIHNlbGY9dGhpcy5wdHI7aWYoaW5kZXgmJnR5cGVvZiBpbmRleD09PSJvYmplY3QiKWluZGV4PWluZGV4LnB0cjtyZXR1cm4gX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0Zsb2F0MzJBcnJheV9HZXRWYWx1ZV8xKHNlbGYsaW5kZXgpfTtEcmFjb0Zsb2F0MzJBcnJheS5wcm90b3R5cGVbInNpemUiXT1EcmFjb0Zsb2F0MzJBcnJheS5wcm90b3R5cGUuc2l6ZT1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO3JldHVybiBfZW1zY3JpcHRlbl9iaW5kX0RyYWNvRmxvYXQzMkFycmF5X3NpemVfMChzZWxmKX07RHJhY29GbG9hdDMyQXJyYXkucHJvdG90eXBlWyJfX2Rlc3Ryb3lfXyJdPURyYWNvRmxvYXQzMkFycmF5LnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO19lbXNjcmlwdGVuX2JpbmRfRHJhY29GbG9hdDMyQXJyYXlfX19kZXN0cm95X19fMChzZWxmKX07ZnVuY3Rpb24gRHJhY29JbnQ4QXJyYXkoKXt0aGlzLnB0cj1fZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50OEFycmF5X0RyYWNvSW50OEFycmF5XzAoKTtnZXRDYWNoZShEcmFjb0ludDhBcnJheSlbdGhpcy5wdHJdPXRoaXN9RHJhY29JbnQ4QXJyYXkucHJvdG90eXBlPU9iamVjdC5jcmVhdGUoV3JhcHBlck9iamVjdC5wcm90b3R5cGUpO0RyYWNvSW50OEFycmF5LnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1EcmFjb0ludDhBcnJheTtEcmFjb0ludDhBcnJheS5wcm90b3R5cGUuX19jbGFzc19fPURyYWNvSW50OEFycmF5O0RyYWNvSW50OEFycmF5Ll9fY2FjaGVfXz17fTtNb2R1bGVbIkRyYWNvSW50OEFycmF5Il09RHJhY29JbnQ4QXJyYXk7RHJhY29JbnQ4QXJyYXkucHJvdG90eXBlWyJHZXRWYWx1ZSJdPURyYWNvSW50OEFycmF5LnByb3RvdHlwZS5HZXRWYWx1ZT1mdW5jdGlvbihpbmRleCl7dmFyIHNlbGY9dGhpcy5wdHI7aWYoaW5kZXgmJnR5cGVvZiBpbmRleD09PSJvYmplY3QiKWluZGV4PWluZGV4LnB0cjtyZXR1cm4gX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDhBcnJheV9HZXRWYWx1ZV8xKHNlbGYsaW5kZXgpfTtEcmFjb0ludDhBcnJheS5wcm90b3R5cGVbInNpemUiXT1EcmFjb0ludDhBcnJheS5wcm90b3R5cGUuc2l6ZT1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO3JldHVybiBfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50OEFycmF5X3NpemVfMChzZWxmKX07RHJhY29JbnQ4QXJyYXkucHJvdG90eXBlWyJfX2Rlc3Ryb3lfXyJdPURyYWNvSW50OEFycmF5LnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1mdW5jdGlvbigpe3ZhciBzZWxmPXRoaXMucHRyO19lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQ4QXJyYXlfX19kZXN0cm95X19fMChzZWxmKX07ZnVuY3Rpb24gRHJhY29VSW50OEFycmF5KCl7dGhpcy5wdHI9X2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQ4QXJyYXlfRHJhY29VSW50OEFycmF5XzAoKTtnZXRDYWNoZShEcmFjb1VJbnQ4QXJyYXkpW3RoaXMucHRyXT10aGlzfURyYWNvVUludDhBcnJheS5wcm90b3R5cGU9T2JqZWN0LmNyZWF0ZShXcmFwcGVyT2JqZWN0LnByb3RvdHlwZSk7RHJhY29VSW50OEFycmF5LnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1EcmFjb1VJbnQ4QXJyYXk7RHJhY29VSW50OEFycmF5LnByb3RvdHlwZS5fX2NsYXNzX189RHJhY29VSW50OEFycmF5O0RyYWNvVUludDhBcnJheS5fX2NhY2hlX189e307TW9kdWxlWyJEcmFjb1VJbnQ4QXJyYXkiXT1EcmFjb1VJbnQ4QXJyYXk7RHJhY29VSW50OEFycmF5LnByb3RvdHlwZVsiR2V0VmFsdWUiXT1EcmFjb1VJbnQ4QXJyYXkucHJvdG90eXBlLkdldFZhbHVlPWZ1bmN0aW9uKGluZGV4KXt2YXIgc2VsZj10aGlzLnB0cjtpZihpbmRleCYmdHlwZW9mIGluZGV4PT09Im9iamVjdCIpaW5kZXg9aW5kZXgucHRyO3JldHVybiBfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDhBcnJheV9HZXRWYWx1ZV8xKHNlbGYsaW5kZXgpfTtEcmFjb1VJbnQ4QXJyYXkucHJvdG90eXBlWyJzaXplIl09RHJhY29VSW50OEFycmF5LnByb3RvdHlwZS5zaXplPWZ1bmN0aW9uKCl7dmFyIHNlbGY9dGhpcy5wdHI7cmV0dXJuIF9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50OEFycmF5X3NpemVfMChzZWxmKX07RHJhY29VSW50OEFycmF5LnByb3RvdHlwZVsiX19kZXN0cm95X18iXT1EcmFjb1VJbnQ4QXJyYXkucHJvdG90eXBlLl9fZGVzdHJveV9fPWZ1bmN0aW9uKCl7dmFyIHNlbGY9dGhpcy5wdHI7X2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQ4QXJyYXlfX19kZXN0cm95X19fMChzZWxmKX07ZnVuY3Rpb24gRHJhY29JbnQxNkFycmF5KCl7dGhpcy5wdHI9X2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDE2QXJyYXlfRHJhY29JbnQxNkFycmF5XzAoKTtnZXRDYWNoZShEcmFjb0ludDE2QXJyYXkpW3RoaXMucHRyXT10aGlzfURyYWNvSW50MTZBcnJheS5wcm90b3R5cGU9T2JqZWN0LmNyZWF0ZShXcmFwcGVyT2JqZWN0LnByb3RvdHlwZSk7RHJhY29JbnQxNkFycmF5LnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1EcmFjb0ludDE2QXJyYXk7RHJhY29JbnQxNkFycmF5LnByb3RvdHlwZS5fX2NsYXNzX189RHJhY29JbnQxNkFycmF5O0RyYWNvSW50MTZBcnJheS5fX2NhY2hlX189e307TW9kdWxlWyJEcmFjb0ludDE2QXJyYXkiXT1EcmFjb0ludDE2QXJyYXk7RHJhY29JbnQxNkFycmF5LnByb3RvdHlwZVsiR2V0VmFsdWUiXT1EcmFjb0ludDE2QXJyYXkucHJvdG90eXBlLkdldFZhbHVlPWZ1bmN0aW9uKGluZGV4KXt2YXIgc2VsZj10aGlzLnB0cjtpZihpbmRleCYmdHlwZW9mIGluZGV4PT09Im9iamVjdCIpaW5kZXg9aW5kZXgucHRyO3JldHVybiBfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MTZBcnJheV9HZXRWYWx1ZV8xKHNlbGYsaW5kZXgpfTtEcmFjb0ludDE2QXJyYXkucHJvdG90eXBlWyJzaXplIl09RHJhY29JbnQxNkFycmF5LnByb3RvdHlwZS5zaXplPWZ1bmN0aW9uKCl7dmFyIHNlbGY9dGhpcy5wdHI7cmV0dXJuIF9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQxNkFycmF5X3NpemVfMChzZWxmKX07RHJhY29JbnQxNkFycmF5LnByb3RvdHlwZVsiX19kZXN0cm95X18iXT1EcmFjb0ludDE2QXJyYXkucHJvdG90eXBlLl9fZGVzdHJveV9fPWZ1bmN0aW9uKCl7dmFyIHNlbGY9dGhpcy5wdHI7X2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDE2QXJyYXlfX19kZXN0cm95X19fMChzZWxmKX07ZnVuY3Rpb24gRHJhY29VSW50MTZBcnJheSgpe3RoaXMucHRyPV9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MTZBcnJheV9EcmFjb1VJbnQxNkFycmF5XzAoKTtnZXRDYWNoZShEcmFjb1VJbnQxNkFycmF5KVt0aGlzLnB0cl09dGhpc31EcmFjb1VJbnQxNkFycmF5LnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKFdyYXBwZXJPYmplY3QucHJvdG90eXBlKTtEcmFjb1VJbnQxNkFycmF5LnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1EcmFjb1VJbnQxNkFycmF5O0RyYWNvVUludDE2QXJyYXkucHJvdG90eXBlLl9fY2xhc3NfXz1EcmFjb1VJbnQxNkFycmF5O0RyYWNvVUludDE2QXJyYXkuX19jYWNoZV9fPXt9O01vZHVsZVsiRHJhY29VSW50MTZBcnJheSJdPURyYWNvVUludDE2QXJyYXk7RHJhY29VSW50MTZBcnJheS5wcm90b3R5cGVbIkdldFZhbHVlIl09RHJhY29VSW50MTZBcnJheS5wcm90b3R5cGUuR2V0VmFsdWU9ZnVuY3Rpb24oaW5kZXgpe3ZhciBzZWxmPXRoaXMucHRyO2lmKGluZGV4JiZ0eXBlb2YgaW5kZXg9PT0ib2JqZWN0IilpbmRleD1pbmRleC5wdHI7cmV0dXJuIF9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MTZBcnJheV9HZXRWYWx1ZV8xKHNlbGYsaW5kZXgpfTtEcmFjb1VJbnQxNkFycmF5LnByb3RvdHlwZVsic2l6ZSJdPURyYWNvVUludDE2QXJyYXkucHJvdG90eXBlLnNpemU9ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtyZXR1cm4gX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQxNkFycmF5X3NpemVfMChzZWxmKX07RHJhY29VSW50MTZBcnJheS5wcm90b3R5cGVbIl9fZGVzdHJveV9fIl09RHJhY29VSW50MTZBcnJheS5wcm90b3R5cGUuX19kZXN0cm95X189ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDE2QXJyYXlfX19kZXN0cm95X19fMChzZWxmKX07ZnVuY3Rpb24gRHJhY29JbnQzMkFycmF5KCl7dGhpcy5wdHI9X2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDMyQXJyYXlfRHJhY29JbnQzMkFycmF5XzAoKTtnZXRDYWNoZShEcmFjb0ludDMyQXJyYXkpW3RoaXMucHRyXT10aGlzfURyYWNvSW50MzJBcnJheS5wcm90b3R5cGU9T2JqZWN0LmNyZWF0ZShXcmFwcGVyT2JqZWN0LnByb3RvdHlwZSk7RHJhY29JbnQzMkFycmF5LnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1EcmFjb0ludDMyQXJyYXk7RHJhY29JbnQzMkFycmF5LnByb3RvdHlwZS5fX2NsYXNzX189RHJhY29JbnQzMkFycmF5O0RyYWNvSW50MzJBcnJheS5fX2NhY2hlX189e307TW9kdWxlWyJEcmFjb0ludDMyQXJyYXkiXT1EcmFjb0ludDMyQXJyYXk7RHJhY29JbnQzMkFycmF5LnByb3RvdHlwZVsiR2V0VmFsdWUiXT1EcmFjb0ludDMyQXJyYXkucHJvdG90eXBlLkdldFZhbHVlPWZ1bmN0aW9uKGluZGV4KXt2YXIgc2VsZj10aGlzLnB0cjtpZihpbmRleCYmdHlwZW9mIGluZGV4PT09Im9iamVjdCIpaW5kZXg9aW5kZXgucHRyO3JldHVybiBfZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MzJBcnJheV9HZXRWYWx1ZV8xKHNlbGYsaW5kZXgpfTtEcmFjb0ludDMyQXJyYXkucHJvdG90eXBlWyJzaXplIl09RHJhY29JbnQzMkFycmF5LnByb3RvdHlwZS5zaXplPWZ1bmN0aW9uKCl7dmFyIHNlbGY9dGhpcy5wdHI7cmV0dXJuIF9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQzMkFycmF5X3NpemVfMChzZWxmKX07RHJhY29JbnQzMkFycmF5LnByb3RvdHlwZVsiX19kZXN0cm95X18iXT1EcmFjb0ludDMyQXJyYXkucHJvdG90eXBlLl9fZGVzdHJveV9fPWZ1bmN0aW9uKCl7dmFyIHNlbGY9dGhpcy5wdHI7X2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDMyQXJyYXlfX19kZXN0cm95X19fMChzZWxmKX07ZnVuY3Rpb24gRHJhY29VSW50MzJBcnJheSgpe3RoaXMucHRyPV9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MzJBcnJheV9EcmFjb1VJbnQzMkFycmF5XzAoKTtnZXRDYWNoZShEcmFjb1VJbnQzMkFycmF5KVt0aGlzLnB0cl09dGhpc31EcmFjb1VJbnQzMkFycmF5LnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKFdyYXBwZXJPYmplY3QucHJvdG90eXBlKTtEcmFjb1VJbnQzMkFycmF5LnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1EcmFjb1VJbnQzMkFycmF5O0RyYWNvVUludDMyQXJyYXkucHJvdG90eXBlLl9fY2xhc3NfXz1EcmFjb1VJbnQzMkFycmF5O0RyYWNvVUludDMyQXJyYXkuX19jYWNoZV9fPXt9O01vZHVsZVsiRHJhY29VSW50MzJBcnJheSJdPURyYWNvVUludDMyQXJyYXk7RHJhY29VSW50MzJBcnJheS5wcm90b3R5cGVbIkdldFZhbHVlIl09RHJhY29VSW50MzJBcnJheS5wcm90b3R5cGUuR2V0VmFsdWU9ZnVuY3Rpb24oaW5kZXgpe3ZhciBzZWxmPXRoaXMucHRyO2lmKGluZGV4JiZ0eXBlb2YgaW5kZXg9PT0ib2JqZWN0IilpbmRleD1pbmRleC5wdHI7cmV0dXJuIF9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MzJBcnJheV9HZXRWYWx1ZV8xKHNlbGYsaW5kZXgpfTtEcmFjb1VJbnQzMkFycmF5LnByb3RvdHlwZVsic2l6ZSJdPURyYWNvVUludDMyQXJyYXkucHJvdG90eXBlLnNpemU9ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtyZXR1cm4gX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQzMkFycmF5X3NpemVfMChzZWxmKX07RHJhY29VSW50MzJBcnJheS5wcm90b3R5cGVbIl9fZGVzdHJveV9fIl09RHJhY29VSW50MzJBcnJheS5wcm90b3R5cGUuX19kZXN0cm95X189ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtfZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDMyQXJyYXlfX19kZXN0cm95X19fMChzZWxmKX07ZnVuY3Rpb24gTWV0YWRhdGFRdWVyaWVyKCl7dGhpcy5wdHI9X2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfTWV0YWRhdGFRdWVyaWVyXzAoKTtnZXRDYWNoZShNZXRhZGF0YVF1ZXJpZXIpW3RoaXMucHRyXT10aGlzfU1ldGFkYXRhUXVlcmllci5wcm90b3R5cGU9T2JqZWN0LmNyZWF0ZShXcmFwcGVyT2JqZWN0LnByb3RvdHlwZSk7TWV0YWRhdGFRdWVyaWVyLnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1NZXRhZGF0YVF1ZXJpZXI7TWV0YWRhdGFRdWVyaWVyLnByb3RvdHlwZS5fX2NsYXNzX189TWV0YWRhdGFRdWVyaWVyO01ldGFkYXRhUXVlcmllci5fX2NhY2hlX189e307TW9kdWxlWyJNZXRhZGF0YVF1ZXJpZXIiXT1NZXRhZGF0YVF1ZXJpZXI7TWV0YWRhdGFRdWVyaWVyLnByb3RvdHlwZVsiSGFzRW50cnkiXT1NZXRhZGF0YVF1ZXJpZXIucHJvdG90eXBlLkhhc0VudHJ5PWZ1bmN0aW9uKG1ldGFkYXRhLGVudHJ5X25hbWUpe3ZhciBzZWxmPXRoaXMucHRyO2Vuc3VyZUNhY2hlLnByZXBhcmUoKTtpZihtZXRhZGF0YSYmdHlwZW9mIG1ldGFkYXRhPT09Im9iamVjdCIpbWV0YWRhdGE9bWV0YWRhdGEucHRyO2lmKGVudHJ5X25hbWUmJnR5cGVvZiBlbnRyeV9uYW1lPT09Im9iamVjdCIpZW50cnlfbmFtZT1lbnRyeV9uYW1lLnB0cjtlbHNlIGVudHJ5X25hbWU9ZW5zdXJlU3RyaW5nKGVudHJ5X25hbWUpO3JldHVybiEhX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfSGFzRW50cnlfMihzZWxmLG1ldGFkYXRhLGVudHJ5X25hbWUpfTtNZXRhZGF0YVF1ZXJpZXIucHJvdG90eXBlWyJHZXRJbnRFbnRyeSJdPU1ldGFkYXRhUXVlcmllci5wcm90b3R5cGUuR2V0SW50RW50cnk9ZnVuY3Rpb24obWV0YWRhdGEsZW50cnlfbmFtZSl7dmFyIHNlbGY9dGhpcy5wdHI7ZW5zdXJlQ2FjaGUucHJlcGFyZSgpO2lmKG1ldGFkYXRhJiZ0eXBlb2YgbWV0YWRhdGE9PT0ib2JqZWN0IiltZXRhZGF0YT1tZXRhZGF0YS5wdHI7aWYoZW50cnlfbmFtZSYmdHlwZW9mIGVudHJ5X25hbWU9PT0ib2JqZWN0IillbnRyeV9uYW1lPWVudHJ5X25hbWUucHRyO2Vsc2UgZW50cnlfbmFtZT1lbnN1cmVTdHJpbmcoZW50cnlfbmFtZSk7cmV0dXJuIF9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX0dldEludEVudHJ5XzIoc2VsZixtZXRhZGF0YSxlbnRyeV9uYW1lKX07TWV0YWRhdGFRdWVyaWVyLnByb3RvdHlwZVsiR2V0SW50RW50cnlBcnJheSJdPU1ldGFkYXRhUXVlcmllci5wcm90b3R5cGUuR2V0SW50RW50cnlBcnJheT1mdW5jdGlvbihtZXRhZGF0YSxlbnRyeV9uYW1lLG91dF92YWx1ZXMpe3ZhciBzZWxmPXRoaXMucHRyO2Vuc3VyZUNhY2hlLnByZXBhcmUoKTtpZihtZXRhZGF0YSYmdHlwZW9mIG1ldGFkYXRhPT09Im9iamVjdCIpbWV0YWRhdGE9bWV0YWRhdGEucHRyO2lmKGVudHJ5X25hbWUmJnR5cGVvZiBlbnRyeV9uYW1lPT09Im9iamVjdCIpZW50cnlfbmFtZT1lbnRyeV9uYW1lLnB0cjtlbHNlIGVudHJ5X25hbWU9ZW5zdXJlU3RyaW5nKGVudHJ5X25hbWUpO2lmKG91dF92YWx1ZXMmJnR5cGVvZiBvdXRfdmFsdWVzPT09Im9iamVjdCIpb3V0X3ZhbHVlcz1vdXRfdmFsdWVzLnB0cjtfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9HZXRJbnRFbnRyeUFycmF5XzMoc2VsZixtZXRhZGF0YSxlbnRyeV9uYW1lLG91dF92YWx1ZXMpfTtNZXRhZGF0YVF1ZXJpZXIucHJvdG90eXBlWyJHZXREb3VibGVFbnRyeSJdPU1ldGFkYXRhUXVlcmllci5wcm90b3R5cGUuR2V0RG91YmxlRW50cnk9ZnVuY3Rpb24obWV0YWRhdGEsZW50cnlfbmFtZSl7dmFyIHNlbGY9dGhpcy5wdHI7ZW5zdXJlQ2FjaGUucHJlcGFyZSgpO2lmKG1ldGFkYXRhJiZ0eXBlb2YgbWV0YWRhdGE9PT0ib2JqZWN0IiltZXRhZGF0YT1tZXRhZGF0YS5wdHI7aWYoZW50cnlfbmFtZSYmdHlwZW9mIGVudHJ5X25hbWU9PT0ib2JqZWN0IillbnRyeV9uYW1lPWVudHJ5X25hbWUucHRyO2Vsc2UgZW50cnlfbmFtZT1lbnN1cmVTdHJpbmcoZW50cnlfbmFtZSk7cmV0dXJuIF9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX0dldERvdWJsZUVudHJ5XzIoc2VsZixtZXRhZGF0YSxlbnRyeV9uYW1lKX07TWV0YWRhdGFRdWVyaWVyLnByb3RvdHlwZVsiR2V0U3RyaW5nRW50cnkiXT1NZXRhZGF0YVF1ZXJpZXIucHJvdG90eXBlLkdldFN0cmluZ0VudHJ5PWZ1bmN0aW9uKG1ldGFkYXRhLGVudHJ5X25hbWUpe3ZhciBzZWxmPXRoaXMucHRyO2Vuc3VyZUNhY2hlLnByZXBhcmUoKTtpZihtZXRhZGF0YSYmdHlwZW9mIG1ldGFkYXRhPT09Im9iamVjdCIpbWV0YWRhdGE9bWV0YWRhdGEucHRyO2lmKGVudHJ5X25hbWUmJnR5cGVvZiBlbnRyeV9uYW1lPT09Im9iamVjdCIpZW50cnlfbmFtZT1lbnRyeV9uYW1lLnB0cjtlbHNlIGVudHJ5X25hbWU9ZW5zdXJlU3RyaW5nKGVudHJ5X25hbWUpO3JldHVybiBVVEY4VG9TdHJpbmcoX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfR2V0U3RyaW5nRW50cnlfMihzZWxmLG1ldGFkYXRhLGVudHJ5X25hbWUpKX07TWV0YWRhdGFRdWVyaWVyLnByb3RvdHlwZVsiTnVtRW50cmllcyJdPU1ldGFkYXRhUXVlcmllci5wcm90b3R5cGUuTnVtRW50cmllcz1mdW5jdGlvbihtZXRhZGF0YSl7dmFyIHNlbGY9dGhpcy5wdHI7aWYobWV0YWRhdGEmJnR5cGVvZiBtZXRhZGF0YT09PSJvYmplY3QiKW1ldGFkYXRhPW1ldGFkYXRhLnB0cjtyZXR1cm4gX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfTnVtRW50cmllc18xKHNlbGYsbWV0YWRhdGEpfTtNZXRhZGF0YVF1ZXJpZXIucHJvdG90eXBlWyJHZXRFbnRyeU5hbWUiXT1NZXRhZGF0YVF1ZXJpZXIucHJvdG90eXBlLkdldEVudHJ5TmFtZT1mdW5jdGlvbihtZXRhZGF0YSxlbnRyeV9pZCl7dmFyIHNlbGY9dGhpcy5wdHI7aWYobWV0YWRhdGEmJnR5cGVvZiBtZXRhZGF0YT09PSJvYmplY3QiKW1ldGFkYXRhPW1ldGFkYXRhLnB0cjtpZihlbnRyeV9pZCYmdHlwZW9mIGVudHJ5X2lkPT09Im9iamVjdCIpZW50cnlfaWQ9ZW50cnlfaWQucHRyO3JldHVybiBVVEY4VG9TdHJpbmcoX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfR2V0RW50cnlOYW1lXzIoc2VsZixtZXRhZGF0YSxlbnRyeV9pZCkpfTtNZXRhZGF0YVF1ZXJpZXIucHJvdG90eXBlWyJfX2Rlc3Ryb3lfXyJdPU1ldGFkYXRhUXVlcmllci5wcm90b3R5cGUuX19kZXN0cm95X189ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtfZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9fX2Rlc3Ryb3lfX18wKHNlbGYpfTtmdW5jdGlvbiBEZWNvZGVyKCl7dGhpcy5wdHI9X2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0RlY29kZXJfMCgpO2dldENhY2hlKERlY29kZXIpW3RoaXMucHRyXT10aGlzfURlY29kZXIucHJvdG90eXBlPU9iamVjdC5jcmVhdGUoV3JhcHBlck9iamVjdC5wcm90b3R5cGUpO0RlY29kZXIucHJvdG90eXBlLmNvbnN0cnVjdG9yPURlY29kZXI7RGVjb2Rlci5wcm90b3R5cGUuX19jbGFzc19fPURlY29kZXI7RGVjb2Rlci5fX2NhY2hlX189e307TW9kdWxlWyJEZWNvZGVyIl09RGVjb2RlcjtEZWNvZGVyLnByb3RvdHlwZVsiRGVjb2RlQXJyYXlUb1BvaW50Q2xvdWQiXT1EZWNvZGVyLnByb3RvdHlwZS5EZWNvZGVBcnJheVRvUG9pbnRDbG91ZD1mdW5jdGlvbihkYXRhLGRhdGFfc2l6ZSxvdXRfcG9pbnRfY2xvdWQpe3ZhciBzZWxmPXRoaXMucHRyO2Vuc3VyZUNhY2hlLnByZXBhcmUoKTtpZih0eXBlb2YgZGF0YT09Im9iamVjdCIpe2RhdGE9ZW5zdXJlSW50OChkYXRhKX1pZihkYXRhX3NpemUmJnR5cGVvZiBkYXRhX3NpemU9PT0ib2JqZWN0IilkYXRhX3NpemU9ZGF0YV9zaXplLnB0cjtpZihvdXRfcG9pbnRfY2xvdWQmJnR5cGVvZiBvdXRfcG9pbnRfY2xvdWQ9PT0ib2JqZWN0IilvdXRfcG9pbnRfY2xvdWQ9b3V0X3BvaW50X2Nsb3VkLnB0cjtyZXR1cm4gd3JhcFBvaW50ZXIoX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0RlY29kZUFycmF5VG9Qb2ludENsb3VkXzMoc2VsZixkYXRhLGRhdGFfc2l6ZSxvdXRfcG9pbnRfY2xvdWQpLFN0YXR1cyl9O0RlY29kZXIucHJvdG90eXBlWyJEZWNvZGVBcnJheVRvTWVzaCJdPURlY29kZXIucHJvdG90eXBlLkRlY29kZUFycmF5VG9NZXNoPWZ1bmN0aW9uKGRhdGEsZGF0YV9zaXplLG91dF9tZXNoKXt2YXIgc2VsZj10aGlzLnB0cjtlbnN1cmVDYWNoZS5wcmVwYXJlKCk7aWYodHlwZW9mIGRhdGE9PSJvYmplY3QiKXtkYXRhPWVuc3VyZUludDgoZGF0YSl9aWYoZGF0YV9zaXplJiZ0eXBlb2YgZGF0YV9zaXplPT09Im9iamVjdCIpZGF0YV9zaXplPWRhdGFfc2l6ZS5wdHI7aWYob3V0X21lc2gmJnR5cGVvZiBvdXRfbWVzaD09PSJvYmplY3QiKW91dF9tZXNoPW91dF9tZXNoLnB0cjtyZXR1cm4gd3JhcFBvaW50ZXIoX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0RlY29kZUFycmF5VG9NZXNoXzMoc2VsZixkYXRhLGRhdGFfc2l6ZSxvdXRfbWVzaCksU3RhdHVzKX07RGVjb2Rlci5wcm90b3R5cGVbIkdldEF0dHJpYnV0ZUlkIl09RGVjb2Rlci5wcm90b3R5cGUuR2V0QXR0cmlidXRlSWQ9ZnVuY3Rpb24ocGMsdHlwZSl7dmFyIHNlbGY9dGhpcy5wdHI7aWYocGMmJnR5cGVvZiBwYz09PSJvYmplY3QiKXBjPXBjLnB0cjtpZih0eXBlJiZ0eXBlb2YgdHlwZT09PSJvYmplY3QiKXR5cGU9dHlwZS5wdHI7cmV0dXJuIF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVJZF8yKHNlbGYscGMsdHlwZSl9O0RlY29kZXIucHJvdG90eXBlWyJHZXRBdHRyaWJ1dGVJZEJ5TmFtZSJdPURlY29kZXIucHJvdG90eXBlLkdldEF0dHJpYnV0ZUlkQnlOYW1lPWZ1bmN0aW9uKHBjLG5hbWUpe3ZhciBzZWxmPXRoaXMucHRyO2Vuc3VyZUNhY2hlLnByZXBhcmUoKTtpZihwYyYmdHlwZW9mIHBjPT09Im9iamVjdCIpcGM9cGMucHRyO2lmKG5hbWUmJnR5cGVvZiBuYW1lPT09Im9iamVjdCIpbmFtZT1uYW1lLnB0cjtlbHNlIG5hbWU9ZW5zdXJlU3RyaW5nKG5hbWUpO3JldHVybiBfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSWRCeU5hbWVfMihzZWxmLHBjLG5hbWUpfTtEZWNvZGVyLnByb3RvdHlwZVsiR2V0QXR0cmlidXRlSWRCeU1ldGFkYXRhRW50cnkiXT1EZWNvZGVyLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVJZEJ5TWV0YWRhdGFFbnRyeT1mdW5jdGlvbihwYyxuYW1lLHZhbHVlKXt2YXIgc2VsZj10aGlzLnB0cjtlbnN1cmVDYWNoZS5wcmVwYXJlKCk7aWYocGMmJnR5cGVvZiBwYz09PSJvYmplY3QiKXBjPXBjLnB0cjtpZihuYW1lJiZ0eXBlb2YgbmFtZT09PSJvYmplY3QiKW5hbWU9bmFtZS5wdHI7ZWxzZSBuYW1lPWVuc3VyZVN0cmluZyhuYW1lKTtpZih2YWx1ZSYmdHlwZW9mIHZhbHVlPT09Im9iamVjdCIpdmFsdWU9dmFsdWUucHRyO2Vsc2UgdmFsdWU9ZW5zdXJlU3RyaW5nKHZhbHVlKTtyZXR1cm4gX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUlkQnlNZXRhZGF0YUVudHJ5XzMoc2VsZixwYyxuYW1lLHZhbHVlKX07RGVjb2Rlci5wcm90b3R5cGVbIkdldEF0dHJpYnV0ZSJdPURlY29kZXIucHJvdG90eXBlLkdldEF0dHJpYnV0ZT1mdW5jdGlvbihwYyxhdHRfaWQpe3ZhciBzZWxmPXRoaXMucHRyO2lmKHBjJiZ0eXBlb2YgcGM9PT0ib2JqZWN0IilwYz1wYy5wdHI7aWYoYXR0X2lkJiZ0eXBlb2YgYXR0X2lkPT09Im9iamVjdCIpYXR0X2lkPWF0dF9pZC5wdHI7cmV0dXJuIHdyYXBQb2ludGVyKF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVfMihzZWxmLHBjLGF0dF9pZCksUG9pbnRBdHRyaWJ1dGUpfTtEZWNvZGVyLnByb3RvdHlwZVsiR2V0QXR0cmlidXRlQnlVbmlxdWVJZCJdPURlY29kZXIucHJvdG90eXBlLkdldEF0dHJpYnV0ZUJ5VW5pcXVlSWQ9ZnVuY3Rpb24ocGMsdW5pcXVlX2lkKXt2YXIgc2VsZj10aGlzLnB0cjtpZihwYyYmdHlwZW9mIHBjPT09Im9iamVjdCIpcGM9cGMucHRyO2lmKHVuaXF1ZV9pZCYmdHlwZW9mIHVuaXF1ZV9pZD09PSJvYmplY3QiKXVuaXF1ZV9pZD11bmlxdWVfaWQucHRyO3JldHVybiB3cmFwUG9pbnRlcihfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlQnlVbmlxdWVJZF8yKHNlbGYscGMsdW5pcXVlX2lkKSxQb2ludEF0dHJpYnV0ZSl9O0RlY29kZXIucHJvdG90eXBlWyJHZXRNZXRhZGF0YSJdPURlY29kZXIucHJvdG90eXBlLkdldE1ldGFkYXRhPWZ1bmN0aW9uKHBjKXt2YXIgc2VsZj10aGlzLnB0cjtpZihwYyYmdHlwZW9mIHBjPT09Im9iamVjdCIpcGM9cGMucHRyO3JldHVybiB3cmFwUG9pbnRlcihfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0TWV0YWRhdGFfMShzZWxmLHBjKSxNZXRhZGF0YSl9O0RlY29kZXIucHJvdG90eXBlWyJHZXRBdHRyaWJ1dGVNZXRhZGF0YSJdPURlY29kZXIucHJvdG90eXBlLkdldEF0dHJpYnV0ZU1ldGFkYXRhPWZ1bmN0aW9uKHBjLGF0dF9pZCl7dmFyIHNlbGY9dGhpcy5wdHI7aWYocGMmJnR5cGVvZiBwYz09PSJvYmplY3QiKXBjPXBjLnB0cjtpZihhdHRfaWQmJnR5cGVvZiBhdHRfaWQ9PT0ib2JqZWN0IilhdHRfaWQ9YXR0X2lkLnB0cjtyZXR1cm4gd3JhcFBvaW50ZXIoX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZU1ldGFkYXRhXzIoc2VsZixwYyxhdHRfaWQpLE1ldGFkYXRhKX07RGVjb2Rlci5wcm90b3R5cGVbIkdldEZhY2VGcm9tTWVzaCJdPURlY29kZXIucHJvdG90eXBlLkdldEZhY2VGcm9tTWVzaD1mdW5jdGlvbihtLGZhY2VfaWQsb3V0X3ZhbHVlcyl7dmFyIHNlbGY9dGhpcy5wdHI7aWYobSYmdHlwZW9mIG09PT0ib2JqZWN0IiltPW0ucHRyO2lmKGZhY2VfaWQmJnR5cGVvZiBmYWNlX2lkPT09Im9iamVjdCIpZmFjZV9pZD1mYWNlX2lkLnB0cjtpZihvdXRfdmFsdWVzJiZ0eXBlb2Ygb3V0X3ZhbHVlcz09PSJvYmplY3QiKW91dF92YWx1ZXM9b3V0X3ZhbHVlcy5wdHI7cmV0dXJuISFfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0RmFjZUZyb21NZXNoXzMoc2VsZixtLGZhY2VfaWQsb3V0X3ZhbHVlcyl9O0RlY29kZXIucHJvdG90eXBlWyJHZXRUcmlhbmdsZVN0cmlwc0Zyb21NZXNoIl09RGVjb2Rlci5wcm90b3R5cGUuR2V0VHJpYW5nbGVTdHJpcHNGcm9tTWVzaD1mdW5jdGlvbihtLHN0cmlwX3ZhbHVlcyl7dmFyIHNlbGY9dGhpcy5wdHI7aWYobSYmdHlwZW9mIG09PT0ib2JqZWN0IiltPW0ucHRyO2lmKHN0cmlwX3ZhbHVlcyYmdHlwZW9mIHN0cmlwX3ZhbHVlcz09PSJvYmplY3QiKXN0cmlwX3ZhbHVlcz1zdHJpcF92YWx1ZXMucHRyO3JldHVybiBfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0VHJpYW5nbGVTdHJpcHNGcm9tTWVzaF8yKHNlbGYsbSxzdHJpcF92YWx1ZXMpfTtEZWNvZGVyLnByb3RvdHlwZVsiR2V0VHJpYW5nbGVzVUludDE2QXJyYXkiXT1EZWNvZGVyLnByb3RvdHlwZS5HZXRUcmlhbmdsZXNVSW50MTZBcnJheT1mdW5jdGlvbihtLG91dF9zaXplLG91dF92YWx1ZXMpe3ZhciBzZWxmPXRoaXMucHRyO2lmKG0mJnR5cGVvZiBtPT09Im9iamVjdCIpbT1tLnB0cjtpZihvdXRfc2l6ZSYmdHlwZW9mIG91dF9zaXplPT09Im9iamVjdCIpb3V0X3NpemU9b3V0X3NpemUucHRyO2lmKG91dF92YWx1ZXMmJnR5cGVvZiBvdXRfdmFsdWVzPT09Im9iamVjdCIpb3V0X3ZhbHVlcz1vdXRfdmFsdWVzLnB0cjtyZXR1cm4hIV9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRUcmlhbmdsZXNVSW50MTZBcnJheV8zKHNlbGYsbSxvdXRfc2l6ZSxvdXRfdmFsdWVzKX07RGVjb2Rlci5wcm90b3R5cGVbIkdldFRyaWFuZ2xlc1VJbnQzMkFycmF5Il09RGVjb2Rlci5wcm90b3R5cGUuR2V0VHJpYW5nbGVzVUludDMyQXJyYXk9ZnVuY3Rpb24obSxvdXRfc2l6ZSxvdXRfdmFsdWVzKXt2YXIgc2VsZj10aGlzLnB0cjtpZihtJiZ0eXBlb2YgbT09PSJvYmplY3QiKW09bS5wdHI7aWYob3V0X3NpemUmJnR5cGVvZiBvdXRfc2l6ZT09PSJvYmplY3QiKW91dF9zaXplPW91dF9zaXplLnB0cjtpZihvdXRfdmFsdWVzJiZ0eXBlb2Ygb3V0X3ZhbHVlcz09PSJvYmplY3QiKW91dF92YWx1ZXM9b3V0X3ZhbHVlcy5wdHI7cmV0dXJuISFfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0VHJpYW5nbGVzVUludDMyQXJyYXlfMyhzZWxmLG0sb3V0X3NpemUsb3V0X3ZhbHVlcyl9O0RlY29kZXIucHJvdG90eXBlWyJHZXRBdHRyaWJ1dGVGbG9hdCJdPURlY29kZXIucHJvdG90eXBlLkdldEF0dHJpYnV0ZUZsb2F0PWZ1bmN0aW9uKHBhLGF0dF9pbmRleCxvdXRfdmFsdWVzKXt2YXIgc2VsZj10aGlzLnB0cjtpZihwYSYmdHlwZW9mIHBhPT09Im9iamVjdCIpcGE9cGEucHRyO2lmKGF0dF9pbmRleCYmdHlwZW9mIGF0dF9pbmRleD09PSJvYmplY3QiKWF0dF9pbmRleD1hdHRfaW5kZXgucHRyO2lmKG91dF92YWx1ZXMmJnR5cGVvZiBvdXRfdmFsdWVzPT09Im9iamVjdCIpb3V0X3ZhbHVlcz1vdXRfdmFsdWVzLnB0cjtyZXR1cm4hIV9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVGbG9hdF8zKHNlbGYscGEsYXR0X2luZGV4LG91dF92YWx1ZXMpfTtEZWNvZGVyLnByb3RvdHlwZVsiR2V0QXR0cmlidXRlRmxvYXRGb3JBbGxQb2ludHMiXT1EZWNvZGVyLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVGbG9hdEZvckFsbFBvaW50cz1mdW5jdGlvbihwYyxwYSxvdXRfdmFsdWVzKXt2YXIgc2VsZj10aGlzLnB0cjtpZihwYyYmdHlwZW9mIHBjPT09Im9iamVjdCIpcGM9cGMucHRyO2lmKHBhJiZ0eXBlb2YgcGE9PT0ib2JqZWN0IilwYT1wYS5wdHI7aWYob3V0X3ZhbHVlcyYmdHlwZW9mIG91dF92YWx1ZXM9PT0ib2JqZWN0IilvdXRfdmFsdWVzPW91dF92YWx1ZXMucHRyO3JldHVybiEhX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUZsb2F0Rm9yQWxsUG9pbnRzXzMoc2VsZixwYyxwYSxvdXRfdmFsdWVzKX07RGVjb2Rlci5wcm90b3R5cGVbIkdldEF0dHJpYnV0ZUludEZvckFsbFBvaW50cyJdPURlY29kZXIucHJvdG90eXBlLkdldEF0dHJpYnV0ZUludEZvckFsbFBvaW50cz1mdW5jdGlvbihwYyxwYSxvdXRfdmFsdWVzKXt2YXIgc2VsZj10aGlzLnB0cjtpZihwYyYmdHlwZW9mIHBjPT09Im9iamVjdCIpcGM9cGMucHRyO2lmKHBhJiZ0eXBlb2YgcGE9PT0ib2JqZWN0IilwYT1wYS5wdHI7aWYob3V0X3ZhbHVlcyYmdHlwZW9mIG91dF92YWx1ZXM9PT0ib2JqZWN0IilvdXRfdmFsdWVzPW91dF92YWx1ZXMucHRyO3JldHVybiEhX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUludEZvckFsbFBvaW50c18zKHNlbGYscGMscGEsb3V0X3ZhbHVlcyl9O0RlY29kZXIucHJvdG90eXBlWyJHZXRBdHRyaWJ1dGVJbnQ4Rm9yQWxsUG9pbnRzIl09RGVjb2Rlci5wcm90b3R5cGUuR2V0QXR0cmlidXRlSW50OEZvckFsbFBvaW50cz1mdW5jdGlvbihwYyxwYSxvdXRfdmFsdWVzKXt2YXIgc2VsZj10aGlzLnB0cjtpZihwYyYmdHlwZW9mIHBjPT09Im9iamVjdCIpcGM9cGMucHRyO2lmKHBhJiZ0eXBlb2YgcGE9PT0ib2JqZWN0IilwYT1wYS5wdHI7aWYob3V0X3ZhbHVlcyYmdHlwZW9mIG91dF92YWx1ZXM9PT0ib2JqZWN0IilvdXRfdmFsdWVzPW91dF92YWx1ZXMucHRyO3JldHVybiEhX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUludDhGb3JBbGxQb2ludHNfMyhzZWxmLHBjLHBhLG91dF92YWx1ZXMpfTtEZWNvZGVyLnByb3RvdHlwZVsiR2V0QXR0cmlidXRlVUludDhGb3JBbGxQb2ludHMiXT1EZWNvZGVyLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVVSW50OEZvckFsbFBvaW50cz1mdW5jdGlvbihwYyxwYSxvdXRfdmFsdWVzKXt2YXIgc2VsZj10aGlzLnB0cjtpZihwYyYmdHlwZW9mIHBjPT09Im9iamVjdCIpcGM9cGMucHRyO2lmKHBhJiZ0eXBlb2YgcGE9PT0ib2JqZWN0IilwYT1wYS5wdHI7aWYob3V0X3ZhbHVlcyYmdHlwZW9mIG91dF92YWx1ZXM9PT0ib2JqZWN0IilvdXRfdmFsdWVzPW91dF92YWx1ZXMucHRyO3JldHVybiEhX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZVVJbnQ4Rm9yQWxsUG9pbnRzXzMoc2VsZixwYyxwYSxvdXRfdmFsdWVzKX07RGVjb2Rlci5wcm90b3R5cGVbIkdldEF0dHJpYnV0ZUludDE2Rm9yQWxsUG9pbnRzIl09RGVjb2Rlci5wcm90b3R5cGUuR2V0QXR0cmlidXRlSW50MTZGb3JBbGxQb2ludHM9ZnVuY3Rpb24ocGMscGEsb3V0X3ZhbHVlcyl7dmFyIHNlbGY9dGhpcy5wdHI7aWYocGMmJnR5cGVvZiBwYz09PSJvYmplY3QiKXBjPXBjLnB0cjtpZihwYSYmdHlwZW9mIHBhPT09Im9iamVjdCIpcGE9cGEucHRyO2lmKG91dF92YWx1ZXMmJnR5cGVvZiBvdXRfdmFsdWVzPT09Im9iamVjdCIpb3V0X3ZhbHVlcz1vdXRfdmFsdWVzLnB0cjtyZXR1cm4hIV9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVJbnQxNkZvckFsbFBvaW50c18zKHNlbGYscGMscGEsb3V0X3ZhbHVlcyl9O0RlY29kZXIucHJvdG90eXBlWyJHZXRBdHRyaWJ1dGVVSW50MTZGb3JBbGxQb2ludHMiXT1EZWNvZGVyLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVVSW50MTZGb3JBbGxQb2ludHM9ZnVuY3Rpb24ocGMscGEsb3V0X3ZhbHVlcyl7dmFyIHNlbGY9dGhpcy5wdHI7aWYocGMmJnR5cGVvZiBwYz09PSJvYmplY3QiKXBjPXBjLnB0cjtpZihwYSYmdHlwZW9mIHBhPT09Im9iamVjdCIpcGE9cGEucHRyO2lmKG91dF92YWx1ZXMmJnR5cGVvZiBvdXRfdmFsdWVzPT09Im9iamVjdCIpb3V0X3ZhbHVlcz1vdXRfdmFsdWVzLnB0cjtyZXR1cm4hIV9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVVSW50MTZGb3JBbGxQb2ludHNfMyhzZWxmLHBjLHBhLG91dF92YWx1ZXMpfTtEZWNvZGVyLnByb3RvdHlwZVsiR2V0QXR0cmlidXRlSW50MzJGb3JBbGxQb2ludHMiXT1EZWNvZGVyLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVJbnQzMkZvckFsbFBvaW50cz1mdW5jdGlvbihwYyxwYSxvdXRfdmFsdWVzKXt2YXIgc2VsZj10aGlzLnB0cjtpZihwYyYmdHlwZW9mIHBjPT09Im9iamVjdCIpcGM9cGMucHRyO2lmKHBhJiZ0eXBlb2YgcGE9PT0ib2JqZWN0IilwYT1wYS5wdHI7aWYob3V0X3ZhbHVlcyYmdHlwZW9mIG91dF92YWx1ZXM9PT0ib2JqZWN0IilvdXRfdmFsdWVzPW91dF92YWx1ZXMucHRyO3JldHVybiEhX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUludDMyRm9yQWxsUG9pbnRzXzMoc2VsZixwYyxwYSxvdXRfdmFsdWVzKX07RGVjb2Rlci5wcm90b3R5cGVbIkdldEF0dHJpYnV0ZVVJbnQzMkZvckFsbFBvaW50cyJdPURlY29kZXIucHJvdG90eXBlLkdldEF0dHJpYnV0ZVVJbnQzMkZvckFsbFBvaW50cz1mdW5jdGlvbihwYyxwYSxvdXRfdmFsdWVzKXt2YXIgc2VsZj10aGlzLnB0cjtpZihwYyYmdHlwZW9mIHBjPT09Im9iamVjdCIpcGM9cGMucHRyO2lmKHBhJiZ0eXBlb2YgcGE9PT0ib2JqZWN0IilwYT1wYS5wdHI7aWYob3V0X3ZhbHVlcyYmdHlwZW9mIG91dF92YWx1ZXM9PT0ib2JqZWN0IilvdXRfdmFsdWVzPW91dF92YWx1ZXMucHRyO3JldHVybiEhX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZVVJbnQzMkZvckFsbFBvaW50c18zKHNlbGYscGMscGEsb3V0X3ZhbHVlcyl9O0RlY29kZXIucHJvdG90eXBlWyJHZXRBdHRyaWJ1dGVEYXRhQXJyYXlGb3JBbGxQb2ludHMiXT1EZWNvZGVyLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVEYXRhQXJyYXlGb3JBbGxQb2ludHM9ZnVuY3Rpb24ocGMscGEsZGF0YV90eXBlLG91dF9zaXplLG91dF92YWx1ZXMpe3ZhciBzZWxmPXRoaXMucHRyO2lmKHBjJiZ0eXBlb2YgcGM9PT0ib2JqZWN0IilwYz1wYy5wdHI7aWYocGEmJnR5cGVvZiBwYT09PSJvYmplY3QiKXBhPXBhLnB0cjtpZihkYXRhX3R5cGUmJnR5cGVvZiBkYXRhX3R5cGU9PT0ib2JqZWN0IilkYXRhX3R5cGU9ZGF0YV90eXBlLnB0cjtpZihvdXRfc2l6ZSYmdHlwZW9mIG91dF9zaXplPT09Im9iamVjdCIpb3V0X3NpemU9b3V0X3NpemUucHRyO2lmKG91dF92YWx1ZXMmJnR5cGVvZiBvdXRfdmFsdWVzPT09Im9iamVjdCIpb3V0X3ZhbHVlcz1vdXRfdmFsdWVzLnB0cjtyZXR1cm4hIV9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVEYXRhQXJyYXlGb3JBbGxQb2ludHNfNShzZWxmLHBjLHBhLGRhdGFfdHlwZSxvdXRfc2l6ZSxvdXRfdmFsdWVzKX07RGVjb2Rlci5wcm90b3R5cGVbIlNraXBBdHRyaWJ1dGVUcmFuc2Zvcm0iXT1EZWNvZGVyLnByb3RvdHlwZS5Ta2lwQXR0cmlidXRlVHJhbnNmb3JtPWZ1bmN0aW9uKGF0dF90eXBlKXt2YXIgc2VsZj10aGlzLnB0cjtpZihhdHRfdHlwZSYmdHlwZW9mIGF0dF90eXBlPT09Im9iamVjdCIpYXR0X3R5cGU9YXR0X3R5cGUucHRyO19lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9Ta2lwQXR0cmlidXRlVHJhbnNmb3JtXzEoc2VsZixhdHRfdHlwZSl9O0RlY29kZXIucHJvdG90eXBlWyJHZXRFbmNvZGVkR2VvbWV0cnlUeXBlX0RlcHJlY2F0ZWQiXT1EZWNvZGVyLnByb3RvdHlwZS5HZXRFbmNvZGVkR2VvbWV0cnlUeXBlX0RlcHJlY2F0ZWQ9ZnVuY3Rpb24oaW5fYnVmZmVyKXt2YXIgc2VsZj10aGlzLnB0cjtpZihpbl9idWZmZXImJnR5cGVvZiBpbl9idWZmZXI9PT0ib2JqZWN0Iilpbl9idWZmZXI9aW5fYnVmZmVyLnB0cjtyZXR1cm4gX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEVuY29kZWRHZW9tZXRyeVR5cGVfRGVwcmVjYXRlZF8xKHNlbGYsaW5fYnVmZmVyKX07RGVjb2Rlci5wcm90b3R5cGVbIkRlY29kZUJ1ZmZlclRvUG9pbnRDbG91ZCJdPURlY29kZXIucHJvdG90eXBlLkRlY29kZUJ1ZmZlclRvUG9pbnRDbG91ZD1mdW5jdGlvbihpbl9idWZmZXIsb3V0X3BvaW50X2Nsb3VkKXt2YXIgc2VsZj10aGlzLnB0cjtpZihpbl9idWZmZXImJnR5cGVvZiBpbl9idWZmZXI9PT0ib2JqZWN0Iilpbl9idWZmZXI9aW5fYnVmZmVyLnB0cjtpZihvdXRfcG9pbnRfY2xvdWQmJnR5cGVvZiBvdXRfcG9pbnRfY2xvdWQ9PT0ib2JqZWN0IilvdXRfcG9pbnRfY2xvdWQ9b3V0X3BvaW50X2Nsb3VkLnB0cjtyZXR1cm4gd3JhcFBvaW50ZXIoX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0RlY29kZUJ1ZmZlclRvUG9pbnRDbG91ZF8yKHNlbGYsaW5fYnVmZmVyLG91dF9wb2ludF9jbG91ZCksU3RhdHVzKX07RGVjb2Rlci5wcm90b3R5cGVbIkRlY29kZUJ1ZmZlclRvTWVzaCJdPURlY29kZXIucHJvdG90eXBlLkRlY29kZUJ1ZmZlclRvTWVzaD1mdW5jdGlvbihpbl9idWZmZXIsb3V0X21lc2gpe3ZhciBzZWxmPXRoaXMucHRyO2lmKGluX2J1ZmZlciYmdHlwZW9mIGluX2J1ZmZlcj09PSJvYmplY3QiKWluX2J1ZmZlcj1pbl9idWZmZXIucHRyO2lmKG91dF9tZXNoJiZ0eXBlb2Ygb3V0X21lc2g9PT0ib2JqZWN0IilvdXRfbWVzaD1vdXRfbWVzaC5wdHI7cmV0dXJuIHdyYXBQb2ludGVyKF9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVCdWZmZXJUb01lc2hfMihzZWxmLGluX2J1ZmZlcixvdXRfbWVzaCksU3RhdHVzKX07RGVjb2Rlci5wcm90b3R5cGVbIl9fZGVzdHJveV9fIl09RGVjb2Rlci5wcm90b3R5cGUuX19kZXN0cm95X189ZnVuY3Rpb24oKXt2YXIgc2VsZj10aGlzLnB0cjtfZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfX19kZXN0cm95X19fMChzZWxmKX07KGZ1bmN0aW9uKCl7ZnVuY3Rpb24gc2V0dXBFbnVtcygpe01vZHVsZVsiQVRUUklCVVRFX0lOVkFMSURfVFJBTlNGT1JNIl09X2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19BdHRyaWJ1dGVUcmFuc2Zvcm1UeXBlX0FUVFJJQlVURV9JTlZBTElEX1RSQU5TRk9STSgpO01vZHVsZVsiQVRUUklCVVRFX05PX1RSQU5TRk9STSJdPV9lbXNjcmlwdGVuX2VudW1fZHJhY29fQXR0cmlidXRlVHJhbnNmb3JtVHlwZV9BVFRSSUJVVEVfTk9fVFJBTlNGT1JNKCk7TW9kdWxlWyJBVFRSSUJVVEVfUVVBTlRJWkFUSU9OX1RSQU5TRk9STSJdPV9lbXNjcmlwdGVuX2VudW1fZHJhY29fQXR0cmlidXRlVHJhbnNmb3JtVHlwZV9BVFRSSUJVVEVfUVVBTlRJWkFUSU9OX1RSQU5TRk9STSgpO01vZHVsZVsiQVRUUklCVVRFX09DVEFIRURST05fVFJBTlNGT1JNIl09X2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19BdHRyaWJ1dGVUcmFuc2Zvcm1UeXBlX0FUVFJJQlVURV9PQ1RBSEVEUk9OX1RSQU5TRk9STSgpO01vZHVsZVsiSU5WQUxJRCJdPV9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9JTlZBTElEKCk7TW9kdWxlWyJQT1NJVElPTiJdPV9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9QT1NJVElPTigpO01vZHVsZVsiTk9STUFMIl09X2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19HZW9tZXRyeUF0dHJpYnV0ZV9UeXBlX05PUk1BTCgpO01vZHVsZVsiQ09MT1IiXT1fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0dlb21ldHJ5QXR0cmlidXRlX1R5cGVfQ09MT1IoKTtNb2R1bGVbIlRFWF9DT09SRCJdPV9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9URVhfQ09PUkQoKTtNb2R1bGVbIkdFTkVSSUMiXT1fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0dlb21ldHJ5QXR0cmlidXRlX1R5cGVfR0VORVJJQygpO01vZHVsZVsiSU5WQUxJRF9HRU9NRVRSWV9UWVBFIl09X2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19FbmNvZGVkR2VvbWV0cnlUeXBlX0lOVkFMSURfR0VPTUVUUllfVFlQRSgpO01vZHVsZVsiUE9JTlRfQ0xPVUQiXT1fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0VuY29kZWRHZW9tZXRyeVR5cGVfUE9JTlRfQ0xPVUQoKTtNb2R1bGVbIlRSSUFOR1VMQVJfTUVTSCJdPV9lbXNjcmlwdGVuX2VudW1fZHJhY29fRW5jb2RlZEdlb21ldHJ5VHlwZV9UUklBTkdVTEFSX01FU0goKTtNb2R1bGVbIkRUX0lOVkFMSUQiXT1fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX0lOVkFMSUQoKTtNb2R1bGVbIkRUX0lOVDgiXT1fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX0lOVDgoKTtNb2R1bGVbIkRUX1VJTlQ4Il09X2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9VSU5UOCgpO01vZHVsZVsiRFRfSU5UMTYiXT1fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX0lOVDE2KCk7TW9kdWxlWyJEVF9VSU5UMTYiXT1fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX1VJTlQxNigpO01vZHVsZVsiRFRfSU5UMzIiXT1fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX0lOVDMyKCk7TW9kdWxlWyJEVF9VSU5UMzIiXT1fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX1VJTlQzMigpO01vZHVsZVsiRFRfSU5UNjQiXT1fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX0lOVDY0KCk7TW9kdWxlWyJEVF9VSU5UNjQiXT1fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX1VJTlQ2NCgpO01vZHVsZVsiRFRfRkxPQVQzMiJdPV9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfRkxPQVQzMigpO01vZHVsZVsiRFRfRkxPQVQ2NCJdPV9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfRkxPQVQ2NCgpO01vZHVsZVsiRFRfQk9PTCJdPV9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfQk9PTCgpO01vZHVsZVsiRFRfVFlQRVNfQ09VTlQiXT1fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX1RZUEVTX0NPVU5UKCk7TW9kdWxlWyJPSyJdPV9lbXNjcmlwdGVuX2VudW1fZHJhY29fU3RhdHVzQ29kZV9PSygpO01vZHVsZVsiRFJBQ09fRVJST1IiXT1fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfRFJBQ09fRVJST1IoKTtNb2R1bGVbIklPX0VSUk9SIl09X2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19TdGF0dXNDb2RlX0lPX0VSUk9SKCk7TW9kdWxlWyJJTlZBTElEX1BBUkFNRVRFUiJdPV9lbXNjcmlwdGVuX2VudW1fZHJhY29fU3RhdHVzQ29kZV9JTlZBTElEX1BBUkFNRVRFUigpO01vZHVsZVsiVU5TVVBQT1JURURfVkVSU0lPTiJdPV9lbXNjcmlwdGVuX2VudW1fZHJhY29fU3RhdHVzQ29kZV9VTlNVUFBPUlRFRF9WRVJTSU9OKCk7TW9kdWxlWyJVTktOT1dOX1ZFUlNJT04iXT1fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfVU5LTk9XTl9WRVJTSU9OKCl9aWYocnVudGltZUluaXRpYWxpemVkKXNldHVwRW51bXMoKTtlbHNlIGFkZE9uSW5pdChzZXR1cEVudW1zKX0pKCk7aWYodHlwZW9mIE1vZHVsZVsib25Nb2R1bGVQYXJzZWQiXT09PSJmdW5jdGlvbiIpe01vZHVsZVsib25Nb2R1bGVQYXJzZWQiXSgpfU1vZHVsZVsiRGVjb2RlciJdLnByb3RvdHlwZS5HZXRFbmNvZGVkR2VvbWV0cnlUeXBlPWZ1bmN0aW9uKGFycmF5KXtpZihhcnJheS5fX2NsYXNzX18mJmFycmF5Ll9fY2xhc3NfXz09PU1vZHVsZS5EZWNvZGVyQnVmZmVyKXtyZXR1cm4gTW9kdWxlLkRlY29kZXIucHJvdG90eXBlLkdldEVuY29kZWRHZW9tZXRyeVR5cGVfRGVwcmVjYXRlZChhcnJheSl9aWYoYXJyYXkuYnl0ZUxlbmd0aDw4KXJldHVybiBNb2R1bGUuSU5WQUxJRF9HRU9NRVRSWV9UWVBFO3N3aXRjaChhcnJheVs3XSl7Y2FzZSAwOnJldHVybiBNb2R1bGUuUE9JTlRfQ0xPVUQ7Y2FzZSAxOnJldHVybiBNb2R1bGUuVFJJQU5HVUxBUl9NRVNIO2RlZmF1bHQ6cmV0dXJuIE1vZHVsZS5JTlZBTElEX0dFT01FVFJZX1RZUEV9fTsNCg0KDQogIHJldHVybiBEcmFjb0RlY29kZXJNb2R1bGUucmVhZHkNCn0NCik7DQp9KSgpOw0KaWYgKHR5cGVvZiBleHBvcnRzID09PSAnb2JqZWN0JyAmJiB0eXBlb2YgbW9kdWxlID09PSAnb2JqZWN0JykNCiAgbW9kdWxlLmV4cG9ydHMgPSBEcmFjb0RlY29kZXJNb2R1bGU7DQplbHNlIGlmICh0eXBlb2YgZGVmaW5lID09PSAnZnVuY3Rpb24nICYmIGRlZmluZVsnYW1kJ10pDQogIGRlZmluZShbXSwgZnVuY3Rpb24oKSB7IHJldHVybiBEcmFjb0RlY29kZXJNb2R1bGU7IH0pOw0KZWxzZSBpZiAodHlwZW9mIGV4cG9ydHMgPT09ICdvYmplY3QnKQ0KICBleHBvcnRzWyJEcmFjb0RlY29kZXJNb2R1bGUiXSA9IERyYWNvRGVjb2Rlck1vZHVsZTs=", Gge = "data:application/wasm;base64,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AiAFQQJ0EAYhBwsgByAEQQJ0aiIEIQMgAUECdCIBQQRrIglBAnZBAWpBB3EiCwRAIAQhAwNAIAMgAigCADYCACADQQRqIQMgCEEBaiIIIAtHDQALCyABIARqIQEgCUEcTwRAA0AgAyACKAIANgIAIAMgAigCADYCBCADIAIoAgA2AgggAyACKAIANgIMIAMgAigCADYCECADIAIoAgA2AhQgAyACKAIANgIYIAMgAigCADYCHCADQSBqIgMgAUcNAAsLIApBAEoEQCAHIAYgChAHGgsgACAHIAVBAnRqNgIIIAAgATYCBCAAIAc2AgAgBgRAIAYQBQsPCxAJAAtBrgoQCgALegECfyAABEAgACgCWCEBIABBADYCWCABBEAgASgCCCICBEAgASACNgIMIAIQBQsgARAFCyAAKAJEIgEEQCAAIAE2AkggARAFCyAAKAJAIQEgAEEANgJAIAEEQCABKAIAIgIEQCABIAI2AgQgAhAFCyABEAULIAAQBQsLfwEDfyAAIQECQCAAQQNxBEADQCABLQAARQ0CIAFBAWoiAUEDcQ0ACwsDQCABIgJBBGohASACKAIAIgNBf3MgA0GBgoQIa3FBgIGChHhxRQ0ACyADQf8BcUUEQCACIABrDwsDQCACLQABIQMgAkEBaiIBIQIgAw0ACwsgASAAawsIAEGhChAKAAtEACABBEAgACABKAIAEBQgACABKAIEEBQgASgCHCIABEAgASAANgIgIAAQBQsgASwAG0EASARAIAEoAhAQBQsgARAFCwtbACABBEAgACABKAIAEBUgACABKAIEEBUgASgCHCEAIAFBADYCHCAABEAgAEEMaiAAKAIQEBUgACAAKAIEEBQgABAFCyABLAAbQQBIBEAgASgCEBAFCyABEAULC2sCAn8BfgJAIABBBUsNACACKQMQIgUgAikDCFkNACACKAIAIAWnai0AACEDIAIgBUIBfDcDECABIANBgAFxBH8gAEEBaiABIAIQFkUNASADQf8AcSABKAIAQQd0cgUgAws2AgBBASEECyAEC9gCAQN/IwBBEGsiBSQAIABBADYCBAJAAkAgAUUNAAJAIAEgACgCCCIDQQV0IgRNBEAgACABNgIEDAELIAVBADYCCCAFQgA3AwAgAUEASA0CIAUgBEH+////A00EfyABQR9qQWBxIgQgA0EGdCID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aiEEIAJBAWohAiADQQFqIgMgBUcNAAsLIABBB0kEQA8LA0AgBCACLQAAOgAAIAQgAi0AAToAASAEIAItAAI6AAIgBCACLQADOgADIAQgAi0ABDoABCAEIAItAAU6AAUgBCACLQAGOgAGIAQgAi0ABzoAByAEQQhqIQQgAkEIaiICIAZHDQALC10BAX8jAEEQayIDJAACQCACQQpNBEAgACACOgALIAAgASACECogA0EAOgAPIAAgAmogAy0ADzoAAAwBCyAAQQogAkEKayAALQALIgAgACACIAEQhQELIANBEGokAAt2AQJ/IwBBEGsiBCQAAkAgAiAAKAIIQf////8HcSIDSQRAIAAoAgAhAyAAIAI2AgQgAyABIAIQKiAEQQA6AA8gAiADaiAELQAPOgAADAELIAAgA0EBayACIANrQQFqIAAoAgQiACAAIAIgARCFAQsgBEEQaiQAC+EBAQN/IABB3NAANgIAIAAoAkQiAQRAIAAgATYCSCABEAULIAAoAjgiAQRAIAAgATYCPCABEAULIAAoAiwiAQRAIAAgATYCMCABEAULIAAoAiAiAQRAIAAgATYCJCABEAULIAAoAhQiAQRAIAAgATYCGCABEAULIAAoAggiAgRAIAIgACgCDCIDRgR/IAIFA0AgA0EEayIDKAIAIQEgA0EANgIAIAEEQCABEBELIAIgA0cNAAsgACgCCAshASAAIAI2AgwgARAFCyAAKAIEIQEgAEEANgIEIAEEQCABEGgLIAALBABBAQsEAEF/CxQAIAAEQCAAIAAoAgAoAgQRAgALC8UMBAR/AX4BfQF8AkACQCADRQ0AAkACQAJAAkACQAJAAkACQAJAAkACQAJAIAAoAhxBAWsOCwABAgMEBQYHCAkKDAtBASEHIAIgACwAGCIEIAIgBEgbQQBKBEAgACgCACIEKAIAIAApAzAgACkDKCABrX58p2ohASAEKAIEIQYDQCABIAZPDQ4gAyAFQQJ0aiABLAAANgIAIAFBAWohASAFQQFqIgUgAiAALAAYIgQgAiAESBtIDQALCyACIARKDQoMCwtBASEHIAIgACwAGCIEIAIgBEgbQQBKBEAgACgCACIEKAIAIAApAzAgACkDKCABrX58p2ohASAEKAIEIQYDQCABIAZPDQ0gAyAFQQJ0aiABLQAANgIAIAFBAWohASAFQQFqIgUgAiAALAAYIgQgAiAESBtIDQALCyACIARKDQkMCgtBASEHIAIgACwAGCIEIAIgBEgbQQBKBEAgACgCACIEKAIAIAApAzAgACkDKCABrX58p2ohASAEKAIEIQYDQCABIAZPDQwgAyAFQQJ0aiABLgEANgIAIAFBAmohASAFQQFqIgUgAiAALAAYIgQgAiAESBtIDQALCyACIARKDQgMCQtBASEHIAIgACwAGCIEIAIgBEgbQQBKBEAgACgCACIEKAIAIAApAzAgACkDKCABrX58p2ohASAEKAIEIQYDQCABIAZPDQsgAyAFQQJ0aiABLwEANgIAIAFBAmohASAFQQFqIgUgAiAALAAYIgQgAiAESBtIDQALCyACIARKDQcMCAtBASEHIAIgACwAGCIEIAIgBEgbQQBKBEAgACgCACIEKAIAIAApAzAgACkDKCABrX58p2ohASAEKAIEIQYDQCABIAZPDQogAyAFQQJ0aiABKAIANgIAIAFBBGohASAFQQFqIgUgAiAALAAYIgQgAiAESBtIDQALCyACIARKDQYMBwtBASEHIAIgACwAGCIEIAIgBEgbQQBKBEAgACgCACIEKAIAIAApAzAgACkDKCABrX58p2ohASAEKAIEIQYDQCABIAZPDQkgAyAFQQJ0aiABKAIANgIAIAFBBGohASAFQQFqIgUgAiAALAAYIgQgAiAESBtIDQALCyACIARKDQUMBgtBASEHIAIgACwAGCIEIAIgBEgbQQBKBEAgACgCACIEKAIAIAApAzAgACkDKCABrX58p2ohASAEKAIEIQYDQCABIAZPDQggASkDACIIQv////8PVg0IIAMgBUECdGogCD4CACABQQhqIQEgBUEBaiIFIAIgACwAGCIEIAIgBEgbSA0ACwsgAiAESg0EDAULQQEhByACIAAsABgiBCACIARIG0EASgRAIAAoAgAiBCgCACAAKQMwIAApAyggAa1+fKdqIQEgBCgCBCEGA0AgASAGTw0HIAEpAwAiCEL/////D1YNByADIAVBAnRqIAg+AgAgAUEIaiEBIAVBAWoiBSACIAAsABgiBCACIARIG0gNAAsLIAIgBEoNAwwEC0EBIQcgAiAALAAYIgQgAiAESBtBAEoEQCAAKAIAIgQoAgAgACkDMCAAKQMoIAGtfnynaiEBIAQoAgQhBgNAIAEgBk8N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AAAAAOBBYwRAIAqqDAELQYCAgIB4CzYCACABQQhqIQEgBUEBaiIFIAIgACwAGCIEIAIgBEgbSA0ACwsgAiAESg0BDAILQQEhByACIAAsABgiBCACIARIG0EASgRAIAAoAgAiBCgCACAAKQMwIAApAyggAa1+fKdqIQEgBCgCBCEGA0AgASAGTw0EIAMgBUECdGogAS0AADYCACABQQFqIQEgBUEBaiIFIAIgACwAGCIEIAIgBEgbSA0ACwsgAiAETA0BCyADIARBAnRqQQAgAiAEa0ECdBAIGgsgBw8LQQAL7QwEBH8BfgF9AXwCQAJAIANFDQACQAJAAkACQAJAAkACQAJAAkACQAJAAkAgACgCHEEBaw4LAAECAwQFBgcICQoMCyACIAAsABgiBCACIARIG0EASgRAIAAoAgAiBCgCACAAKQMwIAApAyggAa1+fKdqIQEgBCgCBCEGA0AgASAGTw0OIAEsAAAiBEEASA0NIAMgBUEBdGogBEH/AXE7AQAgAUEBaiEBIAVBAWoiBSACIAAsABgiBCACIARIG0gNAAsLQQEhByACIARKDQoMCwtBASEHIAIgACwAGCIEIAIgBEgbQQBKBEAgACgCACIEKAIAIAApAzAgACkDKCABrX58p2ohASAEKAIEIQYDQCABIAZPDQ0gAyAFQQF0aiABLQAAOwEAIAFBAWohASAFQQFqIgUgAiAALAAYIgQgAiAESBtIDQALCyACIARKDQkMCgsgAiAALAAYIgQgAiAESBtBAEoEQCAAKAIAIgQoAgAgACkDMCAAKQMoIAGtfnynaiEBIAQoAgQhBgNAIAEgBk8NDCABLgEAIgRBAEgNCyADIAVBAXRqIAQ7AQAgAUECaiEBIAVBAWoiBSACIAAsABgiBCACIARIG0gNAAsLQQEhByACIARKDQgMCQtBASEHIAIgACwAGCIEIAIgBEgbQQBKBEAgACgCACIEKAIAIAApAzAgACkDKCABrX58p2ohASAEKAIEIQYDQCABIAZPDQsgAyAFQQF0aiABLwEAOwEAIAFBAmohASAFQQFqIgUgAiAALAAYIgQgAiAESBtIDQALCyACIARKDQcMCAtBASEHIAIgACwAGCIEIAIgBEgbQQBKBEAgACgCACIEKAIAIAApAzAgACkDKCABrX58p2ohASAEKAIEIQYDQCABIAZPDQog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IAQgBUECdGooAgBBAnRqKAIAIQQgACgCBCIBLQBURQRAIAEoAkQgBEECdGooAgAhBAsgA0IANwMYIANCADcDECADQgA3AwggASAEIAEsABggA0EIahAYIAogAykDCCADKQM4Igp9IgkgAykDKCADKQNAIgt9Igx+fSADKQMQIAt9IgsgAykDICAKfSIPfnwhCiANIAkgAykDMCADKQNIIg19IhB+fCADKQMYIA19IgkgD359IQ0gDiALIBB+fSAJIAx+fCEOIANB0ABqEG8gAygCWCIBQX9HDQALIA4gDkI/hyIJfCAJhSEJQv///////////wAhCyANIA1CP4ciDHwgDIUiDEL///////////8AhSEPAkAgCgJ+IAAoAhxFBEAgCSAPVg0CQoCAgIBwIAkgDHwiCSAKIApCP4ciC3wgC4UiC3xCIIYgCSALQv///////////wCFVhsiCUKBgICAgICAgCBTDQIgCUI9iAwBCwJAIAkgD1YNACAJIAx8IgkgCiAKQj+HIgx8IAyFIgxC////////////AIVWDQAgCSAMfCILQoGAgIACVA0CCyALQh2ICyIJfyEKIA0gCX8hDSAOIAl/IQ4LIAIgCj4CCCACIA0+AgQgAiAOPgIAIANB4ABqJAAPCxANAAsQDQALEA0ACyUBAX8CQAJAAkAgAQ4CAQACC0EBIQELIAAgATYCHEEBIQILIAILmQsCBX8JfiMAQeAAayIEJAAgACgCECEDIARBAToAXCAEIAE2AlggBCABNgJUIAQgAzYCUAJAAkAgAygCHCABQQJ0aigCACIDIAAoAhQiBSgCBCAFKAIAIgVrQQJ1SQRAIAAoAgggBSADQQJ0aigCAEECdGooAgAhAyAAKAIEIgUtAFRFBEAgBSgCRCADQQJ0aigCACEDCyAEQgA3A0ggBEFAa0IANwMAIARCADcDOCAFIAMgBSwAGCAEQThqEBggAUEBaiIDQQNwIQUgASADTQRAIAMgAUECayAFGyEFQX9BAiABQQNwGyABaiEHA0AgBSEDAn8gByAAKAIcRQ0AGiABQQFqIgMgAUECayADQQNwGyEDIAFBAWsgAUEDcA0AGiABQQJqCyEGIAAoAhAoAhwgA0ECdGooAgAiASAAKAIUIgMoAgQgAygCACIDa0ECdU8NAyAAKAIIIAMgAUECdGooAgBBAnRq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IAAQBQsLyQoCAn8CfiMAQTBrIgMkACADQZbPACkBADcBJiADQZDPACkDADcDICADQYjPACkDADcDGCADQYDPACkDADcDEAJAAkAgASkDCCABKQMQIgVCBXxTBEAgA0EQahASIgFBcE8NAgJAAkAgAUELTwRAIAFBEGpBcHEiBBAGIQIgAyAEQYCAgIB4cjYCCCADIAI2AgAgAyABNgIEDAELIAMgAToACyADIQIgAUUNAQsgAiADQRBqIAEQBxoLIAEgAmpBADoAACAAQX42AgAgAEEEaiEAIAMsAAtBAE4EQCAAIAMpAwA3AgAgACADKAIINgIIDAILIAAgAygCACIAIAMoAgQQDCAAEAUMAQsgAiABKAIAIAWnaiIEKAAANgAAIAIgBC0ABDoABCABIAEpAxBCBXwiBTcDECACQfIKQQUQDgRAQSAQBiIBQeYMLQAAOgAQIAFB3gwpAAA3AAggAUHWDCkAADcAACABQQA6ABEgAEF/NgIAIABBBGogAUEREAwgARAFDAELIAUgASkDCFkEQCADQRBqEBIiAUFwTw0CAkACQCABQQtPBEAgAUEQakFwcSIEEAYhAiADIARBgICAgHhyNgIIIAMgAjYCACADIAE2AgQMAQsgAyABOgALIAMhAiABRQ0BCyACIANBEGogARAHGgsgASACakEAOgAAIABBfjYCACAAQQRqIQAgAywAC0EATgRAIAAgAykDADcCACAAIAMoAgg2AggMAgsgACADKAIAIgAgAygCBBAMIAAQBQwBCyACIAEoAgAgBadqLQAAOgAFIAEgASkDEEIBfCIFNwMQIAUgASkDCFkEQCADQRBqEBIiAUFwTw0CAkACQCABQQtPBEAgAUEQakFwcSIEEAYhAiADIARBgICAgHhyNgIIIAMgAjYCACADIAE2AgQMAQsgAyABOgALIAMhAiABRQ0BCyACIANBEGogARAHGgsgASACakEAOgAAIABBfjYCACAAQQRqIQAgAywAC0EATgRAIAAgAykDADcCACAAIAMoAgg2AggMAgsgACADKAIAIgAgAygCBBAMIAAQBQwBCyACIAEoAgAgBadqLQAAOgAGIAEgASkDEEIBfCIFNwMQIAUgASkDCFkEQCADQRBqEBIiAUFwTw0CAkACQCABQQtPBEAgAUEQakFwcSIEEAYhAiADIARBgICAgHhyNgIIIAMgAjYCACADIAE2AgQMAQsgAyABOgALIAMhAiABRQ0BCyACIANBEGogARAHGgsgASACakEAOgAAIABBfjYCACAAQQRqIQAgAywAC0EATgRAIAAgAykDADcCACAAIAMoAgg2AggMAgsgACADKAIAIgAgAygCBBAMIAAQBQwBCyACIAEoAgAgBadqLQAAOgAHIAEgASkDEEIBfCIFNwMQIAUgASkDCFkEQCADIANBEGoQTiEBIABBfjYCACAAQQRqIQAgASwAC0EATgRAIAAgAykDADcCACAAIAMoAgg2AggMAgsgACABKAIAIAEoAgQQDCABLAALQQBODQEgASgCABAFDAELIAIgASgCACAFp2otAAA6AAggASABKQMQIgVCAXwiBjcDECABKQMIIAVCA3xTBEAgAyADQRBqEE4hASAAQX42AgAgAEEEaiEAIAEsAAtBAE4EQCAAIAMpAwA3AgAgACADKAIINgIIDAILIAAgASgCACABKAIEEAwgASwAC0EATg0BIAEoAgAQBQwBCyACIAEoAgAgBqdqLwAAOwEKIAEgASkDEEICfDcDECAAQgA3AgggAEIANwIACyADQTBqJAAPCxATAAvvBQEGf0F/IQUCf0F/IAFBf0YNABogAUEBaiICIAFBAmsgAkEDcBshBSABQQFrIAFBA3ANABogAUECagshBgJAAn8CQAJAAkACQCAAKAKoAQ4IAAAFAgUBBQMFC0F/IQEgACgClAEhBEF/IQJBASEDIAAoApwBIgcgBUF/RwR/IAQoAgAgBUECdGooAgAFQX8LQQJ0aiICIAIoAgBBAWo2AgAgByAGQX9HBH8gBCgCACAGQQJ0aigCAAVBfwtBAnRqDAMLQX8hAiAAKAKUASEEQX8hAyABQX9HBEAgBCgCACABQQJ0aigCACEDCyAAKAKcASIBIANBAnRqIgMgAygCAEEBajYCACABIAVBf0cEfyAEKAIAIAVBAnRqKAIABUF/C0ECdGoiAiACKAIAQQFqNgIAQX8hAkECIQMgASAGQX9HBH8gBCgCACAGQQJ0aigCAAVBfwtBAnRqDAILQX8hAiAAKAKUASEEQX8hAyABQX9HBEAgBCgCACABQQJ0aigCACEDCyAAKAKcASIBIANBAnRqIgMgAygCAEEBajYCACABIAVBf0cEfyAEKAIAIAVBAnRqKAIA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CyAAKALEASIBBEAgACABNgLIASABEAULIAAoArgBIgEEQCAAIAE2ArwBIAEQBQsgACgCrAEiAQRAIAAgATYCsAEgARAFCyAAKAKgASIBBEAgACABNgKkASABEAULIAAoApABIgEEQANAIAEoAgAhAiABEAUgAiIBDQALCyAAKAKIASEBIABBADYCiAEgAQRAIAEQBQsgACgCeCIBBEAgARAFCyAAKAJsIgEEQCABEAULIAAoAmAiAQRAIAEQBQsgACgCSCIBBEAgACABNgJMIAEQBQsgACgCPCIBBEAgARAFCyAAKAIwIgEEQCAAIAE2AjQgARAFCyAAKAIkIgEEQCAAIAE2AiggARAFCyAAKAIYIgEEQCAAIAE2AhwgARAFCyAAKAIMIgEEQCAAIAE2AhAgARAFCyAAKAIIIQEgAEEANgIIIAEEQCABEC8LIAALtQQBBH8gAEH4wAA2AgAgACgC8AIhBCAAQQA2AvACIAQEQCAEQQRrIgMoAgAiAQRAIAQgAUEEdGohAgNAIAJBEGsiAiAERw0ACwsgAxAFCyAAKALYASIDBEAgAyAAKALcASICRgR/IAMFA0AgAkEMaygCACIBBEAgAkEIayABNgIAIAEQBQsgAkEcaygCACIBBEAgAkEYayABNgIAIAEQBQsgAkEoaygCACIBBEAgAkEkayABNgIAIAEQBQsgAkGMAWsQPSACQZABayICIANHDQALIAAoAtgBCyEBIAAgAzYC3AEgARAFCyAAKALEASIBBEAgACABNgLIASABEAULIAAoArgBIgEEQCAAIAE2ArwBIAEQBQsgACgCrAEiAQRAIAAgATYCsAEgARAFCyAAKAKgASIBBEAgACABNgKkASABEAULIAAoApABIgIEQANAIAIoAgAhASACEAUgASICDQALCyAAKAKIASEBIABBADYCiAEgAQRAIAEQBQsgACgCeCIBBEAgARAFCyAAKAJsIgEEQCABEAULIAAoAmAiAQRAIAEQBQsgACgCSCIBBEAgACABNgJMIAEQBQsgACgCPCIBBEAgARAFCyAAKAIwIgEEQCAAIAE2AjQgARAFCyAAKAIkIgEEQCAAIAE2AiggARAFCyAAKAIYIgEEQCAAIAE2AhwgARAFCyAAKAIMIgEEQCAAIAE2AhAgARAFCyAAKAIIIQEgAEEANgIIIAEEQCABEC8L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GiADQQEgA0EBSxshBQNAIAEtACQhAwJAAkAgACgCBC8BJCICQQh0IAJBCHZyQf//A3FBgQRNBEAgA0UNAkEAIQQCQCABKAIYIgogASgCICICQQN2IglqIgMgASgCHCIHTwRAIAIhAwwBCyADLQAAIQQgASACQQFqIgM2AiAgA0EDdiEJIAQgAkEHcXZBAXEhBAsgCSAKaiAHSQ0BDAILIANFDQFBACEEIAEoAhggASgCICIDQQN2aiICIAEoAhxPDQEgAi0AACADQQdxdkEBcSEECyABIANBAWo2AiALIAAoAiQgBkEMbGoiAyADLQAIQf4BcSAEQQFxcjoACCAGQQFqIgYgBUcNAAsgAUEAOgAkIAEgASkDECABNQIgQgd8QgOIfDcDEAsgASgCECEKCyAIQRBqJAAgCg8LEAkAC0GuChAKAAv1CgEFfyABIAAoAgQiAiAAKAIAIgVrQZABbSIDSwRAAkAgASADayIDIAAiAigCCCIBIAAoAgQiAGtBkAFtTQRAIAIgAwR/IAAgA0GQAWxqIQEDQCAAQX82AgAgAEEEahCUASAAQgA3AmggAEEBOgBkIABCADcCcCAAQgA3AnggAEIANwKAASAAQgA3AogBIABBkAFqIgAgAUcNAAsgAQUgAAs2AgQMAQsCQAJAAkAgACACKAIAIgVrQZABbSIGIANqIgBB8ricDkkEQCAAIAEgBWtBkAFtIgFBAXQiBSAAIAVLG0HxuJwOIAFBuJyOB0kbIgUEfyAFQfK4nA5PDQIgBUGQAWwQBgVBAAsiBCAGQZABbGoiASADQZABbGohBiABIQADQCAAQX82AgAgAEEEahCUASAAQgA3AmggAEEBOgBkIABCADcCcCAAQgA3AnggAEIANwKAASAAQgA3AogBIABBkAFqIgAgBkcNAAsgBCAFQZABbGohBSACKAIEIgAgAigCACIDRg0CA0AgAUGQAWsiASAAQZABayIAKAIANgIAIAEgACgCBDYCBCABIAAoAgg2AgggASAAKAIMNgIMIABBADYCDCAAQgA3AgQgASAAKAIQNgIQIAEgACgCFDYCFCABIAAoAhg2AhggAEEANgIYIABCADcCECAALQAcIQQgAUEANgIoIAFCADcCICABIAQ6ABwgASAAKAIgNgIgIAEgACgCJDYCJCABIAAoAig2AiggAEEANgIoIABC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IAIpAgghCyACKQIQIQwgAikCACENIABBADYCKCAAQgA3AiAgACAMNwIYIAAgCzcCECAAIA03AgggAigCHCACKAIYIgVrIgEEQCABQQBIDQggACABEAYiAzYCICAAIAMgAUECdUECdGo2AiggACADIAUgARAHIAFqNgIkCyAAIAQpAgA3AiwgACAEKQIINwI0IABCADcCPCAAQeQXNgIAIABCADcCRCAAQgA3AkwgAEIANwJUIABCADcCXCAAQgA3AmQgAEEANgJsIAAMBAtB6AAQBiIAIAU2AgQgAEGsFzYCACACKQIIIQsgAikCECEMIAIpAgAhDSAAQQA2AiggAEIANwIgIAAgDDcCGCAAIAs3AhAgACANNwIIIAIoAhwgAigCGCIFayIBBEAgAUEASA0DIAAgARAGIgM2AiAgACADIAFBAnVBAnRqNgIoIAAgAyAFIAEQByABajYCJAsgACAEKQIANwIsIAQpAgghCyAAQQA2AlQgAEIANwJMIABCADcCPCAAQdwZNgIAIAAgCzcCNCAAIAQpAgA3AlggACAEKQIINwJgIAAMAwsCf0GAARAGIgAgBTYCBCAAQawXNgIAIAIpAgghCyACKQIQIQwgAikCACENIABBADYCKCAAQgA3AiAgACAMNwIYIAAgCzcCECAAIA03AggCQCACKAIcIAIoAhhrIgMEQCADQQBIDQEgACADEAYiATYCICAAIAE2AiQgACABIANBAnVBAnRqNgIoIAAgAigCHCACKAIYIgVrIgNBAEoEfyABIAUgAxAHIANqBSABCzYCJAsgAEH0FjYCACAAIAQpAgA3AiwgACAEKQIINwI0IABBQGtCADcCACAAQbwgNgI8IABByBs2AgAgACAEKQIANwJIIAAgBCkCCDcCUCAAQoCAgPxzNwJoIABCfzcCYCAAQoGAgIBwNwJYIABBtB02AjwgAEIANwJwIABCADcAdSAADAELEAkACyEACyAADAELDAMLIQMMAQsgBCgCLCEEIAYgCTYCDCAGIAQ2AgggBiAANgIUIAYgAEEMajYCEAJ/IAZBCGohBEEAIQACQAJAAkACQAJAAkAgAUEBaw4GAAQEAQIDBAtBPBAGIgAgBTYCBCAAQawXNgIAIAIpAgghCyACKQIQIQwgAikCACENIABBADYCKCAAQgA3AiAgACAMNwIYIAAgCzcCECAAIA03AgggAigCHCACKAIYIgVrIgEEQCABQQBIDQggACABEAYiAzYCICAAIAMgAUECdUECdGo2AiggACADIAUgARAHIAFqNgIkCyAAIAQpAgA3AiwgACAEKQIINwI0IABB2CA2AgAgAAwFC0HwABAGIgAgBTYCBCAAQawXNgIAIAIpAgghCyACKQIQIQwgAikCACENIABBADYCKCAAQgA3AiAgACAMNwIYIAAgCzcCECAAIA03AgggAigCHCACKAIYIgVrIgEEQCABQQBIDQcgACABEAYiAzYCICAAIAMgAUECdUECdGo2AiggACADIAUgARAHIAFqNgIkCyAAIAQpAgA3AiwgACAEKQIINwI0IABCADcCPCAAQYAkNgIAIABCADcCRCAAQgA3AkwgAEIANwJUIABCADcCXCAAQgA3AmQgAEEANgJsIAAMBAtB6AAQBiIAIAU2AgQgAEGsFzYCACACKQIIIQsgAikCECEMIAIpAgAhDSAAQQA2AiggAEIANwIgIAAgDDcCGCAAIAs3AhAgACANNwIIIAIoAhwgAigCGCIFayIBBEAgAUEASA0DIAAgARAGIgM2AiAgACADIAFBAnVBAnRqNgIoIAAgAyAFIAEQByABajYCJAsgACAEKQIANwIsIAQpAgghCyAAQQA2AlQgAEIANwJMIABCADcCPCAAQewlNgIAIAAgCzcCNCAAIAQpAgA3AlggACAEKQIINwJgIAAMAwsCf0GAARAGIgAgBTYCBCAAQawXNgIAIAIpAgghCyACKQIQIQwgAikCACENIABBADYCKCAAQgA3AiAgACAMNwIYIAAgCzcCECAAIA03AggCQCACKAIcIAIoAhhrIgMEQCADQQBIDQEgACADEAYiATYCICAAIAE2AiQgACABIANBAnVBAnRqNgIoIAAgAigCHCACKAIYIgVrIgNBAEoEfyABIAUgAxAHIANqBSABCzYCJAsgAEHIIzYCACAAIAQpAgA3AiwgACAEKQIINwI0IABBQGtCADcCACAAQZQsNgI8IABBzCc2AgAgACAEKQIANwJIIAAgBCkCCDcCUCAAQoCAgPxzNwJoIABCfzcCYCAAQoGAgIBwNwJYIABBqCk2AjwgAEIANwJwIABCADcAdSAADAELEAkACyEACyAADAELDAILIQMLIAZBIGokACADDAELEAkACyIDDQELQSwQBiIDIAo2AgQgA0GsFzYCACACKQIIIQsgAikCECEMIAIpAgAhDSADQQA2AiggA0IANwIgIAMgDDcCGCADIAs3AhAgAyANNwIIIAIoAhwgAigCGCICayIABEAgAEEASA0CIAMgABAGIgE2AiAgAyABIABBAnVBAnRqNgIoIAMgASACIAAQByAAajYCJAsgA0GwLDYCAAsgAwwBCxAJAAshBCAHKAIgIgBFDQAgByAANgIkIAAQBQsgB0EwaiQAIAQLzwECA38DfgJAIAIpAwgiCCACKQMQIgZXDQAgAigCACIEIAanaiwAACEDIAIgBkIBfCIHNwMQAkAgA0F+Rg0AIAcgCFkNASAEIAenaiwAACEEIAIgBkICfDcDECAEQQRrQf8BcUH7AUkNASAAIAMgBCAAKAIAKAIoEQMAIQQgACgCFCEDIAAgBDYCFCADRQ0AIAMgAygCACgCBBECAAsgACgCFCIDBEAgACADIAAoAgAoAhwRAQBFDQELIAAgASACIAAoAgAoAiQRAwAhBQsgBQsdACAAIAEoAgQgASgCAGtBAnUgACgCACgCMBEBAAtEAQF/An9BACAAKAIUIAAoAhAiAmtBAnUgAUwNABpBACACIAFBAnRqKAIAIgFBAEgNABogACgCJCABQQJ0aigCABAxCwvSAQEDfyAAQfwPNgIAIAAoAjwhASAAQQA2AjwgAQRAIAEgASgCACgCBBECAAsgACgCMCIBBEAgACABNgI0IAEQBQsgACgCJCICBEAgAiAAKAIoIgNGBH8gAgUDQCADQQRrIgMoAgAhASADQQA2AgAgAQRAIAEgASgCACgCBBECAAsgAiADRw0ACyAAKAIkCyEBIAAgAjYCKCABEAULIABBjA42AgAgACgCECIBBEAgACABNgIUIAEQBQsgACgCBCIBBEAgACABNgIIIAEQBQsgABAFC9ABAQN/IABB/A82AgAgACgCPCEBIABBADYCPCABBEAgASABKAIAKAIEEQIACyAAKAIwIgEEQCAAIAE2AjQgARAFCyAAKAIkIgIEQCACIAAoAigiA0YEfyACBQNAIANBBGsiAygCACEBIANBADYCACABBEAgASABKAIAKAIEEQIACyACIANHDQALIAAoAiQLIQEgACACNgIoIAEQBQsgAEGMDjYCACAAKAIQIgEEQCAAIAE2AhQgARAFCyAAKAIEIgEEQCAAIAE2AgggARAFCyAAC4oCAEEAIQACQAJAAkACQAJAIAEOBAABAgMEC0EUEAYiAEL/////DzcCDCAAQgA3AgQgAEGcDzYCACAADwtBGBAGIgBC/////w83AgwgAEIANwIEIABBnA82AgAgAEEANgIUIABB9BA2AgAgAA8LQTAQBiIAQv////8PNwIMIABCADcCBCAAQZwPNgIAIABBADYCFCAAQfQQNgIAIABCADcCICAAQX82AhwgAEH0CDYCGCAAQag+NgIAIABCADcCKCAADwtBIBAGIgBC/////w83AgwgAEIANwIEIABBnA82AgAgAEEANgIUIABB9BA2AgAgAEF/NgIcIABBiAg2AhggAEHQLTYCAAsgAAu+CgIQfwF+IwBBEGsiCCQAQQEhCwJAIAAgACgCACgCGBEAACINQQBMBEBBACELDAELIABBMGohDwNAAkACQCAAIAAoAgAoAhwRAAAoAihFDQAgCkECdCIQIAAoAiRqKAIAIgQoAgghASAEEDEiBUUNACAAIAAoAgAoAhwRAAAoAighBCABKAI4IQYgCEEgEAYiCTYCACAIQpiAgICAhICAgH83AgQgCUEAOgAYIAlBmAopAAA3ABAgCUGQCikAADcACCAJQYgKKQAANwAAIARBEGoiAyECAkAgAygCACIHRQ0AA0AgAiAHIAcoAhAgBkgiARshAiAHIAFBAnRqKAIAIgcNAAsgAiADRg0AIAYgAigCEEgNACACKAIYIgdFDQAgAkEUaiEBA0ACQAJAAkAgBygCFCAHLQAbIgMgA0EYdEEYdUEASCIGGyIMQRggDEEYSSICGyIOBEACQCAJIAdBEGoiAygCACADIAYbIgYgDhAOIgNFBEAgDEEYTQ0BDAULIANBAEgNBAsgBiAJIA4QDiIDRQ0BIANBAEgNAiABIQQMBQsgDEEYSw0CCyACDQAgASEEDAMLIAdBBGohBwsgBygCACIHDQALC0EAIQcCQCAEIAgQLSIBIARBBGpGDQAgASgCHCABQRxqIAEsACdBAEgbIQFBACECQQAhBgNAIAEiBEEBaiEBIAQsAAAiA0EgRiADQQlrQQVJcg0ACwJAAkACQCAELAAAIgNBK2sOAwECAAILQQEhBgsgASwAACEDIAEhBAsgA0Ewa0EKSQRAA0AgAkEKbCAELAAA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ThsQBiEFAn9BASABIANrIgNBAEwNABpBACIBIAIpAxAiCiAJQv////8PgyILfCIJIAIpAwhVDQAaIANBAnUiBkEBIAZBAUobIQdBACEDA0ACQCAFIAIoAgAgCqdqIAQQByEIIAIgCTcDECAAKAIIKAJAKAIAIAFqIAggBBAHGiADQQFqIgMgB0YNACABIARqIQEgAikDECIKIAt8IgkgAikDCFcNAQsLIAMgBk4LIQAgBRAFIAALzgEBBX8gASABKAIAKAIUEQAAQQBMBEBBAQ8LAkADQCAAKAIEKAIEIAEgBCABKAIAKAIYEQEAEIsBIgJBf0YNASAAKAIEIQNBACEFAkAgAkEASA0AIAMoAgQiBigCDCAGKAIIa0ECdSACTA0AIAMoAgggAygCFCACQQJ0aigCAEECdGooAgAiAyACIAMoAgAoAiARAQAhBQsgBSICRQ0BIAEgAiABKAIAKAIcEQEARQ0BIAEgASgCACgCFBEAACAEQQFqIgRKDQALQQEPC0EACwQAQQELQAECfwJAIAAoAggiBCwAGEEATA0AIAQgASgCBCABKAIAa0ECdRBhRQ0AIAAgASACIAAoAgAoAiARAwAhAwsgAwsSACAAQX82AgwgACABNgIIQQELGwAgACABKAIIIAUQHARAIAEgAiADIAQQgQELCzgAIAAgASgCCCAFEBwEQCABIAIgAyAEEIEBDwsgACgCCCIAIAEgAiADIAQgBSAAKAIAKAIUEQsAC6cBACAAIAEoAgggBBAcBEACQCABKAIEIAJHDQAgASgCHEEBRg0AIAEgAzYCHAsPCwJAIAAgASgCACAEEBxFDQACQCACIAEoAhBHBEAgASgCFCACRw0BCyADQQFHDQEgAUEBNgIgDwsgASACNgIUIAEgAzYCICABIAEoAihBAWo2AigCQCABKAIkQQFHDQAgASgCGEECRw0AIAFBAToANgsgAUEENgIsCwuIAgAgACABKAIIIAQQHARAAkAgASgCBCACRw0AIAEoAhxBAUYNACABIAM2AhwLDwsCQCAAIAEoAgAgBBAcBEACQCACIAEoAhBHBEAgASgCFCACRw0BCyADQQFHDQIgAUEBNgIgDwsgASADNgIgAkAgASgCLEEERg0AIAFBADsBNCAAKAIIIgAgASACIAJBASAEIAAoAgAoAhQRCwAgAS0ANQRAIAFB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QCACIAE2AgAgAkEEaiEBIAAsABdBAE4EQCABIABBCGpBBHIiAikCADcCACABIAIoAgg2AggMAgsgASAAKAIMIAAoAhAQDCAALAAXQQBODQEgACgCDBAFDAELIAAsABdBAEgEQCAAKAIMEAULIAAtAB8iAUECTwRAQSAQBiIBQdMMLwAAOwAYIAFBywwpAAA3ABAgAUHDDCkAADcACCABQbsMKQAANwAAIAFBADoAGiAAQX82AgggAEEIakEEciIEIAFBGhAMIAAsABchBSACIAAoAgg2AgAgAkEEaiECAkAgBUEATgRAIAIgBCkCADcCACACIAQoAgg2AggMAQsgAiAAKAIMIAAoAhAQDAsgACwAF0EASARAIAAoAgwQBQsgARAFDAELIAJCADcCACACIAE2AhAgAkIANwIICyAAQdAAaiQAIAMoAhghACADLAAXQQBIBEAgAygCDBAFCyADQSBqJAAgAAsMAAJAIAAgARCGAwsLxjsCDX8CfgJ/AkACQAJAAkACQAJAAkACQAJAAkACQCADQQFrDgkAAwEEAgUHBwYHC0EAIQAjAEEQayILJAACQCABKAJQIgggAiwAGCIJbCAERw0AIAItAFQhAAJAIAIoAhxBAUciAQ0AIABB/wFxRQ0AIAUgAigCACgCACACKAIwaiAEEAcaQQEhAAwBC0EAIQMgC0EANgIIIAtCADcDACAJBEAgCUEASA0JIAsgCRAGIgY2AgAgBkEAIAkQCBoLAkACQCAIRQ0AIAFFBEAgCQRAIAlBfHEhDSAJQQNxIQxBACEBIAlBAWtBA0khDgNAIAYgAigCACgCACACKQMwIAIpAygiEyAAQf8BcQR/IAMFIAIoAkQgA0ECdGooAgALrX58p2ogE6cQByEEQQAhCUEAIQBBACEHIA5FBEADQCABIAVqIgogACAEai0AADoAACAKIAQgAEEBcmotAAA6AAEgCiAEIABBAnJqLQAAOgACIAogBCAAQQNyai0AADoAAyAAQQRqIQAgAUEEaiEBIAdBBGoiByANRw0ACwsgDARAA0AgASAFaiAAIARqLQAAOgAAIABBAWohACABQQFqIQEgCUEBaiIJIAxHDQALC0EBIQAgA0EBaiIDIAhGDQQgAi0AVCEADAALAAsgAigCACEEIAIpAzAhEyACKAJEIQUgAikDKCIUpyECQQAhASAI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CyARQRBqJAACQCAbBEAgGCgCACIABEAgGCAANgIEIAAQBQsgGCALKAIANgIAIBggCygCBDYCBCAYIAsoAgg2AgggCygCVCEeDAELIAsoAgAiAEUNACALIAA2AgQgABAFCyALKAJIIgAEQCAAEAULIAsoAjAiAARAIAsgADYCNCAAEAULIAsoAiQiAARAIAsgADYCKCAAEAULIAsoAhgiAARAIAsgADYCHCAAEAULIAsoAhQhACALQQA2AhQgAARAIAAQLwsgC0HgAGokACAeC3EBAn8jAEEQayIAJAAgASgCYCEBIABBADYCCCAAQgA3AwBBDBAGIgQgASACQQxsakEMEAciAkEMaiEFIAMoAgAiAQRAIAMgATYCBCABEAULIAMgAkEMajYCCCADIAU2AgQgAyAENgIAIABBEGokAEEBC2oBAX8Cf0EAIAEoAgQiA0UNABpBACABKAIIIAJBAnRqKAIAKAI8IgFBAEgNABpBACADKAIYIgAgAygCHCICRg0AGgJAA0AgACgCACIDKAIYIAFGDQEgAEEEaiIAIAJHDQALQQAMAQsgAwsLBwAgASgCBAtmAQJ/An9BACABKAIMIAEoAggiAWsiAEUNABogAEECdSIAQQEgAEEBSxshA0EAIQACQANAIAEgAEECdGooAgAiBCgCPCACRg0BIABBAWoiACADRw0AC0EADAELQQAgBCAAQX9GGwsLEAAgASgCCCACQQJ0aigCAAvMAgEDfwJ/IwBBIGsiBCQAAkAgAhASIgZBcEkEQAJAAkAgBkELTwRAIAZBEGpBcHEiBRAGIQAgBCAFQYCAgIB4cjYCGCAEIAA2AhAgBCAGNgIUDAELIAQgBjoAGyAEQRBqIQAgBkUNAQsgACACIAYQBxoLIAAgBmpBADoAACADEBIiBUFwTw0BAkACQCAFQQtPBEAgBUEQakFwcSICEAYhACAEIAJBgICAgHhyNgIIIAQgADYCACAEIAU2AgQMAQsgBCAFOgALIAQhACAFRQ0BCyAAIAMgBRAHGgsgACAFakEAOgAAAn9BfyABKAIEIgBFDQAaQX8gACAEQRBqIAQQjgEiAEUNABogASAAKAIYEIoBCyEAIAQsAAtBAEgEQCAEKAIAEAULIAQsABtBAEgEQCAEKAIQEAULIARBIGokACAADAILEBMACxATAAsL9wEBA38CfyMA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KAIEEQIACyAAQdDOADYCACAAKAIUIgEEQCAAIAE2AhggARAFCyAAKAIIIgIEQCACIAAoAgwiA0YEfyACBQNAIANBBGsiAygCACEBIANBADYCACABBEAgASABKAIAKAIEEQIACyACIANHDQALIAAoAggLIQEgACACNgIMIAEQBQsgABAFC6gBAQN/IABBjMAANgIAIAAoAjAhASAAQQA2AjAgAQRAIAEgASgCACgCBBECAAsgAEHQzgA2AgAgACgCFCIBBEAgACABNgIYIAEQBQsgACgCCCICBEAgAiAAKAIMIgNGBH8gAgUDQCADQQRrIgMoAgAhASADQQA2AgAgAQRAIAEgASgCACgCBBECAAsgAiADRw0ACyAAKAIICyEBIAAgAjYCDCABEAULIAALFAAgACgCMCIAIAAoAgAoAhwRAAALFAAgACgCMCIAIAAoAgAoAhgRAAALvwcCAn8BfiAAKAIgIgEpAwggASkDECIDVQR/IAEoAgAgA6dqLQAAIQIgASADQgF8NwMQIAAoAjAhASAAQQA2AjAgAQRAIAEgASgCACgCBBECAAsCQAJAAn8CQAJAIAIOAwADAQMLQYADEAYiAUH4wAA2AgAgAUEEakEAQdAAEAgaIAFCADcDYCABQX82AlwgAUJ/NwJUIAFCADcDaCABQgA3A3AgAUIANwN4IAFCADcDgAEgAUIANwOIASABQgA3A5ABIAFCADcCnAEgAUGAgID8AzYCmAEgAUIANwKkASABQgA3AqwBIAFCADcCtAEgAUIANwK8ASABQgA3AsQBIAFCADcCzAEgAUL/////DzcC1AEgAUIANwLcASABQegBaiICQQA7ASYgAkEANgIAIAJCADcDCCACQgA3AxAgAkIANwMYIAJCADcAHSABQZACaiICQQA7ASYgAkEANgIAIAJCADcDCCACQgA3AxAgAkIANwMYIAJCADcAHSABQbgCaiICQgA3AgAgAkIANwAFIAFByAJqIgJBADsBJiACQQA2AgAgAkIANwMIIAJCADcDECACQgA3AxggAkIANwAdIAFBADYC+AIgAUIANwPwAiABDAELQbgDEAYiAUGswQA2AgAgAUEEakEAQdAAEAgaIAFCADcDYCABQX82AlwgAUJ/NwJUIAFCADcDaCABQgA3A3AgAUIANwN4IAFCADcDgAEgAUIANwOIASABQgA3A5ABIAFCADcCnAEgAUGAgID8AzYCmAEgAUIANwKkASABQgA3AqwBIAFCADcCtAEgAUIANwK8ASABQgA3AsQBIAFCADcCzAEgAUL/////DzcC1AEgAUIANwLcASABQegBaiICQQA7ASYgAkEANgIAIAJCADcDCCACQgA3AxAgAkIANwMYIAJCADcAHSABQZACaiICQQA7ASYgAkEANgIAIAJCADcDCCACQgA3AxAgAkIANwMYIAJCADcAHSABQbgCaiICQgA3AgAgAkIANwAFIAFByAJqIgJBADsBJiACQQA2AgAgAkIANwMIIAJCADcDECACQgA3AxggAkIANwAdIAFCADcDiAMgAUIANwOAAyABQgA3A/gCIAFCADcD8AIgAUIANwOgAyABQoKAgIDwADcDmAMgAUJ/NwOQAyABQgA3A6gDIAFCADcDsAMgAQshASAAKAIwIQIgACABNgIwIAJFDQEgAiACKAIAKAIEEQIACyAAKAIwIgENAEEADwsgASAAIAEoAgAoAggRAQAFQQALCxYAIAAoAjAiACABIAAoAgAoAhQRAQALfwEDfyAAQdDOADYCACAAKAIUIgEEQCAAIAE2AhggARAFCyAAKAIIIgIEQCACIAAoAgwiA0YEfyACBQNAIANBBGsiAygCACEBIANBADYCACABBEAgASABKAIAKAIEEQIACyACIANHDQALIAAoAggLIQEgACACNgIMIAEQBQsgAAsbACAAKAIsRQRAQQAPCyAAIAAoAgAoAjARAAAL4wEBBX8gAUEBNgIAIAFBCGohBCABKAIMIAEoAggiAmsiA0F7TQR/IAQgA0EEaq0QTSAEKAIABSACCyADaiAAKAIENgAAIAAoAggiAyAAKAIMRwRAQQAhAgNAIAMgAkECdGohBiABKAIMIAEoAggiA2siBUF7TQR/IAQgBUEEaq0QTSAEKAIABSADCyAFaiAGKAIANgAAIAJBAWoiAiAAKAIMIAAoAggiA2tBAnVJDQALCyABKAIMIAEoAggiAmsiAUF7TQR/IAQgAUEEaq0QTSAEKAIABSACCyABaiAAKAIUNgAAC7ITARR/IwBBEGsiCiQAIAogATYCDCAKQSAQBiIBNgIAIApCmICAgICEgICAfzcCBCABQQA6ABggAUGYCikAADcAECABQZAKKQAANwAIIAFBiAopAAA3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RmFpbGVkIHRvIGRlY29kZSBnZW9tZXRyeSBkYXRhLgBQdXJlIHZpcnR1YWwgZnVuY3Rpb24gY2FsbGVkIQBONWRyYWNvMThBdHRyaWJ1dGVUcmFuc2Zvcm1FAAAYLAAA3gYAAAAAAACIBwAAGQAAABoAAAAbAAAAHAAAAB0AAAAeAAAAHwAAACAAAAAhAAAAIgAAACMAAAAkAAAATjVkcmFjbzE3QXR0cmlidXRlc0RlY29kZXJFAE41ZHJhY28yNkF0dHJpYnV0ZXNEZWNvZGVySW50ZXJmYWNlRQAAAAAYLAAAWAcAAEAsAAA8BwAAgAcAAAAAAADoBwAAJQAAACYAAAAnAAAAKAAAACkAAAAqAAAAKwAAACwAAAAtAAAATjVkcmFjbzI2U2VxdWVudGlhbEF0dHJpYnV0ZURlY29kZXJFAAAAABgsAADABwAA/////wAAAABgCAAALgAAAC8AAAAbAAAAMAAAADEAAAAeAAAAHwAAACAAAAAyAAAAMwAAADQAAAA1AAAANgAAAE41ZHJhY28zN1NlcXVlbnRpYWxBdHRyaWJ1dGVEZWNvZGVyc0NvbnRyb2xsZXJFAEAsAAAwCAAAiAcAAAAAAADUCAAANwAAADgAAAA5AAAAKAAAACkAAAAqAAAAOgAAACwAAAA7AAAAPAAAAD0AAAA+AAAAPwAAAE41ZHJhY28zM1NlcXVlbnRpYWxJbnRlZ2VyQXR0cmlidXRlRGVjb2RlckUAQCwAAKgIAADoBwAAAAAAAGALAABAAAAAQQAAAEIAAABDAAAARAAAAEUAAABGAAAARwAAAEgAAABJAAAASgAAAEsAAABONWRyYWNvNDBNZXNoUHJlZGljdGlvblNjaGVtZVBhcmFsbGVsb2dyYW1EZWNvZGVySWlOU18zN1ByZWRpY3Rpb25TY2hlbWVXcmFwRGVjb2RpbmdUcmFuc2Zvcm1JaWlFRU5TXzI0TWVzaFByZWRpY3Rpb25TY2hlbWVEYXRhSU5TXzI0TWVzaEF0dHJpYnV0ZUNvcm5lclRhYmxlRUVFRUUATjVkcmFjbzI3TWVzaFByZWRpY3Rpb25TY2hlbWVEZWNvZGVySWlOU18zN1ByZWRpY3Rpb25TY2hlbWVXcmFwRGVjb2RpbmdUcmFuc2Zvcm1JaWlFRU5TXzI0TWVzaFByZWRpY3Rpb25TY2hlbWVEYXRhSU5TXzI0TWVzaEF0dHJpYnV0ZUNvcm5lclRhYmxlRUVFRUUATjVkcmFjbzIzUHJlZGljdGlvblNjaGVtZURlY29kZXJJaU5TXzM3UHJlZGljdGlvblNjaGVtZVdyYXBEZWNvZGluZ1RyYW5zZm9ybUlpaUVFRUUATjVkcmFjbzM3UHJlZGljdGlvblNjaGVtZVR5cGVkRGVjb2RlckludGVyZmFjZUlpaUVFAE41ZHJhY28zMlByZWRpY3Rpb25TY2hlbWVEZWNvZGVySW50ZXJmYWNlRQBONWRyYWNvMjVQcmVkaWN0aW9uU2NoZW1lSW50ZXJmYWNlRQAYLAAABAsAAEAsAADZCgAAKAsAAEAsAAClCgAAMAsAAEAsAABRCgAAPAsAAEAsAAC7CQAASAsAAEAsAAAYCQAAVAsAAAAAAABUCwAAQAAAAEwAAAAiAAAAQwAAACIAAABFAAAARgAAAEcAAABIAAAASQAAAEoAAAAiAAAAAAAAAEgLAABAAAAATQAAACIAAABDAAAAIgAAAEUAAABGAAAARwAAAEgAAABJAAAASgAAACIAAAAAAAAAyAwAAE4AAABPAAAAUAAAAEMAAABRAAAARQAAAEYAAABHAAAASAAAAEkAAABSAAAAUwAAAE41ZHJhY281Nk1lc2hQcmVkaWN0aW9uU2NoZW1lQ29uc3RyYWluZWRNdWx0aVBhcmFsbGVsb2dyYW1EZWNvZGVySWlOU18zN1ByZWRpY3Rpb25TY2hlbWVXcmFwRGVjb2RpbmdUcmFuc2Zvcm1JaWlFRU5TXzI0TWVzaFByZWRpY3Rpb25TY2hlbWVEYXRhSU5TXzI0TWVzaEF0dHJpYnV0ZUNvcm5lclRhYmxlRUVFRUUAAEAsAAAUDAAAVAsAAAAAAAC0DQAAVAAAAFUAAABWAAAAQwAAAFcAAABYAAAAWQAAAFoAAABIAAAASQAAAFsAAABcAAAATjVkcmFjbzQ0TWVzaFByZWRpY3Rpb25TY2hlbWVUZXhDb29yZHNQb3J0YWJsZURlY29kZXJJaU5TXzM3UHJlZGljdGlvblNjaGVtZVdyYXBEZWNvZGluZ1RyYW5zZm9ybUlpaUVFTlNfMjRNZXNoUHJlZGljdGlvblNjaGVtZURhdGFJTlNfMjRNZXNoQXR0cmlidXRlQ29ybmVyVGFibGVFRUVFRQAAQCwAAAwNAABUCwAAAAAAAKAOAABdAAAAXgAAAF8AAABDAAAAYAAAAGEAAABiAAAAYwAAAEgAAABJAAAAZAAAAGUAAABONWRyYWNvNDJNZXNoUHJlZGljdGlvblNjaGVtZUdlb21ldHJpY05vcm1hbERlY29kZXJJaU5TXzM3UHJlZGljdGlvblNjaGVtZVdyYXBEZWNvZGluZ1RyYW5zZm9ybUlpaUVFTlNfMjRNZXNoUHJlZGljdGlvblNjaGVtZURhdGFJTlNfMjRNZXNoQXR0cmlidXRlQ29ybmVyVGFibGVFRUVFRQAAAABALAAA+A0AAFQLAAAAAAAAKBAAAGYAAABnAAAAaAAAAGkAAABqAAAATjVkcmFjbzQ4TWVzaFByZWRpY3Rpb25TY2hlbWVHZW9tZXRyaWNOb3JtYWxQcmVkaWN0b3JBcmVhSWlOU18zN1ByZWRpY3Rpb25TY2hlbWVXcmFwRGVjb2RpbmdUcmFuc2Zvcm1JaWlFRU5TXzI0TWVzaFByZWRpY3Rpb25TY2hlbWVEYXRhSU5TXzI0TWVzaEF0dHJpYnV0ZUNvcm5lclRhYmxlRUVFRUUATjVkcmFjbzQ4TWVzaFByZWRpY3Rpb25TY2hlbWVHZW9tZXRyaWNOb3JtYWxQcmVkaWN0b3JCYXNlSWlOU18zN1ByZWRpY3Rpb25TY2hlbWVXcmFwRGVjb2RpbmdUcmFuc2Zvcm1JaWlFRU5TXzI0TWVzaFByZWRpY3Rpb25TY2hlbWVEYXRhSU5TXzI0TWVzaEF0dHJpYnV0ZUNvcm5lclRhYmxlRUVFRUUAAAAYLAAAcw8AAEAsAADIDgAAIBAAAAAAAAAgEAAAZgAAAGsAAAAiAAAAaQAAACIAAAAAAAAAtBEAAEAAAABsAAAAbQAAAEMAAABuAAAARQAAAEYAAABHAAAASAAAAEkAAABKAAAAbwAAAE41ZHJhY280ME1lc2hQcmVkaWN0aW9uU2NoZW1lUGFyYWxsZWxvZ3JhbURlY29kZXJJaU5TXzM3UHJlZGljdGlvblNjaGVtZVdyYXBEZWNvZGluZ1RyYW5zZm9ybUlpaUVFTlNfMjRNZXNoUHJlZGljdGlvblNjaGVtZURhdGFJTlNfMTFDb3JuZXJUYWJsZUVFRUVFAE41ZHJhY28yN01lc2hQcmVkaWN0aW9uU2NoZW1lRGVjb2RlcklpTlNfMzdQcmVkaWN0aW9uU2NoZW1lV3JhcERlY29kaW5nVHJhbnNmb3JtSWlpRUVOU18yNE1lc2hQcmVkaWN0aW9uU2NoZW1lRGF0YUlOU18xMUNvcm5lclRhYmxlRUVFRUUAAEAsAAAeEQAASAsAAEAsAACIEAAAqBEAAAAAAACoEQAAQAAAAHAAAAAiAAAAQwAAACIAAABFAAAARgAAAEcAAABIAAAASQAAAEoAAAAiAAAAAAAAANgSAABxAAAAcgAAAHMAAABDAAAAdAAAAEUAAABGAAAARwAAAEgAAABJAAAAdQAAAHYAAABONWRyYWNvNTZNZXNoUHJlZGljdGlvblNjaGVtZUNvbnN0cmFpbmVkTXVsdGlQYXJhbGxlbG9ncmFtRGVjb2RlcklpTlNfMzdQcmVkaWN0aW9uU2NoZW1lV3JhcERlY29kaW5nVHJhbnNmb3JtSWlpRUVOU18yNE1lc2hQcmVkaWN0aW9uU2NoZW1lRGF0YUlOU18xMUNvcm5lclRhYmxlRUVFRUUAAABALAAAMBIAAKgRAAAAAAAAuBMAAHcAAAB4AAAAeQAAAEMAAAB6AAAAewAAAHwAAAB9AAAASAAAAEkAAAB+AAAAfwAAAE41ZHJhY280NE1lc2hQcmVkaWN0aW9uU2NoZW1lVGV4Q29vcmRzUG9ydGFibGVEZWNvZGVySWlOU18zN1ByZWRpY3Rpb25TY2hlbWVXcmFwRGVjb2RpbmdUcmFuc2Zvcm1JaWlFRU5TXzI0TWVzaFByZWRpY3Rpb25TY2hlbWVEYXRhSU5TXzExQ29ybmVyVGFibGVFRUVFRQAAAEAsAAAcEwAAqBEAAAAAAACUFAAAgAAAAIEAAACCAAAAQwAAAIMAAACEAAAAhQAAAIYAAABIAAAASQAAAIcAAACIAAAATjVkcmFjbzQyTWVzaFByZWRpY3Rpb25TY2hlbWVHZW9tZXRyaWNOb3JtYWxEZWNvZGVySWlOU18zN1ByZWRpY3Rpb25TY2hlbWVXcmFwRGVjb2RpbmdUcmFuc2Zvcm1JaWlFRU5TXzI0TWVzaFByZWRpY3Rpb25TY2hlbWVEYXRhSU5TXzExQ29ybmVyVGFibGVFRUVFRQBALAAA/BMAAKgRAAAAAAAAABYAAIkAAACKAAAAiwAAAIwAAACNAAAATjVkcmFjbzQ4TWVzaFByZWRpY3Rpb25TY2hlbWVHZW9tZXRyaWNOb3JtYWxQcmVkaWN0b3JBcmVhSWlOU18zN1ByZWRpY3Rpb25TY2hlbWVXcmFwRGVjb2RpbmdUcmFuc2Zvcm1JaWlFRU5TXzI0TWVzaFByZWRpY3Rpb25TY2hlbWVEYXRhSU5TXzExQ29ybmVyVGFibGVFRUVFRQBONWRyYWNvNDhNZXNoUHJlZGljdGlvblNjaGVtZUdlb21ldHJpY05vcm1hbFByZWRpY3RvckJhc2VJaU5TXzM3UHJlZGljdGlvblNjaGVtZVdyYXBEZWNvZGluZ1RyYW5zZm9ybUlpaUVFTlNfMjRNZXNoUHJlZGljdGlvblNjaGVtZURhdGFJTlNfMTFDb3JuZXJUYWJsZUVFRUVFABgsAABaFQAAQCwAALwUAAD4FQAAAAAAAPgVAACJAAAAjgAAACIAAACMAAAAIgAAAAAAAAC8FgAAQAAAAI8AAACQAAAAQwAAAJEAAABFAAAARgAAAEcAAABIAAAASQAAAEoAAACSAAAATjVkcmFjbzI4UHJlZGljdGlvblNjaGVtZURlbHRhRGVjb2RlcklpTlNfMzdQcmVkaWN0aW9uU2NoZW1lV3JhcERlY29kaW5nVHJhbnNmb3JtSWlpRUVFRQAAAABALAAAYBYAAEgLAAAAAAAAMBcAAJMAAACUAAAAlQAAACgAAAApAAAAlgAAADoAAAAsAAAAOwAAAJcAAACYAAAAmQAAAJoAAABONWRyYWNvMzJTZXF1ZW50aWFsTm9ybWFsQXR0cmlidXRlRGVjb2RlckUAAEAsAAAEFwAA1AgAAAAAAABkGQAAmwAAAJwAAACdAAAAngAAAJ8AAACgAAAAoQAAAKIAAACjAAAApAAAAKUAAACmAAAATjVkcmFjbzQyTWVzaFByZWRpY3Rpb25TY2hlbWVHZW9tZXRyaWNOb3JtYWxEZWNvZGVySWlOU182MlByZWRpY3Rpb25TY2hlbWVOb3JtYWxPY3RhaGVkcm9uQ2Fub25pY2FsaXplZERlY29kaW5nVHJhbnNmb3JtSWlFRU5TXzI0TWVzaFByZWRpY3Rpb25TY2hlbWVEYXRhSU5TXzI0TWVzaEF0dHJpYnV0ZUNvcm5lclRhYmxlRUVFRUUATjVkcmFjbzI3TWVzaFByZWRpY3Rpb25TY2hlbWVEZWNvZGVySWlOU182MlByZWRpY3Rpb25TY2hlbWVOb3JtYWxPY3RhaGVkcm9uQ2Fub25pY2FsaXplZERlY29kaW5nVHJhbnNmb3JtSWlFRU5TXzI0TWVzaFByZWRpY3Rpb25TY2hlbWVEYXRhSU5TXzI0TWVzaEF0dHJpYnV0ZUNvcm5lclRhYmxlRUVFRUUATjVkcmFjbzIzUHJlZGljdGlvblNjaGVtZURlY29kZXJJaU5TXzYyUHJlZGljdGlvblNjaGVtZU5vcm1hbE9jdGFoZWRyb25DYW5vbmljYWxpemVkRGVjb2RpbmdUcmFuc2Zvcm1JaUVFRUUAAEAsAADfGAAAPAsAAEAsAAAxGAAATBkAAEAsAAB0FwAAWBkAAAAAAAAcGwAApwAAAKgAAACpAAAAqgAAAKsAAABONWRyYWNvNDhNZXNoUHJlZGljdGlvblNjaGVtZUdlb21ldHJpY05vcm1hbFByZWRpY3RvckFyZWFJaU5TXzYyUHJlZGljdGlvblNjaGVtZU5vcm1hbE9jdGFoZWRyb25DYW5vbmljYWxpemVkRGVjb2RpbmdUcmFuc2Zvcm1JaUVFTlNfMjRNZXNoUHJlZGljdGlvblNjaGVtZURhdGFJTlNfMjRNZXNoQXR0cmlidXRlQ29ybmVyVGFibGVFRUVFRQBONWRyYWNvNDhNZXNoUHJlZGljdGlvblNjaGVtZUdlb21ldHJpY05vcm1hbFByZWRpY3RvckJhc2VJaU5TXzYyUHJlZGljdGlvblNjaGVtZU5vcm1hbE9jdGFoZWRyb25DYW5vbmljYWxpemVkRGVjb2RpbmdUcmFuc2Zvcm1JaUVFTlNfMjRNZXNoUHJlZGljdGlvblNjaGVtZURhdGFJTlNfMjRNZXNoQXR0cmlidXRlQ29ybmVyVGFibGVFRUVFRQAAABgsAABPGgAAQCwAAIwZAAAUGwAAAAAAAMAcAACsAAAArQAAAK4AAACeAAAArwAAALAAAACxAAAAsgAAAKMAAACkAAAAswAAALQAAABONWRyYWNvNDJNZXNoUHJlZGljdGlvblNjaGVtZUdlb21ldHJpY05vcm1hbERlY29kZXJJaU5TXzYyUHJlZGljdGlvblNjaGVtZU5vcm1hbE9jdGFoZWRyb25DYW5vbmljYWxpemVkRGVjb2RpbmdUcmFuc2Zvcm1JaUVFTlNfMjRNZXNoUHJlZGljdGlvblNjaGVtZURhdGFJTlNfMTFDb3JuZXJUYWJsZUVFRUVFAE41ZHJhY28yN01lc2hQcmVkaWN0aW9uU2NoZW1lRGVjb2RlcklpTlNfNjJQcmVkaWN0aW9uU2NoZW1lTm9ybWFsT2N0YWhlZHJvbkNhbm9uaWNhbGl6ZWREZWNvZGluZ1RyYW5zZm9ybUlpRUVOU18yNE1lc2hQcmVkaWN0aW9uU2NoZW1lRGF0YUlOU18xMUNvcm5lclRhYmxlRUVFRUUAAAAAQCwAABAcAABMGQAAQCwAAGAbAAC0HAAAAAAAAFweAAC1AAAAtgAAALcAAAC4AAAAuQAAAE41ZHJhY280OE1lc2hQcmVkaWN0aW9uU2NoZW1lR2VvbWV0cmljTm9ybWFsUHJlZGljdG9yQXJlYUlpTlNfNjJQcmVkaWN0aW9uU2NoZW1lTm9ybWFsT2N0YWhlZHJvbkNhbm9uaWNhbGl6ZWREZWNvZGluZ1RyYW5zZm9ybUlpRUVOU18yNE1lc2hQcmVkaWN0aW9uU2NoZW1lRGF0YUlOU18xMUNvcm5lclRhYmxlRUVFRUUATjVkcmFjbzQ4TWVzaFByZWRpY3Rpb25TY2hlbWVHZW9tZXRyaWNOb3JtYWxQcmVkaWN0b3JCYXNlSWlOU182MlByZWRpY3Rpb25TY2hlbWVOb3JtYWxPY3RhaGVkcm9uQ2Fub25pY2FsaXplZERlY29kaW5nVHJhbnNmb3JtSWlFRU5TXzI0TWVzaFByZWRpY3Rpb25TY2hlbWVEYXRhSU5TXzExQ29ybmVyVGFibGVFRUVFRQAYLAAAnh0AAEAsAADoHAAAVB4AAAAAAAAUHwAAugAAALsAAAC8AAAAngAAAL0AAAC+AAAAvwAAAMAAAACjAAAApAAAAMEAAADCAAAATjVkcmFjbzI4UHJlZGljdGlvblNjaGVtZURlbHRhRGVjb2RlcklpTlNfNjJQcmVkaWN0aW9uU2NoZW1lTm9ybWFsT2N0YWhlZHJvbkNhbm9uaWNhbGl6ZWREZWNvZGluZ1RyYW5zZm9ybUlpRUVFRQAAAABALAAAoB4AAEwZAAAAAAAAmB8AAMMAAADEAAAAxQAAACgAAAApAAAAxgAAADoAAAAsAAAAOwAAAMcAAAA9AAAAPgAAAMgAAADJAAAAygAAAE41ZHJhY28zOFNlcXVlbnRpYWxRdWFudGl6YXRpb25BdHRyaWJ1dGVEZWNvZGVyRQAAAABALAAAZB8AANQIAAAAAAAA+B8AAMsAAADMAAAAzQAAAM4AAAAiAAAAzwAAANAAAADRAAAA0gAAANMAAADUAAAA1QAAACIAAABONWRyYWNvMTFNZXNoRGVjb2RlckUAAABALAAA4B8AALwnAAAAAAAAZCAAANYAAADXAAAAzQAAANgAAADZAAAAzwAAANAAAADRAAAA2gAAANsAAADcAAAA3QAAAN4AAABONWRyYWNvMjJNZXNoRWRnZWJyZWFrZXJEZWNvZGVyRQAAAABALAAAQCAAAPgfAAAAAAAAWCEAAN8AAADgAAAA4QAAAOIAAADjAAAA5AAAAOUAAADmAAAA5wAAAOgAAAD/////AAAAALghAADpAAAA6gAAAOsAAADsAAAA7QAAAO4AAADvAAAA8AAAAPEAAADyAAAATjVkcmFjbzI2TWVzaEVkZ2VicmVha2VyRGVjb2RlckltcGxJTlNfMzFNZXNoRWRnZWJyZWFrZXJUcmF2ZXJzYWxEZWNvZGVyRUVFAE41ZHJhY28zNU1lc2hFZGdlYnJlYWtlckRlY29kZXJJbXBsSW50ZXJmYWNlRQAAABgsAAAgIQAAQCwAANQgAABQIQAATjVkcmFjbzI2TWVzaEVkZ2VicmVha2VyRGVjb2RlckltcGxJTlNfMzhNZXNoRWRnZWJyZWFrZXJUcmF2ZXJzYWxWYWxlbmNlRGVjb2RlckVFRQAAQCwAAGQhAABQIQAAAAAAALQiAADzAAAA9AAAAPUAAABONWRyYWNvMTlEZXB0aEZpcnN0VHJhdmVyc2VySU5TXzI0TWVzaEF0dHJpYnV0ZUNvcm5lclRhYmxlRU5TXzM2TWVzaEF0dHJpYnV0ZUluZGljZXNFbmNvZGluZ09ic2VydmVySVMxX0VFRUUATjVkcmFjbzEzVHJhdmVyc2VyQmFzZUlOU18yNE1lc2hBdHRyaWJ1dGVDb3JuZXJUYWJsZUVOU18zNk1lc2hBdHRyaWJ1dGVJbmRpY2VzRW5jb2RpbmdPYnNlcnZlcklTMV9FRUVFABgsAABFIgAAQCwAANghAACsIgAAAAAAAKwiAAD2AAAA9wAAAPUAQeTFAAu2BwEAAAADAAAABQAAAAcAAAAAAAAAuCMAAPgAAAD5AAAA+gAAAPsAAABONWRyYWNvMjJNZXNoVHJhdmVyc2FsU2VxdWVuY2VySU5TXzI4TWF4UHJlZGljdGlvbkRlZ3JlZVRyYXZlcnNlcklOU18xMUNvcm5lclRhYmxlRU5TXzM2TWVzaEF0dHJpYnV0ZUluZGljZXNFbmNvZGluZ09ic2VydmVySVMyX0VFRUVFRQBONWRyYWNvMTVQb2ludHNTZXF1ZW5jZXJFAAAAABgsAACTIwAAQCwAAAwjAACwIwAA/////wAAAACoJAAA/AAAAP0AAAD+AAAATjVkcmFjbzI4TWF4UHJlZGljdGlvbkRlZ3JlZVRyYXZlcnNlcklOU18xMUNvcm5lclRhYmxlRU5TXzM2TWVzaEF0dHJpYnV0ZUluZGljZXNFbmNvZGluZ09ic2VydmVySVMxX0VFRUUATjVkcmFjbzEzVHJhdmVyc2VyQmFzZUlOU18xMUNvcm5lclRhYmxlRU5TXzM2TWVzaEF0dHJpYnV0ZUluZGljZXNFbmNvZGluZ09ic2VydmVySVMxX0VFRUUAABgsAABFJAAAQCwAANwjAACgJAAAAAAAAKAkAAD/AAAAAAEAAP4AAAAAAAAAYCUAAAEBAAACAQAAAwEAAAQBAABONWRyYWNvMjJNZXNoVHJhdmVyc2FsU2VxdWVuY2VySU5TXzE5RGVwdGhGaXJzdFRyYXZlcnNlcklOU18xMUNvcm5lclRhYmxlRU5TXzM2TWVzaEF0dHJpYnV0ZUluZGljZXNFbmNvZGluZ09ic2VydmVySVMyX0VFRUVFRQAAAEAsAADgJAAAsCMAAAAAAADgJQAABQEAAAYBAAD+AAAATjVkcmFjbzE5RGVwdGhGaXJzdFRyYXZlcnNlcklOU18xMUNvcm5lclRhYmxlRU5TXzM2TWVzaEF0dHJpYnV0ZUluZGljZXNFbmNvZGluZ09ic2VydmVySVMxX0VFRUUAQCwAAIAlAACgJAAAAAAAAJAmAAAHAQAACAEAAAkBAAAKAQAATjVkcmFjbzIyTWVzaFRyYXZlcnNhbFNlcXVlbmNlcklOU18xOURlcHRoRmlyc3RUcmF2ZXJzZXJJTlNfMjRNZXNoQXR0cmlidXRl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", Zge = "data:application/javascript;base64,dmFyICRqc2NvbXA9JGpzY29tcHx8e307JGpzY29tcC5zY29wZT17fTskanNjb21wLmFycmF5SXRlcmF0b3JJbXBsPWZ1bmN0aW9uKG0pe3ZhciBuPTA7cmV0dXJuIGZ1bmN0aW9uKCl7cmV0dXJuIG48bS5sZW5ndGg/e2RvbmU6ITEsdmFsdWU6bVtuKytdfTp7ZG9uZTohMH19fTskanNjb21wLmFycmF5SXRlcmF0b3I9ZnVuY3Rpb24obSl7cmV0dXJue25leHQ6JGpzY29tcC5hcnJheUl0ZXJhdG9ySW1wbChtKX19OyRqc2NvbXAubWFrZUl0ZXJhdG9yPWZ1bmN0aW9uKG0pe3ZhciBuPSJ1bmRlZmluZWQiIT10eXBlb2YgU3ltYm9sJiZTeW1ib2wuaXRlcmF0b3ImJm1bU3ltYm9sLml0ZXJhdG9yXTtyZXR1cm4gbj9uLmNhbGwobSk6JGpzY29tcC5hcnJheUl0ZXJhdG9yKG0pfTskanNjb21wLkFTU1VNRV9FUzU9ITE7JGpzY29tcC5BU1NVTUVfTk9fTkFUSVZFX01BUD0hMTskanNjb21wLkFTU1VNRV9OT19OQVRJVkVfU0VUPSExOyRqc2NvbXAuU0lNUExFX0ZST1VORF9QT0xZRklMTD0hMTsNCiRqc2NvbXAuSVNPTEFURV9QT0xZRklMTFM9ITE7JGpzY29tcC5GT1JDRV9QT0xZRklMTF9QUk9NSVNFPSExOyRqc2NvbXAuRk9SQ0VfUE9MWUZJTExfUFJPTUlTRV9XSEVOX05PX1VOSEFORExFRF9SRUpFQ1RJT049ITE7JGpzY29tcC5nZXRHbG9iYWw9ZnVuY3Rpb24obSl7bT1bIm9iamVjdCI9PXR5cGVvZiBnbG9iYWxUaGlzJiZnbG9iYWxUaGlzLG0sIm9iamVjdCI9PXR5cGVvZiB3aW5kb3cmJndpbmRvdywib2JqZWN0Ij09dHlwZW9mIHNlbGYmJnNlbGYsIm9iamVjdCI9PXR5cGVvZiBnbG9iYWwmJmdsb2JhbF07Zm9yKHZhciBuPTA7bjxtLmxlbmd0aDsrK24pe3ZhciBrPW1bbl07aWYoayYmay5NYXRoPT1NYXRoKXJldHVybiBrfXRocm93IEVycm9yKCJDYW5ub3QgZmluZCBnbG9iYWwgb2JqZWN0Iik7fTskanNjb21wLmdsb2JhbD0kanNjb21wLmdldEdsb2JhbCh0aGlzKTsNCiRqc2NvbXAuZGVmaW5lUHJvcGVydHk9JGpzY29tcC5BU1NVTUVfRVM1fHwiZnVuY3Rpb24iPT10eXBlb2YgT2JqZWN0LmRlZmluZVByb3BlcnRpZXM/T2JqZWN0LmRlZmluZVByb3BlcnR5OmZ1bmN0aW9uKG0sbixrKXtpZihtPT1BcnJheS5wcm90b3R5cGV8fG09PU9iamVjdC5wcm90b3R5cGUpcmV0dXJuIG07bVtuXT1rLnZhbHVlO3JldHVybiBtfTskanNjb21wLklTX1NZTUJPTF9OQVRJVkU9ImZ1bmN0aW9uIj09PXR5cGVvZiBTeW1ib2wmJiJzeW1ib2wiPT09dHlwZW9mIFN5bWJvbCgieCIpOyRqc2NvbXAuVFJVU1RfRVM2X1BPTFlGSUxMUz0hJGpzY29tcC5JU09MQVRFX1BPTFlGSUxMU3x8JGpzY29tcC5JU19TWU1CT0xfTkFUSVZFOyRqc2NvbXAucG9seWZpbGxzPXt9OyRqc2NvbXAucHJvcGVydHlUb1BvbHlmaWxsU3ltYm9sPXt9OyRqc2NvbXAuUE9MWUZJTExfUFJFRklYPSIkanNjcCQiOw0KdmFyICRqc2NvbXAkbG9va3VwUG9seWZpbGxlZFZhbHVlPWZ1bmN0aW9uKG0sbil7dmFyIGs9JGpzY29tcC5wcm9wZXJ0eVRvUG9seWZpbGxTeW1ib2xbbl07aWYobnVsbD09aylyZXR1cm4gbVtuXTtrPW1ba107cmV0dXJuIHZvaWQgMCE9PWs/azptW25dfTskanNjb21wLnBvbHlmaWxsPWZ1bmN0aW9uKG0sbixrLHEpe24mJigkanNjb21wLklTT0xBVEVfUE9MWUZJTExTPyRqc2NvbXAucG9seWZpbGxJc29sYXRlZChtLG4sayxxKTokanNjb21wLnBvbHlmaWxsVW5pc29sYXRlZChtLG4sayxxKSl9Ow0KJGpzY29tcC5wb2x5ZmlsbFVuaXNvbGF0ZWQ9ZnVuY3Rpb24obSxuLGsscSl7az0kanNjb21wLmdsb2JhbDttPW0uc3BsaXQoIi4iKTtmb3IocT0wO3E8bS5sZW5ndGgtMTtxKyspe3ZhciBoPW1bcV07aWYoIShoIGluIGspKXJldHVybjtrPWtbaF19bT1tW20ubGVuZ3RoLTFdO3E9a1ttXTtuPW4ocSk7biE9cSYmbnVsbCE9biYmJGpzY29tcC5kZWZpbmVQcm9wZXJ0eShrLG0se2NvbmZpZ3VyYWJsZTohMCx3cml0YWJsZTohMCx2YWx1ZTpufSl9Ow0KJGpzY29tcC5wb2x5ZmlsbElzb2xhdGVkPWZ1bmN0aW9uKG0sbixrLHEpe3ZhciBoPW0uc3BsaXQoIi4iKTttPTE9PT1oLmxlbmd0aDtxPWhbMF07cT0hbSYmcSBpbiAkanNjb21wLnBvbHlmaWxscz8kanNjb21wLnBvbHlmaWxsczokanNjb21wLmdsb2JhbDtmb3IodmFyIEE9MDtBPGgubGVuZ3RoLTE7QSsrKXt2YXIgZj1oW0FdO2lmKCEoZiBpbiBxKSlyZXR1cm47cT1xW2ZdfWg9aFtoLmxlbmd0aC0xXTtrPSRqc2NvbXAuSVNfU1lNQk9MX05BVElWRSYmImVzNiI9PT1rP3FbaF06bnVsbDtuPW4oayk7bnVsbCE9biYmKG0/JGpzY29tcC5kZWZpbmVQcm9wZXJ0eSgkanNjb21wLnBvbHlmaWxscyxoLHtjb25maWd1cmFibGU6ITAsd3JpdGFibGU6ITAsdmFsdWU6bn0pOm4hPT1rJiYodm9pZCAwPT09JGpzY29tcC5wcm9wZXJ0eVRvUG9seWZpbGxTeW1ib2xbaF0mJihrPTFFOSpNYXRoLnJhbmRvbSgpPj4+MCwkanNjb21wLnByb3BlcnR5VG9Qb2x5ZmlsbFN5bWJvbFtoXT0kanNjb21wLklTX1NZTUJPTF9OQVRJVkU/DQokanNjb21wLmdsb2JhbC5TeW1ib2woaCk6JGpzY29tcC5QT0xZRklMTF9QUkVGSVgraysiJCIraCksJGpzY29tcC5kZWZpbmVQcm9wZXJ0eShxLCRqc2NvbXAucHJvcGVydHlUb1BvbHlmaWxsU3ltYm9sW2hdLHtjb25maWd1cmFibGU6ITAsd3JpdGFibGU6ITAsdmFsdWU6bn0pKSl9Ow0KJGpzY29tcC5wb2x5ZmlsbCgiUHJvbWlzZSIsZnVuY3Rpb24obSl7ZnVuY3Rpb24gbigpe3RoaXMuYmF0Y2hfPW51bGx9ZnVuY3Rpb24gayhmKXtyZXR1cm4gZiBpbnN0YW5jZW9mIGg/ZjpuZXcgaChmdW5jdGlvbihwLHYpe3AoZil9KX1pZihtJiYoISgkanNjb21wLkZPUkNFX1BPTFlGSUxMX1BST01JU0V8fCRqc2NvbXAuRk9SQ0VfUE9MWUZJTExfUFJPTUlTRV9XSEVOX05PX1VOSEFORExFRF9SRUpFQ1RJT04mJiJ1bmRlZmluZWQiPT09dHlwZW9mICRqc2NvbXAuZ2xvYmFsLlByb21pc2VSZWplY3Rpb25FdmVudCl8fCEkanNjb21wLmdsb2JhbC5Qcm9taXNlfHwtMT09PSRqc2NvbXAuZ2xvYmFsLlByb21pc2UudG9TdHJpbmcoKS5pbmRleE9mKCJbbmF0aXZlIGNvZGVdIikpKXJldHVybiBtO24ucHJvdG90eXBlLmFzeW5jRXhlY3V0ZT1mdW5jdGlvbihmKXtpZihudWxsPT10aGlzLmJhdGNoXyl7dGhpcy5iYXRjaF89W107dmFyIHA9dGhpczt0aGlzLmFzeW5jRXhlY3V0ZUZ1bmN0aW9uKGZ1bmN0aW9uKCl7cC5leGVjdXRlQmF0Y2hfKCl9KX10aGlzLmJhdGNoXy5wdXNoKGYpfTsNCnZhciBxPSRqc2NvbXAuZ2xvYmFsLnNldFRpbWVvdXQ7bi5wcm90b3R5cGUuYXN5bmNFeGVjdXRlRnVuY3Rpb249ZnVuY3Rpb24oZil7cShmLDApfTtuLnByb3RvdHlwZS5leGVjdXRlQmF0Y2hfPWZ1bmN0aW9uKCl7Zm9yKDt0aGlzLmJhdGNoXyYmdGhpcy5iYXRjaF8ubGVuZ3RoOyl7dmFyIGY9dGhpcy5iYXRjaF87dGhpcy5iYXRjaF89W107Zm9yKHZhciBwPTA7cDxmLmxlbmd0aDsrK3Ape3ZhciB2PWZbcF07ZltwXT1udWxsO3RyeXt2KCl9Y2F0Y2goeil7dGhpcy5hc3luY1Rocm93Xyh6KX19fXRoaXMuYmF0Y2hfPW51bGx9O24ucHJvdG90eXBlLmFzeW5jVGhyb3dfPWZ1bmN0aW9uKGYpe3RoaXMuYXN5bmNFeGVjdXRlRnVuY3Rpb24oZnVuY3Rpb24oKXt0aHJvdyBmO30pfTt2YXIgaD1mdW5jdGlvbihmKXt0aGlzLnN0YXRlXz0wO3RoaXMucmVzdWx0Xz12b2lkIDA7dGhpcy5vblNldHRsZWRDYWxsYmFja3NfPVtdO3RoaXMuaXNSZWplY3Rpb25IYW5kbGVkXz0hMTt2YXIgcD10aGlzLmNyZWF0ZVJlc29sdmVBbmRSZWplY3RfKCk7DQp0cnl7ZihwLnJlc29sdmUscC5yZWplY3QpfWNhdGNoKHYpe3AucmVqZWN0KHYpfX07aC5wcm90b3R5cGUuY3JlYXRlUmVzb2x2ZUFuZFJlamVjdF89ZnVuY3Rpb24oKXtmdW5jdGlvbiBmKHope3JldHVybiBmdW5jdGlvbihGKXt2fHwodj0hMCx6LmNhbGwocCxGKSl9fXZhciBwPXRoaXMsdj0hMTtyZXR1cm57cmVzb2x2ZTpmKHRoaXMucmVzb2x2ZVRvXykscmVqZWN0OmYodGhpcy5yZWplY3RfKX19O2gucHJvdG90eXBlLnJlc29sdmVUb189ZnVuY3Rpb24oZil7aWYoZj09PXRoaXMpdGhpcy5yZWplY3RfKG5ldyBUeXBlRXJyb3IoIkEgUHJvbWlzZSBjYW5ub3QgcmVzb2x2ZSB0byBpdHNlbGYiKSk7ZWxzZSBpZihmIGluc3RhbmNlb2YgaCl0aGlzLnNldHRsZVNhbWVBc1Byb21pc2VfKGYpO2Vsc2V7YTpzd2l0Y2godHlwZW9mIGYpe2Nhc2UgIm9iamVjdCI6dmFyIHA9bnVsbCE9ZjticmVhayBhO2Nhc2UgImZ1bmN0aW9uIjpwPSEwO2JyZWFrIGE7ZGVmYXVsdDpwPSExfXA/dGhpcy5yZXNvbHZlVG9Ob25Qcm9taXNlT2JqXyhmKToNCnRoaXMuZnVsZmlsbF8oZil9fTtoLnByb3RvdHlwZS5yZXNvbHZlVG9Ob25Qcm9taXNlT2JqXz1mdW5jdGlvbihmKXt2YXIgcD12b2lkIDA7dHJ5e3A9Zi50aGVufWNhdGNoKHYpe3RoaXMucmVqZWN0Xyh2KTtyZXR1cm59ImZ1bmN0aW9uIj09dHlwZW9mIHA/dGhpcy5zZXR0bGVTYW1lQXNUaGVuYWJsZV8ocCxmKTp0aGlzLmZ1bGZpbGxfKGYpfTtoLnByb3RvdHlwZS5yZWplY3RfPWZ1bmN0aW9uKGYpe3RoaXMuc2V0dGxlXygyLGYpfTtoLnByb3RvdHlwZS5mdWxmaWxsXz1mdW5jdGlvbihmKXt0aGlzLnNldHRsZV8oMSxmKX07aC5wcm90b3R5cGUuc2V0dGxlXz1mdW5jdGlvbihmLHApe2lmKDAhPXRoaXMuc3RhdGVfKXRocm93IEVycm9yKCJDYW5ub3Qgc2V0dGxlKCIrZisiLCAiK3ArIik6IFByb21pc2UgYWxyZWFkeSBzZXR0bGVkIGluIHN0YXRlIit0aGlzLnN0YXRlXyk7dGhpcy5zdGF0ZV89Zjt0aGlzLnJlc3VsdF89cDsyPT09dGhpcy5zdGF0ZV8mJnRoaXMuc2NoZWR1bGVVbmhhbmRsZWRSZWplY3Rpb25DaGVja18oKTsNCnRoaXMuZXhlY3V0ZU9uU2V0dGxlZENhbGxiYWNrc18oKX07aC5wcm90b3R5cGUuc2NoZWR1bGVVbmhhbmRsZWRSZWplY3Rpb25DaGVja189ZnVuY3Rpb24oKXt2YXIgZj10aGlzO3EoZnVuY3Rpb24oKXtpZihmLm5vdGlmeVVuaGFuZGxlZFJlamVjdGlvbl8oKSl7dmFyIHA9JGpzY29tcC5nbG9iYWwuY29uc29sZTsidW5kZWZpbmVkIiE9PXR5cGVvZiBwJiZwLmVycm9yKGYucmVzdWx0Xyl9fSwxKX07aC5wcm90b3R5cGUubm90aWZ5VW5oYW5kbGVkUmVqZWN0aW9uXz1mdW5jdGlvbigpe2lmKHRoaXMuaXNSZWplY3Rpb25IYW5kbGVkXylyZXR1cm4hMTt2YXIgZj0kanNjb21wLmdsb2JhbC5DdXN0b21FdmVudCxwPSRqc2NvbXAuZ2xvYmFsLkV2ZW50LHY9JGpzY29tcC5nbG9iYWwuZGlzcGF0Y2hFdmVudDtpZigidW5kZWZpbmVkIj09PXR5cGVvZiB2KXJldHVybiEwOyJmdW5jdGlvbiI9PT10eXBlb2YgZj9mPW5ldyBmKCJ1bmhhbmRsZWRyZWplY3Rpb24iLHtjYW5jZWxhYmxlOiEwfSk6DQoiZnVuY3Rpb24iPT09dHlwZW9mIHA/Zj1uZXcgcCgidW5oYW5kbGVkcmVqZWN0aW9uIix7Y2FuY2VsYWJsZTohMH0pOihmPSRqc2NvbXAuZ2xvYmFsLmRvY3VtZW50LmNyZWF0ZUV2ZW50KCJDdXN0b21FdmVudCIpLGYuaW5pdEN1c3RvbUV2ZW50KCJ1bmhhbmRsZWRyZWplY3Rpb24iLCExLCEwLGYpKTtmLnByb21pc2U9dGhpcztmLnJlYXNvbj10aGlzLnJlc3VsdF87cmV0dXJuIHYoZil9O2gucHJvdG90eXBlLmV4ZWN1dGVPblNldHRsZWRDYWxsYmFja3NfPWZ1bmN0aW9uKCl7aWYobnVsbCE9dGhpcy5vblNldHRsZWRDYWxsYmFja3NfKXtmb3IodmFyIGY9MDtmPHRoaXMub25TZXR0bGVkQ2FsbGJhY2tzXy5sZW5ndGg7KytmKUEuYXN5bmNFeGVjdXRlKHRoaXMub25TZXR0bGVkQ2FsbGJhY2tzX1tmXSk7dGhpcy5vblNldHRsZWRDYWxsYmFja3NfPW51bGx9fTt2YXIgQT1uZXcgbjtoLnByb3RvdHlwZS5zZXR0bGVTYW1lQXNQcm9taXNlXz1mdW5jdGlvbihmKXt2YXIgcD10aGlzLmNyZWF0ZVJlc29sdmVBbmRSZWplY3RfKCk7DQpmLmNhbGxXaGVuU2V0dGxlZF8ocC5yZXNvbHZlLHAucmVqZWN0KX07aC5wcm90b3R5cGUuc2V0dGxlU2FtZUFzVGhlbmFibGVfPWZ1bmN0aW9uKGYscCl7dmFyIHY9dGhpcy5jcmVhdGVSZXNvbHZlQW5kUmVqZWN0XygpO3RyeXtmLmNhbGwocCx2LnJlc29sdmUsdi5yZWplY3QpfWNhdGNoKHope3YucmVqZWN0KHopfX07aC5wcm90b3R5cGUudGhlbj1mdW5jdGlvbihmLHApe2Z1bmN0aW9uIHYoUCx0KXtyZXR1cm4iZnVuY3Rpb24iPT10eXBlb2YgUD9mdW5jdGlvbih4KXt0cnl7eihQKHgpKX1jYXRjaChEKXtGKEQpfX06dH12YXIgeixGLGZhPW5ldyBoKGZ1bmN0aW9uKFAsdCl7ej1QO0Y9dH0pO3RoaXMuY2FsbFdoZW5TZXR0bGVkXyh2KGYseiksdihwLEYpKTtyZXR1cm4gZmF9O2gucHJvdG90eXBlLmNhdGNoPWZ1bmN0aW9uKGYpe3JldHVybiB0aGlzLnRoZW4odm9pZCAwLGYpfTtoLnByb3RvdHlwZS5jYWxsV2hlblNldHRsZWRfPWZ1bmN0aW9uKGYscCl7ZnVuY3Rpb24gdigpe3N3aXRjaCh6LnN0YXRlXyl7Y2FzZSAxOmYoei5yZXN1bHRfKTsNCmJyZWFrO2Nhc2UgMjpwKHoucmVzdWx0Xyk7YnJlYWs7ZGVmYXVsdDp0aHJvdyBFcnJvcigiVW5leHBlY3RlZCBzdGF0ZTogIit6LnN0YXRlXyk7fX12YXIgej10aGlzO251bGw9PXRoaXMub25TZXR0bGVkQ2FsbGJhY2tzXz9BLmFzeW5jRXhlY3V0ZSh2KTp0aGlzLm9uU2V0dGxlZENhbGxiYWNrc18ucHVzaCh2KTt0aGlzLmlzUmVqZWN0aW9uSGFuZGxlZF89ITB9O2gucmVzb2x2ZT1rO2gucmVqZWN0PWZ1bmN0aW9uKGYpe3JldHVybiBuZXcgaChmdW5jdGlvbihwLHYpe3YoZil9KX07aC5yYWNlPWZ1bmN0aW9uKGYpe3JldHVybiBuZXcgaChmdW5jdGlvbihwLHYpe2Zvcih2YXIgej0kanNjb21wLm1ha2VJdGVyYXRvcihmKSxGPXoubmV4dCgpOyFGLmRvbmU7Rj16Lm5leHQoKSlrKEYudmFsdWUpLmNhbGxXaGVuU2V0dGxlZF8ocCx2KX0pfTtoLmFsbD1mdW5jdGlvbihmKXt2YXIgcD0kanNjb21wLm1ha2VJdGVyYXRvcihmKSx2PXAubmV4dCgpO3JldHVybiB2LmRvbmU/ayhbXSk6bmV3IGgoZnVuY3Rpb24oeiwNCkYpe2Z1bmN0aW9uIGZhKHgpe3JldHVybiBmdW5jdGlvbihEKXtQW3hdPUQ7dC0tOzA9PXQmJnooUCl9fXZhciBQPVtdLHQ9MDtkbyBQLnB1c2godm9pZCAwKSx0Kyssayh2LnZhbHVlKS5jYWxsV2hlblNldHRsZWRfKGZhKFAubGVuZ3RoLTEpLEYpLHY9cC5uZXh0KCk7d2hpbGUoIXYuZG9uZSl9KX07cmV0dXJuIGh9LCJlczYiLCJlczMiKTskanNjb21wLmNoZWNrU3RyaW5nQXJncz1mdW5jdGlvbihtLG4sayl7aWYobnVsbD09bSl0aHJvdyBuZXcgVHlwZUVycm9yKCJUaGUgJ3RoaXMnIHZhbHVlIGZvciBTdHJpbmcucHJvdG90eXBlLiIraysiIG11c3Qgbm90IGJlIG51bGwgb3IgdW5kZWZpbmVkIik7aWYobiBpbnN0YW5jZW9mIFJlZ0V4cCl0aHJvdyBuZXcgVHlwZUVycm9yKCJGaXJzdCBhcmd1bWVudCB0byBTdHJpbmcucHJvdG90eXBlLiIraysiIG11c3Qgbm90IGJlIGEgcmVndWxhciBleHByZXNzaW9uIik7cmV0dXJuIG0rIiJ9Ow0KJGpzY29tcC5wb2x5ZmlsbCgiU3RyaW5nLnByb3RvdHlwZS5zdGFydHNXaXRoIixmdW5jdGlvbihtKXtyZXR1cm4gbT9tOmZ1bmN0aW9uKG4sayl7dmFyIHE9JGpzY29tcC5jaGVja1N0cmluZ0FyZ3ModGhpcyxuLCJzdGFydHNXaXRoIik7bis9IiI7dmFyIGg9cS5sZW5ndGgsQT1uLmxlbmd0aDtrPU1hdGgubWF4KDAsTWF0aC5taW4oa3wwLHEubGVuZ3RoKSk7Zm9yKHZhciBmPTA7ZjxBJiZrPGg7KWlmKHFbaysrXSE9bltmKytdKXJldHVybiExO3JldHVybiBmPj1BfX0sImVzNiIsImVzMyIpOw0KJGpzY29tcC5wb2x5ZmlsbCgiQXJyYXkucHJvdG90eXBlLmNvcHlXaXRoaW4iLGZ1bmN0aW9uKG0pe2Z1bmN0aW9uIG4oayl7az1OdW1iZXIoayk7cmV0dXJuIEluZmluaXR5PT09a3x8LUluZmluaXR5PT09az9rOmt8MH1yZXR1cm4gbT9tOmZ1bmN0aW9uKGsscSxoKXt2YXIgQT10aGlzLmxlbmd0aDtrPW4oayk7cT1uKHEpO2g9dm9pZCAwPT09aD9BOm4oaCk7az0wPms/TWF0aC5tYXgoQStrLDApOk1hdGgubWluKGssQSk7cT0wPnE/TWF0aC5tYXgoQStxLDApOk1hdGgubWluKHEsQSk7aD0wPmg/TWF0aC5tYXgoQStoLDApOk1hdGgubWluKGgsQSk7aWYoazxxKWZvcig7cTxoOylxIGluIHRoaXM/dGhpc1trKytdPXRoaXNbcSsrXTooZGVsZXRlIHRoaXNbaysrXSxxKyspO2Vsc2UgZm9yKGg9TWF0aC5taW4oaCxBK3Etayksays9aC1xO2g+cTspLS1oIGluIHRoaXM/dGhpc1stLWtdPXRoaXNbaF06ZGVsZXRlIHRoaXNbLS1rXTtyZXR1cm4gdGhpc319LCJlczYiLCJlczMiKTsNCiRqc2NvbXAudHlwZWRBcnJheUNvcHlXaXRoaW49ZnVuY3Rpb24obSl7cmV0dXJuIG0/bTpBcnJheS5wcm90b3R5cGUuY29weVdpdGhpbn07JGpzY29tcC5wb2x5ZmlsbCgiSW50OEFycmF5LnByb3RvdHlwZS5jb3B5V2l0aGluIiwkanNjb21wLnR5cGVkQXJyYXlDb3B5V2l0aGluLCJlczYiLCJlczUiKTskanNjb21wLnBvbHlmaWxsKCJVaW50OEFycmF5LnByb3RvdHlwZS5jb3B5V2l0aGluIiwkanNjb21wLnR5cGVkQXJyYXlDb3B5V2l0aGluLCJlczYiLCJlczUiKTskanNjb21wLnBvbHlmaWxsKCJVaW50OENsYW1wZWRBcnJheS5wcm90b3R5cGUuY29weVdpdGhpbiIsJGpzY29tcC50eXBlZEFycmF5Q29weVdpdGhpbiwiZXM2IiwiZXM1Iik7JGpzY29tcC5wb2x5ZmlsbCgiSW50MTZBcnJheS5wcm90b3R5cGUuY29weVdpdGhpbiIsJGpzY29tcC50eXBlZEFycmF5Q29weVdpdGhpbiwiZXM2IiwiZXM1Iik7DQokanNjb21wLnBvbHlmaWxsKCJVaW50MTZBcnJheS5wcm90b3R5cGUuY29weVdpdGhpbiIsJGpzY29tcC50eXBlZEFycmF5Q29weVdpdGhpbiwiZXM2IiwiZXM1Iik7JGpzY29tcC5wb2x5ZmlsbCgiSW50MzJBcnJheS5wcm90b3R5cGUuY29weVdpdGhpbiIsJGpzY29tcC50eXBlZEFycmF5Q29weVdpdGhpbiwiZXM2IiwiZXM1Iik7JGpzY29tcC5wb2x5ZmlsbCgiVWludDMyQXJyYXkucHJvdG90eXBlLmNvcHlXaXRoaW4iLCRqc2NvbXAudHlwZWRBcnJheUNvcHlXaXRoaW4sImVzNiIsImVzNSIpOyRqc2NvbXAucG9seWZpbGwoIkZsb2F0MzJBcnJheS5wcm90b3R5cGUuY29weVdpdGhpbiIsJGpzY29tcC50eXBlZEFycmF5Q29weVdpdGhpbiwiZXM2IiwiZXM1Iik7JGpzY29tcC5wb2x5ZmlsbCgiRmxvYXQ2NEFycmF5LnByb3RvdHlwZS5jb3B5V2l0aGluIiwkanNjb21wLnR5cGVkQXJyYXlDb3B5V2l0aGluLCJlczYiLCJlczUiKTsNCnZhciBEcmFjb0RlY29kZXJNb2R1bGU9ZnVuY3Rpb24oKXt2YXIgbT0idW5kZWZpbmVkIiE9PXR5cGVvZiBkb2N1bWVudCYmZG9jdW1lbnQuY3VycmVudFNjcmlwdD9kb2N1bWVudC5jdXJyZW50U2NyaXB0LnNyYzp2b2lkIDA7InVuZGVmaW5lZCIhPT10eXBlb2YgX19maWxlbmFtZSYmKG09bXx8X19maWxlbmFtZSk7cmV0dXJuIGZ1bmN0aW9uKG4pe2Z1bmN0aW9uIGsoZSl7cmV0dXJuIGEubG9jYXRlRmlsZT9hLmxvY2F0ZUZpbGUoZSxVKTpVK2V9ZnVuY3Rpb24gcShlLGIpe2V8fGYoIkFzc2VydGlvbiBmYWlsZWQ6ICIrYil9ZnVuY3Rpb24gaChlLGIpe2lmKGUpe3ZhciBjPW1hO3ZhciBkPWUrYjtmb3IoYj1lO2NbYl0mJiEoYj49ZCk7KSsrYjtpZigxNjxiLWUmJmMuc3ViYXJyYXkmJndhKWM9d2EuZGVjb2RlKGMuc3ViYXJyYXkoZSxiKSk7ZWxzZXtmb3IoZD0iIjtlPGI7KXt2YXIgZz1jW2UrK107aWYoZyYxMjgpe3ZhciB1PWNbZSsrXSY2MztpZigxOTI9PShnJjIyNCkpZCs9U3RyaW5nLmZyb21DaGFyQ29kZSgoZyYNCjMxKTw8Nnx1KTtlbHNle3ZhciBYPWNbZSsrXSY2MztnPTIyND09KGcmMjQwKT8oZyYxNSk8PDEyfHU8PDZ8WDooZyY3KTw8MTh8dTw8MTJ8WDw8NnxjW2UrK10mNjM7NjU1MzY+Zz9kKz1TdHJpbmcuZnJvbUNoYXJDb2RlKGcpOihnLT02NTUzNixkKz1TdHJpbmcuZnJvbUNoYXJDb2RlKDU1Mjk2fGc+PjEwLDU2MzIwfGcmMTAyMykpfX1lbHNlIGQrPVN0cmluZy5mcm9tQ2hhckNvZGUoZyl9Yz1kfX1lbHNlIGM9IiI7cmV0dXJuIGN9ZnVuY3Rpb24gQShlKXt4YT1lO2EuSEVBUDg9WT1uZXcgSW50OEFycmF5KGUpO2EuSEVBUDE2PW5ldyBJbnQxNkFycmF5KGUpO2EuSEVBUDMyPVY9bmV3IEludDMyQXJyYXkoZSk7YS5IRUFQVTg9bWE9bmV3IFVpbnQ4QXJyYXkoZSk7YS5IRUFQVTE2PW5ldyBVaW50MTZBcnJheShlKTthLkhFQVBVMzI9bmV3IFVpbnQzMkFycmF5KGUpO2EuSEVBUEYzMj1uZXcgRmxvYXQzMkFycmF5KGUpO2EuSEVBUEY2ND1uZXcgRmxvYXQ2NEFycmF5KGUpfWZ1bmN0aW9uIGYoZSl7aWYoYS5vbkFib3J0KWEub25BYm9ydChlKTsNCmU9IkFib3J0ZWQoIitlKyIpIjtoYShlKTt5YT0hMDtlPW5ldyBXZWJBc3NlbWJseS5SdW50aW1lRXJyb3IoZSsiLiBCdWlsZCB3aXRoIC1zIEFTU0VSVElPTlM9MSBmb3IgbW9yZSBpbmZvLiIpO3FhKGUpO3Rocm93IGU7fWZ1bmN0aW9uIHAoZSl7dHJ5e2lmKGU9PVEmJmlhKXJldHVybiBuZXcgVWludDhBcnJheShpYSk7aWYocmEpcmV0dXJuIHJhKGUpO3Rocm93ImJvdGggYXN5bmMgYW5kIHN5bmMgZmV0Y2hpbmcgb2YgdGhlIHdhc20gZmFpbGVkIjt9Y2F0Y2goYil7ZihiKX19ZnVuY3Rpb24gdigpe2lmKCFpYSYmKHphfHxqYSkpe2lmKCJmdW5jdGlvbiI9PT10eXBlb2YgZmV0Y2gmJiFRLnN0YXJ0c1dpdGgoImZpbGU6Ly8iKSlyZXR1cm4gZmV0Y2goUSx7Y3JlZGVudGlhbHM6InNhbWUtb3JpZ2luIn0pLnRoZW4oZnVuY3Rpb24oZSl7aWYoIWUub2spdGhyb3ciZmFpbGVkIHRvIGxvYWQgd2FzbSBiaW5hcnkgZmlsZSBhdCAnIitRKyInIjtyZXR1cm4gZS5hcnJheUJ1ZmZlcigpfSkuY2F0Y2goZnVuY3Rpb24oKXtyZXR1cm4gcChRKX0pOw0KaWYoc2EpcmV0dXJuIG5ldyBQcm9taXNlKGZ1bmN0aW9uKGUsYil7c2EoUSxmdW5jdGlvbihjKXtlKG5ldyBVaW50OEFycmF5KGMpKX0sYil9KX1yZXR1cm4gUHJvbWlzZS5yZXNvbHZlKCkudGhlbihmdW5jdGlvbigpe3JldHVybiBwKFEpfSl9ZnVuY3Rpb24geihlKXtmb3IoOzA8ZS5sZW5ndGg7KXt2YXIgYj1lLnNoaWZ0KCk7aWYoImZ1bmN0aW9uIj09dHlwZW9mIGIpYihhKTtlbHNle3ZhciBjPWIuZnVuYzsibnVtYmVyIj09PXR5cGVvZiBjP3ZvaWQgMD09PWIuYXJnP0YoYykoKTpGKGMpKGIuYXJnKTpjKHZvaWQgMD09PWIuYXJnP251bGw6Yi5hcmcpfX19ZnVuY3Rpb24gRihlKXt2YXIgYj1uYVtlXTtifHwoZT49bmEubGVuZ3RoJiYobmEubGVuZ3RoPWUrMSksbmFbZV09Yj1BYS5nZXQoZSkpO3JldHVybiBifWZ1bmN0aW9uIGZhKGUpe3RoaXMuZXhjUHRyPWU7dGhpcy5wdHI9ZS0xNjt0aGlzLnNldF90eXBlPWZ1bmN0aW9uKGIpe1ZbdGhpcy5wdHIrND4+Ml09Yn07dGhpcy5nZXRfdHlwZT0NCmZ1bmN0aW9uKCl7cmV0dXJuIFZbdGhpcy5wdHIrND4+Ml19O3RoaXMuc2V0X2Rlc3RydWN0b3I9ZnVuY3Rpb24oYil7Vlt0aGlzLnB0cis4Pj4yXT1ifTt0aGlzLmdldF9kZXN0cnVjdG9yPWZ1bmN0aW9uKCl7cmV0dXJuIFZbdGhpcy5wdHIrOD4+Ml19O3RoaXMuc2V0X3JlZmNvdW50PWZ1bmN0aW9uKGIpe1ZbdGhpcy5wdHI+PjJdPWJ9O3RoaXMuc2V0X2NhdWdodD1mdW5jdGlvbihiKXtZW3RoaXMucHRyKzEyPj4wXT1iPzE6MH07dGhpcy5nZXRfY2F1Z2h0PWZ1bmN0aW9uKCl7cmV0dXJuIDAhPVlbdGhpcy5wdHIrMTI+PjBdfTt0aGlzLnNldF9yZXRocm93bj1mdW5jdGlvbihiKXtZW3RoaXMucHRyKzEzPj4wXT1iPzE6MH07dGhpcy5nZXRfcmV0aHJvd249ZnVuY3Rpb24oKXtyZXR1cm4gMCE9WVt0aGlzLnB0cisxMz4+MF19O3RoaXMuaW5pdD1mdW5jdGlvbihiLGMpe3RoaXMuc2V0X3R5cGUoYik7dGhpcy5zZXRfZGVzdHJ1Y3RvcihjKTt0aGlzLnNldF9yZWZjb3VudCgwKTt0aGlzLnNldF9jYXVnaHQoITEpOw0KdGhpcy5zZXRfcmV0aHJvd24oITEpfTt0aGlzLmFkZF9yZWY9ZnVuY3Rpb24oKXtWW3RoaXMucHRyPj4yXSs9MX07dGhpcy5yZWxlYXNlX3JlZj1mdW5jdGlvbigpe3ZhciBiPVZbdGhpcy5wdHI+PjJdO1ZbdGhpcy5wdHI+PjJdPWItMTtyZXR1cm4gMT09PWJ9fWZ1bmN0aW9uIFAoZSl7ZnVuY3Rpb24gYigpe2lmKCFvYSYmKG9hPSEwLGEuY2FsbGVkUnVuPSEwLCF5YSkpe0JhPSEwO3oodGEpO0NhKGEpO2lmKGEub25SdW50aW1lSW5pdGlhbGl6ZWQpYS5vblJ1bnRpbWVJbml0aWFsaXplZCgpO2lmKGEucG9zdFJ1bilmb3IoImZ1bmN0aW9uIj09dHlwZW9mIGEucG9zdFJ1biYmKGEucG9zdFJ1bj1bYS5wb3N0UnVuXSk7YS5wb3N0UnVuLmxlbmd0aDspRGEudW5zaGlmdChhLnBvc3RSdW4uc2hpZnQoKSk7eihEYSl9fWlmKCEoMDxjYSkpe2lmKGEucHJlUnVuKWZvcigiZnVuY3Rpb24iPT10eXBlb2YgYS5wcmVSdW4mJihhLnByZVJ1bj1bYS5wcmVSdW5dKTthLnByZVJ1bi5sZW5ndGg7KUVhLnVuc2hpZnQoYS5wcmVSdW4uc2hpZnQoKSk7DQp6KEVhKTswPGNhfHwoYS5zZXRTdGF0dXM/KGEuc2V0U3RhdHVzKCJSdW5uaW5nLi4uIiksc2V0VGltZW91dChmdW5jdGlvbigpe3NldFRpbWVvdXQoZnVuY3Rpb24oKXthLnNldFN0YXR1cygiIil9LDEpO2IoKX0sMSkpOmIoKSl9fWZ1bmN0aW9uIHQoKXt9ZnVuY3Rpb24geChlKXtyZXR1cm4oZXx8dCkuX19jYWNoZV9ffWZ1bmN0aW9uIEQoZSxiKXt2YXIgYz14KGIpLGQ9Y1tlXTtpZihkKXJldHVybiBkO2Q9T2JqZWN0LmNyZWF0ZSgoYnx8dCkucHJvdG90eXBlKTtkLnB0cj1lO3JldHVybiBjW2VdPWR9ZnVuY3Rpb24gYWEoZSl7aWYoInN0cmluZyI9PT10eXBlb2YgZSl7Zm9yKHZhciBiPTAsYz0wO2M8ZS5sZW5ndGg7KytjKXt2YXIgZD1lLmNoYXJDb2RlQXQoYyk7NTUyOTY8PWQmJjU3MzQzPj1kJiYoZD02NTUzNisoKGQmMTAyMyk8PDEwKXxlLmNoYXJDb2RlQXQoKytjKSYxMDIzKTsxMjc+PWQ/KytiOmI9MjA0Nz49ZD9iKzI6NjU1MzU+PWQ/YiszOmIrNH1iPUFycmF5KGIrMSk7DQpjPTA7ZD1iLmxlbmd0aDtpZigwPGQpe2Q9YytkLTE7Zm9yKHZhciBnPTA7ZzxlLmxlbmd0aDsrK2cpe3ZhciB1PWUuY2hhckNvZGVBdChnKTtpZig1NTI5Njw9dSYmNTczNDM+PXUpe3ZhciBYPWUuY2hhckNvZGVBdCgrK2cpO3U9NjU1MzYrKCh1JjEwMjMpPDwxMCl8WCYxMDIzfWlmKDEyNz49dSl7aWYoYz49ZClicmVhaztiW2MrK109dX1lbHNle2lmKDIwNDc+PXUpe2lmKGMrMT49ZClicmVhaztiW2MrK109MTkyfHU+PjZ9ZWxzZXtpZig2NTUzNT49dSl7aWYoYysyPj1kKWJyZWFrO2JbYysrXT0yMjR8dT4+MTJ9ZWxzZXtpZihjKzM+PWQpYnJlYWs7YltjKytdPTI0MHx1Pj4xODtiW2MrK109MTI4fHU+PjEyJjYzfWJbYysrXT0xMjh8dT4+NiY2M31iW2MrK109MTI4fHUmNjN9fWJbY109MH1lPXIuYWxsb2MoYixZKTtyLmNvcHkoYixZLGUpO3JldHVybiBlfXJldHVybiBlfWZ1bmN0aW9uIHVhKGUpe2lmKCJvYmplY3QiPT09dHlwZW9mIGUpe3ZhciBiPXIuYWxsb2MoZSxZKTtyLmNvcHkoZSwNClksYik7cmV0dXJuIGJ9cmV0dXJuIGV9ZnVuY3Rpb24gWigpe3Rocm93ImNhbm5vdCBjb25zdHJ1Y3QgYSBWb2lkUHRyLCBubyBjb25zdHJ1Y3RvciBpbiBJREwiO31mdW5jdGlvbiBTKCl7dGhpcy5wdHI9RmEoKTt4KFMpW3RoaXMucHRyXT10aGlzfWZ1bmN0aW9uIFIoKXt0aGlzLnB0cj1HYSgpO3goUilbdGhpcy5wdHJdPXRoaXN9ZnVuY3Rpb24gVygpe3RoaXMucHRyPUhhKCk7eChXKVt0aGlzLnB0cl09dGhpc31mdW5jdGlvbiB3KCl7dGhpcy5wdHI9SWEoKTt4KHcpW3RoaXMucHRyXT10aGlzfWZ1bmN0aW9uIEMoKXt0aGlzLnB0cj1KYSgpO3goQylbdGhpcy5wdHJdPXRoaXN9ZnVuY3Rpb24gRygpe3RoaXMucHRyPUthKCk7eChHKVt0aGlzLnB0cl09dGhpc31mdW5jdGlvbiBIKCl7dGhpcy5wdHI9TGEoKTt4KEgpW3RoaXMucHRyXT10aGlzfWZ1bmN0aW9uIEUoKXt0aGlzLnB0cj1NYSgpO3goRSlbdGhpcy5wdHJdPXRoaXN9ZnVuY3Rpb24gVCgpe3RoaXMucHRyPU5hKCk7eChUKVt0aGlzLnB0cl09DQp0aGlzfWZ1bmN0aW9uIEIoKXt0aHJvdyJjYW5ub3QgY29uc3RydWN0IGEgU3RhdHVzLCBubyBjb25zdHJ1Y3RvciBpbiBJREwiO31mdW5jdGlvbiBJKCl7dGhpcy5wdHI9T2EoKTt4KEkpW3RoaXMucHRyXT10aGlzfWZ1bmN0aW9uIEooKXt0aGlzLnB0cj1QYSgpO3goSilbdGhpcy5wdHJdPXRoaXN9ZnVuY3Rpb24gSygpe3RoaXMucHRyPVFhKCk7eChLKVt0aGlzLnB0cl09dGhpc31mdW5jdGlvbiBMKCl7dGhpcy5wdHI9UmEoKTt4KEwpW3RoaXMucHRyXT10aGlzfWZ1bmN0aW9uIE0oKXt0aGlzLnB0cj1TYSgpO3goTSlbdGhpcy5wdHJdPXRoaXN9ZnVuY3Rpb24gTigpe3RoaXMucHRyPVRhKCk7eChOKVt0aGlzLnB0cl09dGhpc31mdW5jdGlvbiBPKCl7dGhpcy5wdHI9VWEoKTt4KE8pW3RoaXMucHRyXT10aGlzfWZ1bmN0aW9uIHkoKXt0aGlzLnB0cj1WYSgpO3goeSlbdGhpcy5wdHJdPXRoaXN9ZnVuY3Rpb24gbCgpe3RoaXMucHRyPVdhKCk7eChsKVt0aGlzLnB0cl09dGhpc31uPW58fA0Ke307dmFyIGE9InVuZGVmaW5lZCIhPT10eXBlb2Ygbj9uOnt9LENhLHFhO2EucmVhZHk9bmV3IFByb21pc2UoZnVuY3Rpb24oZSxiKXtDYT1lO3FhPWJ9KTt2YXIgWGE9ITEsWWE9ITE7YS5vblJ1bnRpbWVJbml0aWFsaXplZD1mdW5jdGlvbigpe1hhPSEwO2lmKFlhJiYiZnVuY3Rpb24iPT09dHlwZW9mIGEub25Nb2R1bGVMb2FkZWQpYS5vbk1vZHVsZUxvYWRlZChhKX07YS5vbk1vZHVsZVBhcnNlZD1mdW5jdGlvbigpe1lhPSEwO2lmKFhhJiYiZnVuY3Rpb24iPT09dHlwZW9mIGEub25Nb2R1bGVMb2FkZWQpYS5vbk1vZHVsZUxvYWRlZChhKX07YS5pc1ZlcnNpb25TdXBwb3J0ZWQ9ZnVuY3Rpb24oZSl7aWYoInN0cmluZyIhPT10eXBlb2YgZSlyZXR1cm4hMTtlPWUuc3BsaXQoIi4iKTtyZXR1cm4gMj5lLmxlbmd0aHx8MzxlLmxlbmd0aD8hMToxPT1lWzBdJiYwPD1lWzFdJiY1Pj1lWzFdPyEwOjAhPWVbMF18fDEwPGVbMV0/ITE6ITB9O3ZhciBrYT17fSxiYTtmb3IoYmEgaW4gYSlhLmhhc093blByb3BlcnR5KGJhKSYmDQooa2FbYmFdPWFbYmFdKTt2YXIgemE9Im9iamVjdCI9PT10eXBlb2Ygd2luZG93LGphPSJmdW5jdGlvbiI9PT10eXBlb2YgaW1wb3J0U2NyaXB0cyxVPSIiLGRhLGVhO2lmKCJvYmplY3QiPT09dHlwZW9mIHByb2Nlc3MmJiJvYmplY3QiPT09dHlwZW9mIHByb2Nlc3MudmVyc2lvbnMmJiJzdHJpbmciPT09dHlwZW9mIHByb2Nlc3MudmVyc2lvbnMubm9kZSl7VT1qYT9yZXF1aXJlKCJwYXRoIikuZGlybmFtZShVKSsiLyI6X19kaXJuYW1lKyIvIjt2YXIgWmE9ZnVuY3Rpb24oZSxiKXtkYXx8KGRhPXJlcXVpcmUoImZzIikpO2VhfHwoZWE9cmVxdWlyZSgicGF0aCIpKTtlPWVhLm5vcm1hbGl6ZShlKTtyZXR1cm4gZGEucmVhZEZpbGVTeW5jKGUsYj9udWxsOiJ1dGY4Iil9O3ZhciByYT1mdW5jdGlvbihlKXtlPVphKGUsITApO2UuYnVmZmVyfHwoZT1uZXcgVWludDhBcnJheShlKSk7cShlLmJ1ZmZlcik7cmV0dXJuIGV9O3ZhciBzYT1mdW5jdGlvbihlLGIsYyl7ZGF8fChkYT1yZXF1aXJlKCJmcyIpKTsNCmVhfHwoZWE9cmVxdWlyZSgicGF0aCIpKTtlPWVhLm5vcm1hbGl6ZShlKTtkYS5yZWFkRmlsZShlLGZ1bmN0aW9uKGQsZyl7ZD9jKGQpOmIoZy5idWZmZXIpfSl9OzE8cHJvY2Vzcy5hcmd2Lmxlbmd0aCYmcHJvY2Vzcy5hcmd2WzFdLnJlcGxhY2UoL1xcL2csIi8iKTtwcm9jZXNzLmFyZ3Yuc2xpY2UoMik7YS5pbnNwZWN0PWZ1bmN0aW9uKCl7cmV0dXJuIltFbXNjcmlwdGVuIE1vZHVsZSBvYmplY3RdIn19ZWxzZSBpZih6YXx8amEpamE/VT1zZWxmLmxvY2F0aW9uLmhyZWY6InVuZGVmaW5lZCIhPT10eXBlb2YgZG9jdW1lbnQmJmRvY3VtZW50LmN1cnJlbnRTY3JpcHQmJihVPWRvY3VtZW50LmN1cnJlbnRTY3JpcHQuc3JjKSxtJiYoVT1tKSxVPTAhPT1VLmluZGV4T2YoImJsb2I6Iik/VS5zdWJzdHIoMCxVLnJlcGxhY2UoL1s/I10uKi8sIiIpLmxhc3RJbmRleE9mKCIvIikrMSk6IiIsWmE9ZnVuY3Rpb24oZSl7dmFyIGI9bmV3IFhNTEh0dHBSZXF1ZXN0O2Iub3BlbigiR0VUIixlLA0KITEpO2Iuc2VuZChudWxsKTtyZXR1cm4gYi5yZXNwb25zZVRleHR9LGphJiYocmE9ZnVuY3Rpb24oZSl7dmFyIGI9bmV3IFhNTEh0dHBSZXF1ZXN0O2Iub3BlbigiR0VUIixlLCExKTtiLnJlc3BvbnNlVHlwZT0iYXJyYXlidWZmZXIiO2Iuc2VuZChudWxsKTtyZXR1cm4gbmV3IFVpbnQ4QXJyYXkoYi5yZXNwb25zZSl9KSxzYT1mdW5jdGlvbihlLGIsYyl7dmFyIGQ9bmV3IFhNTEh0dHBSZXF1ZXN0O2Qub3BlbigiR0VUIixlLCEwKTtkLnJlc3BvbnNlVHlwZT0iYXJyYXlidWZmZXIiO2Qub25sb2FkPWZ1bmN0aW9uKCl7MjAwPT1kLnN0YXR1c3x8MD09ZC5zdGF0dXMmJmQucmVzcG9uc2U/YihkLnJlc3BvbnNlKTpjKCl9O2Qub25lcnJvcj1jO2Quc2VuZChudWxsKX07YS5wcmludHx8Y29uc29sZS5sb2cuYmluZChjb25zb2xlKTt2YXIgaGE9YS5wcmludEVycnx8Y29uc29sZS53YXJuLmJpbmQoY29uc29sZSk7Zm9yKGJhIGluIGthKWthLmhhc093blByb3BlcnR5KGJhKSYmKGFbYmFdPQ0Ka2FbYmFdKTtrYT1udWxsO3ZhciBpYTthLndhc21CaW5hcnkmJihpYT1hLndhc21CaW5hcnkpOyJvYmplY3QiIT09dHlwZW9mIFdlYkFzc2VtYmx5JiZmKCJubyBuYXRpdmUgd2FzbSBzdXBwb3J0IGRldGVjdGVkIik7dmFyIHBhLHlhPSExLHdhPSJ1bmRlZmluZWQiIT09dHlwZW9mIFRleHREZWNvZGVyP25ldyBUZXh0RGVjb2RlcigidXRmOCIpOnZvaWQgMCx4YSxZLG1hLFYsQWEsRWE9W10sdGE9W10sRGE9W10sQmE9ITEsY2E9MCx2YT1udWxsLGxhPW51bGw7YS5wcmVsb2FkZWRJbWFnZXM9e307YS5wcmVsb2FkZWRBdWRpb3M9e307dmFyIFE9ImRyYWNvX2RlY29kZXJfZ2x0Zi53YXNtIjtRLnN0YXJ0c1dpdGgoImRhdGE6YXBwbGljYXRpb24vb2N0ZXQtc3RyZWFtO2Jhc2U2NCwiKXx8KFE9ayhRKSk7dmFyIG5hPVtdLHRkPTAsdWQ9e2U6ZnVuY3Rpb24oZSl7cmV0dXJuICRhKGUrMTYpKzE2fSxkOmZ1bmN0aW9uKGUsYixjKXsobmV3IGZhKGUpKS5pbml0KGIsYyk7dGQrKzt0aHJvdyBlOw0KfSxhOmZ1bmN0aW9uKCl7ZigiIil9LGI6ZnVuY3Rpb24oZSxiLGMpe21hLmNvcHlXaXRoaW4oZSxiLGIrYyl9LGM6ZnVuY3Rpb24oZSl7dmFyIGI9bWEubGVuZ3RoO2U+Pj49MDtpZigyMTQ3NDgzNjQ4PGUpcmV0dXJuITE7Zm9yKHZhciBjPTE7ND49YztjKj0yKXt2YXIgZD1iKigxKy4yL2MpO2Q9TWF0aC5taW4oZCxlKzEwMDY2MzI5Nik7dmFyIGc9TWF0aCx1PWcubWluO2Q9TWF0aC5tYXgoZSxkKTswPGQlNjU1MzYmJihkKz02NTUzNi1kJTY1NTM2KTtnPXUuY2FsbChnLDIxNDc0ODM2NDgsZCk7YTp7dHJ5e3BhLmdyb3coZy14YS5ieXRlTGVuZ3RoKzY1NTM1Pj4+MTYpO0EocGEuYnVmZmVyKTt2YXIgWD0xO2JyZWFrIGF9Y2F0Y2godmQpe31YPXZvaWQgMH1pZihYKXJldHVybiEwfXJldHVybiExfX07KGZ1bmN0aW9uKCl7ZnVuY3Rpb24gZShnLHUpe2EuYXNtPWcuZXhwb3J0cztwYT1hLmFzbS5mO0EocGEuYnVmZmVyKTtBYT1hLmFzbS5oO3RhLnVuc2hpZnQoYS5hc20uZyk7Y2EtLTsNCmEubW9uaXRvclJ1bkRlcGVuZGVuY2llcyYmYS5tb25pdG9yUnVuRGVwZW5kZW5jaWVzKGNhKTswPT1jYSYmKG51bGwhPT12YSYmKGNsZWFySW50ZXJ2YWwodmEpLHZhPW51bGwpLGxhJiYoZz1sYSxsYT1udWxsLGcoKSkpfWZ1bmN0aW9uIGIoZyl7ZShnLmluc3RhbmNlKX1mdW5jdGlvbiBjKGcpe3JldHVybiB2KCkudGhlbihmdW5jdGlvbih1KXtyZXR1cm4gV2ViQXNzZW1ibHkuaW5zdGFudGlhdGUodSxkKX0pLnRoZW4oZnVuY3Rpb24odSl7cmV0dXJuIHV9KS50aGVuKGcsZnVuY3Rpb24odSl7aGEoImZhaWxlZCB0byBhc3luY2hyb25vdXNseSBwcmVwYXJlIHdhc206ICIrdSk7Zih1KX0pfXZhciBkPXthOnVkfTtjYSsrO2EubW9uaXRvclJ1bkRlcGVuZGVuY2llcyYmYS5tb25pdG9yUnVuRGVwZW5kZW5jaWVzKGNhKTtpZihhLmluc3RhbnRpYXRlV2FzbSl0cnl7cmV0dXJuIGEuaW5zdGFudGlhdGVXYXNtKGQsZSl9Y2F0Y2goZyl7cmV0dXJuIGhhKCJNb2R1bGUuaW5zdGFudGlhdGVXYXNtIGNhbGxiYWNrIGZhaWxlZCB3aXRoIGVycm9yOiAiKw0KZyksITF9KGZ1bmN0aW9uKCl7cmV0dXJuIGlhfHwiZnVuY3Rpb24iIT09dHlwZW9mIFdlYkFzc2VtYmx5Lmluc3RhbnRpYXRlU3RyZWFtaW5nfHxRLnN0YXJ0c1dpdGgoImRhdGE6YXBwbGljYXRpb24vb2N0ZXQtc3RyZWFtO2Jhc2U2NCwiKXx8US5zdGFydHNXaXRoKCJmaWxlOi8vIil8fCJmdW5jdGlvbiIhPT10eXBlb2YgZmV0Y2g/YyhiKTpmZXRjaChRLHtjcmVkZW50aWFsczoic2FtZS1vcmlnaW4ifSkudGhlbihmdW5jdGlvbihnKXtyZXR1cm4gV2ViQXNzZW1ibHkuaW5zdGFudGlhdGVTdHJlYW1pbmcoZyxkKS50aGVuKGIsZnVuY3Rpb24odSl7aGEoIndhc20gc3RyZWFtaW5nIGNvbXBpbGUgZmFpbGVkOiAiK3UpO2hhKCJmYWxsaW5nIGJhY2sgdG8gQXJyYXlCdWZmZXIgaW5zdGFudGlhdGlvbiIpO3JldHVybiBjKGIpfSl9KX0pKCkuY2F0Y2gocWEpO3JldHVybnt9fSkoKTthLl9fX3dhc21fY2FsbF9jdG9ycz1mdW5jdGlvbigpe3JldHVybihhLl9fX3dhc21fY2FsbF9jdG9ycz0NCmEuYXNtLmcpLmFwcGx5KG51bGwsYXJndW1lbnRzKX07dmFyIGFiPWEuX2Vtc2NyaXB0ZW5fYmluZF9Wb2lkUHRyX19fZGVzdHJveV9fXzA9ZnVuY3Rpb24oKXtyZXR1cm4oYWI9YS5fZW1zY3JpcHRlbl9iaW5kX1ZvaWRQdHJfX19kZXN0cm95X19fMD1hLmFzbS5pKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LEZhPWEuX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyQnVmZmVyX0RlY29kZXJCdWZmZXJfMD1mdW5jdGlvbigpe3JldHVybihGYT1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2RlckJ1ZmZlcl9EZWNvZGVyQnVmZmVyXzA9YS5hc20uaikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxiYj1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2RlckJ1ZmZlcl9Jbml0XzI9ZnVuY3Rpb24oKXtyZXR1cm4oYmI9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJCdWZmZXJfSW5pdF8yPWEuYXNtLmspLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sY2I9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJCdWZmZXJfX19kZXN0cm95X19fMD0NCmZ1bmN0aW9uKCl7cmV0dXJuKGNiPWEuX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyQnVmZmVyX19fZGVzdHJveV9fXzA9YS5hc20ubCkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxHYT1hLl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlVHJhbnNmb3JtRGF0YV9BdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhXzA9ZnVuY3Rpb24oKXtyZXR1cm4oR2E9YS5fZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVRyYW5zZm9ybURhdGFfQXR0cmlidXRlVHJhbnNmb3JtRGF0YV8wPWEuYXNtLm0pLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sZGI9YS5fZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVRyYW5zZm9ybURhdGFfdHJhbnNmb3JtX3R5cGVfMD1mdW5jdGlvbigpe3JldHVybihkYj1hLl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlVHJhbnNmb3JtRGF0YV90cmFuc2Zvcm1fdHlwZV8wPWEuYXNtLm4pLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sZWI9YS5fZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVRyYW5zZm9ybURhdGFfX19kZXN0cm95X19fMD0NCmZ1bmN0aW9uKCl7cmV0dXJuKGViPWEuX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhX19fZGVzdHJveV9fXzA9YS5hc20ubykuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxIYT1hLl9lbXNjcmlwdGVuX2JpbmRfR2VvbWV0cnlBdHRyaWJ1dGVfR2VvbWV0cnlBdHRyaWJ1dGVfMD1mdW5jdGlvbigpe3JldHVybihIYT1hLl9lbXNjcmlwdGVuX2JpbmRfR2VvbWV0cnlBdHRyaWJ1dGVfR2VvbWV0cnlBdHRyaWJ1dGVfMD1hLmFzbS5wKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LGZiPWEuX2Vtc2NyaXB0ZW5fYmluZF9HZW9tZXRyeUF0dHJpYnV0ZV9fX2Rlc3Ryb3lfX18wPWZ1bmN0aW9uKCl7cmV0dXJuKGZiPWEuX2Vtc2NyaXB0ZW5fYmluZF9HZW9tZXRyeUF0dHJpYnV0ZV9fX2Rlc3Ryb3lfX18wPWEuYXNtLnEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sSWE9YS5fZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX1BvaW50QXR0cmlidXRlXzA9ZnVuY3Rpb24oKXtyZXR1cm4oSWE9DQphLl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfUG9pbnRBdHRyaWJ1dGVfMD1hLmFzbS5yKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LGdiPWEuX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9zaXplXzA9ZnVuY3Rpb24oKXtyZXR1cm4oZ2I9YS5fZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX3NpemVfMD1hLmFzbS5zKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LGhiPWEuX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9HZXRBdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhXzA9ZnVuY3Rpb24oKXtyZXR1cm4oaGI9YS5fZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX0dldEF0dHJpYnV0ZVRyYW5zZm9ybURhdGFfMD1hLmFzbS50KS5hcHBseShudWxsLGFyZ3VtZW50cyl9LGliPWEuX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9hdHRyaWJ1dGVfdHlwZV8wPWZ1bmN0aW9uKCl7cmV0dXJuKGliPWEuX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9hdHRyaWJ1dGVfdHlwZV8wPQ0KYS5hc20udSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxqYj1hLl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfZGF0YV90eXBlXzA9ZnVuY3Rpb24oKXtyZXR1cm4oamI9YS5fZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX2RhdGFfdHlwZV8wPWEuYXNtLnYpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sa2I9YS5fZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX251bV9jb21wb25lbnRzXzA9ZnVuY3Rpb24oKXtyZXR1cm4oa2I9YS5fZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX251bV9jb21wb25lbnRzXzA9YS5hc20udykuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxsYj1hLl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfbm9ybWFsaXplZF8wPWZ1bmN0aW9uKCl7cmV0dXJuKGxiPWEuX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9ub3JtYWxpemVkXzA9YS5hc20ueCkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxtYj1hLl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfYnl0ZV9zdHJpZGVfMD0NCmZ1bmN0aW9uKCl7cmV0dXJuKG1iPWEuX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludEF0dHJpYnV0ZV9ieXRlX3N0cmlkZV8wPWEuYXNtLnkpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sbmI9YS5fZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX2J5dGVfb2Zmc2V0XzA9ZnVuY3Rpb24oKXtyZXR1cm4obmI9YS5fZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX2J5dGVfb2Zmc2V0XzA9YS5hc20ueikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxvYj1hLl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRBdHRyaWJ1dGVfdW5pcXVlX2lkXzA9ZnVuY3Rpb24oKXtyZXR1cm4ob2I9YS5fZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX3VuaXF1ZV9pZF8wPWEuYXNtLkEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0scGI9YS5fZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX19fZGVzdHJveV9fXzA9ZnVuY3Rpb24oKXtyZXR1cm4ocGI9YS5fZW1zY3JpcHRlbl9iaW5kX1BvaW50QXR0cmlidXRlX19fZGVzdHJveV9fXzA9DQphLmFzbS5CKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LEphPWEuX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm1fQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtXzA9ZnVuY3Rpb24oKXtyZXR1cm4oSmE9YS5fZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm1fMD1hLmFzbS5DKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LHFiPWEuX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm1fSW5pdEZyb21BdHRyaWJ1dGVfMT1mdW5jdGlvbigpe3JldHVybihxYj1hLl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtX0luaXRGcm9tQXR0cmlidXRlXzE9YS5hc20uRCkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxyYj1hLl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtX3F1YW50aXphdGlvbl9iaXRzXzA9DQpmdW5jdGlvbigpe3JldHVybihyYj1hLl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtX3F1YW50aXphdGlvbl9iaXRzXzA9YS5hc20uRSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxzYj1hLl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtX21pbl92YWx1ZV8xPWZ1bmN0aW9uKCl7cmV0dXJuKHNiPWEuX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVRdWFudGl6YXRpb25UcmFuc2Zvcm1fbWluX3ZhbHVlXzE9YS5hc20uRikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSx0Yj1hLl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlUXVhbnRpemF0aW9uVHJhbnNmb3JtX3JhbmdlXzA9ZnVuY3Rpb24oKXtyZXR1cm4odGI9YS5fZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9yYW5nZV8wPWEuYXNtLkcpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sdWI9YS5fZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9fX2Rlc3Ryb3lfX18wPQ0KZnVuY3Rpb24oKXtyZXR1cm4odWI9YS5fZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybV9fX2Rlc3Ryb3lfX18wPWEuYXNtLkgpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sS2E9YS5fZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm1fQXR0cmlidXRlT2N0YWhlZHJvblRyYW5zZm9ybV8wPWZ1bmN0aW9uKCl7cmV0dXJuKEthPWEuX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtX0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm1fMD1hLmFzbS5JKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LHZiPWEuX2Vtc2NyaXB0ZW5fYmluZF9BdHRyaWJ1dGVPY3RhaGVkcm9uVHJhbnNmb3JtX0luaXRGcm9tQXR0cmlidXRlXzE9ZnVuY3Rpb24oKXtyZXR1cm4odmI9YS5fZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm1fSW5pdEZyb21BdHRyaWJ1dGVfMT1hLmFzbS5KKS5hcHBseShudWxsLA0KYXJndW1lbnRzKX0sd2I9YS5fZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm1fcXVhbnRpemF0aW9uX2JpdHNfMD1mdW5jdGlvbigpe3JldHVybih3Yj1hLl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlT2N0YWhlZHJvblRyYW5zZm9ybV9xdWFudGl6YXRpb25fYml0c18wPWEuYXNtLkspLmFwcGx5KG51bGwsYXJndW1lbnRzKX0seGI9YS5fZW1zY3JpcHRlbl9iaW5kX0F0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm1fX19kZXN0cm95X19fMD1mdW5jdGlvbigpe3JldHVybih4Yj1hLl9lbXNjcmlwdGVuX2JpbmRfQXR0cmlidXRlT2N0YWhlZHJvblRyYW5zZm9ybV9fX2Rlc3Ryb3lfX18wPWEuYXNtLkwpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sTGE9YS5fZW1zY3JpcHRlbl9iaW5kX1BvaW50Q2xvdWRfUG9pbnRDbG91ZF8wPWZ1bmN0aW9uKCl7cmV0dXJuKExhPWEuX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludENsb3VkX1BvaW50Q2xvdWRfMD1hLmFzbS5NKS5hcHBseShudWxsLA0KYXJndW1lbnRzKX0seWI9YS5fZW1zY3JpcHRlbl9iaW5kX1BvaW50Q2xvdWRfbnVtX2F0dHJpYnV0ZXNfMD1mdW5jdGlvbigpe3JldHVybih5Yj1hLl9lbXNjcmlwdGVuX2JpbmRfUG9pbnRDbG91ZF9udW1fYXR0cmlidXRlc18wPWEuYXNtLk4pLmFwcGx5KG51bGwsYXJndW1lbnRzKX0semI9YS5fZW1zY3JpcHRlbl9iaW5kX1BvaW50Q2xvdWRfbnVtX3BvaW50c18wPWZ1bmN0aW9uKCl7cmV0dXJuKHpiPWEuX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludENsb3VkX251bV9wb2ludHNfMD1hLmFzbS5PKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LEFiPWEuX2Vtc2NyaXB0ZW5fYmluZF9Qb2ludENsb3VkX19fZGVzdHJveV9fXzA9ZnVuY3Rpb24oKXtyZXR1cm4oQWI9YS5fZW1zY3JpcHRlbl9iaW5kX1BvaW50Q2xvdWRfX19kZXN0cm95X19fMD1hLmFzbS5QKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LE1hPWEuX2Vtc2NyaXB0ZW5fYmluZF9NZXNoX01lc2hfMD1mdW5jdGlvbigpe3JldHVybihNYT0NCmEuX2Vtc2NyaXB0ZW5fYmluZF9NZXNoX01lc2hfMD1hLmFzbS5RKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LEJiPWEuX2Vtc2NyaXB0ZW5fYmluZF9NZXNoX251bV9mYWNlc18wPWZ1bmN0aW9uKCl7cmV0dXJuKEJiPWEuX2Vtc2NyaXB0ZW5fYmluZF9NZXNoX251bV9mYWNlc18wPWEuYXNtLlIpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sQ2I9YS5fZW1zY3JpcHRlbl9iaW5kX01lc2hfbnVtX2F0dHJpYnV0ZXNfMD1mdW5jdGlvbigpe3JldHVybihDYj1hLl9lbXNjcmlwdGVuX2JpbmRfTWVzaF9udW1fYXR0cmlidXRlc18wPWEuYXNtLlMpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sRGI9YS5fZW1zY3JpcHRlbl9iaW5kX01lc2hfbnVtX3BvaW50c18wPWZ1bmN0aW9uKCl7cmV0dXJuKERiPWEuX2Vtc2NyaXB0ZW5fYmluZF9NZXNoX251bV9wb2ludHNfMD1hLmFzbS5UKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LEViPWEuX2Vtc2NyaXB0ZW5fYmluZF9NZXNoX19fZGVzdHJveV9fXzA9ZnVuY3Rpb24oKXtyZXR1cm4oRWI9DQphLl9lbXNjcmlwdGVuX2JpbmRfTWVzaF9fX2Rlc3Ryb3lfX18wPWEuYXNtLlUpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sTmE9YS5fZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhX01ldGFkYXRhXzA9ZnVuY3Rpb24oKXtyZXR1cm4oTmE9YS5fZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhX01ldGFkYXRhXzA9YS5hc20uVikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxGYj1hLl9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFfX19kZXN0cm95X19fMD1mdW5jdGlvbigpe3JldHVybihGYj1hLl9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFfX19kZXN0cm95X19fMD1hLmFzbS5XKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LEdiPWEuX2Vtc2NyaXB0ZW5fYmluZF9TdGF0dXNfY29kZV8wPWZ1bmN0aW9uKCl7cmV0dXJuKEdiPWEuX2Vtc2NyaXB0ZW5fYmluZF9TdGF0dXNfY29kZV8wPWEuYXNtLlgpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sSGI9YS5fZW1zY3JpcHRlbl9iaW5kX1N0YXR1c19va18wPWZ1bmN0aW9uKCl7cmV0dXJuKEhiPQ0KYS5fZW1zY3JpcHRlbl9iaW5kX1N0YXR1c19va18wPWEuYXNtLlkpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sSWI9YS5fZW1zY3JpcHRlbl9iaW5kX1N0YXR1c19lcnJvcl9tc2dfMD1mdW5jdGlvbigpe3JldHVybihJYj1hLl9lbXNjcmlwdGVuX2JpbmRfU3RhdHVzX2Vycm9yX21zZ18wPWEuYXNtLlopLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sSmI9YS5fZW1zY3JpcHRlbl9iaW5kX1N0YXR1c19fX2Rlc3Ryb3lfX18wPWZ1bmN0aW9uKCl7cmV0dXJuKEpiPWEuX2Vtc2NyaXB0ZW5fYmluZF9TdGF0dXNfX19kZXN0cm95X19fMD1hLmFzbS5fKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LE9hPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0Zsb2F0MzJBcnJheV9EcmFjb0Zsb2F0MzJBcnJheV8wPWZ1bmN0aW9uKCl7cmV0dXJuKE9hPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0Zsb2F0MzJBcnJheV9EcmFjb0Zsb2F0MzJBcnJheV8wPWEuYXNtLiQpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sDQpLYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29GbG9hdDMyQXJyYXlfR2V0VmFsdWVfMT1mdW5jdGlvbigpe3JldHVybihLYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29GbG9hdDMyQXJyYXlfR2V0VmFsdWVfMT1hLmFzbS5hYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxMYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29GbG9hdDMyQXJyYXlfc2l6ZV8wPWZ1bmN0aW9uKCl7cmV0dXJuKExiPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0Zsb2F0MzJBcnJheV9zaXplXzA9YS5hc20uYmEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sTWI9YS5fZW1zY3JpcHRlbl9iaW5kX0RyYWNvRmxvYXQzMkFycmF5X19fZGVzdHJveV9fXzA9ZnVuY3Rpb24oKXtyZXR1cm4oTWI9YS5fZW1zY3JpcHRlbl9iaW5kX0RyYWNvRmxvYXQzMkFycmF5X19fZGVzdHJveV9fXzA9YS5hc20uY2EpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sUGE9YS5fZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50OEFycmF5X0RyYWNvSW50OEFycmF5XzA9DQpmdW5jdGlvbigpe3JldHVybihQYT1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQ4QXJyYXlfRHJhY29JbnQ4QXJyYXlfMD1hLmFzbS5kYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxOYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQ4QXJyYXlfR2V0VmFsdWVfMT1mdW5jdGlvbigpe3JldHVybihOYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQ4QXJyYXlfR2V0VmFsdWVfMT1hLmFzbS5lYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxPYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQ4QXJyYXlfc2l6ZV8wPWZ1bmN0aW9uKCl7cmV0dXJuKE9iPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDhBcnJheV9zaXplXzA9YS5hc20uZmEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sUGI9YS5fZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50OEFycmF5X19fZGVzdHJveV9fXzA9ZnVuY3Rpb24oKXtyZXR1cm4oUGI9YS5fZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50OEFycmF5X19fZGVzdHJveV9fXzA9DQphLmFzbS5nYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxRYT1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50OEFycmF5X0RyYWNvVUludDhBcnJheV8wPWZ1bmN0aW9uKCl7cmV0dXJuKFFhPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQ4QXJyYXlfRHJhY29VSW50OEFycmF5XzA9YS5hc20uaGEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sUWI9YS5fZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDhBcnJheV9HZXRWYWx1ZV8xPWZ1bmN0aW9uKCl7cmV0dXJuKFFiPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQ4QXJyYXlfR2V0VmFsdWVfMT1hLmFzbS5pYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxSYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50OEFycmF5X3NpemVfMD1mdW5jdGlvbigpe3JldHVybihSYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50OEFycmF5X3NpemVfMD1hLmFzbS5qYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxTYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50OEFycmF5X19fZGVzdHJveV9fXzA9DQpmdW5jdGlvbigpe3JldHVybihTYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50OEFycmF5X19fZGVzdHJveV9fXzA9YS5hc20ua2EpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sUmE9YS5fZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MTZBcnJheV9EcmFjb0ludDE2QXJyYXlfMD1mdW5jdGlvbigpe3JldHVybihSYT1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQxNkFycmF5X0RyYWNvSW50MTZBcnJheV8wPWEuYXNtLmxhKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LFRiPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDE2QXJyYXlfR2V0VmFsdWVfMT1mdW5jdGlvbigpe3JldHVybihUYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQxNkFycmF5X0dldFZhbHVlXzE9YS5hc20ubWEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sVWI9YS5fZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MTZBcnJheV9zaXplXzA9ZnVuY3Rpb24oKXtyZXR1cm4oVWI9YS5fZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MTZBcnJheV9zaXplXzA9DQphLmFzbS5uYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxWYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQxNkFycmF5X19fZGVzdHJveV9fXzA9ZnVuY3Rpb24oKXtyZXR1cm4oVmI9YS5fZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MTZBcnJheV9fX2Rlc3Ryb3lfX18wPWEuYXNtLm9hKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LFNhPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQxNkFycmF5X0RyYWNvVUludDE2QXJyYXlfMD1mdW5jdGlvbigpe3JldHVybihTYT1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MTZBcnJheV9EcmFjb1VJbnQxNkFycmF5XzA9YS5hc20ucGEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sV2I9YS5fZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDE2QXJyYXlfR2V0VmFsdWVfMT1mdW5jdGlvbigpe3JldHVybihXYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MTZBcnJheV9HZXRWYWx1ZV8xPWEuYXNtLnFhKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LA0KWGI9YS5fZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDE2QXJyYXlfc2l6ZV8wPWZ1bmN0aW9uKCl7cmV0dXJuKFhiPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQxNkFycmF5X3NpemVfMD1hLmFzbS5yYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxZYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MTZBcnJheV9fX2Rlc3Ryb3lfX18wPWZ1bmN0aW9uKCl7cmV0dXJuKFliPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQxNkFycmF5X19fZGVzdHJveV9fXzA9YS5hc20uc2EpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sVGE9YS5fZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MzJBcnJheV9EcmFjb0ludDMyQXJyYXlfMD1mdW5jdGlvbigpe3JldHVybihUYT1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQzMkFycmF5X0RyYWNvSW50MzJBcnJheV8wPWEuYXNtLnRhKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LFpiPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDMyQXJyYXlfR2V0VmFsdWVfMT0NCmZ1bmN0aW9uKCl7cmV0dXJuKFpiPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb0ludDMyQXJyYXlfR2V0VmFsdWVfMT1hLmFzbS51YSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSwkYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQzMkFycmF5X3NpemVfMD1mdW5jdGlvbigpe3JldHVybigkYj1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQzMkFycmF5X3NpemVfMD1hLmFzbS52YSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxhYz1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29JbnQzMkFycmF5X19fZGVzdHJveV9fXzA9ZnVuY3Rpb24oKXtyZXR1cm4oYWM9YS5fZW1zY3JpcHRlbl9iaW5kX0RyYWNvSW50MzJBcnJheV9fX2Rlc3Ryb3lfX18wPWEuYXNtLndhKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LFVhPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQzMkFycmF5X0RyYWNvVUludDMyQXJyYXlfMD1mdW5jdGlvbigpe3JldHVybihVYT1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MzJBcnJheV9EcmFjb1VJbnQzMkFycmF5XzA9DQphLmFzbS54YSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxiYz1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MzJBcnJheV9HZXRWYWx1ZV8xPWZ1bmN0aW9uKCl7cmV0dXJuKGJjPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQzMkFycmF5X0dldFZhbHVlXzE9YS5hc20ueWEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sY2M9YS5fZW1zY3JpcHRlbl9iaW5kX0RyYWNvVUludDMyQXJyYXlfc2l6ZV8wPWZ1bmN0aW9uKCl7cmV0dXJuKGNjPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQzMkFycmF5X3NpemVfMD1hLmFzbS56YSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxkYz1hLl9lbXNjcmlwdGVuX2JpbmRfRHJhY29VSW50MzJBcnJheV9fX2Rlc3Ryb3lfX18wPWZ1bmN0aW9uKCl7cmV0dXJuKGRjPWEuX2Vtc2NyaXB0ZW5fYmluZF9EcmFjb1VJbnQzMkFycmF5X19fZGVzdHJveV9fXzA9YS5hc20uQWEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sVmE9YS5fZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9NZXRhZGF0YVF1ZXJpZXJfMD0NCmZ1bmN0aW9uKCl7cmV0dXJuKFZhPWEuX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfTWV0YWRhdGFRdWVyaWVyXzA9YS5hc20uQmEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sZWM9YS5fZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9IYXNFbnRyeV8yPWZ1bmN0aW9uKCl7cmV0dXJuKGVjPWEuX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfSGFzRW50cnlfMj1hLmFzbS5DYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxmYz1hLl9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX0dldEludEVudHJ5XzI9ZnVuY3Rpb24oKXtyZXR1cm4oZmM9YS5fZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9HZXRJbnRFbnRyeV8yPWEuYXNtLkRhKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LGdjPWEuX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfR2V0SW50RW50cnlBcnJheV8zPWZ1bmN0aW9uKCl7cmV0dXJuKGdjPWEuX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfR2V0SW50RW50cnlBcnJheV8zPQ0KYS5hc20uRWEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0saGM9YS5fZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9HZXREb3VibGVFbnRyeV8yPWZ1bmN0aW9uKCl7cmV0dXJuKGhjPWEuX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfR2V0RG91YmxlRW50cnlfMj1hLmFzbS5GYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxpYz1hLl9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX0dldFN0cmluZ0VudHJ5XzI9ZnVuY3Rpb24oKXtyZXR1cm4oaWM9YS5fZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9HZXRTdHJpbmdFbnRyeV8yPWEuYXNtLkdhKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LGpjPWEuX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfTnVtRW50cmllc18xPWZ1bmN0aW9uKCl7cmV0dXJuKGpjPWEuX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfTnVtRW50cmllc18xPWEuYXNtLkhhKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LA0Ka2M9YS5fZW1zY3JpcHRlbl9iaW5kX01ldGFkYXRhUXVlcmllcl9HZXRFbnRyeU5hbWVfMj1mdW5jdGlvbigpe3JldHVybihrYz1hLl9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX0dldEVudHJ5TmFtZV8yPWEuYXNtLklhKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LGxjPWEuX2Vtc2NyaXB0ZW5fYmluZF9NZXRhZGF0YVF1ZXJpZXJfX19kZXN0cm95X19fMD1mdW5jdGlvbigpe3JldHVybihsYz1hLl9lbXNjcmlwdGVuX2JpbmRfTWV0YWRhdGFRdWVyaWVyX19fZGVzdHJveV9fXzA9YS5hc20uSmEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sV2E9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfRGVjb2Rlcl8wPWZ1bmN0aW9uKCl7cmV0dXJuKFdhPWEuX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0RlY29kZXJfMD1hLmFzbS5LYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxtYz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVBcnJheVRvUG9pbnRDbG91ZF8zPWZ1bmN0aW9uKCl7cmV0dXJuKG1jPQ0KYS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfRGVjb2RlQXJyYXlUb1BvaW50Q2xvdWRfMz1hLmFzbS5MYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxuYz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9EZWNvZGVBcnJheVRvTWVzaF8zPWZ1bmN0aW9uKCl7cmV0dXJuKG5jPWEuX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0RlY29kZUFycmF5VG9NZXNoXzM9YS5hc20uTWEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sb2M9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSWRfMj1mdW5jdGlvbigpe3JldHVybihvYz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVJZF8yPWEuYXNtLk5hKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LHBjPWEuX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUlkQnlOYW1lXzI9ZnVuY3Rpb24oKXtyZXR1cm4ocGM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSWRCeU5hbWVfMj0NCmEuYXNtLk9hKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LHFjPWEuX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUlkQnlNZXRhZGF0YUVudHJ5XzM9ZnVuY3Rpb24oKXtyZXR1cm4ocWM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSWRCeU1ldGFkYXRhRW50cnlfMz1hLmFzbS5QYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxyYz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVfMj1mdW5jdGlvbigpe3JldHVybihyYz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVfMj1hLmFzbS5RYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxzYz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVCeVVuaXF1ZUlkXzI9ZnVuY3Rpb24oKXtyZXR1cm4oc2M9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlQnlVbmlxdWVJZF8yPWEuYXNtLlJhKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LA0KdGM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0TWV0YWRhdGFfMT1mdW5jdGlvbigpe3JldHVybih0Yz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRNZXRhZGF0YV8xPWEuYXNtLlNhKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LHVjPWEuX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZU1ldGFkYXRhXzI9ZnVuY3Rpb24oKXtyZXR1cm4odWM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlTWV0YWRhdGFfMj1hLmFzbS5UYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSx2Yz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRGYWNlRnJvbU1lc2hfMz1mdW5jdGlvbigpe3JldHVybih2Yz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRGYWNlRnJvbU1lc2hfMz1hLmFzbS5VYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSx3Yz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRUcmlhbmdsZVN0cmlwc0Zyb21NZXNoXzI9DQpmdW5jdGlvbigpe3JldHVybih3Yz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRUcmlhbmdsZVN0cmlwc0Zyb21NZXNoXzI9YS5hc20uVmEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0seGM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0VHJpYW5nbGVzVUludDE2QXJyYXlfMz1mdW5jdGlvbigpe3JldHVybih4Yz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRUcmlhbmdsZXNVSW50MTZBcnJheV8zPWEuYXNtLldhKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LHljPWEuX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldFRyaWFuZ2xlc1VJbnQzMkFycmF5XzM9ZnVuY3Rpb24oKXtyZXR1cm4oeWM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0VHJpYW5nbGVzVUludDMyQXJyYXlfMz1hLmFzbS5YYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSx6Yz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVGbG9hdF8zPWZ1bmN0aW9uKCl7cmV0dXJuKHpjPQ0KYS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlRmxvYXRfMz1hLmFzbS5ZYSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxBYz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVGbG9hdEZvckFsbFBvaW50c18zPWZ1bmN0aW9uKCl7cmV0dXJuKEFjPWEuX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUZsb2F0Rm9yQWxsUG9pbnRzXzM9YS5hc20uWmEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sQmM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSW50Rm9yQWxsUG9pbnRzXzM9ZnVuY3Rpb24oKXtyZXR1cm4oQmM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSW50Rm9yQWxsUG9pbnRzXzM9YS5hc20uX2EpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sQ2M9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSW50OEZvckFsbFBvaW50c18zPWZ1bmN0aW9uKCl7cmV0dXJuKENjPQ0KYS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlSW50OEZvckFsbFBvaW50c18zPWEuYXNtLiRhKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LERjPWEuX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZVVJbnQ4Rm9yQWxsUG9pbnRzXzM9ZnVuY3Rpb24oKXtyZXR1cm4oRGM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlVUludDhGb3JBbGxQb2ludHNfMz1hLmFzbS5hYikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxFYz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVJbnQxNkZvckFsbFBvaW50c18zPWZ1bmN0aW9uKCl7cmV0dXJuKEVjPWEuX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUludDE2Rm9yQWxsUG9pbnRzXzM9YS5hc20uYmIpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sRmM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlVUludDE2Rm9yQWxsUG9pbnRzXzM9DQpmdW5jdGlvbigpe3JldHVybihGYz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVVSW50MTZGb3JBbGxQb2ludHNfMz1hLmFzbS5jYikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxHYz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVJbnQzMkZvckFsbFBvaW50c18zPWZ1bmN0aW9uKCl7cmV0dXJuKEdjPWEuX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEF0dHJpYnV0ZUludDMyRm9yQWxsUG9pbnRzXzM9YS5hc20uZGIpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sSGM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlVUludDMyRm9yQWxsUG9pbnRzXzM9ZnVuY3Rpb24oKXtyZXR1cm4oSGM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlVUludDMyRm9yQWxsUG9pbnRzXzM9YS5hc20uZWIpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sSWM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfR2V0QXR0cmlidXRlRGF0YUFycmF5Rm9yQWxsUG9pbnRzXzU9DQpmdW5jdGlvbigpe3JldHVybihJYz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9HZXRBdHRyaWJ1dGVEYXRhQXJyYXlGb3JBbGxQb2ludHNfNT1hLmFzbS5mYikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxKYz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9Ta2lwQXR0cmlidXRlVHJhbnNmb3JtXzE9ZnVuY3Rpb24oKXtyZXR1cm4oSmM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfU2tpcEF0dHJpYnV0ZVRyYW5zZm9ybV8xPWEuYXNtLmdiKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LEtjPWEuX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEVuY29kZWRHZW9tZXRyeVR5cGVfRGVwcmVjYXRlZF8xPWZ1bmN0aW9uKCl7cmV0dXJuKEtjPWEuX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0dldEVuY29kZWRHZW9tZXRyeVR5cGVfRGVwcmVjYXRlZF8xPWEuYXNtLmhiKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LExjPWEuX2Vtc2NyaXB0ZW5fYmluZF9EZWNvZGVyX0RlY29kZUJ1ZmZlclRvUG9pbnRDbG91ZF8yPQ0KZnVuY3Rpb24oKXtyZXR1cm4oTGM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfRGVjb2RlQnVmZmVyVG9Qb2ludENsb3VkXzI9YS5hc20uaWIpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sTWM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfRGVjb2RlQnVmZmVyVG9NZXNoXzI9ZnVuY3Rpb24oKXtyZXR1cm4oTWM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfRGVjb2RlQnVmZmVyVG9NZXNoXzI9YS5hc20uamIpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sTmM9YS5fZW1zY3JpcHRlbl9iaW5kX0RlY29kZXJfX19kZXN0cm95X19fMD1mdW5jdGlvbigpe3JldHVybihOYz1hLl9lbXNjcmlwdGVuX2JpbmRfRGVjb2Rlcl9fX2Rlc3Ryb3lfX18wPWEuYXNtLmtiKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LE9jPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19BdHRyaWJ1dGVUcmFuc2Zvcm1UeXBlX0FUVFJJQlVURV9JTlZBTElEX1RSQU5TRk9STT1mdW5jdGlvbigpe3JldHVybihPYz1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fQXR0cmlidXRlVHJhbnNmb3JtVHlwZV9BVFRSSUJVVEVfSU5WQUxJRF9UUkFOU0ZPUk09DQphLmFzbS5sYikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxQYz1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fQXR0cmlidXRlVHJhbnNmb3JtVHlwZV9BVFRSSUJVVEVfTk9fVFJBTlNGT1JNPWZ1bmN0aW9uKCl7cmV0dXJuKFBjPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19BdHRyaWJ1dGVUcmFuc2Zvcm1UeXBlX0FUVFJJQlVURV9OT19UUkFOU0ZPUk09YS5hc20ubWIpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sUWM9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0F0dHJpYnV0ZVRyYW5zZm9ybVR5cGVfQVRUUklCVVRFX1FVQU5USVpBVElPTl9UUkFOU0ZPUk09ZnVuY3Rpb24oKXtyZXR1cm4oUWM9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0F0dHJpYnV0ZVRyYW5zZm9ybVR5cGVfQVRUUklCVVRFX1FVQU5USVpBVElPTl9UUkFOU0ZPUk09YS5hc20ubmIpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sUmM9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0F0dHJpYnV0ZVRyYW5zZm9ybVR5cGVfQVRUUklCVVRFX09DVEFIRURST05fVFJBTlNGT1JNPQ0KZnVuY3Rpb24oKXtyZXR1cm4oUmM9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0F0dHJpYnV0ZVRyYW5zZm9ybVR5cGVfQVRUUklCVVRFX09DVEFIRURST05fVFJBTlNGT1JNPWEuYXNtLm9iKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LFNjPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19HZW9tZXRyeUF0dHJpYnV0ZV9UeXBlX0lOVkFMSUQ9ZnVuY3Rpb24oKXtyZXR1cm4oU2M9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0dlb21ldHJ5QXR0cmlidXRlX1R5cGVfSU5WQUxJRD1hLmFzbS5wYikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxUYz1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9QT1NJVElPTj1mdW5jdGlvbigpe3JldHVybihUYz1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9QT1NJVElPTj1hLmFzbS5xYikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxVYz1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9OT1JNQUw9DQpmdW5jdGlvbigpe3JldHVybihVYz1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9OT1JNQUw9YS5hc20ucmIpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sVmM9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0dlb21ldHJ5QXR0cmlidXRlX1R5cGVfQ09MT1I9ZnVuY3Rpb24oKXtyZXR1cm4oVmM9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0dlb21ldHJ5QXR0cmlidXRlX1R5cGVfQ09MT1I9YS5hc20uc2IpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sV2M9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0dlb21ldHJ5QXR0cmlidXRlX1R5cGVfVEVYX0NPT1JEPWZ1bmN0aW9uKCl7cmV0dXJuKFdjPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19HZW9tZXRyeUF0dHJpYnV0ZV9UeXBlX1RFWF9DT09SRD1hLmFzbS50YikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxYYz1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fR2VvbWV0cnlBdHRyaWJ1dGVfVHlwZV9HRU5FUklDPQ0KZnVuY3Rpb24oKXtyZXR1cm4oWGM9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0dlb21ldHJ5QXR0cmlidXRlX1R5cGVfR0VORVJJQz1hLmFzbS51YikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxZYz1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRW5jb2RlZEdlb21ldHJ5VHlwZV9JTlZBTElEX0dFT01FVFJZX1RZUEU9ZnVuY3Rpb24oKXtyZXR1cm4oWWM9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0VuY29kZWRHZW9tZXRyeVR5cGVfSU5WQUxJRF9HRU9NRVRSWV9UWVBFPWEuYXNtLnZiKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LFpjPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19FbmNvZGVkR2VvbWV0cnlUeXBlX1BPSU5UX0NMT1VEPWZ1bmN0aW9uKCl7cmV0dXJuKFpjPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19FbmNvZGVkR2VvbWV0cnlUeXBlX1BPSU5UX0NMT1VEPWEuYXNtLndiKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LCRjPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19FbmNvZGVkR2VvbWV0cnlUeXBlX1RSSUFOR1VMQVJfTUVTSD0NCmZ1bmN0aW9uKCl7cmV0dXJuKCRjPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19FbmNvZGVkR2VvbWV0cnlUeXBlX1RSSUFOR1VMQVJfTUVTSD1hLmFzbS54YikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxhZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfSU5WQUxJRD1mdW5jdGlvbigpe3JldHVybihhZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfSU5WQUxJRD1hLmFzbS55YikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxiZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfSU5UOD1mdW5jdGlvbigpe3JldHVybihiZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfSU5UOD1hLmFzbS56YikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxjZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfVUlOVDg9ZnVuY3Rpb24oKXtyZXR1cm4oY2Q9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX1VJTlQ4PQ0KYS5hc20uQWIpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sZGQ9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX0lOVDE2PWZ1bmN0aW9uKCl7cmV0dXJuKGRkPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9JTlQxNj1hLmFzbS5CYikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxlZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfVUlOVDE2PWZ1bmN0aW9uKCl7cmV0dXJuKGVkPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9VSU5UMTY9YS5hc20uQ2IpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sZmQ9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX0lOVDMyPWZ1bmN0aW9uKCl7cmV0dXJuKGZkPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9JTlQzMj1hLmFzbS5EYikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxnZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfVUlOVDMyPQ0KZnVuY3Rpb24oKXtyZXR1cm4oZ2Q9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX1VJTlQzMj1hLmFzbS5FYikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxoZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfSU5UNjQ9ZnVuY3Rpb24oKXtyZXR1cm4oaGQ9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX0lOVDY0PWEuYXNtLkZiKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LGlkPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9VSU5UNjQ9ZnVuY3Rpb24oKXtyZXR1cm4oaWQ9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX0RhdGFUeXBlX0RUX1VJTlQ2ND1hLmFzbS5HYikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxqZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfRkxPQVQzMj1mdW5jdGlvbigpe3JldHVybihqZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfRkxPQVQzMj1hLmFzbS5IYikuYXBwbHkobnVsbCwNCmFyZ3VtZW50cyl9LGtkPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9GTE9BVDY0PWZ1bmN0aW9uKCl7cmV0dXJuKGtkPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9GTE9BVDY0PWEuYXNtLkliKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LGxkPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9CT09MPWZ1bmN0aW9uKCl7cmV0dXJuKGxkPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9CT09MPWEuYXNtLkpiKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LG1kPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19EYXRhVHlwZV9EVF9UWVBFU19DT1VOVD1mdW5jdGlvbigpe3JldHVybihtZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fRGF0YVR5cGVfRFRfVFlQRVNfQ09VTlQ9YS5hc20uS2IpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sbmQ9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfT0s9ZnVuY3Rpb24oKXtyZXR1cm4obmQ9DQphLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fU3RhdHVzQ29kZV9PSz1hLmFzbS5MYikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxvZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fU3RhdHVzQ29kZV9EUkFDT19FUlJPUj1mdW5jdGlvbigpe3JldHVybihvZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fU3RhdHVzQ29kZV9EUkFDT19FUlJPUj1hLmFzbS5NYikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxwZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fU3RhdHVzQ29kZV9JT19FUlJPUj1mdW5jdGlvbigpe3JldHVybihwZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fU3RhdHVzQ29kZV9JT19FUlJPUj1hLmFzbS5OYikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxxZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fU3RhdHVzQ29kZV9JTlZBTElEX1BBUkFNRVRFUj1mdW5jdGlvbigpe3JldHVybihxZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fU3RhdHVzQ29kZV9JTlZBTElEX1BBUkFNRVRFUj0NCmEuYXNtLk9iKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LHJkPWEuX2Vtc2NyaXB0ZW5fZW51bV9kcmFjb19TdGF0dXNDb2RlX1VOU1VQUE9SVEVEX1ZFUlNJT049ZnVuY3Rpb24oKXtyZXR1cm4ocmQ9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfVU5TVVBQT1JURURfVkVSU0lPTj1hLmFzbS5QYikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSxzZD1hLl9lbXNjcmlwdGVuX2VudW1fZHJhY29fU3RhdHVzQ29kZV9VTktOT1dOX1ZFUlNJT049ZnVuY3Rpb24oKXtyZXR1cm4oc2Q9YS5fZW1zY3JpcHRlbl9lbnVtX2RyYWNvX1N0YXR1c0NvZGVfVU5LTk9XTl9WRVJTSU9OPWEuYXNtLlFiKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LCRhPWEuX21hbGxvYz1mdW5jdGlvbigpe3JldHVybigkYT1hLl9tYWxsb2M9YS5hc20uUmIpLmFwcGx5KG51bGwsYXJndW1lbnRzKX07YS5fZnJlZT1mdW5jdGlvbigpe3JldHVybihhLl9mcmVlPWEuYXNtLlNiKS5hcHBseShudWxsLGFyZ3VtZW50cyl9Ow0KdmFyIG9hO2xhPWZ1bmN0aW9uIGIoKXtvYXx8UCgpO29hfHwobGE9Yil9O2EucnVuPVA7aWYoYS5wcmVJbml0KWZvcigiZnVuY3Rpb24iPT10eXBlb2YgYS5wcmVJbml0JiYoYS5wcmVJbml0PVthLnByZUluaXRdKTswPGEucHJlSW5pdC5sZW5ndGg7KWEucHJlSW5pdC5wb3AoKSgpO1AoKTt0LnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKHQucHJvdG90eXBlKTt0LnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj10O3QucHJvdG90eXBlLl9fY2xhc3NfXz10O3QuX19jYWNoZV9fPXt9O2EuV3JhcHBlck9iamVjdD10O2EuZ2V0Q2FjaGU9eDthLndyYXBQb2ludGVyPUQ7YS5jYXN0T2JqZWN0PWZ1bmN0aW9uKGIsYyl7cmV0dXJuIEQoYi5wdHIsYyl9O2EuTlVMTD1EKDApO2EuZGVzdHJveT1mdW5jdGlvbihiKXtpZighYi5fX2Rlc3Ryb3lfXyl0aHJvdyJFcnJvcjogQ2Fubm90IGRlc3Ryb3kgb2JqZWN0LiAoRGlkIHlvdSBjcmVhdGUgaXQgeW91cnNlbGY/KSI7Yi5fX2Rlc3Ryb3lfXygpO2RlbGV0ZSB4KGIuX19jbGFzc19fKVtiLnB0cl19Ow0KYS5jb21wYXJlPWZ1bmN0aW9uKGIsYyl7cmV0dXJuIGIucHRyPT09Yy5wdHJ9O2EuZ2V0UG9pbnRlcj1mdW5jdGlvbihiKXtyZXR1cm4gYi5wdHJ9O2EuZ2V0Q2xhc3M9ZnVuY3Rpb24oYil7cmV0dXJuIGIuX19jbGFzc19ffTt2YXIgcj17YnVmZmVyOjAsc2l6ZTowLHBvczowLHRlbXBzOltdLG5lZWRlZDowLHByZXBhcmU6ZnVuY3Rpb24oKXtpZihyLm5lZWRlZCl7Zm9yKHZhciBiPTA7YjxyLnRlbXBzLmxlbmd0aDtiKyspYS5fZnJlZShyLnRlbXBzW2JdKTtyLnRlbXBzLmxlbmd0aD0wO2EuX2ZyZWUoci5idWZmZXIpO3IuYnVmZmVyPTA7ci5zaXplKz1yLm5lZWRlZDtyLm5lZWRlZD0wfXIuYnVmZmVyfHwoci5zaXplKz0xMjgsci5idWZmZXI9YS5fbWFsbG9jKHIuc2l6ZSkscShyLmJ1ZmZlcikpO3IucG9zPTB9LGFsbG9jOmZ1bmN0aW9uKGIsYyl7cShyLmJ1ZmZlcik7Yj1iLmxlbmd0aCpjLkJZVEVTX1BFUl9FTEVNRU5UO2I9Yis3Ji04O3IucG9zK2I+PXIuc2l6ZT8ocSgwPGIpLA0Kci5uZWVkZWQrPWIsYz1hLl9tYWxsb2MoYiksci50ZW1wcy5wdXNoKGMpKTooYz1yLmJ1ZmZlcityLnBvcyxyLnBvcys9Yik7cmV0dXJuIGN9LGNvcHk6ZnVuY3Rpb24oYixjLGQpe2Q+Pj49MDtzd2l0Y2goYy5CWVRFU19QRVJfRUxFTUVOVCl7Y2FzZSAyOmQ+Pj49MTticmVhaztjYXNlIDQ6ZD4+Pj0yO2JyZWFrO2Nhc2UgODpkPj4+PTN9Zm9yKHZhciBnPTA7ZzxiLmxlbmd0aDtnKyspY1tkK2ddPWJbZ119fTtaLnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKHQucHJvdG90eXBlKTtaLnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1aO1oucHJvdG90eXBlLl9fY2xhc3NfXz1aO1ouX19jYWNoZV9fPXt9O2EuVm9pZFB0cj1aO1oucHJvdG90eXBlLl9fZGVzdHJveV9fPVoucHJvdG90eXBlLl9fZGVzdHJveV9fPWZ1bmN0aW9uKCl7YWIodGhpcy5wdHIpfTtTLnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKHQucHJvdG90eXBlKTtTLnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1TO1MucHJvdG90eXBlLl9fY2xhc3NfXz0NClM7Uy5fX2NhY2hlX189e307YS5EZWNvZGVyQnVmZmVyPVM7Uy5wcm90b3R5cGUuSW5pdD1TLnByb3RvdHlwZS5Jbml0PWZ1bmN0aW9uKGIsYyl7dmFyIGQ9dGhpcy5wdHI7ci5wcmVwYXJlKCk7Im9iamVjdCI9PXR5cGVvZiBiJiYoYj11YShiKSk7YyYmIm9iamVjdCI9PT10eXBlb2YgYyYmKGM9Yy5wdHIpO2JiKGQsYixjKX07Uy5wcm90b3R5cGUuX19kZXN0cm95X189Uy5wcm90b3R5cGUuX19kZXN0cm95X189ZnVuY3Rpb24oKXtjYih0aGlzLnB0cil9O1IucHJvdG90eXBlPU9iamVjdC5jcmVhdGUodC5wcm90b3R5cGUpO1IucHJvdG90eXBlLmNvbnN0cnVjdG9yPVI7Ui5wcm90b3R5cGUuX19jbGFzc19fPVI7Ui5fX2NhY2hlX189e307YS5BdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhPVI7Ui5wcm90b3R5cGUudHJhbnNmb3JtX3R5cGU9Ui5wcm90b3R5cGUudHJhbnNmb3JtX3R5cGU9ZnVuY3Rpb24oKXtyZXR1cm4gZGIodGhpcy5wdHIpfTtSLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1SLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz0NCmZ1bmN0aW9uKCl7ZWIodGhpcy5wdHIpfTtXLnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKHQucHJvdG90eXBlKTtXLnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1XO1cucHJvdG90eXBlLl9fY2xhc3NfXz1XO1cuX19jYWNoZV9fPXt9O2EuR2VvbWV0cnlBdHRyaWJ1dGU9VztXLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1XLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1mdW5jdGlvbigpe2ZiKHRoaXMucHRyKX07dy5wcm90b3R5cGU9T2JqZWN0LmNyZWF0ZSh0LnByb3RvdHlwZSk7dy5wcm90b3R5cGUuY29uc3RydWN0b3I9dzt3LnByb3RvdHlwZS5fX2NsYXNzX189dzt3Ll9fY2FjaGVfXz17fTthLlBvaW50QXR0cmlidXRlPXc7dy5wcm90b3R5cGUuc2l6ZT13LnByb3RvdHlwZS5zaXplPWZ1bmN0aW9uKCl7cmV0dXJuIGdiKHRoaXMucHRyKX07dy5wcm90b3R5cGUuR2V0QXR0cmlidXRlVHJhbnNmb3JtRGF0YT13LnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVUcmFuc2Zvcm1EYXRhPWZ1bmN0aW9uKCl7cmV0dXJuIEQoaGIodGhpcy5wdHIpLA0KUil9O3cucHJvdG90eXBlLmF0dHJpYnV0ZV90eXBlPXcucHJvdG90eXBlLmF0dHJpYnV0ZV90eXBlPWZ1bmN0aW9uKCl7cmV0dXJuIGliKHRoaXMucHRyKX07dy5wcm90b3R5cGUuZGF0YV90eXBlPXcucHJvdG90eXBlLmRhdGFfdHlwZT1mdW5jdGlvbigpe3JldHVybiBqYih0aGlzLnB0cil9O3cucHJvdG90eXBlLm51bV9jb21wb25lbnRzPXcucHJvdG90eXBlLm51bV9jb21wb25lbnRzPWZ1bmN0aW9uKCl7cmV0dXJuIGtiKHRoaXMucHRyKX07dy5wcm90b3R5cGUubm9ybWFsaXplZD13LnByb3RvdHlwZS5ub3JtYWxpemVkPWZ1bmN0aW9uKCl7cmV0dXJuISFsYih0aGlzLnB0cil9O3cucHJvdG90eXBlLmJ5dGVfc3RyaWRlPXcucHJvdG90eXBlLmJ5dGVfc3RyaWRlPWZ1bmN0aW9uKCl7cmV0dXJuIG1iKHRoaXMucHRyKX07dy5wcm90b3R5cGUuYnl0ZV9vZmZzZXQ9dy5wcm90b3R5cGUuYnl0ZV9vZmZzZXQ9ZnVuY3Rpb24oKXtyZXR1cm4gbmIodGhpcy5wdHIpfTt3LnByb3RvdHlwZS51bmlxdWVfaWQ9DQp3LnByb3RvdHlwZS51bmlxdWVfaWQ9ZnVuY3Rpb24oKXtyZXR1cm4gb2IodGhpcy5wdHIpfTt3LnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz13LnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1mdW5jdGlvbigpe3BiKHRoaXMucHRyKX07Qy5wcm90b3R5cGU9T2JqZWN0LmNyZWF0ZSh0LnByb3RvdHlwZSk7Qy5wcm90b3R5cGUuY29uc3RydWN0b3I9QztDLnByb3RvdHlwZS5fX2NsYXNzX189QztDLl9fY2FjaGVfXz17fTthLkF0dHJpYnV0ZVF1YW50aXphdGlvblRyYW5zZm9ybT1DO0MucHJvdG90eXBlLkluaXRGcm9tQXR0cmlidXRlPUMucHJvdG90eXBlLkluaXRGcm9tQXR0cmlidXRlPWZ1bmN0aW9uKGIpe3ZhciBjPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtyZXR1cm4hIXFiKGMsYil9O0MucHJvdG90eXBlLnF1YW50aXphdGlvbl9iaXRzPUMucHJvdG90eXBlLnF1YW50aXphdGlvbl9iaXRzPWZ1bmN0aW9uKCl7cmV0dXJuIHJiKHRoaXMucHRyKX07Qy5wcm90b3R5cGUubWluX3ZhbHVlPQ0KQy5wcm90b3R5cGUubWluX3ZhbHVlPWZ1bmN0aW9uKGIpe3ZhciBjPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtyZXR1cm4gc2IoYyxiKX07Qy5wcm90b3R5cGUucmFuZ2U9Qy5wcm90b3R5cGUucmFuZ2U9ZnVuY3Rpb24oKXtyZXR1cm4gdGIodGhpcy5wdHIpfTtDLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1DLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1mdW5jdGlvbigpe3ViKHRoaXMucHRyKX07Ry5wcm90b3R5cGU9T2JqZWN0LmNyZWF0ZSh0LnByb3RvdHlwZSk7Ry5wcm90b3R5cGUuY29uc3RydWN0b3I9RztHLnByb3RvdHlwZS5fX2NsYXNzX189RztHLl9fY2FjaGVfXz17fTthLkF0dHJpYnV0ZU9jdGFoZWRyb25UcmFuc2Zvcm09RztHLnByb3RvdHlwZS5Jbml0RnJvbUF0dHJpYnV0ZT1HLnByb3RvdHlwZS5Jbml0RnJvbUF0dHJpYnV0ZT1mdW5jdGlvbihiKXt2YXIgYz10aGlzLnB0cjtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7cmV0dXJuISF2YihjLA0KYil9O0cucHJvdG90eXBlLnF1YW50aXphdGlvbl9iaXRzPUcucHJvdG90eXBlLnF1YW50aXphdGlvbl9iaXRzPWZ1bmN0aW9uKCl7cmV0dXJuIHdiKHRoaXMucHRyKX07Ry5wcm90b3R5cGUuX19kZXN0cm95X189Ry5wcm90b3R5cGUuX19kZXN0cm95X189ZnVuY3Rpb24oKXt4Yih0aGlzLnB0cil9O0gucHJvdG90eXBlPU9iamVjdC5jcmVhdGUodC5wcm90b3R5cGUpO0gucHJvdG90eXBlLmNvbnN0cnVjdG9yPUg7SC5wcm90b3R5cGUuX19jbGFzc19fPUg7SC5fX2NhY2hlX189e307YS5Qb2ludENsb3VkPUg7SC5wcm90b3R5cGUubnVtX2F0dHJpYnV0ZXM9SC5wcm90b3R5cGUubnVtX2F0dHJpYnV0ZXM9ZnVuY3Rpb24oKXtyZXR1cm4geWIodGhpcy5wdHIpfTtILnByb3RvdHlwZS5udW1fcG9pbnRzPUgucHJvdG90eXBlLm51bV9wb2ludHM9ZnVuY3Rpb24oKXtyZXR1cm4gemIodGhpcy5wdHIpfTtILnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1ILnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1mdW5jdGlvbigpe0FiKHRoaXMucHRyKX07DQpFLnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKHQucHJvdG90eXBlKTtFLnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1FO0UucHJvdG90eXBlLl9fY2xhc3NfXz1FO0UuX19jYWNoZV9fPXt9O2EuTWVzaD1FO0UucHJvdG90eXBlLm51bV9mYWNlcz1FLnByb3RvdHlwZS5udW1fZmFjZXM9ZnVuY3Rpb24oKXtyZXR1cm4gQmIodGhpcy5wdHIpfTtFLnByb3RvdHlwZS5udW1fYXR0cmlidXRlcz1FLnByb3RvdHlwZS5udW1fYXR0cmlidXRlcz1mdW5jdGlvbigpe3JldHVybiBDYih0aGlzLnB0cil9O0UucHJvdG90eXBlLm51bV9wb2ludHM9RS5wcm90b3R5cGUubnVtX3BvaW50cz1mdW5jdGlvbigpe3JldHVybiBEYih0aGlzLnB0cil9O0UucHJvdG90eXBlLl9fZGVzdHJveV9fPUUucHJvdG90eXBlLl9fZGVzdHJveV9fPWZ1bmN0aW9uKCl7RWIodGhpcy5wdHIpfTtULnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKHQucHJvdG90eXBlKTtULnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1UO1QucHJvdG90eXBlLl9fY2xhc3NfXz0NClQ7VC5fX2NhY2hlX189e307YS5NZXRhZGF0YT1UO1QucHJvdG90eXBlLl9fZGVzdHJveV9fPVQucHJvdG90eXBlLl9fZGVzdHJveV9fPWZ1bmN0aW9uKCl7RmIodGhpcy5wdHIpfTtCLnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKHQucHJvdG90eXBlKTtCLnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1CO0IucHJvdG90eXBlLl9fY2xhc3NfXz1CO0IuX19jYWNoZV9fPXt9O2EuU3RhdHVzPUI7Qi5wcm90b3R5cGUuY29kZT1CLnByb3RvdHlwZS5jb2RlPWZ1bmN0aW9uKCl7cmV0dXJuIEdiKHRoaXMucHRyKX07Qi5wcm90b3R5cGUub2s9Qi5wcm90b3R5cGUub2s9ZnVuY3Rpb24oKXtyZXR1cm4hIUhiKHRoaXMucHRyKX07Qi5wcm90b3R5cGUuZXJyb3JfbXNnPUIucHJvdG90eXBlLmVycm9yX21zZz1mdW5jdGlvbigpe3JldHVybiBoKEliKHRoaXMucHRyKSl9O0IucHJvdG90eXBlLl9fZGVzdHJveV9fPUIucHJvdG90eXBlLl9fZGVzdHJveV9fPWZ1bmN0aW9uKCl7SmIodGhpcy5wdHIpfTtJLnByb3RvdHlwZT0NCk9iamVjdC5jcmVhdGUodC5wcm90b3R5cGUpO0kucHJvdG90eXBlLmNvbnN0cnVjdG9yPUk7SS5wcm90b3R5cGUuX19jbGFzc19fPUk7SS5fX2NhY2hlX189e307YS5EcmFjb0Zsb2F0MzJBcnJheT1JO0kucHJvdG90eXBlLkdldFZhbHVlPUkucHJvdG90eXBlLkdldFZhbHVlPWZ1bmN0aW9uKGIpe3ZhciBjPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtyZXR1cm4gS2IoYyxiKX07SS5wcm90b3R5cGUuc2l6ZT1JLnByb3RvdHlwZS5zaXplPWZ1bmN0aW9uKCl7cmV0dXJuIExiKHRoaXMucHRyKX07SS5wcm90b3R5cGUuX19kZXN0cm95X189SS5wcm90b3R5cGUuX19kZXN0cm95X189ZnVuY3Rpb24oKXtNYih0aGlzLnB0cil9O0oucHJvdG90eXBlPU9iamVjdC5jcmVhdGUodC5wcm90b3R5cGUpO0oucHJvdG90eXBlLmNvbnN0cnVjdG9yPUo7Si5wcm90b3R5cGUuX19jbGFzc19fPUo7Si5fX2NhY2hlX189e307YS5EcmFjb0ludDhBcnJheT1KO0oucHJvdG90eXBlLkdldFZhbHVlPQ0KSi5wcm90b3R5cGUuR2V0VmFsdWU9ZnVuY3Rpb24oYil7dmFyIGM9dGhpcy5wdHI7YiYmIm9iamVjdCI9PT10eXBlb2YgYiYmKGI9Yi5wdHIpO3JldHVybiBOYihjLGIpfTtKLnByb3RvdHlwZS5zaXplPUoucHJvdG90eXBlLnNpemU9ZnVuY3Rpb24oKXtyZXR1cm4gT2IodGhpcy5wdHIpfTtKLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1KLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1mdW5jdGlvbigpe1BiKHRoaXMucHRyKX07Sy5wcm90b3R5cGU9T2JqZWN0LmNyZWF0ZSh0LnByb3RvdHlwZSk7Sy5wcm90b3R5cGUuY29uc3RydWN0b3I9SztLLnByb3RvdHlwZS5fX2NsYXNzX189SztLLl9fY2FjaGVfXz17fTthLkRyYWNvVUludDhBcnJheT1LO0sucHJvdG90eXBlLkdldFZhbHVlPUsucHJvdG90eXBlLkdldFZhbHVlPWZ1bmN0aW9uKGIpe3ZhciBjPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtyZXR1cm4gUWIoYyxiKX07Sy5wcm90b3R5cGUuc2l6ZT1LLnByb3RvdHlwZS5zaXplPQ0KZnVuY3Rpb24oKXtyZXR1cm4gUmIodGhpcy5wdHIpfTtLLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1LLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1mdW5jdGlvbigpe1NiKHRoaXMucHRyKX07TC5wcm90b3R5cGU9T2JqZWN0LmNyZWF0ZSh0LnByb3RvdHlwZSk7TC5wcm90b3R5cGUuY29uc3RydWN0b3I9TDtMLnByb3RvdHlwZS5fX2NsYXNzX189TDtMLl9fY2FjaGVfXz17fTthLkRyYWNvSW50MTZBcnJheT1MO0wucHJvdG90eXBlLkdldFZhbHVlPUwucHJvdG90eXBlLkdldFZhbHVlPWZ1bmN0aW9uKGIpe3ZhciBjPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtyZXR1cm4gVGIoYyxiKX07TC5wcm90b3R5cGUuc2l6ZT1MLnByb3RvdHlwZS5zaXplPWZ1bmN0aW9uKCl7cmV0dXJuIFViKHRoaXMucHRyKX07TC5wcm90b3R5cGUuX19kZXN0cm95X189TC5wcm90b3R5cGUuX19kZXN0cm95X189ZnVuY3Rpb24oKXtWYih0aGlzLnB0cil9O00ucHJvdG90eXBlPU9iamVjdC5jcmVhdGUodC5wcm90b3R5cGUpOw0KTS5wcm90b3R5cGUuY29uc3RydWN0b3I9TTtNLnByb3RvdHlwZS5fX2NsYXNzX189TTtNLl9fY2FjaGVfXz17fTthLkRyYWNvVUludDE2QXJyYXk9TTtNLnByb3RvdHlwZS5HZXRWYWx1ZT1NLnByb3RvdHlwZS5HZXRWYWx1ZT1mdW5jdGlvbihiKXt2YXIgYz10aGlzLnB0cjtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7cmV0dXJuIFdiKGMsYil9O00ucHJvdG90eXBlLnNpemU9TS5wcm90b3R5cGUuc2l6ZT1mdW5jdGlvbigpe3JldHVybiBYYih0aGlzLnB0cil9O00ucHJvdG90eXBlLl9fZGVzdHJveV9fPU0ucHJvdG90eXBlLl9fZGVzdHJveV9fPWZ1bmN0aW9uKCl7WWIodGhpcy5wdHIpfTtOLnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKHQucHJvdG90eXBlKTtOLnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1OO04ucHJvdG90eXBlLl9fY2xhc3NfXz1OO04uX19jYWNoZV9fPXt9O2EuRHJhY29JbnQzMkFycmF5PU47Ti5wcm90b3R5cGUuR2V0VmFsdWU9Ti5wcm90b3R5cGUuR2V0VmFsdWU9DQpmdW5jdGlvbihiKXt2YXIgYz10aGlzLnB0cjtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7cmV0dXJuIFpiKGMsYil9O04ucHJvdG90eXBlLnNpemU9Ti5wcm90b3R5cGUuc2l6ZT1mdW5jdGlvbigpe3JldHVybiAkYih0aGlzLnB0cil9O04ucHJvdG90eXBlLl9fZGVzdHJveV9fPU4ucHJvdG90eXBlLl9fZGVzdHJveV9fPWZ1bmN0aW9uKCl7YWModGhpcy5wdHIpfTtPLnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKHQucHJvdG90eXBlKTtPLnByb3RvdHlwZS5jb25zdHJ1Y3Rvcj1PO08ucHJvdG90eXBlLl9fY2xhc3NfXz1PO08uX19jYWNoZV9fPXt9O2EuRHJhY29VSW50MzJBcnJheT1PO08ucHJvdG90eXBlLkdldFZhbHVlPU8ucHJvdG90eXBlLkdldFZhbHVlPWZ1bmN0aW9uKGIpe3ZhciBjPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtyZXR1cm4gYmMoYyxiKX07Ty5wcm90b3R5cGUuc2l6ZT1PLnByb3RvdHlwZS5zaXplPWZ1bmN0aW9uKCl7cmV0dXJuIGNjKHRoaXMucHRyKX07DQpPLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1PLnByb3RvdHlwZS5fX2Rlc3Ryb3lfXz1mdW5jdGlvbigpe2RjKHRoaXMucHRyKX07eS5wcm90b3R5cGU9T2JqZWN0LmNyZWF0ZSh0LnByb3RvdHlwZSk7eS5wcm90b3R5cGUuY29uc3RydWN0b3I9eTt5LnByb3RvdHlwZS5fX2NsYXNzX189eTt5Ll9fY2FjaGVfXz17fTthLk1ldGFkYXRhUXVlcmllcj15O3kucHJvdG90eXBlLkhhc0VudHJ5PXkucHJvdG90eXBlLkhhc0VudHJ5PWZ1bmN0aW9uKGIsYyl7dmFyIGQ9dGhpcy5wdHI7ci5wcmVwYXJlKCk7YiYmIm9iamVjdCI9PT10eXBlb2YgYiYmKGI9Yi5wdHIpO2M9YyYmIm9iamVjdCI9PT10eXBlb2YgYz9jLnB0cjphYShjKTtyZXR1cm4hIWVjKGQsYixjKX07eS5wcm90b3R5cGUuR2V0SW50RW50cnk9eS5wcm90b3R5cGUuR2V0SW50RW50cnk9ZnVuY3Rpb24oYixjKXt2YXIgZD10aGlzLnB0cjtyLnByZXBhcmUoKTtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7Yz1jJiYib2JqZWN0Ij09PQ0KdHlwZW9mIGM/Yy5wdHI6YWEoYyk7cmV0dXJuIGZjKGQsYixjKX07eS5wcm90b3R5cGUuR2V0SW50RW50cnlBcnJheT15LnByb3RvdHlwZS5HZXRJbnRFbnRyeUFycmF5PWZ1bmN0aW9uKGIsYyxkKXt2YXIgZz10aGlzLnB0cjtyLnByZXBhcmUoKTtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7Yz1jJiYib2JqZWN0Ij09PXR5cGVvZiBjP2MucHRyOmFhKGMpO2QmJiJvYmplY3QiPT09dHlwZW9mIGQmJihkPWQucHRyKTtnYyhnLGIsYyxkKX07eS5wcm90b3R5cGUuR2V0RG91YmxlRW50cnk9eS5wcm90b3R5cGUuR2V0RG91YmxlRW50cnk9ZnVuY3Rpb24oYixjKXt2YXIgZD10aGlzLnB0cjtyLnByZXBhcmUoKTtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7Yz1jJiYib2JqZWN0Ij09PXR5cGVvZiBjP2MucHRyOmFhKGMpO3JldHVybiBoYyhkLGIsYyl9O3kucHJvdG90eXBlLkdldFN0cmluZ0VudHJ5PXkucHJvdG90eXBlLkdldFN0cmluZ0VudHJ5PWZ1bmN0aW9uKGIsDQpjKXt2YXIgZD10aGlzLnB0cjtyLnByZXBhcmUoKTtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7Yz1jJiYib2JqZWN0Ij09PXR5cGVvZiBjP2MucHRyOmFhKGMpO3JldHVybiBoKGljKGQsYixjKSl9O3kucHJvdG90eXBlLk51bUVudHJpZXM9eS5wcm90b3R5cGUuTnVtRW50cmllcz1mdW5jdGlvbihiKXt2YXIgYz10aGlzLnB0cjtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7cmV0dXJuIGpjKGMsYil9O3kucHJvdG90eXBlLkdldEVudHJ5TmFtZT15LnByb3RvdHlwZS5HZXRFbnRyeU5hbWU9ZnVuY3Rpb24oYixjKXt2YXIgZD10aGlzLnB0cjtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7YyYmIm9iamVjdCI9PT10eXBlb2YgYyYmKGM9Yy5wdHIpO3JldHVybiBoKGtjKGQsYixjKSl9O3kucHJvdG90eXBlLl9fZGVzdHJveV9fPXkucHJvdG90eXBlLl9fZGVzdHJveV9fPWZ1bmN0aW9uKCl7bGModGhpcy5wdHIpfTtsLnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKHQucHJvdG90eXBlKTsNCmwucHJvdG90eXBlLmNvbnN0cnVjdG9yPWw7bC5wcm90b3R5cGUuX19jbGFzc19fPWw7bC5fX2NhY2hlX189e307YS5EZWNvZGVyPWw7bC5wcm90b3R5cGUuRGVjb2RlQXJyYXlUb1BvaW50Q2xvdWQ9bC5wcm90b3R5cGUuRGVjb2RlQXJyYXlUb1BvaW50Q2xvdWQ9ZnVuY3Rpb24oYixjLGQpe3ZhciBnPXRoaXMucHRyO3IucHJlcGFyZSgpOyJvYmplY3QiPT10eXBlb2YgYiYmKGI9dWEoYikpO2MmJiJvYmplY3QiPT09dHlwZW9mIGMmJihjPWMucHRyKTtkJiYib2JqZWN0Ij09PXR5cGVvZiBkJiYoZD1kLnB0cik7cmV0dXJuIEQobWMoZyxiLGMsZCksQil9O2wucHJvdG90eXBlLkRlY29kZUFycmF5VG9NZXNoPWwucHJvdG90eXBlLkRlY29kZUFycmF5VG9NZXNoPWZ1bmN0aW9uKGIsYyxkKXt2YXIgZz10aGlzLnB0cjtyLnByZXBhcmUoKTsib2JqZWN0Ij09dHlwZW9mIGImJihiPXVhKGIpKTtjJiYib2JqZWN0Ij09PXR5cGVvZiBjJiYoYz1jLnB0cik7ZCYmIm9iamVjdCI9PT10eXBlb2YgZCYmDQooZD1kLnB0cik7cmV0dXJuIEQobmMoZyxiLGMsZCksQil9O2wucHJvdG90eXBlLkdldEF0dHJpYnV0ZUlkPWwucHJvdG90eXBlLkdldEF0dHJpYnV0ZUlkPWZ1bmN0aW9uKGIsYyl7dmFyIGQ9dGhpcy5wdHI7YiYmIm9iamVjdCI9PT10eXBlb2YgYiYmKGI9Yi5wdHIpO2MmJiJvYmplY3QiPT09dHlwZW9mIGMmJihjPWMucHRyKTtyZXR1cm4gb2MoZCxiLGMpfTtsLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVJZEJ5TmFtZT1sLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVJZEJ5TmFtZT1mdW5jdGlvbihiLGMpe3ZhciBkPXRoaXMucHRyO3IucHJlcGFyZSgpO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtjPWMmJiJvYmplY3QiPT09dHlwZW9mIGM/Yy5wdHI6YWEoYyk7cmV0dXJuIHBjKGQsYixjKX07bC5wcm90b3R5cGUuR2V0QXR0cmlidXRlSWRCeU1ldGFkYXRhRW50cnk9bC5wcm90b3R5cGUuR2V0QXR0cmlidXRlSWRCeU1ldGFkYXRhRW50cnk9ZnVuY3Rpb24oYixjLGQpe3ZhciBnPQ0KdGhpcy5wdHI7ci5wcmVwYXJlKCk7YiYmIm9iamVjdCI9PT10eXBlb2YgYiYmKGI9Yi5wdHIpO2M9YyYmIm9iamVjdCI9PT10eXBlb2YgYz9jLnB0cjphYShjKTtkPWQmJiJvYmplY3QiPT09dHlwZW9mIGQ/ZC5wdHI6YWEoZCk7cmV0dXJuIHFjKGcsYixjLGQpfTtsLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGU9bC5wcm90b3R5cGUuR2V0QXR0cmlidXRlPWZ1bmN0aW9uKGIsYyl7dmFyIGQ9dGhpcy5wdHI7YiYmIm9iamVjdCI9PT10eXBlb2YgYiYmKGI9Yi5wdHIpO2MmJiJvYmplY3QiPT09dHlwZW9mIGMmJihjPWMucHRyKTtyZXR1cm4gRChyYyhkLGIsYyksdyl9O2wucHJvdG90eXBlLkdldEF0dHJpYnV0ZUJ5VW5pcXVlSWQ9bC5wcm90b3R5cGUuR2V0QXR0cmlidXRlQnlVbmlxdWVJZD1mdW5jdGlvbihiLGMpe3ZhciBkPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtjJiYib2JqZWN0Ij09PXR5cGVvZiBjJiYoYz1jLnB0cik7cmV0dXJuIEQoc2MoZCxiLA0KYyksdyl9O2wucHJvdG90eXBlLkdldE1ldGFkYXRhPWwucHJvdG90eXBlLkdldE1ldGFkYXRhPWZ1bmN0aW9uKGIpe3ZhciBjPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtyZXR1cm4gRCh0YyhjLGIpLFQpfTtsLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVNZXRhZGF0YT1sLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVNZXRhZGF0YT1mdW5jdGlvbihiLGMpe3ZhciBkPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtjJiYib2JqZWN0Ij09PXR5cGVvZiBjJiYoYz1jLnB0cik7cmV0dXJuIEQodWMoZCxiLGMpLFQpfTtsLnByb3RvdHlwZS5HZXRGYWNlRnJvbU1lc2g9bC5wcm90b3R5cGUuR2V0RmFjZUZyb21NZXNoPWZ1bmN0aW9uKGIsYyxkKXt2YXIgZz10aGlzLnB0cjtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7YyYmIm9iamVjdCI9PT10eXBlb2YgYyYmKGM9Yy5wdHIpO2QmJiJvYmplY3QiPT09dHlwZW9mIGQmJg0KKGQ9ZC5wdHIpO3JldHVybiEhdmMoZyxiLGMsZCl9O2wucHJvdG90eXBlLkdldFRyaWFuZ2xlU3RyaXBzRnJvbU1lc2g9bC5wcm90b3R5cGUuR2V0VHJpYW5nbGVTdHJpcHNGcm9tTWVzaD1mdW5jdGlvbihiLGMpe3ZhciBkPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtjJiYib2JqZWN0Ij09PXR5cGVvZiBjJiYoYz1jLnB0cik7cmV0dXJuIHdjKGQsYixjKX07bC5wcm90b3R5cGUuR2V0VHJpYW5nbGVzVUludDE2QXJyYXk9bC5wcm90b3R5cGUuR2V0VHJpYW5nbGVzVUludDE2QXJyYXk9ZnVuY3Rpb24oYixjLGQpe3ZhciBnPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtjJiYib2JqZWN0Ij09PXR5cGVvZiBjJiYoYz1jLnB0cik7ZCYmIm9iamVjdCI9PT10eXBlb2YgZCYmKGQ9ZC5wdHIpO3JldHVybiEheGMoZyxiLGMsZCl9O2wucHJvdG90eXBlLkdldFRyaWFuZ2xlc1VJbnQzMkFycmF5PWwucHJvdG90eXBlLkdldFRyaWFuZ2xlc1VJbnQzMkFycmF5PQ0KZnVuY3Rpb24oYixjLGQpe3ZhciBnPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtjJiYib2JqZWN0Ij09PXR5cGVvZiBjJiYoYz1jLnB0cik7ZCYmIm9iamVjdCI9PT10eXBlb2YgZCYmKGQ9ZC5wdHIpO3JldHVybiEheWMoZyxiLGMsZCl9O2wucHJvdG90eXBlLkdldEF0dHJpYnV0ZUZsb2F0PWwucHJvdG90eXBlLkdldEF0dHJpYnV0ZUZsb2F0PWZ1bmN0aW9uKGIsYyxkKXt2YXIgZz10aGlzLnB0cjtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7YyYmIm9iamVjdCI9PT10eXBlb2YgYyYmKGM9Yy5wdHIpO2QmJiJvYmplY3QiPT09dHlwZW9mIGQmJihkPWQucHRyKTtyZXR1cm4hIXpjKGcsYixjLGQpfTtsLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVGbG9hdEZvckFsbFBvaW50cz1sLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVGbG9hdEZvckFsbFBvaW50cz1mdW5jdGlvbihiLGMsZCl7dmFyIGc9dGhpcy5wdHI7YiYmIm9iamVjdCI9PT10eXBlb2YgYiYmDQooYj1iLnB0cik7YyYmIm9iamVjdCI9PT10eXBlb2YgYyYmKGM9Yy5wdHIpO2QmJiJvYmplY3QiPT09dHlwZW9mIGQmJihkPWQucHRyKTtyZXR1cm4hIUFjKGcsYixjLGQpfTtsLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVJbnRGb3JBbGxQb2ludHM9bC5wcm90b3R5cGUuR2V0QXR0cmlidXRlSW50Rm9yQWxsUG9pbnRzPWZ1bmN0aW9uKGIsYyxkKXt2YXIgZz10aGlzLnB0cjtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7YyYmIm9iamVjdCI9PT10eXBlb2YgYyYmKGM9Yy5wdHIpO2QmJiJvYmplY3QiPT09dHlwZW9mIGQmJihkPWQucHRyKTtyZXR1cm4hIUJjKGcsYixjLGQpfTtsLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVJbnQ4Rm9yQWxsUG9pbnRzPWwucHJvdG90eXBlLkdldEF0dHJpYnV0ZUludDhGb3JBbGxQb2ludHM9ZnVuY3Rpb24oYixjLGQpe3ZhciBnPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtjJiYib2JqZWN0Ij09PXR5cGVvZiBjJiYNCihjPWMucHRyKTtkJiYib2JqZWN0Ij09PXR5cGVvZiBkJiYoZD1kLnB0cik7cmV0dXJuISFDYyhnLGIsYyxkKX07bC5wcm90b3R5cGUuR2V0QXR0cmlidXRlVUludDhGb3JBbGxQb2ludHM9bC5wcm90b3R5cGUuR2V0QXR0cmlidXRlVUludDhGb3JBbGxQb2ludHM9ZnVuY3Rpb24oYixjLGQpe3ZhciBnPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtjJiYib2JqZWN0Ij09PXR5cGVvZiBjJiYoYz1jLnB0cik7ZCYmIm9iamVjdCI9PT10eXBlb2YgZCYmKGQ9ZC5wdHIpO3JldHVybiEhRGMoZyxiLGMsZCl9O2wucHJvdG90eXBlLkdldEF0dHJpYnV0ZUludDE2Rm9yQWxsUG9pbnRzPWwucHJvdG90eXBlLkdldEF0dHJpYnV0ZUludDE2Rm9yQWxsUG9pbnRzPWZ1bmN0aW9uKGIsYyxkKXt2YXIgZz10aGlzLnB0cjtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7YyYmIm9iamVjdCI9PT10eXBlb2YgYyYmKGM9Yy5wdHIpO2QmJiJvYmplY3QiPT09dHlwZW9mIGQmJg0KKGQ9ZC5wdHIpO3JldHVybiEhRWMoZyxiLGMsZCl9O2wucHJvdG90eXBlLkdldEF0dHJpYnV0ZVVJbnQxNkZvckFsbFBvaW50cz1sLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVVSW50MTZGb3JBbGxQb2ludHM9ZnVuY3Rpb24oYixjLGQpe3ZhciBnPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtjJiYib2JqZWN0Ij09PXR5cGVvZiBjJiYoYz1jLnB0cik7ZCYmIm9iamVjdCI9PT10eXBlb2YgZCYmKGQ9ZC5wdHIpO3JldHVybiEhRmMoZyxiLGMsZCl9O2wucHJvdG90eXBlLkdldEF0dHJpYnV0ZUludDMyRm9yQWxsUG9pbnRzPWwucHJvdG90eXBlLkdldEF0dHJpYnV0ZUludDMyRm9yQWxsUG9pbnRzPWZ1bmN0aW9uKGIsYyxkKXt2YXIgZz10aGlzLnB0cjtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7YyYmIm9iamVjdCI9PT10eXBlb2YgYyYmKGM9Yy5wdHIpO2QmJiJvYmplY3QiPT09dHlwZW9mIGQmJihkPWQucHRyKTtyZXR1cm4hIUdjKGcsDQpiLGMsZCl9O2wucHJvdG90eXBlLkdldEF0dHJpYnV0ZVVJbnQzMkZvckFsbFBvaW50cz1sLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVVSW50MzJGb3JBbGxQb2ludHM9ZnVuY3Rpb24oYixjLGQpe3ZhciBnPXRoaXMucHRyO2ImJiJvYmplY3QiPT09dHlwZW9mIGImJihiPWIucHRyKTtjJiYib2JqZWN0Ij09PXR5cGVvZiBjJiYoYz1jLnB0cik7ZCYmIm9iamVjdCI9PT10eXBlb2YgZCYmKGQ9ZC5wdHIpO3JldHVybiEhSGMoZyxiLGMsZCl9O2wucHJvdG90eXBlLkdldEF0dHJpYnV0ZURhdGFBcnJheUZvckFsbFBvaW50cz1sLnByb3RvdHlwZS5HZXRBdHRyaWJ1dGVEYXRhQXJyYXlGb3JBbGxQb2ludHM9ZnVuY3Rpb24oYixjLGQsZyx1KXt2YXIgWD10aGlzLnB0cjtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7YyYmIm9iamVjdCI9PT10eXBlb2YgYyYmKGM9Yy5wdHIpO2QmJiJvYmplY3QiPT09dHlwZW9mIGQmJihkPWQucHRyKTtnJiYib2JqZWN0Ij09PXR5cGVvZiBnJiYNCihnPWcucHRyKTt1JiYib2JqZWN0Ij09PXR5cGVvZiB1JiYodT11LnB0cik7cmV0dXJuISFJYyhYLGIsYyxkLGcsdSl9O2wucHJvdG90eXBlLlNraXBBdHRyaWJ1dGVUcmFuc2Zvcm09bC5wcm90b3R5cGUuU2tpcEF0dHJpYnV0ZVRyYW5zZm9ybT1mdW5jdGlvbihiKXt2YXIgYz10aGlzLnB0cjtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7SmMoYyxiKX07bC5wcm90b3R5cGUuR2V0RW5jb2RlZEdlb21ldHJ5VHlwZV9EZXByZWNhdGVkPWwucHJvdG90eXBlLkdldEVuY29kZWRHZW9tZXRyeVR5cGVfRGVwcmVjYXRlZD1mdW5jdGlvbihiKXt2YXIgYz10aGlzLnB0cjtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYoYj1iLnB0cik7cmV0dXJuIEtjKGMsYil9O2wucHJvdG90eXBlLkRlY29kZUJ1ZmZlclRvUG9pbnRDbG91ZD1sLnByb3RvdHlwZS5EZWNvZGVCdWZmZXJUb1BvaW50Q2xvdWQ9ZnVuY3Rpb24oYixjKXt2YXIgZD10aGlzLnB0cjtiJiYib2JqZWN0Ij09PXR5cGVvZiBiJiYNCihiPWIucHRyKTtjJiYib2JqZWN0Ij09PXR5cGVvZiBjJiYoYz1jLnB0cik7cmV0dXJuIEQoTGMoZCxiLGMpLEIpfTtsLnByb3RvdHlwZS5EZWNvZGVCdWZmZXJUb01lc2g9bC5wcm90b3R5cGUuRGVjb2RlQnVmZmVyVG9NZXNoPWZ1bmN0aW9uKGIsYyl7dmFyIGQ9dGhpcy5wdHI7YiYmIm9iamVjdCI9PT10eXBlb2YgYiYmKGI9Yi5wdHIpO2MmJiJvYmplY3QiPT09dHlwZW9mIGMmJihjPWMucHRyKTtyZXR1cm4gRChNYyhkLGIsYyksQil9O2wucHJvdG90eXBlLl9fZGVzdHJveV9fPWwucHJvdG90eXBlLl9fZGVzdHJveV9fPWZ1bmN0aW9uKCl7TmModGhpcy5wdHIpfTsoZnVuY3Rpb24oKXtmdW5jdGlvbiBiKCl7YS5BVFRSSUJVVEVfSU5WQUxJRF9UUkFOU0ZPUk09T2MoKTthLkFUVFJJQlVURV9OT19UUkFOU0ZPUk09UGMoKTthLkFUVFJJQlVURV9RVUFOVElaQVRJT05fVFJBTlNGT1JNPVFjKCk7YS5BVFRSSUJVVEVfT0NUQUhFRFJPTl9UUkFOU0ZPUk09UmMoKTthLklOVkFMSUQ9U2MoKTsNCmEuUE9TSVRJT049VGMoKTthLk5PUk1BTD1VYygpO2EuQ09MT1I9VmMoKTthLlRFWF9DT09SRD1XYygpO2EuR0VORVJJQz1YYygpO2EuSU5WQUxJRF9HRU9NRVRSWV9UWVBFPVljKCk7YS5QT0lOVF9DTE9VRD1aYygpO2EuVFJJQU5HVUxBUl9NRVNIPSRjKCk7YS5EVF9JTlZBTElEPWFkKCk7YS5EVF9JTlQ4PWJkKCk7YS5EVF9VSU5UOD1jZCgpO2EuRFRfSU5UMTY9ZGQoKTthLkRUX1VJTlQxNj1lZCgpO2EuRFRfSU5UMzI9ZmQoKTthLkRUX1VJTlQzMj1nZCgpO2EuRFRfSU5UNjQ9aGQoKTthLkRUX1VJTlQ2ND1pZCgpO2EuRFRfRkxPQVQzMj1qZCgpO2EuRFRfRkxPQVQ2ND1rZCgpO2EuRFRfQk9PTD1sZCgpO2EuRFRfVFlQRVNfQ09VTlQ9bWQoKTthLk9LPW5kKCk7YS5EUkFDT19FUlJPUj1vZCgpO2EuSU9fRVJST1I9cGQoKTthLklOVkFMSURfUEFSQU1FVEVSPXFkKCk7YS5VTlNVUFBPUlRFRF9WRVJTSU9OPXJkKCk7YS5VTktOT1dOX1ZFUlNJT049c2QoKX1CYT9iKCk6dGEudW5zaGlmdChiKX0pKCk7DQppZigiZnVuY3Rpb24iPT09dHlwZW9mIGEub25Nb2R1bGVQYXJzZWQpYS5vbk1vZHVsZVBhcnNlZCgpO2EuRGVjb2Rlci5wcm90b3R5cGUuR2V0RW5jb2RlZEdlb21ldHJ5VHlwZT1mdW5jdGlvbihiKXtpZihiLl9fY2xhc3NfXyYmYi5fX2NsYXNzX189PT1hLkRlY29kZXJCdWZmZXIpcmV0dXJuIGEuRGVjb2Rlci5wcm90b3R5cGUuR2V0RW5jb2RlZEdlb21ldHJ5VHlwZV9EZXByZWNhdGVkKGIpO2lmKDg+Yi5ieXRlTGVuZ3RoKXJldHVybiBhLklOVkFMSURfR0VPTUVUUllfVFlQRTtzd2l0Y2goYls3XSl7Y2FzZSAwOnJldHVybiBhLlBPSU5UX0NMT1VEO2Nhc2UgMTpyZXR1cm4gYS5UUklBTkdVTEFSX01FU0g7ZGVmYXVsdDpyZXR1cm4gYS5JTlZBTElEX0dFT01FVFJZX1RZUEV9fTtyZXR1cm4gbi5yZWFkeX19KCk7DQoib2JqZWN0Ij09PXR5cGVvZiBleHBvcnRzJiYib2JqZWN0Ij09PXR5cGVvZiBtb2R1bGU/bW9kdWxlLmV4cG9ydHM9RHJhY29EZWNvZGVyTW9kdWxlOiJmdW5jdGlvbiI9PT10eXBlb2YgZGVmaW5lJiZkZWZpbmUuYW1kP2RlZmluZShbXSxmdW5jdGlvbigpe3JldHVybiBEcmFjb0RlY29kZXJNb2R1bGV9KToib2JqZWN0Ij09PXR5cGVvZiBleHBvcnRzJiYoZXhwb3J0cy5EcmFjb0RlY29kZXJNb2R1bGU9RHJhY29EZWNvZGVyTW9kdWxlKTs=", C1e = "data:application/octet-stream;base64,hhaHlvbWljZ7InZlcnNpb24iOjIsIndpZHRoIjoxMjgsImltYWdlVHlwZSI6ImltYWdlL3BuZyIsImlycmFkaWFuY2UiOnsieCI6Wy0wLjAxNzA3OTc5MDE1NzU2MzQ5LC0wLjAxNDgzNDc3MjQ2MTQyMjUzOSwtMC4wMTI4NzQ1OTY3NTU2MTMyNzJdLCJ5IjpbMC4wMDI2OTUyNzg4OTM5NzM1Mzk1LDAuMDAyMjQ1MTQ2NDMyODkzNDQxLDAuMDAyMDQ0NDQzOTE1NDI4MTcyXSwieiI6WzAuMDExNzczMTMyMzA3NTk3OTU0LDAuMDA5OTczMDMwMjQ1NTE1NDQ5LDAuMDA4NTA0NjAwNTMzOTUwOTVdLCJ4eCI6WzAuNjkwMDE4NjI1OTk1ODk0NSwwLjcyODUzMzQ1MjE3MDIwNDcsMC43NjYyNTI0NDE5ODY5OTU0XSwieXkiOlswLjcwMDQ2NzQ2MDg5ODczNCwwLjczODI0OTc0NDk5NDc5MDYsMC43NzU5MzMxNzE3NjU2Mzg5XSwienoiOlswLjY4NjkxNDg2MzgwMzUwMzIsMC43MjU3MDI0MzEyNTY1NDgyLDAuNzYzOTAzMzc0MTAyOTM3MV0sInl6IjpbMC4wMDAzODA2MzY3MjI3Mzg5MTcwNiwwLjAwMDI4MTkxMjEyNjM5OTUwODgsMC4wMDAyNTUxODkxMDAwNDUyNTA4NV0sInp4IjpbMC4wMDA5Njg3MjYxNjM2Mzg4NDY2LDAuMDAxMDEyMTUzMzk4OTcxNzc3MywwLjAwMDk3MDEwNDI1MTQyNDQ0NjVdLCJ4eSI6Wy0wLjAwMDQxMzQ2NzUzOTEyNTM0MTcsLTAuMDAwMzU2OTYyMjA2NTQwMDE3LC0wLjAwMDI5NDIwODQxNzI4MjA5Nzk0XX0sInNwZWN1bGFyIjp7Im1pcG1hcHMiOlt7Imxlbmd0aCI6NzE3OCwicG9zaXRpb24iOjB9LHsibGVuZ3RoIjo3MDc4LCJwb3NpdGlvbiI6NzE3OH0seyJsZW5ndGgiOjEyMDY5LCJwb3NpdGlvbiI6MTQyNTZ9LHsibGVuZ3RoIjoxMTM5NCwicG9zaXRpb24iOjI2MzI1fSx7Imxlbmd0aCI6NjgyNiwicG9zaXRpb24iOjM3NzE5fSx7Imxlbmd0aCI6NzQzOSwicG9zaXRpb24iOjQ0NTQ1fSx7Imxlbmd0aCI6NDI5NSwicG9zaXRpb24iOjUxOTg0fSx7Imxlbmd0aCI6NDA0MywicG9zaXRpb24iOjU2Mjc5fSx7Imxlbmd0aCI6MzkxNSwicG9zaXRpb24iOjYwMzIyfSx7Imxlbmd0aCI6MzcxMSwicG9zaXRpb24iOjY0MjM3fSx7Imxlbmd0aCI6MzY2NiwicG9zaXRpb24iOjY3OTQ4fSx7Imxlbmd0aCI6NDMzMCwicG9zaXRpb24iOjcxNjE0fSx7Imxlbmd0aCI6MTMyNCwicG9zaXRpb24iOjc1OTQ0fSx7Imxlbmd0aCI6MTI1NSwicG9zaXRpb24iOjc3MjY4fSx7Imxlbmd0aCI6MTIxMiwicG9zaXRpb24iOjc4NTIzfSx7Imxlbmd0aCI6MTIxMSwicG9zaXRpb24iOjc5NzM1fSx7Imxlbmd0aCI6MTE0MywicG9zaXRpb24iOjgwOTQ2fSx7Imxlbmd0aCI6MTMxNiwicG9zaXRpb24iOjgyMDg5fSx7Imxlbmd0aCI6NDU3LCJwb3NpdGlvbiI6ODM0MDV9LHsibGVuZ3RoIjo0MjAsInBvc2l0aW9uIjo4Mzg2Mn0seyJsZW5ndGgiOjQyOSwicG9zaXRpb24iOjg0MjgyfSx7Imxlbmd0aCI6NDM5LCJwb3NpdGlvbiI6ODQ3MTF9LHsibGVuZ3RoIjozOTgsInBvc2l0aW9uIjo4NTE1MH0seyJsZW5ndGgiOjQyOSwicG9zaXRpb24iOjg1NTQ4fSx7Imxlbmd0aCI6MjAzLCJwb3NpdGlvbiI6ODU5Nzd9LHsibGVuZ3RoIjoyMDAsInBvc2l0aW9uIjo4NjE4MH0seyJsZW5ndGgiOjIwMCwicG9zaXRpb24iOjg2MzgwfSx7Imxlbmd0aCI6MjAyLCJwb3NpdGlvbiI6ODY1ODB9LHsibGVuZ3RoIjoxODYsInBvc2l0aW9uIjo4Njc4Mn0seyJsZW5ndGgiOjE4MCwicG9zaXRpb24iOjg2OTY4fSx7Imxlbmd0aCI6MTIzLCJwb3NpdGlvbiI6ODcxNDh9LHsibGVuZ3RoIjoxMjUsInBvc2l0aW9uIjo4NzI3MX0seyJsZW5ndGgiOjEyNSwicG9zaXRpb24iOjg3Mzk2fSx7Imxlbmd0aCI6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", O1e = "data:application/octet-stream;base64,hhaHlvbWljZ7InZlcnNpb24iOjIsIndpZHRoIjoxMjgsImltYWdlVHlwZSI6ImltYWdlL3BuZyIsImlycmFkaWFuY2UiOnsieCI6Wy0wLjAyOTYzNTE5OTUwMjUxMzgzLC0wLjAyNTI1MTgwOTY4NTEwODI5MywtMC4wMzk2NDgyOTAwNjQ2Mjk2N10sInkiOlswLjExMTkyNjY4NTEwNjI2ODk1LDAuMTIyOTQ2NjQxOTkzMTE5MiwwLjExMTU0NzExNjI1NjY1MDg3XSwieiI6Wy0wLjAxMzYxNDQ4NTAwNjI2NjE4LC0wLjAxNDM3OTE5ODk4MDQwMzAzMiwtMC4wMDcyNDAwODA4MTg3MjUwNTFdLCJ4eCI6WzAuNDAxMDE1NDE5OTAzMTY0MjUsMC40MDM1OTIxNzU0NjQwMzE5LDAuMzQwMTM3MTEzNDgwMTUyN10sInl5IjpbMC40MjQxMTg4NjUxMzQzODE3LDAuNDIzNzExODg2NzY3NzM5MywwLjM2Mjk5ODQ5Njc1NTgwMDE1XSwienoiOlswLjQwMDQ3ODAzMzg2MjgwNTg0LDAuNDAyODM5OTg3MjY1ODA1NDQsMC4zNDAwMjA2NjY1MTI5ODM2XSwieXoiOlstMC4wMDczMjU1MjE1MjQ3NjA1NSwtMC4wMDY5NDg0Mzg5NTUyNzI2OSwtMC4wMDM5NjU4ODQ2NjgxMjgxNzJdLCJ6eCI6Wy0wLjAwMDM4MDU1ODUxNjYzMTUzNCwtMC4wMDA2MjU2MjA3OTMzMDA5NjQsMC4wMDMwMzkxNzc0MzkwMDg3ODc4XSwieHkiOlstMC4wMDA4MTY3MTc2NTE4MjIwMjkzLC0wLjAwMTkzNTkzNTgxNDc5MDIxMDcsLTAuMDA3Mjg1ODIzMTgyOTc2ODMzXX0sInNwZWN1bGFyIjp7Im1pcG1hcHMiOlt7Imxlbmd0aCI6MzQwNjMsInBvc2l0aW9uIjowfSx7Imxlbmd0aCI6MzQxMDEsInBvc2l0aW9uIjozNDA2M30seyJsZW5ndGgiOjI3ODg2LCJwb3NpdGlvbiI6NjgxNjR9LHsibGVuZ3RoIjozOTk2OCwicG9zaXRpb24iOjk2MDUwfSx7Imxlbmd0aCI6MzQ5NzEsInBvc2l0aW9uIjoxMzYwMTh9LHsibGVuZ3RoIjoyODg3MCwicG9zaXRpb24iOjE3MDk4OX0seyJsZW5ndGgiOjcxMTIsInBvc2l0aW9uIjoxOTk4NTl9LHsibGVuZ3RoIjo3MjQzLCJwb3NpdGlvbiI6MjA2OTcxfSx7Imxlbmd0aCI6NzQxMSwicG9zaXRpb24iOjIxNDIxNH0seyJsZW5ndGgiOjc1ODYsInBvc2l0aW9uIjoyMjE2MjV9LHsibGVuZ3RoIjo3NDgxLCJwb3NpdGlvbiI6MjI5MjExfSx7Imxlbmd0aCI6NzE3NiwicG9zaXRpb24iOjIzNjY5Mn0seyJsZW5ndGgiOjE5NTMsInBvc2l0aW9uIjoyNDM4Njh9LHsibGVuZ3RoIjoyMDYxLCJwb3NpdGlvbiI6MjQ1ODIxfSx7Imxlbmd0aCI6MjE2MiwicG9zaXRpb24iOjI0Nzg4Mn0seyJsZW5ndGgiOjE4OTQsInBvc2l0aW9uIjoyNTAwNDR9LHsibGVuZ3RoIjoyMDQ0LCJwb3NpdGlvbiI6MjUxOTM4fSx7Imxlbmd0aCI6MTkzNiwicG9zaXRpb24iOjI1Mzk4Mn0seyJsZW5ndGgiOjU3MCwicG9zaXRpb24iOjI1NTkxOH0seyJsZW5ndGgiOjY2NiwicG9zaXRpb24iOjI1NjQ4OH0seyJsZW5ndGgiOjY5NiwicG9zaXRpb24iOjI1NzE1NH0seyJsZW5ndGgiOjU0MywicG9zaXRpb24iOjI1Nzg1MH0seyJsZW5ndGgiOjYyNSwicG9zaXRpb24iOjI1ODM5M30seyJsZW5ndGgiOjU5NCwicG9zaXRpb24iOjI1OTAxOH0seyJsZW5ndGgiOjIyNiwicG9zaXRpb24iOjI1OTYxMn0seyJsZW5ndGgiOjI1NywicG9zaXRpb24iOjI1OTgzOH0seyJsZW5ndGgiOjI1OCwicG9zaXRpb24iOjI2MDA5NX0seyJsZW5ndGgiOjIxNSwicG9zaXRpb24iOjI2MDM1M30seyJsZW5ndGgiOjIzOCwicG9zaXRpb24iOjI2MDU2OH0seyJsZW5ndGgiOjI0MiwicG9zaXRpb24iOjI2MDgwNn0seyJsZW5ndGgiOjEzNCwicG9zaXRpb24iOjI2MTA0OH0seyJsZW5ndGgiOjEzNywicG9zaXRpb24iOjI2MTE4Mn0seyJsZW5ndGgiOjE0NCwicG9zaXRpb24iOjI2MTMxOX0seyJsZW5ndGgiOjEyOSwicG9zaXRpb24iOjI2MTQ2M30seyJsZW5ndGgiOjEzNiwicG9zaXRpb24iOjI2MTU5Mn0seyJsZW5ndGgiOjEzOSwicG9zaXRpb24iOjI2MTcyOH0seyJsZW5ndGgiOjk3LCJwb3NpdGlvbiI6MjYxODY3fSx7Imxlbmd0aCI6OTcsInBvc2l0aW9uIjoyNjE5NjR9LHsibGVuZ3RoIjo5NywicG9zaXRpb24iOjI2MjA2MX0seyJsZW5ndGgiOjk3LCJwb3NpdGlvbiI6MjYyMTU4fSx7Imxlbmd0aCI6OTcsInBvc2l0aW9uIjoyNjIyNTV9LHsibGVuZ3RoIjo5NywicG9zaXRpb24iOjI2MjM1Mn0seyJsZW5ndGgiOjgzLCJwb3NpdGlvbiI6MjYyNDQ5fSx7Imxlbmd0aCI6ODMsInBvc2l0aW9uIjoyNjI1MzJ9LHsibGVuZ3RoIjo4MywicG9zaXRpb24iOjI2MjYxNX0seyJsZW5ndGgiOjgzLCJwb3NpdGlvbiI6MjYyNjk4fSx7Imxlbmd0aCI6ODMsInBvc2l0aW9uIjoyNjI3ODF9LHsibGVuZ3RoIjo4MywicG9zaXRpb24iOjI2Mjg2NH1dLCJsb2RHZW5lcmF0aW9uU2NhbGUiOjAuOH19AIlQTkcNChoKAAAADUlIRFIAAACAAAAAgAgGAAAAwz5hywAAAAFzUkdCAK7OHOkAACAASURBVHheVL0JrGzZdZ63zlTzeMd33+vu1yObTYqSRcuELCsU7DhBIgeBAQswIkgxYAk2EgYyCc4EW4rEIZRjiTKkAIkdBwkcO1CCKAmcxLCUyIIBBYIUtSmKTfb4Xr/uN9255jpVZ0q+tfauqt6Nxru3btWpc/Zeew3/+tfawRc+9TOViMj+/qHkRcGPUpalXF1cSBQn+ntVVRIlNf05DEP9nVHkhQyGe9Lp9KQoC7m+HkkQBlLkpVyPr/U9tVosSRxJlq2lLAqpJJZGEkmWFxJFoQSByHK1kjIr9L28p16vSafddvdSyDJNJQoD/V6+uV5vyGqZShAEEsaJRFEkQRBu7p/bS2oNCaNQ4iiSKAzt3pOGFEUhRVnac0kls9lMfx7u7evfyrKQdrsjrWZLFou5xGGkf5eqkiRJZDadSZwk0h/0ZDaby+jqUhbLpayzXM7Pz6QoK2kP9iSJYymrShp1m7eizHVesnwteV7IarWy64rdSy2uSRDZfearVLIsk1JKqfgzDx3Y34IwlDC0z0RBKEkoOg/tesvdZyFJrS5JUrff3bP7a0tUk7Uts13vF7/8d6oiy3UyWFwmlwtOJhP90iLLJEoS/eKqrPg2abc6uhjTyVjqjYZ9rtGQ84sraTabMhjs6eevR1e6cPpF/FPZN9f15ioJgkhK91q+XutCRXEozHkUiCRJY3OjCFBRlFIUuS5UFDGxgS58rYagBrLO7PpFnkmj3dGfl8uFJLWEd0oVxFLk+WZSmFfunYEArNdrqdfrcnx8LNl6rQvMtVj8Mi90Xvies9OHKgy1pC5hHMnDs3O9xmQ8tmetNaTb7Uqer6XdauprCEOW5ZKtMymrUgWAjcbXx06IeV9VlZLO5/o3t8/smlG8Wc+IyeHe2U56TyLtmltwQfjrErv3B3GyEawojiQvQ1msUlmuJlsB0C8ubXeZyImss0wXSL8oDPUB7AYricJYGo2mTdJ8ohpgPp9KUdn7D49uSJou9ZrparlZCP1BROIwkOVyJv3uQEL/HZFdW7VFnqoARGEkkXuQSgIpy0qFgM/GcV3vT+8njlXC2GEMFs3ulolHkGoqWAgAglZr2KIw+3mR6zV0F7U7KsB7wz3VBOzC5XwucRzLcjHXBV6nqayKQsIglNU6lxCNx64teNbUvr/IpNPt2cK0mjp3fAcCgJAhwPzLQJD47iK333WO0VDu/Qi43iobwy02L/H9zHbkNEjTCQDPndRqkkQsfCQBc6OaI9B1nKcLmaULfa3VSCT4whd/TueqEZimcXcgq2wtq9RuqtFsO4mqdMIW87m+3mx1pF6ryWw2MfVZxbrTj09uqXAwVuu1ZNlK1qu15MVad+xwMJSz0/vS5LpxLFEQSRKHqsqR0qAspChQkag7p/qiWPKM3V9KXrBzIhUwTA8/sw/YWTqB1fZZ2HF+99TqNcmzTJJ6Uz+DAGCG2NWL+UJ6/b60Wi0Jo1jiMJSkxr3ZlFxdXsrp6bmsVROxoKU+S1apatNlUK0pIulyJic3b0mNZ0tsAcqiktU6VfXP7ma+dGH1+Uozj7rYgUhhWid3As17QncjaGK7Z5GwLDcC0NjRALVGQ4WeO4vqW80QSCSzdC6zdCYdFj8QCf6jr36qaq1LNUVFINIvnH3XRagkXS42trfd7qrkzVVFsXNE+v2B7sCyKGU8mdvEMHmotTiW4d6e3ux8NpPx+EpNRafTkaos9LrsVtThM08/YSqvqnSSx5eXujNM7Yr6IOykIq8kTde6c/guBEvfp75JqVqC3+OkroJXFdVmFzBxq9VakrqZlmarbdcsKtNENq9SrzWk0ahJt9PUSRxNpyoAy/lC0hW2WSQrzfbq8rFYRaE+g96rlLJ3cOB2rM0nwongsvC8wn0z3ONhEO13leACR0yyDOFmEwQbNa7Xj0IJMWoIP8JTVdJyppu/1+p1qTfNJ1AtoLrCrs/iB8FKnw2NogKgbywriZkI20R60UFpM7Jar1QQmHTUGJMVRpHk2Voa7KYoVHWdrnJ9/zpfq31knDzxhKq0+WIqy+VSbXCr2VQ16SWZSbh549C+uBJJ05XuSLX32GB9OdS/qY3MCvUj+LsXAH/PNqkmAAy0hm5SN3FoAAlsVzbbHUHezadIVJuwO1erVBqNunR7HXXmuN7rb96VLEfobeejJfAr9DvcPWZrU63MGs6xmk6uWbGYuQqp7fxg4wSiAXW3l6W7mgkArlPmvDV2P5+1EUoSmTmUKlc/gNFxZg0tigDEtbp+JgjMkY8De1+aL6QKcmk2zDkNPvGNz5jpX+fi3iORqlaROCv0tXiN51xIu9kwB8aZBnyBZrMl67XZeWwsDzwejdQRYdx++hm1nXi/+rA4Zoupqno8dLO/Is88dUtqSax2FbWdrlYqOCwGThL2GBtvu9zUvH6nLrhNDq8xAQx8GAaf8VpEVTW+jnu/F4CL8wuLXgp2U6kaKkoiaTTNmZrOFzIZTVUAEBh2ThzX1F9Sp9JN3Hq1kEa9Lu1WSwIXPSxTmxtuLlRNWUi3P5DxyKIkHoDrpc5kIvD2CF5Y7O9BaHPHnDgLgFeEbdHXe622Ljg7P6mbf8RIvIphAyBVUS6F2Nyks4UEn/rlz1Z4ksv12rx8N+IglmVmO7oxX0vgJhTbGW0cQpFWq6uLyWDy54uFqlluFH9hODQToLbT7+YqVxOCw8UEIwRP3jzQCKReb+oCXV9dqa/BjkEFck3MDBOOkBWV3xGmRxFG7o2/scOyItcdSqjq9axa68oHXiKFBKpt1OHFp6hKFSiEWh0tVK1zQq9HU70WmlEFIIpVYBAC1hqfAa+f7+73uirwDASae0YVI8idbldfL91cpEv8glyWy/nG0bYF3wqAfz51DlU72wKqo+w2wZ5zOhvdnmqeXseioHSRSr3mQlm0VbWW+cyiFUbwS7/6pYq4OEW6y0KWzvFAAGZ5Zo5EYQ4Ho7bMpAY+QGi0zqRWb6kTpzs6cVFDHMvVxaVNQGbxrnr4SWwTV641dOu0O+os8h1PnOxJrdaQWp2wMpar62vFDtYuXlbbidYrK8nXmayyTB1QbC8aYJ3nTgjACzBDJrxVwaL6UNSwBLADBkI/ny91V7JA+n4VgKaFvlGgThtjvkj1WrroGslEErqd7zdZUbIwoewP+zJz/kC6XMnBjSed47eWdrutdzObjNQsY+qYAx9BoN4QPEbu9iP2Ht/LO7mILssRhyYEjJPjE30G1qbf76nGxJncDSUbzabM5iOZzycq2PWkIcFXf+1l/ZpVkemkee+U5566h0cn4SMkmAnChyKUQAWikiYABpaoCqU7MOlGYgFY2LnTyVRfwiFjV6oQJZEKQ6/bU7+CMWhFajbqjabG/9fX1xoqYQIYqG5MBraUXa47uSjVOfImwCswdurGBLgF4xphZN/l3wcAtVimep2kWbdoIgglji24YtK97Z1OZ2oCsAGqSSMwDIAa0woM/IXuoC/NJJZFaoKfNLpqgnpd/tVv13k7P3tsgjWdqWcuDsPwfhFX3BUAm1gbXpMOep0N0DToDeXw8ED/zrxlhN8afhLFWdi7Xq9kOhvrfTIU8/mlb35R7z5jZ7ldjpQDqmReTTFxzm6yRRrrXJpZJVFZSiKANyxeTYaHJ7KYTSTCA1/bwhExpMTOaapoGCOOUdmBARZhpJFBM15Lf7CnJqDeaMl0MtMHAUdgUVCjIJMKFkWxfp86b+jfwHY9Dh8/47SulmZ7iTD8LkBr4BN4d2uRprLOEaZSGhr+Re/DPPAeEQIWbTyZqRCiJ3hFneACRG8h9Zahlv1BX//tDQ4tlvcOKY4kmqVRd9FTJQ8fvqe7foaZA2zTkNC2s/opTgC2hs45fuoHMBeh3Ng/kBde+MBGKFarpTrmPg7GFDDHLPz1lWnkuBar5tUwmPv72je/pJaPxfcQb4YKRSKx42Up6yqQMggkZzIAA516ba0L6aENkPR6XU6eekYdvcvLS1k7ARiNrqVRb0iaLnRBsXdVVUhZrDVM5DtrtbocD9vS7rSl3mhLp9uXJapZoV8ACiDbqWqLRqOli7SBOtlWVeWQNWDkhqpbVCAPzQR4J5BFJBJBEBiz5dKiC40I2ipcLGxZ5Kol1Kl0izIeTfU5zRk3wbC4PpO9w2NpNloKjulfk2SDxBniGUi9lqjvwGfYDA8evqsLvUoX+m/iACWTgMo5peZvcC/q9avpATaOpNvryzNPP61mME3nsvbOJm9SP8We68ot/GI+U80xPDhSlDKdW8QSfP3XX64qB07kzgniD+vCkCt3P4JQMG3cxiJEC5QaLWAWwkqkkzTluaefUfutu9DZ7tffeFWvobPnbDiTaurQUDzs4c3jPd3ZCNBgsC9xUlNQEg0OXKyRh+ICht6htlV4nAnp9/u68AiZRgJohAK/wELW+Xwml9djtblzF4OrdlHtFkgtqanq5/oIqGILUSylE/bJZGreuD2NvY8IO4qk1x/oa7WGxd6AP/g6XIP8BoPIBk3DDl0sF/L48X2HQWDSKkVH0Rn+Z3B+72+ozvEwer0mLzz3vJtSdJlzDItC0oVpPTYU4S7fs1zYwjPYOIeHNzYbvd2uS/DLv/Eyj6sfYLaJmfFt0hJ0yz1uUcoqyyXBBgPvRqG0glAus5VExLfAraXIDz39Qf0iUCi+DM/99OyRvoYaGk/N+2RREACFeRG6qpR2ndAq0c+2Gk1T93FNBYFxenpqOERearJIgZSq1M+4FZG6ey82z7+O6cCpQljuPzqV6Wwmk5kBNhnOn4uvETxFJfV3Q/ZYYK/K2TE+AlHRrbChTRnuH2yAJcUH+HwYqmCiwYC2DfnL1RQiADzvo8f39fo4oAyLXsy0eDPjIx2itG7XJXtE5NbJTVmvlhJHocxnhj4C/eb4I6yh0+bj6yshb4DDd3x0bHMfBep/+RF84ze+XCHXpm7x2m3XhxVhDe6/ec6qhnC6JNCQZ4HaxX4WhdrCsChlPzfk6vkXf2AD4a7SpUxmUxmPR2afXJilUGUYyNX1lYrZyeGBeqfcII4Q3n9Sa2pWj/Ho8WNVsTiVS8UhuFvTAgpMhYGqYUav11Vc33+XJouCQN68c08m04nG8gxLLjnkS1WroZdoKCIjDd+ci88GQd2zQ9udngqo5hjQWg27R/VHsN3lVjPFUaDZTBa9KNaaVOKeHz5+YI6k5hWijYPMc0QOA1CTo1okkUbdBJ3XDvaGJsDrlZQFIWwopYveRuMrKTDfVama4MaxLTzX0URbGMjZOebV5iD42q9/yZBARVS2Lke2LoUsIQP776ZJf6+5GBxbugCtk0CSIJQTvN/ZXAZ7x3J0dEPfu1x6L75Qybwej+Tw4FB32/V4oruDQay6mI4UTr1xclMePX5XVmugXtsZhiuUKgBhrb7JW3joNCKh4nYTy9BsNuTw8Fh3FHE9D37/7FKmM6ISe85dAQA3QfDQAGgYhH295vnNXLHjfbSB2SAlzARzfR9mYkZMALbgEzOnuEWeKZ5PKMmCPX700BZATU65AbB4DeFHOHAOdb6TUJpOyHh/t20CRwTinbnLi1Ofx1PN0PXJKNWGZjpPL6ciFdAwIWUuWbaQ4Jd+/Uuaw2MyvBNIuFVWZOUcDJuDnjkPVeMvEx8crQx1g3MCp4DdoBApOzKRvb1DhVRNg5Rqf0HXiAw0o6h4tWmY2XysjuNyPpODgwM5u3gk6XIty6VpjbPzse4EUDbAJjSPQZ3OuQhFSiew2O56rS61qJIgasj1bCFHB0NZ5+Yj+OhWkzBOlNDcgEGqIsNI8/qKxfvn1meysJDvZPcTpukOdQvvclES1UAuLQyMdrACoiUEGfNwenZmwqJYi0UaOrgRB3USybDDW82GdB2ww1u6bYAqdn2uKXk0MMASDjAjjkNpKp6CMGOKSjm7NLOHKeKafgRf+Y0vV4pvk15QNeXZAjgDYNmVrFU43P0pZmDvCYrCnBvVIJUckntWj7dtu4XoAQEpSZti9wNNFU/G1yogZMj4j5BuPJsokoYHvrc31JQvIM1kbDZuPF5KrVaTVqutOwZ1r7yFKX8vpdvtC+YGiJUURhTihwSyxDFyIE+n09fJ2OIG9rwa86MBdIHNw+fzYBseCCoUn4/VJOqzhyB7OHpGarENYeYQoQ/cwkd4yHr9UBo14ONKiqqS0cj8IdLTmsp2IbIu0nqlrzMXzUaivg0RkjdHzUZTkVU8/1W60ufEacTsKTkGuLwq5PRyrLkaP2ZzyC8kx1bOyxEJvvFf/oLeoeHsWygYN8j5JToR/AUgR4GF2FTwdDmTBKTVSVZdkafKkMHQbFa7aXnxnTBAbxoHb73OBO+d8d7jh7oA2Mz9gz30ggC+kFfQicotXYwjiA+CDWaxvBDV6onBvjiMjx/J0nn65M1C5/TwGRaXlK8+s38mxUAMDmbXtNVkJBLXmhZeapiZKkoZEzWQQFJHy+bMCwmLj/Yk1kZOMCV+Z+viqfK0nT6dTPRZlB+w42vwt3QxkfVyrjuf0Wm1VFv6+4ZrgXDgWHvbP+hZFHT3vXtqmhJyFTpvRFBmhhlxrYQUpPe+XM8k+OY/+nql+LZmroi58PjXErvEjIZu/OdSakx44LRVlWeSlBYaKRZQFZIHgdmwMFZ1l608IBJLr2fCACKHZtHs4NpU5WRFsgjVGsiw11bnit2LEOiogJwM1kfiid9h4yAUXrv4h5zOJio4A3gHZ2ea3lVDobaxa6GqJq/EZfhQ9yYAwL9oGdDIerOjr7PIJKecu7JZdEUBcRb9xgEoiiNpt1saSprjsJl7fa9PcUMwYWiGUV/fvm8+uZIyTaXTMYBpw1PQ8DmS8bWBOsxzyzm72Wolj89P9XW+h/lfr9INqwpYnVFrMW+5rPNUWq09CX7lH/yibXsm0mWQ+FVthXsw7i1dO7uBQHjqFTbSwcMIStNx/NZZKRUCUOUyn5v01SKk2Z6y0epBcZGShEkYS5GtJStzSRLoXaQ7YR1FKgDzqU0UXgavIbnsLjxthIQJVxKJD8F0UjPJVjCazARh0/WZQkLNaGMrlUamwFSlnjpCqc+MGWu2pNsbbniGq1Wm8DLOIf9Dq/LJM67JgAOhz1pPpNK8gJNYnU8TFnak3ZcztUaC2CyqLuB6pZyCm7du2nUTizgY3331W5o1ZYCYnl9d6M+tektzJ6pNALSIJBQ9tc/ZXMVSBAupwlQG/Vt2e//ZPzQBqDlI1c22esqoWp04lyjSX9gNC1vUitc1pkY91zd4Nrl4n3ItKlKgmXqsPjYPYxg52NNInTm9kZCYmd1dSlDl9j3LVBazpS56ErccT46YA+gaKQ9VU5ktNp9FJ9DtTPu5lMCFekkDjzhW1NI9ii4wTlzhcvYsEFgB6r7Z9plOkkI4kJkmzNjxk+lMX0N4jocuRnfbGG3pSZikeQ39JDOIo1aze/WZvTJXQV/MJ5I6+LrXrCl+cOuJJ3SeRtdX8uD+XXs29cFcqO4WS/EUF/+rP+M0NHSwhieLsolXqbR7TWm2SevjaxQSfOO//VpVAeYocmUTGGPbcG7cnq3tIGDgBIVD+ULQ4uVapnMcMRynjsaf0LsktEkGUWTHGuSKq41tNDjXkjpOK9T4PhyKXJKwktl0KtkawoeLRFzoyQJFQSmRcgIDSeo1oz/t6FAm3Kdj1S9wu63R7FjewRM1sf1FKZPxle4UH84prsGOqQFI4VTZDiLBg0ng77P5Qk72OuYLOM0G5UwFKwhVozAIdRmQTZUU4u7To3NkApeLhcymI0lXC/Xsbx4fqx+yf7Av3/ved3TZM6hwfneWuT4Hc+19ANZQF9/lStAADJ5vk0YuS6m3G2qi/Ag+/U++oWEgVDBv2yOcFVKSRSY1CaWGevUTjORqfAzZIJH1YqWxOwuclU0psoVyBPidBV77lJYmbAxgx0nBgbMY2CwQDKFGDSewkqpYy/XFhdGonbmpiF8VCtag1RI3DgjdXmOLGXjbG9cSVfu2MiRHYPOaqjYeY6VOI1lIOIrsfHB2aOvz5UrVrVK/okiWowu9Z5zJw+Mbqg1a7baMp1Nz0Bx1/Ho6lbnD2ucOB/E8RbQpPEmSZgxjOxcyujT7XXPhHZqr2zcwC20HTO8jn8rxAZQP4pzKbJluwlQfjqrGyEtZ7OQJhsO+1AnNHRoWfOa//7tVFZQSV45nhkMogTSRa5/m5DX3M4LQrYyPh12vhy31SJHea8svSAUZw2kTbLVx93D2bLezqzYC5UQRLRGw+2Ni51IuHl8Yhw7ynfNmFSaF+BGJrNOVYv1cF9WMlAPOqJnZWgMNyUDqWMS1I194wiq3aHbf4vlWuyu1elMZQaB2mbJn7Z5xRvk72qbZasnNJ57U3c91HZ96w2B+fH4u144i7pNWLCjCoB58Wcrpw/t6XfWDSsCifGNy2V6whzYsMGy8S+k67pMJBmGzQwA9CuSBTbQgg3liTsAGGOADaCpL54sEn/3Hv6JbMGGnsvscGNjKQinm/z9SFAbS6KAHbCAIndAViVSxSBpIr9VS4ORyhiMFZWy1k8jYFpNoYYdCzpZlsB1t10Ul1hLSxJiQQCbXmADy+vYg2EGwdyYQG6zopDqqpuJYzFpsZgfkz+9y9RGwkUFgRRlZpuGTTr5uFXsynEb9XKMhe3sH6rMgBCw8fgqoIBQy4m0E4PDoSDWBppKhyON45rn6LW+9e0fqtaaRYBw4Q6oah5M0sM6ji/sVzmWxSGR5LCh2oej7aOBunopKWT32ADtQ9gZHArsAd7G0eeb4FE2n9mNHJ+fjaJXgy79sTmDcqEnR2SYJBqtKltlK8kgkny0VX4YnN7g5kNCRRYtVKY0coMLCusl8rRAs8K8yUhypgn9t4T3OYKZgN/Spq2cL4QQtEMjVObkDTMhWAPi87joQO43pTTV7Xr9H5AyIcdpGgRr7OS/JvFlixg8lmABoZZBV2+og3ji+qXH/eDzeAGCoeqIVFhIz0hsM1Dvn+71z+869d/Wyp9dnMuzvK1iTOjwCv4FQMnT+CHwJTW6BypXgAlNlIjNYOGUD7wjAFqCzna9DIyLAq1i1LoOaBzVtOVEVYNb2WfXa78syigTf/PXfqNLFQvLYOHR63SSQvUZHluMr/Z2cwByVaKCARC7CIVSK8ag3KKGvXuFKdi2fUgZnV7hZKs1hb1wKp7OUBVXlEkWQPEROH5ypUGFCGOfnjyxDCEZeVNLqdM0M1ZINEON3PTuZv20Ez10DtM0qbjw2Ad5uYSC7F5o4ztr+waHz+tcW2imIUyozGlAFEwZWQNaRSYbUwa67vLpW9nNIWRqElziR8WzmEkGFap/ShYdeoyrcXlpih8XRTcPi66baqeHaiCy+y9pIok6NKwH3fYZ/C+gpxO2QUFvLQsKd9wb/3T/9TX03WPJ8OtnsXGxTupjL8RMnttOoKpmSvKnk7PzM4N8okS6kT4g4Eki9bo4aaWDsN8NTuqi18xJNPKq4+g74AVxMdUxVUiJWyeXZyBNb9DqPz0AKLeuHvu/3h2qeCOuAgInNtUII9LFNyGiToLUH7l7A3dn9m/g9STa/I6haopUkCiBxLTiKCCsDUGU+m6r58OROhV3DQM4d//Hi0gCawxs3tV6QMZs7Slyeq5DwjMpvBDXk+1kQ5olKKrfI3IPG7Q52xtn0QstbNtVCu06C04z6zBBKPA9UTeT7pEdd503BzT/9H0wAoFFtkGDq/q5h5U5U5UJq8HTr4eGB8vX04WZzaVIYoszZSkKnhvDery6tXm4+GSllqjc42Hypki4dG9irD5IdeZbKfD6SfL2Sy4uRwZlOSiaTKxVOOO+oTkAa4xJUWrbFbmExGLCG/NB413m8SsggpeuSXCyAhqJawrZy32U2nvib3MPB/r7eN946cTopbQSg2+vJo8eP5Ozs8SZTOhweaJTSHQ71euAUk+lYkUh2OptBmc2UnXmyqwpEbj6Ig6gV3aTqyCVtTIi3qtyDXn5NdAMXpdRaDf1OhfUdvoLgWjWRQz+5DjUZyjwWCf7BP/xH5gSCEu2IypjqXkdsBB5V9KqyXLYf2FQQT18xHCQUJTQ0571wkr9aWmgwurreZKGeeuZ5dbbiRksi/c5AeoOhTCYjmU6v5N17b0m73tdY3NOySa4Ar3qyCbx8FhMsXYGmnXtn52KbGSw+MbQJBth7JrmjqzEp3iQsFguNMJRZW4cgGqsjORgOVfO0201V/7qLi1wurq5kslio/8A1GSfHN9SU1FqdbcWPmy98ACIWVD0aCSIH82b3v92RXId8CALgawWM90DEYYvoi0OjOnwER8nf+DxkMgtJnKOOSdJ58KRTwcRaQgwOZvALX/55QwKTWBodw+D1A1qeVAl1w4SDBDuxrwdwBgwvnJhW1Q52FDsGG0iBI0dsxJHMSEgsVY3adyU6wcrvi0I5Pr4hCIrFypWcXj6WfG1oG/ac0Woj3VDPY0mimiQNoz1NJmNlKmlE4e6LZyCcY4C+cY+aso4Dq851GkCdRvdMK/Ia5PdDTFlTtYImmlzMDZVc2UOOdwDJxXiDHgYyQgmjMzzYlKRjMjR1qzCwVTohNNcjMxdEIWT+lG3kF9HBZjCWuJ+NOXMaTkEgRyL18D2cQoayrCu+zzTG+Mq0NWvS73etisuZXmoUgi9/8fOmAZLGJnTSh+h0JEM9Ou6Nu1vTEm63UQWDnSp1d5aSTk3aYKf68mRV0xBDHDuYv2vBR1lKvzfc4PIJeQTAISVmxlJvWhjla/3YQdhOVDEPQUpUU9VU+0K4DEOt6bfvT6XhmLrQvpdLzBs1hZSbZbJy2mmwt7+JEEiVoi1YVMgUzBET5c0n/wIO2WJQar4FsXzqd7Gcq2AfnzyxyQFZCbgJgG4S+iSUpcwcPQ7eHoLmq6g7na4EJNXAN5y3H9Uam2rig/0DTdbppsOPANpOyD5uyTzoBOZlb9DXBQbD7gAAIABJREFUmkY/9vb3de0wwX4EL3/pCyoAYWQ8Pj/wiBtad2+5b7aXvyEmggGgQAWsFPDcWAzb4XmR+bT5hvM2HY80hmdMZjN9aLJ4vr4dxI7v7/cGCoKQ5QJONVID6eetU6eT6b1lV6WjThcJKcrX2Dk4cUki83StiSwtFlkvtLw7W1qGUd9bVdLr9ZWG7ptHdACEGnVpKyBkIgANbangUyFxDdgYW2ssp9hV3vB8hwdHCg17B9ebTBPmLc2MEFE3wwLkFHzD5g4MAvOJj4BQEtLCfwTy5n5hWPMZm+dCF1mF3jmdjYaZYDQAkcqGM+kWttNtWzmbG8HXvvpV4wNsiCDmRVLx4wsKfAMG3ofa9xAVH8SJgoLERE3GlpmC3rVxUPTBC8X2vdQDnVYloI+pVBvGEOZ/nCZCTMwSQmJ8NusVgA+g8KnPVLqaACVauGdgIehZgMPW6Q2kza4KAjk9PxVC3snI7rOrWsLYSsdHN6x0W4UnkuVqaSFnafcHTZ6QVIsptASN16xcnUIXRr9ni5Gu4QB6gXWsXUX74ArGjlVkTw0qyX1788NrgHYIlo8k2OEe2Tvc399sJF4DVj86PNjwJnQmyRE4wYW/+P48iZXoMc/am+Dvf/PXKiRzOqe0e+vg0YDBpAfsnuqfbcxG2lZvXrn3F+rc1Ii9cTAQBi1wjPRhfeUPIRT5f75jMh5ZCKt2y8W+OD6OZcNDIcksuJcPnFSuhfoi94ANpq7PfAmyi+QR7FqYABhCCA6UbStKJfVrpW5w5PV96VxBFc02khFUM5NLFiHohQI/Vn2ny6I5DqWrAx2HkaxWJgDej6Aky76fkM7mkrmlEMPXSlJgq3PnhBU/BqcT/4lB8QjXQzOhxnl+HMtuu72p6Ol2LUfANbU0nUYWTgPwuoXYdt/+2fnZwke0twnl1dVYgl/71b9X8WYFXJwAaFeNONm0XPFNFDZxvJsTKFyL2VjVzXKZS7XjpPS6HVlnqw2EDC/fqzmt1tF7MKHS6MLdIIkbLRGrY5KMdsVAnTMZYOLT6UQoOLHwpia9nqFyPunDZzwoBDdwQ7dy9XUTpWNV8vj0oaxLI3x0+z29h9kqlUWRCT5JS2vrHfpOEwtYyhGmKpJ60tREFaYMU4BfQvGHPZVFKwqHAQj5Jhc+vapm1RNTwQGI222LgaoSwfAbuQ4/S5hBBn0WfOhsuRG7zmKxRV+JgqDZqQA4px4/iIF2zHLfnwhK2Nd/seKBfOcKVFwInw517CMC10ZFpa6ihsAuwE6iJt7sTUfKTZgSSqPZMMKkS5HqB5QKNVYcgSZPpS/c96ENqWIKRAFCtC+RAVAqAI5PhxCgAR68944JSFKTTnco+BCEcQzk0DdnGg5xNMET4PxRsRNtsmOj6VjeeXjPVG09dg2njBpOfEtW0us9cutRQK+hUIsqD4eUfxnNK3SJ2pbDHyCPMOpKDeMZtp68nwefxSPks4VCg7luIloCv61Z0DyCi7bs85Wcn18qSOU1rG/otaEgaTUReQqPiZhGWmfLjTlR4fr85z5ZaYxfUGSxzQXg8DSaVuzJ67s+AnEmjhLlWpfnp9qpig2dV7Zjj44OVQBYNL+APKA3MWcX57Kab+vmo5pl2BQCjWDskK60sm4PgCCY4ACbOPzsVBcqThrWDIJkDzsGFAyzVbdnOTrAa7aWK7Ce0G4QI/YHA/nuW28Iuod7V5qJW23VHkoCsOLPWtyQcLGwli9KR1PwU4KoJtM0l2HbbL+ny0NqZWEo+yKFy3xp0WkExx8YHJ/JkEGuz++ttnnmgE1aQMKmc9KCSYMg64fn+1ldwlbVJ4ldw5pveBh569gvlyChMJt2NMDP/M2fqobDgQI6HmLUG1nDY6dwoTJQxScmoEtpAwKjjfnB/IKFM5584pZ4O4Xq533z2VwdMMbF5ZkcH98y8ojLzYN9K/yJ71CrqV1kdLvGi8MfQX3hCwA1p6RCHcXKx8nsRewlSsNHJLS9WeULa9nWQivhTLruGAFAk2EI0N49JQz1qSXnWSplulAjkFN46dvaQBItSxkMjpXc0nD2Fh4fA9lj4VlcFmm9nuv3ck3UszGJrVEWYBbPNhhYeRk7HXt+PRppfQG+DGZsC9OixlPZPxyo/+Nt/WppKXiKP+ivRAQBBhIF9qx+zNOZJE7T8Frw8pc+a1EAFGpHYkDizMO0XQkw4gsQeK9XX7vgi3a+cqrviSdubRisVhGUydnpY7k4N3iYkDAOY2XP+tq7Zrcv3X5fQzIyb9axA6TPdfqAtRtEckjvHdrR0P/Hef4bsuaOo5q7LBgdx2hOxcgdfbrRMKHSLhwkiJzTpSUPqo4TWS1mMj57bFpMmaiRBEQAUSydVldu33raml7grTv8I3Ya5Nkn+jKZmKM5n/mda00sQDv5HlroWAxvPoWHse22iBSspd18RXFtSzOVfniQBwFgbXT9Mvvy84vHG+r7bDk3n29n+Aprn6sJPvOZv2NIICyRnTdSOu3BhfVypU4JE9VudTc3i0euuXlSvRAelKMvcnC4r5QlwkkkEYcSz/96NJb5Yi5znDAHkHgToQDKjRvSHw509+fw8NUkOM+e0ErBj0raXStv3vALfDm1y54pRJzZbsxKkDjvkVNYStjmqFsCMGO0b43R8arB6hcT5dXFtMfRXoWhtAf70t871AlPhNi/oVlAduxsYYt9dGjdUA76dbm6NG3I31HfBi1TNGPP0+11dU7xkXxnE0/fni3XWn52enlHzi/vSS3pSqO+FYBbJ8/KurhS02T3JzK5tkwngzS+L1Ozkp2dsUNUhkwSfO7zn9bZUS/cefH6e0Snr5pW6sA329KuShlfGVf/+OZNLasGlyeUWixMAOgc5r1QL1SePE5pFtDwW6+/bt0w/fbVCTyU/f19abXNbnrGL9fk3tBK0JtUJapqM4KJhhSYJHetvFxrGTajqFaapWSsVnTqtE4i+oyV9T7S68O5y1LN1mmNIHzDMJF63Qoy6vWOtoGz0BPmct28TdS5C4uH+3tyzsKvxxsufrc70MQYkY3y+13FzujKimabdYPftemmW+MsTCSvAqlC0x48+277nls3bqpZq8VdSWL70MWF2XXC551oXh3VJN4if1p847kaoK9f/eWvmwAExo1naIKioiOHkx6QIx8i5qWMr0cK7LDIB8dHShTBOYJTZxNr4AoDEIZhJAdbCKBgGLUsqBeAhaZagXVr2kZF6/RcaMRnwP64ptLVS7peQEB1RBPXTxXo2iYMYcgkJGIpyQPY68sSodDGavo7jCNy8FDgaoA+NJgscmm4ohMVFBepzDUbmBpDie8tCNlIOtEL0MI/KpcZrVogp48fqDZjJ+7RhlYJGrSPtZR0GG5JHQjwfGabapHNZZHDUgaOp3lGbB1GXTKH95wcWScQmNJEbIzpxHwm2u9UJdrR5hoKmM5J4PmSO02SeP0bv/Kfmw8g72fKKK3I06zVizZKOJLoixqwyfVWUz1WFnfYs8Um+fLgvnHefNWLMtV9TE9xp7ZGs9z8yZNP4VFu+gqQ3KEqBlXuW7DNphPtwKGxNdHBpmbBVW76zhpk2LTxk/d00XkOlesZcukLQ/i3lZif02aimHVYNogMPg00L40TrLOY1RUC5xYyn0wlgBlVBVJrGDDjRyxraTTNqet24EFYo8c0n8hsfq2+Tb42s8GOxFkkZ6Fzl5dCUVXSCGUILE4I6UyUv35PkU3A6KasC8u/pPPDDc2uLCHgmk/gBcA7yjj1m9JgBODLv2ilYWDNPKgPBeEHaHcGJHtJWZTj0tPVwjUiUCCGlKQjd7RdkmE0GsnpY+uBYz0FtwwhXvM9+cAeEATaoGp7RUXi7DtpxkyjBd/PRpNK5BxcPUCj1bVYPYdYoTVDG9XHzvZMIuUyOFbwKsjV/od9s8MUsVLGxt21k6Y2uqAppF3LFVyCRwDKaM9gRy0rCplQsKK4hy3Q7gjLQIb7FhomNcAgBCqT0eJM0nSqwuUx+oBrlqFcXNh8VRGUb/MXet2h1LDxAfS5rTfXbFiPpCwnk+iYVyv8E9cDKedvdkde2PXaqM5ttGtz8IlPmRMYxOSgXUpR6+RDSHR2FU2N0r/P1FasLWNwkOik5aphGvVNmpFyLJ+FajuzQLjlF1NBix06EGGPduMg20fKVAoZ9PtWEeNSrNu3u/QrHEKEAvx+tUDVSM35MOQuIg+5lqIl6zxJVGtplhNwiDGeTmS2oP1bLoFWF3nuoJXIaRhHgwc1e5dWMhYnqiiynKJWS1Er0Y3Q0PXgRZ/eunWi31FvmRklLL2aPd4UjW7yFou5md4qlL2BNcu0Ihn8FdMKlHHtVm3tDc2mk40tHT8zXRzK3tDun5ybLwd0zdA1Dc7Q1ro7TKzgsy4buCVs2ppHNHJyE7JeQAp1JE8rATFBqNW1FNpgfeuuxXj33j2hPRpDETzNNm4n2Fg9jjfge9lMRxZNuJZtnbY1bfI4ga9S0othPrTahFqGXEp4BtC9HPmkqeViLv4NY0lhO7EbB/vqi3htBiCkVURSGlfeHl0ZwBZhoO5N6K/OHjkfxxJd9EkGhVS2kMuuEQ2gDcHzn376Sa3Bz0tHDVtcaTEmzilci9h18KxoWlXRuHnLYmLKgF7mC9d9XJuObLXMyfGehqrqjhRm49crm/vZLJWgQlvbxpxMt1gNDbngU3iNyN+DT3/J+ACqppX7aF+UJEi+fXhJGZRLVqAO2QUsINlCNMBGAzqBufvO3U3hgcewtRG0Q620lWmcWFsaN65gyCipg/LyulbvaJnept28ia063oSGps3U4YP6XK7WsnZ5fhS8z3mjCXx0ENU6tsjuPrVPsCvi1H68yrB18DOmQWv1LXlCoYqaC3c/AEeEhJhM728AFDFOjmh2VbemF2uz0QjacjWTxWqhu5BIiPuk84f1NPL1XCKdZl9VfxybY2gPuhWATntPndHJeCWXlzTTLKXdGsp8ZouuvYRd+Fysnblz962tcHyrUS772Zc9H2DbmBnpr9PWVJ2wQvvX+/YuYOnDvsW7pEXhBigJEVDDtYd9+623N4vtBQBb5DWE5hnCUHlzfvdo93t9RmP6Qr1WmpdTV555xG6l2KLeMp/EWshZ0WZGzQAg1orQzzSAsntc5q3e7ivi5wWCKfeJKN2V2m3U9yQ2RnJbu2qHcnWG+raCK7p00c1MtZoSWaxDqW/zBrG10IhDpDswdQ0YtVyPZZ0DYOUaai5TzAhdR+l6Zg/a7/ZdhRSZTIuqrJGka8ermsmefTquyfU1WooIyW0QoTydTep8g3wLBesGwm9zWpeQPPjEZ61d/C5Dl9/3yG07iYbNQ3dOL139rtlQbXPmyBaK3Lk4fHQ92jB0QfXs+lvDo+STKFRiwyb0zIxcoa1h1Sm1Ch8rcKDszNG8AWGSeHMQg7WMt0ymb7iUkqRyLe2JKHCOdPHgIAIgeeKLu2cmACcNrUETasyWed44maYF33nrexr2bQAWV5sAeumznHmx0n4C3X5zc15AXHedUpVUv1Rvn7MFVqldt9WitXuk5FP9vUlSK5MwLJSGzmBOfA8Cfm+3TvS1x4/W4rrAKb7gR6x1lp6VvUsJNk1k/RVtPYIvfeXLlZVtbVWMMU0GUvcOGO+EUaNNFysNs4z+nSlrB5iSeH7s+uVDG/P964kuVHBchxBWVev66DgGGcIt7HyxdC3dbAdqggaqt1NX1rsXPiA9f9hVVmvI58kNIF+9dlvmy4myYxaOckUq1nIXBufqNThyZjHTmgYrig3UpisbN4rk4vpazRG7iJ4+DBxV/BGtSNZ0c001FZ/xzjFoJyNphtJsOiJJYYvoW8oHxP8BRamuYZYzI8dHplWns2vNIpI4unSAG3gM3wdJhWce9qzx09Ul7Gtbt0aDvgdbck3DdUuP461g8L7Vaq7NLf0Ifv7rFgYqrr952R3Z4gQAL9izblG1JHYY7ODFYtsL2OP28+VcBq4ZBF63TopL9vCzZxEr4OQE4IrjZcDZu33j5+tibbGDrQapNFcPzVqhCayE69vjKWxXF2dWc+cYtpuEk0vCeHaNmq2maQUIkso+CkMVAPB2tB6LrY2dCBq1ptFawYG4aUipbCjbzb5PHwhes2WvTd15CzCT05S8A6q8kDKwOSSli5B3XMUyLfdJZWMuSQjp3MGRiEOZLy2X8vzTH9V/R1eFjEYOfKqbBqG+ksigctHBbvSgf1dG8HazB1942VrFKsGT9qe+nSt2TZsNWAyvPF/NU5ebugCaJyql2VOwfWPl+XyTWYTJwkBV+gJFZa9onz2+0+X7SQE7UHo8nhrDRxsjWMjIhLOJEQT4cjSbIjaHq6fOG86po2ZhUyFWMOinN19QnVNprx3e63sWa2cUjlZRDWCNl3Aotcklx9EEpLcd+4WwV5M01hwStJJ74u+e7Uyih7FMLySquQMh3Nwwb7Sl5bv53m7brktDDOZgNDZnEYR0b9DVMjE8eltUW7RG3TZTv2f/jq7mcj0y36BVt7CTURaRvPvu2/qzldxtB76LYgtuBF/9xtcsCsDW7pBCkeqFCyK9cOjFy1IAehhMlO/Lgy8AGVGvxeFEzsOnMEI/twP79gdDzRYam9UmotGGim2FnnZQkzkmXq35qh9j1hLz057NdQutNzUi8bQo3ZEbziA9fOyUD6ZCKV0bgqkdKsXwLd1R/XAZiG4o8fLAGItkdRG0qwO5m8kyBdKmy4nlQAj7GM0klzA2ehhJNptf0yClwrTAvJZSN3ZBqYkyPwZ9StRimUxNu7YaPWm3fa8lkU7bFnU2WW46sIyvl+LyX+Zcc7JJvrLGVjtjhzysrwZf/cbfNQHQdmdbI0ByguJIBm1MvQrm5jelx2s7usWPDRPWIYC87gXA9wnmtf3DQ+nSbHHnulCdp+NrtacaxsHzo/DS7SAPI+tEu9M1jGfnqOslh0nZYpKjAP9nt+dVJpPZteYR8OEhlHqqlbav92XslfXqwUlqNdoS11pSb9N23XbyUk8D0bhBam3AKvLv9uQc18LoOWJII6FBhD/pxMAXtBO7WkM+PSTK1LDyCrUPAk2xbTO02mg++h/Y7zifu92+pmPXwn6SyXxuQseC67MnNIfmzAbTHhB3/HX5HWi61dw5R+irX/uKQcGal97hlkeckWdqibN+yILZYlt5FQNhoIGh/5zXBqhu466hDm1RtCjc2XvID3QHi2DaeE4bzaTT1M4oKnJpafGDayrBdbSsiiJOK6xYFwsjQCR07SaraG1YdDJQs4FNDNRuB1bKWmwx/VMWFKO6Tt6oRYou4QdQeELaliVoNq2ZBVGWahLXsxfWiZZWQTVzguDz9MP2wHUzBSxzFPoE55WuHpSjhXJ65gpvK6tqunVijp0ueM3g86p0YJYW6exQuSsLLSejVKYT0xJZuiV+UP/gN2m3ZZA0g6wtguVDRF4Lvvkrlg0E/dpNzbLg3qtHvXsuOu/1Vat4+6SMrSmRZ6C6YloXe/tDJHydP58H5YMpCzRMKZneXKOuvXI4mFE7XxOCavNGu/lVRmv3tcxT/A6R89lDVeetRkda3Y5C156tzO70i6EkDxcLJwmoWyqRS99W9C9yQgO1n8YUhE+txoHuPfgA/lkrdUrxF0yFE31girSlbJHKrZPDzXvpilJPzPepN2yzYMcJ7dZrzABOrD1YFFXSauJbbMM1rZwCX3ACoJ/17Vt2tOtiUUq6sM89erCljK1TYHsneDunifE+Ht0LLFFW8Ku/bKVhas93agPmKX36TANwsILf9fxOpA4uTlJnsH9gZ/8wIQvg3FCuri5kMrGKFL8o8PTNHtJQwhbfTt1w+LXr20vOIQ4iKSrbbexSlfAg18YIy2wpBZPBjlCnkEjBVLPf9fxs3jmNrmpSBWgjmkXRKKKm6tsmnwqmunr2AJ060SLy1M2n7RwDBbrs0AV6TxgFy4reDHNw5/s53wd0j8HVPSm13/N9hA3sod+hCmUZSiyASbZhimobmqmmI2pwGUJqGv0hTyZMJrUo4gLOm4i8+bq1ng3jQPJ1KKVjCOXrrWYI4kxrMX13EL3WL3/lc+Z7aysXvYZNOHRn9zukAl8Bx0uua4u+RgWP0ZgryRx2bSVYpm49JRuPH3SPXTOkvToO3057Wl8+hQAoG5mCI2HRbQfN85msi7XMSaPSjTumUyj4gjt7T+FSnzGzk010QrT7uQOxEpInmcTaso5boDzcGjmHIdiGHs6iJAsFTDIqk6yd7d4AJwy7apxDTXFUmEQgblPJy9TM3niykoYjoRzsWXhm9xJKsaq742A99o+uARvZzv14NpJ0DX3bNiCbuF7fmmeaYTPQRt6Jvr7cOpHrNbUCOJoteXR/2ySSz1RUHO8cUhl89ZesWTQnf/ldzoFDemqFEwBtoOgLzrUrh3mWVqZMNswaKESudYmCNr623cXIBrnaBVkcBM6CNxvWq9gcC3IFZCcRAL9bZ+uRZCUagJM7aKE6cExbxySOcwVPdKLVJDmkS8mYlrYNQtfJ1PkdhGNbQbGcBrokECOR8D3agYOrBXbqh39uupT5yibPllqvLLafzgtptyLNafTaTgDotlpw+KXv6+DOZdbcPdnFLVN3PBur1l2s/KKmTovZ9x8fHem/FKp459BD9bxeFqGUTjO8d29rGlSYQrTyVtiCl79sx8YpMcE7OOooGeCiD7zjHDIJTApTh000mrMuqbZ3sQU2zEB/3pxxpp90M2iHLm3RQTsr0HoGYYUrmS2utH0rnD5Gs20dutEk8OjopqLdu9z/20fip+3RoXpvrqUJtc40n/BMp3Qey3ptx9BJwPfDxaeV6kqxehJSOE7G4M1lNKLnP9PFzuMQTWutN0+t8fPltbWIaTVO5PDAfIDQZeXKPJBVStLHxGgX4dQkXLBdKNrKwG6aOsCtrFaSV65jKuccD0x79LuH0qbppoaEUO1sztFIcWBa7v69rWbQeWxWsoMaS/DFz1suAFvtM11UmXCurF93unf4LKGeF+CYPYQ0Zjqs/4+dmkVkQENnt9b+TL+dFWJC9SCIHZvjq2d1N5GRS0i0mE2z++PtOF2Ud1WyWhpWYH34iQJoyb6T+nRaCk5fLbHG0p0BZwjXpHBt7kZXa8nWoGaBwqy1OuSSQp58mkwk4eJUZjNz+u68dS5BgDCQbKkpjbvVrunxslfXJgCcL6gC0GzJDQftuiMVlY84G5Mwsolpd20zMA2o/0oMS2AQ0eFirNabNh9CJzc/mo0t/MuzMR7d31YB16K2JM4snT56vwC0Wwj5DhL48pc/uXUCXXhnE267jUFsvtuvZiMMCh+70inXyo33K9fPdbegWxaDBfc7D+y+TkxszbtsZNYD0M7q4wBEyI3WPk3/7CheOGo22ahbTgVdSrsDYoettAkDb/ffS8eRdguiBX9rq4O0XNoEzKdXMtjj7AIMujVuJu6nrT1n5ZXRSpbLqdUORol0WpBU7JAIKno5gROj0WiaSaQ6Sb+fHTk2Hyhd+MnGlFDkaZXUWW473qfhe72d5o04tSHlYi4PkS5l6uoZ+UzkXgdX8G1jG7FR0BirZbUFhWZb08Lfdg/d1Hv93Of+Y+cD0AzZnCgmoV6z8wD1IYjBd0iJm/PrtP0KPfrskKOGI4CSHfNh43BoNtA6WNsEzedjLfiw1jMmATyzecc8eCztbt9RxBzoMZtsTADk08eX9zSV2mhAcoC2DrHDx/lg9TYZ2oqmMEdutea8XJhHtiizSSrPvEAhaiDr9VAWMzJwhZ5ZKGEmEpEAMhPQae2pUJGn0IIUbRtL9Y4euWXztunDT4LIvh/V7wf0L38ugy+19zk4Hy7z3kJWSlLZP/QHUKykKLdRQq9rsT1Amp+/+o4AzKf0M/Kng76/MMCKp3bu6dOf+1s6G6jt7VHtIudn53LhDjXwbUd5n1XhGg180z7b2bT+oUkhJWPefh3u28khKgAOq18uxwqFEt5sqtECy6drpjA2kAVI2WuNZoNaAIorIpnNRnI5wqFD06RSqwVS1xDLxdwVtG77Gbs4HnEAVSCDfRyHlVS5edGNVi57Q/oBcx8cXlXJfM7/U6k1Iml3alYOTuaTPgPaI8iOjeF3JbBolnGrBbmutrFzbF3P/dPW8q6a1wTC8xLtt93zAugUwqLsH9niJQndS7bkmV7XqGPK/HXs3/l023ZvMp7LbGoCs053U3xW2+kJo3qvL3/FTEC3Qz2bK5ioKnn99e/Je/fNqWkk9Mx1hZeanXKlVdrjJtUaOhzDpgM/0hR6tT1Yz5VMmzftRriWpFZKt0ebNdf3pgHbxhw7yKB60JPn/bsTvfWGtdpmJLO5xeFVlUqz1jSmcO4YMUFNsrV55PQsBBUF8Dk4zKTVSmTichmh3JCjG8Tn1DK8IFeXE5mMOd8IlBFh5G/W0o7zA7akHavywWnepcCXjn7OexXYqYzs4ccu58IbBqW+77S30YWVWF3qgdEuJE4o0d/uZF+Rp3UOa6/NtgLCMTseCNL+zTsDh3In2pfg5z7/k1VeLiSJ7JAEkyzO9hlvnCp2h6ul0MnAZhYpYQ7VMXYwNMzWKHegR5BsBMD3vLcQ0L46aVBZU0iz5cinLHDHHS0jlVbNoAWos1+60MpOFM+VpwjOXZWp1GK6ZlCmTVEH5wmZvVP773bGak2zag/8YCrsdBQVzl5d9obGr4sTYOhMEN6y6BppY5VbSMqZibSn99ivq67RqETPL3KL4yQcfCNyeZVN6xZ9/O0pZD6qjhPDAHbqcvWEFmUa04nVFuR9VG7fPlDV+eart8tqfQ4sUUbtwPsG7Xx208F/+zM/rp+MAlTdDhqh5/P5bp/vv4bvKg0SuGlBUojkY/s8fhvFH4zDfVNXmI5ax80QlasBjRE8AAKbyM1qEEi7S6eunvbs8egkKVceFiJkLSmkVqNBw0rRyiQk5MLummayJJb5zBsiAAAgAElEQVSFTRrDC5NABQ+AD06tywCW9Bi0MwhbbZox1xSHN1ZPLPvHDQFJ48SUO2+dScgciZ1ZkGeVnpUMBO3PBmx1TLBgIDUiMzMUadgaono5loffsW8OpIo5qAKnzzkNWs00NwQ0czl+iSVQ0MkGpBh/Tb+dabPvhxXP+DY+WwHA57E+TNv1DD718k+Y6OgXbO2FNnt0YZUvJPQf80AQXrwmSzSQrWQ+RX2WcvPmidRrtus6vlrI0Fr7KuCdciWNRiiHe0/ra5NJprsezkAVGnnSSCqurGu9VKlerSo9xh5cn+hkDvtnxr2WslrbJFk20T7HwYm1pKOZQEBwhByWMKPI4T/gy4jcvt2QXi+WZgv7TRdxFHq1KbocXacyG9NxzCaWnDsaUDOWDcvuNVu2YaBkhbE/YMPPtitgcQIhpTlyXKdRD6UMtohdGEOTp4TLfCrMTbST1o191xJt7Ow23c4ZyXY4hU/Nb8vCuFaekx63dVAf5pNfMgGAnLmbDCIWBdfXBbM52gzPMeBBex1X7gxXL7SHeOLkRbcrRVYLH5vvCFe2kEatIR944fu0uwc76pVvvWINmqkJrM30XBscI6+FAEcAgMqSU8dInSIgBkpdno/0mPb9PdM2s8VEbji0DAwDZo02lK5g/yTSdiXnV6NLWaR2ksdgry3tJkzfQJZrehgZvuCBnDgoZL4IJVsHKug4h1T/JHFd6k1bBCBYnS+JpNWxMK8Uz5ii8BQzwuucYmL3ukoD+jbunGJiPqX6lQ7a9kjkZgGcg8mpJP4o+WZzi/kXeSjF0s33rlZXcirO8k4UsCsAcPz9sFp9e6NBnTsqxv1IEqjZpCliqfax42KvdBZK3REiID0ylstrWa23xaMnN25bEOF6A969+67cvv2Mqqj3Hr8m9eZCM4XQq3Wi1tbtm1Zsmvotyczh+W55g/6gqtHkTAiVlDdQmTkiXLu+ts7hR0eGnlFZjW+hVc8doGW7obPLU+ME6gFRlsAqSyhrbQkD6hlgLpMWp/0MQJLzMZzDVQTX0u24phHu4EvocqTXqa/QHsfOkS5zzlZC021ZOtoBXSuXzYyRsIodsMPvQeTO/d1x8AuHmHLugZodt3nTBc+w+0YqkXYE4FM//5NOA1hSxI9azc7w0S8MQ6kyB3IEdrAiqlkp2a7HDhctHC8d2rI/xbNWN61ACz/Utk18ohM3nVHDZ6qq1Vwr9NrUdqc1OTh8UUaTC7lw8CopaZJunXZPrsdXms+g8WMdBmyYC86UpzrpOQK+5oC+gEqmxP7T75/8vtlbSCmcj6hNXgp7FrSKZUYz6+yZWfau3zNTpzwB+ge1a7JY0LeQLur2XB7niBOOszHB9a9Zj8Olng6Cembna8c1x2rSLqVu8q1jWCLFygRVi1c4IdSFESEYhSv72vQaEmoxXY5GKD0nwqKOcwf4V6VCrcNOZPLpX/jrKgAeivUCwG5ThJ9mBhAXSY1qyhQBcN1EOeliMtEHAeXqijkttHjzJ4TllUGU2EScPp34bk+PqJ0tUlk7m3qwd6AhFx00hsMjefDonqyypaQr1+oUiwx5gkRN2dRsHETJeiNQirgmn1wCyhpDOnpWracFmYBVUdzTHVyIb09DyRVhUiSPzs41RQ28TVmaVQAHslhO1B8YDiB5kOWEaQTAFGlmk78N+vbcnvxCEwsaQOq8agbRqF9hlGtBiLaHn5rgo4HwtRBARluPtCEUDmXQsSpgopPx1Uo8GNhseM28za/UGzvpZHgErvLIkkLbCAGTRiLaj+BnP/lX9a8WruwWEWxTquDdnnGsOL5vZlREuosY2nB55rDwhrVoY/T2TLWRgACwUWNSLbRxhLJ5HYgy7KOKhxIlK1nlQ7kYTzQl7TdykiztNHFCrDASMCV9/CgQYFTibZ/LWC3nOx226CjKjqHWDo4cZV197TyCKaHgAhPQrLXUD8LnyDM0BpuCZ7WdX693pSwX2s0c00VBjIelPV+vUbfA/fzqrU2r2qFjRxO1oDVwFrm2b/qEbwPn/3pkURP1H70Bmwyf1WL4brsp/W5LHtyf6+c9zAy3wNtzd3qcxDFFIpBObP7rNYtG/KglJPm2vwc//Yl/35JBO4dGtVt2SrenP1lzYfsQGqBZB3iBytxVp6oR0g27kLfesKTIsN+WTttUT5zYbjs4rku354CaWiSX45FhCk5D1WNKqm9Is3kkd++fSibUxNtRbYxaaIuCswuqWHNHoFBpM+h37Oh3BY7o9LGSukuSLJaldgvBj+F+uYavjdOOpRt/xvr/+QSTHn8b1GRJG3dnk307Ge33E+MXrNVZs3P/yEzaHOUptG5XyeNYPd1OQ/b3e9rfEEEbDm3jgF4ihKOJaQpa4A2GMI4xszb6nZ5xKNx44/VzKbJKZhNKzOxdjdZ281pXdH9Q9/ujAANotqY++PQv/gdmAopteTW/ryravTrgxnXM8ALQik2qPvj892szYlqbvvfoniwm5ARy6TT2pN1w1UOBE4rDRNodqE8id94dK8x782TgwkhSm5aQkbAmd99911i8cNhijquJZJ1d22Rlc6nVBtJt0emb08FS2R8ea/jozYXyF8JEFvOVNojgf0ZTj1AzPJ6B/8H7sOwsgrfxJgiANlC3rBgDRw0B8CVkzWZXD5pC8BYLM3Ortb2339wTb6cvXaaQNnp7wwPrDCaVDPpWK0liikOyR1PrGFKvH8igDwQdS7q0++T42LZLOPH7m29Zhq/K7Tg7E6Qdta6zZ2vXar0/GcSj43/5EXzmKz9lGqBuaU8/aFXuM3Fg357hgxN0ODRphDwJfYuBM3T3u9aCdW94rOFdVqSSNFw8mli5E2O2ZCLnmn/w9ev1ONfO4LScIRN4tP+0JqN8P587d/+1ZAWNohNZF4UsF5eO5BjLjcPn9OflckSVuJycPCP3H9nZPLP0QskVLAwnkuGcUoXEUCqWlpXZUyvpwzWCUIi3sA5evA+NMluM9MCIVQZBtq7lWPxdGz7s9EcOpZB206IAX/oGOLNaz61TWRjIcNCWPE+FU05xNof9Z/X9hKCdDi32Yul3baMhsABPm7Vxp7E+fngquWYVwU9Mo2QrHEhYSx4i3PoG9pDUWTjnHmDq5b/3N/TKCJIyUxy5vFDEy74UlMmrN9C/NrRr9bmgyhrIS9Zr8tB19az1JHE564MbdmNJfSlhYm1Zzy7v60IlMY6f6y8Qp+rEsXsbzZrCG/3ukXRcr8Lnn/2YXF6fyqtv/aF2zmommCHCwFzqSV/Lrj0P7+mnPqTVPYzx7L76EvrsYam1f96swOMHe2CQ4CKmBxZWSpseeUe/PWtXG0rdTunyvEktC19ZqzuHzHmIOWmMZa//rKKUpJMZF1eXMluuBIoY+Y963ZzUve6BNGq0v3UtXGhnBy8hqiRzZBK+stiJ3S8uFwqiAVdz+0Q11Nr4aiirDHK5m8gjjI6gE+L77GiAL37DNAB0691kBf3x2TmsrlG/HYNHIwO3g5i4oCVNHKRcZPzQpZMxJxw+KCKDI7NRi2y6SZdS2meHJnnnk54/XUUHs3IqJwcvSiNuy3JxLU0XR+8Nb6n6ny7PtIFVoM2mLdmi0RQ8EufcsGMyR/eta5LJoZwBDR/soAkV+jxVTqGFO4bs8Xfa0KIZ+N+obYE2g9De/x5H1dPRzZ7iCzBa9bYSUetNkjHmF/jTw+mtgZ1HYAEnBq7fb6feUAHa7z4v45lp0B/66I/L0eGT8sq3/5n+fu/+azIaGY1c51JZSfgyDcndYmbZtgYRqtr2NBETQMYqhWtB17RtSV7wha8ZEmi9bbfuIUfEz92JV702J3TxLnMGay7ujUP66gwV56YyJUod/h0EMs8M9Dl+whAq+HyVU0uwYrptWs4a0eSEkzIbT9mBiulKCZHL1Ui/Ey+f0WkdKEXtqVsvyvnFuxJkpfR6+3Ljxm15886rWqnTbpnXfDl+T1odM1M0l6RwUm1/NpN8lcm+I2om7USFDpAIdJO6BsLh+ZJ/rUzLGlsBXS811FUeZBBotzJ/esl6NbXw2JVcgViO3IEQvU5PhQh2MZHhZHampvF4z+7vZP+W0EXl3/jz/4n+/tp3f0+eunlbn01qpj3feecVeeWV39ksZOZIrnQpBS8gv18FdBN1fIiIBJjTesqItlwEA6cVKrofwc998d/R31otmDE78QGMGJ94CPJNORLOlu80iVRfnREtAC5EUk5dJ+xsJZ1DUzONXihUqq7LhWSVebqUh9eiUnrdutQT6uJm8vxTH1e122x05e13vyWrYqo70Jeb3Tz8gHBuEADLwd6RrMZTeeLJZ+XDH/4zMplOtNoWYivj/OqxPDqzJlXvvHtHaKjAAj1560Bmi7Fk7n1kGgMXTdRbHdUIkCw52o2KI6qVdP3pPdxo6VG22t8niaVPKjsOZTodaWpbn8u57WU5FJi9jKLwdf1wE+g+SuRRygtPm81vKj0+lI//yGc3i7L9AQeOTuLXsna8Q/72P/3Wf+UOgqZHk1PnESxmuwHwE0JWRm2nv2C6nFlXs51TxIJPfuEnK60WadDt2qnGopJGm8SIZZKikDIo58yFoaycd7qgPakQ88O6EUnP7QboSTs47Gio8tSzVvGSFtC6Lda1jlW5zKbXWq7MiKsDDd/WRSp7+wdSVhNlBfkawzikG9lMnbFa3JHR9RtK32429uT48FmBMHJwcFNef+cPtYhkvrzSa6bZtXS6BhSt0kKatVhuP/mifuefvPFHkrhzjCkKoRwNHyEhS6nFoJXMOPYNgmwt0UjBt3YF/7A2bbGkjr17PbbdFkZ18fUARBG+F+EqpUUczx7KXt82SxKQ6AmlHb6kv/+Fj/81KXP8l0Bee+NP5UMf+pjQU/D08Rsbufid3/s/7XtiDrq0l1OX4ezvg0CyVi5p5fAA3yVMcys7ZwkGn/9P/0ZF/V67S+PlLUbcGzQkcPBvvQkRdJsXoMpHVWNVl/3+89ZBPK/k4j0LT0ajiXQ1rBMZHJgaO7xxJAcHLGwlb9z5tqJuOEi+83i6GMvJzedlleGA5XI9e2RJKOfADXp9xwBaSUxTZLX9dgaPHbEW6WGODLqJG8xq3EYSJVrQUeAc2QlbjCefeknOr87UtlNzQBcyQKN2q6FYR7+NczkxGDmqaY9/f4QNmwZ4WoEph2VYbUMsRYl5wOyUssomVounGEZlHdV2UrplDtEklHt3LZS80d+Teb7QvsRZuZCjzrFVIe2cIXhxadpF/TLXBPLs0sg7w2FbspK2MaYBel1DE33CSsvbd1IDwU/9rOEAVMlEzmNEXb7wwi3NNKHycaTIfjHmi/WmE3VQdSUOeopkIQDptamjdDlRsgXjuZc+aA92clOPW339zW/LvQdvaOvTdJVp330W58bxkbx59zWZzM+k3WnRtVDr7v0p23TOwMvGlI2uxtLTjlpIOdQscvpLNR86MVKTXuum/vzMUx9RRhPOy+nZXTk7fywtmkxyb09/wHoFSyX3H78t7WZXPfKbN29Jt3Mix4cvysXFqxrmvfHWH8gqh5OQy/3HKwEHgAxK7X5WWlUO7d0Ys7mdPcSYzu5Js3Wg+QPOEzCGMoUtJqDa3qUQ+d5335UoNNAmAdSJCgkdPAzG75E9/l5JXUQboBsnkXE1suPlA0H70G+ZFjRT3bjxDimk1axJw0H5RHnBT/6MJYMiMkzlWtLUjni/eUI8bmqEAktPBQeO9OADCNz+HhECYVRLarl1uXj3nfek5o5BffEjL2i59dnoPXXO9CbjWClLSUwDJ4fLFw2J6JeDOaqTiWe3Qjj1HTUN08ZpxA/Q4g84/CExt50+FjmMuxb15NaND9iiJHWZja91Fz/3/Efk9pPPiuC8ishbb/6J7B0casKm233GVsyWxYUGu9C4yOuvf0vOzu5LUcxkkr4iUdLRCOPswkLOR6d2r71OTdCgtlhXKmTMH5uIbl6aXOo2tA1tqectcczelvL14P6ZzmlSb2ooWo/b0oi31C40eEqqvCJKc80p5lt0r90upesIuo/POX5+OyI5kIMDW6dGM5Tgb/5tywUUVZ1gZ/NOEKR1ZhKt4ZrzDzutgRwfW2UKFSp4wKBxf/Wv/C25+/ZrWjP4v/0vvyWVcs8r+cD3vSDvPXxdioAafbvZXGh50pBVOtt02KIFUbMB3TqRTifR5Mj+Hr0A7ZbGY04lmUkQEY/PJYzp4x9Lq1GXPv2KVQCoJYzk8mqq2TZdjOaxduJCA3zg6Q8p2PTii39J//bevT+S45MPa1f03TEfP5A4aUm95Uh57o/ZmoOkMxlPRvKv/tXvyunVY+UetLruNA+XuKL6ttEgcVXIwYEJUVGRuWQSrfk05jMvFkpAQeu8996W108jKw6JCMp9SmSsJc1Oqn5ybSag3+BUV8fZcJnc8YL1AO+vyWCPiqbFxnzzmdE1p7zQq8CBQT/9sz9e1RJ62r6/URTHp0K8IP3a6bak2aSsyVrJdXorCaNKBr0jOTm6rTcD0+bVb7+mP799565bapFbzzi1nGTqRfP5wXBfrmdw8GYy6DnmTjbSvreaRq1CObkBz2Dbs2A89ZrAiCrYc5xZkh9SWY+gVr0lszmL3ZDrsUUB7XYsJ4c3VWP8ux//1Ptpb+VUJGhpN5Dd8e1Xfl+G+0fy5O0X3vf6bDJS/IJE1qvf+7aMZlN55+3XJWnYxlm4xk/4U0mykqRWyeFhT4pqqozodptSditE0QM8STi5jX917eP1Ukk+1u3cFikOu1ILtmXey+yxZGkq/e6x9DtGLAGI0zQzvQpjwK5Cu5jP51v8gPcNh02hubnP8wSf+OTPVHyQlOozT9/WNiiYgKOjPfXGGe/ce1uuR1dy42govd5Q/q2//O/p61eXV/KdV/9Y3nvvNXWGTk9NFV4+ApUxDDpqu3x/t6Y2EpuH54oEggd48InzeNgxnBbOwx/sY3aQVFMBxq2zE0SgUS0WluEjOiHUAgpuNY4V2Xvu6R+QB49N9cVxam3aA5E/88J/KE89ZT6JH3fuvCKL+Ui+7yM/YrkIieR3fvsfy/7+sdy+/YIM905cD4Oa5BnZwExbznzv9T+V86uHMl+ONa3NmKe2MzsJCGdTRtOHEkaWNeT7bz8zkBotaTkhpGnR0SK1BXrwwP7NiJSiXGVy5dBBHFPK5PxoxA5mLjjL2dHb0prcOrZno4cTiS8GJ6X5QQQDX9GXjfF68KWf/2RFDppSJ3++z4de/LDM5qcynVsih6PToI0znnryOfnwhz7mrlnKP/8X/6NMZ1cq1RcPDfx5681HInEm62UhTecMRg3zlseXV/L0czeFU8nwVCnBYqe3BpxaAgvITvcKYyqDcg3pGLWYHjs0UMZUdOViZJk/JUtouRg1eTYxcZDJ8aErygRdpAUdJ5o8qMve3g158cUfkP39PdnbO5Tf/u3/XR4+fE+eePI5ba3yF3/sr8ibb/6JYv2wnZTizUXDQAb9niyXE5nNJvLH3/pdDRtpUBGEc1kXmTw+tV18ckxL2X1Jl4FGPq0a/XtyefOdP9LsJ3Az/YX0uWrUJ1rjKH8eYxg1tSrr4elb+h47om4rAHXn+cOOtg5v1iaekaZzkTIRKeh5aNVMu6OUhRbU+BF87etfqD78wY8IR555AQChY5cysft7J5KuZprseOmlH9bU52x0qqb59be+IxfXZzpRz95+UW4cPinzxUJ+/Zv/hUznY1W3vjg0zS/U5iH9f+0nflobOJ5fPNQiDz1RZHJPF6rTqQlsKQozaFjVadnN0raO9rVWAFLKYhUpBy/LyeRB+aqk6zpt9bttSWj6SIPHiwsZXc2Ur/Dgfl1arUDPJWDcu/e2fPSjP6j9ilbZI2k020rzDspntdFjll3Lhz/85zWUvLx6ILPphR7/Uq/1ZDKdyo3jJ7RC98HD7+n1sso2QBg0peG00eX1PSllrsROMIuz00tNWvk2+oSAw7094URQxv2H75gDStjo/B8WdTRzGUBshqO8W+7CVQ/tVBezVtMru5c/+9EfVQE+PzMf4/TqvqSF60DKN/2vv/VPKu2wVQtk6Vq+rVaFfOhDL8nBoe36/cMTPZtPhPNvlvLdV39fX5/MRtq8yCpo4QG25Td/87+R0WWx4fJNsUGByHC/ralffj48OtaM4Z2735LUqU1C226PkrBYOr26NNuV2vhh39XCW9cotYuweYWdDb01jGVFgWpJEsVsOeyddbpWPCBf55KvmKRA7t65kqQW6UFSjCefeEq6vSO9/7t372g/o/l8Ih/7ob8gt54YSr93U96596rm1knnrrOJHgTR7dyUl178S1KW1A2uZTqx9m0XV/Yv9vdDH7RWbn/ynT+Q88t3ZTa/knQ9ckwp+hnZ6h7sfVBxgLOzP7XfD24oo4q08dzV9R0Mb8id+3+6Ie0GgTufcD0XStIV3dvxY0jo0bRShTJfScuFx889+WG5nFzK9dTMzXxxLcGdO6/qnbzx2htyeWHJCACG4xuH8vEf+zH7PWrKgwfmVN29+z0ZDl2bsvGZ3H94JpeXF4qQPXjnbQ3jYM/6UmWqWii2aLehUFuowoGdqFioVL7D5Wx1rvZsNB7Le++mcvMpkReef0oOjwdK6W53rIU7/4OpL6ZrWdKocgkQRFcvWqtSsl0oB1CrgqOWrOnV6xhMj+/PZLnI5OjI4u3bT31Enn/uB+V69FDu3nnPqOBVIX/xx/5t61xQhfLtb/8/mmkEoAJrwCw9+9yLcrj3jIIs1FDOphbK9ga35Prqjpyd35cXX/y4vvbuu7awF1cPhB6AYbySx+evScP19SursXYQTZcUs5hQ3L75g9LvHcndO+ZUHx3vS9+BXK+9/Yo8PjXQByew2ejJe4/eVM6iH6PRXMmzjFo9eV93F9rPWEgPptKQ4O7d7zovK5CxyzgBEFDbN5/7jt9WhIikQgj94EvmHV9cnMvv/f7vyp17r6tqjVzxwoOHF5L6uNapKys0Nofw4JDjTA2w8J7uKj+T2Xwp4wkl2SLf99JNefqZnjx12zRAp7un0rxMx/LocSoPHp7LegXyVlciCN01X/zADTk7g6MHevZQTViz2Zcf/eGf1qPl/u//63+WO3fekZdeMoxgf3hbvv8jH3ct2JZy951vybv3X5Ef+9FPuLmcyyvf+j9U612P6IhmlcuAMlkKO9+SQ/uuAPajH/s39XPT8Tty565Bt3v7t/RfCmbPrx7JzZMntNro29/5l/p6XoxFlE7vzjjKeppiLnMOpLSeRMP+kfzIx/6yIpbX08tN297Dw+fkzr3v6nU+8qHv13+pon7tjW/J917/Az3UKqi9v01cxjnEgFf+UMqqMjL76Hosg+GBfPfVb+mFxqOJImJog7Ozh/Jn/xxeMu44ByrO5fLyUn7/D/6lvH3/D7U6VgEacucSyoN3Z4IvQojacAQ0PWErX8kP/fAPyXe/85aSQdqtlrz32OznYsGZwaCOtGgptWEjJoH/GT/w0SfkX//xQ/nwR25oW5pXvnVfXn+tkFYrlxvHNKomRDV7+Nztj8ssfdVd95Ec7r+gz/HkyZ/T/gBvv2XfORgm8oEXflRaza4ez3Z6dl/7G33/99nuZVxc3JHr0T05OnxaptMrbcGGdsGZvLp8oEKJNmI8fnQm/f6B9A8yOT1/Q5JoTzmNywU9+iH7W/9jDNl0Ybsb8wOgc3jgjo2jPD2FncTxeK71mxzLwfBJff+HP/iD0na9mjHJ9r8u/eae0+W10s//2b/4r+Wv/8SnN5HW47P7kq0W8v++8s837w2+/Sd/qAJwcX66aeHKLQJE3Dg2Z+n2My9J13HSkMI3X3tFX///GvvO4Eqv87znu733jt62YXshKRaRpiiSimrUbNkpsp3xJMpkJskkcX7EHiXjSIntKI5n4klsR4oY1ZiUElmSRXFlUiR3ye273F0AW9AvgIvbey/x857vYqlRnMmZ4RDAArd95zvnPc/7lDfe/nO4AsS3lQizXFamSxcuXEbYP41gyIRWg1ZuGiYmJqXy5l594cJVhCMmOF02ZDIFJBJBwRVY1fJO8QVccDlsYqeSy6o7450bd+D12pBJ2zB/ZBaFSlVo3ARXnC4etbqo1lSh43aFhFPH4fUoDIPvKRE+AJ9vDI26DoKwiSU9/QGOH1Pb3V8/yIpSx0BeHAN9hgZV5RCiK3HV8wDJ1Dq202oCEgr2uWbhdHrh9cRENs7OYnJL/XupkpY288z0KaTzl6Xh1e3Rooe5QXldnTVAqViF3z0OjzOKrh7cQZbWY4//ot6yfjAB6tUChvwAr18pr9SoodkgSKZOK4VCBtrLf/GiqgFW7ov8mcPlsSMRGsHYyKh8P7P/GDxeP86+/B1UKiUsrywIO8bhMaDZ44fOGW5EZruLQq4CX8CNaEyha1Ud35ibO4R4nFLxAVZXb6DH8y7P8DoS4nC54HAQlWeBN0Ak5kCrYUazoUCWhYXrwnHPpruIRkZw4hFiAV3kc224Pepi1+kaIshlF5Wy+jtKxMRpewDMzczCZqM/obpQt2+tYmQ0KEcxh3U/Zqb3IRpVPYSfG+JK9i7fuv/7b4lodWV1CRev/m/5jWBYJ28YPYhFDkMbcNKZkN7NyI3Dkwd5CKnUHUSiYxgdnQF5hEQ9G1211xPsqlRqUktxDP1//K641Dkc84dPy/9DoTmsry/C41Q/P//mWTkhHT72lHw/NX1o75WXC/ehff3bfyyvkNWvzU4VibJJMnSN+PQv/i05o3/npa+LsJFjafmaFC1Wu0XOoMl1miXw3/o4esKLjdUMJmajIrXmaNZ5TjYiGvXD7eaLInKXQyDoAAwNIZFyUB8wgFkJPqt9WKyK/08qFilMxXwF0zNMCrPh/t0iKoRgHSZ4fQyfMMDpNmMrqTpg1aoD20k180zmopgzcTz//DwCQYZGcLtyCl7RkahwDW7nlJhMHTl6Er1fx0gAACAASURBVIfnT/01l/f/78f1WhaLd1+VPXtlYxGBIE0vBjCbHRj0yEzmKqIbWZrmZUJS87+4+IZ6AiHBaEI6bXXyIqhxuNn5U6ecWr2CSkFnR+9lIPYxMXIEsfAEdtNbSGe2hb62k96U7cdjVSef2MgE/KEwQoFx+HwxaF/9lkoNq8r+RsqSATNTB7C7u7nnCUxqGMmK6rWpSBQOVvC7O3npltHsYXwyIjBpu1VFvtAT5w2CEiQo0rlKwpY1IxIJ4tQuIUkOwX6DsQGPJyKKljt3N6XLxaRwFnfQmkI1F5ZOpwfapGUzNanaqQJKbrG676Gpo8V+nxmxmGp4EFhavV8VUmVsxI3JyQjyRXVnEQLfTtbRbDI+zqUoXf0WPvqRzwjiGYuN713x1bWbGBvdr9vJqB9TsTyMmX331Gi3q7h45cvyo42kOln10YEvyKwDgl+cgA6JdikV+2hU2LHz48QRdZduJO8gX0oh4FPmGryBBloafd1HSDPSK4lAGD2a9IZV74E2kBwFj1MVn9uZK+g0bIKeqgtIFzYbTh1RBav2wrd+b8CLHgnE0emqO6jTrSKd3UZNZ59WyiRuNOB0kO9uQCigJkCxqKFayQkcS8IFiyMOu8OKbDav5Nw6p4ATh6sJXbYmphygibi0Uy0mVApWhOMsEk3Y3urCbKXlGztl7IuriWczuxSz19RDKBiBycI7io7iRhTyhIOpzrWLFdytW9cEZ+AIhyOo13cFIt5Ns6BtwO1WKGGtXmRioPD3PvyBX0MqtSFHuFIhIxOVx0vlmqJhZMyOWHRG5x7yJnDIpJqbPQ63R+El+dw2AkG1hVy78T9Rq6ckIYSDQRGZTB6TswMQUhnawO5s04KH+UzePfldo6lkef0et7EeXA6/HANJah1QbKqVoZmzOoWPlPkHBJBemysmf2+oDu5LyvtwMD+AbmUklMoEePPcy/JVrrCCVlvNVu4Z2WIWrTap1rR2b0r1rxl6sFoGoIyLI5tpgeHQkuRJXwuzBp9nQv7m7l0iWqwnFOuy3RiIJRsx8ZnZqPDpiBEwfJln7lMnPoCtnWUVlFBcg9sVQauVQ6O5pVCxdhQjiRk562dy1+B2TQq3b2x8FOndKuhstrb5Dg7ufwIXLr+IXF7hFj5fEPHIKWHusEGUTm9idkp1A4Wccu86qtU8fJ4wErFZ+H0RRCIJcTrl462urAng5HKTckbNICeeGX6fH9HIHAL+BMolBQDNzBxRE6GQQza3hngigEr9jnoutAT1tDuGUbGMfW2gXuNEcmBnuw2ne+jHYBNZmrTm5YL0EQ2N7dnyGy15wSNoakFcgfQ1s67VqDVyyhxT3F8NQvd/9xATaULF+tC+9g3lFk4PGs6wYIBn/D4Wl8+jVCYzlkkeFfg8Uamy/d4RMUrg2NlOS2VcKRPhimB0ZArh0AT6gzbevvQT+R3SxcIRjyBoCremF68LlVoKk+On4HQyHNGAQwcfxtLSRfiDIdxffh1MNBemjk51stvmMDtzHMeOnMHde7fQpCc/G1eGHuZmnxBAplxRvYsLl1/GTuqefE2rNz4Pdfwj8Vncu7+IVlNXHNOguluTD7hYTGNi/DDCwYScLIKhKKanD6BUUavHzVvXcfPWmyI/I0+x1WjC42MLnSGXTDyn7pAdSg9a3YJIyfiaLE6FB5gtPTicTWxuKPJMMKRWiqYITPto1J2oVNVdyctOf6KCHgdLkQhRUbKRiLwOC9VGKwOrI6Ork9QdrxnJbWygR0ELBkjtqMkZjKgVUYlp1KokAl9OgJnpI2h12GNXmyiX2uu3zqPVqUpytdXilA+71VJhT26nWzpi21sUZ6hECzZogv5Z0dwdO3YKS3cuYz25DJu9ilaTWTnFvbydQZ/snRBGEkfhdKrlk+xZfyCMciUnBlXJ1HV43BF4dUoTJ0lP25G9jEuY103uQBB+3yhmpufBIEmbxY/UbhLnL/wvaTZxFMtbe6eAI0ceFe/9YYF68cpPUW+0EA2PIZ1bFdNFbmMPn3xKhVB1ujh9+lERx164eBarazeQz61JsreJCWRdMoOIX6jbyWzU01Fs/j0kLhwfilAMsDuaKJUI3wLBYGzvovLCJjd1wS3JHs2hg+kwMJraB/UcbAoNJwB3J/YwOLx6ZuBAuoNsB+vScs0ubeFCUa3uDpdDajh2XTlBtJ++/l017UxtpDNq2c7mN5FKFxEMukGihtXEOBWLFHu0jOvqrNpSsYdYjHAkTSJMsFviCIbY8nTj8jVFY262ttSs1mi4oGoMEjgpAefRj8oagjRPPfbL2EmtY3NrCWaLC7uZW7BZvbBaiXsznjUr5olLdy9i/uAHMT0eBheUcknD/Pwjoire2lbL/uXr30YspoqgUGAfktuL0kZ22iPSjjXp7JpcrohQKIwLl96UWoMUdPHz7RoQDY1jbuqgWObxhmg2Szh67LToEgkInb/4IkiLL5TKyOzUMDrikS5fPOFFrUaFk6K6MQWcg+/d4c4hm82J4igYUCuARMeLwZWasJmsUvrwv3ptaAXDS0R6HpNOVaaSvLfoyB7D12CpwGjUt3Dho6nHY6OuVlGrAEedfsZC1tGFIhevvDIgVene8iVk82oClMtWtHsN2IQSRhYlDZzUftRoFtBpqNnKI1QiHlN2b0ar7Ikca+t3UaqoSpuJ3CKXrpf3tHukj3Gb4FGS2D3358ce/hXkcttYTX4XLtcBDMBu3INAawYpZ7JZFPJVTIw9DqsljVAwgbGxKSS3V6TQyWbVHkouXCSqmzca4nLBxb5WeIa0wdej3VsaXG4bopH9eP2t7wlpg6AVj6Qh7ximEgdA3+RMLi3afuUHSEMKrngu5Er3RBzr9ag79fXXFO7/2CPvQU8/3g5NHlweeh5VpJ5qtHIo6Axq1hIejx2BoFP8j8jmKRSpKGrB6wogX+jDbjcLTD4cEm6laQj4I3DpIRUevwmFclYXnmT3ZH4s+riCD0exsLXnusKfaTduvSHz6fVzP5EIFA4ibO1+Vnz9OZqNtnIC4xgYkNnV24l9J/bv2yeo226qgrLeZaIHYH+gPuRYZExkTQajQ9nDCuybR625Do+HBkvKPIn2qt1uS87ERqMb1Rpbx8TaqVSuiIcQK2MqvkLBKPweuorSKZP1SBf5fE24ixx0/o5GVR/c555Hrcq7mNtBDuVKEmb9A5kaP45qLSsQ9U6GxR6JFySklOC0RhB0TeLEsdOySl2//pZsY6xL7E63TIBCYRP1egFjEwowC/gUr/DNN97CE08o6Dy5tSHsG2YDOT1tsaHjcX5qUtHAtb4LK2tssNn2+vSdHvOINTRqZCBRP2mAprd9+T6yWXWdyGXkquDxhOTvyb0slvMwmOowmulEQtm5qos4PM6QQAzsFrJZRD6F9sI3/+OAb6zWIJe+KHw8q5VGStTBUf5MYkNO2p42SxRWQwT3lq+jhzI6TZvYsB4+PIK799Jyt3NQq24y0fbdBocukiTvbiirIxmxULotvX+1HPGi2sA2NPfgeoNc+rbIzXQrftRrbEUzq6crEyDgoQxLExEEVxR+MGW9Z+52MYdPnaGnxk9ibvawZBrcvH0Jq+s398KeZ6fOoFbnksykEE16++Qp3F44J6LOZq2Bz3z6H0kdZHf5ce7cD3Hv7mWMjhzAM89+WhjJPKa9+Gf/QZ7ryLy66Feu3pC0MA6LRX0moUAcTqcZxVJStpWnn/yU/JwoYL5QFHCtWFSk2U6XKy65FDohpGdFr6Nb3GtuMcriYMe1r1vJ+XwuRGPqPdfqO6g3U7C7eCOpVZ03Wbdrg4P5ShbGyujG3r/3h78xIG3JSv2fTnA3Gb0yAUjQiIVmYejbUSopnJ0vnhMgFBhDs1XG+HhECJmlUgabSU4Umtc5EIk74XONy0zk4OMNs3BhoFWsCYGAE26XWp6MRlqzj6HeXEelWhI/XLPZg57u6lkqs7LtSpGpaU2E/TPI5onFs2DixXOgofvycuvqddSWNTezHxMTM3Ken546LBfshz/+L/JvXte4TCbeSV6PD072EOxerC5fQy6fxu2lCxhNHJRgjMcffVr0ecnNZelMzu5T/X7VxdPZOJUtSTW5dfsq7q5eQq/HLU7fsBkO7Q9idvqQ1BHlQkkK6oOHD4kmkb87HKtrS3IkHhpF0gbXqFFG3kYuXxLeBUe5xoJR7eVuyTNgn8EOq70lE4CDIBJHb5CTLZBCWppJc4iH0pf+6LPyCq22+J5TKL8fSxyF16kKlfWN+zIBpM3a7WJlbUGwfyZc8sn4HO2mZe/cmtxKCh2aI55QDGIKGOr1hli70Z3THzDB7wsh6Gexxt57XOLWtrbuIRAiDljF8cOfxdzk43sfjLh39Lr4zve+glRqSy4q1TqSoiV6/qGbqQVOp2ojOxwGHDt6Rs7uk+PHxZHk1df/RP6NaFk4lBDDJl4Uh8sj6WNkGhH4KlWyePUvvy+6Q9LJT516CuFwTCDWLk0smw24vXE0qqoSd3rH0KqlZS9++eyfYgALXHqfIh7bh0qhKyxu4iaRUBStdl1ArZHxEZisqmjlSG7eRjK5JDWYprUQCo4gGhnFvbtvqzRTSsy6jLAnLVxN9LEphUEsry0L748sYmYUtvsPtgD+O3sVA9LJJYE0CO3f/cE/lAng9ngQDs4gFt4vlfDU1CzW11Zw+tRpXLl6GRsbGyJscNhtuHb9ovoAPS4xUGA2Hx8kk1VZdexjd/odaXTkdUDGHwiioUvKEjE7YjEnotERcfrsdpxYX+MRJ4NmO4mJSRfGxmYQCz2EcFA1L3h0KVeK2N5Zx+LSW8jKcaqJIweOIhadgM3mxJ//8L/h1LFn0e2l98Qr2WwV8/OnlBdwqSVt37EJxQcglpDcXBU5OCHsYDACt8eHVColFnbc73/446+hVisgEtyH9zzyfgGJyA/IZzeQyeYQDYURjipkMZ3JIqrrEb75rd+Wo5hmVnUJV6oPPfe3kd7JiGuZV/f/iUSjaNSLopwajp3dPKqVDFLZy3A7E3IXx6MJWUE4Fm69DsPALJ95qaw8EqlDsNlDsNr8wlkUo04KXll4SulaRw9N9AgCDerQoDuNffFL/2zQ73twZP4EfHrL9z1nHlGJmforWly4jWxWsUs3Nu5jfWNFvuZRix8ee/082hTLGxIgnclW4XDaEAgSO1Dn0Vp1IM2g0VEfWq2ScOCcjgnB/vk8PjcZNgHpOt545zyyhSUcnX8aOztq62E9YLORH6DcSbdSNyU1JOgfRSI+A7fTg5u3ryMWHRHzCYpGpiZPCrgUj40J3pBKbaNRL2Nu/3F5zB/96H/I1kGQiJU170xavnOlohnEZvoutL6yibMa7Aj6EsJjUKzg6zh98jHp2ff6qiieGFXFYLvZx+qGYiUz50h+1mEV30HAM4JoeFYMLDmoAj508DjqetyO0+3A1XfewtrmEsI0t5Y9vYbRxPzeBMll7sLvCYi0jjpJcgpsDr/EznJwZez1zJI34PerYjhXSkrvpNWnzd673ET+7e//G7nOf/czv4Z4LK6ygdtk0tikQ8jvr169gHv39DfU7YqDF0ciMYNKNYNaXZ0zeaZtdbsCw05PDfvQLd3xQkXGR8IROJ30zOuiVYvAaiY7SBMgh2d1nz+G9Y0F4QuS+s3qn6NerYjKh1bxExPjCIW8wuodG5vGSHwGKyvLGB2bEDDp3aNerwqfjyOf20VqZw2HDj8s36+sXMPCnbdRqWYxOXZMGERcYnnHUfCxk1lF0DcqDbL15IJk8NE/wG7xo1wuC82LLmBjcUXH7urgy+ljT8OtHw1XpBboIZ0ltF6F1xmE2x3A08/8kvzNjatvSw0zQg6cPpY3r2E3u4ZifRthb0ReU06v5sXIQjfWctnssEt4ESHvmb1wSbbsyX3k8AdGRBQrcfPVjPQAVITOAJV2HtoL3/jKgPm1D595L9o6AyUY9MPlsqPIpgjlyC99GVldfMgz+9BQkiggLWXFuo3OV3rhRcx7bExdCAqM/b4ZKfosFqUDIOPnnRtJ+N1jyBfy8gHHE3FpfZIXqGkVhCNOlIqEMtX5d3OtILzCVpPR8EwkoZ8BewsdTE/Pw+cL4MRJ1RMfG5/d+zDvLL6DnZ11kWcdP/4Edra3cPjoMfWYm7ewsPQWytWcWOCwA0it/pNPfATpzC5+8Mq3EQlNyMSmswdbq5VKQaWGGG2COtKCbs96VZec+e0+fO4f/FN5jqvXz6NeTwvf7/7KElwCghnx8U//ln6K0JDcWMFb576395onZseRLayi2iQXsQBoJNPSi4HyvSZqHXU8N/Rr0Pp1seO1iB8yBC4nslitlQW2npqal2tT1QmmBMJK1QKaeoC0VigUZAWgXx1RveFgxfvGGz+Wb9c2lhSCJIOyzSYPsJKrQwOGTI5ndubvsvCgns+GcIiqVysmJ5QOPhj04sTR5+Tsef7tV4SESWgmly2K+nViihOEqSJ2nDr1CErlNRg0OnOqivX733sF8/P7BKOfGJ/HV778ZZkA3iDj2zlJuMXoRaCJOcLqzgj5x6VQyheziIVjguh94pOK81csruPWwnlp/Hi8TowmJhAIhOBxH0WrVUa5mhSyLI+ETz31KVy89ArW1m/o0i4HUplVVOrZPZf1VpsdT54JjBiYqnj0zAewk17H7NQBxX4eDHD08KNSoBIbMdsf7PvDz/361deRK96Q0IhSVU3+bq/JtKO9a0PXVOL9Iu9qZaSXoRkYE2OEi3oLXTIa8iUQikzJ5B0KcHjj7LWQubktLi4NNjY2cerUSaFsDcfubgrnziviYmp3lZodCVagr065siNtSaJWpSLt1emBqzz+2bMn0yUS8clF+PhHf0Meg/nD6xtr4AkhFAxiafEWRscmcenS29L4CYYIPMXgcoZx6uRzWFtdFmozW9TkEayv3ZY7lIWj2ezF2dc4OVV+EZ1BwpEwUrv31Gsx0fWTK1MfJtjFLfzJJ5/H1MRBuRt+8AN1CpifPyOADu/wSGxqLwDiZ1zT9z4R1jE5QQSJero8YfzZS1/BZvI+mvrST6Ilt0wb84Doe8C8QSaMG4x47Mz70GxWcfLk03A5HNhYXxeCaN+gsId3j5W1t5BK30CzpU5S2UJTHMhk8EpqYQF5Wt3iXqaSjR7HxbI8Jx+PJhZah/7J0wiHHgRLM8BjaDtLr0Lt5ZdfGbAH/v5nnkM0Gpdk8EuXLmJ8fAILi9fkji9W07JkDLSK7EflisKc200juDAMnanrDXUk8fpMmD80JWrYifEzKOQLQjYhmhgKxTEyEkUkEsPNmzdhMaucgFu3v4ujR57D/KHT0ku4fuNNLC8vCycuGg1gN31LBVwbafFuwU52S4wYrGY34vFRsZLj3V2qpIR122yWxe3DZvXg5PGnpf6YnnxQSPF13rzxNmb3HZG6okdnVImP/9n26c9cGXErZaxME8nNNYxNTMrf/dcv/yGa7dKeHtIixzcFADnsHmWiKRzGPuamj+KTH/tVvPjSV+DzhhALjWB3dwfl+g6effbXhTvICZDN3VXBVXQSI4yun/drjSr83llRTxWqWWVdoxHzf6D2YTveSRl6u42WLjFPRCYRDsaRTK5gM7mIj37w7ysS7wsv/Omg2W7KHjqUf62sL4n9STAQkIuTyt1WDWbdt5bdOvb2LSY7WszYyZf1KBkaFQ+QGPEjHFVEi31zT6p/Y9PHaBRMm72BMycfxsrqPRTo2F2vYunuWRw/+j6cP38OszPzcrQkX7BQ2IHbHUapekdeAidiq+mWi0xjCjKNxscmxMEjHIoguXUfU1MTuL2ojqoWgxOf+PjndH+Dn7+4qe11OfOHgxGYbQ4xg/p/D6qUG9hYvY9bt27gfc8+j69+7Y+lkC2XK+ibGW3P9zt01lTB0fyuUa3B7fRJ1e5zKk5Fq9sT/4JKcxteuy6/G58Vm73ddAFma1rSRjttI0q1LdlCXY4YCORxct3bUixukq85d6me4utjrRRw+1HgqkBdhe7vQGUTBa6aniuofeF3f1NC3+amjyOTTaFQysiLDYei6PQUwEFThq2dFNK7uzJr2MAQZ4uBhkpZLdFGFkB6TszRowfR6akC8ujh5+XOp45+c3NXHvvQ/BySm/eEDMJIWa7Ld+6/I312ysWiYUq6fWLzUqkq8wWyh/jmjdoIcsU8+m0KQlV6F9u4VNR84hOfld+9cuVVvPbay/I1fXN++Vd+VbagkK5kHuoTchl2KntSIE1MHcDq/QWsry3J8vn4kx/5mXnQahSF5TQyNi0B15tr93Ht+mWkdjZRrCjZlkobr0OztmCg7Z0gcWbRVNDskXdqW2LqgHBAbbeGAR3HB6i2VCfT7whj0Gtiauowej01GdmvYDgWC9lKfQchvwLoXO4REZtwpHYXUGzoTa4OG2l6z5+UsT23cAPMGgW4qpeg9f/K8fR3fv+fSxFIkgJjVIeDfH6fz4bkVk5cNaCTGLlHs/hTLloGdJoq8dPtciEcVGdOf8CDAwfVWdvvUwjXZnJLrGFoqRbw21GtNhCLjCKTVu3iSj2HWq2GSrmMfbOHsLBEUOlBeCVlauzx8wRQqdRFqcuLRyIrARKHk1l947LShIIT+MY3viqP63EbEY7sk6X96acel0kTTajXlE1v7uXwrqzcxeTUfoQjCaR31nH58jk0WnXsP3BM/mZsZFrql2KhIKFYZFCXS1WsLN+DR8ds2/2m+C2SXFPvEowywOvzoFKlaRN5fC0YDQ5xRSFP0ml3wWShGqmDWqONoUMPfQ5sRitCYXWaET1mLY2APy6IZaeTF0Rz/uB7Qdt9jkvXXsXWzopMAtZp0tBrKcNvGfpEYP3ChhtX9lYjDe0LX/q8ZLW022VplRIGrVWZ60vzwbJYoWgmYtHizwGDuFDrfkE0U9bo1kFY2IBwyA2Hk+7dDlhtYaQzO3va+56sM9Swd+C0m6VxRE4AIWWfN4Er198S7gGFj2TIkIXENjOJp+xJPPnEh9Bqt1CtlJBK7SgbObpoWAwIhYKIRmbw6HuewMrKHWElffMbX8Xo6Bg2t24jGjkk7pmPP/Y4PG4PgpEHsCs/G8rDz5//IaJxMmVDcNsd8tqvXP+pBFjJihjch42NNaF7SZhEhy7fyjOIFizkCPY0Ru+R8rYPF2+clc99ZEwvrDV6DvaRzRKFM8DpUP16Fq2kpg0MTTGYJp3LyCbTgFuD0l06bT7UG4W9sK6An94JBhzc/xxGR2l4VUKpqIC6V89/H61uDbVWWQA2I00oyw/EoOix2feg6NS+8KV/LStAOpXGMBqGF6nXoa2JInJSDjYkEHC5Ip7PwdloMpiFns1uWyjolIRL9ul3d1UFO9AMaHaSMGhcHpsCevi8EWxvpTE9cQg+r1/OqzcXXxV+gc/nlj1qK7kt+/ahA6SNuaU+KZWK2NxclySyYqkiyzrzB4k6ut1uPP0L7xdhidcbwTe//t/l+e/cvYzJqVlB+h46/QSmp2eRz+pLtoPLsgavjxX9HwhZlP5BRw+d+rkJEAyOoVAoIZ+jiWRXWsMUjfZaPIm4pPax6sKQ+SOncPGqmgBWp7rQVjudxjRxD201+7J1cbDI1Qb8jLJwexR2ws/XauzBaVMNHhWPqwps3mwel3pMo+bGLzz163A4PXJjVcr3sJvZxrXFN9DWjSM1K93V1DbBiVvKUlrWh9dLMY4B2m/+9j+RCcAcvLbsGwoAJvmRtGw1uMyrrhOXDqvOTyeDhiASBzEAj34ODwZHkcooTh47WOSVVqu9vfhZBjHR7ftjH/0EvF6f/Pwvf/ptVdSxuaFVkYjNiOGRWQ9fYju5UlZ3v9HUhj8YEB99h4Oagzi8Hj9OnXpc5Qo12/jJWcVI2tlZg9FE5/GW4PsnTjyMmi7lsjstQoviUv3K2e+KJoFnfiZ18whL2Jcq476xieEEaNDfv9NCvkCfXkLgtJ/1y/9d+kXtm6totFtoNroYmLjfGuBy++W1UoFM3sHQw4fJZEReu90Hdym5E1xRHzn+YXkPmd27MBotqEoRyFRQpqUZRFtphMJJnv+gOm6XigtIZbawS+l9vQiY1cXPl0syEZxWmm8PsLqclJVAJgCzZBvs1HVIR1ZgCvn3XH65XHGP4gRwMI5kQNFiG8GAXyxP6X7B/Y37JF9UodwUjzx2otjLT6fVasGjYsjtQrZSxWR8Ag6HD/sPzMsJYHNjFWwRc09ny7JSLeIjf+PviZOYzaZg3NdeOyvNmbIoZAoIhtgYUdQmToC5uRmMjbLuYPRrE5cvX8Z7n3wa5954FUt3LsoEoJ4/Ek7g2hWCOUroc+rMw0IkuXP3mljY8KzOwjQUimJqchZ3lm4JT/GTnx4KRoEXvvafhDzDNjG3sGKBTFy6oSs0rjNg/LveA+gz7Kot0niP36XyB8hv7DfQqJHZSyMHA/J53SGEeUVE/TQTzhx+Ruzq6rUSqPcbjkIuLXVXwG+G0xoUTkDAX8VDjyr08c7SOewWFJcgV1yFkYpg/XQj2Qn9LpaXN5Vlz7/8/D8eSC4evfllE6aDBZsJKuGbg2EHYk5mtkg3kEWGGrxgOpFUes5M+SZ6x1mtqtCSTtNy2EySZ88ln3d5MDiOaqUoq86z7/8Yrlz7MUqlPHL5HRgMTnz4g5+SY6QyVgSWlhaEf9Bq1cVM2s1JxiTOfgdzswfgcgX0CbD3Oe19sXDrivAYzr/1sugFuNTTlMHrDWBxYUE8+gMBn/LjazUxNj6pitpwGDduXJK//eBH/o5Y1nEZpQnGuddfQbFUFAPMvO4LwFVIJgCKYBAEh0FX55LAadO9qISxp6enlCp5WUE8HqewheixSF9Bk9mAsehDCvjpdFDIq2KZK66mO6dZrHRwpesYhSAPjrguzxzu37+srpD+PCz8TFY7zEZ6G3b34Gvt81/8F7o0rI5Bj7Em6oFIPiRmzUHWLy/83tC9dISYEQAABJlJREFUakl14q+z7cul1KQnV9EnMLdbUV55fQ0upw2JeGjP2cJipTAkgoP758UwiqvKxUsvy4WoNyripEGuHx3Hhv6/t25fkv2SxNTxcVKvlPjh2JHH5I5Ver8HXa5vfl2hfY8/8RQWF94R/UKjVUWtUsLhI8rihhNgefm+nD6eeeZZrCwvY3NjHQcPzUv72eslSueUi+B0B9BqVNHrtFGt1XD2le9LgUsYutbKiWlkIKCusMnZRVkPsWRzTGLfGG5tYx3AAOneXnwOaXRE+UbjDwypqmVa33YRCx2RFPBqtYRiMQOn7hTmZLQeBoiPUKhD3WALAS8re9YURj2dhatvA04v00MG2MrSJJo+CxaZBBzN9l9VZJwALCzI+xMTIbMqMPi1zaKCCyQhg1ExYjE5EFIhGbBM9Bpav5GbNjAbZElPp9MwQxE+p6dUi9SoqWAGMYWMHZA3ePDgcYyNT8md/OMfvSRGDLR7IdWLEyAWTYgQlUOWLqKg2gAu0RJQ3h3GaGJKkDyOd/vi/tF//qL87LnnPoTbt2/I5CmV83K3DplPLFZJsWYPgkcmp9OJ5OYG4vGEOJmS7Tx/5JgwhthjX1y8jbm5A2If+xc/+A46NJru92BzKM62iFCJ/gXIUqSRRRsNpoT1KQkzwG5nyhp9FQ3wuC3ixk44nfUE5Voc4pdkcqDf7cKgcwkYhNltm8Uml1cgop8gOAECesu412brXRW3FMfSKYxjaPjNr1N5EkjasJgGsOj1ivY7v/uvJEm0I2nYqtBjgUbwgi+UwyZ1gTr6EXEbZgdzIlAWxgqbLc/NTFo6duhl0aqrF7BvdlyqWNWAYMVthVvk0lbM7jsh1SgnwEsv/ok8xi995nPI5dJSf2xvb0jFvbZ2FxPjc3InJLfWMDKaQCgUESg1EBgVv0FOqPSuAlOymQIuXX5Tvo5EmNIxnPGSeiVBUBzt1gCHDpzYmwBsgJXLRSkK47ER4UeIq6imYW3tjnw+9N9jPUHKGKv6aqWm+/Xy+OsSYw2314K2UbmQ15ijLLTyBgZaG0bNIkJZTgAOAjk8fQUCXqSzaQmbSng8KNfa8NgD6PT72MllpTvJEfCOwmE2w2I04aGjj4gNXb58FW67IrkUSyvoDxiAxUArNp3onTiQx+EwoIV8qSF4hUy4L/z73xpQ18cPiS/UbjWL+2aj1YddtxRlIcgjn8fjFVIBsfBulxZlpEhbsbq5rtrBPQMsRgN8gTZc1lHZr5jGOTLqEiybQcfx+AHx1BfPWiOXQxoqDrCyfB8PPfw+TE/vx/raMiYmZ9Qr1Mduagu5XAZ37l4XC7YDB2bVXu6LwGxmZlEHiwuXxPGLdxHvGlb1ZO8Mk8z9Urj+7ATg9sGax2pxCfzM/Z4TgG+Id7fPG5bX12jSD8krJxZxUCnnRPHLXXJIhrXT+0Y2pzY6+jLL7CPFseii3a0LsYWvLxGNweHiZ0KaWx9ePYWFf29qN2HodWE0t1GquFCvVwSlzVT1YM6QYh+bDWZ89Nm/KehspZSBwTwMqeyg0byDXq+Ejt465u9zIrD45CSQG6XUx/8BNkFfM0qJRRAAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAAgAAAAIAIBgAAAMM+YcsAAAABc1JHQgCuzhzpAAAgAElEQVR4XnS9aaytWXoW9nzDnscznzucO1ZVDzX0bCaZjlCUf0H5E1lYIFsGxbKxPMhDEyuYliBABhQCHgAhQgxCGDm2MTI2xFEwoo2hu6u6u6prvLfufM+85/mbkud519rnVJOsVvU5d5+99/d9a73rHZ73ed8V/Mo//NvFZHCIcrmMj714E0WegiNLc+zt30AURQAKvPXtD/R6EATY3tvDw/fvoVStIopC1OoVbO/soFarIy9yPHjvHQzHM9x7eITZPMXW3lW0t3bR3ezqO4rC/uP/LZdzRMUSUbHS38a9Y7S7m9jc3sLh4TNEUUWvn5ycADk/BIznmfueFGEYohWnmCbANA0RBCEatQqiKNZ/cSlCnuXIshRZXujf9W6IYhkAOVBKZ/jC51/T9/3+1z4AwhCD8RzD4RibW9t6vkopRqdV0z2nqzmOD5/ZfUzGaLY6aHW30Gw0EWQLzhDiECjXWggQYjHtA0WOIACCKECgT3JGc+RpgTwH4lpF856uFoiiAMui7t4FJFmIMCgQIEeymqNeDpAhRjlIkBUpkOWIglzvL5VreO0zn9ecPHxs93j45CEyhIjjKj752mdRrlQ05/P5HN9+630EFAB9OApxfb+umy0K+8JabQPtdlu/f/joCKvFAmlqC3X6/ClKlQbK1Qo6G01sbu+iWq0hjEKECDCczPGvfvt37XvauxKceq2E9vY+SpUqGtUKxrMlUi7qaoQo5+QB2fgEcWNDAsARRE0TgMMHWMyXGA6HyEwOUKrEGI3mqFUqGA8nWBVAWCqh2ahpEsIgQqkco8g50bl+D8ICeThHjKoEIE5m+PRrH9P3PXg2Aqf6+HyIk7M+dne2tXBhEKDTrCEMEixnUxwdnyEKgOl8ho3uBmrNDqr1OsrFXALA/4rCljpLU81nGKQolasoigIZBTLlPGtHYZ6HKLIEcVwgDAMUYQ1plum/UmwboFWNsFyMbU5QmAAlqe4vDnO88qnPo1Zv6O95luJrf/AVuz5ivPyZP4JKlUIVYDqb4iv/1v7W6nYQ/A9f/sni1q1bmrDtjToaVZvcIs9QqXb0Ie6C0/OJ/QGQIBw+vo9SpYk4jtDq1rF35Toq1dr6PbNFiqcnY7zzxldx7fqOLUalg/5wqPfcuPsxLcp4MkOezYHEvj+YnaPd2UC9s2kPE7X0s3/2DEmSI1kuEEQhTk7P0Go3UC6X0G61cXR4iqdPDpGsUtRaXdQaVZTKJVQqJeRFgSLP0e60MJtPJQDlgAsaI1xN8cmXDnSNaVrBeW90IQBbXQRhgFIcotOsaPEoAIPzM71/sVqhVm+h1emiXKkiLijEGbIsQxBUkKVzpIlpVN4nBSBJEmkp7nh+H2VgkcVANkMRxMiKAHlY0mcalVgbh6MUR1jMbY64+KW4jHQxw2uf/QLi0HZEuVSSFuH4xutfx8Htl9HsbKBSq2E6X+L3/s2/198oaNR07VYLwV//8k9KVu/cuYNmw1RPtQwJwmJBSQaarTre+Ma3ZSauX7fJenTvHaSrBHEpxs7+Lvav3UC90dD7ubD90RSnA9MWm90Onnz4Dup1W9RVskKvf4L2xh7CcgPZcoxsNZV6rqUjdLd2EZdN8t97PMbtW9cQxrH+zTGfTnF+8hRFvpLGanY2MZ3MMByOpO5n8xWKqIxKpYxmvYokTbQQrY0OEAFFuEQ6z5AlOcr5ai0Ao2VIlYNHz45xdHSKTqOKOA5Rr5Q0JxpBoN0bxTHyLHQLAiDkTkwlaBQA3gcHr7tYJajVqojCQBstikrI0oTijTDIscoCrJIUpUod8yXfe7GR+H3L1VKvZckC1w5eQKPZBI1Jtcr/L7CcjjEa9HDlyhVds9bo4PikjyTN8OjZIVZpivFojuEslcbdbNKsA41mA8Ff+bmfKqKgQBRSykJ8/OOfXE80dxtVH8e//J3fxp27LyIKI9y5c1dqbDocon9+iHa3hb2rB9jYoso0iT0+HeArX39Xv9+9cwv7O9tAUMF0PMBiMgKCFE+fP8PTR481CXv7V/T5BqbY3tnF1s6ePvsH33qin6VaDdVqBbtbHQnAfDrCdHxmJivPkOYB0iyQXzEcT6Tqo1JLAkA1vOIk1iOEUYCwFGM8HKNUrqOcJ7h5xcxMGncQRCWc9EY4OjlFq1xClmcSgGbFdlkeBMhzW/jFbIUszyWs3MtxFGAhmaTNz2TfVwvu2gDNZhWlUkUWQjY7T3SvvPc8KJtfFFckIMsVhWapa9D8XL9xF41GA1Fo101WS7zzjT/Aq5/7LpyfnqBRsddfeNHWjhrvq2+8o99XaYKj3hCjCYUzQBACtbKt0fZmywRA/yoKlKMcpdLFTtu/cl27mU7Vv//K76HZbOLKles2WVmBu7fvaHLH0yG2t3fQ6Zi6rlQrODrp43d//1to1Kqo1KoIwwgv3f0YmvUaRv1TpMlUKpB+wDvf/CrCgJqmg1vXt9DpdLGxuYVatYL7R6ZCD09PtYs4Ibtd01TjYQ+r1Ry9syM0Gk2EVJ1BiEq9ifF4iMlkIH8gDmOUoghJttLfo1IZo9EEs8kEs+lUpo9jZ/cqOpub6A1nOKaJKZcQxRHqlRgxTJvleYbVyu3uLJQApDkQRkAYV/XdaUr7vMB8JcWgUatFaJQSOXTc+UVGE0BnNkAeVLEsSlosmQU5B8CNWy+iXrd745JlyQr33/mmtMfO9bvIUUaIAqVipve8/MonsUjsPt986z5GswV6wwlmyyXKpRZyCVCBKJvrGttbbQR/5ct/wa6WUU2a/ahXyxKEQe8UeZbh4698Fk8fPcBiMcNsOsb+/lXZPn0sL7C73UV3c1tOYKNm9qs3muL/+v039XtcrSua2N/dR6NWw/W9HWRZIvt3enqEcf8cy8Uczx/fw+7uFjY3Omi1Gtjb3cejk6V2KspVlOMQZ6cniJ2WGQ1PsFrMEGQzVGsdlEpVBFGEcqWEuFxCHEVYLmd6P7XZcjGTCkcYIy1inJ88+08EgE7gfLFEki4RhwGisIRqKUPE7SwNEGGeRnIWR7MUYRQhjkzAqrUYacbFT4BkhvGqBE5HpQSUY+5mCjO3eoAiWyILTbumQQXVRgetzoY0KE3tqHeK9uaOoo4nH76LdLWSEFy5+RKqjbY002gwRhiuUCoSScjtuy8oUOKd/tt/9w2b+7DAcJIgD0pAFGkD5StzJrfp4/zlv/SlghelRKTJCgHtqhurSV87gKPZ2rAFT1MMh+eISlVN5ubmDgrKoZPaP/KHPi8hOBuM8ca7TzCfL5DR2Yoi7NC209YA6DSb2N5oYblMMOqf4OzkUK+/+sqL+ODdt1CrlXHj+k08eD7S62Glikqljmq9iaPD53ptMjhBlq4wODvBlavXJChxXFLYyQlZzQeoVrlLQuRposlRQKWfMU5OjvDWt76Fz372VX1frRJJTTKU1U4JQsxmfVRLAYLMee1hhDmdNqriNEWgXRWhUY+RLSb6fkYA8vLlcNmGqJRzmSveWBCVkSUJShTaahNxrSENTPXfO3mmnd5odTAdDxX6abF2D9DZ3NHv/bGZh3TZN3PDcJCm9sW7KNJM3/X22+/q3sbzHKNZAsQVaSZdR84q0O00EHz5L/6MNAAdFAqAHxQECgAXlo5eu+W8cj0W5HBRv0VxBZXWLkqghBb4zKdeQbVaxWA8w3uPjhV/H50PEccxrl+/qc+WoxDVckXS3qzVkMwGWM2GWK5WeOWVO5gtC2RFiPPDe1itLOYPoxJQspB0PJkirtRx/PwBVosxKnGMJAkQSQXHuHHnJbRadR+QIVmtECHTzuLdZxTWgIId4dtvfgO3DvbtsemZO3VLXyHPCtup6QxBSu89kC+RhuYR5hStIJBXX6pUEGZLh5sEMg26b6etgqBAtd4WNqCoJGRIaBebL1bStHnGhS2Ep5TpBAd01EzT8l6Hc1toapBev49aKdOcMwzkl925fUvXnc1mOD7p2TpNV+iNZojKdQRhJAWYrxYKTeu1CoIf/9EfKqi+Gs26VCZHSluUZygjw2R4qte2d64qvvRD8TgD8ihGpb2tiYmDAs26ee/c9XGto7j2+KynSepsbqEUxyiXq0hThkoBrm1vIUyGKJYj2cyXX74p8GO6yPUwYVzD+dEDgToJcYiwitFoZIIZ0pHKcN47R63aQrlSRxjFaG/sc44UojbqDYVeq/kEk9lCE0UlVGPcXirhwf33sbdtgrWxua2FiEsV2V46W6vlDOP+CaDFoYOVInEePn/w+eRYBvToKRjUAPw9RFoUqFdsPrTz4zI6DG+DGPP5Sv4VtdFwMlFktLmxgVKppJCPGpOrlTs8Yb5McXR0gvfefgulakO+xgt3DiQAtTiT4BK38OPkbCgN0B9O0R/Psb17DXGJzmaGk8Nnen+1XKIA/LAUIjVAq92W3fMjWM10gcnoTF/gB9VOv3duAhBEiOpt2R3uNI5apYRao4Gls3HL5VKT0u4auMMhew2gXS2jU04RpubIfOazH5cg9CcmbEkeypumg7eYDjA8P5R20JQWCzmPFKz5dI7ZbK5JbW5d1+QxhldYmuVIkxyr5VzvJzZAJCxJVkjnE4SZYRMvffIzWCxXyFBCo9XC0eP7ev+MmjDlrimQ8SdvvCgQEGmMQvkA/JlkvGaI/f0DCffpySFu3rqrHc1rrRIueIHFKsNstpRgTqYTLRJN1d7eLtqdjhxNPV9gJpdzfO/ePdy791Cvt9tNXfPK9QP5Us2qzUejXsVy4QSVWioIMFtk0sbN9paceEZ6H3zwvt7fbTUR/NRPWxRA+1MrG+zrL96qMF5dIl0usL2zh+lkuLb142Ffk0qzAtmxJkA1zfBocIqwUke5tSEBqlerWMxnugE/8rBikUeRYbtRoAoTgM9/12cxmWcYzkwDJLlpJQpbuRQr/uXCcjx98DaCIsHV/X1Bz5xQmpzHjx8jK3Lc+eTnNZmEXLMkRbocy2YmWYJqraUdUQpznDx52wTgE59WPJ6HZVTrDQx7J1hMJ0iWU2zvXdWi9o6fCBjjuPPSq0I95VfSF6CkEvKVaBd49uwZrl+3jTOfM7KgWiaoG6HebuPk8MgEdmNbgtao1wQYxXIWbUxmZha++fq3MZ4ZEFSvN7G3v4NVKjwH1ZKhiw0HVlDoqX05BuMJsjxAd2PbNEC2xPPn5kNF3CQ//aWf1YoHSNF09iZbcTEKtKolPRxHo8oQbH1fOHr22GBOREiiMuL6/xtTxoaW6cvjMkq1uhZ+f3cXk/FIYR1HmqbIitgSAukSnfISrbI99Bf/+B/DdEFHx8Kko3PzWIkyFnR2igJ5EaLVbODw8T0kqymu7e8gikziudufHx3r90dPHqJao2krod3dR57SBFCoMswXiczQyfEhwtTs5R/+o1+Uo1SghHKtivPTI7RabfkGdD45J6UwlCBy0NyslksMzns4uHmAY7egwv0V7tEHMGGh197ZuYp6izY9xGK5EGBEePz0dIQoWKHOkLlSQSI0z+U9mOTgPBydM3gUekezQdNDuJrfXi7ZwnBB/YhrdLBXMiFEF7udjnwgbqrD508RgvNLAfiZn6XjKhtN5y0KCy06selqmU6DfWmzYjtRDyVbaGqKKvPxk6eIKm00NsxR5Kg3WyhV61gsFtjotLXzilQZIORFgmRl8W6aztAuZ+iWiYwBX/zPvhs01aOpfc/z84F+JslSoZauH9kC9I4fSwNQABiWEvLkjuqd9eUM8n0BQ74kxQfvvyt8gg+nqCUsowgiHB+dYGfLAUFzQrxE62I5b3vXbmM07KHZaiuJQuFbTKco+8io2cBiPsdivkDIHENWoNm1OdjcM1RuPh6i1mxb2CtUtaY5pUagABDcodAN++do1GJUqlXMnK/Cz4+nS/lNtrlsTmhqnz15gFVeQr1aWW9MmjjmDzhKtTZmkxGiuCQHutluYzEbSK4Wy0RgFOc/+Imf/FJB548CQBWdZpZgoCAQS+fNBkStCKI4D7lZq8jGzycjqcynT58ji+sIY9uFhBgpAJtbFrZwEXj3WUL4lf5AgWq5iuFkiFIpQKNUIEr6eu/nPks7DMws+sEksQdCkGM+NRXonhHD8+eoxMC1/S2FTUyGcHfMkxCrNIfDaxAXOeIiQyJJZ2YstlCwCNA/7yFgLkLXyARYlStl865LROYiqU46ZRTi85PnAnE4aLMts1mg3elilXHzyEghy1aax5rLj9BR1OxrbgM52hRW05ghemcnyNO5FptOIgcF2D8rgyG+9fjwCZaLhbRvVG1KK5Zc7mA8GrkQFlitLOdQqpTRbreQrEzg+P45w0LdTYHgh//8j0mu6PW22x19AW9M0rucYYv4ufD3CSoxgRGbK6r2mzcOdKGHDx+iiOsGsKSpPs+U7pXrhhoSTKH6ymi03Jgu52iXq5gnM3Tq0dp+3biygelkjieHI107Jwgkh8jnkIHp2LCBxaSPxbiHrc2GfJR6o6nFGkwWWCYMtbjLAwSaZCZFCeAUSJnoKleFEA56fcA5oBRcOYwuJm/W6siKTBFIwQQOd99ygVHvSNc/OLgufyRQ1rGEx48eIi6VFMLxXmgetrdNI9BsKNUeWMqaGIJbBeR5gdFwgNn4TP5Fo2mYi56ROMlogPFoYrtbJoVwNHMIKSqlsj7PMRoO19/LDCjDd6r+aq2K8XigfAkHP0tntVwqI/iRH/3xgqqLadxdh7/7i/dGllDg2Ow0ETq1rxdWM3n2N24cYJnkmK8yJT2mU7tIuVLD/vUDqTgmPzjxy7l5qNwlFAgmOfhbrRwjdjj3Sze30RcUO0atFKCoWoqTD9ppt7WrZpMe0iTDcj5CnqzQZJq5XUWj2daOmixSRQ+MFrhwAj+CQKAMZYE+QFBEFpNnMyzPDYQ6uLGnnMJ4xawcxSUTTkFPiwkl7pl0tcR4YKHxy698AkmSytYSAZxMOSeWeq5WDHPY3OjK7hIC9yaakYAWwfkHXgP0Tp8pIba5uYf5YqE5m4zGuuf5colUJtTQVwoQw1i9Z2L2krA2r0XHt+X8OS4055t+BTWYB6b8Ggc/9uM/WdBB4QRVKgZNxnFZuHu1zlDNfIDxpQiAEzEf9kFTWCmVEDJpUgSo1pprn4EgCdFCZrG6GxuaGKaRE6fSw1IkwMLwABOKejlGSUQH7hbLdGWxS1x0NtBqdeVU5elYpAaie8v5BJWIIWxTJoDXmS2W2u/0fhmzc1dzUphr0C6m7aWaZV4ACRbn5hXfubkhDKI3JwISY2ezrYlmyBgy0BUhhBrguYT/5dc+jdVyhX6/ryRS7nYmF1cCHQDdLoUyQK1SlpawHcDQNJNZYXKNWrR3forTo8daJAJG/Bvfs1h6/IH5F5sL+lWz+QL1elnPNhgOsFxyge3r21s7yFwyiXMof9T/0YWX/j6C//Znf7ZgTp6DNyMEyo0S06AVLmJdiQo/FvOpJPrk+VMtXFQqIYijj0hXo8GogA8cod6oa2EYMtHL1QiJiOWoV2tSe/LcRUyJFatSADhS2ATUyzV0W5vGAGqYHaK5mY17BlARX3c32NloyxwxflYoSSBotdCiit8QRyJaRCHj9xzzM8s4vnhrRyHacFnBPCnQbdfNJNDeSpEUymaePH2gBNfLr7wqqJu5AyrKqFTRzub75SjSH2pUFaKVS3SyDf2kSVHY6MZisVTy6vTomT5LR9A7fBQ+u+cYK7cE1LJcdCbUOGgBys0NVLyAFcB4YJENTU9Zms+cew4RSjRZOYKf+/LPFSRRUFVkaYYF85luNFrNNZTZUvgCY/BQJXORCnvYMZGs1XK9AHQqyQ2o1hp6zUK4ArlAAxtbO+Yg0jFh3p7qjGM2n2E6na9DLQoMTcXw/BzT0RTXrx1g74oDlATd2g6nJuF//PdiOkZULksgmVTiPdLp8gASBc4cPE5OgHnPNMD+Zk2p3QVaWGZQFtDNmGJ87s5h/0xJq7BUxiuvvGrsnizHivQupy2NjWQfpTBzNOoVPROTYvQTfHTFv9GR5uL0z8+xWq0wYQ7AEUkUMmcZ5ssFjg5P9F15XCXRAK3NbSOULBcytdw8HNQ484kJADVmKWZizPAKuneMSGiGCF5JAPyikGXiB+0KM3R+XDswHF83nGaKGPiguvHeOeZ8rxOx5XwmrUHPWIuccwdUdKN+cAE4CdwVhGt9BJvlpE6VJYx8cGokfu3J4XMJABfh7osv6msqYswwKUMAiB6uA7Fyo1bRryFfkZuNnrcHaEQRU3Kn0M5KZhaB1EqxyC+obiItQoQFF4ZhIeNic44H/TOcPH+MerODK3t7psajkgCbGu28hJrfbW47OX4cxN2J1PF56JzZwpojSD+FvtCTh48xmUy0sbRhMmA6MxzkbDAUrq/nrtbRajVRqtQUDdBJpBbZ7HTXm3A8ONM9BuQjin+YSBNykOSysX1F8xz83F/6i2vor3JJ/dOzDANzACeTGRotW0yb+KqSH6bOIqVgmT6mFB4fH+khKuXyOpfthcBLqL6EKprqKKK65+7ycshkCH0P2znE8zkoucPR2ACVFp3BQAwm4uaVakk7Tsma5Uq7iFqInyWylud8+CUC+jpEPCukZhGLN7glW1l4WWaIWKmi3NqSALRq/GyOVDH8RXKHO1QLlyWKsyncqyTT/SizKiYQmSH2LBxcfN4nzax4f3xmN58MEYlVvPG1r8oX2Nm16CkxDFy/f/jhh0jdNqGzS0EKo7Ic77MTM597O9souTVcONSQaWRmTHlXbaKBcay10ZwyKv3SX/jpggsqJ81h+ba7I9RrTB4UcoKIvvkQhh8mb0+7IwrRJDrlvE8tdpbi+OhoTSiltMsGrUUIglqlspWcMaYOx9n5udGq0kSs3JXTGkwg9Xp9pPSoaxGarU3U6w006sS3KYBrBeSAWKU4XLbO/pYmJGQEykQyLJxNmIHMUIjNq5dkAurdHWEatWqs52JSiECQAVfJhWZ0dpV3zvCz1agrqUPtQq3E1xui2QVot5h8YuRhgkrN4W0yUcmHj57i8YMPFOptbFgYSA//+TNj9573e6i3uuh0NtDutOU3DcZL9PpDmQyO2zduaE4p3FIf1GpVE0AKHc08zZ4n/SiV/JM/9eNrE2C8k4sc9nwyxsGNA1PJWaCHkhoJAuzsXNXi074wiUTb7ZMYpoo5WSlms6nMAXcDvX4DP3LtGD6IvHruLqe++RlO+Gg8dqGSrQ1Jpf1+T+ak3bKH2tzY0YN1u1vSRIR9eW3eI22u5yfS1lGgoxIJgQXSZSKnjCQNaozl0jTAaDCXSWh0Omanme6NIiWPqDnorYvv5ya3yte1mFpWhDQTCvE4h8xbQEkeDgJrlTLNBZ3DmGxumSSFx6sVvvYf//1aRZP/SFoYR7qYIMkKOeFXrxkfs14n6znA/cenIrlMp2Ymru1f1c+q/B/TsJxbC39tmVcipULXpJAEP/pjP7IWgMueqS4ut9NUH334m7dua7dysa9cPdDD8D8mJ4zfamO+mOkGvM2xMKSw8MX5FQxveE/0C7x22N3dNXueJtI6egA32QRVPPzc7Wzi7PRYMbPek2W4cfMuaMK4uyrlmu7Gu5ze8aIypf3zPEL+3O4QMzcBGPRnzqQxC5egP5oLPydANJvPUa1URMj0KpS21TQ0fzq2TZYJO8lS20xC9vT4pHjnmit+57Nnj3WvFOgnjx5oUzGK4JyQum2LFylbqEVVtpBMqlQpYwr1o6fHZiKd+ayL+m2DGkJIn4Q4Fsdw2j8TcNXdMCeamy/40T//g8xrUkTX7B//JSsxHO1ftIOUpJ3dXbSaLswSM7WKvb19waecaELJ3FVcLCY3tCMZkxYFRnTiuEOQYzAwtcUdZ2BRhIE4Bika9bq+R4sytFwAuYX+ZqoVA4eSdKUdORz05FhRG8mxLFdw4yYdRdKruHvpeUdaQHMOI8sS5gUY0u50LReQZxPhFIsVWTsVDEcEdhhdLBQOcy5okqLI1OtyvkAcV1Au1+Rsci4Ie4spRKiLfAZn96oVOn4r4Qgkc0zGA8v5M7ScLY1QEpeUT7AMoJFLT8/P9Xu9UcX1a/tS8VxchrK8v2atLFCMQ+l5Uc5o+5wGWC1Bx5o5C84VNznpezRT1DTBn/2+7y0868Rw6AC5S7pU4gsg6Lxv6Fep5CIFl/o9uH4DL33sY8K8ueupYjgI3ZJUwlEthwJn5g7gFxnC3iRppiRSvXOQ3CHN4nYO7TvHkyeP0Gw0tMDjyQgH1+5gvphKYMajnmJ6C7NiXNnfRbfblaqVY+vwesbJcbmmCh2zOoXi9XrNYe9BIhs8GC6Uu+922rL5vA4cB2G5WKHiWLhMhFnRSYFGo4ZcMXxNDF5qsEa9LC9cAtyo4fzsBFMymmdjh6s47IJEkiCQWmZE8fSZhaUcp2dnsuk0szdv3NBrEvZSSSioIGFXC4DiQiN6f4smQDwGF/VIQxLw837In/2+P2UZCnm1majZfhDPFr8tZ1ZqpFh/PUS4CFFrtLHR3cSNg+vY3TUqNz1USh1DLw3x9y004wQPR9N1zCw4NDBqld7KHTGZ2O6gOZkZSDUaD+VTcIzHFrbRhlEFrlYG1ESEceIIV/Z2cXDjjmDYWrWs5MlsNoF93IAYql4uHt+/tpe5lY+NZ4wkMty4vmvU7SLXLhWMTMZv5sMpFp24rBpywUxKvKSJhDhZ0oRwl2eYz8dWzJEytVxHvz/QZuHicD64sbyb/PWvf3U9zWLEZUu0urvY29/HYjFXCMj7Jx2cn18sjBFkOQ83mPam0+vQRosOLGozpzxHr3eO4Ae+73suPmVcyPXIoxA3XPyfO2TO//Hw+FC7lF/W6w9w9cpV8flfe+1TuhB3oE8c0QEiVElyCXcqb2pK2o/j53Hpfc6BqVFJt9s55qQt5V1TfZ2dna4RL+Lt/CxJqpQ1HwrWK4Syy6iUY1y7em29iPTI5Q/EZZz3qT2YXPPWsmIAACAASURBVGFixR6aBSXSPKFRu4t0qvdz8qjaFd7JSbV7r9etNrIcM31MwkrZObtDDHrH6HR3RBaVrEqTzBQ2skKIaJ43Q0fHPSxmU6eZcjx+8lCO8+3bd/H0+TGGgwF6wzGa7a6QVIaU1LgV1RSaypcPRpNDfrpLWtFBoQbk7pdprFTlh/X655iMx5a5/OEf/H49vcLAS0ANX6On7ndixdldvs6QqNejanLYZJBjY3sXbcccpi383Oe+IISPg2gY1XOaLKWyjBLucvv0rpMLhJDO1jJJxZrloDeufcuEhotANh3vgELC9314/20jU5TL8ieIvzM0Ypi3tbmpHU6hajQYs5fRbG5iPBpqMmivY/dsX/vam8r9X7lGSlck+y+sg5kzhsqMbBJiDKatWs26aioZVlGRMdfBqIeLQgiYAlmtuvsHaXRn0nbkLpBTybCbgNebb72t62rTxKEWTM9cZBhPTOu+9fZ7iCqEmjPs7uwqvBRHYc3gIvfZxtJhEPxdjj2dbUVpxsEY9PsSfsH1P/Hj5ASac1Nydt07YOdnp+udmPjkOsOvnV0M+gMtpkqdiDdXabftYblrrl65qVQkxwsvvqDQhGGjHsxxEHljXADLYNku5A5bLBIBPrxZJlCYEm02icv7nWoRgsGpImKj02KdYKx8Q61WEgOJjg8vqZ2WZdjZ6cor5y6i70Ani2gaSE8D8Gu/+hvCN27eZkRBkIU2me+vKoanmdKud+CMcvF0YmPm88/x5je/ijt3P6YqJxal8JpXrxoxJE3mMqXyGQAcHR5hMhppse/du6+QeHtrA6U4EJeBg3mG50dWh3h4MkDEuJWU+k5b2ca9K9f0DNw0GpcSPmWH6lpUlWIymiDPU803nVc+y87eFQRf+pmfMAEgq9ctIP9NBm7iPHGqvPfe+bauQeklAENAxksYySGcJF9DoPAjDHDFxaVf+MIfUsy7TgRRi7gsn8JDpjrdteSdksnjYWWXAaHjQzXLmJiADgftv1LLxOJXiQAhvc6CDpE1gcnoVDuFWomsWy6eIbsG1dIHec+RJFm6RvNADVIqRfKJqP4bLMTIFnoGapLF0nwRCgdt0GDQw/vv3cetWwcOHi9Ef+Mc7OxYyEXHj1qFfhPn89Gj55iMSWJNhHCSZUatQWGeubQ5P/fgiaWqnzw/1T1wfPwTL4saRnSWG4WCTh/iMnTPiEjryoKY5QKL2XztoJdjg6PJiQj+4n/3pbXVpwfq0UBCp5wAP95+x6p8piPP0bNQrCyvty2Qxe9wv9AUBArG3Tsv6oJ379xdYwfE2WlvDaFjssYBH8w60kN1uXIrvDBAw8gQF+VW1A4K6zRtppb5fVT55hOQmUP62UK7VEUpTOmmS3n/RMl4nf/zd/8l7tx9DbUG+wqwv0BZ7yW7mQJCRrI3Z/S4BwNzTIkW0j+ZzudKuV67cU0mYjKeokEcnncWmL/AqqA4rglDYWHLaEQSq3EcVcsYkEJuAk0hp7PJ8ezYoqNnx/R9Qjnc1BTbO9vgx3l95SocSccLptjEjj5GDEUUNKbeQ5bpc8NWtOmDH/vRHzIfgJi3y0Dt7pvaGg5d5UkQCNqdOVBiRaawMwlKz3YJXV7YccKiNAdcdI6NrU1nmw3i/NSrn0HL1aHTG1VRhAd8VuZBW2MKS+IQFbNcmxMG916FkMIQjPRgAmg4u5i6JIGQNCHQn/G8wbP8nPkkU5Wnz13DCXrsdKI67ZpMQKdJe0wcIxMUToGm7RwPR8pYLpZzE+A0xtaVXXOy5nMhc8QGfBk4S+a63QaCsITpdIIHHz629DTNXF7g7XfeEUZBbUE8Zbnk8xYqgDkfmHqfLJZoNw1XuX3rBrrtNg5PzqVN1pnaNF1HUySHcCgLqsSaC8/DAI2aZXb52eCv/ZUvSwCGoyGGDpzRXzVRBmcy2TOgA+McQ33Y1RHOZyvj2KmziAkBizM16aQs11qo1mlnA7F6NjYsqfTqq5/D3lZH4eJFZpzAEJslFFoko02Ri08Bob/hwlVXLEG1RxjW4FXL9lHdUcp9xY4P8ZQ1FNvZKGpKhYYMcVPMSEOm6VCennUNxBOAKpNJJdYXJkb2YF5kNrbogaHp9AiLVYopa+/Aer7+Ouy6cvWqNFyZJXRyZolOZsJC3vzmm2i2WXpOvkKGt998Gxsbm9i/ui8w6/j0dI2E9ic2p0wMtVh+VuS4ffNAAvD4yXOpfmZTObzZ5HwcHT6Tn8Pnvxz6MXFF9jefQRrhl37+fy6U5hxOlcXzgyqD083dyUHJ9b9LOMplxfRMRWrHKdEAEChZDwfmMKThbmBCR5h9o44rV25iZ2cbu3t74s7Jc9X+zXDzyq6Bz/ID/KIT06ZWCDAam3SLLasCTgqAMZn0idA4c9wt0jCKPHLnmTP34B0m5QLXxEtSy/n5KCRDip9l8qZAkE+F+PG7GM/PV3MMJicoQlKtciznjEYi9M/OREtrtbu4cmVPz0o84cnjxzg762MpYiYLVFLsXd2xwpUsx7Nnh6K5EyxiZHJyaugfkbosMJbWMmXcZNpgZ6srwSLbV2xuRQPUdtFaOA+fGcnFI60KFuTzECo3oTw5fo7gF//Xv16wZowq+PC5fchP+lL4jUsiXAaBXPxOk6Fcu1Q2d1QoGPj8/ESLTXXKcIuAkhcAfjtFivaT/sF3//EvSk17lW88QbKGzYlpu2pj9S8gidFV/NB++nvjzjRRUVWGMXj5C0kP3rEtiO8v5eMYIdKuyRIrMxtWzy8aGlPHTNLwGiSdZDMgn+n6i+QcE3ftjc2rKDJS1EJVIFcqbSWkWEuwWE5lMr/5xjdVXxBENUG93JUEprZ3Ntc8/XffelMAErOqG1s74vv7MU1IIZ+qM4kEXWBTW7mBrW0D7S46D12kmM9PLUXsk2ws5KG58drL/gYEv/A3/2phtirHhI0bCIg49O3yZpYad4OLzJiV76NapjOm+HLQWzNZqE7991DtcHChPNMliENNyJWrN3H3hRelAvUwvJeiwPmZPUCDmUTVJdADt8obFYq6DKFXe9ITLnKg4HGheS0lRXwyamnhT63OHWD+Av2L0cjyEsdHhGCpEaaqoKk3KdTUbglaCmkZ3cyxu8VGDAHKETGGisrcdK+kcuU5Foupqo5ZL8CWOJyLrb1r+PD+fe169jTobhDUqUo7nR2fiLdfYTaTPsEH99dz7VlGLPrwvQK6nQ1FNfYcF6p/3eCLfsrIcigM+WgWUyXTTCuyyprajI5i8Av/y3+/jgKoBZhs4KCHSIDG/7HbvYCI+fe0oJq0hgP8Yjo1Uru+jp6tTVydmi9W8GGqSsZTmptQqNv+9TtrAbh1+452rS/+WJPj0hkKkPoU4LTvNVWGa3svqer1AsEkMsZp4/8MYvbPoHwDJ6BWxmg80k6czs6xTEzlNjsM/6posiFUmKDXP5ZjWa7G2N19xdU2LNCp3tX7aa+5K2U2SmUcPnpPfhMjhumE9PgCcZU8hp6IL29/+x1FO9P5BPt0GlW6XmA8Zg+Eiuh2Dz+8h4nLmVCTkWjrr6V6C0d85Vxvbm5Kc6uQR4mquYWmNNmOI0DEj5R+3aegUoJaNW1W8Tv+p79qjCARP5yXv7llu/Hk9GQd79+6+cJaKgmTWrhoUyuMP02wcp6mfZ9x9XQzjiruAzi+TpRLErpMsEyIZbv0qSpfS7h2zVgxr772af1M0yl6g2fKDYRlQ8dWaiyVYau1g1LEEI42PEApZGKGNCjz/judLQnmajVTlpL/kWyaI0USDBX+cVRKe4jCCqplEmJYes1GDDMs0zPEoTmvpaCEyFH6+71TpWK5ays1FmxmODs9x/vvf4DzsYVdYUzzavELTQEjDdLODm7cFF+f8947H8ufOj05ESgmcMoNalqmjylgTMlzkBFMSPyll17SyvsIinCzN4tzxxFgNHDK8jdmQN1O8ICdSCk/9Of+tF6mN8rWbRzKpLER1LOHa5X9uc/9Ib3GR2FYwxDCEz80oYmFcp7rTlvDIkxKHXcbB3EG0bHZ6eLKDWPYJIly+2dnVmzBcrKw0pBDxxCPhBSOGzf30es/USbrdGzvTSUA9I5bKvKIyeRPczy6f4TNDtPWXexudeWnhBH9gbpFC1HqScNYMJoJLESiLWdKiS3kGCWUyi3j5+WpMn2MTMg+Vnrb4fvCG0LSvmMlqYjwVeptnA4Y5gEPHz5YP/tkcI4XP/4StrbaAq2sQijF66+/I7CKPokW81KN394VI3mo4mlp93l6eiLz+vFPfMLExGldgj5eA58dW0ZxPpsKFeVWJBuIqX+20tNcM33+J774hyUAL9y9jf6ZpXzXw8GJpRLLxlqo12q4ds2qXb36V8c6mQ7a24v+QlR/XvXP2ABBkUZfKVaOu7c/pcll27ZTCsDpsXLoRARV5++ufefFl/T+q9d3MF+cSVBnmUUrtIsq1zo/FSmyKS4h8O1v3MONqy9ge2MPL3/yE6rbi0LStGwLCMAjni+NyNjfnlhxMSKUYyaNQjFxJeTMOjpcguYpd2aSyBuxfhI+ux1yAqry9s97fbz/4XPVFfoy92ajgzu3rgt/aDbYYewC+Hrvg8eKqmROLUTR/TDzxxI7W8g5JlOLAqazmcrTbr/wkpxij4BOxwbS8bvff/cte+9ipXZ4VPve0Xa+pFr+BP/Ff/5FXY29dAhwXB6bu9ew0facMpslFXp0WutWLxKGMFL40u0alcoGacguxcva42SFY3b7dLO92WaziBJu3nkBh4eHOHx+qHq60XiARmtT3DeOV161Lp4s/j0+eYjuZgMTV9tmDRepdvuqjiF5VeycYiaTUA5KuHH9loT24OAAIJSqFj1GQ1MKNbxob7NYEEiJxAuk7WerFj41d7jfmaw09irUogyGVSE2OhVsbbbVfIopi7e+/b487te/Yd26cpRw69YNYfg72xtq+Wbp8Rzv3XvgCkSWMifP19EYsMt2N45Z7efE50mYuqa/QwCKmD9DdT/e+ubr9msYy9mr1ckjdA1AnKlW5PY93/NfFwwzOHwSwn8J1Z4ve2elC5s0Vaq2y32dnxVCLFV5S9V4lelXN7zNYg0A8wS+tw8nvhyxFLqKT77yGgajEcajsYTo8aPHOLhxAzduGg39+o3b+vnhowd49/4bWK7YWtZ2RQBCvhFipm/ZCpZZxDzH0+cPAXYezWfY6V7FrTt3cOvmLdy+sa/wr1xh5ZElm8KQLd5sYlaZaarFir0JILCFWiJdGfFEC1lQmDNUZXaElOi7SKDd3aprUQlmHR0fywH96tfekCn78PExdnf3DWFs1dVRzcq1M3xw76HunxQ8Ntx84+v/cT2HTLHTj2HVE+v79NwhIXTjK3Lzeej97MzqBji+9bo1hazWO3p/neX7Un2WQCMwp9rAP/W9JIQY1k62in53NoV18X6wyYMfLF9m5o2DD5eurHuYD/t04Wod9YYtFEM8RgAD125mY3NDUDKF5uOfeNWKSFwV7cnJKQ4Obqx743iN8uu/+VtymvgACcunHV+wHFe1oDQBytMTL6i4h2QlrrwWg4z/2B/9I8oqbm6wj6/116FP4us0UxJXiBiy/VvGfn0LgU2kdovezTD2kqnzzF+pYW0QPneEWr2ELDF13R9NFAb+3//m91GpsUcQS7UyXLm6JwFgR1Wyj4mbUMVTgB4/erDOjhJ51KIHZpKM30gEtKREDwXTuwzUJp6v+PTpY32OOB6zhWRCea4AKXmm2SIEf+bPfO86DPQhID/IBIfvjcd/sxTKJp0d5eipXth7vkihYZGkH4Qgq86bZYcOCYvLAA76J2jUW1qURruDg4Ob2Ny0rFmnuyEnVJXKeY633rLuHe+9fw/D6VTOEMmPimP5dEWmHroUJvESwwjdTbKECevG6shlCR0WeBDuDXHn+gHqVZaHsSuKIZQS2loD/fNTVJtd/c6Mp9DEjGRO6+Ql8Ms9hyqhRbVihMF6Q4JBIeq1CMQ5RItLbAEPnx+jN2L2L0OvP0atxsaP1EQ5dvc25S8R++cznRybk8vhoWzuVr9Qk9lUIR9RV/EVXBbUd07h55i7oWPJTclq65jAnBpjsUmWo+qRUPN93/99FziAC9R9FtCzZdd3c+kX2rKLQcSMObmLwXQqa9xWSyMxWhWQxbTj8Vi7ntAyw0DWIVRde9QvfvGLxjRuGL79j3/5H+lnVGbDyYZsPk0Hx0QCQd7+CrGaWdAxDNHuGshC5I7Oj0q2S2VMHC2dThJZQ2Ty7G9vqE2MRm6VQNYvIUC10ZL3TLu8ZFo6YK0dNU5ZjatF/VbegRqD5tK32CnQahIJDdBpm0/UPz/Go8dPrS3eaK6wkcLKeyEsTA7h6emZhdOJEXA5iCso4RUS6rU5J0OIzjMdVoPAXdu6ZCnHUeQeRxSlgHK+a3XSyFgWXxI4JOFirucH/twPSAD4IB4jtkUamY1wy+rVB/9GhzFiw8T5SE4Uk36qPbtoIiI2rkeuWEEzn+di+/oxmjKsKisPwOJRIlscf/K//JOqtj0+PMTO3h7+1e/8ti32ZI6NrV0JE3vqchB8MkxiqXQ0dwIXjg9Lk0IVay1iYk0SGz8J76BHPR6Lkr670UWr4aqiWVNMFk/JtIUnimjrqRVoaAxcn5ZWcyir9TO2kpUacVfXKnVpsHrDh1wpNtq2I/ls9z98pIXhHHziU5/B6ekpTs966l3A+/XDN++k9mm1jGBD2riIoCxfC62zi/IdhZWM27DtSMElV4Kw9LVrB0JjT52GUV/I7/+zFxqADg1hQj9Y+OB3tee3+78xA8ZBmpfDWpG7nv983STWIoednR0sFmNV8XAwNTlfWuLiyeOH+NjHPoEdOjv0+l95RcmTN77+VWHqPrl01usLG+BgBpAIGfMGymqRK6++ehZ5UPNwYsjuoVPgkGC1huPEpDmre9gNNBVHoenQNra5K5MxnEwUknqauRJdDn6W4+0EwHiLJLW6yV4y5Aq0OcghIHk0cjmNrY0mdrdot+cYDA7x/PBUVcVUy432vkK70cT6C/jwWULOUM217+H6WH3iuRJ3FHJqS9YqcHjOAufAV3mzBpNCRoHn81I4Hzy4Z+tAxPTP/aAJgDlCFzl9TbQ6SpqFaLrF4+88J8C/ld4ymzczc3Z51Bu1NbxLShjxaF8n9+jR+5jPrUUKnb4rV67ihRes4JPECh4U8fab1up0b8+iik53UwjZ08ePFSrqnhXOhIgZQ2vxTRaNBm3Fpb4SifF1ucoIoCbBogBxgbvbO0rocNAk0EdgVTDjdfb2ta5k1tVDxRuXWFNeE3CKVOqWWNMHCh8dLQ42yyJuQE99Z6eG2YxU8gXOT4lq8nwCYJGU5YAzZifC6jWxtaovafcyxCaLl2ZFDSnSBLTCNAvs1qrVc2ly/q7mVq5bOXkdNDGMNGh2JxPzSxg2Bv/NDxkpVBJxqREk/12ooMP+5m2yB018fQBRQLJS0oQCcCFAVrxgqox8N0m2O2ziu77w3dYYgrAtGbeXqMjf+OZXtTuJYHHUXLULqVxXrx4o9DlRuzVgOhnLXk5n5LtZToL2mli7bUoyjJ3WYMa+ylay9OqbCMRGstNEaO70/EIyI53+QTbtp1++I/tu5e62y7i7Pdy6TkQFtNWuCpk4QamqCmpO3crlQ5Jkju7WvgSsUoqwmJ0qO0kBPTrtSQOQGVSLAtl3P4TeiYpXX/s+M5FREiA08gxT0FlKu3/hhHtEUNlTts3V34xBrNpBN3/B9//A9xRE+TzjdX1l2o8J25PYol69dgO8sB9qpOR7DLPhkfOG2WjJJtMKKTnUj6/EXrUWVpIfEKGiXfHqq592WSq7zje+8TVFBBsb24oGHj9+ZDc7nqDZ3NAO+OD+e3qNeQG2fa1UmSRp6vvYwla5oYJxPu219fihXa2WrXaQpexs+07tQaeI3znss8zKzgtI2V4mA5qxUdoIwNy+dUfmg6gb/RuJl+v1aqQ0lnbxZ6huZUxy8euY8eRglECT5SuWSewU/T1P8fTxO8rD9M7Z/dzYQByW7zfzQsH3HUYWcwr+QqVj9YY1zeBY72TXl1D3mHNtqMks22LEGltFMpiD7/3T/5U+p35zl8rD+drBwVVJ2HQyxe6e66fLCt6TUzR52IBq540yTZaN5+9xoR9+eH9tG7tORflk0PHpQ+x2r8rz//wXvksl5YSDOd57902FKWTqcFzdNyCIanFjY0uL9VxpWyiNy1NEmMHjIisrKLVb0kTrUAedzGFmgUgnwSKRT6jSRVzdw747aIFgF505VvDQm75287Zo52zhVpazG+OFuy+g2TJt4J9XNBYxbFLXfYyEGfokobSN3htYUQy1yYxHv/jVornKyeszviJv1vMyuWSe20f17buuTacLi0pgDTGsfxGE+VOAOJY6p4AhO814vm4Qwdd8hEEcIfiRH/1B3QrDjetXrfTIjy4bQzlxkSDMLG/OcdEE2eJQqnSX/LMLp5nUIAfBGtt5Fn6QT7DZ5YFMEa5c28fTJ4/w7W9/S3+bzUeCV30J8/r8gjzEtWt3FTqNHGTNhaSCoqNj5eluVp0zyHtkLGzRAKndFfy7r/yuBE/Cov4GpfUJHex3ZNk1+x42t6Zg8Uwi+geKMGq1ddxNuNgKTayMfN1LWVCzzRPvTT8jtpK3Q64M87+oiF4n3R35xheesOmU1za+dpLfRQSU/99s+bJ+S04xJ+LH4VNLQpGdbHPIG3J0c1d7OBoMEPyNn/+bulXa2C2XBfRfwtJkn4+/rF60wA5KFRBBqZ4vFG74oSagziWYTldYLafqkevH5z77eZmI+/fvqQHV4NxgzCDMxMPzBRXKwOmCViRCNba/Z6lpFlcQmKnX6lKnrE+gxiRFSj6IyCBWaaRkCzF9HSlj3jbnhI2j1GX0EgIqOhjTsqznD0PEIe2+LZjKqx03cbacymcwRlQJZTKRfAcw39TSN8IgVuDmzAo/AjxztC1qC3EYHZfRTIsNhoocrBtkZREH/YmUfRt5StYlQshsbu/lcNiQCmWVTg5DcMF55cRlFQeDcwR/7x/8PfsWUaYuFpAvLSZ8QPtChlZ+BAEPSnAdGbjQBGOoAS7dOJMmHmE7OjrThBL58uPq1etyTP7d738Fuzub6LRbSgkza2eInb2zcA0qkyVbqphEffwl6+9frTJtHKtwUjCUD89oPkzvG+fETe6cAk0wx6VePYDj1ayaJpEwoS6j1jzTmzmZJPY4mE91WgoH43FvXoSy5VY+ZsJl90qBMeIqTWQkHIE+yIP770modMwdu5yqdYw99OVWOt5jZ8MtdU9HJiCXTbSKBbusEf42o75kEasbO7tmspmfIS2eaXcP8TO0ni4tpAz+/t83AeA/eq4Umf+uN9ku/aI8nKrvUpra1Kh3AuXhk2hxIbn8N+NcTdSETBUeGGHhCsfXv/U6ru5fk9Qzr+DRrPPzI1t0d3CFgxJATIJt37iYbQeIbO/sS8iqFevG7WntOm7NNUlQriJZyQHjTpBTpdu0ejq5gm4C/WEWvkcPJ5ZRBX0aK4ylvbbQkrg7ibOsn1jOLVmU5Fa/yGf1+Ik0A88pyANVPjNyYYZCUQMLRdgkulLV5uF38xorVxjC1+YOXudRe+tN6L58MrDm3QyxOVhz6IdHcxlNsSyMgBnvg4MREIcoYf/wl3/ZpoM96S41heJrQqTUBKontO7yIBed0mvZs4VUle9px/dx8fk5jk67qwKSnUuVx1/75tc1Ua12R+XUPtnEmyV0o758aj3viiszNmMkLYy5fXtNE61WbE3TQI7BnCLGwfWPY3v7Kra2toUbULCsqZVBt97p8kJM7eEJLq52fF2yxlO3yFriYhHD7zQNph6cP0Ovd4InT99DiYWaSzqDZt/tzIUC46nNQRTlaLU3RChRGXce8/wZKS1GFuosombYF233yYyik8thfY9t5SXsWaYqav3NF9FcAuO9UBD2NYo++Y3mBnhDLA3w87/4SwYFuwMW/CJbN40LcMeXGumCIjaywJOdrNjMkGnUj2oAYgOEdCVILhQia8ePWTqXfW2yKWIYYeA4bDsbm4amuYclcqYJZPFlwni9WPce8nZcMKhwANJcCQy1cOvWx7C1uSfhmE5H6LOFvcrC3H9q5KjOUmsWjU75cK3v+N0M7YgtcLezGyoFvtvprtu5v/Xm78seLxYjLMjBL2xXm9ZhGJqj7/r12Vl/9GGIFMbarWpWQi6fK3WjE6muKYv5mpvpzWqTTTncWJIWF4UYXaLaUYgUDTjvc+GIvR4PEIztwnCP1iiX8bd/6RfW3V0SdyrG+koePNBONKm/GC4+pZPEB3MJn7UAMWxyIAwzaQZhXtCdmdlTNrDRwGp50cKMZc86EtVZE9o820FW16+QKzdnx1rDsQ6eHUmoN6zTd3fjKq5du4XtrV2RUGgDSdOWwLhkj2kP0whrIqurIxCCqMwfmbQ6nRXlckP3y7Ksxw8tQ/nk8TfXpiQvYsSO7sabr/BADJlIE/o0ZVtaO8KWp5CsVqyz8E2b2VmFBaz20GwVs94org2sDr/yVC5nsob05nWIxUXafv1B54Pwb+qT6FP88p3MVCh0/sW/a1GAYdBmSyYz+8J43cbhO1AGRzc2lWmTTpvmGSf8LMEetojV95TIh8/w+JGBOlo8rQXDKpIobNfovSoMvaDFFe64NpI0s8LlHZwzQpsvx4atz0EHjLF6Gbdvfxrtdlflzx8++NAhjdyBruj0kjNji+33hHXQtIbXLq5X+9dArCIjzvbx7tv/QVh/kY/XTp86outrXL2++8owsMWsVjviVzKJMxycYzEfGkijPknsH1RGs72pOH5r+8JX8qgj8X/P9FVlMzuROZKorxe4nI71GT9WcBFXkPbjpiRm48vuONe/8Et/w+21CwHwiyRs2f2VHPfLY9Tvu1iWWpPskqoYsH4wUdFx5eH8CnrwcOhzxwAAIABJREFUR0fW8kwL7dA08tUuNQkToERnRb39tRu8KmCrNmPJe4GnylO+vmBoVuHBwWoQ9fInv1udTbiYxyeHdm6g0qreNbOFVsjmDnbmtThByiO4knWSTBiiNRstVKts3Z7h4YdvYTo0ICom+CXqm9UNKpKi4DDL5lbDCxfPLFzMzXZT46k9nfgGduYC5yeK62LtdC+du+DzJzyIkkU2HDwHgDY+KllfwnVRjS8TF57i6WGua+ml1rT+V0UmP/+3/vradffp3DS3eHO6cEelOQz8sgBEjkZFwGc6ngpW9ACGFo6EEMfysd63dlL4WrjgWq4kzFE31ockVytN5R082uYzo+PpmZkUoY8XhziKiLKaWOVNyGqjGj792p9Q+1XuONp/qjo2qWT2S4M8Tyd1Fptf5DB4n3wOtc1dpsq07e3todc7Fjr5+OFbCHnyiDSkC/s8QdUJlPrxuAJW7nIO7lc6yckqwYRNHwp65IYvTBcssCnpLGbWYXpcQkLp9hTpeF5+GfqRrjcYufOWnRN4mdNJk8HBaIg4iEUYAKMqnqvIwYRd8Ff/sh0Zw3E5KWNPeNF90Xf/8O+t8wg0d2SanYVrhRB+MEQ6PTFk6vjomeJqtjnz4/qNO9oldjPrNkWYzAZIMxalmLB4ExDz7D6njhKvFNyx6XFUQb3WRaXcVPbt1sGnlPEic0aTyC5hEdE64wFql8rhskIWz6ghpZxXZMxMh3Bza19pYR6nc//+G8ZinpytNa0CSXVmsVBxOSNFy+y8V8u+mQSdYmoUHezIvn/sD+Tp3CtiBTWdH0hfik0o/Mhz04TMFfjzi2kulN8IU817v2+FuwYP+2EmlfdCOFwRhosYWJXEwTxI8D/+tb/0nwiAdzYmi+G6N1+7fZEL4Ifb7S15vDYB9I43PtIAgilLj2KdnBxaezLXO5ifV/NIVRURz6YqdxVJCTN7XADDEBJ3rDyJlgyLKKY+6yWaF+HeUh3t1h4adSaLImx0DjCiAEwnRmpx0PDlglG/641T5yfLBETnBaLA9Ws35eVPJme4/+HrJjwrS9lqBBbyUbDI9GEvYzs42k4e4fCdOlRYK5fJqpJ9a1zeB08FV5ErmU7scC6o10bJmcL5bIFqyYgrKnrJC+x0rFmUbxQ5mlw42eJpKGQ0TUiugBjQrDJumUMvR/h//98sDKRKnLh6Mn/xacLGxSYfpdgfYGh/3dm21KzSpaWa0Kx1P3yXC/CIFjts2wUvCsHZLIGDzh0XnELAsUqYgeShUs5DdidocW96Rc0qW92TSwDVKm1sb95Ap00NQ5tcV3qViJs/soXvJxGFC8QYnbi8+vi5LqL6ux0pKqGh2uy2u8pNHJ18iONDHoRNahYbK9gBW8PxuSaXqr13PtIhFL4zubxcR4zR/GZ02aw61yhyhChtLnnwZZUt5XiqCP2IS3m9XdcZ/ej4HGcnvhUMAboE7QrPILQ2/xysnPLD0+YYwrLZRKvRuei7nLrvYfvYf/TLf8dFAbkweY6qO/wx4bErTj/4flD+Ajs7bIdC1VpCs9W1fPylUMPAEJ93JmJF5OyCt76z7QpMIqpM4gj23smsp0OlMmc7ufO1OJwYZ6tzrxUCy+nXqhvY27mDbmdX93HeG8rZIhoZl91RcwrJ3G7VgZAkVaSu05cDm6z2VM5rs06C6CP1CDw7eyJBVRqGBA9HvBiO+2CrfaKMx8c9DPoTeefK+jmtctHD30TYled9hD9Jf4kp87jGdDOQxRe+Uly2bCITZNbjmOzqkc5IrEc1jIcL1F3DhxwXuYCm2+Vs/7+cJ4K/vVbiSai2+SoIfvkf/0LBo9bFB3AL5hfZM2VsZ36ULdRu72rymULe2mQHy0vZONuH61y2ERVz0ZL8UEc1ddygB0wBsB2f5XOEMXeKkzyvNdRh1F7zWTe7RoRWcx/d9jU0G1tSieLkkxGsnWGNJXRsqrPZUv8+1GQfRH8SKDuGMbfAVuylGIdPXrfTwlUgYn19mYf3rFpuDtKsqAXOzoZ49OC5tbiXqTJyCuv19Fz0C8hnpNZUCxvfvJ61/wvUmxVU2zz5lC1iLtnykv88f7oq63Sp8vlO3ZjTYzaezBbY27kAi+KYhJsIbOAxm/PoOSvrYwTME9U51I3nH/3Tv32BA1wSAGoBZrc8IYFhy+URRcavowPYqHcRcDd+pNeHVedy0AGi9zNfXNgo1uNZeLgS8FMwxnckC3OuTODWrBuhZr7LmDtZk55zVMP1q5+x4o7CDoYm/Uk0LmX/jENIIMnT3hl6yna7Uixvfti+lbzFRjXCdNpD79xXIZuvwmGHT5sgklsoejp9gNUK33r9scu62kEapMszu8mR5BNsdHl2X1W5jE5rA+PBSE5fp1NHERSYJVMJV4+NMF2s7v0dZv/YQYyDmAud2ig0/oGHi9uXrDSbXHDMZhlm0xTNGhtYslFViC1/jkMaIPgnv/rzBgVrgj+6yCVVoDjb+JHFpa/imyBb4oum4IIg4TJiLipYI14XQYL1zEeBqePNf0S6Lv/Df+bi0DAEzl+oVDqolBq4dfO7xKljStovkna98H1LlnlB4tf5RCF/4b/Zc5grR0KIehItehhPzjGfWtm47WA3NyoudQwcdRG5MCsnhzOxlEbDMUYjHjebo7vdlo3ubNSVDPPZRf7NkOsA9QpPVskwT9ghPcOjIy94wDxz5VwFFz5HKWeiyxpylN0zsutXb3YG5+jrfqslc/TyYIHlIkO72dXRMZwT1jJKgEmY+pVf/wWXG7PdeHkYdu/BEx6GfAE5Xs4MfuRIUf8FAmicyna+gf+3bkxCY8DL5U7jnBUugn9tHbrru7w02AI0atuCaK9deUXH1pBAwQkuqWrJ3/dl8MfjGkYH43utt5F93/ZGW2ydZ8/eUhMHS+naNT1f0h9dS1+DptPyEEykzXByfu7OISKd3rSnP70rD5Zi5ajGYDFXJGGHcgDdBvsMMA3NZtMhxmuMH3j3vp3zm+TEMgLkbHNPM8JuZq43smdyex6ltERsAkAcZDobYXfnijW7LFjA4zYKQbBf+fVfvNiXlys7JCHutMLv3J5yli/5BP683O/4vBcSThIbRV4mrXlgwkOnvIQcpo9I1rry2YVbds3CnSe8u/OSBKAoyvJ06YdYUsuo4CZoliS67KBa+EiNFWA+m2B7y6qSJtNztY7v9x/Y+YmZEVc52ev+SJc6Wqnzpm6IajjBs5NnAsTYurZcqsnOb2wY9jFbDqwQw4lbRX2DrcR+NhxJgzKtTaHhARJ+nPRMC7FUTSRXnqymnonAdq2Jzua2Dqvm8ORW/n7oaHN0UKnhdnavyPwIkCM8nBgpJvgn/4dpABuXftU/L/7tAY3/7/fauYNezfI9Pvbm7zym9DuH90h9CtbDmf6S3/mTjpo/ci6KTLr3915CqVRTzz3Ldfr2swbrCvp1jSTpHEo78KBFLxDsRKqqIdNsDx6+IVvPxhPqdSAkjmEu2UQWxzM8JQtZi5KwmaSZSGLv7z5807qKF7H1AnbOJt/Lotp2izxKI6gyZPSap8i46HMdDk2r6ZthlwkKlc2wWwrc5tH3Suo6Vb4uArnU6pd5hSKxHsuEzHf3r6u8nyI7cfWGdFaDX/mNCwH4js23bubAi1JVEo/+/xuXF5zvYSeMdfMi98VeQOg4OmTSnZ1zYS4M27+QPQIsa9F08lirWruanZ0XUIqrOO/xvACiY75wxOXIGSO7rqNK7ToB1y5YGbd+e3Mbg4ElqQ6PLMtng+EcW84ZtSwpbLeuMh73brtyMJhoZ2ZJgNl0ieGEhzlzl7NYlb0FLrALHuFKB5MpbWYFdzpXjTGlHs0jTBdz0cJ9OtjfRdkVpFTYjte36neC0GnvyrEdje1+Rpc4mx+7yT5GdugU8wLkMayLQ52vp16Gv/LrFgVoXEoY8J9sv85dEPm2ru5tOgnrkrq3hbYXLjeJuDDZlvhmyMbBGNbvVvvKi1tQJyu1dbPXvFBawYXrktWymvmNLkmbZfT6PTVmpHfM66Q5u3zZ7r/QSmzlYUCPpXrtOLb93S7ev/cflKVLVgxT3ed8nQCnpRwi1RF1BRbpFLOlAS6Pn7HXL4+n55kFPCuQCSOWqBOitSdLlgbrrpbsPEqqWYqovEJ3k7WLPI+YZyS4buhFiOV4hdwJMtd5srAiGDpwuQvFy4SNwxirlCn1iI1ubLGXQ1RcKrndMNPDZtb03cTedgLEjifEM2iCgl/9zb9VRJGRHzyD1cuDdZ3yi0Pc/nJB6IUEsCKG/31EKvyXyJKY6vLhE4sY1gnYizSA3bDOwiXD2LSNL4xgxo99ALg+Nw++W39TQYoAH8bYrgmS1KUPX51guhPFGO6rodVypa6bpKJ9+OB1TGfuPL7vKIwpsbI4onljTyE7ypUaYLqy/r2DgeqSzBFMjXlP4U0LJn6mGA8MbZQgMzwlHVwFnTGaza01DY44iTqHpuxHzGYPFyaTXEiOyWyCeslYWZY6D4F6inySqzuZhOQSZ8MfN2fhLs8ysuN/GvXORcNMasdf+xdmAnSalqveXa9dZDw4PcMaEXSi/RGzzon2nvclL8l9ERMWBjQ5UCOIVB3jU69RWJZTxiNdeVYoZytY+w2264ucBz2Zk3T37h/Xa9Qk2nFsMC22j2kmCsOFD2HCzc/pGDYxcugEGq/+/XtfEfQsAXWniNippSHqLYbBPFRziflqLOEZz0Y4GVj5Nql7XGBefz4DNjdJw24KFpfPoH4+hn2EqKu+QieZKmFEM2kLWSQRVimZvgus5kwkXeQCSKmvtys4Px0LuOFg3C+WV418/xDNhm1MCrUfzbolfIaTMYhYsq3tbJBonrY67C5Gp7uG4J//luUClDa8dHAkX1vogGhb+csp05QHIjmBCAqrnct9TtzfwZrkaNkvLgxP6tVkKO/OxIvF1wE9cpcA4YQL83N26iLDWKBR29Rxcdub1qaNGLz37n0Juqd4mS9h/oR0lY5kYYNqnlReQb9Hj32AGWP9wlfU8mfJzkngwRalnvyeIFwgiFYS4nlC+pchmqPZsVQ9Wyyzh/bt6zVpuSKor7OYhjGQGMuevRaHi0pOkErODhc00rFwzDtwNGqXED3nAySLBKuUVLNUqW71LqBfRlaVq7quV0pqJc/h8wOMAujvrAG1gEWk9rysgwh+61//fQOC1GyoDPbJ8YM3v47H1xGBqf4Al8qQCYiwweKlXICoSl79CYdnNHAh2UpTuhiKfICySw5xwol6JSntuBEsOGjjd7fZbZwIWNsOP/AFmE79r0kYrrGDjsPlbnNFG+UyS7YTLOanGPSfYz4bmlA6AaBDSZMiciurjGM6fmQMJeIy8F4ePn2AqGJz9NInWnqNbfXJkWQWlZ9fJokoYElGr95AF2ZOK+wXrKqnlUMRXevW0rbyF0dnz+SxV0vWpk+LJCYzG1lYA+7lkqlswzPYj8gaZZiQ8dxCW3y2hLFIiXQ9chENGOOMFlisXN1lpYrgX/z2P1x7YBdtIcn1TxHmF+r8cq5fztz6U+zhT2oT7d0lZ841kPQ35Dnw/LftTEPXae9rtTYqZcc6dg9EKhNjby+A5UoL165+Ug/kDtR2LF/X6sSkUsM6YtrvnCwuCgkebJPOnvuPn76Bgo0niwSsaGdUwsFsJjUSj1jTAy6t1S3z7UQZqcnOTs+QxyYAL39uB7PZEr3BBI3GNnau1ZFlS5BQ4/2lhTuSbiEnkMfaMarIkYG70oQ7TdkriGf/eO6+3Q+7mJMXwbkiRELoV48ZGEE1SWIsF3ZKmaqOjPJo1wiJ49hZSlbenmOry96LPLPRCChJOkPwa79tGsAvjN2Q3VglYuWuzeplARD8udYItsPo+V5G+izV4+yv+w7/9yy1pATfQ+m1tmzeftFOW89BA+ns9lrtPQmAjkulwVWMbk6eTIq6itsB1owg1P4tiMAFYKRQ4vfnY3U5Pzl9hkSNn3jK53Ld0JI0betlNMJkPMFwyGPk2H1rrOPodU1OaJxgNR/gk6+8KE+ebWHG4wUqbdLS+Pclak36NQGOexYxpEkdUWDnEPAIuSzvK/9RhDVpO9p/Zh6pIeKQGUCz60kyQ5qx9pIOimNIqzkUTU9TZNg8dTz/S7V5zEraWrLzGZtHcNebE8FT4QMejsF8yT/7zb+7FoALdI51AgWq7oBnE44LIbnIyzkyqTMVl3Ajk0QvPO4G1yeLig/PDtvcnUvnA5jUW07B6OZGs7MLb+/dxO4uK3RpFmwneHSYBZcilrgXJMAChqx9PLUG7d352QfygKeTgfHz3Rm/7pgETCfs5p26E8oSjHSiBxM9jABMKVA7uAhVJFPiHTwRtNEo46WXr1oX0nxl5edhgMncRQzjPqZTE3zWFeTZOdtRo9nsqPkTs6AJG1UWrHO8zPK1RWu5bqYBuipRJyC1LGZIVmwDZ2tjjCZGR+yJzGIUCpExgtgPyfockIxjm22+nBAIMj7AZZvtuWdcjMuDb/ROl69i4edIvWInrouu1aaW195sxviYjqM7xWKZYZXZZFNlUhrpU1Bdk0dINdx2hNJWx3bCwcFrOqKda+zrFYwkZnV8XiBMA9Ams1n0SiXhZBfpcIj+QyOFpFbKpjrB2XLd/YMEFk7WbGGspOF4aiGmziY2sgbNycmp8SYqFTaFIuO3wP7eFsISkRNmN5do+NRcQMp6gdN+H8mSuznFYjXU6SBWiZQjDf8AtcrHUCnxCF42u7wg4DbrjkYncIw+GVW4JXLG80Tf0W3T3ofIUp5l4JpFOOYsUc1VMkSzdsXRyNhmnjUCS/kUwT/7jb9TXMbsLy+4j8H5mmJdtX81h0Nnzylta71yi8zO2vXS6Pv3STIlxG5XK0NVw9tvfl0TSs+cJ2DxGPnd/U1sbvI4k5p6+3BQXUkArr+MyZQ18RdHowl9lEagzbfIRGANzztKeCLYCjsbW5hOBxiOjpEuhhIWDwSRLTMeWwsW7YiVOVWLFUOyxI6GkU3NEUTWBpcNK4xqxraz7B/AQyUyvPryd+Pp8bexsRGhCBKE9BMCYDK3kHGVRDjt0+9gNRW9eWtzK75CfIJKeYJaxR3pSjPiFqIUb8gsRdEW5nMTvCxn/qOC3nAi8m3HpYHT1LWI17vsxeWKPZRSkUH9ejLk9Bsm+NV/7jQAX/kOLJiT6hs+LxxJkxx5vm4xvQlAmrL3XYyouGhyHCh3z+IQdtasyONngwkiYXoI1+5VHbV1Fm6MrZ2GFo7RiG/B3m7bYZTN5r7aqBEa9iXTFlKZcMoW+lMyuT2kbXK1ZZ1M+ugPDtnyxBFLvXMb6IiXnqOsnfaolgPx9EX7qvG+SZcnRY1snhCd9iZOXCv7Zq2DdnsD3c4Wdrb38PZ7X9GJoOVahnojUpfxzobDMUJ2NONBnIY33Hv0nlhQXJTtzgEG4yeoVngETQnTmflgtSp9gz7KpfcQBn8CKCz8jMObaNUPsMpGdkxtwBPDY1Rr/Dw7kNfEJp5M+1qbwXiERu2KY15zrsyH0qkpXgDMYeMkXnjyVJNSe+Q2OPVdLplkmZ2yWnqeqRMHRNYuVxCzvMs8c9HBdcLlEts7trNJpvBFJTF7+Tr6M73eOKIatJ1/7ZpVAo/GKyuhIqTsIGvaWFXfOLarVeJwt6VytFjlOxo9kcOXpgzJrOLIOqzaczJ79/zIcgGn5wRtSpgvrQ9PrcTDJVj/X8XuNnsnBOrH88F98gNZwmWt2pjg2ejuYrnoI8lmWKUj3Li5idlyhStXytJi5VqA3uAYtXoHkykPrbIyMo7JnCqdTZyshT4PsLQdSsq7YQN0Yv2RDUXGE7+2gIj8AdYtWi7Ak6dajTtrTMd6JXGDso+iFepMXM5APQ8up4Nlpy+Bv2KaOGcpdocilKI6quWuJVlUNcO41zpc0oNdJxxIRc6ncuhIq+ZQG1W30J7zRz+Dx677Iao2+fYFvfgYt29/QX86YQs1UbZcYadnC0lr+TN47eYJ5PBQRx4mee/eV9Cu8cTuCpYrVuMQZyDn0PyJp4fP8eihETBYWSQBWpDmRR+AnTlZbUSApSNwhcfjHh491fv39/bRqDV0aBPTr4enR0DIAtYhbt+9pQjB74lyNcez4zdVKpbnBvfqsUnRarE8nAvOWokc8+UFEWU89gzeMuqOF87wjSDQdPmvEUd3sNGxxI8O6aAQJjPUq9ZDgfgEKWu1KiuZ/OY2AatWNskIuuADeO+WZ9FykGfvrVGncUELp6RGChEvqm1V5uTKuPThgOfUm6rxloUYPAdVdn19oqYd577OLGhXkDdn5oRsH46z/pkjZbC/r0MUGQuziEOgEO+ZajpQkyfujDde/x3Uq20hX9VaGTduvKwWtp1uB+e9M1XXvv3eN9Ykz6Pj51oU7hj+ZK99OrviBFK1sjS93dXh1RyddgdbWzvY37um7Obv/t4/VWu8UiXCSy9dkQDElSEPwEGSjzBb8HMxULA8S0Gwvmfy/xT3pbGN32d6D0VS4k3xEiXqPkajkeYee8bjseMztuNkN0nbBRZF87Uf+qVAsUBRYPt10Xabtgm2RYvudreOjyTrpLkTx7c9E0/mtObUNaLuizcpkiIpksXzvv8/NU6b9EDR/IyBNRpKPH7XezzHXhlhf1jgbQyK2zv2kSmqTyOaGtS1WWj0xAITrw1S8+3oDz0ix/rC+lvyGLuUlnVY2liTUMh+uVyR1L5mEFpIMtHRgOWvXz/oBvLqNK3eGPk62wIHVm3GLjZfNHnxeoQzDXIDFrVQQ2sRsF5tsGKMnFTKsgJjboPD6EzpVXKQP5Bj12btgM+nLd/OgMrFb20uo2HI1onuP4Cl5XtSPmWfm4UkVvJEe3B1EX19fQL0IJ4+HI6iqyum+gIEfXq8CgtvNnDj5rUWhW361lVDqbMslG0GiOrxw6KN2uLkC7tC/CyWyjhx/KQIOzhdThQKaaxvLqrBRLWAgcGQLAaxmm+jKaQbtWYe+xUyg9nAYodRPzMaXPO0zeX24XJ3wFB4k/dYr+tiKxQrcNgC2GuU4XL1ot3uwZnxl5BIraO7awgbO4stggsfPz33t/JzvE5Z9GHLmAG3zJDx/1q9AMtfvfbvmlqls6j+n0TtulM7CLwwtia5+ebgxLZJ4UalVXmfccEwfTQh22pQpEeOmVaKno5xfzOgYpdKihH8YzyR0LscHnT61TGkZBBVmbbMzN2Wrhn1fr740pdx6ZO3sb2zCpfDKdBo7m6OYCiKcLhXNHO2tuM4cvgUJifVeUSDuiYWF+dFmGptc1Hq8lyUCw/uSnzACWf84/cFJKOgsAOzG8EDVBhw6c6NxWKaAYnFbQO1SgPVfTJ/adBQE5cyfi5+PxlLNaTTvIKYgtlweNJnRF2Eb9WQTJWwtk75N0rkHzCoRMBaACVl2K0KEfd7euV0PjL2JBztbgz06RVgjrn4NOaXb2G3lJRAs96sorCbEaGMXCmOZkM/JyHF/OWr/6ZpSpOa8u6mAJTNenA3t9gwxrOQgiVRpFGRU58+bfyYVUPl4isil8Pk/IswgSB27PC4nIa8i/7iKqXa607MzagUnM2mr+HkqcdbAeqn0x/J99rbFf/XHR0URk2nn1ZtNsR6R8T0kpZ2fn8YoWAAXV0HHyp/luYVpb0yLl/+EN2GnMrlKx9IddDsa7C6xx1QKhVQNDwK2pqUdtM6h9/HngQ3AeVnayju8WqkGxlhWxqdcwtoOdiGfG7PUOluQ7SH2kRWdAY6kExWxIiSNvIMuRxWmkwCTi91jyl9a8VAX0yuSgJDuQmqlQZ8rmEpcgUCUbzw7B9hJ7Pe4lh+cv1nwkje28vJ0Z/b3VKF1r0SnB16nVNmzvI3b3xTTgBeGAQNmqNBAqL8Z5YAVQLVHHpVaEitFSad5IdFIEzJFZPTJyIJdq3KNdGBcHgIJ0+exfbWpvjlcHdSdv7GtUtSHuVhQQ8+DrphUHuYkX+2oKobIuIk6p02HBo7hVhsAHfuXhP5GnIX6Xh+ZOKE0KLYOmatnWVjiQkcTDVtmJ6+LkURjsu//kDS0L1qQSaRgSirZkw9d7ZJWFFDZxPKNTgwhGBnQH4PDaa2UxnZBG1tKoXPxU9WdbVWwW5eQSJCexNXMha49C5u2OlFRP8Es4BpVvTqsHfo11zAJoXe662K4HSzQm1EByanpuDzuxAKq+Mrx1ZKncdY969WCRxh3ECGNPsGddSqPG3qsPyXV5QezjvPvN/NX6LrV/H97KFzHMjD6aP4fVUE1ZxPQJnmmS/2rXyU9gQoTJBIaBFicuqs7CKmjokdegYpkXRzY0ZKpJxYys33908Kz//ar68gl+OR1kAgokehw0np1w709Q1JRW98/KyIOka7QlI0oglzV1gFls3BTII7jc5gC/MzWF+PY3VdtfXp7sVqIC1f1KqYun7UQeSO1vfP6L2Yr8HT6YDL4UY4GBMwayq9hWQ+YcQ4ak/L95BKMXcnXoBtWVOJnNkGj3Ylp5YrZbS3+8RZlZtF2dm6OJwurYTydN4t1KVQZhfKfB1BLxeFTXiaHI+dV6AMA0kaYt1fuKZi1jYrcoWciE0LLN3YxxIDvPb6Nx7q6xkUYgPzvyfdOFKKVQadTQVzHGjoUmWcOcwBQoj1ci6UxUXDtGCfrp82dHerDiE9hMbHpiS+2Fhblb481S85bPa6QLsoDMmAbmTkOG7fvohimZMWkC1CXACH7jQnHnnkWYRCgVYw9/CE53bZbGoTP956rSjCThRLpvsGQRLFYqaFqeO9X6m2IZtLy6Tm8klJ/XisB0OaldjbLS0ihsftQ7WqLV1bmxckBDEOCYeiuDF9VRbTtiGAyQ4qyxg8skmM3d3NtQAynEwr3chYbWwjjy8ip1K5UhS5PBs8sDlY5u1AJruPcJDOam0YH+kTe7r+3mF0BvxYWlpvKY3/wZe/qpukwyV/rk5/gGvT7xtIZT3jgAkMAAAgAElEQVRV65Y9WN5806wEMmhT7Vn5cNlIYd/Z+DRJxuCdboJGmN9z5PKKxWd+LiJGzSYWH7ABwuJOm4gyMxfl8eX1dqKvd0R2NHceRzazKneoeWiQDl6vqQMpR3dMIc/+zjB8XPEWRueKyj1+8lH4vD6RqefOptTc7xo3b1wXAgnrEVQ6PXrsJGZm7mB+/i4ymW1URPmcqFy10WWTRk81i3T4ZAHY1KeAgzUAxjEMXIcHxxEMdCOZShl1gXXsFgtImvqHFnXu4OtUB2/azOlnnUglxHKGLWSaa/icqpzO0WiUBdDh8jTQFTGk3+plpLN78Lrtslmi4UGxk38Y0MNMKNI1hMnJKYyMjMrrFOGqZhMXf/0eSnspVPZTsHz3O4oI4jt/SOBDjhwRTxBVLgNizXw8mZHeuLYUVXErsUPK0QFujPVurnSHAVtugsodHZg8ckKeiU2a+bkb+oItwNTkGSmwcHz44c/lrjfNKjqM3U7YWHe0Bwtzd6SXzvHVr35NHEbWN7Yl0PMasum/cxW0Opv6tqenb2Ju9rZ8nUxtiDcwo2bqDgtljRg/Bnpa+Ba1U3bgOOp1O4KBkHgNMwVdWdWK4pEjx5BMJeTDJs+QI5dPCzmW9y+vQr3O9MpkH0QrmNrRpKK5OdqN07itXsLg4IR8e6/GimVNwKS8RulNwFrE6LCmzBz5vMZJ8/P3RZTij//+P0Q0GpPfz0ISx8bWGizf/bYuAB6vZQOBSvAEh8ms5dfLq1r9ku9b3RIYaSmYFOsOSSlYGjUHj0oeh3yDrOy125sC/Dg0NoWe7j5cuviu7C7W0umFawpKsVdPoQOyWTwul0jGyB+rDYVdFZxsQN/AyAi1gII4efKR1vO2vjCWNVu91PgjXzAQoHPWZzGLNH/mKcDx8aW3hRFUKG4bbVN1G6HeLyNqUtyqVQtu3dYiDfskdDuhDxHp2iUqlRAmarVL2ZuDG4UbiKdrE+oqwiszndlR0wf5PQ4pVPEoU1q3cZ0abiC8Tnkd0g2Fw+dxIdbjx+HDF6Q7eOfWdfn+o49ckP8zNTWt5Do7g3L3r65tiNzN6KjK8nNkstuw/Pyn35WPishUs9tkPoCmh4xgOcIRdfHioPJ1rGdI7nUNqthtS6NS0aIFj1iKEJjpn8sVgd8XwvHjpxCPz+Pq1fcQDAYEqNEVGRKziFBIj73bd25IkGP6DQcCCgHnlZHOPpCr5NjRp+V7k5PH5TrR108xSIIf+NwHOyiVysoOozM3B4WWmZ8fvJcGMpmEtKFff+O/SqWNWQAXLptODFIp0eZo1wDK3u7ExUsfyIkn2E02oQTXWENfX7/I0GZzWakSJpMpKTiFgkEpLRM5xYyr3d4m5WjRB2w0sbK8I5utg5qInHRDDYztdAbYamxdl4XEoXxCL2yWdpHBOzKhJ8POzgY2NlbwxIXnEI9r/PXyy1+QbunMzCyyIpVnw9TUKQSDYfj9nbC8/qohFYsmNjfTLaky/nCllhC/OQ4CFx8etJLNFRKqrp1Kyj1OcIQpQdZoaK+Ao7u7D6nUmhxL7eyD20hQpJoHWTdueTGmpg2DLopP0E2bi5KI4o8u/hz29qbIx5Pz1hXRuOCx80/Jm/jNkc1mkc8VQQvcBw/mJK4heIMFIk4qM4BfvvWWFGlIle7rV62Cu/duS8+gu1uVzG/duoa9SklEJml5o8wiN7Z29MNl78PpcAuIkz12DhpwUxvQZEZz58tnWaU1jsryqyQt4V/abCsWuYDo1qJ0NWoimGm1iaJirYbZAgd7E7xS29uYEdgQCoYxOjzymWZuLpcV3MPU1AkMj4xIx5MnzvzcPMbGtE/A12P5wQ/eEPdwthXjC4toWg8qfoxMheok6Q/TIfoG6t9tVsqzKi2aFKNGg/Yt5lRYMHF4CtvbCodiGVLfyJ4oaxEAWq5Th5jmCX6cOvk4ol2aw07fui4cN9bqOQS1KymfTdquPEIjEZ2w8+efaS2AdCaLbJaRch2pZAqnz5ySnUblDub81N3JFwoCzZ6dnVGQJ7GMjVpLLXxk9LDsKDMeWd9YQ6GQQ3x5Busbq8rFL+xiJ7kFFktLRU4ui/oEv9A70S3XDNNIltvZnjbdPyg2RXwCVVW5sEXpnERYi1VMrLkptK7BnztgYBH9y9fPz4pObjzhtDdBlFBQFkqsKyZ4R25Ac/gNOV36J/F1jR8ak3oN9YTFOJqnejgKyxvf+ctmfJVBEIHZThQrFBvQVRt0Rg1ZVQvanQqYTBpkRY87TG9uHBqfwP37d4WBQjnZySPq9El5to0NjfQpiuhzB1qRfmG3Amt7Q+71aGQQ0e6+lo0Jg5v8bhbptEqxmf6EgYAf/b1H5ISiQyg5DCdOnpYj2vTK4eNZTCIdmne5unBVZBIoLE2oVP9ATHR52cImIOT27WmcffQxfa7PAqDkeyz8JFJZ3Ll3TU6sYCCAt99/TRdniUEwuQNk6exhr1iD02OF2+WW5y/n2+BmfCCuKX5kMilFEXMhELBi1uT3avIe2MGkGPXDC0Cg882GUN9M5bGymGURPeyRa9ZvXIOdnXpa+9ydCBhCUFwAbndALHmovsYzmUCvTDqJPPUJ/uxf/+OmxaaedIW80Xs2VKsHKOZsHOOJJDlmZZx79GmBHNvEj8+Cwi7tyffg9bpE349agIyA90opheU1G/B6wrKQsrmCYeTM5gibFFb4/FEheDJr2NpaRHdPDDuJhRau3Wwbd9g74fd2yc8US5oFPPLI46Kpd+v2HYyODiPWY5Q4Gw1c/PiSmCyfPq2ZB23tunq65TUr9Etr+8lEAiHDs5ABk7RP98sK9mzX6DybLyIaCUi8s7j0QIJKngaL8TltLzeqyOU2kN+lPT2LPKw7EF5ekmaMLiSqhNEuVp0+dznRbCE2lOHEarlXTDhoWn2wEnPGnFDraF8g5SooLYJX+3bBBvIEFgcR6hAZtrhd4R50RcIIBiKIhKIy+YwnKPpturIury3A8i+++U8NvGMDfv9nI2SnhaYOWolaeBDHM099UXZNJl0QTX5+mNxx62txVKrFFvesmKdEC2vYCkDgkc3TY2N9GU63Imt386q7z+ifb5itTKuVMmpNhAL02NW7PZ3R7MNuJQtGzSazuT1JqXpiIxKgcZVTy09wAYRXV0ryAQ0MDOHdd99BJBJGdzQKWwe5BQGj2sc7sSZCUzs7CqV68snPSf5PazdG9uTpk+W7tbWNxx87JwuGf2cNgePylUvIZFNIpjZR2N1CYTerWEihk5MPQE1ljfTZY2C8JCirhlUKWyxEcRDBw8/HbmPXkdeiYTIhRB0DlU2sRYtWQRXxirivys8b9YSHYBUSfMrCr1vg9/oQjYS0/2KzYmioD6Ojh5Fl5fLr/+lPNGGibLnRTSNNi6OjzYPjxzS1yGZyQsNm+kehg2Rqy+Dl1UT6lBXLA39dYtg65KQQckmTPekcgkGPBEFsrtRrbPsSIdRlsI4IDfNgZPCkZh7SJbZgbV1zdJ/Hi05/vyB0Hhi2ZyMjR2TyWWjmdcJjUUrIxBd2dSEa7RIwhOk6ckAUBdY3NtAbi+GTX/1KDCo4Tp48LZPPI7ezUxevOX78kx/j5S98Ua6VW3fU3WRrc02aK0zpSmWigfZElczj7RDzasKxUhlFBTfqbB6FpC/PWoJA6wyOBGMe7ZwSrqVEUJlYwfHRmr4DVjtldQ2+QEWBI1zAQps3jCMYDHLQDYz2vvo77FIHUAs8K4IBH/r6uhGJkh5Wg+Wv3vh6k4RB7pjdXd0Jjz/6Ffk/TQt5V5IttLmp4EZayTNQ4aSrEnYDFpIhxbzgAE4mSBpDFEJBuyRZap2ADRdCqRwuj0T7XCxC/GQue+Y5bG9vSPMnlVqHpU2D0nBkCoFAt0T09+9fkZjh6NGzYqrMXXnhicflcexaXr9+E8PDQ+jujspRL24b5T34fbwzP3vKkegRX5zXebY0MDo6Do/X1/JPYkA3PX1HlEKZ7q1vrLcIHWIZI6QLG2I9A7g3e8MQnrTD6+4UqLfX40MiuSqkz1AoKq9vbeOBnACmOJVZ96fAhdzthjagzoFBBmmrIxTwyWlJxTBRJKcaeSaPkqEowitLWUHsf2ggWSgw2GWQapfrgfqBPT2acodCIVi+/UMlhowOnoDTYJlSoGBzawmhzkE50lh8MYGT9PR95qln8a3X/kZarowR0ikKQWpVy2w0CFbQ6JvzJCABc69ygFplfs/J5JXCShYt0Dm6wkPY2lyB06m70ixK9fQdw/bOtpwyu3n9PaFIrzR/nn/+BZ1Ag8CSzxMDqBSqaDQgpdtcNo/bt67LxBIbkEptiuYu/fUW47oAzp57XOzh6LT9m+PWrWmhbV2+fElOFQ4GphIEWq34/PMv4ubNGwKooSmEiaJmBM/BQs+9+zzN2uQKSiS25PRgrLHfUDyCyZ4y3T8VtGK8r3oFPq8GlEQ88ef2DHFvwUlSOalGPeFdyfvVpFOBssKOosiX0NSUMMMxPj4Iy/d/+mqzLzYuxgelkqYHHPlCAgM9J1QFzO3GmdNnWv9WKFXx4x+9KSkSV1R5j6IHDIyUH0hAJdMVihlxEEJNHVy77SC9GR2dkKBqY4O7g3el7nS/J4xA0IdyKQOL1Y0eo4FUKFSQy7ME2kC7oX/DKJP5+ueefEqeix8KJ2d2Zq6lPcSAi5PBChtVvikJwypjOBST64Ol3NW1JXnup595UTQP2Thiscvr7QBrCjNzsxJgmZL3uXxOM6LkmgRlXGxnH31KnpsRPHciRaKoJ8iYQj7PfFpeP6Vxs/m8oJHZN5DPp4NtZ2ItNa4yawo8cVpC0BLXMIbQzi1PgvZ2ehQHVM1Eglst2kmv0CCLsKDGRc80X9huFXYlG2KWKa4/V25eaX78qx9JFO73elra89z5f/z3/lHLAdyc/XKFKptN/OCH35OsgZp91UoO9OMxizkUIWae6vF2G7lnBeFwBOGIti3lw37yZcmHf/ij18VO3byneZQyIGFdnoMLIJliwMbIdx8TR05hZ1P/jXcphQ7GDo3C43bB6XLIkXnn9k1ZnCzKkIU3OnoEh8Yn8cnF9yQPdns64PUEpao3NKxMY46uqFYdzcEPMpvN4P79e62aBv/t0KHDoif88aV3JD3jhBPIwsVDb4JIJCrdTe4+vnaOvVIVxXJeFguzKX4+JJVy8H0xt2fpXEEzpuSOEkdkUsVOriaoHj6WCy/axU1lawlF8qozBzGMHCa8Xn631SqntgnYkXPj6//+T+WQYWDn8/hblT9+78sv/4PWAuCuN1/I1lYCy8sLePqpp5FOJfHt77yCRpNVOzPwsGJwcFIoVqNjE1LB42oe6OuXHScfSKWBt9/+Oba3ltDRbsPU0ZM4ffocrl65gs0ttWLhMHfAY+efNfoFFgwNaiHok0+uYKB/AG+//X1pIDEzUVWQOo4fP4fBIdNdLI9EKo1sKiEpGDGB9BzkAjz72PnWhzZz/7ZY2PT2atuaODwCSba2dgRjwBPr7t3rGBk6JAvvnffegMVKoooDuTytXBnhK4+SOT53KwGk4swBG0oVyuDyXifBRnmAorbaRtZwEw2j3c58vxWwtmR5CJ7VE5XcA+5oH/0p29Q7mPWSkGkh22ggmdTTnCLflLtni95kdZkCFCyGWf7tf1C1cDYwzAob/062zuef/rtSROED5h8cNINCQZ80cIi5SyZ2cHP6sggbDQ8daX2YZPfQE8Buc2FsbEJWdqfPgdt3VPZsfW1J7tRwJIrRkSF0RzUw+f73/5sUUY4e0yvHFJAk6KGvl/Qmw2DRkEzl9fGLX3xPeAAEYbIE63AEMTQ8hmi0B26PE7k827J8ff2yc7e3U8jnsuIivr9fRNFA+fYPjcni6InpSbBbqmK/WsbM3IIUYbgYB/r7kEjo0f3xxbckY/F30gx7Hy5Hl+jxUL2c6mW8onYSm4a3Mq9HIqcJnyfBJCCcw0xmUyBtvFLKDAyJQjZSRyHfGIqpEouZfMY6O3p1+ERVnKdFU9rJpLRx9Hb3t6xqOH9d4T5Juc1FxW4uG26sHlq+8R//tMlqEiPZYGeP7BDCoLgiz5/7Usuvlp4AjELNsb6xKTuCzZiNzXVpqHBXmBAutnw1VrIgEoyoLn5RXyDvXPb7+b0LTzyJTr8XDvbCG8DK6rY4cbldHYjFtEXMkUzl4PeqTRyDTZowczGUSrt4772f4tFHL2BgcEiLNQ8WBN62ub2F8+cea0mu0HCS/858fG52Glsby5LTh8N6KvUNTklMwffED9jrdmB2YRFrq3HtSNqs6O2JIb6s7tupFBtgZayuLcDeYcNg/4gsMOIKqhXaxLbLKchYgF6EDPpEUUQApyR2GIk9CaX7qo8syONdqoHoiXvQEziwrReId70BCkhpS16vEsYRJL8w+wj5dUOpQaoFEXFY61BYnFExFpm4v/jP/1xOAJYOQ8E++fDNgOzF574Gt8HRa82E8QWPRX5QlDe7c3cGPp9bApypicOyUzLZHGYXlqQA0ai3iVrVteu/av0as4Hy7LPPCWCS5o8c+/tlyctZGn3n3ffg9Sikix3D/v5ugVqVilXsJAii6BCSxvraCp544hn4TJIcBZzWVsWljHBrNoAYjRMxw51GC7r44n0kEmvYLeQQi2kcMHX8tEqwsZ1bqYmTGhE5D+IrWF6alff14otfwdy8to+HBoYkxWQa+mBpFvfufiIxEQUuiEdkFZDFHp6GLNbkczlpmlEFRZlUar9rdziwt1cQNBR39E7iQFOZP8+GGLMLBofSUSlRM0DV0kVs0tBCJsqIz8NFFiCRheV5p1ecx53EZogZV5vhycRTpAbLK9/9V00BVgrKtQt7xjHCI+Pl574mL/7Slfdw4eyz8qb5htOZnEzY/ZlZwdGdO/sEvN52ab5wMFAkPcvv0Rbsp7fuya5jq3fisJZm5+bv4djUUYQjMUHymAFMNOJHPL6Eq9euS677/HOa4q2urGJzY1WCVQZcHF3RTmQyaVy48Iw0U2gqIfiEh9TOMtkCctkcdnZ2UMiTXVRFpULvXinDyNUVieidb3e5Ee2KYnLiAGbN6+NBfFUtY9usoOHl1rZ2A4nGdbTb4HbZsby+gY8++qFE6tJSLithhZh/S5sDY6NHcOXqr3WH1ssQMVMe3yyH+3rlGG80yCqqC+eAg32PXMGI7OX1qpmVKIYaJWRWPhkTcNAk0qwpDA2MagOuqcIYxHY5nZSqc0qQyrG2HoflW2/+ufT7pATJFcnVZJQffd5ebdla7Rgd1iYPB4+we/fuS1QaCASl5ehyWeX+yhc1EOQH8/ACYLA2MzuH7mgYI8P6gfP1sVQgDZeEBn1LS4tSM6cpAp+b4AdZMLO3ceLUWek3XLz4jnyvp2dQzan8QeHZsyjEBZBOJdAV6YI/EMCHH30gV1qbrSGm0NKhs7fLhLI6lssl4JDCCfCHf/B3Wu+RX9CSpVjiUa7IYA7usERSi2KDAyMioedztiFbrOD9934oFnJkADWFSKAKn1RhUzYRq5Usn9dEOFJ0DYiqBq1u7HC52qX6RxiZyeLaK6uvEX2FTAV1Jgr8QwFvTrq5AJgBmdpK/D8zKLfTI5kYsZAEr3ARDw4STm5BJpeE5bU3/1yuABZkfARdCv1YV91g3ymZfA4aMhzIvTVQyNMNlAIJ7YjFulGVVicDIXUBE/ZRs4o7d+dx6NAYVleWMTWlkCUiZUMhr9LDa6xmZVuedontLcHtbW6w5VrDbkHTKAKPz5y9IDHARx++JZcbK2B8rsOHjsiS9YvZkkVMmYiv46I4cWoKa+trSKQzmq5VqlhaWpIrhkc6ewlMFTmeeEy7guaIr6xjaXlWTsZoVIkqnK6+bl0w+bK6cvN7Thvw4aV3sLW9imIpg3YSXC1ssBm7uaHqJfJYlw9dYV5nTsnh6UdQ2ivC0aHkGhpdHmAx7SIdKz6ERkrI185j3hSWNKWZVBpOf5ImFAwKA4Ew2Bpm846NO572hNGNDo+hpycGy1+/8WfyE1wZTKXMQXjX+MjjAjqkAeTYyDHMzikrlqM72o++3m50dnrFJJqiBYJrq+yB6ZQsAlsHCvmMPCE/bLPnT6nV9fUVWTzEzwlQ0kBBXb92TeRZ+CebWxYIFsfAwLBg7fkGVla0csc3wKrY+gqPZNYS2uU1MC4ZHDos/fNz55+S6LpQzCEcDEkVbmEhjsOHjyAYDMLveVj78DPzj5/98kfIZYvw+wk7Oyf/yFLuUL8GWOV9wuIBsw2/vr2Ora0V3Ju5LO+b8UCtYiif0OeH4s5tNglQySk0rVx5RRGLWNkrGH5GB/k8m3EM5IgwM51dGceIkysV1ehOWmCnkcDbgy5ivWbwMtuAYCAoeEqintbW1lqCoOxbWF5981/KAhC1DZoiGTq4/N7o4HlpILA92oaDDODo5CS6uw/IojwdCR9jTYBjY21FaNehsH5QLFZwcgjEMMfxk6fkeyMj4wKFchpCEL98+z3kMmkkkoolsNncIoMWirCu34AvEEE6qWxeN7Vvmw0xYqYKJiN4bgCnh/Lr/DBs+NzTL0p07XB3wOPokNTv7mwch4b7EOj0YWU7h4HoZ7kD5mv81hvfwsjwGHK5PMbHFcbOI3uoj8LLB4pnEk0bK3hufgbvf/wTZDPreuy6ulRqtlYTsgpfo7B5Qn7plVBZhR1epn7bW8uGcoeqqPN3Es8g/2/YDYVVPUWYNXjcHRoUGuLSTLmzOc20TFONplDLmgKJI7G1p7tXNgMHYXSyAMSJWtqoGn16XSF50nBoTCJHtjKPTz72mUlnyZcRJY+fQnEPS/FFmfTpG1d1cjweBDq1bdrhICc+h53tFDxuEj4s6B8+Lcf5wMAoGvtlued+/OOfoVSkNh+DowZOnHgcS/F50RLwd0YQCEXxzOfOI5HUNzBzfxrZTAK7hR1pLJmSNIPDUyjtVZDN5vDEk88LE5fri7uVeNdiFWBjjXNoRtGf3fv6t1yhhHQ6iZvT1zBsFJW0zawFpgfxeTHLoj9xNpvHp59eEzmWXG4LqeSyoJ/aLAafwObExMQZUesaHx9HOpvATmpdZOo87ojQ2FZXGFcxwq/JexEtJEnjeJ9SDYx9BF69hv1LsyrBoznshr0M/07qW7FAJrHGWJ1+l1wlNKfo6e6RAJVydZZXvqMxQBvoQnFA/GAeGgoPt2RFXnhWO4TJjGrXOtodKOQpiEwW8D7ef/cXkudbDcQuJ4THVKmYBCzcqYQ/EyGrLzedI/zZgaOTx7EYfyAfIOVWGDUHOoNwON3YLZSwua0nQe/AKJ556nNymqTS9M614/atXyGb2cb+fgn9AxMIhQgro61LFeUqF3Qd5898ljj5P5voh7+XSKYE2DnYr7hDfiLMsi9dvCj5NPX6jhhE0zt3b8uxzKrc0aMncP/eLWztLGNvbwepBEUk95FNaTwV7Q5jYuKcEEr7BwcQMswhef1dv3lNpOjW1ubkJGCWwufixLBzyuH1hVCt8ZRuIhLukRL04iKBMxYJHind0xnQJlahwOugIcZZPF3IDCrk6SKu4tcshTPAJmDE8uq3vylamkLbpiYurdYNeHes7yg6iOAgWOKxF5DNFyTwyqQzCIe65Z6/cf0ylhZnJTqW3wE+CR29KZpIpQ6KEpclaLPbfC3B4610EfvVIgaHD4mnDTUJ2AEkpbuvdwirBgw9nddr5ZGzFzA+MiAnUyanEXl8aVq6lEzvBgYmQSMrwQO4uPCKGOw96D38rybe/PeHFwB3SXa3DpfLhvv3KO7QlLt9aUljkLGx45KGbm5t40tf+jI2N5bEpHpzMy53LYthJYPE0hUJYfLoGVn0PM3Y8qV1TE9XFLMLc9jZ2cLszHU06lXksopSFn0igyLm9vqE3s5B6rzL5UF8+Z5UBwkY5WvzeNjd0TiQxE8WjNjUIiaDiq78RxprmPUnwSC+angGUfG6VjN2t1fRqZHwMfl95dIWHjv7R6qdl82hXq1hbXVdVTXLdAzdkScjTt7pov6+wpvN3U6CIjHv1qYb7Y4OdIaC+PXla1LU6e4dM1w8G4hEgjhx7BHx+93cILMmg5e/+BVEIp9l9pqTtbC0gHw+D483JKzYoGDiiIA5gIX/7078wwuAO2dooB9bqargH7gAKmXiHWq4du2qLC52S9lrp3IKtQkmJ49iMX5HsgyeCjs7y1LboDIYATZ8bxMTR2VCCc2u7hcwNk4+Q5u0qpPJLayvzUgMVjHEoPiaKPMuk+7vFNAtT1IuIH4/X9xALlsQMOjubg3PPvVFpNLbiC/fFhYWx04ig70ygTLm6c4r3XQz3dcFIEJQzQasDkq66CpjUOh2aizQqNsx2KdSLRz7lRqWl5ewujynYsVeu5QXqWkvP0vwZLBb9HjVrFEt1wKhvpaC5t0Zgj3ahLYUCQeRTChyxucOI5tLiZhjNNaHl174wm+dQ9aDuAi527qCFGs6CFT/Tye+tQBSaczMLUnxZHLqsLiQMMgql/LyXj69ecPotQPXblyGz+cRZPLTT72IW7c/kfYvH7eTWBHSKyN7TtrhiTGxbRE3MzG/VHEIOVV2uYgHsPTgttzLtcoBwcbkR3i9AckiRGSDGZvTK5C1jg5W9DTVtFjcePS0Iri48wvFNNbXFyTY42NbdrZC+aOBpAeWV177RhPWDGCpy2rmHWuO9rZwS0d3cOC0TLw5Ln38thgqC87fShatkhhiMQ2QiNlruW61EQHcjlC4T95AOlsjHRInTp2ROzWxs46V5bicIqKpb/jo8fe88PmXfutcsljS8oP+v53x3/g5KpEx/yfMiguA/XPWKfjCeV9/8MHHOHFM44p33/+lLICzj55HT8+ALACCX9hQunHzkiB8MxktcI0fDsLZERPBBqbWLDVzobAYxOCZZBgCSenVuLT0oCUBX68xvbUjEOyC1+A+lvcIxbegSYOrikiUDDAAAAS/SURBVCqPclRMxUsAJ45qTYPo7M3tZUE7mQW+TFo7hbyOLK98R9vB3KGkQHHsV3lMECNIFG4bklt5nDhxQQggD+YNE6NaET4frU7JKWygu2dE8HCkYPGPauYWpcm0v78Hr29YKNwc9g4nxieOoi/WRcUALMzdx8rSPF566Q/xy7d+IBDqJ556Xh77G14V/4+m+Xf/mu0ET6C09Pd1EAhLybV9TE/fxqmTx400ixgAqpkE0BUh2LQpLKmdnXVcu/6hdAbbLA4BzPT1W6UQw+IZy+u8/wm22S1QYYT9j3bJOAhd39hkNqbFKYfDB0uN/QSXpI40lqrV+Xe3pjUySEKwCZ5R/ma14uik1i1S6QQ2t9fQ2xuTEjVjDVMGgD0VWQD0uCfRw2pXEYP9GqlHMeTShZZx0t6BmaUof3W0N+APhlStusaScEhQLuZgOdR0OHG5e4yybxts7Xo/8+gfH+1HsdyAw84Fp+vwnXf+FofGpzB6SOXhfh8LYHFpGYtLK5g4PCli0fH4HCJdMTkdx8cnMTdnbIL9XSkyMeeemf0Uwc4hnD17DtlsEvdn5rGTpL6fQt2cbk1LKQDVRBnDQ4cN3qMNXZFBxOPsq+QkA1jf0Mif7CHyGWTT2KoIB7SuIvz/Zg2BULcKWplazPQeMiTm2Djq7zsk4FNmGD09fSp53ySeM41kekcyBMsbb6pABO8jVpj4AKJJvM5BSWVMkgqVq4USbvSn3S4bGJmyFtAV7UYhlxOGy24pL7+DHbjJqVPygrt7R6UQcv3mp/j8s0/KVfEwNJOMntm5RV2xqTsSXY8bC6C1ov4/frG0soql1XWMj44Lz29hflZUyk6fOCMLYHEpLvnhyGgM165dQT6XRonFmLoV3d296O8bwtVrl+VoP2VA6a7e+J4sAFrQWu0lHBojC4mV1za4HDFpSs3PLkjPgehr4vj2qsxADJcQSwkDMd1IuwVyCmoIdfdI4M1JZ1XRYlQCWSY2BaGIoaSe0ODguND2WHvJGphKofRxATgdqoi5V67LxJsjlUiIFy7VNmwGdJo9cYo0Mfc0bWZCgRB2CHIsqfEhW8qRSA9ItbLaHIhEe1XP9qEL++EFsLpGFpHelfaOqpSMe3u09v77GPT/SzLV7WSBJ4uFB3PYSmzL3U1lMAaIHMPDvSjkC9hJprAUXxIxKbbPe3r6sbm5DKK7/b6o9BtyuU3cuXcd9QYlbBNwuzsxMUFkE3Xd2VYvYSW+JhG7kkYMtL6VOAl6HJUBwzOZKmUelwNOD8Gh7dg3HhsMRFERmVum81osq1TtqO/XkEgWcezoKQwMDGJ8fELiD3IcLT/4yV80Cf3eWK1gZESpw0SXsuNXJJihzvZkBR1OlloPWLUEFzBDYGOi0xdCKr2JXFYnkYOdupGxo/L1+LgGhr9tcAHEuauIePXaMTI0Jvfq72ssra4ivrqGeqkqdzyrmLySWNIdiP2PtYXltXWUykQhl7C4EBfhx2pFMYH7NZ3I7Z11SSlJcecCIMnUzAi8Lh8i0QBSiS0UKTtvSoISObWpqXm+wMZXgZheONuJNbChu5cYRG5IzX7MlJFf7xbT2nYW02hK5oaQL+xhdS2PP/kn/0wev7h0D/8dBaHfttLp8GMAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAAgAAAAIAIBgAAAMM+YcsAAAABc1JHQgCuzhzpAAAgAElEQVR4Xsy9B5Se53Xf+Xu/3nub3ge9saCwgFUQRZGiSMqKVRxbsS0lzsnmJLt7Nsl6fZyT6vU66yPLTmzLVqe6RJGUCBaAIECCIEH0NgCmz3xTvt572Tz3BSkrIhXqWF7rwcGZme/t73OfW/73f++nXZ251K3VyjicEYwmKy6XnZNnfoQaI4Pb6YkNU6vWiQUD8tnJsyeplot4vQHodoiE+2g0GnQw4HE7MZltuOwWjEYD7U4Xo0GT49Sot/SfS8uzZDPrGDCRzV4jFLARHuilL7Tv7X1/kb9kU0n8ofA7nvIv/uyz8nk+V6JSreF02uTvbDaLx+Nm0+YtZLM5Gu0O165fkW0ej5dKuUI+n8FoNNLptOm2O7JtaHCQZCpPs12Xv61m6B/oo14vYzRaSCYTrK0kZVu0N8qObSM0m3UazSb9fRvl83K5gtPhIplaYW1tiU2btpMv5mRbpVJkaXmOdtdIMODHbAKDwUSn06HRaNNqdeh2IZFcZbhvBLoQCPmo16u0Wy00gwGT2UG73aHWaKI1291uKr1CIhknEh3B6/PTbDV55cSTOGw+YuERKhX93Y0O9tNsNjl56iguhxOXQ59cj2+YcqnEwNAEnW5bXROfy4bZZPqpl/7y8SNUKwUcNiM2q49McoqBfiu+3jH87kkc1uAvcu5pNpXUdTGbze943v/nD/5Qn4xYL2PjE6TTCfk7nc6wujzL6PgEK/FVUpkURpMFi8VClzbtVpvl5SXZ1+/30Wo1iYV1IcsVyrRaDVKpJXbtugmDwUi1WsTjCaChkc0WuH5tntv3jchCyZeLuD0eOdZicdBqmiiVcxTyOYwGoy4U9QbDg0PUGnVyuQLFUpag343ZbMJkNFOtNmjeEMJqpUrA79Oft63RajdwOmxomobRZKaLgVpDF1gRAPXLpcsnRUL7Bkax2uwkEgnml85jM3mxmN2MDI+IlHXabV579VnK5QwbNmwSiTIYfOzYvlNWfLHSpJDPyskH+6K0umD6sRLg4KEf0mrWoLoORgMmk5nNmwN4IjvlGJ9j5BcqANlMjmajSST2zhrgLQHYtmMXbrfnbQG4eOE8xWKBbrcrgtw16CtNCZKarL6+Pi5cOI96tHK5gKZBb6yHLhr5QpFyqUCxmGB4aBCDAdwePwbNgMFol+c7e/oMsbCdgcEIzVYLg6mNw6ULf62mNFCSaq0qfzfVSwTGxjbL541Gi04nL58pTWG32VldW7/x3syYTEZ8Ph/1WoVuuyXCqobX55PnUf/bHY1KrYn2n/7wP3bfd//dTF17k96eMULBXix2D3a7i0xqVQ5ut6r09g4RCvdw9KUf0mjW6bRbuFxOAkFd0rZtuwOjelKgVK7Q7nTwul0iAOolGTWlvgocf/VbNCoNUU35YomxiUH94cYDBEMHfqGTL/dSKNJqt/G9tSJuXKFcrpHNlvjud74rn2zfsR273c7o2LD8/dSTTxGPLzI0NMDAwCBnz51mfT1Js6HbsZGxIXk3hUJOtJ4awUBQVLCaJDU0KpiMRux2C06XF6PBhN8fZvr6NZZn52WfgdEg/UMRGvUaTldIVmena8If6OPM6SNwQ2AsZgvBQIR2s0WhmEPT6pjM4HZ5KZYqFAs5CqUymlFjdHBCVnun06LVKNJqdrBanJhMBrpaV959s9mg0WqjffXrn+uGwx6Sad3GeJwRnE4fgUCMTDIOmgGH206306ZWrRGNDpHNrcvJlDbodFts33kPbrvlHSev3YULFy6wc/s26o0Wx17+IqurBYq5hNjPwaF+tm+/nVojztj4B96zAKhVUG+0cbusP/OYYj4vts8X/LFpUfbv8uVpThw/inqxagRCEdwuD/2DA7g9Ln749DOsr6+zZctGBgYGKJYKPPPMD0UAPvjQgySTSdrtNrV6ncXFeaxWC26Xk0qtTrPRYG11gWDAicVsxmazYDCaZFK8nhAXL17E0FUqucPohBuf3023ravo9z/wMc6cO0ldTBecOfc6BqMZt9OFVc04BprtMrVqCZNFw2yy3TBzsJ5KiolSoyfcS7ulFqquRQyahYDfJb+nswXabf382lPPfL5rtyvHx8D6uq66Yz3j9PZOkEkuiaR0Ox0Wls/Ltg0Tt7B12y3MzF5k25ZbMBpNIkkGTcNk1DXAW+PJHzwjvw6PDGGzu4hGYrzw/JOsLM9gMZYwm7309HjZddMdojItDgvDw7vfdUJbrRbHjr4kL3Vy405sdvvPFIBkMs/M1Ckq1Sb3Hnj/2+f9nwmAUpFXpqYoFot84uOPMzs7z/T0tBx/7NgrWK12Hnvsw/L3D3/0jJiKbKGA3WqX92EyaWRScQxam+GhIUDZ2zalUolqtcV6YhmnLYTDYSXWq2vQWDSC3e4hGh1EM3So1Bp4/FHZprTHq0f1d9lqlWSulBY2W8yiVdRYXV9VnoluZjST3Ivb4RB132rUaLWNeNw2HHYTHbrkckXZpj3xzT/oBvw9DA3eKivlyvUzWGwewsEequUmWrfBlamXxeM3W52i4nbfei837dotEvuzxtLSMsdeeZVsLs0jj36cRqPO0uI0x4/pUUbI58JmNxMOefB4/fQNTTI0uONdT6ls4itHj+Cw2zEYrKAZ2XebLjCFQhGr1Uqj0cTtdtLpdInHE8xcOUlP3xj+UIRINMhyXFfjB599Bp/bw9Lyoi6ko+N4PB55sVaLjWKphMVi5kMf0rXSufMXqVTKZDMJTp48x0c+8jgul4t0Ji3bT5w8SSqRplDIksumsVn0d9PXG5RV2W43KJdL1GpN2m0jnW6FO/cfoFYp0+kUMRg0opEeBoe2kkyuUCrr5w1GxuSeKuUMV6fOyWcq6uh0OzLZJrOVerOO0WxgbV33A+xWJyaDCZfdTi6TpdMEh8dFs1HF7bHj8TjF1KRSGWUC/rB7z53/kEq1gNfvx+1x8/yhZ1lcnCMSHCDgjzA/dwKjoYnNZiNXqBOOBPjIo5+h3lC25SdX/d+cvVa7wyvHX+fSxTN8+NFPiL06dOgpUqkVmo0Sbpd+rKlTJXzDSdt+0wEiYd0O/4/j3NkTZDNJitkEdpsNq3tQvOxiqSbhlXoxkUiUleVpfL4gFy6cxe92MD2zhNPjoVRpsLq6RqlUJBYNoky37gKDy+PBaDJRKecJiLkw8Wu/9rG3b+EHTz1Du93EZlV2PIrJZCJfKDE5Mc7xE6+Ry2fZt2cPV69e5+Qbr1OtZAgG/TSbZfENmi09LCyXVMjXJRD0cOcd75PPnj/4IkNDXsbGRuntm5DP2u0qpXKZQKCXRCpOuVIQrZLNZjAYNWq1huzX6jSx2u0i8OpfKp3RBcNgxmq0Uc4VZD/lnhlM+sOGoyFMZjO5XBntr7/477tGo52HH/oMTo9Tdjx8+DWmrr4mO0+OjVKtNmnWU3Q6NVFvmtmBwx5k100Pyz6x0E/b/1ani2bQuHZ9kddPHJb9PvL4b+BywCuvvkKpmKVYTtPtZjEpL9lpx+Vxsnv3x0SNvtuYunyWq9cvYddadM1OXC4ft+zeLzb4e9/7Fu1WDZvFRjQ6wLmzZ6DbpNMxYraqVa3bw5XVOJFwUEIypXmUarbabBK+looZvB4vDpf/JwTgia9/g0ZdqV/Yvv1Wce6OHnuFD33oQ6ytrRGJhDCb7UxdvcKT3/+W7Dc02EsmE8dqMWO1WeV4X8CFphnZsnU/42OTXLp0mdMnX2d4xM/o8AY5rndgRLTx6Ph2rk5dkM/U4jv55gtUq8r+V0Rwq806brdbtjdaLfmvIrVioYTT7Cbg0bctL8clSlHD53dI5OX2+ESAtS995Y+6NmuQweGt7NhxM+rdv/jCUV1qtMzb81AuJ+l2qqhl4/MPEgjpoIWSWJ8bbFaLqF+lNitV3Su224102iqGhUJR9y/6o35ee/0UyeQU+aKusob6zXg9USoVfZXs2/dr7yoAasMPfvA13HY/wfCgCMvA0ICoteWlNZaXpjn1+ityvNXuodNp0Olo1OodavWyfK4AHK83KEKjYnM16vUa5UqZXD4hYV84PMjHP/EJAgGvbP/Od78jQptOZrn5llvx+0N881vfFtUejfp57PGPi0P2hS98gWRykU6nLv6J8sTtdl2gx8eGSGcVnuDkn/7O71GptDl+/AXOnX2V4UH9Pt53QH92jy/09jtYW12S+7NYbBw59E3KtSotBT51u1jFf1NeRpdao0G71aFeaeK0mXBaPXRU2Aesraxgs1sE6HO6XBjNNrodDe273/tCt9XSiPWOywUGB/pYWUmRSqexmNs0GgVMRl11LMXXGejvYcPEbpG0dC5HKBhhfmmJHVsmWVxWqtnC0GAf9aYONLjsOpBRqtTJl+qsrSzRbNRZWTlHo6kAkxUG+mI3QAqfhJ87t39IwJN3G8oWv/H6a1htLvbt2YsKc1stuHbtCutrSywvLpJJJ7E59JhbU6GPmvhCGa/Hw9panEAwKqt+aFDHHS5eOk9XvcZuAwWkmEwOfvO3/gm9vTp+cOToy1y9cobZ6Xm2bt+Ow+7h8OHn6O/rke33HXhUVu3Bg0+Tz6dotys47A4Mxg5mc1fi8qGBXpqtDlZbiEcf/TUq5TaZbIonvvo5quUaW7f1Eg5PMjyygcHhcTlvJp0QUEn5XioaeOmFr2FzukllEjQaOkJndbplkvVnLNHttHCYrLS7HZxWXYCLRYUbdLDbnTidVtFCaFZdACYnd+F0ellb11Ewt8vB0tKSqK1CfhaXKygTPje/zq/9w9+hkMvIJBrMVomFVUSgwjKHXaNajeP1RLjppgfIl8q4bCZRX08ffJKbbr5fzn/l0mHarSa53DrN5jKhYIxYxEmt1qF541Hu3PtpEuvL5LKLjE/u+QmBeOH5gzQaRQLBAcZGh1leXhTHStnFVGKNfD5LMZ+TM9nEE9bE2LeVN6RWw+ryuwoANGnU6/gDUR758EcZ6I/x9W9+nWIhy8rKHGsrcWI9feQLBehoOOxW3J4Qd939gJz729/+grwrq9UF3SpOh4VOt8nGDVGJlErlOnZHH9lciqHBzRSKRZbmr1KtNNEw8Niv/Ar9/SNUihl6h3R/IJlYY309QTgck7/PnXuZ9eSC/F4oZpSEY7E7KVfrNOu6mbMaFOKnD7vZLRpDaW+F1ShcQiGQKpoQAbjp5ntkx+OvvkQwHOPypRPY7X5sFqXulRrtMjU1TTA0xGd++3dQmlrBufV6i7n5S/pVFL7QaVKtrTExuRvNEGRx6YqARROjmzCbLHzru3/KyPhNBHwhivkUjWaN+fk3GOjx4/Wa6LRdLC6v0DfgY6j3XtbX4yIYDlcAl6f/xuPAwsI84XCIIy88Tax/hGRyWYG9Ym7SqSztZptiIS8rxWK1YTaaCISjnHz9FarVCnRb+ANe+gcmsFntNBVa1mnK/auRyeYplcr8+q//Jj2xEF/6yl9TrVZZXZkT3F4NhS6qVVSvFDHavNxx2z2CIl65cp52WyF0bolG+vuiNBopHC4j6gbLlZoc32ioyKCNUTlAAlhVada79A9Fcdm77Nv7MPV6lmB4SMLBk2+ckP0q1YzgDtACg7pfjUIxLaGuyeqUfVpNXTOYDSZqNeV0tnHZdH/gLUQ84FPmoYP21a99rrtx4y2CUL366mGxGcpxsdvagjNrmFhZ1e33rbcc4P3vP4ByQG0WSCqYtdlkeWlWMGv1cvbuuZ342qJMSLerHK81XQotdnFglldmueXm+1lYmCKTTZNMKJDoNjrtHLOzM5TKRSbGo8TjZTaOuLBaB4n2b8fm+OkcQTKxwrWpsxTLRVFnNruXS5fOUchn8Huj+PxhDAYFlFTZsnUrVy6dZ3r6GhazAbQ2FquPWLQXfzCA1+dlfn5O7lXTDDidTh5+6AOiFb/xzb9GOZ8GbNhcTonvlU8larRbx+rQ7fD46C7S6SSrKylRt5OTg6Jy282m2Gu3tyZRU7etkSvWRGup+1Pb1tdWabXaBAJuwkEnIyNb8ft89PVtIJvLkc3lWVlblkSQChmN6tKaguYb1BsNMrkCZptVIgG1GtvNmsDS5WJZzq/yBU6bgvV102w0tOU5RQDUDrV6W+DCVGoBq1XZLf1FdDtW0qk0I6O7xT6rWPuuew7wyok3uH3fburVsiSGzp1/k4kNO7CajfKQyVxVMHGbxcLZM8/LRQvZokjuyPgErbZNbJgKq8KhGOcv6d5uOJCVB6oWYcOol43bH1Upkhv/31YCb/8yNzvFyKjukJ45d57jrx7GbDHh9wQF6lUCtXv3bRKLz89Oc/XqlNhAZRIMRiObNu/CarVJomc9EZfzuJxuCYfVtvPnTnL61CFJzBQKdTAitr7VquKwmwXhM5o1HDYPfn8P9VqLlWV1Hg2fTyWKPFjMPrleNKYLitdlYHYxQaFYwOWKCo6vvPRaqS7mY/eerQz0j2G3OWnUG5jMFuKrqxz4wEf40pf+VJJPKv+gRq1exWIxEF/TkVVNCbc48B0JW9UoFWpiBl12jwhDraov6KDfg/b5z//HrnpJunpwU69XsNnaGI0a4VCffJ7J5Mhlm/QPTFJtKK/WwY5du5mbuc7Y6BC9Pb0YzXbRBh7nj7Nu1+bWBO5ci89gNXdZXFyQ1HG7s8bg8H5yuWtEoj1s3rBPVP/S4lFcbpNAmwatl55oL0OD49gd/p+e+Xf55Mixl5mbvYrF7KLb7oqttNncaBixWx0sLl6jt7dHUD7l9O7ddx/NVoNcvoD1Boy6YeNGorGwhKfFQoWTbx7l7Ok3WFpax2RVk65RLhUxmxQAY8XpUOifRzB3vz8i6V6LWb3TLkODwxJ+VkoNRkYj9PUof6oLBrM4ytdnr+mLI18mGu5ndnpGBGBkbAfj49tYnr9KKh3n1j0fwGzVw+2DB78jDmsiuUYunyMU8Mj915UnrCbfpBI+dZl0BWopu7+2lsNpc1OqlFS2gUolR6VWRvvSl/9dV0GjuVydbkeXHqerLSrQYDDQbptZWdbDNbPFicfXR09PPz29vaTSRRqVAnfcea+EI++UDmg0dSlVKNa1aydotZWz00UzdvF4Quy/40P4fWFJZLzyypfpizRIZs3Uyk22btnB4MidPzMieCc5UIJ47NUT5DIJXZvUSiTXE5RyRaLRCCazRqWqBN3F3fd+UHCAlZU4H/jABwXZm5mdIZFIsf+O/YKAHjv2PKvxWZbXdA1RrylsoYOmFXE5vBjMHarluuRPmg0z5bJy/kwEg3Y8bpdM0no8z4EH7qHbyVMoVvAHeyVLtrA0TSJVJOh30moaWVtOE+sJ8euf+l8wmIwszV3G6+/BZnNIpKSEQPkOn//r/5tysYjRYsPjsgq2X1OJKq2NzWamWCnicTtQacxmsysCoBZ1q6mDR/VqUTdjf/Cf/teusgr1ekliaoNBec0KMDBJjNzpmCUXkEzoGS6ny8OOXXfidLol3ajgXpW7/o1/+FuyPVdRKV1otMFiVAAFokbrtSrXrp8UTNxkBavFxOT4zQz0T9A16P7B4twRXHYFDtnlYZ3WAG5PkOjA3e9ZA7y14xtvniOxNo/L5WV9Jc7C4hwupwKOXHItlcULBHuY3LCDSqUkBIq9+24XLaUgYQV+7dx5C9/7zrdxOc2kUovYzE3xptPZsjjAJlNHHEqbzSrndHt6KOQbNOt1cTYfeGC/CO+582dZWUzw4cceFCKNyx/h+rVzJJIrmI0dGu2m2P9qqUKp0GFisocPfuC3OHL0q4yN3MJaUvej9t52QCb/qWe+jpEm0zPXMVkVgCUZfhEoi1V5+CasVhVG6/a+1VJ+XZlKtSFmwWSyYzAaxHxr/+Hf//OuoD/iGOihg0r1GlXs6gmK5Ck+gMr85XJJYcMo52JsbIvYP2U3g/4Id955N4dfPs6Om2+jXikQCuoEh4sXztJtNzl35hVS2RWcTgcetwero43JZOWxR/657PfMs1/FZlwjHPTj8VgpFKu0q0027fwIdmfk5xaAtw44cuRlluanCUdiAsrMz14TKLhvYIhgqIdysS4A0fTMRRxO5STZeOTDHyGTyWO1OTh65DD9/SHJ7pVKmRsv1igwrQELmqGKzWpAYSloQVG7y8sLKBBuYnyzCMylC+c5cP+jRHtibNmxXfyDs2fPcOXKK2idOo2OwgOyoqqrBRNj4/q7C4f76O/fIqpeTb4a3/3+VwTeLpULLK8sC3ydzScwGa3YbDrc53I7MZt1HEX5KaWybhpWVxN0uya5jlqgamFrv/t7v9U1W2xyApXDV6dwOEwCWNAxC4DhcYclHdzutEQlmswqB2CXiVEY9/ZttzE81Cfh4dzcNP6APmFry3OoDN6pN1/GYtb02L+UZKC/H78nQqtdYdv2/SyupSgWljB18tgtTfr6/JiMSp3B4NjPRgX/Z5Lx/HPPszhzBZvbLyhlJr0m6F2sp18w/ZFhxfhZ4ty504TC/Wzeuo0NG4ZJJtN84Ytfw+9zEQ75xZdZWrzCwMAokegA58+/QaVcpkuLPXvfh9/roNG0kMlkeOH5pwkJ5GsQCDqxtsRNO27nwMMPy/tUQwnA4sKr5PIlzBYrRkOX1UQJw42cgeIgDA1tw+1xMj754wTZ+bMnyGRWyRdX5TzxtSU53mTUhabR0IkidrtZsn8qYlD/lUNsNhtIp6uSkczndL9P+9e/948EL/C4QxLbvkXhsyshqCsQx8nY6CayWd0P6HTKWCyKO+jF4QzicPiJRPtR+YRIVAkKzC9MszQ/K6rp3KnjctzQ0DilchLMBRxmD8FAD51uXk8jW/vQjCbazTX8wTqGjoWI18XA5OOyWvT/720o/oEin+j3qpyhDn/yx/8BzaQmMkxBULoGRqOZyQ3bf0oAHn7kUex2jctTs5w/d4pMKiUmYWRkWMySCqcUZKyAsGIpL6GrMhXDIxNcvHSRs2dPkUom8bk7DA5v4OqVywR8DsKRXj7zT/93ua9sNk8mvc7M7JssLlyXc5qUcBYqoHWxYMRm87Nj201s2XaLpH3VtdXkq7Ecv65Pdjsv7z9fTIpf0mxq8rz1RhG/T484XE6bmNtsRs9jqCggvqhzP2qtOtr/8bv/vOv1WOQCKhRR5EZlZ6yWLiarA9pWxse2kk6v0GwVhQgyOnIb+fwcN998F52ukf6+fmbn1yXdqyR3afE658+fZjV+XfyJvbvvJ5fLEF+fo91eF1ZLwNd74xrq2ibUY3s8ZezOLh6bDYfFKVGHZtOpYu91KMWVTGZwOz1YbSZBNJ/8/l/T7liEYmUxKd6dQfLoKtRqNTWhsimUTfk3mzZvw+mysLS8xqEXfkQisa4Cbh56SAEzdYGV1U8VaaTSSdEUoXCEQlHF/gY0zUqtvIKmmSU5lFxbZmign4ce/ThjE5vEX1Bjauo0y8tXWV7So5RSrS6kDQxtWc2dRpkHPvAYAX+IqatTWC0d6rUaCwtXxH6rbKRMdLCHdCYh+Q01lCnKZBYFgPN5bILw+nw2gZKX5/RFrMJZ5QQbrCa0//AH/7qbz1fwB1QKUQ8zbBY79XpOJq/TMuL16yGOy6X4Zg4GBm7BZndiszvYvlWPwRvNDqfP6KQRn9dJOpUkvnSNYKhXIE6jyc7K6izLS5dxO9sYNQ2XOyhsYq/PzfzSCvfdOSoQ5WI8zcbRAfqH7gXTew8B3xKSXL7K1OWrDPZHmJo6xdkLp6g3lQOkoFv9GV1OK52mBavFJZj55MZtghoqIo5KKU+O9/HVr32DbDojK0iFvj6fn2RyjdW15f+uO+s4HD5WVtYIBPw0mlVxDJtNqCjKFk28Hjfbtg0T69mCyxdiYmICn9fL1JVZIlEnp0+9LveSK+RIK9Kpwcgjj3ycM2dPsWPbTqLRGMlUklOnnycWGSadWqRSrlGpVIn2hCgUMtQaZWw2n3AG1L0pU60cwXK5zED/EBZzUyIShSco2D6TLrO0FGd4dIhisasLQLOhZ8l6e6Pk8wVikTFRFZl0nLqijDscuL1mCSk2b76DnTvv4sKli3JMJOLD5XCxvLLG4OAYLz73DDXl0S/qaurmm/disXrIKyw9PkOpuIbR2MVqUzTygGidbFHBqw02bhggYDdTbRm4ff/fzvYvL6cEY19bnUIzGVhYikumTHnGYWHvtpmdXsLj8or/c/Oeu8Sx27R5BwP9emZOhdVPP/0MyfVVjMYO+UIah93L6uoiLredTC5Bu63MoUVUb6PZIJ0oyySEwl4sNjf9/cNs2bQZm8PLzh2bhdHr8+n2OqVoeJ0m5UqeQy/9iA2T23A6XWzetEOAq1Q6xYk3nsQkvMCOQNrbt+9hbU0nsSh6fb6QENh+fV2PFKwWo2Lx4ffr0LlBy9LtNjEbjVTKTWrVppBClI+nwkLti1/+g246nZXkiAJ0Fhan+ejjv8N3v/9FOUGtmqfTamM1m+ia6ty1/zcZHRlVOAap5JpIXSoVJxQaE6z7zJnXSCVWxaTctldn09SbJY68/Jy89G6nLli5y+XA4QbNZBMt4w9YcFiNDPdE2H7rr2A0/myu31ur/Wf9XIonSSVXiS9dIbsWp1iriV+gJkqhhdlsUdKm3Y6Jfbd/EF8gSP9AL16v4+3TXr22yMHnnhWNkU6qpFNL2NOa1qJUWaXbdWMyauI3FAoNiup/PoPH56XTNvNvfvf3iUb11K5CVIOhH0PaJ04cJ5dfFfJK80b2dNvWXaJp3hrPH/oajWYBk8El4JbCMUR4UvqEK36m8q0EAu7acTiaWEwB0QjDQzfLPjOzhyXvYjEasFt14fYFVLjf1gVAfRCL9YndarV1ByEc2su58wdFelw2B/VqjXJT9xwff+xfEo6EeOnIV9BwoGHGZlXhXQ9nTr0i6sfvD2AwWBgaGiOXy7GeXGN+9qKuWbI5QlE/Hq8JjCpF6cZlteD3WNk4OcawkEPfPR38XiZf7aODNRpHXz7M3JU3aRvNgjJmcknCPVHW1+YFiZAAACAASURBVNIM9G4knVwWXyAc7uFTv/2pnzi9ut+/+usviXPc3xvj4uXz0KmSy2ewWIw4nB7xJ9TvBQW5drpcnZrCavVyx137+eQnf6zJXnzuRZweK/v23SnXWFlZkZ+ra4vUqg2GhkfFn/qb49jx5ygWF8S3aLd0FtDe3fdx6fIZifeX4pflMxWONupVXC63fg+zupbYsmEPqfQ0FqOKFBry/tWcqKGcYREAl1NJjp47N1kM1OoGWm2H5P2r1QLXLh+i1W1Trr6FpJlw2P0CNypCqMngwO0IYjIo+6bIHstCeDQa3TQbXUKhHtbWl0llFFHShEIHnW4IRb00miWcDj92i4Hx4Ri9sQH6hu56r3P8nvf78hNfol5ISSq73m68qwC8/4OPksmsMTExKoSRt0YuV+XKlQtcPP+G1ETUqlUpEHG7/JTLRXp6I9QqdZbj61JUokzDr/yDT3Df/fvlFE89/SydaoF7Drwfr9f3dmBz+co1ceAmJyaEzqWqfUKhH5NBluPzLMWnhGKWL6Qk+rj37seZm58hFutnZvYsieR1cQpVer1rMArQo6DhaDhCKlHDZXfcCKZy0K0wMb6dYGCj+BLan/+33++qkC8UjuLy6KHDti0P8dKx54lFdUlx2uycOvsD+d3tsUkK1aC1sNoiwoi12Cz0RraIljBoVs6eO0o2u4DLFaZZtzE4NMLiwiztbkNSlw6XWjUQiXnFG7VarHjdDnoifm7asQ+L4505ge95tt9hx6eeepJ2LUUmlaDaqOL0+UiuF/j0Z/41R4++zNzMZUKhmABB/f399A8NMTAQYX5xleHBHnEOZ2anWVTef3IRzVhl+toyrYYq6HAyOBgmlSwIIWNhMcXY6AaC4TCPP/64pNRnr1+lkM9z7vRpHvmVj+L2eEUzKhxAjYnxEd44eZLdt976U3e/nphndvYS+UKSQiGJz9OPPxCi3amRycxQqyuKWId8OU2no6j9vXS6NVqtDJm0S+J/h00H+5yOLlarh107dP6C9tWv/eduq6mrFrvDzdZtu+nt2Yzb5WZhaZn4mp4iVfTwdrtGPjctRAeLyqO3XAII9fbsxGzUMBosnHjtoJAzC8UELrcXmy2AyxGkWMmynpyjTZNqpUMoYBGYNBpWKVv95npDIW6+efcvXABeOvIi9UoKh0URXHIkkks4vQru7fArH/sXXL58iVePPYfFZMTlcBPwB/EGewRgUTQqxXrq7Qtx7fo0AZ+fxcUprl9/ncW5dfGuxyeGQWuxtp6hrBI/5S52u5XVeJqHHnlUagMVi8modamUyri9LrL5MvVGXdg+jYYyiyk0OgSDYR577CM/IQTx+ALTsxcpFVWt4DpOh5PxiXEG+sc5d+GInCNf0cvUDN2IRC0qEVWurmE1R7BYjRLq1ioV4UsqTuDO7e+XukPt2ee/2M2m02/z+VRu+967Po5Pqakb441Tb1Ct1ZhfPEmXKi6bhtXqlwl/8AE9B6DGX/3VZ8VJyueT4mH6AiGiUX01v3Hmdbw+C2gdHDYvrUYLp63L4IDOfR/sv1mctbvv+/jfZqG/47GLizNcvnhS8HlVCKL8mlq9htEU4p77HuLb3/46xXz8RgWThR07bsVgVqnxdcl5TG6YZHi4j6V4goG+CK+/cUTqGE6fPCTqtrcvQqVSY24+STKRlQSQImsqGlanqfHYP/gEe/fdxn/5g38vZVw2h0a1UWF4ZCfJ5BKB0ACXLr1OpVznjtvv4cCBnyyQia8sc/XaabKZFCZjRYgm27bdLovy9LnDggvUalVxrqOhUXkHheIaLmeMVGaWWM8QXk+Aixd1om+npdjNbsLhUbQvf+X/7dbrDXEgVJwc7Q1gt/dLpux99+oUroMvHpSf68lLb1OOHTYnZksbt3cHA31hhgeH+eyf/SeMbXXxHJGeqIR4KntVLDVYT69hsSps2orPrQSgTTSoGKoGHnzwH9+YuL+94/dOEqBo2FcuvcnC/GWpzDGZDdx+5wd46qlnGB/fwJkzJ2k1C+SzGYwGK3uVk2awcvH8GcYnJugdGGNkWKdjvTWOHHqSixePC5N5diGFUTOTTBTottuMTYzRatXJZTMUyi0ioSgbN22Cdp1gIEClWmYpPkMwHCKVyVGv6iwjTbOxaeMWwSPuvUd/92o89fQ3ZII1rUYw6CUaDuH1xTh7QZFfWwSDfVLIOzd/gXBomNtv+zAnT77E8MhGZmZPyTmcLsW/qHNt6pLQ8x0Oj2hg7U///N92LUYbths2QjGmQtEhHV+2aLQbDQFxVPyuCCGKdl2u5FhdXmJoeCORnq3MzF6gXM7SbbUwaiahYputZowWDZPJQraUJV/UCQx94SB2s+5rWEwWEaL9+x7AF9b5b3+XQ7FmlZFUHEU1jp84idlolrT3lctvUKtVGJ/YxfzsPP2Dw8TCARbm5zjw4APCjnprLC+vcvbUQQm/rs3GqTaatGp6nK0m2WBUBJsKw6MjUggyO7fErTu3Sy3DLbfezMrKOqFwiJXEDBcv6yXnmyZvES2STK4LdhIKRtm+fdvb1/zyV/4LNrtGT7SPvfse4KUj39NNaGREMAMlcIqLqYbfH6Pd7EoomkqvMjNzBpfTI4UpV69ewO/3CqGlr3cL2uf+/Pe7KmnhsDokbGo1u1KXpqTEF1QnqtHpKqBIe1u9jI3t4MyZo5w+cxy3z4rbPcDo0KRcXDFnFRt2LTNPJByl2qrQaasoVUNTFSwmI5MjQ+QyOvg0ODRIJOBhaOLnT/n+bYTl4pVZMXP9vXpc/KNnf4jJ2KG/f4PY5J5YFI/TweHDz3Pz3juIRX9sEpPJApcuvMTqynUW4kt0DVYq1RaZbBmt3pB6SlWQ2moV6esfopgv0hONEYs62bxltwjayuoquWKSSrVEodilv2+CXC6F3+sXtvKWzTvxeHQ8QIWgR1/9inj6O3buYXx0p3AfVcinqpWX47NcuXJRGE7773hAaiUKuZx4/opzqATlR888Ieeq1rJS22GxasSiG9E++19/t+ty6A+n6tkVbKvCQJU7VzhyKDhKu5tjdHQzmzft5o3XDzExsYuXj77I+voCRosqazLjcgXw2fSXmVUFmSpE7FapNat6PN5pCUnx5q3jRMNe4elv3Kao1E3MN5g4f5sJ/XmPnZ5VTpPG+KgKpWa5evUKHpcDi8VLKNyH1+Pg0oWz2B1mvP4Ik5O6bVUVDyeOPc/a2nXKpQz1pplyrUA6XyWZyDHYFyKTqNBqq5p/txTEqpoJk2aiJ+anry8iIFe+pChzeanzV9zCaGwzmUwCvy8kqfedO/ZIziEWiwk4dfbsMRbjF8UEFApr+DxbufWW/SwtX+PkyVeJ9QxIbeZt+x78qVehFuCFCyfJplIsr14U3OJjn/iXsp/2R5/9F91O24TTaReCh4rp1dhx0/14HYoP4CS+vkgiscL7DzwiaqxQSHHo8GHS2WXq9SJDw5N4nGG0roHlpWuS4YuvzdM2K1qS8kcNUj2sC8AYAz1BMNoIx3Zhtf9iG0K8V0HI5ErkMhlGR/XydDVKhQqZTJZyrSUEj/OnTtHbF2a7wuX7Y5LpzOUqXLr4MuVijvX1a5SFyq4yoEmdEuaxYTW1pRpJATX5Yl5CtKAvTPAGBPybn/43PPvsl6VCSSF2KlFjtgWoN6rC4cim8/i8fmE5P/roRxkcGhJW0+raWYYGb+bqdVV6VhN0r79vnGJJ16aqcOXWW+4XAVI9Ht4a09OX5R6WFxa4fPVNfvMf/SvJuUiW8bN/9h+7AV8Pjcay7N/TE6JadRLtmWDv7r1UylXmF25s6+8j6NEF5JmDz3D16hnJkxstbQKeMb0QsdOhXEyquk2sLosUfzosBkrFOlvHB6SNTCSsKMkGQrGtONwD73XOfqH7Kb/rzTdf5s7bfxp0eu75IzicTmxmC9OXz3DP/Q8Q64+RzVVZWrgqdHezoUmlWqRSrVOsNrg6vST1BKocy2xSKS4DFrtbvHxVzOp1BfG47ASjG3j4oYc5cuRJstk0dodFOoaoUve1ZEIQVZUV9CvadlvD6XKwY/utjI5ulOymygg+8c1/K066YjRt2bz3Rt4mRyo9j88fwm4OsnnLLVy7eoZ0SmUpYX7+EuVKVShyn/70//m2T6N94ct/0fW6Q9Qbq5hNbZwuMw7npEzs9m23cH1GTyFaTB02bNpEudyiUStIffzB578hktWhQV9so6RGlb1SYIUqZuzrD0nM63NZiAV6xC72BJ1SfaySP4MTv/iGED+PlBx84XkeeN9P38PB5w6zvnQds9GGxx9gw9YdjI0O8uQPnmCgfwOrqzNSIaSKQRYWTzOzMCdwuOoVkM+vYTG48LoVUdTAypqqRlKOppG1RI6773xA+PjlUgq3W6GlBtrdGqlUWXIUit6u4vz+vgHBU266aTff+95X+fSn/zemLl+Q3kWZ3CKF8iUCgXGGBjfy6uvfxmI1EQluIBQcpF3XpGWN4l4GvH2cPnMYVKWVQeUwzPzqr/62vKY3T76G9hd/9d8kF+ByqjADrLYuvT03iUR2Ok0WFucZGbkJj9NKKp3AbLYy0D8sYVUyHafSUDVwHlRYuLC4Jjlq5WE220UJVxTP3WYxY7fYxNFstiESdNE/sB1/WEca/77G4vIyiUSBrVs3S53DW2Nqao5Trx+mkC+x9b/zArdv382pU8+Sy5eZnp+mVmnQ3zfC8OA4MzOXqTXTYkJVRVK9DiZTm0wqQywSEDu/sFDEE3QIJKscuVqlRV9vQLxxhTNUahky2ZqobYX579x2K4ODIwQCAZ597ltkMyol7WXPnr0cP/48GIpgyGMyO2m2a3jcEdECieQVAt5B8Sn27HxckMel5WlSmQWh6FusTskc3nvvwzgcLuFHaj94+umuUlmFwiqRSIR8YYGtWx9iZXVegJHZ+RlhmqrsVygwKgxaRUAYGdwg1a5K2bW7LbpNTcKry1NnmI9PybucGBllpM+Pw2qV5gXFiiKdanh8YQ7c/5No19+XEPzw4IvceuseLBY7PreJmZkFIYA89eQ3hf08MbmJWM8gly68yNXrVyhUW4yOqQXh5eKF17FbNDZunmQ5PkWn4yQYCLO0MoXZ4BYOwfVrZ6g09MymprWJ3ogmCtma0O97Yv04XU4hqRSKdSLBfkZGN7Bp83YWlxY4feqEOJMq5R4I+jn1xpv4Ak5JXE1NH8FgVt3FtrOwqMLYDPt3/zMxOSajiZWVRYrFNImUjuaurqWJREfx+dx84AO/qt+TEgBVLaJCQatSI7FBcfyWl6+jACI14WrkCkuCA6hRLHYY6B2TIkSV+1Y0X2VP6agKFFUoYcHpVHTkKj0BG5smJwVJXIznCUYDTE7cwsCAHjb+Mo1EusSR55+iIn0E5iQauPOue3n91ZfxB9qcn7oudHmrLczlS9MM9gZxOS0CFiWS87Li1MQrXr6qF7BaHSwsXEVxLpdX0rRaqkCjTW9PgHpdFaJ2ZbX294YYG90qqfcHP/hxKdJRPYNeOPysONmqsOOO21Vxi1MSSeFIlK9+/d9RrbQZGtkoLKOFhQs0Gxlu3/0pwV4yqSVJUqXSC9L+TrqlGS0CAPm8AXbuvE0IJdoPDz7TVbBl+0Y92S179CSBah2XSqRwOjycOPk9+UzBnrtvvovLV6d49KHH+P6T35FiinpnhXqty7Vry7hcVmkZ53K7sZhNjA2F8NptkvhQTZjqLY3b7/qIcAp/2YYSgO9/88/EzKVTRXyBGCMjoyzNn6fVqVColKTp1cr6Mu2Gi3opx749+/lHv/2P+da3PkchU6fRKGG1Kz/HSTG7KlU9Cq27MBWXKAJDnd7emPRebKjmDkYHsZ4eNoxtYs+e/bSabWEoKfraE9/8nPRhrFe69PZGeOihj9JuNsSZUwU23/run+D06yiiETu9oR0SuvdEh3A4dD/g29/6gmz3+EPCWlKfbduu92O89Za70Z597kddr9fDvj23y4eq1dzSqp6nfvXEs/LT7w6TLazTabX45K/+EzENM/PzIqGthr6vashgNbulPrBQWqG/J0q3pciXdul9l0+t6F0qzBoPPKRTwX8Zx1e/8qdkU2XK5bqUkClepM1cp9YsUVDNGWot8sUyprYFl83I7/27P+aJr3+dXPo6ZkNbGkL2D4zh940wc/0ie+/YTyoxw+LSCidOTeFyWzGYOthtXjRUmKz4EA4OvO9X2b5NUcbhytQl3jzzGmtrs+JUq5By1449xCK9jIxOCCn3O9/+Kh3i2Kw28rVlTAaV77fisg1yz90PcuXyq1RKRon/hSKkGQXZVf2NJjfsJBzq0X2AQ0eOdRW7ZGRkkpt3bpdJUkJQKFclRn3xpe9iVsBFMUGzWeO+uz7B4MAQKkGkbn722nFZHeohFHChOAIBnw+nw47LoqDKIGcuKOpSib6Il7GNe/EHeohEVPOkX77x9a9/g1x6jWolSwuNVq3Lpi2D5IupG6ydJpOTd3H96lk2bdrFtu17+f6TnyURTxCOxiRZY7P0Sk5haXmJ++67jXj8KuuJLG+em8fmMGCyNKWiSBFMraqZpsEm3VMU3r+wMM2ePXfwnSc/z9JySpJqykfzukLSQWTuRki+besm0ukbldmmJiaDk0YtIz0CQwHFBfRImHnl8rT4aKIlzHojL81gZNfOO+jpHVA+wI+6/qCPXTt0tTA7e43JyUmpKZlbmCGXyzK/cJl8QccCPnjgNxjq72UlUZBmg0de+Aq1eklaoxpNqjbOQcDrE1Ak4DaLncyXWlyfXsZicTO+aSujo+/eCez/b5FQUYmqtFU1Darj3A9/dFDyHqrZpcpsNqoKxi4yMBSQShqrI8b42F5eP/E8H3r447x8/AXK5QT33/O4dCdZi8dZTazhsAUYHIixbccOzp05TCaXIVFsEF/K4PbViQaDGOnQF9tMKpOFbg23LyB+QcAbYGruCosLK7IIVe3fhx76NYGQjx49wq5de6UO49z5Z/F4u/i8m6VYdHr6rHRi27ZFJ6Go3kzXr83h8brJ5fWmUypzaLV5cLlCjI1NoF24ckXCQFUXmEkpW15h/906nLgcX+bQy9+9kTlqEQtvlKqeW2/awfnL16WyZGXxOrNCVVYdJ3KEQz68TifRYACb1YzLYcBmMDOfqFAqVZjcdDuxHh1W/WUYio/h1X1bTp85x/T1q1L7UKqkWVlekcpnug16evxYrarCNspaMk+xUGdobJSllRkJ78bGbmJ2+go33bxb2DoKptW6Za5fn2c9cZ5UPiMdxIzWjtRQKszf0AGfZwNrCVUa3mJoVDXq7BOwJplOM78wi8uq0ds7QrmUpVw1CGvIH3AyMaYnz1qtNVQ5T2J9iWotRSwyjtPpYfNGBdvrrX4KhSrZXBarRXUSaUoNhmI0q0pv7fK1ha7P6yKVSnFt6nWBEXtim+XAajPH7OIVIYcocMLl6BUvXzmNU6qPXn6ZwYFxLk6dxOnuCMqnMHSLwUzIF8RiVgQLCwHvCOcW1rhjz71iNn5WE6i/D6FQZVKqaPLFg38lTaNN1i65YpnUWlnqArW2EeXLDfQPUK8lSWdMBKNuSqr/r8ND19Cl1bFIk8079+/n9TfOEAzGqFSy9PREuHr1DWbnL4tjrcrQfe4mQb8NQ8dJT+xW4qvnsTodjI3sYcfOO7lw6RRzM+fIZxLC71c9hRLrOemgUm9lhGpvs9lx2I14fIqko1rttNm4YbdMqgKXAr4w169PsRpfFuevUtELf1WzaatDz9n09/nQ5peWu0P9ehn4W2M2nuTN4wcpVbKYLLr9UHUADuso9UaGdqtMo6EaJRjJpGcl62SQcmi1mqz4nE78Xhcmo01KyXr7d+GPbcT5Lt1E/z4m/W9ec3GtQsDj4IVn/pJGu0Ct0aauWDrpEp/8+D/hzZPHpa2Nw+bnrrv3ML+UxuX2UMylMdst0o0kFN7Kxk1bqNWbXLt+nQPvu1c6eV+6fIlnD35ZsHubw0i1XiYWcuJ2drCYHBIu+sNhXJ4ebMRYWp6XRpGJxCpWgw2zWTXqVBG2ibVEHo/fTSjce2P1V0QA7t6v8ykuXToijGKXUzW7uMTiXFzYQgr0SaZK9Pf3Sejo9esCUKkm0KQ7xDuMpfUMBw9+AbOhhsFkl2JRv3dY2oyrECSxnhX2qfIYjWbVBEpvvaZSPwGXlb6eKHaHj2qtQ7vjYtuO/QJ2/DIO1fehXm7w6otfQhVDzS0mWFpMs2/vHsYntnDx/Gnm5+aJhVW9n5mh0V6Sq2ksDjPtZg6TPcJDD3+Cc2cvsmPnVqmIVqysaq3OF77wOYqFNenypZna0jrP7TDLhEb8ASkb8/d5KRU0kmuKbFoQkEf1+rN0mnqPUSUAHY1Utky4R5/8RGIRt6ufxz/ySWoCzc/icoYEwne7Yxw7+l2J9X2BIVqNHK8ce0mOC4V153vr9pto1Cs/LQCq2VOl1uJ7T32eeqOG3VzD6ogIEmgwhdi8cR/Hjn1TV+XdLh5XVMgjueIiVmuLgFuXrrFh1T1LlV05abVNOGwx0FaY2HjvL50MXFtYpzfs5+yZoyQTRbz+KFcvH6M3Okg8vkq1VcZjs5NL5fnYr3+KyxcOYTAFBLrNF1Z46EM6La5SazM9Pcfw0AiVcpovfekPxXdQvQa8XlXlk5JoyOM14vLbCDiDjI9O4PZGqdeUdiiRSc0wMb5Buo42S0npXqK6q6l1qq6pkFSHx8L5C2eEb6mKXO6++05m5y5LjB8KDDM8shU6Kq+gikU7wihOrKs+jxDrHZCvBFBDgKj/UQOUqw0yhSpHj/6QZqtKt5PD6QxJMeX4+G52bNLbl526MMtVxbVvFoTnp2kVmq2ylE97XB5CPocwUY1Gp6hSj9snpeHjm+/7pROAuXiKkT6din3o6EnqpQyTwz2cOvcKhWKL8X4LJsc4s9evMzzkwuWeoNJKMz9/lX/wiX+JUmxKGZ448QanT7/GLTfvY3Vtgbn5s/RE/dJDgRvE11SyKPkRf7+FSrrFYM8mHnn0k3z1ic/J9R95+BPk8wusLC7r7fmrGaHYGS1hOqj0clqSTvMLC3i9YQaHQvjdEUbHN4qHb3eo4pEfs6rXVxalF4IMzcTEBt2/e2u8owlYzxQplqu8+sqzdFB9Ai3SCm10+Cb6e2L43HZUIesf/dH/hdnQEgjZ7bEQiigVqWruCtKleqy/H6/XK1mtRKrB5HiMnuEfN23+pZME1Wev2GZh6iVm5q5RK+foiYS57a6HeOnoBd7/4AGq5Rx2p4/FpTlKZQObN+oqVdnY5577Dp/8xG+8/Vh//Ce/T8+NZtB0LCwszElLWdVAMzTgo5YzccsOvUNboZSUxM6HH/4Uzx18Ao/DTrOSlgyqajPXaGq0OibpP+B22Wm06uRzeil4/+Ambt//IJ22Kl7V0G7UeNZVOxijkfPnzoqAKv9gYGiIV14+JOBQs5V/Zx9gfjUjYcnLr3wHt9vHYx/4qHj/amTLUL/RCfTwoSe5fu0UVqtyEjX6+sNS6qy+TsXQaTEyECLkD5BXqJrNgd/nJTqsP/Av4yjdwAIunzrCqQuvMRAOsWPXvfh7x2hWa5hvdOVUqvOJb3yeoSHVM0mjkE2xZfNGnO4gA706y/mtr8v50Qt/SWItL9/Usb6qyriKuJwOrG4jCld2O3vkiyo6bRvbtm3m7IXjNGtFekMRYTArSp7y3Fuq+aM9IF/QoQpxd950u8DrgUAEt9uLw/1jytrb77bbFbLv2mqc+EpcNEqprAuNAqpEABbiqa4Ca/7mWFhX/WSMzM5dIuxzs2liy9ubsyUoV1UBpGr+FOfSpVPMz1/GHzASiQawWawY2npbtGjQxVB/D+evzBIOBeiLjRAb/rv5XqC/jUC9duI45Uqb+++9k4sXr7I8d5SFxWU+80//rbByvvnEXzI+uhGfz8HElpt5/tDzrMTnyJTWcdvd0hQyFBriwQM/yS2Ynr3MydMHpfFWYj0t3VdqFU0aZhVrDaHGqRBOjd7oVvoGe7ioOJVm2DgSo6Ho22azLEaVTFPxO9ioVuHDj39ShEx1cPlZo1LKk0rqDUAVMqkqtNVQXVHUotZmFxPdvphfEjdvDeXpKwdCOYNux083gq42IJVVHS7nefno01htHUymLn6fE4fNiuHGN2i4bF02jI8wu1xicHADLpeHSOzvnv378wrD97/5NXbf9TArq4u0KkVeP/0iWrnJxz71zzhz5jjXr51n59Yd3HHPh1Cd144df5bLUxcoVqp0qJNJ5xgZHudDD36c/r4Y167OEAqHSWXmWVw9Jpm4uZk02UwFi8lEf2+QhcV1IXGoBlImo1tqLB758K/y7MHvEgg4pE2PSqOriinFL9i05W4xO2pMjG/BH4gRj88JV/PdGmisxhdZX4uzvr4qKKJalCq0V0PhmyIALxw61FV5b7PJzPBQv7RRea9DfQvYt77zBNVajnqtgM9rERRLJY0sRjP9Mb/YyGzJK9UoqitYT+8vXw7gW1//IrtuvZdL509gMJTIN0K0K6qRRYN6u4Db6sbrCzA5uY0fHfoO5VJZGmNY7V0JiV02CwN9W9m27RYqxbKkw1V/YEXhGhvbytMH/5KzZy7jtPsI+lTvxQ4enx1VrW7UnMRi2/BLTWCMv/jLP8ZmbdEnKWPVUk/v6/vhD39MWsdLS1a6vPbac+zbp2duf1YHlalLZ4X2rmoC1HcTXbx4Wjq0qZyP6gmlvXbiza6yC4MDI0xO/M+/sEnFzAo1q9dUo0HV5myeS5cvsLx8TTqIdFXbEYNGJBgg7HWwdfMYZmuUalMVoCpV+ZOg03sVtr/L/Q4dfp7JjbeyPPMal6cvS09Ej9ONL+TgzKnXuG3fvSzOzkueY0mFdOqLIvJ5yoWCdAVX4aLR4CAUDBKLRLg4dRV/0IvN6uLRxz/K5/7rvxIkVTXFUN+pZtJawvcr10w1+QAAIABJREFU1yp4vCHC4X75LgNlDlQ7O6NWw+Mx4HZHKJSKUt71G7+hgz2qI5vKSIaCPy4g1XkF795G59qV89KAQrXPm5+flnCw1S1KmznthReOCBCkGKm7dutly70hvefsOw1V+6DGpYtTUgKVzsRJJOIy+Y2Gap9WEJJI0Ocm5PPSG/XRaivGiwunr4fIDSDi73JCf95zqwRLKRMnX87S7mpS+p1NZxmdHMBqDFAqpMXDbrXN3H/fAY4df1nSsNMzM+TTaRyeALFoDx3VhiW+gMXg4FOf+bTcxg+e+Rq9gwYstg6ZTBeD4gMWNCoN1e5d5fLVTwuRkF55ZLcZyKaWZKGoNrLKe08k16XC+r77HuLFQ89y4L5H3u4hoHoNqcTV8I1uqf/jsyvH78Txl8jnc4Lr5LJ6SKgEQDXB1A6/elwEYM+tu3GYjayky/QG310A9M5z8KODL2BU3LdsXFqs222qE3YZTVKPXWKRKE6riZDfI2CEZrCydedPc9Z/3sn6u9j/L/7ij/H8f+19WXRd5ZXmd+65w7nzqCvJmizJ8mxjbBMwEKgiJKmkEtJUV5KqSpc7Q6/OSg8PvVYe+qWeuh9q9cqqlXS6kko6FQjpEAhhSCCEIiRAKKYY8IRtedA8Xt15Oueeudfe/5Vkg7FFYhwD3jwgWUdX955/n/3vf+9vf1/Eh8GhbXjhwNOolivYtGEHo6EyGT/X09WWhD/7yB148flfwaLP6DrM9N00HJ6LoH682phDb1c30t1Rrr3Hwn5G/Xp8SVhuFVMzJ5FO9/IDUiyoPJ9ISCzH9CFE2gLM5EF6PwJNTc80RVtKyUjjh879NEg7MLAZ27duZT7CTKoTErV436DXtHyfKIG85wffFGAUbgzVmKVUWAvSxPQ0O0C2S5QYyQkuZGKOGHjkoYcY3mQ5TfjkAjN+EJUqgSioWpVJZaF4FaQTKfiUJCRZwcjI2yN8eicW+3yv+cO7/wE96/qQK04zGicT34rDR1/B1pFuzJaKrL0XDRMdO6mREBNIFZbr44ZRwG8j4I0g5NOR7cggEvQiHA3x5HGxYsJwovDILVRqoyy9R+Fa0w2E/AQXy6Ejm4ZtkxKYYBEnoini9AWq8Mti1IvuMT1A5ZKGdKoXS4VZhIMhfPRDn0JXW6/gjZ+LqGxp8YmZ9dGf/QjFcg51laINQEzhZKQm+pa9gPPdKKEDQkDRGTiOB9VKGQcPPonOToWxa64tThKhoMJnfgIqpaJ0xJSwbZfQx7kS7Zknf4KaVsd8oYz8TBGDG4MY7N2K01MLKOQK2LhpAKfHJmG0DPT1BOGYRAblQzIW4nN2tdbCn/7JJ5ErFLhD2pX1YqmoY2a+gLomwe/TeVSe+r9qq4pMWvSf0x1JHhGjhhBt4YYuY2RAHJOnJg8yLIzIKhtNwhK68BHjekMUe/y+EFLpXqaj9fk9otiWn0G5UOKkb6B3E4NayIgLeWpKHP+UcApWW9aWuB3elgM8f+AYU8i98OLDSERD3ANwpSZ3AOPRJBcaSLSBZEkIP0ggxq7OfiTinUilRIHkSrSf/fwHPGWTjpI+XzdyJRr4dKAEXERDcRw6egz1ag3X7xqBVxnAiZMHUWmUkIzRNJQfu669FvVyDdGIeIrDoQAi4TQmp8ZQKC/h5HgeSpiQOnHUawS/9yA76EcyGcXCIs1jJBhLqKoW+rtElCTRp3KZCDQF+QMBP7s7O5HLzcAkWlmNWMSjiCYcSD4hBklhvlokFVMJiUgnIiGSsRNyOMRslkimuQKotqVziK18zQ5A+9DzvzvMDnDgwGNIxAR5sV/REQ0rSMU7mIZsmZwx3TmCSDiBTRsEzu1Ktl8//j0ooSQM2MiXywh5bOYNJA0FGhhdmpvBDX/6aTi2B7nZk+ge2M6sXdVKAa+++izmZqbx2b/5DzxXyGYsYX5+FsRLIPvimJ8/gaauwxfyIpoIodxYYHgZ8ShRR7VEZNL5AlLxGBRZRIeFxQq+sv/v+CEjDQLCaUxNHWcthEpVtN43bN6GmjqN+cI0JMfDB0S1SnFaQkgOMYWvLJvM07yQm2MG12Zb+Ip+/205QFUDnvr1owyTohmCllZBJpmEEjRZG5iQtMRz5/OGUW042Lp5NzZs2HIlr/vKe3vlpV9h7w2XfouqFOe49JorjkPVc7DRRCzhw9xCE+NTU9i6eSuKZVGl80g0v9+C1tRQKlq4bd+dyGa60d3dg0IhD9+ybtPsOCYmx9E70M9JY8tZwqmJMwgFosxd7HHDRFbC9uk7/x2e++0v0FTrDO0jNXJ4hYOlkkILaU0RgF6PlEbrKvCbZ+5mbT1TJ+kVIBwk8kgP0qku5qkldkxCAkeiA7huzw1XvAMQkyhJw77TdmT0FzDtaXi9CRTKRCJls3af1qpzfd80WoxEJgl4IsqWnRDWZYaQiHUgHErgur0380jZ5PQok13nq2dQVRcZCdTSLeZasAwJuuZhtbC/+ssv8Ud6+ulHGIJu2RYPo1IziiybFWJXF3UAOvcTRqBU0jA9cwzHR59mrB8pg9D+QsUOevpVlUQI6qyMQZp0t9x8J1KJ8Iqg9Dt9g3/f19+/fz/uueee3/fX1/x7DW0MplVEMvoBjM28jKMnhbRdJBqB10t6fzrKZXHGqjdVECOczcOm/fg3f/4fmQeA7NSZw2i2KlgqjaPcmOeiDm3PlDPUSiL8R5UkPvuXX8KJ48RbvMCDIXNzQusgXxT8gp2d4tR3UQegci9RwdXqBh574gf8S6EAKWZo2DDci3U91yAWzzL8Kx6mM+w7Q/Oy5jv9Ni+8XA5w9tuiZPm+x76BpfIo+rp7GCNJtLJN1UChIJhUTNtEvVpET/IaXL/3w8wgSs7S0usoVBYwPnMUNbUETW0LRHn9qJU1eNwAC3Td8fHP8twiqZMRg3m5rSDqWsT7EGI6OpMGSN8KEnb2GybodMsEHnj4+1jX2csjToP9W3hAYSF3FLfs+/O3eduvnMv/GA5w5OSLOHzqN3w8o8elpysL29FRqtTRatFxrogwYdNcCY7uRUeiD8l4F/bdfCPXC5575QG+gVWtCdv08FZskTAV/WeRHiqdAhIrCmfEDlYoiNAfCvQimxXdX72lr80Byk0Xrx8/CMtosmaAQa1Erx+7d+xDJi3kyN6tJhzgn9pNFlEBPXbit9i2RWDrL7WRRvO/vPh9LFDNn/QF/UEM9PTDsFUmk8gXRQSgjqzHFdE0IMeRjQ9geKSfeQteH3sauuNyFCaTSSTSbCLYPoVwsalRxvDAtTh18hiPm5Wrgicg4M3wnAAZaSqtKQLM52t4/dgr/EvlmuCj6+7ejluuE1y072ZbjgCFpRzCUQXBoDjKPfWru5jE4tZbP4lAm1/3UnzOo2eewYlTB9AyJKjNMjqSGcQSRNHvoFAl3EBVsHcQb4AlFp8Er0i1hRjGLVc8ybJPRqHS5CMemc9LY2HUO/BAbdQhS350dgxgemqMHaBebwn+ZWm5mA+Ws7uoA5BGIMm+0hM/MX2GRZLJejJvn8b9UtzAS/0al2ILePnlX6KjY6Ddm7/4O/zRz/8ehWKO+/HDg30IhxVoeh227WJ6tsD1lJbWYtIJeqrjkSQCQR9Mw0Y4Rswj4m/kS3U+ZtKAKZUSyQHItIYNRY7xoGgykcKhIyLhzERSLGhFZtgu1LVsATVxPRzbweTMGCt2vlcWnz7X+RwgNzeKzp7NmJ0eR2//haeYLFtDszmBpaUT6O/7qJCMvYA99sw/QDckJnmkUB1SZMTjCid9pXIDpbJI6ghz4JXC6MykEY2GuJVuUcpPLRy7DEOn/oHJXEbBQIw7lsum1ZuIBru4H0MJQqNZgKtLTIl3tgMQDf1bRgDK/Buq6P0eev13iMez2Li+H9HwhSFIF/f/K+uKsx3g1PFD2Lh1uWFFN5sqIG9GRP2+n+CJ577H5Nhay4KqiT1Z8UcYOVysNJktjQQiafFJiT2qpDAyPMBTV2T1poNKQyx0LJJkBTDTJMEqmXsENElMkxe0nTgGEVYlYdo1bgBJhsALEJ6QTVJQKlykFExOYNgOfvfqq7jtxiu/qPP7LMzZDvDU44/hlg99EJq2hHhiBIX8DDIdl6ZIdOjkI6jWKihVF1AqiURveKCfx+dKJADNsvail0BkVI5pIqhEuJ6/fjDL9Xta7EqjjHgbAEpNHeIWkkFRR2IuYuIFqperiLbHvwKKxRPONORK9DTUISTWNhoXr5aoff8Wk0H0RqisQP7y3Asv4ea9u5mR8r1m5zjAE4/h9j/7RPtT23jwgfvwJx+6EemU4DI6fPxhSAigVJnDYP9eDPReu6bbQcezUm0Kx07/KyamBTPoUH8/An6Ce2uwbA8Wcg00tQosi5DApE0sIxZOIxwW9zzbFUS5Rk5gI51Ms1KI43iZVsaxgsy+Tg5AjztR3g71b4HiF9jORlWDrtFEl45iWTgbcQWSHP1Fk8A1fcJ38UXsAHd9Ez+590F85m+/eNYnoRAdx30/+RZ27OpBJrkXnR3r8eiT/8g4iGx2HXZu/SBCgYt3OQ2zjn995T4uk88uTkI3asyluHlkAzySyd+Xqh6YtKBlQh8F0FI1+H1hJu7csWMjdLOMhkqDNx6hdO46TNVD/MK1KvEHiK3ZVG2eMejNrkcoJEQj1ZrYyomx3OsTfEUkR084zasO0C4Fv/z8M7j+pnPpaufmDuBX/3IAW66twnZCGBujuf9BqEYe3V09uP3mv+IZPxLBeLNR7CTlbi9aRg5LxQWcnHgRC0viGLe+dxhwhZi27A+jRXxMmiHEo9rNHMeMYnhoG4IhGbolmFiaagN+JYBGUxXNI01GLBhjHCYcCfVKA4osZgTW9/eiq03EQVI+lDDSfMKyA3R2Dlx1gOUt4J5//ifs/9Iya7lYzsLSJI4fPwDIDWzY0MEZOolfVWs2y+F89PbPQ8L5Fn/ZHYhj0YdybYzVVo6feokxh4waprO+N8BcvhTWc0Wh0iZ5bCzMlpAJj3C3b3hwI5RQEZbrYC4n6vlev8y0MbW6mCkI+uKATcUfjaFpsWCGKexIKXV4425MTxzjU8fpMcEoQsjterXBwplXI8D+/fif/+O/48zp13Hb7Z8550FuNqoIR+KstO33VzA3dxrHjo8iGu1FuiOMTRv3QZbWUglt4MzkS5jNHWqPekkIBhSE2mG7Wq9jfpFmLGVUijXYrSA7x03X39xe8AVMzgtET6lKRFxdyKR7UaoUWexTcoKAQVBvwmD2oDu7nrUeiRjSaDV44pjmDJ597nHuKJI+ERnVHa46wP792LN7O/bd1IOOdA8Gh87PWv7E4z8kDQ7EEgoGBragt4f0EilWr43r+PTE3WjqRJzt8sg8dVJpwULhEManRHjP5UTRRas7rD/Q3ZVFLKZgcDCNlrnIYX92YQ7hMKmYUvWPWFxEVHB1l/katg7fgi1b9iBElH4ePwr5eSwtTjF1/VNP/oyFKVmqr5zn6HXVAfbvxw037MHIJi8G+rYhFgsjm93N1K6GVme+XxJqkqHD4xdgylaL1D7f/oDLs6/8b26PBwMhRgan0kGUKyqK5RqKZR0NVYUHXpaip+4gYSyv27OVgbZE0FVXcxibEmggEoROp/2wzCqWFoTjbOzZjY50HzKpbkZrZXsGcPzoi+hf38+fh6aMn39RiH9QH0BTrasOQDnAvn17MTTi4MO3fw4e2rNL80imtvJ0sBI+z9DlORvF2r958bX/yxf7FYmHSlJZwaUwPVfmPKBWb8C2XJg6EW8QCNSDoJ8ANzKGh7NomRZahomFpXlOPk0jwEUfIuRyDQkk/PHvP/ef+TVlj2hsnTl1GKfHD8IwVAbuUuOIFp+M2N2vRoD9+/Ffv/xFbN81hBxBplwbQSXBVGuq3kJQia4cnda+1OdeuVR6DcXSGTiuDdNsMhNoOiPq9nN5URSan68w2xqhfQWujxTAgY4UjdV5kEyFWOpmam4O5ZoKxyVhbAl+b5QHWPO5KhSvaGT99Z1fRiotIOWU/E2cPIpXjz3O3+utuuAmVNLwyaGrDkAR4L/8py9j797tsKwmDrz2IrZu2YmZyRJyi4v48Mc+ddF1r1TmkEhceOStoc6g2ngJTU1Q7/ro6NfSkcsXOfTbtjieVcqi1KtrLkv1WEYD/b1d2LKFqHWbmF+a4XIudfiqdReqZsBp8wAWizWkgr3YNLAXG0cEFS+Ng5F6yMsHSbkVaLRKPLATjQgHuRoB9u/HD+76J0jUUSOBpod/iHpDKITHY2Hm1b/jzs/gB9/7Nvr6+3HbR84FvyzOv4ZGs4ENIxfHD5ju6ygUj6FSo2SQCB/o7zRQqYl9vdju5/i9FvKFMkLBGNf7OzOi/ZeIrUN3j8AIkLTP1LwQnKg1bMwviLn/GNH4jHwAQSWEbdv3wB8g2luRZD786F3M4uI64jVI//GqA5yFCXziFw8zPp9oYUjpmyhgqFxK2sDd61JoNFrIdvRh567fn+jy9NSPWSBiqdhCX18c9aYO1zFhOwYsQ0GjVoJhm7zwlLg5th8hxQfJ8WNdVy9CwTTWdfWjoR9hpfKWrmF2SWA0FnI6LBXYvP4D6OjoRCbTwclgOJLBM889gqOjL7W3ARXxaAqK76oDcDv48/v3o1xaYLr7el3H1m2bMDs7j81bBrlrR6qqqjYL25IRi8bR2flmdc+L7hPtC2yM48DBZ3jhiY8pnQ5gZq6IoOxFzzqF6VyrdQuNhg6vLIPwHgZpEMCDbGYIwYAPmzbt5MGbUDiLQ0fvY7KHYqOCYrnOOoW0I9y67xMYHNjBiGOyu3/ydbHFVOagNRrimHh1CxB4gI9/+COMp+vt60Bvbx8GN2yDYRTRqJXhDxAYU0zu0lNZLB5CNvuHOMBxjI4dxMJigyFgVMsnkY2w1w+/34tUh8LDHPliu37fIMGtMBQ/CUvMQpHTMHQSgN6C3XtugaZO4YUjv+ZppnIxx91ACv9d6WHky0VOAm+/+W/w9AuPol4X6i+WbTCOMJ1JXd0CyAHu/MTHcM2ubTx3v3Xnbr5JWnMcwTCxbV16bsNS5QyOnHqa4fa1WpNbv4m4jFgsDZ9PYvIG3aqxZkC54kW5mkdnfDs7AI2VUWS44+Nf5cUldbIX2rrOqrqAQjnPLV/e45Uu5PMLK8EpQEml6fBpJJGMs4Tf1RygnQPMz4xjXd87z2F8+MRPIXlczOVICJpgXyT2TDzLNqIEqw+SdJ3oL1RrKiN7ZU8EC/NtAEnAh5CS4V5EIpzA1p3XYnzyKJaKk6i3iFfQw3wDpQqJRNgs6E1DKBJrm1H1MMnDKPEk8TYYVx3gUmAC17r/E707cfQcPnkCNgj0Qc0bhRO1liZOAn3rxMBGm5QN0eAQaxBs2vhBHDr8K1imxScUj1dmB6BoEY17sVhcRLm6yGolHh4ikeBxxT5PRNLUfiYiaYpouXKOK5LF+hKkcqXkkkbd+9UulwOQiEapegonx09CaxpoGYKpIxGJwnF8sEhtio9mcXh9RO/WRCqTQjTUj6HBvSw/s8wF9NST96Cra4C7feGIKBotFqZQqVWhhMQ4fjiURK2xxDoDqWgHC0YVSkvQDVGHaKgNjM2fhPTqod+5u6+5jmVgRoaGV0aQ3i8OcbkcgO7nU898h29rvdFgsseWQbV/j9D+NW2eCyRx7UAgxjh/XbeQjGfxgRtIYEvU+yuVKkqFaVi2ilhYAEjHZsU+PzY9xv/f2WZj7Uh38swgSdKOjIjOItl37v07HuUnk5597nlXCQaYlpxs6ybqcr1/7HI4QKOZQ7UyjqViDkWq9jh+EFbPtjSmfSUL+EJcovUHZGZYIY7gaDSCSpFqAhr2XPdJTEy8vrIwunaSu5F+Xwu6biJXMrFYqmNk/fXw+xTmDd6y6Rq+PhYTvM2ETSC7/+d/z//PVyrCAeibzZtGkO0QI8PvJ7scDqBpJczNvYDZWTEKHouK+1xvzMCmIxkjL0m72UA0piDTEUEw5GW6PQrzVPDRNAmZzAY0KqL9SxaJTbG2o+UGUcyb0MwM6zP1rRN8zmRj06/iEx/5Cp596QE+xuYWl+DxOXDbrGLS9My829crRoXfj3ZZHKBxAg4Iuz+HSm0BSjDEyuNErqW1SGkcSMQFVm9qRjjJlq19kL1iktfQvdB1LzTVj1jEgdmKIZZY5LmCSnMGwUAPHIja/9zCPCSfhkgwi0ZdlId1j4gyyWAWi3N5wONC8roIRp2rp4DL4QDLD1aldAqJFC3UaVYKF9rLKdTq4oi3zN0TUMSCDQ+tws2mx/08Ct7R5WduZsvUUK5P8e9DikKWO4Q8LAFLCgswXWJroaAfgClZuGHHxxFqD6088fT9fJ0ceJskUe/FCHE5HeDc+7fAql7Hjr/GmsFDg9dgcu4AKnmRpff3SQhRkicFAFdmvGBuXkZHN5Avn2C0MFkgsGnlZQmcars0ZRTgxZ9eEkJf1E7+2I1/zY2jU+OH4boWGq08H0mvFoIuE0HEGx+e8YlfYm5+Gq4jwXY9UIIOdFL39NQQ84snmZKzSMxkljDHWcUeVhtHAZfmAQWwIxrazHSy/DuSAdOKQDd9KNdzkCwFcW79Sig1ptGdGUalOc3XEpL4fe0AVCH7whe+gLvuuovZ0S+nkQOQLZVOc3GGxu5Jm8G2/VzPj7SrgdSjaFmLkOUICDwqTIbkiPZ1JCRoZoaGO1AoyHDhYCEnqn6O5MLVBYyNZOhnCkfZGYJKEL72gOlVB2g7wPRUW1MnvRnReG8b8AlUiscQT26E5GnfsQt6SXuSFhe/tt6Yw8LiszzqBanOBJWE8KX6vs8rFrfZFDMEdKSjoyEJWUWC4rieiGUhk+4cgK6eJHRd5BGvnxBMbfWamAUkzQG3LejFjtAeGfNJGg0mvjkC0EsSPy2dEi7vM3E5nz/xt86OAKOjj7PqNiFlMlmqhdCZmRalCQmiauf196/hTdKx7uIjdK8duYc1/GxHzANSexgwoOtVrufTIng8JPBAo2M0+Olh5g8yqhZm0+K9hEJhpNLCYWqNcUxOKyiVm9BUIJYKwNfeTiq1MiyiJ6cxs7hwItm13+wARltDrFSsIqz4uWL1XrWzHeDY8UdZnIEIGLdsuY71dfwB2juJqKEO26rAr2Rg20IBnM7nZKXSImv6wpNlMAmRYl/IXj/+/1Z+bFgOKvVFJoUmIidNzbPML3X7uF2bFH/DI4VgUH5A+31EJHYezyCGh25dea0Tpw/x1zRAOjohoEUEK6cXoghCVcfuODCRs7ChS+ARW2rbAWgUmUaTSPiRbPT0afgDYu8Y6r/47Nu71UHOdoBHHvwWuvtkhAIp9A/sQDgURSCYwfEjL8O0dGzffi0WFuZQqy2hZixyUYVZOLKCYr9eq6G/dwOCwTT8gSi3bc8XCcrlSZw88xjUlsj2q40aAoEEqm36Nq/XhL+dj6RTER7wpGZRRxYI+EUEINWP3r4d8Hq6sJCjKNGEqqkYW2gXiVwvurpEhJAcEw1VAE/72y0f1/XAcmweFpFOnjzthtpS7mqzgqV8ni+mIcJsJwlIBBCNxBCPXPq++B/bcc52gHvv+Ud0dnUgkaHPHwVchUbtmMbFI/sRCsdw7dabcezkq3CgQdXKSET74aceu1XnQc5s53rE4kMYPyMkW4c2LOMEbRw/LoidRBPIQbkuQn+tIfbuVovwgT7IkolIJICOtIyAIkq3qYSoC8SivfD5qHso1qJQIkIJMS9oySJST82eQTSeZNLOgGdVEjIesOAjnIHeFFAzghJLsnAAOiIsLk4j3z43prP9PDmaSWeRSHYgGIqCdgLirnkv2dkO8NqBX6JcrmP7zvVQgqTYSdp8L0A3SLzRRSzWAdNU4fPHkE2uQyhAjOBiYSrVCWYGoRr80MhtKBYm4NoNJm32+xVE4imcOnkQxeooRw7D0mBZHsheL5aWNJiOSPY8UgSRCGEAxZPemV3NJXq7huDz0YyCilxBSMew4+hxlOr0Gg4sx1jZmiiik5yd7BC0THQMw34ZprmcqAJnxmqQfvmLh91yaRF+ZVUjgGTi+gY2sRPUmw1s3iL4fuNR0qh/7zjB2Q5w9OCTOPjKGP7is3fwZ63WxzAze0Z83aBR6gATYvb3CQaR4fVCXOOtrFw4zkyg01NH+JKF+TmWDvRFKEn0oFAghk8xoyf7bHT1eBmtS/Bw4gowNRG2b75hN2xSZpcDyKa7MTlRgNevwhcQR8JCxYeWRTokNlPDU1mZxCHIoqEQPGYNHpnmB3yIkJQv9wfmoBs0o+BA+v73vu4qVFvkDNNFKEwq2SKcBEMxSLKPtedGNu1Y+aydFxCUuOBducJ+eLYD2MYUHrr/18h0ZnDdDXtwZvxZmJa4kQQUJfMrQcQjW6CEvNi88UIwcHriaHJXtGuPHXkekAU0u1QXe3+p5OXTViAstlyyakNhuDiRUtMQp79d7btu9whOHm8DRnrFNNFibYkRRKZrwLA1wF1GEQl6GbIt6xKYzxewsVeATEol8bdHJ2ehtyzMlJvCAchTqXSYSiQheWR4vQrve7TXUCjs7huEPyCQK+s6hSr4e8GOHDmCr33ta/jqV7+KzSMR/PzBp1FvL9CtH90LVSti+7YteOHFpxiJQw8DU7GGR7Bjxy1rUEEXDqCqRUyMExSMGDtMlBu0mLTnR2FaDjSjAcMQC6jbRN9GZ3+R9XuFRhQ6EqtqoLq3BdkjIRn3woBY1KDsZ8VXsv4klY7F66XiIvwbrXaOML+EucUS8iUZdaMsHICw7zRXFo4EueZM3xN9aSicwIaRLWi2RMJC0uSZdJx5a94LdrYD7NxJNXULd3+b5vdcfP4r/w3zsyexrldseYcPvwS1IbbJfTf9xZo+fq12BKoqevgNdRKWKYmBENvmvj/Z9AwpgthotFqwdC8sqb1gBp0GRPnXMeuwXBNhEp0gc110dYqpZNdDMjM0GUysMGm1AAAMhElEQVRVAkEDkw678PskaK3AigPohoP5nHDIak3D6QkR1aQHf3qv6/cFUa+WEFAIjCASD8NysX79CNb1iOOEpmtcjfIFgkgnY8xX09t+E2u6G1fgRec6AOU5FLYV3P3tb2DrTkF1H21n4HPTNYQiUdz4wTvf1idZXLyXr6+UKvAqPkSi7cSx0t4KlryoqQ2UahIiYSKE9KBM0mys9BGADBH6VUM83TQttGwbhzOCVEIiBtC2PjBFA6/Jcr7CQ0KgxTd0UQRaKmkYnxT5heMEIT3zm6f5rECMWLLs8rTpUn4J6QyVQ8FOkMlmoWkqZ62U0Z5t72YneLMDiE/2/K8fRrEgeul3fPbzWJg/gmq5hdnpUdz+sf1rdwD3NHK5KeSXRNauRGixNFSrFhrEAcAFHRd1TWU8QE1bPbYtl3zHJhfgOHV4PDQX4OHOnj8goa9N9x6MUj1CnCLCfhu240HAS8mgA9cNQfYSvpAKSx7kKuaKeNTMlMkTxNIjj/zUbbbVpb1eCYWiACTQ/k9OIHuF51BSEo6sLn6mowsf2LN37TfjCrzyrRyA3uqTP/8xtl5zA3oHLq6l+NYf7SQTPk9NvcR4Pzr/L+WrPNlLrJ9en4SgIkI+Lb0t6cz/v2w+OYMF0htua/8kEzTNG+dagKJEEI+JKGJDhQQ6UdiIKWJrWWyTQycYDgbkijQVHEEyPoh4tBfZzDAeefReSPfffy8LzpAAhEGU4EwmqDITVVBREEtmUCqLYgVZT6+AG5mWB8ND61m+7N1qF3KAS/eZCLvXi6nJB5nIcWa2gHCEZGAEv4/s1REK+lCt64inHE7GDcPFxGmxFvFEAJLHQCTqIhIVx7j8ogddnRQN/PD6bW4KaS0R1mWPxTpDyxZQEowWJvRQODiCHZv/7crPasUZSPfd9yOXRozZB02R7JmmC49MHSiHz4uU/cvtpoJwgmGYloxPfeLK1AFc6+JdHgcQ72ZmalWUwnDKTDcjogIwP+cgQLqLDRn+gIO56dUFvGaX2POpW0iqH5Yuwdve3ikpj3BdQVizrQxOX5MT0OJ7iCCaKo5VA+HARqRTgwgHwqxQQhFJuuee77uWvSo7SkiTVoOqWBI7AXHaky07AS0+GeHSd++68gWhLuQMl9MBxPsQpM2G2US5+jqKRZLdoQWiTE4c+6amhDwPVfGWLZG2uSzcbMu2Eh9oKEjQsDA8HgM+H3X/ROh3XEEEGfBGYTkeTgCJFJr/TdqEkeGdiETiWJwVfQN2ANsmCAKVer1cByDR5HYZmi+qVAscQkj7jrjm9l5/28qbezc7wR/LAZrqLA+EVioNtNrbLtUHLIOO4UT37mVxLrXZhBJyWYiauo+OK8GVxDZARvqDtVoZPi/ByIXHlIotdHeJp55OCKWKwUdIsmY9gA998G+5m5nLjbLzsQPQD0kmlf5INJZFo5znApASJBVsH+r1Ctw2/KirW5Aj9faLSHDVAVaf1LV81Wjey/kVUcWQ5CwJRVGfnqBh9brYEqgQFAzT3ACxgYqnV9OLcB0BJ0+mZRDfIzUN69XlmT+S8VndDro6A6hWxOsRr1BT9bFCKVk62UBv9038tfTd73zLJdFEMqr9Dw4M4cypUyufJRojQShxBu3oFBml1ye88I477sTpsQkMrR94VzaKLmcEICBmpfIaTHsUHo9I2Lzt/bla0xnFozaMlUXy+gSdGwk+8kIxUMxltdBgiBjCiW7egW01sDC/zP+rINuxOuZntCQGnLiOgZYegNqiGo44ObAjxYKQvvn1r3ME8PpkDA2J6djxsTEWKsx0ENzYA1VtIdxmqF7+5R27BHt4Oi3wAvEozbVf+kYRvbmzhdGJTjUSvDjiZuVTsvZuG+H0BoX1y+UAlnUCxaJoLJHJvimxqJIHxQJN/hDARBy3iYQiGFK5bEzTwxQdSEfI31aJILlZMkNXYVmri0lOEI+R0CV1ID2Ix33QdRuldj1D1SmiBBGLilqD30fKbz5I3//ut11NN5HNJhFQgqxhQ1avTXEXiYiLeUrFobl1Fy3di1S6i1udN+y7GQ2GMglLJS6tlsDZi0+FjbraHnCInluKpnN1tSnCXTIiUuRTk2JebqBPRK3ziZm98w4g4GGOnUOhNArXqSFfLEEJWswARlYqVLhqR5Rw/oCXQRq6XuGnn/ZwvSXDdkiST4R6vy8OJSgiNpFDEULJ549D9vj4tcg6MoJabtnGxpc4x+Pfp2JeKMiLTybd/c/fdbnG7Ir9g5zAtsqMHSfTVBOJ5CpXHvWolGCCsQK7dgm69FJFveSLv/Lu219oLR2nJ6Z5fK0rcy53HzmAbYsBq4BXgmqQLKoooXZdQNTqnXcAYuSgldYwNvEianUB1aJg3pGVoGnEE2ygUiqiI0vjYMuy7i5XDMkqZYMfvEpF58VftnBERA4SmSAhb35dyQQc8ZBQb4CIpJoqjZcReKSEZrOFQMCPcFBBZzoB11UgPfCT+1frj67B5WAwmQAlfwWGSy/vP7F4F2PWYrEUdu+5EeFQkI+Hl8OOnCD4NLB5w3oOX+czSqipxUq2/KHadPvnvf6ddoCpqacwMCDa6IeOPsEwM0rmyKjmT2XdcqWMSDiM3GIZ63oiCIfbT6bb5BEu2B7U2+pgzabIvfx+E74ArZX4npI7r7wqGdPSEowt0DRRzqbmU1M1USqtXrNzy2ZYJiA99NBPXRIpItO0CiJhEZvyeXFOJNVqcoJotJuhRMRtu7O9/9PPk4nYO7L3n3eBdQNK4K33/3JNRLFkbG05wh/mAIIOXti5hNHNZh5zcwJm3jLFIlB7lo54tHB0zqcsvlYRNXy+TnN4Wx3eEEehoME0LKzrEtsZnRAaqs1RWZJXt1z6WUgR/EWGOcF/wzaDcB0vi1BQZCRoOS3+sq3vW8V42qYL6dFHH3aJSqRRLwqcGFeNCJq8ugTJFO35JG6Uwbq+PoY5EUx5oH/9BRef9i7DshB8C6WRUqWEVIL07C1W0PpD7fI6AL3b5UGNNzOGj47+WPANGzrz9gQU/8q2mkqJtnJTbTLOgEVbXImfZE4G6XtCHQ36oDaFM9uu2Pep3q+3OkDFILJUSji92iBtoFVJuEJFEEKRtTSxRfT2iC2EWEnILFMXDjA+fhqBgA+EDKLFXzaiH+no7GKAIdWdo4k0OrOdPFIUT8aYZiQSiiMSPj9lumasnkvP5wSvHn515W/t2LEHXs/5KZmOnRrl60YGh98y/POtsV3I8tq3pD8sAqy8dZwZfWrlm6YuMnzqqhoUYxm3JxZP9koM9MhmIyiXRcK2rAnM37gmLCsAvd36pX/q7xZZ/8RMC+u6gNxiAunMaoLnp3qAJEI7K4c6DmoNkb8ZdhGEK6EA39t+Hfp3OnJ6XA9aWgPSN77xv1xKRMiIVjQYCHAZmCyWSMHn9TK2LdNuP9K/kxOMnnkJN+37GGPaw8EQwqEQDMPAgcPPcEWR9qdYNCGg0mdFgGq9hkazzoUn27GhaRoGBoYRi4nE7uwSKH2v6wbOTAmu/G0bLy15BQ2GLtsfIiBNT97rBx9DMBQCOQBN8BCEW9N1JoAm0qdmez9e5ufz+Swu5ND4TShEqiO0MA7z/1DllhYuElF4K1BVH7OBksVDGa789fZKIOQXOQBZuTQlIgk8cNsKJqZT5SSQLBzyIZ2MtOsN4rW8Hh3SAw99w50ar/PiLxs5QSKV5j2JzNeuC2c7BB9uqTLB1SzCCt56s+guUbj7zW9/ikyqE8FQhEGMywniddfcuPLac4sityAHWLYNG9oTri5588o/8xenJwQLJtnI4KVR8Fp+vT/UAajAYmoC6/fbZ3/JT3cwTh06cd+q7bFv+tpFix8mgtbReSUcWg3XApMpjB4Imu9TlDC8PhH+R0fFiYC0GzOJDsSidMwjB2nrCKp53jbU1nITyQPVFgUkMr0lIjSRTAb9IvzXmjpi4QCkRx77P25LlVCtqPArXizOVRCKKBwF4tE0FpZy6O8X4BAy2dcEuG/t5VBFyhYfvu1z/LNnnnuY/2+39zD6+tN3fPGc/b3WqGNhcR6bNmzC5PQkkqk0IhHxBt+INDQtC62WgWgkxH3ys8+2K2/oLb54YwHpfJetzQHoaVl+tTdjIY3mJB/lnvutIGN2/ETtIk4j9abBi0TCzj5/WwiIj4Fi4QYHY1hcrCLgJwieeIeW7aKliQWPxFJwHRcnT5RhmKs50vBAD3q60iwYkS9STV/kDNu234FTEw/zSW1+UYBQfDI9NBIW5tugkRA5oYe1C8n+P6Ilrkgcr4EDAAAAAElFTkSuQmCCiVBORw0KGgoAAAANSUhEUgAAAIAAAACACAYAAADDPmHLAAAAAXNSR0IArs4c6QAAIABJREFUeF6MvQeQJNt1JXayKquyvK/23dM93v2ZP99/GAL8oAMUIlakxFDEakFSQbfCBilqAZIACRILLghSAAgaAdKCAX6SKykkcem5JAwBwn/vxvZ0T3tb3ru00rkvq2awu1LoIT66p7oqKzPfffede+65N7XP/c4HvVA4hu3dBsZDD+u4dG4Ruh6Ql3YPa/LTtbryc2gF4HnAyHKgBXUEtAAiRgj94RCZdBqpVApuQJ8cT9MCiMcN+XevO5Cfg4E5+XskGkXp+Bj5fB7D4RDWaATHteXvM9PFyfsunCggGglh9+AI9VoDISOKSrsP19WgRxKIJeIYmSaqtRaCQQ39gTpfz3HguQ48z4Nrq+N++MMfw4/92I9Pjv3ss8/ive/9aTiOI68FgkH5adsmLFu9FnQ9BEI6dHjQAMT4u67eFwmpn45tw/NcdZ/8z/H3Rmcg98yBBgcBhGOJyXd/xy+eA2gB6MEAgv45uFpo8hYjFIIWUN/F6+HQdXWvTVPdU97v8Xnx30Y47L+uIaCmdDK0/+OzH/NM/0T3DtsI6kFoGi8POH96ToyA/9ra2Zl8KGJEMDTVTekMIZNfyKgLCoTURAcjMTiOi2AwgFD4/gVYIwumZaNUKiOVyqBeryIQ4HeqM4vH4/J31x4hk46BF5xJRuRvZ5dnJ+dwc3VTft+vKoOyXBqlBhceHFdNQKmmDDdqBLCyfBq2ZaFyvA/LGv1/GoBL49N4Yw04w74cw+x3uAQAz8VMcUZdqxZAIKDuleb/lJvlj/7Q8u+RCcdSv488NXn62AA8DwH/fssfPEcmNOjfDy0YhAP1GQ7eDxqI5zn+PKn75so1e9C04OSceG5GWM2HOnf+XZsYNxeD9sIX/sDb2qnKanY9D+WauqFyLp6HqxeWYBghDEcjbO3sgpNPy+TF8e+GEUEkElUnp3sY2RoCoGV6CCbz8vpw2ENIDyISS2I07CMSUcZy9+765OSOy00kYlGYjoOpqSJSsRDCoSA818VULinvW5kvIhjg3ASwtn0g53F3pzo5V1cz0Ov14Gka9JCOarMpN9LxXFw8d1beVznal5+/+Iu/jJs3b+ETn/gE3ve+9+HSpUv4qZ96D2+TDE4uh2Y7COs6Rt0GLMfGVHEG40Wk+ytRC2qy8jmCIbUaHcfDyFKvdXxDMB1lnKFwFC5XN1f75Ggyi/L+qKFWrAsXRlRdu5yLzAnvrFr58pqmwXXVv2Vyed6+QfGec+h6yP+uyccmdioGwAPc26nCth1YjoZGs49KRd1Yju/5rmswjDBMy8LRcVkmvt9tY+nEilir7TgYDgYIavdPbGjRXYYxQhCDvnLFqVRanWgghGBQx+bWNtervNZqDTD0b9TC4rxcVDykltPplbnJuSxPp+T3RquHRqcnN3p9T51rp69uOLei8bB9N3liYR6RWPT/twFwcjPJJKKGMu5+S3kTjkQ8Lj/NgfoebhMyOKm+4XBBybVqwMCmQVgY+F4zFPJdslizBs8LKMOjcehBBILquh+c/PHqlbdxJYsZaPKTg78HtPueIuQbovzNvwdjgxh7eDnmc3//Wc/xt4D13QYsW1nh2vquuoAgXUoQz7z5MsKhkOwz9+7dm9wMGkEsEhR33+31eSZq/x2pG2DRWCx1krFYEtV62/9d3YRGswkvmILmtNFoq7XFGxA3dCSiOsJ6AFEjhMWFBXiejZUZdfPrzR4q9aY61z2FXyr1nmwlXDmOvyo83yXTADhatZL8/MAHfw03b32nB/jlD/z3sCcTpz44NgCr34Jl2YjHYnBtE67jqRvrcdPx4DouWkNlCIl4FIORI/v4UN0GOJ6aKs3HRrY1guavUP59vN/LPffdNLHZeLiu43ul+4ts8kf/M+rfygjGRkQTmUw4vQGNjn8nGLAtaN/8m/9ZjmiarlzQbqkt+3swoGF980Amn4Or+J3veAK2ORRPsLOzK2CN4+rDVxDmah+p/XI8emYAljnCcfc+IPTckVirEQ7CBreS++/v9/qYmc6h1x+iUtqTP8QCHizbwjve9oz8e74QQiwaFkO8u3kgr40NoNXhuXEr4w1XNyrku9NoSF14LBzAcGT+vxjA/yA4YbqYRzypvFXUMOS17Xt31Y3zAZ4YgKtmd+ySGz11MR5ohL7l6crQA3oYLsHoxPn+BxMZUEYwBqHReFYdy3PAyZfh/yQI5KQGfVdPj6AOq74zEFT3+z4cUUYwnnz5G49lDaBtvPJ33s721sRN2I6Le/st2P5+dHhYmbhwfvC7nroIezREs1lHq6Vcezo/g8cfviC/lytqhYUCBBxAcxjCwHRRallqv7JNaH50IashEIA5UgCJI5FQ+GBkjnB0sIuwZ0HXw8hk0njkymWEQgGcXszLzSxVGzg8Lsu+ynNutgfoDmiUmrhifr9p2wjpOtIJBSRpABw//74P/EcY4Ld/81cnLpwGoD6XRLfXxfbaHQFPAc9RUYGtztn2AHvYR9v0xFv2+gOE/W3DCUagB3nzlQFgDBTpEXx8wGNw4okjxriD1xOOqL1fAKn//kDgvtGMwWcwEJQtQQ1et28E9CK6Ds/3hIwK1IbhIeC6CLg8fw/a9utfkC1ge3sTnPwxct08qGFk2jJpR4clWeFEuslEEnM59XU21E3lOH3yNIr5BIaDLrpttV/SzdT7QbiOjcHIRbmtXOT87MLkc+Wa2r8HloXvfdMT2Dkqo9NsyWsvvPoGkjEdRiiAbDaDJx55SF4/u1SQn4fHFRwcl+UGrMs2oGG/pLYF+PshgRuH1VfbxIXTpwQl//Offd9/ZACf/tRvIGxEEE8kJ0ic12CZJtZvX1fH9SfEGg7l3gz6PXn5uKO8ocRZvpsPhtV2FdR1mQxOtKxoCUt94OYbRYjhnY8fdN+A+Fmb4NLzMN6maQQ8f3ogXiKPQzzFMTYDGkUwFJos6u/YAniKnHzfaMQA+GHu4ddv3laTMbQEEB7XO2IUPGHXeSC+AfD4w2fkvTu7ezL5iwuLGA1bEgPblglrqLwDx16li2iMgPB+TG+a6oadO72AuZkpNJvdyQm/+OoNnD+7grmZIj72O3+ARy+dxM7+Pt75zJvlM2dPzkP34+4XX3lDXru+fiw/a76Rjbeukanc8mjQlfO6dO60RBaRMPD4W945iQKe/9rnoQUY1gURiysvRCBK97v6xiuTawn7sfn4bvQ6ylh3mip6moTBXF16VBZBKBJBKBicRBi2f+18PxdVSGdsrzyTHoooT+F6cBgleJ7wGDJHEum5stDl3TwJ/0S0IMN1Ff8zMuMY8xfjk6eZkMNQ16YMS/v3//unvYceUu6be+316zfR6aqbVq3XMbAgoRvH3PxJ2LYlK4Tj3d/zlPys1BSwkxugmSgUihgNB9jevAkw7tVs1HoeXIRheimUDtfw6COPI5dViH5qakoWTiIdRb8/QigQEgvn2Ng+wsHeDmq1CpYWZhHQFEi9eHZJfj7/8mtyITc3K3Jz2n1GMh56vS6SybSsIMe1oHm8YBuPPqy8yKDTxrWn3jExgC/97f+JSFRNfMLf/5Wb1bD6xsvyOidRXiEuch1w8gPhCMqNDnoWz8u9bwC6mgQ9HBZvwlCSw3bpVQNyHwWsAAgbKlYXAsfnUcZE2Hjyx/eXC5WnNV7VdOnBYAiu5iAcNsRTj4cQXzRizwO/ncGFNtkuuHX1oL3w5X8nZxGPhpBKJrG/t4N7W4eyV9Tqym22uwMsnTiJWCwm6JbjzY+fl5+FdEYsrdnqYeWkQtr9bg+2PcK9u69NTsZ0Q1i9t6fQsBaSG/AD3/f9iIQdZDIpJBNx4AHU61l9WKaBIJSLHfmRVq+rfjFCykgPjkrY2z8UT7W6XUGrO4JpMQpwBazKDQ4HJVIJ6WQsIzhzagXNWhUPP/mdBhBLKuAVi8bEZY/d5Naq8ozWaChGxhvNMRioVb9dVtuOeJCgIr0Y53NN0gBikftbJTkJDnpKTg7BtngrQf4+WOaqdxXL+qABjMM5EkeT38efYZjtLygen2CSkYl8l+NCf8CB0wic/4cMk3OmAeihELgHmf0WXNuSm/fcyzfkDfF4SiyUF63pYZTL23jX937fZGJnCkmEw7pY72Boo9Eoy2fGlnZv7RYsT13Y7bVt+XlcVkCR410/8C7UanX8wPe/Ha+/dh1nz55UNzkSFTKHBjvs9hCOxCS89EJRsf5erYxoVMXoz73wioR9O6UO9KCOo0oTvX5fSCEO21YeLJtJiRGfObmMTruDSw+/eeIBvvqlv5mAt7gPRB3LRVDX0Go0BQe061WZZN4jYuQhDYJGWGvJthKOqbCNzoDeg6wc9/1YRBlMwAe/luXIBPF6JoTThPnjpCvcwkke44LxDk8gNx48tu9EJmxiIpOUY/NzYmQErD6gZ9RAhnU8aNBiAFH/xMlSj/pdRIwoRpaFV16/M3H3PAHb6aPRUnsejaDfbchewhXV7bZxfHiITF7t8//41Zdx+coVeI4l2KB0eIhGrYJay5V9z5bwyMPs1H1ccO3qeZw9vYJyqYK/+Zu/x/c88xY5Vj7vbxXzK3JT3QDJEh1uj1uPh+deVJ5mt9xHOpXG6saOrCoSNNVafWIAC3NTwosvzM+h3ergoUfeOjGAr/3D30IPR8TgxvtxNByB67nod0kDa6iXj5RBmcqzOAEdpXIZI98guNdzjENATi4NjkPCsJAuxxPXTGzlozbl0hUGGNPYsp34QHY85QE/hhdjeoD0EeTvH2vMZMaTMVn5BI+ybfE7HgB/3Y7vtf76337aWz69ok7SZ+U01xXmj+7y68+/DoexO/dQCUU9VCs1uaBn3vY4BoMhXnruZTz22EWEQzpqbRtHRwrZzy0solypwnNMjNrHeOLpp3Drxk102o5wAD2olVFIh9Ec6XBGI4m54wHecODhqxdxXLVw5cIMbt7awFU/1FxcUDmBUEjd3P3DY+wfHKE1Uu53HFXuHx4gHotja3NNANXK0jxsx8b87BRa7TauPvrMxABefeEb6A99txgMgkxgKKgjZETQqtdAL7l664asWF571AeK2/uKMCMS1/2tgROr8Jo3MYBI9P42YFqKSBqHb+OQjmB7zPPLQYkhxkk1iWyVKQQfSLQ9GO2PDUo+p2vf8b6AZ0sQSP7EJF8zjgL+/NnfEdtJppKYmZtCQFNkBd+ciqdhuja++dwLk8mHD84IEF1Pg+HddymnLpwXFHpwVJcLHLlBpKKKtLhy/j6d+8K3r8sqaAbvZ8RCRL9M4Bzt4uSJZZ/X0JDyV1U45GBpoYh6syvb1cgK4c1PXMTeUVkZwf4uet0ujPSCfOdUPoVvvraFH/mhZ/DxT/0bec/Ksgo/c+kYeoMOLl39ngcM4JuKQtaIGSKSRCF46/ge797qquQyBr0WCX8hVehCGyPyKS0UZhfQadbgcd+27QniJ57gYJKLxsebLZS7j08miSACfx/tP8j1j8kewQnjxBMTPtp9hu8BXCffpWnexIvJQvEjDMscwPQBvbyRW8xf/cnveQqFaiB/PDsO8knTNjqYnZsRt0V27rWXVcglgFcAWA+MDovZ+/uSYURl3xuammwndKuXz80KjZyKKSxghAysrm+KEW5UbQFO43i6123BHA1wauUUuFcmohFx9wRLiXhQbjpvSiSmEk0ry0VUqw006iXs7R1MuJYf+IHvx8kLF+U9ph+jbx8o5nB1/Y6kii9evQ8CX33hm+j7ezojmHQqi2g0hoEQS8DhgXL/rSa9XwCuNUTXBnzmGGkfgHU6KiIaWTxGWrxAOBxGkC5b0ybgVCb7gZkbs33/qckXUsvfRrg4aZgPxvacuzEoVL5HbTmCV3wgSA9gmn6iz8cEzU4b2pf+7LPiAfiaHmSSJoCArtgrnh/DqHQug4hhSJj44os+IaJsAFslBbRmMhoqdeW6l5cWMJWfx927t+Tf586dw0PnTwghZA4UhiCjd/32Noa2hsqYQR6pFGcoouhnczTExTMnsb15AD0SQr3ZBjzuvy5OzE/Je3pdNUHVinLFsNX5/PCP/FeIJVMozk7Lvx1bR6/dx3G5ivVtRTMvrlyaeICXX/w2Dg9UyjseT2B+dknyOkYkIZ4lm1fkU7ValfvSqFfQajYwHKqbyokZmhYs20ar06V4AI4TQLGQmQA9Jn0Yqw9H/AzzCOOY3Ncg+OuIU8gtiJP6IGFMw+M2wJ9Bf1UzJCUA5bAcFSGNs5RBneG08ja2ZUMP3Gdcmy0fA7z81T/zCNAc20I0lkSr0xMAuDg/LZPf6alJnZ8/gWg0JEj/1ddukHBWN5zofkORMAnDxrlzV1DIF+FZI0SjBpaXVGjIsbSYwe7WLhxTGQHHnbVDVNvqyjv9IZ5+8iGcP78Cz7Kxt6uOu7G5h0azg+FwhC6JHW8kjOSZU/OoVWqYnUmi3elib/cA99Z2EY0EkS3O4J/84DswND2cu3wW0MLMTEnsbbkqVr6zU76/Bbz4DclbcHA/T6ZUSEgD4GBoSSZjNBzKTxrJpbNL+MKXvoJsKoY767tiABzVRlORNoAYAIfCeET4FgLBCExLGapgAZe5FwUCidzHJNZ4ZUuyybcEoXR9wMgJHvte0xqKt6GB0AD4Nw6LRmlZEyNR4E9dpxjk3/7JJ8UOacF7xw3E4lHowSAOD3cF+Jw+tYJwOCqooFCcEg8xHPZx444SZIxNtGcG0e/38cjVKyjmY2i0hrhwWk2+ZbpYWsqpDJTnYNDroLSnMoo373KSNbT9FO4PvvvdiDBhM7SRmM7ieLuCmeV5dI4VsPzWC69jc2NDfr94XpFB0YANR3NxeHgs++trr9zGwokFibPf/rY3yXto0IlUAmEjDCeoEj231r/TAMjQSe5cyKCcysrpCqhyQpmf4CgU1fYznY1KKPriC8/Lv195+SUMRzZGVAXpMQGA/D6OcDgAsp8keOiWbZe8hI5Bfyg5hkiEEYijtgp/jBNDBIv8jxQvB+d/TBhR3CKkkj/4nfI+2rrL1e9nZS1rgmccj2Gs+owYwEG5ie7Akrw+3U6r20YyqvZdGsGlC5dkP4mEQ3LBDEe6nTbWttQK1RxlzV0nhSC6eNNT6qZznJhNyjEy2Tzurm9hcSYrAOpwZw1jJdIr17dVXjwQFFLpR/+bH1EfJjmEMNBrYmAFYWg2ut0OXn7h28gVZ8UoiTcGbUVYdQfKW61uHaDesjAymbJVh/q+77qCqWIW5y6cAaLKAK7fum8AX/viXyCZSAmPzxWWSGYlnq7WW2g2W/L7zOy0GMXZs6cRCrjCrCWSykOQ8OHY3NzE1772jyjXe0ik1OpXNymAvp89HVPBY9aOxwn5n/8PRSIkuERi5q96HoogmINGMXggAxv3cwjyd//CR75XajXq8hmTPA/TwAFX4IIYwNdf8VczL8SgmxrA0HUYzOjBkQTJo489pq5DJdqEqSqV62i0+9DcEfqdY7iOhWDyBLrdIRanxulMD4yAYokUTp1R7GGvcSSeJBrWJDVr2y6+9u03kJtSUqv3/NP/Ei+99JpsQXNz01g+sSivN5o9CTtr5SOcfexNMCRCUPueXW1gf38H5dIh7mwcyUStbh0LOeMMTZxdVpjh6sUFLC8vITm3DFebwY//+I+DesDPfOpfiwGEIiqBQ9VSqaJu2vbGuuCXk6dPSSq6MDMvNziXjqPZ7mN5aVa2TkZOTPz8+Z//Kbq9Ifp+rjudnYXpOhjZ5gSRE3Az5nf9aEBNrFrhrq8SGgM4brcq9RxA0H8PtwzZ2x0Sd46Ei4wMKBl7UFvQbDTEQ9Cou11/6xVQ6YPF3/7IL3jHtR50n1XjCYRpkZpyETEDSMZ9Td7FSxOUzT2pVKrIew737sjki+jRBQ4bKvS5dnFRtguObCaGbt8VL8OxcmIekXAQIR+YfP6LX0ckGkelpfbCS+dVrL9zUMaVi6fE8ubnFpBMJhCLhJCZXUS6wElVBuB2e3CHJtZWX8fLN7bEAG5vHkkEQYIm5YtSQwEHj14+g8vnTyA+/9TEAH734x9BMpUTVzwYmuj1BiiXVIhpjQaC8tPptOAaJm9CkSgymax4oWQyLquUotbxnliplHH9+i3EYxHcursn5BBdPClyStYklH0gJCPnwolU2j4Ntqt2d94tLkYOspwUuxAjcDultx7jFsexxbPyPcQBlMJRwzGiBnM0lHmYJI0fpIXf+xM/Ia8X52cl/Ir5mSS+plkdmSiOoGbBiITli6PJnEixKOBg+pcXfbS/Cj9ikvebjlpJC7M5qFQE90EdrXYPPV8m89anLqJVa8AZqZX2xuo2inOn0HMUxVurVsUQSYBFIkEBgWPq9OL5k/ied7wdwXAIiUIRnjkUQLT68guS2dw56mB1W1HOWjg6kW1FAkNYlroDP/e+X54YwId/5Zcwt7AgN65eb6DVaqPvZ/oodFlYUDxGNBpGJJ6WdGvad/G88dwWyc7Jvh9iAiiAXr+Hg/193L2jcgk9h0khhrBJWZXNRlVi+6kpFalk02mUKzX0hkO5Btu7L+FNJhJwLdvPWFJ16UvBGCl4LkJkV5nh88EnwSY9aL/Xk8SYGNADMjH+m3/XaABLZ84qi7JGYgTjcWZ5Fp7L8EFx3sQEXjCObrcrwk1Ks9XrFqwhka+L/aOmCCM4RkM6Gg0LM0VF/DQbCBsxuEFjokw9s5CD0y9hcVmll2+u7aDcCyMQtNDveZKHGPXa8NwuAgEdI19fx4WSDJoIBxz80A//oHyW5E+3NxD3+9r1DVRaQ2STBrLT89jar6DbH+DoeAce1B76yd/6xAMG8AG4zNIRqFm2TADFKwR0cExMc/9HQAgzLeALQIMhFKfm0G63kEqmJTLgFkC6mZ6EBk8gdnh4ACOawlf+8SuIRmLiqru9Jrr9Ea49fEXOhZ+hd2y2FI45qtQnmoFxipk5Ep7DeNAIdK72gJ9dlIVKtk95b26H9987+XXyi3AFf/AHn/MIcGglRli5+pRhw/D3GrjKergKt3f2ML+4gs11hRlWTp1EOmWIe+40yxhYQ1i2h2q1I25ubATUQmaiKhtnMNNGYYRmwHbVBVw4NY90XLk1wzDw6htrKA0VX95vK4MkuxUyQhIOkcCIR1PifbLBHt79w+8SVzwQkaiDZCqFr3/1ywhH88jm81g5rdLdBFQbm1uCjJ+7cfc7DOC97/0ZjHo9xGJRRH32kRrHEb1iPI58VtU7dAYjpBM6zJGGKMUqkbgogtPpHEj3Sqzux+sMJ0kCxSI62n6K/WD/AOvrd8TdcxTzOeTyBcFZHN1OR2obhqYyslpb3f9AIHQ/M8gwj5lH6gL4f75v13z5GF1+u6kWJ+d1jCnESwVUvcE4WaR97nPPekzRckhGy+sLutWDDDuYSRph/+AYRJP8rq0NZVVnzy1IDv/s2bMqVJIsioa9/Q0Jv2qNjuABx/ZgWho8x8NsISZ7aCRiCILtjbhCLFw8syAhEWsL+t0WbtzaRt9yUB9Qx6YkzdzX+BlbVoDK0zMdG3FUePjD/8U7YfaHomhiuNqo13Dr+st47PEncPXpt6vr61voDkf4uy/+I44rXfyLn//5iQf4p//1D8mRKWxhZpGTR9Sey+fku7nvEj0zP8BtyPVJF16PHmLtgwoXx1rCsBEXY2ZMns0qQLy/f4CN9XVQEMph2UNEIwZicYWZCOA8JrtcVyTl9ChD08T+cXVSJ0BpuYSnnEhZ/WruRONHI7cIyJUXaTbqImVX1SDKc/A8x2CTeQHt2T/5X5UiyCYA8VBIhNBqlNBpKLYsk85i/6gsTGHf1GEOB4hHWRGjI5PQEDMMESTk5+YmtGOpsg9oERwclsQu+FlyAXRPDz90Wo6rGDQXrZ4y39msIVU/2VRUSIx72xUct3wg5Cqgx6wb/N+DHt20h/bAwdLySQRGyjAzuUXM5FRI9ty3vo3U3Ckszs/i2tWLWJpdkPqHkemiO7Qwc+L8xADe/3M/I/wHdYqJZByZtArvOn0TtqtuHieDySUOTlJgIoO3xUMSC8QTKUmdx+IpBHyiRg8EhKbmPn58rCjlZDKJ46MtWTj83mKxKAYTixFQegj6wo6NvYqShTEKaquJ5RarP1BMovtJvIGPWchQjnMNY0lcIBiWop8Hw0k51tgAptJxjEamXEijtCU69u0tplWpUtHRt+8rTUg/pgwX0xmVLuVIJJKIpDKTfavWbIjY8+C4ijYtcTRCOqvEnE898ZB4C3IJtqvjqKw0hJdOTkm4WMzEBe3eXd/DQVMdv00BKqFFEHjntZOIRFRtwd995VVYwbhkvzLpMDxLbRmXzy3h9TfuoNJyMLc4j3e98+04QVZSNHsabEuDFpmfGMAn/vUvoVbvYHFxGuGQIRwF3WyrO5TM5ZjZI8lC0MX/mM/nPOgMlwM6dMbW9AKxmIhKOIKhuBBAm5vbE3YxYuhSAlevK/Abj2iCGXLZrGyPUT9zeFDuTkSeZV8nSc831iGEfX2iYDeLOC2AXqdzvzRtRAX2/bBSzmcM8l1HuAcxgEQkjJjhkwuOjfrxJtrtriRCOu06olnFuLUpgWKIGNRQSPOCgUhQQywZF9dIwscNhCehSYvKIMfFG6++ARchBFjkFOZP4InHFa9weFRCtVpHz1caveMtV7A0pwiU9Y19DEwP9d4Q0TH96Zh4+PIKXE1HPBLC7Xv72C/VZS98+snHUSkfYDQa4fHHruCv//rzcpy9qoNC1sAjD1/G4tICTp0/A2YfHX12YgCf+tgHxbVyexK84Qs6q42ubwCqro7Aa1z7xz01Fo3CtchlUIFjKn2fyLzCCAYNxOMxrN5ZQ5s1dMyZzM/jxNKCRBoM3dLJJHqdpoSF4ZCHRCIui+6oOtZUamj3B4J9qHDmEIGo7UvQRwM4PnBnMkvS0FKz4AqoHbOXeiQmW/x3aBACAWh/+ed/5hlhlTRoV5Xb56hW6spNR3JysGarhZEwL22GAAAgAElEQVQ1QkhzMV/wS8EiMXFfccOTolBaLz1IrdmfZKtavb64o9devSPHzc3Mw+z3kc4VUMwoRm759CmpBzw4Um58JunhxKISitxZ3UKp1UQAAZz3CSF6g4WFaRWqeUC948HsNfHIQxeRSkZwd21VJptu+tVXb+H23kAmKBLy8PjVk7j00EXMzS0A8SW85z0/KkTQ73/8V8H7wKHrHkKhBCr1ttqPXcrXKaf06wDI1zNfkEyq/Y0g2E8bSxzvuogaSvK+vnZPIgo9HEVQ08UAzl84J99jmyZGwxFGAxa42jDCSmswBoyy6AYjmbRxhRNfY8grn+8oD2I5liiWOEy6KmYBib+YCLItBH1wz78Ta/H1Mc2s/fu/+StvnIioH24I2CPbRitzQhlR0rYGiuolcjVbihxZnJ1CMhHFWD5Fr5BJ+PJkESJC3Cf3nFa3Ixmw44pfIOkMsDB3Gql4EIuzSmNuunSbhtDBYQwQDmmw+jWx2rV7u+j1R9Cj08JScv+fns+K+6V0ndxSq2cKWHzLo5dwdHyMmamCiE5feOkNvHqvClcLUpsqRmAYDoqFAv7ZT7wPP/pjPyYG8Ae//xtwrK5gGyGP9Ag6Q1eMWW46y9kosvBzBbyB01PTYtx08fw54Ep1XRCNW6b63N7uvkjNuAc/dOUK5uZmBBel0hnh7MPhEG6+cUNYTHIu9UZLpXK1ADpDUzSU48RbOEBVlgN7OJDQWM7LF+ooUsiErVF1pAzVfaColMBVcj7+lk0ASOPQnv3Dz3m1gw1MFfMCYujyKfQYaWpibNfB0KaKZIiIX8vWbyvLmymmUSz4RQIsZAwFkU2pcIbJHJ5Ud8CLkJckBqeBbh004ZgOHn3kYdEJzE4plXE4UqCIHprmIhEJin6tWVE6wv3DOnpeCiEjCW/YwsnlFVh2GxpoVA4CoQjWt7bwfW96Env7e0K0XLl8Hjdub+LWTgs1P/0ZClKCDRTTMfz0e98vBvBHzz6Lz33mt2ANWVbOMIm4J46+FQRrHHt90+fAubh09Pyy83Q6I/eF30XD4aIRwO1r+pr0UH2KWy1cunwZs7OzEk0QU3GSSRil0lmpqyQm+Lu//jO51mhKJZtCkfulYRSVs4ze7DQkGnJGPQz85BQLZ2xNg6mcEWyEEPR8itzfWiUR54+R5XsLcwTtTz73Ge9we0OEmNwGMtkEukNWsQJGag79URdBARs+8+QnICmaYAEUv3yumEQsSteiaMzZYlLeTp7fcTT0KPXyaw4r1T3hzw9LXTTqVVy+cBb18j6++21vxWDQQSy1IOXbHNlMEp7ZEEXwYakJM5AQXeHU3EmcmCugUFQMWqO6K9Lv1bXbwqY98dAlbOxs4NFrV/Da66vYrw7R7A3RGwzEyGWr0sL4l+97v2CAP/qjZ/G/fe730GmVpBQsHFUaRC8YheMFUak2JNVLZQ2BH9PlpGS4h/f7rHlIyF2Rgg3XxbA/kLtFKXij2RA6mGncs+cuIOzXA0bjKZjDHmJxtQ1Wy4pW//w/fEl+RhJpFQKTK8ilWUoEijpCehjeqIuBacoCo8rY1MirKPJtaLvQqFjSIIK73pByPgVOmdDm3iDiEV8won3mEx+WyqCj/T0BQHKQQBLJRAzhpEqgwO4odumB0GPoC++iYeX2pzJRRAxFgtDKJcPFKpagDr6XN8T0q2hev/26hHD7pTZLGZBJqMzaU48/LMfq9mxk8ioXwBseMzR0+n3cvr2Jnl9h+653/zBm83lhH6lFLJfrImOvVg6xOBNFs9nGqZNLmJudwo2b63jl9gHa3S5S2YwUTraaPXz0o7+pDODZZ/HHn/0kBr26CF9iSSX+YHlWpz+SErj5GabCPdzb2petcDAy0Ws34PikzDizJyGvZQrZQgMIBFxR+VIXOTOr6GS6f5aIMyHEtG7M7xVAr/DC88+JsdzZ3MdsMY9+v4dISEfQtRCSFC73eGUYPdMUVEKZg4WgAG6GnBF6UTVz6A6GCOpkCm2lQaQB+M6ArKL2+7/5IY8nTAOwHQ96JI5QLINIdhaRsI6QiAc8OQC/gH0CiLJFGBlNqv3K8xANUwJNHBBGlGU3LBgpH2LUqsByXJx7+Jrs572GUqLcXl0VrrxUtwX15pMkWIAnn3hE/l6rVBGMq+2l1+khHtOwt6eEIclUBo88/pTUCy4trcgND2ka1jYPMew38dDFs6jWOjg6uIfvevM1bG7t48b6EWrNulTeUK7FPMbHfuv3JwbALYAoXqRnqRwsJ4hWqyXglIUuNCRS4MelEnZ2d+Q4Q7pgjbn8qNwX8iGyXixTGEAONqeg+6bkjcUpnOyZOZJoQcRTGRi+m+cx+v0BKpUSqpWKfC9T40aAlVY2AmZXgNs4va0IMaAxtOGSolfdAxAO3g/XJ3oCn+QjiH9wUPugfea3PuD1BxZKR4eT/DyJhmRhCdlpJaKkDt6zlYyJk8YVzUkb7ypU53L/n5+flTo+KQI1O7h39wa6bRcOgYrbw+WrVxFPxFErK23e1u4BNxEcHvVkleQyDIE0nD2jElDmyMVBSXEEjUYb2UwI+cKcNJKYzmWRTsdQd1wk42Gkwklh3ZLZKSTiCQwGPewfVPDQ2Sls7WxKYmjvsCqqYxc2otE4fvGDvzEJA//w078pWw8Vt7FERnQA7WYdK6dOY2YqIa6XlUOkr0vlKjY2t1FrtARojVu2EIRy0ohhOAq5LAjcaKDmoCvfSXccicZQmJ6dGEy+MCOULZk7IxJFuXwo6ed+vwPN0xEcdSdkEKd6zEk0mRgR5M/OI4xMfMGIprxyQAilkGCHTrspAF/4C/Egfjr4s5/8kNfuDGD227DNAfq+VDueLgrPTVGnXCSduOaBos12qy4XbYQiQpOOM4iXL12WL66WD1GtHMMeddGu9/HoE1fQabdQPdrBybPnJcHD5hBiXJ4rjOHhfhOePUQsmRC3dvWKSg7ZpovjShuOxrKnAM6fWQBBqKdHJcbmMOKOcBGNag/5VAGPP/IwCtkUhjYFoTVcv/UKqg0TJR/RVxoNoV1b7S6OSh1cOTMl9YCSVydjFjSk/qHfaSObzyKVjCMRjyGVTgtqJ53KSImrntx+qXQM04tiQFUTiSLYmC4W4JpDSfLQuLnlcbBJhuzJ/urJ5Qrosa+CFJGGcXS4h2icCSO23jHkfpvNlkjJqPOXamq6c4alHtD2jaAvJWMB8QZyT0KRiRqJ0QYjBEYqzKhysBpZCn6f/fSvy5l16lU0a3WkE2Hh9gOhGHqBjGS0RFYcCEqzg4EUSQBtX4WTiMWwsnIS08UpAUCb62vqBJwKWj0dntnH8oWzgKNiVzb9YEatO+hKT4Ltfcb+Fg6PmzjcOpzcmHA8gSceu4xBzxT8kU1nMRqMkJ8viJvuNCqIZacQCYVhOX1RJ9drbfR7A1y7cArXrlxCMsNUdBCa2cfzL38LO0dDDCwPR6U9AUqGkcDq3W1cPpVHLJ4UkMTSsn63C2s4QChkIJ1JIxYPixIqEg7LhMaTKUTouqXGz8Gt23eksQY9guVqSKWiSMUigu7t4QidDoFiDNk8MZUnPAPp2rn5xUlFz57fg8mIRkQXQU/APAFHl91JHBeVgy2/6plGxvhHrfieSVGI8gJMZ0fGmIIFo9I2Rt4mWzcne5xRlAqiP/70Rzxy9KlEQlgtjnKlgna7c7+PTEIBMs82pc0JlbUcYT0k5U5PPvkmtYX4rWBmswE06yXp8HX9xh4uXjkjdXqOz17lMhmpNdzY3kWz2cHuwZawV0es999TMjOYQ8SyKTxx7aoocHlRKydPI5EsouWXn1PmFDACkq8gzmAzpdt3biGfKSIXD+ChS6cxHFh47NojuH7jNdzb6aDO7a52jHTYQt+J4vbqDs4tphAmngkZ2NrchK4FJG0bibKm3xaSi+F4IkquIqom3wdztNidvX3s7x+iTYIrlRSKNR6NYNBtgBWCHPW2iU6nhZMnV7C7syUeZ35ObXXNehkOoyTWJETjAhJFd0EjMMLo9zrYWrsp7w34VdquD8gHXJTjPYF+WgtCNyIIjPsK+AZgcHsIBiVUFLWx31NC++ynPuLFDA8zxaISGUhoYcmX12oNNP1OWzQiM5THoN9XByCxEoqiWCiKUmZvV0mqp+IO4hECI3WTzpxdhjkaoeHr5Xc2lesfdVt40zPfj9W1LZTKNZTLB3AtD7V2BSM2f9g6FlYxHEvhPe/5J7h1axXFojLEWCwtiZlWrYJQgislLnzFvc2SpGZDYR1zhQyuXj2Fve0dvPVNT+LgcA8b2y2UOibqnTYimiWu9ObqPi4s51T2koGtZSGWSMtxBuYQeQo/43EkolFZZZFQSHoEkQDjJHI/Z8Rx967yfEwfJ9JpmL02d0z/fFMYmANs7RzK+xkpZf06gogRFnDcbTcxv3QSjUZdPGlxipXQmgDRRnlftHx235Qw1tAZFQ2U7oLA0yenhpR+062rr5VcBiONcQUS1cyOR0J+HAZo0D7x4ffLaRqxPEaDBi6fV7o8Zr2YtOAoHZdRKVVV6xWWXrtRMYApauNEm6+Mona4hmsXl/DUk4/gjRubEnKwPs8IWCKUoNttD0eoHh9J2Hfq7EOyT2WKGXzxH76Gks9/mwOqhCg1U8Yyu3QBP/juN+PrX/0m5leUgCIRNeRmNyplTM3PIhg2cG/rQOLufC6JRCKHWAhIxSjwTMueeev6KorzpzA1u4DtezfRG5p46fVVnD2Rl0lk60Juab1+E7qwkjoScUOlUNnrkMkTtmYJBpBOJ5HPqChlfUOdJ8UutjVEp1GblI4RVBZnpiVT+vqN26i3Bsjn06hUFaN69vQp4WCmpsgQelKMwowi28zQZR8d7mI46KPTUO+v+epoytBcmXg13ZlcHn2/aRUrig3yDePaAfZscB0BsOMx9DgfAWif+Mj7vM7AQyyaQNAb+po3NfHXLqsogHsIc+R0G7sHVRwf1xHQo0gX53DrlpI7MW9QSLChUxC5ZASFpNpOEFTMYFCPIJkIIpUv4ptf+wqCoSyuXbuMYc9EPB1HLJXCrdVVcYelcksaLxL0sQUM99rpmROIx3WUjveQnTorxjU3m5Bz67Y72NsrY+XcebxxZw35XEbKzdnSsViMoNkcYjQaYtDr4e3f/QwuXn4If/mXf4ty6Qjffn0d507O4rhcR7dZV8CSvQrySWHipBZQsJch7WF4XhSGMtQl/ihXVWg5rl5qSt/DkACuVCaPE0sq9l+/t4lGq6uyo+2mNJJixnV6agq5nEp+zfgLqjg9J5nZwVBR8HvbmxIiHu1vwNBjiCaSiu71XGSKU0K6KSIqoBI+VDRZlMipHZ+Dzbr4L5J1FNL0weYVGrT/6ZMf9XjxdEtGNIpUIgZdczBTUGxYPBZFIR0QjkAm2lYKlINyD81WH8eVOiqH67BgYDbJNigBnFlZBAUR+ZiOuTnSu2Fs7Ryj49Ox6ak8TMuDEUsJy0ViplI6Rjafw73Ne2h1FHCiMJMkVH7utHDw3DPL5QravRGsUQuxWBZnlmM42i+hXO9AN/KoNFsIhzXMzU5D13R0WxUhltJJFnskxFh+6id/El4gjJ3dQ3zkk/8Lzp+axcbaJpaW5hCOhmHBgmdocC1OJGMg5t+5KKjCIdp2RbzC7qYMY5nJW1hYgEldoh+C1ao1TBVySCVTqFSrqNRVOMvaxlGfPL4rIa0sMAnfPCk6WTl9Dob0B1IhXiQSFklZ6fgQtWpdcgEkmVZOstVNQIxojOyJF5hWJhVNb9Jrjbu/krVUzbM4CMB7dki6lmqf+dTHvIE0PiDApKrUQNJfvezTx9w0VbhC7ZoOEvEASI3TQskHcL/6xte+BvYJSMRCGPVVfJmOJjE/O43TZ1dQKe+j12lj4cSydPAweDxao55AMMBq4waMoC399HhxjVYb9XoTu35EEUmmML+8gqHZxMLsSRxWGmjXOjBHVcRiGVw+P4s6cUSrDU3PwWHHLUqkdRea5mBp6SQO9zZw5fJFvPbGa9AjEfzkf/ezIrl9z0/8Ai6cmlVSsoRy96X2Mcjzd7sMUQ8Qj+YQCUclWygV1I6FZrMpLvfkyorUGyRiEaSyKoPJ3gNscMXB7JyQQCy6ZTGmxzrDQ/GYiXQB8XRB6i4jQU+ih2R6GoYRw9LyCZlghqBf/MLfodNuy4JIZbMiNvE8C9VqGblcXlrCsCqIfEPKr1NgE6t+W/UtmKSvWZsw7KNSVyDeNvLQfulf/pw352fkxhaiPuChmMsLccKRTVF0EYLjqtCDOQKO1Ztq/5sr8ItoU2FEAiH0ukOcvngNyWwSiVQEQc/CdMbA9m5JlEcNWpFgDwUWI7E0GrWSuEmOXqcuurZaSSlowvEUTlw4h3AkgqPdOqrNLmZySZjsWOp28bYnL+P1m2uotTsYWgmksszT23BGZVx46AmpOmYIGk9nocUieOLSBSydPIP3/NQHcenUFJLJFFxy7RFKwfoYWGwM0UG1SsFnUQpAmK3kyh0NhlhfW5XzWl4qIpNM4MSJZQF49FTUUrBb2DhTyKZZVAkzrqen5UQyXNzbO0QsmRejIhBPJFKIxFMgMUQJPHEG992N9VW8/JJqUzOS7UgtoEIxJyrksWKIhkbVUSKZlIwhB+ek22xIowhhYgc9hLOKYuYQAzh/et7XmAWkvSq/VACD54pQlPFxzM8pxxNppJIZSRxtr78kHoA0MAdXARW0rh1AblpJv2ZPLKKQi0vsToutHZcwt3JSwNTx/gY8PSPtYXvdDrr9njSjIhXK409Ns1zdxd0794TUmTl1AeawKug4EUliaSrJrkM4KtUkq7Ywl8Wff+l56IEwoEXxxJU5yUdsbu3CyGRAeX0qkYVjhNEoVfCTP/7P8J5//mE8fH5O5FjcvuLJBIwExRMaqItd39hFpcaWNw7mZmdFc8iOpgxrg84AId1FPhMXniCfLwiGMmLsgczIpiLuPmIERTfBvAVLwpSw1RTiaXZa9VM62NtAoTgrHiGTLUikceWhi7JPf+WLX8D63Ruw3JCwfjwOgSK350w2I0ZFKpmgkV4gn88KPS+iEMdG7egAjhGTtPU4nSOdS5jE+h//1fs9Fn3msioBwvrxB7tYDb2AFEZysGCSNSXz84u4fv1VeS2XCOCRq2dR8Pv5fusrX1WWSoGCHkMyPaXKseTzabzy6nVk0hlZ4abLvU6DG/R77o26IoTcXL0J1wsJdghHozhsDWH2WkilioJPyPJx6EYSSYNUj4Pu0EEEA6xvbgF6EkelKgqFPOanM8gVpiRk2t1fx7WHr+LW1hZGXQvXLp7FH/5fXxIegNW/FEskU3EsLLEkXpesZb3JlWyLgXIh9AeupHeDmgvNGQjlTWna9PyiMGvUA44bPhwf7MjqJqMqmTtmVswhas0ustmceB0OVjpREEKFVSw9jUxhDsU89QIBAZEMsb/4+b+FF1DEULujmMOpmRk5Bvd9EmAk2LiyubhoBKx0IjvJ/ILMyWjot/u7X4iq/fa/fr837uFLylKaGnqq1arjK1+DQRY6GHBDMUzPKXnYay9+E9l4Eru7a8jE2Acniu9+x1swO8tVEMO/+9O/AAIGNF9WvnLuURw3mGwJC2gytB4GQwdmgCtPQ7dF8AYMWw0snzqNvZ17GNghbFcUdZlJpYVcCQaVMXI1694Q28c9aOYA8WQSAbuPs8tzaHRMVDo2NF2JWMnHLy3Nolzew8ULF/D3X35O5GvJZAS3145wfjknVT48r1NnVoRHYBjG1vblakOMkuKJ4dCRhllMGpG2rh7tSSYykcwopTOTP7GYhGwMw8KRuGrUwK5nniYNrSgkJdNKgSYZP3YbYwsakmRk6DLZrFRp5+dWUMimoXmqsvfu3VWsr91F229HS0yezWSkHoHDtj1FWScSch3MKNIYuT1UayrVLO/zt3SeF1lD7bd/5ac9WiPVqEy1EpGzwwXdoRRsupR1O9D9Dl5MJHR7NbS7JpLRBHKpkGS8bNPDmUuXkc2p2FjzuoiGVFPEv/6rL8DWYxKWhFwbM/PkGjw4IQWajspVFkNh0DdlxQxaDeiJLLLpKNo9F7WuOh8iXm4zqgGTi1gkjGIxjZ3DBtLxNA62buFtb3s7NKuPV+5QFMK0D5F5HOdWZrCzswHHHmFudgE31zaRjum4uVnHhZOzom5iM+XlU8vyPVJ3jzBq9TpMEV2yf6Ipe2wkFMDO5l1YQ0vAIu8Z1bfCEQRcsBlpPOZ3+Q7FsHfIukkbI7MriZh0poBQOCIhJruGlI8UiZZOK/k4ay9I6CTSOczNzggb+Py3v4mNrS0pepG9SQixuCTqFDfBsDkkXprMpQhOxgWnIkE30SZIZvb2gechaB/9wL/wmKE7MatUKIWimhTuiUzY8AMjPwdP8MFJbHVbk87cx5WGgKvLJxcwPTOtOOw428CZQmHOTC8iHGyLeKJbr+Hm6gEiMSp8M4hE57C+eywJEu6RbK9i2iaivrpmv+534xwofoKVOhwzU3lpMNFp12QFZqfSODhqI6zHwJbAl88sYmd7F13HQK91LNsJV4ymUTPfw8Vzp3B3ax9nT5/E5//xRTGAdCouNK+RSEEPsLgkK5Q4xafkgSu1mrj6Qi4lwHTYGyART0nrl2anJUpdc8CMZUrAHOv+Ke3u+8qoSnVfKqRZq8esIsUezNKxtm97U+klmXvIF1VoODu/KJ4nn0tjfm4OB3t7+MY3voGDSgOJRGzSIp6JNXYfZ5JKeUaG4iExgvEDJFiSLluQ7Yh3iPrKYIpatQ+8/2e9WDQixATH8lxBwp1pv7GB6AW1gLiTm7fXhVzigwYGTgDH1bqkP9u1GhLxCDJJsocaZmfnUW/2EQxEMFVM4pFL80gnojg82MfWwTabnOCw1MPQVjXvDqLomy4c00KBnT/sAYYDU5Id1PyxKpiAJhEzhIJNpxLiyhiKMjM5PTOLja1dBAMs0iDDFYQ36gj5NBj2cf7MIra291FtDRFwTTzzXY/hWy/dEje9unWI//yd34vq8Y5sA9QkEjlLx1PbQ73dx9RUQdLDcSMgIpChcP45IYAYVoYNXVb94qKaPKqIXI99ijJIZbnaWW9YF4PiSMTTEs/vbK0jnU6pFn2jgeT9KRRJZvJiROlsFtceUhXVm9t7uHv3rlz3caUmuQ8OZijHvYXoZRipjSVn3IqlbIzeifEZS9Y4+R5Fr2GRq2m//qu/4I0nnyogjnG1CW8ux+LsDHKFGRSzSUnVio7fA167uYqjw0NJNa4sTMueZVlD0c2xuRSZw/Onz2AumxBC6vi4hoWlKdzb3pH4mU8aMfsD3L69Jd9THUJqDGWVn1hBp9NDq2Ni2KfWHVhcOgFzZIp3GrdT4/ZBl83/WMkTNVIC/jrdjuyHtUYTBkaykm6t7ogBPX71FF58fRXhQEAM4Aff9b2oHe8gEmejrAVhCIuFPMq1Biq1hhSHxKNhhGDCNU0U81PQQhF89RvfUH36YSEeMQT72J6ORITMakTK0pj0onCEkQ0lcvVGR7xqrVoW1RUHu6mNRZ0Ec9x+6C2KM4qJncolBIju7OygUq1JGpuDhsBQUbl7w286yecMRJUn4OLyq7HTPuDktiFzTNEpewT9+ofe75FMkHIh7iXkh0ivdppyUiP/IQ4nz1xEvVQSDRwTJQsnFnFndRWNWgMXzymUz0e8kP2iV6AFDwZdrG3vQbNZVuZA12yp/O0MKF4I4PufPIPD3R1k00nslntoOyFpvpSbVokQ1gWw3kCMcKGIYMhQOXdm0JosPXORyhRECjYzOyvKWqZf+SwhxYgNROF0cNxCIRlC2PCk+JI6RZJv/e4Q++UannnLYxh2myjOzovw5PHHHxWt3Te/+TyOqnVonolMMoyAq2F6ak7O48VXXkOn20exkJKtgRPAtHW1tImzp88KCGNYSLA3PbuEb33r62g12+j4PY1CUiUckbIxhmeZbE5WKdvcRyJJCWv5qJ2Z6VlpRsHxwosvCZjkYCNMPmuBZd80vExG0cmquaQnnARDQ/VMA9U0ijQ36xZ4vjQA8Qwf/+ivenQNjEkZV1KdytGRChOxFVx77CkMOlU0qhWFWB1HCiWJhNm4IBYmoo2hUMji5MllZEhg8CQHDkqlfdWASnMwNPt46bU7wgCySqhWqUsCJptUSRQLIWRyKTBDtrGxj3gyK4KTYi7Gyk45Jp81xDaurNkjNWuNerAoDfdId7L/gC1InBlIPqcnlc2Bffmov9c0E41aH+lCDnOZILYOu7i7sYm3PnVFBDH5wiw63S6efvopob7b3R6++tWvwLWHAvgoP9PDIdxZuwd72EUxl5JSeSPJrieKJWQI3W4ci/cxwuy6tiSyLza/fP75b2FIFK4FJW1MI2Rmzi//lx7JLDEfR1rLyysoFqeFo6GiZ3X1jrCMzGKOyZ9OuyqJMOZEKNCl9+AEE4tx0DikdxC5k3hc/iYNwFwqrkY0gF/z+AVkiMYdLCkdYvaK/fLou59605vRru/K/lM+3EMmqayN/WajIbYjC0rcOu0DSIKp/XoPBappMjHpTy8nY4Rwa21dCUd26tjd30Vy5qQIGfk/qTzig44icYm1250ezp5axKCnJFH9bl9y5ZE4S7eImuMY8IZYFnoDE6EYsYEpamULOjSHrF1H0rmZRAzVWhXDbheBWBLnFlLY2G/h9to9XD2/gGIui/WNHZy+cBHhaAJnV5aws7uPWrWC229cl5VKQSmzap3aFmKJAqKZBQkPA54Jj0kvglS7L+EzFwh5AbJyU9MKG/zFn/4xwrGMeBAWzVLUyTIyjmwmPSnqZDPu+QUmvxLI5rLI+91Xv/LlL/nVvcoLtFoNSSxxggN+2xh6Xj6Sh1XO3HK4uGlUNAD2guYgqcXUEBlN7eMf+zXxEtS2j5OFw74ppAHj+Mcee0IqYJrVY3SadVHisOCCbqbXqi3Z9bIAACAASURBVCHojrB8QnEDJC1SyRjaPQeHjQE6fQV6PCeIZDyulMFeUB4tE/Jr567f2EShQNVNAoORhZ31dcTSRVlNZ1bYB0idVb3ek4mOJQ14egrdVkMe5EBZNZ8FNOp2pEm1ascXwHwxDy0UFg3jqFdFPJFBb0DlDCN8DflMCAeVDu7e28ETV/jYmwjagy5SmUUJrzbWbosnWVk5jZtvvIGp6TxajQN4dg+ankZfUyROJhEV9o0Yxwjp6NSPBEQz/c0Vx3CO3oqtaTkIGkcjhY/ICPZ6NpLpuKD6yxcvotWqi+GIW89msLCo7i2xy5e/pCTjnA+Gwf1BDwHYKBZnJjUEbI0rZWRUDLvjUJA5HdVjedyYkoYrXvGj/+qXvHHbsQcNQFa4Azz+xFNwfEFoafeeAD1egLRtpx/xPKmyycTJgAWkzn+3WVKNkgMhtFsuBqOukBaWFp7UALL+cXu7IfG9afVRYN5hwO4hJkYeu3QqKROfGcSbzPTrUamEkUs1LlEtma2+aPQ9Twfl6SmuJquHiJA6YdRrdenfw/2Se26/b6HebAlC58ZYa/CBE3U8eWUZa/f2JBydmylgYDmYmVvEiRNL6PVGkkiqHK2BhBgi04IJmM2U5o4islQFtPRSN2/dxEPnmb0MIhENIRlPYmp2Ubay1197BbuHKrfBrutMvh3tHmM4GOHptz4NVvdyoqKxqEzU9My8uOt8cQpbG+oJa9V6TYpIVefzsDycg2lnzgNb3XIO2XjC0xSZxQNw3+8PFHs4fsYg6WdRBX/ol37OC0cMSVty9P12Iv2uomevPfoEHDjy3Lxm+UCxU3xfp43DvU3Jnp258JCkR5kwsvo9mFGyezq6zb50ERmwYoYdrIIGOo0eckkNvcEI9UYTCaLdYgEVv98Qy7HY+YoVuurZQgYK6QgOSg0MLUfAkaQ6u32EdUeA4FShiF6nAUMPY2T2kEimsLR8BkxR3FldQzCWxFIhjv7QlOiA2j22sdPRx93tMh69vIAbt/dw9ckkAsM5aV4RiwZRKMzi3voNpIwgmqOIFImy0aXOLhw8l2wWkXBE9H10p5sbaxhYfOjmEvJ+/WQ+W0A2VxDPQzXVc899Q44vntF//l9QT4i3YIs+NqIgMzp2x5SX8fE8jALYn6DZViVhhVxO3tttV0Ubwb4EbFxNV6THMpMnlEnLeV9LTkZxbADqBDxoH/3wBzz2pImzG9QDT7Eb9YYiWmSzBUuP4GhddQg95z+8kZ2n6bKDfvt2hpJT6SjYldV0NKTSCRzvbUMLhJXrDxgo1SpwrSBmijEclOvC6hFEstvX5m4ZfEpWr9eQCGFqel7IoWTcEKzBWrjjhoVQJCSa+lyM3TiY+u5LrM1MW71SQmF2DuXDdYksNveJKcIigGKqmVK1Tr2Cyw8/jOs312D3m1g/qOHs8gxaVgTF9CGqJQ2zM7OCyruNI8k/UI1LFpCxfqffQ/n4AImpeQHMMSOFRqMFs9eVdjKkkbWgLe57dnpKQNv0NFO8YRzs70rZGgEjS9k4VBOtAIYmQ7+kKJA4qEKmB2SEwLBa9gzm/20yirZEBsRY0tLGHk0MwIgmYPmVH41uH91Of9ItPCdNrCDXx0H1lPZbH/2g16c78IsaoDmThzidOncZqURGXMja7ddQzPA5ejoo6iTCJrvnUHfDrhlB6ulMdPuqgigaTyLsdJBKGHBIH1sBlKmjpzTJdlBtDjA/kxEC6vXb23J91N6z5RzpT2rzTLq1/kAeH9dpm9D8pFE64iIWjQtxY8SS0jiBxuO4AXlUjaaHpJSLW0Cv20OMTa3MrlT60KuwwQUXWbVUwhtrW0hGwygubSGCy2gctnDi1CnJ8lWbwKkzp3FqZQF3X30BM4tLmJmfxcJUEp//0tdweFRFJleQRAw7d7SbXVTqZbTazFgGEI/lUMgnkEr5TbGlhYsr4aKqtFLgbGdrAz2LrWRYYxDAysqStJUZ9/vlqo5JuZoCf4ymMqmsbJ/EG+yqQq/HSK5arUgoSX1gtdmRMJ6fyuZU1nBhfkHk7Bzl8jG03/v0r3i0IHEH8j+CNpUtisZySCQWZaJf+Pa3sDKfx9xMXmRMtDy2hmO83etbMgnyaBQjjLCmcgpcoVurd5FZOIFksog3NreRChvyrF8j4T/V0rNRqnUlhUv0HEskREefigYE0B0dtJQ617ORjg8EEwRC0+ohFUYYg35H/m1ggHw2gnJlINy9x9iZ9fFOBO1OAydOnJA4XB6KxQdS6JRO9fHC6+tI5ww8/eQC1l4vCQ5oNVvIFOcwNT2NSDSE+dwUEPQQCjryNwLmve0t3Nli1ZODaquPaEghqIiuSbcP1zUlgUZ9AMEkE0iqwzdLt9mlxESn3ZXwmU9tI+BmPjMUZju8aeH2GdtT1Cm1BZ4nWwmNgqBT2sJJ23gKTrryfia4uJhpCGwxU65WBVyaQ1NwBb97bmZOOA4aohBBv/vbH/KkcwgBwqR/HMu0SBiEgVBK4v6b11/BwvQUrj6s6vd6jYq4WQIfVsj2eh1psmjEU+g3S+AzggVTDHhyMZihJLaPm1JEyjJr9vzvNesS6pEDpzUfNQbIGkSufPBRWNq607BafEAkH/Skezh74QxGtofuiN2wbCGKeLFsgNBrt+Qxbf6DfbB3WMLFCxexfnsV6Wxa5NJM1tBdO4GIMHvPvXoT/9n3PYLXX70hD8X+3meexvXr9+TxK8yXLy+fgmsO0CrtIJvJyuRNz83Ls5RZHfTK9Q0J9eLJkKSpo+G4lMLznkmfJSkGvf94Vz6Mg5NIboIdUtTTzQIolUrQdP/5ywFXKN65hRNIpXLynaSm2XONYlD2Rmq2m+yTKpPIFDCJOzbH6vXVw7fYnsaIxtVziCQEVBhv/JhfhoFSF/C7H/+Qx4c40jrHBsDJD2gGHI3dZ9i0yMHdO9ehewOcPX0GrkfRhylZKumYwbxZJIII691D6vl0fLqWxZh4NMCgUYZ+4imUaj3RBRqGLlSrNIU4ZkFmEF0rKHstc3CpZBTnlovY3D/GoGvBHCrgo0d1PP3mp+GaTRwdV9HtqQQIu3jxMW6UYx+VKkoObYQQ0tmFLILK4Q4uXjoDm1WUmo79oyqMqCH5ildefRXL8+wObuCZx9+GL33jVdxc30O+kEWKGULG0WwpY9oymWxizUGmIRhUeXh23mZegpOyvHJGqakdekHl5cgDMAooHx+iWJwSjyDH4MTyFy2I/f09EaFQF8B2eqSPZ+ZPI53JS9KIaeMQvZCuHjl3WK5KrSK7uLALSCIRFWkYGUgaEw2MfAaNYcF/Art0GfeZVYaBwhr+m4+/35MeN9IgWVlJKl2QUC8YTSEYSeLundehuTbSYQePPvqoPD2LRsNJl2J7eeQJH2/GlGhQxbmui3aritXnvwwz/zBM20Or50rxKEMyuqu23zAyFY+i2uqC3tkxh5LtTEc1HFb6srITcUqwmeXSUa8PkUgYiESD2NvaxdRMDgPbxXRhTmJnWnWzVYbmdXD53DT2d49xXA8JYGXYxJSvY9ElJtDtjfDf/sx78elPfkTaurLzKWNrsm/LK6fk6eYUinSbZXiDFpLpBBJso0oqutaWyl52CNus1JR+oN9Df8AsaNd/lrKOWDQpghQWgm7dW4fpsE5wgEazKovf0A0wCiNm6DSbcm6pXB5BI4ZUdlqirNnZRXldD6q+REHNxn6pKhGWPH3EdYWoUs8dDuDo6EgJckRyl0AmEZcOq4oSVtu7PLyCDOHL//BvRanICZOSIb/OnA2S9WgOejSNl196Dm6viuX5ObmJrZp6And29jSyxTmUj/f8J2nygYYBDEwbhh4QWbdlDaBnl/DG+i4i0mDZQyqbx+buoaQsSSylM1lZtWzEUEgaSMYCiCSSqNb6aPdUcqgwlUE4oGFhbhardw+luCMeJlJ2ML+wgLBuy7ODOy0TsZiB5ZOziIccHDWHUtBhmR5q5M7bNaQL05jOZvF/0/WeQZal53nYc27OOfa9nad7enrybF4AC4IJJZK2VOXyP7tUZf+zftgu/XCRVQQlihLFYgRFlijIYhApkmIUIRMEsMAuFuBid3Zmd/JMd0/ndPvmnJP5PKdn6R/Wqdra3Qndt8/5zve97/M+IRG2ILe5jXcebCCdzSAYjmuyRrJmdiaJSDCgkIWDh9/GP/wf/kd882vvwOq2wRdOw44eDk9yeH5gcgMdlgi8Ib86AwJWjNgN+iPi97F1JbpXKjfw5NF9DEZ0Jf/70IcGDTcmNKPOapYxPMf7SQ/jjhKkP2GjprAKn9ehjo2EXJ7lH3x4V18/mZoRUsvagCBenlxKwySFEDfg1JGUsRcj4hcF5f/vAqD3zpOdKm7cuKo+mgKM0sk+LsxnJcgYcu6dWvz7AcSoi9Gkqy2MmHxnDAzaxM/7cAZ9GE3d2C2UEfW40Gx31LuzrYr6nBhQ295n5CsVSVMkQqY3Hs9AeuV0GyUszMewPBfEoDMA3Woo2IzGotg7buLoeBtufwzjTgu1zhjRkEtf02J3Sx5Vr7Zg2K36wSn0mEnG0e33cSEdwO2Hm0jaJii2W2hPrAh6PXjOYGzDwD/+x/+ztIvV0rGoX+RJUGbGftxiY7FZRCIVQ7lch8NpYOPZExlMMzeI98/r5lsX0EQukUzLRe3b735bBRw9fbhr8tilYph7SqWYw9rlGyKOlApnGE3t8IXiWFlZQ9DvQ61aRq9D+r4hQ0qSP4gtbGw8FahEAQrnK6wHWJNVSgXtspzUsvYhb4DQso5SAkCUkNEW79f/9Zem0tXRgUoul3aA//ydW8Xrr93Evfv3tcVwAazMz8LrdmLcb8IbisPu8iEQDAgLN0AzI5ogO3CUq2Jn1/T0zy6uYkDpU3cgcEfbZ7Wq8TEHIvtnFbTqFQzgQtDFRAwzlbtQaWmKiMkQF7IRiTcOD07hjGYl8DgtNrRLcDZQr7Mj6ZsydKqXRjwrLTDo5+tzqmPJn+zj2vVrmI5aePD8CJeXF7B1uI+QzQOfz4737mzi0qUVTCdDtW8dpoUPGli9uIi57Dx2trfEC6jm8xgPKMvySk7GKt8bDJmmET1q+KeoVEpyIaO+MBoJI5XOaAvn+c1rf28PZ2dn2r4XL6zp+GBB+HxnGw6reY88oQTiyVkkkwmcC7Sw+eQTET84x+db/sabb4hnsL9/ZKa7KNL4PAeABh12B07OTqU6Nu3s6W5qTgSJLXBxGr/8cz855eTLdKnm6vCI0MDrC59/HUcnp2g0GtjfuI8vfOY1tGpltXokNNB0kbSlQaeGaDwpunN3YBEGfViqYiYZQ6OXg23sgGNiw8qFBG4/PUGtaLqBLS6votnqCncot/qYiZnOJCRkMvsvrmCGKbxk1WIIf9CvaaRh92Jj8zmiAQdi4TCa/QnqjaHy/cLEEEZjhMNumSuUynXEAw70bAH40MKz/Qpev5JV4si3HzyApTnAxGpg0G3jMNfAD//oD+Fw77nmAQ/vPcbaxTRW1i4hlY5qjEz188W1NRw825BNi8vpRbVDNlAbyewCcrljuYCzxggFQkim0+IvsD6hbJ4PkEUdcw8eP/oEDmcANptX+H2lUhSoQ3PrSHwOgQAp40A6GYINTR0rhKZJN//k7j3cfOkSFpcv49Gjh2g1WqjWGp9axbN2oHchC00KVlrnxF7uhMmECQSxRTT+r3/6v6kQpYcUx4kvLo5iP/eZ13Gwv28KFHP7+Ic/9kVp/IV/+2ZU9fM6O9tFKh5DbdBFtdRCp92HN+qB1eVBc1CBk4ke3TF6bQd6hgeNyjGurl6AjdQohwO7B3mxi6JBPy4tpnFv40hbXSIe1Irm8GnY68Bpm4qeVqm3Rf6g9oA9d7XZ0jTM5fQgGQuAsXfUvUwdExzlu/ixl+Zw59EW0pklfHJ/ExcuzmM6aSJfmqJfK6A37qNBY2qvDXtbdOtKweMPqYZ5eO8Bbt28hm6ngeWVVVnPM7iKSB3BomK+jr29I5E/OgNm8/SFUbAo4yKjv7/H5cGjBx8J3eTv8cXhmF3WLYMRzs79gXo9s6DjMCwcjmnhsLInYFY628XCwoJUQJVyGR/dvouVtYvIZjNot8wuaev5jnaCWovUL6dG0i8Eok+3N1E8D+x0nwdLzM8vwfg//4lpF0/nTS6AoMeBaDiA2SVThLl/fKxVWTp8in/04z+OZ9sb+pCJ+BJ8Tm5HBfnkD/tjnFZqKBeKUrom55dVUbf6p/BYTRZKseTAxG6ByzbGW69ew87RKZrtHg5OK9pCafNyfW0esIdweLCFSCwu5xH+EFYMYUymeLhdPCevTJFIxkxFzJggjRVev1syrowf2Nh5rrk6zah3ToqwDWoY+5cx7TXwQ5+7jv3DPTRrQ1DCdZI/QTo7h8cPHmNqdeq4igQdoEaP9DMGQdFQgmobuqiRgUz3UbqhcNT84BNmIfbg8NrhdvmkxuXxRtNp1io8psqlPHa2N4SQ0oSCjGAOs4jk8esdHR5paDUaWRFKziPo8wpkI1zONjl3uImV1dVzY6oODo9yktKZ5m889jgAcuh7Hx7nRdqNRiM653kP2aof5/eVochr1BtgNrMM41/+zE+ZtA+D+bLmGcUFQLw9u3gZe0dH2H9+D+FAABfmF3FwWkAskpBXfizsURZOt1tGqzFArsHRa98MYZYwoY9Jr2Dq5Rw+tAbEv8ea0b/12g1sH+ewf1AQNNzpDxDxexEJeGTmRFaPx2GD1zmG0x1Gq9vVUKnRGpkGjcYUE6aNTBnY4IE/4IEdQxWW8XAE+ycHGI+tGDo8mEv7sL+f03ZdKRdRKTQQn51DP/8EDu8MahyrOhxi/jJskgZStUoRyaxZvM1nZtCjooaUr15PSeQmX8KBpdVlZQm6vGHc/v6HgooJ9nBn4nmrTAExrFrY3noKp5fQcEBkEUKxDocHrda5s3dnBJvDD3cwIcYzH14mHUetlMPR3hbWr16T4xjPfyqPCoWSWMqcRHKnUFbyFNg9zGngxWe4tHQBhfwJ8oUjCUrpUcDL5wnB6/bB+Nc/+6WpNOQjk+uuBcDCzmbD0sWr2N7bx97z+0gnZ5FNJdAUz94sEllX0OeW1/FpDUOKICxO6db4Sa7QktUNJYLf+eQ5uv2Rotzojp0MOFFudlAqtSQTS4S92rrlQ9Sfwj7t4+LaAtr02htMhGyNp3ZFslmMETgQ9vtYrLUwHydqRwxhCHcwgumghUAwgruPDhEOWpE/a2F+JgyvfYRadwK71SG/Ap/HC8tkjHrfjLsfGU6k4lHcfbKBarEsSzdXgJM6D7KpFJ5v76iKJ9GEbVww4FGV7XQ5VWlvPH2OuaUlZDIpRMIRtW/ESrjoyAYu5I9RLJdwYYW7Kyv3CSrlEnyBGA4PDlTEsXaIpufF75tNRXVkjAcd3P/4Q7V0Fy9d1m5N4svpaQGlYk7Kopdf+6wgatZm27v7wjS4SHg0sRg9ze2pHuDCZbwcF0AkkoDxyz//s1OGQhAHIN+dBgg0IciumHamxwcnKEtMOMVSNo3c2b5cNHm1+wNEPR4NWC6uX8Di4oz0AaNzXzqnBciX86h3J3j3e/fF0iV7aGF+BgbNjW02nBwXdBM5NyCzmWc/bEEM+03MpMJoNsiwMaFeaiIoyCQgREKJ3zWFzzqWySVRMioTCU0Thj5pAF5jggadtqxWzAcGOCxP4PJ4NDBZX57B/kleYBD9hRJBJzoDA4VaC5X8EQr1PgbdDrxuD9KzabjJwR/2sbt9oLHywuI83HarMAzy+klKjaVmPwXBZmfnEAxT3WPg8PBQ4Zgnp6zITbKIiC7TEbxutyr9x4/uw+mN6ziIxDOSoZOBnYwF0aiZGYXlYh7XbtzSC8TxN0WlrAnOTk8wO78ALwdwLqfc1La2tsyQjnpXpNODg03tDuwM+P39/hCcTIP7rS//wpTghwShghvpb2dFan4BR3vbZnoGGaZjk1DAdpH0a/reesNeXF1bUZQM+YQkcWx+/AHGlqDg01QygFy5jbbhk37v9KwsYgfp0pGAQ1Dmaa6MyZQnmfGp573FFcTx/paGTm4vBypjTOBAMJTWjZCZk2FFxDflPUQyFdGRUCjUZbY8H7PjrNbGWQM6oyme+IE1H+5vVxENBdAY2HB8vI+D/R2kZhb09Xy2MXLVBlZWr+Dy+gr+4v/5Go5Pi3IK87vdWL+yro5IeT7DqUwu2QLPZrMo5fYRTyaRZBA2hz1wymeIdG/OQlijUOZGL0MSSWnvym6AUXakwzGGhkERPM4CoQQi8ZQS2vgiGqOOdIOsh1jv8LicnZ9TKyrHtnPTJ8LMcwsLquPIB9zY2MTJCWuBNqykylsM+L2+Tz2cqR0g7Gz8uy//wpTQLK8XbnOaQ4+H8gqYDCdwh0yImDV/4fjs3BEUWgBeC319nUhEU7BbqaGro9WZikrNLTBf72HsdOP+JrsHRsxNEfeZ4g5SuuoNGkRwemYGG6sNsjnkikl41GIbY/XSLI6OirBPLAqGJpSaSifEVGJ9kYj5MJzQPQuYC03gHA5xfy+HsTuDRDKIrH+Aw3wN9fYYg1YbxWYPQZ9HngTxREKSrhtX5nH33nPh971BT4DXznEerWYbQ3IB7FaEIiSYEk1jHt8IyWQMkVBAOQekrtMjgP0UM4EG/bHOfIIxBGF4Jnf6lI43MByTCGrec05XnU4z4MkXTCIYSeplYjHHWqzXrOHkcAv+ICVjPtTrprN4NpuS2fTx0aHmFZwqkpGlxBOPH1ubWzoO2GG4PabiiDZ2mpQyo4COo5MxjP/wG79otoEGcWb6A/DDmXgxC47paKIdwKSMA60KjRpNxunIMsCEtqgj1t4jBMgsinhQG7YFUzbzTVxYv4HDoyIarQ4ePD3B2G6FzzbEwlwKw/5AQghz9VlQrrcleOAuwSKIRlVWuwVebwhex0QOXiymgl475rIpTK12rGRi2Do4hdVDpLCKGb8drd4EjakLuzvHuL4SRa7SxmI2jbv3N9GHB9ZxFTaml48niAS80gOwHexb7MgkU6jUarh2eQ3f/O77aDDkqtUS/4FU6uG4j8tXbyKXL0i+fvXKJUzlxjGRXDscMS12eAYf7m/LepY7JC+XL4yd7V1ZzDV7Y0QiQbW4FptVrmTR1BwCwajGvxTSEMzaeb6FCjOJ3GaW4cbj++JPvvzqqxK2cPfK5U5k58eLhWM0Fkf+LKcjokpDCpl6mAX+i6jbkTwDpzB+61f+xfTFB+Qf4I1/ISH2+8MCacjQFVXMYOHlkVbeZgxQ7zQx6UyQL+RRqZ7hwvwKEinaonOyyJg0JxZW1vH46TY+2drXLN6w+xAPOuC0kBwx1pCC+l6qcEqNHprdHhyDBlxerwYZhs2JmN+BzEwau/sHQt98wTDCHg6dhoiF3WgPWZDaMbZ6RELpdXs4qQ7hsI6RjAbxhZdX8J0PHuO02MTAZkGvNsB83IG9Yh0hSsK8frV4dDYlcmdxenB2so8nj55i7ebLKOVLqLc62pYZcROL0kKf2UgVzesvXbokMgfHtKFQAHabGb5BuVhP8nCH0lJF8eHx5Qrr7dzbO9TPSRZRIBREKJKSbpCDtnjYbJ3JBXy+eyRix8XVBSTiMbz9tf+CH/3xf4SAx67hV6FYwGA0gcfjMrEDB/2T+th8Tu8GU5TqOJ8CUnr3ApH9dAEQkGBVm0omdcY3qiV57HGbGPSnePDIpIPNxCOYX5zTNIqTqGqzjlYjj153hNHQDEt+663XkI2FwAlfv0H+3SmaY7esZT7cPUHYE0PAQ2rWqeRVLK54vm1t7yJf7WFMBy2jgyl75UBC1KfV5QxK+RM83y+ZhBM7EE8kEfS7cJCraTgybDYRCLkQjQTw8SdPMTubQb3RFbdvbTaKrTy3fTseP94GLDSZtKPWaIuxxEIixO7EMcXOwbGGNoye3dnakxdwMBJURjG1dYyS8wX8mM2mVFDl8xVcWr+so8tlZ3TbWMUiqe00buKUlGPk/OG+sH/uJiySGa5FKjntd7nQ7cLrPYjFU6qRbOhpN6a9zIOHG+iPrIK1F+foWWyF2+vW3EBRvN2e6QR2zv0jdvFCQLL1/Ln4jy+S0JQjfP7naPFn/Ntf/bkpzw2Tx07uGVDMHSEQSotQwJl9sVDAydkhlmdnsLC0gvvf+5Zp6GhlwpgVCZ8H9lgAdx7u48rLryPo9WI2YIfR6yF/fAcTVxpjZwTvPcrDTuZOJMzxnFq6ZHJGFrOPnm4iV2wiGPLDNqaxoll3xKJBuKxDWcucFtsilCRDDF3sweFP4fi0hFanjFvXriDiseMsdyjaOOHpQq2P5dUFfdTNZ1u4sr6K/XwTzXLJ7Bgm9DkeSd0cCvgFRe8eHsvJi+d7pZyXEUO12UOz3Ue1wuibnuoVr8+FC8sLSgvLnRYkzWbFT4o6Hzq/P+XhHNykUkmpcg529lGvVtUJyZWs34ONvoDhpHSBHBnHEmn4XTY4jA7CkSAePnyK09wp8sWWyCFUcMWjQSWW0bSKWIQo30Q+p6SKU8bGKJ+xuA/tdg/be0cirXJnpFXci6NCjuH/5hd/RjVAIpaQNp2ra/vpfTTrbay//NanFuOP7n8IDyXj3MJ6TQSooD0vDr2REC7cWMTv/DnNIQxcufUmRvlD3Fpbx97BPpz2GqrOKB4/7MLjo2GTD7VaS/6CDGdirUH0ijlDVrsHo0FbqpmFTAZzab/CDzsDC4pNOolUkcrMY/fJA8wuLoPkZS87i6Ad40FfgI3V6ZXKptUtYjqywE8qldWpXMTHm4fwOgyQMMm2if7A8YhfY+psKoCz0zJq9DmymhFuIa8VlUYLU7sHTS6E8+weonhcNK+9/pqUP8+ePoWVD8PJQQuPQBpJNETgZIt9/dbLcNjdMpqolusol8totrtweoMIRplJOEGQzqdOO2zkH09amJ1LI18sIp8v4e4nm4KSKaPn5ScDA/yg8gAAIABJREFUKRLFgAxqh13CEjNdzTSCGA7GaFFIE4rjydMtVMumzlNvOC0L2HVRZv8rP/+lKflqs5nZ80GCgZ2nD3Hh2mf0x0nP4sPn5MjPGLlqXtAi64NANKgvArsN116/iN/6/e8gGI0iMz+Ho3u39RB5LV27iI5lgMO9Cfr9NsKegPpnTAwEAy6peoj+VaqMTmHWSRvhaBJ+1xjZRAj1Rl96gaNiGcZorDqFnzmUXcPJ6RnmkxFYJ+T+WzFo1RFOZdVBHByewBdy4PmTPVx94w0tkM3NPcyloqh3eeYXkEqn0G0WpSZamktiPDTgDUbQHTuwufEAFgwQDQZRavbR6vdRrzSEsHFnbFRK2r3mFi8om4+jEXYLLO5KZyeaQnYZmDEaweW2qSWNRmJyRaV9DEfiVldA5lJ0T/H76ZN0Jto5Y3jG/RYCkbD8FfeP8jjLldUGRmKmKUQ8GtWCp50+6za6mrOyJ7eSdRVnBrx4L99793tyejHF3n8fE2j8y5/6P0Q/uHxpRVM8EjbzxRN4vBF4/HF4/G4VMEe7GzTkExnK0W4glUwgkIjBEUuj1a3j5RtZWap87/1NhKMhTAaMlq1g5uKSip9ao467T46wEM4gHI+oZiDLqFIqIpFOCdw5OCqgSN6A3Dn98HtciPtt6A0ZTmFFa2xHtztEKhnBoNfDUakpPj3fGhZ79+8/xsW0D+3BCJFkHJ3eFG4b8Pz5PuzRJKz9HuIhIFeb6pznm+tzjsV34BbKXeClG1fwdGMPF6++gUcPPxKVe2k2ieHUJneOarWJbqsri1xCJ8XcMbJzc0jOLqjFokchfX86zZoGQ40anctc4u5ZrGM4rIyKnZHC12anHa0XVocXHn9QrOjcwZbItjb0RY+jaZV26NlFPN/aRY4BWTy7DQvWL9IKl5YwkCIrFPCZtHRSeywWNOo1JZ+TJ3B8nNO9PTrcV01i+gp5YPyHL/+MFkA0kgApATwP9vc24QkkEIhmdCZzw9h8dBfoNORazS9IAISXy+NAo2Piy6mUH9vP9+FweJGMBiQKnUmlcVo4Eydgr9RAsdRBNj2LUb+rtpJCB57rDpcb+WJdW9Vsdkbw6oXZOEb9NkIBN0Zw4tHGAeKhAAyXB+N+R1TsfKUFuzGVYGVktYKRgZ3JGNcuZHGUb2LYrSPXMFCtlhB02bB+aQ4fP+A5z9Ep8xDa55m6Y5lBZxMB5E7LWL76plJIvvWNrwmyXV2aR7k9RLtvZgN1mkzkNCNkWrUSwuEQgil6+wSFMbCVY01AVw5TkcztmQrICcgTJYEjJPNoO+xODxx8GBYncocbevC8z8RlWLSRKEPWPo8Tuoq1aVR1HltD46pYNCDthGJj+x3hAnQeFekUUxWJXPD5fBlbGw/hpHrae5718O9/9V9M6TsdCprWLgx6PtjfEmTqjZhOX+xvN5+YnUC3VkAyEpKPD+XLBGAJgzLG3GXv4eTwRKPIxcVFhAMMg4yg1GwLe291m3iwXcFsOiMHcvIH8idHSNCdxOKUyrdQrKrVSieiSiejYzYr5Ecbh2q/mHvX4aJh5T80sX2XlUcLPX3OcP3KJZzkKvj8Kyu49/SYoyf0Jm4Nm0Y8Hvx2wBGUAymNLqbDGjLzq4pu4wjWYzQwGNvhCqVVI7z7ztflmXxzfQmGK4CjXF5FHNtiEjJ5HFTLBditZq7wm595Q2c+2cm0nSdh8/jkBFY6fLqYOzdCt03zDDO6JhxLYzq1wO0Lqd0u5o/h93oEFA27XaGG9ESQqthiUWFJ4UepStYRaw4n1lYXVXgST203SgK3eNFCjou1Wq1id9804X7y+GMw34FRtawVjH//qz83nU56WpFmUTRBgy7XRONs1LXNwe504PT4QKZIg0EHszNp8ex5jQYt1MpmAATFF5fWb6kCFb4PA/OLGRRrbbSbdTzKlZHPjXB5dQ50+q7XmkinYrJdoxsY3/pP7j0VqjaXScHvdSMecqLFLdfuNWfrbrcStvmZWLRyiLV3VJFh8kzci6DLgNXpx972HrILczg6PANDknjm99ptOZ9ItXRGzvxITGdfKIFWs4JkPINu/QzxVBoWVxiHByc4PtrGab4Ih2WCmXgI4fQCGl1zByAtnbQs4hkEhRxaBD587q3P6u2lOrdcpellVUceiZqL80kRNdoNE9FTALTNJddUu92N3Nmh6Og8KljN85yhySa5BhbxNczUUx5JrC3IzL58aVUt5nRATwcCaXbT9MrhUOVfLNdQq7fw8Ud/q9qFOw2nhxKe/MG/+7UpWwuqeswdwICF4g2aQwxMelJmNoFHj+6pf+w0auLVERTh+PWFa9el1Tm5YLLffP7BE1iiPswuXcJ43Ea3UkTbTX77VK0cJ4CplFerkEbInc4AkVgIO3vH4rpx8kdkjJM5avP5dkytBuJeKxqNHtKZFI7PijKkateqyJcbaA04PQshEfLIJcwfDmI66uNg/xiDsVVeP3wbHXTk9LlAToaTeohJC+0eWUzk8TNboIVLly+iO3ZhMrHij/7o96X+oZomE3EgnFqQkykfHg0he/2Ohiv0BKIjGfl4bKnfeutzCsI4yx2JlNppdQXl0vvH7SLwNZIKihcZO2QY9ybMA7bKd4hHCJ8GfQvo+UfJPt3ZmcPII3lsWDEe0HXNomhcdQ/s0sYdcSfo+0SexqDX1/Dq5KwscOjRg7uavXgZi9PrwvjDr/z61EqXp/Pc+Rcp1owkm4LOlnSiDun8evKY1KM6SqcnMlq6urYglMvjsWkB7fztHTBjMXXxEgKxFLq9ph46Id2hzYtSx3SvVAIXqPbh+dRHpdHG0uK82EdsQzX6Jd3KbpjSc1KbRk2xggMBr9xKyVLK13qI+gzBoGeFjqLTstk4nj09RDhsh5d0rUYLAa9HbuEc7MhOLcA0MANOqpdeLADy+C1TdAiC+exYWn8Vo7EVf/xnf6bdi6zbbruBuflZrKxdxvbmU1jks9PWA+SxxyOJi6DXaQvNu3z1lhRETBCh4YY/kMDi8kVV4mTrtJpUR08x7LZVk9CB3eVh6LT5dtLwmjR5zgpcLq8yBacWC8oVFq0GSuUc0mlTPp5ORGCZ9iR549dMpRIqBFWIEcLvT/D2178q9PQF5Y8zA+N3fvOXphRn8mLxQZcNcgH4YXhGmWIxCKR495139N/rCzFkM2mEfE5pAQp3P5QT14jz8yU/FrNrsFgDKBfKyGSSqDVr8rXfO81ja2tHXsEz8Sx6hgXHJwUM6XA9meLll19C7uRYDFhud247fYqtCPndCIQCyhFkn805d61O/wKTP8CYV1q7NNstXFy/pISPRrmKRmuIXL6KaMAliTrFLt2haTXHBUBnk1TMi6eb+6p92PJSqm4Z9rF27VXBul/79m0U8qeIs53cOhKwxJn8/i6HLTTWaGkno7EUkUKOoumjoDApnx/0W+ax4PYGEYmmFF2fyc4qRrfXa8sPgJV8sVQxg6OFCJh7MVnFXieDuk3WTzSWUK31bPNYCSVUDXPQFPQHhR4quk/GlVYsLS9j0DNJq2QKccf5+tf+i4ZplP2bXoV2cwFwYsXxIFcv+3AWHh5/WNvilH4657//3nvfZG4pPJYRvvgjX4SNyZbVnEnsDKYBmwP+eBWdmmlFngxEEYmb0qbTwjGePi+iP+jg+vWL5s2auNEb0i3kQBXupbVLCk96cv9jLK6sieuWjPjlKM7bQlLkkKPbYEITuWq1BpvDpkKH349nZ63VR7/VwtxCCp1eD6dnLXjddBvrIxiIotaoKoaN2zKL36trs3jyzJzx03qOO92gVcOt178gW/g733mPmfBwEWXMVzCRBUwRn/uRfyDjxkqdkXqMgXebZpJdk7TZpycBpWjU+Plp/5ZGMp1FNsPaiWwpwuk5ycdKpRqqzAUiU9cgtX6ioQ05BcQErIY5OGL7RvDsbz94Imp9pfp3Jk9WKwJer0wlrl9fRf5oWztIIpFQFC2BLvIbp1MDu3sH+OD772tTmMIM0NQC0Nt/bgQ56rUxJjvonPDpdgfQ7XfF1X/29KH2g4zPjatrq7B7/HLNGnlDqOTPYOXUi2ygaQ/RgBlBwzHmnY/vqN25fMXsHOq9JtxTN6qtIRrtIU7yeWnlOdIkF+6Tj24jlkgiGonCNukI3JhlUegPoMVevN5WKFOxWEKx3JB7xkw2i2I+p86CP9jy8iwqBUqsRhptD/oN8Rg6IwMeu1UwcNDrwmzKhXJtKPiXIlQ6ebDboAKakPUH330PQ7sL8URUAZKMzeEbzjOW3EF2JQRn+PBZQ5CWxUXAt6teKUuLx1+/uHZJ4JfGtjqj/TqvWYDv7J1KmWRxuEUEYcQ88wfmFxZF0+ORxRExHyrbOfIot7cPUGt0YbCnn5quZNevraGv2J4pEoTQXQ64PYSFIbrbkyeP8Xxz89M4XHovGr/z6/98OjYcsNndAjZ6HSZU9QTJKoCAZzWrfQO4fPECbn/4kc5KJozdfO3zqhGcNrtm35yMObwjWO0T+KdOdAj05OhpN8AXf/iHhEOXm12ZQtNgYdwfotIYahp3clZEu9vRNke8/ez0WMAGo89o8ZZNm2FRlWYXtmkfTrcH5EKwdeqyQ7JZUWlWEQmFzfZsNBZO4CZOUa+Z9nJ0jLC7YGdsiraAMbIpP0rVpjyBWV8Q0eNDWLu4pl3l/ffeFgwcS2RkZFUuFlSd04/gNHeGNiV04hOyIKMrh2mzS86CUkRbVAAzJNKOm69+Vrss71k8Qi0f1cZdVOtdFAtFbf+xREx29PlSGdnZBXRbNWQyCZFjiPuTwl9rtPDo0ZZCIfm1z/JlXL+y+imfw+NkhgGZyQGEAh75L/At56K5e+cjlCsdYSDyE/ydL//z6cigibF5znCuTwk2ARiLk46WBrJE6uwu2Nx+PHps4gEkNly/+aqSr/kDlfI5eL3kA04RnB8gtzGAMTKwtppBPB5Bq1yDN+DX1tUcAsUqjRMrwupdXh8KlYa4a7RNYU9dLZ1h1G3C6vTB7Q9hLk24d4iA14Za2/QzIELJipY+APy77lhUFCh6FbarTXgDYYwmA/RaHe0KvXEDnW4XM/EMphO7blLQORDoVahUpc4hJjDsd2T/TlBl48kDfe3uxKGaIH96DIdtis9+/osSdNL2hbQyizz7xzouOWzr03Tr3D6WLR+RTT6spdVr2iEX5rOwWwcoFGooFeu6b9QeEngjUMPvw11kNLGImUznNY/brvwGtnTPNrZRqbeUSk4hTKtF7MSPQMCUomtoFIvgAiNpiN90e/iE1nbNJp482ZKvAi/j9379n037tDCfUOFph9POdTjFqNfE6lJKbUahYaDV6+HCxSv45JO7qt5LBZojRKSndzmsGofK8XLYw6jXUVr3lZWsPsikPxUzhlO5YCymdLBnu/uYDg354/NinVCotZGORzDtt0UtZ2I3P02t2YbP7cD6haxcPqjJrzcpRSNJwxx+VJtWBGJBDHtN1Cp10FKXhR4VS2OQfzjC0FKDw/Cj1zA19nOZRaRDXqGUhsMnUwpO07gt37z1Mu5//LHs8ukESotWmlBwUFWvllWw+gNRfPd776k65w0lrkA4mJa6/G8FUUqEOVHyWSq7qJ81FEvh8voFuZ9UK2VBtA3Z8k2xuraK3DHZU0wW5XzAp3aUvMi5LKeBDsHR5CIcHJ+oOC+W2uh1x1IerVxYUEFqs5hDI9LOaDrJ1nFv9wDPt7ewublrFvg0qPjdX//SdGo15crDEav+CTJh01duNCaG34DhCAmomFtYlID0o48+QLlUUlHCUScvp8PQyJGX20ZLeRONopcUR6W0KuNWQR7dR3fuo82vG4iBliYEgEgBp+kk2yqueP6d3OkxcnVW1wOE/S7cupRFi9x5mVe7xYJlUcYcv/3TMroDqoRJIe9h3OuiOxxodtbpVDF21RAKLqJRKGLQm/w3F0C53oR9Osarr7+G3/vKb8EfSiI1k8bIZkOpWNFuRq9gFojXb76CO7ffRzCewc7OjpnurfRRi85dvkjddk1xriSAlEplzM4v6qh46wde09dg/XCwd4zD/T05fF67cVXWdI16HdFoUogsRTG8mJQajsaF4z97tqXFSsV74awhRrHFMsbiktkWTkdDLC+m5E7OCJpmsysPh7e//i3zSD/v7rQA9BcsLlj7ZTiddlGXOHTpjixqByUzgoHVNTMZdOv5hgQVrVZD/azHx/qB8SwGpj2SLDyYXZjD6UleBZfF6UUoYIXbRj19HcZwjI3dY/HlI8sX4TAsKJbzsMAtzj2/DuNmDo5PpfcnK8hj6Qqh8zgs0ifyyJmZWzCDKGodTIZtnNUHZmCDww0ve/RhU8EMQ5IrnCRz2jChwHTY0g27tnoVxoiKnxV8dP+JRuJnpTrcnOvbgc2nD/VzzixeEppIkysf43ToFUhPgF5Xn+OVz/0o9g629X1JUyfjhu2aSa2aKGyTLw7tYvkmMjrH7bDj8z/wigglJG/sPt/EtZs35P3HopGHMUe2eljjvnYil9uJBMWt3bFEN5x2Mgl6MKCpJWcANkQiYb2YTFLhcTGTMo246WpGTeHbX/+6dkYpxSxjcweg4dP4PKEqEKGej9ZtPM/MmDTD6oDH4ZRSZWZuXhOj77OdoOdt+UwFEfvbcrmiWPfVi2vy+aWJ5NDiRtTvBW1yyDuMeIMqkkjoeLD5HMFEHPV6DzNJJmFNcefjXdNbLx7T0VKu1VEpFuGyDMWgubQyLyygUSmIds1jgEheIOTG1u42Bj3awliQSrvhcPmwt5+HDRZ0ydIl8DHpKZjB7fcim5nDuF3FKl1E6FruDyvdKxGOolo5RfHsFLV6HYnssmoNBlpTjcMagKzfvb1jWNDBpWsvyVaOk7dCkbZv5rSQHELC10TgmjWCYuQY1LUDcOck6vnSeWI61UcsMkmqTaZm1QYyC5nCG+3Ow7FsXthVcOvm9yoWywKOaJbBY5lpJ7zMNPIpLq4sCJKnuxi/N9vmt7/xNYwtDhMf4gL4vV/7SbMNJB3c4UCry7ZJvwKbjInMrYyXnT62Hg/ml1Zw7+EDHB/t6qFxiMP2i7jBZ155SZMt9sKkkVrsFtGReYZfW5tDgz59nAzuPdf4mFzG0DlrtdAcIZqM4WvfeYDVCwvwuu1Sz1BNLBaO04qwz4NMkrE1PnUNsk0JRZArlkTYJBz9bLcMf8Ai/nzptI6w06FKvdjuK8rW7bDA4fHKxnXcreDS5cu4/3RbdOrTkxLS0SA6vZayE6lCOt7bRrlBW5oumMCdTiaV/kljSj74aNgrLj/ZxFQFbWw81nFArJ44BT0DiBI2G211PLViXsXwzExau8nrb74qqJkKntXVdekhh0QXHYyfp7G0OW2NxWPwetlROMVlYBHNBcmQ7Bem0IqS8fmwvnZBQ7ruuW6QUPPR0QH+9r33ZNZpWtcAxn/8N1+a8o0g1mxm2ljQ7U8VpaqznUmXbG2oj6PnnIKLxmicJ4EapBnTtrRSxc3X3pRjmKZMhkU4NOVUvFZnE4JY7Q4rdncOUC7lsDo3i5OzM7laxBNZ2HweWB02PNi8g5OcW3N1jsMPT8sacpwd7WAmGYWPSWYeL9bmIyjWWxiSQm6nUaUNPfbBoyF2imXTvbvWxUtX13BwmsP959uI+AOYTWRhd3Em68KkV8ba5Sv4xjvvKZF08cJV1PJHaDXLSMZTsnujnOrunU9geMOqiSajqTgEvVZdmkLmEjHC7uKVyypmKSWn0pcFm93JuQpRv6EYzVTmsMcnmYQuHbPZWVnFX1pf1Rk9MzOrnp/KIdq9sJis18y6x+unMxrDIMyqvlptwelmqEVbOwSLY/5dvsgL87Nqe5nbQDk4X7rf/vdfNp+pLymkUiFfv6sdwHzDSVRkO0iOPfF16vPIMuXVb9f1wV2+ILrdjoQJ1O7zigR9ghjZKvLt5/lnJ9RI5gnZBr2mipNTARxT7O89wU/8yA/rQz26+xDRmTn0hg2E4nPYPHkCR9CO/Q0rAh6HzKLCXgcOclWUKg0Yk67gz7l4AIOJBfMzQXHqDZdbGnyqkTp9RtEY6JC/N3TgaH8fN9cX8ey0gGGrhqjLL2YxVbjdyj6u3nxZw5JStYWVtRuwTDv4+Pb3FOTYbZURTWZw78G2CspGhzZvHYlNSEbh9krwKxSMSkZOoyYGRt2/f8ekhtmdZn4BBzvdHlpt5gIwi4BRMVWlmizMz6v2uXLtiqRgFLfw4oCHUnemhBDDp9EV20HG03AHaLX6QlLZdjKYggMuj8eOhVl2X3Z0OzUhh8wIyOVO8dW//EMMxqaDaCw5q8Vh/M75AuBf5iiYa2FA4YLd8WmvyB2CvSUvh4eZACMMOw3cvHZBkC4DGGlZQvDI5nIKUCLLmCPKfq8FO3/BYCyaV+YJt65dMAcnjQrK5S6a9RJSSwty1zg5KcCT9auNyZ9a0a/k4PUE0IUL4UgUm88em3bwdoogfHDbJ7h14xaK5M6dnSBMhxHSsbmNThgobUMxn0ex3Mf8TBTDQQePNh8hGb2AS1evolM8wI1b15Ur0OwC0WgKzWYex4fHqllo+ERZV5mC0dYAD58eqMrnRG866ethsJpPJ7M6RpMzaVRKeeXz0QOAgywzSsZM9eSby6OPcG+ftrjn8W1z2RkmceDV11/SfeZWz4sLrNXqCRMgcJNMxgXCsQMiIkqv4Vanqx2chplaODbTUDIS8SIQ8Os5vP2Nv9bv7e4fy1c4HDG7N+Mrv/LTZibsOdlwRMiPQQOTv48hp0BB1mOGobEqW5z1RZoxB7C3vYHByNBwgoVbt9fXmc23lIia+HAOi+BKRsh89vWrevNPT+oIe0mNtmDz4WMkFi+iO+ygP7UjmHBjNjGLr/7Nf8WoP4bf7hdvcOXKSzKX6HSJObhkGEFaN7HwRrUBh8+PSqWAdMgBhy8oxNECOw6Pd9EolgF3EHOZBL7xztv/zQXw8iu38K2330WzVddghcKVmewigqGQQq6fPt2RGxiL3KUL85qOcgdYWb2EKuNc/D51AuQclCqVc2cUmwpPvvW1esNECpmQwjFvl2rjNhKxsIppinNu3FzTERrwBTAYjWU5w3kLI2bn5uigOhVhlfe61x8JxiYBlFBOPJaEixZxToteWhpEUnhKqPr9774njsLM/JoZAE5lkBaAuRZUXRJw4G9yLj017HoruZWZ/5gVaSLilU8+4+PJWCXFmQVHtV5V4oY6B7pys7bo1PTDUOz5mTdeV4jD/m4OqiZo/hgNmPN8WppanbB7XFjKROCOxfHHf/6nQM8Jj9dAMjaL1YUkGgMLRqQ716krPNPWurI0i/5wIiva2YhbgJM3FNYMgN0Y7VpJzeKwKhrx4v7jjzDuTnH55heQP34m8oTbE0Q4sYCLK1n87m//EbsrTEY1RMJRZLKLylGk2zgp4BfXFnF2VlRhSaNKmTgGg2jXTcv95Aw7m7oKYbp5SXhLOHbU1wMlj0A5vudOLCR89Jp1GVNyfmJ32vDS9csam1M3sX9wrGfDaFtemZmIagkexXsHpiN7ozfS7ILkXj7DTot+QlQKmZK77773XaG3bEP9UfPrUAGmBTBmkjS3KcO0i5OdK5eTisDziBGbVdErXOHpTBqdvjLANJU7PToQz55mhYRe04o6mapu4DXqVfHZz74lB677dx8hHnQqu8cYD+BOLSjQaffJE3MAsjSPQNiHjx88Q6s/wOLFFIZkLDFA2jWDcqWM7IU1HB2y8rXg6HAX9UYN8wsLmNrtiFj7pkO22wxS5A9PKJaext3RFCvLM5JaKfcgtoZqmQYYQ2Qzy4inZpFOuPCnf/I1sYD3th6IlZRIZ7C4fAH+oJn/E4vGNA4mVE4r2tGwC0ayzM3SWzmJSrmAw+0NdDga7o1Qqtak2ePPzwwjvhB0CSWuP1W+0Bi1gukizvwjaghsdgMXV5elpqIXodsXgNNmw/LSjInUngdocTjU6pkV/cqFRe0yVGnx6bnOLe340P/iT/5Q5FSH28x6oLeiaOG/9ov/auog3VtWZxMM+3SaNPcEi8Nr6j+cHgybObz85udMXvl0gr/55of6M299/gc1QNnb2YRhM0Qb404RjZjpY536CT73mVc1fj06rqLfbsM56cJn6SC8eA2lRlvnLBdAMjsjkcZpiVBuH4FIEMG4G/FkCJXTGqqFnsKqFhbmMKITVm2IUvEM48lA6B9FFTE/DSdHkmnR99Djdalg6/dpPFUROlYoncEytWBijaBZO5NLx3xmSa4fzWYF73znI+ye1lA8y+thzGTiuHzlstovxstlszP48PvvK8iRx2G9cqZjLRqOCiImQPTs8QPVH6PJBAVC092eJnY8u9kWsrirN5qyvOFghguVXYXe8OyMADby/DOzs+r1SS932gykUin4POYDJweCtjLNzhSXL18yd4JGRQbTw6GJHqZpPTcc4Kt/+efo9ac6qrlz2WjoyZ2aC4CwIwWXTotJUXqx5fO/J4wZnxLsuKH2gtYj3PLr9abkx2987gtSptCPnvbxvOgqunrhIiqlY/zID7ws+dPz7aKZnDkcIR12I+C2o9Luq8rnNsdF47CQ/GDB0+092brPZGJw+geiW7XPWNC5MLJbYJkOEQkE5cxB0IyOYtQUkO3jdJvG1ZZpV/IssmU5IaRbR7FcgTE23TiSiSxqXULVpJnHcWFuFjNzWewdHOI//ac/QseIKe5+b/sIyXQUb7zxJjxOh7wIaHD9x//5jzX2Tc6uaA7CKSo9gZKpFFqVgo6GcDShjoohW/W2mdVL8Ic2M9OJRXA6L848ZLDZbWPYbcEbCIjOxQKT7eDSypqEoqwXomE/5uYXFWzFRccCsd4cigHk8RDgMUfTDJQgL4Ct8J//8e9/6v/YbI8xnJpHC9lcxi/93E9P7RaGOplvvdtBw8jz5Hl6+I0GSGdnNcee0D52wsq+h0q9pyLLBpCNAAAbqklEQVSk1ejhpVdviXpEHHx7awPF/KFIl//L//TfC1vY2zmW753TG8JoaPri2KZMyxkrp4fnOFMzRt0+9k6O1I0who6YwnjCoOUuMKSdrR8TC/kAXRkveXwWpEIeVed7h020ek0M+xwVM1JmKMyB7uZxrwcuPwUmdTjR0w1kzXNaNj2Iaev++stX0er08Gd/9uco1EcIJbMwRj0063UZM7z6+mekx1tdWcY7X/0T5JtTuMjyidFVtSMTR04XSWWbTSc00+DMn93QcGzgIJfXmU5IlxcHSlQcE2nkhsvUdnYRrU5T/gzkFVJWRjcQuoMyuYQBFETwaKXL2QqDIQguMWaOf57XaNTHeEybHtMOjsehFgC5hyML+gPuACYnkfwH49d+njZxJEhOTWKhYbZ7nEZRNMorSPKlOeeBy+NTaCTn60KaugP4fCGtQJ597IFpouwcnUm7f+3Wm/p7lHaNJyZzhnH0jDw9ePpEti4zhHddVtSajFC1oNWuYzYzR/iJPCAMHR20Sm34PH54nF5Me2M4/W6Ui0WszKVQK1cQDvhwUmnik40zbXlK3rabcveI26n0cUak2Kc9LC4uiM69Twj1fAF85rOv4dH9x/jw9m04vHEMmHgyHWv+T3g2m2F6h4E3P/M6vv71t0FXfOIcyWRarS0ZRwShQoEAItEYxr2mEDm+2ZSx84bvHJrnvIZGtG6rmSxe5g0w2IHiF14EgOinzIVMqTy/DnWb1ADYrYZ4EjyKiSHQlk8JI5pADlUb0HgzljBl6nduf6hcodOTE7WgrD+cPlNZRNGr8av/6ien5ADyRjk4umPhpwdv8u24ZSsYwmZHKJxAezDVvJrnDBnA/IIWi1P8AY/fh0ePnwqt6zRfaNGAQDihH4gAEd9I1i+ZzIwkzYV8HqPhKfzeqPDzfrsrz0Fuk6FIHBPrBENrB+1BAV47MB97BXMzWQVH/de//CtcWlvBoD+Sy8akU8e7HzzF2O6H1+cR9YsO3xRuTMV4msKhBbCIcqWFMb9fp6VkzjfefBX3P36A3FkejfYIxVpDgU6jfgse2rwWKAD1iwLGmoiDsGx2QYjk2dmxMHr2/ay8PW4PogEWzxRqMJrFKkbvca6Aess8CsgxKNPfh+lm3LaHlMt/atcuqTexAAI5PBbJhqLRtHU6QDjoE2bAq9ljd2YyibWz1Ew5GNlHh4empS/dTB7cu4fJdGKO3xlKvbRkdn6/8Ytfmr7QirnBaDXzC/LXyCOjrSohRl52b0ygCLfs0XnwEREnun7whzBIX/I5cVasI5FKoNVoSARJ4ILt4enRJtq1IeLpNFJpsxUJ+c1VT+nUmF6AXp/Gos1GFe1qA5GZNBxJGzwO5ojaEA4m0G+0tb2JfaYppBUOzsBtNrl905/Qa7eg3h2gWmhgzD59/TKurc1ib2cPCwvz0hu2Bn1pCReWFoQlbO8cqHUjoEVBKL28KfVOpxKSYne6IxyenKI/Ml8OijVXVtd0HvMmHx6aAZhzcwtK+8LQpInz7Z1Q8D0YyuWbhZhs2ydjVGpV/Rmpu9lqKv2bL5UhX0C20gzHYOgEnw0VSFE/ef0OnBZqehGJWPLzFM87iUgkLqyfbKmT05zcxHjt7+8Ke/D4GZBJQcoUxm/+8j9Tze+csoflWcQBhBkgwBWpMSirRUdQZxV/j4xS/psTNmIB6hrGTaxc4I1tY+XyTdy7d0+5uNTD0/uWI0lG0DidHhxsbyOWjMPjNcEiOmgrCqXTkqqlUi2YGvY+BMS4/G64ooyrbSBgT4q93Ku3MJeZxdb2NuLJNGaicZ3R33z3NmrNhmJsJzYX7n94D3aPG7deuoW1i4s43HqK5bUl1Bp7OMjZlSxOfcGD27cRTs1oxl5vD9DpVrGysi4xKqVhDL3iwqg3Grj/8DFiyaSURJfWr+qYOGOMTCAoDoMZ5OyQ6phbPGfzfEHYilIccpKjdR5gtTkxZgtYrWlUy5vOv8tJIN3K2QWwiOOIWxc9fQwDC0vLiPpsaHYYt8MAjonYxVxAoVBUOgXWaTSnolEFKXIcnPExsVgP0l+QlvqGA8Zv/MJPT1kAuq0jFXxaDEycVANo2omSE8eL3EFenG2zdRn16e5lwGXt4JVb68gV6qIpzcwuwh8gE3WKux99hEajjk6jhNTMohYAP3AyEdabS+s0MmdolZ6c8cNpD6DSqIJBrXR9dgW8MDw9hSVyZEo0Mh4OIn9UwNVLF3UTF5dWFP/ChIzN7SPUmb7lsGNosWFr6wCpZArRoAu3bl3H3vMNJDJzOCueoNGxw2GMRFJ59vHHcAUCQsrc4RTa3Tpi0TjslKx1SSYJiDnMhXrn7l21tT/4xf9ObRaZxKyo6ZRC82ZO5vj/MtwY0mOop8/NvEA+eVLAad6ke+2hm0gfR4eH52EOpt8yzak5h2HX8AKG558nRY8Pj+GWlVJZrmJNPvwpiR/UJDrVZjKFhKgOuzVeXABsWSeYaHd32Mib8MD4yi/906nLStyZUfHMADK3e0rDBlR58HeYbmVzy8SAHAFeU6p7LRb4nCMszsVkjfp8hyLQsXj72fllRKJxrcSH9+4gf7ILT8BceZFQTImhywvM6T3WTjEa9eAP2tCnHo5nWCAj2hJj2Wk+VWrWMBz1MJiU4TECuLV6We6dLGBbzAJwubG5fYpKlUbNPKQUgosPP36GucwMsnEf1i5ewMnRIXyReTi9LhRKdYw7VcG8xZMTJDLmsVSod7G1s4Xl5XVYxz0Mu009EDp57u9sKbY2NbeCy9duIBSKSGXLbf7gcA/7B/vq6WOxc1yeUTKcslKzP56oNWQBfXRyLPyeC4AXARzmMDN4mz+/KQzxyxGFQBXHwy+u1197Vf9JXwXea1rJs+XjxSOMdRVTQUm15w7ABckoGafHjXa7DR4RxCG40xj/9y/8Ex0BDjudpCYY0XeNl+U81WrED0eZshlZbg42iBxaYLd0sLKQFDiys1+Qtq3Z6aNGQMNqxerqZYRDUXx8m8YRpGbRIs0cNjF6vVE7gzGlKteJRMpEqKg1ZC8biy6ITMFatNFtoN1rymhyamshYvfj8tIiNh4+wfr1m3i48QTJ2WUUThs4yx3AH/SIHvbg4TG6oyEuX5hDKOTGwsIsGvUWPP4MQpGgnDc61VONfMnenTlP//7uBx+rJXQH0/A7mGLCsAkSZzmY6YqHt3RhVV57fAlI1ybx4uBwXy7p5ODxptNgUn+vY/5MnFrq1wO0ji2jea6U4kvFWsb0YTNns8RWaAzJ1lLsbKKI3RYWl5aUs8CHz2OEtUmtVlORSPDrYN/EYlioU3vJi8YbtIinhyALQRJfeBUZGsUFwBZCmbPneXY9AglEp4Z0lRrB8QIOlgGx2W+Ou1UsL8ThcTsVzkz8uTecotUbYcIaYjpFJjOLufPkyyePPkGr3UarM9IUzIIuRn2CRztYXrmkDiBCDV6H+ncrYqGkbgI5b/UuHTUYgEB00sCrS+uyUP/k9m1p8zzzCwBZvmOvFoAv6BUP7vsfbsPqc8Fn6SMaC2I2k9WkMhTJKlPY4QSe3b8tP1+6h7jPx7Dvf/RQI9jTch/pCNVJpu6uQwpcIKgFsHxhVYogWsn6Walbrdh49hT7e7twB5jQxeg3k6HDY4bFGl09+TApuOWLQP8fvpF84Hy5+HtaVBy6cUt3OoQb8P95ZLArSCdj2u0IRnHn5D8EjHisHh8eiuto6jpHarVf2PYrTkaOZeZxXquY3YLxH7/8U2ZqmMYEZATVJbUajlgAQqZH2m58nEVHdUCMulUkw265afcmpm1spz9CqzuU2SF/IE62Wo02MtkMlljQAPjk/n2c5sxe2Aayesc42n0m06T5pcsydhpZyDMAUtGUpmzd4UhMm1IpB6fPBtvQwI21ixgNxtg824LDQz8gDn0MGEMb6jwXfR4EQiF849uP4Ax64aHpFKaIRVPKzJuZmRO8HI1F8O2/+SvtAOxA3IRfAXx0f0thFTS2qjf7WJ7xY29vT8yneDKu8/xHf+wndN4GgyFRsTgW/9533pGgZqpQBxPdZAvIxVMrnkgpJENOLgon42InOD0zZds8wxnowIfLxWCaOBia5ZNRTRST9cD83JzGz6SdERPg1+fnopy8VGRwJOS9xIvRdNpxBwNNZ3lxJ+dFiz7uBloAplPoGGPqAzp1TCweMHKGK4kPk4Ubr1gspSlTwDmSvo6mDox57UqcMZWNi4ihoon15WdjgfmNX3rlDblT8CoUD7Xlsf2ploso5I+wsHwVhnWCWJrFIathAz67AYs7hCq1ghML7C4nhpUSrl69gPc/eQdGyIpwYAE2ODGqWzXyddDmTn1wEO9+5wk8Yb/cP7jDXbu8ru+/OD+LSIwxLl4tAF4s2Kj5D8Xi6I0s6PQtavn2do9QPnmumkg3bgK88tpLWFlZMaNYAqZk69GDe/p9ysG5u9DBkwRbRdD4vWjXq1qE9Aii3S3PYS6QetPkCrL6Z6fgoC5ThhDn5k98++uUgJFHaNYofPAvyDisPcg1UEI65yNls+Vjt/EiH4itJb8+6yLeWx4PvDhN1ALojEjYJGpmiNvnOy/WGHnOXpKxpLxIxowGbNwCzMo3lBZOf1pu6uFzhZqXIXURVTIaK0+oN7AJAXO7+WemyOdPxahhwJHNzrdhhInbLACZhzcTcaM/MeDxhtFoN2REYe8NsL6yiIkxxsfH72MysiDoz6BPQ4eOG7WzkuJXuXWyCv/+fdMafXZmXgMaau14LS3MwbCM5Bl8tG8mm8wSmOr3P10ALl9c5lTPnm5iZ/eZMofYkjKa9srlNWH1V9Yvy4SB14d3Pzk3Z+opdZTIIN9eUrj41jMBnX79RAT5oIjicZfQ3bLYdO94f0g143+TkMN73OQYWwgsw6RNE0kWhVSCcwc9v9sq/rhwmo26kFAWfXzoFILwIv1eC8tmE4bCz0Awy/irP/iNaS5vJni4A3M6ZzoNc1uiUfSLixqzdNQLr4dMYQvs50TOnUMGOpu5eJwhiEImt3AebOfulEMg4jdvFEOPTJu0iulsRYDiMI/epApvKGIaHE0GwKAOu48Ioh39aU+Ucmu5ghsv3cD7+++AWIzTaSJng4YfjkkQk4bJnE3ETLeTdz+8jezsmtqwYTePZNIUZrz5ubdE3Hz7699GMGRu+6srS2bce8AHhzcCty8qC/qNJ09RrFZRqzRRrtD3p4fVFdLRrbhy5brG4yz0NrZ3VKjpbZMuYKJOgA9BhQeAQxZohkVFGzsBTiv5kpEHQLMLXtL4K7h7JC9lXmQ+6fu4PEJkKdhTcDUBOWofKG1XzCzNoSD5GeFf+RJZTD8h7ij0fTRNKfj1WZS2YXz1D39zSgDh44d7uHbjmj4weXCkhfM6PdhHIbcrtu2tl28pSy+cyKjIu/fQTLQeWcxFwSkc1azmZVINe31DcnBeQR+HMw4ZPKXSSb1Fj588R7F4hokFsPrscNhdaFZO5Hc7tRkIhOlV2NJQyVZs4ZXXX8X3dr6pmiORIGEV6J4ldY/HTc7ArZiZy+Bb3/gWDJcTc/OXEKaY9cqKhlK8qFHZOzrD8VkHjkkJubMG1tcX4fX4EU/E4PIFxPKlzGp7cwPFap0pWLp5vNo9HplTzM4tyBmESKA/FBZljQMgpqbyokcPx8m6GwLZrGoTyfrd29vX7spKn4ObxYUFHV/m0zc7Me4adF0zL73ygnxZtPPiW8x5DC/WAXxIfH4kn+ySj3BON+POJWR3MpFJBT87j/f97Q0Y//aXvzQ9POSgYIBoIoXLV69j2GelzhFrXlr9eu5QmPry6jJCsZgmcfcfPjU/q9XsCkhhJjGCl/M8GJl7Qq/PmzVCPOzTcMkYdpGeSSGWiKPW4O9BocuPn97X3yXvrd8vIR5fRrNfRCDCuXgfw2oLV1dWYXFYsNs+wnDCrayCwkkJzlEGgyaNldQIY2llGR98/0M0ukO8+ebnZNf20s11NOodVGoVvVn7J0WclYYoHm7q+166tKQZfGomo4EXuQVcAFQHM9aGk07B5FY7ojGyagnY2FHMF/DoyYa2+eu3riMS4RjXhdOjQ9gcdsW1US/BeoHX06fP9PZyF+Bxq59Zzp9erK6uwjjfLajo4c2RD9O5Cyg9E2hkRf0hZwcSmrJ2IBhyHg0vNbAk/rSZ3RBXk5cZtG0urJnMHPZ3n4u5bPzk//6/TtnqDRnxDgPpTBZLnOVXyijkD9SbErFrnm3j0vWrSKaSKJWrKJdrGE0Z8U7+vdcMkh4NPg0tslpMthBNEFj9JiMerWA7Rli/flUf5PiEhRGJGQ4zoq16qpxfblVc6YGgC6NJHwPU0Tku45XXXsZBa0MECKeH9jRTnOVaKB+P0C0Co1Ybt16+qgLwHoWQ7TZeeu0tMP/01VtXML98AY8ePdKb8dHdZ7C6gtjdNhcy2bh8W9fXr6u6z2Ro/W7gow8+xOTc7pWzEe50JLuQM8hz/uTE7GpI9FQQpMOOubl5eIj1m04AOsvv3vm+hkTrl69jb3dXY2CaOfIBEwnkdePmDYFbnCtEIuYxJgt/h+nf/ALs0exAz8WicG0VjC7n/yeR1HzQtKchD5PZSp1OUy4j8URKHcHx4Z7JG/jKL//MdGubQ5Dxp7mytIQhyyfgGiu9gwTREAmcbBusdlW4pBezeCEXnxseWb7c6klqFFhstaFZJ19uguyM+cNYRgwvoFo4idvv30UiOyvCRKtvsmYdzqkKIFq60TUjHPehWDhAv9VElnx5nwdtZxG9wd/l3tvpoGVBo+qWBL1y3IOl00E8FUckGla6FruLV978EQwHTbz56k29Ld9795vK7+n0nXCFkvDSKxhA7uhIxtgvvfK6Rsmzs/QutOC3f/sP9PvMOubDpHaAi5Njal7bu3toNFsirfIhEYLmmczcBd50dhqEqu99ckdFISlhbjct2li5j+QNyHvEopU7JKXnxP95P9jr8+JiEih2jtDSKKPD/EWYgBy3fQpOKB3TIM8wcHRoHs/EBehTQMSQRxxTTLkA2J1oevmVX/3ZaSFfVmvCzDtW6XyQUb9NXD1WjcHYnEbCtdKJtAMcBr1YtZJAn6OGJDOqDdS2Zsd4NEEs4lM7OBm0tVCu37ypYvGD791Wq3JWKmBx/XU43FxkLAqnoKN1p9tEu15ClZaq9S6u3aBbmRWNEX1+LLAHD2GxpXGwbZ6HO4+PYfTamJ8zRak2w7x5a5fX1WFkEmEc7D/XgIW8he7ABSezebzmuVs4OdUCuHHrZeRPjzC3mBGM+nRzT/mIC9mYhjYOB80i+PfM4nFrawP7R8doMwzgRYvGfxPDl2vn2Jz1GxZzVs+ga0a8El6f8uXiyLindpILgWrqZDyhxcQCjgHdmiNYGZ1rUbFHzyUuRsYY0++PL6F+ZptdxSK/brdLVPQYjZqZvKrFwzBwRgMHQqofyEnQAuBZVCxW9MV5hiQjNjhsfJAWzCxfEyW51aQxEftcIkgmZ4h9MwEZ87szDsYCp9MqsiNlYaxok3EaRo4x7jVw4eJFeFxO3H7/I51brV5XfbP516249dLn5U/E8fNoOka9QEy7i4Pnu2DHFM1E4LAH4PANYA9tw+u6hMePcmhXyIMH6vmCFgDpX5FIWgQL0+hygnQsqAWgxBObA75gVDXC4Fx5y4wf7mz188/DBcAt8vHTXS2A1eUsCmcMeTIhcr4AvNF3Pvw+upwM0bo1lVH7NqRoRqRMh44JikNoY8M+ny0qCa4volv5gnEB8C3mDkC5PQkd6WRK4k66sWlhCVAy324O7ZjDxN2Y546NNRfrk/NRPh1JyQBm/cZKn89JCaUCpii2cal74DFmfPuv//N0Z+u55vbcVoxhA/PzphuFyxeF3eGCN0SQpKeRIy86X9JDd0yFy3iCZtu8ARbq4CjXZhqVYejtJ3bQqpfPPWpJEyfxYSJ/Pw5xSOeGMVL+jm4sDCyuXhXlqVEq6gejJQtNHRPZtBI9x1PWDkO0W1Z00ceo7YD3fGJZPsshFI7hxquvASNq9anTniAW9ouJy2Aq5gEz6p7uHp3zPpmyb07L+BbxunfvgQiXzkAMK6sXhZHkTw/M8e54qGNsZ3tHlPPD4xO1d+Q58NIAy+ZQgCR7dtrf8GfmW8+zWhdnLucJrYLbplNEolG95Tw23HYWueabTctc8w1mS8dcpCG658HbLErpqvYi85ktqZ5RrYx6rYJy4cxUL/HND0fhkTGEOVFstmowvv6XfzB98uiRekbZt44aOldWr97QIKJSrWJ5zVSrcNjAAoX0cV6look69foTdPsT9Ce0STE/AA0guWo5qiR1ir79+hot3hADY41FG2ByOW3qiHmz0ua40hXyI+Ixt/J0OoVHH9/RuZXOzKkPHk9G6PaqyOUPUcy3kYrOwOelP0EbuVxZ3cy1W6+DVinFs12tfIpJeXEBkGzRZ99JIkrdZOJyVsAxLI84XrSsqzXHKFbK+OwP/JDs2kqFY7PYHVCpY8XelgkinZVrSGcy5oNQm9jVAqCE7NPrHO178f9qFM+TvLm1kv/Hi4bP/Hx2i0n9MmsqHmkmxsL/J9bAgtT8rH/f57/42iSRnBzvi4dBciqBH6J/4Rit43iUmF4NPFaMv/6z35uSN0ZzI4I5dmOMIClHCpN0iofPbeOzP/gT+rVGvaJACVNBAOTPitoKy+diB3YMtHaTNH5K9NApP7xGvQx3YFZ0M6vLJjoZvewd+rMOpYI2ey+Ih8DqXBbDgQ0hr0f9K+lQTBo9PjzRAiCWsPXsUJ8hOxuDwzY0A5a7LQRCUcwurGF2JoWdnQ3Mzs6If6+HdbynrTSYmNX/Hx+YaCGLo1uvvHYungD+8k//AoYjoAXgC0TFChr0GmYhzIVus6LXNUOuuVNUP527d9El7fu8nePxQ3dw83mbn+EFAVehDjzr1UY7ZDVLTIADNr48Huf/29O5rDQMBlF4YhrbQAiKiCKtCip4eRFd+WyuXfogLl3oQnSjICooIpK22pCQKFX5ziTNJotAMkl+5j9zOXMinTU3sEmqAfLI+lFaloAVfYH1j4pSfHOiCmoHTFi7ujjX89i2YDF7JjCUl2v1JYKz05O/oijt+fFBCJI+dFq4UAgffZY2H/2KbEmocXh07HtXMGcvT3cCEdx0PJpYPf22Sd7wB0P22UjKYnVViKQoQ5KB+v5LBkRJg8dmC0D7Pu7ZcmXWtvp969rUDna21UL1OqzUgoVX4k1BsLc399YLQwGnTkBU4IsS0YZsmNvGOkSSzFaXF533SJaMkIsF++VZtihyNB/HS7a7v2fXl166jtMVi3oLEoygC4iDkJH3JdlErN/igSSJ7e3dGTof2VgDqPCWIG28j/9976byBeDVPv+WfpkUb5KmYimBv2RDt2MxSiJI1SBU2VC6sR9UD4UPWh6AjgNSiVdvmFpaSa6WXMHaYLM1YTYqHlvoB/gHJX3Nrl1gVbMAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAAgAAAAIAIBgAAAMM+YcsAAAABc1JHQgCuzhzpAAAgAElEQVR4XnS9abRt2VUe9u329N3t7+s7vVdVUpVklTokG2IDsgAZkwTiYWIG2GaQIcsDULCCQzC2A3YIIBuTgG1sIsNIhhN+2E7AMUgQEdGoL0lVkqp771W9V6+93enb3YXvm2ufVwyPLCjdd8499+y915prNt/85lzeP/v5ny2ybIW8yBH4AYoiQzmWiyWOHt5DXK1jMh2iAFCr9/DN7/3z+kiW5RicHKJ/fA+zxRKjyRxFASxnI/1+c3MTi8UCR4cP9dqLdxCEATJ4SJOF3mvUI3heiCxLkeY5xtMT1Gsxrp4/i7BY4LHLlxBEVXz808/AjwJs7ke4sPckBv0h7r02wNZGG0HgYTo6QJJ4+s5et4UXvnYDe/ubiKo1bLZrWKY56tUqVssleJPHw4k+G0WpftZqm3jsicfxpc/9nl5vnbqKqNJEq93DcjXXe5/51O+jt7WJ6XiITruDdqMOeECe53hwcKTPXL9xHXkO5EUA3/OQA/B9IAhj+PywRoGcE+X5mg+OSqWC/b1TqFRjwK1BoxKhUonXf1WNIgyHQ4S+hxQxcvhI0lR/y9Fut5EmK80lx53XbiII7D48z0OhFQQm4wEKeJjPp/A++kv/qGjEuX45XebYP3NBH7p5/SVEYax/L5Zz9I9PsFjOkBUBarUWwijEe97zHi14UeS4d/c1HD58gFW6QiX2URQFAt+3Gwh8DAYDDMY+vMBD4QVIM7uZRjUHgghpkiPLMswWQ1QrngQgype4evk8VkmGj33mK/BjH+cutbTg924fon8/xRsuXUWz1cBydrIWgHqzgd/97T/E7k4NG7unESQDtLfP2ILHNeRZhlt3D/R6c6Omn8kiRb6a4/SFy3p9+ux5NJpbqNWraDVskT7+//yunosC0O300KjFyNIMvh/guH+iz7zw0kvI81ALz8HfeZ4PP/AkAPw3uBicOL4OgE6nh2aziSiKEMchKDacu267BRS5/huNhqgEof7W930kuY/FyjZro9HQPPMnBaBwV++fHGI5n7k7ASaTga4bhjHGkxGq9YYJQKeaI/ILzDIP1XoTYVyHH1YQBiFeuXFdC+h5wHw2w2QyRpr7SJIcQRTqpt/+trdhNBpjPJliNh2jSEZI0gS9Xk/CQe0ymy1wcDiXoC1z7gAPeV4gCFLUGw0s5itN5nI1RjWOcOHUNuJihfNntpFkBT755RsofA8bWzGC0MPJ4RSTwyWuXb2GKI6AdILFwh623trFjZfvYzZ/iE5vC2GxxNburn43mqfasbfum5ZKxwP9LJITePkKX/9N37YWgHZnC61mAx5WmM/mePa5z2rn9gdT9Do9RIHt6DCsYLZc6d+ff+ZL8LxAQuZHoX5GUawd6AU+Nz8Kqgg/QBhEaDRrCIIQGxs9CUkcRfBgC1uLIySrhbQW92+9UoUfRsjTHIskxyorkGc5ur3uWhCyLNH3+x4XfITh4ASLuWk7n5uRmsj3pW3zLIX3q//8I0WrAkTIkVPZVJpriZkucmxv70qFvPrqDaki3sxkYl9ItZqDZqNAmha4fPmS3p8O7upnu1VBlvsoPNvtL75wC1G1hSRPJRgZYlQiIK5WJSDLxQpZMkMlCnCeAuAlOHd6C0ma4/efew0JCvigeisQhTlmJwmuXL5sk5bPkBeJrhPHHVy/cYThtI+4XkUlDuFnS/3uyhse0wQc94d6/dKLL+tnvV7Xbn/Xu96l171eF61mHb1uF/2TY733yu0XZT6GwzkajRYKqtrC1PlssUKSFnjl1j1N7CpbacKzJNECl+bAdj7/rJBgbGxuwPc9NBt17eIwDNCm0HkekuUMq/kMRWr33mj19DNdJRhOl5oNjm63q43Vbjd1bQodx907N7CYz1Gt1dFstTGdTmUyaHaSNMO9o3tOAGIgopajzfIi/YQf4otf/LJN2rU3YX9vX/aON3brles4PjqGpK3IkRYRVqmHKDZb1G3X0WrVpAl4i/QVOF69/RBpUaDCHet5mCcFqpUYcb2hHTafzLBK5qhEIU5vdxB5Ga5cOi0T8KmvPUCKHFnKXV5gOU+BxMO50zuIwgj1SoHCCUAUt/Ha7Xvwogam0wGy5RitVkcW8G3vfDfCMMSZC+d1T//qo7+qn8mSCwX8J3/2m/T6/PnzqNcrODq8j4f3bus9vxJpxw2GM9RrTalXTjzXdLGyZ6QAcIJXyRJZTvMQamE1TwlFGPCcXQ78CBu9toRj/9Q+opAC4GO1mGFycgjkGTzfR6Vak9DOpvSxCr2XFqF2M4Wowl0EmpIW+scHWK7Mv1otpqg3Wvo3x3hqG3c0G+BoeAKv1jQBaEQFQq8wVV9pSoVQun//jz4r2zSczqQ23/a2d+H0qVN6YC3oKy/J4Tg5OkLhR0gL+gw+Om2zq/U4QZKkKPIC1TjE4dEIYRQgyXKsVrZbq7UKiqCCxWKJxXyJ6XSoHXt2dxNBkeLK5TNIM+AzLzzEKs+wnFPFURNNEMLHmb1NJwAestScNe6oW9efR+/Um7C93cPLLz4vp43jybc8LZNx9sJFvf6VX/lXEuo8SxDHEd75jnfr/dNnTqNajfHC155Bt2ta8ehkoHk4Oh5jY2Mbuextgfl8IbPI3x0Pxlqg+WKp13meSeDo8KUpnW1ugJp8oUoYYW93S4tbr9URhR6SxRQzmqWikGYLwhDzuS08CjM5caUKL6qD+ypJEtRrFZkPztt8NsVyOTMNm6ba/ePxCMt0joTaoShQ0JlEAbph3v/yTz9SNCtA7OVyUKL2htQIheCTn/yUooI0apn/mEwRxg08/Za3YHdnGwcH9zCbTkxdpTkOjs2enhwcYHt3F51OFV6erb1SL+hIwotiKhvFMZ9nWKUJlqtcJmCxnMgB2tlsoxoUeOzqRaSFh08/d1caIFlpbjAaniDyPXRbdanNzW4L6dzseqUa/gkBSJMpNnvb+h0nplavowhs13z6c6blOu0mzpw+jYrTYnymOAoxm538CQE4PjlBmkWo11vaYeYFA7kXorPRxa3b95DlBRbLlXP0TGvyHtM01yajJqV/1Wo2sb3ZxQZteJFhtZjj3p1bCPURz5y6NNUiU7tWqnWbNHiYzqlhzARQ20rwkWE6G2ntqpUqptOJVD6FZ5UtkFD1h/bc4+UctWod3r/4xY8UnRp0Ua5KtWl2huOZ517GeDTEIrWQxTZ+gFpkF+50urh6+SI2eh28dvtVTGameq6/+KJ+LsYPcOrcJUSVKmqVCEHURbVWxeZ2D+lqgZOje4iCOlKsMJ1y9y8xm0+R5ym2uk14RYJrV85TZ+ILLxxo4ubUAF6A8bivSYt8Xzv+ytl9LKYDTfbe3i5uvvRlVJp7OH/5CpAtcObMOd1TslohTTN8/gvP2kSs7Kl63SYeu3ZNKp7jwf0HaDQbqNWAuosCnn/hhjmuYQf1egOrpduZtL+9TSyWSxwdD2Rf5/Ml4riiHUwtQG1ATVirmTpvNuuy+xvdtu758OE9jId9JMs5GvWaBKBWqytci12YxwmgX0HnzvOr2NjckjYbnFhEw/vxOCexRW+T6QST+Ri1HXOAx/MpVkmKoFKRkCRZBu9/+kc/U1SDBdqNir6s0dnRh2nfKQAcg/Fccb7newi0c3h7wO5mVw/TqNXwhisXJCB377yG0YmFRFGQ4vrLLyOKq9ja28O9ewP4YYB3vvNtsq8cXhFiuRxhMjzEbLbCarXCYX+EPBnDSxd497vfpYjhKzdswRktUB7HizFWyxVC36Lrvd4GxseGNzzx1Jtw6/qXEVd7uPiGJ+BjgavXrul39DWITfzm//nbel3ZtMnZ7DXx1JNPYjE3Ib7x8nVs725jNDrAhUsWQt648VCLFVXbin7Go74mkiLTaDZxfDJAWgRO9Rey7bpGpaIQl9oiqlQQULvEITY3eijShXbo8OQYy+UcWbJAq9mSYmk2WxY9+BaGcvH5lQzBGc42Gmaa7t+7g6PDezIZcbWCybgvNV9p9xQ5lWO0Whr+wO9imM75//mP/IxwgCCfy3Z1ds4oDKMavnnnCMdHBxIAOmJZkaFeqenhGfZ0OwRh7Kay5Qz7u7s4e/Y0xi4mHh7fR7vTkQdKO/Zbv/VJVOs1NLsN+C4yeOub346IAsv7KgoMju5iuUrx2v1jzCbHeNPVPfhBBfeGkTTCdJogK3IkyRJJXmBG/6TI4acFZn3bCe/4urfh8P6raHVP4dyFi0iXI3z9n/1z+t3N6y9LAD7/hef1ergwX+TM6S28653vxGu3bun1H37iE+htb2N7r4fdfdsU9x+O5NQRS6Imm80num163ePpVI5tFNfh+RFCxv0STgZw9nghY/Uaw+sAtAT1eg2L2QhHD+5jtVrILAYB/YGGno8YQq1OLdTQ5uPCc6RZgmazgwf372C5mOs/fmG1WUe2WiCNq0KfqnFVAjBPDRhKKYS5RQglWOT94i/84yJLpmjVGKsC7Y3NtcQ8PKLXWKA/nkltjoZDSSftkr7ABFO2hA4UFzCZz3Bmdxv1ehWL8QE6vS6iuCEpPzoZ4zOf+hy6my1NkP1tB15cVbx79fI5ZLk5iDeuX0d/OMD501uoV2JCaSjyFMcDet0FCh+oNVoSrOPjI6zmCzjLhN2tNmaTAbb3LmF/fwf0AQi2cNx57VV5zvWu7fybt1/Tz3ajio1uC/du39Trbm9XvkmGFXb39kxYximG/WMgqMvsEAcpCg9pZuGwJrbwZe6I/oW0tx4dtVRz1msxvPO1+F6WCFibUZ0TuskSWllUKzUDc+gExlV0X7ceWb5SJJHlqbQfnemFrmMmmT5iQZMID5MVnaXcogBGDnmBtMj0vTn9An6WJuDnf/anCu52wsG8uXZ3E1tbPe2qew/6+mLGuIvVSqqWcPGM0s6HzVdSf3yvWuei6nnRjU1Sm9VMKB0jCX0eVZycDBQevfTCV/Qena4gqsljnS8TeLnthLjeQZomeONjl1GpVnF092XkOSMAKS4UXoTBvDBgJyMoWqBWDXF8eIwin+Ps+csI45ZAnJPDu5iOxvLGL119o36ePm+YxSd+92P6ub29iVP7u1jNxnodxrHi5vsP76HZ2dJ7C+IUjDTCplSwVkyuUy7gjMMLK9KenAfPK+QzUKip9k0ACIBlOD64J4eUQq3FKzI0Gy3NR7Pd1kLRXzH/wXYtd36SWuQxXq6QORNTVEKknoeASOpqicAPEXo+OvUmlqlpODqfq9US6cowhYL3UeTw/snP/L0iqjQMXlwt0Gh1tJsZU56MGMqkOKFNph9IHEDOxgrL5RJpRvSLcWmARrOtL+XYalZR5D5q4QLddhWNpnmv40mKRpN2KdLDHx/e183evnlDQkHJjqs1Pbifz+EHIS5fuSY12G53EUSR2dcsx4vPfV5+ySrNZS9D7mqq1GWK+bSP3b3TyMI2nvnCs0iWY2w0bYdee+ItCKMI5y5d0es/+OQn9LPXaWNvbxuruQkATRsfeMbvW2ay2cQE5PRzt3oeGu0Nh7H7SL2KvHnuSuIe3PEcUeDbDi0KLKZj7fgiz7STKYiMEGu1Ji5feUwLzXk4OTzQHNMnE3LKsM7zMHLCSWEahwEmDjFsUdXTL5qM0KRvIJtKQKyCpMhcmEutsZTmpjmaLxcyRd4/+Zm/W6xWDE9COTIe8WZNgC+4kY7Iip6x0L5U8ToXSFLlE9PO5W1KcMqkRGwxcc1fodWmBjABODoeYrXKcHAylWk4d/oU5pMxJvO58G/+VwlD3HjlJlbLBVaLCdrUHoWHoL6Di5fOwisKbG32pHbny6VicF5rNOwLMVPIFEL/bm9fxIs3XkGeJBgPzUG8eu0pxdFXrj2h18996fM2WaGHjY02Vi6RRRvc6nThhzWcDE0oHj7gri0Qh4FUK2Fm7WihoqYNBsOBNEC9VoWfZ1gsF5jPLFSmliPKyftV0iZNsXvqDE6dPouT4yNMJ2NtMAo4tSa9fe79zPlxo2SOPAzAzM08S7HMUwFrdS9ENQzRqjBsTNCs1eUv9EcDOY4M6zkIEJWC6QeE1AN4/+gffLggpJhksba3vNQgVLjCcIuDzoOFI1Uslwt5ypS4KAp1Qe4UZbtcZqsT22SE+QyddguNVl3agz7AcpXh3vEUuVNN1WoDl87vYavXwWDQF/TJReYu4eI+87nfp2FDo3cGcWcHtZASXWiXjcZjXCL8TOzbL5C7DGPsU7Uucf9ggt1zb0CynCJ2dnK+XKFareHJN5oA/Nvf+E271zDEhXMXUIlstrPVBLv7+3J2+30TgJmeO1eoVm800epsaMGoDV+6ft02TlxBnRnIbtdCtvFIcTrzHBZhFEIld3f3MRwNlJHjxFMoaAKIalL4OMH98QDcrxkxBB9YEQSRKS0wKBLkSYooL9CsGA6w1eoKX9BiJzPtctPZNhYCrux14AUIvBTe//yzP16ksjN0rhhjNNZ/UGs05fVSUnPiAFRjs5nZPs+TY0jTwTCH8CajCI4Kk5VRiCCdSi3XahSqAERLJ9OFBGC2NNtEmLNVCeU4PXblPJJVoh0wm8/lcN25+5rujZNdZQydL3FMQUroRK30wLRvxO3rFZdaDXMldu7c6+PsG55UfiEW1s3rExuv4U1OAP7db/z7/18B6G1tIc0LgVStdldaR5pwRVWaSdtoQYoc16+/JCFvNpqoxFW0O1175uUCcSXW4tLHUlgHDw8e3ENeZDJHHHFUwTxZyPOfLedYcJmVhPPgO19jHloIN0uXSBcJvCRFo1JdC0CjUkderORz0LGk9qF3lBIBVAracKs48GQKmLn1fukjf6dYCS2ydfeD6loAmLhRfsAlF7jwy4SZMYuVmdIt8X/m+CksvW4bYUEkMUU276NarWBne1Nyl3kBFssE945nyHKTRN5qlQiV+AAZulGmdG+vVUOKKibzhVC0M/u72vFEFpnmJIx598ExaGolmPMpiuUElWoFnVYFtWqMo8M+rr35nTIHu7sWyoUMkQB85dln9PNgYM/S7XaEbqZLc3AX02Ps7p9CEFVQqXVwdPgAHvMk9IFWSzltFF4uEO31CbF7wt81ZvciJdGq1eo6Yhr0+1pcLeB8JrVMjZkgx/FsIK1rTmGOkDwDeIiCCElk4aSum6Va3Pl0jFoQoxJYlrFbNxMr7ZEyP5NhuiL0S+zB5pmmnfB+mSeghlWYSgFIEnqblp+G71KXJIe40Obo8BA7e6ckyd1OB5PpVPn9RVII9ZLghLGwad7Q6R2HJiYzBF6xVqs15zSdTFLh+0wtU3MUSSpbx6xhLciVZt1pWgy8yAMskwwHhyd419NPomD8pwtaPvzll6/rQb/63HNIlyvUKx667Yps8HiS4crjb1Rkcfb0Kf3Z2UtXFDr++q/9cxOAqS3q5StX7DOFeckMjbe3tzGdLzAaTY3foM1RYLGYa2FL28qFo+BxNBumQadMbDn732i05Sj7Yaj54TxWG3V56LcPqOFySo6Fa7aScv4YRppz7NgFfGZiD2mGU63NtbrXbk6WiGtVLKkdaLILIC18xHRm3Vg46JqaKdOMe/B+8ed+vJBXmtGb5mXdBBNvntETjTGZzBTLxnGMU2fPodEwm3Nw1F8LAFUNPLM/1TiQKo+DAt1mzfBvxvzNltT70XCOVVoICl0yNGFM6xC1GCkqoY8zW1U06nWcjJeYLRLtdqZlmTTRjmAKtdXGY49fQ5bmePjwYJ2mvv7VL6JCxPHrvwFhVMH45D52ty2UoxPKyCZ26vT3PmuAEN2XJrGATpkOX+Di5WtYzBboDyzHUK1ZZo2qn9yIkn+g3V6xZ6cgEG08OT5Bs91BltJULeUPtLsd3XuZQXy1f4B5nqBR68Dj7pc7WWCWUENmxibyzKG28NqEjD5PFMR6Dg46g5KbOMZoOlWun4M4AqhZ08RCU5+EFPcVcrhjeL/wMz9e0AHKXM5ZdCb6A1RVTtVPlYErZK9On7sgQeAgzr1cUeUQLh6vJVJAMW2dVBpteirJP3X6jLDowXi1ZsVQ+1BKKeU5nck8kcY4vVFBq93C9Vv3MZ4sMBwXGE8XqNKhJMpVjZEtJkZycNm8yxcM7+8f3QfyBI89/iQyv4Jb17+CjkuYxBUSPIBKYJP0m7/7KVu4SoTNjTbOnT2t161WHZ1uT8K/uWXv0ZFjaEbhHgyHmIyHmgs+GyMTDmL6dF6nWYGgVkWQpFjN53IyR5FlB7nRZKdDZvuIGzBbyPAxRZKtREHje4HvISpow83XoLMX8rP6+4TZcA3CuhzT1ULakACWRgFFVeVg1FZuRgqEYPT/7aO/WBAapcRx0hLG9+4L0hxYLuhI0Jud6SG2904pXOSgQJhTY/8OHKZ75849/Z5qibZN9i0lmETVnyt1Smllijb0qS14DQc0JEtpkDPbDdQbVbx4/RaG4zkOxykiPxRnTtlH7oI4VoqV8CYzgX/qTebZp4uRHMRTp8/Bj6t46fkvodc1hHNr77QmcDl4oNe///nn7P2tHnZ2ttHrtfWaTiMXqUCEnZ2zNp95KgeOyS06gVLdbvj5CkHo4wsvfknazQ/qqLXbCseSFcPTFUaLKUYOMyRBjDt1t9GTzu1PBwJm6BUJ6iqAelQRmldyCZk+HiymBiV7PmIHp1OZ03FcMnPION/5c7UwllktcxJkJDFZxA/QV5HR/5Vf/oViOp0rDiYClzoCpB648MQHpNrml5KaVWu00XK59XICqAlo++gQZpJOU0lHx0cYjx9x0mi/mYzIxAd0WoK73oWatI1+nqJRreD0TgPVSgXXb90R1WwwJU8uBPl+VKNUv/SiSVzlxCXzBd79tj+lCUQyQq/TwGu37qDa2VLQQqydo9be1nNWcnv9/CsGBe/v76HbIRJnNnMxH8sBXC0L1JubyLOl0sgcDKFEtHRwNpNTju2FF167Lgc1QSAn7nDaR84LMmpSwG3p44B0LnjaADbXKfw8R+wToTNDXKEA8F4cSZQQLkUy8CqoBAXShItOr99WnHMfBYZEctPRRPA/M+30KbjGLkEVV5B6BbyP/NxPFbxRhgocBHnqFaMwkaHCUeakqSnopii922grOuClK5WqPHWCSSSSdrpGvrhz/y7Gk4niZ30P1aJPpy5B7KBTZvjobMoDLgqFg2Hgo9OI0WvXcXgylOY47k/FtSuzW5xwJqcKl76djUZ419Nv0XW8ZIiNbh1/9Iefxdb+Ps6eOVM6w7h/bDuoGpjzunQwK6OAalxZw64W5nFOyJM0xy6umABUo4psMxUtn5/hlh8UuHPwEJOUGH2qheGzrgouGlAnEbakgxl+ijikBjN1XWRL3VfkB1i4DcT3fWExnmJ4ao0kXaJGjiHZTxR8EU1KDmFVi8//o1tN3060NTnpPiivzF/otXyRP3bbfuq//7GCKlnxIWNlskadZJO6xcxgFIYWZlF1cifxprgLxPhl4iMU04cxObUAQz9J7mIpkGQ2M6bOcMZda6RIqlNmGAmTVqstUAvRZDRoN30f2z2CIj7mKyOUzJaZyJAWy9Ij9xSS8t5oEnjf2465Q47j9mYTX37mGdTbHVw4f1HJJI5X7w6EgLWrthN8JwgNJrkYZjkIl6EceX+D/onQSglW7DRAXMciSTBLyMvjvEXIXSZqSPvNryYZliEddz9xfQoF04hU6LS/kTlx3PkcDPH4WQqNqB0ZSbdEDS2eZ3qemoeDm6TMMEp4nAbgRl7j9S4XQZVf4gicM8Lp2oxejpS4xC//058vRBMqgAcHx3I8ytFtNZWAiYJAcTV3KWPgGcEgOYFmj4lTR1FNDhLBDkqym901YCLKF/PRea7QypkpUcWIPXAnMIblBIgw0SB6yE/ZDhDIRNWamhNFKJV6kqrO2DuR1CJHpxaKlnbzpa/JLzl95rxqGzi++tJr0lYUEo5ez+3qRlNzQPNExJNCPR70MSU9S9y8EGG1aaSUJIEfVZC5yaQJmmcrTFYzNGokiiwwmg6kxjl3HjOGtM+ieBcG7oCYyhKJl+n3lShGkeby1Jmt5ODvlb0jKytZStg5WvWm5k/hIc2FMyPUkHQebNZMK3A+6ZtQCDxCv84EDBhe0gn85V/6xwWpxnRueJNl8QD/OFk59eEXzqOnA2EePUfpA1ALU/1TqnlB0pCoCejxUoVRVdsWChVCDUZjTKYmRLSrUk0hvV5/nQ4mWkeBI7hRRhX8TCmeFI7xeCJUkvdDpwbOW479FKd3N/Dw3j1MZhNcvXoNMRcPwKeeeU6T2q6YA/fUU4/rZxgzc0aQxuDWyZiJmwGy1VxaSZ9xNnmeUyMtMHXPtcoSLF0GtMqpJ3eRhFnj2bJKxKja8nv4v9yFFaQL0uTNPNJDXyak5fE6HlarVGalIHfabahG1UAmiyIy+T7MS1BYOVI5kZSBTNlH8QucMAXNuhJV3OyMtsYKO314/+wXP1KQq8ady7RrmerVAzsfgDdvWjZHupohJ4SoneoKHQg8FlYEEcRVLIjn07mjb8EbzTLbVQErgJg8mgqsYEjIa5NcQZIJAQr+R8EiFMxrMDMZaZdbbcKjYWGTpJ42lvy5peUo/GwBL19iMZkzRsJjj11FGNkifuaLX9Uue/IJSwe3WsapH4wmGE2IR9hFyKcLihzV2INPZzMp0GvazuxP+kgKt6O5UVDgnguju1HN0lrkASh9bYQRpCHq9Y7MJm+SaptO3WQ2lppvNulTJcqwFsVKhNEqcyouA0vOhUUJNuQzcZcTu2PxCUO8wrOaA6dfqxsdc7pRgIl7A4jt+ZapB3/5x5D5T/y3Hyq42xrNjmWsXMzOD5FTTqGgenVXRZIs5Fwgt53CXcmRZ4VQPRsG/PDiKVnBTt+HFVsExr603zbxI1QqBqmW//F9qjdqC4aeFA45aDRBzIe7ZBGTJgQ7WOnCSGWpxAoLKkLGgqKYkSPw+NWLYF6D4/bDvr7vwjkL7V69Y1nCfn+ENLecpCZIhAp6zr6iEV3rtdsAACAASURBVI5OaBox91jAMsMsmYuIsvSAE5dEavuxFj/1CzT8Nsbzke6N5W9xpWmVQVZOZWo5COR7UUjofNPWM/1Lk6sQkBvTXZ8+CjeOBMvLlbE102iCEVSZhudy22tC1TOnpaKgprxGiaTOlkskd2bw/psPfaDY2d0VEYG7jjnkcqsR4+bglyp3TQ9SQIZT6X/ip5U7ca0jpyoZDvHBxIejRqnQrvtKG9OmqmooLxQpcNDrns5Me9QaRp7Q9RU6WjEJ71NkCTlghTJ7SslmpIfZ9xB9zJYLDEdTIaxvfuPVdeh6NCZ8na7v8eYdw/DpaHHnyJQo1GNBC+TQlkIeEkplzBKSWzjCcMoauwJ5pYejbAG/iDRHjEyoPejkLVImeFgaFqFVN/4AF46RhGB6z+aG/EkOZkJpdiQg1IZU1L6PVZKI0aPPkG7mHHVCmGUGUMEe2d2yttTQVRQOnZX/xDzC/Tla1Qb2u3u6d+8f/P0fKzqdjlQvSQrcvWVIWDJkiX2X6lcorE+4mrtutU5U2J3Zzqfjw8Fwj08pkIk2KqcDxMRPhh5LtsSYZaHFwBITLHhwZAri4HxooloUEi5aHIZrFJIAUOmvlGAUsxccvXYbo/4xZvMM9WqGS+f20dvcUdFpKQA3Xrlvu6RtwA9zGyxIKQWAmASdLiJ9xDX4/Cf9ibQrHX6jpZmAsl7hgeMRxFViGYSPCoVgk+kYeeApm9iqWUaQ0RXveaH0tW2yejVG7BGvWGE6M01WklKohXkDq2SBgMWiPv0McvxL34ocf2rLKqKQDqzNP8m0Sf+R2SBqu9XYQK9JMq8lir1f+MhPF4QaWRrFh1TK042MRNA8FxGCMKiNTOVMJbWJjJfS5lhIaE6ZyYPFqKUgkV/AKGI4mQku1WJtbCBW7Zzl5MtRRibK3nmeIGcWSBL9o0mi/0EKeSYTY9ZNIReAUzs7Qggf3j9Er13B+bP7iGvm7V+/9UDaZ+VbbJ8z9+7i4lq1+ro4meVaTWeKnGNIn0KJosRMpYupqfLnzvqFFACSRohRiOY2EdZQrTXQrLek0nm/8DP4vG/PgBpFjtSWKbWtRVMpvQutks3KIxPrNIdL25Q4AE1rNbMIgWM2W6IVtVCLyLlcqVagTNmzImu2WMD7yE//ZKFMEkmdNhP6QZtb2u67d2/rCxjKceJLm1gufKfTNk3gwKM1jVnl5hb20AwY4MO8wVSOoASgtyliiewYETq3O6hh5AdkpgqZbaRdTpYLzBYzkS7EbnGTw6KKiovFNzttwcvjwUj1iefPnUGzs6HrPfOlrxhyFhlYVW9ZgofCxd3eapmvwPCSi0xB5HPrPdLmGCk5s0aOhISIO7caSjWnfi4fhEArIWCWeWUMzWT6qopoPM+894pPrUtNRsCHXrYR7mki+CKFaR5z/mh6bIN4Pnc8aysLhGnVkj6SkwD5PEeYBUgZwRGXKf03auaqUfo5jh88lHn2fuxv/c2i0WZxgi+2T69nE0W1rcQD6dD375gjw7p0JnP6RhYl/k9JYg27khwOkNja2TZ8QOAS8QOTSU5m+T7/TapVvdGxPLfSq5kygZwAMpKoEYKATCUPk7lhCGIjs7yqWsVJv+80lyefYHBstPCve8fbsbmxIYmfz8ZKSZd8+Oe+ynK2DHcPrDj08rXH9JNgVRSH6DqYm44XGTj0SehMchC0obDqXlMugBuBh0XdAUtVFqwV8DJCtAn8VYLQ0caCONSOpp3nf+2oigWFKS9Qj+jAUeCXJfSh+bW8vfLs2r1eVEVG85d6KDIPyWSJZGJoYlTEErIgNC5hRAp6mUFcCxmJqhluvPi8sBfvJ370hwvmpvlH3N1dJwBcDCsCAa6//Lz8A0ogCYzciaPBYG3zGRZylyo6UJFEQ1kn5gesBNz0o2oI1ti3mRpSw1aFMYb4N2m6lJqkOqcNFn+ON+pH6G1uGyu2YNjUwmAw1t+QKMIdVNK+ep0mOu02XrnxEo77A+3MRt12Ou3kfDrB4bEJ8fZZCwfbLbKfHnncvAa5jv2+kTX0tw5o4k7l5jBHi7/wUXRsrpKADivpWgvttSjL0AiYlAFWhmWhGgSo04eivV8YnE6njt57pR4rgydn0WfSKxOK6MUVrPpzLPsLZKvlmoG11d7EeDTRnBz3x4hrNfENODa2t8R+4ji6fw/DQX9dD0CtVmEy7+f+4d8TI0gUYY+wbEWERPHZRpYH7x8dyTkhskDQSLRsAgouG2ZQrO1wVcq8LkQsd7YkNAxVKELAqFRNSj8LHg0U7vC6yjmwnm21wmwytqiDu5/FjqqaIQcwxPFJXybrTW98Upqg6ZI15OxNRgO8+sp1pF4onmOZWJF373h9vKdF6DKbAa9v/gURSaZ81bBBr00DUKUbimcsKS/0RX0jkjWKbBeOvZWCYPpD0nZpRsWsQpgGvXnnY0n9ewEiCrdSWAWCOBb2wtKt2TEXNcfseCotqg2EAt32JqENaRB2C+EYjVwUBV+cxMCzzcWcyuuhE2r4svuIYGoiuT/8ge8t2KGj02Wdui9foMz3H7vWLlxcOifE0KnlSx4bHbIydjfn0WwiM3VlAQKdPvoPcSVC4sJKoxR62NzYQsHdpSxjU5WsSoiy0mY8kiCQg2gVtYYCdrd25OXeu8uUM4tSzQlczpfiD9iD01P3RG3f3DuLKtHGMjKZThWm1WqmlQ6mprWOjk4sSeTyGEyDk8/IAs6F8H2Da5XVW63kLGYuc0iQ5V5imwUhQLiAWovZuZVfoIIQMXmLceRsfQE2jPFZFuctUUwzpCO2drG8BzUIRz1u6n5KHKDldjZ9n/sHhzjq2zVLbiKfk25QWY5ODMMiOxtcE0Yb+neei5zqfeiDf72gDTL2iSe2a+nFk/zI97gYxKJV1RKy7MhBr85hpG/QqDcxLQkkLq6nWmAKkjtXiaQlY3zf8PGyqCKOwd43YhxVqvJM5c74kStypP1mVbDRqCuCZT30B0Nh5qUHzAlvO/tNAaad2+qVpNYEuYtiuptb8iGY+uWYOjq3YGUVX/oIFf9bPYN2Idus0I+hP8LB3EAAzHNz4JZZhvsLhog52uxD8LCPUb5Q+trvNNGOjKvfq9eELxCTW02XWKQrLKdLmQMxgx1CyPCX6r/TairGJ6OITrPuw43ZnEwp55yWURcxAwcB82NsGKGsbsRK5IYV0LjPkolFrev96Ic+WDBJI1yYxQ5NJoBc3O6qSDjJ8/lEtpkVQKXqJ6evdPBoD1csWc4y1bI5mTOV5nZft9cTO4g7iWpVPHixi6liS/TQ/rJSY/UNNQwpUZla0DAJVT4A1TSVDpM3kWP8NpxHz+ZULBc7tbPpcPjXZThE6tAq6n9PnbeeQAzEmLeYs85OTrBV8srxzRI8eHCAXrMnrzwIWGJeFUbBSZxOZ7h54xX9XavGFjshggazAgHCZg3NMEatWkExtw0l8IjwLJFGdzd+trK2LQp5rXijUmvIrzJK+RwDV59g+X9GR+abCP0TF4C1iZHVJJAs2mmtGUAKxYWnlDUdVMM+vA/8wPcWShGykRFz3zWDPdm/pkwMruHR2VSVuXJQRHCwSVSOgOhaSALFowpU4eCKDuxzDB9pV7e2uq6CNseCO8f5FPwcWcPMF1C1U/CYwOD7dOKYVyjpUawfZAqZPgBNV61SUT6Ai3Lh/HkcHDxAu95QUoe+RwmOKAaXtrPIhDw9jp2dPTmsdCg5iFPQLHKypy6dzVCOg9qw3WrJVPJvlPIemz1GlqFKwaVWY8p7xRL2QItSb7A0vLTKhm6SAKKOJ6uVQsX+yck6v8HCFM2B0159Z/PzlLD7I4CgfKZGvapr8xrzObuYVFBx9D3eWpmj4Rox8pIA/PAP/Y1CnD3y8tjQ6HVgDGlMtEFK/FBA6LgwNnbhnhI3jsghH8ElZkpGEKWSkigH0sGrFklUSrjBWqmQOqZFCjEejZy54A7P1WmDC0yePbGJuG54OkmXVNu0fxQ+fmer1VDzpWvXrgoWHp0MMZlNNdGPnssWnjg5BxdHAhF4ykyWySXOB4kubHxR4gCsj+ToMLop8QlXHVWrkt/nqzqXQpPSxrNTyGyh8nnm4kugigKk4lIxm8zhZXgt0IwhIokhYYRKraZC2ZIVXBJr2HuAC0tRllPqhJkcC+UX1LqOoWqi5yq7vYnitu5RJBIevA9+4K8VVDus+tXErJsRAMN+3/AAtTgxdgoTN+uslMtU8YuE+SdmOtiZSlkvWGlZOdllIyRrfGALQT+DqeMxy5jYvOnkxJzAyUS2rwQm89zHzvYGmFDiMyyXCWYTVi1b1y+ic3HVFpP8fpI87z841u/oqnAxOWoutVvmK8pU6snJseUa3PPL33GmS/kRNcSwZg5kJluNXYFU+QEL2cgADuiY0ta6nU4quJXQG3dSz5ym+j03mMrHjSWmn/K/nFAyNB8M+/IDKBiKOBw51bQR4WHWKdrC0rgwkllHWKziZvEOcw+1uhzTcvMyyiHhxPsr3/3tBXPlYol7PvZ2rBRattnBpKVPwJsVlcjdqEAK13LEiA6u+sapQ4Iw9MzZToVja5t8PEsGlbZceQJHESSsSxycNzocjTEajsWdpwZKE9sdChqVU+fE+Vp4KfWcKWe7fqtBB66B+dKVTrPkzWk2hj6crHJXc6dxnPTZ9CpDzZVZMZam0KiU2i1c03nQcWwUttfHWLkTfrtXw0WY8SQ/QqlsBzeXERAXphQ4VisZ/cyAuvJ6rqbDCLdhtMY5Voul8ipOEa+RWd4X160UWGkCt4PoT8nZd+Y4YSaT/IPv+97vLFiebVQlOg5mEzmo/vllbD1ipWNWmUu1pVHegcsDMbcvFcPy7qiCMPIVf6uPn76v6lKYDLfoqJiafJTop+2ya9FsnJycYDwZw/cjlx30sVyQG+iJe081TMxCDJ9ma51LGI4M5Dk4PFL3TMbyJdVrXdLtFl41/NqVCSajoTmt5DXI3HvaUaW2qtRNi1BoGGKV9Y18b0l/RCEwi7py1e2pZpIcP3UgcyV0jkNhG4AFt2T1AFNW61bqxgt04Z6lgwkNO5TRiQaLXRssXC2TPm5RRaARxmLRCrUMTcvURQ+lYPF3RBMZunsf/JvfX3DCxUQtmTVOAOgFa3Ic/KrFZQ8a8dlDwaH6sjwzmhhJm6oFsEtRxVEzUIgoXKVs7+9blY4mM4pVWk7JLMkgJcg0HI5w9/4DNVhk1ww+MCWX98q4n/NhYY/Z71Jls3RM90MaeRRL2KxPh8XCVsxqC0+yBwe/k45tKaxKTrFItsFQ0CDsmUtrEzKmoBDJsy8hCmnqeb6YySmcumISsqrE7+PCuoUhusfSOKrmUoMadStQ6RoXXOggKeJsMFWaIicIDJVpltc8i9cJc+mTuYddk06nbPHjOBf8XWdzx57rB37gews5EorpPU1WOUoBcOG+diZ3ChEmLZ67sCpl2MrMlZKNxg4UUeKDHHb7RpIzuOP4oLYzFlLNhRY2kerkA3Ncufq48PnhZKrJYDdOjvF4aP4GM2yVqqBa/ps4geELllIloFUUvDYX1fB5Dqp5quS66u9YA/gotma8zBY2HOQdkOhKzVEyaROHA1CNkzHNPgEcXCCmkksBoC+kMvpkhclwLNUrahznRzaeAmimgTWVHLTR4vur6aVlYVlKznVZO4/uemQ+ER2kUDG1XlNijFomUi/AUjAYbJaEUIbIhuPYfSrJxtzCD/3gDxRW5GA7o2yTxg+VGUL6ABQM7mxWrRSkd0dcREcxpgApAWEPc/eudQrNKeWO5GmvDW5W/xwnFLS17Lil5SkKdeik6mL4xp1zzMQTTUVUA9O11mXLA7UDf0eVS3GkAK0XweXTNze3jKksdq3tdCKLvHTpWbM1nRa8WtWCq48OX7PDaNm8yL3nN4w7wB3KhhGsslWz5igWU5dD9QeKfKy9G/H36XhuQFtUtQ1BpEl9f9k8woFLDsChpozdxmKijT4PBYNCTr3AOeE6ULjodOpeS6YVkUYKjPt75V1ejwXr0zboP4nq96Mf/mChRs1rXN/q0UxNOnDGAQ96QKdKJLUKm4zKRWYOWcJc5Nu3XxO4w/C1DCH5eXa14nVY4LiuOEahzJ2FM5YyLQczfLPZVD7Axs6evFtVKhUFDo6OZOMpsBQIaiNh3ZK0TIkO9imSRywuo00WHTM+eFk+VVYLHx4eo7fRQd2xh+n/MI623kjmUzRcvx5en6ALwR0xf8lIdmBUmd4ti0fHozHu3rsHNmRotLqW05eyNW0QO64iv18pFZJm3O4w/IIbxvUAWodwLNML159b23bHIjJyLVFY42By0Onk95ZCw/BSPIq//eEPFBQka95ttG8Oo4aVxYlcSUsAUdWWMC7DN0GNIm6yQYRLHz8g28bIIAKg3Iq2Gk2FQLTP1Dp0jHhDy+VU6pwJlI0OJ4k5CcfzI0Ai5zPVbpsxDcuqGsW0nMeyT4+RVPUUNAFsH1sLBW3zE0b5smYOfI6RU/3k/nMQYCpJsHzNbCIFaTwcrjdRc3sbCdFO5ze0XJEsn7/TdF3HWZNPU+kArNFwhKOjQzVvqresZwDvX9MtzWkaQMQabaZHTrF8hNJRLhlCLqMvFjWxBe12F1KT9eu6k2rRE0ZpvjS2dj3DZbfRV6uZkETvh3/o+wtNviNvGCZuo2wFQ9Vo90tcuSLJIoCzBk2YYdMX22KQXKGHqtRsscS3t1o+7vzp3LV0XTuKZhq0kx2f0OJ34yhwtFuERXMkVF0OyJUnTQdFkHC2Rib5+dAL0G011AE0Txfqw8dRhn8r3xUGuOXl4vOz46F1OyWRY0nWEqt0nASfcn2F2Aiai1yNuEONex/HvqqTWXBpOydQtk4JHqprOrlVK2szP4DdwqyQRnPFHS2YmPxBt+Nde/kyN1M6g7wGmdSlwJah3eHhkbS2FZfQNPCsAdsU2rwuJ8PwMl0sDJ/4wA/8pYLqqVxMNV90dHDStlV5+7r4gcgZ+QN205HDoz3V6PWPjWC5jpvZdpZp41L6KPkuzVsKGXfTdDIXcMPvI2pF3FsCyC4gLuQsQ8iqupY4urQjneizrqUq/80YXHWKaY7drR6CnI2jDOKVDRUTz+Ho7maJQTSaFcwcFEwBmE8mir1LOhnb0nGw9Yo1YnA8fHVRdXE8S8E8+g8GALXbLYFPhHNPBgNR27jZDJ2jU2YLpAIXOoGhweUUkrJukGtI81n6TeJOvK7Ue+iElsCZoayPWNnMfJadQ+kHlcK0WiZW3/jDP/g9xXyWCViQ5DsTUFMFrL1XNo7Soku4TcKITJUcQKp7tn/hKHdtq9t1fH6rtC098NliviYm0M5X4rrAId6oVJqjhPPzxOb5U0UOYSCCA7WF2TATLlOrzBs4B44MZhaW0HKpfpCtVMxJI4qnDB+TGqoncI2gDw6RrOZYOkHhtpcpiWPVQmqRnHZiW3FjR60E10pNO1q48hokwTjN0NvoKmKYLxZ47c4d8QxUZRSyiCYWp4ALoaKaIAATZobfs+LKHGcrqE2E82uNpFGYOnZr5gA7OrxMYZefMwbVmrckM1iG6As5kH/cP/AnfuyDxcOjoSRTO0SJB6fzHLJHtfx6NVRqiF6Xhxyw6NCKQEYDFjlkOHKaoOMaJZW2eeGuQVkrE0x6hoR+wAKVikHEtO1kvPJAhMlwqEnQQkTWNYQTxqwdNQQn06TUaOXloFNIgWIalDlxNp/SgtOzJrGlyqwmvWkT2tFopHCVRE0OdedQ8Uqyhr5LiLUsZCnDYwpkGTITObReQtYYmgkaZlgZDbDLiQkzy7R5TkJ9XbEUVSKjfKlu0J6JEZJaxylKY7hngqgaRleeTgexNAFCDF1yjJ9TxtTBwXytnoVlxzBGF7ze3/s7P1g8fHgC5pc1yqeyF25yBWTq/+kv8CIc7BQiUIhOW1TBct1QwjWSVIlYjNAhgVS/lFqqqnKsUpY+UfMYBXzFsI7ZvXpDfDwCPvQ5Ou0Ng1W5tXw6dYyzM25/d9uPehlZx5NMjhUzkAK5nOPGihuGX9W2NXSYuh1P7ICPxR1q08Dtx3MADF+QjDnSqWBl1jOwLsAoTesWebzv0GPTq5ryA2HMEDRVNzCec0SSiEK70FDWSs1CS20qciUingXgnDbdgiWPNd+uSpkMKQpQuSnXbV/LQlG3bIZ+uheO1FpGGCmBLgrAd3/n+1RhzUtUq/W1UyKJcc6EcHjZfINwS6xATkbJjGUrFLfL1KuPzZXI8HWJEH6fuPRk7yxfRz1n+pS5dbaJIXAym7tYNhagw9ZqcqsIpCQZorpxF3XqitAzU+2vrxtkCEqfwNrHsfnGo0QKd7Yhh46z4FAq4eQFQHocB3c7xX7h2Mt8j6igTbq1sVu4NLE2htud4goyOeQSZaTRzxYTtcKh3dW9qhqIxbQVi/kJ0MjuW8cxtdvhDnZNHt0Ur1O7jKS4eKWtL0kx8hFc3M/sg9V3PHLg7Pwiq/6i4GpDfce3fkNBIkIJHqgIgepHwmD/FvHCqUZV4VJdkdjoqnJ4TS4WW8trolzGrd3qGstFVcUr1R1SdS2XK1f3ztNFuqiQraIG0DxggtmrQPRschLu3rmr3PzxyYEWnTZSExibzSyf2ErVXUGE+PXslpEJolW61AFBJSRcQrfk4ZlwkjZdVcm6yJzMgdB0kP7lvK8Sk6dzxjQr6WLSDCKkmiCqnJOhrXgOlmMosYtHu5GefiCWkvr1ynENpbVogCNiEQ5bUWjohLTETibzqRDUdXMJpyHU+fP1WUed6PJIAMp2NrrnzMyq9x3vf29BKghvbpWOBeiUo9a0xBAbI5TqJmLzCDfRtEnl++IKlI0aHbhRAhG28404QXuVLEhnMk9/f++0dp3oSULQrKqWTiPVNwkS5dFo3GWs3KGdfPCQvHZm5BwOQK3gbtzCMrP/XFkRIdaTZMAVu3FoIpw5Y96fnUhK1jKdUgoHIyGFZ6Rxu46nnHzV2rtwjfaZ982mkIJ4xTOwrUjNIjYZ1a2bN4vHbU7CdXcPFlGx81miJtj0U8rcQbmrrdra9f0TD8NC77LyiugkhbVcE36+BPN4L68v2jmzvaOGHd63v/+9BTtVlBNROoDZKkfgOn5SRVv2yke3t7nO3pWdQUXJJue9XAK3EnQ6qLpLJ1DeOhkuy8X6esQKeP4PnSAuGjWQdecwiJZnE1HlEftW5o91haRNLSwUezQ8LaDtQlOzjXrZuJlNL0w7RC7XYF00rb2dBEGnkAxUBq/voApXv2CjTnEwz8/Bdadnv3b8RFyxHc8W7dQyBMbkkKmk3XIKRCGFyFEAmR2sxWqRpzoHtojXqWnJuvG0Pd8jOpvqJSpG3LV3bXeX+D5zJ/wOhr86U4Ch5etMAH2isrWdmYvgPxaAwncZLkcq5AVCnx6qZdBard4a83+kXqzYwBwcNpay3c2SmbJJFF9u8pydPMdwRMavOYJUs1UxbPj3TGeaU8myLDqEr958RT/l6dJDJphCZykMsLXJ1qimOYiHl6AKqWrs7F2LXfeOJMWxi0xoL5UUIQBTcKLccSuuP5Lb1KJ8yVRFsRpOcUzc2QL0MZqNms5LUEKHO9O1dWGnb3rmpIPTpNGJtZp/a6FTDgtl2XSTibMCu6fOIUsJ1pAWR/QTAsyYyKKvJeF1mAHLyEU7dz5aqbXLiuqyHzD7GtTK1LKk2nEvRGszzoL3nf/Z+wvh+OXNBbareBBDp+O6Z9QZ61LyIzRbrByyHZE5xoJKs5kkcuaGqVsO5qJFc3acNvLe+Fmyh8vdS2lt9ja06Co+4U05r5g4+oO7d0mzU5KI2olkUcuz+6oDYAsb1cYrOrD7ooOU5inCnD0PGA5G664m5MMZbco+Wy+jASKgauJs4WK320AQMT6v4eDgRNqADSs5uKsJ9JT4P1Vx7CIEZvWohhlvGzXC/o8qhsJXppZ5/YOjE6vQlZmtqcSLQsxwjura0FBjG3EcH1kpu1r7a1PaM5w/Zyeg2cmh1ilUn0vZWeTRhg4DIq72XaxxEC7z3X/5O4tITBL7slrtUfqo3TXkiwCIUCPdanXd5jVZGt3b8P2JysT0Hc7DLhsplm5I2SvIdUrRZynlmR9qx7G4g7uHoAnZSDxGjjuZdpWYA4EOkkz44IwOmOApv3Nza2PtfLLXPyeiEhudjbaU5BE9iyvY8BwSmPtOE7F9jVLF9jnS2XkdMviTxObm+NCVki+sa8jWhvlIAnBZvuVAMNIkWIFEsKfVZOW11RSyGbYXsOU98QeGhgtR5XW9iL2VqqLPMdwrF4rPUZpDns7GwXoH0uOisrG1i1yYtHINyfQ5Ck/p3Os+X9dhQ4JBPsD3fu93F9s8o494v2cefAktNlyhJI+VNG+b6aq47EdgZAPXroQMk9JpKYEMayKZr02D+gDIbyH65TqQrVZqIk2PmgJATLzkBJJQSSyA1yWewP/Yi5i7fzKZKrQirkDh5EFRZUQgsgoPc5yMDWrVeYKuR4ELz9bCWrHFFf2cXD5nA3iaB3H5PI/RaFi95NCdNzwYHGvHl/W04tq5I+tUCQUPS6VbmQ9pKCTm9xIu53xQKGtVRj5QgwlOCn0g5gYYkZU+k+J96Vu3Od0Fmy020KLQmellISyHBFj/coNhpRMOE4hw3QJHKXZqWwrA1hbPrbVkRM1luPgHZZ6Z2IipMxa3GexKUMf3yjiT9X/MQ5vJKCWNyo2ADXc0x6hMNDEnXjZAZmNKdf2y91r0B5gvWLIamecIWM0gD5LgDfPsAX5O7edmCwmX9FK1uoaClaOgWfAorInOGLSmbszt8DsBVhBrN7lQzrpxlUWgBqEy1EsTCp9rEetsHO+Lza3IBObgQpXkEINocwzpkCVszpyroZSOl1GPHou/SZVTF/RGXaltHhhlk8e5sH+yiocRR8lnZKNMDpo4O6rKtA4rlWwYhb4cdszcIyCITKR15eV2qgAAIABJREFUVFj2Ovzw3/oAD5gqiahqYBTX7MFqrrFSmjkiqIpHrGzaFtqdGMLb0Vk/Lh3rdD75AVTR7ArGwQMerEGCFYNyMPHD6tkyV0C7bllHq61TltIdW8O4vvT7OZlUb9Rc1ABLHofmklhqVqmkD+lUrgef0wCpGl9mqDgNxAygJpU9+14HlXq+tXBLE9LgbD4arpqI+RK21i0bUkjVlrZVLGVgMlvqnsZEWD1LDJVrIZ4fewjyuFmH1fPXPC6OG5DtYTiTfE2vvSzvmrjT2MRx5NnLbr+XuIsdUcNDP0uaGwOYR44nfbCaO62Nzyqz/iMf+mtFWX/Oh1y9DqULA0eHJqbvzueLa9adgoM3X8aiqzRA6fyXTCGSOzjZJYBxfHxg3bICevE2qexiwYyf5ffZ+ZMEEE82VhlKZcki1Js0UxWZKIWBOrhirkW32Ji5A5N+xu46L5g0MsbL6fJRI4jyjB7n7JVOktrDccfVHPUaJI6we3lDzZz1va6TOP0dpspL2yy2kdt4bPbExA3RU+4qnnHU5wGOCU1DWZRCsix7E29hxNrHPMdGl46wCTy7lPK+Gw2Ge8xF2Hzfu/+Cfl669GYVh5aj70rd2RKGpoWJJQ4e5L2xYeaLQyazbELpUeMD3t/84F8pyrQmPxSFBlFydLqmJpn8MOaPqTs2auKYTq0IQsmbnN0nXEjl1DvRMaJipR06OT6UwFho6ASAZkIsH0tCya/QuUB8Tc8+VJzLUmdl0JzX3GhYb0H2NeI9LBlplGbF2b1mvW0OWpGog5gJ+Niyaytzvqo8NlVkzrm1WXEFn2xSqZi9CFCNTUss/KEWeslGWWyX5yInCmCtYpGPwlI6gQWRuiYWywz9/snrahwsl8EiGtZHrNx5MLvb23IIyYDa3bb6g1qdsC21hDmq589d1c/xeOwOiTaBmg6dCSissqv0xVjrsOMOjdQHfU/leBwqbiUA9zc+8D0FKVdi17JP/54drMDB3DKHTuuifVHHSwINjoGSmGkwJgtttyNGunw+ETg6KGVVMKlgVIX1Bs8btM/ySBY66Oa98rjUulQYq2Bo23iUOmPhvdOnrErJefNMI9PvUHOYosDhCQ9EdmBMvSJfgtwF0rbITeNRsya0Ru9ymwStrj0jtdJkNVpnB7mT6O/kCc8Icuzo0Ox04fGwB986eTkql+eaPaXpQv5mmJOlzNPReaI3aewrhZOq/mGyMCNGUkG315GPxKoeNsdW7F6z3ZmkExFvVq4FnUqPFcnw+TZQjU3o2HqGg6Ed6eqlE/nKqy/i6be9c72eT7/tHaLfcfyLj/5jJKsc3of+1vcVrDWjoeUCbG49OjeQ6l4CUDYjUmzPYk77TlIH6OnStoj16lxQnl7FkRL6FcBjf8DGBqwf3NrdRsxj11xRJtvPEsHif5WIDSLpBBJEmVt9fRhgY9O6alO10ox0uk2RJHRmH+HexI5w14LIXAFBbsxcnc6xdo8NQctyW0wvNHvJQxcRZurqZc+2EKKWznlQlr0XVaxTqBxe5u7Fd3AQqzu5XMe3c75YjcTeR9RZOR1nDxn7u1tzT20iFr9EfkXO9VaPx+paydwyoQlgvwAWivKkU9NWZW2iDqxqbaNaa6LV6KqegcNa9Vp3do5Xbn5tfRo6Xz/11DvQdV3Tf/rn/ms74ONDP/pdRaexrVBDdOkyS6ZvNDU9p7OSWEu34ZDerakegxXF90HhrZC7Dhrs8sHBbKBsdMlQyQkBk5ThyJrukAN2v6YN5n+BX0O1zhr8AOkyRzVsSzgPHh6j3+9jNDoS7rC/t4ur166KREpnb7nMMJrYRJUdy/3V0GmtlYol7XcmjKRzcaROAHKsELCyyHnghHYDhKhXW+uG1DoVhMifq/jZ2zqNOLDzgJ58whpVb29u4879O/j4pz4mE3V0ckvdVenBN2otdJunFe5xA7B+MfBcx7DpTAWqPKXki1/4lKp4t/Z2MWc/woVT287JZY/CvDDB4qCJ43j7W79BKeeS1sdTUN/69Dv0O1vHkZpNcxydPMAqW8L7iX/w/UU9bAnlk/ftMoBi+Lj8/nzp6tdy2iR/Hdfb8SpK3QD+CoGzn/VmmQhhSncpNaZJrzCNu0IYsQTLQsYwiDEZMlkTazIePhhKTY7HKeiEbm/uSohoG3nG8GpB7Nz64+9s7+DS5auWQ6AguIzfw4d3UWHTRZW9E4dgt64ScOHE2cFPHFHLOVM+MJz0188vqJV7Mw8EXTNMDiqOviUoOsRWd2edfePpGxwqYOHv41Dz1B8fYTA5xmI1M9+HIZwjuXaaGyi7zgSoYau3h43eNm7fvqn4njgMC02mMzNtE9ejuVJpSCupDR+11cyepVWrodfdWZ8pPJ/O8K3f9u1rAfh/P/k70igcg9GxfBnvJ3/2R4qunCU7FaxU3/wQT5vmyNjQ2FX9TBcsMLQL86Qua4Loo9NqCMWLIla2mOY4OLrnOmKbmu11tuAzjEy5Y+077tw7wNHxWL31Iy9STT1VuBwkYukKLUhxdtSvBNhgQkrwrzGE6FewPLw89aTIVwJqiowNFhdSx2XoWrKbWVPIhVoV9oyNdg3TxVhdwDmIySfLTBjSik2EOcqjWELgPV/3DfiW932brk1cgBXA//rf/hpZI4rTlySVrubwogDTxUR2/GjwwHUes27p1UoHDXYP1QkmPFM4lN9i/RKYq4lUOiZug4tu+JOJpiRjGb1rec8TV1VZPSnhOn3+8UvX8N73fttaAA6PHmKuYlbg05/9hE5t937iJ3+giEL2r2OpUaFJKEdW2I2QFVP27WOipaw26XU7Vs1CAWi7VnHyUssTtJqy+YzXVSmzoAYpMBmzM7i1SqPDEzfoKDGGpZ8gr1LhECekTDiRRElhyzL2uWGXMPIHrbESB3/HGJ+DZxHwb8nuYb6E3jztpU0iBZhHt9qJIamL8cezITw2eHLqVPY2g3a/ewtVFZ8WqNcj/IVv+8/x7ve8R/G8fIzIIiaOo6MH+MQnfgOHhw8wWYykGckeOh5YBTIHCzOIOvPIV44AbNVn3VJaza60z2LB4hIifbbQTMuTXURTwlCzDOmGrnHEZMokFcvozb+aDvq4dNlOSOVgLqE0hXG1yiIAeH/3J/+rIgjIOKWkLjCaGN6syWH7U05ECSaJpNBQnT6HqsZds+JalVQmW4BSgnvt05jPWNyxwoUzj+HF529pQWllmi2zX9/4jd+Iuw+/glZzC836BqajhUK7Lz7zeSzIwVeNJbODTAc3UGNiSrkAerxsJWfpYfoFJbpIFa/0tSNVcjL39603cKPZFYZQVuR86dkvuXvmEQspBjOjhRMfYDjMbJ6aXLChtmsERYoYkb/Ll9+oli/URG9+E08rAb785S9ox4+mfZmuyYIl9qSbBajzZHE9S0dVU0889hZ8/tk/kpZigYh1QLNDMLXgQU3haVncGftG0Nns7KDZYqbPNG1/aE73zVu3QH+17CnNY2rp0wmTcAm2MufDo3IZzXg/+T/8UEGoV8fDp+Ts2wRw+KFJ62I1RbdhPAASN2kPKRjueF3ttkYrwiIx7j13MpMhf/F9f10XVwXsH8exH/3oLwnsYNQSVewhn37rn1EiqMgiTczJ8UD27YWvPa/zi3jsKkGjN/+pr8O3vP+7cPf2Syq/unjlKQz7d3Hj5Rekghn+vOPd79N3/vq//mdaPB7AxMX/rv/irz7iSiEWc2nuBH3nlLWI+aNP/d945dbNNe/v7oM7YiRTU5VM6ZXD+3kaCEfVxZKtRgcXz9j3jCcDpBmLRxg6FxiMD9RSrhLXEHuObRUxjOviTU+8FS/d+Joyo8zIDkYDnAwOMF1YkwyefhrH3HDmL+moZqHFVopWnujy4KG1vZ0vc3X+KNPB1JTkcJZDnVDDAMtFosZXKg378I9/4HXnBnL3uvIqOTSxJQycmmrW2X9va82SpQrWgquR0gkaDWu9XqlYXErtLOqSiyae/eonlJpttsgXNDX12NV3oBo2dFI3/2vVu0LC7t25q/bqTzzxZpFQHn/TW9cPUv7jha9+CrVqS87V3v4+/g1tMIBbdx6gt9XG3k4Lj1/9Bv39s8/Y6WBvefpPq4Djs5/5pF53N0y9nTv/Bty/f4CvvPCsXs9WE/h+jAfHdzEvsXY6sx5PLuGG4eFXJthU3TsufM7ymVg4nc6W6O48fZxmj7tbLXEzHsBRQatGahsdyT1phaPBHS0KKegHx/SdeDbyTPPE0FjDxdmMMk/tXkTHtbpnHSI1xc1Xb6inoIplUzrbJI+UHdzp1xiApXtmjwZyXf7SX39vwTx04DD+pquOJfs2cinO1XLqkhgxTu1fRq+3rS+hfX3x5S/ryDOlXB1Qz+TJcHKAnZ0LmC+n6zAm9wc6Bmanu7VuxNBtbuKpx75u3SGLWohs2b29c+ht7CPwzWO/desGDg7u48UXX5Wmes+f+dP42lefxTd987egu7GBQX+AL7qDoK/ffA5JPkFvL0UjPI9WtYlv/LPfqXsO4o4g5H/5yz+n19/+HeYl+34FL738Ao7cCaA0u+NZH4cnt1FWO89nZvr6wweIqyk6bVIvY8RxHTub1nDSCLSZTE6vswm22eXikBjKWmA7ui3A3ta+Pn9m97yqe7/83Bdx6eJVnDlzHrdevalFvPnKizh35gJO7Z/RZz/2yf+gn002omLrFxfvb27acXl37t7F4fED9PumiVMssbNrDT84hmP6IGYOBpOHFvn95e9/X0HyhNQBz+rZ2Ma4RMs8EiuX2N+5ZCFMCLQ3awjd8bL9/qFq6rggJ/2xnEFKFUNFDmLSpHit284Ec1G1WeZVEhp5yMF284qKJOiIfsd/+tfWjR6oBl+58SyGwxO8dvc1HBwcYGuro9+///1/Fa+9dh2f+9wnVDK+SnzxC2/d+aL4f7NpA+feUMFqFuE73v9fInWOVHfjtEzG//7r/4fu8XBoPgDDyIvnn8LxwNrNjmcPcNh/DfXWFmZLA1p8h3/Qp1ESRwrQyCkNlzij2qWjuVhO4aGLi6cv4/zZi7j/8C4mYzaA4uetK/nZU+dx8dwlab4LFx9DrcYqohYePLgpAO23P/Z/4amnnsa1a4/jzt3bOr6e4/kXnsHW1mlxDThu37GTz+DFYHOM6dR8guPhQxSesYc4mBonD5BjthgZEPTBD39PMV9apQ5Vxmplf7y5v429LfNsSVJQd0xl2djQ0L5kMGSRJzt4ZhiOR6g4n6FM/25unbKGEcolk80Ti2N/8bxJrO45By6deguazY51AmNXrciaNt9/eAcvvPRZIWGB38b29jkd33YyvI2tjXM4c+oJfPkrn5Ht3ugxbWq2cngyxvbWBczTMRazke57b9dYM9Z0KUVZADRb2oLv7Z9CjiEmS/OBeKxN7qWYr8jacWnushw7Zq29naJiB18E6NVtRxfFzNBJVxpPNFDdVXTE/VDzSI3ba21hf+8MTu+dkcPKHodb22extX0ar978inD6l9iiV4mlFb75z/+FdQPJV29dF95Qdj995daruvbde/fVVaVkan3txefWfRD4+7BmDbc07zq4JIf3s//8R4p5shRnTTiAkx5+SI6fbFx55Cgpz3MsXAVOiQhS6tlAqWwUWS8bTfmJKxwxVg7TlQrv2FyBTNplKkfS93m0SuiOVyEiyRCQHbIy8ef52eP+fXnTRyfWft732Y2UtndXqeP9vS4IrLDmZNxfYD5e4eHBA7LyEVRy1N0pUdO59SYMc4sKlksLB9mSbX/vAjY3zbx95pmP48KFy7h/cAdJbkJxeGB5hG6nZ4ddk37lijHe/sY/p99963u/E3/w6U9iPBnqrIC0WCgfopM6RdIkx4AM4qFwEeIF9KH2d07j9KlzePzxp9etXv/oD34PV649hQuXrum7f+e3fl0/3/aOd6lMv4Tom47W9swXvoBnv/plVTlxkDMxn7v2dWqFt/PoRLd0hsliAu+nf+mHdIoI6X2GAzyqDt7q2JGpBRw2oBOsWNxoUqR40+okRN2aO1W5vWG7gd46D4diGpJDWtMjaNSU3ZtMFtjs7Sq5UYmbakZ51L8lEGexmoDBw2pJokeuU7iIBTTjjjtGhUxmCicbJrKcu4JaZNU+/cMlNtrncHDyVbA2ocgruHrVThW9ePFJOaZlV/C/+1N/W+83mgs0O1U0W2XCZVOh3NHJMZJ0pjP+ZlPXlbtCIS7WXjgnrxFvY6O5hwtnrwgc2tjcVB3jv/nN/5U5TsHCjO6+/k+/zwghMYthFnj5+stS95NxXyqdnUbf+tZ36DtOnX5Mp5aNh6alGB1tb7O1S66S83Khd/es5c6v/eq/RKu9oQIfDvZVvHDhgq1FwqJfhqwenn3u81ikM92X94N//1uLuLapBgbis5NqpISHp1CFhyRz0UfjgS5crUTodszbrxHF8j2MhkcIeVRqebJmEOqwAhaPiujhvGWWjfM7avGWwI5GbRtPXH0Km5u7uHH9RZCGxetRYG7dfgmT6VBOJB0ngx7IOuoJs7h86RLmKzaQIvW8gB8O1KKNI094xu4Wjk9uodUgl7CN06fNkfrK8y9IK12+cFGvn/vqp+1vwDbttO1zbG104YddDMcnmM55wJVryVpY0aZ13mBjR9tp3BDVyLKojXhf5JD3/bm/iEuX3oCf+YX/DkXAGkMmpQxe3ups4dqVN+Hua68gd3j+255+O+7cu41XXnkRO9t7EvI3P/m09Ut2hJqyIuvK5TfAIwjjB/i93/kP65bwZFyx40lJuB0O+7h48bIY1hzkfpakmU9/5hPy77wf+YffWtTrXVG+A3+F8fgRpYhpYo6EUKO6bPNL2Ccvxu07D9CsG/edDmKr0ZbtG02WYOOEBwcPRFw8eDhZAzRxQMn0sL99BRvUEuzZ297CH33+t9ft1Kz5EXvssI5urrasluOuiRBSZy8Dzxeq1u3sgnUsVEhhsJBK5Vgmc5z0j5DM2lgs7yLwzq4JlPOV9dDhUbkcW1u2cDrdpBhiNDWVmeUT1Os87HqKo76BY5GbD5V2q2Kn7C+UI1vZXIX+BkI/Ri0mhY3KMdEhT6uU6W3j8NQrDWxvnhKp4/6Dm0iyBYqshs3eGWxs7ILZ2fGoDx7lQ9P55JNv1nd/4XOf0c8rl6/ZSa2OFv78C1/T+6/cuo7z5zm3Fqp/7Hd+E29/67v0Ow5yA8paho//zr9D7iXwPvw/vr9gked8Zkec3X/oqmhrpHOVWShuP1P7alYoXluIJOPJViSMeJhN55jPLOfuFQ64SCxPwL66HH7AQyFWolWTr9frsef/XMfHW66bTZ3Y+oQvGS/n6HYc1l9tK6lT9a1sanOzKm5+FFnvHDJ3m227R7Zl2+iewbBPUuqhHNSzZywcCsNN7G8/juHQMPHBiS34zVsv4mhwCw3X939EX8hnQ0ciaaYBFnOXSawY1EzkkEOm0yGIodA6HvTA28yx2dsXGMMsHDuhKOfCWswU2Nu5gGaTR+IA8+lceZSN7j62NnakWZ/98hexWFg7eY6LF40Q0h/0cXp3H71uV0543/UHePkGD8Mgt9EQ2a8+/yy2th/xO44P7+h9dkU5GR+BlEbvp37x+4rZ3FqZ8UGYo+52zA6W+MdwROYPF92ObQ8DHv9OM0HVYuqZIVzL1QPMpjYxJ307iTwIWfnDneqaREVMmFgLmsGQKdGKEjdswkUSKb1rppKJXNGhLLtdEYHc6p1WzWCe3cfeXk/2jhnFdvMCHnuDSXs12kRajNRI6g/+4Pdw7uwlnDlj/sz2pqn+W7du6uf58xa/3713E//+4x/FzGXLZskx+qMTwGMnMwu3MlcYEobMcQSoVC1XQci8LMDhZ7l4zH2wacRWuyeWUb3akH3nw83miR04QQ/y/+vtO4AkPcsznw7T0zlN93SYuJNnNmvzggICRbBB5gCHc7grn111oerqfDYH9l25TNnmbOM7wAQbbIywsLEBoWAFkMQKsUGbVzsbJufQOceZ7r563u/vmZUsaeVz+KpWo+npnun+/y+84Qla2Xd2blxSYbvNA7+nC15PEDMzU7JQiCbmmJgY126gG8ODY9J67u8fwODoPqn+nfnxC8gUSmKAxfHjV1/eKsu7XG7JNAIaQujy+BnlJ/DxP/pQw2RMSF2fq9xkVFEwByVkOahkycqU3eIQMEhTvJgybPK8Rl26gQQ36Fv0It3GzS6ViUvhpxk0qioUO3RcNtwdjLC1emBoqcoF4ZGjOlyqGcRGkscRkPYtt2Kie3yedmxWrfB6LYjGp6VOz3PV39a7BZlipy0c7ITZ7JTWalfnKNbWVLRP6XvGAOJ4wkJMj4oNrt28hOX1ia1mkLGFVK0y1uPzKG8y+t+EqaGZSGtiDgR7cqjiStNij0qk9BZgqkizSS90BiV2zQkh6t8C5ebxRps5ZSQ5tzCHYrEmx6dBbxVyiNMWRD5XEJIOx8jwbvlKPCLjgSZtvXm/zp87hY/+zC9hZGxMCDWf+ezvIaspprEc32JWzq4c6VwcQkj+3T/9eXE/E3EC0QxWAR5HUSsIKUFjRURkvGDWSr3KTk2liARlkujASc3Inzn5xgZz4opsf3od9YLYC2c/X9nRKAligkSJFDJJdiA26nojzK0+WExedASHZQIk41HJHMb6j0rMwSLTwsp1ZHNx6ZAZjFXc9e4PqAlZq6JaLcHvGUW2sABziwsm0/bn2vqAAP78a6oi2NnTJUcTJwzHHXuPiDz76YuPo7wFydKugcUopdZ6zSUrX5hLdg2JuwEYWq2wW4JyDXpCO2TrZqDLWkdzXL95Ft1dw1tCjs/94G/hdtHC3o9Ll67Kil5amhHp3lBYZVVrq0p+r6+vH2OjuxAMhsTh9MKFc7h65YIEz0eO3omHHlYt4N/51G8hp2EJ+D31F4hRyOYS0jVkMUv3hW/+10Zlg8hUdTNZieJoMWwDLGhlqmBUNeSEL6/x8QiW1CnSCGFhxZJKIYVCoqOi2Kb8UUqWiKihqIaRrkRkkKaCYaQ2IeFTVhj1VmzWONsNQN0iN59ACUKzCsWMAn1IAY4xQDuy+XV1rOiB7q4uuB3qQjGd4hFitXrw4olvYnjwKPx+lSotL63IhN05dkC+//4JVREkzOrStWfhsKqjwmV3C1dxZukSNhoqDdvQENNkHXPSFvI+OdNrGwYYNERyId2At82J/fuPYi2ygN0jh7F3zwG5ToTUv/mo4+bEuNQ3OLGffPJxWUwk6ITDXYKT5Odxu71SDhc/5VtYPgGt3Hvp0jlkchns33tQ/syPTp2QMjkHd0qCS5r8TU5gxgu6z/zFv28QVsU2Ly+tUUP28kWZ3JqSLBUJGGrrKJkRuYKy9SsXJMkbGqQ5qcCK3UMOFjxMLXa0afp6TNnokKmrW3nYyHMymar04buDfdIoYtWPx02hXEA2V0SFEbKIaihVLZvmPkocQa2Rg9cVEFIFo3C3oxuHDhyRpg7hVn39vXht/FWNcKkCVLeLxS097r3rp+T7v/72V+VrtrgKq6OBbF7x7wgE5e7JiRFNqV3BILA5ikd4BYvApqCRLeK6BbkoDahriK9U8L77348HHtwGYrzFXX/Lh0+fOoHZuWmsR9axth7RFpReagAcoyNjGB4cxMTEDUkTZ2dnJPXr6OyWRdaUjKFUD72JxYvQZBATjaIGCClXywpi//tf/khDZ7BJX12oSLoG8kUF6LBbVVrValKAEYWmof4ft3pFBuFQdCpW+TQRo2oK5lYrKps0a6BGrorOWxgdc9Js2NCoqdXgMIfgdASkAUTqF9MlIn6zhSSS2ThSeQZiTS3eOtiRZExhNbcq/wAjJWuMaPf24Y69x/GjH7+IpcV5FMsUkbTAavPAZmNrWE1KsoS8njAeeO/Py/cXzp2Ur0trM1hPvLa13VNazmYO41f/3W/iz//6M/KcaIxBWAPdnQdkoUSiKcVnqOsQX1ENGEPNgHff8z7c/6BqTVdoW9fCruq2acXtJsTzzz2JO/YfwHPPPyMxSzqlJm9ewzV2dnThwfsfRpCWv2YznnnmKfm5OoJ1YsxFx7PTZ06Cz22OWDIqRhocbFnJTv3f/2BPw2rphkEDG2zU1B+r1XPwuVUJ0mxqQy4fVTfCRPwg+9kWTTBRCSmbWpzIFRMw6hsw6lQaaDCqYlBTVCqTTTKKQDmn0kcuba+7B71do2Ig6XJ5cGX8stS80/lFqVal87Q6q6HN5ZDaAKtbfClXqLW1Tfx9CAXUwS6tZP4NFo+gK0pfge1QbuVNJq/b2a2MFjVhhtFBlWNfvHwGjjYTZhcV+YIppt3Shngqgly5gDo2kEktSwpcStPt1I0Wg1czutChUlLXjUre/f1D+MkPflhSsFQ8iumZKenK9fQqzMDtxt/+zdcFuUO4G1c9JW9pz+Pz+XH21TNCMg0FQ0rMUtxXVcl+aWV5Sw2A+IKLl87h/Q9vYwLPnntVCnrLq0tKa5ENwE985l5xrtmCVhkyQEMFK80z1evhTqAaQEr/Rq1ely2Eg/sflBuZScVxfVpV1YqVHGxmL1r0hHST5qXiirUIu4ebKBfzUgQxCqLFDBuhUCYXwuFBzC9OSaBYB1k/ZAh5lMiTnqlqDpt1pkRUyjCjRR+C39t0EVNACo5MJoGNWgoDPYeRykQEqFLX4G0BPxtRlG3W5G+0LiG1ANi1zOdVdkB/RGYRRPzOL04gs5FEuaxWz+SVTRAEOja6R0nU1euS8gqzSXiIxAxs4KM//XNYXlqUrZo36aMf+1mhcL+TMTV1Uyx7O7t6pCcxNzu5pd3c0alSWSqScDz9xN8K5sHldiEaJWpaw1suLeLIsePyHPZLLl48B52GzXj11R8ptZX/+X+oD9COqrbynXYVBMkEYJOBrWJuLA1WCg1wOjwwayyYfbvulm2W7cd2dxitrSrSZgmXI5pYQiodRSIdwcLSJKBXtQZl4ULjv+BzAAAgAElEQVTWrRO5QhzvuuMRCZBYHWP1i3HAhYsnpWvXv2NMiJ/VjSLOX3xZzq0cI/+GHrtG34XVtagyXC6to6eriX8zip9QW5sbpXIOkdgUMnnVMvW4LQp7WFHBUbP10d4exujITjGuPHfhWekm7tt7N2w2F+LxBGbnbuL7Lz4vGcnU5Jwcib/23z4hFzYSieCVHyvAicDna0UM9O0Qt5STp07KjSOH4J573iPlcl6Drs4eMdPgc95qXDh3RlhZTZJusyA1MrYX8di6yOwTPPuNv/yy/Ar+zrHde9EeUAHv1776RQRCIaSSqpJJkQyCQsnXPHDwqDoCvvjo/2qU60lZIV4Xu2n04G3mt02JWFUDD/mHUK3mQZ9eDgEU8MxxeHH38X+zhQY+e/aEpElrsXkUK1mUKlm0mgyobujgdvqF9ZrNaZ01dwAPvPdjYr3OXWZxaR6FUlYw7G3esEDBOVm6u0aE5LG8uiCVrqW1S+jp3CPxAuFf7KTlNUIKewXh9j7kKwmcOfs06roEymVVBTPo2RwCchmV0xtq6mso1I1Dh44pIkldgTGZ3tGfcHVVvfapp55TIlclpqrKmZPcO+5qOp3S9GMnk4imBx94H+666y7k8kVRTz9z+jTi8QiGhwZFrGJ9bQkf+5lfAr0TeA0z2bTwDyg82dRXYJOI+MVmM61J7T3/6mn0Dw3J++Q4f+6CfM3lCjh65BDaAwH4Ax349rf+CoGgyow4ousrIDKYw+6ikokRukef+ESDgMSmVnChkFY31RaWggQrhD5PpxQmmJvG4kvQ1dg4MuHYkTvR7u/eKvSoP1NHOquCyJtTV7Fv5zEtwwC+9/dfR9DXi4Xl8S0K9O6dd2J4cA9S6ZjsFg6HW+xeJqeuwdfWIW1SxSRWgeSl8dMoFLMoltfR3jbA5BTLa+Mwm7xo86jdyyrS7y0Ihjpw8+YVUDxi/MYJ+VkhX5FUspBVR4BNo7/5fV3o2zEsPIPmYP+eANemnOtv/9an1BFW4qQzwR8ISAeTNRRbM8fX6VEqZLFn5wg6u7tx5MgRxONxnHn1NF568QdyjvNMZ38jnYrLLqsMO00YHBxCONyhUuQGz3aPpM3UACYphlU/rlqPxyNOZjIxGsC1awrGduXyZYyN7ZTfzxEKB5BKJrZEva5evril5PaLv/yfZafVPfrEJxtsTXqdIwJ+zOciyKRV6ZGyqhwM0Ggbw9lvswQxNDAqj3d39t+iQEGmsOY3uEkvP/LfqQm4TW2KRCOy8q9PvKgMKAHsHLkTd+x919ZFr4HpZBnWVta5N5HNJzG/fE1WVyqTgMfVKTvN1NwpWG3sYZMKpYfL3gGz0Y2RoQMiTcPJxOoffYqMLXVcva4mANFFhH/dd/e/le9PnHheTduaHktLcwh3UGhCWd489NAjsNrc+M7ffR1Wqx0nTpyW1m0ynhDX8fYgAS/KeKEpNsVFUikVoG8UZPv2t3fA4XQiHotJLMBcX1w96lWEw2Hs2rlHbHGGhoYlUKRcfbCDi8qAVCIipW4KSKWTcZCZRD4FbX1EVU2vl5Yw2cqEuT391JPY0dcPv09Vc8d27cTMtApqOfr6hkXiTz5fq1VNgMef/7+NjQpVORTJ0t+mSqMspqQz65qreB6HDt4j24fLHkBXh6qfNwfRLpG1RZw485RE8KF2lT3ce4/KhZvctpNnXkKlWkA6s7yldc/d58M/+R9f9/vIFZxbvAmfLyBdPf5OungmUgsIBcYwv3gDc0uTMLYkMNx3pyCVGSx2an93emYCi4szUgvYrBswNrobBQ0YMTt/TS7uoX3HtAmh4pWB/hGsra/i9Ek1UXhuj+3cJ9W3aFTFD6+Nz0gTh4Us4vrcHsUM4vHICJzUcCqZ8aw+evSQaPtEY+uiUxiJrGFtbVFiCErYpFMlqeX/xsc/gRdeeFaOAq5+VveoKtIs9FC5jMFp03ntdRfqTb7JpkmfS0vg3N4ewtWrF7cUwzu7umAhsNelcBMi2Ht94kJjeXVOkCmcVX6fCiA4u9xuJ85ffEl4bGLm0GLBju6dGBtRVbQXXnpK8HVyZtbZtzegMzQoruEcI8P75eY3u1Pf+s5XFALISFYLZWQJzAjgkff/qiBsmiBHwqBS2bjk/K9e/KFQyyjH7nA4kEivKyfRwgoMLWmgzpTMjqP7PwiPQ818BqzXrl+W5gpt7gKhLklb5SPTsKLM7VPV7h//7jfkKxsu1NDnccHBNIxxgMvuRyisjpYTPz4h1TPa4hEXaTYzgFO0OT7GodfVYGUVsaY0D7xeB+6//2EkE3FBEvHoSKdS6O7tAxE/H3j/B6XlS/SQ0+WFy928OeoWsZjGqD4WXUOX5nJKyj1jH94zjlQiBY+PWYAP2XQSJa3+7/G1Y2ZqQo4QFoyGR4a23FA9bq/Y/OiWVqYbjHAVF66BdFYBDzm4/bqcfuQLceXY2dBhdOA41iMrqBJRsgkkcjOC5tnRuR+7xw6DXafV9UV5fTjYLaigitZOfeXks+jp6cHCEsEfGQnuWk1tOHLkQZhbWrEeXYDX0y5bJM2djTpOPMq0FXD5xisol7PYRBLQF7FZzygRp5oZLQY73nv4P2297+b/EFF09uwZ7Nq1Z0vkkaUrXgzGCRw3Jy7K17NnT2NublqArxztvh6UKznkC5sCLZObW6dnYR3RyDI8be0wGpsWuerGy8QpZMVOx2rn0QRYrXrRWWBg+Cv/4VPweLYFGyj39oPnn5cshje5u2cHDhzaJnP+gw+kPSCuJHQt0Qg6K4sKEzg/M4Ode/eiXFAleW8ghKmJ61u4gfHrVxCJrqKne0CKeXe++z1MxxsNdsLSmQh0OsqbKOaqmgBR+UOpTFRRoKREq/z1OOwWl6SENqsbB/fdB7dLfbh0RlXFXC4vIpFlrK2rLZRSawyqkullWMyq1jA2chCnzz+BzvAQGIhVykm0+/o1kqUJk7OXUCzlkM0nkKlcR6uFeEADQv79SCTGpcnEpsuhXT8LvZbbv9WF4+NSABG3EpXhNBG033vqW1hbXURF4wGSlMprMbcwBT3twJneptUxmUoyWLXf4pG8LVS9WaejCp3SOMFo7VKF1coqYAPtvk74fG04dvxhOVp5/DWFPmLxOBbnZ9HTuwNDwyNb7+/tPssbf/b0k48jGAyKPD6Hv82LPFe5Vrq/eOW89BJ4nFAGgHGOLhZfbTCI8rjbhJV78bXTUnHjIJCip2sUq+usNStAQ64QxaZGUWlzheFyBiQNPLRfgSJvHcl0QgI0atizuRSJrctxsLa+ICuGjadgoB3T8xfgsHukHDzQc3RrxvJ3XZ+6gHI1i/7+3RIsLay+gspGBi6HD4XCMmyWLgkKezuPwm3ZrmHc7sKlsooA43GqneDpZ57G9Oy4dBzlcZcHS8tTKBSpcawxpIpKFi+TSoh4o9jRiQEI/6MKZYRdtbbq4HG3IJ3KyjHKyUYLXE4KYhfEfa+ux4MPPQKfPwyfz4fV1VXcvHkTCwsLePDBh4R9xSBc7HjYt30H4wuf/UPJHGhevW/fYZw48SS8zm29h7b2IC5cPKekZQ0N2B026BaXJhu+Nv6AVbwG6rqSwgxL4yQoQVgytSznLlGo+SJFmwlo0KMjtFPAGWwk7Rw+CqdTFYKIRGlrC8Hb5lVeAhqIdGpmQihQekMdhbyyirVYjZhdviLlXEKpO0P9Qgvf0bVHftfF8RNycfv7dqJUzmBq8RTKlSz0ug34PAOSeXAm9/ceRyvevOX7Dq6d+luXr8CqiShdvHgG9YbC4+3drQLG737ne3LzqF/Mzhqla0Q/WE9pFtUH8XpdAmptbaX4dUHQ0tQHyuVKSKQyIGVblG90DUl3f/Hnfh0mc6sUZ4aGBuV3nDl7Du0+H/r6VMXvjUOoXwToajvxtWuXBeh6/fo1YSHHYqvyEruzVfSWiaZuNVjhb++U4hHH5fHLqGzkobt0+XSDAU8suYxEZgl7d969ldpxVnKUKgxwFFGUOXGrSW3fXHl7dx+Hzerc0tHh4+vRmMCP3W8oez7z/GMIBnsxt3Rqy4bGYfej1UIhRWWzunfXfUKdomgFsXRzizfk73KXmVkk40dJ0xkMFTjshFD1yISx29phb3GhUE3D9ha9/ze7mDOzk/JwOLwD165dRamsgjkGZqwrEEN3+cpJCSZPnzovP0uJCENOEEcc/Kw2DU1sl7OfNq86dHYGBS1EKhqpZIUCMQQ1cTPjjspGmV4j2VD7j6oo/+M3Po3pmVlxYR8bU+n2m02A1ZVlJOIxqQQmE+rIZTzjcNpw990PCGbimWf/bktKjj/nZORiZT9nPRKFHFffevJPG26HCx63yn9Z2m1qzDTP9PEbp4jHlZXf27tLa6lCAA5Mawgja/P4YdHgyPw97Eu7CIGSoSLuZ194TAKjQim5VTWkBFuLVdmrU5No9+hDEsWbNV0evk5k4kpZ2CxOnL7yjHTXSmWWNVfh8+yCyWhFV8du6AhV4wdt9W+lPm96BW958MUXFNa+WNzEocP3KCwCV+GZl0XUoVwsCASbg9JHTeXwbDaPpSXF4SOwZWRYpc9F7fhkYUehf+iMbhKyBicq6yBkRVHAQqxhdGrn4CSh5A0Vzj7w0MdkES4vR7Bz1xiGBrcp3nxuJBLDysq89Ae6OneAzSOOuq6Kzq5uDA4OIBzukSLeldcUkJRobvYKmgJShw7ciZXVZei+/dTnG+3+PljNKqhpqmrzRX5fUOrQ129e3FL38noCIhjNsbq2iOm580LQ3L/rPQi0q4uQTCsN3M2NvDRTbFaFHTx97hnB2Gdz65K3c3D7M7SS6090sQ6DO+4VMGq7+83P88ml11DeKAoIYz12GYf23SPSaDbjth3tRqOMFo2Je7sJkNbSt3SuCJvZhJMnX0Rv7xAuXjqJSpXdTTMi6zlR3xbzRtEZpmJJBeurVD2jrE0de/aS4lWCla3nTWr5ql0kn6cdT0U0Ek2SNVC+VS9iUCTZJLKa/k+NgSRjDIpqB+Rv9PWOqEKW0YDDh1lRVfHJn3/tLzE82IdFDdc4Pz8hj2ezJfh8Tpg1ncehoV3I5BZQrZIqt4GuLr/sPhyDAwcl5tN9+4k/afR07xSH75mZGQTaw8jnSiJucMd+hSxJZVYQT8QlO+jvH0OukEE8EUU+tyLi0YwDukKD6AjtwOT0OfR075Kgjlh40dXX9AV4dBE9e2PiDBqaMMOB/cdx4frTgkegUIUOFlhaHRjqVZo7HPw9a4k1iRmImKX83PTSDSws/whGI1FHFXjdAXSGVEUx7H59oertJoFyJAcyFFcwG/HX3/yifF/MKyKlw+7Fa5dVmhXupmGVkpozGJ1YW0uJlA0D2j171SqtVhTnrqq1m0nhWl5ZFUuetjaXxEMUeiQlnkrhsbQCvGxWy8LlJ9aPYNgqhTLqeukesmnEMTAwgs6Obnzjm38Du10n0vHMiC5dUqms3d6G0ZFdojXE4fHaceWq2gF4HzrCIRHP5nA4rCLKofv2d7/aiCeyKFeVinZz2xc1Ta0UzHON2kG8ARu1PFyCqlEO1MzFHbR+revR263OLOrUcDSVsbdvZAWR6DwisYUtyRYKHyfz09jReQRBn3p90+uu+Toxb6ptCL7N66BEvBGJQgLXJy+jUJ5AvVGEXl+Fx61q4EdGfuXt7vnrfpbPqpz5tSvnQXXQWFShf9iS5hnP835tVcHje3Y0HUK4C5ixsMCiVF0YTql4Avfe9y5sbuRQqmwgkkxq3D0jpmcXkUyxTUwmsRH9vSFRYnc5W7C8TgQUkE7nRHtYp6+jqyMsEy0U6JJdmZVOehCurKp02tTilUyDNQuDfhMXzt9QN9moEzJKZ1dYAuN733sf/uqxv9ImAItfBewcVQwpdiFFmf3b33m0EU/GkUixRm7YioJllthtKG/k4XBR2oUsHBO8nj6hc3FQl0dElus1BPy9UhMgL74pIfNmdyFfSGNy5jwaojtEPYIS9C05hAMH4HZ3wGJQJsdvHCRsrCfXpIbAEqbX0wmno12kV6gNNL/6hEwA9jN6Aw+jK/DOwBcXL6gVEl1fgl5HRRLN1FJvRDKVRbawiRuTE3T4w+gIjy3liMoJMD1DzR+6hWzC7VR9E6vVgJW1BGoGgxwbbodJNJFoZsECTrHIXj239yA2qnlEkiSvMu30o1wsC3qqpUX1Ytr9AZGH6evbJVXQyckpiYgSqawcI6GgWukzs+oIMFvaYLNYRYGUIxDy4fT5V6Wi6na4kSksolLNiDYBORp3Hr8fus9+/o8b6by6+bwhTlEIV4GQ0VRCRziIXGkZUhtqsPgTgsep2L3C69Mbpf7d3qY4aGynblmd3HIXKZ7sslqQKyQxPX9eOoulakECyXxeoVSJDxjsen0doPkryM6ZXZlFvcHIuw6L2S0CCuxYykVJX4fZrFrMTttBUS0bHXgXLLptC5w3m5C3Pnbz5mvIphQAtJBLIhpdxdxiAvMr6rEjdyhFcmIVS6UGZmaiso2ScRwM2EXznxOAKumpfFlg9WRSuV1Wwextlg1SMqa+sd1GPqQeMY07QTVSBt+sCNY2CnIvOjpCSkxLg9/HY0l0dPaKAxn70XUd0+kiujp2YH19QSYAidtNEVkeE9DnkM8pXYPNRkSANeYWKyy2TeVk8Fu//WsNu8MjevUcphYDhkc6BB+gN6mKX67AaJdvBAj5xjDYp9Q6RHmr0YDV4hGeIPv4HF73tigBPXDIJiYJwW5uFZLn9enTgivQ6z0IB0OIp9bQHRxC0LdNG7/1xjD4mlpeFMnVdHYOtXoZ7b4uKdU6bGGBkTN9y5fVVkjAaNC/G/2BbVrU7W7+G3/+yivPYmbyMq5dnkD/sGoRcydkuXxtPYZcnvr/7OopSHyLQWEOOzrbRPaeHclIPInCBgG1dZhaaTVHKBp9kapIxiLw+zwoaT7IbFzxSOUC8vs6UK8p2RtG8uT0U5WUqGQeq2yM9Q/uFXhXNpdEZwdLu5sihMWgNp5UC8FiL8NoUuggKpMSVNPUfgiHwtKR1H3ms59qNHQmDA9rdOnGzNa1cNjVY6nUkrwRbs193XtQLXMLDkrxpznE0EgjHbzdxeZWfmP6klLu1CkbmURqHQPde+Fxqq3rjYMyaYl8QVqjxCMwyl5YPQGnrQPBwKikV6VSCutxtRXq9SXBOHaGDqPHr3amf+y4fu0CfvzKU4K7LzVr676gRnjViSDG0nJK0z/QQ1dX0fy+/aNw2s0w6mu4PjmNYrkGh8uLHBs0DUrysn9REywAXdeMZoWwZtBHhnU6RS8CNoXM0txhM43CEooKboBRz3N+QIJv6i9wfPiRn0IiSQCNE1/6ypdQ0Ch9yfwS7PZtiRjK1ikrOYLiCvC7fNB9/bGvNhwuki8NqG5m0TConBcNBiea9ctmHQ6rU+r3dx//WSE/vnG8MeB7qwuezqYwMXdFUxLZRE/3IJxOvxAvTRr58tbXlutUJd8UsCWPkKVoQnLlhfW/kbPXbKLAggkWsw+Z/DL6e44JJrBYnYHdGoCltR2DHW9eUHm7SXHyledw9uzLsLu8WyvKRTs8UUprRSKRxsKSYkyLb5/GIB4a6hLSrK/NKh2+RDoHvcEKnd4MtycoZV0il2amrwrFvE5kfV1BzJkp5KmRXKblPendVThdFpi1yJ1S+qwXCM8AlMpTccfdd70XPT1qos8vzmFlTTX0nvr+YwozoQXWDK2Uw8sGKrWM6A3qvv7Ylxv61qREwNxKTa1NlTByBNSK5OM+Tz9sVhdGBo+IDh8HV/34xClJ+XZ07brtIsvks1iJLaOQj25hBLq6yJrpEcvW0kYZbRStvIX0wHCIH6E55bgbsOx5aZJVuRSsVr/QxcvlhvTjHRaHNI9W4heha5gx1H8IXss23e22b/KWJzz6ja/D5dAjEV9DtlCC2+GQN8PaCCP/9QhFKtTKjcXWpD3b1dEOS6sOoyMhgcsXKzwqsqhutsBibZNdkq3cQiEBCkpFNMYTfQmorFIokLzRgs0qtRg2RT21RXNvI82bGk6/8PO/gnC4EydeflGMqYaGR+WakS3ECuKVa5cktTxx8nuAXvU8eIyIInqNXM0W2FotgurSfeVrf9Aw2+uSQrCL1dq67b/LVUVOAGXM2cUjfv/Q/odw8TUlVmSztmFwcJ/oDFtMt0e73pyfRCwxj40NFkfUGxseOIK+8BDEzJhmTG9IAhejC9Iepq4fUT42G1HAVBulhOoqLGYViPaFt7d6xiYXJp6XLqXJaIfX1YE2x5sfL2+7C5w5i3CwDZOTl7GyOAOPV00kimSxuDK/wCygjkQiidXVRan5k+3LtsDQQFA6hrkSW8oFlLmV6XnRyY2kiyhbwGVkiwr2zvfKm8maR1N5hUhlirE0K7NWTV53aGAnPvxTHxHxh4nJm1hZXZD3de89DygEtbaAnnvxu5hl4axcEu0FAm8ZxZD06nd3S+9F96d/8fsNq0O/Vcs3mFJwOXrgcnRgbvGcqH647EFBwFDxAnXKveiwWS9hdOg+8RdgIcgjbmJvP14685yQLGsNAj3Fhhz3HPiovIhiKxR4eGMCuBidFn8CEkC627cLPJlqBaurk3Ic8FzdPaDw/Rzz0Smkc7Mwtzilpdvm7kTAvQ2OvN37bP783MUr6Ohox83rCnTJZcWFkI5FRdUkmSR8vSGTl8IOHInEqpg69fWGJXArlNNS1hWAp6aXqAAkSjbG6nQiFo8J2rpUzEi2wB2Fg/V8ZYpFRxVuuU1qmQG/+fFPYmTk9UcbM4Ym5IuvP3HyOeEB8ubHUyvI5xUFkMPeGoKO4hqf+9LvNFih2jV2CN1dg5iZvyqYPNKIMoWbcDiMsFk7hbnLlIu1/GQ6IrP98IEPiVECRZPfyThxjhOggjqyogrOcudddzyy9dLm7FUUTDUSeWL8KzBx1Tdq8Nj8UgjKVMqIxBalbMwjYu+QQilx8Giaj03B0eoUpdGgFiy9k/d463MKpSpsFipvZeVfZ5gZhx7PP/MdwUiur6kjQOjgWgwwPnFDXjPQ64fDYZMMJ5WKalrLFG6kQ7qSrxV1HSPNLc3ipyS7A422xIiysI0JoA4jndRERpeVPAOOHTqMzo4w3v+BD209L5stIpmKwGa1wu8P4NLVH+Ha5MtwWHvlSJiZv4FNTens2MEHJODUvfjyU419ew4LivZHr7yEOvLIFdRstjtoWWKG17VTtlm+41hiEfV6GWaLC/cc+QismgbvO7m447OXJM1kZ21zU93i/nC/XJANWsoSCcTyVqOObCUHF29goybU8WIlj3QpCb8jJBd9emlSQbhRkS7inoFtJA37EBWKSQhx9Z9/TE9NYnVlEpk02boNaRItr/GYquPm7ITQ1kb7Q3A5qIMUl88Qy5RQLmWEuUwVFPZHuCXnudrpmbTZQCGflyIPMQkUmRSTaIJoagoB5HJ6BDbfarQgEOzAkYMqC2uaVYyM7MP4+GUMDKmy9Mr6JcRSi1tE3FKpigxT1wbwCz/9X6SGoJtfmmxcvHRe/sB6/ApcribnvxUOrXfs9fQjkYwgk4zA7rYK347jPUc+egsq5vYXej4yK7hCihy2u7ZrBSWee/WKwMT8DpZB/2ElkEkQ/7HyxxUei6+BreSA1ycrYHzqFfR0Kf682/xPwwW81SfR9CGQTGZForbFQOr7BuIsGC2oGghRzeRN3Pe++1AqxLA4d0mUTTZqekQSMYnACTjlKmUNIZMvyLW325yIJ5MiDUPNQw6XkEaUURfHoUPHMTl5XWoR8wuzW7vksWPvRpu3HbHYMuJynzLCbDp4fBSRhErruYOXK1rGwqqhsUMQSbov/NlnGtCxcFAV1o2hKUsqDJ+C1Lrb/WHkMwRwNGQCOGwhSb3effjBf9QEeLMLyw+frRZRIym0VpIJ8GalZOXzsT1eOv0Mdo8eRyS2LGwe6DPY0aXiAI91GwVz+2n55s+YXkyiq8MLEQjXBht33KCSsTSmpq7h6uVXsG/fQSSTsa3Uq1Bi6beED/7EQ+jo6JKYgekk6xTR+LIgpBLJJJxED2/WEU9kZGdl04jznoEgwbFRIrCdrJJS1l6lcl1dA2gPhLC4SHSxck6ZmbmOXbt2ys+ZLlI+hmRSttRF4UTIuQbccagTa+tR5ApFuJ1eCUrbvQOcAL/bMJmrsFk1QaiaAdm0gi3pjAmRdKXBEbl0bGaYbSY4PV64bB5BynA7T6XjW42Y213wW893WTHUHKCzR60Oj+32kXqykBMuXmR9DbsGR+TPcUeYW78Ol0PFIj6H6lX8/4xYqhmAGSSlCvpVrs0xPR8VsmYyEcWZUy9hdfmKPE4Fc0LrubVvbNpRqpbwyAffj5ER9f6SyTh++MPHsbo6o0rllLuzOaWoM7W4LJVEXudmOZ5FMuoMEiLHx/gvmylKGi7XrKGTRUmlECKZKZXHYbURekaHMnIrNpBKZaHTt4o7K0e1VkQgrBl7gQIWdU6ATzUcTla2CGMmaEFtn7VaAR1h6vm1oJClg5YqItjsDoQ7OsW0UWdsYHRoP9rfYZDF+JNz+ZZFhZXkDKxEFBkssLSobtvbDU4Ar80hx0Hz9zSLUCmtAubRTJRu97ve7ueFMrC6uobBvu3sYfwGu5h1zM/dxNzsDMrFFflATqcFmVRF6gN6Q5ukuKO7dqOjoxsH9imOxPj4OVw6/wMFo2/QFKsqJJCGgZR5AnBzqG0Q8EodBHUEMMAl7Ew0GsSXQL3jbG4DO8f2Cu6fNLPHvvl57Nq5F/193UKvz2YVQogUcZpLlUtVpHN5xLMk56oJHgoMiIeg7o8+/8mGyMPIw8wSW7BZS8DtdKC7Q334TLIiBopsW9pdPhqdyuP3v+8nxef2jSOmuXeJ1LmR6h/sJm4ilctIL99pd6JYTaNY4b8cKtUGPFY/fO6Q7AgTM1MY01b3P+Um/mNfy3is6doZi2dFw/fWCce8W3sAAAglSURBVPB7v//Hkr+3tqgc3tCiVh4jeuoVc1jNbWg12yWt3blrPz78wYfl8dmZCVy+8APksgmhuBNtRNculorFIsOg/JV4DnACmQ2tyIjBRAOmpiezxSXdzvXVVXh9fhw7pjQIvvLV/y1fDx44AI/HKSRcDiKeGUxy5DIlsduN55ICqOEEYP9H94ef+2SDFSsqWJEinkmvIuBTbt4d7eosjUYpPMA3QqVrH+655ydgs9vFVKk5uC3nNDUrEio8disKVR1MhjpOnfl7edrePe+Si1Usx+BwqYkTS1APsAK/uxceTzvGr12Bx+XGjk6l7fuvOagc13Rbn51dFslWcvl+eOKH8jZS8RVksnn42lxYWpyGXquyEZuZy2YF9k7aOpnLdrdPVu/Hf/3jMOqBpcUFXLrwovgUcAeobJaQK9F0uyrkDxpokhPIFjIBLrLbtjqRjK9pfRjdViEqHotKTYbdU45iKSFeS91dIViox0zTTFYhS1nY7Spgp6UczaRSSYVtCIQ6kCpUoHv0sc81CL5ktY0iCgZ9Aa0ab8/ntSGTpeQIJc8a6O07gtGRw1uOIbfenCobNtmMBI0dGjft+tQ08jkiiWjH7kdLi13Em0xmlm3tEvB4bAFcn3sNXaE+0HhhNbIqQVBQE3D815wAdM9LJlRX74XvP4/+/h0YGBx53QRgakZyy/T0DcTWVJvYbDWIpHy9VpbGD1d2NJ4Q55Nf/IVfFh7BnXe9FxfPv4RSiZItRdTqG0hksiIvT5ZSifUMKqOZHKiKxA7TPheqFULrVEdvaUkRbsg3JHqb4BIqlZBBzUGdJeoJEY9IJLCxxYxioQArtQgF01iRPgRb2Da7R6RzZAIwTSEQoZQrwONRqBU2PEpFdei0Eewhggk2DPbvE0jTrYMbTq5UkjOHozkBosmUACazjEY569p3KDnYclxUrMXmpW4RYyUygyj5wgnUlJX717z5/FvspRfz6ox86qmnZQJwJc4vKiTOvr17pKLHVO7q1cu4eVVJzdscNmRyamXRYo+w8FQyjVwujQ9/5GfEDtfvD2J1bUF6+cF21gBq4khSLJNvUZbznguCk8DldCCVyaHN61Aik9U6Uqmi9BsoHNke7IahRVnLsE3fZBvVqhsi3kEuAQcribeagLFqyYnFDqPd4ZKSs+7zX/yjBoEDG+WyOHn4fNtRbzZflZd0do7C5xtEm9sj6pRvHM0JQH0as9G45cknpu1VqoSpN1Sl00Y1j4XlafnQHB5XQHrlTptrq+7wL3nj6ebNrCqT0wytRfUMiCcS6O4MwudTn+/Tn/4sHvnQT7xO7Fosbmt1KfVOTt7A1dcuy4SlTzEbRi6PXyYAI1Ta3nPb9fqccIoQhAsuF1G/jNBLiEdXpGPo8auMhdXNsqhV1LCxoRTZ7FaSQogcVuFug7zETEayk1yVx7G6Ury+ZA4bdcqHQN4AXV43iTbOaWo8dZQrShCUsjlGU4sIfkkdwG5pwOdT6QFr9PmiHfU6Kc8ZON3t6AjuxsiAimbfOFijLlU3pAcduMWomM/LFMsyGZqCE3yM+jY3Z67KNlVhg6SRR3f3iGgNU9n7X3rki+x66lDQvPZKt0wAj8uBKYFdAZMTN3Ds2DGhdqe5VddqOHXqlOxaR48cxMTETQnslMmTHtm0UuHQt7CppsSyGXBXawUMDQ7JBGiKaZEvmUmtwue1oC1AyxgKYyq7WMq7M01kUE5vBjKDae/ChVgpqaMgFo2iWGHjaBPGViqIqF7pRrmKzg4qqZKCVxHKuN3hRiqdVgisUnXL5obHAj2XZQJ0hMygKEgNZlTLrSJjMjK6C3MLKs/dO3YPfBrO7NYbRBod3xi3xM0Ntktf3xEcn5xE0OeHz+tBplAB8YC5bFpQK5x9pFatr05ieIAt4Q64Nazhv/QkYDajZVr44YuKDk7UM0WXIutKQYP6PPQhOnr8LrmAHI9/77vydXh4EMtLC9J4iUYiQoCl4CWHBHEbm9J4YRZhMdMr2YVioQKP247O7h55fjKxjEC7FzY7XVWohaiUVtiVpb4AY7JSgY0kk2gHmlr1mnUNhbrILygiFlfpnsPlkCCylC+KdhELeoFAO6gWzq2+OVbXIiJhy2G2qaBf9+i3/qzR5qGK5wZazU4M9CtKFsfK+ixajGYc2nvnVm56ayGHE6BULEuKZzYbxDzx1sEJ0BHqVJ0sYeVmxee2WCpsARevjZ+SCQCUYLFZEAruVbbx/4yDFTfCqziW1xPCyAloW/03/vIv5HEWVZqWq/yeXbhQKIj+wWF421SB6Ytf/pwomOzo7cXyCpnMSt2jUMhhbUXFCRSFID6EdRK6fjTqZXjb3MhmKjC1KtdS8gjIHSUvkhOAw2JxwWKhKJUbmfSa7CxsETNYbBpes/xNniWzjjponE1sAY/ppoFkCfqGSWxqOarlInr7VQeVC1W5pdQRiyWUwCdVXl/88RMNBjb9/eS/EeNWQCS+JOpaxw8riTGHmfq1lHlVgRJl/yjnFgz0CsOHUS2DOjp23zpy5U1J+9yaRTsnDINAVtJ44dLJJGKxq3DZjHJ+eto64fep6tk/5yiXNrbIEm+cAF/50p/AbLH+gwmQSsQRCIdw8OARODRBhc9/4bNyY3gGi5O5JuNNv9/oukLhSCDL9q3EARTC3kDfQL8Ea0TziBtYoYyW1hqoR7xzbIfU9jdrOs02h/KzzQVAV5KKSNgSRDIzo3ZkkmRtDq9oD4rkruY/m8qkpERPAStqUJdyefQPD2y9T2ogaLUkIakyBf9/kzu4dNcGwkEAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAAgAAAAIAIBgAAAMM+YcsAAAABc1JHQgCuzhzpAAAgAElEQVR4XsS9ebQtZ3UfuGs6p8545+lN0tMsJGGMMQT3cuK4HbdZ7l7OSsLyCAhDgxlicBiFGSQEYrAkkLCEJGwkIHbstBPHQ3q5Y7tXjGNDMEJIQiA0PT298c73nvmcmtq/PVTVefe581/3p/V07z1Vp4bv298ef3tv55de97PZkcNXkg3H98hxXMqyjCaTSf55HCcUxQn/XakG+ec4j/CP8COj0XBMSRrz306aEhH+EUVxRITTHAf/K+5HGX88GfQoTVMKKMmPzjVDPi/wXHLzrzhErsufjyN5nnHm5ddL+PpEQaXC9+K7OXJ+lsr5SRLTPfc8QK9//evz7z344IP08Y99SP525Xp4rjRL+fthrZqfe3htjX+fb9X556i7S1maUhKNKfDk/hk51JnI7+NJQmkqc+T5FXJxvXqdkjii4WhMcRTnx0ejPrXaLfL9gE6+cEafJ7819bsdWpxt0dmNDn/ouB45nk+rsyH5nk+u61C7WeElScmhOHNobb7J537/yScpo9CWS74PArDLz8+tUq3eJi/w+SM8WJImShApJVlG0SSmoOKT4zi84BhpgkWW30EANtIYBKQEEEXy8UUIAB8Pe/JCblJ8f7Yhk+45Lvm6xoFfyQmgN44ocF0hAFw3ywgE4Ho++fgCf8+lwJf3SZUAJuMR3XX3fQcI4GO33izvnUx4QVzPJU9vHNZCfXaiCwlgsL9Djr6/m054gjEb53bknTy/RlkmxJcqsYY1WZQ4SWgyjnguO50uDQYDyhwiX59ZJoVof29AWZaS78l14kQW3sbKTI0C3yPP86hSq+Sf45fOxgs0GA7IcTyqVud4A4d1mVvnLW/+pQzE2R/LAlWrNaak2YU1puLywIaOk4gq2F2lMRqOKM0ywoXGE1lAlxxK0wlmnf+OlACYcMrXVSLqd/bJ8xxKx0PyfdmxZQLw5CPKHJdSR6hhohzJqdTyp8FxjEazSS4TW8GhjACSJKI777znf0gAeB5MKgaI3ka73aIgqJKTCacD97L3q4NQ+EEzGkVy3HWDnHiMAPqDEbmuS1GUUJIod/B82t3dIeynXmkjVQKPibpaDSmOB+T4AUVjIYQsScnxXDq2vECVinDm/d4uzc60aX1ji8C5vQgcyqUJhXT08HL+Hptb6+S8/W2vyxKdSBBCTD5VPI/Zx8LiKp+cZK7Qt04uqAxs1MZoNKIEC52C3RU7mCAKeNfhReVzEICJAL5miQB0NvPrzrfrvLHB8bEYHrmUEF7Eo8AlGscJxZkQrY1En7HVxA7LsHmE6fCtZNKwWBcjgNvv/Iy8bzSiKJqQ62QU+A7vUj/waDzs8fGlFREBFd9jMTkZYncmLJIWZlpyD6xillKSiohLEl0wRxZpPJaNIfvDI093/Pb2Lo3GExqp+PU8l5pNuSY/G8lGnfRL80xElxxaoWgypsFoSFEi52Qp0XCckE9yLoj50NoyYeHB3TGcD9309lwE7Ha6lKaYLaH6haVDzPZtQJ7wi1eEyrHoIITxeERxkjIHKOsNbpaoeCCKoQOAFLCDszT/aWKku7crt4lHvGgYs6YDBNABTLa6FGNHYRJ0Uj2/kM+OTmSzXifHdS4ggJiSWCb+9js+d4AD/MaddzErT6Ihs1v87jqycKOhsHMMZplEVKuLDmDijz+ryWd4Wt+T8yZRUohL8llfiBJ7S5ccR7gL5uL0mXPMqrGrbUAc8QbD8OV5vLjgzpjTVi0gz3VomPnkxANe+Pz7kx5zh8lkTM16wCLdq+hzfuh9b814i+gE+35IO7tdiv5eUVpaPszXAHsHIRgB6KNQqITQH/SYwkHt41FBmSAAfTM+bhOXP5kqWXhxI4B0MsgPz7dFTgYBZJtMCFRE7Gs8Q4xJwXORR74nRGsyG7If3wC3ClSnwcTL6jh0++13X5QAcDisiDIVR2Ma9Pb5K4PeDv+E/EyVY9aZy4AxurmCGVRVtjpE7bpslE53n1kxhuvVqT8YUKUq3+31R1SthDQay7zt73d5roOgQmmSMVGkql/wCX7Cf1USjzxjbbhuWONNgs0FkYTBGzOOyXdk84EDzM7NFdMPJZkJwIbjkB/UyFUt2B5yoPpBpGdiEXKKxKLEMS8wNkakrAuL7VCx6DhH5t7lyWWWqOwRD93ZlQmORyUCmG/xOdULCCBWHsG6J75TvAFVq6KfXEgAZWvFDwL65Cfv/AcJoBL45HsOxWCp/WkCwLU9tULmlkSeQlwwv8V9ldjAfdpsOThMACZmOz2xrFISQgHn9LyAOp0O7e7ukusFVKlUp8QaCAALCe5abcv7NVLhgrEn16kELu9wiLfJoJsvab0RUjIeUK3R4j0O4i5veOcD73lLFk8iqoZy4aBSJc/1WVELqoXswbH9kVCS5xdm4AQPBgUPSmCWUa/f4+9juCUzcDKR71b8gBKdLCwuFh9jb2ebzcADBMAiJyAfRAlxBBYPSwAvr7tqOB7zIbxgRUWAaeWVis8TKkMoBSbWxQjgXTeJGXhsbZW5BkxPx5Hv9LsqorBr97f4Xat10T2EoIXAW+02WxD4fqgK5DQBjClOYtrc7vP5UABtbvhaJQJIlYMOhqN8Qas1WZOqKnwQrdjpYuKKNeYkMWHhbUBXSdU0bzRMbMlR573veGNp/xA1W02mSHmZ+fwiUZxSfyQv2R+OeScHahvjBUChWFgQQD4S2RjQy4Rpi0mX2+VZyv4ByLCd7U0+buwLv8/Pz8heqdbZrMuSiNxKhYJajcajCRMAdtp4EuVK1oUEUK34TNT8sio2oX1/6lOfPcABPvzRT/B5i3MzfD8QQLUK2TmhQW+PqqEseL+7xz/TLGZdwZTLeq1OgXKgZq1BWRZTmiQ0HE3YnMY4ceK0XGNkPomp6af2zCwTBThDEAS8o0H8JgIHak1DImLh85EllCVjno96WM2JCpZPvSbriXkGAcB3YAvjfPi9v5rhJCwe3CZ4HCwSxtLqIdXa5TaDwYTZbXcwrYHy8qpfoNMtCIAdQsqmvdxmTXnX4x4gCuMAO1tCAONhwb7mZ0EAWIRQ7HrmPj5VdOdBW8YYlfSOTBVD35fVhm5gOgBeXJTOi+sAt3zs07xQjXqNFap6vUZzs20hHuNaWLy+KIQ72+v8MwyD3LCdacv52IkgDgzMSRTFbN4+89wp5lb66HwOOIbnh5TEI2q253mhTOnlK2VZbhrvD4Z8zThKCEzAiHo8GrKOBRPXuJ8+CM20G8z1YIHVaoXFBH2NCcBOxE+zZ6E8hUrxYa1OuIFXaeirTfsHusNIzJ0spb3egL1hEWxbNg1153tiwsmCK1VAacigaUNT7tOwP6KhylzcaF4VlmqlmhMAZGylFvLLDHQyRpOIzURc1szNsFrNiTff+UpE4FZ333X/AQ7wyU/fxe9Xw4K6LrPZdlveOU1ilv1QPLM0psl4THu7G3Tsksspjce03xERUVdRikUzEdXtiZKMceLkGSLHp/E4Jserkufj+sUSQIKJMxGWkkcQA+bIwqcggIoHCwz+g4gVVehsXeVK2FaFyCOq16r5omPOwEmwBKbDOTf92punCABsZ6zkiQmAODCZX2/IZIzjwkTh3aFK0X5vzBquDScZU5xmFCVwqGDxIfPAw9QG5t2oXsdYZOLGxvP806cqzS2IklWvh+Sq6eWHUJBEBzACgMWCgV0z6A/4JS9GACYKsIPvvffBAwRw51338XXE4MgoDCvUasIXUch4XDtWp1aoLnEQgC1iEsOLCNGQUebJfO3trhP0rElSofMbolRGZc2VxFcCGweLD4JlgsZ5E7D1fErZPARhdfaEY9qAXoEx05ph7mUD1oQppvxWqcN6TaomvXPrB97BBGCKRiUIciug7OwBIaTJhBqtGXLV5ML34NmqhKBUaKEycTYmoz5t7whLd/SFyMloEqUs43Cu7Yw4BWsDq/NoOBzQ3u42LS8eY9YVBC7LQ15k1yU/mPZE2k6DKFvfELbcarXFTIL+ofMRVkUx6nb36b77HjpAAJ+9+342eaNI3K6Yi1pNFnF1ZTZ/r+6+sGH4BuA1ZblqTjJX3eNZRkPWjRLa2+uwTIYusbkFAgCHlLnC3hHnGJhhSnEED6GnDjS5ZRQPCVy43+vQeCL3BlfEwPzBqoHuwxaW57G44oU3dzITlczflAMvysh57zvflrXqhZsT2j8uAls3t5vZ6QMGzpLt791Hws5qDShLFfIrQjQ4Nh4PsYyUZQHFUZ/iifqu0wlN4oxAO0NVgPhqqmXvd/dV1hWBnUYoGq8HM9D12HsFXcIPquzXx8QxWwtwP3FF76tDCdwAnxnbYyJ0hPN0ux164IGvHCCAe+/7gkz4ZJzrJiZPDx8WnwjmdFd9/Hu7+9RsVKjZaLEl0OkNqK0BouEoot29bSbg3d3CD7C3L5wOIhJKYjmmgs/jyYgXbjwuOCk4gxBCwT3LbB7vWq/X840b6AY1l7vr+qyDDIZDCsOQ1BfG13Te/Y63ZFA6bCCStNfdp9n2HFUDoTLIf3HkyIMMx30Kw8L7hsgWFrPWnKfJpNBMxfeeEjimiZVUF5xfFl7ECYIuKY0j+d5Mu9hpceTQGD5xXWgoc+JHEGquqtMlVhMHC763I/4EU6LElQzLo1CY4jSiz955MBh0zz2/yd+FPiM6Df7JOx87eiyfo25HlEAQAEajUct3MUQPxrmNHQrUa3fu3Fb+XVgsPKfqiuX7JWCdwjnEXyJeVWj0EDnDYT834/AuGCAAvCN0NZsLiGrEYIyjQJ/b6+xTxWIlDvSDMFfB8D3n7W+9ke8cwgHkuDSa9CiOZTHAdo+sFC8+7Pc5zLqzs5E7Plzfo/ZsmycZwxajUm3p5MmL9XpiHaROkLNk+NixGBgTjhwStVsL/BN6xXgkJhYikBZOhZsU98RYXhQdAbsPAwu2vy+L0u3sM5fAMTOhTBn8HxEA7oWnhiyPY2G5h9Yu5Z/8bFDC4oROn5aFnUQjvgfYvausFpul39/m41EEZw92YUTw64GVIgjE8wHfRwydQaOuSSLisuSC76tiDPZdV5FkATmsh4c5UecdezDjmBfeRlhtqPvdI1/d5lAucZ7zljf/fOapN4lfZjKkRBcFoSEbzVqb1hYO8Z/9kqdpNBxQk+1mdQ6xiQmbNKaEnZgqE7FAWGPMHsK0FSibGkQq6QgVtTR44vJ4+oQS3TFxnNJ4PGZ/RUuDJHiZmfYMT+bW1gY/497eLis/2El5hM6eJUvozjvuPSAC7rv3LnaAxTGeOqMUSiyUVvg8gsJ8Wl5Z4XucPSuKWLfb5wnu9UY0nqTUbtbZ9IvHYhlgt9qudBBcYkIQpS9JhABy4jJfRUmX2tsXQsIGnZ9bYnFcLUVkwZ0RRs7lu4udLoSBIfdi0mVFvDycN7/l1RmlcgJMCDyQEUApxkLxOKZaTWxci1xVXZ+qsBIqFdZOoT33xhNqqRcqcUccDOJYRwKdICLfh8IkDwYOwCwRbNYBywN7L0RLEmvYF4AK6CQZdmTCBICXWVtb44kH20QMAIsNBRKj09llsQUfQj23fUUHwC684/aD4WAQAAacXlHUFz+82vJpJnoSCP3osUv493PnNvl5zq1v83tCpCUa5BmPezQZKB4A4VxlPwYsEQJDsAw+mIyJF8PkN4gaC471GAxFb8CYmxFffikMwASEzYRFZ11MXB15sGpY8ttA9y4ZCeS8+W3/QrcoNE+ozHDdCaKh1vAIO85iz011I8YRJkPvopo5AkOgypEBPyBW2nAuyajXRLanaYUiNTOjSUIj1RkKy6JwM6vKwYSjwUSOKo40XrCwIOKirNlaLKLX68hOUQLAJIFLygQlFyWAB+4XAuj1xHIRYhPRtKd2Pn5/yUteyZ8NBrIw33vqJCtZGEKcmB5EHseURAkrsRIWRzDJFG4NboGA4SNQlm+EAmvEhmsWFAiwpK8xd9Lhe+LogZ7klawkbIoRe03l+QqHnIYEfvVdP59hAuFZwvADlyABwCpq9UIjByHUw4CSGLH94sb4jkMBWwN4gJmZQolL/TE5imBZWhBdYjSEUimTOpnENFTgA+7vB9gFAVEG2yhlxiALgR1pMW7ILjkAMSD3NzKDiQkQSkaIUHLYWQNQ/PIKNAHS57N3HhQBd90truDRaKiyGRxF7jUYivsX46orX8I/Z2ZELDzx3adyAtjdExGEkSUyf9jlpuQZAfBz2nkWXtXoJX+nJAIQxatWPOZMMG8xdra3qNFoSrTPlxCzDcQTyqPXKyyKMtIIirzz9nf8bObqxPCXnLEsAFNrcZl44lE1VJY8Uo0Vyg2QGUmFJwy7a3npKMXJgDy3SpkX5U6MZqvNqJrJCA4h0VTBAYbDCf/e64iSWKZ2k5sgDmORoPrhoMcs1WzvekNBGFHEJhwGxALYLHaX+cwNMYPv3nffQUfQxz/1QVkwxkSI8jYZiakGAkgTmdirr7qWI2+rK4t87DtPfI+ajRrValV69ukn5RpweqmTC15EEyXw84tFBTc42LyYuDYsnA0pWTM5X7KcwqoEc8YTmTfDU/D1zFl2gZ/EFGOI2HarPQU3c976NhBAsdBeFWaXsAuEVB2C0wXU5VKgEahIFON8+A5CjT6zl2ZjPjedHC9mccL/qecJHCRTAoM5ZxCuNHFoOBpQ4BZOHgOXwr0LAsALQxwMR0IsNnGB4hKYfvWpoIhikRCpxHXxCvWmOHVm2zP0yU/ecUAJvOW2D/Bxh3xK8S8e03goc9HtyUt7bkYL8yJ6rr7qCv65sbGhWrxDzzz1vZwA4tSjhIM0BTjD4v62WNDaDWCDL8Jjik0AXcD2dC0sInscr4F5WsYIgJuOBceIdzRxZAsE7yhELAgfgJWGzgO/65ve9GrezmD9GLVmaqBYSml6pT2vweZeqtp5TgEJWKHHLzI7I3ApjDiBLM2I3AIoknEcW5bJ9arkB0LRkcootSb5s+EQXjGHYGmY2QQ//EhBI5B7GDOzKnY4FiCs1eLv0BdsB8zOigKFIM/FCOCmW97HxwNfTGJWiFX07GwXQa5KIFzmZS8VXSBLC1fw1772tfz9mYiwVEwAsssNWY1YAgYU6rBa4RnBQkTxhBDCNnc3tmA1DHN3uxGTEQA8kTJkTrFJ4I4vcwfcEzoa5rLZbjEIFQOWAxOA79WZbWOE9YDZuucl1GgXpg+O+W6Nuv0tSsYFy2JXZlYjygAn96hRF7aIqFOSwqUqOwjoFhkuQ6TgZoXelLsoDVqex+xhN0sYdTIaFlAzxJjsXN1Z5R1ks882N2vXyYEdsby6TB+9+WMHOMAHP/pR/vrOzjZzDkbYQoEDQU1SctwC4YTPrr3qWtrv7NNlxy/NHTKPP/Yted80pc1tscXB9sHuwZEs1G7oKPjtcxNacYuQ0yACG+JoA6HB0lDwrjrpcm+oKoG47/n1dSbyXl/WFPEB3BsKaqOEL2SCedtbxBFkA4sPR4Trj3PosO/DWwcTpcnmR7+PyZXJmIx65BJYlMT5m3XsMtnlfgDnhly5UFLAaYTbYJH6aqIA4hyGTbFAeCIKjZr9CmqOwR3a64p5ZRFaTFDFC2iSRHkoGzI3A9GV5CS4HDCBq2urdNP7fv0gAdzycd6HUJo4RJ5Aj4D72clhYHJjdYUrGPWGG66jWlhjLra1eV7mJYro8e88oQRgzByucBFxFpcASzddR+ZkxIqexT74duoKhgxHfAOKX/l4p9cjH+KUsQkZ9TQgxxwsSRiEgzEcjagNwEoptO285hf+VU4AoJpqvcIuSB5+xAtY01hBJRBZZNo5qI2xecC1MwTMoVoIQIloj4JokcszRWdC7WVXpbF+jhpGRNVQ5DSGwaUNeIHPEA/odUQjhxXhV5ADUOgNBZ+RazAgxReXKXQaDIiCm957kABu+7SggpGsgXfb29vOdxz05GqlxsGYMZuhAFfUWNt76UtfUkC7FYcAmfzwt77J10PU0oJeYbXG7NzkOnOztPBWVhVk02w0aTjoU63e4N1rAyINGAcsMhaY5zlJqN/rlwz8UvgQSG1FZEN8gwAKt1BGzhtv/PkMzhiwIvi/DbmDCzcX6+Qj+Kyj1Qx5wWtBKEBRZcEptGVluQYU5UVnhKzDtrCxNDgrjAAm45hggSACaEESLygIwNyqYHsS50doBvrEiOPx8AlAVgMJbC/F0PZETCNftVsznXE+ds7szCx98KYPH+AAt336s/ymUNRAAPApDBThhIikjZ7G3oGXAEG8+PrrOSq4t79Pa4dWNKsqolMnT/JXoCSabgK/BAb89o26AESBXTaoVqvV4ntbvB5s29zksgFEx4Fi11G3N/6GEg6OFSBErY49e94oGQsAx3WppSF9O+a86Q2/lImdKgOKgmmoacnRUJsLaWauQYO9mNldvV1Kx4qAaxMqNcQsfq+1GuQr1zBMPSwFEACQsNi5rJ26Dg1GEn9IUklrkpcCWxPFzsxAWAYTPdc0anZJKau3vARTjvDSFiuwC7daTfrYzZ84KAJuvpXqtSb1+xABgKtNaDi08GvxvpE6r8wjurwIy0ce+tLjEjMAsQHniHHixElGNmOYvEeEDsP0FPk9ZcKA+LHr4fNz6+fy9Uk0uorFhovZgkFQmNnZ5cJtXeXNaQ6zWi1k4oC+Bqha2XHmvO3N0zoAdqNBrPJ0JrD6JKHWjEC06mB9pVFvgr2L3I6jklcDOgArOdWc/UKZwkNiJ2M3NJvi2DAgye6eaNvAASKsjEgYzDy4jZnAIJs1opYbfXniR0Z9dQUbChny2wIl9sitZuOiBHDTBz/MpzCOnwGrUCJFiQU20AZSyzAsIAMUsXG1666/ThY2SanX2eXw64kTz7EiOjs7y+lffFwvJsGgSDKCkoSWV9bo9KmT5BjMHUk45hJlB6Oij1WuTxSoi2SQ3FeiFhUSdtK/j3zOzs5IRBQ4iUaDuQsSXTCcd7z1l6eUQARQTEkwTRW7ABQZqk+9Eho4A04MBB6AoBXHTK1WoW6vT5hk189osyOesaU5CaDEUUa+V2GWDg5Qdmzg+HgSMwvHGGr2TCWo5PEHTJIln9RCRbjmgIqMuurG7Q/6TLRlDmAKKbKGbv3IbQc4wK/fLLmBEh532JQ0jx2QtTag+WOAyHAPvxSYuebqq/kY5PTSgpin586dy+325559VrANCYCsoqRBseVcRJi8oxGDWMsoHoi8PFFE0VhG4PwcgIalCEbts2JtXAHvvrq6nCOIecE10MhhdYTI3/suyQvATUCWiLHn3ihH4E+521KVQ6YexpUlfC6bggr5bmtqFO+QALtPZOd40ueomuOJXgDcfZriu4ZUgcUM/JsokL1Bj/bVOyig0OK8ifrdJQtHACy8q7KUtnclAgd0sgBCCniV2dBhUKE7L5IYcsvHYQVgQqFNg1gjBlFgWKIOkDidHvIYXfH76/axzKeqIpfq9Qa9/GU/xN/d3dtl5RWm8aOPfpufc3dPcAtiPRUQLjwzL6CCbnBGVMoC8uBddKBYFuBZft8esroias/PUrOkSDcghvP5ySjW2AY4M7yzzofe/04hAFUuIDsNUwaFyZC3OEcibxLDZ+ULWLkcjy6v0yqJhzQtUsXn5+cEt1+FM0mjgXHG/m0mlqqibwnBC9Entrcl3s4p0HkQBNQm8rMSVsUxBWApfqYpbSMM7CFnD7kC+Bz3UvEBOYjU9yShuz97MBbwnpt+XeHgavuXspkixUgIPlKzfBxixQ9paLyUnkujviwMFMSXvPgH+XfsSjzraDikr3/tb2lurkXr65ssHmHiRkkBovG8WaRIc0qa4faAkLRdDYV6Z2899xIqFQkXdhEunmdRUx5Vcw2DqC2qxgv693P70Q+/d0oEMIhCKfIAXAksMUYse3qYjQ5ApFN6NEwI73onpfYMwIqCVStimQ5NVKlBuLXdXqD1jSKYYoplpVrhCZAMJIBM1Q7XFGuYVRAplmLtAL+YYrEdgnw2kZYrqpTRPRcJBt12+z2ykAofwr4c9jsce3AU6zc3v0yukzIoxpwyUHy7HYkMIhsKA4GgSy6XANgVlx/P7f5HHv5b/uy5ZwX8KgARAEY0o8lvsPvYLCysUhn2Phj0GCQ7v7QkxDWJ2EVf1kOWNWPJVqm736HRGDmXLgUXBIqcT932kan1hJlh0asLlScLdcalUCW/BDsgEIyAt6xEHJquA7YO64eTEj3oCgqELGUKj0Yps/kwNFw9QrBi70MBNIgU2LJ5uEwLtwgZ5xhqXoLZ2eyLV2wgonxC+BcngNvveoAadV9M2hTJGUVhijQtdqk5tU6ffoFGox5bCtFkSK3mAgNq8qHi78d+9B9TU/MILZfg4Yf/ju8xGuP8wp2LiCWjkTReApLCPVKd2Pk5iYAeWjvEooA3lEYRQbDY4dgowxI6Gw4gJmxkaVu4Wh/Suf3TtzIBYPHY1ZgTAEyWQvHBWsK5gZPNscAU7yLtSJQtfpAS2gBhybwoQzRi7yHCmhbxA8ycAahwZAzEvm22hLIxAKbEc0EhYkw+IGTRJCcA003m1MeP7wB9CxMT9jDCxmJTa0ZTv5fnJn7uIvUBPv+AgEIlm1cI26wAJG1YGjrQ0RjfeeJRTrbcOLeeK65LixILWV1bo85+lxXWq6+6KjfrlpYkHvHVv/pLIcYkpW53SO1ZCTDBJQ9E0mRSWFP1dj3H+h86tMjXMiUQJh0IMNciMqKeYhaN3UdKANiAFfVDIIupUW+S8/l778ggQ4d9oVx4uezilmZlLMLi+GUCEBtc7VlO3St82KBOk90WwavXhZ0L0UV5uRlXWWCaFV698xtbVPGrkvdnUSIknyhnMDZppiSuaUBWeAjByoVohACgvGGAO9xxkdSwz99/Px+HAmkhW7uHYQnsufFzZ1fs/LLpu7Z6mH0fIGrE7DHgeTRAxg1qJp448V0+1u8PqduL8uSNKB6w3gV4mY2wXmRGAYWMUfYO7iCpFDyayHAAACAASURBVMirnP2WTPEkY6AOQuJQmtvtIjv4yJHD5Nx99ycyQ5PgwiLDZMmRCmaOHWTKmtsWSg3XAzCfspIfU1gJRCDsvvD78wJVASFXAGk8oSgR2YnsVR5OEfrc3ulI/gBqCoBDwVHienkGcu5d03AwdBbg5zHMkmGlVRWfrpahgTZ+MQK4/wEhADhVzNMJdzPPRSlBEyBQDMvY4WIQ8vC0tiJAVbjFzTqB/LU5PX7JcT4exxLP6PWGtLVlFgGgbDsc+QQWwQYSwhER5fdSaLtxNTsncH1KldBzp5OKh0OHDrPzB064NcUzyvcycu6446NZObAAZ6v5zJNUASDq+OhpVYqR5u/5Zr4oxg9arYEu+PqM3ZdbjQYyaYiGWawByjwUR5h/lh9g2a84d78rXCnLPBobCgkxB46HS1wAw72A68h9VJNPCzCJ4QXrjTrd/omD6eGfuv3Tujgy+QImkXuUy+JY0QwTbwa6xPMsLgorl9eX64Q1eDxlIhrqu1hcEFne6fTo7NnT4mjKiJ57/jlGTNnz89wh5qHvU0MtASSuWK6+3sv0Eojkmqan4VCz2SB4PrHpYLrOz01bCM6nP33zdDQQ2i5Sn6tVxuxjGP7e6ttw5s5OkZpUcSWixYiTRkPKo7BogCauk6H+6RI6iuUjzCsUVUhICGROod74fWNbbHp4BFO1hYGuYUdQKfcOmTE2TD+FgggRgKRXc3VjJ2JnQ2O++yI6wGfu/ixr4L1en71lrD8oVG5xabF0D5mX3T0LSonfAGN1RcrqYJi7N07EQsEIq0KYxw4f4Z99tt+HFPgBbW5t03ef+K4qyyU4HkSgXhOxFlgFQnzloE9CNc5QcqipBaDwFZjpVtQKHkwEsMqmoPMpVQLzCYTCxPhtsELI9FIpEl/YK1ArOftn9ySQvYBP9aiFQgSoL5DGmgouV0ZgFTsCGrbDNYeknMxY/eqDkSz2Fde8iH9iwnb2RGaPIrhkFfuPKiTKji0aZho/K6paoAKmI8QP8hRMp0H9HIw0iuiBi0DC7rr3c/LeiN6p+1UMGYeKrN8ChLq1vcsBp24PuAdZ4LVVQOfF3bbXF/HG+fuoITSOaG1VRN3a0lH+CYSRjwQSRfk88q1H+R26nUIHqJWScCyhpIzqwXXAudlNjuzmC6qEzWuGMzyNB8zAj976oZwDsNbPODVZNAAiy0EJcEOwMkvbNqKpBj4hQIKvoSjEzNwC9bgsSpHxYvhg1MGTvEAARGE+KYZP5eqxS4+Tryxsd78DuCllTiWHju1CRipbxi7Fwzc0WYJj4eoKDutSMgXnmAIGNysTQHJxArjlttv4uEUs2WxVfcVyEOyd8XN3R4jWiBO/c7TPGbP4OLu5SWGloQWfZFJnFchqEbv5uTYdOyJucmyKJ554mPGUzz7zXH4r4P9sVGo1jisAf2giDcdqyIbGs6ZJXqvQvoOKYUJsbs7NBwrGdW6+5SYmAGNXUAItHy4raeQ4J4YsZlOwSGTA5816yGy12WhQXZXACOwzRl6AAkeUlfZBGBGCRCCwAVlo1SzOerOVx8oBqoQGi+xiR13GmKSREo2xYOTm8QRmGW1uiyPJlEAoapZIacQG1n7P5w4WinzPTQIKBcuEc4WLM2lcAhzFBuIcGDta1qYcTbVMH3CnDT2+OLeS50JsboplMFDucOTQCl11pSqGSUxOJhbAU4otxLmbm4WSCNALRnhBqb4KzGvoYi5RvVQ2j9enVYTYkbBj45nnniPn1o99kAnAzB3gAYwAPPbbF8Or1Hj3drRAgh1p1lBbUJwmsy2JGGIEFS9X+CLFoaEQBLR3lv/RmCLF+M9qYQWDP/EF/Brt7++QF4RUrQGN5Ag7teCPomCNhYH9bWihCQtkAVtnOky302UigUl370UKRf7qO9/Ft51fmGdzliFhCoRF7N4G3LoYhuvr9yWJBMPCx/yHKm6GXeTjiqje3pQQ79xMm9bWxHJAnaDrrr2W9vf3aFuLT+Dz558/ld97v7fPQaPcvatHDBqPpzCwLCt+TlFAAx5RzKcNoKOcO+64jfMCbKSx1ZuBUwIFCQvPVn1mSeDcZX+yVvFi9u+6tFCqQgVXaQ5b1u+kyFvTgA3YFXQFjKomTITKzvHZCClajAFEBRJReDjxwUq/oiYEe9OEteNeJufhtYTCh8qiZppaqBiLefMHP3IgGviGN72Rr2OmK+rzWfLJIS0Pi+N58EknDa5Ww1CU6yQC5cTPBa5qmUEV2SCbm2f5J2IT5jyDdfSia64jjqFoxVS803e/+/18fZBxbKk/5eQR1GUy6wrz0+0PRCw4xGu4qdzICnbYBZkAytEo30WatSzK9tY+V8gwxWqSaHpUCbAoi4KgjMMvYztZblAATUxNhgJpO2eCMLPmIVpCakWrkvDEMSAEbFhw9qi5i9wClHCBqGrPznHufdlXbpBmk5sguA7gUqXiVqjv9753v/cAAbz+DTfyeXCyYEHnFxbo8KHD/AyLJevETDSroAJXq2UQgRtgADTTUX0E0DhD8bqOYCn28p0IsxYlYKG0Eh09cpTNuEZdRA6+982H/y4ngP1drU+Ur6BYCxLtF8As1zQaDBj9hH97e1sEzx/OKosDnO/ccustU2YgTAjITdTjxEvNzEihqL3dLRqq8jUscQUcA+gBt6/VmwQ71QZSonJ8oaKCISqMrcKta+ga1BLgFy4BFlGyDiMGqhbVZjLZLUEFASaPllZWWc+wfDy2AtQ3IGCIjLAgnDcXVPIqpuc2t+mzdx6sE/grb30zhfUGB1igtWPyRKuHHC0qptnCY1KhZCJ72cqUGXFXqlX18xNzpRw+pz5+/k4pO9jmDMkg83MzND8/y4E1RASfeurxfE7PnhfvoxmGluq1sw0HkoBCDAkEHCOCrXNzwnWatXAqCYWv8q73vnuKAFDjpqU5gGY+sF+c82YNCVyqTsVerZg2d7Z5sUOFK+Pi9WabKprU4GhiCMKfUBiZalGqRJWaKBVPlz08f19LogFbanY2dlOigZnDR6TKeWdfdkWzOZPnwiNqiP+63S7tqjkJ4lpaXGLM3y+/8a0Hq4TdeitD0Ps9pJVlhFjF3Owsi4RZRUPhPlhcjlFoZI1hdGrZYC5YKUSenuoPMBNHkyHv8qEWfqhr0Is9jqwUu+wIGo06tDA/R3OaYod32Nh4ISeAk6dEycVil03Tc2dOs1XF9RhKNRpw7sLiEq0dOsxs1NW5tws6773p/VMEgEXh3RxWaWlB/MaSFyTyF8MgUXaRCaqCwEMH87eUo5Z5mmjKDpB5TqBwOVdQ5HmWAlGkLk+tZgkImI162GJolJSWExaXZDH1+7LgrmYRNeri3WIdoD+iw0cvyWsVIiBkXMFcx/9vBCCEKfeSKqPiZAIhYACtgzLv/CzKbZDjYDvS7oFJhccQHrz+cMTuZYxT52Qxl5fEEVTxKwRkkwWY3Cwmv1ql+ZkZevbZp2T+SxXUeyOZr2V1TK2uiuMpRZheazVeuD6WjHLi5AkO6JWH8/4PihWQjzTOvU6H1sQ+tQFwIcZUcmiWsuYLFoeHL5eDTanwA1hFEYR8c7wBUs8ss0VBkp39IgNnpNm30QQmpTwmoFmtsEpBmFFqRDOeMIvHIngaS4iSgGZmmixuzAowNozM4V/+33/1InUCb+V75IorO6/kvldccXk+D0YAcOzkk0+AteNv4ZLgGsOxfBeiKFfYNO8BShvD1VB7wPOo0RBbPYmHbB1BiV1YkPk34sDvgfpINku4CYg7hH8R9wCXQ3mZcigfcDJ+Ni7YFedKMz/n294hRaIAR9anzQMXhw8VaV48+crSymVi8TnXuEeIFGyvBDjwAhSUFIo1swWII5R9qbdqBNPQlCNzO1sKFb+4NYQYjXI4NFymVvBhrGHZsSqt+A7bwviZAQmbUG+wxbEDjOWVo7wjAYX6uV94Hd1443TDiLe99Vfku75yHKCHuCxuRscvKwhgT4kUnkBdtZxo4NTZ7/R5YesN0Z9U+vHvAHNgNOoN6nSlyBQ8mXmeYyB4ACFC40TFHhX8ANH6+fN09qxYEhjDXjfXM8q1A3juSz4Mdajyd8JKQM7b31nUCYRcg+JnnrOVZfF/A9kDGSeaZIETsJ0FlgriAJsuB00A97owjSvwfFqckxcbRwA7yO+QvfNzkpBqI48jAOWqyiHY+XAI796IJolO5mzhpzcCGg73aNDvSBEGq1TuiVIZVJv0/vffdIADvPENPy+L1JxjU49FgDq22rOlqqmaHd/pCltH7f8MvREYBynPxBh89frB925cxULpqN/Hi1OpsgiwIaV2pT+AjZpW+uRn1/l5/PHHcmLH55G6myWfYVpHczWnwYpJ4fw8IHfLrTdnQPHagI1vPnbDutkxmHBMAKWHw99c+lFRN1OZp7BNlRzhX+eRwXOoNXbR3cNFMqOkOnvAwWkWC05t1mQSUNfOIhLwrVQq8tf2nryobyXagZCpWLaMhGBRHKGvfnXHETaL+kXves+7DhLAG3+Oj8/OiSMI1chg2fBjKw5RJk8WOdX8/5FaR/gMWUV8DLGRocQyJLlGRAN3PFFOhfo+DD8v+WEAJoWYRIDLzM0yyqqpSvW3Hn6Er4MAEM/FxrlS6dhpOW8YQQmXl3L+8S6f+NTHp3QABkIoy7SCiHlShio9KAhnA1olPHDsCEENv1INQdQDNA5gWALIqWZbzBJwA5Hd4qeG7DP5is/isRAmAlKmN8BKWZjVjGJ9jETLt4DWQiUaVAuDgwtxg2FfJsTVe83MLdLb3/Hug44gJYCGIpXg17dkSpVGfJ2qAlj3csIqopGmcHHNv7EWaoIyrMNV+BdSxDBEcSuOA+cPT6PVJcY5Za46pwUi1jckBxEDhTKefeb7XP6O35N1LxmwLuaVe+E+0H/Kw/nYbdN+AGi9BtgY2a7Vbwy68kImU01GYYGg9Eh12ML5g7CwvZxZBxAnke4c1M6FHZuRx3EEDPbXa/Ihag7yBPioVqp1jLKEfC4qSTSr8Kq5Za0s5jjU1qodvX2UiEk5hWpfy7ntatFK1w/pwx+96wABvPb1IgIWFxa5rAv8Iaa8AsfATSRYLxA9Y2dXFNbROMqLL3EPA7ZwMraWsAFQPMJkfF7lS2HucLsb2APfA1gFcYdAoVv4zLKr8LvpX63mdHLOqVMnCIBRcO+dzbNTjiukwvEzMWbyAgJ473umS8VWgiLdaGO9qG+HXbO7JUoPvFzlgXh2G7saMXq166GFctMopW5XzSlj5jkPMf+AWgHlJgiUqbnFpVcTQjMLTtlQZ0q1JUQRNtVrBmib2tmeaV6iUQnhOvAgSnXT973/4wcI4FffJbGAZiNk8AREDxDAsjAF63RVHKADGAaUOeN0Vll0YX6W1jfFXEUwx+oDLqq+gkJarVaVd7fFGYDkwaMCEmaiRAhMN5WT0jgWsViterS1VywmIrIcDExT2t0W7hA2hdOWy9GYDmHr53zkg9OwcLBoq1PPGUGlumLcBIoncDoayKXfZIq5K0USa7kXuD31mJVRh4goysUXnj8RM1kefmWKRWAKOX9w6uT+BWARPHL8jCoNuW9jXlgwdmjea8cIrwSbiMZynue26f3vv/kAAfyb9/+6vJ8WbnRdPI/W4NXuJTg+NycKoSXCIqeC3cVLRwgNpTAgCk6dktLwqDlsOkyvK2LtkqMSAGJEdcl3gggpcIwmMnHOsJQdPIlFJ7ECGOZUY1NccwoHo0Kn4/dVNBLEbLsh37fhvOedb5vSAbjwINgySrLlWDc5vaYmDGcRlQbarFkeIb4bK5Io4wZHenmrgsWqfSkTRo+b3iFpWeooMtmolxA2KvVwMaxwE8oKYKwdOkTVmjzbeBxyYWRWArviZex3REE7tHqUfu29B4NBD3zx3/Jx7jjHTi3kIQjRL2qaF35nL6DvEaqAY3A4GpHKKMkJAISeF3oqyXhppQfwrTWMQCHMQmZLUQzEP0rAzlLZFMcVRbbcZAJ/9/p9Jgoon91Bl2olEWKZzYtz81w4ojyc971HCUAnmbN2rSqGBmoqgcxwV/vmtLWRhHm9GDrNAY8CkSMTpQ30WCbKC0myiGVzFk4XYAMxuJmBDjNVrJImH0eHLXP2xKKTBFpTWG6gpVccCX4wvkFrHrUawhLhWXzXTdMVQh566CH64z/9Y7kEJy7wQ+cirKr5kDi+sS72N0rAGAF0tYxc7k/BcbVoGYKl89vTqp8IUskcwclVVF3TNldTmwTWiA03mOX5K4eYccwKToALoCuYjWNHjkmNJL1inimtzirn3e98yxQHAHvmvLEmkCcXuA1zUMZ0mbiwgj5D4BopdXtFObUS4efh0ilXMTej0rQtnZDyw+SdwkAUTDSAXzrkqwvYStCZmYRnb6hekKU1jqqhBsFIq5BYefc47tKv3zxdKBIE8Hu//2WeKLbb9V5AQNXRTFM7gOF4RyN5FiUtF6iwcjWMdVDuARls2Ul5rMOCG1rkOl+xFEWgiuonTODqW2Bimz/COkerhLvA5yi8xVgAz6OVxcJngWOGkuoNRtTX1DW7n3PzR96SlVv9oStW3qGrJP+zpEaTSJTCWjiNLOU8QaYoj0uc2ABl5yalcjQ4KewzLuOi90hV4SsHgwynAGXL0qWR3WIaNZJLMep1ke0oAXPt9drTL2iIO3YIESBcIZ7Ieevrz9OtnziICPqtB6VQJOIHlk1kaF7LkeD75F0sjVwLkVbUQsKEyEsLhF7OhbjEsHA1t5wtlYrtd/vsxgWns1GuIeQ3JHGmTHRMAKIes+Vy5IjgDS3l3opcSMLrtPh2vvDb75riANVwTJvnNUP3Amo5dkzMreefLVAl+Nt1EA6WlC9UvLLBac/mwdN+Q1wuztEePqX6OONI2FRXkzdksrQLKWvkclXRIFScqCiwpBKsy9KqTNzySoMJGT4rAw13dsV/8OwzW/Sx2w72Dfzc5wUTaBMiThotoFnKkrI6vVYZ3HoR5i+uvxhKB7n7RgBmxlmamtywICCk0fFfZX9u6Xek1/Nhq++s8wsgDcP64FxT8K49T7kWceBdUOTziw9ORwODKvLLNftVPV52oWefEvu7tz8dP0L5UqlWBfdnQQCrR2bywgrrKjfRrMEeiCt4jUWpsQzZali4QC1HfoIGy+otQ93CwLXWNULNQC7JgNolhNRowZ+B+UBoVl56/TwWAhxjiT72sYN1An/zvk/yeVZkmXP3VQwiBmHDnFcWKSxnQwFLYMPWTZJaZM4sC9uIhmFuilbGcTTDYOdQKck2z4qCCFDlrlz9U95cOAC4R701LQLMryN0VVIu8fenPjHNASD/j1wiO6Xbma4TeOIZsTuDUnVx/htFivVty82Orr3uCDXMckikE3aU7JOxe0TEjJUtzksBZsoKv/jOrtxva2cnl2PYkJOB7PKtDe01GIlrFnZ7Y0YI4JLLl1kkAC1krPDsGTl/Zc2jd7zldw62jPnMLfniyfXggdTgUrFJ8xzBYuGLDVE2m9FSjxenlExri2HcgUGkpdK7g35XOrKWAlxRSZlqKY5gUnKZM+Fw13fhj2hAVR5Q7G2M1btqfzsf/uB006ipb16QhW6mx/8vBKAFEZB2Has9f/IZSbGOxsIRYLe3F4UArvmBIwxKRUoXunFinDsnVsP83Ije9a//kF7/+l/OXxft4+9UArD4BXM1NUnN3sYXcn++mWcl4sgvCAibLuz/lwQAArTeDMWziDnCwbpSEg1/9pEPSTi4PIzloFRZeVgypAEh7BiSM7lgMVyNmriJY6urS3kIGSXlMZotVM+WW0Ks+vr7oCsstuyqjB1ZMGTrdgaahTMk2jsvL2Tw7COXKGLGSalWN13ickoSpJuHed8fx5GM3KuuqNArXv7XdOON0wTwmc9IoUgb3J9IA1+1kv2c5zZaAetyYWcNmeMaKH+LwfkL5o20Kit6XdmzBQXBAoHeZPEXfv9SsMjqOKOEXHnANwORCoXx0CFRAm3knCrLaFth6XbM+ciHpUKIDZgR9oKYvPKwnWGFGu3Y4sKK9LpNU+p2CgWx1V7gqBaGIXfjFBBqLV3uA3Gj5V10DvxKoaUOx+LS7PXQV0Aes+rWqeWJMtqeFV1la0sKNMNaXD2qxnfwGJHzHFWCn6RqKDV9H3vkXv553TVvoP/tp+8/gAe4/747+Dh7KtFqBiJAvWjlIFdeEMPiHiVHjaGmoGzkwJmSEme1fszlDcKwABHujb+5OFWJ7VuASeZCnE/l1jr426qvSQW26WrhZYyGibSCAD7ya1MEAIq3WnbmBbOTzdkAlGl5LC6h1aooLxbA4ZdhU8ao29gQlEy5pecLwFPO1eTRoBSaTsThgqILkmWEUPIyrc5LIoVDoiOcP2NOlYyeeVosh8uu63FoGE624UjuMVIxcsP1V9FPv+r2f5AA4DIVpRYJKfKmZaU8txLMGilhGMpJnWW0tc2XAWBMqYUZWC4mzfyACbDARZQXyLqvIZKKYagkKKbGucUjUIw8rAziKNVh5Pe65ZZ3Z7bg+AALaelQ5c5V4r/X6l1aOMluMTMzp9g5NEsstOByXkCrJdwEsXybTOxYyz2EcojRKxU/Mk5UDYfk++YpDMnzxXIY7Mu9OnuGKAYmULgCqqon6ZhmZ0NaXBLFcmdTcvJnZo7Qq//lVw4QwOfu+pg+o5SxyZsyodBUaZfnJe3zF7mYEgBrwmxXraJlnlC+S7FIRXRVK6fgUKnQRilUQCkJgQcXmHpMSHzJwrlmtyjvequqYmvn3H77dHYwZIhQDOr3THsCY7XVLyzOLHUApIy5FYLgRag2c6eN7fpavUOu2qLwcGqTLcpIF3VYmClWYh44fanbi/ugqqksuKuTMR5qbh2zXdEVAHzAM80tpLS0IkShyThsafzLf3GQAzz0RXEE5ZNT2vnlicvZuBIFO7T0S0YK2Ah5VnXpmoV5ZxFKAcXYsKSXcrwEdZlsZHBrM/SiVON/6plRfW3a1CsKchRJLfk7fuKTkhxqOxfxavMzm0fQyopsbogpV1eUjGHSwZ7txUIFjvI1J1L/DmN2TnZhrQ4RoDFz5AxnQtH9gSzScFyUP0MCBwaKJVhsxHMjqnii6LmuBn7MWuXmlUJIS4c75PmAjqGcq1gJ/Z46dZwd+uXXTvcNfOihB+nf/Y4UiJAYBp67qMtfbrVi2ARj57mChwDSFCHYwhbV/aW6shaHUgFZDoHD3yCt8QoREJcwFrGj/RwvUN3NYYXv2frgPshTsHvyfS/4nvPhj0znBaBujJk51pLFqGVXS59OtyAH0nTCwR5wPLiSIxLcACBkec5aLFbA4vJMDi+Hw8m4wXAsx/c7Z4o+BRzQgaNnh9JMdn3FdakRyMRay/udPanKHUcrVKmIP8GykeMM1UxkB7Wa18irOD167as/TK8vgUJBAP/hD35LFgfcRiFutjvLtY9AHxMUsdTU+TJxmKOHb6NxjqxUuzfvMJoXtWBt06aYo+fc26jwBOcNPPgdFR1djrNgVRltrddhAi3ds8ADFPWd7YbO+98nhSJtoBCSsYzZmYWpurJGSUG1lPLFKNoRN4VA9GwcowOo1sqr1iT6B5NwUX3YwAOoYjhBY8ixvPww2qOMKjSYFIUnfHeWPVtJrKhj7lmQUTVIWdSg3pAsqGm9cIYIQHScbHGEMkkDShIFYFbnyPdalGR7dOPP3nqAAH7/332es4dQxBILXgZrlNm0uW5HWsl0OALBgCOlpWQ45Cxql9AJStzIe87MyDxYJzM2+UrZ1pOox3NpZemYIEvHzUuZF+VSSVBWOBFOLg/zYuLTMv6Qp+6Wm6f9AKg6bfLHUqBwYrMd0NamKGoz89O9e10ffXEl/y1KLHUJhFGUiDmyqg8Fm5YLFmZcGh7JHxint8V8HE/q5HrIL/Cozb0HIOuH5HCwSLCHEy0u2WyKvdtql7OYheDghePnidBSTRVIJ+aJ7ey/QL/yumlPIDjAf/zDB/m7HkCo4GaM1tH0tFJkFHkKGChthyEt5WSBe1psizOgVXZjHrFg6ODZaMjzVbCJMjR0RD8hBZKmPo3GexrwKGMmit9rmpyDQtYoY2fDSuyw2FIz21Lqp6jhAuee85v3FAUi5ESmE3lIhHlLLcvOnNaU5gXZZexnZ0dFj5M8U+pTpVaqqh0jxUrOqThistUrwMDLLgALR21h1MkbZRF1u1tMAOhwIecq1IsxgJqEQR4KysqTKjQsS4WIsJe6PeEgaQqrA23UpOMWhjSzgKXxCN34s18+yAH+4PN8XPIKUVqmKBFjMDQcNwWsUhHCC5HBZNm/ms7GC6HKWL1RRFh7/S7NzgZEjgJCuEVtobQJAZX63JYyoPi9tGPahWAdbmuvYqsaeDQqxSQs5C5rNm2xOF946H1T/MJxi/oAxq5gDWA3nD4t9e/R/NHGJEazaOmZA/lVbliVOXBaaFVPT8xAnxq0clhCtoCORfEOIdQ85gVDqdmivoBZHWlS42bUGKgp1B2JH2KsmTe5lpyhFJz274vRyCmmqt+mWlWuOaumqEMn6I2/9Ef0+rIn8KEH6Uu/czefJ0qdNJ5G+JmftVSUydVoZqg1eRGJs4k1kYcUu1pVnmWqOKoL7gaOMdJ8QOlRbAOLZT79/MPSmrkqDsqAof09JOXkZ7MSGdbU9zHEtpDrc6b1BRaCc/d97+SGETaQkGAvU84owfGegj3KBSTx+cwcuoRowmcpZwCFoE3pCSuqF1SqNEksR2BMlIpnK8kOMTTK8wr3c1gTZbLTRaayTCa3tFAqzhtJccsaGSb/BsOeIpSREiXvtzgjk4IG2T/7Mw8e8AM88NDtfNwUcOAczANYxik0tAlmoy4WB7D89p7m4IFSkCbaOdSXRBF5ftkQrZaKFq6XXOhUeeGPUqu4MgbTmmDYilsxj7LYL8ceeE6UWrjZxoW5gZ/6zTdNewJ9OIK0PJzai1/9VAAAIABJREFU1IyPy4qWsQB+lke9NkMZOAd3rizKkcSsEKoGXlcgqVulXr9GrcaYMXd23KU2tVounTun6Vbw8G2JuYcu4ka48Bw22uZjl0cvzV/Onvv9CQVocukleQLqypyIioW5Zfrnr/rIQUfQ/eIIMoAqm1SVJiOQMq9U0Nnsf511RDVtQco+EpOe6HdgSqQBQqph2cVYuH4FJl+IYXmeomUM1kaAtcBeFC5f6acg3IQBKNZFjHsnFhzA6h7a+jmf/Nwb+GiilcHYnDAUqXrfpFIWwsD5PpsigKAyS2FNGyEoq8YJKJ1eruaBz4ajRp5RM56sk+tamzd5yIW5ArS43zEwBsAOKgKihAZqz4c6iWlmi+NQquZkvwtMfso4+0ZTRNbCao1GwwmdPn+W/s0bHzoQDr73XqkTyH19HJlkm4ud3aKItZSpL4FfWGzKsM7m/HvuCCzMYYsVANjKu5ezo1yylMq8GhouoN8v6gaXCkGkMdWqZeDolAyYWp8y4MT8MrzmcUTOx+9+zRQHQDcQ+6BeUuj4hTxFwZr5xYSDd4AMyigIUhprBws+n9rcywajVmsStOetHTR1lkVfWznGbWEx6jXhFL5X1Ao+eXKdiz/UGz4Fmg4G+xsyb3Z2RB0tJHlo5TDPFkKhW5vCdjfPx9RspFQLXWpoC9znnhWwJECkH3j/bx8ggM98VlrHriyvcRMIlM3lkixsPhUcwDx8BvnyOZ6BLmlBjsHnJo2VUBQ8pL0p1zR4V+5aVpdznjFd9gmoqCuzdHPSof6hDSjaErsQoil3Ip2mBPyFEvqFyHE++8VCBMCOZfUHuz1MaUXry8Fkg8/ewJh7/elgUFgFkBRUDvZUMsmyce5DT5UzQJmztvCd/Yj6fRENRzWCieQNG9vbCkBBXaDcGom5gANGV2Hex45Knx7Iuuef3STPnXCEDSpctYqkVrlid3+Hzp9H8uc8feADv3URQMgn2MxEOBURNXTttmJNNmmYaAuvWh9BOJtscKNoDfFauzbpdyA71HQl87JyTqQrnVd4AwSSgQTuJaKlHDUovm+Ko2+6RYkBlCOJTLyqd8CHUBgB8gXn3t+b7hmEVq6JOi28eEzIEN7cll1QDZQla7VLfIZ276DKjOBsQSJFsVMatRkaDuvUaEwoVgRLr+fRyefFBEpj0bQxFpdFzlk5E10y/lEJWnk9/Y31Lm1vyfPs76HKdi33JezuPkJLC+oJjNHM0qVqZUJJBMy8S1dKNxdKoxr9+D87GAz6jd8QTOCxY5ew1QM07fYOrIksT0jFcVgn0PzZQZNxIZfcFc6QLvVD2qKKZaATrgsm+Q+iHFo/H5zCeEMtp1+0ly3sfZMr0FPMwQNCL0PIOFH3AsIpAnAXmIEP/B/TiSEwqdgUoYS8tEpk8tUJaU5rCAQl3B5eYjSSTl9ZEpBfKbEXhXDyOZpnuLPdz7t/z88iPUp2TFczZpBbb2N+XrRsBIXw74knnuPdB9bqBROKNVg1p+VbJDlUCLDKpp9LTjogxxWOcVxRZ6PBf6J/9pPRFCDkS1/6En3poYe4AtrRY0c4px4tZ8+dO8Vu1r7i+YUAtFg0t2rnGp2569X89uj/Z8qy4Qv4IWz+9Sf4JgjENgK+w7+XwsHl7GCzUEzrz6FlU05+ubh1gMsJ0Xr75DNM5Nzzu78wpQOkFFBYEW25Gc7Rzt5GbptbOlJVj9s993bh2EDnsIz6E82I5dx3qRyGYW33AICMFF6OglQzWsXy9CmphTcu1ckfjeDfBhhEQKFBgE7eFapXxVSMULQyC2l5SV8YmPhDR5jbdAZwTsE7CQeJPEPdvGij5+n1r/ujAwRw3+elY8iVV13BmIjV1UO0pJ05UOEUTRcxvvSQxAx6XeFE06XkZXYhCpCvxwuBnxcuvMp69jUo/gDnSonbIhtaLjAdKyjTUZF9VKzqhc4eW4Mca1AmgC/8+zdlVb+ogIXK22az7u6MaH6+mUeQ1s9Z+vF0jrkfoC0ZEhszGqpSxy+DYI/yngQuYg6i7OXtVQA+MWBDyo2mIeOKvMNJ7FMc+YS8wsEItfhiDlQdP3aVLIDmRvqBfAfTFNRFOfKQwq0NoC3Pzs965DsZJZN1uvE1vzelA4ADfP7zn2MT0CqEgAAuv1zQRC++/gfyadvfF1P1T/74P/LPwaCTO3O63V2uS8hKmdnfiJPYjrcsZ42RXLjY3AdJC2/nNyx77zzJ9RPCKRFG+Xf1h+C4nWGh9aKngFzd+eLvGh5ATkVDKHPfRvE0KhiY9LDq045mvdoDzswe5WIKvr9LcZ4oCq9fi1uxY3S64kYeD7a4yheUYlzP6v4WJma9MPnSCff5RQmaVBFDzXqUm6OeLxbDQGv1iLYtTiF2fsB34aETGeDgHgVejznKZDBHr/nFOw/4AT5/nxSLPnz4KJeFm5mZoTUtwnT0iMoPInrqaWkRb1wL7fFsnDzzDA27UqPv8OHL+OPnn8H5aovrT1cjiay4lyFjmplVxgmWTbeeFptaXisaPzDBK1Fc4EXOn+sf+sX54u9KfYBEy7Gzo0Jx9LVmSfnAORN0o+yTo6Fduyh0AqQrw+7vJ0FRaTvu54UgR0ODhBVpUigL4/nCzrEz+UUc1LaXKws+EF3BOqVGVnBUWWIJdlpKo7hHVfOraHOnNEEWMZhASo4XUTTap5rqLllSp1/8xYOYwHsekN7BwDgiJwHdP5YWJe7xmHYFx++G9VtdPkTbuxvM7vO5qKLtPBxcDtWDFo0maNZU6EWDgXAPBKkwODewBPpE1tF4iJ5FxZLBycNNs5KEnj8ppuzyKnwvBSdG7iJbYqhsrgU07AoFGkHqFpgpwKrGvV8oKoQE/pBg06JwA0bsFB4o/D0Az81SOrJc1MTH5xNOZ4atlbGGHPsV8mNkuaLhpPrmIzhyEMlDXgA+w46sUbMpaWadXREBVmMHv7ueKYToSSQzgvYpQUVAoeOe4gQDjUBylFGeeTiucpFFEJOVinWUi6AX78UI4NOfE1TwkUPHCXUSgLJta2Dqe08+lq+IEQXq+TRqTRoAKaUK0dzCvBRyRstbrQu8tnaI+upGH6gyiU6h/I6s75XK4CVjrppSTv607qA4f2cPPYViRhttaKdyfF5vSU2DVrtFrXbhjcUx80Hg9wtRbM5vPyTVwvlE1ycUa37pS/4J//21R/6M075sdPae5V+XZi/n1GsbYb3NjRmuv+YH6Ovf/C8cJUS9n95ojxKtKeA4iO3DdKnRXm9dbPQKyrhpDxyNnTfrxcMjtRvDD9D0Wagdncldv8F5d0GqZVk10shp5bH4IfBVjtH7CXlaN2jQ36GMQkKZmZ/+mbsPiIA77vk4J6ocP3YlF29aWVmhy45LdbAyIOT0GQmKPfrYtxnBiyaQ7JjJiLa7m9RuttklfWRZRMDa6nIOjzdRvaPgGiCxtjYFaYVCl2fOPMN4icUFqR+AkeXw8oSseqrVGbDgDhBBSCYZxwNqzkwTQBmlZWltphw4f/wndxywAlCf5vlT36Urjl3PD2AOjefPPsMtXxZmVuin/pdfyB/w/Prz9P2nH6ZDq8fpm9/6U4qVf6EJs9Xms3azYXWWG0rIYqJhRJ88sOixeQwLtgZIGQbcpqGZnugfRKhikpGnQR5f7+dkQ4pHqkxqIAo6QVBpUpJGOV6x18no515zF934+ukycX/z9a/y/Q4dPkorSytsZxsSutMtxSjOCVp5Z0d28TMnn2VIXX/Qo7E2mMTnFjRiZJTul1BD03PWJSyOuESMKGcZnXzhe+WIPF9/YeFw/n1kCsPzIJVbC1wg/t7bRSVxZEgXzjSu/AaQCbupfQKq2Cqb4DPnoS9/KNvXNmP4wK84FFTRijyh66+Q1qd1rce/vims1hSi7z/1df47SqUjBh7C3LwsatyYXRQYqWa7NptA5ViuQEfPRyEHmeCZdkH5KBFTqTQJJVnQqAk9cuv1GRpw9a2iG1g6RgRTsHzjnlzH9yL+rBY28+cfaT3ibpfoly4gAKSHf/Phb/J3r7/ueqrX6+wIWt8Q5bXcqtUw+RZYefy7j7JegNz87b2zkiiLrc5hX4BTJFyOsaJmpbWjQwS0193lyqGwos6cfVpPLfZlu7VIvgd9x2MlnOfzAnCfZBSjpiFkSqFzIBA0mnQ5QQZjeGF28F2/+Y6srmVNmGqbAVWqkuVz2doNeR4fFnh7RxC3fXUFT0irerIfRDqKDofFgzer4o7FsN4DDsw8LvuGh41oqGBQLRxGtbBIbPS5yfIGVYJF8lyh6nPnH6V4oh20KtCEcZ2Ca1iHmwwvyuXi0XNPFE0L1Q7GPfqFn5tGBYMAHv72w3weQCQgHrSzsRC4NcrC8Rff8MN83v6ecJsnn3kiz1uYW2jQs8+eYOU10DC1yx5UmRer1mHhZdRKttKunCKiCng5wF+pwP+g/hTVN1GvuDw8TzK0MUyJNszAJBkXbohSDgPOdb7w5Q9mCOH7Glman1nI2V47LBdgJFrfEdnX7csuQ7XvOBqSH1ZpZe4KVuB298ShA5YGBA7i8kw0I4mmVcNWHrJFubW9/XUOEi0uzkl7uLjILPLZvEpoOBhxKzW+LlXZvMSIXHmOsGVIW4ficUjY6CAAZnnQA8z08tqsF4yTMf3cqz8xpQMAEvbIY1J777tPPUo9NMVwEOUTMXRo+Qid25T3bzeES83PrXGRx+98/2GKVPk8cnSBky+AJto4g6KPDnNVqwyCxpAYUoZHUUMoz6sLC4NCUu1LqOAIgSbZweZcskAUX8sNqBYiYiq9mMCNLHqK4yhOaa7hsryHOHC+/B8+wJ+hBi9OWpg7zLi5sNpmbJwNyJL1zTPsxu12ixKlq0vXcDXNn/ixN/EuOPH0X+Tf2drfJr+CsukD2u9sUSWoUas9p53AMzq/fpaG412eiNUVCejs7UmaF0YtFFu339+lCN3HmQAqOQFQVYjl2FFxDEFr396V84YdOI0WaTzuUTRaJ99Dw6oG+VWPglqFfuZVH7iAAB6iv/zqn/N3N7fP0X5nl99nOLJK5IUyPNT6f5ceu5rqlVV6/vSTXLNndgEdu7Q8bELU2xLCXD20QFZW5/QZUaQ3Ns+p/Y/oXNECbzTcZxEJJdTw/bVwhoABxLDdPT8/z3kb3OB7NCa0z7U8RmvIbW31DD4GwrLyc1xTeDIm5/f/qMAE1qotarXmGQuImj1lbDrfnU25AtlqC3XZZT9M9RD27y5trBcFDP/u8a/S6oo0UH7h1Lf4ZxiukutVqepXaRIPaDjp0I++/DW0fv4xqvo17gFg42+/8YdyWxRNQoXR5hxNJn3a7Ugy6A3XiQURei+m/hAmqkPDnizU/OxlvAue+N7fUl9rGzXacixs+PSqH//1A1bAJ+/6IB0/9iLqD/ZZo0aZl15/m2FUOQJZ09gCr0mTaMCdOON0TI2GsGQogrxQlFBHk1ivueZqMk18e0fmZ1fLzKBWELKaMRy3wqVgURWNS75GI7ZKsjQgg+Ibghj1knnHa4VwuK45sohSvAuHaLNkIrpOhXMzx5Nhbk1ZGpnz37/x76esAMCtDYaMpkhWBx8PeOWlN9DayhFaPvQSnhwzzSgBEHNCf/vNv6B17WjBL5TAEQRv/D6XV0Xr9Zna4by+wOEjN9Cll7yYX/47j/2f9MiTf01BrUs/eM3/So88+ac03xLvW7u9wm3nMcbjDm3tSQuVo4c89ln4dDmbmzAGtjcyXsArLn8liyT04YHlgvHkU3/J9fq8Skyv/plpEYD0cBAAagxYw2kEUQB/A0u99ipxBZ89dyIv/TKe9CiOXLrkyIvonCaoem6Frjz+Unry6W/Q+gsCUIVHMbfFtUBDiwtYOOx1TUq5/kgXQD6/y9FVvC/apXg51K2iWL9GTSuNSr5UDuSFMnj0yNV5mXtc4/il19I3vvHXfD0oib1SFRbnq3/9u0wAttDwfjVU63/4239FczOFVn715dezNn7izNM0016l4bDDuwDZPZME8C+ixC0qVMHpYUmd1eAycmie/vE/+leEfAOMM6efoie+IyLjfEds4UqtT/WquHgvPSTEAflqBRPwe28A1u/S6ioUzy7jAuLJvBRZ7KuWHLc4HHzFFS9l/8NVV7+C/vwvBfABTOKrfuIgJOz2z98kk5Q41G6hUYSTZ0wdWRMRhe4j1u6l091m+PqRw1cyTAuOmJVlgFOgQI7pG18XkcLNIdQOlCQN6DMIUkXMtpFMY2My7vIuXl0ulOHRAAk0oiS0603qDhGOrlFU6iPAyCvWAR2aaU2X+X/JD/wI5yGgZ+O31NKxeXX+63/9t0wA0rECbdfWqasdOC6//Eo2uyxTCD3vADY4tf58bh2Mkn0ajJ6kNN2lMLyCkkgosxJ2aTis5iBIPztMlUpCR1eP09LiVUy1Z889ST1VKJ85Jdk9c/Oq4EUDunT55fxZq7lIVYVg9wa7dHr9O/z52pqcC4wAxvrGt8lNJftnNNxjNrqycpyOHBJnzkte+uP886mn/m+68sqfoNe9TnoEYYAD3P8VgYSJAqaGu9bvQXXUHPgZJ1yW1VrVry5fkifAvOwH/xFfA1HPr/7Nf+Lf6+ECV/8e9LoMMsFCVLXvEmQxTOdKxSeYlXBdB346VefP3Ma41tLcGoNHZucLBb3b36FqKLhDrgKWFnkbECGXHJU5gSjb3CjyNipovfPVv/l9JoDhqM+a5PzMPF13jey8x77zCGuVJ9ef5r8nkYTfWi3N2QPuP8XDw0YV3/uEAzMKdvAmVFcHjqF9V+dexCHdq678n+n06SfpO0/8GV+zR8I5wCFdRdg4CbT2lIJKmwbqKYRuMtRiEVdfdS1/5+TJRyljkzQglyR1HGyTd0N7lua1sudoLDZ94K3QK/+nn6cbbywIANHAP/+r/8zH9zpbjJdDcodZMeUGDPBGQuHE+2JzTFD5TO5KTYW1g/M9++yj7BuYbR1iSBzGzo5YQ5PI+iKjGISlvrcobNRp0NvOu4bhXHgWYU4nNKHl2WMcF1lYKXoUQ/eR9DfkUKY5bJ7fNWkTpXJvvMNMe17qOidImYvJefDLYgXY+PF/8s/pb771p/xnoy5Uhuhdkgy4BiD63szMljODYCph12PC4aEqPGYh5LZW8j6yegXvyMsueRXVa/N05vyT9PDjX6Sw0mQgSUqyi8faMqY7bFJFYxHDyZDGZvu7Ncb6YwzVH9Fql8xVbXY5gtmZZdRurdHcrCii9RrSxBBgqikBTHsC/+Kr/5mefuYZLu0qkC14s8T83NgS4sFoNWXyLaEWXkJD5OT9BVO8S0bd/iYdOXQVAS3c2d/NC2Ww0soj5d5JXJswTWhpeY2xIHlZXnRi8QaUabCu3RTx2J6VRW3VEYp3GVrOuKQ0oR0NV/t+yom4W1okGxvi2OKL6fz2KbryqHh5nfsfet+UTwk19ypVw69Nw7+r1SHtdybUbiPVqfA2NWoe1etokAxkDwI3GvpMZ6miTpqOdsmaJLuUZhOaazeZqtMUYeSEJpDZXMNvj5ysSpkzJvP+9od7FKfA3vuUJXuUTeTezWZMqMO0MK+7IUMQCHZwhaqhnINAm5WuXV0RpRIwsVe88l8fsALuul9EgNnY6Ooxp6XWT54qevcB9oVhrFm8n/LOYMUYyMkfdiUQNje3mHvidnYFT9lXLobfrcwuvIIoAIl+hTDrbGxsfTfPHlqeF10E4Xcpl6eZVxWE0aEsprTb6eX9n+VkOJtQhZ0oGl5Q9eULX7kpQ+wcQRl0uQK4wiJvcMHOz4pSww89fEbmBxE2P2+MSbVaQskkIL8KpaaUNdStUBheQv3eKeqxPwUNCwYM+oDyFEUQHfICaSxNlLa3CkdQlmnoNAVATTN0UEI+7pGbVala22LlyPdmaH9/SNDAl1Y0eBP0CBp1lg2p353QZBwQyqb77oiaTYde/soPTRPAQw/SZ+77kLwfSRduyOWivp+WtXF9SjSXD9VQkOghBRikc5ik0mfUqLWpXVvhHcngUfWpbO+KGTgYoYWuBJmAeB6OkPaN5FfEPerUrCN+kXJYelxKmE0mKcv6za0tuvpq2cW8xgDt+oiAJrQL135W2rwaVUWqGsArPMCZkSv5lT/46JQIaLcD3jUYM+1FTt9GmJgnxhGnyFCjdM2apHhF8S4nSlZD6XMPZxFGq/YS8lxhVSfPfpsmky5HAC23Diai5cUFwQp1e0/TZDRPQUU8bioaaX8fHUH1ReG9cgdU9dpUDcXMWpj7Ef55fuNJWl4VvSCLnxGoei2kQHMPfArZDApDj17xIxdYAUoAIPoAu4lrH7qEYpCwsRHORTFoDBcTp7mTW/vnOFYx2xAxNFDZ3gib9IqXSlT15Kmn8kCU6wvi98Tz4hASVHARWa14VSYEJNiYrQ5HmA10A3WdgLZ2z7IpbuOaa1/EeiujsrRkPs9h6pNVD8dco9hGJUARbMVL/F9fvW2KAAQLENDqSoN2d8SrdnTth+nUub+jGvveseCd3DcfeC326MG3DdY/uzAibZhJ881jFChC5/nzwkJX5zKqVaQhoiRFyCs8ewoTDuTOgKJJg8bjOu1sneNd7HiV3AxCkmiaCCHWq4sMlFhbFtgWhhdo/8HhGQajILw8o+zU9YRgJskMvewVHz4gAu558FN8HCBQznLy4WIVjrY/VRdJCMFq+AZeSSfKfNZ1UKTh5T/8T3WRJXPo3PkTdHbjCUojSYMX01ayeSxamIwTGg3jvLAGLhAGhTiwwtewhmxgwaGXjKMuF9uuq8IJVy8ykPaF/UojSrSY17wM/gwE4PtDqqpbtdm4lIKgRZcd+acUVsQHwDg516UXTj9Bx478ED36uHjoMFBC/oVTz5PLVUXhXjxDszPHaXXlOurunqaMvsHnWTXtsDLDuIMkC2kwnssDNNu7Mqn9nmz15fkG+TSh7z17io4dPUz9obyENHRWDIFW2CxKuWccWuahQFAUKQs1qWQ0Fl9DNWzTy15+kAN85Q+kQghMONHy4auX/VEunF2kxsmx+Tn0VBZKvvSoNLOEaEJkE3LZCjTUa7P0dw//Fz4OpRoDfpVBf4v8MKR4NKJKtc66kZv5ec+AWiA+AQSnLNS7ob2HuTVNMqaFeaxbhbOagppP11z1UvraN/6Mer0dml84zv6Yzb3TrMTbiKIeOf/tYYGFp5HI+muv/Eman1/j1i2dUs0/HFtavJa+/difULNVKkPmeLSxsUHoRw+2efxSMSExBr0/ozg6QeunX04TR7ToZnMhDwejhq4lfPT64sp0oj55/oiSOKR6NaMwSGm7A3SRBnwcj1raDn1rR3Z0oE0d8P0qyqlzWHuXKR4dPGphm3fUUANTqB52MQK4/8t3sLY+GQuOAezSNGRr18aLq1wLaWcYnEqmfgOgnAAtR83fG64TnwB6EJnz6PmTgifsdMXshU0PH89kLC5k5C0C8ub7cu3dvU2q+nN5Qen+SEvndceEWkUgPK6J7EFJRtm9Cv2TH/upfA3wy2PfeZSiZMi6A1oB470wnj/9PXK++vXbMku9BrXOtldZpvHEBmgGfRltbiPa1aHOPrJWXLrs0h+dusHJU49Rf7DDfuvaBF0wZTePkiG7VQdRh0Jlw4jTWysYbiuTwQxyaBLBshhSwxf5z8QSSN/e3V5EPVZDYOaMubAERncw4Hv6LpTYJo2G27SyJDrAcLBOaepSrdGjhrZOicZVimOH1lYvpaPHf2rKEYRw8H1fvp3NNPjnjSV7nsCzzG7HtTmKhqJYWs5V6vrIM8NSQSCYi9R4Q7r0yI/QqTNP02gkC7y9K4G0pfk51n9QI3mmCdvcLC84iWpUreAdPVbWgGG08fQL36J2LaT19UIE4Bg6qsJng8SYitYoQEAJHsmUIlpiUAlK39ck5cCe96//+52ZRbBwoVq1SfBxy3Qr7t1D0YYb6MknRY5fe83LpghgZ/tZGo86XFeis9WhMGhSZ7BBO5P1vNjNwrzY4tDeLQEyTdEeRZs+aIHImVAyjPCELQ15r28PqaMpZJDPRgCZIouaNYSdgYbxaWVRTL3trbMMI0+5cAWKVSW0rM6Tl/4QPIJX02tfWziCQACf+twH+LsMWEGOneMzR8AYR0OqKiwNqWDA5OH6HJFD2FlnxHIhxYcgOhQ3jdI4/HPPrTNXadSxaxEP8Dj4Y6NRn+GCFnCWWWMorjGUhUTOiE6cfkGaRCOZF673CITm0HiAQl0oLeNQXZNhW82jLG539qV1Db6wuLBGRw4VOpPzyOMPZXMzh6muHqy9vbN0yVEBPKxvANhQDBDAwsIqHTtWXABHTzz3ML1w9nGWl1GpK9XS8Rty+JejfuuY6+po7yFW8xV67iCbN6bZmhRz4FBlhiQVhzr9CcWKQQyCJjVqonH3hmIyNhvaNQzBYk+4w97+KVZ2EKq2OoiB5j9c/6Lrqdb+QXrtBa7ge1UJHKoOALFhkDYghGw0m3IPeNMwBIgqi1hDgUKdbCkCAasD4g0fpgQCuPLyl9Dps48JYCUARrEoioHYPhxD5cJT4Egz2vlre182JeQ+2sSYLwImPMcuIvFQlofNnSibRd7F8uJxcja2v8GnnzkrjYqPHL6B2k2ZYCOA9Y0XaDDYp4U5+JQzWlk9zBCpE6fEJ7+/d5ommaB9F9uX5H1r48yhaLLNhR3dVDx9aRJSI4yoP8T2nhA8hBjWFeX73/tvxURrU2j0H2zUNEzM2rmihR1ZiPaMOJFYEUtbVA1hSoltzk0nNQIHRQ5K5OrK8kUJ4O7fknLxWFA4ZyBTrciila7HcVQ4xWhoSTxzfPExra2AzTCOhLu9+Lofp2eeE/ic64gnb0eBM7g+OE2O+OFOZIjwlbEYk6Lampbyh0JoaCtcD0EypLLDHDeRjneA6xwEYP6MwbAow4f6R84jj0swCAOh0smkSj/44lfSqdPP0+amgTMcbozkukKp++qka1KIAAAQXElEQVSC7Y/PsG4Aux8YNbzMbGOVxhozKCc3LM60ud0LOoZYb8FqZY5qtUU6tHIZbe49x9Dwze0/yQkgHgk7bzT+n96uLDauswqfGXu2O9sdz2aP7XiJHdtJmqWL2ogWIlRBVRVVfeKJvvHKG+88IiSeCqISFYHSCqkVSFSFCiK1BVVtSUOzNWmceBnv63j2zTPjge+c/5+54zqxEkTPS+vM3Ln3/ve7/3/+c77znT5ysyOH3UiDxqeepUh0jC59/Bo9cvYFWpr9sAUAQ6338cQY9/JLbWRahaVm+BitLHxEI8e+w+vryy+/3DoXkkEAADuBNWngjOvH4GDwrEqhbtY1IjJVqRjaQ2ENYMHIpo28Hi+P5U5aZrdvPHm+1YAyny9x7mBlbY7KlTzrInXZ2p45QIfEE9r05nI7ZBh+yufbyml7NhBbwPixikEirYy/RSeIU8h83bKGlsqy4xBDjgMNwRTD6tN/X2jmi+2AgssRIFT1goHiUIKMOKzHjFGmALqXkxbWb6gfw6qDvafOdtqpuxmlgHGUnF0oKcN0Jb8N5y8SfYwiZi+lUogoNglTEGTpYJ9d/jtTm7pUM4hc4Tr1RsTXMHwucqjsmcsTIzOs9P4UJ3Hu9kV+azEF9kQkomiG+5TcHZostTUHLCPRAQBQwn795i/UgOU5yYPYBpI5MKzz2sCZhEdtKiKjDXUOSkVDxw24VF7VUA4OHOMcP2z67i3mVMAxhGFpgHtQLhd5x4IiWcQ9tQgFvlPIy/YVgEYrej7O0vkD7GA0huKtK1pmlMrsSGq+Rq2u6yug/tpZ7GO7+M9XeQZ48uwL7PygyYJ2WBYXPuc4trb5JeHMJdc/aw/Gf7twwsnyGwPk7A5QVwPJDBuNDIVodTVJ3V1Nipgmxfpf5GOKuZlWX8FiIU+FggBEK5SY3h5a3xRe4fGTpzi0ms1sUDAsqWFRMNPrmExnmQ0VooYog09Q73BPMcPmfmadAQCAN//0G/462rkgpCo+gJyrYcm9ow8BLBIS36MBKrhaeHU3EbCqjo4/Kqdvujjah4n9+heXCEmyWFSiqEgCofl2qagUSJrdTGjR3E34ApWq+DpoOberfCGbXWYXdB7B22zvcvCMhWIVeP+pnbaGA5JbWkFMd4DH5UKF3ZbLLzc1TRs5a2mMAJq1QdnsVT4J5GPxdt26+wH/vbz5JTnVPhV/e1xhnmb9Rp3iEXXTHFtAbhpbKpOiCQmL5jNfUkNVEKOaKJPZonhikjbWRMi5Pya7BZjT7xCemy1CxVKRdlK3yGNEKRKFmsQu3bn1IXv+TmgANCp8jX19j8ixHgAAUydC0YgNADid6D8IAFhbdbdvsIM1izeXE7Al+hK015QpNejvZU9+c2tZYv6slCaAcTu8NDHxFKXTyxQMHKFupZJ2+do/+PMTk4/xdtkMhHhKXl5b4p3EyNAYgTa2ujbH03qtioaQ+VZkUIO6srvNcRctBgHAYSkGN1FP74j9Awi6VyB+H00vS6ViSz3VVqmmmojPtw3oFoTXa/Jm7e5maDt9iXIFOEUNWlWVLJFQnKceL7c3hxiTkwGAbpsNcOLBS1bc+Ejf0/xbldIquQ3Z30smS6bYlYXPOV4+EB8mtykRyKV54RFqmQE0WzR7+rlDJ+zOLQwmPOkSOZ1eqtVK5HFKICiSOMkUNCJ45TqU2imwvN8H+PmvfiIPDxW6djvTqvD/eAut7XJZd9BuY/k5xBS+uPmJaqDdoETvEfK4/bS4cp38quGGyy2Kp7BSyU7PfftFhuKVW1eYfzHY3y48RcVQKpOm2bkbVFN5BZsNncREvFu3kW0oEMoASShZn0P3XizkIcMnPaAwA5SKUE+HH2epKm42RWyvUt6i2m6W3ODsOYNUKm3Q9s4fLcCACqY0HVKZ3dZnNqpST49JPm+AqNGuWm3WPGQoCrXLkIcGHyI5e42Gjz5K1y7/rdUAYXhEIpH9Q8clUNSs09KCsISkKETeLJ8/TIkB2Tksz8suRBdDYg/uM3UZd9uz7rgJyx/7AfDTV4QSBn8D06nXG6CwWgJPq1oAfJ5TTbTjyt9Ao0bt2Oq6SnwPSwiKbUFmBYhAoTtz4hyfAzR3CDpaTaIf0iNpdVMvCU1aXP4XjwfMbhNHvFjepMpugXcQCAK5XeAF6EKUHEdEq0rlDYzofD7PM5t1e4nxsmXTNyQUzBE5dAB30cbWe3wSw0BiQ9AFL193jEWLt71mnoMV0AXu75tiypYL3rEFALZGlBzOQarvbpDLQJrWTkvJ61QqSjKnUoZaiEytY+PHObiCOkNtSwvygMFc0ZmvSGRAGD7dUPxyktsjgNM5dYnJY3/+cADAWw+dHjCCUJKlxTLOnD5NSD3wNGqGeMs5lBDQ5gpZCqjdB6jzKxuL7PRhWs7mdpgNpDl9U6Mn+Ji0ynmgA4jh6qJwKMS+ALKPuUKRUuk0bykRD5lRQMdxfrUD2U6rNjlqsMxggkqVAlUq4BS2i0ZAlm3stVPs1WpVyd8UxC/4w1udXcNGhqPkMuy8joQV60Q/EESgup3VVlxb//vY2Fn2mrtsR6lRz5Cj+yh1c1sXXEgnypMzV6iolDWwbtfqyCJ20dTxUywy5fKNEjWhl2tSNS+RR0xjOiCDB2R4VbClW8gRD2v7Z4BXfy9CkZChETEoNwX9Me6CMqlYwfh8oK+fypVKCwCHnV91u+GvqbwUAyBgOGl1c4fW1sXLx4t26sQJXrehtqaLR65ef58/h58V9KG2kmh1XXYRukIIAED+AT8C3iEio7j2XGGbM6wwREVrDVXNpS66AwC5UpZGRmIUMv28negxTXZsWqKDtiyFQqM0n7xM4Uibvx8LnyeP41EVPLGWlIsOsNU0AMamzlG1kqdtdSOjk535Bb5gpS9gd+hATyeYDhv4wz7fD4Bf/u4V6rLjDfHybOjzhigcEm/92WfOH/Zz9/yc9y0q2qIBoL+czqTpbhK+FmauLs5rQJlkbVV2Qjhsaekq2btku9yjWNql6iKVyjWuqoL5fXEGCMDr83X2Dcxk16lSTTFtDSAAiYV5GJgBXnmtrROIaa2/P94qQBwf1es2nJcKDQ3LG5fOzJPP8wSFg/sfGi6mHWkSD/zeDy2XWWcA9ESHyQx3drp66NF+gAMPAgAOR8/hvtgIDQ0c4akYdv6cOLGwsoThOagFQ0wFSyQIsIalyfT9LqVcB/tHeAYAgI4gLiRnKRQ0KRSCM4tSLxstLFxpycH0xSf45Jvpqy2mNM5TrbqUH2Anw4Mlqq0LAAAwmJp7VCikCOru2hgA0nRZCSoH3AoATRofHaSCpOE4y3Vq6kdUrd0ml0MHYvbfogZAO7b9AM/ja/9qBwB+e4E+vizFoajjg9MJADwyKWu21Uqqkw6o4WJtkEfMzvbs97opAACWzmzTnbsiPoG9+cbaKnncburvQ4metCCZm5ftOGxw4Dj/d31brhXLJ/yvbK7MxFBUAQX8wxTwt9v1QJWE3/5mnQGgDUCwXXjL0jjSZqdgwEVu1ckrEYvwg/c4pNBgfOz797qfh/73Tz54ix28eK/s/ydOSbzg67CDABCPxWl2/s59AaCDsNmcal3DAttyxZg9YFoV7bD7SGcyND17hx9ksZylhYW7HErWW0jI1cwnr7V+BltC2J5NSdU5hYvg9w1zqj6d2SDD0565AQTO1CpHv1LJCitIaSsIACwJ4mDAyfJuQN744Bk+EGlJRJNMs5No8NWbw3TZKV701e9g+pEFcX0lSbPTH3PePhiS40499r2OQ3CxVk7+YQP6IJ9bAYC6AO2ZI+iEII3PcFOY982dZgWA14OdkDz9ar1BhvLS8TZr3aP9sm3WX0NGb3pW+IEy1gbzDObnb9Po6ATNzU1TOqPTuZjq0xypVRH01nFmENRzHzN++hNSLAs1s5W1ZUpnhGMBf87vayvCcfDqwtvtvoFhM0R70PVRRIenzjzND14bAwDCkRbZGHxWr4O7h2kQjkxnSfn+wUM5FDJ0hfwWzUz/taXpQ00Z6MfP/YAdldmZN2hrtUKR2CSFY6MUVnn+B3nAh313PwC2s7ob2R6FzQA3XcipUHWsp7OL6mG/DaZ1SrWsRZdPMJqGBxLUBAPogINnkknK5XMU8PvJ5zMoqBJN2Kq9+5c3Wjw+AADWpF3O/mHcsVX3eLDr6qJoeJSODHYuW59+9mdqNETzwOGQ9d+t+J22dy62WcEeN5SoKuQzHOT3uulIQsKqnD2q1amXRZmRXQJFqcje5F6jQP7AkNKuwVTTyRXQpMeNFeEGZjKLtKuLPFwVqpRwI3Uy1Qxg6ZDGAIBNnHz+/w6A119/nVaU/F005OdyeXmIMs1bAVCp7lKhVKZI6N6+jhUADrtNqNpoan1Epuf77Wc2t1O0tLpKk8dG6e5skmZn27mXTEb6JVvrMuQlRD0HRLcCXIqmbXLiSbp0WXM4u7mZNwJrrW7sAAAefMAnPD+kApCIuHlti85/S+jW2lBskS+tUcDT5v2BkoWaf4ku4rYEACsLn5JpmtxZHLaTkshVpbRDXS7w/sCDq1BZNaH0B9BKxsNrr46nbywQ9UQ9FPAfYb1gmNPdQ4GQcP//V7POAG0ANClq+lVO5OAzbKclz38/AOgjEU+REjJ55DojgabWKPE+yDa3t2lpVVciNen6zY94eP1egzQAdAFKvV7hiiIAADsGJC+7u3p4N1cuV8jjcVNDiXAiPoA0szYAwfavKz/r4I8sziIkLF95/IkT5DOClEqvUa6wQ/39QSm3Mk4zF8/vU+u9Hd07NykQ6qOVhW2mY2fzOYrGemknvUFfTn9G8bhMoeFwQPULtnGsO2DKwOzsCFfO7WrHF/Z2vVRr7FHAO0huZ4A2UylKLi/R88/9kJJzV6gvnuDYg9uncwvW4UQ8olNGZf9gfxUAMr0eBID9qSQ8WGs/4QcF4905pTbi81E8ZqnzAxFna4sWVyQOAP9sZu4SF46CF1sp7oj2kYE+RO1HhwAZ8hfQYk6nFDvJcLO6yshImLw+kfFD7SZT3lV6nQGwtd4mHKA0D8JC6Ordl7A6dE2KRiWoMz70Uut+Fxcvk7Pby8UUvI3ZrNH12+K1ogAB9CaYAKBJvgDq1SQr53Y7yDDkLdAA2N7xUm9UmD5rm+JuFQr2Vns5j9NBPcqROXniLH/uCykZ8I6noGlaAFhnM2X9tf0AuN9DtNIvOnOKD/ro5fv3AwA+R+Tz8xvXeExnkyJeBSvms8zYBgsKLCA4dgCC9A4G86eL9mpBWlxEhlKu2udVPRujfnKqVLb+Pdu77/24BSMoaaIhQ7Ekm1QBAFrBCRohA4cA0rGRl+jmF+/Ib9hsFA1PEoQvV9ZTNLsgYU0YADAxFeILK6iu4qhrR18+uUlhIfX2Bml+XtZICCVpq+12c1weAPB7nSyqCIUvuwrAPPPN7zIN2uk9KCSshZlwroNX3AcBgE7UtC7ukP9BahYOWr5QYkfaSijhe1fHQ4kMDTmHjwzT9Mw09fUmKGDx1LFWv/2O8BRgusvo+WfkJbz4/psMBBBAurtBTHFRo9bWCQQQ+mJxyuZT7CyWChXq7Tdpt1LnJNp/ALlJ++WNYzkzAAAAAElFTkSuQmCCiVBORw0KGgoAAAANSUhEUgAAAEAAAABACAYAAACqaXHeAAAAAXNSR0IArs4c6QAAG4JJREFUeF5lm1uPZNlRheNcM7OyqnvsQX6Af8cDL1yewB57Znrac2Uwg0eIiw1CCJBsi6stDLwYyS/8FYSAme66ZWWeK3xrxc4qxLRysrsq85y9Y0esWLEiTvXxR99Y12UNvdaI4BURVVX7LxGxLGssy6LXmh+qqoi6qqKu62jqKqr84rr6c3xHH9VvqnzPC1Z1VE2re/D9ummjaZuo+Xneb56nGKcphmGI0+kU4zjpDmvFepaY5zmmaY5lmbX2ZV1i1fqWYDVVXUVb19E2TbRtG13bRtO0UTdNVFUVKxtgZe/LAHwZIywRi43ARUIf8r/ZDJvnRvpHtUZTVdFwg0af1u9tgGKw/Ppq89i2NkbUtY1cN1oUi2zqWtcYxzFfNgL/npdFm2LxXB8DzMscy8z9MIINr01xMFpbHZ3WlwZonxggV1S998k3Vi6y8lrWiHzJilEFZ6LFp2dwg+IFnHzXNtF1beAv5WSKBxQvKN+XEcu1dAp1BKde1TJk3dgAw2mIYRx1wt7okjbjs5UPIr1Av5+n9E4OB8/Evnhm7fXhAV0nQ/BznUGedPXe77y1zlgxXzHbCDUXwgBVugyni5cUd4g1mqaKvmtj03e6GdeYpkkL1vXSc/wVvIeF8552ZhlYRKFkA/D7YRh18jqY9DaFSu21sAF7gT1lmkZ9NvBOXa7Setqmjr5to++76NpORn50anty9fKzb622InG1yhNYVbVENFFFWzXR1o3cdiluhzs+McB208uFcUOdGrGpcHj0HDbC9TEO95rKu1yLmG3k4mxMRpxme10VWniD+2IkjpcwmOcYhlMM4MM0pAEUu3J/b77T4WAAsEDGY12cfsGy93/v7bUsyu8+aRlgjeiqJvqGE27kZhPxOE+KdRbTtbVv0nVabXHbp5uX2Tl1Tm1ZBGincYwBY8m966iECcUAgJvBTPiA+yp+MYJDgFM/Hg9xOh71d+EX+8dguH7Xxrbvz96ZmCcvkRcnuFUffP6OQHDOxWEAFoqRqnUNonvTdDICC5qnyS7HAqtV1sYIbYuFa3uSNpYBrwPz5pQR1jVOwxgPp1MMI+FCrNXKAMW17frYpc7N9/aApmQJn/7x+KB31sSBsC+AmbVwKJsO5Pd3FEr5GeG74HKN6sPff3cVessAa6JqxvsSwoK+amLT9nIjDEPIFMQV6gaG8E343UgIJAZo25XRGxfm1uM8x/E06HMzccjvAScZKBer0yc9ZvrCQ/QZ7kFmGGI4HWMYcX9nAQ5MwNy1in3CgLV5f3hUJqGSsVj7R3/wQsHKjZ2+wIHiBcaDZq2jr8GC1ieVSG6rAmpzxGrExr0Jg2Lsck9njozfNYwFAkT7rUOgsTkTGM0TwAbebUTi11gziiOcQ3KZ7bENHvlk88kNvNbi+AnmsUT1yR+91I9tpXxXbl1jnWSVqPEEpcQkKyIs5ghG92KAKcZhjHlKFxZi2/UfiZQ9ArfXz/l+8M9G5Ig7PU2XxXsASG5ZiBZu7wwwOATmiW/K/Tl5HSo8IcmbuI7SURIbZeA1qk++//4KvVK255eFFWZGWCeTI0DRK+AknLuT/mT8gdxJWp4YoJCXM1vL2AbZAT95XTLPuulkGA5CnlHCh81raUtMyliwwEmnP414waDNtjXkx0h/JkdiiMld5G0+k7ohxOqoPv6Tj0xYRe7Wx3dWRShMDgm9YHRJbwuwmSZxkymm4WQPSA86b+DsCSXlmQLbAzgZxzsv/KwwRqU/GSq08UEGJsSMAeMwGATHgStlmNhLTIvNTM8u5QXJ+QBHAeTH3//UR79CfjCAmTvgVy0wjohligTHQncfgc1nM8cKKVFe5kRgZo/JxgBoECNr2JUfyRHu33W9DJA0TRvve37WCCtO4xAnNjxCj0flfzY/DYDg6PWnpyTfTOQ/U1hzW0Vf1jAY4NM//sxRyMaVS20MG6KKaq0T+WF3CY7pmgImO2csE6jMogbFJgYorM3MzExPwZZUVqevLFhH32+jbbtzulQx03Y6MVghmyd7lNpAm8f9M/7zcDO/Z5wXcwKyJZw4XB2I2WL12R9+vspllAYAM+dLu08WRcJC53flaIEWXNvMDPdjMbikgGmeTEq4QUKHWZzdW4BI6koDsEk8oOs34utnUA67Pi+VOWv+O0PgbACtu+z2Scml7AKd9zpMgwkye4HS5He++7uCwP/nNqKMJYc6lkRwJmJ81oVcCbZgWSyzY9Oo65uBEyIbaTDortkg15kEXOd4J3W1ZnwFG6YJysxnfC8M6hogM0DxgKwCXajlvXPjqiGSh2Qd+ugBGOCT3/54daohRl0BnrNl8YSSTnLhAJBcD+aVQOUqK2vIjPPijwUD8AI2DE8Yx5OMZmd1TNawPRkNxM8UqequzlrAROixEqRgYh1ZbMtrsxxuvHFhT+6qaBilXNYyP/rofdd9cpeS500TS429kucTKEkvxDnx/n88ofMJqqwVwkN27CmlkiNHI0qQ4ohhjMA1EDkefbSUyLBGL171QBYzpZozebMAYsdOb5UBHKLak0ut/IzJVDlgQr364P33DHcyx//N7yYvtrBjyaFCnKsKG8ezF3SUnB0GMGfHeKC1Q8V1OO7ddihBlXDidDrGkTQGiOayFC4VhqIA60SQwBrUHaN3WUcBOodY4QnFIDbKU48unyqkD+P9L3t8771vreT34orFZk5V9gK8wXmzsWKTHoAB+I9qrd9s0gAua10XTK7308AWgDCy5QhJXmCK7qEtOJ1VdbR1F03bR1NbKeoafubvFkS3+uOUrLAAL8Qr/DOBn6h7dWa5FmnSczjYd1980wkgUf+JduWYTsOxcE6CLxMCx+NRpIdFgODb7TY65e1cpHh7ua6lNJOeJdbK0Mj/JZ4UrXGZo15dNGnzqkCtOcLxG76XPOXRED79mVpEhqRKtRyG56m+WKoAt88KVUn1rOKdl28LOlW9nnlzSjYJIkVGKjncNT3EhxBYkrRsgjAQFU2r2fpFBUIk8QsDwLGKTgDnoOYQ/0gR1V7pE3TKJTRcdfrynHZJpUvMGLciVCFEGW5Q62hjWZpYZrwgwSa5keTat1++k0zwicunSMptDCakH3g4qcsft7z16G6EB2Wo6GW65VmBSWrqYCJ1RFRtI/HurDuWqi0FVQszGMlg5jIX8ZRAmWKeAVAIV2oBGAcvAf0bPAbcaSPWLpYZtYoUDDalS6d2WL314puJAGYaRSGWLHbm8KQuV3rSATJtFoFDbB4hgkygfJ1Axb34POmOjFD8ljq/z8KHXaZUJYNS7GSKtVKEMEeVV8dmC8CuMS3HGKejDMBp4BV4Xtua53N/sIPTX+c25rmNZX40gLiCdr1G9Y1335JELruYpqW8VKiVZSoAy7nfIFUEjOIJrKIou6V2VwlLjCNZzZPpdeb7Kmku6yglrygyBhgHqzuZUaQLthFdzwbnmJZTjPNJIcCGrUyTKh9tjJtx8uPYxDzxd/DLHpBFoVPjWy8wwBn7H70gKzqTDpehhd8XWivczqaEkJ4SU0VPyf2WuVWXI6NNoyVrwIkqLwskNooXqNbnPvOsz8ErLJXPUVVz1B0Ah4pE5hhUo+MZKNM6/XOmI4Wx8TbGoYlpAgjT/Z8ItUIEQLCQA+FyNkhUb6c66w6M6wOlF9hfUWgTzU1Bs8Z+kq5KTSjaOxs0HUJGM10LRTmLJCdN1+smVeANtcUYVQOAJviB9rWlefChcARjaxN14PpNjKcqppEsgJLsekKlf6HH3/7gxVkROjc2FIduOLi2z1TFexpBzYbeRMXVndliqQBLLQ4ym6Nbt+ePyUpKYeeCxR0iS2AWKBBd/eLkveES46Ww8e9d3MAXzP3NIuexiuEEb1ljGteYUbhU4BVSVUf14YeIoq7QOGmdem48RZlMZXbNwgppNGw25OoscEDkJBjzOsW0jMrN2oiYcUpSMogkZ8dglqZqXwm5DaIFlKQ8C1SzAm3d8SnSmam2koU9p3jXioId2vxwWmIcKOY4g8w8ed/qww++tZYTdnPEkrYJh+NZJ6iCyKdc8ixe4OIl64aYY44xxmWMaR1MShzqT06zyISP1JTtWMtzLXIurAoLU2Xqwkv3RF1eGsW5rWhdclZKNDCattc2wrjGJC+wuKMoVIr/37V8++XX1Rco3VVpcUXSzxzspWbjMbmBmWGhmnbzpcLBp1iqSe+ivnLbRwLjwih7fAsZxumnaRb6pOIFi5CazwGOFFWsGsWXDk+njQNqRT1aMTyS3IIXTgoXYYMktkabnngNoVBQRuae8JCXL39rdXMxga7Q0jOHL7dJ4MimowwA6ZBcXX4XsSIhy91hfSyGE3ETRTesTFAKReWUbAJWZVIDYSGHFz3Q1Wj2FUhlsLrC5pL94W0L7DAmZQcLHnmfmXCo0gswQAm/iOqDD7++0lwoTUapPgXx7V+P5aQ4tslOo24Q5AdDJPrTJlcXzKA4zeRzPGG2y4lrcfpeGO6+2UBXG2l9CjGFv2Pc/zjzVv8cA5TyvxRDyNu8GrytgKWLIT6PR8GZFAaTPeBswA8+/s31OBxd4kpiNpgZLZ/+MbqqpKWub6vo+yra3n+vW/CibB4sKS00DzAI+LPxIvqbxGm7JZO0MZVOcAmBlOELSJpbmDjbZQyUruINtA65jO1UAgA9NqyWXxLSp13r6sUHv76exmNMs7W8x5Z0ioeAEhVZxcm7x15Ov9/Qd291WHJ9MGDl5N3dhadbRsMjCtV2JVjyMGHU93iRNXsMJdDCk7lveptjOvlDyvMyRjGCk7uBO5UpjFS8RbWWXihNeIAbttU3X/7Kiqtq2oLq6qzSlfGXJlvkvXrsdn94dx292uKd4heCoiKFFKg0WsZXLHXY5bL3iKc/aXUTSmCE7o8Mr0KoyHSIIy5wFGoYu4zgSLZPzUf4ks0alOyk9udukDZdqDCGSQP81otfXgErCwmAlV3fhMJuX6+NPAADdB1xjwhShEq7P6BISTrOg7yppEw2xwaKlueRFrM1p1ifiEhSdqWMA8afkjXEBdIA6hBqoAPRxcYtypGEWE5ZXvcok9kixoRHDS6i+vp7GICTfzSAczEn7zSClekPqOjoq+g3EWQjMTPmcBQGbUwMNyijPObjruP3ztelrjCbNJgVkeK82OxPJgidm6JFBzC6W+5mI6pFZhtA2AQzJYyGJU4nwq8MdtkEZ2SThauo3v7wV1bctqg1RTmVDocis5hSxuKJq25DVQbiziI53FQzQhQuSm+gLiFAsbJmoWIdEILFNEepB0rvIQWax+50qkeWtcxBzPDs1yq9W2QyUmoXdWUtUu17kuE8x+nEAAUd5OxlFDAgbT7aIqqXn/7qOi1Dxj8pxBMW4tU6/SaWsYl1dt5vdfq0iycRHDG4nOAoIGn9Hg5AqYphHJNo/BNQnDW8QM+ieHpDjtacK0zTZvN9a0Wlq8Q1t/02LrbPYtc/j77f6T7j+BCHh7t4eLj3EIZ6lVw3UbBQ8QSJ6uVv/9q6QFtVbEA33Th0N4WNt2kAx2C3qaLuSDtzVPXiYaSs7U1AOBkU3Tb6jhGVXZ4+afYknPAAU4qnOWXCBq1NeuzNnMDKrYlQSvXZ7WV9234XlxdvxrP912LbX+rew/AQd3ev4ubuVdwf7tRTnMUQuUYWYYJ/k4nqxUe/uoLgOk0ATgSnVFRtVEsXy1Qrh9oATbR9HWvlmpwTdMOTv5n+MlSFAVQwZc+vME2ETkTUvt1Ek4azS+PC9AzoOzzEMD6ISHEPvEkn+OT0rBSTmfaxaZ9H3+4dfvMUx9N93B9u5QlwGwwgpnn2ptKHiKje/eA3VsAMtAasur6ACXI08YWsVMc4uO3VbujgoMKexL0RJ4Xa8gjz/pKHPadnQ4APXbOL3eYy9rvnsd+/EX23zcEnx69S6rrEw8NtXN9+Ebf3X8bD6S7G6STDmOUlz89+AXWB1R6KIwswZIgybKnKFlBm8zno8FSzrD789K2VJiGb7zdMe3mokNa0XJu5vdMY9/eHOJ6OsdZTrDWbtovykjDRrsoQUoXPkV2mNQkJZow2MsBXn/9SfPWNX4rNZudUqwIH43keYBiOcXPzRfznF/8uQ5yGB9UV3KNpMYCzD39IodPo6TYNuqaXO6Ui7xWxN8fjVC/k8C5h/kff+87KHF3ft+LlbJzXZrORNQGTu9u7uL6+idu7W2lxZABqAEKG+RlOXwbYcOLGilLVEbvCg3Yjg/bNLi53X4s3nv1i7LZ7xbY7R9582/ZyeXnBzZdxf7iJ43AQJphqQxHxPHMWcENlvDrO7jOcJ0ySThcjuJ2W4oy8IaL6u7//q9XuZw9g43gAiHs8ASg3cXNzE69evYrr6+sYppOUWTg84UIFt1SjDLDd1rHdMdiAO9LXN1FB6dl0eEAXdfTRxD42/fPYXzyTF2AAsILuEvcVaEl8YXNMjRLHxgHjw31Ms8PCahM0z+yuTLGYXDnXF7FVYzNpBHEftMaf//xfNB+A47ZgAJMaNRraKOu/ev1FvH79pTZ/f38vygxeUAcAlpz+GmPU7RoXF03s973CCZc8nRBTqfWdEcQk21009UU01SYu92/E1dWb0XWb2G52Mv44InmfvJ6mzw1wHaZDjjLAaThIFue1rACcDUDWghfwvXPFqdE89ADGagFUZxgMyh6rf/u3n61MWgFwyvOa0vDpE4evrv8rbm9fx/3hoL4AtyFcuh62aPcXMLURGzKEtGn4PFOjbqBI+BRR4aQBwqvou31sNpdxuX8z9nu84UpTIotO31xBnSV6jAsTIiD7l3ovTVsMpalV6YylNY5qRLhtz++W7QDGIadc3becMPTP/vUnMoDYVio1uNzhcBt3d6/j7nAdp9PDeeyFlKfhY+oBAKlzU0JxbzEgCx9PjlruLsqPiRGxv9tdRl1to+su49nlL8Qbb/xCXFxcSjQ50+CUxIwJ1/H65t/jcPzSvAClSFOpruwkpKTYWvqY4EknQ+BJTLG6SCuUWLjy03/+wSqr07dbmMHxBCZuBvrSBrfbuBXG7jQl3lex2TYCTjBDIZcxCei1CXq45LxUHmoEP+j8dpzQJmKl1ujjYveVeP7szbi8vMrCydTXcwHQ6IjTcBe3d/8Rh+N/xbw8JElC6CDc6P+5ruCzVKs+lGS1+XCG6wZYoadXFDb/+E9/uUI2cKfTQIydNHwofYBZ3ommhBUaWlZ8FpmLIme328bFBa689QjNfNLJkN9xdVy634D0bTw83MXh4Voh1LTbqKJTiIAVfbeLq8s3Yn8BmyN0cO3BU985I7yuUwzjTUzzdSzrQczVa6pjGml+FC/L0R2M0OKd5iYiaqWfWTrhaAc/+emfr4AJAIMBHCfuCyCTYyVOkYWrH7ecbIC2Vkt8v7+SAcjTLNqtaVxvE9vtRey2V1FVnQxwGu9z+GEjGYz8TcsNq+02+9jvnylbCODGo/BAUxxZBAG2UXN/DE3I0v/z5pG/Cxc4T5WQqjFCS8pjLyn+ZqhImPmbf/jeyukr1eTTGWwewMMAbppRY4OcVHiT+/UtIdDJC+ANZdKMOp9FsTg8oeu2GX+j3HvTX8hAjw0Xh5ZG5bqdsoG1AYObQdEPRZzv3zCTAOfwbKF6l8wPapTfekPREmjY4gn0FNcKwHDp72q4iuqvfvS5VHdPU0ApQxXU8eFBg49Sitg8qmLKzabN5Pw2dlsTHC6maY5mF+tKW8pKBYaQqEL1Jo95rvAwIE05vOhCCBIj/MiJMBgquEJIHo9kAdx/9LMAm23sthfRI04EGPEQp9N9tszzuQcJJflAliQ3SFSpCbKP+Rc//HxlQZvNhfIncU6+v7+/0U3xDlt/dCcWyty3IkK7HdUefL73cwOceHspFRY6y/cIBc0Adp1Od3exj+3mwhUixQ65OCfQy5Clx+UIO2sBHsx6iPvDK4UHJ3+xu9SL8AM3AO3D8Tqm6ZhNVgzsUttjuBZtRd31MshWP/rbP105FVITP3w43Mf1zau4vX1lA4wnkRDchs1zihe7bWy3bIoZHjq4fmZIo66c/mRVmP8wLCe13ZroQHo4NVV+iyu30m3ChU8DbW9nQuORG6os2gY76RE7FVQXz7QGjHQaD3F4eC2WaMbnZwgIBXtmeeIkexloHhjgxz/94brfXUnUYPgJ0nN984U9YDhoApQLctOL3S6urq6E/BpozPk2MIHfgxeeAbQUDnb0Gwx2GbvdLj2hj90OLyBzzCp7OWlKY+JX0vw06lEYStr7+1t5ob2PMp3472J/gQGe65oYko0fjq9FlDQ5kn1KzSTkU2l4ONzA43xO39WP//FHq+JoDRU+t3ewvut44PRJibOBcLvdaPPPrt6IzRY6a7xw7x5ugBZHEzLl9RRgSIUX+72Aku+w+efPvhq73UUKpShHdkdzfww4nD3xy1f/fY79UrLjSZz+bnupECNLkJ2G4VaeIK9JusP6/DwDnSmP3/GdvtuoDql+8Nd/tgKAdIWPp0McH+70LvUmR1Ww2P6C038WFxdXHlwoj7ClgqPmVj7D53xrD8D9+R7WBu2J/6/A+vbkfI/YetLTtYu71FMcDneqQTDA7f1rTYW46uRRPeMJ2OW1eIgKrrCkvOfGC10n5ovJchA6g66JGhXvLqrv/9l39egsQ43EO/nXoFdQnAeQ+ri42OnUiGFPiDxOb6q3SPWW2URSt1IbxdE+nj//ioDreDITvLp6HlewvpbZHRcphcqWp0IB4uvrV3Fz+1reiAHWalD7y6pVTqZKGHE8e5jSjU8/OLWLtusF4ISGvcPYRGaCllef/8GHem6wzAUYQNzMMBMD9T0LwMVUt/Nl6X0ecfdA5KxyV6OwOdDMIq8un8kAnMbDwZ5FZVhwRAg/Hg1YWQZQQB0A4+tXcXd/k1riEIuIkL1AjZInnWbcGy3ScpzVanAGMgb/GMZDHIf7GKggnxxW9Z3vvvCTo/kQsquw8lCBR8rdC/Sp60EGla8XKmj4e6nL5R1Vcy47iVVO+/LymQDz/h46fK8F6FlhTYa75vepGa1ZwMPDIV6//kIGUNW3jjIAapTaZGUirDRyADpKYU2W+pHe3aaPbeobCCkA7jChNTK8kf3P3/nsHT82l7lHRUgZW1GjMedvclSWcGDzFzuywZVCwq7vxqlZ2UkRTchcXhqpAcfb2xvFNl6gmQD6CPkoHEBIbOMdGIA54pvbV3E43EiEYfML80FQYPUjSq/QPN+9ZGaDeFV6ahSStlO6huShGnm6TK07eXpE9dl3/dxgMsOc0HgsScuISnkYkXQG+m7z9YgJrtrK8zwYcb8n/UGMFp3+7e21ThaXl04gPZD0ae8HKJWaIEkTD1fex+l00KJp3qBe8yyPBiEZlaGniAgLtS3PBqlgrRQO+91OqbvvKNUJU6vMEkOywv0fU8WVyfIyomIAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAAQAAAAEAIBgAAAKppcd4AAAABc1JHQgCuzhzpAAAcBUlEQVR4Xl2ba7MkaXWdd97rdi7NzETY/87+ALZkQB7PCGERKMLWB4essGwMIwGy5LAAoUGMMVh/ycL09HSfW90yKzMdz1rvW1W4h+KcPl0nK999WXvttXcWP/zOv5/Xy0UsF220TRNlWcU8zzGOU0zTHFEUMUcR/WmO4zDGOBVRlFUU/HyaYh77KOZT1FWh3x3niNNpjNPpFNN4inEaYxzn4FJcqyh5X+j9bV1GV9f6t8PxFKdxirIooqqKiHnSfRTB30v9PIJ7maKYuZhf+m+OmKLUfXJfvFXXKSOqsoyq4n7LGKc5htMUp4nz8XtzFP/1P357Xi+7KwOUurZuOCJKLlA3Mc5FHPpT9ANG4Yb4QAwwRBljNBU3Wut9HAQDnIY+xnGwMed0czJAMkIZ0dY2Zj9MMZzGKIuIquDmJmygf+MedH7+cNOFbJkMYAfx4r7y+3UdHd4vjMfh+2HUi3vUdf7kj//tvOyaWHRNtE2lX+LNOnzB4euom1ZePw6TXnxQVVaB3efxJAPgTd47zaUsbAMcZQRHUmnvJw/Zg1PU3GhVxThFjOMYRfKlDCAnlDKYbonLyCm8fI8+fBmzDs97bSwih6+cp6wKXYuDE2l8nWYMUETxx3/0jZmD44mmLqOuynR8wrWMum6iaVtFAWc/pgho6iaI1CLGZIDKB5kLHQYDDMMxxqHXUcuqTqF4Ce9pOkUxT/KQUgrvTmPEPOp7Z2COgOwURxDGxDgyQFE6vTCCgySmlCZOBe5pjn44KYqJBDu4iOKPvvX7MwdvKg7vN9u4zjluvG6aqJtOH8TvzlFFXdf6nbYKv9raeTbjSUcABpjGIaVSE2XNexJ2zFNM4xjzZPxQnnLz08kvu18etVe5L39fEtKk4FzISLNSwofn4BmDuIYOWRYxTYS/0wwHCZIw3x9+EwNwoDJqwCkZwBfjXWUUVRMVAFk3EYUBpawqGWBRF7Hqqlh0ndIEA5HzIwB46iPwsoCUuCVtfDPzNMY0jUqhui6CiCJk+bmSnxAtcUiVDODDK1q4TuHDk178Z88DcmMcj30MJwxpvOC4RMBp5KsjhPeSZsUffOP356apoq542QhEgW4mhSGHLptWUYAR7BG/t6uKWC/qWC0xQB2nyQfkcHgffOC6GIDoAAypHuTgNGKIwYava1UR5y93OCnUbQCHN0YnUogA/o639VmjjQmG9P0Qx36IYcjgy2fYAHo/4YJzcQIG+Pjjj+emaYIowLrcLBGWMOZcYjh43XbRYIQUrrynqSKWLRHQ6gCUI9lcXh6CnzTn9EghK+un94ynqEpwwBiBcVXeJleEWlXCKcC/101+H9eaY6TkDn0MvE4nG+DYK99JQ0WjHJJB1VhjkJ2j+Oijj2asrzyXhTFAKlUJChRgZR1128oA3KzrrWstJZAqwI2qRJaVABIwK4tRxrVxcsnKJQ1XDFEXsyKAVMMASo9x0LX5ufAoGaBp6miIFiG7seZ42MfxcEje7+N4HKIfBuf7SM47ClJldw7kGvLRxx/NsiwewBAYQdFQnUuWOUyp8KcqEAFnQEqlTZUqgSfAg+eEJ+WcgFW1S0aTXfV1jnIao+Y9YEzV6ucK5+Go3+NeinAJJR1qKpaiABCcFer73S52u20cj8c4HPs49o4Ae580STihc/vwvtWCCPhw5uAcOIehSh8RIUs7L/1+AxA3k29KeJCs6WubYdVlGW17SS2TnwRifHBGYaKEulK3MrDL4Rjj6RRlgfH4vIT+JWBZBVEAW8SrfX+Ml5cXvQ4HDHCU9wE8ip0Or3zL7MaO0h3wvw8//PqcD56NQDo0TRvdYhFt29kLZyrmQpuZoq1DzqY8SwBHFBCqGAGKrbKJoYmuTIgEdkZ9AaBS64IBJj02GtfDiAJqpZu5xuFwiOfn59hutzo8AAjLyxReQHsu6jZ88pid+vWv/Y4MoLAmd0UdGx28Wy5jtVrHYrEQ+AiNE4IaWGxdoz6lj9w1Iiu8IVJwhaaJrut0HRvDjFOeSCVP9Tzluu/Rpc2VIEWAmB3Yw2eeBHb7/S62u13s9weFP953z0HJdbpnFp1JksLB/4viq7/7ldlk4xJqhJyM0HWxXm9ic3MTq9VK3jRh8aHJLzVO6fACHL2owQSgDwHRWS66WMuYrSOh8g0qmtK1FJg50tKdZyanFFKTw2eOQn1Cfr/fx47Dp/BX/Ve0ZP4gUnz+LFeAVIWoAl/76ldoUxIqnpPdfL+qlQabzU3c3NzEerOJ5XIRLYid6iRRoHIkzxMF5gCUmVQPdeNdW8eiJRKcEuQyZ8R4A9x8ImJc4wFYGU/AWihK3OERAeQ1DU0fu91eoY/3DXxwADpQSBTkLjnMWfpb0UpqKNI+/Ne/K3M4lH0QDiEHEMINRljGer2WEW5ubmOzXkfXtalEZcIEDuT2NLerl+YF1ghwURlAcQzAHzxG3vJOpUfiArqSjGuigwHMUeD1YxwPx3hO4Af6qxz2p9iLBY7qXVr1MPWZJRKVOt/JzlLE/cHHX5/NqHjRx0MsYFGjc0SlBzDrYrlcxeZmE7e3t3GzcVqAyL9d2q5KTQKuStzALbC96J6fvh4D8JnCipYybCLF51Pi+DfwwSDoMkqK7Xa7eH5+0de+7+M0jjLk4TiIcVLFSGF1raLYOBd6zsvNliLg23/48ZxzN4ewyQONij2K+QVoWLVb6OBEwu3dnSIDz5lL+KCZf7uT8+8S1jmk1cMlji7KPFHujBUmglOcRro2btb5SufHH5xDvu/2e1UAe9+khwjoTwBwoQ7WmOUI4Gz8Lu+1AZLe8O++/Y3Zqg0IDkdPcJCaCysu6SVUdm513SLWYMPtnSpF25HXDu2qxGjOsYzgIlJZsUlqThQAKIbOVYPuDSxI9zDzTRUz/cU46uDbLaRnH8f+aLUJ+jvwgv3huFmpK3pP/5JSjfcNvSnzaTi5V+G93/rmhyRbKmlJ7EhNR1YhsFZOE/HnJJS07SK6bqlyuVjwWkS3wOrkbh/zDM8voyrwAs0TmgN0N5WyILxP55KpKMkyVvodAJrc5fCE/Mt2F/vDUQfgOsr9I6HvNBAV5/4p6WqwULigzEP0xxQxpHiqYMXvffUrqojX/betl375EhCpO3RIZq5AiBHPqEbL1SqWq2VU9Ryn6SA5TB2XKBn9Qhe3a1cSKG1BpPDCPHSclOJEk2WMOZe+KfaHQzw+PsfT81apIbCsCvH/XAUsc1kaU36n1MNYeP9Az6CUMQbok7/y5X82q2xJfiEEyXW3xxYTkzSSAMi9PaBUqToQBSJQdR2L5VJN1TAe4zjs3Y0BYnhmmqNr2ri/u48v3d/Her0M2nB5SICQJK3UZCllrpotjPm83cbT07PALoufAyqPOMBBaZD7nHO7I4GGLvF4bphU5XLh//KX//lsgYIf0q9dI67fJa/k3jw1QliXyrBabcwWOXzdqB4/vzzH/ngw4lJu5lnG7FqI1TruANCbtchRbmxyn3ERQExgrOthI0oYALgXBuB5Eyg4gaOAakDZtOCROYnfQ97DHfh6ZrGA8b/8nS/PuTeW4uomyTUyg1UKS0VGIhdmWo4CiBIsjw8nTwEqdWOp7nJR0so4sTJWdG0s2lpGQBo3x8+9/kUDyG039JcSahZ4jOPRVQD0xwBEwXAa7MgrdprPliscOEFEZNms+Fdf/ReKhsxATTcTJc3SkcoWjUil8sJh1D8UdGaN2CK/Q35BUAhLbkxlDD5xFjOMzH6ZEcoYfM+MoK31vbtQ2Kbrfk4H/92V4nSCBu/0mYAijRARB1ib1LmynbsfNXCTUhJDDT1Vo4/iw3/zNe4vqa95oHDp203Xcy0Oawa0y0ktbptW/348HlRiZnVipdgkzUpG5jMnyLJWmjdk9kd0LdsmlguGNItoqOG5P8lteJLWrVdC2o5KCfqBzAZzyT5LbgngFfbyPLhkyYyUKb717W/OrtVJEU7f5zTIBhBNVnhlJQjvI4OV0fc0IzuXJnnb0WFKnRoToba7TbM5vJPoLx5X+1ypa1REtK21v9ynZNlbAxkTGzABcJMcltir+AzeTkqQ2S0HhwANnlYlQygC/uRP/4MiIA8scvjnqHA+pT5BVsx6vVFaINQf5A1LYIAWqg36IUSEl1OGCuGy6Q5Qn5u6NqUeUXFWjRI1V2eZy3RCJZBdBIhDQWrcgDm/swzm6sPfObwPbtKVO1ZwoPjOd/9Lao7dwOQ6nMHnAiiOAE15UHaRmsUcVeR1YYYkGawIaYwgQaViuLKI5epWue9u0bhyrbNYaMljsSmO/S7222d52ZUqKbt4P+l915qEO1HfkwA4lWGc5BmEKX6+luT0737yHY0WLRtlvSzfmEdKvqnUUoqqJw2AEpfxI7WrlCtACpIjbyrXu1iu7uL29n1FAyGZ291LiiUpPfUfzBR223exfXmIAQPkw0nbTwZIZTbfulPjAoD2/iVCfA2aPA9dlWLf/eTPTIWlzCR2pPbJpZAczkWRw7hT8+DjLJbmhke6J+zOkeBDUjmWcXPzQazXr3S9c9kVwNr4OYzdBo+x2z7E7uVd9P0uxpTfSr+kOQBgILnkL0hUqhqc4RLiTKiT15NShTPFdVR2qyi+98mfztc6uU+bVV9aXauyeaKTL6j+LI+tFQVEA0KEv5Ln1vLbWCzuYrN5P9p2HWrD08HzcOL686nzh/1TbF/exvHwIu/roLlFp6ujlEn3d0Mza1zm0T6nO0+eUs1XG6wUgOyZT2iyhQF+8IP/NJs9XTo+NICqRLrqoq6dw0Zu1JtegMIHGdHxM6SHIeggNsnhnRqg+iY2q/ei6+4iitoRdZbBLIhmxR587/ttvLy8icP+JY69yc5+y9djTGnclaNGacnncB+a5kAZnaISPDBOHsEp9D0UzANXUf3//j++czZAHhnSgkJEmnoRTbOIumo9jFT4uaSoyVF2UHYGv8K5nckH3l8u7mO9ej/KamHvC0C5N+ej1SgoOH8/xmH/EIcDwAe3t+j5+PgS25edaKwU5GR8CSR0lohLpbYVzF2TMxFcnMDWJ89KKDGt4U0VxY9++t2U+RcAVE4pCqCmhFbW6z28VC+d5OwpQHSsa3DJf8wSV7Favhdte4c0au+f9ckLY2MaPI6H6I+PcTw+i+qeJosXMLynx5d4fnpRrcfoGuNDm9VWS1qO03SKfgQTLHZ4amUJLYsvzj3vPXiwU0Xxtz//Xrprx1OGPEnLV+RDH3Q1+Li00JcQvkQQbWwbXXcbi+5VlNXy3O2le0iexwgcHkaH5x/jdNrHyOFHVB7P+az9bRUBouR1ktis9p1HZL2UIVAfUcVzCY3lNEq/ck4euePoT3/xyflfsjasvPL/2Sia5KRJSnKxr5fE9fPAKfeOAMwquu4+2uYmoiCCEr1O7bREFtVnWORj7A8PMQzbmKY+TjM8fS8NQCqQRl97AZ9b8bw35INhGL2SMi29QjOJPOK7bI1keS+34cXPf/XnStrz+OscxPmb/08yvzLAlVETsrtxqatOoAf6l2UnWUuNTIovUsUUlebpIXb7B4Efh5+ij2E6xKF/ju1hG4dDr9YXjdAdXN4d8qgOb6siaBaYGKMqzUWPvMhyF2d6XaeM4n/+w/c9F0gjrnSSFO0GrLOjrR2dSZFtkQ6WLEhetc06VityfxNF4eluVo5zzT+NffSHp9jt3sXxuIvTqVclmeZD9OM29v1zHE87pYLCF5BjhgADPFm7FEkbp+j7XBKz/p92LPi1FO55zyCnea4ExS/+4YfnFPjtw+bpSULVdPCrybJnBwITK6+57i+X98GrrhfmETnszywNfQ5J+10c9y+a74P4/YCosYtDv43jsI2ohqiamcm8lkumAr6fPe0FB0qzdwEcGT4DeMWUmXu38a1K5/baQKgK8tn//sE5nV1EU8G4WmK4psJmh661Bpa8xeHpctuuxPi6xU2qHmaPeX7ITG8Y9nHYPcR+9xjHw04qkpUeBE/ElIOqQFVP0S0rveqOeo8BGHwgaqIk0/9f83u5IRG3vENkY5h/ONWFaDkF/v6X378CyGSAK/4vppXYmywrlfaymsLhNPlt2+jaZXSLTSyWt1FXizSgTJVYXRjUlJx+juPhOQ57xloGuN1+K62fxmec3dLCaxbLJparKqr2FFVziqkA6fsYRjo8iBfcA3DMVYwSR9rxqryblOtc2g8y202S26e/+AtHjSzkftD83572TC1VgXMZTB2gtrCSARoGEQw/b0V6tEx1xpZMTjAAWiGkhmnuIZ6eHuPx6TFeGG8fDkJyb3zZS8wSUZmL+hDNYoy6HWMKlzvjA8yTrTLPFqDt8JaqRKxl4SJtn6WBrpulZBGi4aef/fmcG57QIMJrLpn/58Lmrz547uQkoqTVNVhf06wFfJOGGZkWeY7gWQCM7yQiwx+8/sUXb+KLd1/IEC51nhNCxBoNZ1vNGKbYRbsYo+mmmEtSgPC/Ynii4G6MqEIqw+2N2KzX5JDprBtqezWx0OInP/skS6GpTGU5LKuyOW9EYD3EyKOytFViYyGSdnrl7S3vaDAZ7jSng5gghq6WK4Xg88tTvP78/8abN7+JL96+8ZqLxE3YHGoRZKoVMRqnnVKgqtEYzTwt/ycsgh7X3jqlh1mkHmS9utdeE9c49lsBLBhDGmGI4iefYgD/yTpw/v48nk6M60xzNCdkuZpmqU2DTM/xCDnz8UndIkrw7e295HPGafwdpYju7Pn5MX7z+h/j9et/lBG2GCB1fyC4ZpEtEUC4UyLdc3AY0d3SM0ZXI/aJCg1YkdWWi3VsVvdxe/NerNe3UdWlrtEPuzhiACrO6RDF334KEbpMhq5KwhU1viwXWh0m39n2oMwValW1FSqZC/Bxu8kQ9f7V+/Hq1XuxXt3o4HiICGGziwj4/PNfx29e/zrevv08tjv4Pp6xpA6naGoaMS9O4bljz83nEDYNv3R3zATpQGnC2G7ZxO3mVdzc3KW5hQc9VJjhdIjjsIvi735GFfBgJEvK1yqwyEZuLzUSo9Qt5U3XVvM76W7SBC1+Eh33r96L9z/4J3F3e+/3sxWSV9uQs8GAt5/H69/8Ot4+vInd3gag1iN4ECXq2tIKP1yfLjG/57yIkTZCFP6M2Zk5dF2sluvYrG9js7k9r/rknSdSCLpd/N2n39d4XDp63qdLahCAkiWmPBSl5OF97fVJ1bXoqRKX1A4wgl0CDs9rvdqcFxWs4XmdHikd8HsHCD6/EwfgZ1QH7f3sUJqNOwCeJG11jqQEX70NqohLO8TGAE+hMMB6fRObNL1igrVcMs7z4heVo/jRT743WzFJuqBoK6FiJUhRIcS0+CAATEoQB2fSQ2hLKwCZp1FhvtncxQcf/NP40pc+0ARZbDD16dwwKHxQ3XdIq/sb6Pye4vn5SaUR4xyOe6dEkrKzo4g4fkfy+lUaeITnXYauXWhixcvGsEEwAnMH6Ql//Td/Nue66HBz/28y4bm8vWtVNUtKHEirdB2g1thLyVuE/2p1E3f378f9/ZdkJD7MN2tv8X4YIPjhzTSeGRi184MRHh/fKTIen2mTtw77NGmCK2A0ITkVIy1c5M3wy05irVRcdAxbVkqDzWajxS8Zge2Wv/7xf57dwdE7d9FU7AK38j4XNyqj+zN386jpvLOXUNprtgY/bYXVyOGA0E3c3r2SB7xwmXh5aVnd4DkonSiT4ANs8OXlOR6fHuLh4W08Pr0TNhApOEJjN0112AnEKGCPq4G3VYnQtJUisZZJN2M4OlSqgw2xWq80gSr+5tNP5vMbALeGKW8n+gs/B5nh65bBvPundRgNSj3MJNwwgBsiws/lEeC7ub2PzXqT5ny5hb1sdxNZRBA3hfkohe/evVMEPD0/CCh1YFQijbTYBWYWmIejGCBtfKUZJwGm5crzhqnX/jTOE2OlHC8UBcWP//77M6Vm0S1jvWTKy/CiEybstrvYbp/F10UaOJyWJ3yxvH2Rt7wxkPd5IT2s16MIMwnurrxfng+i98MV4AfLtcrd9uUlHh4fhAPwBLyvfNckqNeWByCpfYDeRjmLukn38zjfRtBmGXvQ2mHwfeMcDWfbNoofffrJTB5Tp29vXsVmfadUQFxgA3unCexB4UkYy9taPTOo8SFcNHd7GAOUxdJ+jx+JyZHCNbSwkJab+Dm7BctsgO0u7f0Sfc/x8kIU0DF6/k9K8rvkv4ceuRtMi11ZEP0tIzDQJcUv02ltyCOL//hn35v58M3qLm5uyNdbGYDOj3zD6n7ywmqPWts0foYH4GEO4amrNTs9TyTCY36hxes09uYaeA2j4kHwZLVGQNkoAna7g6Qwb3UcYrd7icent/EsMKRXSO2wxM88UHFjlJ8PyEMeN31pxxicIfwZvjLib72yW/z0s0/mxYIyQb28i+XyxgyPRx3EaX0BdQJXMzktLgoEkbyIFpYWDjIQWhzliMOTJqSAdgrSNrhW4ER0TooOb5isxDmY9XMQj7s8eH338CbevPl1vLw8eS5x9ZiMZ/6uCC6N3hQ7l9xE9LkP7TKwA40ReE6ybaL49H/9xdw15CmoeBPL7kYIDt31gxF+kNJLi6nuppUXchwjUCoJU9pZjIGl4QoCzXO77IZIBiu8t8MfbZRr67SR9sfer1Vdl0qkMiLg9ev/Ew8Pb5QG+XE+l1ZHiifUNoDJ1rV8ZxJjI0Dk3GR1iy4KBBFxexlhLQMsFhv9HakLr3vzw1tfWqNR41ELLHMzRHiSAjAsWCDWzTN86YRYfkGuU2WYEaSuMqG1lyx6GYBeX4I3tHkcBMREwNu3rzUqQ3yhqeI6HHq3A6i3MpbHbXnXPHd5WetImy6aXLP02UTx81/9cGZCQunqWvZ3iIKNZnpUF/KUw/EQpA/vRoeyAtrzNeciqQImQDY4sJTcoXdvgAFkhPzITepA1eh43qfn+ihxLDzRFEn0gDLv5f2Hh8+j7/emuWyo8hmnQamx3T6Jr4iya8eAOeHlSdKzlJdKpvaWWdX77Fd/OSukRBYQEdbR1MtAHEFq1sOPJ+Z+l6GiKoG8zwsWaLAhtDg8NwcO5CUpUVOBoIHn8pSJu0wY4iE95sIgdJ/WXiSPacTdx8vLYzw+fhHH41aCymLJ2K6V8dARti9PSoNcihntmXy5DGrnCGarOahpP5Ps4rNf/qWYIBQYJUVaXjTn5WKATEsP6cmNRuLG0q92kTo1gx3hDd8GZPi7gOi8dpdX4a+eCbh60BH80AMP7PtijLT6Yg7AGtxWDRNzw3keomk8HlPqsDusZ4bAD8ru5RkoP7HqEV9+2CvrFWqGPvvlfwPn1ceXQe2uk4qKV9PBteTg/Z3lCh6N9geXpgIY0AA9+n/6cPGE88Kv6XGmwXk+kLIzPel50p4f+Q8Q5gaH61L3jUHmBfs9s8OtJtEsV3O9gTV5nhlIa/OuUK77ADXVRymbDKHIoEphgJ//4q9mSp4PTr9uAMnjJ1vT63F6cCKpO15F9zYYeQqa5y5L211Xuy+X7y/jtaTEprneqNq/vdIE/VSYARjvkhYsYjE2JwqG004aIPdJlF6eF2R874cuMYBIGe07FYsoSFtuuZ3/fztyxG2zVmGQAAAAAElFTkSuQmCCiVBORw0KGgoAAAANSUhEUgAAAEAAAABACAYAAACqaXHeAAAAAXNSR0IArs4c6QAAHK1JREFUeF69m9mPJdlx3r/cbubda+ueISkZ/mv9IL9ZgCVCpuFFlsThjAWa0EaABmgLerEkPxh6M2c4w+7a6+4398X4xcmsqu7pnp6RBBd5UT1Vt26eEyfiiy++iOP98q9/3kXRSGEUie/xKLFXEIQKgkBR6H7ne746dWqaWmWRK8sPaupC8SjSdLbQZLxQGI7k+b7U6Y2vrpO6rlPbtmqaxr6XZaX9bq+7uzsdDgeVZam6ruz3WZ7reDyqKDIVRa6qKtTUtdquURhIYRiIx4SBpziJlMSxrVHy1NgzavusuqnVtu55PN/zPPmeZ+9ze+nk/fdf/mUXjUYKw0hhENr30SjWyDYe2r+jKHYG6Fr7UBaWHneqq8wWsJifaDo9URQl8rzgcbNYwbOHyRZR17XbSNupKEttt1s93D+oKAr7fVU5A+R5pjTLVFWlyjJ3xsEITaUg6BQGvnzfUxj6SsaxxkmicDSSJ//RAFWNQUtbL5/Zta0ZgO3zxaFgLO/nv/iZeUAQRHbigR8IgyTxWEkyNutiAE62qRuVLKrIdDxuVZZHJaNQi+WZ5oszxaOJGYATqKraHuT7ziBsnI3wYrNFUWqz2Wq326lpWvkcqSdxQFmaKs1SM1hZFsrzXEWRqqpy+V5rBggCX6N4pMkk0TgZaxQn9qy268xwRZmZ8fi384TWdv1kgE5N28n7i5//aTcauT/GPiwkiiKNk4mm07mS8USjaGQ/r+tGecFicmXpTnm+UxR6WiyWOjl5ofF4YZ/D5vMsswdg1LZhw2wkU57lqjnlotTxcLTN8YULj8djRVGow2FvIcDJ2WaKQsfjXkV+kOe1ikK3+elkqiRJFMexEvvb2Fyb9aXpXlmeqqwwQmWh0LUWi3YgfGEs72d//pMuHo0t5vkxMYJHjMcTTSYzM8RohIcEvesWzgDZXnm2lefVmk2nWi5faDY7VRjGtuE0Te205flmkCObOhyV5ZxMpbIo7ecsgmPByMuTpUajSNvtRvv9/hET8JDDYac828v3WyVJZJufzua2bow2nc6UJBPbWJ6nOhx3OqZ75QWhVJg3EQa2efd/CwPvT3/6XzpiNwxDi1g2ikHGk6nbfJwoCkMF9nvZB4EBPKQodlJXKI4jzWcnms/PlSQzeV6oIi8MzMxrsvxxU/wcw4ABPI/nEpvRKDJPwrUfVnfabTfOgLzL85SmR1VVplHkaTKJNZ3MFPcb5jPm86X9jFBlfRjscNwqzY8uFMAesMAh8iMOeD/+yY860BvX5RUGkZLxVLPZUpPJ1LyBBwS+ywK4Ew/gQ5sG9y0VBp1hxnx+ptnsTFE0Vtt0yvPCQuZ4TLVZr7Xb7c2d3StX4IeKk7EZl5OfTCYGXLe319psVhb/htyEX1VJajROQjMALg9usSG+z2dzzeZkosjWyOnvD1sLBfOCulLb1Pb+Ry8gBP7jf/7dDvT3SXsYIBzZKc7mJ2YE4tIwIPB7hE6VpXtVdeFSUUjYNBaXs+mJFssXSuK55AV2+mwiTTOt12utV2tLeVmWqiwKW+xkOjMvI8x4Ees3N6+13a7NyICiy0Cgv6fxONZ0OtZkPOYX9gzP8y0EFvOlfRaAl2ZH7fZrM0JeAKiAr8sGZCHwwDDgD/7d73S+71w8IhWGseIYAFyYEQwIOaUgUFNXStODDoeNc8dRqCSJFQStoqDTdLrUYvGRkvFSfKalmrpRURYW0w8PD7q/v9duu1VVFgojUBxXJuOQQj3td1s9PNzocCDL5JZ62SCfhZGnk7EWy4Vms5nxAoAV44zHUzOA4QAhkx203tzb5xRlqrqp1HWOE4BR4AqG8P7t7/+rjpN3YMIpgKpYeK4pYYAB4rEtAjDBpUiBdZ0pHuGOgKSnJA4MBOfzl4qThbklMT7k/yzLLQyuri51f39nZIrnEW72vOnMPGy3xUu2ynMyRGanxoZIo6S/+XyuFy9emhEw0PFIGm0Ux8lj2GIQ4n+7e7BQKKvMOIQ6QNBtnoOpm7Y3gOV/SBCxmJg1MQCbH49nCoKRy915pixjYSA0HuCbK06miRaziRbLC81mLzQaTS2kjHHhBUZuCgPCq8tXur6+tM3BMjEA3jafL4xjbDZrFXlqIIu3kQY5KZjdKIp0cXGh733/tzSbTc21j4ed5XkzwHRuHAa+YABoqfqoogKvMEAD9psHEDoYwvuDH/3rjljExX0PDMCtXQiweQzSdb5D7hxWlqvIj2aAMJK55HI519nZhU5OXmoyPVUUJvLgFZ4zgCM+hbabja6uMMBry/N4VZJMdXJyoZPTMzud9XplYcDJguQ8D95AzHL6P/jBb+ujjz62kCmrwg6EzydM4yRW05TKs4PyAiqd2ncH2IAoRGig5ISA5P2nP/w3XQTdhcd7vr0JI7B5DEEsG6JnpC/yqcsCZIAo8jWbTXR+dq4XL79nBgBAw6CvCXreNdDc4/Gg25trXV5+ZWkOojQaTbRcnuni4oWd3mG/13r9oPX6Xvv9yhAcd5V8nZ+90G//i3+pkxNod2QYgGEx9HicGFBDltJsrwomWFFLHI0MwU4dAXBZwJFiX96ffPJ73SgeW+73g6hHyto8wjHESE3dGrPjYbizA5TaCMliMdfF+Qu9fPF9CwH3N+R23xD8sRAy9lfYxi5ff2UnTTEShBg/1snJqaazmS0O1wcIH+6vzRNIa3jh2dlLff/7PzDcMf7ge7Ye1kq2ojYA8BxNT9U0hRmBkID3D0WalUMezNeX98lnP+xGUaIEd4/GtgDArm2dEYh/DAARwQPaHkjC0NN0MtHp6bmdDIubThaG7Ear+6rLFR7O7fCe3W6ru9tr7fdbM0DTesYKcWGYncsGnXa7la4uv9RqdWtptGlkIHd+/sKeG0aBgaOlxygyDAGP6rrQMXUgzSHx303L6fuuMDMnAAytUpH3yac/7NgkqY94BLwgDFRTMCdX3DQGgOR+qTW2RgW2WJzo4vylTk9fajZ1OZhsQt6243/25dAXMMwttrMsU1lUyvJC6fFoRrNsMCEljgzEXv3mV7q6fqXdFtBtrTKdzRaaz6gBYnUe+NIpGiVaLmCi8I9WZXFQJzbdWpjwRrISz3D1hSuvzQA//vSHHW7O6UMt8QbeyGmBxA6FXVECwASBNE5inSyXOj15YXE/X5yaAQFTWNtw+oMNnLXxAmpw91mAapY5Voh3sUFcFFoNmB32a7169bmub15rs92pKmtjo3jIfD7TbDqzGqJuKY4gRwvNFwvTJ+SxOXCj7esbCrzEDsbty2GCheonn/37ztIfBojJ6a4wYtOH497yPjS0qSlsGo1GgRazqc7PLnR+/rEWixcWPk902gkObvOPxadzOUKhDwfimtSIQTDGYX8UIInYQXGz3T7o9u61Vut7K5l5PwgOa51Op+byGJqQBLvGifvZZMpBwD+oAB3ys1Er6QWXqewA8Gz26f30Zz/ueChxBAs0V/GoqwE+AAX+DhIXJkZMJ7jbQudnL3V+8T3N5mePRnPA9/bmn+LAeDj/6z0BUCQfY+D1amVMEaGjaSujwlbSZlSQR2OOg7Lj+MNYka0b7jI2fIAOsxcisG1KNa1jknhlGMQGzDUVaAtu9DrHz3/xF2YAAG8Aq+G0sJSr+lLzgCiUZtOxKUBn5x+bAaC/MLrnwOdC4N1fQzHiUqMTScADGOLq/t5iNsv3OqZHIyzUG4QeBrAMhNQF/w98J8dN5zo5OdNicdZrFqhGHEKrqgb9C3N9wgQs4Jnsj2xF9vP++m/+Z0dOxUpUTJAG3N9ZqzTtrypT1DbFUWBIi7XPzr+n07OPjS+4ktYh/5uu/3UjPBcjAFm0ARjir7/4XPd3N4baZZWqqtH/RhpPxqpJYzXuTDlOpXdUWZeKRyPHIV58bKU7lSdrRiyBNtcN2YOizVdkbFaqykroIgAn2qf3v/7ubw0DyKkwK9yetEF9gGum2VZNldrpT8axJuOJxuOlTk4/1nJ5YbjhwyI5Kov995/+YI7BC8jNpECIzxdf/F/d3lyqLDM7PUIR1+ZwqOCcvEYM59rsHowG4wHz+akZgXiGXsMaqV6D0Lcq1fNbA08TfFpZ5mla2efjud7/+Yd/6FizpboStffYpwgKEIBqr67NlMS+gR/pbjo712x+ocl08cz9HfF5Ar73RsFjTQ4XB5Hv7m701Ze/0sPDrYmfYA1xHlickqLRJWIDtP3hQfcPl6qqoxJkMaPscyNs8BTSGxjmhxC50LIKRgQ8if2yRJiFAwcKwIvPf/3rjhNh8wAeoGMiYsPpoKfh/qXGCeg/02JxrsXipYEfpx+Q+vq8/21OfyBGJpQinWWZbm+vdPnq11YDWPXXUtxggL5Ymp3acxBjb25/o4eHSzVNZtoAvCCJp5bT215W78j7YWQpE3KFaEqKdiSvcdTaQ1kO5L2+ujKFjPhBjAT0LFWZrod4geyVaxz7RkAWs1MtluT+F4qTqSM+z3L/+8/9rWxgBnA9AMrk25vX2m0f1LVQ28jimNDCywgFwmW/X+vm5itttneq69zoL4BsmqbvqWtrtT3B8U3YmWg2X5qRAD28k7BGHhv6BN7V9U03AKATEHuwqRrt9httt3dqqoNGI2k+SeyBy+VLLU4+UpLMzdIO/Fz+/zZfjxjQGwDDr+5vtN8+mLuCzpwQhxKwkbhXi49bXd9Aj28sNcMMod/QaEDP62qBRPACwmYy60Wdydw8gMxDyIEVZDgL2FeXV5YFjFTQRYEn07goSm13az08XCvPN4r8RrMpHOBUy5OXWi4/6j2Av313/Fsx5EjgW7TYVWQYm9pgs9lovb5Tmad9ERaramqtN2uj1GenF6b5UejcP7zS/f21VYmEHtVkAhB7kCQpiRNNJgtNZ0h6J05yAz88T1VZGtlyVDx1atOXv3ndEW+IDZAQ1jpw9u1ua9Y+HB7kq9R8OtbpybmWJx+ZAuwywFD5vXn6z5NBL8I+GmHgG3CA1ereSmQAkFOlEAKwyqoyJopXLZenRn39QEpzDHZrjQ/yehiiaI9M7YHPzKYL61HM5yeaTOZWX5DlDOeKXPvdzg42Pe4N67zXl9cdkjQGoMixk7Fcm2qzXVsIZOnGUsqM6u/kXCcnH5n8FRltfrPyM7d6tAX/eJKgn6dB1x8sTAG+fP2llcmkYTaCR+GNECSMR+FFvR+NQpV1pqyAGtPx4b2kYCfoGkOMnaLN6WMMGKNlAAyQ59rtNqY4YwDSvbfabHpNkOjpTx9LHXbabB50PK6ttIwCxI+5TpcXpvxiXcf/n4qfYYNfT4e4/Jsg6AyQWwag7OW0KVAsC/CFDEbBYqpRrFEEsamVFamKGq5SWrjCEDyFRnUhO5AnKD3AOUFvTBIzjjFPC4G9hQDewM+8NMs6AzCkq45Fldof9lpvVtofVqqrVIHfmea2mJ/pZHluFn5sp/X8//2bd78ZGODwb0eFC61X99ps7npQwkqUrk5PHPp5hKjv02c4arvfKC22KptUdUuFitF8hf5YyWihxPqTzoOoO8jQ9uyWLNH3DXPXLrOnlWXZU3+ER9f7I/Y3m3tLgVBga1qMZ1ouz60DRFy59Pd19H8/F3jygicMKJSlRwOjoTHK4gdeYdmi7Runch5zTDfap/fapysdsq2KKlVdNfK6kSbxmebTMwsFlKCsPJgchqTnKzLhlzRIKFAT8ByvqiozgFNsXF7e7rfa7zdWUbF5kNWapROnEwIq3871n4P/2wZwMU5aIp+DP+9OpwOGoB7RdaZ1vtV6d6u71W+U5vQoaJ8zLzDVcnph4Vk3hQ7pxkIGThN4IwVeZNUg9BujWDlc1bWtDJpI6sEAdG9A2RAiMqZFnlh+pnFioEfcv8X7P0yD3zQAmYYGKjkZidvVEnB016J7DpgDkBKiFENUi3erV7q6+1yZGaBQUdbyWnSBhaYTlKFGGcowinDdylcgrwvUVHxGK6/z3UBHXddGhUFJPICmJXU6XRQyQ2wDEq43yMaHCYu3Xf3DNPhdBjiaF0ynkz4DkcowssMANu7U3Kc5g7opTS67uvtCd6svVZQ0PiA3lbrWN2GHmQaKO95rHWgmQTpEAt9aA+6/XRuwN4DrlZtcZRMcxJ1nHsBrcPf3lbsfPn3HhoZM4OqAxrRACAkpDpHTNUKHtOq6woYPZgByeauiyrTe3uj67gtt99eqamYA0C1Kda3XHxiymJsA4bQ7U8R9+WYEwFC9AeLeAM+UWxsb6UtPNs6JD6f7rlP+dpt/lwFqa47AzKjrqdo6RMwevflubbrR2JoslhlbQnSv29VXunv4UnmxsVjHY4uitk2SMSBEwwxQ23jqGjYf9GHgyTKtkajeAM/jzSUiZ/1v2vgT4fl2/P9tD2DOgM1DTMjzrjWeG8DhunjIfHZmjJPc7sKh1DFb63b1pVabVyqrg703y5kiaeTLlb9EEHiBlM7myRCR73iC16tCDZkDURQMMAM8465PRO5pqOg5nn/3zb/tAW5gio7xenVnIYCii4szdlOUB3Pf+Yy0i9jRG6AtlWYrPWy+1Hp3qaJyvYo0Y26pVehTAUaST9+PTjDEMlTkTzSJF4qjiatzGNXJUYu6JwO84d7Ppqne3vg/bvPvNgBTYre3rzWZREawECoph9HycENoLYyOk3JNzUp5sdN695U2+9cGgGme6XDMVRaNQp85I8poxyKtMm5DJdFCs8mZktHUQDVLD0qPjOZV7zAAk1rfVNZ+6Pfvspjjgo8gOLTMqQTv766UjGm0TGyyxEbybAqsl62ZXOl7lvAANr07XGp3eG0c4JgddTjQtqMUhjbT4g/Mg6oKowUahTPNJ+dm0MaKrJ0pztZXhAg9j3d3wu+P628Pem9b4k0DcBLo/dQbSGAmZfk0Zd1ckuv1UWr3fTzrAdBUSXXMbrTdv9L+6Pr/acbkGmmQYSu4APJ+aMVSWdFZjjQeLTWO55bhXOd5a1MqXlmVzgDPiM3/TwNQDg99e9B7FMeWFcZMjvSDGcN63KAmw1cbC4P19kppRmXoJHNi2oBwBBegs+0rhyQVpTyNNAqZIvNN6bLBiSKXVxTFowd8G+T/kIe8NwKe8wDyc+NA8OH+3hoc4ymzSG5SFbHSEaInscU1NelMg/o7rbev9LB+ZVmj6yrGhYQW2LWEDPrGWLT93fsPxm/UwSU8i31UZapD73A8drA84s6Kgz79/fN7wXMi5OoO0uDxcDB3R8Bk40+ky43DDjKbY6t0kXId0rWlwfUWtWpnfUA3PUrBk1h5TJ7n722GgBa5ETy4hBusgiJT+Hmr1cokMRuFe8b6PnTS3x0Lvm4ARmJZBL0+JsSeCqyv45CrIOkkpdrtb7XaXGp/uFdRHk1Fpg+A9BWPGOlhWJMZ4cJmGahxaMMzvygvNFU4ZQQnO8i7vbvriLlhGtQtwrbf04NvAsRvS4Le1gNcq5y+P3kcAwzdqfd5HvEPO4Qj7Pd32u5udDzSIDlY1Urqo2ZhPAfeYO83A9SqGrwH6Q4jRzaZAg84Znt5D6tVx+aZBn2T8z9t7n2L+i5e8KYg8j4DvKuzPKTQpxDIcmaI7s0TUKxoksirxMQ8ZTWU2P6KsbhOqhtfdUvvE0oN5aZHSNGXyttsth0dVsOBXt56Fxj+04zwtiTWaw8ZjdfchiKcB1AIDe7/tnfxGe4uAKkwzTba7eADSOR7tV2mwG/kB9A/97fWkadh0tL3RFJD88QIjNU3KutC3nq97gx8AMC++BkMMdTo//Sa4LkBhkGJ2qpBJjrjmOmQ2OV94yHDhImrR57SoMMBJ4pstNtjgCvlBR0lpkrJBgilVj+6UX0229D3lJjgBwuc/IfA0si7ubnpBmUXA1D+MjX66BEmfjyB0nf3hDdPf5gUccUQAiWKc60YBmf6vRuNRcxzm3eNTTfBhmhbGfAR/7vDtQ7HO+XlTl2by/fJBp38gJshHCjtsEBV3aoo8Tp3R4B6wEpyJkU///xXPQ/wbeMIIIYJvUxuJfEzYPxwenwC0PdJ4jwcZYeOEMNQbZMqiihPXRnrBrfJSrzIUCNTdimXmVPKsq1tnCxwzDbWTldXKYw6xRHjMG7Nj6dfwwhpijrvs3lhpkXRBf733/8tl2kex+Kgomjp1NXOCEyRPtfrvpkqm7jZs6H3d4Xc9RgzwHalpk7lB5SnVHC1NVvieKYkdqM3zgM4fSgvwxP3OkCDs611s4llAI/OEOO7vKzHwaiUTZk6zBkGpdsaQzqP8P7HL3/R4V6Mt9Fj454AtblNXmKEt/jB+1Sh9zPAN38zKMIYgG4w4qs6NH4mO5Cya2t2jMfDPSRGW4hbqkSatWz+xsrio02E5qYIWZOEmyfGB/AayJ2DQYwzuD31QlPRDW+VF9wZ+qs/s+YoAMTUNnPC4wlDEGPryABOgyd8E1H5rgag/8C06DE92BR42zK4RGeY8A+s8Urf3xVEbuipril8rpUer5Xmazc/xBWekt/BE5x8ZrfdIl54Ng6Ocfrs0DEp5qsspKxo5P35n/3UMIC4o9tKJ4UQwAsYOoKkIDKCD8+p8oeY4rsMgksaApOC+ltjaIKkQZqb5sZ2ciweJTp+TI1u6BEPuFGaXivNIEFchmB+mZa3K4bcUJQzArdL7D6Dj2EwAByBeiBWU4cqyk7ef/vpZ8ZR3DW0yN0Ys8GCmQ0fLhau/QzLGgTL50D4YTX4yRTPB6Rw2912Z+rMbE4H15W+VpMY8Dp5/GnWEDrLZlfKsmszBITI3V4preyFXdpAtEcmIKM5DyA1ujlhRFaeM5anRA1y2X/97MeGVWwEq0GK2DAzd0xwLxfcGkGs6Icg6Qa9PQv0lkjyvBv+2B59vKfjxmYtBHZMgxc2+IjRn2ccZ9jnc4YAJGN1CBm3ygsKITcrxGcx+WEXo6y5allUkEJaasOUuKsoQ3nCsybyFMn79Cd/7KZm/eHKXGghQHsZA3CRicEo8wAbqXeX+0w1eo86NBjgOZdzl7XctLbd7Xs0QGkjriaNW9PlOR12+dq9qOYwAFLWg4ryWmXJtRqmwbkkyegLXuKaq84IbjzelJK+HKcaJAx8n8nRGAP8kQ1JORzgyhyXpiY2B7xk/m7OENLYBimH5sggoDw3wBtsrb+g6Bjd08S4jaUYR++bsPu9nR4GAAdcNTqc/LBxBps5fa7CMtnB3zyorB5U124oGnLkqj93CcJimnDyHSbw90P3y/EA0joePZL32Wd/ZGfjegFcX+PO4FQLmwQ5tZY4rbGhH2iZYOAFztZv0NVhFBZPGUbn+NkwL2wn0YcAgggGYOQe0B0arg41eF9j/UDEUJSgqqLZSSeIlj3T5lyuZBrUuT4nbXeCXM/csACtgM8Z7hADlHZRgkFpL8IAf9g9dV+4kZlYE3Ru09fM3k6NHQ49u+FSNRvs9fRH2mzRxgK4ytYbylVlz8thFuAuOtgMcFlpuVzYFT1n5L6Qsc24cXe4f2kSOJvfWRg07dHoL/yBDQ6H2N+L7A/UyLN5D0WUvbg8aVSY5wTyfvLpf7DeoMufDBcxY8M9nplt/gkAnUxll6ntdomb1Xf1w3Cv0HWZh1z8FBZ9bdbP6rvpsMyu0NC7Oz1lnsdNnD43AI2QsqB9tu97gFzX49RTSYzu0+PnMuRwGYJ4M1/uDwUvwAAQpeEy9dNtcozgffLpj1zIwAaZnuzToPEBI0JuFoAX2YE5AYwwoLS7a+gWPpy0w5NeWOmvqToK0F+csIuNqQ1I8/Dzcy5czh3QGiA5nOD0uS6bphvL+TBBNt927sImm8PNHUg6HaDjVOV6mnAAM0AHUWIm4M3L1Hjq/wPT4FdqijUy7AAAAABJRU5ErkJggolQTkcNChoKAAAADUlIRFIAAABAAAAAQAgGAAAAqmlx3gAAAAFzUkdCAK7OHOkAAB1cSURBVHheXZt5kyXVccWz9npLd0P4CxshAcaWBQyMEEgIUEgRdniJ8B+K0BeyFmaml7fW7vidc6sHueFF9/TrV3Vv3syTJ09mZV9/+v7SD2NM0xyR5ZHnWfC1LPoWWZZFlud6L/IiMr7rlcWyLHrxiSLPoirz2DSVXlynG8Y4XYc4X4e49lOM06zPt20b+90uqqqOeZ7jernEmzev4nQ8RN93MQ1DLNMU2TJ7Icsc87LEOC+6xjBHzNw1L7QW1pez9oKfi8j1ytPas9B/WRZsje8RS7r2HNkfvvp4ufZD9P0Y07xEkedRFDYCF1lfbHiOXDdeIsc0MS+hDdhQEbUMUMd+W0dVFtH1QzydrnE49XHuhhimWddr203c7m9is93GMs9xuVzi9eu/xel4jLHv9JrHMZZ50vvcY1oNMEdM3Jc16FDY3LrO4tkYmX7vzbO4nJ+TAdjBaoTsv3/3ydJ1Q1w6G4G3yiLXBjBGWZX6N7/HSUZei40gb9D5+xt2a6oibnZtbJoyhmGMh8M53jxd4njpox8nndRm08bd7V3s9zdaCAa4f/MqLudTDF0X/fViI0xjTOMUwzTp9Kcli0kH4Jd3xKlnOnU8gTV54z6V9Wd7STJCcnGONPvjv71UCFyuXVyuQ4zTpI03dRV1VUZdV1GVZWSZ3a8b5himiDkroijKqKoqiqKwEZY58myJbVvHblPHNE3x8HSMV/eneDxeo+ODWRZtU8c7d8kAWRbd9RKHp8fou6s2fzlhiGuMQx+sjc9hAHlfcnuF4BqiKQxkgPVAnt9L7v8jj3j2gHmK7E//+dUyT3Ncuy7OnNIwymoYYLNpoq3rqCoMEDGNYxAuGGFa8qjqJjabTTRNLaOBB7FM8p62rmJZbIAf3hzi/umiz7JojEoIbHe7WJYs+v4a1/NZGwYPLqdj9Ndr9H0fXd/HFQOAN3gdBhAmrZ7nENBJr7+TQ+IdjnnjAR6wOqtiVyGW/ek/vl5woREv6LogHAAcTh0DbNomeUDEPE1alDBjXCIvKm1iu93ISBiBoCgIoapQ/B6Op/jhzVM8HDAAODPLQNvtNupmg9MI+Pqui3EctXEM0F2v3nzXRzeOCr1FgGyAk8N5P3Z1nfzbcNS/ZJQUIjJS+lTa/DKNkf3x33+1lKVdmJjlpngBV2dTayj4posBaeTvRqFxWTXRtG20jUNF+FGVz0Y7X65x/3AQBoAhM9GbZcoAGHAcl+hw/Y6Yn23gyzmuyQOuWg8hwCfZlVGff3BQb82QNv//XP8ZB+wQyiqcPPjCK/ufP7xcyqJQCuFiYEDPCQ+jNsupekNVFGURVQE4hk4L4CSs87JSGNRVFVyLlFiWRmb+7ng8KY7rpo2qrnVi85IFduYauL0MANCNU3QAYd/HOBJurMUvMGgFP8KN9Tns+N+GXZ18DYkfh4X/cJYnT+MQEwb4r+9fyLDkbVyXD2BZjEAagx/w+6ZpFQ4AWFOXivXrtVOeX7IiNputjADKe2GTOcXiNMe12s0m9jc3wg7i+Xwd4+lwUfpj0+wFI4zjFAuxwYGMgwD6fDGGEEJsmnWtP6+R8PZ7woPVICupYfPy4OGtAf7tt58uuDQ3Y8GK5aLQTbqulyHI93XTxG6zie3W7g7aA1rnSxfDnMVmu5MR8BJin5PMi0zeAsrjBZP+bhv7m33UzTa6cYn7h1Mcnp5kJDY1Euzh8KtkaA6jj+PpFMfTWR4xDKRGXDh5QHLv5AwJG1ZvsIfIR1L4DmOvMCZLZb/76hcLVsYybKrGzWsDmsOhFzYQew2o3zY6aVgfeRQXJyvkZS0XhzcIDPNM2NA0ZUx9H48Pj/F0vIq9AYDb/T7yoonDqYuH+/s4HOwF3IvQgSu0mybKskyH0cXpdJEnnK8pRKbJhEZ8BJ6yyHufD1wQsYbJLJcfZUC7P3vOvvnlLxb+oZQQxLyJEDFMOPAeHxjwkoXfv+UGIkuFKfEwZSIqxKizQBm77SZubrZRFVmcT+d4OpxjCgzcCDzLehNdP8Xr16/i/s29vWAcZUToctM0SpmFjJBwh3A42xDDOIjcYDBRZcJCobEagtM36D1vHmxJmxeN/80vP1tmUU6MwCZtCIVDAS3m4rOBqMdqGKjQwlaiVFeFbnrpJnkDp0JmIYPsd23sNo1imhSbV21stnuQM5asivO1j7/95a+qBfAArgOrUwjoVUWZwBUvxG0vl6uMgOeyTgzAYWMAMsYwsuHJWASV1sn3MeDNxP8KnmSM3371xeLYAHWJjUEWA7/qCrcHE3J9iJzMC3zCAI1IEt/JEoWKH4CNGyuUKjykSCmy0CllRR23d+9Gu93FFGU8Pp3jr3/5czzcv5Gn9aPTHanOdQnXsOdhGEASbGKDnK7LAcf7MM7mKMOo0JQRpkHF1YABcH0MsoYJ2PHt1y8XNow1VwuRJsCDpsqjbUqdNPcQZb50MsJa1a1u2gJYMQsUATI+gweotpA3+BrXYYmq3sTtO+9G2WzjeLrG6x/+Gk+P98oUhMRKesTdobBsNG1SVWtkUdeNPMQMnPWbmzhtUkMkpB+cTlevwMP8n7Ex++bLzxblxcmuQ3wIHYNTdHnbNoVwAdSFKJGWOG1qgU3bmg7XzgywOjZRlnVstq1OD2+iSMJdIUTnbpYH3Ny+q5B6vH8Vh8N99L1DSLyfPK8QJt+7KtRrCZfT+72MwC8IHfCDgwGrhPAygP8NmAsbdK1kgOQF2dcv/nmRu+im5gDO4wBiFm2NC5dRiygBMpPwAKbGieGWGIFFyUgz709RlE3c3t1F29ZCYsgm71+ufbx5OseclXF79w86xdPTfZyOT4rhcQZUcWeMwel58VyXNwgLMsR2u1OWgXdgANIk2IC7y+3HQV7hShJgTMZcjaraYons688+WEg9+oPIYxGRsRGMA2Xg3iB5thDfpB6nP2guGyIClRqrSu6Ja4H0GICUhxcoJCo0gl61wbkbo93eiGFez4e4Xk4xTUvkpYGPU8PTrtdebo33EQUYsS5TlsAARSkjXS7n6K6IKX1MQ6d0R/mNBkEhxeGubm+vSprAt198aA+YKDdB01yugtUpMDghGyAiTxnCadds8XA6Kz/z9xWlc8WrcRprm6ipFCl+2jZu9jvF66s3j/KCkbKaxYwQrl6LwpCcMKmPa4IppxN/a2DG2KB+WfpeGEDsbiJMJhnAlWQnLxqmJUZOXRhisARmF4VCRPb9lx8LBC0UsaCQMfggrljkhU6uqbKo5caswlIYf8sp3T8exPTYwI/dEwSnxjA3KOLuZqeMcDp38XC8xvGc0hkIPaYyHAMkDgDyA2gYQO8nhqPDWqhG8yiLMnJlilLZYOg76QkUWAofU6W/M4CodPKC7NuXH2MOERrci4uAB6QUXrhemUdsmyK2VHwVuqBPjgsjTx2O53h4fIzL+Rq5St2diY7IFCqSjYlxANWgri/MAX549TpO53NMhJZOlqxTR1OZZ6hAUjpLjC4ipeNO96/rOtoGb2uFR5TRaIt8J+WtipFP3wfH70W7YYK/fvGRyLJEzcrVnhjgTGk6KjVmyxQbMTsQvxEQSQ9MuiA6wtPhKDoLkEGJi5KsYI/hWuO0iOGRDln0bm8y9L9//lscjoeY9R/Mk7xeRCEdz7oEqVZrAtQAYOkWvdZAKFBkbbfOCrj+09NjXC9X845V10xiKBmPcBLNH6fIvvr0Z8uqmAJWIh6oLoQDFHkao2AhBZygic12o1PiZAAYXmQD3JTXdRh0upLKMKpkGP5m0mJEmpoymnYTZVXH6zcPiv8CNqmQIoUtPvFp0N/vdy6yBom31AvkdSM77s/G0RcpyMj5jw8PWgtubiJlfVOey8lD72XMKbJfffL+shYvFg152c3Ju2U+y22dxzFQqY2sOIEVRWDIxVfkKyBm1RDMAZC3zdE5MYTWIjI8Y1r0WYB2u9vKXZHlrugMlL59F0UOMKJKUaGSfVa+8lYb4EB2u1t5FS7+9PgojxRwFnCYUocmWq+QMjdABcu+/NefLFwctsaJYQB0OqVCTp/43yBythY50AbFBcixIeKC2IE1ZYzBrmx16G3cubp0PY5rE3Z4Dz/ttxsZAKRm86Q+uPsokWQSPnmNpeJeVFeY4NzO7wmB/c2tfj6ejvHw+KQw4STxgFXQUZpOusCIB3zxL+9JEgNwAEHcXwKoanOQdxZybzeuzAgFxA5ODt7e93NckKwmg91AzslQhErhDsZSSpVLAZ4UKRa3LbYUsd9vxSaJ2dPZYih9AZfivQ4BtaltqSIrg+C1S40WsKuS+9/c3CocIGkCZYgRBsdz5QEOb+g1a8Gbss8+fm+xcGG6y8nhCVJNlynKPItt20gIIQ6t0kwWK3Xyc3QjiErDwhe2Pu/S2IiLwvOMw8YWNTxQjXIZQKAYWTw+Wh7HU+AZfA4KjTcBwAAoleCPXRydQh5we6uMANk6HCyy4OrwGULB6dKdIzV2eO+Tj/9xcevIKYj0Q9qTNhBzNHUh98f6pClSpFna37e8OH1iGFe39mjmpXQ5U8yYwJiFpWqN7g8ZZtPEdncjfLEgenY1J3HIFDzPwKJaeEAWuH940ka5J5ve7RwCEDC85nw+67vqCK0oSenZj1Rl1oUB2K7ZFQaohdxF5vwP+KECAVzsEfTt1SAhDBz3VH/EJJYFaFaSZMXabSnX7VzD3KJPPQDkqbo2d2glk6NCWRlSbQDokVliVhi2m1bGeXx8EgmD4qJXgiHbzVbhAKcgZUpXVF/RapHID8ed+ghigi9+/p5FUXkA0nYlA9SlBZG1TYZbc5LePG6fePaIEVRIiJkp69EhSgYl9rj2SkCgp7g1SHy9UsGdA7hoazjGLtq6SUA7SgM8ni6KdzwFmixPjCWuF6rOXowVw3Ly4Ah0nPv5yyowBhGxg1BZbEhhEJF9+en7osmQjzwn/iEhEW0FML7NDKxa5GiY4oqWT66WFGYhk41z+mKVeA7CaVGllGpBRSdDssepuRaEpj9HkQOGdKJ2cUPXOCcL+ZTfPB4EuC6EkNNQnsEWPGN07a/+BEC9i5YmTcl9bWR4gapClcXOOuojpkPJvv3lB4taywIGTiqiiEVlMCSEm4pAzLNueOnGuAwRg1wKGcqVlQwg7EAtgr1VcjfR3twqc54b+WGoq64/DPQAOxmKLtOOcMut4wFiT8ezOlacHIcDyySsKMkJDUJE4mZeKhMAhq5KLeh2dJuTATwCkBqpCaCzP3z98WJV1Z1e3JwmuNSguhY/4AvgQwTB/VF1OHmuJ4WFzenC4RmBFhZY+n0HmCwDkrtyc2lLiiUM2CACyt3dbWzbMioZigZJHyd6hZLAyPd4yVbrPF9OAsM1w+Bt7WYvMFQ3ezFtVpMlqUQYwt0khwgHl3338iM1V54HC5CwkhymTo8Y2BK95O8pBjyBEnNMxRCekOYKOJm2Nt0lC8g7IFUaoPDcgXKDJC43PU7Hs3Q8ylrEk/2uccFEeLGBpPFhTHoTAB3Go9q7XK8iYBq8yCudPmAIeCOKiPLKCGaXJlBJYEnCT/blL95XFlAVRvuLnJvQH+GBRRO/rq0xAICC2xtMABcWoJaYxApLX5KoE1eHxYEnMEBNlSRVFtHicDqJ/nJtPq+U2Da6L16Fi4M75HB1n9rGRZFi2ziAoelLYIBNW0t9QuFehVGMCJDaCDaqeiHTEtnLn7+/SAJHiUkVHK6MBOaeYHIX0Ud7AjgA88NwCJFsVsMUpQcrTIBMiBBTCAn0xed2lsAQTLGL4gGUxkhgeJ3a8Ujr8Hh5oBWm3W4nRucGCiRp1Q7zKOtW3oGSjaF5ufLDEGv73wqWPr8a4Fef/EzFkE/QgMfNRT+TmmsNwPXBpR/j6dQ9pz5Ox0qtswALNuhlOvVtWwlQAUmLLMznwCFGLQSXVC9ymOQNuCkHAhZJjRL54dV6pmhZdJrODPZcSu+GDNBQqvv6HvOxEVgfJ49yBaYQEmQFyWy/efHhopwtirgOEXgKCFdCC1w7QMTztZ/j8YQqLBdQjLvISZoCnZzEBqkhiGdACSRYZ4RwfVQh8rjndlx7HKG49BAlfzGAUalQ2u0YwmgVOizcjVvGeZIgIvRn6KpSCT3Pgyg8Hon7QL4Il1Veg0ApdKgFvvn8w4VyURxZ2cDTE1DPPCaphFwIDs7fkAUezwxJWJ83DsAGkcJRkGuX0jRU1TBZq0KjP17CaUNwWAR/u7bbz6g5Z8vbfHGiuD1CDIWYCypvBrEG+l1UaBT72O93MjR1BBRY4mmS4u0JbuxArqgl+NkGePHBQgqS+0vhYQqKKnCIEjpcZlHQ8FSR5JL22NGBSSMryNIas5uikcuS/xM3SGFF/JNaLaOZO7jqo7cH3XZ6ZFYAZOekcFmMjv4P++PaUBWUIgxBXIsFltDjrSvFPPUlkgElviQy5y4znneJw8ESug7gmxcfLsSRRRDHHiptMQ8WQmF0pXtvwoGIuAxLXAdOGYHBkyUwM88IohY5/3umJ5eirAJLBMYUWiSlM8dvasCzUlidTkcBFbkfvMD1d/B8qlERTZfWK3cpqjbKukkSmic/OBAMqgmXBm3REpuk/FRJHo9nrSH79vOPlpUa4s5KUwghMQm8KFQYApOrMjeQ5XFCA+h8AmtPn4WxeTxAE1wYJ5EN1/OEWOaJEGYOkhHUgRLtrtQUoRyGvbnUHhwKm03c3e7VfXLTE6kLXkHjtBGHELdYqDPMC1Y5jJD0+I672Jz809NB+iVgmn3/xT8tngRNVBgUSAbgZGCEnKCIDFogslU/xamzFqCKLSk9ZA2BHu4KX5foQD6EITqdcbqaR1Tf3tOgoD4GQE94c/8gkkOBImwhKxSZWu0UPNIHMmOSWOrqDSg/aQjKrXILM5p2a1GZofsRfddLS3h8PCjcsu9fJiqcRmCVW6dRALihO1xS40cU0vJoduZSgBhdY1LMLm2QgeoinWuCJOGFT8NaAG6J22sEJqVaUq60RzCoqCRlnS9nubiLL4YZyApWfqQMbbYCR8AZUiTRI6VEYh6Lr8CM12IA2ntwGsKAyTWV0+dLZN/hAWn2d21rqT0eizZPSYzlitI5F2+RLgApGma1uPjuRS6xkcU9PCmykxihSmAVVKaofCHBSzYnOyQiBkpLyUkjss8NTyk7VJm03N6JW2aNqkItfZEiricJnemVUgbA28AjcIA1AaR4Hooxkhnfs+8+/2AhrteB4h9PW7rCA8iIORMjgAVDwH2Y/4UVanASaqghyVyTorj8Kp3jjg4Vx/U4A352ey/KkhX3YvIDLwFUMRr9R0bl+CwD1NYQ93GnvmMj75F2SQhA3JhWE1YscYZrDKO8C69cW/TgAB5AQzX77vOfqhZYB44zZO8MOmsNXXJUvvYFYHWeHGVTIDUyuBUht9jJ09BfpDQRHNFn1xKwvJ4hjJk0axFku2mFP5oQmyY1NBBb1iFsGqOwPjM3tEQzxJubm7i7u0mV3wqKb6dKWL9keqbYKNHTIIcKqauB0AZ48X4avTMQSsRgfIV8Lb6McEmuNi0VNU2upIUlN/cGPahAynPMuZS2XE4BQyHlEMADtu0mdkyMB7E5aVB6nT1Y63xTRcgWn/d0F0YABG9v989TqvJM1Q54qMfoAVsKIA4BsqfpVUrwC1zgKN0w+/YzJDGah6sBPIqCAdaOKwDGDRFMcSNOGKaIgZQGGVLQeIqrMzCk0bwwBCpXPK81+YrsGAAJizFbJBgZh/QnDcBNFu6PBkDfD4KG3I0RfA+MUMUNirL0QLNFd4LYgxkqlNnluqtVRNLz+SQDgDXZN5+9Jz1gVW44fT8j4Hay6nfUXumDhao6UoqIzTK574cBQHgyQg8ZcV1Ar8GobHmKkhc2RteHe0rM3G6iKGphxAUlF7JyHWQEeROGbDdaz7W7SgRRayylUg1e3+zi9vZGRsBgLpAKrcNjth6xYS/SIDRzeNI8QfabTzGAKxJwAOvxYQ8lmel5mMCcvS6yxA0YmCDFmXRQGjHUrDk/sUrXFuuN1ZXXtJn1OawuHbDdSObCXR+fnlJHt9DCPbnu+OUAIEiAouaZMaTUYkZmGhGlm5u9JHK8FcPwZbLmaRANe2ua5KiONNfIfv3pT6wIJd3++RmA0m1szwt4tBypnHqbbOCq3x+EeExUe9LgqcUxQKlpsnVcDcanhlgaXPT96BR7vIbqkEYpxiTPq9VNwaIwIB0X2jAGYFCSUhpDkH0wDkQJTIAf4FUUUpbhPfMAqFKFMklyOBL/F4VlxjNDnORqAFyoqKi8yjRWQn2Alk/h4vE36X96QsRTXEA6aUtdQfAnKKsRNdKgZW/1FkNZGrNuwL2QuZgHeOTJkoeDPEAjt1SeUm54yMJZSB7UEyIGS4VDD102KNJhgh8wicIsgtpzYjQ+RPoNnPwxGUBDF3iAmliSxnnwCEGj0gA0Z6w5fMlYuKMfkPKYCu00C6Ge1caIS2RUj2gIRRUFtUVqnx8loqSnUZLIoa5teuIEAxxObok9d6Cls1gCX0UaPWTVoSAxP+SRWX7nBulGZfHtzV4eAXMkrOWhSWQF+TGANAEM8M1nP1HdtoKdj9Cv54eR1mHkVFeruSmS5CEGJOiUraKosihVP6Du5pHpiRDiG3mbnGx3RfuTFJ+UJkpgNuZw1NM8mjSjDwAYwvjcdKEBS3rzuB5xjNeA8mABouh+t4n91tPrq5LFNfQZZo2lByQDfPf5+5oUFUzgC2nzanPRME1z/4711O3V3LCfH/PmZ0+GU6GVhIVdFlEVJaDrPBaP+MmneGSO0+Ie7h7TbHWVuA46k42YJaRZosVfLwoBTAMX0KgeImeSttZBCdp4nlMCB1ITRe18ym93ldn8M4j+7oufalZYaJ8eR0PTUxdGw9BpdC7V9+oQ6Wkuam4MgYZHZyYNPigkoMRMhziLdN0cZ1SkK6P3uYBqv9t7EgxyQw3f+xlDiRTBNGkbN3fvpKkPJlQv6X7mH3q8JslaqhBZL8WSZoaYUmvS8CaS3aAqEAOQSZSJlI2GyH7/5QcLQOOS1sGgYeU0NCH1Zfbpa+BIoolBcZo9V8yJvzUAeDKrhPY84RxDRxnqWQJ0AmYHGWnx2MvVfH62VgDK4xFkhrt33hUg2uWND+b+HnyWoKoc79GedXQOLVFzzAi06WEvaQzp5J8HKMGA33/5UXpy1A0OD0z5oSe5qNpf6ZkDob9nbaQNpcbjTBigGTBLKPUHIGM+b4hxWGLsspgHhpx4nx4e8dmmGYDOgFo1mjFAuWWhEkR3N1E1fvxu5Q7kbKZH0Az8HAOzAwbTlQprVhGJX2KsR22tQFmGs1TuQans+5fJALS41O2AN9uieIMGH6x6e9xEdTes0NqcR1FZiIumvEBEJR3CC/oYujmmC62aIrKl0HDUdrPRZy2O9ELwdrOLrCik2GAAqlJO1FkgPTCVxtz0LNGaAiV6QHxcpGnAQw0dizLKUcvag/D0iUA19QwkisKZXfnRwFy7p25ueIZ43b+BDjq8tsxU78N4SIE5RoCY0MnEMEN0lynGC2yfxTWxA9jaVqeAAbg3J9hsoMRlorrWC7ivxmQkoJis+Xkhx7AeeqKsZnpNMW8jcL11SsUzQf6M4h/PSaP08oCvPwUDLFGRZuQ2qRyGQmq+Po2VYlFUYhBe2JCGDdVxxSPwAGUEemKIITbA1C1RZqa9xD/VpOb0Umtb8VvVz0rOyh5lpDQHsHrB2lVe22troSb0ZzQ39TPXx3rNXFyLYEwe0lwNIH7x1Sc/0yMzPn0/6mZxNE1V6DEUxAk/VKUWuOX5Z/FRKVBeYfkMHIAVCmm7kUShU8EAkq8JLwGfJ700RJHDPD1opZYWM/4D4JfSlowFCLq75Ij0BBjzyDJAgxckD0gzSqvKpXmCZACP37iQy2iNaTcpxv1gwvp0uBmU1BjJ0ekZHA1L+LXQh1NvETUmaQAgr7q7Y0yMoqVylAkuihS3r21gvsQGRVuNO6QnaKsmvjWa30l84Wd5Y3qA4nmsRyFgrcLTbpb4vXl/V3eIQWoNUVtn5AD+D1jkoxYVK5XPAAAAAElFTkSuQmCCiVBORw0KGgoAAAANSUhEUgAAAEAAAABACAYAAACqaXHeAAAAAXNSR0IArs4c6QAAHPNJREFUeF51m+mPZNlRxePtuVZWd89Yxiziv0IgMGAwYBYPIxvJ0oCxwWBsxsh42O0PCBDGrB8Qi1gkQAIh+Ddg3D3dXV25v/2h34l7M8sgRpOq6srM9+6N5cSJc+MlX/7SZycz/380C78mZknCv2wcRxuGwcaxN5smm/j3OFo/TjZO/gU+WhSpLarC8iK3YZr0maEfjQ8lE5dLLE0yK4pcL740DJNN02R5UVhVzawoK0vS1Nq2sbo+W9911vWDnevGTufamrazYdAq/XpZZmVZ2GxW2awqreLaWWZpmuj9ydgDe0ltmkbr+07X5mc/9tYPoyVfeeeXJj7MRoawqUkb0220wHEabRoHM35yoQGjYAi+xYcnyzMLBsisG0Zr+8GmcbLUUstYQ7heludWFqXxk/v1fW95nttiubLFYmFJmljT1HY+n63rOutlgNaOwQA4hKuxyTxn86XNqsrKIrM8SyxLfPPuwNR/JomN42B931rXtdYNbH6wbhws+fI7n5syVjiZdcPgXpGFUy0ZA/Bi83oZGw/GGnlvVJSkyWSzMlMEtP1oddvr73mSWpEl7gt+l7cry4tK12naxrIst+VyafP5zAgXj4DGum5QBOB5jNB0XNMjLsP7hRugLHLLUrY7WXoxwHXzcuLY+8b7ThuPr+R33vnclPNtM92s7QYtjMWmKS+MEw0wyNujjOIxwu/jMNg0DdponhMBkwyA97gyf8/SVJ6ezWY2m88sTXNru16bIwXms5kVRabrd32nv3c9YTsGI3jI4h48nIfwx/MeYTjBLE1SpRFWIgX4/Djh2MHDnvSdMABpPFryW19yA/AlbtC0vW7IBjEA72UZgUw4e7iTEhiJ2+JZwmscOt0OAwyTWdsO1va9TYNHB7lJvhLms/lc38RIXUiBssxjtCo8cUQ/eKTxk7U55rAuDBDWxl6n0ZIE7weHpb5SzCUsYn1EKtE7TdZr/f5KfuOLn50AEzw0WnKNggA2bF7WzvlpyjG8jpEIR0fN8RIFbsxMC8bDRIGNg+VpYkVZKvyrEgwotLm26yxTPrMTXU34outP3MtkUN+8G5wwx8lsk4375kPex2soVdkw2yYOPLUVscEA/Ex+7Vc+MyVJJkQFLbkRi1e4kQppoggoc4yQWMFCwYveb0CkEH6A5Dj0WggAN00YEwP0eo/wJG95DxwohfiZNU0TIse9xyIv9xcIJ/Ka7qNNOqD6Jx0P4ouNyutUIcAa0Nbn/LueDuAB0cRrsuSdtz/NLg0jWEoOsp3EKwLWYOEhFTAAkcDnlZuDFxogREYYehklDdchlKMBMhvD5ksrykIIztLO51oeBMhIH67tIOvGvfwePO9J4Jvy7ZpNSfTuaAMGUKjzfccLvR543w0QDPiltz812RRqJsj/4CXsC+itMAXkUsIvFQ8gNx0rzIrUMUKRE77HjUgBDJMmozZZzWZKASKuaVo717WibFYRFYWiAq9jfIwQvaXl6rpela7e9hD3DPe/x8945fnGzceqJpPx3jtv/9ykP+rCboiYElqM6mm0ouMqFyU/xQNkZbMyMyM4HMCuRgCKMEwevFxWpRV5rk23TWunurY0S20+q2QAsMESjID3rwaghFGmlb/BAHgZj3toX8NdqeEU6FLKHQSjRwPLgTB96Qufgd6Feu8bIgrSLPdXml3KyijyA/g5IjsGYhwvdVQxDNC0XkqF2IlZladWlc4HKIVggZggzKzrLcszq6pS71EeeYm8hFxm8+eusQZMGQeFt3BBaeBcROUxpIbDZUiVYKxYtoM/VeZxQvKrX/gFxbE2Hi9wMQK0EoDEY9BJ0N+RXWUJIqTowPupAFIVInAJricDFJnNy0xpwmYhPtiHhQtjglE8WFk9EO+A3E2DNUNvdddYC5G5AFhEg8BRFO7BwwEcQwCEa+oDATC5vlPk5Itv/4Iz+ggKDyzHp8l3UgFDsFn1AQK3UCkiL4+fBW1Vatyo3KfIUquoIBlgR79QiC6L0ATg49pEFr2GiHnC5kdrx8GacbBWJOYafQ8C3KuBctqrg1eFQL4dCC4bx+sR56YpteQLn/+5KdJeRxmnEP4zoChGIC8DEKk2RzocQlHee8jDgwFEWzECQEm/AHWtZgJRNSmkAj2Tbun3xZOwNzzejL01lLQLA4WOUzIzsQLxABnAydClDYgAEKuA3nBnqkdg87w+/4s/O8nL0XOitdzkCnheSgK9DLmPn7CRPhmbJ9Xa2FHy0zs3j4LEyiyxxbwUHe6HxLrRZAhxLnWL7inuPwB6Y68IaBNKW8j4yctsDj7p81cDRF7gCHhhAKESZJ735gZI2fyYWPLZn//pCVCKBqBuD5CXYATdJJS+/2ME4jSYPEaAlvMgBVgUEVBBpIrMZmUuRghQNj1ly0sv2ACVZdnwdr2mQZ7vMip1jMxRJRTMgTiJDUb80mcCAl7hwL1tvnlIPb+rIMBjPvOpT0wCugBysjxdlyIh5FVoMhwPYo8d++3QfFxx98IDVCIhSmBAQRsM5cbTqTV0enBcUF8g6L0ANFi9RUD2gdKaTTakIsmWTpOIGddR6xtIhwDwEn0qbNqgyByeZusw3tjlgjfg0Cff+tjFAHyfjRMFAqSQ3xFNA564GBHA0aOCno+yRU8Q2JmaJMcQUd+csI0dJhUlJBk5ycJiGiqlPAUJsD6drLXBhsQ7UTZNY5Vr86J3gSFFl5N5dKguyGBQkTWijKjRWsNncdAnfuonvtEAgXxcGFP8bGh88BKLJW1iAyIskAoUeLaEFUJZiok0AthfoRJ4NUIGkGWFU0mJHBFnHMwmvC8w7GRc/uiNGQYgjCd5kcrhjZnvTWKNHOkNm4BYa0acAe+iDSZLPvbmRy4GcBxwVHOC4eVR9f4CaPT1LnxwUd5Xaex6kZrI4dVUqa0eRXBEdMCaWDa1oFwcQL1DiCpfnRuZdBMgjmBSIFbqSzB+WGdoaxXyQbpwwuYs0kHY9QgYp5wWyr6aoTc++kMCwYgDF/Jw4d2E1PVifNa7OZiblyLYYdt2UnGarrtUU8/nUZZHB4z5J/ASiDn4CreCxsff5Kmg7amrf0CJFSmB3ytlQ7S54S7k90Fliu2zVzqvMsGxGOCjP/FhRYDf0D90/c/BRMJowAR1hhigKGQI9x5hN1jToOS0lw5uiLqgREr3mCtNobdQWXI6K/ElsEQ8parA5+SuUBaJsAFt0sumaE8o0TDVuH5FRygHqmD/m58EziuM/8k3ftilhocS2PXr+g0rO/sDiNx7GExqkRbqngQDqCASMMZRUQGgxpAgxWXoUCpDyjqV5npECU6QbhAiQRvAwL2aJzpI7jMplwNL5ZpB1MFBwuVgAEl6l4hxMSHyFtnxY2/+yOScPnzpQf/8sO6L/0uluXxSdhS6kltZ6OBUBbyRQe1xHPBySK9AJ+ge9Z7cnUmOep7yQW0GfQCj5F7yqE6oxBhALfdl8zRtHsGAaszza5Q9jOiIk0HTZF0ff/NHJrzrZU+x4MQnRIRaX9haABb16tDSgPgxrNUs6W9wtiBFBwAV2FC+IEN0glJjeEXmSKq5sCreAC8QOAbjSrPlM0HjYy2KXzZOC42YEvhETIUQ+vrehapeDeANoFny5hsfnujwVL9DbkVMiLigG6sXdwOJklwM4uisUhgEkIjo1zLtBojKklRcMUh0oknMc+QVDCBADhiAJxXWqhKOHUQCmiH9SUZzlRf6KUYbGGI864hb/gZke2CQ5Mc/8n0hAoJCIi3AtTv17oRl2JwHf+jTL+3wdfMqh+CE+pwoTUcp03HLGZyXOdpsiRnBmLH8ylgC5SuCq3KoTHvJlaKLAfB80C6SoG6rz78cxriDHhokGkMS+oc/9J2T8lqcP9TmUJ9doHADBOi9gKKXOEd1vhujomudtERdMHaIIlAXNdeNobKYPChl8S4ygAMtBosM1MP2yu48XQMYyvu84snQVSyNspgbwrHOK0NqyYe+7zt0NOabj573vEKZuZbFcIQYmKc3LNFwTmQAKkCKnIPkeK0P5Um1373vRD0KlqnjBhK1VGEwx1cKaGKIaDwviZH9em6L5lKKL8rVdeMXvhF2HZnCNTrMkg//4HdNXnK89LhkRX1HlgpdU8gZfkhrD1RTPFtl0VtNNVJtp98x4CXvlZ++ERkg6A1xgQ6AwQATBzROqXXWJ/bmxgq+C9J2UI7HUDVC/juNcSHGo8jZn/58SYvAInHCj/3o907UX71iPklsuAogjqLXahBlsVg+43GUA+mknMSgEWucKwQOjnAx9TrNQSnSuWRogDxbMz8Q4eCSZjmc/HivwSFJ0CHEcq8s0knUVVOIZVUpET4ZJfJQU7w8v/HG97sBctA0tKVB149NzjS6Ckw10KkqmkFgZPGc0G9CKhFJnkK+sSheBvImatnphYmpiiInF0nLu0v9e/IDlUhf+x5NEikuHpR4eomU6eTKT4xIpEv6QbsktoR0DBTbm7XBko99/AcmR/zCVZag2Kreq44DUn5E5XwAwQSk9/z38ndtRUmdonCFl/DDqzpOD10ZB3Bj39o4tLxjScZ1dGoX+Gfg7OG4DMqtaOLeHN5y5ojoqkMZV4YgV3nO5v1aAjlpHFQxehBPcX1WhzK5UuzcnCz5xFs0Q+75KHzGTV1a7cDX41E4Px3R3X06TxgTmwYXHiLfr8pK6o80xHCCPI2djtA4rp7Qe9LBxgT6HDQIcjpJ1WytV2tbLVdWFaX1Q2/H08FO55PoNvfjXmywzMEs/A6hC5KePJ5blvjeiswbOM4mcTjH5IfT3pKf/vRHJiwUCU7ctDTLS712BA+dS9ALQxnhu1OqzY+DN/GRHnNDPwP0CjExZeJzKGHQgkmN1tqxsbZvPLU4Uk9SbfwD7/9me/zosf59Oh9tu39l+8O91e3Zhy/YHBEnfRAZPZ5bxIhkPcFQWWGzaqZJFPbVdLWd64Mlb336I0q3KGP7+TsNiSOoen9xfXLMqwNgx4YIpfl8YUVeyQBtwzxAZrPZXP0/34k0EYvjdaKimi1k8JZJkOZsh9PO9uedneuzpkPY3HKxsA9807fYk0dPlJ54/9X2hR1OW+u6xoct8vKiDpOL2oNa54BHqkxOzsYBXPAqB+IMQ2dtX1vyUz/zA443QU9XreYEFyBTaXHBhjAqi5lKpG+m01DD5uaRLaqVJVaovBdZZfPFUq2ya3y9Zn0IN2YINpvHtl7fSpk9HPbWdrXtj1u727607eFen2OMBY8u5gtbLdc2ny0U2kQJ0yOUTYyymJEelYzhs0wunLiAg1iaCLTrurW6qf2gVlojfQhssrfkzbc+qOM/Pwb3g4pYEVwh8rkg1XZFADk+iaQs50u7WT229eKRzedry7PyoiniAweb3urmbMfjXlL17eaJ3d68ZhxK3L16YU13tK5vbH/a2e5wb/vzVlEgISXNrCrnNp8ttSapxsHL5PRqsbb1aqP38W7fOQljvd7fuFBzOp1sfzwowrq+Fd4MY6dX8vFPf3By2TtuPuh2QRiJqaFMUq8dBibK0hbVwhbzG1tUayuLuW5e17W1bRtOfQp1eOS9OMJoVlVzWyxuLElyDUIdjvc2TI31Y2PHeq9X05IGAG2q67qBV8rnfuj0XpmXtpgtbDkHKG8UJaGXlaEl7nKwUje2Oxxsu93a8XxU+vRjZ/3Qegp84pc+qIORq0zlhMX5c1ByApWIZ3c0IWVR2bxa2axaSWt3EbLVfA9VhJSpSpAXCQ0RJLdp8AaI7+R5pe9sd3dWtzvrx9rq7myn+uhhLjqNAlzacv5IRijLmTXNSd8j9Pl3nnI9j5KqJEVzCS7sgVQg9A+Hg+32e1UQDIjn266xuj1a8tbb3z2poQhH4I5a8Tg8IH04BfLwwzikSWVVsRAwRh2AdpbyiP7HYuYzhp9KAWWlCKGe835lWVqqlp9r0P257c8Y4mh1c7K2gyME9TnNrMhmNq/w8srFTUUhvMXTAj4AIGYp//ZxH2o+xsBYp/PZjqejHY8Hq9taadAPjTXtyZJP/er3fIMBqOneNYUuSzfwBkXc/zKnx428KohlJX5TNkfdJiRnFd4p7Obmsa0WG0vTgnEhx5KkUD0/nQ/28tW7drd7ZufmYMfzzlrKXGhyNFyZMQi5tvXi1tbrjS3mS21eXUVowHqRIw47nARBxlgHzkU/IBL2h73t9lthErjTDY0lv/ibH5pcRooKbSAzMCn9DeDxk1uRjKC/x5MWgI+bUY+lzqTumVnJ2JufBt2sHytPcwygE2MaJeaIJnt1/8Lutk/tcHqlsDyet3aq97pfYrklU2FlNrPN+jV732sfsCdP3ifgo8pQvUi50/Fo293W9jsqSBewAw5ShiN91yZJgf1+Z4fTwZrWjZD88ld+2DEgzNd5CEGJC8vTUptythdkKzQc8XlMUFhVLm1WLXxuCIZHjxBEThgaqQBQiivIiH04LZopLfaHrd3v3rNTvRORabrG9odX1raIqYUlU255OrPXH3+Tfdu3frvd3j62Jeywmvlk2zSpamzv7+3Vqzs7n076GxUIiowhGcbs2lY/qQSkw7k+WdvWlnzxdz86xc1TdvKcAwymL5nlWdqs9HDTAcgAejLB2YpcZElhi/nGlvONwpE67YcYUNJOoVjmM8uzSgjuA4uN3iNn82yu6gAXwBv8x4Dky7v3bLc7WDJBdCqRq9vNIxGj9Xptq9XKVqu1jEDZFtjVZzvs95cSGhiYIqJtYJkAH6pyoxTAEBCx5Mt/9EkJIuqqUpoK3zwbX1LiFjcBwFB5GViureHLba0N3qyf2Hp564OWVADamoGcO+rABJQGMDEqkdO0LPLgatJUmI0MUFbyGO0NpfHdd9+1p8+e2TQCbjNFIAOW73v9dVsu4AQ+WosxiAZCPd4fucz7Ae/2MExT10HH6FVhSDXXFVtL/viv3hFb5iZEACGPAUDc1dyRVyNtccBwGOx8PtrptFOzcLt5zW7Wj2RADSQPHI5QzvbW9WeVsfkMIy60oLaD7h79vGCEvMATvKUFb84nN8C77z5V1xebNFjn48e38j5rIbxX67XdbgDGG1ssiVTHlctaewxQW12fxEYl2lIGw7CVNIe/++c/npSbD87+QPJZRW7PQ6jSQnqLS37B6iAwMEJFwPo2sL5BZYx8Vm3va11ns36/VcVKYqbL7354SeMDHQaY6vNZUUGIvnjx0l68uNNJUxxm4nB1vV7Yer2y+XxuM0jQYmmr1Y3d3NzICIzgxRkDRUDfKyWcnPmYPOlLZMYUSf7tP/5ucqtcQ8ePqQptvqq4kTOta12lLd0r16nx87mTECJCBjjfW91tbZw6W8w29ujmm21W3gg4vcR6VYExbu9f2bP3vm7396+0UGjrbru37XYnfdFPongeIbP5vLTlamGbm1vvKaDBc2aPMQzpSmPGmF2Q6EP5O5+OSltwIJI1vK+1/Pt//uPVAAIvP5snJOHhq+XG1qtHorAxzAkpDAAgaoy2Aum9FwAn6vZg53arfn9R3dhm9X7VcUqkT2rEo7NW3n/+4j27u3th+/3WDvuD3W93dv9qq6igSUPtKUpG7XJjqJq1LJc3ttk8svX6kS0WS1su18IExm9Yp9Ktaex8PtnpdFAauPdJhThsNVjyL//6NxPo7ANvsSucVNuhl3jfgXDmx18jDzAc7XC6Vz7jHRa0Wj6ysvCure5Odq63Noytvzd/bPNqLUIjsVXCsD8XgAG2W0rYS73g7P7a2fF4VFll8zyRkkn1cSZImhJ5rG9JGqw3dnPjGMHawRjf/NExABVKm+/VK0TxNPnbv/9T/evSC9ARMoRQgQMe3lQDNuIgMyrMj6d7p5KMuc6WtlrcWlkuxOCa7ixW13ZHtdPzammLamOzcqWSKnKlg9TODse90uDu5Xv24uV79uruTrhAmTod/amRvCAlXZDR6EzoNSBDouRywNpu1g6I6BFqzMj/M/W+UXUCsyYE2YR5xSC5/8lf/K6YoLfBHH1TlpzK0oHRbREBAKLzAUpZbaczQHdUBMxna1suNipZyOYoNqgtTQcpaSWoLGY3igSqgaQ0DVD08tB2iwGe2fPnz+zFi+cKWUrV6Vhb3bQaxsb7KrHhnNIldx+uVuNFWixWKosYgL/pHuEpER/K1EGcZo0wKEw1+f2v/ubkm86sKONDSFzsxlbLW22MUkithifgga6r7XimbT0JK9QS06ggltB81EeVwbYH2RmUngSSq/kjGQJWKJld/TobPdj9/Z09ffpf9uzZ11Vl4BmHw9lOZ8bpRysKIsDHcOIojpc71JxoBMr3XFGLA6sZ/AL6zZrpVGtJcESBgDVPLfm9r/76VFZsHKABZEoB2hqhY3mr3yll5JWUF6Y3WzcA7I2WlM+AEbwvBYYIaI7WKgJChyhaPLflbCNiFcUNgVXb2OGws6df/297+vS/7XDYihDtDyc7Hmh/O80a41R/0sNnkkkPP6HyAxoiFHZIJGw2t8IEKgMKF1F7PG0VmWgBSgMOa/7wT35jKjAAeUbzUlbKeUAN7y8Wa/X9bgAfh4FKnuqDNkd+u2JTCtxUa6GdyFfdWQZDvYltLAxzVlJRPESjlAVg3d1R/5/ZbndvR0SM3U4lsWnOljNmW/njb0yNQ8WZP/DSFmVywLG0xWJlTx6/Zk+evK7KgFOh6EhviKrqNs3EXZKv/flvT4Q+OaGSpj5+pbBerW7tZvUosMFgAAaX27PV7UleBAARR8hHPyBFbnI1hjwGsQArnTIbD1wwLUY1QIn2aoAXCVFQn3b1ECrD3csXdn+/VfOSJKNVFUd1vRomp7I+08hRucvu4ILjAVrla6+9rkhAoCV9oOcIsDhQa0FP/Nqf/86kQwXm/TXNOdOmwIDNzRMxPXmY+AsRwMYIS8piVGYFSOoa/WkNnRP2nQyApX201fsNjceFQ5WHU2Z0a3B3DEFn9/y9p/oJOaJKLRalDAAHicIJm4/TK+CctIC8ElOkIjg3cGme0I9tMPSbKqcUQHrySS5vX0F/APB287p4fgWqxsdp1NFxqPHgcRRtzM//YutMBBAJ8ZgqbtyPxMMJ/SUF/AErmCF4QPi/vHtpz8UQX2r6DDDbbNaK1OPxlR0O96K5GsMJOOCzgg6IEmZKzgFgs3SV3o26JAYXgFgtqALvqAq4jMVTmAuVPUSMzeaJGBcXixzbH5bwozE/z2N01Q8opCOEB5/ipKlTXz/tiQby2L9On+FBeD9UmE0hbLy8e2Evnj+13e6VGF1elHZ7C9GZW13vbb97KUKmp0sxgEbt4/OMbgRNjujc0A9ONYzBAxd8VqP6pSV/8Ee/PolesnlKCKSG3L95op+gapwT8HBFFYI5uiHoJKXP6XC10CTmw0ms0HVcnt3R3h94nhzGi/WZ54NPjv57qPCd3d09t/0OwlVLnaL729ysecTTzvAQdZVugEsXGA5GvIe4TqZiAPUgl0fmguz3tT/7yqQHj3kAOcjM8Gs2z799iPnBOL14tIOc1/hElQNAkXavo/X//7947ujlz73Oi6fHTida5UbscL/bihvIAA3VZBKar5ZLUeNxQHxBPQ6b1+nOdUTXS2OMvDBnpMlTZ5PxOenkL//6q5MeZqwAPzo/RAbIz0Ib8qdFHhiAL/d4rRYQgu5ETmydNaYSlOWrGcLsfjiCQwUCwWl2dru9QA8dgCHLyA53u50dqAgHusKzIg9jk89EbJoyO8AYrj9FHvsYWnvNNAZciA99XFJQc4JUHh+wSP7hn/5y4rwOL1I+YFHggNA9zt8pCnzCkIVQslyfR9rK1I3NZit9J84MxTyPRngYonoytOu08ZcvXyrniQKdKYxTEC/3osTwA4gXwIsmTz7TGnOKzzyAkzi0Sy/DynHxBE8NYZE/2RROtMNjtWHk7n8ADJSHVWuwTZwAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAAQAAAAEAIBgAAAKppcd4AAAABc1JHQgCuzhzpAAAbwklEQVR4XmVba68k11Xdp55d/bhzZ+6Mw38DISG+BMchtrFJHDsvh0zsCBEeUYREiBCRDF9AAgnEIyQBCwRIIKIgfkGceB4ez9zb3fWuQmutfarbiaV2z719u+qcdfZj7bV3hVde+dRsITULiV4WbJrN5nnma8JrHG2aJn/Xv+d5shnv02jzOFpqk2WJWZ4lliaJJSHYHBKbLZgliYUktTTPrShKW282dvvyjq2qygLvOdvheLC2bXi9NE2szFNbFRnfszRwLX0/WtuPNoyTjRPWh3XqNc1Yn68nrg0f2My1JEmwYNoT1p4mwbIstfDqK5+czbRASzLDv/U1XHS2aZptBADjyHcskAAAEL5Gm4fewjxanoQFgJAkNgMEApBakqaWZQCgsKpa263LS1tXGzMuLrG6PgqAebAkmGVpYoVvHj9jl9g0Nz9ONoz42Xx9+B3WNxEI/C02iU3gu9h8vAZ+H4JZmgbL0tTCZ159gQBgkQEAhJSLxgsAjJNffBAI2PBHABgHGwHANJ4sABcGALgO7haSMwByAnB569Kq9YYWkGaZNXVtXdfaNAHMCUbDU0phmLiEHwrWMwwTrQEg4N98cW06Yewc7/heYsG/P/kVZl03TfgKr736whxCoAUAgAAAQmLTHAxfiQAMQ7QAnXy0gBEA9J3N02BpCFZkqaVZylPF5icuHaeQWJZlbgGVXV7esfUaFoDf59Y0jQ1DZwaLmkfYH48hCdoI12jGjXbdYG03WNeN1g9yCW0edzq96846zvguYLH5VAB89tVP8A5JkJnKCnR6AGGEFcDkhkFmtmx+pjvgd1j4NAIAxICMG8W1cF1YERYms0utyHOr1msHYEsLyPKC5g8wZXv4joAIPEm8EAcm64fBmqazuu6s6Qae/mnz3L5cwN8BJH4mmLQonD42LysNr73yvCyAvoiTwwcKitEV4HvcaPSzxf8RjEa6AADANeBXACDegADwdGDWieUAoKrs8vYd26x3BCAvSmvblgDghBC0DEFtGvmuiKQ1dF1nx7qx47GjFcjvPWph74YAHWMAIxkth3EAJ48AzZcD8OmXPy43hZ/6h/g30YmZIQQGIJh+tAJYgn7WQhF8AqIJQZSJ0a2wBPwdNmOykNWqstsAYLPjfcqitKZprR96AgAQ44aZZQgi/L63pm0JAKwAcYDrOAt8zE4EBMDJ8pQFlJlwvwT7Y1zKLLz8qV+d9aH8VP/WH/CVCq0Zh7KkG0XkCAIWoEOAmfK2Cwi4HjPJMNg0j9zczwFQlgyCbddy0YzOcB8GPARZWB98vrO2aa3tOuv6nuAzM/mm9S6LAfCB0R/Wrc0DDVn7GQC//sIvzwk2GdMFAcACBAR8WbEBX0JUdyDOUpD8XJvHZvGC1UZfw4kAgBGBMkmY/+/cvrLNdocYbWVZ2rE+WNPUzOVYeJoiJQfru14bxqttres7j0cxDZ8sILqKgm7cPKHg2plLGExlCbhH+MQnfnFGTpSP4AN8MVpBSjDSLJdJ0xqU1uYJbmE6hUhK4s8kJMi3wQAu/os8At+vVpXdubqy7faCn5VFYfvDjR3rI80c4CEwZmnG00eGaNqGcQKfk5e4C9I9/MS5v2XzyJ/6Of5/efd4RwCedwDgdokDIRMBTgAAAS23LM0tyU4A2AzTdhAAgJ+8CAkC48iFwQVwJTG1kdddrVZ29+qebXc7bbbI7Pr6mR2PR572PM6W5bmtypI+DCsAOAAClgC3EACe3pYg7ofoP3sWXN4YG/AVcA8Eaxzq87/2S3OSBYPFpXgHCO4jsHfwgjQBAIXcgRZAnGn2AAHrmBaXQGrSK1oBSZFnA3yVANy9Z9vNjhEcWePDZ09tvz84GQJVTWy1Kmkd8C24x7GuFSzdCmJ6VQbzQEfKy0TKzZ5TemYj5yQpqDl4wPMfRwzQ6RMAgqBgGMAQ8X9YQQJLyOQGDoDqBlmBwFBgFAAyVdxROVykBNeNFrDZbOnzqBE+fPrUbm725AM4XeBc5BlTZp5nTJE1099RIMHNeLb08MV9eRdPg9FKFmtxxLB+xifs8YXnf4VpEOknAARYgNNEgkBXUCYQMEpzMZLOBlcAY5QViDlOCnrwVe2cm2JwShOrypVdXd217WZr4zDR358sALROgj5qBbg3rgk3QCxAVgAIZKWMATr1GBuYmv1zBWnPDDR/f2EfL7/4cdoKg0WknR4Ql7R4Zl7MGE6Y8A5XgAuwOGGBEgsoJySRB3gdQQsgAFe2Xe8IVk4Antn+IBeIBRfWhaII5KksC8uYUielQwZFxANZ2lK5npG0aP5LXRBPHgBEzvP6ay/NLGsjm/JCwjOGM0T3MX4xWAK+HxkjY0HCjYOX40RjFclzcTrMfD6NdB8Qn6s7V4oB00Qm+PTZtR0OKIg6m6aBVgCzx9oAGig0XAdugWCKoHhwwFCnMMgC/EiE3OW0D1BgnXokRMx0sIAvv/n6DBorECKNlCmdp5eYV5kp6CriBqwk6QbBeq/KlBpFjlTvyzVoaSHwRO/cvmO7zQUXDV7w7HpPH48AwIT7vrMeFjHPlme5raqVVasVN9K1jR0O+s55nUJXY6bWpmMhJUKkspiAILgjCL791pdmIEqzo5AQ/Ui5NtLJWC94daugI57pGkJCNyA9Zb0unh4p9sLCTEIEqPDFVgCgLL6+2dvxWJMOy38BQEtAEO3yorAiL1hNIiiiRqAbNGCQPQkSsgPWHLMCY3V0b6ySAJzqFfCM8Pbbb7q0otM/RU4HgNVYzK9el0d3CU5CXPEhAOf1uUdqptUlcILqZgJg5wBUa7u+ubHD8aj0iWXHbMIKEXGgYJmNU1XBBddDfQB63FBQQZ0AdxAd1uZVCYoZ4rsAD9aEmoSHeP+rXwKJ9RJUysu5JIbPUOfrxHVRyWMqcFiy4uJZxrqrH6DagPq6fnDOxNwMkfdBhQEArAQCydPrZ1Y3kMTmhT3iGogrotUo0z3tQXkikYF7CYTj4WA1RZVuYYrRhVmKO3DYeO56Ba3jy7/1hVl+EYmDMmvMsdTOvLrD36HchHkOfS9zZRkM6pozl44kRsGlKwCkvMxIvQgjOS1gt70gUEVZEoCuhxwGF8kJOIDue7A+pTFVfjGd+YlC1wsz6wSQJbFFWAKKL1lA1ASjEgSLjNVmePPLn59Zg7vRRJg9MaqGz9KTybgFAIChhxDSq+SkdgU5LfEKXMcllijJChsgwc5ySmJQhLqu54Y/ePIBTxs+juIIVoL/YNJLkHPNT1WprJVpkiAYdQkAIH0R+sKZOzhw0tZc8J1GC1988/Mz1s70RgrsFuAERv4P2Ur+RyqMs4yCKDIIJCzVnwo5M3xebsHfuXKr8hXVXsJCqChWNHuY96NHD60bepbKKJZgUVgPdUkwSy+ApAgjSGoT+BvEgyKDnGfWdo3Vx4PHg0iWXM90oRQWozVNFt74wutzNHOgDkRVOsrfVUBITjrn28SBn0uKligh8FR+qhwOrO31UrpUZqjWW8pvh2PN0370+DELIVgAQMDvSKGjEOPFD061dwFU+T2liwIE3BOWCReAXhAth2SJ2cUV4zN5PHz2c6/N0UfyFHqe58cIgvNq1xN0sktQlDWcJCilUfF/5WL2BBYABCyqy/XmgorR4XCkpTx69MiarqU1AAAUXmSWXlqr7I1aBKK8KHvGGkWWKeuWQgXtkPpBh/TYOQBijOe6QfjsG5+RJOYFxaKZRS3abUE8wDUFbMKppAwlam9nAmTkCN4TwOoUPEdLktx2F3csz1e0AGzgwaP3mctZeucFrQS1lERPV5wE91JU8fRZoToA7sL4G7gN4gC4AoNi39NS2aagNOZK8xtv/CZF0UV4/1n5IIoKbhFaAC6i3K7KkImbL0nZajyk6BLlmaW5ymiYYd9BG8xst7uyotxQ36MFPH7ArMLTBEGB/0NvoK/7mfF0o8h5Fpdc64uqDxVt5xEgRwKgk88vjRLFtvDG6y87ANK6ZCGxweAJcQkJ2n4MlIvMdEY6IvMi8ciC5WVqWQEAgA/abCBWhe12d61c7ShulquVPXn6IZkf+gCxr8C7U5OQNBfTqcplL9v9gFh/LK0ypWJ1tSIrRbBWmb2IpHCbN15/kTFgif4uJCx7iowAwW75LLKEeDLnWryTFTezCEKeS2eEyJKEwi4u7tpqdUFpe3dxQU0QvooIDeuRUcnX2VcEADDtsx4lLC7WMNIiHX7m+YwpOfIQ+j6qXeobp25T+OIXPyUAliByajjGqB4NgGQpRn7X4iJ9Pn+PyjApKPJ0mVpVZZYXubSFUNh6fceKcsfuzsWtC6vrvfXoDIF5eqpVwHIl1+t9KdHepGENEztVnoTQDEPQTXLykqgCqw8Q3TX2DSYL97/y4hy1wBjPYKYIQIoZUeIGj0YfD3+FOgGcXSckU0XFhwIGahDyrocFCpzBqnVq1RoMD6eSW7W6a0Vxy9p2sO1ua/vDM+s7NEc95Tr/OHcz1fwnNZhUfGmLudu6WsW0G+Vv6heZ6pHz9A7Kdv/+y2KCXt1xI+TyEDpVzpJHZ5CycitX8EfQ2t5CAtVGF0eOR96tSUlbG7rZhl60OU0nKyuzalNQUpun1NbVPcuzu1bXo202G7u5UQwQ5zhv0kT7E6WO6s652HFSgc5lMsl5avuddYIAyhkI4Stf+Q0GwVgv6zRV8LC9HKWwFHV8akUBf8RIATam05LogWCDhgXSTm/ziLyHV28WesuKSQExL+if6+pjFuZ7djhMlMb2+6eKAV65RXONpIykyPW+hbV7I3TRAD14O4ldCNnSCFkkPg/44K337ysNnvS/U8rRYnDiQNBLSqhCmVmWTQ6EOACaHmhtjxMUndFSoG5IfeDyo80BAJgVBcrQwqryF2ye7toRAGy3djxcU96KjdAlNUdySWf8eYnf1dYYs0XKzuRSfuW8I8S/dBtAffDW2yBC3h1e3CDq7T44QlaH01aExhRIXiBd4eLi5gMB8NY2SuQwsSxSKlI6y4pgOSwoqazMn7NpvLT6MNrFxc6aem/jiCDozdGz3caco5AYLVv5L9KQhY5w+6cs5TnpjN67/ZBbzBbe/tqnFwuQ2OnFDvksNo5AgA2NlqSoviBXAwC4gqwFhcow9idKbBAxkHO1ZGiG85yx5GSfIV1bVd6zedhafRzt8taFdd2eleVHUjJyuUu25yfLPE7ChIgfN+wUl4XOzwKwkBdpJOczBL/99VddFneRkFC60sN2L9gTRI+BYGQp9DlYgWoG/DewH6CjoTgNS2CbWhtGPo7jLPg5zypbr65sHnbWAIDLC2tbAAAe4FSdZ+1FFAspqdYkMi7fs4nsG54MaWdm7/IU5JylLxT67OfYUvudb7w0xy4qLqwFUGH3uSmZPpsndAMMQ8XmidYF1oUSWM1Rb5nPsWfozRMvhZE1yqKy3fqe2XhpzXGy3cXWmuZGEyKxdmb2yagdMIczTmkN6mUCCJ10rA8IxlmYUFp3hzg7+SWD4G+//oefdAvQRVUk+FTGEvhi09QbKATJj8o1Q+CuSZLYG/CurdNZpFXIaBkAKCvbVM9ZmC6tPioI1g1igLQFxJJTAPZGjA86kWLHZi4P7FSkRclMge8UL2LWOIFxqi/C733r+VldYRUwqvllapqPcKHEA6TLANyoeD2Ym/iCukJqlMaZothmR78Av8+p4JRWrZ4zm6+sqWfbbLdqiCCVYtzG+wI6O5e5vQRnFYqNewV4qgRdpXbaEIE5aRsnWzmlUbPwzT99gS4Qld84J3CeOjhrtURlnSTk72lAcARYUmFjcKEl4PNJ7S3NDcYGBRojua1XH7NhuLLjIdh2d0uEyfU9bBqANM3R+qHlYAWu/pHNxv7/eVM0KlpuFTGVMoEufF6RKv4q/NE7L3FIammJuzbAzjDDNmQuBTv5Dro/mNrQScWgp7jhMSFq8xi8YtWm0lj3MCvQ5Cju2jjcsboOtttdErzNemOb7YZdIOh6H374ge0P12SIvNdStClDsfRxYCOPiXkxCjL8TrResnbnE3F87tvvgAe4bu8FUdwJANDYnCa0SHbY8JDERC0wFuvR7zx+oHrTpIlGL3UasrQsya0kAJdWNwIAGG+3O9ttd8wuEDJubq4JQNc1vLcqOa1Fc0kqhk7zQJHgKD3TfeMLKXMBQn9HcP7knc/NnOJMMdp20gPBnxGFI82FKQ54jercaBLzRFFi/iYBcrOUDWF4Sto+rYpcI7csu7JhuLSmDXbr8jZb5KDEVbXipuACh+PebvY3VjdHjdcETaBRnvfeIQ4CwTMGTvm8MtLS3/S4QStxJWgZlfnOn39phiKj+T4MQqhuV0sLiRYcv7e+b+SPkwjPQsHiQCPRPXWZXVkhIcKsH6Y+ArSAJLeQFJamt63rd9Z2qd299xxFEQxEqB+gxigkretoBSyUIMWJrJEA8e9kjXhXIaXibBny+IgLeCw4jwd/9hf3Z3ZL0HfLMQskPX75jz4DOtsRAKYqbEpQ+k1FiVEmn3qMCFzQ/2QHKI7mEdUBylTc55Z1/draNtjV3ecY2WHSIwFGGlTxA1eAWIINIvhqagVVIUpunDwau7ovAyWZpuYUBVacfVR1uPDGSIT+6m9/d4b5x+FGdk3OSkhOhQT0/Grr+poFCxbGkRmcJi+K/I2gCPfQ/A6HnpPBsjyxMOc2DpnZVNg8abQmJJU1TWF1Pdn24sL6EeNvtQe7SGvj5NmgTk6PjQp3nvyoQByHKTnTwvQIeR8kSkoSfsY+AA6+LPXIu0z/+P0/FhE6Sx3xApgY4elPrXUDGpdwAeT90lblhWVpybzddnsbp0ZRmSEPN+gsSaDyBpvG1MYht8TWlmcrxoFpSm2/n+1m37HtXXcHltKaKPWhKmp/7vdJYn2Ldnmn6yMY01qUiU5T4do0GqBwbVoBq1kNe2HcJ5I29BjCD979zkwUfR43qtnKfBI+sBnejOktsyLf2qrAiFtidXNtTXvjlZxMnPQFvD5BcYPFAgTcfGWrcmtFUWFm1J5dD3Z93Vi5XlndCgCOx2KDrkkwa2SF5WlO6wMAcT4ZVkAA2PyMVBm9gpQujTkENmeUE3320QfC0RLGfr77g2/N6vCeVCH6M30a/g4TixPc0vOydGVpUrAM7nhymNxCitR4DF0A0yCJ1FtqdAYdAOBhyqOyJFnbzT7Ys5vWilUpAGhhktvigAZ8XiNtqgmkLI8uvAAwAACrxMkrx3MiPFUbnB0mDnK4/5PCa3SPLby///432B2Gv9BviJhPa9O3FWQ0dY2NSJ5WEaR5wL6f2NnVYJQ/uWETp8cZWxKdBHuQbL8hfuzsZp/Z9U1vaZFb3UoUleaHzm4cecU8AII03G7N1Ay3a1r0/46yPO8Rwv/F2KT/Y+OKBQKBkcU7zZL/g4V/evf3Z87x5yveBDegCfIklGNxEw4asSurE/LCmuIpAOBgwzJNriQMX1ZqVfsqTTATCH+EYru165vMnl13BODY7GlJur4PWbKJAX8uWEFW1ZZrRJPjWN9Y0+7JSKX3e7bhyO5pgJtMNCvUbEXs8VZf7HmGd//9mzO6tEWOAIXJzBglxeUR+EBFNbXtavBSAfpsMPt3g3eMETRl9qfJcZy6miPq4+VmhqEIsydPG06gHpobdnbj5jXfr8EmPmqTlVaWayvykq7XNJgoq5X/Kb74XKOr04gXGNORypxRi0QQjJJSpNDhP/772zM376dPQdRn8CVz15SrEXVj1/M0SSo+zhgB8ZPpD6YGVgkzF2mRfzo5cSY4z6V98GFrjz+4gSvaob62lkFQOoTKXhVSMQYwsGUlrVwzQRiYRIzymsDrlRgke+cOosUujfsYsNhvbuG/fvjOnOfV2enrlGH6ID7I/xyC4LBk9GdEdMQDpUl8Ll/EwmFqMDlMc8GdQGqgH+IEkAKlCmMm+sHDp/bw8RObw7wAAJIV5xFA0E4gIKhhSAoBNGW5Hbu+oMmRDMV6BewVrzjSH5s1kQLjGsgu4X9+9JczLgo0EPnnqbNh1MZBfyc+xQUTXjGHI/dnaUUAWB3S78XeZLK5JfibvKRfIoWCDmf4br6xkKzIAZqmt/d+8r49ePjQpjApCLokhpjEYaY8Z0pDEaPUVlrua41Ps+GgmB6ZiTAyp2ePINGzRjhrpHDUxjUj1jqwgP/9v7+Z6R8UQ/EFPLuDcTVPMUhdWWVFvrE8XVuWIRKv1Ali1AcHh0Qu5YZVFnwO6KItwDSF7hCGHtYWAnw42PHY2I9//J69//ABqg1r+iODF8bhMEmKIWkMS8B9pAXoGYIsw/iM2ueU4KhC4ZmCmu21lsG0/ggAsc+hBor0SnW8Mgs//NFfe2sMgQRNDERV0F34lQTMsthanm0tS9eWJCAxmOMfbOBUJ1raaHjkPtExcXYQiyT1XJqbiAkwfww+GEfiAMCDh+/bFEbrBozKpJwbwgQphiQAAE7eS0lRdMaTEyfAIAQAQMkMAI7HG2vbAw9Q5C3OivkjP+QY6hSz6Pu3//zOjPSUZpPlGdJUfKoKSGMROyuyjW++tGAFyQ5H1gdlB2wekTp2b/SkCRYp1sUARB4Ot4F2OHIy5L333rOHjx8y6E02WVHknB3C+Bx0Q9JZ7+YoANO+lgFuRHkOT3ctJ0rb9sj0KABQuCl9s0nKEhkZSpUjXJvB87v//AczZpmKIrGyzGxVrKwsocrs+MrzDYMXTi9YTvJA7Q9phgAMHGjgiVO9jcUHyt4oipwey8V6EMAw5/uTn/7EPnjyxIoSvB3PEq0oikAZQrpbulWxs7MoNf4oLXqR9dGOeOy2a6xra4qrbXOwYZSIQv4CeR38hjMDqCEQHxC3Jgt/972vzWh0lGXOMfaq2lm1umVlceu0+YDTRURHESOShNNHwAERYUTNV37qID6lpXzAQqYaqzDSUT7+OtACfvr+T+3ps2eG5wZW64pPiKyrysoSwdafZ15aW16gOwtlQwbj8yiXMRXWYFK0tvoIAeWG6TtqF1g72uV8csU1BLHNwcI//OCtGU1PBJ2qXFu1gv9dWJlv6QLYSAi5zTM2j6DmIgQQHmqalPxyZQnrfDDKDb+LTCDae9oMTiRawIMHD+xmv7eLW5dskQMAzANneRQ0KOIt2kvU82MngOIsAmDbWM0nSvZ2OODRG8hoRwZCETgUQl4E8RE+FVGsgb737lszJsMgVIJursoNXSDP10x9yOvYPPp7elgJQQQSNmgrMgCGCHBxIIxcD0a5tbzYWFlsrCiRYj1qO1GJADx8+JAzQrdv3+aUCA6Bktcyjxif9pIgG4XXCATnBemOeJS2cQktAlDb0EvC06QZvu6db0p0ng3+5V+/OusJitynsVWtqWIrWQANY/BpLa8ODdnCCyRuCukPAKCWwOlj81tbr2/ZagVLKlWMOE+HpI5R90ePHtOMb9+5Y7sdYg44hJRkCagKejxxf94nKlXqQEFBkj6Jf8MNbvbXtARxApTP4AdgjYMNePlkvFSt2f4fUAQboPzflJwAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAAIAAAACAIBgAAAHN6evQAAAABc1JHQgCuzhzpAAAHW0lEQVRYR01XDY8c1RGs9zUz67XDh/JDg4SifBhzBxg7MggBUhKjSEiRICRCUZTwx4C73Z2Z917nqnpmxUkj765nXldXV1f3hD+9embWO9CBAAAGmPmHGAJSCAjB0DuvjtYM3XjxjgCEqIvPtNawrivW2vS5topaK3pvOjfGgJySrsgrBoSPX71n1jqsGwIDXy8C8Ici4EFbdyAMtv1LEBYCrAO1eWAHaGgEsK76LcCQYkTOGTknJAZncs8/fc94sDVjFEQLCBY2GvgQdCNZ6nv2nQA6ajPwMWHeQJE7hOAA6op1XdA3AMy85IgYo9gls+H5ZyyBZ8fTGNxB+KlEzhAEQPrNeDkDa+toqldAZzn4MUaEEFWutS6oy4Leq1jMKW6Zb+UmgI8/vzEdysNbR2hwEDpwC9gbWm06dAfAMjjVLIFoYvTtXwJsWJdFDJgAuKaUtUgyL8GLLz5USmSBDOyl8DIweVK/iYmlkkB3qZCfgBAZOClzAmnWXYwCMMM6NSCO9KQDAFIigM+fexU9JYFgrfVZ9/P7LqbuvylbB0EyqegQswDwN4pxYf2XWSIkALbZHpiZs7tiiggvPiWAPSXvBqm9NbWm6i8Wmuj3+Azkbamyp4wYk3fDVhq1YV0lRD4vPbGrVIYgLVCM4aOXH1By12wZhMF48RD+7Q/t2fMAUd2quoGfVQawDAzAzwTYlL0AkPodAMum+x4evf3wmW2a3EzIhUYANBH+pZSQcnKQZIAGQQC9oXbvex4YQ0ZKWf3OW0ShkSUvnUqgfrjWD+HZzR+d2E5SdyfcXa+KUhlHZo15BO8zGN2RR6tTujonBBpMlLikS7/D6y/M7K0EsyjjEqCn7/9Bbbh3grKn6RDQ1oYUWaKB7ACYFY1CUTZQ7s1yzpQJgMHJYNNtqndIQM/oPQK8CPrp+793H9i83u3Wy3D92wVDXiUo10agijcQYkGCDEjJEGKFBQY3ZLYbdWEJvSa0lmDdHTc8u3EALjq6nRvOVXxezI0vaV9ZKd2tlTy7jXZl2xEiQZiyl+/TU3tCqwGtxi2Hh19vbn9nPuWo/I6+tYyLj/Vky3V0NHRSGlhPFnBzMlKuum8dQAACRrlJCJttkGUG5wX05nYfbj/4rVHtBOD96g/sozNmthjL1WBYQa+mjXrWnrkLzK3Yv7Mj2DX+f7sQ+Z0g+gZCsW4/etfo15zdHDg6JnBsJuQSdREEM2+dplK3IUXKOdOTsnTJeK0156V4935nwYeWANDjuCJQBzfP3zFOLU4sakHBNTYThiEhDwkhMQdqY18ufCJ6MGZNAPzN3Y0WK/NyJTlDDM5G4e4je+D3B4C3L35jzIy114AAR2ZCGaIuMZDY0SxBv3benrGyUhd7K7NcWmJUBvcG9j7pbpVa21ucOuZG9Mm71kirzIK7gNcx54A0mADIA5hZdFPyem800/9pOfKOCoNvP7wlp4ySJqRYpLF5njUlr3rjKZ988dRoGO6u5NKJC2yl1BGzydloLiVnlDxI8TmNGIYDci7SASfmvJ6x1jO6rWrTFDOGfEAKg2x9ns+Yl1kz5OqQf/7qpdEs1DZcpVrHsi5otjz4ZEVIDakwc7JCbQwYygHH6U08Ob6NcTzK/80a7k8/4f78I9Z25lDetinSHzXiCYJi98nKkhrCt9+9NtWM9/WOy+WE+9MdlnoGYkXKhjLw8l4nnQ7gLTw5/hqHw68wDpMon5czLvMdluUkEK1T3BzxlMcuQp+S7rYN4b//+5bro74sywV39z/hdPoZtS2IqSMXwzAGlOItF2JBySOm4QmOh7fw+Pg2Hj16LDOikFubBWJefsayXtAaW1dbgwtSK7zvF10AfvjGeBNXp8t8j/P5XkCoi1ygViwSIuteENOgsUsQ4/AYh+kNHI9vYCjFdwBbsSx3uCw/olbWm9n6XHHfcHf0+dMR/v2fr22ts4LOywWLRLIihIZhiBinQeLjOGbQnEcfu6lIiCmNmEYXo2a/rajtgtbvZVq/fJlRqWlSwuDjP/zz+6+sthV1XaRQLZFWkRIwjhnjOKKUooB70FIoxEnf3YySDIxl7J2uehEQnsHOovCofL1t0Sc0HX2xCf/412tj7WjHbBMC4LAZhoxpYqsNyo7iY+alMONHOEyPPGta7PZa5olcUBvPaCiFG1KUhbc6e801Z3zIaZn95ru/GlerZSb9F5lJLgnTNGEaSTkzp5t5pux9Bp+mA4ZhFJXciriAsoPOFwp41kgmcywf9dQ7vcHHuK/kSWeHr//+5VWEpIn1GYdBABigbEZDqmks48jfJwxlAEvhc6BhrasETC9gC8fYlAhniu8snDMMziHirkunDK//9srUEp2z3l+fhnHANE4Yx4OCMDCzp+EQAGcFO4uC0sRhW/WGy3zC6XznRhToIdyOfAXnVsRLLyPcJ8gEB9lfXr80rVOa8XQ7BiEDB0zjUSwQAG8mZXTC/b3PX7uZkb8/sgzLesLaZ+0NMfHMjkTKI8RG0b64bcwP4P8PRsgwgxEULxgAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAAIAAAACAIBgAAAHN6evQAAAABc1JHQgCuzhzpAAAHx0lEQVRYR01Xa48cVxE9t/vefs2sYyP+JokUOxaYh4jEQ0JEgHg4SMAHEE7sgJEiFAX+EJG93p1Xvx8XnVPda691PTvTs12nTp06Ve1+98nHscgzBO+RJCkAhzk6RDjwJ+FxERHAsixYlhkxAs4BaeKQONh7fcM+d86tfw19qivrf3afiHk97pc/+2EssoAQPLz3cATheBI4lyBNLBB/lnk2AArEIFEA7CohR32e8Np6FDBGxBgVmEkw+DTbq/vJj5/EPPPICCD1SMgEXxO+pghpguDJg0NU9ovSfDfoBsI5AnJIkkSH32PgeV4wzROmabaj94uAuR/94Lsxy4IB8CnS1CP1GVJvgBg8+BSJIwgGsCzFdbT3FtSYUFnSFOn6AQOO44CuHzAMI8ZpsuAbgCff+yiGkMFABHgfkPIELzA+TeCVkdPNddJU74mBsPiZY2DqRddXBhAxDAPattXpBWAWIyyFtPP48YcGIGTI8xxZniOEFURqOhDFq+jIBvViojURvq15RHpXAtNM27Wo6xpN22EYp3eCRywE8OjhB1GZhiAAZVmhKEtkWaZMCUASZr0cVJI8I9ggdki7jkRnQKXZuCj7uqlR1w3arhf1mxi3V/fo4fvRlJuI+rwosNvtsdtV+p06IL2mcsCnTnqhcH2abj0oFuyHYFnjWbRfLhfUpL8fMZPzaF2hlp4XuMcffWBNvmZhIHJU1R77/R5lWYpyq6tpQGWgYFdgKkNiiudN53lW9k1d49I0aNsewzSp5mpntqK0MBsAMxHrXbYPSxKyTOWodnsUZaGsk5Q1TtQtmadgCcoolyjhdPOhH9A0LZq2Rdd179BPwiKmacI4jQLhHn74HfM9BaeaaT4GIssK5GWJPC/gQ4qIWZRn3qMiuLJACGxdazu1aozohx5N3aDrO3Rdj67v1X5UP7Mex1GtOZGBR4/ej0S1CUnK3lgIGYqCJQiqG1Hzu8EH7HY77HcVSnaO2pdlSuWG8zJjUHBm36LtOpWEmVv2k0Dwd/fk+49XBlQESZpspIl1hkToYOiXRTWUTrJcXVPo8H1mVr4a1DxPGIYebcsOaAVAJqRDBowF99OffxwtewssHUss1IS12RJn8/2EdAeVhuCYMTuBmmBLUgucFRLZvChA31v2wzisWY+YGFwamOB+9ZtfcFLcBZfnW9uLbv5zWCQ26SIvUJb3EEJ+1/v88rIw4xZ9WytzUa0sab2TBLcF5XubqhHu908/seZUAeh8NpI5uSJIeeRgRJJEZZzlO1TlA3if26iNbKcR49CiqQ9o6hO6tl2Dj+Z8BBhNgCwNgytpiv5Pf/m1GDczsilI6m2EEgCn32y68BTle8izK7gkrPsBKaXYjqgvB1zOR1nvNA7m90xjBcD76Z4r4+wa97dnv113DxsgZED2u9JPFqSRhDOgQp7fh09L7QvMZBw79P0RbXtA3Z5wOZ9wqS8Smu7BzLUDzLbg3Fm2ta37/B9/WAGYEVkptmm/+TzFR9Wz9ldI0iCNTFOPrj2iaW/R9gc07Ql1e5bqCcAWkBnzZNOPSmdQ7Qvrq3vxr6daqFYzvrNLs2eC2pRfoWD2obS1bZ7Qd2dcLreom4NK0A4njEuLOdJkWH8uH7PsWdWUxtbM3wL41LqObrAucKJEsGhKpL5Anl/pMHtenMYebXPE6XSD4+lW9I9TB+cnJH7EjAHzwrOqHVxqqC+aFSes9b77/J+fWnxzAwHZDIk0aSsKtNy9lM9NSetWnNB1F9zcXOPNzWvVfV5GuGSG8wOQ9JjjKPuWgOkfvtQ9bFUzQbrPvthKYMG3fW8Lzj+0Q6fLUZQVyoI3StF1LV69/h9evf4Gp/NRxhPjhIXZo8MSJ8AtmhNZxl1jj6rYwwevFp+XCe6zF0/pB9tasYqQfc+th4E9klUHZbXD/fvfwq66Egtt1+D6+hVeX3+D04kibDBOHDw9hrHDNA9q4yRNZNs7jvjdPVRVdQfCPXu+MUDbXWsTbSBxP9SOyO3YZ7i6eg8PHnwbRVHJujn1LpeTsj8eDzgcb1HTiHrufy2GsVcXaJMKQbvFfneF/W6PUiBSuGcv/hhTeXwGh9Tsl/u6c2v9bdLRevf7e7i6uq/xzOsaOPL4AefzCW9urnF7eIO6YSs26HtrR5oRFxpOTU7XXbXTNOXq556//HPMAheOQird1mUaBgVoY5YA6IIVqmqn32mjXLs0LWKU+93cvsHh8AbnyxFNcxETGwssshYZ3YfzpBQL7uWXf41FXiHPKtVdQlSHcCnhlLNHNltQMgUnndwbNnPh9znzzyrF2pb12SbhyMHEPWIRaN4vyzMb42UB9+VXf495vkMeKniWQVuwuaI9F9iMFxhpgmyYLgypPetx62nbGqfTEYfjtbTA0hDAOPYyJX6ZguRKz2SKIof799fP1xKUBoBC1DPAGogilBnx+XHdAfjU5P3doxcXDW0/bYPL5Yzj6Y2mIrtgHAb0QydDMmPj9m1/z4ch99V/vojB5/Bpbs+DWjp5MdfiwaDMPM/pA0EZ2JORMcDM+mFYAfAh5Izz5YCmOcktx4mPZL2WD7aDWF2ZVBd8/d+XMU2YpRkOA1Ll3Ihz7oNEG/jYZg8qNjFtdbMN9y0A1rxpalzqkxgYBvqCdYn2v3WdMx3ZCvd/ODjnglpda2UAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAAIAAAACAIBgAAAHN6evQAAAABc1JHQgCuzhzpAAAILElEQVRYR4VXW48cRxk91fee2+46zr+HB3hAIYoiApibCUYKKFJeokh5AUQk4tjGuzszfb9UVVfB+apndyUesNSatWa66tS5fN9X6vM/v/CbcofNZouy3CLLckRRBO8cvF+QpgnyvEQcJfBQcIuDtRbzrDEMA9q2wzD08nfXNuj7FrMeAW8RJwppGiOJY0ABi1uwLBbOLXDOwXkP9buXv/BlsUW52WJTbpHnBeI4hpcfWSRJjLLYIElyeO9hzIJ5njGOI4Z+wDCM0FpjmifZnGAMAWBBlsXI8wxJmnJ/mMXAmBnWGizLIoDUL1/8PAAot8JCwdMmCbx38M4gjpUAy7ISzik5ed/36LoOIzc3RtjSekbXt5jGAcbO4KGLPEOe58iyVBjQZsY8E/AMuxCEg/rk05/6otisAHbraQlgWSWIsCk3KIs9oGJM04ymblDXNYZxhPeQf9M0omtr2cB7izxLUZTcPENKBiIFrSeMUy8SXVhQH338Y0/aCWKz2csnNSMApRzyPMVuu8N2e4U4zqG1RdPUON7fCxNQEZSKMA492q4SAJECNpsS5aaUtZIkEQa48Th10CsA+kB99PGPfJrmoj03z7MymNA7oXG7KXF1dY3d7hppWsBah65tcXd3i6ZpBEAcxRjHAU1bYRp7ef9w2AuIKI4Qx1xvwawHTFMvPghmFAl+4klRkqRIkwxxQsMoKAVkaYLD4YBnz54LA/zeeYjjT8d7VFUlOsZxIrrW9QldVwuA/f6AsiwQJ/EKwEKbEcaMoj/TJCn47Fc/89yciyglWwtdaZxgs93i5voZrq8/QFnuhEpqTteTBUoxTrMsxhPV9RHn8704vCw3KMpC9M/yFJHyWJyWZJFdiaHzUL/+7Sc+iRNEcQKuzoxSw7IocH19g5ub59jtrpBljGcihuMGjKJkv+sx9L1oXzcnnE93wgajVzABeS5AyCaUA8AHsjmBq5d/+o2PxEiQOBFhHAHb7Q7Pnn345PSpUEt6eAJrCWJC27Q4Hu/RNGd0XYNh7ERjHiZJE5RlKWwkaSwHUyrEhkzyUX/98gueO1SnxUr2Wb0OuwNubj7E/nATKqFIRAB8MTDFmlDXFd69/QFVdZScC72LFRBclwBocIJnTYn4cB2vxE/q62++9lyMxcMaLQCKnOa7xvXVc2y2h7U8x+KRwAABOJGhqk549/Y12qaChxegzjv0Qw1rp1AHxGMx0jST//M3PL144G//+HsAYGYpFNaMyNJYondFAOUeSZo90B/Kjhf3U+vz+Yy723cYx06ARXEshmzaI7QexIQEkKWZxLwotwKA77OnqH99/z3VkGhM04B57pFEwOFwhavDByjE/anQLwSs+gUjTjge73C8fy+VkNVNQqQ8hrHBNA+IIiWblwWL2UHk5I8I3hgN9e79e08+rNVSJvU8IEki7HdX2O+v5YUoSmQhWf2JB9iQjsfb1fmTpIOHodsn3WGae8l6EjMRWxTFNtQS52C0hrEG6nQ++WVtJmTALUZ02m330h94+ii66H9h4GLCWQrPPI1iSkogCVlmdEOFdjgJCwpsyYVUTLZ0v4QI8veqbhrxgDYa1hgwaZwJ6NyLeS7mC9tfUhA8wIenvsjD01kbAByrN6jbo2gdqVQYlI2lHDANCVTbdZ5I2JspBd3KGYADCKNzof6p/sGEi1REmjdiBZUSwWIS/NT2J9ydXqNqbsUrQT4F7ziZKCjPPVKoru99WJz1X8mGzCk/Lye/fM+NQ/sNALgwH7Im3TOKRLJlMai7OxzPr1G395j1JNbwngBjRJ5VN0KEBKrve8+Tsl8LAHF76AfSF+Rgq/1X+oWxZZHUsLux23EHfjKyizNoCaB6jaY9CUjWCBYf5dkXUvglCnWg6zpP2gUEqXzyhM0DO0/1fwpgGFqRQDpqlorRCKAb7nGufxAGGNFAHdtyDOUTUHFjLFTTNF5aZvS/IMLxH5m4SHABQPeP0v8V0ixdJ58Iy6IxjGecqjdoOjYnAgiLManyWJqYMTyd1nmAAPg81f4iwQMBaxN5jCHLd+j54QChqGn0wxlV/QZdfy8lWTqt52TMlHgYSxMbqNvbW8/cs9dfFnmU4wn9xLLOf5dmxBSw7LJwXd4JdUBjHCtUzTt03S2sHWhbeOdh+djwaErw9u2bFQCnov8D4qEOcADhCThi69DlVg9JQhzLeoumfY+u/ze0buC9FiOSfiMA+OmgvvvunyIBWWAB4t/BE2HYDKZ8ZIKL8CQ8uZYYcjDlWEYQQQbnDMapQdvdoutvoU0LQIf4PjAAYUF9++03nm1ShtKylClGQHBKepKMRxdQzjCC0d2MofcGcaRkCqKMy8Lx+yRR7IcTtOl5LZEmFd4NmxsLqK+++lIYKIoSmw1Hc96OAgjpATTlOieKDaSgBAkIgC1cYZFBgxLSxMsyYBzv5fT9eJZpmHMGx3wQhGMSFKxVUF/85ZUMpUVRhPl/xyZUPrIgEkSPpWAFwJFM5gcbbk8sQqFsL2K6aT5iGN5jnCoBwOrI7zgTBhNHcEsE9erVHzxrf16U2O322O/2KMrNw+Ry8cKTWhTczLFLz9IFeYEN1ZCjnYGxbOtnTPMd5rmGNmzVer1thfuAc5yIIqg/fv7CU29KsNtxaNjJ5PJgyJiteC3PqxG4wGKtNCMai3e/y+VjWShLh1lXMKYSMIslUAvH657jndDAcpSnIr9/+ZmndjTh5YpeFqUwkqX5qmuYhpmIyzzI1k0ATEheZOt1zsKaAbNuoU0Dazs4N62zQrjuO2+kXTO+5r9Z/A+WFQVAFxJTOwAAAABJRU5ErkJggolQTkcNChoKAAAADUlIRFIAAAAgAAAAIAgGAAAAc3p69AAAAAFzUkdCAK7OHOkAAAcgSURBVFhHTZfpchzHEYSze44F6cf2D5uiZEbIlkAGTQUv+fksEZidow/py5qFEIgNzO7sTmdnZmVVp/s3f++9d3VJOWWlnKWUlNLg62EYNI2DxjErKWndi7ajqSfujaq1anl40OPjg7b1qmPfVWpVqV1VSTqfmfOgnLNy5tlSYsXelT7+/Kq33rj2F4YhK6XsBQAxePFB0zD4h9ft0LIeasoaAdCalsdHLY8PWpdF27ZrL0WlSU0KAGwon/+9OSmxYG9Kv77/IRjo3eh4aB6yGZEBTJqm8RmAXcv1UAXwMKqUpuuyaFke/X9dN+2lqrSuxhp+Dtyd/30d66lVpf99eNO56Q/UTfk4BF3+YR40TbOmCQaSjqNYBhjoytr2outy9WtdV237ru04dNSqWpvaKa+RsPWT+t64X5W+vn/TczLOuCkZALuGERYZx0njNMX73nTAbxrUNGi5bnp4gPpNBepL1Xbs2vZD+1HU2gnCLMvvb4vXWpQ+vvtXT2iRetCkbqBIgfYwAQvQPY2jPYLuXYN6HrUsm377/Zu1v5kZE/IeELVVtdbNBtfsuhroEQz8cv+m91aEZWAihx6GYjlGqmEwfQCYTiaUJ/F6AMD/f9O6bdYcmaAdJmCEaxZCuv04VJCnHHGvtQAAFdACgNFl0oOqLgMACJUxjqPmeXJlQD8Aruuub9++6brtT5qzY5cYwHv3wniDEt35X0owA+Bffv6ht5MmFB8zO4eF5l2wA9fvMGieL7q7uxgILt+LdBxVy3WJHbfm9+w4SpoHJS8MgH3bvPsDc7pKutKnt38BMO1JmoakIcso0ZGHGsDlonme/UKk61Z0XTd/x5mjFEBqDT+75pMXO47jBBBAWfwJgAPhNGCWNI/QnW0qymxZV9WGHKPpBwzXfPZ4XV12yHVWu39ns1XYi++jOfQXl2YwwHX6eP/aOYD5eLHzeSCAslqTtgMApFu1D5wPNmgmOnUgkzBdUzmKn0EV2fW1KznKZ+8WGfjcAA2AKvjpux4LJw2JJEyaJ8ovud7XvWo96ln7LB7xRvfgz1qPY7h839RP/blHP4CBab5Yin0/vGh49GTgv/951Vl4GJKmnDWNvAZLuB/VAPYCbVjUIWq6CSSqJ/O7aVIt1RTbgFCawqCwdLncKedRpRbfDwPisab04WRgHLLmIRZn96Cn621H5Lr1dYmGBJQpxmq9aCYbUnKNr/seJQiAUp2kmJY0BTwA0B4JDOATHoD6c/dIwA7ZNQAomeiUyVQ/B+Dorbuj+wILtWi5YliSNRKT3Q7jrGmeLR+/ARhS4If05d33nd0RQNCPF0BnABiLtsd97lHXGDFFTyBSMRa7vcyjF0BnagpvRPzS4OiowQCsOROOYt+kz/evu9twpr8HAByL6wFxA8DiPDSSMamfEpADJJsZskQwNbn/s3jvyY2M30ElPYDf0LbJj/T57evOjvgxLFgCDAgDOxLEtHRr01SIWaLhnAnnHfGwnHSZI6ygBaMxNUH/4IEE40ajul5X94/05QTgUuTlvpOEgS1BiVCj7gMEEsgaruS/NWVK2qz33eWiy2V+mifQ363cSRczB2YFwHVdlb6+/b7HjAaAbgZcBT1pL11HCQa88DlW8WUi13q7j9APNjcdjHp3wfW0bpJzitA65xGAUEHrtuq6AMAmPAFkFkKObCPtNcxIaNymJK7xCAaCTnTlffSE3c5n8Qs943Ixa9FNY5c3JmhQsJB+ffe6YxTu3wLJg6mrIbyAm9EdZtAV+gFwmyMpU+SIKYi+0K37yxcvPFndQFiEc/SjHBnh0te330UbOTtXDKbhdr7rxUByDq08ANdTywY9TGoVSgFQzlmwerf4gfbNDEEZ8mFkA33j8BiXvgDgbJ03jQFAJrCAE8/9m3CJydFz/9mcxmFysgHAw2qj9inf4k0B4uXLO1dHGmhwMS0R2wbw+f7VOTnHoYTDCQ2PkkSvrqbqc0PEMazc5jzSDqPtjmzmPL7LNBUGhWZaMQBCDsKIHlG8uMvwBiCGhzg8cAZiPPOOPTE3AtD3uocLMj7OEIxdNC1eNCwPGq3ZC/vBoCrnwt9evnB5usSt/zMAYYw4RhFKMd245cSwSutnBMycJbqOnc+z65tvsjiBFSiJYCRikmL2i6q4u7uLfDgBmAFi/NP9q06smuDTiB7QDYr5HwA9ACS6GUHCNVNRjGbENbMDbCCjG42NGpVCNmBEuib7Qxoqxn3k40//7BgjfBgx7PHqllyJBzNYBCAY6BXtCZjRWQ8fMWrF72MaOnPi2TgfzUwGhzzIlD78+x9Pp2OfWc+jVDxK6s8AYEiOEFkxZvkkrZP2090cyXD5zahkis+WzxKxNHLk0F4Opfc//gUgJp3I66f/jBSmPyqiNwIr6I+yjZMTAWXjQe0tJZ0Tg6l3IPm8GZOQx/M/G80fuCYRcpfV8gIAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAAIAAAACAIBgAAAHN6evQAAAABc1JHQgCuzhzpAAAHtklEQVRYR1VXaY8kRxWMzDr73rXxT8S2ACGDtSxaLsvGeMWlxSBkkPyBQ4AE4pDXSEZI/BrEzs5s9/T0WUdmQsSrmlmm1eqqmqp88eLFi5flfvnhByklQF9+0viNOuZV7wHneA/PgcxlyPMCWZbB+Qw+y/R823VoWvv2McI5hyLPUOQ58jyD9w5cIcSIPnRoQw/30U8eK3QcQDiFIIiIKFQRcAm8znOXoMBFQQA5nPPwecZH0Pa9grd9QIwR3nvkuUeeeR1zEa7Rx4A+9OhigPv5kw+UljHAPwcvFHdM6J8CaSi5YFGUAkBqfOb1bN8HdH1EF409xmTWZI/BlX2KCCkYC2TpZz9+P/EO5sj1SZt39pDFjS+BMUYy71GUVgIBzixCH5ldRIgDWNLFRJyKizgkEcWuMew+/NF7DKsFkvNw/DD6CMCg3WqDx2SgLHLRyhCmBY+QIACiOIYhIMP+/4eBpTVGffKDd60EPHGsk4FQ3rrRBMgD0eqAqqAGKCpTp/cZuVZGbQwSF+vLLMfQFsGCjvGYpfvh43eGa4IwZO8VmAtSnAQ0qAKZd6hyj7IkA5lYM4RAn6IBSL1qbQsbg6PCbCnxDCQP9/i73xxEOLaBgbDHTBfjEjzKnENZZCoBWzBBShM7fQpoEQSEdSdbQ3FvAYhRPpMcHL/vfufhLQAzBFOsEDrW2PBbB5iqyzxXCeBIvXnBKLIeAcHRA2BiHjpIJYwmbMve2sJ969FXbzWQotVZJLG/JTKnnu7DoFqYuchYMuogh6MIFRGI+qixjO1hPemJwrorgOX69a992bpAQntZeGwvWziEiK4P+uVCbEOCyzKv7HUuQB7OUTcRTGYMRzQCqMTujsXGwwdfurt3cAtrk7ubuRaDiwVZLBRUiw0+wFbMRrul2QS2oXWIMWWg7ZmxExLcg7e/kGgcI+0yJRkJEFUSY2U0lxFYLoeTQwzzwskPyIoxT+OhZnJkea7/GYBRYwM/X3nrdVXJak7KX2rBl4DEMBgHM/BO3UAbGIMxLbKisgwmJrAClas8ZsvmsiML7q23Pp9EzzDZDACFlxAlPNOR/GAwHS5E92W9jYPRQ0bvH9pXl7m2H3RD0AZwdBb39oPXExHaZDNvH+kPzFrVIeqBods6BrgU4JwcXt3Cvic4/vHZmHjNWDEQdi4mFCfCPfzGmwIgO6WrjdZhhTHTQAbvONOH+yTSHil1CJGzvxMLZVmiLkux13YtYmBZBgGKMWY/iFH7gh7u0TtvJm0q+M+hFLdt5nMUeYXMcfQWKPLCvCFx1lOsEafmgP15h65rMaknmM/mCDGgac5q63zYM4wtrtJK3BF938E9eu+NJHokFutpOy5QFRWmkzmqYoo8GzYgJDxGlGWl8uyPN7jerXE47aSRSc17cwueF6iLidajCbE1m6bBuTmj7Rp0fQP37e+/kcb2Uc+qXzMFrIoadTUTC5yRLBVdkL95dgfg5rDG/nSDGAKqcoJJNUeRl0qA5zxmMQngfD7jcNzjeDrg3J7g3v/pFxPrZH1qNVY5uO/LShRZKTDMqipLTCcT1NUU3heIEdgfttjcXGJ32CL0PYq8xmJ2H/PZEnU50VrsKpVg0MbxeMDhtMfpfIB78vEDiVBBcgvmh27gL7PXtSzTEKqribJSu8aEw2GH7X6Nm/0WzblBntW4v3oNr95/DdPpXGXpmhZd14kB1v10PokBAfj4D99LVDcDceGqnNpeb+iCsjAA3I1SOKxr7gsJkHvAtm1xPB2x3W5xc7ODQ47V8h5efeVzWC5WqOvatmt9h1675gZNcxII/ro/Pf2FGCAAUlvXU9VsdLiyKBU0ph59aK0sea3su7bF4XDEfrfDZrPBerNB6BOm0ylWq5WALJcrTCYsGUH06o7z+YS2NSG6p//4XaJRMGhdTzCp56KZdhVjkIlwCx4Tt9LtwNQMHrmy315f4/LyOa6uLrFZb9T/RZFjNpsp+Gp5H4vFcuiaKADMvOsareeefvZ7laCq2HIzzKZLsaAxHDt1BbfgNBo+kHmKcYrcl+i7gO31BhcX/8HFxTOs1y+UHctFEExkNl9iMV9hMiUL3krRN+j7Vqy6v376m2TBp5jPl5jPWDcDQAboERQnp1sXWk0+tmeR1UghYb/f4fnzCzx79m+8eHGFw2Ev4NrIOYKvVAIKknHGxKLcM8D9+ZNfpcnEHGyxuIfF/J60IADgOwDnfHHrXGSCoiyySq11Pp2x2bwQA1eXF9jeXIteBuDLB7XM52lQ0+kMZVnY9k0AItxfnv46UTTz2QLLxSuiq6rYZjay6AtyMk3EMOwL6Zjmbm3T4mZ3gxdXV7i8fCYwFFiIbL1GrceFyGItD6n1IsNBpFe+v3362zRj9vMVlov7mM2WKMpKG0oDwCnGSTlO8HGb7hB6OluD/X6PzWaNq6vnAtB1Zw2rrj+r5vR+loPdZG9U9p6ovcHfP/ujAVisMJ/fMy/P2fd6VzJb5puwpuXgaNzthIi27XA8HrHb7bHdXmO9vsL2eo22O9uYTgweVL7b905Z/bghcHD//NcnaTZbqAQTDh5SpK3OaDw5aEYyJ70nWPCu63E6nZQ9AVCM2+0Gu90WbXPisIX3w6s9XwNUwrtXfm3k/nfxv9Iq1dJStK0pAAAAAElFTkSuQmCCiVBORw0KGgoAAAANSUhEUgAAACAAAAAgCAYAAABzenr0AAAAAXNSR0IArs4c6QAAB0pJREFUWEdVV9uOJFcRjHOt7umetdl/tARCQsY747W5SLMaS5aMkEBCRtzkB/sJnpB4QTzAAwibhW8B70533U7VgYg81bPMqnZ2u6tORkZGRma592/frXAeFQ5rBdZ11bUsC+rKa4XHiug9YgwIMSJ3OxwORxyOR8SYsK4LUBek4BCDA1BRyoq5LChlsbPqqrNQV/CHd3nv4J7ffkcA4AIqIBB8YF3ag+sCVyuCdwghwIeAnDtcHQ4CkFOnwwnAuwrveMaKMheMc8E8l5bMilqZJi8opHce7vnNt6vzAbzEBFmoVQ8tyyogDOCdQ/AEQCYS9ldXOB6v0XUEwKQXwleGSymYphnDOBmAtWWvFAnSKXvPuLfPvlm99/oPLyvHBqJeSsIPeZ9zXkzs9/sGYKfMrAws34JZwUeMBFCKPnsze1HvAwLPe/beOwIgehsI572qZOXg4XapdkIfsOs6HK+fqBxkqZRZQcjcNI0YhxHTPFs5WXtS76z2PIMxyKh777vv1BCcqDWKiSzqhkr5rcCy8mCWw0AQMKm/Pj5ByhnTPGGaJrAWda2Y5wnjOGImADFjzxGE6CWAVlJ3c/OtSuF4gqDQGgi/gagOywKUhZ1Biuv/AaAejO5RKmeWKsM8G4CFJaAAqYMK/lEHOK+k3fc+fLfB4sGWna5guiALjwCqBOrgxABFGGNWrYdxUDBmS5ExoEBME8rSOkHdIgJUf2nq7u4DFhhVCrY6mUhIE9vTqQzzYmUgm/yctT8erpFy1xQ/YilGubLzDuvCbrDybEyoDUU/Gfdw9x/9oOpU66VGD9Vin9mnQKEWKEiB9Or/w/EaKVID7HcLvi5rqzEfXFHUkqOY2BgyBkwH7v6jH1ZT5qNC9SWfr03BMhfSap3BO2PK2O8PKtM4WQeILWqFbtqSYinoCzNBzFNrV0tUCb+4/34NjTKGlQ6aUZhqzcEYVO2psrBjElLeCVA/jPqeHUSRysRkPlZSMsk2JYAyz1jXYrZMAHcvPqwW0CuwmNDlZKt2gPm3tBFiE2pCTHuQ8X4YFJjmYp5hDBizNC9cvGLrDDKjOHcvbiszVz1at15A8JMGQK0THIIGEr0iI+cjqgvqAvo/T9zkZKzZ4/wxe7chJ9dUK/5vBN3fPxNP1Jxo3vTgN/lRo5YVvwspIOWIlPbourcQQofSJqdFs5DbVCXVDE796NqS1K0V7uOPb8SVQktHnHoOMTJjfkB342gtEhjZyV1EzlfI+W04t79kJRbFQgOt2bAZkHK+tHaTOdwnP3q/8iHzeaszAaREumlOzGBBWWZUTcaKEOkDB6T4FKhXAiaGNvouxmssGqCNHAO5MeV+/JPnAqDgZCIAIVSkCLGABkDLSeXFewNSPCCGp3B1b35x8futn+08q4hZ8OPPVooK99NPbytdT/OZNQlVAIKA8HkzH6OSbEgJiP4KOX4DDocGgJ0iOZuLytata5vAmqWZsV3m0qe/uREABjML5u9tdJoHbCLkNCQDrGUKBPAUDkf9X/eJJXaDBbcB11q5sXxpjW0w/vqLD6ptJ3ZZfRiIdTXD0CRrQDzoFwEx7FUC757ID7hPzDOn4tksl2xePIVMtFbXLtA6jv/+/Ld3NWjy8WDCtXHK4I+O1VxLvUuS6QN7ePc2fHgLXXdASkl7wen8WkC2PfHieATD4bMNoYbO/e73n9TIJUQbkWW/bME1PrWgSYCXhbJGOOxQcQ24oyyZzHHsDkOPaR7aLsnhVPS8ua0tPEy2tQzcH/74MzFg45EAKDRmb2NT9WcLqgMWc8ya4cD+32FekrmcSmUzQBt14Q4wo64EwHbmyk4AtPJgfkB9/ukvv7howMkO7V2gzcaLh9teRzdkqTK836OUjH4EplLkE2/6v+2Jk8DbJkxRsnS2exoAB/fnv/6qxqYBAlCmWlCsVTTZCjVhmxBXNU5C5zLmOeLUr+i19cx6hptOCEn3ljIKxKU79D5gWrMh5eH+9tVnNYoWVnhrJQsmF9Cqz7+oXgaP8CEBiBgG4OuHEf04qv7mohEpJtWZywgBLLy0lpVmSE0PLsB99c/PK1+3VH6uZVI6ASV4hG1RsswDD+Z3Gesa8HCa8O9Xr7WUVlcRYwST4W/STItmR0xTj2kaMDcgekfQDhLg/vGvL9pazj3NI4asHve+g6tmMPa+kBBibvQnLIvDq4cz/vP1K8xlRkgRXc6X4BQrjYuB++GEYTgJCBcTrm7bbum+fPlZjcmLtpx2SOlKABxSe5tpdY8dYtghhAznEuay4vXDSRerxSW167JKsA036mccB5z7B5zPrzGOJ21Fb+4E7u8vf1lzSujyThPOss+y122EcebHuEdKB8TIno+Y5wWnUy/6WUKu6TQj9bi2sMe3pHN/wvn8Cn1PFlgKe2PSVvTly5/XnLjd7JAiLbXTsLksJ6r9Hileo8vXsl0C4Gt33w9aRpKeZ/aP/b29LzIYzanvHwSgH86YxkGvbXTa/wK6cRtPxpmOCQAAAABJRU5ErkJggolQTkcNChoKAAAADUlIRFIAAAAQAAAAEAgGAAAAH/P/YQAAAAFzUkdCAK7OHOkAAAH0SURBVDhPRVPRbhQxDBw7m+SOIv4XKsGVqlTQQzygCnjgCVrBF9LexTaMs+UeslllM+OZsVf2Hy8CAUgAiMgVEXAPDAuYczmGOcYYMDNEOEQAEYG83+8CTiAwd4I9CSaYnwNmJDS4j7xDAlVArve7CALXFe4IWwlIltyyEg64GSutBAK5+vAm4EACPU57gtMdAgq+uVOBkQ4qAlWBvH33mqZPYOaQMD6nApajX1ZG+KRcz2R3eR4M7X91CuZdmQTcM6z0zG80xGPNp+wuXwXlMzimOwEEru8qU255IvgHSnYuhVxevQz6muDIZBOsp6BUNf2mZ5YPQZDABXJ9cx5uI/1pCZQCSAlIyiVIVwtTCc9yXGYUkE+fL8L9yLig6igVKItgWRRLqdCyzJurQsrnjLjNrsm37zdJ4EEVA7o4aivorWOzOUOv2yQY9ghjoeCccMDYUof8uLuN4QeM8YjAAWUJtLag9y2ebV9gu3meNsZ4wHH8QRbLwOdcyP3vr0EwK0AMtSp6a0nQ+xl626IUhbOIPSRBBKdx/jNy9+tLmB3SQimC1mrKJ7D1DZY1A48Dggu0ymmcsyI/72/Dw6AC1LagtYbWOlrtqLVl6/gDmT8COELEIcpRZqdkErC3TP0JTGBdKiSHgn4HPI6TQA1F57yQ/C/XynwwsmWrAwAAAABJRU5ErkJggolQTkcNChoKAAAADUlIRFIAAAAQAAAAEAgGAAAAH/P/YQAAAAFzUkdCAK7OHOkAAAJUSURBVDhPPZNJixRBEIVf5FrdP9kFZmhFcPQyiIKgoHgYRAZEwYPoSfw1zkxVdW1ZuYREVmtBkIfiffki4iVdPn/E1loYbaC1glIKWhGICIpQiwAQAQyglIKcM2LKSLmALp6cs3MWzjkYY2GMhqkgghaI2kACYRSklBFjxBpjhdDjw0MWsfceznk4a2CMgTUKRuvqZgMwcskIIWAJAWGNiDGDzs/u1xYE0DRNPa0xFeTsyY3Y54J1XTHPM+ZlOQES6OzsHmulYK2rACnnHbw18M7CGl0dSN/LMmOcJsxLqLAYE+hweMAyMK10hYhYHEntvK8QrQk5/QOMm4MggAh6enFgwmnq0rM2tXdjHfxpJkSMnGJ1IC0sYd4cpAR68fJZdbBBFEipOkTndjDGA1wQowgHLMuEEJZ6c8oJhQvo7btLJqj/YqV0FVq3B5HGus6YxjschxbLMiKmiFIymHm79MPVq+pAka4Qrd3p9l1d2zR26IcbjHOLsE7IOYFZgiUaBbr69HpzoGTncnsD7/bVRcoRfX+Lrr/BHHqsaQZz3tKqTT3p4/UbltuVkuHJ2jy838M3O3Ap6I8t7to/OI4tQpzAnE4z2pJL15/fs1ayKomxhbOShT2aZgf5pnlE293WmqYeKa/QRtXASdHXbx/ZGAer/ZYDJz929ZQHJOvqjx3a7gbHocO6LgAxnKS38aDvP7+wsxLfZhM72cAWZ3kBa1wxTgO67g7D0GFZZ3DJ0EbXB0i/fv9ga2VwIt4g4kQGVJgrYJrGKh7GHiHMKDlDaYm/xV86gElAU9lrHQAAAABJRU5ErkJggolQTkcNChoKAAAADUlIRFIAAAAQAAAAEAgGAAAAH/P/YQAAAAFzUkdCAK7OHOkAAAJySURBVDhPLZNLj+Q0FIU/O3FeXT3//2cgzQZmQBqBYAHDAtCI2UCzoGGa6a6qLjuJEz8usqsVWYkc3y/nnnOj3n33ldxMB8Zxous6jOnQShNjYl1WrHPYyzPzfCGEDZEECJmE5IR6+81rmaYD0zQx9ANd19M2hhAi8zzjnMM5y7I4cg4oJVDKc6xLffn2CxmHqSoYh5FpnDBdT4oZa6/FZXk/10KtqSpSDqQUUG++fi191zGU4mnicHPLMEzkLDg3Y+0Fa5+rAkhorUBlco7k0sK7b9+IaVu6vucwHbi9fVUBoFjXlfP5xPn0xLzY2oJuVFVRfBDJqB/f/yCN1hjTcnNz4Pbwir4fywm895yOT5xOjyyrI6YNpaBpFErpaqX68PtvUlS1bVN9KIkY078AioIjz8/H6sG2r4hEmqZBXc1A3f31p5Tnstn3PUM/0rYGEWHbPM6VCC3btrD4C3uB1A5U/Yj65997KcVt02CMoTUGrTWShRB2tm0lBI/fF6w74uZT3S8mK9Goh/8+iWkNbdvWpXVT+ywKysF996S8V8jFPXKxjxWaUkayQj08fBJTJrBtaSpAU8QVh0MItbDkHuKKnZ+w9jPrthBDJEVB3d//LV3f0ZnuqqAYVAFCjDsxhmpcAczLE9Z9xnvHHsq7jLq7+0OKeWUOCuTqsKqAFAMxbmQpd8+yHpmXR7y3L4CE+vjxQwX0w0DfDy8+6NpCipGUroCUVlZ/qpBtc/XHCjGhfvn15wooozwMBdBRBkuQOuuSI/IC8PsZ78/s+1y9qYCf3n8vRX4F9GONssZYrpxAMqiipMRp2fcifyFETwyB/wHIy83c0ClbYQAAAABJRU5ErkJggolQTkcNChoKAAAADUlIRFIAAAAQAAAAEAgGAAAAH/P/YQAAAAFzUkdCAK7OHOkAAAHZSURBVDhPTZPpbtRAEISrZ8yj8yPiSJACUciuAAVIXpD11Qeqmt0kXo1sa2e+rqpu29fr92VmaI2rY5omWGvIAjwK67phnmcsy4rdHZGFyw9VsIfbqzJAgN47ep/Qehdgj8Sy7jidZszLgm13RCSyEpmJqoQdv32sZoVmhDQB+tSJhGdhXjb8Oy1Y1g3bvsM94BEIrgzYj7vPZUiYFewNhHaigJkK5hXrxsOO3UNW+EyI/bq/LlAKIXIHmDW0PsnGsrkOD/msnq+ATNjjdwJoocB7ZKoy5RTsTUVKpneGO0Bc9vvhRhYaoPpMnisBZJYCk195Vq1zhwgI2J8DAaw1DjA4StdGqslQ2iN9AlgOemcW9nT8cs4AqkAAiDNDZmDfXYDiYaaMppx4mP/Z8xlAAwQE8WbqSHhgVetcw8XO8M4CHq622vPxZhjDkC6ZVFDMYijYnCqg+eCcXNSxM/YkAMuOi1KHVwKGd250tswapncTmnHUzxn8PVxXsTe0fe6E/KqlpWA3T00gy/TOGemjgAfs8f5T8YUWlJG2XZTYy0clNVLBj24EqTb+vPsgwCtkwMZFqaZgaYEAXgySSdGeHW6vXhSMQR4zMTrJj+qiggC2kko5OWx74j/rKe5tVS7CTgAAAABJRU5ErkJggolQTkcNChoKAAAADUlIRFIAAAAQAAAAEAgGAAAAH/P/YQAAAAFzUkdCAK7OHOkAAAIrSURBVDhPRZJNaxRBEIbf6u752g9/rAERDxt1IVEIhCg5ePAg3nIQBAVFEMF/I7txZnaS3Znp7tKqniW99MxsV9VTb1U1vXtzzswAQx4M+QIBhgiGLGAMIjN8iAisXmqXoMARdH15lsLVkkAEiSMYY0DGapCPAohql6dAY4ygq4t1ilXMw7KGYK0FGQMmwHMCaKAmjIgRoItXpyxyVLx4TkuyO3cEEDwkOClQf90Ana2fsVC1D5MKeUsPrLMwUoIhMGlo6tWkWNv14vRJAkSpK0HkLUtUaB8sgcykjggke1JKq9VJmsIUyPEBoI4mqRFA+j9BSRAMWj1/zFCiOfZfv8U5tTUmRwXIuQCm/Pz//OX5CUudxjhY62BNBmczWB1fxOh7hBhSOfIzRrkywhA86PXVU3YuQ6Y7R56VyLMCzjqtcn+4x6HfazPFbo3Tfo3jgH44gK7frznPC+R5Ckw7VxXSnPv7O3R3nZZVlTM9l+zD0CuYPt685bKcqbFQSA7rpBSL4CO6rkNT1/A+oCwqFEWhysaxTwo+ffnAs2qO+WyJsqyQZRmMtVqv9xFt02C72agSqV8AckMjB3g/gL5+v+HFYon5fImqrOCyPM2eCX4MaNsWm80fNI2oGGGtSReMCJE96MfPz7xYPFJAWZRwooCs1j8OHrtdi9vtFnVzi77fg1kmcrxYDPr1+5sqqKo5iryAc04BIUT0hx67boe6/otdWytAsqb7kfY/apVHM5p9C+IAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAAEAAAABAIBgAAAB/z/2EAAAABc1JHQgCuzhzpAAACDElEQVQ4T02TMYsUQRCF36vumdm9vR+7GOjewQbKoYhwgcEFIhwGJkYmBgaCgcKBP+h2drq79FXvHg40M8N0ffXq9RvuXmwdNPSLMAI5JUzThNVqjWkakYwgHLVW1FrQaoW7w1Vxtds6VYUzBDAahmEMwDgOMBLuFWUpKKXAW4PKSYLX11vXBpoA6iMYkdMQKoY8RKdWC8qyoNQOCL1m4H7/zEUSpI+ibgImjOMKOWe01rr8clZQTwCCL1899+gpFeoecv99YEIeJqSUUZvH3P97ALTYz5vXO22H6aWTQgUtI+cRpBR4h7QaIN1jDDr45m0HhI0xRldjSYAVwAxvRHMBWrgfAJdbDr67FaAX07qFulLSCBsYJzgM7oB7B8Tz2e73d1fdRAOSAJTnumfkJMCFcHESvSyqT+8EP9zvXcV99eK+Q4BLGDcdEME5fTvL1Oifv9y4KWnWUyB3JZHIUWy8hCPFEcbscv9kdnj39dttALSetIXeDGANbxOq8ykHzeupkU7NwO8/7jwlC4AiHMfgCcSA5gOWxVAa4iil4Kykj0rw56+Prp9HgUkm2QNI5X9EbRnzsWdAm3WMyzJjWY6okQUHfz/cu/Ku0OSk5K2Q0xq0CbUaSpF1MogRouNxxjw/4ihILeDDn0+uP2/Iq1g5X2DIG5it0dzQWp9VLmsEdT8cHnGYD1iOM/4CcwswaWK7HsQAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAACAAAAAgIBgAAAMQPvosAAAABc1JHQgCuzhzpAAAAnElEQVQoUy2OsU7EUAwEZ/3s5K85GgTKCRqk606iQRSIT4wXvVwsbWOPPdb1+mwMbtPd7GfsZpbetotpMxt9Qm2DTAj0uj3Z3QfgczA3IyAktL1f/Di3ozAKETM6gc/bi81ORDMqyBEHwPEY6P71YdRkwrIWVYk0f9qZan3/3BzDZA3WdaFyoHgop1q/f3ePFEsltRSZU9FIE2r+AQEjWgGV1e8hAAAAAElFTkSuQmCCiVBORw0KGgoAAAANSUhEUgAAAAgAAAAICAYAAADED76LAAAAAXNSR0IArs4c6QAAALtJREFUKFMtzbtOA0EMheHjGdvLqxOqsBIFaVFapCgdoqGC5yHMxTM7RrOK2/PpN708H1w4QpgRY0AggvtA3zpa66D1+OhzVBGoMEIgjG2DNUM1A63rk3O8A5W90Luh1AKb4PV09BAimAUTzrxZRq0ZrXfQ2/nkMTKYFUQB1gpyvqG1iuED9H45u7CCedlBqQkp/aL1CsBBH59XV10gsiDsICOl216a7+j758tVHyCimGdWkfIfzArG2PAPR/VriHVCGeoAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAACAAAAAgIBgAAAMQPvosAAAABc1JHQgCuzhzpAAAAvElEQVQoUy3N70rDMBRA8ZPRZOn7P48fBPEPIuIEp6CvsM21N83N0lzp6vcf57i7+xuLscf7wFxnUhJSGlGdqLPiHp5urY89IURaa4gMiJzRMlGr4l5eH68gbnvMjHE8M4wnVGUFH/vddRG3EbOlsIAj0zRQLhn3/fP1vwgYjZyFYTwgciJrWsFSCMEDRimJlI6McmDKgtt/vluMEe89zhm1ZrL+ktJpBW+7Zwsh0HUdm43RWqFcBNWBrMIfGbGKPlWQOEoAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAACAAAAAgIBgAAAMQPvosAAAABc1JHQgCuzhzpAAAAkUlEQVQoU02OwQrCYAyD09cfioqojO0wcA/pv7aJ9N8OO+RQ8iWNfZ6DDAJgSArujuYOjwCZsPl9EToAkELzwK9tHYhM2DLeBBGSekMBbdsbOvCdHhITJBEkPBLuAc9EJGHrfAKSiAJYpvaGdbqrxlBEFnAoj0ZbxqvE2sA+st5UOitAwebXoDJrJLvQjQLq/gMlGrBwUp25tQAAAABJRU5ErkJggolQTkcNChoKAAAADUlIRFIAAAAIAAAACAgGAAAAxA++iwAAAAFzUkdCAK7OHOkAAACoSURBVChTNY69CgJBDIQnet6Pr32IhaCihXK9hbYWV4rvpKt7t5uRRB0IIZMvQ+S4W1JJkIRJrUhkJZQK2W8WVNUfIKD8IXVQNuuWOSuohGfIFwJsNmDbejq/rpsuGy3h0K04nRawEpkgp4SsGVT1Lqdzx6qqUZWVn8UYMcSIlEfkNEKu/YVNM3eACoQQ8ApPjCkipQFyu/esmwazooQ9a8sQHojD24EPQJ57wInfP5EAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAACAAAAAgIBgAAAMQPvosAAAABc1JHQgCuzhzpAAAArElEQVQoUzXOsUpEQQyF4f9kkll3n3gRbJa1WCwEsRMEGysLSwsrwae6906WDJoq4Xwk0el0TJPRmtMjcHdMAhIQOt8fUwgzx1tM+J/LhC4PtxNIbSKzqIkqSejp+S5nU0CBWSexP2Do9e2cZkb9IeuIHdDIeqF2fXw+Zt1ttV4V7hhDjJGMTPT1/ZI9Ou57mt2QGROs68a6bejn9z1739PjgLcDUrBNsLIsC1fTZzYczHDyJAAAAABJRU5ErkJggolQTkcNChoKAAAADUlIRFIAAAAEAAAABAgGAAAAqfGefgAAAAFzUkdCAK7OHOkAAABASURBVBhXBcFBCoAwEATBnt01/xa8BhTvkg8E/KAZq9T7bq+FWUQIXfdh+0NhKhM943SkqU2EhOY73FpSJYT5AaoeFj3gv0ycAAAAAElFTkSuQmCCiVBORw0KGgoAAAANSUhEUgAAAAQAAAAECAYAAACp8Z5+AAAAAXNSR0IArs4c6QAAAENJREFUGFcdxzsOQFAQQNE7b2L/WqKiU9Lq2JMM85E43ZFp7KshQBKZyDIPBVCVRDiy7Wu1pn+e15DzOkq1w8Mxu/kAMRoht4QO6vgAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAABAAAAAQIBgAAAKnxnn4AAAABc1JHQgCuzhzpAAAASklEQVQYVwXBQRJAMAwF0J+W6v0PZIWxKIZ7sEhHM1Im3qNh6s07B60PRDJo2WbrQoBqAecLtB/JYgzQeoP5BKV1tLbxeD9BEcYP4LMkzdssivEAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAABAAAAAQIBgAAAKnxnn4AAAABc1JHQgCuzhzpAAAAO0lEQVQYVx3IQQrAIBADwOz3RY9aEPrQbpJuqXOc2KsXKVAJvUbc1yhJeJJI849etpEnhNiznaAN2fgAMwkuYHFiHgsAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAABAAAAAQIBgAAAKnxnn4AAAABc1JHQgCuzhzpAAAAQklEQVQYVyXIMQqAQBADwOT7VlaCcIWt2NmJ/xL2NnsRdMrhukyWCqrC8ABbm7+o0h/7sdk2pERmgNd9miQyOyIevCXdLCJpeWu+AAAAAElFTkSuQmCCiVBORw0KGgoAAAANSUhEUgAAAAQAAAAECAYAAACp8Z5+AAAAAXNSR0IArs4c6QAAAEVJREFUGFcVyLsJgEAQQMG3P28DizYxvUzECkSwA7G0Y8UJR3pfSsUID9wnZD/WEhy3Rvxx3VuFJ2YJZcjznpU5o9oYAz7iGQ7hnzpEpwAAAABJRU5ErkJggolQTkcNChoKAAAADUlIRFIAAAACAAAAAggGAAAAcrYNJAAAAAFzUkdCAK7OHOkAAAAbSURBVBhXY5w0ueU/M/N/BsbV6+b9Z2ZmYAAARtsGrGDQAycAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAAAgAAAAIIBgAAAHK2DSQAAAABc1JHQgCuzhzpAAAAG0lEQVQYV2NcuGDa/////jEw7tm75f+fP38ZAGx7C/Wvw8Z6AAAAAElFTkSuQmCCiVBORw0KGgoAAAANSUhEUgAAAAIAAAACCAYAAABytg0kAAAAAXNSR0IArs4c6QAAABtJREFUGFdjPHJs7/8/f34wMJ44tf//nz/fGQBz5wx4FluSCgAAAABJRU5ErkJggolQTkcNChoKAAAADUlIRFIAAAACAAAAAggGAAAAcrYNJAAAAAFzUkdCAK7OHOkAAAAbSURBVBhXY5wxpfv/z1/fGBinT+78//PXTwYAZj0LQEoCgIsAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAAAgAAAAIIBgAAAHK2DSQAAAABc1JHQgCuzhzpAAAAG0lEQVQYV2OcO3vC/x+/fjAwbtq86v/PX98ZAGnQC6xtdnczAAAAAElFTkSuQmCCiVBORw0KGgoAAAANSUhEUgAAAAIAAAACCAYAAABytg0kAAAAAXNSR0IArs4c6QAAABtJREFUGFdjnDGz4z8HKzcD45ZtS/+zs3EzAAA+kwX5HkOXfwAAAABJRU5ErkJggolQTkcNChoKAAAADUlIRFIAAAABAAAAAQgGAAAAHxXEiQAAAAFzUkdCAK7OHOkAAAANSURBVBhXY5i3oPc/AAZ2AstmcHmZAAAAAElFTkSuQmCCiVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAAAXNSR0IArs4c6QAAAA1JREFUGFdjWL9u2X8ABxYDAyYuHw4AAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAAAQAAAAEIBgAAAB8VxIkAAAABc1JHQgCuzhzpAAAADUlEQVQYV2M4cWrvfwAH/ANPEyoF+wAAAABJRU5ErkJggolQTkcNChoKAAAADUlIRFIAAAABAAAAAQgGAAAAHxXEiQAAAAFzUkdCAK7OHOkAAAANSURBVBhXY5jU1/IfAAX+AqTQAe//AAAAAElFTkSuQmCCiVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAAAXNSR0IArs4c6QAAAA1JREFUGFdjWLx4+n8ABqcC3V9yBbYAAAAASUVORK5CYIKJUE5HDQoaCgAAAA1JSERSAAAAAQAAAAEIBgAAAB8VxIkAAAABc1JHQgCuzhzpAAAADUlEQVQYV2NYvWbefwAG8AL1Roy/rAAAAABJRU5ErkJggg==", y1e = "data:application/octet-stream;base64,hhaHlvbWljZ7InZlcnNpb24iOjIsIndpZHRoIjoxMjgsImltYWdlVHlwZSI6ImltYWdlL3BuZyIsImlycmFkaWFuY2UiOnsieCI6Wy0wLjAyNjYyNzA4NzM0MjA1Mjk0MiwtMC4wMTk0NDMwMDY2MTk5NDMxNCwtMC4wMTc3MDkyNDk4MzQ4MDMxOF0sInkiOlswLjAwMDM4OTUyOTExNTkyNzI2NTk1LDAuMDAwODM5NDI5MTY1NTY4MDUxMSwwLjAwMTQwNjg4MDM2MTcyNDA5MjZdLCJ6IjpbMC4wMDM4MTQ3MzQwMjM4ODQzOTcyLC0wLjAwNDc5MDc2ODQ1NjA2MTQyOSwtMC4wMDg4MjYxMDQ0NzgzOTA5OF0sInh4IjpbMC41OTU0MzQ1NzM3Mjg1ODQ4LDAuNjIzNTUzNTE3NTA3ODY2NywwLjY2MTU1ODIwOTQxNTQ5NTFdLCJ5eSI6WzAuNTk4NTcwOTU5NTM4NDg5MywwLjYyODg1MzIzNDE5NDc5NjcsMC42NzAzMDIzOTA1NTIxNTU4XSwienoiOlswLjU5ODU4MzI2MzE4MzE3OTcsMC42Mjc2MDk4NDM2Mzg2OTIxLDAuNjY2NjczMDI1MjUzOTczNV0sInl6IjpbLTAuMDAwNDE5NjY0MDc5NzcwMjE5NTUsLTAuMDAwMjEzNTk0MzExMjI2NTczMywtMC4wMDAwODk1NDQ2MTgzNDY0NDY0NF0sInp4IjpbLTAuMDA0OTE5OTg5MTQyOTE3MTE4LC0wLjAwMzkxNDk2OTg2OTc0NjEzMSwtMC4wMDI4ODI2NTcwODAyODg1XSwieHkiOlswLjAwMDE3MDA5NjE0NDQ0NzU0NDQsMC4wMDAxOTY4MzUwMDk0MjMzMTM5OSwwLjAwMDI4NDg4ODY3NTY0NjQ2NjldfSwic3BlY3VsYXIiOnsibWlwbWFwcyI6W3sibGVuZ3RoIjo4MDMxLCJwb3NpdGlvbiI6MH0seyJsZW5ndGgiOjgxMTAsInBvc2l0aW9uIjo4MDMxfSx7Imxlbmd0aCI6MTQyOTcsInBvc2l0aW9uIjoxNjE0MX0seyJsZW5ndGgiOjEzODEyLCJwb3NpdGlvbiI6MzA0Mzh9LHsibGVuZ3RoIjo3NjQ5LCJwb3NpdGlvbiI6NDQyNTB9LHsibGVuZ3RoIjo3ODcxLCJwb3NpdGlvbiI6NTE4OTl9LHsibGVuZ3RoIjo1MDQ4LCJwb3NpdGlvbiI6NTk3NzB9LHsibGVuZ3RoIjo1MjM2LCJwb3NpdGlvbiI6NjQ4MTh9LHsibGVuZ3RoIjo1MDA3LCJwb3NpdGlvbiI6NzAwNTR9LHsibGVuZ3RoIjo0NzU4LCJwb3NpdGlvbiI6NzUwNjF9LHsibGVuZ3RoIjo0NDc2LCJwb3NpdGlvbiI6Nzk4MTl9LHsibGVuZ3RoIjo0Njc4LCJwb3NpdGlvbiI6ODQyOTV9LHsibGVuZ3RoIjoxNTg3LCJwb3NpdGlvbiI6ODg5NzN9LHsibGVuZ3RoIjoxNjMzLCJwb3NpdGlvbiI6OTA1NjB9LHsibGVuZ3RoIjoxNTYwLCJwb3NpdGlvbiI6OTIxOTN9LHsibGVuZ3RoIjoxNTQ1LCJwb3NpdGlvbiI6OTM3NTN9LHsibGVuZ3RoIjoxMzg0LCJwb3NpdGlvbiI6OTUyOTh9LHsibGVuZ3RoIjoxNDUzLCJwb3NpdGlvbiI6OTY2ODJ9LHsibGVuZ3RoIjo1MjgsInBvc2l0aW9uIjo5ODEzNX0seyJsZW5ndGgiOjUwNywicG9zaXRpb24iOjk4NjYzfSx7Imxlbmd0aCI6NTE1LCJwb3NpdGlvbiI6OTkxNzB9LHsibGVuZ3RoIjo1MTksInBvc2l0aW9uIjo5OTY4NX0seyJsZW5ndGgiOjQ2MCwicG9zaXRpb24iOjEwMDIwNH0seyJsZW5ndGgiOjQ3MSwicG9zaXRpb24iOjEwMDY2NH0seyJsZW5ndGgiOjIxMiwicG9zaXRpb24iOjEwMTEzNX0seyJsZW5ndGgiOjIwMywicG9zaXRpb24iOjEwMTM0N30seyJsZW5ndGgiOjE5OSwicG9zaXRpb24iOjEwMTU1MH0seyJsZW5ndGgiOjE5OSwicG9zaXRpb24iOjEwMTc0OX0seyJsZW5ndGgiOjE5NSwicG9zaXRpb24iOjEwMTk0OH0seyJsZW5ndGgiOjE4OSwicG9zaXRpb24iOjEwMjE0M30seyJsZW5ndGgiOjEyNywicG9zaXRpb24iOjEwMjMzMn0seyJsZW5ndGgiOjEyMCwicG9zaXRpb24iOjEwMjQ1OX0seyJsZW5ndGgiOjEyNSwicG9zaXRpb24iOjEw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", k1e = "data:application/octet-stream;base64,hhaHlvbWljZ7InZlcnNpb24iOjIsIndpZHRoIjoxMjgsImltYWdlVHlwZSI6ImltYWdlL3BuZyIsImlycmFkaWFuY2UiOnsieCI6Wy0wLjAwMzQ0NTc1MjY5ODM4NzIyNSwtMC4wMDM1NzcyMDc5Nzk0OTM3MDE2LC0wLjAwMzU1MzUzNzU2MzE2OTMyODddLCJ5IjpbMC4wMDE2NTc1NzMwNDc0MTY4MjA1LDAuMDA1NDI3ODQ1NDM0NDU4MjgxLDAuMDA1Nzc1OTY3NjAxODUwNTY5XSwieiI6WzAuMDAxNDk3OTIxNDgyMTI5MTI2NywwLjAwMTU5Mzc0NjE2ODA3ODk3MjQsMC4wMDE4NzE5NDkxNjEyNDA5NjcxXSwieHgiOlswLjAzNTI3NzI3MTIwMTIxMTEsMC4wNDIzMTc1NDk3NjcyMDI0NjQsMC4wNTgwNTQxMTY0NDgwOTMwMjRdLCJ5eSI6WzAuMDMxNzM0NTgzOTAzMDI0NDUsMC4wMzgzMDM4ODgzMDAxMzMyNCwwLjA1Mzg0ODIzNjk0NDkzMV0sInp6IjpbMC4wMzU0MzAyNjY1NTQyODg1NiwwLjA0MjQ5MTY1NDM4NTQ4MTM1LDAuMDU4MjQwNDc2MTAyNTIyMzNdLCJ5eiI6WzAuMDAwMDkwMTQxNDYxMzE5MjEwMDIsMC4wMDAxMDk2NzQ1MjQ2MTU4MTEwNiwwLjAwMDA1MzIzNjExNjM4OTMzNzk1Nl0sInp4IjpbMC4wMDA3ODE4Mzc1Mzk0NzI1NzE1LDAuMDAwNjkzMDM4NjQzNzY3NDc0OCwwLjAwMDc2MDYwNzY1MTAxNjIzXSwieHkiOlstMC4wMDAwMzMwMDU3MTMzNzM1NTA5ODYsMC4wMDAwODAyMDI1MzY2OTI4MTY5OCwwLjAwMDA3NzkwODQ0NjYyNjk1NDg1XX0sInNwZWN1bGFyIjp7Im1pcG1hcHMiOlt7Imxlbmd0aCI6NjAzMSwicG9zaXRpb24iOjB9LHsibGVuZ3RoIjo1OTQ4LCJwb3NpdGlvbiI6NjAzMX0seyJsZW5ndGgiOjg1MTksInBvc2l0aW9uIjoxMTk3OX0seyJsZW5ndGgiOjk1ODAsInBvc2l0aW9uIjoyMDQ5OH0seyJsZW5ndGgiOjYwNzIsInBvc2l0aW9uIjozMDA3OH0seyJsZW5ndGgiOjYwOTEsInBvc2l0aW9uIjozNjE1MH0seyJsZW5ndGgiOjQ5MjksInBvc2l0aW9uIjo0MjI0MX0seyJsZW5ndGgiOjQ1NDMsInBvc2l0aW9uIjo0NzE3MH0seyJsZW5ndGgiOjQxNjgsInBvc2l0aW9uIjo1MTcxM30seyJsZW5ndGgiOjQ4MzksInBvc2l0aW9uIjo1NTg4MX0seyJsZW5ndGgiOjQ4NDQsInBvc2l0aW9uIjo2MDcyMH0seyJsZW5ndGgiOjUwMDgsInBvc2l0aW9uIjo2NTU2NH0seyJsZW5ndGgiOjE0MDgsInBvc2l0aW9uIjo3MDU3Mn0seyJsZW5ndGgiOjEzMTcsInBvc2l0aW9uIjo3MTk4MH0seyJsZW5ndGgiOjExMzEsInBvc2l0aW9uIjo3MzI5N30seyJsZW5ndGgiOjEzNTEsInBvc2l0aW9uIjo3NDQyOH0seyJsZW5ndGgiOjE0MTIsInBvc2l0aW9uIjo3NTc3OX0seyJsZW5ndGgiOjE0MzYsInBvc2l0aW9uIjo3NzE5MX0seyJsZW5ndGgiOjQ3NiwicG9zaXRpb24iOjc4NjI3fSx7Imxlbmd0aCI6NDM1LCJwb3NpdGlvbiI6NzkxMDN9LHsibGVuZ3RoIjozODMsInBvc2l0aW9uIjo3OTUzOH0seyJsZW5ndGgiOjQzMiwicG9zaXRpb24iOjc5OTIxfSx7Imxlbmd0aCI6NDU1LCJwb3NpdGlvbiI6ODAzNTN9LHsibGVuZ3RoIjo0NjQsInBvc2l0aW9uIjo4MDgwOH0seyJsZW5ndGgiOjIwMiwicG9zaXRpb24iOjgxMjcyfSx7Imxlbmd0aCI6MjAzLCJwb3NpdGlvbiI6ODE0NzR9LHsibGVuZ3RoIjoyMDAsInBvc2l0aW9uIjo4MTY3N30seyJsZW5ndGgiOjE4NSwicG9zaXRpb24iOjgxODc3fSx7Imxlbmd0aCI6MTk4LCJwb3NpdGlvbiI6ODIwNjJ9LHsibGVuZ3RoIjoyMDEsInBvc2l0aW9uIjo4MjI2MH0seyJsZW5ndGgiOjEzMSwicG9zaXRpb24iOjgyNDYxfSx7Imxlbmd0aCI6MTIzLCJwb3NpdGlvbiI6ODI1OTJ9LHsibGVuZ3RoIjoxMzAsInBvc2l0aW9uIjo4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", E1e = "data:application/octet-stream;base64,hhaHlvbWljZ7InZlcnNpb24iOjIsIndpZHRoIjoxMjgsImltYWdlVHlwZSI6ImltYWdlL3BuZyIsImlycmFkaWFuY2UiOnsieCI6WzAuMDAwNjc2NDE0MTExNzA3NjYyOSwwLjAwMDAzMTU5NTM1OTA2MDY0ODksMC4wMDAwNjg4Mjc4OTgyNjg5NzAxNl0sInkiOlstMC4wMDI1MjU5NTUxMzIwNTM0MzU4LC0wLjAwMjU1OTA4MDk0OTQ0NjM5NTIsLTAuMDAxOTYzNTQxODAxNzk2MDQyNF0sInoiOlswLjAyNTcwMjY1Njg5NTY2MDExNiwwLjAxODAxMTAyODM1ODgxNDYyNiwwLjAwNzg0MjYzNTU5NjAxNTgwOF0sInh4IjpbMC41NTI2OTAyMzU4NjIxNTY5LDAuNjc3MzI4ODQyMTA5Mjk5NywwLjgwODAwMjAzMjAwNzA2NTddLCJ5eSI6WzAuNTM3NzA4NTUzNzk3NjQyOSwwLjY2Mjk3Njg4NDE0NzczNjcsMC43OTc0MTkwODczMzczMzUxXSwienoiOlswLjU0ODkyNDc1MzEwNDYxMDUsMC42NzQxNjQyMDIzNjMzNjk5LDAuODA1Nzk5OTIwNTMwOTE1NV0sInl6IjpbMC4wMDA2NjkyNTM4NjU0MDg3NjAxLDAuMDAwMzg1NzU3OTgyNDkxNjQ1MywwLjAwMDA3NTc4NjI5MDg2MzI5MjYzXSwiengiOlstMC4wMDQxMjYxMzQ2MDk0MTE2MTE1LC0wLjAwMjEzODk0MzAzMjAwOTgxMTQsMC4wMDAxNjgxODQ1MzU0MDAyMjY1M10sInh5IjpbLTAuMDAwMTczNTY2MDM4NjM1Njc1MjgsLTAuMDAwMDk3NTA5NjY3MzI0MzY4NjUsLTAuMDAwMDg1OTU0ODMwMDkyODQ4OTVdfSwic3BlY3VsYXIiOnsibWlwbWFwcyI6W3sibGVuZ3RoIjo4MDgxLCJwb3NpdGlvbiI6MH0seyJsZW5ndGgiOjc5MTcsInBvc2l0aW9uIjo4MDgxfSx7Imxlbmd0aCI6MTY1MjMsInBvc2l0aW9uIjoxNTk5OH0seyJsZW5ndGgiOjE1MzEwLCJwb3NpdGlvbiI6MzI1MjF9LHsibGVuZ3RoIjo4MTY4LCJwb3NpdGlvbiI6NDc4MzF9LHsibGVuZ3RoIjo4MDkxLCJwb3NpdGlvbiI6NTU5OTl9LHsibGVuZ3RoIjo1MTc3LCJwb3NpdGlvbiI6NjQwOTB9LHsibGVuZ3RoIjo0OTEwLCJwb3NpdGlvbiI6NjkyNjd9LHsibGVuZ3RoIjo2MTcyLCJwb3NpdGlvbiI6NzQxNzd9LHsibGVuZ3RoIjo1NDk1LCJwb3NpdGlvbiI6ODAzNDl9LHsibGVuZ3RoIjo1MDU0LCJwb3NpdGlvbiI6ODU4NDR9LHsibGVuZ3RoIjo1MDk2LCJwb3NpdGlvbiI6OTA4OTh9LHsibGVuZ3RoIjoxNTIyLCJwb3NpdGlvbiI6OTU5OTR9LHsibGVuZ3RoIjoxNDc1LCJwb3NpdGlvbiI6OTc1MTZ9LHsibGVuZ3RoIjoxODU3LCJwb3NpdGlvbiI6OTg5OTF9LHsibGVuZ3RoIjoxNzk0LCJwb3NpdGlvbiI6MTAwODQ4fSx7Imxlbmd0aCI6MTUxMCwicG9zaXRpb24iOjEwMjY0Mn0seyJsZW5ndGgiOjE1MjksInBvc2l0aW9uIjoxMDQxNTJ9LHsibGVuZ3RoIjo0NTgsInBvc2l0aW9uIjoxMDU2ODF9LHsibGVuZ3RoIjo1MDMsInBvc2l0aW9uIjoxMDYxMzl9LHsibGVuZ3RoIjo1NzgsInBvc2l0aW9uIjoxMDY2NDJ9LHsibGVuZ3RoIjo1ODIsInBvc2l0aW9uIjoxMDcyMjB9LHsibGVuZ3RoIjo0NzcsInBvc2l0aW9uIjoxMDc4MDJ9LHsibGVuZ3RoIjo1MDgsInBvc2l0aW9uIjoxMDgyNzl9LHsibGVuZ3RoIjoxOTAsInBvc2l0aW9uIjoxMDg3ODd9LHsibGVuZ3RoIjoyMTEsInBvc2l0aW9uIjoxMDg5Nzd9LHsibGVuZ3RoIjoyMjcsInBvc2l0aW9uIjoxMDkxODh9LHsibGVuZ3RoIjoyMzcsInBvc2l0aW9uIjoxMDk0MTV9LHsibGVuZ3RoIjoxOTQsInBvc2l0aW9uIjoxMDk2NTJ9LHsibGVuZ3RoIjoyMjQsInBvc2l0aW9uIjoxMDk4NDZ9LHsibGVuZ3RoIjoxMjcsInBvc2l0aW9uIjoxMTAwNzB9LHsibGVuZ3RoIjoxMzAsInBvc2l0aW9uIjoxMTAxOTd9LHsibGVuZ3RoIjox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", wSe = "", F1e = "", N1e = "", Q1e = "", Y1e = ""; function OF(A, e, t, r) { const n = { externalResourceFunction: (i) => r(i).then((s) => new Uint8Array(s)) }; return t && (n.uri = e === "file:" ? t : e + t), A instanceof ArrayBuffer ? GLTFValidator.validateBytes(new Uint8Array(A), n) : GLTFValidator.validateString(A, n); } function _ge() { const A = []; onmessage = (e) => { const t = e.data; switch (t.id) { case "init": { importScripts(t.url); break; } case "validate": { OF(t.data, t.rootUrl, t.fileName, (r) => new Promise((n, i) => { const s = A.length; A.push({ resolve: n, reject: i }), postMessage({ id: "getExternalResource", index: s, uri: r }); })).then((r) => { postMessage({ id: "validate.resolve", value: r }); }, (r) => { postMessage({ id: "validate.reject", reason: r }); }); break; } case "getExternalResource.resolve": { A[t.index].resolve(t.value); break; } case "getExternalResource.reject": { A[t.index].reject(t.reason); break; } } }; } class M1e { /** * Validate a glTF asset using the glTF-Validator. * @param data The JSON of a glTF or the array buffer of a binary glTF * @param rootUrl The root url for the glTF * @param fileName The file name for the glTF * @param getExternalResource The callback to get external resources for the glTF validator * @returns A promise that resolves with the glTF validation results once complete */ static ValidateAsync(e, t, r, n) { const i = ArrayBuffer.isView(e) ? e.slice().buffer : e; return typeof Worker == "function" ? new Promise((s, a) => { const f = `${OF}(${_ge})()`, o = URL.createObjectURL(new Blob([f], { type: "application/javascript" })), d = new Worker(o), v = (l) => { d.removeEventListener("error", v), d.removeEventListener("message", u), a(l); }, u = (l) => { const P = l.data; switch (P.id) { case "getExternalResource": { n(P.uri).then((p) => { d.postMessage({ id: "getExternalResource.resolve", index: P.index, value: p }, [p]); }, (p) => { d.postMessage({ id: "getExternalResource.reject", index: P.index, reason: p }); }); break; } case "validate.resolve": { d.removeEventListener("error", v), d.removeEventListener("message", u), s(P.value), d.terminate(); break; } case "validate.reject": d.removeEventListener("error", v), d.removeEventListener("message", u), a(P.reason), d.terminate(); } }; d.addEventListener("error", v), d.addEventListener("message", u), d.postMessage({ id: "init", url: ye.GetBabylonScriptURL(this.Configuration.url) }), d.postMessage({ id: "validate", data: i, rootUrl: t, fileName: r }); }) : (this._LoadScriptPromise || (this._LoadScriptPromise = ye.LoadBabylonScriptAsync(this.Configuration.url)), this._LoadScriptPromise.then(() => OF(i, t, r, n))); } } M1e.Configuration = { url: `${ye._DefaultCdnUrl}/gltf_validator.js` }; function sZ(A, e, t) { try { return Promise.resolve(new Uint8Array(A, e, t)); } catch (r) { return Promise.reject(r); } } function $ge(A, e, t) { try { if (e < 0 || e >= A.byteLength) throw new RangeError("Offset is out of range."); if (e + t > A.byteLength) throw new RangeError("Length is out of range."); return Promise.resolve(new Uint8Array(A.buffer, A.byteOffset + e, t)); } catch (r) { return Promise.reject(r); } } var AR; (function(A) { A[A.AUTO = 0] = "AUTO", A[A.FORCE_RIGHT_HANDED = 1] = "FORCE_RIGHT_HANDED"; })(AR || (AR = {})); var qS; (function(A) { A[A.NONE = 0] = "NONE", A[A.FIRST = 1] = "FIRST", A[A.ALL = 2] = "ALL"; })(qS || (qS = {})); var cc; (function(A) { A[A.LOADING = 0] = "LOADING", A[A.READY = 1] = "READY", A[A.COMPLETE = 2] = "COMPLETE"; })(cc || (cc = {})); class q1 { constructor() { this.onParsedObservable = new Oe(), this.coordinateSystemMode = AR.AUTO, this.animationStartMode = qS.FIRST, this.compileMaterials = !1, this.useClipPlane = !1, this.compileShadowGenerators = !1, this.transparencyAsCoverage = !1, this.useRangeRequests = !1, this.createInstances = !0, this.alwaysComputeBoundingBox = !1, this.loadAllMaterials = !1, this.loadOnlyMaterials = !1, this.skipMaterials = !1, this.useSRGBBuffers = !0, this.targetFps = 60, this.alwaysComputeSkeletonRootNode = !1, this.preprocessUrlAsync = (e) => Promise.resolve(e), this.onMeshLoadedObservable = new Oe(), this.onSkinLoadedObservable = new Oe(), this.onTextureLoadedObservable = new Oe(), this.onMaterialLoadedObservable = new Oe(), this.onCameraLoadedObservable = new Oe(), this.onCompleteObservable = new Oe(), this.onErrorObservable = new Oe(), this.onDisposeObservable = new Oe(), this.onExtensionLoadedObservable = new Oe(), this.validate = !1, this.onValidatedObservable = new Oe(), this._loader = null, this._state = null, this._requests = new Array(), this.name = "gltf", this.extensions = { ".gltf": { isBinary: !1 }, ".glb": { isBinary: !0 } }, this.onLoaderStateChangedObservable = new Oe(), this._logIndentLevel = 0, this._loggingEnabled = !1, this._log = this._logDisabled, this._capturePerformanceCounters = !1, this._startPerformanceCounter = this._startPerformanceCounterDisabled, this._endPerformanceCounter = this._endPerformanceCounterDisabled; } /** * Raised when the asset has been parsed */ set onParsed(e) { this._onParsedObserver && this.onParsedObservable.remove(this._onParsedObserver), this._onParsedObserver = this.onParsedObservable.add(e); } /** * Callback raised when the loader creates a mesh after parsing the glTF properties of the mesh. * Note that the callback is called as soon as the mesh object is created, meaning some data may not have been setup yet for this mesh (vertex data, morph targets, material, ...) */ set onMeshLoaded(e) { this._onMeshLoadedObserver && this.onMeshLoadedObservable.remove(this._onMeshLoadedObserver), this._onMeshLoadedObserver = this.onMeshLoadedObservable.add(e); } /** * Callback raised when the loader creates a texture after parsing the glTF properties of the texture. */ set onTextureLoaded(e) { this._onTextureLoadedObserver && this.onTextureLoadedObservable.remove(this._onTextureLoadedObserver), this._onTextureLoadedObserver = this.onTextureLoadedObservable.add(e); } /** * Callback raised when the loader creates a material after parsing the glTF properties of the material. */ set onMaterialLoaded(e) { this._onMaterialLoadedObserver && this.onMaterialLoadedObservable.remove(this._onMaterialLoadedObserver), this._onMaterialLoadedObserver = this.onMaterialLoadedObservable.add(e); } /** * Callback raised when the loader creates a camera after parsing the glTF properties of the camera. */ set onCameraLoaded(e) { this._onCameraLoadedObserver && this.onCameraLoadedObservable.remove(this._onCameraLoadedObserver), this._onCameraLoadedObserver = this.onCameraLoadedObservable.add(e); } /** * Callback raised when the asset is completely loaded, immediately before the loader is disposed. * For assets with LODs, raised when all of the LODs are complete. * For assets without LODs, raised when the model is complete, immediately after the loader resolves the returned promise. */ set onComplete(e) { this._onCompleteObserver && this.onCompleteObservable.remove(this._onCompleteObserver), this._onCompleteObserver = this.onCompleteObservable.add(e); } /** * Callback raised when an error occurs. */ set onError(e) { this._onErrorObserver && this.onErrorObservable.remove(this._onErrorObserver), this._onErrorObserver = this.onErrorObservable.add(e); } /** * Callback raised after the loader is disposed. */ set onDispose(e) { this._onDisposeObserver && this.onDisposeObservable.remove(this._onDisposeObserver), this._onDisposeObserver = this.onDisposeObservable.add(e); } /** * Callback raised after a loader extension is created. */ set onExtensionLoaded(e) { this._onExtensionLoadedObserver && this.onExtensionLoadedObservable.remove(this._onExtensionLoadedObserver), this._onExtensionLoadedObserver = this.onExtensionLoadedObservable.add(e); } /** * Defines if the loader logging is enabled. */ get loggingEnabled() { return this._loggingEnabled; } set loggingEnabled(e) { this._loggingEnabled !== e && (this._loggingEnabled = e, this._loggingEnabled ? this._log = this._logEnabled : this._log = this._logDisabled); } /** * Defines if the loader should capture performance counters. */ get capturePerformanceCounters() { return this._capturePerformanceCounters; } set capturePerformanceCounters(e) { this._capturePerformanceCounters !== e && (this._capturePerformanceCounters = e, this._capturePerformanceCounters ? (this._startPerformanceCounter = this._startPerformanceCounterEnabled, this._endPerformanceCounter = this._endPerformanceCounterEnabled) : (this._startPerformanceCounter = this._startPerformanceCounterDisabled, this._endPerformanceCounter = this._endPerformanceCounterDisabled)); } /** * Callback raised after a loader extension is created. */ set onValidated(e) { this._onValidatedObserver && this.onValidatedObservable.remove(this._onValidatedObserver), this._onValidatedObserver = this.onValidatedObservable.add(e); } /** * Disposes the loader, releases resources during load, and cancels any outstanding requests. */ dispose() { this._loader && (this._loader.dispose(), this._loader = null); for (const e of this._requests) e.abort(); this._requests.length = 0, delete this._progressCallback, this.preprocessUrlAsync = (e) => Promise.resolve(e), this.onMeshLoadedObservable.clear(), this.onSkinLoadedObservable.clear(), this.onTextureLoadedObservable.clear(), this.onMaterialLoadedObservable.clear(), this.onCameraLoadedObservable.clear(), this.onCompleteObservable.clear(), this.onExtensionLoadedObservable.clear(), this.onDisposeObservable.notifyObservers(void 0), this.onDisposeObservable.clear(); } /** * @internal */ loadFile(e, t, r, n, i, s, a, f) { if (ArrayBuffer.isView(t)) return this._loadBinary(e, t, r, n, a, f), null; this._progressCallback = i; const o = t.name || ye.GetFilename(t); if (s) { if (this.useRangeRequests) { this.validate && Se.Warn("glTF validation is not supported when range requests are enabled"); const d = { abort: () => { }, onCompleteObservable: new Oe() }, v = { readAsync: (u, l) => new Promise((P, p) => { this._loadFile(e, t, (c) => { P(new Uint8Array(c)); }, !0, (c) => { p(c); }, (c) => { c.setRequestHeader("Range", `bytes=${u}-${u + l - 1}`); }); }), byteLength: 0 }; return this._unpackBinaryAsync(new gI(v)).then((u) => { d.onCompleteObservable.notifyObservers(d), n(u); }, a ? (u) => a(void 0, u) : void 0), d; } return this._loadFile(e, t, (d) => { this._validate(e, new Uint8Array(d), r, o), this._unpackBinaryAsync(new gI({ readAsync: (v, u) => sZ(d, v, u), byteLength: d.byteLength })).then((v) => { n(v); }, a ? (v) => a(void 0, v) : void 0); }, !0, a); } return this._loadFile(e, t, (d) => { this._validate(e, new Uint8Array(d), r, o), n({ json: this._parseJson(d) }); }, s, a); } _loadBinary(e, t, r, n, i, s) { this._validate(e, t, r, s), this._unpackBinaryAsync(new gI({ readAsync: (a, f) => $ge(t, a, f), byteLength: t.byteLength })).then((a) => { n(a); }, i ? (a) => i(void 0, a) : void 0); } /** * @internal */ importMeshAsync(e, t, r, n, i, s) { return Promise.resolve().then(() => (this.onParsedObservable.notifyObservers(r), this.onParsedObservable.clear(), this._log(`Loading ${s || ""}`), this._loader = this._getLoader(r), this._loader.importMeshAsync(e, t, null, r, n, i, s))); } /** * @internal */ loadAsync(e, t, r, n, i) { return Promise.resolve().then(() => (this.onParsedObservable.notifyObservers(t), this.onParsedObservable.clear(), this._log(`Loading ${i || ""}`), this._loader = this._getLoader(t), this._loader.loadAsync(e, t, r, n, i))); } /** * @internal */ loadAssetContainerAsync(e, t, r, n, i) { return Promise.resolve().then(() => { this.onParsedObservable.notifyObservers(t), this.onParsedObservable.clear(), this._log(`Loading ${i || ""}`), this._loader = this._getLoader(t); const s = new xR(e), a = []; this.onMaterialLoadedObservable.add((v) => { a.push(v); }); const f = []; this.onTextureLoadedObservable.add((v) => { f.push(v); }); const o = []; this.onCameraLoadedObservable.add((v) => { o.push(v); }); const d = []; return this.onMeshLoadedObservable.add((v) => { v.morphTargetManager && d.push(v.morphTargetManager); }), this._loader.importMeshAsync(null, e, s, t, r, n, i).then((v) => (Array.prototype.push.apply(s.geometries, v.geometries), Array.prototype.push.apply(s.meshes, v.meshes), Array.prototype.push.apply(s.particleSystems, v.particleSystems), Array.prototype.push.apply(s.skeletons, v.skeletons), Array.prototype.push.apply(s.animationGroups, v.animationGroups), Array.prototype.push.apply(s.materials, a), Array.prototype.push.apply(s.textures, f), Array.prototype.push.apply(s.lights, v.lights), Array.prototype.push.apply(s.transformNodes, v.transformNodes), Array.prototype.push.apply(s.cameras, o), Array.prototype.push.apply(s.morphTargetManagers, d), s)); }); } /** * @internal */ canDirectLoad(e) { return e.indexOf("asset") !== -1 && e.indexOf("version") !== -1 || e.startsWith("data:base64," + q1._MagicBase64Encoded) || // this is technically incorrect, but will continue to support for backcompat. e.startsWith("data:;base64," + q1._MagicBase64Encoded) || e.startsWith("data:application/octet-stream;base64," + q1._MagicBase64Encoded) || e.startsWith("data:model/gltf-binary;base64," + q1._MagicBase64Encoded); } /** * @internal */ directLoad(e, t) { if (t.startsWith("base64," + q1._MagicBase64Encoded) || // this is technically incorrect, but will continue to support for backcompat. t.startsWith(";base64," + q1._MagicBase64Encoded) || t.startsWith("application/octet-stream;base64," + q1._MagicBase64Encoded) || t.startsWith("model/gltf-binary;base64," + q1._MagicBase64Encoded)) { const r = iU(t); return this._validate(e, new Uint8Array(r)), this._unpackBinaryAsync(new gI({ readAsync: (n, i) => sZ(r, n, i), byteLength: r.byteLength })); } return this._validate(e, t), Promise.resolve({ json: this._parseJson(t) }); } /** @internal */ createPlugin() { return new q1(); } /** * The loader state or null if the loader is not active. */ get loaderState() { return this._state; } /** * Returns a promise that resolves when the asset is completely loaded. * @returns a promise that resolves when the asset is completely loaded. */ whenCompleteAsync() { return new Promise((e, t) => { this.onCompleteObservable.addOnce(() => { e(); }), this.onErrorObservable.addOnce((r) => { t(r); }); }); } /** * @internal */ _setState(e) { this._state !== e && (this._state = e, this.onLoaderStateChangedObservable.notifyObservers(this._state), this._log(cc[this._state])); } /** * @internal */ _loadFile(e, t, r, n, i, s) { const a = e._loadFile(t, r, (f) => { this._onProgress(f, a); }, !0, n, i, s); return a.onCompleteObservable.add((f) => { this._requests.splice(this._requests.indexOf(f), 1); }), this._requests.push(a), a; } _onProgress(e, t) { if (!this._progressCallback) return; t._lengthComputable = e.lengthComputable, t._loaded = e.loaded, t._total = e.total; let r = !0, n = 0, i = 0; for (const s of this._requests) { if (s._lengthComputable === void 0 || s._loaded === void 0 || s._total === void 0) return; r = r && s._lengthComputable, n += s._loaded, i += s._total; } this._progressCallback({ lengthComputable: r, loaded: n, total: r ? i : 0 }); } _validate(e, t, r = "", n = "") { this.validate && (this._startPerformanceCounter("Validate JSON"), M1e.ValidateAsync(t, r, n, (i) => this.preprocessUrlAsync(r + i).then((s) => e._loadFileAsync(s, void 0, !0, !0))).then((i) => { this._endPerformanceCounter("Validate JSON"), this.onValidatedObservable.notifyObservers(i), this.onValidatedObservable.clear(); }, (i) => { this._endPerformanceCounter("Validate JSON"), ye.Warn(`Failed to validate: ${i.message}`), this.onValidatedObservable.clear(); })); } _getLoader(e) { const t = e.json.asset || {}; this._log(`Asset version: ${t.version}`), t.minVersion && this._log(`Asset minimum version: ${t.minVersion}`), t.generator && this._log(`Asset generator: ${t.generator}`); const r = q1._parseVersion(t.version); if (!r) throw new Error("Invalid version: " + t.version); if (t.minVersion !== void 0) { const s = q1._parseVersion(t.minVersion); if (!s) throw new Error("Invalid minimum version: " + t.minVersion); if (q1._compareVersion(s, { major: 2, minor: 0 }) > 0) throw new Error("Incompatible minimum version: " + t.minVersion); } const i = { 1: q1._CreateGLTF1Loader, 2: q1._CreateGLTF2Loader }[r.major]; if (!i) throw new Error("Unsupported version: " + t.version); return i(this); } _parseJson(e) { this._startPerformanceCounter("Parse JSON"), this._log(`JSON length: ${e.length}`); const t = JSON.parse(e); return this._endPerformanceCounter("Parse JSON"), t; } _unpackBinaryAsync(e) { return this._startPerformanceCounter("Unpack Binary"), e.loadAsync(20).then(() => { const t = { Magic: 1179937895 }, r = e.readUint32(); if (r !== t.Magic) throw new O0("Unexpected magic: " + r, Z2.GLTFLoaderUnexpectedMagicError); const n = e.readUint32(); this.loggingEnabled && this._log(`Binary version: ${n}`); const i = e.readUint32(); !this.useRangeRequests && i !== e.buffer.byteLength && Se.Warn(`Length in header does not match actual data length: ${i} != ${e.buffer.byteLength}`); let s; switch (n) { case 1: { s = this._unpackBinaryV1Async(e, i); break; } case 2: { s = this._unpackBinaryV2Async(e, i); break; } default: throw new Error("Unsupported version: " + n); } return this._endPerformanceCounter("Unpack Binary"), s; }); } _unpackBinaryV1Async(e, t) { const r = { JSON: 0 }, n = e.readUint32(), i = e.readUint32(); if (i !== r.JSON) throw new Error(`Unexpected content format: ${i}`); const s = t - e.byteOffset, a = { json: this._parseJson(e.readString(n)), bin: null }; if (s !== 0) { const f = e.byteOffset; a.bin = { readAsync: (o, d) => e.buffer.readAsync(f + o, d), byteLength: s }; } return Promise.resolve(a); } _unpackBinaryV2Async(e, t) { const r = { JSON: 1313821514, BIN: 5130562 }, n = e.readUint32(); if (e.readUint32() !== r.JSON) throw new Error("First chunk format is not JSON"); return e.byteOffset + n === t ? e.loadAsync(n).then(() => ({ json: this._parseJson(e.readString(n)), bin: null })) : e.loadAsync(n + 8).then(() => { const s = { json: this._parseJson(e.readString(n)), bin: null }, a = () => { const f = e.readUint32(); switch (e.readUint32()) { case r.JSON: throw new Error("Unexpected JSON chunk"); case r.BIN: { const d = e.byteOffset; s.bin = { readAsync: (v, u) => e.buffer.readAsync(d + v, u), byteLength: f }, e.skipBytes(f); break; } default: { e.skipBytes(f); break; } } return e.byteOffset !== t ? e.loadAsync(8).then(a) : Promise.resolve(s); }; return a(); }); } static _parseVersion(e) { if (e === "1.0" || e === "1.0.1") return { major: 1, minor: 0 }; const t = (e + "").match(/^(\d+)\.(\d+)/); return t ? { major: parseInt(t[1]), minor: parseInt(t[2]) } : null; } static _compareVersion(e, t) { return e.major > t.major ? 1 : e.major < t.major ? -1 : e.minor > t.minor ? 1 : e.minor < t.minor ? -1 : 0; } /** * @internal */ _logOpen(e) { this._log(e), this._logIndentLevel++; } /** @internal */ _logClose() { --this._logIndentLevel; } _logEnabled(e) { const t = q1._logSpaces.substr(0, this._logIndentLevel * 2); Se.Log(`${t}${e}`); } _logDisabled(e) { } _startPerformanceCounterEnabled(e) { ye.StartPerformanceCounter(e); } _startPerformanceCounterDisabled(e) { } _endPerformanceCounterEnabled(e) { ye.EndPerformanceCounter(e); } _endPerformanceCounterDisabled(e) { } } q1.IncrementalLoading = !0; q1.HomogeneousCoordinates = !1; q1._MagicBase64Encoded = "Z2xURg"; q1._logSpaces = " "; Hn && Hn.RegisterPlugin(new q1()); var iD; (function(A) { A[A.BYTE = 5120] = "BYTE", A[A.UNSIGNED_BYTE = 5121] = "UNSIGNED_BYTE", A[A.SHORT = 5122] = "SHORT", A[A.UNSIGNED_SHORT = 5123] = "UNSIGNED_SHORT", A[A.FLOAT = 5126] = "FLOAT"; })(iD || (iD = {})); var yF; (function(A) { A[A.FRAGMENT = 35632] = "FRAGMENT", A[A.VERTEX = 35633] = "VERTEX"; })(yF || (yF = {})); var Yl; (function(A) { A[A.BYTE = 5120] = "BYTE", A[A.UNSIGNED_BYTE = 5121] = "UNSIGNED_BYTE", A[A.SHORT = 5122] = "SHORT", A[A.UNSIGNED_SHORT = 5123] = "UNSIGNED_SHORT", A[A.INT = 5124] = "INT", A[A.UNSIGNED_INT = 5125] = "UNSIGNED_INT", A[A.FLOAT = 5126] = "FLOAT", A[A.FLOAT_VEC2 = 35664] = "FLOAT_VEC2", A[A.FLOAT_VEC3 = 35665] = "FLOAT_VEC3", A[A.FLOAT_VEC4 = 35666] = "FLOAT_VEC4", A[A.INT_VEC2 = 35667] = "INT_VEC2", A[A.INT_VEC3 = 35668] = "INT_VEC3", A[A.INT_VEC4 = 35669] = "INT_VEC4", A[A.BOOL = 35670] = "BOOL", A[A.BOOL_VEC2 = 35671] = "BOOL_VEC2", A[A.BOOL_VEC3 = 35672] = "BOOL_VEC3", A[A.BOOL_VEC4 = 35673] = "BOOL_VEC4", A[A.FLOAT_MAT2 = 35674] = "FLOAT_MAT2", A[A.FLOAT_MAT3 = 35675] = "FLOAT_MAT3", A[A.FLOAT_MAT4 = 35676] = "FLOAT_MAT4", A[A.SAMPLER_2D = 35678] = "SAMPLER_2D"; })(Yl || (Yl = {})); var II; (function(A) { A[A.CLAMP_TO_EDGE = 33071] = "CLAMP_TO_EDGE", A[A.MIRRORED_REPEAT = 33648] = "MIRRORED_REPEAT", A[A.REPEAT = 10497] = "REPEAT"; })(II || (II = {})); var _H; (function(A) { A[A.NEAREST = 9728] = "NEAREST", A[A.LINEAR = 9728] = "LINEAR", A[A.NEAREST_MIPMAP_NEAREST = 9984] = "NEAREST_MIPMAP_NEAREST", A[A.LINEAR_MIPMAP_NEAREST = 9985] = "LINEAR_MIPMAP_NEAREST", A[A.NEAREST_MIPMAP_LINEAR = 9986] = "NEAREST_MIPMAP_LINEAR", A[A.LINEAR_MIPMAP_LINEAR = 9987] = "LINEAR_MIPMAP_LINEAR"; })(_H || (_H = {})); var aZ; (function(A) { A[A.ALPHA = 6406] = "ALPHA", A[A.RGB = 6407] = "RGB", A[A.RGBA = 6408] = "RGBA", A[A.LUMINANCE = 6409] = "LUMINANCE", A[A.LUMINANCE_ALPHA = 6410] = "LUMINANCE_ALPHA"; })(aZ || (aZ = {})); var kF; (function(A) { A[A.FRONT = 1028] = "FRONT", A[A.BACK = 1029] = "BACK", A[A.FRONT_AND_BACK = 1032] = "FRONT_AND_BACK"; })(kF || (kF = {})); var Hf; (function(A) { A[A.ZERO = 0] = "ZERO", A[A.ONE = 1] = "ONE", A[A.SRC_COLOR = 768] = "SRC_COLOR", A[A.ONE_MINUS_SRC_COLOR = 769] = "ONE_MINUS_SRC_COLOR", A[A.DST_COLOR = 774] = "DST_COLOR", A[A.ONE_MINUS_DST_COLOR = 775] = "ONE_MINUS_DST_COLOR", A[A.SRC_ALPHA = 770] = "SRC_ALPHA", A[A.ONE_MINUS_SRC_ALPHA = 771] = "ONE_MINUS_SRC_ALPHA", A[A.DST_ALPHA = 772] = "DST_ALPHA", A[A.ONE_MINUS_DST_ALPHA = 773] = "ONE_MINUS_DST_ALPHA", A[A.CONSTANT_COLOR = 32769] = "CONSTANT_COLOR", A[A.ONE_MINUS_CONSTANT_COLOR = 32770] = "ONE_MINUS_CONSTANT_COLOR", A[A.CONSTANT_ALPHA = 32771] = "CONSTANT_ALPHA", A[A.ONE_MINUS_CONSTANT_ALPHA = 32772] = "ONE_MINUS_CONSTANT_ALPHA", A[A.SRC_ALPHA_SATURATE = 776] = "SRC_ALPHA_SATURATE"; })(Hf || (Hf = {})); class j6 { /** * Sets the given "parameter" matrix * @param scene the Scene object * @param source the source node where to pick the matrix * @param parameter the GLTF technique parameter * @param uniformName the name of the shader's uniform * @param shaderMaterial the shader material */ static SetMatrix(e, t, r, n, i) { let s = null; if (r.semantic === "MODEL" ? s = t.getWorldMatrix() : r.semantic === "PROJECTION" ? s = e.getProjectionMatrix() : r.semantic === "VIEW" ? s = e.getViewMatrix() : r.semantic === "MODELVIEWINVERSETRANSPOSE" ? s = he.Transpose(t.getWorldMatrix().multiply(e.getViewMatrix()).invert()) : r.semantic === "MODELVIEW" ? s = t.getWorldMatrix().multiply(e.getViewMatrix()) : r.semantic === "MODELVIEWPROJECTION" ? s = t.getWorldMatrix().multiply(e.getTransformMatrix()) : r.semantic === "MODELINVERSE" ? s = t.getWorldMatrix().invert() : r.semantic === "VIEWINVERSE" ? s = e.getViewMatrix().invert() : r.semantic === "PROJECTIONINVERSE" ? s = e.getProjectionMatrix().invert() : r.semantic === "MODELVIEWINVERSE" ? s = t.getWorldMatrix().multiply(e.getViewMatrix()).invert() : r.semantic === "MODELVIEWPROJECTIONINVERSE" ? s = t.getWorldMatrix().multiply(e.getTransformMatrix()).invert() : r.semantic === "MODELINVERSETRANSPOSE" && (s = he.Transpose(t.getWorldMatrix().invert())), s) switch (r.type) { case Yl.FLOAT_MAT2: i.setMatrix2x2(n, he.GetAsMatrix2x2(s)); break; case Yl.FLOAT_MAT3: i.setMatrix3x3(n, he.GetAsMatrix3x3(s)); break; case Yl.FLOAT_MAT4: i.setMatrix(n, s); break; } } /** * Sets the given "parameter" matrix * @param shaderMaterial the shader material * @param uniform the name of the shader's uniform * @param value the value of the uniform * @param type the uniform's type (EParameterType FLOAT, VEC2, VEC3 or VEC4) */ static SetUniform(e, t, r, n) { switch (n) { case Yl.FLOAT: return e.setFloat(t, r), !0; case Yl.FLOAT_VEC2: return e.setVector2(t, at.FromArray(r)), !0; case Yl.FLOAT_VEC3: return e.setVector3(t, S.FromArray(r)), !0; case Yl.FLOAT_VEC4: return e.setVector4(t, Ir.FromArray(r)), !0; default: return !1; } } /** * Returns the wrap mode of the texture * @param mode the mode value */ static GetWrapMode(e) { switch (e) { case II.CLAMP_TO_EDGE: return We.CLAMP_ADDRESSMODE; case II.MIRRORED_REPEAT: return We.MIRROR_ADDRESSMODE; case II.REPEAT: return We.WRAP_ADDRESSMODE; default: return We.WRAP_ADDRESSMODE; } } /** * Returns the byte stride giving an accessor * @param accessor the GLTF accessor objet */ static GetByteStrideFromType(e) { switch (e.type) { case "VEC2": return 2; case "VEC3": return 3; case "VEC4": return 4; case "MAT2": return 4; case "MAT3": return 9; case "MAT4": return 16; default: return 1; } } /** * Returns the texture filter mode giving a mode value * @param mode the filter mode value * @returns the filter mode (TODO - needs to be a type?) */ static GetTextureFilterMode(e) { switch (e) { case _H.LINEAR: case _H.LINEAR_MIPMAP_NEAREST: case _H.LINEAR_MIPMAP_LINEAR: return We.TRILINEAR_SAMPLINGMODE; case _H.NEAREST: case _H.NEAREST_MIPMAP_NEAREST: return We.NEAREST_SAMPLINGMODE; default: return We.BILINEAR_SAMPLINGMODE; } } static GetBufferFromBufferView(e, t, r, n, i) { r = t.byteOffset + r; const s = e.loadedBufferViews[t.buffer]; if (r + n > s.byteLength) throw new Error("Buffer access is out of range"); const a = s.buffer; switch (r += s.byteOffset, i) { case iD.BYTE: return new Int8Array(a, r, n); case iD.UNSIGNED_BYTE: return new Uint8Array(a, r, n); case iD.SHORT: return new Int16Array(a, r, n); case iD.UNSIGNED_SHORT: return new Uint16Array(a, r, n); default: return new Float32Array(a, r, n); } } /** * Returns a buffer from its accessor * @param gltfRuntime the GLTF runtime * @param accessor the GLTF accessor */ static GetBufferFromAccessor(e, t) { const r = e.bufferViews[t.bufferView], n = t.count * j6.GetByteStrideFromType(t); return j6.GetBufferFromBufferView(e, r, t.byteOffset, n, t.componentType); } /** * Decodes a buffer view into a string * @param view the buffer view */ static DecodeBufferToText(e) { let t = ""; const r = e.byteLength; for (let n = 0; n < r; ++n) t += String.fromCharCode(e[n]); return t; } /** * Returns the default material of gltf. Related to * https://github.com/KhronosGroup/glTF/tree/master/specification/1.0#appendix-a-default-material * @param scene the Babylon.js scene */ static GetDefaultMaterial(e) { if (!j6._DefaultMaterial) { An.ShadersStore.GLTFDefaultMaterialVertexShader = [ "precision highp float;", "", "uniform mat4 worldView;", "uniform mat4 projection;", "", "attribute vec3 position;", "", "void main(void)", "{", " gl_Position = projection * worldView * vec4(position, 1.0);", "}" ].join(` `), An.ShadersStore.GLTFDefaultMaterialPixelShader = [ "precision highp float;", "", "uniform vec4 u_emission;", "", "void main(void)", "{", " gl_FragColor = u_emission;", "}" ].join(` `); const t = { vertex: "GLTFDefaultMaterial", fragment: "GLTFDefaultMaterial" }, r = { attributes: ["position"], uniforms: ["worldView", "projection", "u_emission"], samplers: new Array(), needAlphaBlending: !1 }; j6._DefaultMaterial = new Zo("GLTFDefaultMaterial", e, t, r), j6._DefaultMaterial.setColor4("u_emission", new xt(0.5, 0.5, 0.5, 1)); } return j6._DefaultMaterial; } } j6._DefaultMaterial = null; var sD; (function(A) { A[A.IDENTIFIER = 1] = "IDENTIFIER", A[A.UNKNOWN = 2] = "UNKNOWN", A[A.END_OF_INPUT = 3] = "END_OF_INPUT"; })(sD || (sD = {})); class oZ { constructor(e) { this._pos = 0, this.currentToken = sD.UNKNOWN, this.currentIdentifier = "", this.currentString = "", this.isLetterOrDigitPattern = /^[a-zA-Z0-9]+$/, this._toParse = e, this._maxPos = e.length; } getNextToken() { if (this.isEnd()) return sD.END_OF_INPUT; if (this.currentString = this.read(), this.currentToken = sD.UNKNOWN, this.currentString === "_" || this.isLetterOrDigitPattern.test(this.currentString)) for (this.currentToken = sD.IDENTIFIER, this.currentIdentifier = this.currentString; !this.isEnd() && (this.isLetterOrDigitPattern.test(this.currentString = this.peek()) || this.currentString === "_"); ) this.currentIdentifier += this.currentString, this.forward(); return this.currentToken; } peek() { return this._toParse[this._pos]; } read() { return this._toParse[this._pos++]; } forward() { this._pos++; } isEnd() { return this._pos >= this._maxPos; } } const L1e = ["MODEL", "VIEW", "PROJECTION", "MODELVIEW", "MODELVIEWPROJECTION", "JOINTMATRIX"], K1e = ["world", "view", "projection", "worldView", "worldViewProjection", "mBones"], eXe = ["translation", "rotation", "scale"], tXe = ["position", "rotationQuaternion", "scaling"], rXe = (A, e) => { for (const t in A) { const r = A[t]; e.buffers[t] = r, e.buffersCount++; } }, nXe = (A, e) => { for (const t in A) { const r = A[t]; e.shaders[t] = r, e.shaderscount++; } }, Su = (A, e, t) => { for (const r in A) { const n = A[r]; t[e][r] = n; } }, iXe = (A) => { if (A) for (let e = 0; e < A.length / 2; e++) A[e * 2 + 1] = 1 - A[e * 2 + 1]; }, fZ = (A) => { if (A.semantic === "NORMAL") return "normal"; if (A.semantic === "POSITION") return "position"; if (A.semantic === "JOINT") return "matricesIndices"; if (A.semantic === "WEIGHT") return "matricesWeights"; if (A.semantic === "COLOR") return "color"; if (A.semantic && A.semantic.indexOf("TEXCOORD_") !== -1) { const e = Number(A.semantic.split("_")[1]); return "uv" + (e === 0 ? "" : e + 1); } return null; }, sXe = (A) => { for (const e in A.animations) { const t = A.animations[e]; if (!t.channels || !t.samplers) continue; let r = null; for (let n = 0; n < t.channels.length; n++) { const i = t.channels[n], s = t.samplers[i.sampler]; if (!s) continue; let a = null, f = null; t.parameters ? (a = t.parameters[s.input], f = t.parameters[s.output]) : (a = s.input, f = s.output); const o = j6.GetBufferFromAccessor(A, A.accessors[a]), d = j6.GetBufferFromAccessor(A, A.accessors[f]), v = i.target.id; let u = A.scene.getNodeById(v); if (u === null && (u = A.scene.getNodeByName(v)), u === null) { ye.Warn("Creating animation named " + e + ". But cannot find node named " + v + " to attach to"); continue; } const l = u instanceof da; let P = i.target.path; const p = eXe.indexOf(P); p !== -1 && (P = tXe[p]); let c = st.ANIMATIONTYPE_MATRIX; l || (P === "rotationQuaternion" ? (c = st.ANIMATIONTYPE_QUATERNION, u.rotationQuaternion = new Ze()) : c = st.ANIMATIONTYPE_VECTOR3); let H = null; const T = []; let q = 0, b = !1; l && r && r.getKeys().length === o.length && (H = r, b = !0), b || (A.scene._blockEntityCollection = !!A.assetContainer, H = new st(e, l ? "_matrix" : P, 1, c, st.ANIMATIONLOOPMODE_CYCLE), A.scene._blockEntityCollection = !1); for (let j = 0; j < o.length; j++) { let w = null; if (P === "rotationQuaternion" ? (w = Ze.FromArray([d[q], d[q + 1], d[q + 2], d[q + 3]]), q += 4) : (w = S.FromArray([d[q], d[q + 1], d[q + 2]]), q += 3), l) { const m = u; let I = S.Zero(), N = new Ze(), k = S.Zero(), R = m.getBaseMatrix(); b && r && (R = r.getKeys()[j].value), R.decompose(k, N, I), P === "position" ? I = w : P === "rotationQuaternion" ? N = w : k = w, w = he.Compose(k, N, I); } b ? r && (r.getKeys()[j].value = w) : T.push({ frame: o[j], value: w }); } !b && H && (H.setKeys(T), u.animations.push(H)), r = H, A.scene.stopAnimation(u), A.scene.beginAnimation(u, 0, o[o.length - 1], !0, 1); } } }, rM = (A) => { let e = null; if (A.translation || A.rotation || A.scale) { const t = S.FromArray(A.scale || [1, 1, 1]), r = Ze.FromArray(A.rotation || [0, 0, 0, 1]), n = S.FromArray(A.translation || [0, 0, 0]); e = he.Compose(t, r, n); } else e = he.FromArray(A.matrix); return e; }, J1e = (A, e, t, r) => { for (let i = 0; i < r.bones.length; i++) if (r.bones[i].name === t) return r.bones[i]; const n = A.nodes; for (const i in n) { const s = n[i]; if (!s.jointName) continue; const a = s.children; for (let f = 0; f < a.length; f++) { const o = A.nodes[a[f]]; if (o.jointName && o.jointName === t) { const d = rM(s), v = new da(s.name || "", r, J1e(A, e, s.jointName, r), d); return v.id = i, v; } } } return null; }, aXe = (A, e) => { for (let t = 0; t < A.length; t++) { const r = A[t]; for (let n = 0; n < r.node.children.length; n++) if (r.node.children[n] === e) return r.bone; } return null; }, VE = (A, e) => { const t = A.nodes; let r = t[e]; if (r) return { node: r, id: e }; for (const n in t) if (r = t[n], r.jointName === e) return { node: r, id: n }; return null; }, oXe = (A, e) => { for (let t = 0; t < A.jointNames.length; t++) if (A.jointNames[t] === e) return !0; return !1; }, fXe = (A, e, t, r) => { for (const n in A.nodes) { const i = A.nodes[n], s = n; if (!i.jointName || oXe(t, i.jointName)) continue; const a = rM(i), f = new da(i.name || "", e, null, a); f.id = s, r.push({ bone: f, node: i, id: s }); } for (let n = 0; n < r.length; n++) { const i = r[n], s = i.node.children; for (let a = 0; a < s.length; a++) { let f = null; for (let o = 0; o < r.length; o++) if (r[o].id === s[a]) { f = r[o]; break; } f && (f.bone._parent = i.bone, i.bone.children.push(f.bone)); } } }, AXe = (A, e, t, r) => { if (r || (r = new r4(e.name || "", "", A.scene)), !e.babylonSkeleton) return r; const n = [], i = []; fXe(A, r, e, n), r.bones = []; for (let a = 0; a < e.jointNames.length; a++) { const f = VE(A, e.jointNames[a]); if (!f) continue; const o = f.node; if (!o) { ye.Warn("Joint named " + e.jointNames[a] + " does not exist"); continue; } const d = f.id, v = A.scene.getBoneById(d); if (v) { r.bones.push(v); continue; } let u = !1, l = null; for (let c = 0; c < a; c++) { const H = VE(A, e.jointNames[c]); if (!H) continue; const T = H.node; if (!T) { ye.Warn("Joint named " + e.jointNames[c] + " does not exist when looking for parent"); continue; } const q = T.children; if (q) { u = !1; for (let b = 0; b < q.length; b++) if (q[b] === d) { l = J1e(A, e, e.jointNames[c], r), u = !0; break; } if (u) break; } } const P = rM(o); !l && n.length > 0 && (l = aXe(n, d), l && i.indexOf(l) === -1 && i.push(l)); const p = new da(o.jointName || "", r, l, P); p.id = d; } const s = r.bones; r.bones = []; for (let a = 0; a < e.jointNames.length; a++) { const f = VE(A, e.jointNames[a]); if (f) { for (let o = 0; o < s.length; o++) if (s[o].id === f.id) { r.bones.push(s[o]); break; } } } r.prepare(); for (let a = 0; a < i.length; a++) r.bones.push(i[a]); return r; }, AZ = (A, e, t, r, n) => { if (n || (A.scene._blockEntityCollection = !!A.assetContainer, n = new Ee(e.name || "", A.scene), n._parentContainer = A.assetContainer, A.scene._blockEntityCollection = !1, n.id = r), !e.babylonNode) return n; const i = []; let s = null; const a = [], f = [], o = [], d = []; for (let l = 0; l < t.length; l++) { const P = t[l], p = A.meshes[P]; if (p) for (let c = 0; c < p.primitives.length; c++) { const H = new Ut(), T = p.primitives[c]; T.mode; const q = T.attributes; let b = null, j = null; for (const m in q) if (b = A.accessors[q[m]], j = j6.GetBufferFromAccessor(A, b), m === "NORMAL") H.normals = new Float32Array(j.length), H.normals.set(j); else if (m === "POSITION") { if (q1.HomogeneousCoordinates) { H.positions = new Float32Array(j.length - j.length / 4); for (let I = 0; I < j.length; I += 4) H.positions[I] = j[I], H.positions[I + 1] = j[I + 1], H.positions[I + 2] = j[I + 2]; } else H.positions = new Float32Array(j.length), H.positions.set(j); f.push(H.positions.length); } else if (m.indexOf("TEXCOORD_") !== -1) { const I = Number(m.split("_")[1]), N = J.UVKind + (I === 0 ? "" : I + 1), k = new Float32Array(j.length); k.set(j), iXe(k), H.set(k, N); } else m === "JOINT" ? (H.matricesIndices = new Float32Array(j.length), H.matricesIndices.set(j)) : m === "WEIGHT" ? (H.matricesWeights = new Float32Array(j.length), H.matricesWeights.set(j)) : m === "COLOR" && (H.colors = new Float32Array(j.length), H.colors.set(j)); if (b = A.accessors[T.indices], b) j = j6.GetBufferFromAccessor(A, b), H.indices = new Int32Array(j.length), H.indices.set(j), d.push(H.indices.length); else { const m = []; for (let I = 0; I < H.positions.length / 3; I++) m.push(I); H.indices = new Int32Array(m), d.push(H.indices.length); } s ? s.merge(H) : s = H; const w = A.scene.getMaterialById(T.material); i.push(w === null ? j6.GetDefaultMaterial(A.scene) : w), a.push(a.length === 0 ? 0 : a[a.length - 1] + f[f.length - 2]), o.push(o.length === 0 ? 0 : o[o.length - 1] + d[d.length - 2]); } } let v; A.scene._blockEntityCollection = !!A.assetContainer, i.length > 1 ? (v = new Dc("multimat" + r, A.scene), v.subMaterials = i) : v = new Wt("multimat" + r, A.scene), i.length === 1 && (v = i[0]), v._parentContainer = A.assetContainer, n.material || (n.material = v), new Tf(r, A.scene, s, !1, n), n.computeWorldMatrix(!0), A.scene._blockEntityCollection = !1, n.subMeshes = []; let u = 0; for (let l = 0; l < t.length; l++) { const P = t[l], p = A.meshes[P]; if (p) for (let c = 0; c < p.primitives.length; c++) p.primitives[c].mode, rA.AddToMesh(u, a[u], f[u], o[u], d[u], n, n, !0), u++; } return n; }, EF = (A, e, t, r) => { A.position && (A.position = e), (A.rotationQuaternion || A.rotation) && (A.rotationQuaternion = t), A.scaling && (A.scaling = r); }, dXe = (A, e) => { if (e.matrix) { const t = new S(0, 0, 0), r = new Ze(), n = new S(0, 0, 0); he.FromArray(e.matrix).decompose(n, r, t), EF(A, t, r, n); } else e.translation && e.rotation && e.scale && EF(A, S.FromArray(e.translation), Ze.FromArray(e.rotation), S.FromArray(e.scale)); A.computeWorldMatrix(!0); }, vXe = (A, e, t) => { let r = null; if (A.importOnlyMeshes && (e.skin || e.meshes) && A.importMeshesNames && A.importMeshesNames.length > 0 && A.importMeshesNames.indexOf(e.name || "") === -1) return null; if (e.skin) { if (e.meshes) { const n = A.skins[e.skin], i = AZ(A, e, e.meshes, t, e.babylonNode); i.skeleton = A.scene.getLastSkeletonById(e.skin), i.skeleton === null && (i.skeleton = AXe(A, n, i, n.babylonSkeleton), n.babylonSkeleton || (n.babylonSkeleton = i.skeleton)), r = i; } } else if (e.meshes) r = AZ(A, e, e.mesh ? [e.mesh] : e.meshes, t, e.babylonNode); else if (e.light && !e.babylonNode && !A.importOnlyMeshes) { const n = A.lights[e.light]; if (n) { if (n.type === "ambient") { const i = n[n.type], s = new y0(e.light, S.Zero(), A.scene); s.name = e.name || "", i.color && (s.diffuse = Ne.FromArray(i.color)), r = s; } else if (n.type === "directional") { const i = n[n.type], s = new IA(e.light, S.Zero(), A.scene); s.name = e.name || "", i.color && (s.diffuse = Ne.FromArray(i.color)), r = s; } else if (n.type === "point") { const i = n[n.type], s = new ag(e.light, S.Zero(), A.scene); s.name = e.name || "", i.color && (s.diffuse = Ne.FromArray(i.color)), r = s; } else if (n.type === "spot") { const i = n[n.type], s = new nA(e.light, S.Zero(), S.Zero(), 0, 0, A.scene); s.name = e.name || "", i.color && (s.diffuse = Ne.FromArray(i.color)), i.fallOfAngle && (s.angle = i.fallOfAngle), i.fallOffExponent && (s.exponent = i.fallOffExponent), r = s; } } } else if (e.camera && !e.babylonNode && !A.importOnlyMeshes) { const n = A.cameras[e.camera]; if (n) { if (A.scene._blockEntityCollection = !!A.assetContainer, n.type === "orthographic") { const i = new SA(e.camera, S.Zero(), A.scene, !1); i.name = e.name || "", i.mode = Tr.ORTHOGRAPHIC_CAMERA, i.attachControl(), r = i, i._parentContainer = A.assetContainer; } else if (n.type === "perspective") { const i = n[n.type], s = new SA(e.camera, S.Zero(), A.scene, !1); s.name = e.name || "", s.attachControl(), i.aspectRatio || (i.aspectRatio = A.scene.getEngine().getRenderWidth() / A.scene.getEngine().getRenderHeight()), i.znear && i.zfar && (s.maxZ = i.zfar, s.minZ = i.znear), r = s, s._parentContainer = A.assetContainer; } A.scene._blockEntityCollection = !1; } } if (!e.jointName) { if (e.babylonNode) return e.babylonNode; if (r === null) { A.scene._blockEntityCollection = !!A.assetContainer; const n = new Ee(e.name || "", A.scene); n._parentContainer = A.assetContainer, A.scene._blockEntityCollection = !1, e.babylonNode = n, r = n; } } if (r !== null) { if (e.matrix && r instanceof Ee) dXe(r, e); else { const n = e.translation || [0, 0, 0], i = e.rotation || [0, 0, 0, 1], s = e.scale || [1, 1, 1]; EF(r, S.FromArray(n), Ze.FromArray(i), S.FromArray(s)); } r.updateCache(!0), e.babylonNode = r; } return r; }, dR = (A, e, t, r = !1) => { const n = A.nodes[e]; let i = null; if (A.importOnlyMeshes && !r && A.importMeshesNames ? A.importMeshesNames.indexOf(n.name || "") !== -1 || A.importMeshesNames.length === 0 ? r = !0 : r = !1 : r = !0, !n.jointName && r && (i = vXe(A, n, e), i !== null && (i.id = e, i.parent = t)), n.children) for (let s = 0; s < n.children.length; s++) dR(A, n.children[s], i, r); }, dZ = (A) => { let e = A.currentScene; if (e) for (let t = 0; t < e.nodes.length; t++) dR(A, e.nodes[t], null); else for (const t in A.scenes) { e = A.scenes[t]; for (let r = 0; r < e.nodes.length; r++) dR(A, e.nodes[r], null); } sXe(A); for (let t = 0; t < A.scene.skeletons.length; t++) { const r = A.scene.skeletons[t]; A.scene.beginAnimation(r, 0, Number.MAX_VALUE, !0, 1); } }, uXe = (A, e, t, r, n, i, s) => { const a = i.values || n.parameters; for (const f in t) { const o = t[f], d = o.type; if (d === Yl.FLOAT_MAT2 || d === Yl.FLOAT_MAT3 || d === Yl.FLOAT_MAT4) { if (o.semantic && !o.source && !o.node) j6.SetMatrix(e.scene, A, o, f, r.getEffect()); else if (o.semantic && (o.source || o.node)) { let v = e.scene.getNodeByName(o.source || o.node || ""); if (v === null && (v = e.scene.getNodeById(o.source || o.node || "")), v === null) continue; j6.SetMatrix(e.scene, v, o, f, r.getEffect()); } } else { const v = a[n.uniforms[f]]; if (!v) continue; if (d === Yl.SAMPLER_2D) { const u = e.textures[i.values ? v : o.value].babylonTexture; if (u == null) continue; r.getEffect().setTexture(f, u); } else j6.SetUniform(r.getEffect(), f, v, d); } } s(r); }, lXe = (A, e, t, r, n) => { const i = r.values || t.parameters, s = t.uniforms; for (const a in n) { const f = n[a], o = f.type; let d = i[s[a]]; if (d === void 0 && (d = f.value), !d) continue; const v = (u) => (l) => { f.value && u && (e.setTexture(u, l), delete n[u]); }; o === Yl.SAMPLER_2D ? od.LoadTextureAsync(A, r.values ? d : f.value, v(a), () => v(null)) : f.value && j6.SetUniform(e, a, r.values ? d : f.value, o) && delete n[a]; } }, PXe = (A, e, t) => (r, n) => { e.dispose(!0), t("Cannot compile program named " + A.name + ". Error: " + n + ". Default material will be applied"); }, cXe = (A, e, t, r, n, i) => (s) => { lXe(A, e, t, r, n), e.onBind = (a) => { uXe(a, A, n, e, t, r, i); }; }, vZ = (A, e, t) => { for (const r in e.uniforms) { const n = e.uniforms[r], i = e.parameters[n]; if (A.currentIdentifier === r && i.semantic && !i.source && !i.node) { const s = L1e.indexOf(i.semantic); if (s !== -1) return delete t[r], K1e[s]; } } return A.currentIdentifier; }, uZ = (A) => { for (const e in A.materials) od.LoadMaterialAsync(A, e, () => { }, () => { }); }; class eq { static CreateRuntime(e, t, r) { const n = { extensions: {}, accessors: {}, buffers: {}, bufferViews: {}, meshes: {}, lights: {}, cameras: {}, nodes: {}, images: {}, textures: {}, shaders: {}, programs: {}, samplers: {}, techniques: {}, materials: {}, animations: {}, skins: {}, extensionsUsed: [], scenes: {}, buffersCount: 0, shaderscount: 0, scene: t, rootUrl: r, loadedBufferCount: 0, loadedBufferViews: {}, loadedShaderCount: 0, importOnlyMeshes: !1, dummyNodes: [], assetContainer: null }; return e.extensions && Su(e.extensions, "extensions", n), e.extensionsUsed && Su(e.extensionsUsed, "extensionsUsed", n), e.buffers && rXe(e.buffers, n), e.bufferViews && Su(e.bufferViews, "bufferViews", n), e.accessors && Su(e.accessors, "accessors", n), e.meshes && Su(e.meshes, "meshes", n), e.lights && Su(e.lights, "lights", n), e.cameras && Su(e.cameras, "cameras", n), e.nodes && Su(e.nodes, "nodes", n), e.images && Su(e.images, "images", n), e.textures && Su(e.textures, "textures", n), e.shaders && nXe(e.shaders, n), e.programs && Su(e.programs, "programs", n), e.samplers && Su(e.samplers, "samplers", n), e.techniques && Su(e.techniques, "techniques", n), e.materials && Su(e.materials, "materials", n), e.animations && Su(e.animations, "animations", n), e.skins && Su(e.skins, "skins", n), e.scenes && (n.scenes = e.scenes), e.scene && e.scenes && (n.currentScene = e.scenes[e.scene]), n; } static LoadBufferAsync(e, t, r, n, i) { const s = e.buffers[t]; ye.IsBase64(s.uri) ? setTimeout(() => r(new Uint8Array(ye.DecodeBase64(s.uri)))) : ye.LoadFile(e.rootUrl + s.uri, (a) => r(new Uint8Array(a)), i, void 0, !0, (a) => { a && n(a.status + " " + a.statusText); }); } static LoadTextureBufferAsync(e, t, r, n) { const i = e.textures[t]; if (!i || !i.source) { n(""); return; } if (i.babylonTexture) { r(null); return; } const s = e.images[i.source]; ye.IsBase64(s.uri) ? setTimeout(() => r(new Uint8Array(ye.DecodeBase64(s.uri)))) : ye.LoadFile(e.rootUrl + s.uri, (a) => r(new Uint8Array(a)), void 0, void 0, !0, (a) => { a && n(a.status + " " + a.statusText); }); } static CreateTextureAsync(e, t, r, n) { const i = e.textures[t]; if (i.babylonTexture) { n(i.babylonTexture); return; } const s = e.samplers[i.sampler], a = s.minFilter === _H.NEAREST_MIPMAP_NEAREST || s.minFilter === _H.NEAREST_MIPMAP_LINEAR || s.minFilter === _H.LINEAR_MIPMAP_NEAREST || s.minFilter === _H.LINEAR_MIPMAP_LINEAR, f = We.BILINEAR_SAMPLINGMODE, o = r == null ? new Blob() : new Blob([r]), d = URL.createObjectURL(o), v = () => URL.revokeObjectURL(d), u = new We(d, e.scene, !a, !0, f, v, v); s.wrapS !== void 0 && (u.wrapU = j6.GetWrapMode(s.wrapS)), s.wrapT !== void 0 && (u.wrapV = j6.GetWrapMode(s.wrapT)), u.name = t, i.babylonTexture = u, n(u); } static LoadShaderStringAsync(e, t, r, n) { const i = e.shaders[t]; if (ye.IsBase64(i.uri)) { const s = atob(i.uri.split(",")[1]); r && r(s); } else ye.LoadFile(e.rootUrl + i.uri, r, void 0, void 0, !1, (s) => { s && n && n(s.status + " " + s.statusText); }); } static LoadMaterialAsync(e, t, r, n) { const i = e.materials[t]; if (!i.technique) { n && n("No technique found."); return; } const s = e.techniques[i.technique]; if (!s) { e.scene._blockEntityCollection = !!e.assetContainer; const w = new Wt(t, e.scene); w._parentContainer = e.assetContainer, e.scene._blockEntityCollection = !1, w.diffuseColor = new Ne(0.5, 0.5, 0.5), w.sideOrientation = gt.CounterClockWiseSideOrientation, r(w); return; } const a = e.programs[s.program], f = s.states, o = An.ShadersStore[a.vertexShader + "VertexShader"], d = An.ShadersStore[a.fragmentShader + "PixelShader"]; let v = "", u = ""; const l = new oZ(o), P = new oZ(d), p = {}, c = [], H = [], T = []; for (const w in s.uniforms) { const m = s.uniforms[w], I = s.parameters[m]; if (p[w] = I, I.semantic && !I.node && !I.source) { const N = L1e.indexOf(I.semantic); N !== -1 ? (c.push(K1e[N]), delete p[w]) : c.push(w); } else I.type === Yl.SAMPLER_2D ? T.push(w) : c.push(w); } for (const w in s.attributes) { const m = s.attributes[w], I = s.parameters[m]; if (I.semantic) { const N = fZ(I); N && H.push(N); } } for (; !l.isEnd() && l.getNextToken(); ) { if (l.currentToken !== sD.IDENTIFIER) { v += l.currentString; continue; } let m = !1; for (const I in s.attributes) { const N = s.attributes[I], k = s.parameters[N]; if (l.currentIdentifier === I && k.semantic) { v += fZ(k), m = !0; break; } } m || (v += vZ(l, s, p)); } for (; !P.isEnd() && P.getNextToken(); ) { if (P.currentToken !== sD.IDENTIFIER) { u += P.currentString; continue; } u += vZ(P, s, p); } const q = { vertex: a.vertexShader + t, fragment: a.fragmentShader + t }, b = { attributes: H, uniforms: c, samplers: T, needAlphaBlending: f && f.enable && f.enable.indexOf(3042) !== -1 }; An.ShadersStore[a.vertexShader + t + "VertexShader"] = v, An.ShadersStore[a.fragmentShader + t + "PixelShader"] = u; const j = new Zo(t, e.scene, q, b); if (j.onError = PXe(a, j, n), j.onCompiled = cXe(e, j, s, i, p, r), j.sideOrientation = gt.CounterClockWiseSideOrientation, f && f.functions) { const w = f.functions; w.cullFace && w.cullFace[0] !== kF.BACK && (j.backFaceCulling = !1); const m = w.blendFuncSeparate; m && (m[0] === Hf.SRC_ALPHA && m[1] === Hf.ONE_MINUS_SRC_ALPHA && m[2] === Hf.ONE && m[3] === Hf.ONE ? j.alphaMode = et.ALPHA_COMBINE : m[0] === Hf.ONE && m[1] === Hf.ONE && m[2] === Hf.ZERO && m[3] === Hf.ONE ? j.alphaMode = et.ALPHA_ONEONE : m[0] === Hf.SRC_ALPHA && m[1] === Hf.ONE && m[2] === Hf.ZERO && m[3] === Hf.ONE ? j.alphaMode = et.ALPHA_ADD : m[0] === Hf.ZERO && m[1] === Hf.ONE_MINUS_SRC_COLOR && m[2] === Hf.ONE && m[3] === Hf.ONE ? j.alphaMode = et.ALPHA_SUBTRACT : m[0] === Hf.DST_COLOR && m[1] === Hf.ZERO && m[2] === Hf.ONE && m[3] === Hf.ONE ? j.alphaMode = et.ALPHA_MULTIPLY : m[0] === Hf.SRC_ALPHA && m[1] === Hf.ONE_MINUS_SRC_COLOR && m[2] === Hf.ONE && m[3] === Hf.ONE && (j.alphaMode = et.ALPHA_MAXIMIZED)); } } } let zS = class FF { static RegisterExtension(e) { if (FF.Extensions[e.name]) { ye.Error('Tool with the same name "' + e.name + '" already exists'); return; } FF.Extensions[e.name] = e; } dispose() { } _importMeshAsync(e, t, r, n, i, s, a, f) { return t.useRightHandedSystem = !0, od.LoadRuntimeAsync(t, r, n, (o) => { o.assetContainer = i, o.importOnlyMeshes = !0, e === "" ? o.importMeshesNames = [] : typeof e == "string" ? o.importMeshesNames = [e] : e && !(e instanceof Array) ? o.importMeshesNames = [e] : (o.importMeshesNames = [], ye.Warn("Argument meshesNames must be of type string or string[]")), this._createNodes(o); const d = [], v = []; for (const u in o.nodes) { const l = o.nodes[u]; l.babylonNode instanceof jn && d.push(l.babylonNode); } for (const u in o.skins) { const l = o.skins[u]; l.babylonSkeleton instanceof r4 && v.push(l.babylonSkeleton); } this._loadBuffersAsync(o, () => { this._loadShadersAsync(o, () => { uZ(o), dZ(o), !q1.IncrementalLoading && s && s(d, v); }); }), q1.IncrementalLoading && s && s(d, v); }, f), !0; } /** * Imports one or more meshes from a loaded gltf file and adds them to the scene * @param meshesNames a string or array of strings of the mesh names that should be loaded from the file * @param scene the scene the meshes should be added to * @param assetContainer defines the asset container to use (can be null) * @param data gltf data containing information of the meshes in a loaded file * @param rootUrl root url to load from * @param onProgress event that fires when loading progress has occured * @returns a promise containg the loaded meshes, particles, skeletons and animations */ importMeshAsync(e, t, r, n, i, s) { return new Promise((a, f) => { this._importMeshAsync(e, t, n, i, r, (o, d) => { a({ meshes: o, particleSystems: [], skeletons: d, animationGroups: [], lights: [], transformNodes: [], geometries: [] }); }, s, (o) => { f(new Error(o)); }); }); } _loadAsync(e, t, r, n, i, s) { e.useRightHandedSystem = !0, od.LoadRuntimeAsync(e, t, r, (a) => { od.LoadRuntimeExtensionsAsync(a, () => { this._createNodes(a), this._loadBuffersAsync(a, () => { this._loadShadersAsync(a, () => { uZ(a), dZ(a), q1.IncrementalLoading || n(); }); }), q1.IncrementalLoading && n(); }, s); }, s); } /** * Imports all objects from a loaded gltf file and adds them to the scene * @param scene the scene the objects should be added to * @param data gltf data containing information of the meshes in a loaded file * @param rootUrl root url to load from * @param onProgress event that fires when loading progress has occured * @returns a promise which completes when objects have been loaded to the scene */ loadAsync(e, t, r, n) { return new Promise((i, s) => { this._loadAsync(e, t, r, () => { i(); }, n, (a) => { s(new Error(a)); }); }); } _loadShadersAsync(e, t) { let r = !1; const n = (i, s) => { od.LoadShaderStringAsync(e, i, (a) => { a instanceof ArrayBuffer || (e.loadedShaderCount++, a && (An.ShadersStore[i + (s.type === yF.VERTEX ? "VertexShader" : "PixelShader")] = a), e.loadedShaderCount === e.shaderscount && t()); }, () => { ye.Error("Error when loading shader program named " + i + " located at " + s.uri); }); }; for (const i in e.shaders) { r = !0; const s = e.shaders[i]; s ? n.bind(this, i, s)() : ye.Error("No shader named: " + i); } r || t(); } _loadBuffersAsync(e, t) { let r = !1; const n = (i, s) => { od.LoadBufferAsync(e, i, (a) => { e.loadedBufferCount++, a && (a.byteLength != e.buffers[i].byteLength && ye.Error("Buffer named " + i + " is length " + a.byteLength + ". Expected: " + s.byteLength), e.loadedBufferViews[i] = a), e.loadedBufferCount === e.buffersCount && t(); }, () => { ye.Error("Error when loading buffer named " + i + " located at " + s.uri); }); }; for (const i in e.buffers) { r = !0; const s = e.buffers[i]; s ? n.bind(this, i, s)() : ye.Error("No buffer named: " + i); } r || t(); } _createNodes(e) { let t = e.currentScene; if (t) for (let r = 0; r < t.nodes.length; r++) dR(e, t.nodes[r], null); else for (const r in e.scenes) { t = e.scenes[r]; for (let n = 0; n < t.nodes.length; n++) dR(e, t.nodes[n], null); } } }; zS.Extensions = {}; class od { constructor(e) { this._name = e; } get name() { return this._name; } /** * Defines an override for loading the runtime * Return true to stop further extensions from loading the runtime * @param scene * @param data * @param rootUrl * @param onSuccess * @param onError */ loadRuntimeAsync(e, t, r, n, i) { return !1; } /** * Defines an onverride for creating gltf runtime * Return true to stop further extensions from creating the runtime * @param gltfRuntime * @param onSuccess * @param onError */ loadRuntimeExtensionsAsync(e, t, r) { return !1; } /** * Defines an override for loading buffers * Return true to stop further extensions from loading this buffer * @param gltfRuntime * @param id * @param onSuccess * @param onError * @param onProgress */ loadBufferAsync(e, t, r, n, i) { return !1; } /** * Defines an override for loading texture buffers * Return true to stop further extensions from loading this texture data * @param gltfRuntime * @param id * @param onSuccess * @param onError */ loadTextureBufferAsync(e, t, r, n) { return !1; } /** * Defines an override for creating textures * Return true to stop further extensions from loading this texture * @param gltfRuntime * @param id * @param buffer * @param onSuccess * @param onError */ createTextureAsync(e, t, r, n, i) { return !1; } /** * Defines an override for loading shader strings * Return true to stop further extensions from loading this shader data * @param gltfRuntime * @param id * @param onSuccess * @param onError */ loadShaderStringAsync(e, t, r, n) { return !1; } /** * Defines an override for loading materials * Return true to stop further extensions from loading this material * @param gltfRuntime * @param id * @param onSuccess * @param onError */ loadMaterialAsync(e, t, r, n) { return !1; } // --------- // Utilities // --------- static LoadRuntimeAsync(e, t, r, n, i) { od._ApplyExtensions((s) => s.loadRuntimeAsync(e, t, r, n, i), () => { setTimeout(() => { n && n(eq.CreateRuntime(t.json, e, r)); }); }); } static LoadRuntimeExtensionsAsync(e, t, r) { od._ApplyExtensions((n) => n.loadRuntimeExtensionsAsync(e, t, r), () => { setTimeout(() => { t(); }); }); } static LoadBufferAsync(e, t, r, n, i) { od._ApplyExtensions((s) => s.loadBufferAsync(e, t, r, n, i), () => { eq.LoadBufferAsync(e, t, r, n, i); }); } static LoadTextureAsync(e, t, r, n) { od._LoadTextureBufferAsync(e, t, (i) => { i && od._CreateTextureAsync(e, t, i, r, n); }, n); } static LoadShaderStringAsync(e, t, r, n) { od._ApplyExtensions((i) => i.loadShaderStringAsync(e, t, r, n), () => { eq.LoadShaderStringAsync(e, t, r, n); }); } static LoadMaterialAsync(e, t, r, n) { od._ApplyExtensions((i) => i.loadMaterialAsync(e, t, r, n), () => { eq.LoadMaterialAsync(e, t, r, n); }); } static _LoadTextureBufferAsync(e, t, r, n) { od._ApplyExtensions((i) => i.loadTextureBufferAsync(e, t, r, n), () => { eq.LoadTextureBufferAsync(e, t, r, n); }); } static _CreateTextureAsync(e, t, r, n, i) { od._ApplyExtensions((s) => s.createTextureAsync(e, t, r, n, i), () => { eq.CreateTextureAsync(e, t, r, n); }); } static _ApplyExtensions(e, t) { for (const r in zS.Extensions) { const n = zS.Extensions[r]; if (e(n)) return; } t(); } } q1._CreateGLTF1Loader = () => new zS(); const pXe = "binary_glTF"; class hXe extends od { constructor() { super("KHR_binary_glTF"); } loadRuntimeAsync(e, t, r, n) { const i = t.json.extensionsUsed; return !i || i.indexOf(this.name) === -1 || !t.bin ? !1 : (this._bin = t.bin, n(eq.CreateRuntime(t.json, e, r)), !0); } loadBufferAsync(e, t, r, n) { return e.extensionsUsed.indexOf(this.name) === -1 || t !== pXe ? !1 : (this._bin.readAsync(0, this._bin.byteLength).then(r, (i) => n(i.message)), !0); } loadTextureBufferAsync(e, t, r) { const n = e.textures[t], i = e.images[n.source]; if (!i.extensions || !(this.name in i.extensions)) return !1; const s = i.extensions[this.name], a = e.bufferViews[s.bufferView], f = j6.GetBufferFromBufferView(e, a, 0, a.byteLength, iD.UNSIGNED_BYTE); return r(f), !0; } loadShaderStringAsync(e, t, r) { const n = e.shaders[t]; if (!n.extensions || !(this.name in n.extensions)) return !1; const i = n.extensions[this.name], s = e.bufferViews[i.bufferView], a = j6.GetBufferFromBufferView(e, s, 0, s.byteLength, iD.UNSIGNED_BYTE); return setTimeout(() => { const f = j6.DecodeBufferToText(a); r(f); }), !0; } } zS.RegisterExtension(new hXe()); class HXe extends od { constructor() { super("KHR_materials_common"); } loadRuntimeExtensionsAsync(e) { if (!e.extensions) return !1; const t = e.extensions[this.name]; if (!t) return !1; const r = t.lights; if (r) for (const n in r) { const i = r[n]; switch (i.type) { case "ambient": { const s = new y0(i.name, new S(0, 1, 0), e.scene), a = i.ambient; a && (s.diffuse = Ne.FromArray(a.color || [1, 1, 1])); break; } case "point": { const s = new ag(i.name, new S(10, 10, 10), e.scene), a = i.point; a && (s.diffuse = Ne.FromArray(a.color || [1, 1, 1])); break; } case "directional": { const s = new IA(i.name, new S(0, -1, 0), e.scene), a = i.directional; a && (s.diffuse = Ne.FromArray(a.color || [1, 1, 1])); break; } case "spot": { const s = i.spot; if (s) { const a = new nA(i.name, new S(0, 10, 0), new S(0, -1, 0), s.fallOffAngle || Math.PI, s.fallOffExponent || 0, e.scene); a.diffuse = Ne.FromArray(s.color || [1, 1, 1]); } break; } default: ye.Warn('GLTF Material Common extension: light type "' + i.type + "” not supported"); break; } } return !1; } loadMaterialAsync(e, t, r, n) { const i = e.materials[t]; if (!i || !i.extensions) return !1; const s = i.extensions[this.name]; if (!s) return !1; const a = new Wt(t, e.scene); return a.sideOrientation = gt.CounterClockWiseSideOrientation, s.technique === "CONSTANT" && (a.disableLighting = !0), a.backFaceCulling = s.doubleSided === void 0 ? !1 : !s.doubleSided, a.alpha = s.values.transparency === void 0 ? 1 : s.values.transparency, a.specularPower = s.values.shininess === void 0 ? 0 : s.values.shininess, typeof s.values.ambient == "string" ? this._loadTexture(e, s.values.ambient, a, "ambientTexture", n) : a.ambientColor = Ne.FromArray(s.values.ambient || [0, 0, 0]), typeof s.values.diffuse == "string" ? this._loadTexture(e, s.values.diffuse, a, "diffuseTexture", n) : a.diffuseColor = Ne.FromArray(s.values.diffuse || [0, 0, 0]), typeof s.values.emission == "string" ? this._loadTexture(e, s.values.emission, a, "emissiveTexture", n) : a.emissiveColor = Ne.FromArray(s.values.emission || [0, 0, 0]), typeof s.values.specular == "string" ? this._loadTexture(e, s.values.specular, a, "specularTexture", n) : a.specularColor = Ne.FromArray(s.values.specular || [0, 0, 0]), !0; } _loadTexture(e, t, r, n, i) { eq.LoadTextureBufferAsync(e, t, (s) => { eq.CreateTextureAsync(e, t, s, (a) => r[n] = a); }, i); } } zS.RegisterExtension(new HXe()); function lZ(A, e, t, r) { return S.FromArray(e, t).scaleInPlace(r); } function gXe(A, e, t, r) { return Ze.FromArray(e, t).scaleInPlace(r); } function XXe(A, e, t, r) { const n = new Array(A._numMorphTargets); for (let i = 0; i < n.length; i++) n[i] = e[t++] * r; return n; } class HV { /** @internal */ constructor(e, t, r, n) { this.type = e, this.name = t, this.getValue = r, this.getStride = n; } _buildAnimation(e, t, r) { const n = new st(e, this.name, t, this.type); return n.setKeys(r), n; } } class CE extends HV { /** @internal */ buildAnimations(e, t, r, n, i) { i(e._babylonTransformNode, this._buildAnimation(t, r, n)); } } class TXe extends HV { buildAnimations(e, t, r, n, i) { if (e._numMorphTargets) for (let s = 0; s < e._numMorphTargets; s++) { const a = new st(`${t}_${s}`, this.name, r, this.type); if (a.setKeys(n.map((f) => ({ frame: f.frame, inTangent: f.inTangent ? f.inTangent[s] : void 0, value: f.value[s], outTangent: f.outTangent ? f.outTangent[s] : void 0, interpolation: f.interpolation }))), e._primitiveBabylonMeshes) { for (const f of e._primitiveBabylonMeshes) if (f.morphTargetManager) { const o = f.morphTargetManager.getTarget(s), d = a.clone(); o.animations.push(d), i(o, d); } } } } } const XI = { translation: [new CE(st.ANIMATIONTYPE_VECTOR3, "position", lZ, () => 3)], rotation: [new CE(st.ANIMATIONTYPE_QUATERNION, "rotationQuaternion", gXe, () => 4)], scale: [new CE(st.ANIMATIONTYPE_VECTOR3, "scaling", lZ, () => 3)], weights: [new TXe(st.ANIMATIONTYPE_FLOAT, "influence", XXe, (A) => A._numMorphTargets)] }; function z1e(...A) { const e = (t) => t && typeof t == "object"; return A.reduce((t, r) => (Object.keys(r).forEach((n) => { const i = t[n], s = r[n]; Array.isArray(i) && Array.isArray(s) ? t[n] = i.concat(...s) : e(i) && e(s) ? t[n] = z1e(i, s) : t[n] = s; }), t), {}); } class ei { /** * Gets an item from the given array. * @param context The context when loading the asset * @param array The array to get the item from * @param index The index to the array * @returns The array item */ static Get(e, t, r) { if (!t || r == null || !t[r]) throw new Error(`${e}: Failed to find index (${r})`); return t[r]; } /** * Gets an item from the given array or returns null if not available. * @param array The array to get the item from * @param index The index to the array * @returns The array item or null */ static TryGet(e, t) { return !e || t == null || !e[t] ? null : e[t]; } /** * Assign an `index` field to each item of the given array. * @param array The array of items */ static Assign(e) { if (e) for (let t = 0; t < e.length; t++) e[t].index = t; } } class Jr { /** * Registers a loader extension. * @param name The name of the loader extension. * @param factory The factory function that creates the loader extension. */ static RegisterExtension(e, t) { Jr.UnregisterExtension(e) && Se.Warn(`Extension with the name '${e}' already exists`), Jr._RegisteredExtensions[e] = { factory: t }; } /** * Unregisters a loader extension. * @param name The name of the loader extension. * @returns A boolean indicating whether the extension has been unregistered */ static UnregisterExtension(e) { return Jr._RegisteredExtensions[e] ? (delete Jr._RegisteredExtensions[e], !0) : !1; } /** * The object that represents the glTF JSON. */ get gltf() { if (!this._gltf) throw new Error("glTF JSON is not available"); return this._gltf; } /** * The BIN chunk of a binary glTF. */ get bin() { return this._bin; } /** * The parent file loader. */ get parent() { return this._parent; } /** * The Babylon scene when loading the asset. */ get babylonScene() { if (!this._babylonScene) throw new Error("Scene is not available"); return this._babylonScene; } /** * The root Babylon mesh when loading the asset. */ get rootBabylonMesh() { return this._rootBabylonMesh; } /** * @internal */ constructor(e) { this._completePromises = new Array(), this._assetContainer = null, this._babylonLights = [], this._disableInstancedMesh = 0, this._allMaterialsDirtyRequired = !1, this._extensions = new Array(), this._disposed = !1, this._rootUrl = null, this._fileName = null, this._uniqueRootUrl = null, this._bin = null, this._rootBabylonMesh = null, this._defaultBabylonMaterialData = {}, this._postSceneLoadActions = new Array(), this._parent = e; } /** @internal */ dispose() { this._disposed || (this._disposed = !0, this._completePromises.length = 0, this._extensions.forEach((e) => e.dispose && e.dispose()), this._extensions.length = 0, this._gltf = null, this._bin = null, this._babylonScene = null, this._rootBabylonMesh = null, this._defaultBabylonMaterialData = {}, this._postSceneLoadActions.length = 0, this._parent.dispose()); } /** * @internal */ importMeshAsync(e, t, r, n, i, s, a = "") { return Promise.resolve().then(() => { this._babylonScene = t, this._assetContainer = r, this._loadData(n); let f = null; if (e) { const o = {}; if (this._gltf.nodes) for (const v of this._gltf.nodes) v.name && (o[v.name] = v.index); f = (e instanceof Array ? e : [e]).map((v) => { const u = o[v]; if (u === void 0) throw new Error(`Failed to find node '${v}'`); return u; }); } return this._loadAsync(i, a, f, () => ({ meshes: this._getMeshes(), particleSystems: [], skeletons: this._getSkeletons(), animationGroups: this._getAnimationGroups(), lights: this._babylonLights, transformNodes: this._getTransformNodes(), geometries: this._getGeometries() })); }); } /** * @internal */ loadAsync(e, t, r, n, i = "") { return Promise.resolve().then(() => (this._babylonScene = e, this._loadData(t), this._loadAsync(r, i, null, () => { }))); } _loadAsync(e, t, r, n) { return Promise.resolve().then(() => { this._rootUrl = e, this._uniqueRootUrl = !e.startsWith("file:") && t ? e : `${e}${Date.now()}/`, this._fileName = t, this._allMaterialsDirtyRequired = !1, this._loadExtensions(), this._checkExtensions(); const i = `${cc[cc.LOADING]} => ${cc[cc.READY]}`, s = `${cc[cc.LOADING]} => ${cc[cc.COMPLETE]}`; this._parent._startPerformanceCounter(i), this._parent._startPerformanceCounter(s), this._parent._setState(cc.LOADING), this._extensionsOnLoading(); const a = new Array(), f = this._babylonScene.blockMaterialDirtyMechanism; if (this._babylonScene.blockMaterialDirtyMechanism = !0, !this.parent.loadOnlyMaterials) { if (r) a.push(this.loadSceneAsync("/nodes", { nodes: r, index: -1 })); else if (this._gltf.scene != null || this._gltf.scenes && this._gltf.scenes[0]) { const d = ei.Get("/scene", this._gltf.scenes, this._gltf.scene || 0); a.push(this.loadSceneAsync(`/scenes/${d.index}`, d)); } } if (!this.parent.skipMaterials && this.parent.loadAllMaterials && this._gltf.materials) for (let d = 0; d < this._gltf.materials.length; ++d) { const v = this._gltf.materials[d], u = "/materials/" + d, l = gt.TriangleFillMode; a.push(this._loadMaterialAsync(u, v, null, l, () => { })); } return this._allMaterialsDirtyRequired ? this._babylonScene.blockMaterialDirtyMechanism = f : this._babylonScene._forceBlockMaterialDirtyMechanism(f), this._parent.compileMaterials && a.push(this._compileMaterialsAsync()), this._parent.compileShadowGenerators && a.push(this._compileShadowGeneratorsAsync()), Promise.all(a).then(() => (this._rootBabylonMesh && this._rootBabylonMesh.setEnabled(!0), this._extensionsOnReady(), this._parent._setState(cc.READY), this._startAnimations(), n())).then((d) => (this._parent._endPerformanceCounter(i), ye.SetImmediate(() => { this._disposed || Promise.all(this._completePromises).then(() => { this._parent._endPerformanceCounter(s), this._parent._setState(cc.COMPLETE), this._parent.onCompleteObservable.notifyObservers(void 0), this._parent.onCompleteObservable.clear(), this.dispose(); }, (v) => { this._parent.onErrorObservable.notifyObservers(v), this._parent.onErrorObservable.clear(), this.dispose(); }); }), d)); }).catch((i) => { throw this._disposed || (this._parent.onErrorObservable.notifyObservers(i), this._parent.onErrorObservable.clear(), this.dispose()), i; }); } _loadData(e) { if (this._gltf = e.json, this._setupData(), e.bin) { const t = this._gltf.buffers; if (t && t[0] && !t[0].uri) { const r = t[0]; (r.byteLength < e.bin.byteLength - 3 || r.byteLength > e.bin.byteLength) && Se.Warn(`Binary buffer length (${r.byteLength}) from JSON does not match chunk length (${e.bin.byteLength})`), this._bin = e.bin; } else Se.Warn("Unexpected BIN chunk"); } } _setupData() { if (ei.Assign(this._gltf.accessors), ei.Assign(this._gltf.animations), ei.Assign(this._gltf.buffers), ei.Assign(this._gltf.bufferViews), ei.Assign(this._gltf.cameras), ei.Assign(this._gltf.images), ei.Assign(this._gltf.materials), ei.Assign(this._gltf.meshes), ei.Assign(this._gltf.nodes), ei.Assign(this._gltf.samplers), ei.Assign(this._gltf.scenes), ei.Assign(this._gltf.skins), ei.Assign(this._gltf.textures), this._gltf.nodes) { const e = {}; for (const r of this._gltf.nodes) if (r.children) for (const n of r.children) e[n] = r.index; const t = this._createRootNode(); for (const r of this._gltf.nodes) { const n = e[r.index]; r.parent = n === void 0 ? t : this._gltf.nodes[n]; } } } _loadExtensions() { for (const e in Jr._RegisteredExtensions) { const t = Jr._RegisteredExtensions[e].factory(this); t.name !== e && Se.Warn(`The name of the glTF loader extension instance does not match the registered name: ${t.name} !== ${e}`), this._extensions.push(t), this._parent.onExtensionLoadedObservable.notifyObservers(t); } this._extensions.sort((e, t) => (e.order || Number.MAX_VALUE) - (t.order || Number.MAX_VALUE)), this._parent.onExtensionLoadedObservable.clear(); } _checkExtensions() { if (this._gltf.extensionsRequired) { for (const e of this._gltf.extensionsRequired) if (!this._extensions.some((r) => r.name === e && r.enabled)) throw new Error(`Required extension ${e} is not available`); } } _createRootNode() { this._babylonScene._blockEntityCollection = !!this._assetContainer, this._rootBabylonMesh = new Ee("__root__", this._babylonScene), this._rootBabylonMesh._parentContainer = this._assetContainer, this._babylonScene._blockEntityCollection = !1, this._rootBabylonMesh.setEnabled(!1); const e = { _babylonTransformNode: this._rootBabylonMesh, index: -1 }; switch (this._parent.coordinateSystemMode) { case AR.AUTO: { this._babylonScene.useRightHandedSystem || (e.rotation = [0, 1, 0, 0], e.scale = [1, 1, -1], Jr._LoadTransform(e, this._rootBabylonMesh)); break; } case AR.FORCE_RIGHT_HANDED: { this._babylonScene.useRightHandedSystem = !0; break; } default: throw new Error(`Invalid coordinate system mode (${this._parent.coordinateSystemMode})`); } return this._parent.onMeshLoadedObservable.notifyObservers(this._rootBabylonMesh), e; } /** * Loads a glTF scene. * @param context The context when loading the asset * @param scene The glTF scene property * @returns A promise that resolves when the load is complete */ loadSceneAsync(e, t) { const r = this._extensionsLoadSceneAsync(e, t); if (r) return r; const n = new Array(); if (this.logOpen(`${e} ${t.name || ""}`), t.nodes) for (const i of t.nodes) { const s = ei.Get(`${e}/nodes/${i}`, this._gltf.nodes, i); n.push(this.loadNodeAsync(`/nodes/${s.index}`, s, (a) => { a.parent = this._rootBabylonMesh; })); } for (const i of this._postSceneLoadActions) i(); return n.push(this._loadAnimationsAsync()), this.logClose(), Promise.all(n).then(() => { }); } _forEachPrimitive(e, t) { if (e._primitiveBabylonMeshes) for (const r of e._primitiveBabylonMeshes) t(r); } _getGeometries() { const e = [], t = this._gltf.nodes; if (t) for (const r of t) this._forEachPrimitive(r, (n) => { const i = n.geometry; i && e.indexOf(i) === -1 && e.push(i); }); return e; } _getMeshes() { const e = []; this._rootBabylonMesh && e.push(this._rootBabylonMesh); const t = this._gltf.nodes; if (t) for (const r of t) this._forEachPrimitive(r, (n) => { e.push(n); }); return e; } _getTransformNodes() { const e = [], t = this._gltf.nodes; if (t) for (const r of t) r._babylonTransformNode && r._babylonTransformNode.getClassName() === "TransformNode" && e.push(r._babylonTransformNode), r._babylonTransformNodeForSkin && e.push(r._babylonTransformNodeForSkin); return e; } _getSkeletons() { const e = [], t = this._gltf.skins; if (t) for (const r of t) r._data && e.push(r._data.babylonSkeleton); return e; } _getAnimationGroups() { const e = [], t = this._gltf.animations; if (t) for (const r of t) r._babylonAnimationGroup && e.push(r._babylonAnimationGroup); return e; } _startAnimations() { switch (this._parent.animationStartMode) { case qS.NONE: break; case qS.FIRST: { const e = this._getAnimationGroups(); e.length !== 0 && e[0].start(!0); break; } case qS.ALL: { const e = this._getAnimationGroups(); for (const t of e) t.start(!0); break; } default: { Se.Error(`Invalid animation start mode (${this._parent.animationStartMode})`); return; } } } /** * Loads a glTF node. * @param context The context when loading the asset * @param node The glTF node property * @param assign A function called synchronously after parsing the glTF properties * @returns A promise that resolves with the loaded Babylon mesh when the load is complete */ loadNodeAsync(e, t, r = () => { }) { const n = this._extensionsLoadNodeAsync(e, t, r); if (n) return n; if (t._babylonTransformNode) throw new Error(`${e}: Invalid recursive node hierarchy`); const i = new Array(); this.logOpen(`${e} ${t.name || ""}`); const s = (a) => { if (Jr.AddPointerMetadata(a, e), Jr._LoadTransform(t, a), t.camera != null) { const f = ei.Get(`${e}/camera`, this._gltf.cameras, t.camera); i.push(this.loadCameraAsync(`/cameras/${f.index}`, f, (o) => { o.parent = a; })); } if (t.children) for (const f of t.children) { const o = ei.Get(`${e}/children/${f}`, this._gltf.nodes, f); i.push(this.loadNodeAsync(`/nodes/${o.index}`, o, (d) => { d.parent = a; })); } r(a); }; if (t.mesh == null || t.skin != null) { const a = t.name || `node${t.index}`; this._babylonScene._blockEntityCollection = !!this._assetContainer; const f = new Hr(a, this._babylonScene); f._parentContainer = this._assetContainer, this._babylonScene._blockEntityCollection = !1, t.mesh == null ? t._babylonTransformNode = f : t._babylonTransformNodeForSkin = f, s(f); } if (t.mesh != null) if (t.skin == null) { const a = ei.Get(`${e}/mesh`, this._gltf.meshes, t.mesh); i.push(this._loadMeshAsync(`/meshes/${a.index}`, t, a, s)); } else { const a = ei.Get(`${e}/mesh`, this._gltf.meshes, t.mesh); i.push(this._loadMeshAsync(`/meshes/${a.index}`, t, a, (f) => { const o = t._babylonTransformNodeForSkin; f.metadata = z1e(o.metadata, f.metadata || {}); const d = ei.Get(`${e}/skin`, this._gltf.skins, t.skin); i.push(this._loadSkinAsync(`/skins/${d.index}`, t, d, (v) => { this._forEachPrimitive(t, (u) => { u.skeleton = v; }), this._postSceneLoadActions.push(() => { if (d.skeleton != null) { const u = ei.Get(`/skins/${d.index}/skeleton`, this._gltf.nodes, d.skeleton).parent; t.index === u.index ? f.parent = o.parent : f.parent = u._babylonTransformNode; } else f.parent = this._rootBabylonMesh; this._parent.onSkinLoadedObservable.notifyObservers({ node: o, skinnedNode: f }); }); })); })); } return this.logClose(), Promise.all(i).then(() => (this._forEachPrimitive(t, (a) => { a.geometry && a.geometry.useBoundingInfoFromGeometry ? a._updateBoundingInfo() : a.refreshBoundingInfo(!0); }), t._babylonTransformNode)); } _loadMeshAsync(e, t, r, n) { const i = r.primitives; if (!i || !i.length) throw new Error(`${e}: Primitives are missing`); i[0].index == null && ei.Assign(i); const s = new Array(); this.logOpen(`${e} ${r.name || ""}`); const a = t.name || `node${t.index}`; if (i.length === 1) { const f = r.primitives[0]; s.push(this._loadMeshPrimitiveAsync(`${e}/primitives/${f.index}`, a, t, r, f, (o) => { t._babylonTransformNode = o, t._primitiveBabylonMeshes = [o]; })); } else { this._babylonScene._blockEntityCollection = !!this._assetContainer, t._babylonTransformNode = new Hr(a, this._babylonScene), t._babylonTransformNode._parentContainer = this._assetContainer, this._babylonScene._blockEntityCollection = !1, t._primitiveBabylonMeshes = []; for (const f of i) s.push(this._loadMeshPrimitiveAsync(`${e}/primitives/${f.index}`, `${a}_primitive${f.index}`, t, r, f, (o) => { o.parent = t._babylonTransformNode, t._primitiveBabylonMeshes.push(o); })); } return n(t._babylonTransformNode), this.logClose(), Promise.all(s).then(() => t._babylonTransformNode); } /** * @internal Define this method to modify the default behavior when loading data for mesh primitives. * @param context The context when loading the asset * @param name The mesh name when loading the asset * @param node The glTF node when loading the asset * @param mesh The glTF mesh when loading the asset * @param primitive The glTF mesh primitive property * @param assign A function called synchronously after parsing the glTF properties * @returns A promise that resolves with the loaded mesh when the load is complete or null if not handled */ _loadMeshPrimitiveAsync(e, t, r, n, i, s) { const a = this._extensionsLoadMeshPrimitiveAsync(e, t, r, n, i, s); if (a) return a; this.logOpen(`${e}`); const f = this._disableInstancedMesh === 0 && this._parent.createInstances && r.skin == null && !n.primitives[0].targets; let o, d; if (f && i._instanceData) this._babylonScene._blockEntityCollection = !!this._assetContainer, o = i._instanceData.babylonSourceMesh.createInstance(t), o._parentContainer = this._assetContainer, this._babylonScene._blockEntityCollection = !1, d = i._instanceData.promise; else { const v = new Array(); this._babylonScene._blockEntityCollection = !!this._assetContainer; const u = new Ee(t, this._babylonScene); u._parentContainer = this._assetContainer, this._babylonScene._blockEntityCollection = !1, u.overrideMaterialSideOrientation = this._babylonScene.useRightHandedSystem ? gt.CounterClockWiseSideOrientation : gt.ClockWiseSideOrientation, this._createMorphTargets(e, r, n, i, u), v.push(this._loadVertexDataAsync(e, i, u).then((P) => this._loadMorphTargetsAsync(e, i, u, P).then(() => { this._disposed || (this._babylonScene._blockEntityCollection = !!this._assetContainer, P.applyToMesh(u), P._parentContainer = this._assetContainer, this._babylonScene._blockEntityCollection = !1); }))); const l = Jr._GetDrawMode(e, i.mode); if (i.material == null) { let P = this._defaultBabylonMaterialData[l]; P || (P = this._createDefaultMaterial("__GLTFLoader._default", l), this._parent.onMaterialLoadedObservable.notifyObservers(P), this._defaultBabylonMaterialData[l] = P), u.material = P; } else if (!this.parent.skipMaterials) { const P = ei.Get(`${e}/material`, this._gltf.materials, i.material); v.push(this._loadMaterialAsync(`/materials/${P.index}`, P, u, l, (p) => { u.material = p; })); } d = Promise.all(v), f && (i._instanceData = { babylonSourceMesh: u, promise: d }), o = u; } return Jr.AddPointerMetadata(o, e), this._parent.onMeshLoadedObservable.notifyObservers(o), s(o), this.logClose(), d.then(() => o); } _loadVertexDataAsync(e, t, r) { const n = this._extensionsLoadVertexDataAsync(e, t, r); if (n) return n; const i = t.attributes; if (!i) throw new Error(`${e}: Attributes are missing`); const s = new Array(), a = new Tf(r.name, this._babylonScene); if (t.indices == null) r.isUnIndexed = !0; else { const o = ei.Get(`${e}/indices`, this._gltf.accessors, t.indices); s.push(this._loadIndicesAccessorAsync(`/accessors/${o.index}`, o).then((d) => { a.setIndices(d); })); } const f = (o, d, v) => { if (i[o] == null) return; r._delayInfo = r._delayInfo || [], r._delayInfo.indexOf(d) === -1 && r._delayInfo.push(d); const u = ei.Get(`${e}/attributes/${o}`, this._gltf.accessors, i[o]); s.push(this._loadVertexAccessorAsync(`/accessors/${u.index}`, u, d).then((l) => { if (l.getKind() === J.PositionKind && !this.parent.alwaysComputeBoundingBox && !r.skeleton && u.min && u.max) { const P = ue.Vector3[0].copyFromFloats(...u.min), p = ue.Vector3[1].copyFromFloats(...u.max); if (u.normalized && u.componentType !== 5126) { let c = 1; switch (u.componentType) { case 5120: c = 127; break; case 5121: c = 255; break; case 5122: c = 32767; break; case 5123: c = 65535; break; } const H = 1 / c; P.scaleInPlace(H), p.scaleInPlace(H); } a._boundingInfo = new Md(P, p), a.useBoundingInfoFromGeometry = !0; } a.setVerticesBuffer(l, u.count); })), d == J.MatricesIndicesExtraKind && (r.numBoneInfluencers = 8), v && v(u); }; return f("POSITION", J.PositionKind), f("NORMAL", J.NormalKind), f("TANGENT", J.TangentKind), f("TEXCOORD_0", J.UVKind), f("TEXCOORD_1", J.UV2Kind), f("TEXCOORD_2", J.UV3Kind), f("TEXCOORD_3", J.UV4Kind), f("TEXCOORD_4", J.UV5Kind), f("TEXCOORD_5", J.UV6Kind), f("JOINTS_0", J.MatricesIndicesKind), f("WEIGHTS_0", J.MatricesWeightsKind), f("JOINTS_1", J.MatricesIndicesExtraKind), f("WEIGHTS_1", J.MatricesWeightsExtraKind), f("COLOR_0", J.ColorKind, (o) => { o.type === "VEC4" && (r.hasVertexAlpha = !0); }), Promise.all(s).then(() => a); } _createMorphTargets(e, t, r, n, i) { if (!n.targets) return; if (t._numMorphTargets == null) t._numMorphTargets = n.targets.length; else if (n.targets.length !== t._numMorphTargets) throw new Error(`${e}: Primitives do not have the same number of targets`); const s = r.extras ? r.extras.targetNames : null; this._babylonScene._blockEntityCollection = !!this._assetContainer, i.morphTargetManager = new U0(this._babylonScene), i.morphTargetManager._parentContainer = this._assetContainer, this._babylonScene._blockEntityCollection = !1, i.morphTargetManager.areUpdatesFrozen = !0; for (let a = 0; a < n.targets.length; a++) { const f = t.weights ? t.weights[a] : r.weights ? r.weights[a] : 0, o = s ? s[a] : `morphTarget${a}`; i.morphTargetManager.addTarget(new lD(o, f, i.getScene())); } } _loadMorphTargetsAsync(e, t, r, n) { if (!t.targets) return Promise.resolve(); const i = new Array(), s = r.morphTargetManager; for (let a = 0; a < s.numTargets; a++) { const f = s.getTarget(a); i.push(this._loadMorphTargetVertexDataAsync(`${e}/targets/${a}`, n, t.targets[a], f)); } return Promise.all(i).then(() => { s.areUpdatesFrozen = !1; }); } _loadMorphTargetVertexDataAsync(e, t, r, n) { const i = new Array(), s = (a, f, o) => { if (r[a] == null) return; const d = t.getVertexBuffer(f); if (!d) return; const v = ei.Get(`${e}/${a}`, this._gltf.accessors, r[a]); i.push(this._loadFloatAccessorAsync(`/accessors/${v.index}`, v).then((u) => { o(d, u); })); }; return s("POSITION", J.PositionKind, (a, f) => { const o = new Float32Array(f.length); a.forEach(f.length, (d, v) => { o[v] = f[v] + d; }), n.setPositions(o); }), s("NORMAL", J.NormalKind, (a, f) => { const o = new Float32Array(f.length); a.forEach(o.length, (d, v) => { o[v] = f[v] + d; }), n.setNormals(o); }), s("TANGENT", J.TangentKind, (a, f) => { const o = new Float32Array(f.length / 3 * 4); let d = 0; a.forEach(f.length / 3 * 4, (v, u) => { (u + 1) % 4 !== 0 && (o[d] = f[d] + v, d++); }), n.setTangents(o); }), Promise.all(i).then(() => { }); } static _LoadTransform(e, t) { if (e.skin != null) return; let r = S.Zero(), n = Ze.Identity(), i = S.One(); e.matrix ? he.FromArray(e.matrix).decompose(i, n, r) : (e.translation && (r = S.FromArray(e.translation)), e.rotation && (n = Ze.FromArray(e.rotation)), e.scale && (i = S.FromArray(e.scale))), t.position = r, t.rotationQuaternion = n, t.scaling = i; } _loadSkinAsync(e, t, r, n) { const i = this._extensionsLoadSkinAsync(e, t, r); if (i) return i; if (r._data) return n(r._data.babylonSkeleton), r._data.promise; const s = `skeleton${r.index}`; this._babylonScene._blockEntityCollection = !!this._assetContainer; const a = new r4(r.name || s, s, this._babylonScene); a._parentContainer = this._assetContainer, this._babylonScene._blockEntityCollection = !1, this._loadBones(e, r, a); const f = this._loadSkinInverseBindMatricesDataAsync(e, r).then((o) => { this._updateBoneMatrices(a, o); }); return r._data = { babylonSkeleton: a, promise: f }, n(a), f; } _loadBones(e, t, r) { if (t.skeleton == null || this._parent.alwaysComputeSkeletonRootNode) { const i = this._findSkeletonRootNode(`${e}/joints`, t.joints); if (i) if (t.skeleton === void 0) t.skeleton = i.index; else { const s = (f, o) => { for (; o.parent; o = o.parent) if (o.parent === f) return !0; return !1; }, a = ei.Get(`${e}/skeleton`, this._gltf.nodes, t.skeleton); a !== i && !s(a, i) && (Se.Warn(`${e}/skeleton: Overriding with nearest common ancestor as skeleton node is not a common root`), t.skeleton = i.index); } else Se.Warn(`${e}: Failed to find common root`); } const n = {}; for (const i of t.joints) { const s = ei.Get(`${e}/joints/${i}`, this._gltf.nodes, i); this._loadBone(s, t, r, n); } } _findSkeletonRootNode(e, t) { if (t.length === 0) return null; const r = {}; for (const i of t) { const s = []; let a = ei.Get(`${e}/${i}`, this._gltf.nodes, i); for (; a.index !== -1; ) s.unshift(a), a = a.parent; r[i] = s; } let n = null; for (let i = 0; ; ++i) { let s = r[t[0]]; if (i >= s.length) return n; const a = s[i]; for (let f = 1; f < t.length; ++f) if (s = r[t[f]], i >= s.length || a !== s[i]) return n; n = a; } } _loadBone(e, t, r, n) { let i = n[e.index]; if (i) return i; let s = null; e.index !== t.skeleton && (e.parent && e.parent.index !== -1 ? s = this._loadBone(e.parent, t, r, n) : t.skeleton !== void 0 && Se.Warn(`/skins/${t.index}/skeleton: Skeleton node is not a common root`)); const a = t.joints.indexOf(e.index); return i = new da(e.name || `joint${e.index}`, r, s, this._getNodeMatrix(e), null, null, a), n[e.index] = i, this._postSceneLoadActions.push(() => { i.linkTransformNode(e._babylonTransformNode); }), i; } _loadSkinInverseBindMatricesDataAsync(e, t) { if (t.inverseBindMatrices == null) return Promise.resolve(null); const r = ei.Get(`${e}/inverseBindMatrices`, this._gltf.accessors, t.inverseBindMatrices); return this._loadFloatAccessorAsync(`/accessors/${r.index}`, r); } _updateBoneMatrices(e, t) { for (const r of e.bones) { const n = he.Identity(), i = r._index; t && i !== -1 && (he.FromArrayToRef(t, i * 16, n), n.invertToRef(n)); const s = r.getParent(); s && n.multiplyToRef(s.getAbsoluteInverseBindMatrix(), n), r.updateMatrix(n, !1, !1), r._updateAbsoluteBindMatrices(void 0, !1); } } _getNodeMatrix(e) { return e.matrix ? he.FromArray(e.matrix) : he.Compose(e.scale ? S.FromArray(e.scale) : S.One(), e.rotation ? Ze.FromArray(e.rotation) : Ze.Identity(), e.translation ? S.FromArray(e.translation) : S.Zero()); } /** * Loads a glTF camera. * @param context The context when loading the asset * @param camera The glTF camera property * @param assign A function called synchronously after parsing the glTF properties * @returns A promise that resolves with the loaded Babylon camera when the load is complete */ loadCameraAsync(e, t, r = () => { }) { const n = this._extensionsLoadCameraAsync(e, t, r); if (n) return n; const i = new Array(); this.logOpen(`${e} ${t.name || ""}`), this._babylonScene._blockEntityCollection = !!this._assetContainer; const s = new SA(t.name || `camera${t.index}`, S.Zero(), this._babylonScene, !1); switch (s._parentContainer = this._assetContainer, this._babylonScene._blockEntityCollection = !1, s.ignoreParentScaling = !0, t._babylonCamera = s, s.rotation.set(0, Math.PI, 0), t.type) { case "perspective": { const a = t.perspective; if (!a) throw new Error(`${e}: Camera perspective properties are missing`); s.fov = a.yfov, s.minZ = a.znear, s.maxZ = a.zfar || 0; break; } case "orthographic": { if (!t.orthographic) throw new Error(`${e}: Camera orthographic properties are missing`); s.mode = Tr.ORTHOGRAPHIC_CAMERA, s.orthoLeft = -t.orthographic.xmag, s.orthoRight = t.orthographic.xmag, s.orthoBottom = -t.orthographic.ymag, s.orthoTop = t.orthographic.ymag, s.minZ = t.orthographic.znear, s.maxZ = t.orthographic.zfar; break; } default: throw new Error(`${e}: Invalid camera type (${t.type})`); } return Jr.AddPointerMetadata(s, e), this._parent.onCameraLoadedObservable.notifyObservers(s), r(s), this.logClose(), Promise.all(i).then(() => s); } _loadAnimationsAsync() { const e = this._gltf.animations; if (!e) return Promise.resolve(); const t = new Array(); for (let r = 0; r < e.length; r++) { const n = e[r]; t.push(this.loadAnimationAsync(`/animations/${n.index}`, n).then((i) => { i.targetedAnimations.length === 0 && i.dispose(); })); } return Promise.all(t).then(() => { }); } /** * Loads a glTF animation. * @param context The context when loading the asset * @param animation The glTF animation property * @returns A promise that resolves with the loaded Babylon animation group when the load is complete */ loadAnimationAsync(e, t) { const r = this._extensionsLoadAnimationAsync(e, t); if (r) return r; this._babylonScene._blockEntityCollection = !!this._assetContainer; const n = new w0(t.name || `animation${t.index}`, this._babylonScene); n._parentContainer = this._assetContainer, this._babylonScene._blockEntityCollection = !1, t._babylonAnimationGroup = n; const i = new Array(); ei.Assign(t.channels), ei.Assign(t.samplers); for (const s of t.channels) i.push(this._loadAnimationChannelAsync(`${e}/channels/${s.index}`, e, t, s, (a, f) => { a.animations = a.animations || [], a.animations.push(f), n.addTargetedAnimation(f, a); })); return Promise.all(i).then(() => (n.normalize(0), n)); } /** * @hidden * Loads a glTF animation channel. * @param context The context when loading the asset * @param animationContext The context of the animation when loading the asset * @param animation The glTF animation property * @param channel The glTF animation channel property * @param onLoad Called for each animation loaded * @returns A void promise that resolves when the load is complete */ _loadAnimationChannelAsync(e, t, r, n, i) { const s = this._extensionsLoadAnimationChannelAsync(e, t, r, n, i); if (s) return s; if (n.target.node == null) return Promise.resolve(); const a = ei.Get(`${e}/target/node`, this._gltf.nodes, n.target.node); if (n.target.path === "weights" && !a._numMorphTargets || n.target.path !== "weights" && !a._babylonTransformNode) return Promise.resolve(); let f; switch (n.target.path) { case "translation": { f = XI.translation; break; } case "rotation": { f = XI.rotation; break; } case "scale": { f = XI.scale; break; } case "weights": { f = XI.weights; break; } default: throw new Error(`${e}/target/path: Invalid value (${n.target.path})`); } const o = { target: a, properties: f }; return this._loadAnimationChannelFromTargetInfoAsync(e, t, r, n, o, i); } /** * @hidden * Loads a glTF animation channel. * @param context The context when loading the asset * @param animationContext The context of the animation when loading the asset * @param animation The glTF animation property * @param channel The glTF animation channel property * @param targetInfo The glTF target and properties * @param onLoad Called for each animation loaded * @returns A void promise that resolves when the load is complete */ _loadAnimationChannelFromTargetInfoAsync(e, t, r, n, i, s) { const a = this.parent.targetFps, f = 1 / a, o = ei.Get(`${e}/sampler`, r.samplers, n.sampler); return this._loadAnimationSamplerAsync(`${t}/samplers/${n.sampler}`, o).then((d) => { let v = 0; for (const u of i.properties) { const l = u.getStride(i.target), P = d.input, p = d.output, c = new Array(P.length); let H = 0; switch (d.interpolation) { case "STEP": { for (let T = 0; T < P.length; T++) { const q = u.getValue(i.target, p, H, 1); H += l, c[T] = { frame: P[T] * a, value: q, interpolation: CI.STEP }; } break; } case "CUBICSPLINE": { for (let T = 0; T < P.length; T++) { const q = u.getValue(i.target, p, H, f); H += l; const b = u.getValue(i.target, p, H, 1); H += l; const j = u.getValue(i.target, p, H, f); H += l, c[T] = { frame: P[T] * a, inTangent: q, value: b, outTangent: j }; } break; } case "LINEAR": { for (let T = 0; T < P.length; T++) { const q = u.getValue(i.target, p, H, 1); H += l, c[T] = { frame: P[T] * a, value: q }; } break; } } if (H > 0) { const T = `${r.name || `animation${r.index}`}_channel${n.index}_${v}`; u.buildAnimations(i.target, T, a, c, (q, b) => { ++v, s(q, b); }); } } }); } _loadAnimationSamplerAsync(e, t) { if (t._data) return t._data; const r = t.interpolation || "LINEAR"; switch (r) { case "STEP": case "LINEAR": case "CUBICSPLINE": break; default: throw new Error(`${e}/interpolation: Invalid value (${t.interpolation})`); } const n = ei.Get(`${e}/input`, this._gltf.accessors, t.input), i = ei.Get(`${e}/output`, this._gltf.accessors, t.output); return t._data = Promise.all([ this._loadFloatAccessorAsync(`/accessors/${n.index}`, n), this._loadFloatAccessorAsync(`/accessors/${i.index}`, i) ]).then(([s, a]) => ({ input: s, interpolation: r, output: a })), t._data; } /** * Loads a glTF buffer. * @param context The context when loading the asset * @param buffer The glTF buffer property * @param byteOffset The byte offset to use * @param byteLength The byte length to use * @returns A promise that resolves with the loaded data when the load is complete */ loadBufferAsync(e, t, r, n) { const i = this._extensionsLoadBufferAsync(e, t, r, n); if (i) return i; if (!t._data) if (t.uri) t._data = this.loadUriAsync(`${e}/uri`, t, t.uri); else { if (!this._bin) throw new Error(`${e}: Uri is missing or the binary glTF is missing its binary chunk`); t._data = this._bin.readAsync(0, t.byteLength); } return t._data.then((s) => { try { return new Uint8Array(s.buffer, s.byteOffset + r, n); } catch (a) { throw new Error(`${e}: ${a.message}`); } }); } /** * Loads a glTF buffer view. * @param context The context when loading the asset * @param bufferView The glTF buffer view property * @returns A promise that resolves with the loaded data when the load is complete */ loadBufferViewAsync(e, t) { const r = this._extensionsLoadBufferViewAsync(e, t); if (r) return r; if (t._data) return t._data; const n = ei.Get(`${e}/buffer`, this._gltf.buffers, t.buffer); return t._data = this.loadBufferAsync(`/buffers/${n.index}`, n, t.byteOffset || 0, t.byteLength), t._data; } _loadAccessorAsync(e, t, r) { if (t._data) return t._data; const n = Jr._GetNumComponents(e, t.type), i = n * J.GetTypeByteLength(t.componentType), s = n * t.count; if (t.bufferView == null) t._data = Promise.resolve(new r(s)); else { const a = ei.Get(`${e}/bufferView`, this._gltf.bufferViews, t.bufferView); t._data = this.loadBufferViewAsync(`/bufferViews/${a.index}`, a).then((f) => { if (t.componentType === 5126 && !t.normalized && (!a.byteStride || a.byteStride === i)) return Jr._GetTypedArray(e, t.componentType, f, t.byteOffset, s); { const o = new r(s); return J.ForEach(f, t.byteOffset || 0, a.byteStride || i, n, t.componentType, o.length, t.normalized || !1, (d, v) => { o[v] = d; }), o; } }); } if (t.sparse) { const a = t.sparse; t._data = t._data.then((f) => { const o = f, d = ei.Get(`${e}/sparse/indices/bufferView`, this._gltf.bufferViews, a.indices.bufferView), v = ei.Get(`${e}/sparse/values/bufferView`, this._gltf.bufferViews, a.values.bufferView); return Promise.all([ this.loadBufferViewAsync(`/bufferViews/${d.index}`, d), this.loadBufferViewAsync(`/bufferViews/${v.index}`, v) ]).then(([u, l]) => { const P = Jr._GetTypedArray(`${e}/sparse/indices`, a.indices.componentType, u, a.indices.byteOffset, a.count), p = n * a.count; let c; if (t.componentType === 5126 && !t.normalized) c = Jr._GetTypedArray(`${e}/sparse/values`, t.componentType, l, a.values.byteOffset, p); else { const T = Jr._GetTypedArray(`${e}/sparse/values`, t.componentType, l, a.values.byteOffset, p); c = new r(p), J.ForEach(T, 0, i, n, t.componentType, c.length, t.normalized || !1, (q, b) => { c[b] = q; }); } let H = 0; for (let T = 0; T < P.length; T++) { let q = P[T] * n; for (let b = 0; b < n; b++) o[q++] = c[H++]; } return o; }); }); } return t._data; } /** * @internal */ _loadFloatAccessorAsync(e, t) { return this._loadAccessorAsync(e, t, Float32Array); } /** * @internal */ _loadIndicesAccessorAsync(e, t) { if (t.type !== "SCALAR") throw new Error(`${e}/type: Invalid value ${t.type}`); if (t.componentType !== 5121 && t.componentType !== 5123 && t.componentType !== 5125) throw new Error(`${e}/componentType: Invalid value ${t.componentType}`); if (t._data) return t._data; if (t.sparse) { const r = Jr._GetTypedArrayConstructor(`${e}/componentType`, t.componentType); t._data = this._loadAccessorAsync(e, t, r); } else { const r = ei.Get(`${e}/bufferView`, this._gltf.bufferViews, t.bufferView); t._data = this.loadBufferViewAsync(`/bufferViews/${r.index}`, r).then((n) => Jr._GetTypedArray(e, t.componentType, n, t.byteOffset, t.count)); } return t._data; } /** * @internal */ _loadVertexBufferViewAsync(e) { if (e._babylonBuffer) return e._babylonBuffer; const t = this._babylonScene.getEngine(); return e._babylonBuffer = this.loadBufferViewAsync(`/bufferViews/${e.index}`, e).then((r) => new P9(t, r, !1)), e._babylonBuffer; } /** * @internal */ _loadVertexAccessorAsync(e, t, r) { var n; if (!((n = t._babylonVertexBuffer) === null || n === void 0) && n[r]) return t._babylonVertexBuffer[r]; t._babylonVertexBuffer || (t._babylonVertexBuffer = {}); const i = this._babylonScene.getEngine(); if (t.sparse || t.bufferView == null) t._babylonVertexBuffer[r] = this._loadFloatAccessorAsync(e, t).then((s) => new J(i, s, r, !1)); else { const s = ei.Get(`${e}/bufferView`, this._gltf.bufferViews, t.bufferView); t._babylonVertexBuffer[r] = this._loadVertexBufferViewAsync(s).then((a) => { const f = Jr._GetNumComponents(e, t.type); return new J(i, a, r, !1, void 0, s.byteStride, void 0, t.byteOffset, f, t.componentType, t.normalized, !0, void 0, !0); }); } return t._babylonVertexBuffer[r]; } _loadMaterialMetallicRoughnessPropertiesAsync(e, t, r) { if (!(r instanceof mr)) throw new Error(`${e}: Material type not supported`); const n = new Array(); return t && (t.baseColorFactor ? (r.albedoColor = Ne.FromArray(t.baseColorFactor), r.alpha = t.baseColorFactor[3]) : r.albedoColor = Ne.White(), r.metallic = t.metallicFactor == null ? 1 : t.metallicFactor, r.roughness = t.roughnessFactor == null ? 1 : t.roughnessFactor, t.baseColorTexture && n.push(this.loadTextureInfoAsync(`${e}/baseColorTexture`, t.baseColorTexture, (i) => { i.name = `${r.name} (Base Color)`, r.albedoTexture = i; })), t.metallicRoughnessTexture && (t.metallicRoughnessTexture.nonColorData = !0, n.push(this.loadTextureInfoAsync(`${e}/metallicRoughnessTexture`, t.metallicRoughnessTexture, (i) => { i.name = `${r.name} (Metallic Roughness)`, r.metallicTexture = i; })), r.useMetallnessFromMetallicTextureBlue = !0, r.useRoughnessFromMetallicTextureGreen = !0, r.useRoughnessFromMetallicTextureAlpha = !1)), Promise.all(n).then(() => { }); } /** * @internal */ _loadMaterialAsync(e, t, r, n, i = () => { }) { const s = this._extensionsLoadMaterialAsync(e, t, r, n, i); if (s) return s; t._data = t._data || {}; let a = t._data[n]; if (!a) { this.logOpen(`${e} ${t.name || ""}`); const f = this.createMaterial(e, t, n); a = { babylonMaterial: f, babylonMeshes: [], promise: this.loadMaterialPropertiesAsync(e, t, f) }, t._data[n] = a, Jr.AddPointerMetadata(f, e), this._parent.onMaterialLoadedObservable.notifyObservers(f), this.logClose(); } return r && (a.babylonMeshes.push(r), r.onDisposeObservable.addOnce(() => { const f = a.babylonMeshes.indexOf(r); f !== -1 && a.babylonMeshes.splice(f, 1); })), i(a.babylonMaterial), a.promise.then(() => a.babylonMaterial); } _createDefaultMaterial(e, t) { this._babylonScene._blockEntityCollection = !!this._assetContainer; const r = new mr(e, this._babylonScene); return r._parentContainer = this._assetContainer, this._babylonScene._blockEntityCollection = !1, r.fillMode = t, r.enableSpecularAntiAliasing = !0, r.useRadianceOverAlpha = !this._parent.transparencyAsCoverage, r.useSpecularOverAlpha = !this._parent.transparencyAsCoverage, r.transparencyMode = mr.PBRMATERIAL_OPAQUE, r.metallic = 1, r.roughness = 1, r; } /** * Creates a Babylon material from a glTF material. * @param context The context when loading the asset * @param material The glTF material property * @param babylonDrawMode The draw mode for the Babylon material * @returns The Babylon material */ createMaterial(e, t, r) { const n = this._extensionsCreateMaterial(e, t, r); if (n) return n; const i = t.name || `material${t.index}`; return this._createDefaultMaterial(i, r); } /** * Loads properties from a glTF material into a Babylon material. * @param context The context when loading the asset * @param material The glTF material property * @param babylonMaterial The Babylon material * @returns A promise that resolves when the load is complete */ loadMaterialPropertiesAsync(e, t, r) { const n = this._extensionsLoadMaterialPropertiesAsync(e, t, r); if (n) return n; const i = new Array(); return i.push(this.loadMaterialBasePropertiesAsync(e, t, r)), t.pbrMetallicRoughness && i.push(this._loadMaterialMetallicRoughnessPropertiesAsync(`${e}/pbrMetallicRoughness`, t.pbrMetallicRoughness, r)), this.loadMaterialAlphaProperties(e, t, r), Promise.all(i).then(() => { }); } /** * Loads the normal, occlusion, and emissive properties from a glTF material into a Babylon material. * @param context The context when loading the asset * @param material The glTF material property * @param babylonMaterial The Babylon material * @returns A promise that resolves when the load is complete */ loadMaterialBasePropertiesAsync(e, t, r) { if (!(r instanceof mr)) throw new Error(`${e}: Material type not supported`); const n = new Array(); return r.emissiveColor = t.emissiveFactor ? Ne.FromArray(t.emissiveFactor) : new Ne(0, 0, 0), t.doubleSided && (r.backFaceCulling = !1, r.twoSidedLighting = !0), t.normalTexture && (t.normalTexture.nonColorData = !0, n.push(this.loadTextureInfoAsync(`${e}/normalTexture`, t.normalTexture, (i) => { i.name = `${r.name} (Normal)`, r.bumpTexture = i; })), r.invertNormalMapX = !this._babylonScene.useRightHandedSystem, r.invertNormalMapY = this._babylonScene.useRightHandedSystem, t.normalTexture.scale != null && r.bumpTexture && (r.bumpTexture.level = t.normalTexture.scale), r.forceIrradianceInFragment = !0), t.occlusionTexture && (t.occlusionTexture.nonColorData = !0, n.push(this.loadTextureInfoAsync(`${e}/occlusionTexture`, t.occlusionTexture, (i) => { i.name = `${r.name} (Occlusion)`, r.ambientTexture = i; })), r.useAmbientInGrayScale = !0, t.occlusionTexture.strength != null && (r.ambientTextureStrength = t.occlusionTexture.strength)), t.emissiveTexture && n.push(this.loadTextureInfoAsync(`${e}/emissiveTexture`, t.emissiveTexture, (i) => { i.name = `${r.name} (Emissive)`, r.emissiveTexture = i; })), Promise.all(n).then(() => { }); } /** * Loads the alpha properties from a glTF material into a Babylon material. * Must be called after the setting the albedo texture of the Babylon material when the material has an albedo texture. * @param context The context when loading the asset * @param material The glTF material property * @param babylonMaterial The Babylon material */ loadMaterialAlphaProperties(e, t, r) { if (!(r instanceof mr)) throw new Error(`${e}: Material type not supported`); switch (t.alphaMode || "OPAQUE") { case "OPAQUE": { r.transparencyMode = mr.PBRMATERIAL_OPAQUE; break; } case "MASK": { r.transparencyMode = mr.PBRMATERIAL_ALPHATEST, r.alphaCutOff = t.alphaCutoff == null ? 0.5 : t.alphaCutoff, r.albedoTexture && (r.albedoTexture.hasAlpha = !0); break; } case "BLEND": { r.transparencyMode = mr.PBRMATERIAL_ALPHABLEND, r.albedoTexture && (r.albedoTexture.hasAlpha = !0, r.useAlphaFromAlbedoTexture = !0); break; } default: throw new Error(`${e}/alphaMode: Invalid value (${t.alphaMode})`); } } /** * Loads a glTF texture info. * @param context The context when loading the asset * @param textureInfo The glTF texture info property * @param assign A function called synchronously after parsing the glTF properties * @returns A promise that resolves with the loaded Babylon texture when the load is complete */ loadTextureInfoAsync(e, t, r = () => { }) { const n = this._extensionsLoadTextureInfoAsync(e, t, r); if (n) return n; if (this.logOpen(`${e}`), t.texCoord >= 6) throw new Error(`${e}/texCoord: Invalid value (${t.texCoord})`); const i = ei.Get(`${e}/index`, this._gltf.textures, t.index); i._textureInfo = t; const s = this._loadTextureAsync(`/textures/${t.index}`, i, (a) => { a.coordinatesIndex = t.texCoord || 0, Jr.AddPointerMetadata(a, e), this._parent.onTextureLoadedObservable.notifyObservers(a), r(a); }); return this.logClose(), s; } /** * @internal */ _loadTextureAsync(e, t, r = () => { }) { const n = this._extensionsLoadTextureAsync(e, t, r); if (n) return n; this.logOpen(`${e} ${t.name || ""}`); const i = t.sampler == null ? Jr.DefaultSampler : ei.Get(`${e}/sampler`, this._gltf.samplers, t.sampler), s = ei.Get(`${e}/source`, this._gltf.images, t.source), a = this._createTextureAsync(e, i, s, r, void 0, !t._textureInfo.nonColorData); return this.logClose(), a; } /** * @internal */ _createTextureAsync(e, t, r, n = () => { }, i, s) { const a = this._loadSampler(`/samplers/${t.index}`, t), f = new Array(), o = new MW(); this._babylonScene._blockEntityCollection = !!this._assetContainer; const d = { noMipmap: a.noMipMaps, invertY: !1, samplingMode: a.samplingMode, onLoad: () => { this._disposed || o.resolve(); }, onError: (u, l) => { this._disposed || o.reject(new Error(`${e}: ${l && l.message ? l.message : u || "Failed to load texture"}`)); }, mimeType: r.mimeType, loaderOptions: i, useSRGBBuffer: !!s && this._parent.useSRGBBuffers }, v = new We(null, this._babylonScene, d); return v._parentContainer = this._assetContainer, this._babylonScene._blockEntityCollection = !1, f.push(o.promise), f.push(this.loadImageAsync(`/images/${r.index}`, r).then((u) => { const l = r.uri || `${this._fileName}#image${r.index}`, P = `data:${this._uniqueRootUrl}${l}`; v.updateURL(P, u); })), v.wrapU = a.wrapU, v.wrapV = a.wrapV, n(v), Promise.all(f).then(() => v); } _loadSampler(e, t) { return t._data || (t._data = { noMipMaps: t.minFilter === 9728 || t.minFilter === 9729, samplingMode: Jr._GetTextureSamplingMode(e, t), wrapU: Jr._GetTextureWrapMode(`${e}/wrapS`, t.wrapS), wrapV: Jr._GetTextureWrapMode(`${e}/wrapT`, t.wrapT) }), t._data; } /** * Loads a glTF image. * @param context The context when loading the asset * @param image The glTF image property * @returns A promise that resolves with the loaded data when the load is complete */ loadImageAsync(e, t) { if (!t._data) { if (this.logOpen(`${e} ${t.name || ""}`), t.uri) t._data = this.loadUriAsync(`${e}/uri`, t, t.uri); else { const r = ei.Get(`${e}/bufferView`, this._gltf.bufferViews, t.bufferView); t._data = this.loadBufferViewAsync(`/bufferViews/${r.index}`, r); } this.logClose(); } return t._data; } /** * Loads a glTF uri. * @param context The context when loading the asset * @param property The glTF property associated with the uri * @param uri The base64 or relative uri * @returns A promise that resolves with the loaded data when the load is complete */ loadUriAsync(e, t, r) { const n = this._extensionsLoadUriAsync(e, t, r); if (n) return n; if (!Jr._ValidateUri(r)) throw new Error(`${e}: '${r}' is invalid`); if (qR(r)) { const i = new Uint8Array(iU(r)); return this.log(`${e}: Decoded ${r.substr(0, 64)}... (${i.length} bytes)`), Promise.resolve(i); } return this.log(`${e}: Loading ${r}`), this._parent.preprocessUrlAsync(this._rootUrl + r).then((i) => new Promise((s, a) => { this._parent._loadFile(this._babylonScene, i, (f) => { this._disposed || (this.log(`${e}: Loaded ${r} (${f.byteLength} bytes)`), s(new Uint8Array(f))); }, !0, (f) => { a(new mS(`${e}: Failed to load '${r}'${f ? ": " + f.status + " " + f.statusText : ""}`, f)); }); })); } /** * Adds a JSON pointer to the _internalMetadata of the Babylon object at `._internalMetadata.gltf.pointers`. * @param babylonObject the Babylon object with _internalMetadata * @param pointer the JSON pointer */ static AddPointerMetadata(e, t) { e.metadata = e.metadata || {}; const r = e._internalMetadata = e._internalMetadata || {}, n = r.gltf = r.gltf || {}; (n.pointers = n.pointers || []).push(t); } static _GetTextureWrapMode(e, t) { switch (t = t ?? 10497, t) { case 33071: return We.CLAMP_ADDRESSMODE; case 33648: return We.MIRROR_ADDRESSMODE; case 10497: return We.WRAP_ADDRESSMODE; default: return Se.Warn(`${e}: Invalid value (${t})`), We.WRAP_ADDRESSMODE; } } static _GetTextureSamplingMode(e, t) { const r = t.magFilter == null ? 9729 : t.magFilter, n = t.minFilter == null ? 9987 : t.minFilter; if (r === 9729) switch (n) { case 9728: return We.LINEAR_NEAREST; case 9729: return We.LINEAR_LINEAR; case 9984: return We.LINEAR_NEAREST_MIPNEAREST; case 9985: return We.LINEAR_LINEAR_MIPNEAREST; case 9986: return We.LINEAR_NEAREST_MIPLINEAR; case 9987: return We.LINEAR_LINEAR_MIPLINEAR; default: return Se.Warn(`${e}/minFilter: Invalid value (${n})`), We.LINEAR_LINEAR_MIPLINEAR; } else switch (r !== 9728 && Se.Warn(`${e}/magFilter: Invalid value (${r})`), n) { case 9728: return We.NEAREST_NEAREST; case 9729: return We.NEAREST_LINEAR; case 9984: return We.NEAREST_NEAREST_MIPNEAREST; case 9985: return We.NEAREST_LINEAR_MIPNEAREST; case 9986: return We.NEAREST_NEAREST_MIPLINEAR; case 9987: return We.NEAREST_LINEAR_MIPLINEAR; default: return Se.Warn(`${e}/minFilter: Invalid value (${n})`), We.NEAREST_NEAREST_MIPNEAREST; } } static _GetTypedArrayConstructor(e, t) { switch (t) { case 5120: return Int8Array; case 5121: return Uint8Array; case 5122: return Int16Array; case 5123: return Uint16Array; case 5125: return Uint32Array; case 5126: return Float32Array; default: throw new Error(`${e}: Invalid component type ${t}`); } } static _GetTypedArray(e, t, r, n, i) { const s = r.buffer; n = r.byteOffset + (n || 0); const a = Jr._GetTypedArrayConstructor(`${e}/componentType`, t), f = J.GetTypeByteLength(t); return n % f !== 0 ? (Se.Warn(`${e}: Copying buffer as byte offset (${n}) is not a multiple of component type byte length (${f})`), new a(s.slice(n, n + i * f), 0)) : new a(s, n, i); } static _GetNumComponents(e, t) { switch (t) { case "SCALAR": return 1; case "VEC2": return 2; case "VEC3": return 3; case "VEC4": return 4; case "MAT2": return 4; case "MAT3": return 9; case "MAT4": return 16; } throw new Error(`${e}: Invalid type (${t})`); } static _ValidateUri(e) { return ye.IsBase64(e) || e.indexOf("..") === -1; } /** * @internal */ static _GetDrawMode(e, t) { switch (t == null && (t = 4), t) { case 0: return gt.PointListDrawMode; case 1: return gt.LineListDrawMode; case 2: return gt.LineLoopDrawMode; case 3: return gt.LineStripDrawMode; case 4: return gt.TriangleFillMode; case 5: return gt.TriangleStripDrawMode; case 6: return gt.TriangleFanDrawMode; } throw new Error(`${e}: Invalid mesh primitive mode (${t})`); } _compileMaterialsAsync() { this._parent._startPerformanceCounter("Compile materials"); const e = new Array(); if (this._gltf.materials) { for (const t of this._gltf.materials) if (t._data) for (const r in t._data) { const n = t._data[r]; for (const i of n.babylonMeshes) { i.computeWorldMatrix(!0); const s = n.babylonMaterial; e.push(s.forceCompilationAsync(i)), e.push(s.forceCompilationAsync(i, { useInstances: !0 })), this._parent.useClipPlane && (e.push(s.forceCompilationAsync(i, { clipPlane: !0 })), e.push(s.forceCompilationAsync(i, { clipPlane: !0, useInstances: !0 }))); } } } return Promise.all(e).then(() => { this._parent._endPerformanceCounter("Compile materials"); }); } _compileShadowGeneratorsAsync() { this._parent._startPerformanceCounter("Compile shadow generators"); const e = new Array(), t = this._babylonScene.lights; for (const r of t) { const n = r.getShadowGenerator(); n && e.push(n.forceCompilationAsync()); } return Promise.all(e).then(() => { this._parent._endPerformanceCounter("Compile shadow generators"); }); } _forEachExtensions(e) { for (const t of this._extensions) t.enabled && e(t); } _applyExtensions(e, t, r) { for (const n of this._extensions) if (n.enabled) { const i = `${n.name}.${t}`, s = e; s._activeLoaderExtensionFunctions = s._activeLoaderExtensionFunctions || {}; const a = s._activeLoaderExtensionFunctions; if (!a[i]) { a[i] = !0; try { const f = r(n); if (f) return f; } finally { delete a[i]; } } } return null; } _extensionsOnLoading() { this._forEachExtensions((e) => e.onLoading && e.onLoading()); } _extensionsOnReady() { this._forEachExtensions((e) => e.onReady && e.onReady()); } _extensionsLoadSceneAsync(e, t) { return this._applyExtensions(t, "loadScene", (r) => r.loadSceneAsync && r.loadSceneAsync(e, t)); } _extensionsLoadNodeAsync(e, t, r) { return this._applyExtensions(t, "loadNode", (n) => n.loadNodeAsync && n.loadNodeAsync(e, t, r)); } _extensionsLoadCameraAsync(e, t, r) { return this._applyExtensions(t, "loadCamera", (n) => n.loadCameraAsync && n.loadCameraAsync(e, t, r)); } _extensionsLoadVertexDataAsync(e, t, r) { return this._applyExtensions(t, "loadVertexData", (n) => n._loadVertexDataAsync && n._loadVertexDataAsync(e, t, r)); } _extensionsLoadMeshPrimitiveAsync(e, t, r, n, i, s) { return this._applyExtensions(i, "loadMeshPrimitive", (a) => a._loadMeshPrimitiveAsync && a._loadMeshPrimitiveAsync(e, t, r, n, i, s)); } _extensionsLoadMaterialAsync(e, t, r, n, i) { return this._applyExtensions(t, "loadMaterial", (s) => s._loadMaterialAsync && s._loadMaterialAsync(e, t, r, n, i)); } _extensionsCreateMaterial(e, t, r) { return this._applyExtensions(t, "createMaterial", (n) => n.createMaterial && n.createMaterial(e, t, r)); } _extensionsLoadMaterialPropertiesAsync(e, t, r) { return this._applyExtensions(t, "loadMaterialProperties", (n) => n.loadMaterialPropertiesAsync && n.loadMaterialPropertiesAsync(e, t, r)); } _extensionsLoadTextureInfoAsync(e, t, r) { return this._applyExtensions(t, "loadTextureInfo", (n) => n.loadTextureInfoAsync && n.loadTextureInfoAsync(e, t, r)); } _extensionsLoadTextureAsync(e, t, r) { return this._applyExtensions(t, "loadTexture", (n) => n._loadTextureAsync && n._loadTextureAsync(e, t, r)); } _extensionsLoadAnimationAsync(e, t) { return this._applyExtensions(t, "loadAnimation", (r) => r.loadAnimationAsync && r.loadAnimationAsync(e, t)); } _extensionsLoadAnimationChannelAsync(e, t, r, n, i) { return this._applyExtensions(r, "loadAnimationChannel", (s) => s._loadAnimationChannelAsync && s._loadAnimationChannelAsync(e, t, r, n, i)); } _extensionsLoadSkinAsync(e, t, r) { return this._applyExtensions(r, "loadSkin", (n) => n._loadSkinAsync && n._loadSkinAsync(e, t, r)); } _extensionsLoadUriAsync(e, t, r) { return this._applyExtensions(t, "loadUri", (n) => n._loadUriAsync && n._loadUriAsync(e, t, r)); } _extensionsLoadBufferViewAsync(e, t) { return this._applyExtensions(t, "loadBufferView", (r) => r.loadBufferViewAsync && r.loadBufferViewAsync(e, t)); } _extensionsLoadBufferAsync(e, t, r, n) { return this._applyExtensions(t, "loadBuffer", (i) => i.loadBufferAsync && i.loadBufferAsync(e, t, r, n)); } /** * Helper method called by a loader extension to load an glTF extension. * @param context The context when loading the asset * @param property The glTF property to load the extension from * @param extensionName The name of the extension to load * @param actionAsync The action to run * @returns The promise returned by actionAsync or null if the extension does not exist */ static LoadExtensionAsync(e, t, r, n) { if (!t.extensions) return null; const s = t.extensions[r]; return s ? n(`${e}/extensions/${r}`, s) : null; } /** * Helper method called by a loader extension to load a glTF extra. * @param context The context when loading the asset * @param property The glTF property to load the extra from * @param extensionName The name of the extension to load * @param actionAsync The action to run * @returns The promise returned by actionAsync or null if the extra does not exist */ static LoadExtraAsync(e, t, r, n) { if (!t.extras) return null; const s = t.extras[r]; return s ? n(`${e}/extras/${r}`, s) : null; } /** * Checks for presence of an extension. * @param name The name of the extension to check * @returns A boolean indicating the presence of the given extension name in `extensionsUsed` */ isExtensionUsed(e) { return !!this._gltf.extensionsUsed && this._gltf.extensionsUsed.indexOf(e) !== -1; } /** * Increments the indentation level and logs a message. * @param message The message to log */ logOpen(e) { this._parent._logOpen(e); } /** * Decrements the indentation level. */ logClose() { this._parent._logClose(); } /** * Logs a message * @param message The message to log */ log(e) { this._parent._log(e); } /** * Starts a performance counter. * @param counterName The name of the performance counter */ startPerformanceCounter(e) { this._parent._startPerformanceCounter(e); } /** * Ends a performance counter. * @param counterName The name of the performance counter */ endPerformanceCounter(e) { this._parent._endPerformanceCounter(e); } } Jr._RegisteredExtensions = {}; Jr.DefaultSampler = { index: -1 }; q1._CreateGLTF2Loader = (A) => new Jr(A); const NF = "EXT_lights_image_based"; class qXe { /** * @internal */ constructor(e) { this.name = NF, this._loader = e, this.enabled = this._loader.isExtensionUsed(NF); } /** @internal */ dispose() { this._loader = null, delete this._lights; } /** @internal */ onLoading() { const e = this._loader.gltf.extensions; if (e && e[this.name]) { const t = e[this.name]; this._lights = t.lights; } } /** * @internal */ loadSceneAsync(e, t) { return Jr.LoadExtensionAsync(e, t, this.name, (r, n) => { this._loader._allMaterialsDirtyRequired = !0; const i = new Array(); i.push(this._loader.loadSceneAsync(e, t)), this._loader.logOpen(`${r}`); const s = ei.Get(`${r}/light`, this._lights, n.light); return i.push(this._loadLightAsync(`/extensions/${this.name}/lights/${n.light}`, s).then((a) => { this._loader.babylonScene.environmentTexture = a; })), this._loader.logClose(), Promise.all(i).then(() => { }); }); } _loadLightAsync(e, t) { if (!t._loaded) { const r = new Array(); this._loader.logOpen(`${e}`); const n = new Array(t.specularImages.length); for (let i = 0; i < t.specularImages.length; i++) { const s = t.specularImages[i]; n[i] = new Array(s.length); for (let a = 0; a < s.length; a++) { const f = `${e}/specularImages/${i}/${a}`; this._loader.logOpen(`${f}`); const o = s[a], d = ei.Get(f, this._loader.gltf.images, o); r.push(this._loader.loadImageAsync(`/images/${o}`, d).then((v) => { n[i][a] = v; })), this._loader.logClose(); } } this._loader.logClose(), t._loaded = Promise.all(r).then(() => { const i = new hy(this._loader.babylonScene, null, t.specularImageSize); if (i.name = t.name || "environment", t._babylonTexture = i, t.intensity != null && (i.level = t.intensity), t.rotation) { let o = Ze.FromArray(t.rotation); this._loader.babylonScene.useRightHandedSystem || (o = Ze.Inverse(o)), he.FromQuaternionToRef(o, i.getReflectionTextureMatrix()); } if (!t.irradianceCoefficients) throw new Error(`${e}: Irradiance coefficients are missing`); const s = HD.FromArray(t.irradianceCoefficients); s.scaleInPlace(t.intensity), s.convertIrradianceToLambertianRadiance(); const a = i4.FromHarmonics(s), f = (n.length - 1) / Xt.Log2(t.specularImageSize); return i.updateRGBDAsync(n, a, f); }); } return t._loaded.then(() => t._babylonTexture); } } Jr.RegisterExtension(NF, (A) => new qXe(A)); const QF = "EXT_mesh_gpu_instancing"; class bXe { /** * @internal */ constructor(e) { this.name = QF, this._loader = e, this.enabled = this._loader.isExtensionUsed(QF); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadNodeAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { this._loader._disableInstancedMesh++; const s = this._loader.loadNodeAsync(`/nodes/${t.index}`, t, r); if (this._loader._disableInstancedMesh--, !t._primitiveBabylonMeshes) return s; const a = new Array(); let f = 0; const o = (d) => { if (i.attributes[d] == null) { a.push(Promise.resolve(null)); return; } const v = ei.Get(`${n}/attributes/${d}`, this._loader.gltf.accessors, i.attributes[d]); if (a.push(this._loader._loadFloatAccessorAsync(`/accessors/${v.bufferView}`, v)), f === 0) f = v.count; else if (f !== v.count) throw new Error(`${n}/attributes: Instance buffer accessors do not have the same count.`); }; return o("TRANSLATION"), o("ROTATION"), o("SCALE"), s.then((d) => Promise.all(a).then(([v, u, l]) => { const P = new Float32Array(f * 16); ue.Vector3[0].copyFromFloats(0, 0, 0), ue.Quaternion[0].copyFromFloats(0, 0, 0, 1), ue.Vector3[1].copyFromFloats(1, 1, 1); for (let p = 0; p < f; ++p) v && S.FromArrayToRef(v, p * 3, ue.Vector3[0]), u && Ze.FromArrayToRef(u, p * 4, ue.Quaternion[0]), l && S.FromArrayToRef(l, p * 3, ue.Vector3[1]), he.ComposeToRef(ue.Vector3[1], ue.Quaternion[0], ue.Vector3[0], ue.Matrix[0]), ue.Matrix[0].copyToArray(P, p * 16); for (const p of t._primitiveBabylonMeshes) p.thinInstanceSetBuffer("matrix", P, 16, !0); return d; })); }); } } Jr.RegisterExtension(QF, (A) => new bXe(A)); const YF = "EXT_meshopt_compression"; class xXe { /** * @internal */ constructor(e) { this.name = YF, this.enabled = e.isExtensionUsed(YF), this._loader = e; } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadBufferViewAsync(e, t) { return Jr.LoadExtensionAsync(e, t, this.name, (r, n) => { const i = t; if (i._meshOptData) return i._meshOptData; const s = ei.Get(`${e}/buffer`, this._loader.gltf.buffers, n.buffer); return i._meshOptData = this._loader.loadBufferAsync(`/buffers/${s.index}`, s, n.byteOffset || 0, n.byteLength).then((a) => Q2.Default.decodeGltfBufferAsync(a, n.count, n.byteStride, n.mode, n.filter)), i._meshOptData; }); } } Jr.RegisterExtension(YF, (A) => new xXe(A)); const MF = "EXT_texture_webp"; class DXe { /** * @internal */ constructor(e) { this.name = MF, this._loader = e, this.enabled = e.isExtensionUsed(MF); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ _loadTextureAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { const s = t.sampler == null ? Jr.DefaultSampler : ei.Get(`${e}/sampler`, this._loader.gltf.samplers, t.sampler), a = ei.Get(`${n}/source`, this._loader.gltf.images, i.source); return this._loader._createTextureAsync(e, s, a, (f) => { r(f); }, void 0, !t._textureInfo.nonColorData); }); } } Jr.RegisterExtension(MF, (A) => new DXe(A)); const LF = "KHR_draco_mesh_compression"; class jXe { /** * @internal */ constructor(e) { this.name = LF, this.useNormalizedFlagFromAccessor = !0, this._loader = e, this.enabled = Yd.DecoderAvailable && this._loader.isExtensionUsed(LF); } /** @internal */ dispose() { delete this.dracoCompression, this._loader = null; } /** * @internal */ _loadVertexDataAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { if (t.mode != null) { if (t.mode !== 5 && t.mode !== 4) throw new Error(`${e}: Unsupported mode ${t.mode}`); if (t.mode === 5) throw new Error(`${e}: Mode ${t.mode} is not currently supported`); } const s = {}, a = {}, f = (d, v) => { const u = i.attributes[d]; if (u != null && (r._delayInfo = r._delayInfo || [], r._delayInfo.indexOf(v) === -1 && r._delayInfo.push(v), s[v] = u, this.useNormalizedFlagFromAccessor)) { const l = ei.TryGet(this._loader.gltf.accessors, t.attributes[d]); l && (a[v] = l.normalized || !1); } }; f("POSITION", J.PositionKind), f("NORMAL", J.NormalKind), f("TANGENT", J.TangentKind), f("TEXCOORD_0", J.UVKind), f("TEXCOORD_1", J.UV2Kind), f("TEXCOORD_2", J.UV3Kind), f("TEXCOORD_3", J.UV4Kind), f("TEXCOORD_4", J.UV5Kind), f("TEXCOORD_5", J.UV6Kind), f("JOINTS_0", J.MatricesIndicesKind), f("WEIGHTS_0", J.MatricesWeightsKind), f("COLOR_0", J.ColorKind); const o = ei.Get(n, this._loader.gltf.bufferViews, i.bufferView); return o._dracoBabylonGeometry || (o._dracoBabylonGeometry = this._loader.loadBufferViewAsync(`/bufferViews/${o.index}`, o).then((d) => (this.dracoCompression || Yd.Default)._decodeMeshToGeometryForGltfAsync(r.name, this._loader.babylonScene, d, s, a).catch((u) => { throw new Error(`${e}: ${u.message}`); }))), o._dracoBabylonGeometry; }); } } Jr.RegisterExtension(LF, (A) => new jXe(A)); const KF = "KHR_lights_punctual"; class wXe { /** * @internal */ constructor(e) { this.name = KF, this._loader = e, this.enabled = this._loader.isExtensionUsed(KF); } /** @internal */ dispose() { this._loader = null, delete this._lights; } /** @internal */ onLoading() { const e = this._loader.gltf.extensions; if (e && e[this.name]) { const t = e[this.name]; this._lights = t.lights, ei.Assign(this._lights); } } /** * @internal */ loadNodeAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => (this._loader._allMaterialsDirtyRequired = !0, this._loader.loadNodeAsync(e, t, (s) => { let a; const f = ei.Get(n, this._lights, i.light), o = f.name || s.name; switch (this._loader.babylonScene._blockEntityCollection = !!this._loader._assetContainer, f.type) { case "directional": { const d = new IA(o, S.Backward(), this._loader.babylonScene); d.position.setAll(0), a = d; break; } case "point": { a = new ag(o, S.Zero(), this._loader.babylonScene); break; } case "spot": { const d = new nA(o, S.Zero(), S.Backward(), 0, 1, this._loader.babylonScene); d.angle = (f.spot && f.spot.outerConeAngle || Math.PI / 4) * 2, d.innerAngle = (f.spot && f.spot.innerConeAngle || 0) * 2, a = d; break; } default: throw this._loader.babylonScene._blockEntityCollection = !1, new Error(`${n}: Invalid light type (${f.type})`); } a._parentContainer = this._loader._assetContainer, this._loader.babylonScene._blockEntityCollection = !1, f._babylonLight = a, a.falloffType = ci.FALLOFF_GLTF, a.diffuse = f.color ? Ne.FromArray(f.color) : Ne.White(), a.intensity = f.intensity == null ? 1 : f.intensity, a.range = f.range == null ? Number.MAX_VALUE : f.range, a.parent = s, this._loader._babylonLights.push(a), Jr.AddPointerMetadata(a, n), r(s); }))); } } Jr.RegisterExtension(KF, (A) => new wXe(A)); const JF = "KHR_materials_pbrSpecularGlossiness"; class mXe { /** * @internal */ constructor(e) { this.name = JF, this.order = 200, this._loader = e, this.enabled = this._loader.isExtensionUsed(JF); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadMaterialPropertiesAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { const s = new Array(); return s.push(this._loader.loadMaterialBasePropertiesAsync(e, t, r)), s.push(this._loadSpecularGlossinessPropertiesAsync(n, t, i, r)), this._loader.loadMaterialAlphaProperties(e, t, r), Promise.all(s).then(() => { }); }); } _loadSpecularGlossinessPropertiesAsync(e, t, r, n) { if (!(n instanceof mr)) throw new Error(`${e}: Material type not supported`); const i = new Array(); return n.metallic = null, n.roughness = null, r.diffuseFactor ? (n.albedoColor = Ne.FromArray(r.diffuseFactor), n.alpha = r.diffuseFactor[3]) : n.albedoColor = Ne.White(), n.reflectivityColor = r.specularFactor ? Ne.FromArray(r.specularFactor) : Ne.White(), n.microSurface = r.glossinessFactor == null ? 1 : r.glossinessFactor, r.diffuseTexture && i.push(this._loader.loadTextureInfoAsync(`${e}/diffuseTexture`, r.diffuseTexture, (s) => { s.name = `${n.name} (Diffuse)`, n.albedoTexture = s; })), r.specularGlossinessTexture && (i.push(this._loader.loadTextureInfoAsync(`${e}/specularGlossinessTexture`, r.specularGlossinessTexture, (s) => { s.name = `${n.name} (Specular Glossiness)`, n.reflectivityTexture = s, n.reflectivityTexture.hasAlpha = !0; })), n.useMicroSurfaceFromReflectivityMapAlpha = !0), Promise.all(i).then(() => { }); } } Jr.RegisterExtension(JF, (A) => new mXe(A)); const zF = "KHR_materials_unlit"; class BXe { /** * @internal */ constructor(e) { this.name = zF, this.order = 210, this._loader = e, this.enabled = this._loader.isExtensionUsed(zF); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadMaterialPropertiesAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, () => this._loadUnlitPropertiesAsync(e, t, r)); } _loadUnlitPropertiesAsync(e, t, r) { if (!(r instanceof mr)) throw new Error(`${e}: Material type not supported`); const n = new Array(); r.unlit = !0; const i = t.pbrMetallicRoughness; return i && (i.baseColorFactor ? (r.albedoColor = Ne.FromArray(i.baseColorFactor), r.alpha = i.baseColorFactor[3]) : r.albedoColor = Ne.White(), i.baseColorTexture && n.push(this._loader.loadTextureInfoAsync(`${e}/baseColorTexture`, i.baseColorTexture, (s) => { s.name = `${r.name} (Base Color)`, r.albedoTexture = s; }))), t.doubleSided && (r.backFaceCulling = !1, r.twoSidedLighting = !0), this._loader.loadMaterialAlphaProperties(e, t, r), Promise.all(n).then(() => { }); } } Jr.RegisterExtension(zF, (A) => new BXe(A)); const GF = "KHR_materials_clearcoat"; class WXe { /** * @internal */ constructor(e) { this.name = GF, this.order = 190, this._loader = e, this.enabled = this._loader.isExtensionUsed(GF); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadMaterialPropertiesAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { const s = new Array(); return s.push(this._loader.loadMaterialPropertiesAsync(e, t, r)), s.push(this._loadClearCoatPropertiesAsync(n, i, r)), Promise.all(s).then(() => { }); }); } _loadClearCoatPropertiesAsync(e, t, r) { if (!(r instanceof mr)) throw new Error(`${e}: Material type not supported`); const n = new Array(); return r.clearCoat.isEnabled = !0, r.clearCoat.useRoughnessFromMainTexture = !1, r.clearCoat.remapF0OnInterfaceChange = !1, t.clearcoatFactor != null ? r.clearCoat.intensity = t.clearcoatFactor : r.clearCoat.intensity = 0, t.clearcoatTexture && n.push(this._loader.loadTextureInfoAsync(`${e}/clearcoatTexture`, t.clearcoatTexture, (i) => { i.name = `${r.name} (ClearCoat Intensity)`, r.clearCoat.texture = i; })), t.clearcoatRoughnessFactor != null ? r.clearCoat.roughness = t.clearcoatRoughnessFactor : r.clearCoat.roughness = 0, t.clearcoatRoughnessTexture && (t.clearcoatRoughnessTexture.nonColorData = !0, n.push(this._loader.loadTextureInfoAsync(`${e}/clearcoatRoughnessTexture`, t.clearcoatRoughnessTexture, (i) => { i.name = `${r.name} (ClearCoat Roughness)`, r.clearCoat.textureRoughness = i; }))), t.clearcoatNormalTexture && (t.clearcoatNormalTexture.nonColorData = !0, n.push(this._loader.loadTextureInfoAsync(`${e}/clearcoatNormalTexture`, t.clearcoatNormalTexture, (i) => { i.name = `${r.name} (ClearCoat Normal)`, r.clearCoat.bumpTexture = i; })), r.invertNormalMapX = !r.getScene().useRightHandedSystem, r.invertNormalMapY = r.getScene().useRightHandedSystem, t.clearcoatNormalTexture.scale != null && (r.clearCoat.bumpTexture.level = t.clearcoatNormalTexture.scale)), Promise.all(n).then(() => { }); } } Jr.RegisterExtension(GF, (A) => new WXe(A)); const ZF = "KHR_materials_iridescence"; class SXe { /** * @internal */ constructor(e) { this.name = ZF, this.order = 195, this._loader = e, this.enabled = this._loader.isExtensionUsed(ZF); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadMaterialPropertiesAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { const s = new Array(); return s.push(this._loader.loadMaterialPropertiesAsync(e, t, r)), s.push(this._loadIridescencePropertiesAsync(n, i, r)), Promise.all(s).then(() => { }); }); } _loadIridescencePropertiesAsync(e, t, r) { var n, i, s, a, f; if (!(r instanceof mr)) throw new Error(`${e}: Material type not supported`); const o = new Array(); return r.iridescence.isEnabled = !0, r.iridescence.intensity = (n = t.iridescenceFactor) !== null && n !== void 0 ? n : 0, r.iridescence.indexOfRefraction = (s = (i = t.iridescenceIor) !== null && i !== void 0 ? i : t.iridescenceIOR) !== null && s !== void 0 ? s : 1.3, r.iridescence.minimumThickness = (a = t.iridescenceThicknessMinimum) !== null && a !== void 0 ? a : 100, r.iridescence.maximumThickness = (f = t.iridescenceThicknessMaximum) !== null && f !== void 0 ? f : 400, t.iridescenceTexture && o.push(this._loader.loadTextureInfoAsync(`${e}/iridescenceTexture`, t.iridescenceTexture, (d) => { d.name = `${r.name} (Iridescence Intensity)`, r.iridescence.texture = d; })), t.iridescenceThicknessTexture && o.push(this._loader.loadTextureInfoAsync(`${e}/iridescenceThicknessTexture`, t.iridescenceThicknessTexture, (d) => { d.name = `${r.name} (Iridescence Thickness)`, r.iridescence.thicknessTexture = d; })), Promise.all(o).then(() => { }); } } Jr.RegisterExtension(ZF, (A) => new SXe(A)); const _F = "KHR_materials_anisotropy"; class UXe { /** * @internal */ constructor(e) { this.name = _F, this.order = 195, this._loader = e, this.enabled = this._loader.isExtensionUsed(_F); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadMaterialPropertiesAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { const s = new Array(); return s.push(this._loader.loadMaterialPropertiesAsync(e, t, r)), s.push(this._loadIridescencePropertiesAsync(n, i, r)), Promise.all(s).then(() => { }); }); } _loadIridescencePropertiesAsync(e, t, r) { var n, i; if (!(r instanceof mr)) throw new Error(`${e}: Material type not supported`); const s = new Array(); return r.anisotropy.isEnabled = !0, r.anisotropy.intensity = (n = t.anisotropyStrength) !== null && n !== void 0 ? n : 0, r.anisotropy.angle = (i = t.anisotropyRotation) !== null && i !== void 0 ? i : 0, t.anisotropyTexture && s.push(this._loader.loadTextureInfoAsync(`${e}/anisotropyTexture`, t.anisotropyTexture, (a) => { a.name = `${r.name} (Anisotropy Intensity)`, r.anisotropy.texture = a; })), Promise.all(s).then(() => { }); } } Jr.RegisterExtension(_F, (A) => new UXe(A)); const $F = "KHR_materials_emissive_strength"; class IXe { /** * @internal */ constructor(e) { this.name = $F, this.order = 170, this._loader = e, this.enabled = this._loader.isExtensionUsed($F); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadMaterialPropertiesAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => this._loader.loadMaterialPropertiesAsync(e, t, r).then(() => { this._loadEmissiveProperties(n, i, r); })); } _loadEmissiveProperties(e, t, r) { if (!(r instanceof mr)) throw new Error(`${e}: Material type not supported`); t.emissiveStrength !== void 0 && r.emissiveColor.scaleToRef(t.emissiveStrength, r.emissiveColor); } } Jr.RegisterExtension($F, (A) => new IXe(A)); const eN = "KHR_materials_sheen"; class RXe { /** * @internal */ constructor(e) { this.name = eN, this.order = 190, this._loader = e, this.enabled = this._loader.isExtensionUsed(eN); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadMaterialPropertiesAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { const s = new Array(); return s.push(this._loader.loadMaterialPropertiesAsync(e, t, r)), s.push(this._loadSheenPropertiesAsync(n, i, r)), Promise.all(s).then(() => { }); }); } _loadSheenPropertiesAsync(e, t, r) { if (!(r instanceof mr)) throw new Error(`${e}: Material type not supported`); const n = new Array(); return r.sheen.isEnabled = !0, r.sheen.intensity = 1, t.sheenColorFactor != null ? r.sheen.color = Ne.FromArray(t.sheenColorFactor) : r.sheen.color = Ne.Black(), t.sheenColorTexture && n.push(this._loader.loadTextureInfoAsync(`${e}/sheenColorTexture`, t.sheenColorTexture, (i) => { i.name = `${r.name} (Sheen Color)`, r.sheen.texture = i; })), t.sheenRoughnessFactor !== void 0 ? r.sheen.roughness = t.sheenRoughnessFactor : r.sheen.roughness = 0, t.sheenRoughnessTexture && (t.sheenRoughnessTexture.nonColorData = !0, n.push(this._loader.loadTextureInfoAsync(`${e}/sheenRoughnessTexture`, t.sheenRoughnessTexture, (i) => { i.name = `${r.name} (Sheen Roughness)`, r.sheen.textureRoughness = i; }))), r.sheen.albedoScaling = !0, r.sheen.useRoughnessFromMainTexture = !1, Promise.all(n).then(() => { }); } } Jr.RegisterExtension(eN, (A) => new RXe(A)); const tN = "KHR_materials_specular"; class VXe { /** * @internal */ constructor(e) { this.name = tN, this.order = 190, this._loader = e, this.enabled = this._loader.isExtensionUsed(tN); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadMaterialPropertiesAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { const s = new Array(); return s.push(this._loader.loadMaterialPropertiesAsync(e, t, r)), s.push(this._loadSpecularPropertiesAsync(n, i, r)), Promise.all(s).then(() => { }); }); } _loadSpecularPropertiesAsync(e, t, r) { if (!(r instanceof mr)) throw new Error(`${e}: Material type not supported`); const n = new Array(); return t.specularFactor !== void 0 && (r.metallicF0Factor = t.specularFactor), t.specularColorFactor !== void 0 && (r.metallicReflectanceColor = Ne.FromArray(t.specularColorFactor)), t.specularTexture && (t.specularTexture.nonColorData = !0, n.push(this._loader.loadTextureInfoAsync(`${e}/specularTexture`, t.specularTexture, (i) => { i.name = `${r.name} (Specular F0 Strength)`, r.metallicReflectanceTexture = i, r.useOnlyMetallicFromMetallicReflectanceTexture = !0; }))), t.specularColorTexture && n.push(this._loader.loadTextureInfoAsync(`${e}/specularColorTexture`, t.specularColorTexture, (i) => { i.name = `${r.name} (Specular F0 Color)`, r.reflectanceTexture = i; })), Promise.all(n).then(() => { }); } } Jr.RegisterExtension(tN, (A) => new VXe(A)); const rN = "KHR_materials_ior"; class Oy { /** * @internal */ constructor(e) { this.name = rN, this.order = 180, this._loader = e, this.enabled = this._loader.isExtensionUsed(rN); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadMaterialPropertiesAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { const s = new Array(); return s.push(this._loader.loadMaterialPropertiesAsync(e, t, r)), s.push(this._loadIorPropertiesAsync(n, i, r)), Promise.all(s).then(() => { }); }); } _loadIorPropertiesAsync(e, t, r) { if (!(r instanceof mr)) throw new Error(`${e}: Material type not supported`); return t.ior !== void 0 ? r.indexOfRefraction = t.ior : r.indexOfRefraction = Oy._DEFAULT_IOR, Promise.resolve(); } } Oy._DEFAULT_IOR = 1.5; Jr.RegisterExtension(rN, (A) => new Oy(A)); const yl = "KHR_materials_variants"; class Qx { /** * @internal */ constructor(e) { this.name = yl, this._loader = e, this.enabled = this._loader.isExtensionUsed(yl); } /** @internal */ dispose() { this._loader = null; } /** * Gets the list of available variant names for this asset. * @param rootMesh The glTF root mesh * @returns the list of all the variant names for this model */ static GetAvailableVariants(e) { const t = this._GetExtensionMetadata(e); return t ? Object.keys(t.variants) : []; } /** * Gets the list of available variant names for this asset. * @param rootMesh The glTF root mesh * @returns the list of all the variant names for this model */ getAvailableVariants(e) { return Qx.GetAvailableVariants(e); } /** * Select a variant given a variant name or a list of variant names. * @param rootMesh The glTF root mesh * @param variantName The variant name(s) to select. */ static SelectVariant(e, t) { const r = this._GetExtensionMetadata(e); if (!r) throw new Error(`Cannot select variant on a glTF mesh that does not have the ${yl} extension`); const n = (i) => { const s = r.variants[i]; if (s) for (const a of s) a.mesh.material = a.material; }; if (t instanceof Array) for (const i of t) n(i); else n(t); r.lastSelected = t; } /** * Select a variant given a variant name or a list of variant names. * @param rootMesh The glTF root mesh * @param variantName The variant name(s) to select. */ selectVariant(e, t) { return Qx.SelectVariant(e, t); } /** * Reset back to the original before selecting a variant. * @param rootMesh The glTF root mesh */ static Reset(e) { const t = this._GetExtensionMetadata(e); if (!t) throw new Error(`Cannot reset on a glTF mesh that does not have the ${yl} extension`); for (const r of t.original) r.mesh.material = r.material; t.lastSelected = null; } /** * Reset back to the original before selecting a variant. * @param rootMesh The glTF root mesh */ reset(e) { return Qx.Reset(e); } /** * Gets the last selected variant name(s) or null if original. * @param rootMesh The glTF root mesh * @returns The selected variant name(s). */ static GetLastSelectedVariant(e) { const t = this._GetExtensionMetadata(e); if (!t) throw new Error(`Cannot get the last selected variant on a glTF mesh that does not have the ${yl} extension`); return t.lastSelected; } /** * Gets the last selected variant name(s) or null if original. * @param rootMesh The glTF root mesh * @returns The selected variant name(s). */ getLastSelectedVariant(e) { return Qx.GetLastSelectedVariant(e); } static _GetExtensionMetadata(e) { var t, r; return ((r = (t = e == null ? void 0 : e._internalMetadata) === null || t === void 0 ? void 0 : t.gltf) === null || r === void 0 ? void 0 : r[yl]) || null; } /** @internal */ onLoading() { const e = this._loader.gltf.extensions; if (e && e[this.name]) { const t = e[this.name]; this._variants = t.variants; } } /** * @internal */ _loadMeshPrimitiveAsync(e, t, r, n, i, s) { return Jr.LoadExtensionAsync(e, i, this.name, (a, f) => { const o = new Array(); return o.push(this._loader._loadMeshPrimitiveAsync(e, t, r, n, i, (d) => { if (s(d), d instanceof Ee) { const v = Jr._GetDrawMode(e, i.mode), u = this._loader.rootBabylonMesh, l = u ? u._internalMetadata = u._internalMetadata || {} : {}, P = l.gltf = l.gltf || {}, p = P[yl] = P[yl] || { lastSelected: null, original: [], variants: {} }; p.original.push({ mesh: d, material: d.material }); for (let c = 0; c < f.mappings.length; ++c) { const H = f.mappings[c], T = ei.Get(`${a}/mappings/${c}/material`, this._loader.gltf.materials, H.material); o.push(this._loader._loadMaterialAsync(`#/materials/${H.material}`, T, d, v, (q) => { for (let b = 0; b < H.variants.length; ++b) { const j = H.variants[b], w = ei.Get(`/extensions/${yl}/variants/${j}`, this._variants, j); p.variants[w.name] = p.variants[w.name] || [], p.variants[w.name].push({ mesh: d, material: q }), d.onClonedObservable.add((m) => { const I = m; let N = null, k = I; do { if (k = k.parent, !k) return; N = Qx._GetExtensionMetadata(k); } while (N === null); if (u && N === Qx._GetExtensionMetadata(u)) { k._internalMetadata = {}; for (const R in u._internalMetadata) k._internalMetadata[R] = u._internalMetadata[R]; k._internalMetadata.gltf = []; for (const R in u._internalMetadata.gltf) k._internalMetadata.gltf[R] = u._internalMetadata.gltf[R]; k._internalMetadata.gltf[yl] = { lastSelected: null, original: [], variants: {} }; for (const R of N.original) k._internalMetadata.gltf[yl].original.push({ mesh: R.mesh, material: R.material }); for (const R in N.variants) if (Object.prototype.hasOwnProperty.call(N.variants, R)) { k._internalMetadata.gltf[yl].variants[R] = []; for (const y of N.variants[R]) k._internalMetadata.gltf[yl].variants[R].push({ mesh: y.mesh, material: y.material }); } N = k._internalMetadata.gltf[yl]; } for (const R of N.original) R.mesh === d && (R.mesh = I); for (const R of N.variants[w.name]) R.mesh === d && (R.mesh = I); }); } })); } } })), Promise.all(o).then(([d]) => d); }); } } Jr.RegisterExtension(yl, (A) => new Qx(A)); class nM { /** * Creates the default options for the helper. */ static _GetDefaultOptions() { return { renderSize: 1024, samples: 4, lodGenerationScale: 1, lodGenerationOffset: -4, renderTargetTextureType: et.TEXTURETYPE_HALF_FLOAT, generateMipmaps: !0 }; } /** * constructor * @param options Defines the options we want to customize the helper * @param scene The scene to add the material to */ constructor(e, t) { this._opaqueRenderTarget = null, this._opaqueMeshesCache = [], this._transparentMeshesCache = [], this._materialObservers = {}, this._options = { ...nM._GetDefaultOptions(), ...e }, this._scene = t, this._scene._transmissionHelper = this, this.onErrorObservable = new Oe(), this._scene.onDisposeObservable.addOnce(() => { this.dispose(); }), this._parseScene(), this._setupRenderTargets(); } /** * Updates the background according to the new options * @param options */ updateOptions(e) { if (!Object.keys(e).filter((i) => this._options[i] !== e[i]).length) return; const r = { ...this._options, ...e }, n = this._options; this._options = r, r.renderSize !== n.renderSize || r.renderTargetTextureType !== n.renderTargetTextureType || r.generateMipmaps !== n.generateMipmaps || !this._opaqueRenderTarget ? this._setupRenderTargets() : (this._opaqueRenderTarget.samples = r.samples, this._opaqueRenderTarget.lodGenerationScale = r.lodGenerationScale, this._opaqueRenderTarget.lodGenerationOffset = r.lodGenerationOffset); } /** * Gets the opaque render target texture or null if not available. */ getOpaqueTarget() { return this._opaqueRenderTarget; } _shouldRenderAsTransmission(e) { return e ? !!(e instanceof mr && e.subSurface.isRefractionEnabled) : !1; } _addMesh(e) { this._materialObservers[e.uniqueId] = e.onMaterialChangedObservable.add(this._onMeshMaterialChanged.bind(this)), ye.SetImmediate(() => { this._shouldRenderAsTransmission(e.material) ? (e.material.refractionTexture = this._opaqueRenderTarget, this._transparentMeshesCache.indexOf(e) === -1 && this._transparentMeshesCache.push(e)) : this._opaqueMeshesCache.indexOf(e) === -1 && this._opaqueMeshesCache.push(e); }); } _removeMesh(e) { e.onMaterialChangedObservable.remove(this._materialObservers[e.uniqueId]), delete this._materialObservers[e.uniqueId]; let t = this._transparentMeshesCache.indexOf(e); t !== -1 && this._transparentMeshesCache.splice(t, 1), t = this._opaqueMeshesCache.indexOf(e), t !== -1 && this._opaqueMeshesCache.splice(t, 1); } _parseScene() { this._scene.meshes.forEach(this._addMesh.bind(this)), this._scene.onNewMeshAddedObservable.add(this._addMesh.bind(this)), this._scene.onMeshRemovedObservable.add(this._removeMesh.bind(this)); } // When one of the meshes in the scene has its material changed, make sure that it's in the correct cache list. _onMeshMaterialChanged(e) { const t = this._transparentMeshesCache.indexOf(e), r = this._opaqueMeshesCache.indexOf(e); this._shouldRenderAsTransmission(e.material) ? (e.material instanceof mr && (e.material.subSurface.refractionTexture = this._opaqueRenderTarget), r !== -1 ? (this._opaqueMeshesCache.splice(r, 1), this._transparentMeshesCache.push(e)) : t === -1 && this._transparentMeshesCache.push(e)) : t !== -1 ? (this._transparentMeshesCache.splice(t, 1), this._opaqueMeshesCache.push(e)) : r === -1 && this._opaqueMeshesCache.push(e); } /** * @internal * Check if the opaque render target has not been disposed and can still be used. * @returns */ _isRenderTargetValid() { var e; return ((e = this._opaqueRenderTarget) === null || e === void 0 ? void 0 : e.getInternalTexture()) !== null; } /** * @internal * Setup the render targets according to the specified options. */ _setupRenderTargets() { var e, t; this._opaqueRenderTarget && this._opaqueRenderTarget.dispose(), this._opaqueRenderTarget = new Ta("opaqueSceneTexture", this._options.renderSize, this._scene, this._options.generateMipmaps, void 0, this._options.renderTargetTextureType), this._opaqueRenderTarget.ignoreCameraViewport = !0, this._opaqueRenderTarget.renderList = this._opaqueMeshesCache, this._opaqueRenderTarget.clearColor = (t = (e = this._options.clearColor) === null || e === void 0 ? void 0 : e.clone()) !== null && t !== void 0 ? t : this._scene.clearColor.clone(), this._opaqueRenderTarget.gammaSpace = !1, this._opaqueRenderTarget.lodGenerationScale = this._options.lodGenerationScale, this._opaqueRenderTarget.lodGenerationOffset = this._options.lodGenerationOffset, this._opaqueRenderTarget.samples = this._options.samples, this._opaqueRenderTarget.renderSprites = !0, this._opaqueRenderTarget.renderParticles = !0; let r, n; this._opaqueRenderTarget.onBeforeBindObservable.add((i) => { n = this._scene.environmentIntensity, this._scene.environmentIntensity = 1, r = this._scene.imageProcessingConfiguration.applyByPostProcess, this._options.clearColor ? i.clearColor.copyFrom(this._options.clearColor) : this._scene.clearColor.toLinearSpaceToRef(i.clearColor, this._scene.getEngine().useExactSrgbConversions), this._scene.imageProcessingConfiguration._applyByPostProcess = !0; }), this._opaqueRenderTarget.onAfterUnbindObservable.add(() => { this._scene.environmentIntensity = n, this._scene.imageProcessingConfiguration._applyByPostProcess = r; }), this._transparentMeshesCache.forEach((i) => { this._shouldRenderAsTransmission(i.material) && (i.material.refractionTexture = this._opaqueRenderTarget); }); } /** * Dispose all the elements created by the Helper. */ dispose() { this._scene._transmissionHelper = void 0, this._opaqueRenderTarget && (this._opaqueRenderTarget.dispose(), this._opaqueRenderTarget = null), this._transparentMeshesCache = [], this._opaqueMeshesCache = []; } } const nN = "KHR_materials_transmission"; class CXe { /** * @internal */ constructor(e) { this.name = nN, this.order = 175, this._loader = e, this.enabled = this._loader.isExtensionUsed(nN), this.enabled && (e.parent.transparencyAsCoverage = !0); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadMaterialPropertiesAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { const s = new Array(); return s.push(this._loader.loadMaterialBasePropertiesAsync(e, t, r)), s.push(this._loader.loadMaterialPropertiesAsync(e, t, r)), s.push(this._loadTransparentPropertiesAsync(n, t, r, i)), Promise.all(s).then(() => { }); }); } _loadTransparentPropertiesAsync(e, t, r, n) { var i, s; if (!(r instanceof mr)) throw new Error(`${e}: Material type not supported`); const a = r; if (a.subSurface.isRefractionEnabled = !0, a.subSurface.volumeIndexOfRefraction = 1, a.subSurface.useAlbedoToTintRefraction = !0, n.transmissionFactor !== void 0) { a.subSurface.refractionIntensity = n.transmissionFactor; const f = a.getScene(); a.subSurface.refractionIntensity && !f._transmissionHelper ? new nM({}, a.getScene()) : a.subSurface.refractionIntensity && !(!((i = f._transmissionHelper) === null || i === void 0) && i._isRenderTargetValid()) && ((s = f._transmissionHelper) === null || s === void 0 || s._setupRenderTargets()); } else return a.subSurface.refractionIntensity = 0, a.subSurface.isRefractionEnabled = !1, Promise.resolve(); return a.subSurface.minimumThickness = 0, a.subSurface.maximumThickness = 0, n.transmissionTexture ? (n.transmissionTexture.nonColorData = !0, this._loader.loadTextureInfoAsync(`${e}/transmissionTexture`, n.transmissionTexture, void 0).then((f) => { a.subSurface.refractionIntensityTexture = f, a.subSurface.useGltfStyleTextures = !0; })) : Promise.resolve(); } } Jr.RegisterExtension(nN, (A) => new CXe(A)); const iN = "KHR_materials_translucency"; class OXe { /** * @internal */ constructor(e) { this.name = iN, this.order = 174, this._loader = e, this.enabled = this._loader.isExtensionUsed(iN), this.enabled && (e.parent.transparencyAsCoverage = !0); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadMaterialPropertiesAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { const s = new Array(); return s.push(this._loader.loadMaterialBasePropertiesAsync(e, t, r)), s.push(this._loader.loadMaterialPropertiesAsync(e, t, r)), s.push(this._loadTranslucentPropertiesAsync(n, t, r, i)), Promise.all(s).then(() => { }); }); } _loadTranslucentPropertiesAsync(e, t, r, n) { if (!(r instanceof mr)) throw new Error(`${e}: Material type not supported`); const i = r; if (i.subSurface.isTranslucencyEnabled = !0, i.subSurface.volumeIndexOfRefraction = 1, i.subSurface.minimumThickness = 0, i.subSurface.maximumThickness = 0, i.subSurface.useAlbedoToTintTranslucency = !0, n.translucencyFactor !== void 0) i.subSurface.translucencyIntensity = n.translucencyFactor; else return i.subSurface.translucencyIntensity = 0, i.subSurface.isTranslucencyEnabled = !1, Promise.resolve(); return n.translucencyTexture ? (n.translucencyTexture.nonColorData = !0, this._loader.loadTextureInfoAsync(`${e}/translucencyTexture`, n.translucencyTexture).then((s) => { i.subSurface.translucencyIntensityTexture = s; })) : Promise.resolve(); } } Jr.RegisterExtension(iN, (A) => new OXe(A)); const sN = "KHR_materials_volume"; class yXe { /** * @internal */ constructor(e) { this.name = sN, this.order = 173, this._loader = e, this.enabled = this._loader.isExtensionUsed(sN), this.enabled && this._loader._disableInstancedMesh++; } /** @internal */ dispose() { this.enabled && this._loader._disableInstancedMesh--, this._loader = null; } /** * @internal */ loadMaterialPropertiesAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { const s = new Array(); return s.push(this._loader.loadMaterialBasePropertiesAsync(e, t, r)), s.push(this._loader.loadMaterialPropertiesAsync(e, t, r)), s.push(this._loadVolumePropertiesAsync(n, t, r, i)), Promise.all(s).then(() => { }); }); } _loadVolumePropertiesAsync(e, t, r, n) { if (!(r instanceof mr)) throw new Error(`${e}: Material type not supported`); if (!r.subSurface.isRefractionEnabled && !r.subSurface.isTranslucencyEnabled || !n.thicknessFactor) return Promise.resolve(); r.subSurface.volumeIndexOfRefraction = r.indexOfRefraction; const i = n.attenuationDistance !== void 0 ? n.attenuationDistance : Number.MAX_VALUE; return r.subSurface.tintColorAtDistance = i, n.attenuationColor !== void 0 && n.attenuationColor.length == 3 && r.subSurface.tintColor.copyFromFloats(n.attenuationColor[0], n.attenuationColor[1], n.attenuationColor[2]), r.subSurface.minimumThickness = 0, r.subSurface.maximumThickness = n.thicknessFactor, r.subSurface.useThicknessAsDepth = !0, n.thicknessTexture ? (n.thicknessTexture.nonColorData = !0, this._loader.loadTextureInfoAsync(`${e}/thicknessTexture`, n.thicknessTexture).then((s) => { r.subSurface.thicknessTexture = s, r.subSurface.useGltfStyleTextures = !0; })) : Promise.resolve(); } } Jr.RegisterExtension(sN, (A) => new yXe(A)); const aN = "KHR_materials_dispersion"; class kXe { /** * @internal */ constructor(e) { this.name = aN, this.order = 174, this._loader = e, this.enabled = this._loader.isExtensionUsed(aN); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadMaterialPropertiesAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { const s = new Array(); return s.push(this._loader.loadMaterialBasePropertiesAsync(e, t, r)), s.push(this._loader.loadMaterialPropertiesAsync(e, t, r)), s.push(this._loadDispersionPropertiesAsync(n, t, r, i)), Promise.all(s).then(() => { }); }); } _loadDispersionPropertiesAsync(e, t, r, n) { if (!(r instanceof mr)) throw new Error(`${e}: Material type not supported`); return !r.subSurface.isRefractionEnabled || !n.dispersion || (r.subSurface.isDispersionEnabled = !0, r.subSurface.dispersion = n.dispersion), Promise.resolve(); } } Jr.RegisterExtension(aN, (A) => new kXe(A)); const oN = "KHR_mesh_quantization"; class EXe { /** * @internal */ constructor(e) { this.name = oN, this.enabled = e.isExtensionUsed(oN); } /** @internal */ dispose() { } } Jr.RegisterExtension(oN, (A) => new EXe(A)); const fN = "KHR_texture_basisu"; class FXe { /** * @internal */ constructor(e) { this.name = fN, this._loader = e, this.enabled = e.isExtensionUsed(fN); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ _loadTextureAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { const s = t.sampler == null ? Jr.DefaultSampler : ei.Get(`${e}/sampler`, this._loader.gltf.samplers, t.sampler), a = ei.Get(`${n}/source`, this._loader.gltf.images, i.source); return this._loader._createTextureAsync(e, s, a, (f) => { r(f); }, t._textureInfo.nonColorData ? { useRGBAIfASTCBC7NotAvailableWhenUASTC: !0 } : void 0, !t._textureInfo.nonColorData); }); } } Jr.RegisterExtension(fN, (A) => new FXe(A)); const AN = "KHR_texture_transform"; class NXe { /** * @internal */ constructor(e) { this.name = AN, this._loader = e, this.enabled = this._loader.isExtensionUsed(AN); } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadTextureInfoAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => this._loader.loadTextureInfoAsync(e, t, (s) => { if (!(s instanceof We)) throw new Error(`${n}: Texture type not supported`); i.offset && (s.uOffset = i.offset[0], s.vOffset = i.offset[1]), s.uRotationCenter = 0, s.vRotationCenter = 0, i.rotation && (s.wAng = -i.rotation), i.scale && (s.uScale = i.scale[0], s.vScale = i.scale[1]), i.texCoord != null && (s.coordinatesIndex = i.texCoord), r(s); })); } } Jr.RegisterExtension(AN, (A) => new NXe(A)); const dN = "KHR_xmp_json_ld"; class QXe { /** * @internal */ constructor(e) { this.name = dN, this.order = 100, this._loader = e, this.enabled = this._loader.isExtensionUsed(dN); } /** @internal */ dispose() { this._loader = null; } /** * Called after the loader state changes to LOADING. */ onLoading() { var e, t, r; if (this._loader.rootBabylonMesh === null) return; const n = (e = this._loader.gltf.extensions) === null || e === void 0 ? void 0 : e.KHR_xmp_json_ld, i = (r = (t = this._loader.gltf.asset) === null || t === void 0 ? void 0 : t.extensions) === null || r === void 0 ? void 0 : r.KHR_xmp_json_ld; if (n && i) { const s = +i.packet; n.packets && s < n.packets.length && (this._loader.rootBabylonMesh.metadata = this._loader.rootBabylonMesh.metadata || {}, this._loader.rootBabylonMesh.metadata.xmp = n.packets[s]); } } } Jr.RegisterExtension(dN, (A) => new QXe(A)); function FW(A, e, t, r) { return Ne.FromArray(e, t).scale(r); } function YXe(A, e, t, r) { return e[t + 3] * r; } function A1(A, e, t, r) { return e[t] * r; } function vN(A, e, t, r) { return -e[t] * r; } function PO(A, e, t, r) { return e[t + 1] * r; } function PZ(A, e, t, r) { return e[t] * r * 2; } function OE(A) { return { scale: [ new wo(st.ANIMATIONTYPE_FLOAT, `${A}.uScale`, A1, () => 2), new wo(st.ANIMATIONTYPE_FLOAT, `${A}.vScale`, PO, () => 2) ], offset: [ new wo(st.ANIMATIONTYPE_FLOAT, `${A}.uOffset`, A1, () => 2), new wo(st.ANIMATIONTYPE_FLOAT, `${A}.vOffset`, PO, () => 2) ], rotation: [new wo(st.ANIMATIONTYPE_FLOAT, `${A}.wAng`, vN, () => 1)] }; } class Q8 extends HV { /** @internal */ buildAnimations(e, t, r, n, i) { i(e._babylonCamera, this._buildAnimation(t, r, n)); } } class wo extends HV { /** @internal */ buildAnimations(e, t, r, n, i) { for (const s in e._data) i(e._data[s].babylonMaterial, this._buildAnimation(t, r, n)); } } class dI extends HV { /** @internal */ buildAnimations(e, t, r, n, i) { i(e._babylonLight, this._buildAnimation(t, r, n)); } } const MXe = { __array__: { __target__: !0, ...XI } }, LXe = { __array__: { __target__: !0, orthographic: { xmag: [ new Q8(st.ANIMATIONTYPE_FLOAT, "orthoLeft", vN, () => 1), new Q8(st.ANIMATIONTYPE_FLOAT, "orthoRight", PO, () => 1) ], ymag: [ new Q8(st.ANIMATIONTYPE_FLOAT, "orthoBottom", vN, () => 1), new Q8(st.ANIMATIONTYPE_FLOAT, "orthoTop", PO, () => 1) ], zfar: [new Q8(st.ANIMATIONTYPE_FLOAT, "maxZ", A1, () => 1)], znear: [new Q8(st.ANIMATIONTYPE_FLOAT, "minZ", A1, () => 1)] }, perspective: { yfov: [new Q8(st.ANIMATIONTYPE_FLOAT, "fov", A1, () => 1)], zfar: [new Q8(st.ANIMATIONTYPE_FLOAT, "maxZ", A1, () => 1)], znear: [new Q8(st.ANIMATIONTYPE_FLOAT, "minZ", A1, () => 1)] } } }, KXe = { __array__: { __target__: !0, pbrMetallicRoughness: { baseColorFactor: [ new wo(st.ANIMATIONTYPE_COLOR3, "albedoColor", FW, () => 4), new wo(st.ANIMATIONTYPE_FLOAT, "alpha", YXe, () => 4) ], metallicFactor: [new wo(st.ANIMATIONTYPE_FLOAT, "metallic", A1, () => 1)], roughnessFactor: [new wo(st.ANIMATIONTYPE_FLOAT, "roughness", A1, () => 1)], baseColorTexture: { extensions: { KHR_texture_transform: OE("albedoTexture") } } }, emissiveFactor: [new wo(st.ANIMATIONTYPE_COLOR3, "emissiveColor", FW, () => 3)], normalTexture: { scale: [new wo(st.ANIMATIONTYPE_FLOAT, "bumpTexture.level", A1, () => 1)] }, occlusionTexture: { strength: [new wo(st.ANIMATIONTYPE_FLOAT, "ambientTextureStrength", A1, () => 1)], extensions: { KHR_texture_transform: OE("ambientTexture") } }, emissiveTexture: { extensions: { KHR_texture_transform: OE("emissiveTexture") } }, extensions: { KHR_materials_ior: { ior: [new wo(st.ANIMATIONTYPE_FLOAT, "indexOfRefraction", A1, () => 1)] }, KHR_materials_clearcoat: { clearcoatFactor: [new wo(st.ANIMATIONTYPE_FLOAT, "clearCoat.intensity", A1, () => 1)], clearcoatRoughnessFactor: [new wo(st.ANIMATIONTYPE_FLOAT, "clearCoat.roughness", A1, () => 1)] }, KHR_materials_sheen: { sheenColorFactor: [new wo(st.ANIMATIONTYPE_COLOR3, "sheen.color", FW, () => 3)], sheenRoughnessFactor: [new wo(st.ANIMATIONTYPE_FLOAT, "sheen.roughness", A1, () => 1)] }, KHR_materials_specular: { specularFactor: [new wo(st.ANIMATIONTYPE_FLOAT, "metallicF0Factor", A1, () => 1)], specularColorFactor: [new wo(st.ANIMATIONTYPE_COLOR3, "metallicReflectanceColor", FW, () => 3)] }, KHR_materials_emissive_strength: { emissiveStrength: [new wo(st.ANIMATIONTYPE_FLOAT, "emissiveIntensity", A1, () => 1)] }, KHR_materials_transmission: { transmissionFactor: [new wo(st.ANIMATIONTYPE_FLOAT, "subSurface.refractionIntensity", A1, () => 1)] }, KHR_materials_volume: { attenuationColor: [new wo(st.ANIMATIONTYPE_COLOR3, "subSurface.tintColor", FW, () => 3)], attenuationDistance: [new wo(st.ANIMATIONTYPE_FLOAT, "subSurface.tintColorAtDistance", A1, () => 1)], thicknessFactor: [new wo(st.ANIMATIONTYPE_FLOAT, "subSurface.maximumThickness", A1, () => 1)] }, KHR_materials_dispersion: { dispersion: [new wo(st.ANIMATIONTYPE_FLOAT, "subSurface.dispersion", A1, () => 1)] }, KHR_materials_iridescence: { iridescenceFactor: [new wo(st.ANIMATIONTYPE_FLOAT, "iridescence.intensity", A1, () => 1)], iridescenceIor: [new wo(st.ANIMATIONTYPE_FLOAT, "iridescence.indexOfRefraction", A1, () => 1)], iridescenceThicknessMinimum: [new wo(st.ANIMATIONTYPE_FLOAT, "iridescence.minimumThickness", A1, () => 1)], iridescenceThicknessMaximum: [new wo(st.ANIMATIONTYPE_FLOAT, "iridescence.maximumThickness", A1, () => 1)] }, KHR_materials_anisotropy: { anisotropyStrength: [new wo(st.ANIMATIONTYPE_FLOAT, "anisotropy.intensity", A1, () => 1)], anisotropyRotation: [new wo(st.ANIMATIONTYPE_FLOAT, "anisotropy.angle", A1, () => 1)] } } } }, JXe = { KHR_lights_punctual: { lights: { __array__: { __target__: !0, color: [new dI(st.ANIMATIONTYPE_COLOR3, "diffuse", FW, () => 3)], intensity: [new dI(st.ANIMATIONTYPE_FLOAT, "intensity", A1, () => 1)], range: [new dI(st.ANIMATIONTYPE_FLOAT, "range", A1, () => 1)], spot: { innerConeAngle: [new dI(st.ANIMATIONTYPE_FLOAT, "innerAngle", PZ, () => 1)], outerConeAngle: [new dI(st.ANIMATIONTYPE_FLOAT, "angle", PZ, () => 1)] } } } } }, zXe = { nodes: MXe, materials: KXe, cameras: LXe, extensions: JXe }, uN = "KHR_animation_pointer"; class GXe { /** * @internal */ constructor(e) { this.name = uN, this._loader = e; } /** * Defines whether this extension is enabled. */ get enabled() { return this._loader.isExtensionUsed(uN); } /** @internal */ dispose() { this._loader = null; } /** * Loads a glTF animation channel. * @param context The context when loading the asset * @param animationContext The context of the animation when loading the asset * @param animation The glTF animation property * @param channel The glTF animation channel property * @param onLoad Called for each animation loaded * @returns A void promise that resolves when the load is complete or null if not handled */ _loadAnimationChannelAsync(e, t, r, n, i) { var s; const a = (s = n.target.extensions) === null || s === void 0 ? void 0 : s.KHR_animation_pointer; if (!a) return null; n.target.path !== "pointer" && Se.Warn(`${e}/target/path: Value (${n.target.path}) must be (pointer) when using the ${this.name} extension`), n.target.node != null && Se.Warn(`${e}/target/node: Value (${n.target.node}) must not be present when using the ${this.name} extension`); const f = `${e}/extensions/${this.name}`, o = a.pointer; if (!o) throw new Error(`${f}: Pointer is missing`); const d = this._parseAnimationPointer(`${f}/pointer`, o); return d ? this._loader._loadAnimationChannelFromTargetInfoAsync(e, t, r, n, d, i) : (Se.Warn(`${f}/pointer: Invalid pointer (${o}) skipped`), null); } /** * The pointer string is represented by a [JSON pointer](https://datatracker.ietf.org/doc/html/rfc6901). * := /// * := "nodes" | "materials" | "meshes" | "cameras" | "extensions" * := | * := | * := "extensions"// * := | / * := W+ * := D+ * * Examples: * - "/nodes/0/rotation" * - "/materials/2/emissiveFactor" * - "/materials/2/pbrMetallicRoughness/baseColorFactor" * - "/materials/2/extensions/KHR_materials_emissive_strength/emissiveStrength" */ _parseAnimationPointer(e, t) { if (!t.startsWith("/")) return Se.Warn(`${e}: Value (${t}) must start with a slash`), null; const r = t.split("/"); r.shift(); let n = zXe, i = this._loader.gltf, s; for (const a of r) { if (n.__array__) n = n.__array__; else if (n = n[a], !n) return null; i = i && i[a], n.__target__ && (s = i); } return !s || !Array.isArray(n) ? null : { target: s, properties: n }; } } Jr.RegisterExtension(uN, (A) => new GXe(A)); const lN = "MSFT_audio_emitter"; class ZXe { /** * @internal */ constructor(e) { this.name = lN, this._loader = e, this.enabled = this._loader.isExtensionUsed(lN); } /** @internal */ dispose() { this._loader = null, this._clips = null, this._emitters = null; } /** @internal */ onLoading() { const e = this._loader.gltf.extensions; if (e && e[this.name]) { const t = e[this.name]; this._clips = t.clips, this._emitters = t.emitters, ei.Assign(this._clips), ei.Assign(this._emitters); } } /** * @internal */ loadSceneAsync(e, t) { return Jr.LoadExtensionAsync(e, t, this.name, (r, n) => { const i = new Array(); i.push(this._loader.loadSceneAsync(e, t)); for (const s of n.emitters) { const a = ei.Get(`${r}/emitters`, this._emitters, s); if (a.refDistance != null || a.maxDistance != null || a.rolloffFactor != null || a.distanceModel != null || a.innerAngle != null || a.outerAngle != null) throw new Error(`${r}: Direction or Distance properties are not allowed on emitters attached to a scene`); i.push(this._loadEmitterAsync(`${r}/emitters/${a.index}`, a)); } return Promise.all(i).then(() => { }); }); } /** * @internal */ loadNodeAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { const s = new Array(); return this._loader.loadNodeAsync(n, t, (a) => { for (const f of i.emitters) { const o = ei.Get(`${n}/emitters`, this._emitters, f); s.push(this._loadEmitterAsync(`${n}/emitters/${o.index}`, o).then(() => { for (const d of o._babylonSounds) d.attachToMesh(a), (o.innerAngle != null || o.outerAngle != null) && (d.setLocalDirectionToMesh(S.Forward()), d.setDirectionalCone(2 * ye.ToDegrees(o.innerAngle == null ? Math.PI : o.innerAngle), 2 * ye.ToDegrees(o.outerAngle == null ? Math.PI : o.outerAngle), 0)); })); } r(a); }).then((a) => Promise.all(s).then(() => a)); }); } /** * @internal */ loadAnimationAsync(e, t) { return Jr.LoadExtensionAsync(e, t, this.name, (r, n) => this._loader.loadAnimationAsync(e, t).then((i) => { const s = new Array(); ei.Assign(n.events); for (const a of n.events) s.push(this._loadAnimationEventAsync(`${r}/events/${a.index}`, e, t, a, i)); return Promise.all(s).then(() => i); })); } _loadClipAsync(e, t) { if (t._objectURL) return t._objectURL; let r; if (t.uri) r = this._loader.loadUriAsync(e, t, t.uri); else { const n = ei.Get(`${e}/bufferView`, this._loader.gltf.bufferViews, t.bufferView); r = this._loader.loadBufferViewAsync(`/bufferViews/${n.index}`, n); } return t._objectURL = r.then((n) => URL.createObjectURL(new Blob([n], { type: t.mimeType }))), t._objectURL; } _loadEmitterAsync(e, t) { if (t._babylonSounds = t._babylonSounds || [], !t._babylonData) { const r = new Array(), n = t.name || `emitter${t.index}`, i = { loop: !1, autoplay: !1, volume: t.volume == null ? 1 : t.volume }; for (let a = 0; a < t.clips.length; a++) { const f = `/extensions/${this.name}/clips`, o = ei.Get(f, this._clips, t.clips[a].clip); r.push(this._loadClipAsync(`${f}/${t.clips[a].clip}`, o).then((d) => { const v = t._babylonSounds[a] = new qc(n, d, this._loader.babylonScene, null, i); v.refDistance = t.refDistance || 1, v.maxDistance = t.maxDistance || 256, v.rolloffFactor = t.rolloffFactor || 1, v.distanceModel = t.distanceModel || "exponential"; })); } const s = Promise.all(r).then(() => { const a = t.clips.map((o) => o.weight || 1), f = new lee(t.loop || !1, t._babylonSounds, a); t.innerAngle && (f.directionalConeInnerAngle = 2 * ye.ToDegrees(t.innerAngle)), t.outerAngle && (f.directionalConeOuterAngle = 2 * ye.ToDegrees(t.outerAngle)), t.volume && (f.volume = t.volume), t._babylonData.sound = f; }); t._babylonData = { loaded: s }; } return t._babylonData.loaded; } _getEventAction(e, t, r, n, i) { switch (r) { case "play": return (s) => { const a = (i || 0) + (s - n); t.play(a); }; case "stop": return () => { t.stop(); }; case "pause": return () => { t.pause(); }; default: throw new Error(`${e}: Unsupported action ${r}`); } } _loadAnimationEventAsync(e, t, r, n, i) { if (i.targetedAnimations.length == 0) return Promise.resolve(); const s = i.targetedAnimations[0], a = n.emitter, f = ei.Get(`/extensions/${this.name}/emitters`, this._emitters, a); return this._loadEmitterAsync(e, f).then(() => { const o = f._babylonData.sound; if (o) { const d = new wO(n.time, this._getEventAction(e, o, n.action, n.time, n.startOffset)); s.animation.addEvent(d), i.onAnimationGroupEndObservable.add(() => { o.stop(); }), i.onAnimationGroupPauseObservable.add(() => { o.pause(); }); } }); } } Jr.RegisterExtension(lN, (A) => new ZXe(A)); const PN = "MSFT_lod"; class _Xe { /** * @internal */ constructor(e) { this.name = PN, this.order = 100, this.maxLODsToLoad = 10, this.onNodeLODsLoadedObservable = new Oe(), this.onMaterialLODsLoadedObservable = new Oe(), this._bufferLODs = new Array(), this._nodeIndexLOD = null, this._nodeSignalLODs = new Array(), this._nodePromiseLODs = new Array(), this._nodeBufferLODs = new Array(), this._materialIndexLOD = null, this._materialSignalLODs = new Array(), this._materialPromiseLODs = new Array(), this._materialBufferLODs = new Array(), this._loader = e, this.enabled = this._loader.isExtensionUsed(PN); } /** @internal */ dispose() { this._loader = null, this._nodeIndexLOD = null, this._nodeSignalLODs.length = 0, this._nodePromiseLODs.length = 0, this._nodeBufferLODs.length = 0, this._materialIndexLOD = null, this._materialSignalLODs.length = 0, this._materialPromiseLODs.length = 0, this._materialBufferLODs.length = 0, this.onMaterialLODsLoadedObservable.clear(), this.onNodeLODsLoadedObservable.clear(); } /** @internal */ onReady() { for (let e = 0; e < this._nodePromiseLODs.length; e++) { const t = Promise.all(this._nodePromiseLODs[e]).then(() => { e !== 0 && (this._loader.endPerformanceCounter(`Node LOD ${e}`), this._loader.log(`Loaded node LOD ${e}`)), this.onNodeLODsLoadedObservable.notifyObservers(e), e !== this._nodePromiseLODs.length - 1 && (this._loader.startPerformanceCounter(`Node LOD ${e + 1}`), this._loadBufferLOD(this._nodeBufferLODs, e + 1), this._nodeSignalLODs[e] && this._nodeSignalLODs[e].resolve()); }); this._loader._completePromises.push(t); } for (let e = 0; e < this._materialPromiseLODs.length; e++) { const t = Promise.all(this._materialPromiseLODs[e]).then(() => { e !== 0 && (this._loader.endPerformanceCounter(`Material LOD ${e}`), this._loader.log(`Loaded material LOD ${e}`)), this.onMaterialLODsLoadedObservable.notifyObservers(e), e !== this._materialPromiseLODs.length - 1 && (this._loader.startPerformanceCounter(`Material LOD ${e + 1}`), this._loadBufferLOD(this._materialBufferLODs, e + 1), this._materialSignalLODs[e] && this._materialSignalLODs[e].resolve()); }); this._loader._completePromises.push(t); } } /** * @internal */ loadSceneAsync(e, t) { const r = this._loader.loadSceneAsync(e, t); return this._loadBufferLOD(this._bufferLODs, 0), r; } /** * @internal */ loadNodeAsync(e, t, r) { return Jr.LoadExtensionAsync(e, t, this.name, (n, i) => { let s; const a = this._getLODs(n, t, this._loader.gltf.nodes, i.ids); this._loader.logOpen(`${n}`); for (let f = 0; f < a.length; f++) { const o = a[f]; f !== 0 && (this._nodeIndexLOD = f, this._nodeSignalLODs[f] = this._nodeSignalLODs[f] || new MW()); const d = (u) => { r(u), u.setEnabled(!1); }, v = this._loader.loadNodeAsync(`/nodes/${o.index}`, o, d).then((u) => { if (f !== 0) { const l = a[f - 1]; l._babylonTransformNode && (this._disposeTransformNode(l._babylonTransformNode), delete l._babylonTransformNode); } return u.setEnabled(!0), u; }); this._nodePromiseLODs[f] = this._nodePromiseLODs[f] || [], f === 0 ? s = v : (this._nodeIndexLOD = null, this._nodePromiseLODs[f].push(v)); } return this._loader.logClose(), s; }); } /** * @internal */ _loadMaterialAsync(e, t, r, n, i) { return this._nodeIndexLOD ? null : Jr.LoadExtensionAsync(e, t, this.name, (s, a) => { let f; const o = this._getLODs(s, t, this._loader.gltf.materials, a.ids); this._loader.logOpen(`${s}`); for (let d = 0; d < o.length; d++) { const v = o[d]; d !== 0 && (this._materialIndexLOD = d); const u = this._loader._loadMaterialAsync(`/materials/${v.index}`, v, r, n, (l) => { d === 0 && i(l); }).then((l) => { if (d !== 0) { i(l); const P = o[d - 1]._data; P[n] && (this._disposeMaterials([P[n].babylonMaterial]), delete P[n]); } return l; }); this._materialPromiseLODs[d] = this._materialPromiseLODs[d] || [], d === 0 ? f = u : (this._materialIndexLOD = null, this._materialPromiseLODs[d].push(u)); } return this._loader.logClose(), f; }); } /** * @internal */ _loadUriAsync(e, t, r) { if (this._nodeIndexLOD !== null) { this._loader.log("deferred"); const n = this._nodeIndexLOD - 1; return this._nodeSignalLODs[n] = this._nodeSignalLODs[n] || new MW(), this._nodeSignalLODs[this._nodeIndexLOD - 1].promise.then(() => this._loader.loadUriAsync(e, t, r)); } else if (this._materialIndexLOD !== null) { this._loader.log("deferred"); const n = this._materialIndexLOD - 1; return this._materialSignalLODs[n] = this._materialSignalLODs[n] || new MW(), this._materialSignalLODs[n].promise.then(() => this._loader.loadUriAsync(e, t, r)); } return null; } /** * @internal */ loadBufferAsync(e, t, r, n) { if (this._loader.parent.useRangeRequests && !t.uri) { if (!this._loader.bin) throw new Error(`${e}: Uri is missing or the binary glTF is missing its binary chunk`); const i = (s, a) => { const f = r, o = f + n - 1; let d = s[a]; return d ? (d.start = Math.min(d.start, f), d.end = Math.max(d.end, o)) : (d = { start: f, end: o, loaded: new MW() }, s[a] = d), d.loaded.promise.then((v) => new Uint8Array(v.buffer, v.byteOffset + r - d.start, n)); }; return this._loader.log("deferred"), this._nodeIndexLOD !== null ? i(this._nodeBufferLODs, this._nodeIndexLOD) : this._materialIndexLOD !== null ? i(this._materialBufferLODs, this._materialIndexLOD) : i(this._bufferLODs, 0); } return null; } _loadBufferLOD(e, t) { const r = e[t]; r && (this._loader.log(`Loading buffer range [${r.start}-${r.end}]`), this._loader.bin.readAsync(r.start, r.end - r.start + 1).then((n) => { r.loaded.resolve(n); }, (n) => { r.loaded.reject(n); })); } /** * Gets an array of LOD properties from lowest to highest. * @param context * @param property * @param array * @param ids */ _getLODs(e, t, r, n) { if (this.maxLODsToLoad <= 0) throw new Error("maxLODsToLoad must be greater than zero"); const i = []; for (let s = n.length - 1; s >= 0; s--) if (i.push(ei.Get(`${e}/ids/${n[s]}`, r, n[s])), i.length === this.maxLODsToLoad) return i; return i.push(t), i; } _disposeTransformNode(e) { const t = [], r = e.material; r && t.push(r); for (const i of e.getChildMeshes()) i.material && t.push(i.material); e.dispose(); const n = t.filter((i) => this._loader.babylonScene.meshes.every((s) => s.material != i)); this._disposeMaterials(n); } _disposeMaterials(e) { const t = {}; for (const r of e) { for (const n of r.getActiveTextures()) t[n.uniqueId] = n; r.dispose(); } for (const r in t) for (const n of this._loader.babylonScene.materials) n.hasTexture(t[r]) && delete t[r]; for (const r in t) t[r].dispose(); } } Jr.RegisterExtension(PN, (A) => new _Xe(A)); const cN = "MSFT_minecraftMesh"; class $Xe { /** @internal */ constructor(e) { this.name = cN, this._loader = e, this.enabled = this._loader.isExtensionUsed(cN); } /** @internal */ dispose() { this._loader = null; } /** @internal */ loadMaterialPropertiesAsync(e, t, r) { return Jr.LoadExtraAsync(e, t, this.name, (n, i) => { if (i) { if (!(r instanceof mr)) throw new Error(`${n}: Material type not supported`); const s = this._loader.loadMaterialPropertiesAsync(e, t, r); return r.needAlphaBlending() && (r.forceDepthWrite = !0, r.separateCullingPass = !0), r.backFaceCulling = r.forceDepthWrite, r.twoSidedLighting = !0, s; } return null; }); } } Jr.RegisterExtension(cN, (A) => new $Xe(A)); const pN = "MSFT_sRGBFactors"; class e2e { /** @internal */ constructor(e) { this.name = pN, this._loader = e, this.enabled = this._loader.isExtensionUsed(pN); } /** @internal */ dispose() { this._loader = null; } /** @internal */ loadMaterialPropertiesAsync(e, t, r) { return Jr.LoadExtraAsync(e, t, this.name, (n, i) => { if (i) { if (!(r instanceof mr)) throw new Error(`${n}: Material type not supported`); const s = this._loader.loadMaterialPropertiesAsync(e, t, r), a = r.getScene().getEngine().useExactSrgbConversions; return r.albedoTexture || r.albedoColor.toLinearSpaceToRef(r.albedoColor, a), r.reflectivityTexture || r.reflectivityColor.toLinearSpaceToRef(r.reflectivityColor, a), s; } return null; }); } } Jr.RegisterExtension(pN, (A) => new e2e(A)); const G1e = "ExtrasAsMetadata"; class t2e { _assignExtras(e, t) { if (t.extras && Object.keys(t.extras).length > 0) { const r = e.metadata = e.metadata || {}, n = r.gltf = r.gltf || {}; n.extras = t.extras; } } /** * @internal */ constructor(e) { this.name = G1e, this.enabled = !0, this._loader = e; } /** @internal */ dispose() { this._loader = null; } /** * @internal */ loadNodeAsync(e, t, r) { return this._loader.loadNodeAsync(e, t, (n) => { this._assignExtras(n, t), r(n); }); } /** * @internal */ loadCameraAsync(e, t, r) { return this._loader.loadCameraAsync(e, t, (n) => { this._assignExtras(n, t), r(n); }); } /** * @internal */ createMaterial(e, t, r) { const n = this._loader.createMaterial(e, t, r); return this._assignExtras(n, t), n; } } Jr.RegisterExtension(G1e, (A) => new t2e(A)); class KH { constructor() { this.materials = []; } /** * This function will read the mtl file and create each material described inside * This function could be improve by adding : * -some component missing (Ni, Tf...) * -including the specific options available * * @param scene defines the scene the material will be created in * @param data defines the mtl data to parse * @param rootUrl defines the rooturl to use in order to load relative dependencies * @param assetContainer defines the asset container to store the material in (can be null) */ parseMTL(e, t, r, n) { if (t instanceof ArrayBuffer) return; const i = t.split(` `), s = /\s+/; let a, f = null; for (let o = 0; o < i.length; o++) { const d = i[o].trim(); if (d.length === 0 || d.charAt(0) === "#") continue; const v = d.indexOf(" "); let u = v >= 0 ? d.substring(0, v) : d; u = u.toLowerCase(); const l = v >= 0 ? d.substring(v + 1).trim() : ""; if (u === "newmtl") f && this.materials.push(f), e._blockEntityCollection = !!n, f = new Wt(l, e), f._parentContainer = n, e._blockEntityCollection = !1; else if (u === "kd" && f) a = l.split(s, 3).map(parseFloat), f.diffuseColor = Ne.FromArray(a); else if (u === "ka" && f) a = l.split(s, 3).map(parseFloat), f.ambientColor = Ne.FromArray(a); else if (u === "ks" && f) a = l.split(s, 3).map(parseFloat), f.specularColor = Ne.FromArray(a); else if (u === "ke" && f) a = l.split(s, 3).map(parseFloat), f.emissiveColor = Ne.FromArray(a); else if (u === "ns" && f) f.specularPower = parseFloat(l); else if (u === "d" && f) f.alpha = parseFloat(l); else if (u === "map_ka" && f) f.ambientTexture = KH._GetTexture(r, l, e); else if (u === "map_kd" && f) f.diffuseTexture = KH._GetTexture(r, l, e); else if (u === "map_ks" && f) f.specularTexture = KH._GetTexture(r, l, e); else if (u !== "map_ns") if (u === "map_bump" && f) { const P = l.split(s), p = P.indexOf("-bm"); let c = null; p >= 0 && (c = P[p + 1], P.splice(p, 2)), f.bumpTexture = KH._GetTexture(r, P.join(" "), e), f.bumpTexture && c !== null && (f.bumpTexture.level = parseFloat(c)); } else u === "map_d" && f && (f.opacityTexture = KH._GetTexture(r, l, e)); } f && this.materials.push(f); } /** * Gets the texture for the material. * * If the material is imported from input file, * We sanitize the url to ensure it takes the texture from aside the material. * * @param rootUrl The root url to load from * @param value The value stored in the mtl * @param scene * @returns The Texture */ static _GetTexture(e, t, r) { if (!t) return null; let n = e; if (e === "file:") { let i = t.lastIndexOf("\\"); i === -1 && (i = t.lastIndexOf("/")), i > -1 ? n += t.substr(i + 1) : n += t; } else n += t; return new We(n, r, !1, KH.INVERT_TEXTURE_Y); } } KH.INVERT_TEXTURE_Y = !0; class ya { /** * Creates a new SolidParser * @param materialToUse defines the array to fill with the list of materials to use (it will be filled by the parse function) * @param babylonMeshesArray defines the array to fill with the list of loaded meshes (it will be filled by the parse function) * @param loadingOptions defines the loading options to use */ constructor(e, t, r) { this._positions = [], this._normals = [], this._uvs = [], this._colors = [], this._meshesFromObj = [], this._indicesForBabylon = [], this._wrappedPositionForBabylon = [], this._wrappedUvsForBabylon = [], this._wrappedColorsForBabylon = [], this._wrappedNormalsForBabylon = [], this._tuplePosNorm = [], this._curPositionInIndices = 0, this._hasMeshes = !1, this._unwrappedPositionsForBabylon = [], this._unwrappedColorsForBabylon = [], this._unwrappedNormalsForBabylon = [], this._unwrappedUVForBabylon = [], this._triangles = [], this._materialNameFromObj = "", this._objMeshName = "", this._increment = 1, this._isFirstMaterial = !0, this._grayColor = new xt(0.5, 0.5, 0.5, 1), this._materialToUse = e, this._babylonMeshesArray = t, this._loadingOptions = r; } /** * Search for obj in the given array. * This function is called to check if a couple of data already exists in an array. * * If found, returns the index of the founded tuple index. Returns -1 if not found * @param arr Array<{ normals: Array, idx: Array }> * @param obj Array * @returns {boolean} */ _isInArray(e, t) { e[t[0]] || (e[t[0]] = { normals: [], idx: [] }); const r = e[t[0]].normals.indexOf(t[1]); return r === -1 ? -1 : e[t[0]].idx[r]; } _isInArrayUV(e, t) { e[t[0]] || (e[t[0]] = { normals: [], idx: [], uv: [] }); const r = e[t[0]].normals.indexOf(t[1]); return r != 1 && t[2] === e[t[0]].uv[r] ? e[t[0]].idx[r] : -1; } /** * This function set the data for each triangle. * Data are position, normals and uvs * If a tuple of (position, normal) is not set, add the data into the corresponding array * If the tuple already exist, add only their indice * * @param indicePositionFromObj Integer The index in positions array * @param indiceUvsFromObj Integer The index in uvs array * @param indiceNormalFromObj Integer The index in normals array * @param positionVectorFromOBJ Vector3 The value of position at index objIndice * @param textureVectorFromOBJ Vector3 The value of uvs * @param normalsVectorFromOBJ Vector3 The value of normals at index objNormale * @param positionColorsFromOBJ */ _setData(e, t, r, n, i, s, a) { let f; this._loadingOptions.optimizeWithUV ? f = this._isInArrayUV(this._tuplePosNorm, [e, r, t]) : f = this._isInArray(this._tuplePosNorm, [e, r]), f === -1 ? (this._indicesForBabylon.push(this._wrappedPositionForBabylon.length), this._wrappedPositionForBabylon.push(n), this._wrappedUvsForBabylon.push(i), this._wrappedNormalsForBabylon.push(s), a !== void 0 && this._wrappedColorsForBabylon.push(a), this._tuplePosNorm[e].normals.push(r), this._tuplePosNorm[e].idx.push(this._curPositionInIndices++), this._loadingOptions.optimizeWithUV && this._tuplePosNorm[e].uv.push(t)) : this._indicesForBabylon.push(f); } /** * Transform Vector() and BABYLON.Color() objects into numbers in an array */ _unwrapData() { for (let e = 0; e < this._wrappedPositionForBabylon.length; e++) this._unwrappedPositionsForBabylon.push(this._wrappedPositionForBabylon[e].x, this._wrappedPositionForBabylon[e].y, this._wrappedPositionForBabylon[e].z), this._unwrappedNormalsForBabylon.push(this._wrappedNormalsForBabylon[e].x, this._wrappedNormalsForBabylon[e].y, this._wrappedNormalsForBabylon[e].z), this._unwrappedUVForBabylon.push(this._wrappedUvsForBabylon[e].x, this._wrappedUvsForBabylon[e].y), this._loadingOptions.importVertexColors && this._unwrappedColorsForBabylon.push(this._wrappedColorsForBabylon[e].r, this._wrappedColorsForBabylon[e].g, this._wrappedColorsForBabylon[e].b, this._wrappedColorsForBabylon[e].a); this._wrappedPositionForBabylon.length = 0, this._wrappedNormalsForBabylon.length = 0, this._wrappedUvsForBabylon.length = 0, this._wrappedColorsForBabylon.length = 0, this._tuplePosNorm.length = 0, this._curPositionInIndices = 0; } /** * Create triangles from polygons * It is important to notice that a triangle is a polygon * We get 5 patterns of face defined in OBJ File : * facePattern1 = ["1","2","3","4","5","6"] * facePattern2 = ["1/1","2/2","3/3","4/4","5/5","6/6"] * facePattern3 = ["1/1/1","2/2/2","3/3/3","4/4/4","5/5/5","6/6/6"] * facePattern4 = ["1//1","2//2","3//3","4//4","5//5","6//6"] * facePattern5 = ["-1/-1/-1","-2/-2/-2","-3/-3/-3","-4/-4/-4","-5/-5/-5","-6/-6/-6"] * Each pattern is divided by the same method * @param faces Array[String] The indices of elements * @param v Integer The variable to increment */ _getTriangles(e, t) { for (let r = t; r < e.length - 1; r++) this._triangles.push(e[0], e[r], e[r + 1]); } /** * Create triangles and push the data for each polygon for the pattern 1 * In this pattern we get vertice positions * @param face * @param v */ _setDataForCurrentFaceWithPattern1(e, t) { this._getTriangles(e, t); for (let r = 0; r < this._triangles.length; r++) { const n = parseInt(this._triangles[r]) - 1; this._setData( n, 0, 0, // In the pattern 1, normals and uvs are not defined this._positions[n], // Get the vectors data at.Zero(), S.Up(), // Create default vectors this._loadingOptions.importVertexColors ? this._colors[n] : void 0 ); } this._triangles.length = 0; } /** * Create triangles and push the data for each polygon for the pattern 2 * In this pattern we get vertice positions and uvs * @param face * @param v */ _setDataForCurrentFaceWithPattern2(e, t) { this._getTriangles(e, t); for (let r = 0; r < this._triangles.length; r++) { const n = this._triangles[r].split("/"), i = parseInt(n[0]) - 1, s = parseInt(n[1]) - 1; this._setData( i, s, 0, //Default value for normals this._positions[i], //Get the values for each element this._uvs[s], S.Up(), //Default value for normals this._loadingOptions.importVertexColors ? this._colors[i] : void 0 ); } this._triangles.length = 0; } /** * Create triangles and push the data for each polygon for the pattern 3 * In this pattern we get vertice positions, uvs and normals * @param face * @param v */ _setDataForCurrentFaceWithPattern3(e, t) { this._getTriangles(e, t); for (let r = 0; r < this._triangles.length; r++) { const n = this._triangles[r].split("/"), i = parseInt(n[0]) - 1, s = parseInt(n[1]) - 1, a = parseInt(n[2]) - 1; this._setData( i, s, a, this._positions[i], this._uvs[s], this._normals[a] //Set the vector for each component ); } this._triangles.length = 0; } /** * Create triangles and push the data for each polygon for the pattern 4 * In this pattern we get vertice positions and normals * @param face * @param v */ _setDataForCurrentFaceWithPattern4(e, t) { this._getTriangles(e, t); for (let r = 0; r < this._triangles.length; r++) { const n = this._triangles[r].split("//"), i = parseInt(n[0]) - 1, s = parseInt(n[1]) - 1; this._setData( i, 1, //Default value for uv s, this._positions[i], //Get each vector of data at.Zero(), this._normals[s], this._loadingOptions.importVertexColors ? this._colors[i] : void 0 ); } this._triangles.length = 0; } /* * Create triangles and push the data for each polygon for the pattern 3 * In this pattern we get vertice positions, uvs and normals * @param face * @param v */ _setDataForCurrentFaceWithPattern5(e, t) { this._getTriangles(e, t); for (let r = 0; r < this._triangles.length; r++) { const n = this._triangles[r].split("/"), i = this._positions.length + parseInt(n[0]), s = this._uvs.length + parseInt(n[1]), a = this._normals.length + parseInt(n[2]); this._setData( i, s, a, this._positions[i], this._uvs[s], this._normals[a], //Set the vector for each component this._loadingOptions.importVertexColors ? this._colors[i] : void 0 ); } this._triangles.length = 0; } _addPreviousObjMesh() { this._meshesFromObj.length > 0 && (this._handledMesh = this._meshesFromObj[this._meshesFromObj.length - 1], this._unwrapData(), this._indicesForBabylon.reverse(), this._handledMesh.indices = this._indicesForBabylon.slice(), this._handledMesh.positions = this._unwrappedPositionsForBabylon.slice(), this._handledMesh.normals = this._unwrappedNormalsForBabylon.slice(), this._handledMesh.uvs = this._unwrappedUVForBabylon.slice(), this._loadingOptions.importVertexColors && (this._handledMesh.colors = this._unwrappedColorsForBabylon.slice()), this._indicesForBabylon.length = 0, this._unwrappedPositionsForBabylon.length = 0, this._unwrappedColorsForBabylon.length = 0, this._unwrappedNormalsForBabylon.length = 0, this._unwrappedUVForBabylon.length = 0); } _optimizeNormals(e) { const t = e.getVerticesData(J.PositionKind), r = e.getVerticesData(J.NormalKind), n = {}; if (!t || !r) return; for (let s = 0; s < t.length / 3; s++) { const a = t[s * 3 + 0], f = t[s * 3 + 1], o = t[s * 3 + 2], d = a + "_" + f + "_" + o; let v = n[d]; v || (v = [], n[d] = v), v.push(s); } const i = new S(); for (const s in n) { const a = n[s]; if (a.length < 2) continue; const f = a[0]; for (let o = 1; o < a.length; ++o) { const d = a[o]; r[f * 3 + 0] += r[d * 3 + 0], r[f * 3 + 1] += r[d * 3 + 1], r[f * 3 + 2] += r[d * 3 + 2]; } i.copyFromFloats(r[f * 3 + 0], r[f * 3 + 1], r[f * 3 + 2]), i.normalize(); for (let o = 0; o < a.length; ++o) { const d = a[o]; r[d * 3 + 0] = i.x, r[d * 3 + 1] = i.y, r[d * 3 + 2] = i.z; } } e.setVerticesData(J.NormalKind, r); } /** * Function used to parse an OBJ string * @param meshesNames defines the list of meshes to load (all if not defined) * @param data defines the OBJ string * @param scene defines the hosting scene * @param assetContainer defines the asset container to load data in * @param onFileToLoadFound defines a callback that will be called if a MTL file is found */ parse(e, t, r, n, i) { var s; const a = t.split(` `); for (let f = 0; f < a.length; f++) { const o = a[f].trim().replace(/\s\s/g, " "); let d; if (!(o.length === 0 || o.charAt(0) === "#")) if (ya.VertexPattern.test(o)) { if (d = o.match(/[^ ]+/g), this._positions.push(new S(parseFloat(d[1]), parseFloat(d[2]), parseFloat(d[3]))), this._loadingOptions.importVertexColors) if (d.length >= 7) { const v = parseFloat(d[4]), u = parseFloat(d[5]), l = parseFloat(d[6]); this._colors.push(new xt(v > 1 ? v / 255 : v, u > 1 ? u / 255 : u, l > 1 ? l / 255 : l, d.length === 7 || d[7] === void 0 ? 1 : parseFloat(d[7]))); } else this._colors.push(this._grayColor); } else if ((d = ya.NormalPattern.exec(o)) !== null) this._normals.push(new S(parseFloat(d[1]), parseFloat(d[2]), parseFloat(d[3]))); else if ((d = ya.UVPattern.exec(o)) !== null) this._uvs.push(new at(parseFloat(d[1]) * this._loadingOptions.UVScaling.x, parseFloat(d[2]) * this._loadingOptions.UVScaling.y)); else if ((d = ya.FacePattern3.exec(o)) !== null) this._setDataForCurrentFaceWithPattern3( d[1].trim().split(" "), // ["1/1/1", "2/2/2", "3/3/3"] 1 ); else if ((d = ya.FacePattern4.exec(o)) !== null) this._setDataForCurrentFaceWithPattern4( d[1].trim().split(" "), // ["1//1", "2//2", "3//3"] 1 ); else if ((d = ya.FacePattern5.exec(o)) !== null) this._setDataForCurrentFaceWithPattern5( d[1].trim().split(" "), // ["-1/-1/-1", "-2/-2/-2", "-3/-3/-3"] 1 ); else if ((d = ya.FacePattern2.exec(o)) !== null) this._setDataForCurrentFaceWithPattern2( d[1].trim().split(" "), // ["1/1", "2/2", "3/3"] 1 ); else if ((d = ya.FacePattern1.exec(o)) !== null) this._setDataForCurrentFaceWithPattern1( d[1].trim().split(" "), // ["1", "2", "3"] 1 ); else if ((d = ya.LinePattern1.exec(o)) !== null) this._setDataForCurrentFaceWithPattern1( d[1].trim().split(" "), // ["1", "2"] 0 ); else if ((d = ya.LinePattern2.exec(o)) !== null) this._setDataForCurrentFaceWithPattern2( d[1].trim().split(" "), // ["1/1", "2/2"] 0 ); else if ((d = ya.LinePattern3.exec(o)) !== null) this._setDataForCurrentFaceWithPattern3( d[1].trim().split(" "), // ["1/1/1", "2/2/2"] 0 ); else if (ya.GroupDescriptor.test(o) || ya.ObjectDescriptor.test(o)) { const v = { name: o.substring(2).trim(), indices: void 0, positions: void 0, normals: void 0, uvs: void 0, colors: void 0, materialName: this._materialNameFromObj, isObject: ya.ObjectDescriptor.test(o) }; this._addPreviousObjMesh(), this._meshesFromObj.push(v), this._hasMeshes = !0, this._isFirstMaterial = !0, this._increment = 1; } else if (ya.UseMtlDescriptor.test(o)) { if (this._materialNameFromObj = o.substring(7).trim(), !this._isFirstMaterial || !this._hasMeshes) { this._addPreviousObjMesh(); const v = ( //Set the name of the current obj mesh { name: (this._objMeshName || "mesh") + "_mm" + this._increment.toString(), indices: void 0, positions: void 0, normals: void 0, uvs: void 0, colors: void 0, materialName: this._materialNameFromObj, isObject: !1 } ); this._increment++, this._meshesFromObj.push(v), this._hasMeshes = !0; } this._hasMeshes && this._isFirstMaterial && (this._meshesFromObj[this._meshesFromObj.length - 1].materialName = this._materialNameFromObj, this._isFirstMaterial = !1); } else ya.MtlLibGroupDescriptor.test(o) ? i(o.substring(7).trim()) : ya.SmoothDescriptor.test(o) || console.log("Unhandled expression at line : " + o); } if (this._hasMeshes && (this._handledMesh = this._meshesFromObj[this._meshesFromObj.length - 1], this._indicesForBabylon.reverse(), this._unwrapData(), this._handledMesh.indices = this._indicesForBabylon, this._handledMesh.positions = this._unwrappedPositionsForBabylon, this._handledMesh.normals = this._unwrappedNormalsForBabylon, this._handledMesh.uvs = this._unwrappedUVForBabylon, this._loadingOptions.importVertexColors && (this._handledMesh.colors = this._unwrappedColorsForBabylon)), !this._hasMeshes) { let f = null; if (this._indicesForBabylon.length) this._indicesForBabylon.reverse(), this._unwrapData(); else { for (const o of this._positions) this._unwrappedPositionsForBabylon.push(o.x, o.y, o.z); if (this._normals.length) for (const o of this._normals) this._unwrappedNormalsForBabylon.push(o.x, o.y, o.z); if (this._uvs.length) for (const o of this._uvs) this._unwrappedUVForBabylon.push(o.x, o.y); if (this._colors.length) for (const o of this._colors) this._unwrappedColorsForBabylon.push(o.r, o.g, o.b, o.a); this._materialNameFromObj || (f = new Wt(Tf.RandomId(), r), f.pointsCloud = !0, this._materialNameFromObj = f.name, this._normals.length || (f.disableLighting = !0, f.emissiveColor = Ne.White())); } this._meshesFromObj.push({ name: Tf.RandomId(), indices: this._indicesForBabylon, positions: this._unwrappedPositionsForBabylon, colors: this._unwrappedColorsForBabylon, normals: this._unwrappedNormalsForBabylon, uvs: this._unwrappedUVForBabylon, materialName: this._materialNameFromObj, directMaterial: f, isObject: !0 }); } for (let f = 0; f < this._meshesFromObj.length; f++) { if (e && this._meshesFromObj[f].name) { if (e instanceof Array) { if (e.indexOf(this._meshesFromObj[f].name) === -1) continue; } else if (this._meshesFromObj[f].name !== e) continue; } this._handledMesh = this._meshesFromObj[f], r._blockEntityCollection = !!n; const o = new Ee(this._meshesFromObj[f].name, r); if (o._parentContainer = n, r._blockEntityCollection = !1, this._handledMesh._babylonMesh = o, !this._handledMesh.isObject) { for (let v = f - 1; v >= 0; --v) if (this._meshesFromObj[v].isObject && this._meshesFromObj[v]._babylonMesh) { o.parent = this._meshesFromObj[v]._babylonMesh; break; } } if (this._materialToUse.push(this._meshesFromObj[f].materialName), ((s = this._handledMesh.positions) === null || s === void 0 ? void 0 : s.length) === 0) { this._babylonMeshesArray.push(o); continue; } const d = new Ut(); if (d.uvs = this._handledMesh.uvs, d.indices = this._handledMesh.indices, d.positions = this._handledMesh.positions, this._loadingOptions.computeNormals) { const v = new Array(); Ut.ComputeNormals(this._handledMesh.positions, this._handledMesh.indices, v), d.normals = v; } else d.normals = this._handledMesh.normals; this._loadingOptions.importVertexColors && (d.colors = this._handledMesh.colors), d.applyToMesh(o), this._loadingOptions.invertY && (o.scaling.y *= -1), this._loadingOptions.optimizeNormals && this._optimizeNormals(o), this._babylonMeshesArray.push(o), this._handledMesh.directMaterial && (o.material = this._handledMesh.directMaterial); } } } ya.ObjectDescriptor = /^o/; ya.GroupDescriptor = /^g/; ya.MtlLibGroupDescriptor = /^mtllib /; ya.UseMtlDescriptor = /^usemtl /; ya.SmoothDescriptor = /^s /; ya.VertexPattern = /^v(\s+[\d|.|+|\-|e|E]+){3,7}/; ya.NormalPattern = /^vn(\s+[\d|.|+|\-|e|E]+)( +[\d|.|+|\-|e|E]+)( +[\d|.|+|\-|e|E]+)/; ya.UVPattern = /^vt(\s+[\d|.|+|\-|e|E]+)( +[\d|.|+|\-|e|E]+)/; ya.FacePattern1 = /^f\s+(([\d]{1,}[\s]?){3,})+/; ya.FacePattern2 = /^f\s+((([\d]{1,}\/[\d]{1,}[\s]?){3,})+)/; ya.FacePattern3 = /^f\s+((([\d]{1,}\/[\d]{1,}\/[\d]{1,}[\s]?){3,})+)/; ya.FacePattern4 = /^f\s+((([\d]{1,}\/\/[\d]{1,}[\s]?){3,})+)/; ya.FacePattern5 = /^f\s+(((-[\d]{1,}\/-[\d]{1,}\/-[\d]{1,}[\s]?){3,})+)/; ya.LinePattern1 = /^l\s+(([\d]{1,}[\s]?){2,})+/; ya.LinePattern2 = /^l\s+((([\d]{1,}\/[\d]{1,}[\s]?){2,})+)/; ya.LinePattern3 = /^l\s+((([\d]{1,}\/[\d]{1,}\/[\d]{1,}[\s]?){2,})+)/; class T6 { /** * Invert Y-Axis of referenced textures on load */ static get INVERT_TEXTURE_Y() { return KH.INVERT_TEXTURE_Y; } static set INVERT_TEXTURE_Y(e) { KH.INVERT_TEXTURE_Y = e; } /** * Creates loader for .OBJ files * * @param loadingOptions options for loading and parsing OBJ/MTL files. */ constructor(e) { this.name = "obj", this.extensions = ".obj", this._assetContainer = null, this._loadingOptions = e || T6._DefaultLoadingOptions; } static get _DefaultLoadingOptions() { return { computeNormals: T6.COMPUTE_NORMALS, optimizeNormals: T6.OPTIMIZE_NORMALS, importVertexColors: T6.IMPORT_VERTEX_COLORS, invertY: T6.INVERT_Y, invertTextureY: T6.INVERT_TEXTURE_Y, // eslint-disable-next-line @typescript-eslint/naming-convention UVScaling: T6.UV_SCALING, materialLoadingFailsSilently: T6.MATERIAL_LOADING_FAILS_SILENTLY, optimizeWithUV: T6.OPTIMIZE_WITH_UV, skipMaterials: T6.SKIP_MATERIALS }; } /** * Calls synchronously the MTL file attached to this obj. * Load function or importMesh function don't enable to load 2 files in the same time asynchronously. * Without this function materials are not displayed in the first frame (but displayed after). * In consequence it is impossible to get material information in your HTML file * * @param url The URL of the MTL file * @param rootUrl defines where to load data from * @param onSuccess Callback function to be called when the MTL file is loaded * @param onFailure */ _loadMTL(e, t, r, n) { const i = t + e; ye.LoadFile(i, r, void 0, void 0, !1, (s, a) => { n(i, a); }); } /** * Instantiates a OBJ file loader plugin. * @returns the created plugin */ createPlugin() { return new T6(T6._DefaultLoadingOptions); } /** * If the data string can be loaded directly. * @returns if the data can be loaded directly */ canDirectLoad() { return !1; } /** * Imports one or more meshes from the loaded OBJ data and adds them to the scene * @param meshesNames a string or array of strings of the mesh names that should be loaded from the file * @param scene the scene the meshes should be added to * @param data the OBJ data to load * @param rootUrl root url to load from * @returns a promise containing the loaded meshes, particles, skeletons and animations */ importMeshAsync(e, t, r, n) { return this._parseSolid(e, t, r, n).then((i) => ({ meshes: i, particleSystems: [], skeletons: [], animationGroups: [], transformNodes: [], geometries: [], lights: [] })); } /** * Imports all objects from the loaded OBJ data and adds them to the scene * @param scene the scene the objects should be added to * @param data the OBJ data to load * @param rootUrl root url to load from * @returns a promise which completes when objects have been loaded to the scene */ loadAsync(e, t, r) { return this.importMeshAsync(null, e, t, r).then(() => { }); } /** * Load into an asset container. * @param scene The scene to load into * @param data The data to import * @param rootUrl The root url for scene and resources * @returns The loaded asset container */ loadAssetContainerAsync(e, t, r) { const n = new xR(e); return this._assetContainer = n, this.importMeshAsync(null, e, t, r).then((i) => (i.meshes.forEach((s) => n.meshes.push(s)), i.meshes.forEach((s) => { const a = s.material; a && n.materials.indexOf(a) == -1 && (n.materials.push(a), a.getActiveTextures().forEach((o) => { n.textures.indexOf(o) == -1 && n.textures.push(o); })); }), this._assetContainer = null, n)).catch((i) => { throw this._assetContainer = null, i; }); } /** * Read the OBJ file and create an Array of meshes. * Each mesh contains all information given by the OBJ and the MTL file. * i.e. vertices positions and indices, optional normals values, optional UV values, optional material * @param meshesNames defines a string or array of strings of the mesh names that should be loaded from the file * @param scene defines the scene where are displayed the data * @param data defines the content of the obj file * @param rootUrl defines the path to the folder * @returns the list of loaded meshes */ _parseSolid(e, t, r, n) { let i = ""; const s = new KH(), a = [], f = []; new ya(a, f, this._loadingOptions).parse(e, r, t, this._assetContainer, (v) => { i = v; }); const d = []; return i !== "" && !this._loadingOptions.skipMaterials && d.push(new Promise((v, u) => { this._loadMTL(i, n, (l) => { try { s.parseMTL(t, l, n, this._assetContainer); for (let P = 0; P < s.materials.length; P++) { let p = 0; const c = []; let H; for (; (H = a.indexOf(s.materials[P].name, p)) > -1; ) c.push(H), p = H + 1; if (H === -1 && c.length === 0) s.materials[P].dispose(); else for (let T = 0; T < c.length; T++) { const q = f[c[T]], b = s.materials[P]; q.material = b, q.getTotalIndices() || (b.pointsCloud = !0); } } v(); } catch (P) { ye.Warn(`Error processing MTL file: '${i}'`), this._loadingOptions.materialLoadingFailsSilently ? v() : u(P); } }, (l, P) => { ye.Warn(`Error downloading MTL file: '${i}'`), this._loadingOptions.materialLoadingFailsSilently ? v() : u(P); }); })), Promise.all(d).then(() => f); } } T6.OPTIMIZE_WITH_UV = !0; T6.INVERT_Y = !1; T6.IMPORT_VERTEX_COLORS = !1; T6.COMPUTE_NORMALS = !1; T6.OPTIMIZE_NORMALS = !1; T6.UV_SCALING = new at(1, 1); T6.SKIP_MATERIALS = !1; T6.MATERIAL_LOADING_FAILS_SILENTLY = !0; Hn && Hn.RegisterPlugin(new T6()); class aD { constructor() { this.solidPattern = /solid (\S*)([\S\s]*?)endsolid[ ]*(\S*)/g, this.facetsPattern = /facet([\s\S]*?)endfacet/g, this.normalPattern = /normal[\s]+([-+]?[0-9]+\.?[0-9]*([eE][-+]?[0-9]+)?)+[\s]+([-+]?[0-9]*\.?[0-9]+([eE][-+]?[0-9]+)?)+[\s]+([-+]?[0-9]*\.?[0-9]+([eE][-+]?[0-9]+)?)+/g, this.vertexPattern = /vertex[\s]+([-+]?[0-9]+\.?[0-9]*([eE][-+]?[0-9]+)?)+[\s]+([-+]?[0-9]*\.?[0-9]+([eE][-+]?[0-9]+)?)+[\s]+([-+]?[0-9]*\.?[0-9]+([eE][-+]?[0-9]+)?)+/g, this.name = "stl", this.extensions = { ".stl": { isBinary: !0 } }; } /** * Import meshes into a scene. * @param meshesNames An array of mesh names, a single mesh name, or empty string for all meshes that filter what meshes are imported * @param scene The scene to import into * @param data The data to import * @param rootUrl The root url for scene and resources * @param meshes The meshes array to import into * @returns True if successful or false otherwise */ importMesh(e, t, r, n, i) { let s; if (typeof r != "string") { if (this._isBinary(r)) { const a = new Ee("stlmesh", t); return this._parseBinary(a, r), i && i.push(a), !0; } r = new TextDecoder().decode(new Uint8Array(r)); } for (; s = this.solidPattern.exec(r); ) { let a = s[1]; const f = s[3]; if (f && a != f) return ye.Error("Error in STL, solid name != endsolid name"), !1; if (e && a) { if (e instanceof Array) { if (!e.indexOf(a)) continue; } else if (a !== e) continue; } a = a || "stlmesh"; const o = new Ee(a, t); this._parseASCII(o, s[2]), i && i.push(o); } return !0; } /** * Load into a scene. * @param scene The scene to load into * @param data The data to import * @param rootUrl The root url for scene and resources * @returns true if successful or false otherwise */ load(e, t, r) { return this.importMesh(null, e, t, r, null); } /** * Load into an asset container. * @param scene The scene to load into * @param data The data to import * @param rootUrl The root url for scene and resources * @returns The loaded asset container */ loadAssetContainer(e, t, r) { const n = new xR(e); return e._blockEntityCollection = !0, this.importMesh(null, e, t, r, n.meshes), e._blockEntityCollection = !1, n; } _isBinary(e) { const t = new DataView(e); if (t.byteLength <= 80) return !1; const r = 32 / 8 * 3 + 32 / 8 * 3 * 3 + 16 / 8, n = t.getUint32(80, !0); if (80 + 32 / 8 + n * r === t.byteLength) return !0; const i = [115, 111, 108, 105, 100]; for (let s = 0; s < 5; s++) if (t.getUint8(s) !== i[s]) return !0; return !1; } _parseBinary(e, t) { const r = new DataView(t), n = r.getUint32(80, !0), i = 84, s = 12 * 4 + 2; let a = 0; const f = new Float32Array(n * 3 * 3), o = new Float32Array(n * 3 * 3), d = new Uint32Array(n * 3); let v = 0; for (let u = 0; u < n; u++) { const l = i + u * s, P = r.getFloat32(l, !0), p = r.getFloat32(l + 4, !0), c = r.getFloat32(l + 8, !0); for (let H = 1; H <= 3; H++) { const T = l + H * 12; f[a] = r.getFloat32(T, !0), o[a] = P, aD.DO_NOT_ALTER_FILE_COORDINATES ? (f[a + 1] = r.getFloat32(T + 4, !0), f[a + 2] = r.getFloat32(T + 8, !0), o[a + 1] = p, o[a + 2] = c) : (f[a + 2] = r.getFloat32(T + 4, !0), f[a + 1] = r.getFloat32(T + 8, !0), o[a + 2] = p, o[a + 1] = c), a += 3; } aD.DO_NOT_ALTER_FILE_COORDINATES ? (d[v] = v, d[v + 1] = v + 2, d[v + 2] = v + 1, v += 3) : (d[v] = v++, d[v] = v++, d[v] = v++); } e.setVerticesData(J.PositionKind, f), e.setVerticesData(J.NormalKind, o), e.setIndices(d), e.computeWorldMatrix(!0); } _parseASCII(e, t) { const r = [], n = [], i = []; let s = 0, a; for (; a = this.facetsPattern.exec(t); ) { const f = a[1], o = this.normalPattern.exec(f); if (this.normalPattern.lastIndex = 0, !o) continue; const d = [Number(o[1]), Number(o[5]), Number(o[3])]; let v; for (; v = this.vertexPattern.exec(f); ) aD.DO_NOT_ALTER_FILE_COORDINATES ? (r.push(Number(v[1]), Number(v[3]), Number(v[5])), n.push(d[0], d[2], d[1])) : (r.push(Number(v[1]), Number(v[5]), Number(v[3])), n.push(d[0], d[1], d[2])); aD.DO_NOT_ALTER_FILE_COORDINATES ? (i.push(s, s + 2, s + 1), s += 3) : i.push(s++, s++, s++), this.vertexPattern.lastIndex = 0; } this.facetsPattern.lastIndex = 0, e.setVerticesData(J.PositionKind, r), e.setVerticesData(J.NormalKind, n), e.setIndices(i), e.computeWorldMatrix(!0); } } aD.DO_NOT_ALTER_FILE_COORDINATES = !1; Hn && Hn.RegisterPlugin(new aD()); function r2e(A) { return { all: A = A || /* @__PURE__ */ new Map(), on: function(e, t) { var r = A.get(e); r ? r.push(t) : A.set(e, [t]); }, off: function(e, t) { var r = A.get(e); r && (t ? r.splice(r.indexOf(t) >>> 0, 1) : A.set(e, [])); }, emit: function(e, t) { var r = A.get(e); r && r.slice().map(function(n) { n(t); }), (r = A.get("*")) && r.slice().map(function(n) { n(e, t); }); } }; } const OC = { CopyMeshes: { buttons: [ { code: "c", keyDown: !1, ctrlKey: !0, shiftKey: !0 }, { code: "c", keyDown: !1, ctrlKey: !0, shiftKey: !1 } ], callback(A, e) { const t = A.getSelectedNodes(), r = e.shiftKey; t.length && A.copyNodes(t, { instanced: r, onSuccess: (n) => { A.selectNodes(n); } }); } }, DeselectMesh: { buttons: [ { code: "escape", keyDown: !1 } ], callback(A, e) { A.selectNodes([]); } }, DeleteMesh: { buttons: [ { code: "delete", keyDown: !1 }, { code: "backspace", keyDown: !1 } ], callback(A, e) { const t = A.getSelectedNodes(); A.deleteNodes(t); } }, Undo: { buttons: { code: "z", ctrlKey: !0 }, callback(A) { A.undo(); } }, Redo: { buttons: { code: "z", ctrlKey: !0, shiftKey: !0 }, callback(A) { A.redo(); } }, SwitchGizmo: { buttons: [ { code: "1" }, { code: "2" }, { code: "3" } ], callback(A, e) { const t = { 1: "position", 2: "scaling", 3: "rotation" }[e.code]; A.root.setOptions({ selectorType: t }); } } }; var Mo = /* @__PURE__ */ ((A) => (A[A.String = 0] = "String", A[A.Number = 1] = "Number", A[A.Boolean = 2] = "Boolean", A))(Mo || {}), gV = /* @__PURE__ */ ((A) => (A[A.Value = 0] = "Value", A[A.DynamicData = 1] = "DynamicData", A))(gV || {}), $2 = /* @__PURE__ */ ((A) => (A.None = "None", A.PX_PY_PZ = "PX_PY_PZ", A.PX_CY_PZ = "PX_CY_PZ", A.PX_NY_PZ = "PX_NY_PZ", A.PX_PY_CZ = "PX_PY_CZ", A.PX_PY_NZ = "PX_PY_NZ", A.PX_CY_CZ = "PX_CY_CZ", A.PX_CY_NZ = "PX_CY_NZ", A.PX_NY_CZ = "PX_NY_CZ", A.PX_NY_NZ = "PX_NY_NZ", A.CX_PY_PZ = "CX_PY_PZ", A.CX_CY_PZ = "CX_CY_PZ", A.CX_NY_PZ = "CX_NY_PZ", A.CX_PY_CZ = "CX_PY_CZ", A.CX_PY_NZ = "CX_PY_NZ", A.CX_CY_CZ = "CX_CY_CZ", A.CX_CY_NZ = "CX_CY_NZ", A.CX_NY_CZ = "CX_NY_CZ", A.CX_NY_NZ = "CX_NY_NZ", A.NX_PY_PZ = "NX_PY_PZ", A.NX_CY_PZ = "NX_CY_PZ", A.NX_NY_PZ = "NX_NY_PZ", A.NX_PY_CZ = "NX_PY_CZ", A.NX_PY_NZ = "NX_PY_NZ", A.NX_CY_CZ = "NX_CY_CZ", A.NX_CY_NZ = "NX_CY_NZ", A.NX_NY_CZ = "NX_NY_CZ", A.NX_NY_NZ = "NX_NY_NZ", A))($2 || {}); const zd = "le5le", Z1e = zd + "_arcRotateCamera", n2e = zd + "_arcRotateCamera", mSe = zd + "_freeCamera", i2e = zd + "_highlight", yC = zd + "_skybox", s2e = zd + "_ground", iM = zd + "_auxiliaryMesh", XV = iM + "_light", hN = XV + "_range_point", HN = XV + "_range_direction", gN = XV + "_range_spot", a2e = iM + "_camera", BSe = zd + "_alignLine", sM = zd + "_route", o2e = sM + "_line", f2e = sM + "_point", WSe = zd + "_boudingMesh", SSe = zd + "_mainLight", USe = zd + "_fillLight", A2e = zd + "_placeholder", _1e = zd + "_targetMesh", d2e = zd + "_selectorMesh", ISe = zd + "_activeMesh", v2e = "https://drive.le5lecdn.com/2023/0713/1/1/flare.4d7eaf70.png"; var se = /* @__PURE__ */ ((A) => (A[A.Number = 0] = "Number", A[A.Vector3 = 1] = "Vector3", A[A.Quaternion = 2] = "Quaternion", A[A.Matrix = 3] = "Matrix", A[A.Color3 = 4] = "Color3", A[A.Vector2 = 5] = "Vector2", A[A.Size = 6] = "Size", A[A.Color4 = 7] = "Color4", A[A.String = 8] = "String", A[A.Boolean = 9] = "Boolean", A[A.Null = 10] = "Null", A[A.Void = 11] = "Void", A[A.Unknown = 12] = "Unknown", A[A.Texture = 13] = "Texture", A[A.VideoTexture = 14] = "VideoTexture", A[A.CubeTexture = 15] = "CubeTexture", A[A.StandardMaterial = 16] = "StandardMaterial", A[A.PBRMaterial = 17] = "PBRMaterial", A[A.GradientMaterial = 18] = "GradientMaterial", A[A.Object = 19] = "Object", A[A.Array = 20] = "Array", A[A.GridMaterial = 21] = "GridMaterial", A[A.Vector4 = 23] = "Vector4", A[A.ShaderMaterial = 24] = "ShaderMaterial", A[A.NodeMaterial = 25] = "NodeMaterial", A[A.MirrorTexture = 26] = "MirrorTexture", A[A.WaterMaterial = 27] = "WaterMaterial", A[A.CityMaterial = 28] = "CityMaterial", A[A.Scene = 1e3] = "Scene", A[A.TransformNode = 1001] = "TransformNode", A[A.AbstractMesh = 1002] = "AbstractMesh", A[A.Mesh = 1003] = "Mesh", A[A.InstancedMesh = 1004] = "InstancedMesh", A[A.PointLight = 1005] = "PointLight", A[A.SpotLight = 1006] = "SpotLight", A[A.HemisphericLight = 1007] = "HemisphericLight", A[A.DirectionalLight = 1008] = "DirectionalLight", A[A.ArcRotateCamera = 1009] = "ArcRotateCamera", A[A.FollowCamera = 1010] = "FollowCamera", A[A.FreeCamera = 1011] = "FreeCamera", A[A.OrthographicCamera = 1012] = "OrthographicCamera", A))(se || {}); function RSe(A) { const e = {}, t = [], r = []; A.forEach((n) => { e[n.name] ? e[n.name]++ : e[n.name] = 0; }), Object.keys(e).forEach((n) => { e[n] ? t.push(n) : r.push(n); }), console.log("map: ", e), console.log("same: ", t), console.log("diff: ", r); } function VSe(A) { return Object.keys(A).map((e) => [e, A[e]]); } function CSe(A) { console.log("x: ", A.x, "y: ", A.y, "z: ", A.z); } function fq(A, e = (t) => { }) { try { A && A(); } catch (t) { console.error(t), e(t); } } function u2e(A) { return A == null || A === ""; } function Zn(A) { return `${A || ""}${ye.RandomId()}`; } function OSe() { return ((1 + Math.random()) * 65536 | 0).toString(16).substring(1); } function ySe(A, e) { for (let t = 0; t < e; t++) { const r = A.clone("clone", null); r.name = Zn(); const n = Math.random() >= 0.5, i = Math.random() >= 0.5, s = Math.random() * 200, a = Math.random() * 200, f = new S(n ? s : -s, 0, i ? a : -a); r.position = f; } } function kSe(A) { const e = {}; [...A.meshes, ...A.transformNodes].forEach((t) => { const r = t.name; typeof e[r] != "number" && (e[r] = 0), e[r]++; }), console.log(e); } const l2e = (A) => Object.prototype.toString.call(A) === "[object Number]", P2e = (A) => Object.prototype.toString.call(A) === "[object String]", c2e = (A) => Object.prototype.toString.call(A) === "[object Boolean]", XN = (A) => Object.prototype.toString.call(A) === "[object Array]", V0 = (A) => Object.prototype.toString.call(A) === "[object Object]", ESe = (A) => Object.prototype.toString.call(A) === "[object Function]", FSe = (A) => Object.prototype.toString.call(A) === "[object Undefined]", NSe = (A) => Object.prototype.toString.call(A) === "[object Null]", FH = (A, e) => { const t = { [Mo.Boolean]: "boolean", [Mo.Number]: "number", [Mo.String]: "string" }[e]; if (typeof A === t) return A; if (t === "boolean") return !!A; if (t === "number") { let r = Number(A); return !isNaN(r) || (r = parseFloat(A), !isNaN(r)) ? r : 0; } if (t === "string") return A == null ? "" : String(A); }, p2e = (A, e) => { if (e === se.Array) return XN(A) ? A : []; if (e === se.Object) return V0(A) ? A : {}; if (e === se.String) return P2e(A) ? A : A == null ? "" : String(A); if (e === se.Number) return l2e(A) && !isNaN(A) || (A = Number(A), !isNaN(A)) || (A = parseFloat(A), !isNaN(A)) ? A : 0; if (e === se.Boolean) return c2e(A) ? A : !1; }, h2e = (A) => ({ string: Mo.String, integer: Mo.Number, float: Mo.Number, bool: Mo.Boolean })[A] || Mo.String; var H2e = typeof global == "object" && global && global.Object === Object && global; const $1e = H2e; var g2e = typeof self == "object" && self && self.Object === Object && self, X2e = $1e || g2e || Function("return this")(); const hg = X2e; var T2e = hg.Symbol; const XD = T2e; var efe = Object.prototype, q2e = efe.hasOwnProperty, b2e = efe.toString, vI = XD ? XD.toStringTag : void 0; function x2e(A) { var e = q2e.call(A, vI), t = A[vI]; try { A[vI] = void 0; var r = !0; } catch { } var n = b2e.call(A); return r && (e ? A[vI] = t : delete A[vI]), n; } var D2e = Object.prototype, j2e = D2e.toString; function w2e(A) { return j2e.call(A); } var m2e = "[object Null]", B2e = "[object Undefined]", cZ = XD ? XD.toStringTag : void 0; function XU(A) { return A == null ? A === void 0 ? B2e : m2e : cZ && cZ in Object(A) ? x2e(A) : w2e(A); } function TU(A) { return A != null && typeof A == "object"; } var W2e = "[object Symbol]"; function tfe(A) { return typeof A == "symbol" || TU(A) && XU(A) == W2e; } function S2e(A, e) { for (var t = -1, r = A == null ? 0 : A.length, n = Array(r); ++t < r; ) n[t] = e(A[t], t, A); return n; } var U2e = Array.isArray; const yy = U2e; var I2e = 1 / 0, pZ = XD ? XD.prototype : void 0, hZ = pZ ? pZ.toString : void 0; function rfe(A) { if (typeof A == "string") return A; if (yy(A)) return S2e(A, rfe) + ""; if (tfe(A)) return hZ ? hZ.call(A) : ""; var e = A + ""; return e == "0" && 1 / A == -I2e ? "-0" : e; } var R2e = /\s/; function V2e(A) { for (var e = A.length; e-- && R2e.test(A.charAt(e)); ) ; return e; } var C2e = /^\s+/; function O2e(A) { return A && A.slice(0, V2e(A) + 1).replace(C2e, ""); } function xm(A) { var e = typeof A; return A != null && (e == "object" || e == "function"); } var HZ = NaN, y2e = /^[-+]0x[0-9a-f]+$/i, k2e = /^0b[01]+$/i, E2e = /^0o[0-7]+$/i, F2e = parseInt; function nfe(A) { if (typeof A == "number") return A; if (tfe(A)) return HZ; if (xm(A)) { var e = typeof A.valueOf == "function" ? A.valueOf() : A; A = xm(e) ? e + "" : e; } if (typeof A != "string") return A === 0 ? A : +A; A = O2e(A); var t = k2e.test(A); return t || E2e.test(A) ? F2e(A.slice(2), t ? 2 : 8) : y2e.test(A) ? HZ : +A; } var gZ = 1 / 0, N2e = 17976931348623157e292; function Q2e(A) { if (!A) return A === 0 ? A : 0; if (A = nfe(A), A === gZ || A === -gZ) { var e = A < 0 ? -1 : 1; return e * N2e; } return A === A ? A : 0; } function Y2e(A) { var e = Q2e(A), t = e % 1; return e === e ? t ? e - t : e : 0; } var M2e = "[object AsyncFunction]", L2e = "[object Function]", K2e = "[object GeneratorFunction]", J2e = "[object Proxy]"; function ife(A) { if (!xm(A)) return !1; var e = XU(A); return e == L2e || e == K2e || e == M2e || e == J2e; } var z2e = hg["__core-js_shared__"]; const yE = z2e; var XZ = function() { var A = /[^.]+$/.exec(yE && yE.keys && yE.keys.IE_PROTO || ""); return A ? "Symbol(src)_1." + A : ""; }(); function G2e(A) { return !!XZ && XZ in A; } var Z2e = Function.prototype, _2e = Z2e.toString; function Fm(A) { if (A != null) { try { return _2e.call(A); } catch { } try { return A + ""; } catch { } } return ""; } var $2e = /[\\^$.*+?()[\]{}|]/g, e4e = /^\[object .+?Constructor\]$/, t4e = Function.prototype, r4e = Object.prototype, n4e = t4e.toString, i4e = r4e.hasOwnProperty, s4e = RegExp( "^" + n4e.call(i4e).replace($2e, "\\$&").replace(/hasOwnProperty|(function).*?(?=\\\()| for .+?(?=\\\])/g, "$1.*?") + "$" ); function a4e(A) { if (!xm(A) || G2e(A)) return !1; var e = ife(A) ? s4e : e4e; return e.test(Fm(A)); } function o4e(A, e) { return A == null ? void 0 : A[e]; } function Nm(A, e) { var t = o4e(A, e); return a4e(t) ? t : void 0; } var f4e = Nm(hg, "WeakMap"); const TN = f4e; var TZ = Object.create, A4e = /* @__PURE__ */ function() { function A() { } return function(e) { if (!xm(e)) return {}; if (TZ) return TZ(e); A.prototype = e; var t = new A(); return A.prototype = void 0, t; }; }(); const d4e = A4e; function v4e(A, e) { var t = -1, r = A.length; for (e || (e = Array(r)); ++t < r; ) e[t] = A[t]; return e; } var u4e = function() { try { var A = Nm(Object, "defineProperty"); return A({}, "", {}), A; } catch { } }(); const qZ = u4e; function l4e(A, e) { for (var t = -1, r = A == null ? 0 : A.length; ++t < r && e(A[t], t, A) !== !1; ) ; return A; } var P4e = 9007199254740991, c4e = /^(?:0|[1-9]\d*)$/; function p4e(A, e) { var t = typeof A; return e = e ?? P4e, !!e && (t == "number" || t != "symbol" && c4e.test(A)) && A > -1 && A % 1 == 0 && A < e; } function sfe(A, e, t) { e == "__proto__" && qZ ? qZ(A, e, { configurable: !0, enumerable: !0, value: t, writable: !0 }) : A[e] = t; } function afe(A, e) { return A === e || A !== A && e !== e; } var h4e = Object.prototype, H4e = h4e.hasOwnProperty; function ofe(A, e, t) { var r = A[e]; (!(H4e.call(A, e) && afe(r, t)) || t === void 0 && !(e in A)) && sfe(A, e, t); } function ky(A, e, t, r) { var n = !t; t || (t = {}); for (var i = -1, s = e.length; ++i < s; ) { var a = e[i], f = r ? r(t[a], A[a], a, t, A) : void 0; f === void 0 && (f = A[a]), n ? sfe(t, a, f) : ofe(t, a, f); } return t; } var g4e = 9007199254740991; function ffe(A) { return typeof A == "number" && A > -1 && A % 1 == 0 && A <= g4e; } function Afe(A) { return A != null && ffe(A.length) && !ife(A); } var X4e = Object.prototype; function aM(A) { var e = A && A.constructor, t = typeof e == "function" && e.prototype || X4e; return A === t; } function T4e(A, e) { for (var t = -1, r = Array(A); ++t < A; ) r[t] = e(t); return r; } var q4e = "[object Arguments]"; function bZ(A) { return TU(A) && XU(A) == q4e; } var dfe = Object.prototype, b4e = dfe.hasOwnProperty, x4e = dfe.propertyIsEnumerable, D4e = bZ(/* @__PURE__ */ function() { return arguments; }()) ? bZ : function(A) { return TU(A) && b4e.call(A, "callee") && !x4e.call(A, "callee"); }; const j4e = D4e; function w4e() { return !1; } var vfe = typeof exports == "object" && exports && !exports.nodeType && exports, xZ = vfe && typeof module == "object" && module && !module.nodeType && module, m4e = xZ && xZ.exports === vfe, DZ = m4e ? hg.Buffer : void 0, B4e = DZ ? DZ.isBuffer : void 0, W4e = B4e || w4e; const ufe = W4e; var S4e = "[object Arguments]", U4e = "[object Array]", I4e = "[object Boolean]", R4e = "[object Date]", V4e = "[object Error]", C4e = "[object Function]", O4e = "[object Map]", y4e = "[object Number]", k4e = "[object Object]", E4e = "[object RegExp]", F4e = "[object Set]", N4e = "[object String]", Q4e = "[object WeakMap]", Y4e = "[object ArrayBuffer]", M4e = "[object DataView]", L4e = "[object Float32Array]", K4e = "[object Float64Array]", J4e = "[object Int8Array]", z4e = "[object Int16Array]", G4e = "[object Int32Array]", Z4e = "[object Uint8Array]", _4e = "[object Uint8ClampedArray]", $4e = "[object Uint16Array]", eTe = "[object Uint32Array]", yf = {}; yf[L4e] = yf[K4e] = yf[J4e] = yf[z4e] = yf[G4e] = yf[Z4e] = yf[_4e] = yf[$4e] = yf[eTe] = !0; yf[S4e] = yf[U4e] = yf[Y4e] = yf[I4e] = yf[M4e] = yf[R4e] = yf[V4e] = yf[C4e] = yf[O4e] = yf[y4e] = yf[k4e] = yf[E4e] = yf[F4e] = yf[N4e] = yf[Q4e] = !1; function tTe(A) { return TU(A) && ffe(A.length) && !!yf[XU(A)]; } function oM(A) { return function(e) { return A(e); }; } var lfe = typeof exports == "object" && exports && !exports.nodeType && exports, RI = lfe && typeof module == "object" && module && !module.nodeType && module, rTe = RI && RI.exports === lfe, kE = rTe && $1e.process, nTe = function() { try { var A = RI && RI.require && RI.require("util").types; return A || kE && kE.binding && kE.binding("util"); } catch { } }(); const GS = nTe; var jZ = GS && GS.isTypedArray, iTe = jZ ? oM(jZ) : tTe; const sTe = iTe; var aTe = Object.prototype, oTe = aTe.hasOwnProperty; function Pfe(A, e) { var t = yy(A), r = !t && j4e(A), n = !t && !r && ufe(A), i = !t && !r && !n && sTe(A), s = t || r || n || i, a = s ? T4e(A.length, String) : [], f = a.length; for (var o in A) (e || oTe.call(A, o)) && !(s && // Safari 9 has enumerable `arguments.length` in strict mode. (o == "length" || // Node.js 0.10 has enumerable non-index properties on buffers. n && (o == "offset" || o == "parent") || // PhantomJS 2 has enumerable non-index properties on typed arrays. i && (o == "buffer" || o == "byteLength" || o == "byteOffset") || // Skip index properties. p4e(o, f))) && a.push(o); return a; } function cfe(A, e) { return function(t) { return A(e(t)); }; } var fTe = cfe(Object.keys, Object); const ATe = fTe; var dTe = Object.prototype, vTe = dTe.hasOwnProperty; function uTe(A) { if (!aM(A)) return ATe(A); var e = []; for (var t in Object(A)) vTe.call(A, t) && t != "constructor" && e.push(t); return e; } function fM(A) { return Afe(A) ? Pfe(A) : uTe(A); } function lTe(A) { var e = []; if (A != null) for (var t in Object(A)) e.push(t); return e; } var PTe = Object.prototype, cTe = PTe.hasOwnProperty; function pTe(A) { if (!xm(A)) return lTe(A); var e = aM(A), t = []; for (var r in A) r == "constructor" && (e || !cTe.call(A, r)) || t.push(r); return t; } function AM(A) { return Afe(A) ? Pfe(A, !0) : pTe(A); } var hTe = Nm(Object, "create"); const vR = hTe; function HTe() { this.__data__ = vR ? vR(null) : {}, this.size = 0; } function gTe(A) { var e = this.has(A) && delete this.__data__[A]; return this.size -= e ? 1 : 0, e; } var XTe = "__lodash_hash_undefined__", TTe = Object.prototype, qTe = TTe.hasOwnProperty; function bTe(A) { var e = this.__data__; if (vR) { var t = e[A]; return t === XTe ? void 0 : t; } return qTe.call(e, A) ? e[A] : void 0; } var xTe = Object.prototype, DTe = xTe.hasOwnProperty; function jTe(A) { var e = this.__data__; return vR ? e[A] !== void 0 : DTe.call(e, A); } var wTe = "__lodash_hash_undefined__"; function mTe(A, e) { var t = this.__data__; return this.size += this.has(A) ? 0 : 1, t[A] = vR && e === void 0 ? wTe : e, this; } function Dm(A) { var e = -1, t = A == null ? 0 : A.length; for (this.clear(); ++e < t; ) { var r = A[e]; this.set(r[0], r[1]); } } Dm.prototype.clear = HTe; Dm.prototype.delete = gTe; Dm.prototype.get = bTe; Dm.prototype.has = jTe; Dm.prototype.set = mTe; function BTe() { this.__data__ = [], this.size = 0; } function Ey(A, e) { for (var t = A.length; t--; ) if (afe(A[t][0], e)) return t; return -1; } var WTe = Array.prototype, STe = WTe.splice; function UTe(A) { var e = this.__data__, t = Ey(e, A); if (t < 0) return !1; var r = e.length - 1; return t == r ? e.pop() : STe.call(e, t, 1), --this.size, !0; } function ITe(A) { var e = this.__data__, t = Ey(e, A); return t < 0 ? void 0 : e[t][1]; } function RTe(A) { return Ey(this.__data__, A) > -1; } function VTe(A, e) { var t = this.__data__, r = Ey(t, A); return r < 0 ? (++this.size, t.push([A, e])) : t[r][1] = e, this; } function mq(A) { var e = -1, t = A == null ? 0 : A.length; for (this.clear(); ++e < t; ) { var r = A[e]; this.set(r[0], r[1]); } } mq.prototype.clear = BTe; mq.prototype.delete = UTe; mq.prototype.get = ITe; mq.prototype.has = RTe; mq.prototype.set = VTe; var CTe = Nm(hg, "Map"); const uR = CTe; function OTe() { this.size = 0, this.__data__ = { hash: new Dm(), map: new (uR || mq)(), string: new Dm() }; } function yTe(A) { var e = typeof A; return e == "string" || e == "number" || e == "symbol" || e == "boolean" ? A !== "__proto__" : A === null; } function Fy(A, e) { var t = A.__data__; return yTe(e) ? t[typeof e == "string" ? "string" : "hash"] : t.map; } function kTe(A) { var e = Fy(this, A).delete(A); return this.size -= e ? 1 : 0, e; } function ETe(A) { return Fy(this, A).get(A); } function FTe(A) { return Fy(this, A).has(A); } function NTe(A, e) { var t = Fy(this, A), r = t.size; return t.set(A, e), this.size += t.size == r ? 0 : 1, this; } function qU(A) { var e = -1, t = A == null ? 0 : A.length; for (this.clear(); ++e < t; ) { var r = A[e]; this.set(r[0], r[1]); } } qU.prototype.clear = OTe; qU.prototype.delete = kTe; qU.prototype.get = ETe; qU.prototype.has = FTe; qU.prototype.set = NTe; function wZ(A) { return A == null ? "" : rfe(A); } function pfe(A, e) { for (var t = -1, r = e.length, n = A.length; ++t < r; ) A[n + t] = e[t]; return A; } var QTe = cfe(Object.getPrototypeOf, Object); const hfe = QTe; var YTe = hg.isFinite, MTe = Math.min; function LTe(A) { var e = Math[A]; return function(t, r) { if (t = nfe(t), r = r == null ? 0 : MTe(Y2e(r), 292), r && YTe(t)) { var n = (wZ(t) + "e").split("e"), i = e(n[0] + "e" + (+n[1] + r)); return n = (wZ(i) + "e").split("e"), +(n[0] + "e" + (+n[1] - r)); } return e(t); }; } function KTe() { this.__data__ = new mq(), this.size = 0; } function JTe(A) { var e = this.__data__, t = e.delete(A); return this.size = e.size, t; } function zTe(A) { return this.__data__.get(A); } function GTe(A) { return this.__data__.has(A); } var ZTe = 200; function _Te(A, e) { var t = this.__data__; if (t instanceof mq) { var r = t.__data__; if (!uR || r.length < ZTe - 1) return r.push([A, e]), this.size = ++t.size, this; t = this.__data__ = new qU(r); } return t.set(A, e), this.size = t.size, this; } function bU(A) { var e = this.__data__ = new mq(A); this.size = e.size; } bU.prototype.clear = KTe; bU.prototype.delete = JTe; bU.prototype.get = zTe; bU.prototype.has = GTe; bU.prototype.set = _Te; function $Te(A, e) { return A && ky(e, fM(e), A); } function e8e(A, e) { return A && ky(e, AM(e), A); } var Hfe = typeof exports == "object" && exports && !exports.nodeType && exports, mZ = Hfe && typeof module == "object" && module && !module.nodeType && module, t8e = mZ && mZ.exports === Hfe, BZ = t8e ? hg.Buffer : void 0, WZ = BZ ? BZ.allocUnsafe : void 0; function r8e(A, e) { if (e) return A.slice(); var t = A.length, r = WZ ? WZ(t) : new A.constructor(t); return A.copy(r), r; } function n8e(A, e) { for (var t = -1, r = A == null ? 0 : A.length, n = 0, i = []; ++t < r; ) { var s = A[t]; e(s, t, A) && (i[n++] = s); } return i; } function gfe() { return []; } var i8e = Object.prototype, s8e = i8e.propertyIsEnumerable, SZ = Object.getOwnPropertySymbols, a8e = SZ ? function(A) { return A == null ? [] : (A = Object(A), n8e(SZ(A), function(e) { return s8e.call(A, e); })); } : gfe; const dM = a8e; function o8e(A, e) { return ky(A, dM(A), e); } var f8e = Object.getOwnPropertySymbols, A8e = f8e ? function(A) { for (var e = []; A; ) pfe(e, dM(A)), A = hfe(A); return e; } : gfe; const Xfe = A8e; function d8e(A, e) { return ky(A, Xfe(A), e); } function Tfe(A, e, t) { var r = e(A); return yy(A) ? r : pfe(r, t(A)); } function v8e(A) { return Tfe(A, fM, dM); } function u8e(A) { return Tfe(A, AM, Xfe); } var l8e = Nm(hg, "DataView"); const qN = l8e; var P8e = Nm(hg, "Promise"); const bN = P8e; var c8e = Nm(hg, "Set"); const xN = c8e; var UZ = "[object Map]", p8e = "[object Object]", IZ = "[object Promise]", RZ = "[object Set]", VZ = "[object WeakMap]", CZ = "[object DataView]", h8e = Fm(qN), H8e = Fm(uR), g8e = Fm(bN), X8e = Fm(xN), T8e = Fm(TN), Nw = XU; (qN && Nw(new qN(new ArrayBuffer(1))) != CZ || uR && Nw(new uR()) != UZ || bN && Nw(bN.resolve()) != IZ || xN && Nw(new xN()) != RZ || TN && Nw(new TN()) != VZ) && (Nw = function(A) { var e = XU(A), t = e == p8e ? A.constructor : void 0, r = t ? Fm(t) : ""; if (r) switch (r) { case h8e: return CZ; case H8e: return UZ; case g8e: return IZ; case X8e: return RZ; case T8e: return VZ; } return e; }); const vM = Nw; var q8e = Object.prototype, b8e = q8e.hasOwnProperty; function x8e(A) { var e = A.length, t = new A.constructor(e); return e && typeof A[0] == "string" && b8e.call(A, "index") && (t.index = A.index, t.input = A.input), t; } var D8e = hg.Uint8Array; const OZ = D8e; function uM(A) { var e = new A.constructor(A.byteLength); return new OZ(e).set(new OZ(A)), e; } function j8e(A, e) { var t = e ? uM(A.buffer) : A.buffer; return new A.constructor(t, A.byteOffset, A.byteLength); } var w8e = /\w*$/; function m8e(A) { var e = new A.constructor(A.source, w8e.exec(A)); return e.lastIndex = A.lastIndex, e; } var yZ = XD ? XD.prototype : void 0, kZ = yZ ? yZ.valueOf : void 0; function B8e(A) { return kZ ? Object(kZ.call(A)) : {}; } function W8e(A, e) { var t = e ? uM(A.buffer) : A.buffer; return new A.constructor(t, A.byteOffset, A.length); } var S8e = "[object Boolean]", U8e = "[object Date]", I8e = "[object Map]", R8e = "[object Number]", V8e = "[object RegExp]", C8e = "[object Set]", O8e = "[object String]", y8e = "[object Symbol]", k8e = "[object ArrayBuffer]", E8e = "[object DataView]", F8e = "[object Float32Array]", N8e = "[object Float64Array]", Q8e = "[object Int8Array]", Y8e = "[object Int16Array]", M8e = "[object Int32Array]", L8e = "[object Uint8Array]", K8e = "[object Uint8ClampedArray]", J8e = "[object Uint16Array]", z8e = "[object Uint32Array]"; function G8e(A, e, t) { var r = A.constructor; switch (e) { case k8e: return uM(A); case S8e: case U8e: return new r(+A); case E8e: return j8e(A, t); case F8e: case N8e: case Q8e: case Y8e: case M8e: case L8e: case K8e: case J8e: case z8e: return W8e(A, t); case I8e: return new r(); case R8e: case O8e: return new r(A); case V8e: return m8e(A); case C8e: return new r(); case y8e: return B8e(A); } } function Z8e(A) { return typeof A.constructor == "function" && !aM(A) ? d4e(hfe(A)) : {}; } var _8e = "[object Map]"; function $8e(A) { return TU(A) && vM(A) == _8e; } var EZ = GS && GS.isMap, eqe = EZ ? oM(EZ) : $8e; const tqe = eqe; var rqe = "[object Set]"; function nqe(A) { return TU(A) && vM(A) == rqe; } var FZ = GS && GS.isSet, iqe = FZ ? oM(FZ) : nqe; const sqe = iqe; var aqe = 1, oqe = 2, fqe = 4, qfe = "[object Arguments]", Aqe = "[object Array]", dqe = "[object Boolean]", vqe = "[object Date]", uqe = "[object Error]", bfe = "[object Function]", lqe = "[object GeneratorFunction]", Pqe = "[object Map]", cqe = "[object Number]", xfe = "[object Object]", pqe = "[object RegExp]", hqe = "[object Set]", Hqe = "[object String]", gqe = "[object Symbol]", Xqe = "[object WeakMap]", Tqe = "[object ArrayBuffer]", qqe = "[object DataView]", bqe = "[object Float32Array]", xqe = "[object Float64Array]", Dqe = "[object Int8Array]", jqe = "[object Int16Array]", wqe = "[object Int32Array]", mqe = "[object Uint8Array]", Bqe = "[object Uint8ClampedArray]", Wqe = "[object Uint16Array]", Sqe = "[object Uint32Array]", gf = {}; gf[qfe] = gf[Aqe] = gf[Tqe] = gf[qqe] = gf[dqe] = gf[vqe] = gf[bqe] = gf[xqe] = gf[Dqe] = gf[jqe] = gf[wqe] = gf[Pqe] = gf[cqe] = gf[xfe] = gf[pqe] = gf[hqe] = gf[Hqe] = gf[gqe] = gf[mqe] = gf[Bqe] = gf[Wqe] = gf[Sqe] = !0; gf[uqe] = gf[bfe] = gf[Xqe] = !1; function kC(A, e, t, r, n, i) { var s, a = e & aqe, f = e & oqe, o = e & fqe; if (t && (s = n ? t(A, r, n, i) : t(A)), s !== void 0) return s; if (!xm(A)) return A; var d = yy(A); if (d) { if (s = x8e(A), !a) return v4e(A, s); } else { var v = vM(A), u = v == bfe || v == lqe; if (ufe(A)) return r8e(A, a); if (v == xfe || v == qfe || u && !n) { if (s = f || u ? {} : Z8e(A), !a) return f ? d8e(A, e8e(s, A)) : o8e(A, $Te(s, A)); } else { if (!gf[v]) return n ? A : {}; s = G8e(A, v, a); } } i || (i = new bU()); var l = i.get(A); if (l) return l; i.set(A, s), sqe(A) ? A.forEach(function(c) { s.add(kC(c, e, t, c, A, i)); }) : tqe(A) && A.forEach(function(c, H) { s.set(H, kC(c, e, t, H, A, i)); }); var P = o ? f ? u8e : v8e : f ? AM : fM, p = d ? void 0 : P(A); return l4e(p || A, function(c, H) { p && (H = c, c = A[H]), ofe(s, H, kC(c, e, t, H, A, i)); }), s; } var Uqe = 1, Iqe = 4; function M9(A) { return kC(A, Uqe | Iqe); } var Rqe = LTe("round"); const Vqe = Rqe; function QSe(A, e, t) { const r = A.getScene(); let n = [], i = A.material, s = r.getMaterialById("le5le_alarmMat"); A.material instanceof Wt || A.material instanceof mr ? A.material instanceof Wt ? n = A.material.diffuseColor.asArray() : n = A.material.albedoColor.asArray() : s || (s = new Wt("le5le_alarmMat", r), A.material = s, n = s.diffuseColor.asArray()); const a = A4(e); let f; i instanceof mr ? f = new st( "alarm", "albedoColor", 60, st.ANIMATIONTYPE_COLOR3, st.ANIMATIONLOOPMODE_CYCLE ) : f = new st( "alarm", "diffuseColor", 60, st.ANIMATIONTYPE_COLOR3, st.ANIMATIONLOOPMODE_CYCLE ); const o = []; o.push({ frame: 0, value: Ne.FromArray(n) }), o.push({ frame: t / 2, value: a }), o.push({ frame: t, value: Ne.FromArray(n) }), f.setKeys(o); const d = r.beginDirectAnimation( A.material, [f], 0, t, !0 ); return d.disposeOnEnd = !0, d.onAnimationEndObservable.add(() => { }), d.onAnimationLoopObservable.add(() => { d.stop(); }), d; } class YSe { constructor(e, t = []) { this.materials = /* @__PURE__ */ new Map(), this.addMaterial = (r) => { r.id = r.id || Zn(); const { type: n, id: i } = r; return this.materials.get(n) || this.materials.set(n, /* @__PURE__ */ new Map()), this.materials.get(n).set(i, r), r; }, this.getMaterial = (r) => { for (const [, n] of this.materials) if (n.has(r)) return n.get(r); }, this.data = () => M9(this.allMaterials()), this.matDispose = (r, n) => { let i = n.getMaterialByName(r); i && (i.diffuseColor && (i.diffuseColor = null), i.diffuseTexture && i.diffuseTexture.dispose(), i.emissiveColor && (i.diffuseColor = null), i.emissiveTexture && i.emissiveTexture.dispose(), i.diffuseTexture && i.diffuseTexture.name === "video" && i.diffuseTexture.video.pause(), i.dispose()); }, this.dispose = () => { }, this.root = e, this.materials.set("image", /* @__PURE__ */ new Map()), this.materials.set("video", /* @__PURE__ */ new Map()), t.forEach(([r, n]) => { this.materials.get(r) || this.materials.set(r, /* @__PURE__ */ new Map()); const i = this.materials.get(r); n.forEach((s) => { const { id: a } = s; i.set(a, s); }); }); } //删除某一条材质信息 removeMaterial(e) { for (const [, t] of this.materials) if (t.get(e)) return this.root.getAllScenes().forEach((r) => { this.matDispose(e, r.scene.scene); }), t.delete(e); } //----------- updateMaterial(e) { this.materials.forEach((t) => { t.get(e.id) && (t.delete(e.id), this.addMaterial(e), this.root.getAllScenes().forEach((r) => { let n = r.scene.scene.getMaterialByName(e.id); if (n) { let i = e.type === "image" ? this.createMatImage(e, r.scene.scene) : this.createMatVideo(e, r.scene.scene); r.scene.scene.meshes.forEach((s) => { s.material && s.material.name && s.material.name === n.name && (s.material = i); }), n.dispose(); } })); }); } //获取this.materials列表的信息; allMaterials() { return [...this.materials].map(([e, t]) => [ e, [...t].map(([, r]) => r) ]); } //将材质绑定在Mesh上 bindMaterialToMesh(e, t) { if (e && e.length > 0) { let r = e, n = r[0].getScene(), i = n.getMaterialByName(t); if (i) r.forEach((s) => { s.material = null, s.material = i; }); else { let s = this.getMaterial(t), a = s.type === "image" ? this.createMatImage(s, n) : this.createMatVideo(s, n); r.forEach((f) => { f.material = null, f.material = a; }); } } } //将Mat解绑 removeMatToMesh(e) { let t = null; if (e && e.length > 0) { e.forEach((i) => { t = i.material, i.material = null; }); let n = !0; this.root.getAllScenes().forEach((i) => { let s = i.scene; s.meshes.forEach((a) => { t && a.material && a.material.name && t.name && a.material.name === t.name && (n = !1); }), n && t && t.name && this.matDispose(t.name, s); }); } } //创建一个图片贴图类型的材质; createMatImage(e, t) { let r = new Wt("standardMaterial", t); return r.name = e.id, this.matImage(r, e, t), r; } /** * 根据前端UI的交互传过来的materialData修改material的属性(img); * @param mat StandardMaterial 材质实例对象; * @param data MaterialData 信息 * @param scene Scene 当前场景 */ matImage(e, t, r) { t.diffuseColor && (e.diffuseColor = A4(t.diffuseColor)), t.emissiveColor && (e.emissiveColor = A4(t.emissiveColor)), t.type === "image" && (t.diffuseTexture && (e.diffuseTexture = new We(t.diffuseTexture, r)), t.emissiveTexture && (e.emissiveTexture = new We(t.emissiveTexture, r))); } /** * 创建一个视频贴图类的材质 * @param data 属性数据 * @param scene 当前的场景 * @returns 实例化材质 */ createMatVideo(e, t) { let r = new Wt("standardMaterial", t); return this.matVideo(r, e, t), r.name = e.id, r; } /** * 根据前端UI的交互传过来的materialData修改material的属性(video); * @param videoMat 材质实例对象; * @param data 信息 * @param scene 当前场景 */ matVideo(e, t, r) { t.diffuseVideo && (e.diffuseTexture = new xc( "video", t.diffuseVideo, r )); let n = e.diffuseTexture; n.video.autoplay = !1, n.coordinatesIndex = 0, e.emissiveColor = Ne.White(); } /** * 修改实例材质信息 * @param mat 实例材质 * @param data 属性数据 */ chargeMaterialProp(e, t) { this.root.getAllScenes().forEach((r) => { let n = r.scene; t.type == "image" ? (e.diffuseColor && (e.diffuseColor = null), e.diffuseTexture && e.diffuseTexture.dispose(), e.emissiveColor && (e.diffuseColor = null), e.emissiveTexture && e.emissiveTexture.dispose(), this.matImage(e, t, n)) : (e.diffuseTexture && e.diffuseTexture.dispose(), this.matVideo(e, t, n)); }); } } var Rv = /* @__PURE__ */ ((A) => (A.Grad = "Grad", A.OutLine = "OutLine", A))(Rv || {}); function MSe(A, e, t) { const r = A.getScene(); An.ShadersStore[A.name + "VertexShader"] = `\r precision highp float;\r //attribute\r attribute vec3 position;\r attribute vec2 uv;\r attribute vec3 normal;\r //uniform\r uniform mat4 worldViewProjection;\r //varying\r varying vec3 vPosition;\r varying vec2 vUV;\r void main(void) {\r gl_Position = worldViewProjection*vec4(position,1.0);\r vPosition = position;\r vUV = uv;\r }\r `, An.ShadersStore[A.name + "ragmentShader"] = `\r precision highp float;\r varying vec3 vPosition;\r uniform float distance;\r uniform vec3 changeColor;\r uniform vec3 center;\r uniform float pow;\r float distanceTwoVec3(vec3 p1,vec3 p2){\r float d = sqrt((p1.x-p2.x)*(p1.x-p2.x)+(p1.y-p2.y)*(p1.y-p2.y)+(p1.z-p2.z)*(p1.z-p2.z));\r return d;\r }\r void main(void) {\r float alpha = distanceTwoVec3(center,vPosition)*0.8/distance;\r gl_FragColor = vec4(changeColor,pow(alpha,pow);\r }\r `; let n = new Zo( A.name, r, { vertex: "custom", fragment: "custom" }, { attributes: ["normals", "position", "uv"], uniforms: [ "world", "worldView", "worldViewProjection", "view", "projection" ], needAlphaBlending: !0, needAlphaTesting: !0 } ); n.name = "shader" + A.name; const i = Ne.FromHexString(e), s = t || 2, a = A.getBoundingInfo().boundingBox.center, f = A.getBoundingInfo().boundingBox.maximum, o = S.Distance(a, f); n.setFloat("distance", o), n.setColor3("changeColor", i), n.setVector3("center", a), n.setFloat("pow", s); const d = []; return d.push({ name: "pow", unifrom: s }), d.push({ name: "changeColor", unifrom: i }), { mat: n, rtList: d }; } function Cqe(A, e) { An.ShadersStore[A + "VertexShader"] = `\r precision highp float;\r //attribute\r attribute vec3 position;\r attribute vec2 uv;\r attribute vec3 normal;\r //uniform\r uniform mat4 worldViewProjection;\r //varying\r varying vec3 vPosition;\r varying vec2 vUV;\r varying vec3 vNormal;\r void main(void) {\r gl_Position = worldViewProjection*vec4(position,1.0);\r vPosition = position;\r vUV = uv;\r vNormal = normal;\r }\r `, An.ShadersStore[A + "FragmentShader"] = `\r precision highp float;\r varying vec3 vPosition;\r varying vec3 vNormal;\r uniform float distance;\r uniform vec3 changeColor;\r uniform vec3 center;\r uniform float power;\r uniform float k;\r uniform vec3 viewPoint;\r float distanceTwoVec3(vec3 p1,vec3 p2){\r float d = sqrt((p1.x-p2.x)*(p1.x-p2.x)+(p1.y-p2.y)*(p1.y-p2.y)+(p1.z-p2.z)*(p1.z-p2.z));\r return d;\r }\r void main(void) {\r float modelDot = abs(dot(viewPoint,vNormal)/(length(viewPoint)*length(vNormal)));\r float alpha = distanceTwoVec3(center,vPosition)*k /distance;\r vec3 color = changeColor;\r alpha = modelDot * pow(alpha,power);\r gl_FragColor = vec4(color,alpha);\r }\r `; let t = new Zo( "effectGlShader" + A, e, { vertex: A, fragment: A }, { attributes: ["normal", "position", "uv"], uniforms: [ "world", "worldView", "worldViewProjection", "view", "projection" ], needAlphaBlending: !0, needAlphaTesting: !0 } ); return t.onBind = (r) => { const n = r.getBoundingInfo().boundingBox, i = S.Normalize(n.centerWorld), s = S.Distance(n.centerWorld, n.maximumWorld); t.setFloat("distance", s), t.setVector3("center", i), t.setVector3("viewPoint", S.Normalize(r.getScene().activeCamera.position)), t.onBind = null; }, t.setFloat("distance", 20), t.setVector3("center", new S(10, 10 * Math.random(), 10)), t.setVector3("viewPoint", new S(10, 10 * Math.random(), 10)), t; } function Oqe(A, e) { return An.ShadersStore["gradient" + A + "VertexShader"] = `\r precision highp float;\r //attribute\r attribute vec3 position;\r attribute vec2 uv;\r attribute vec3 normal;\r //uniform\r uniform mat4 worldViewProjection;\r //varying\r varying vec3 vPosition;\r varying vec2 vUV;\r varying vec3 vNormal;\r void main(void) {\r gl_Position = worldViewProjection*vec4(position,1.0);\r vPosition = position;\r vUV = uv;\r vNormal = normal;\r }\r `, An.ShadersStore["gradient" + A + "FragmentShader"] = `\r precision highp float;\r varying vec3 vPosition;\r float distanceTwoVec3(vec3 p1,vec3 p2){\r float d = sqrt((p1.x-p2.x)*(p1.x-p2.x)+(p1.y-p2.y)*(p1.y-p2.y)+(p1.z-p2.z)*(p1.z-p2.z));\r return d;\r }\r uniform vec3 mainColor;\r uniform float maxHeight;\r uniform float minHeight;\r uniform float alphaK;\r void main(void) {\r float height = maxHeight - minHeight;\r vec3 color = mainColor;\r float alpha = (vPosition.y-minHeight)/height;\r gl_FragColor = vec4(color,alphaK+pow(alpha,0.5)*alpha*(1.0-alphaK));\r }\r `, new Zo( "effectGradAlpha" + A, e, { vertex: "gradient" + A, fragment: "gradient" + A }, { attributes: ["normal", "position", "uv"], uniforms: [ "world", "worldView", "worldViewProjection", "view", "projection" ], needAlphaBlending: !0, needAlphaTesting: !0 } ); } var zi = typeof globalThis < "u" ? globalThis : typeof window < "u" ? window : typeof global < "u" ? global : typeof self < "u" ? self : {}; function Dfe(A) { return A && A.__esModule && Object.prototype.hasOwnProperty.call(A, "default") ? A.default : A; } function jfe(A) { if (A.__esModule) return A; var e = A.default; if (typeof e == "function") { var t = function r() { return this instanceof r ? Reflect.construct(e, arguments, this.constructor) : e.apply(this, arguments); }; t.prototype = e.prototype; } else t = {}; return Object.defineProperty(t, "__esModule", { value: !0 }), Object.keys(A).forEach(function(r) { var n = Object.getOwnPropertyDescriptor(A, r); Object.defineProperty(t, r, n.get ? n : { enumerable: !0, get: function() { return A[r]; } }); }), t; } var wfe = { exports: {} }; const yqe = /* @__PURE__ */ jfe(Jge); (function(A, e) { (function(r, n) { A.exports = n(yqe); })(typeof self < "u" ? self : zi, (t) => ( /******/ (() => { var r = { /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[3].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[3].use[2]!../../../dev/sharedUiComponents/dist/colorPicker/colorPicker.scss": ( /*!**********************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[3].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[3].use[2]!../../../dev/sharedUiComponents/dist/colorPicker/colorPicker.scss ***! \**********************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.color-picker-container { width: 320px; height: 300px; background-color: white; display: grid; grid-template-columns: 100%; grid-template-rows: 50% 50px 60px 40px 1fr auto; font-family: "acumin-pro-condensed"; font-weight: normal; font-size: 14px; } .color-picker-container.with-hints { height: 380px; } .color-picker-container .color-picker-saturation { grid-row: 1; grid-column: 1; display: grid; grid-template-columns: 100%; grid-template-rows: 100%; position: relative; cursor: pointer; } .color-picker-container .color-picker-saturation .color-picker-saturation-white { grid-row: 1; grid-column: 1; background: -webkit-linear-gradient(to right, #fff, rgba(255, 255, 255, 0)); background: linear-gradient(to right, #fff, rgba(255, 255, 255, 0)); } .color-picker-container .color-picker-saturation .color-picker-saturation-black { grid-row: 1; grid-column: 1; background: -webkit-linear-gradient(to top, #000, rgba(0, 0, 0, 0)); background: linear-gradient(to top, #000, rgba(0, 0, 0, 0)); } .color-picker-container .color-picker-saturation .color-picker-saturation-cursor { pointer-events: none; width: 4px; height: 4px; box-shadow: 0 0 0 1.5px #fff, inset 0 0 1px 1px rgba(0, 0, 0, 0.3), 0 0 1px 2px rgba(0, 0, 0, 0.4); border-radius: 50%; transform: translate(-2px, -2px); position: absolute; } .color-picker-container .color-picker-hue { grid-row: 2; grid-column: 1; display: grid; margin: 10px; grid-template-columns: 24% 76%; grid-template-rows: 100%; } .color-picker-container .color-picker-hue .color-picker-hue-color { grid-row: 1; grid-column: 1; align-self: center; justify-self: center; width: 30px; height: 30px; border-radius: 15px; border: 1px solid black; } .color-picker-container .color-picker-hue .color-picker-hue-slider { grid-row: 1; grid-column: 2; align-self: center; height: 16px; position: relative; cursor: pointer; background: linear-gradient(to right, #f00 0%, #ff0 17%, #0f0 33%, #0ff 50%, #00f 67%, #f0f 83%, #f00 100%); background: -webkit-linear-gradient(to right, #f00 0%, #ff0 17%, #0f0 33%, #0ff 50%, #00f 67%, #f0f 83%, #f00 100%); } .color-picker-container .color-picker-hue .color-picker-hue-slider .color-picker-hue-cursor { pointer-events: none; width: 8px; height: 18px; transform: translate(-4px, -2px); background-color: rgb(248, 248, 248); box-shadow: 0 1px 4px 0 rgba(0, 0, 0, 0.37); position: absolute; } .color-picker-container .color-picker-component { display: grid; margin: 5px; grid-template-columns: 100%; grid-template-rows: 50% 50%; } .color-picker-container .color-picker-component .color-picker-component-value { justify-self: center; align-self: center; grid-row: 1; grid-column: 1; margin-bottom: 4px; } .color-picker-container .color-picker-component .color-picker-component-value input { width: 50px; } .color-picker-container .color-picker-component .color-picker-component-label { justify-self: center; align-self: center; grid-row: 2; grid-column: 1; color: black; } .color-picker-container .color-picker-rgb { grid-row: 3; grid-column: 1; display: grid; margin: 10px; grid-template-columns: 20% 6.66% 20% 6.66% 20% 6.66% 20%; grid-template-rows: 100%; } .color-picker-container .red { grid-row: 1; grid-column: 1; } .color-picker-container .green { grid-row: 1; grid-column: 3; } .color-picker-container .blue { grid-row: 1; grid-column: 5; } .color-picker-container .alpha { grid-row: 1; grid-column: 7; } .color-picker-container .alpha.grayed { opacity: 0.5; } .color-picker-container .color-picker-hex { grid-row: 4; grid-column: 1; display: grid; grid-template-columns: 20% 80%; grid-template-rows: 100%; } .color-picker-container .color-picker-hex .color-picker-hex-label { justify-self: center; align-self: center; grid-row: 1; grid-column: 1; margin-left: 10px; color: black; } .color-picker-container .color-picker-hex .color-picker-hex-value { justify-self: left; align-self: center; grid-row: 1; grid-column: 2; margin-left: 10px; margin-right: 10px; } .color-picker-container .color-picker-hex .color-picker-hex-value input { width: 70px; } .color-picker-container .color-picker-warning { color: black; font-size: 11px; padding: 4px; justify-self: left; align-self: center; grid-row: 6; grid-column: 1; }`, "", { version: 3, sources: ["webpack://./../../../dev/sharedUiComponents/dist/colorPicker/colorPicker.scss"], names: [], mappings: "AAAA;EACI,YAAA;EACA,aAAA;EACA,uBAAA;EACA,aAAA;EACA,2BAAA;EACA,+CAAA;EACA,mCAAA;EACA,mBAAA;EACA,eAAA;AACJ;AACI;EACI,aAAA;AACR;AAEI;EACI,WAAA;EACA,cAAA;EACA,aAAA;EACA,2BAAA;EACA,wBAAA;EACA,kBAAA;EACA,eAAA;AAAR;AAEQ;EACI,WAAA;EACA,cAAA;EAEA,2EAAA;EACA,mEAAA;AADZ;AAIQ;EACI,WAAA;EACA,cAAA;EAEA,mEAAA;EACA,2DAAA;AAHZ;AAMQ;EACI,oBAAA;EACA,UAAA;EACA,WAAA;EACA,kGACI;EAGJ,kBAAA;EACA,gCAAA;EACA,kBAAA;AAPZ;AAWI;EACI,WAAA;EACA,cAAA;EACA,aAAA;EACA,YAAA;EACA,8BAAA;EACA,wBAAA;AATR;AAWQ;EACI,WAAA;EACA,cAAA;EACA,kBAAA;EACA,oBAAA;EACA,WAAA;EACA,YAAA;EACA,mBAAA;EACA,uBAAA;AATZ;AAYQ;EACI,WAAA;EACA,cAAA;EACA,kBAAA;EACA,YAAA;EACA,kBAAA;EACA,eAAA;EAEA,2GAAA;EACA,mHAAA;AAXZ;AAaY;EACI,oBAAA;EACA,UAAA;EACA,YAAA;EACA,gCAAA;EACA,oCAAA;EACA,2CAAA;EACA,kBAAA;AAXhB;AAgBI;EACI,aAAA;EACA,WAAA;EACA,2BAAA;EACA,2BAAA;AAdR;AAgBQ;EACI,oBAAA;EACA,kBAAA;EACA,WAAA;EACA,cAAA;EACA,kBAAA;AAdZ;AAgBY;EACI,WAAA;AAdhB;AAkBQ;EACI,oBAAA;EACA,kBAAA;EACA,WAAA;EACA,cAAA;EACA,YAAA;AAhBZ;AAoBI;EACI,WAAA;EACA,cAAA;EACA,aAAA;EACA,YAAA;EACA,wDAAA;EACA,wBAAA;AAlBR;AAqBI;EACI,WAAA;EACA,cAAA;AAnBR;AAsBI;EACI,WAAA;EACA,cAAA;AApBR;AAuBI;EACI,WAAA;EACA,cAAA;AArBR;AAwBI;EACI,WAAA;EACA,cAAA;AAtBR;AAwBQ;EACI,YAAA;AAtBZ;AA0BI;EACI,WAAA;EACA,cAAA;EACA,aAAA;EACA,8BAAA;EACA,wBAAA;AAxBR;AA0BQ;EACI,oBAAA;EACA,kBAAA;EACA,WAAA;EACA,cAAA;EACA,iBAAA;EACA,YAAA;AAxBZ;AA2BQ;EACI,kBAAA;EACA,kBAAA;EACA,WAAA;EACA,cAAA;EACA,iBAAA;EACA,kBAAA;AAzBZ;AA2BY;EACI,WAAA;AAzBhB;AA8BI;EACI,YAAA;EACA,eAAA;EACA,YAAA;EACA,kBAAA;EACA,kBAAA;EACA,WAAA;EACA,cAAA;AA5BR", sourcesContent: [`.color-picker-container {\r width: 320px;\r height: 300px;\r background-color: white;\r display: grid;\r grid-template-columns: 100%;\r grid-template-rows: 50% 50px 60px 40px 1fr auto;\r font-family: "acumin-pro-condensed";\r font-weight: normal;\r font-size: 14px;\r \r &.with-hints {\r height: 380px;\r }\r \r .color-picker-saturation {\r grid-row: 1;\r grid-column: 1;\r display: grid;\r grid-template-columns: 100%;\r grid-template-rows: 100%;\r position: relative;\r cursor: pointer;\r \r .color-picker-saturation-white {\r grid-row: 1;\r grid-column: 1;\r \r background: -webkit-linear-gradient(to right, #fff, rgba(255, 255, 255, 0));\r background: linear-gradient(to right, #fff, rgba(255, 255, 255, 0));\r }\r \r .color-picker-saturation-black {\r grid-row: 1;\r grid-column: 1;\r \r background: -webkit-linear-gradient(to top, #000, rgba(0, 0, 0, 0));\r background: linear-gradient(to top, #000, rgba(0, 0, 0, 0));\r }\r \r .color-picker-saturation-cursor {\r pointer-events: none;\r width: 4px;\r height: 4px;\r box-shadow:\r 0 0 0 1.5px #fff,\r inset 0 0 1px 1px rgba(0, 0, 0, 0.3),\r 0 0 1px 2px rgba(0, 0, 0, 0.4);\r border-radius: 50%;\r transform: translate(-2px, -2px);\r position: absolute;\r }\r }\r \r .color-picker-hue {\r grid-row: 2;\r grid-column: 1;\r display: grid;\r margin: 10px;\r grid-template-columns: 24% 76%;\r grid-template-rows: 100%;\r \r .color-picker-hue-color {\r grid-row: 1;\r grid-column: 1;\r align-self: center;\r justify-self: center;\r width: 30px;\r height: 30px;\r border-radius: 15px;\r border: 1px solid black;\r }\r \r .color-picker-hue-slider {\r grid-row: 1;\r grid-column: 2;\r align-self: center;\r height: 16px;\r position: relative;\r cursor: pointer;\r \r background: linear-gradient(to right, #f00 0%, #ff0 17%, #0f0 33%, #0ff 50%, #00f 67%, #f0f 83%, #f00 100%);\r background: -webkit-linear-gradient(to right, #f00 0%, #ff0 17%, #0f0 33%, #0ff 50%, #00f 67%, #f0f 83%, #f00 100%);\r \r .color-picker-hue-cursor {\r pointer-events: none;\r width: 8px;\r height: 18px;\r transform: translate(-4px, -2px);\r background-color: rgb(248, 248, 248);\r box-shadow: 0 1px 4px 0 rgba(0, 0, 0, 0.37);\r position: absolute;\r }\r }\r }\r \r .color-picker-component {\r display: grid;\r margin: 5px;\r grid-template-columns: 100%;\r grid-template-rows: 50% 50%;\r \r .color-picker-component-value {\r justify-self: center;\r align-self: center;\r grid-row: 1;\r grid-column: 1;\r margin-bottom: 4px;\r \r input {\r width: 50px;\r }\r }\r \r .color-picker-component-label {\r justify-self: center;\r align-self: center;\r grid-row: 2;\r grid-column: 1;\r color: black;\r }\r }\r \r .color-picker-rgb {\r grid-row: 3;\r grid-column: 1;\r display: grid;\r margin: 10px;\r grid-template-columns: 20% 6.66% 20% 6.66% 20% 6.66% 20%;\r grid-template-rows: 100%;\r }\r \r .red {\r grid-row: 1;\r grid-column: 1;\r }\r \r .green {\r grid-row: 1;\r grid-column: 3;\r }\r \r .blue {\r grid-row: 1;\r grid-column: 5;\r }\r \r .alpha {\r grid-row: 1;\r grid-column: 7;\r \r &.grayed {\r opacity: 0.5;\r }\r }\r \r .color-picker-hex {\r grid-row: 4;\r grid-column: 1;\r display: grid;\r grid-template-columns: 20% 80%;\r grid-template-rows: 100%;\r \r .color-picker-hex-label {\r justify-self: center;\r align-self: center;\r grid-row: 1;\r grid-column: 1;\r margin-left: 10px;\r color: black;\r }\r \r .color-picker-hex-value {\r justify-self: left;\r align-self: center;\r grid-row: 1;\r grid-column: 2;\r margin-left: 10px;\r margin-right: 10px;\r \r input {\r width: 70px;\r }\r }\r }\r \r .color-picker-warning {\r color: black;\r font-size: 11px;\r padding: 4px;\r justify-self: left;\r align-self: center;\r grid-row: 6;\r grid-column: 1;\r }\r }\r `], sourceRoot: "" }]), P.locals = {}; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[3].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[3].use[2]!../../../dev/sharedUiComponents/dist/nodeGraphSystem/searchBox.scss": ( /*!************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[3].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[3].use[2]!../../../dev/sharedUiComponents/dist/nodeGraphSystem/searchBox.scss ***! \************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `#graph-search-container { position: absolute; width: 100%; height: 100%; left: 0; top: 0; } #graph-search-container #graph-search-picking-blocker { position: absolute; width: 100%; height: 100%; left: 0; top: 0; cursor: default; } #graph-search-container #graph-search-box { width: 300px; height: 400px; position: absolute; left: calc(50% - 150px); top: calc(50% - 200px); background: rgb(72, 72, 72); border: 2px solid black; z-index: 100; cursor: auto; display: grid; grid-template-rows: auto auto 1fr; } #graph-search-container #graph-search-box .graph-search-box-title { border-bottom: 1px solid rgb(40, 40, 40); margin: 4px; padding-bottom: 4px; color: white; grid-row: 1; } #graph-search-container #graph-search-box .graph-search-box-filter { margin: 4px; padding-bottom: 4px; border-radius: 4px; width: calc(100% - 16px); grid-row: 2; } #graph-search-container #graph-search-box .graph-search-box-list { grid-row: 3; overflow-y: auto; display: flex; flex-direction: column; } #graph-search-container #graph-search-box .graph-search-box-list .graph-search-box-list-item { margin-left: 4px; cursor: pointer; color: white; margin-top: 4px; margin-bottom: 4px; } #graph-search-container #graph-search-box .graph-search-box-list .graph-search-box-list-item:hover { background: rgb(30, 30, 30); } #graph-search-container #graph-search-box .graph-search-box-list .graph-search-box-list-item.selected { background: rgb(230, 230, 230); color: black; }`, "", { version: 3, sources: ["webpack://./../../../dev/sharedUiComponents/dist/nodeGraphSystem/searchBox.scss"], names: [], mappings: "AAAA;EACI,kBAAA;EACA,WAAA;EACA,YAAA;EACA,OAAA;EACA,MAAA;AACJ;AACI;EACI,kBAAA;EACA,WAAA;EACA,YAAA;EACA,OAAA;EACA,MAAA;EACA,eAAA;AACR;AAEI;EACI,YAAA;EACA,aAAA;EACA,kBAAA;EACA,uBAAA;EACA,sBAAA;EACA,2BAAA;EACA,uBAAA;EACA,YAAA;EACA,YAAA;EACA,aAAA;EACA,iCAAA;AAAR;AAEQ;EACI,wCAAA;EACA,WAAA;EACA,mBAAA;EACA,YAAA;EACA,WAAA;AAAZ;AAGQ;EACI,WAAA;EACA,mBAAA;EACA,kBAAA;EACA,wBAAA;EACA,WAAA;AADZ;AAIQ;EACI,WAAA;EACA,gBAAA;EACA,aAAA;EACA,sBAAA;AAFZ;AAIY;EACI,gBAAA;EACA,eAAA;EACA,YAAA;EACA,eAAA;EACA,kBAAA;AAFhB;AAIgB;EACI,2BAAA;AAFpB;AAKgB;EACI,8BAAA;EACA,YAAA;AAHpB", sourcesContent: [`#graph-search-container {\r position: absolute;\r width: 100%;\r height: 100%;\r left: 0;\r top: 0;\r \r #graph-search-picking-blocker {\r position: absolute;\r width: 100%;\r height: 100%;\r left: 0;\r top: 0;\r cursor: default;\r }\r \r #graph-search-box {\r width: 300px;\r height: 400px;\r position: absolute;\r left: calc(50% - 150px);\r top: calc(50% - 200px);\r background: rgb(72, 72, 72);\r border: 2px solid black;\r z-index: 100;\r cursor: auto;\r display: grid;\r grid-template-rows: auto auto 1fr;\r \r .graph-search-box-title {\r border-bottom: 1px solid rgb(40, 40, 40);\r margin: 4px;\r padding-bottom: 4px;\r color: white;\r grid-row: 1;\r }\r \r .graph-search-box-filter {\r margin: 4px;\r padding-bottom: 4px;\r border-radius: 4px;\r width: calc(100% - 16px);\r grid-row: 2;\r }\r \r .graph-search-box-list {\r grid-row: 3;\r overflow-y: auto;\r display: flex;\r flex-direction: column;\r \r .graph-search-box-list-item {\r margin-left: 4px;\r cursor: pointer;\r color: white;\r margin-top: 4px;\r margin-bottom: 4px;\r \r &:hover {\r background: rgb(30, 30, 30);\r }\r \r &.selected {\r background: rgb(230, 230, 230);\r color: black;\r }\r }\r }\r }\r }\r `], sourceRoot: "" }]), P.locals = {}; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[3].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[3].use[2]!../../../tools/nodeEditor/dist/components/log/log.scss": ( /*!***********************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[3].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[3].use[2]!../../../tools/nodeEditor/dist/components/log/log.scss ***! \***********************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `#nme-log-console { background: #333333; height: 120px; box-sizing: border-box; margin: 0; padding: 10px; width: 100%; overflow: hidden; overflow-y: auto; grid-row: 2; grid-column: 3; } #nme-log-console .log { color: white; font-size: 14px; font-family: "Courier New", Courier, monospace; } #nme-log-console .log.error { color: red; }`, "", { version: 3, sources: ["webpack://./../../../tools/nodeEditor/dist/components/log/log.scss"], names: [], mappings: "AAAA;EACI,mBAAA;EACA,aAAA;EACA,sBAAA;EACA,SAAA;EACA,aAAA;EACA,WAAA;EACA,gBAAA;EACA,gBAAA;EACA,WAAA;EACA,cAAA;AACJ;AACI;EACI,YAAA;EACA,eAAA;EACA,8CAAA;AACR;AACQ;EACI,UAAA;AACZ", sourcesContent: [`#nme-log-console {\r background: #333333;\r height: 120px;\r box-sizing: border-box;\r margin: 0;\r padding: 10px;\r width: 100%;\r overflow: hidden;\r overflow-y: auto;\r grid-row: 2;\r grid-column: 3;\r \r .log {\r color: white;\r font-size: 14px;\r font-family: "Courier New", Courier, monospace;\r \r &.error {\r color: red;\r }\r }\r }\r `], sourceRoot: "" }]), P.locals = {}; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[3].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[3].use[2]!../../../tools/nodeEditor/dist/components/nodeList/nodeList.scss": ( /*!*********************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[3].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[3].use[2]!../../../tools/nodeEditor/dist/components/nodeList/nodeList.scss ***! \*********************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `#nmeNodeList { background: #333333; height: 100%; margin: 0; padding: 0; display: grid; width: 100%; overflow: hidden; } #nmeNodeList .panes { overflow: hidden; } #nmeNodeList .panes .pane { color: white; overflow: hidden; height: 100%; -webkit-user-select: none; -moz-user-select: none; -ms-user-select: none; user-select: none; } #nmeNodeList .panes .pane .filter { display: flex; align-items: stretch; } #nmeNodeList .panes .pane .filter input { width: 100%; margin: 10px 10px 5px 10px; display: block; border: none; padding: 0; border-bottom: solid 1px rgb(51, 122, 183); background: linear-gradient(to bottom, rgba(255, 255, 255, 0) 96%, rgb(51, 122, 183) 4%); background-position: -1000px 0; background-size: 1000px 100%; background-repeat: no-repeat; color: white; } #nmeNodeList .panes .pane .filter input:focus { box-shadow: none; outline: none; background-position: 0 0; } #nmeNodeList .panes .pane .filter input::placeholder { color: gray; } #nmeNodeList .panes .pane .list-container { overflow-x: hidden; overflow-y: auto; height: calc(100% - 32px); } #nmeNodeList .panes .pane .list-container .underline { border-bottom: 0.5px solid rgba(255, 255, 255, 0.5); } #nmeNodeList .panes .pane .list-container .draggableLine { height: 30px; display: grid; align-items: center; justify-items: stretch; background: #222222; cursor: grab; text-align: center; margin: 0; box-sizing: border-box; } #nmeNodeList .panes .pane .list-container .draggableLine:hover { background: rgb(51, 122, 183); color: white; } #nmeNodeList .panes .pane .list-container .nonDraggableLine { height: 30px; display: grid; align-items: center; justify-items: stretch; background: #222222; text-align: center; margin: 0; box-sizing: border-box; } #nmeNodeList .panes .pane .list-container .withButton { height: 30px; position: relative; } #nmeNodeList .panes .pane .list-container .withButton .icon { position: absolute; right: 4px; top: 5px; } #nmeNodeList .panes .pane .list-container .withButton .icon:hover { cursor: pointer; } #nmeNodeList .panes .pane .list-container .withButton .icon .img { height: 17px; width: 17px; } #nmeNodeList .panes .pane .list-container .withButton .buttonLine { height: 30px; display: grid; align-items: center; justify-items: stretch; padding-bottom: 5px; position: absolute; right: 0px; top: 2px; } #nmeNodeList .panes .pane .list-container .withButton .buttonLine input[type=file] { display: none; } #nmeNodeList .panes .pane .list-container .withButton .buttonLine .file-upload { background: transparent; border: transparent; padding: 15px 200px; opacity: 0.9; cursor: pointer; text-align: center; } #nmeNodeList .panes .pane .list-container .withButton .buttonLine .file-upload:hover { opacity: 1; } #nmeNodeList .panes .pane .list-container .withButton .buttonLine .file-upload:active { transform: scale(0.98); transform-origin: 0.5 0.5; } #nmeNodeList .panes .pane .list-container .withButton .buttonLine button { background: transparent; border: transparent; margin: 5px 10px 5px 10px; color: white; padding: 4px 5px; opacity: 0.9; } #nmeNodeList .panes .pane .list-container .withButton .buttonLine button:hover { opacity: 0; } #nmeNodeList .panes .pane .list-container .withButton .buttonLine button:active { background: transparent; } #nmeNodeList .panes .pane .list-container .withButton .buttonLine button:focus { border: transparent; outline: 0px; } #nmeNodeList .panes .pane .list-container .paneContainer { margin-top: 3px; display: grid; grid-template-rows: 100%; grid-template-columns: 100%; } #nmeNodeList .panes .pane .list-container .paneContainer .paneContainer-content { grid-row: 1; grid-column: 1; } #nmeNodeList .panes .pane .list-container .paneContainer .paneContainer-content .header { display: grid; grid-template-columns: 1fr auto; background: #555555; height: 30px; padding-right: 5px; cursor: pointer; } #nmeNodeList .panes .pane .list-container .paneContainer .paneContainer-content .header .title { border-left: 3px solid transparent; padding-left: 5px; grid-column: 1; display: flex; align-items: center; } #nmeNodeList .panes .pane .list-container .paneContainer .paneContainer-content .header .collapse { grid-column: 2; display: flex; align-items: center; justify-items: center; transform-origin: center; } #nmeNodeList .panes .pane .list-container .paneContainer .paneContainer-content .header .collapse.closed { transform: rotate(180deg); } #nmeNodeList .panes .pane .list-container .paneContainer .paneContainer-content .paneList > div:not(:last-child) { border-bottom: 1px solid rgba(255, 255, 255, 0.3); }`, "", { version: 3, sources: ["webpack://./../../../tools/nodeEditor/dist/components/nodeList/nodeList.scss"], names: [], mappings: "AAAA;EACI,mBAAA;EACA,YAAA;EACA,SAAA;EACA,UAAA;EACA,aAAA;EACA,WAAA;EACA,gBAAA;AACJ;AACI;EACI,gBAAA;AACR;AACQ;EACI,YAAA;EAEA,gBAAA;EACA,YAAA;EAEA,yBAAA;EACA,sBAAA;EACA,qBAAA;EACA,iBAAA;AADZ;AAGY;EACI,aAAA;EACA,oBAAA;AADhB;AAGgB;EACI,WAAA;EACA,0BAAA;EACA,cAAA;EACA,YAAA;EACA,UAAA;EACA,0CAAA;EACA,wFAAA;EACA,8BAAA;EACA,4BAAA;EACA,4BAAA;EACA,YAAA;AADpB;AAIgB;EACI,gBAAA;EACA,aAAA;EACA,wBAAA;AAFpB;AAKgB;EACI,WAAA;AAHpB;AAOY;EACI,kBAAA;EACA,gBAAA;EACA,yBAAA;AALhB;AAOgB;EACI,mDAAA;AALpB;AAQgB;EACI,YAAA;EACA,aAAA;EACA,mBAAA;EACA,sBAAA;EACA,mBAAA;EACA,YAAA;EACA,kBAAA;EACA,SAAA;EACA,sBAAA;AANpB;AAQoB;EACI,6BAAA;EACA,YAAA;AANxB;AAUgB;EACI,YAAA;EACA,aAAA;EACA,mBAAA;EACA,sBAAA;EACA,mBAAA;EACA,kBAAA;EACA,SAAA;EACA,sBAAA;AARpB;AAWgB;EACI,YAAA;EACA,kBAAA;AATpB;AAUoB;EACI,kBAAA;EACA,UAAA;EACA,QAAA;AARxB;AASwB;EACI,eAAA;AAP5B;AAUwB;EACI,YAAA;EACA,WAAA;AAR5B;AAYoB;EACI,YAAA;EACA,aAAA;EACA,mBAAA;EACA,sBAAA;EACA,mBAAA;EACA,kBAAA;EACA,UAAA;EACA,QAAA;AAVxB;AAWwB;EACI,aAAA;AAT5B;AAYwB;EACI,uBAAA;EACA,mBAAA;EACA,mBAAA;EACA,YAAA;EACA,eAAA;EACA,kBAAA;AAV5B;AAawB;EACI,UAAA;AAX5B;AAcwB;EACI,sBAAA;EACA,yBAAA;AAZ5B;AAewB;EACI,uBAAA;EACA,mBAAA;EACA,yBAAA;EACA,YAAA;EACA,gBAAA;EACA,YAAA;AAb5B;AAgBwB;EACI,UAAA;AAd5B;AAiBwB;EACI,uBAAA;AAf5B;AAkBwB;EACI,mBAAA;EACA,YAAA;AAhB5B;AAqBgB;EACI,eAAA;EACA,aAAA;EACA,wBAAA;EACA,2BAAA;AAnBpB;AAqBoB;EACI,WAAA;EACA,cAAA;AAnBxB;AAqBwB;EACI,aAAA;EACA,+BAAA;EACA,mBAAA;EACA,YAAA;EACA,kBAAA;EACA,eAAA;AAnB5B;AAqB4B;EACI,kCAAA;EACA,iBAAA;EACA,cAAA;EACA,aAAA;EACA,mBAAA;AAnBhC;AAsB4B;EACI,cAAA;EACA,aAAA;EACA,mBAAA;EACA,qBAAA;EACA,wBAAA;AApBhC;AAsBgC;EACI,yBAAA;AApBpC;AAyBwB;EACI,iDAAA;AAvB5B", sourcesContent: [`#nmeNodeList {\r background: #333333;\r height: 100%;\r margin: 0;\r padding: 0;\r display: grid;\r width: 100%;\r overflow: hidden;\r \r .panes {\r overflow: hidden;\r \r .pane {\r color: white;\r \r overflow: hidden;\r height: 100%;\r \r -webkit-user-select: none;\r -moz-user-select: none;\r -ms-user-select: none;\r user-select: none;\r \r .filter {\r display: flex;\r align-items: stretch;\r \r input {\r width: 100%;\r margin: 10px 10px 5px 10px;\r display: block;\r border: none;\r padding: 0;\r border-bottom: solid 1px rgb(51, 122, 183);\r background: linear-gradient(to bottom, rgba(255, 255, 255, 0) 96%, rgb(51, 122, 183) 4%);\r background-position: -1000px 0;\r background-size: 1000px 100%;\r background-repeat: no-repeat;\r color: white;\r }\r \r input:focus {\r box-shadow: none;\r outline: none;\r background-position: 0 0;\r }\r \r input::placeholder {\r color: gray;\r }\r }\r \r .list-container {\r overflow-x: hidden;\r overflow-y: auto;\r height: calc(100% - 32px);\r \r .underline {\r border-bottom: 0.5px solid rgba(255, 255, 255, 0.5);\r }\r \r .draggableLine {\r height: 30px;\r display: grid;\r align-items: center;\r justify-items: stretch;\r background: #222222;\r cursor: grab;\r text-align: center;\r margin: 0;\r box-sizing: border-box;\r \r &:hover {\r background: rgb(51, 122, 183);\r color: white;\r }\r }\r \r .nonDraggableLine {\r height: 30px;\r display: grid;\r align-items: center;\r justify-items: stretch;\r background: #222222;\r text-align: center;\r margin: 0;\r box-sizing: border-box;\r }\r \r .withButton {\r height: 30px;\r position: relative;\r .icon {\r position: absolute;\r right: 4px;\r top: 5px;\r &:hover {\r cursor: pointer;\r }\r \r .img {\r height: 17px;\r width: 17px;\r }\r }\r \r .buttonLine {\r height: 30px;\r display: grid;\r align-items: center;\r justify-items: stretch;\r padding-bottom: 5px;\r position: absolute;\r right: 0px;\r top: 2px;\r input[type="file"] {\r display: none;\r }\r \r .file-upload {\r background: transparent;\r border: transparent;\r padding: 15px 200px;\r opacity: 0.9;\r cursor: pointer;\r text-align: center;\r }\r \r .file-upload:hover {\r opacity: 1;\r }\r \r .file-upload:active {\r transform: scale(0.98);\r transform-origin: 0.5 0.5;\r }\r \r button {\r background: transparent;\r border: transparent;\r margin: 5px 10px 5px 10px;\r color: white;\r padding: 4px 5px;\r opacity: 0.9;\r }\r \r button:hover {\r opacity: 0;\r }\r \r button:active {\r background: transparent;\r }\r \r button:focus {\r border: transparent;\r outline: 0px;\r }\r }\r }\r \r .paneContainer {\r margin-top: 3px;\r display: grid;\r grid-template-rows: 100%;\r grid-template-columns: 100%;\r \r .paneContainer-content {\r grid-row: 1;\r grid-column: 1;\r \r .header {\r display: grid;\r grid-template-columns: 1fr auto;\r background: #555555;\r height: 30px;\r padding-right: 5px;\r cursor: pointer;\r \r .title {\r border-left: 3px solid transparent;\r padding-left: 5px;\r grid-column: 1;\r display: flex;\r align-items: center;\r }\r \r .collapse {\r grid-column: 2;\r display: flex;\r align-items: center;\r justify-items: center;\r transform-origin: center;\r \r &.closed {\r transform: rotate(180deg);\r }\r }\r }\r \r .paneList > div:not(:last-child) {\r border-bottom: 1px solid rgba(255, 255, 255, 0.3);\r }\r }\r }\r }\r }\r }\r }\r `], sourceRoot: "" }]), P.locals = {}; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[3].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[3].use[2]!../../../tools/nodeEditor/dist/components/propertyTab/propertyTab.scss": ( /*!***************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[3].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[3].use[2]!../../../tools/nodeEditor/dist/components/propertyTab/propertyTab.scss ***! \***************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.nme-right-panel #propertyTab { color: white; background: #333333; } .nme-right-panel #propertyTab #header { height: 30px; font-size: 16px; color: white; background: #222222; grid-row: 1; text-align: center; display: grid; grid-template-columns: 30px 1fr; -webkit-user-select: none; -moz-user-select: none; -ms-user-select: none; user-select: none; } .nme-right-panel #propertyTab #header #logo { position: relative; grid-column: 1; width: 24px; height: 24px; left: 0; display: flex; align-self: center; justify-self: center; } .nme-right-panel #propertyTab #header #title { grid-column: 2; display: grid; align-items: center; text-align: center; } .nme-right-panel #propertyTab .range { -webkit-appearance: none; width: 120px; height: 6px; background: #d3d3d3; border-radius: 5px; outline: none; opacity: 0.7; -webkit-transition: 0.2s; transition: opacity 0.2s; } .nme-right-panel #propertyTab .range:hover { opacity: 1; } .nme-right-panel #propertyTab .range::-webkit-slider-thumb { -webkit-appearance: none; appearance: none; width: 14px; height: 14px; border-radius: 50%; background: rgb(51, 122, 183); cursor: pointer; } .nme-right-panel #propertyTab .range::-moz-range-thumb { width: 14px; height: 14px; border-radius: 50%; background: rgb(51, 122, 183); cursor: pointer; } .nme-right-panel #propertyTab input[type=color] { -webkit-appearance: none; border: 1px solid rgba(255, 255, 255, 0.5); padding: 0; width: 30px; height: 20px; } .nme-right-panel #propertyTab input[type=color]::-webkit-color-swatch-wrapper { padding: 0; } .nme-right-panel #propertyTab input[type=color]::-webkit-color-swatch { border: none; } .nme-right-panel #propertyTab .sliderLine { padding-left: 5px; height: 30px; display: grid; grid-template-rows: 100%; grid-template-columns: 1fr 50px auto; } .nme-right-panel #propertyTab .sliderLine .label { grid-column: 1; display: flex; align-items: center; } .nme-right-panel #propertyTab .sliderLine .slider { grid-column: 3; grid-row: 1; margin-right: 5px; width: 90%; display: flex; align-items: center; } .nme-right-panel #propertyTab .sliderLine .floatLine { grid-column: 2; padding-left: 5px; } .nme-right-panel #propertyTab .sliderLine .floatLine .label { grid-column: 1; display: flex; align-items: center; } .nme-right-panel #propertyTab .sliderLine .floatLine .short { grid-column: 1; display: flex; align-items: center; } .nme-right-panel #propertyTab .sliderLine .floatLine .short input { width: 35px; } .nme-right-panel #propertyTab .sliderLine .floatLine .short input::-webkit-outer-spin-button, .nme-right-panel #propertyTab .sliderLine .floatLine .short input::-webkit-inner-spin-button { -webkit-appearance: none; margin: 0; } .nme-right-panel #propertyTab .sliderLine .floatLine .short input[type=number] { -moz-appearance: textfield; } .nme-right-panel #propertyTab .textInputLine { padding-left: 5px; height: 30px; display: grid; grid-template-columns: 1fr 120px auto; } .nme-right-panel #propertyTab .textInputLine .label { grid-column: 1; display: flex; align-items: center; } .nme-right-panel #propertyTab .textInputLine .value { display: flex; align-items: center; grid-column: 2; } .nme-right-panel #propertyTab .textInputLine .value input { width: calc(100% - 5px); margin-right: 5px; } .nme-right-panel #propertyTab .textInputArea { padding-left: 5px; height: 50px; display: grid; grid-template-columns: 1fr 120px; } .nme-right-panel #propertyTab .textInputArea .label { grid-column: 1; display: flex; align-items: center; } .nme-right-panel #propertyTab .textInputArea textarea { margin-right: 5px; margin-left: -50%; height: 40px; resize: none; } .nme-right-panel #propertyTab .textInputArea .value { display: flex; align-items: center; grid-column: 2; } .nme-right-panel #propertyTab .paneContainer { margin-top: 3px; display: grid; grid-template-rows: 100%; grid-template-columns: 100%; } .nme-right-panel #propertyTab .paneContainer .paneList { border-left: 3px solid transparent; } .nme-right-panel #propertyTab .paneContainer:hover .paneList { border-left: 3px solid rgba(51, 122, 183, 0.8); } .nme-right-panel #propertyTab .paneContainer:hover .paneContainer-content .header .title { border-left: 3px solid rgb(51, 122, 183); } .nme-right-panel #propertyTab .paneContainer .paneContainer-highlight-border { grid-row: 1; grid-column: 1; opacity: 1; border: 3px solid red; transition: opacity 250ms; pointer-events: none; } .nme-right-panel #propertyTab .paneContainer .paneContainer-highlight-border.transparent { opacity: 0; } .nme-right-panel #propertyTab .paneContainer .paneContainer-content { grid-row: 1; grid-column: 1; } .nme-right-panel #propertyTab .paneContainer .paneContainer-content .header { display: grid; grid-template-columns: 1fr auto; background: #555555; height: 30px; padding-right: 5px; cursor: pointer; } .nme-right-panel #propertyTab .paneContainer .paneContainer-content .header .title { border-left: 3px solid transparent; padding-left: 5px; grid-column: 1; display: flex; align-items: center; } .nme-right-panel #propertyTab .paneContainer .paneContainer-content .header .collapse { grid-column: 2; display: flex; align-items: center; justify-items: center; transform-origin: center; } .nme-right-panel #propertyTab .paneContainer .paneContainer-content .header .collapse.closed { transform: rotate(180deg); } .nme-right-panel #propertyTab .paneContainer .paneContainer-content .paneList > div:not(:last-child) { border-bottom: 0.5px solid rgba(255, 255, 255, 0.1); } .nme-right-panel #propertyTab .paneContainer .paneContainer-content .fragment > div:not(:last-child) { border-bottom: 0.5px solid rgba(255, 255, 255, 0.1); } .nme-right-panel #propertyTab .color-picker { height: calc(100% - 8px); margin: 4px; width: calc(100% - 8px); } .nme-right-panel #propertyTab .color-picker .color-rect { height: calc(100% - 4px); border: 2px white solid; cursor: pointer; min-height: 18px; } .nme-right-panel #propertyTab .color-picker .color-picker-cover { position: fixed; top: 0px; right: 0px; bottom: 0px; left: 0px; z-index: 1; } .nme-right-panel #propertyTab .color-picker .color-picker-float { z-index: 2; position: absolute; } .nme-right-panel #propertyTab .gradient-step { display: grid; grid-template-rows: 100%; grid-template-columns: 20px 30px 40px auto 20px 30px; padding-top: 5px; padding-left: 5px; padding-bottom: 5px; } .nme-right-panel #propertyTab .gradient-step .step { grid-row: 1; grid-column: 1; } .nme-right-panel #propertyTab .gradient-step .color { grid-row: 1; grid-column: 2; cursor: pointer; } .nme-right-panel #propertyTab .gradient-step .step-value { margin-left: 5px; grid-row: 1; grid-column: 3; text-align: right; margin-right: 5px; } .nme-right-panel #propertyTab .gradient-step .step-slider { grid-row: 1; grid-column: 4; display: grid; justify-content: stretch; align-content: center; margin-right: -5px; padding-left: 12px; } .nme-right-panel #propertyTab .gradient-step .step-slider input { width: 90%; } .nme-right-panel #propertyTab .gradient-step .gradient-copy { grid-row: 1; grid-column: 5; display: grid; align-content: center; justify-content: center; } .nme-right-panel #propertyTab .gradient-step .gradient-copy .img { height: 20px; width: 20px; } .nme-right-panel #propertyTab .gradient-step .gradient-copy .img:hover { cursor: pointer; } .nme-right-panel #propertyTab .gradient-step .gradient-delete { grid-row: 1; grid-column: 6; display: grid; align-content: center; justify-content: center; } .nme-right-panel #propertyTab .gradient-step .gradient-delete .img { height: 20px; width: 20px; } .nme-right-panel #propertyTab .gradient-step .gradient-delete .img:hover { cursor: pointer; } .nme-right-panel #propertyTab .floatLine { padding-left: 5px; height: 30px; display: grid; grid-template-columns: 1fr 120px; } .nme-right-panel #propertyTab .floatLine .label { grid-column: 1; display: flex; align-items: center; } .nme-right-panel #propertyTab .floatLine .value { grid-column: 2; display: flex; align-items: center; } .nme-right-panel #propertyTab .floatLine .value input { width: 110px; } .nme-right-panel #propertyTab .floatLine .short { grid-column: 2; display: flex; align-items: center; } .nme-right-panel #propertyTab .floatLine .short input { width: 27px; } .nme-right-panel #propertyTab .floatLine .short input::-webkit-outer-spin-button, .nme-right-panel #propertyTab .floatLine .short input::-webkit-inner-spin-button { -webkit-appearance: none; margin: 0; } .nme-right-panel #propertyTab .floatLine .short input[type=number] { -moz-appearance: textfield; } .nme-right-panel #propertyTab .vector3Line { padding-left: 5px; display: grid; } .nme-right-panel #propertyTab .vector3Line .firstLine { display: grid; grid-template-columns: 1fr auto 20px; height: 30px; } .nme-right-panel #propertyTab .vector3Line .firstLine .label { grid-column: 1; display: flex; align-items: center; } .nme-right-panel #propertyTab .vector3Line .firstLine .vector { grid-column: 2; display: flex; align-items: center; text-align: right; opacity: 0.8; } .nme-right-panel #propertyTab .vector3Line .firstLine .expand { grid-column: 3; display: grid; align-items: center; justify-items: center; cursor: pointer; } .nme-right-panel #propertyTab .vector3Line .secondLine { display: grid; padding-right: 5px; border-left: 1px solid rgb(51, 122, 183); } .nme-right-panel #propertyTab .vector3Line .secondLine .no-right-margin { margin-right: 0; } .nme-right-panel #propertyTab .vector3Line .secondLine .numeric { display: grid; grid-template-columns: 1fr auto; } .nme-right-panel #propertyTab .vector3Line .secondLine .numeric-label { text-align: right; grid-column: 1; display: flex; align-items: center; justify-self: right; margin-right: 10px; } .nme-right-panel #propertyTab .vector3Line .secondLine .numeric-value { width: 120px; grid-column: 2; display: flex; align-items: center; border: 1px solid rgb(51, 122, 183); } .nme-right-panel #propertyTab .buttonLine { height: 30px; display: grid; align-items: center; justify-items: stretch; padding-bottom: 5px; } .nme-right-panel #propertyTab .buttonLine.disabled { opacity: 0.3; } .nme-right-panel #propertyTab .buttonLine input[type=file] { display: none; } .nme-right-panel #propertyTab .buttonLine .file-upload { background: #222222; border: 1px solid rgb(51, 122, 183); margin: 5px 10px; color: white; padding: 4px 5px; padding-top: 0px; opacity: 0.9; cursor: pointer; text-align: center; } .nme-right-panel #propertyTab .buttonLine .file-upload:hover { opacity: 1; } .nme-right-panel #propertyTab .buttonLine .file-upload:active { transform: scale(0.98); transform-origin: 0.5 0.5; } .nme-right-panel #propertyTab .buttonLine button { background: #222222; border: 1px solid rgb(51, 122, 183); margin: 5px 10px 5px 10px; color: white; padding: 4px 5px; opacity: 0.9; } .nme-right-panel #propertyTab .buttonLine button:hover { opacity: 1; } .nme-right-panel #propertyTab .buttonLine button:active { background: #282828; } .nme-right-panel #propertyTab .buttonLine button:focus { border: 1px solid rgb(51, 122, 183); outline: 0px; } .nme-right-panel #propertyTab .checkBoxLine { padding-left: 5px; height: 30px; display: grid; grid-template-columns: 1fr auto; } .nme-right-panel #propertyTab .checkBoxLine .label { grid-column: 1; display: flex; align-items: center; } .nme-right-panel #propertyTab .checkBoxLine .checkBox { grid-column: 2; display: flex; align-items: center; } .nme-right-panel #propertyTab .checkBoxLine .checkBox .lbl { position: relative; display: block; height: 14px; width: 34px; margin-right: 5px; background: #898989; border-radius: 100px; cursor: pointer; transition: all 0.3s ease; } .nme-right-panel #propertyTab .checkBoxLine .checkBox .lbl:after { position: absolute; left: 3px; top: 2px; display: block; width: 10px; height: 10px; border-radius: 100px; background: #fff; box-shadow: 0px 3px 3px rgba(0, 0, 0, 0.05); content: ""; transition: all 0.15s ease; } .nme-right-panel #propertyTab .checkBoxLine .checkBox .lbl:active:after { transform: scale(1.15, 0.85); } .nme-right-panel #propertyTab .checkBoxLine .checkBox .cbx:checked ~ label { background: rgb(51, 122, 183); } .nme-right-panel #propertyTab .checkBoxLine .checkBox .cbx:checked ~ label:after { left: 20px; background: rgb(22, 73, 117); } .nme-right-panel #propertyTab .checkBoxLine .checkBox .cbx:checked ~ label.disabled { background: rgb(22, 73, 117); cursor: pointer; } .nme-right-panel #propertyTab .checkBoxLine .checkBox .cbx:checked ~ label.disabled:after { left: 20px; background: rgb(85, 85, 85); cursor: pointer; } .nme-right-panel #propertyTab .checkBoxLine .checkBox .cbx ~ label.disabled { background: rgb(85, 85, 85); cursor: pointer; } .nme-right-panel #propertyTab .checkBoxLine .checkBox .hidden { display: none; } .nme-right-panel #propertyTab .listLine { padding-left: 5px; height: 30px; display: grid; grid-template-columns: 1fr auto; } .nme-right-panel #propertyTab .listLine .label { grid-column: 1; display: flex; align-items: center; } .nme-right-panel #propertyTab .listLine .options { grid-column: 2; display: flex; align-items: center; margin-right: 5px; } .nme-right-panel #propertyTab .listLine .options select { width: 115px; } .nme-right-panel #propertyTab .color3Line { padding-left: 5px; display: grid; } .nme-right-panel #propertyTab .color3Line .firstLine { height: 30px; display: grid; grid-template-columns: 1fr auto 0px 20px 20px; } .nme-right-panel #propertyTab .color3Line .firstLine .label { grid-column: 1; display: flex; align-items: center; } .nme-right-panel #propertyTab .color3Line .firstLine .textInputLine { display: none; } .nme-right-panel #propertyTab .color3Line .firstLine .color3 { grid-column: 2; width: 50px; display: flex; align-items: center; } .nme-right-panel #propertyTab .color3Line .firstLine .color3 input { margin-right: 5px; } .nme-right-panel #propertyTab .color3Line .firstLine .copy { grid-column: 4; display: grid; align-items: center; justify-items: center; cursor: pointer; } .nme-right-panel #propertyTab .color3Line .firstLine .copy img { height: 100%; width: 24px; } .nme-right-panel #propertyTab .color3Line .firstLine .expand { grid-column: 5; display: grid; align-items: center; justify-items: center; cursor: pointer; } .nme-right-panel #propertyTab .color3Line .firstLine .expand img { height: 100%; width: 20px; } .nme-right-panel #propertyTab .color3Line .secondLine { display: grid; padding-right: 5px; border-left: 1px solid rgb(51, 122, 183); } .nme-right-panel #propertyTab .color3Line .secondLine .numeric { display: grid; grid-template-columns: 1fr auto; } .nme-right-panel #propertyTab .color3Line .secondLine .numeric-label { text-align: right; grid-column: 1; display: flex; align-items: center; justify-self: right; margin-right: 10px; } .nme-right-panel #propertyTab .color3Line .secondLine .numeric-value { width: 120px; grid-column: 2; display: flex; align-items: center; border: 1px solid rgb(51, 122, 183); } .nme-right-panel #propertyTab .textLine { padding-left: 5px; height: 30px; display: grid; grid-template-columns: 1fr auto; } .nme-right-panel #propertyTab .textLine .label { grid-column: 1; display: flex; align-items: center; } .nme-right-panel #propertyTab .textLine .link-value { grid-column: 2; white-space: nowrap; text-overflow: ellipsis; overflow: hidden; text-align: end; opacity: 0.8; margin: 5px; margin-top: 6px; max-width: 140px; text-decoration: underline; cursor: pointer; } .nme-right-panel #propertyTab .textLine .value { grid-column: 2; white-space: nowrap; text-overflow: ellipsis; overflow: hidden; text-align: end; opacity: 0.8; margin: 5px; margin-top: 6px; max-width: 200px; -webkit-user-select: text; -moz-user-select: text; -ms-user-select: text; user-select: text; } .nme-right-panel #propertyTab .textLine .value.check { color: green; } .nme-right-panel #propertyTab .textLine .value.uncheck { color: red; }`, "", { version: 3, sources: ["webpack://./../../../tools/nodeEditor/dist/components/propertyTab/propertyTab.scss"], names: [], mappings: "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", sourcesContent: [`.nme-right-panel {\r #propertyTab {\r $line-padding-left: 5px;\r color: white;\r background: #333333;\r \r #header {\r height: 30px;\r font-size: 16px;\r color: white;\r background: #222222;\r grid-row: 1;\r text-align: center;\r display: grid;\r grid-template-columns: 30px 1fr;\r -webkit-user-select: none;\r -moz-user-select: none;\r -ms-user-select: none;\r user-select: none;\r \r #logo {\r position: relative;\r grid-column: 1;\r width: 24px;\r height: 24px;\r left: 0;\r display: flex;\r align-self: center;\r justify-self: center;\r }\r \r #title {\r grid-column: 2;\r display: grid;\r align-items: center;\r text-align: center;\r }\r }\r \r .range {\r -webkit-appearance: none;\r width: 120px;\r height: 6px;\r background: #d3d3d3;\r border-radius: 5px;\r outline: none;\r opacity: 0.7;\r -webkit-transition: 0.2s;\r transition: opacity 0.2s;\r }\r \r .range:hover {\r opacity: 1;\r }\r \r .range::-webkit-slider-thumb {\r -webkit-appearance: none;\r appearance: none;\r width: 14px;\r height: 14px;\r border-radius: 50%;\r background: rgb(51, 122, 183);\r cursor: pointer;\r }\r \r .range::-moz-range-thumb {\r width: 14px;\r height: 14px;\r border-radius: 50%;\r background: rgb(51, 122, 183);\r cursor: pointer;\r }\r \r input[type="color"] {\r -webkit-appearance: none;\r border: 1px solid rgba(255, 255, 255, 0.5);\r padding: 0;\r width: 30px;\r height: 20px;\r }\r input[type="color"]::-webkit-color-swatch-wrapper {\r padding: 0;\r }\r input[type="color"]::-webkit-color-swatch {\r border: none;\r }\r \r .sliderLine {\r padding-left: $line-padding-left;\r height: 30px;\r display: grid;\r grid-template-rows: 100%;\r grid-template-columns: 1fr 50px auto;\r \r .label {\r grid-column: 1;\r display: flex;\r align-items: center;\r }\r \r .slider {\r grid-column: 3;\r grid-row: 1;\r margin-right: 5px;\r width: 90%;\r display: flex;\r align-items: center;\r }\r \r .floatLine {\r grid-column: 2;\r padding-left: $line-padding-left;\r \r .label {\r grid-column: 1;\r display: flex;\r align-items: center;\r }\r \r .short {\r grid-column: 1;\r display: flex;\r align-items: center;\r \r input {\r width: 35px;\r }\r \r input::-webkit-outer-spin-button,\r input::-webkit-inner-spin-button {\r -webkit-appearance: none;\r margin: 0;\r }\r \r input[type="number"] {\r -moz-appearance: textfield;\r }\r }\r }\r }\r \r .textInputLine {\r padding-left: $line-padding-left;\r height: 30px;\r display: grid;\r grid-template-columns: 1fr 120px auto;\r \r .label {\r grid-column: 1;\r display: flex;\r align-items: center;\r }\r \r .value {\r display: flex;\r align-items: center;\r grid-column: 2;\r \r input {\r width: calc(100% - 5px);\r margin-right: 5px;\r }\r }\r }\r \r .textInputArea {\r padding-left: $line-padding-left;\r height: 50px;\r display: grid;\r grid-template-columns: 1fr 120px;\r \r .label {\r grid-column: 1;\r display: flex;\r align-items: center;\r }\r \r textarea {\r margin-right: 5px;\r margin-left: -50%;\r height: 40px;\r resize: none;\r }\r \r .value {\r display: flex;\r align-items: center;\r grid-column: 2;\r }\r }\r \r .paneContainer {\r margin-top: 3px;\r display: grid;\r grid-template-rows: 100%;\r grid-template-columns: 100%;\r \r .paneList {\r border-left: 3px solid transparent;\r }\r \r &:hover {\r .paneList {\r border-left: 3px solid rgba(51, 122, 183, 0.8);\r }\r \r .paneContainer-content {\r .header {\r .title {\r border-left: 3px solid rgb(51, 122, 183);\r }\r }\r }\r }\r \r .paneContainer-highlight-border {\r grid-row: 1;\r grid-column: 1;\r opacity: 1;\r border: 3px solid red;\r transition: opacity 250ms;\r pointer-events: none;\r \r &.transparent {\r opacity: 0;\r }\r }\r \r .paneContainer-content {\r grid-row: 1;\r grid-column: 1;\r \r .header {\r display: grid;\r grid-template-columns: 1fr auto;\r background: #555555;\r height: 30px;\r padding-right: 5px;\r cursor: pointer;\r \r .title {\r border-left: 3px solid transparent;\r padding-left: 5px;\r grid-column: 1;\r display: flex;\r align-items: center;\r }\r \r .collapse {\r grid-column: 2;\r display: flex;\r align-items: center;\r justify-items: center;\r transform-origin: center;\r \r &.closed {\r transform: rotate(180deg);\r }\r }\r }\r \r .paneList > div:not(:last-child) {\r border-bottom: 0.5px solid rgba(255, 255, 255, 0.1);\r }\r \r .fragment > div:not(:last-child) {\r border-bottom: 0.5px solid rgba(255, 255, 255, 0.1);\r }\r }\r }\r \r .color-picker {\r height: calc(100% - 8px);\r margin: 4px;\r width: calc(100% - 8px);\r \r .color-rect {\r height: calc(100% - 4px);\r border: 2px white solid;\r cursor: pointer;\r min-height: 18px;\r }\r \r .color-picker-cover {\r position: fixed;\r top: 0px;\r right: 0px;\r bottom: 0px;\r left: 0px;\r z-index: 1;\r }\r \r .color-picker-float {\r z-index: 2;\r position: absolute;\r }\r }\r \r .gradient-step {\r display: grid;\r grid-template-rows: 100%;\r grid-template-columns: 20px 30px 40px auto 20px 30px;\r padding-top: 5px;\r padding-left: 5px;\r padding-bottom: 5px;\r \r .step {\r grid-row: 1;\r grid-column: 1;\r }\r \r .color {\r grid-row: 1;\r grid-column: 2;\r cursor: pointer;\r }\r \r .step-value {\r margin-left: 5px;\r grid-row: 1;\r grid-column: 3;\r text-align: right;\r margin-right: 5px;\r }\r \r .step-slider {\r grid-row: 1;\r grid-column: 4;\r display: grid;\r justify-content: stretch;\r align-content: center;\r margin-right: -5px;\r padding-left: 12px;\r \r input {\r width: 90%;\r }\r }\r \r .gradient-copy {\r grid-row: 1;\r grid-column: 5;\r display: grid;\r align-content: center;\r justify-content: center;\r \r .img {\r height: 20px;\r width: 20px;\r }\r .img:hover {\r cursor: pointer;\r }\r }\r .gradient-delete {\r grid-row: 1;\r grid-column: 6;\r display: grid;\r align-content: center;\r justify-content: center;\r .img {\r height: 20px;\r width: 20px;\r }\r .img:hover {\r cursor: pointer;\r }\r }\r }\r \r .floatLine {\r padding-left: $line-padding-left;\r height: 30px;\r display: grid;\r grid-template-columns: 1fr 120px;\r \r .label {\r grid-column: 1;\r display: flex;\r align-items: center;\r }\r \r .value {\r grid-column: 2;\r \r display: flex;\r align-items: center;\r \r input {\r width: 110px;\r }\r }\r \r .short {\r grid-column: 2;\r \r display: flex;\r align-items: center;\r \r input {\r width: 27px;\r }\r \r input::-webkit-outer-spin-button,\r input::-webkit-inner-spin-button {\r -webkit-appearance: none;\r margin: 0;\r }\r \r input[type="number"] {\r -moz-appearance: textfield;\r }\r }\r }\r \r .vector3Line {\r padding-left: $line-padding-left;\r display: grid;\r \r .firstLine {\r display: grid;\r grid-template-columns: 1fr auto 20px;\r height: 30px;\r \r .label {\r grid-column: 1;\r display: flex;\r align-items: center;\r }\r \r .vector {\r grid-column: 2;\r display: flex;\r align-items: center;\r text-align: right;\r opacity: 0.8;\r }\r \r .expand {\r grid-column: 3;\r display: grid;\r align-items: center;\r justify-items: center;\r cursor: pointer;\r }\r }\r \r .secondLine {\r display: grid;\r padding-right: 5px;\r border-left: 1px solid rgb(51, 122, 183);\r \r .no-right-margin {\r margin-right: 0;\r }\r \r .numeric {\r display: grid;\r grid-template-columns: 1fr auto;\r }\r \r .numeric-label {\r text-align: right;\r grid-column: 1;\r display: flex;\r align-items: center;\r justify-self: right;\r margin-right: 10px;\r }\r \r .numeric-value {\r width: 120px;\r grid-column: 2;\r display: flex;\r align-items: center;\r border: 1px solid rgb(51, 122, 183);\r }\r }\r }\r \r .buttonLine {\r height: 30px;\r display: grid;\r align-items: center;\r justify-items: stretch;\r padding-bottom: 5px;\r \r &.disabled {\r opacity: 0.3;\r }\r \r input[type="file"] {\r display: none;\r }\r \r .file-upload {\r background: #222222;\r border: 1px solid rgb(51, 122, 183);\r margin: 5px 10px;\r color: white;\r padding: 4px 5px;\r padding-top: 0px;\r opacity: 0.9;\r cursor: pointer;\r text-align: center;\r }\r \r .file-upload:hover {\r opacity: 1;\r }\r \r .file-upload:active {\r transform: scale(0.98);\r transform-origin: 0.5 0.5;\r }\r \r button {\r background: #222222;\r border: 1px solid rgb(51, 122, 183);\r margin: 5px 10px 5px 10px;\r color: white;\r padding: 4px 5px;\r opacity: 0.9;\r }\r \r button:hover {\r opacity: 1;\r }\r \r button:active {\r background: #282828;\r }\r \r button:focus {\r border: 1px solid rgb(51, 122, 183);\r outline: 0px;\r }\r }\r \r .checkBoxLine {\r padding-left: $line-padding-left;\r height: 30px;\r display: grid;\r grid-template-columns: 1fr auto;\r \r .label {\r grid-column: 1;\r display: flex;\r align-items: center;\r }\r \r .checkBox {\r grid-column: 2;\r \r display: flex;\r align-items: center;\r \r .lbl {\r position: relative;\r display: block;\r height: 14px;\r width: 34px;\r margin-right: 5px;\r background: #898989;\r border-radius: 100px;\r cursor: pointer;\r transition: all 0.3s ease;\r }\r \r .lbl:after {\r position: absolute;\r left: 3px;\r top: 2px;\r display: block;\r width: 10px;\r height: 10px;\r border-radius: 100px;\r background: #fff;\r box-shadow: 0px 3px 3px rgba(0, 0, 0, 0.05);\r content: "";\r transition: all 0.15s ease;\r }\r \r .lbl:active:after {\r transform: scale(1.15, 0.85);\r }\r \r .cbx:checked ~ label {\r background: rgb(51, 122, 183);\r }\r \r .cbx:checked ~ label:after {\r left: 20px;\r background: rgb(22, 73, 117);\r }\r \r .cbx:checked ~ label.disabled {\r background: rgb(22, 73, 117);\r cursor: pointer;\r }\r \r .cbx:checked ~ label.disabled:after {\r left: 20px;\r background: rgb(85, 85, 85);\r cursor: pointer;\r }\r \r .cbx ~ label.disabled {\r background: rgb(85, 85, 85);\r cursor: pointer;\r }\r \r .hidden {\r display: none;\r }\r }\r }\r \r .listLine {\r padding-left: $line-padding-left;\r height: 30px;\r display: grid;\r grid-template-columns: 1fr auto;\r \r .label {\r grid-column: 1;\r display: flex;\r align-items: center;\r }\r \r .options {\r grid-column: 2;\r \r display: flex;\r align-items: center;\r margin-right: 5px;\r \r select {\r width: 115px;\r }\r }\r }\r \r .color3Line {\r padding-left: $line-padding-left;\r display: grid;\r \r .firstLine {\r height: 30px;\r display: grid;\r grid-template-columns: 1fr auto 0px 20px 20px;\r \r .label {\r grid-column: 1;\r display: flex;\r align-items: center;\r }\r \r .textInputLine {\r display: none;\r }\r \r .color3 {\r grid-column: 2;\r width: 50px;\r \r display: flex;\r align-items: center;\r \r input {\r margin-right: 5px;\r }\r }\r \r .copy {\r grid-column: 4;\r display: grid;\r align-items: center;\r justify-items: center;\r cursor: pointer;\r \r img {\r height: 100%;\r width: 24px;\r }\r }\r \r .expand {\r grid-column: 5;\r display: grid;\r align-items: center;\r justify-items: center;\r cursor: pointer;\r \r img {\r height: 100%;\r width: 20px;\r }\r }\r }\r \r .secondLine {\r display: grid;\r padding-right: 5px;\r border-left: 1px solid rgb(51, 122, 183);\r \r .numeric {\r display: grid;\r grid-template-columns: 1fr auto;\r }\r \r .numeric-label {\r text-align: right;\r grid-column: 1;\r display: flex;\r align-items: center;\r justify-self: right;\r margin-right: 10px;\r }\r \r .numeric-value {\r width: 120px;\r grid-column: 2;\r display: flex;\r align-items: center;\r border: 1px solid rgb(51, 122, 183);\r }\r }\r }\r \r .textLine {\r padding-left: $line-padding-left;\r height: 30px;\r display: grid;\r grid-template-columns: 1fr auto;\r \r .label {\r grid-column: 1;\r display: flex;\r align-items: center;\r }\r \r .link-value {\r grid-column: 2;\r white-space: nowrap;\r text-overflow: ellipsis;\r overflow: hidden;\r text-align: end;\r opacity: 0.8;\r margin: 5px;\r margin-top: 6px;\r max-width: 140px;\r text-decoration: underline;\r cursor: pointer;\r }\r \r .value {\r grid-column: 2;\r white-space: nowrap;\r text-overflow: ellipsis;\r overflow: hidden;\r text-align: end;\r opacity: 0.8;\r margin: 5px;\r margin-top: 6px;\r max-width: 200px;\r -webkit-user-select: text;\r -moz-user-select: text;\r -ms-user-select: text;\r user-select: text;\r \r &.check {\r color: green;\r }\r \r &.uncheck {\r color: red;\r }\r }\r }\r }\r }\r `], sourceRoot: "" }]), P.locals = {}; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[3].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[3].use[2]!../../../tools/nodeEditor/dist/main.scss": ( /*!*********************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[3].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[3].use[2]!../../../tools/nodeEditor/dist/main.scss ***! \*********************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `#node-editor-graph-root { display: grid; grid-template-rows: calc(100% - 120px) 120px; height: 100%; width: 100%; background: #464646; font: 14px "acumin-pro"; } #node-editor-graph-root.popup { grid-template-columns: 100%; overflow: hidden; } #node-editor-graph-root .wait-screen { display: grid; justify-content: center; align-content: center; height: 100%; width: 100%; background: #464646; opacity: 0.95; color: white; font: 24px "acumin-pro"; position: absolute; top: 0; left: 0; } #node-editor-graph-root .wait-screen.hidden { visibility: hidden; } #node-editor-graph-root #nmeNodeList { grid-row: 1/span 2; grid-column: 1; } #node-editor-graph-root #leftGrab { grid-row: 1/span 2; grid-column: 2; cursor: ew-resize; } #node-editor-graph-root #rightGrab { grid-row: 1/span 2; grid-column: 4; cursor: ew-resize; } #node-editor-graph-root .diagram-container { grid-row: 1; grid-column: 3; background: #5f5b60; width: 100%; height: 100%; } #node-editor-graph-root .diagram-container .diagram { display: none; width: 100%; height: 100%; } #node-editor-graph-root .nme-right-panel { grid-row: 1/span 2; grid-column: 5; display: grid; grid-template-rows: 1fr 40px auto 40px; grid-template-columns: 100%; height: 100%; overflow-y: auto; } #node-editor-graph-root #propertyTab { grid-row: 1; grid-column: 1; } #node-editor-graph-root .button { display: grid; justify-content: center; align-content: center; height: auto; width: 14.2857142857%; cursor: pointer; } #node-editor-graph-root .button:hover { background: rgb(51, 122, 183); color: white; opacity: 0.8; } #node-editor-graph-root .button.selected { background: rgb(51, 122, 183); color: white; } #node-editor-graph-root .button.align { justify-content: stretch; text-align: center; } #node-editor-graph-root #preview-mesh-bar { grid-row: 2; grid-column: 1; display: grid; grid-template-columns: auto 1fr 40px 40px 40px; align-items: center; font-size: 18px; background-color: #555555; } #node-editor-graph-root #preview-mesh-bar #file-picker { display: none; } #node-editor-graph-root #preview-mesh-bar .listLine { grid-column: 1; height: 40px; display: grid; grid-template-columns: 0px 1fr; } #node-editor-graph-root #preview-mesh-bar .listLine .label { grid-column: 1; display: flex; align-items: center; font-size: 14px; } #node-editor-graph-root #preview-mesh-bar .listLine .options { grid-column: 2; display: flex; align-items: center; margin-left: 5px; } #node-editor-graph-root #preview-mesh-bar .listLine .options select { width: 115px; } #node-editor-graph-root #preview-mesh-bar .button { color: #ffffff; width: 40px; height: 40px; transform-origin: 50% 50%; } #node-editor-graph-root #preview-mesh-bar .button:active { transform: scale(0.9); } #node-editor-graph-root #preview-mesh-bar .button:hover { background: #3f3461; } #node-editor-graph-root #preview-mesh-bar .button.selected { background: #9379e6; } #node-editor-graph-root #preview-mesh-bar .button img { height: 24px; width: 24px; } #node-editor-graph-root #preview-mesh-bar #play-button { grid-column: 3; } #node-editor-graph-root #preview-mesh-bar #color-picker-button { grid-column: 4; display: grid; grid-template-columns: 100%; grid-template-rows: 100%; } #node-editor-graph-root #preview-mesh-bar #color-picker-button img { height: 24px; width: 24px; } #node-editor-graph-root #preview-mesh-bar #color-picker-button #color-picker-image { padding-left: 7px; padding-top: 8px; } #node-editor-graph-root #preview-mesh-bar #color-picker-button #color-picker { transform: scale(0); grid-column: 1; grid-row: 1; } #node-editor-graph-root #preview-mesh-bar #color-picker-button #color-picker-label { width: 100%; background: transparent; cursor: pointer; } #node-editor-graph-root #preview-mesh-bar #preview-new-window { grid-column: 5; } #node-editor-graph-root #preview-mesh-bar select { background-color: #a3a3a3; color: #333333; } #node-editor-graph-root #preview-config-bar { grid-row: 4; grid-column: 1; display: grid; grid-template-columns: 40px 40px 40px 1fr 40px 40px; color: white; align-items: center; font-size: 18px; } #node-editor-graph-root #preview-config-bar.extended { grid-template-columns: 1fr; } #node-editor-graph-root #preview-config-bar .listLine { padding-left: 5px; height: 30px; display: grid; grid-template-columns: 1fr auto; } #node-editor-graph-root #preview-config-bar .listLine .label { grid-column: 1; display: flex; align-items: center; font-size: 14px; } #node-editor-graph-root #preview-config-bar .listLine .options { grid-column: 2; display: flex; align-items: center; margin-right: 5px; } #node-editor-graph-root #preview-config-bar .listLine .options select { width: 115px; } #node-editor-graph-root #preview-config-bar .button { width: 40px; grid-row: 1; height: 40px; transform-origin: 50% 50%; } #node-editor-graph-root #preview-config-bar .button:hover { background: #3f3461; } #node-editor-graph-root #preview-config-bar .button.selected { background: #9379e6; } #node-editor-graph-root #preview-config-bar .button:active { transform: scale(0.9); } #node-editor-graph-root #preview-config-bar .button img { height: auto; width: 100%; } #node-editor-graph-root #preview-config-bar .button.back-face { grid-column: 6; } #node-editor-graph-root #preview-config-bar .button.depth-pass { grid-column: 5/6; } #node-editor-graph-root #preview-config-bar .button.hemispheric-light { grid-column: 3/4; } #node-editor-graph-root #preview-config-bar .button.direction-light-1 { grid-column: 2/3; } #node-editor-graph-root #preview-config-bar .button.direction-light-0 { grid-column: 1/2; } #node-editor-graph-root #preview { border-top: 1px solid rgb(85, 85, 85); grid-row: 3; grid-column: 1; width: 100%; display: grid; grid-template-columns: 100%; grid-template-rows: 100%; outline: 0 !important; padding: 0 !important; user-select: none; overflow: hidden; } #node-editor-graph-root #preview #preview-canvas { width: 100%; height: 100%; outline: 0 !important; padding: 0 !important; grid-row: 1; grid-column: 1; } #node-editor-graph-root #preview .waitPanel { width: 100%; height: 100%; grid-row: 1; grid-column: 1; color: white; font-size: 18px; align-content: center; justify-content: center; background: rgba(20, 20, 20, 0.95); z-index: 10; display: grid; transition: opacity 250ms; } #node-editor-graph-root #preview .waitPanel.hidden { opacity: 0; pointer-events: none; } #node-editor-graph-root .blocker { visibility: hidden; position: absolute; width: calc(100% - 40px); height: 100%; top: 0; left: 0; background: rgba(20, 20, 20, 0.95); font-family: "acumin-pro"; color: white; font-size: 24px; display: grid; align-content: center; justify-content: center; user-select: none; padding: 20px; text-align: center; } #node-editor-graph-root #log-console { grid-row: 2; grid-column: 3; } #node-editor-graph-root .LightInformationBlock { width: 280px; } #node-editor-graph-root .InputBlock { width: 250px; }`, "", { version: 3, sources: ["webpack://./../../../tools/nodeEditor/dist/main.scss"], names: [], mappings: "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", sourcesContent: [`#node-editor-graph-root {\r display: grid;\r grid-template-rows: calc(100% - 120px) 120px;\r height: 100%;\r width: 100%;\r background: #464646;\r font: 14px "acumin-pro";\r \r &.popup {\r grid-template-columns: 100%;\r overflow: hidden;\r }\r \r .wait-screen {\r display: grid;\r justify-content: center;\r align-content: center;\r height: 100%;\r width: 100%;\r background: #464646;\r opacity: 0.95;\r color: white;\r font: 24px "acumin-pro";\r position: absolute;\r top: 0;\r left: 0;\r \r &.hidden {\r visibility: hidden;\r }\r }\r \r #nmeNodeList {\r grid-row: 1 / span 2;\r grid-column: 1;\r }\r \r #leftGrab {\r grid-row: 1 / span 2;\r grid-column: 2;\r cursor: ew-resize;\r }\r \r #rightGrab {\r grid-row: 1 / span 2;\r grid-column: 4;\r cursor: ew-resize;\r }\r \r .diagram-container {\r grid-row: 1;\r grid-column: 3;\r background: #5f5b60;\r width: 100%;\r height: 100%;\r \r .diagram {\r display: none;\r width: 100%;\r height: 100%;\r }\r }\r \r .nme-right-panel {\r grid-row: 1 / span 2;\r grid-column: 5;\r display: grid;\r grid-template-rows: 1fr 40px auto 40px;\r grid-template-columns: 100%;\r height: 100%;\r overflow-y: auto;\r }\r \r #propertyTab {\r grid-row: 1;\r grid-column: 1;\r }\r \r .button {\r display: grid;\r justify-content: center;\r align-content: center;\r height: auto;\r width: calc(100% / 7);\r cursor: pointer;\r \r &:hover {\r background: rgb(51, 122, 183);\r color: white;\r opacity: 0.8;\r }\r \r &.selected {\r background: rgb(51, 122, 183);\r color: white;\r }\r \r &.align {\r justify-content: stretch;\r text-align: center;\r }\r }\r \r #preview-mesh-bar {\r grid-row: 2;\r grid-column: 1;\r display: grid;\r grid-template-columns: auto 1fr 40px 40px 40px;\r align-items: center;\r font-size: 18px;\r background-color: #555555;\r \r #file-picker {\r display: none;\r }\r \r .listLine {\r grid-column: 1;\r height: 40px;\r display: grid;\r grid-template-columns: 0px 1fr;\r \r .label {\r grid-column: 1;\r display: flex;\r align-items: center;\r font-size: 14px;\r }\r \r .options {\r grid-column: 2;\r \r display: flex;\r align-items: center;\r margin-left: 5px;\r \r select {\r width: 115px;\r }\r }\r }\r \r .button {\r color: #ffffff;\r width: 40px;\r height: 40px;\r transform-origin: 50% 50%;\r \r &:active {\r transform: scale(0.9);\r }\r \r &:hover {\r background: #3f3461;\r }\r \r &.selected {\r background: #9379e6;\r }\r \r img {\r height: 24px;\r width: 24px;\r }\r }\r \r #play-button {\r grid-column: 3;\r }\r \r #color-picker-button {\r grid-column: 4;\r display: grid;\r grid-template-columns: 100%;\r grid-template-rows: 100%;\r \r img {\r height: 24px;\r width: 24px;\r }\r #color-picker-image {\r padding-left: 7px;\r padding-top: 8px;\r }\r \r #color-picker {\r transform: scale(0);\r grid-column: 1;\r grid-row: 1;\r }\r \r #color-picker-label {\r width: 100%;\r background: transparent;\r cursor: pointer;\r }\r }\r \r #preview-new-window {\r grid-column: 5;\r }\r \r select {\r background-color: #a3a3a3;\r color: #333333;\r }\r }\r \r #preview-config-bar {\r grid-row: 4;\r grid-column: 1;\r display: grid;\r grid-template-columns: 40px 40px 40px 1fr 40px 40px;\r color: white;\r align-items: center;\r font-size: 18px;\r \r &.extended {\r grid-template-columns: 1fr;\r }\r \r .listLine {\r padding-left: 5px;\r height: 30px;\r display: grid;\r grid-template-columns: 1fr auto;\r \r .label {\r grid-column: 1;\r display: flex;\r align-items: center;\r font-size: 14px;\r }\r \r .options {\r grid-column: 2;\r \r display: flex;\r align-items: center;\r margin-right: 5px;\r \r select {\r width: 115px;\r }\r }\r }\r \r .button {\r width: 40px;\r grid-row: 1;\r height: 40px;\r transform-origin: 50% 50%;\r \r &:hover {\r background: #3f3461;\r }\r \r &.selected {\r background: #9379e6;\r }\r \r &:active {\r transform: scale(0.9);\r }\r \r img {\r height: auto;\r width: 100%;\r }\r \r &.back-face {\r grid-column: 6;\r }\r \r &.depth-pass {\r grid-column: 5 / 6;\r }\r \r &.hemispheric-light {\r grid-column: 3 / 4;\r }\r &.direction-light-1 {\r grid-column: 2 / 3;\r }\r &.direction-light-0 {\r grid-column: 1 / 2;\r }\r }\r }\r \r #preview {\r border-top: 1px solid rgb(85, 85, 85);\r grid-row: 3;\r grid-column: 1;\r width: 100%;\r display: grid;\r grid-template-columns: 100%;\r grid-template-rows: 100%;\r outline: 0 !important;\r padding: 0 !important;\r user-select: none;\r overflow: hidden;\r \r #preview-canvas {\r width: 100%;\r height: 100%;\r outline: 0 !important;\r padding: 0 !important;\r grid-row: 1;\r grid-column: 1;\r }\r \r .waitPanel {\r width: 100%;\r height: 100%;\r grid-row: 1;\r grid-column: 1;\r color: white;\r font-size: 18px;\r align-content: center;\r justify-content: center;\r background: rgba(20, 20, 20, 0.95);\r z-index: 10;\r display: grid;\r transition: opacity 250ms;\r \r &.hidden {\r opacity: 0;\r pointer-events: none;\r }\r }\r }\r \r .blocker {\r visibility: hidden;\r position: absolute;\r width: calc(100% - 40px);\r height: 100%;\r top: 0;\r left: 0;\r \r background: rgba(20, 20, 20, 0.95);\r font-family: "acumin-pro";\r color: white;\r font-size: 24px;\r \r display: grid;\r align-content: center;\r justify-content: center;\r \r user-select: none;\r \r padding: 20px;\r text-align: center;\r }\r \r #log-console {\r grid-row: 2;\r grid-column: 3;\r }\r \r .LightInformationBlock {\r width: 280px;\r }\r \r .InputBlock {\r width: 250px;\r }\r }\r `], sourceRoot: "" }]), P.locals = {}; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../dev/sharedUiComponents/dist/components/MessageDialog.modules.scss": ( /*!*******************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../dev/sharedUiComponents/dist/components/MessageDialog.modules.scss ***! \*******************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-container { position: absolute; width: 100%; height: 100%; background: rgba(0, 0, 0, 0.6); display: grid; font-family: "acumin-pro"; top: 0; } .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-container .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog { align-self: center; justify-self: center; min-height: 140px; max-width: 400px; border-radius: 10px; background: white; padding: 10px; display: grid; grid-template-columns: 100%; grid-template-rows: 1fr 50px; } .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-container .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-message { grid-row: 1; grid-column: 1; margin-top: 20px; padding: 10px; font-size: 18px; color: black; } .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-container .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-buttons { grid-row: 2; grid-column: 1; display: grid; grid-template-rows: 100%; grid-template-columns: 100%; color: white; } .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-container .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-buttons .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-button-ok { cursor: pointer; justify-self: center; background: green; min-width: 80px; justify-content: center; display: grid; align-content: center; align-self: center; height: 35px; border-radius: 10px; } .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-container .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-buttons .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-button-ok:hover { opacity: 0.8; } .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-container .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-buttons .\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-button-ok.\\---------dev-sharedUiComponents-dist-components-MessageDialog-modules__error { background: red; }`, "", { version: 3, sources: ["webpack://./../../../dev/sharedUiComponents/dist/components/MessageDialog.modules.scss"], names: [], mappings: "AAAA;EACI,kBAAA;EACA,WAAA;EACA,YAAA;EACA,8BAAA;EACA,aAAA;EACA,yBAAA;EACA,MAAA;AACJ;AACI;EACI,kBAAA;EACA,oBAAA;EACA,iBAAA;EACA,gBAAA;EACA,mBAAA;EACA,iBAAA;EACA,aAAA;EAEA,aAAA;EACA,2BAAA;EACA,4BAAA;AAAR;AAEQ;EACI,WAAA;EACA,cAAA;EACA,gBAAA;EACA,aAAA;EACA,eAAA;EACA,YAAA;AAAZ;AAGQ;EACI,WAAA;EACA,cAAA;EACA,aAAA;EACA,wBAAA;EACA,2BAAA;EACA,YAAA;AADZ;AAGY;EACI,eAAA;EACA,oBAAA;EACA,iBAAA;EACA,eAAA;EACA,uBAAA;EACA,aAAA;EACA,qBAAA;EACA,kBAAA;EACA,YAAA;EACA,mBAAA;AADhB;AAGgB;EACI,YAAA;AADpB;AAIgB;EACI,eAAA;AAFpB", sourcesContent: [`:local .dialog-container {\r position: absolute;\r width: 100%;\r height: 100%;\r background: rgba(0.1, 0.1, 0.1, 0.6);\r display: grid;\r font-family: "acumin-pro";\r top: 0;\r \r .dialog {\r align-self: center;\r justify-self: center;\r min-height: 140px;\r max-width: 400px;\r border-radius: 10px;\r background: white;\r padding: 10px;\r \r display: grid;\r grid-template-columns: 100%;\r grid-template-rows: 1fr 50px;\r \r .dialog-message {\r grid-row: 1;\r grid-column: 1;\r margin-top: 20px;\r padding: 10px;\r font-size: 18px;\r color: black;\r }\r \r .dialog-buttons {\r grid-row: 2;\r grid-column: 1;\r display: grid;\r grid-template-rows: 100%;\r grid-template-columns: 100%;\r color: white;\r \r .dialog-button-ok {\r cursor: pointer;\r justify-self: center;\r background: green;\r min-width: 80px;\r justify-content: center;\r display: grid;\r align-content: center;\r align-self: center;\r height: 35px;\r border-radius: 10px;\r \r &:hover {\r opacity: 0.8;\r }\r \r &.error {\r background: red;\r }\r }\r }\r }\r }\r `], sourceRoot: "" }]), P.locals = { "dialog-container": "---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-container", dialog: "---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog", "dialog-message": "---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-message", "dialog-buttons": "---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-buttons", "dialog-button-ok": "---------dev-sharedUiComponents-dist-components-MessageDialog-modules__dialog-button-ok", error: "---------dev-sharedUiComponents-dist-components-MessageDialog-modules__error" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../dev/sharedUiComponents/dist/nodeGraphSystem/common.modules.scss": ( /*!*****************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../dev/sharedUiComponents/dist/nodeGraphSystem/common.modules.scss ***! \*****************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__hidden { display: none !important; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__port { border-radius: 20px; width: 20px; height: 20px; align-self: center; display: grid; grid-template-columns: 100%; grid-template-rows: 100%; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__port .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__img { width: 100%; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__port:hover, .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__port.\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__selected { filter: brightness(2); } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__portLine { height: 24px; display: grid; grid-template-rows: 100%; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__port-label { align-items: center; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__inputsContainer { grid-row: 1; grid-column: 1; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__inputsContainer .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__portLine { grid-template-columns: 12px calc(100% - 15px); } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__inputsContainer .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__portLine .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__port-label { grid-row: 1; grid-column: 2; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__inputsContainer .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__portLine .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__port { grid-row: 1; grid-column: 1; transform: translateX(-12px); } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__outputsContainer { grid-row: 1; grid-column: 2; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__outputsContainer .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__portLine { grid-template-columns: calc(100% - 10px) 12px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__outputsContainer .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__portLine .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__port-label { grid-row: 1; grid-column: 1; text-align: right; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__outputsContainer .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__portLine .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__port { grid-row: 1; grid-column: 2; transform: translateX(2px); } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__frame-box { position: absolute; background: rgba(72, 72, 72, 0.7); display: grid; grid-template-rows: 40px calc(100% - 40px); grid-template-columns: 100%; box-sizing: border-box; }`, "", { version: 3, sources: ["webpack://./../../../dev/sharedUiComponents/dist/nodeGraphSystem/common.modules.scss"], names: [], mappings: "AAAA;EACI,wBAAA;AACJ;;AAEA;EACI,mBAAA;EACA,WAAA;EACA,YAAA;EACA,kBAAA;EACA,aAAA;EACA,2BAAA;EACA,wBAAA;AACJ;AACI;EACI,WAAA;AACR;AAEI;EAEI,qBAAA;AADR;;AAKA;EACI,YAAA;EACA,aAAA;EACA,wBAAA;AAFJ;;AAKA;EACI,mBAAA;AAFJ;;AAKA;EACI,WAAA;EACA,cAAA;AAFJ;AAII;EACI,6CAAA;AAFR;AAIQ;EACI,WAAA;EACA,cAAA;AAFZ;AAKQ;EACI,WAAA;EACA,cAAA;EACA,4BAAA;AAHZ;;AAQA;EACI,WAAA;EACA,cAAA;AALJ;AAOI;EACI,6CAAA;AALR;AAOQ;EACI,WAAA;EACA,cAAA;EACA,iBAAA;AALZ;AAQQ;EACI,WAAA;EACA,cAAA;EACA,0BAAA;AANZ;;AAWA;EACI,kBAAA;EACA,iCAAA;EACA,aAAA;EACA,0CAAA;EACA,2BAAA;EACA,sBAAA;AARJ", sourcesContent: [`.hidden {\r display: none !important;\r }\r \r .port {\r border-radius: 20px;\r width: 20px;\r height: 20px;\r align-self: center;\r display: grid;\r grid-template-columns: 100%;\r grid-template-rows: 100%;\r \r .img {\r width: 100%;\r }\r \r &:hover,\r &.selected {\r filter: brightness(2);\r }\r }\r \r .portLine {\r height: 24px;\r display: grid;\r grid-template-rows: 100%;\r }\r \r .port-label {\r align-items: center;\r }\r \r .inputsContainer {\r grid-row: 1;\r grid-column: 1;\r \r .portLine {\r grid-template-columns: 12px calc(100% - 15px);\r \r .port-label {\r grid-row: 1;\r grid-column: 2;\r }\r \r .port {\r grid-row: 1;\r grid-column: 1;\r transform: translateX(-12px);\r }\r }\r }\r \r .outputsContainer {\r grid-row: 1;\r grid-column: 2;\r \r .portLine {\r grid-template-columns: calc(100% - 10px) 12px;\r \r .port-label {\r grid-row: 1;\r grid-column: 1;\r text-align: right;\r }\r \r .port {\r grid-row: 1;\r grid-column: 2;\r transform: translateX(2px);\r }\r }\r }\r \r .frame-box {\r position: absolute;\r background: rgba(72, 72, 72, 0.7);\r display: grid;\r grid-template-rows: 40px calc(100% - 40px);\r grid-template-columns: 100%;\r box-sizing: border-box;\r }\r `], sourceRoot: "" }]), P.locals = { hidden: "---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__hidden", port: "---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__port", img: "---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__img", selected: "---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__selected", portLine: "---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__portLine", "port-label": "---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__port-label", inputsContainer: "---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__inputsContainer", outputsContainer: "---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__outputsContainer", "frame-box": "---------dev-sharedUiComponents-dist-nodeGraphSystem-common-modules__frame-box" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../dev/sharedUiComponents/dist/nodeGraphSystem/graphCanvas.modules.scss": ( /*!**********************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../dev/sharedUiComponents/dist/nodeGraphSystem/graphCanvas.modules.scss ***! \**********************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphCanvas-modules__graph-canvas { width: 100%; height: 100%; margin: 0; padding: 0; font: 14px "acumin-pro"; user-select: none; overflow: hidden; cursor: move; position: relative; background-image: linear-gradient(to right, #4f4e4f 1px, transparent 1px), linear-gradient(to bottom, #4f4e4f 1px, transparent 1px); } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphCanvas-modules__graph-container { width: 100%; height: 100%; left: 0; top: 0; transform-origin: left top; display: grid; grid-template-rows: 100%; grid-template-columns: 100%; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphCanvas-modules__frame-container { overflow: visible; grid-row: 1; grid-column: 1; position: relative; width: 100%; height: 100%; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphCanvas-modules__graph-svg-container { grid-row: 1; grid-column: 1; position: relative; width: 100%; height: 100%; overflow: visible; pointer-events: none; z-index: 2; filter: drop-shadow(7px 6px 2px rgba(0, 0, 0, 0.2)); } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphCanvas-modules__graph-canvas-container { grid-row: 1; grid-column: 1; position: relative; width: 100%; height: 100%; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphCanvas-modules__selection-container { pointer-events: none; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphCanvas-modules__selection-box { z-index: 10; position: absolute; background: rgba(72, 72, 196, 0.5); border: blue solid 2px; }`, "", { version: 3, sources: ["webpack://./../../../dev/sharedUiComponents/dist/nodeGraphSystem/graphCanvas.modules.scss"], names: [], mappings: "AAAA;EACI,WAAA;EACA,YAAA;EACA,SAAA;EACA,UAAA;EACA,uBAAA;EACA,iBAAA;EACA,gBAAA;EACA,YAAA;EACA,kBAAA;EACA,mIAAA;AACJ;;AAEA;EACI,WAAA;EACA,YAAA;EACA,OAAA;EACA,MAAA;EACA,0BAAA;EACA,aAAA;EACA,wBAAA;EACA,2BAAA;AACJ;;AAEA;EACI,iBAAA;EACA,WAAA;EACA,cAAA;EACA,kBAAA;EACA,WAAA;EACA,YAAA;AACJ;;AAEA;EACI,WAAA;EACA,cAAA;EACA,kBAAA;EACA,WAAA;EACA,YAAA;EACA,iBAAA;EACA,oBAAA;EACA,UAAA;EACA,mDAAA;AACJ;;AAEA;EACI,WAAA;EACA,cAAA;EACA,kBAAA;EACA,WAAA;EACA,YAAA;AACJ;;AAEA;EACI,oBAAA;AACJ;;AAEA;EACI,WAAA;EACA,kBAAA;EACA,kCAAA;EACA,sBAAA;AACJ", sourcesContent: [`.graph-canvas {\r width: 100%;\r height: 100%;\r margin: 0;\r padding: 0;\r font: 14px "acumin-pro";\r user-select: none;\r overflow: hidden;\r cursor: move;\r position: relative;\r background-image: linear-gradient(to right, #4f4e4f 1px, transparent 1px), linear-gradient(to bottom, #4f4e4f 1px, transparent 1px);\r }\r \r .graph-container {\r width: 100%;\r height: 100%;\r left: 0;\r top: 0;\r transform-origin: left top;\r display: grid;\r grid-template-rows: 100%;\r grid-template-columns: 100%;\r }\r \r .frame-container {\r overflow: visible;\r grid-row: 1;\r grid-column: 1;\r position: relative;\r width: 100%;\r height: 100%;\r }\r \r .graph-svg-container {\r grid-row: 1;\r grid-column: 1;\r position: relative;\r width: 100%;\r height: 100%;\r overflow: visible;\r pointer-events: none;\r z-index: 2;\r filter: drop-shadow(7px 6px 2px rgba(0, 0, 0, 0.2));\r }\r \r .graph-canvas-container {\r grid-row: 1;\r grid-column: 1;\r position: relative;\r width: 100%;\r height: 100%;\r }\r \r .selection-container {\r pointer-events: none;\r }\r \r .selection-box {\r z-index: 10;\r position: absolute;\r background: rgba(72, 72, 196, 0.5);\r border: blue solid 2px;\r }\r `], sourceRoot: "" }]), P.locals = { "graph-canvas": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphCanvas-modules__graph-canvas", "graph-container": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphCanvas-modules__graph-container", "frame-container": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphCanvas-modules__frame-container", "graph-svg-container": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphCanvas-modules__graph-svg-container", "graph-canvas-container": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphCanvas-modules__graph-canvas-container", "selection-container": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphCanvas-modules__selection-container", "selection-box": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphCanvas-modules__selection-box" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../dev/sharedUiComponents/dist/nodeGraphSystem/graphFrame.modules.scss": ( /*!*********************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../dev/sharedUiComponents/dist/nodeGraphSystem/graphFrame.modules.scss ***! \*********************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__port-container { margin-top: 6px; margin-bottom: 6px; margin-left: 4px; margin-right: 4px; color: white; grid-row: 2; grid-column: 1; display: grid; grid-template-rows: 100%; grid-template-columns: 50% 50%; z-index: 2; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-box-border { grid-row: 1/span 2; grid-column: 1; width: 100%; height: 100%; border: transparent solid 4px; pointer-events: none; box-sizing: border-box; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-comments { display: grid; grid-row: 2; grid-column: 1; padding: 0 10px; font-style: italic; word-wrap: break-word; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__selected.\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-box-border { border-color: white; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-box-header { grid-row: 1; grid-column: 1; background: rgb(72, 72, 72); color: white; text-align: center; display: grid; grid-template-rows: 100%; grid-template-columns: calc(100% - 74px) 30px 7px 30px 7px; align-content: center; overflow: hidden; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-box-header-button { cursor: pointer; align-self: center; transform-origin: 50% 50%; transform: scale(1); stroke: transparent; fill: white; display: grid; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-box-header-button.\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__down { transform: scale(0.9); } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-box-header-collapse { grid-column: 2; grid-row: 1; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-box-header-close { grid-column: 4; grid-row: 1; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-box-header-title { grid-column: 1; grid-row: 1; display: grid; height: 100%; width: 100%; align-self: stretch; align-items: center; margin-top: -2px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__right-handle { grid-area: 1/2/3/2; width: 4px; background-color: transparent; cursor: ew-resize; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__right-handle::after { content: ""; width: 8px; position: absolute; top: 0; bottom: 0; margin-left: -4px; cursor: ew-resize; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__top-right-corner-handle { background-color: transparent; height: 4px; z-index: 21; cursor: ne-resize; width: 4px; margin-left: -6px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__top-right-corner-handle::after { background-color: transparent; cursor: ne-resize; margin-left: unset; top: -4px; height: 10px; width: 10px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__bottom-right-corner-handle { background-color: transparent; height: 0px; z-index: 21; cursor: nw-resize; grid-area: 4/2/4/2; margin-left: -2px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__bottom-right-corner-handle::after { background-color: transparent; height: 10px; cursor: nw-resize; top: unset; bottom: -4px; width: 10px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__left-handle { grid-area: 1/1/3/1; width: 4px; background-color: transparent; cursor: ew-resize; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__left-handle::before { content: ""; width: 8px; position: absolute; top: 0; bottom: 0; margin-left: -4px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__top-left-corner-handle { background-color: transparent; height: 4px; z-index: 21; cursor: nw-resize; width: 4px; margin-left: -4px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__top-left-corner-handle::before { background-color: transparent; cursor: nw-resize; margin-left: unset; top: -4px; height: 10px; width: 10px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__bottom-left-corner-handle { background-color: transparent; height: 0px; z-index: 21; cursor: sw-resize; grid-area: 4/1/4/1; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__bottom-left-corner-handle::before { background-color: transparent; height: 10px; cursor: sw-resize; top: unset; bottom: -4px; width: 10px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__top-handle { grid-area: 1/1/1/1; background-color: transparent; height: 4px; cursor: ns-resize; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__top-handle::before { content: ""; width: 100%; position: absolute; top: -4px; bottom: 100%; right: 0; left: 0; margin-bottom: -8px; cursor: ns-resize; height: 8px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__bottom-handle { grid-area: 3/1/3/1; background-color: transparent; height: 4px; cursor: ns-resize; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__bottom-handle::after { content: ""; width: 100%; position: absolute; top: 100%; bottom: 0; right: 0; left: 0; margin-top: -8px; cursor: ns-resize; height: 12px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__expanded { font-size: 24px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__collapsed { height: auto !important; width: 200px !important; z-index: 3; font-size: 16px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__collapsedHeader { grid-template-columns: calc(100% - 37px) 30px 7px; }`, "", { version: 3, sources: ["webpack://./../../../dev/sharedUiComponents/dist/nodeGraphSystem/graphFrame.modules.scss"], names: [], mappings: "AAAA;EACI,eAAA;EACA,kBAAA;EACA,gBAAA;EACA,iBAAA;EACA,YAAA;EACA,WAAA;EACA,cAAA;EACA,aAAA;EACA,wBAAA;EACA,8BAAA;EACA,UAAA;AACJ;;AAEA;EACI,kBAAA;EACA,cAAA;EACA,WAAA;EACA,YAAA;EACA,6BAAA;EACA,oBAAA;EACA,sBAAA;AACJ;;AAEA;EACI,aAAA;EACA,WAAA;EACA,cAAA;EACA,eAAA;EACA,kBAAA;EACA,qBAAA;AACJ;;AAEA;EACI,mBAAA;AACJ;;AAEA;EACI,WAAA;EACA,cAAA;EACA,2BAAA;EACA,YAAA;EACA,kBAAA;EACA,aAAA;EACA,wBAAA;EACA,0DAAA;EACA,qBAAA;EACA,gBAAA;AACJ;;AAEA;EACI,eAAA;EACA,kBAAA;EACA,yBAAA;EACA,mBAAA;EACA,mBAAA;EACA,WAAA;EACA,aAAA;AACJ;AACI;EACI,qBAAA;AACR;;AAGA;EACI,cAAA;EACA,WAAA;AAAJ;;AAGA;EACI,cAAA;EACA,WAAA;AAAJ;;AAGA;EACI,cAAA;EACA,WAAA;EACA,aAAA;EACA,YAAA;EACA,WAAA;EACA,mBAAA;EACA,mBAAA;EACA,gBAAA;AAAJ;;AAGA;EACI,kBAAA;EACA,UAAA;EACA,6BAAA;EACA,iBAAA;AAAJ;AAEI;EACI,WAAA;EACA,UAAA;EACA,kBAAA;EACA,MAAA;EACA,SAAA;EACA,iBAAA;EACA,iBAAA;AAAR;;AAIA;EACI,6BAAA;EACA,WAAA;EACA,WAAA;EACA,iBAAA;EACA,UAAA;EACA,iBAAA;AADJ;AAGI;EACI,6BAAA;EACA,iBAAA;EACA,kBAAA;EACA,SAAA;EACA,YAAA;EACA,WAAA;AADR;;AAKA;EACI,6BAAA;EACA,WAAA;EACA,WAAA;EACA,iBAAA;EACA,kBAAA;EACA,iBAAA;AAFJ;AAII;EACI,6BAAA;EACA,YAAA;EACA,iBAAA;EACA,UAAA;EACA,YAAA;EACA,WAAA;AAFR;;AAMA;EACI,kBAAA;EACA,UAAA;EACA,6BAAA;EACA,iBAAA;AAHJ;AAKI;EACI,WAAA;EACA,UAAA;EACA,kBAAA;EACA,MAAA;EACA,SAAA;EACA,iBAAA;AAHR;;AAOA;EACI,6BAAA;EACA,WAAA;EACA,WAAA;EACA,iBAAA;EACA,UAAA;EACA,iBAAA;AAJJ;AAMI;EACI,6BAAA;EACA,iBAAA;EACA,kBAAA;EACA,SAAA;EACA,YAAA;EACA,WAAA;AAJR;;AAQA;EACI,6BAAA;EACA,WAAA;EACA,WAAA;EACA,iBAAA;EACA,kBAAA;AALJ;AAOI;EACI,6BAAA;EACA,YAAA;EACA,iBAAA;EACA,UAAA;EACA,YAAA;EACA,WAAA;AALR;;AASA;EACI,kBAAA;EACA,6BAAA;EACA,WAAA;EACA,iBAAA;AANJ;AAQI;EACI,WAAA;EACA,WAAA;EACA,kBAAA;EACA,SAAA;EACA,YAAA;EACA,QAAA;EACA,OAAA;EACA,mBAAA;EACA,iBAAA;EACA,WAAA;AANR;;AAUA;EACI,kBAAA;EACA,6BAAA;EACA,WAAA;EACA,iBAAA;AAPJ;AASI;EACI,WAAA;EACA,WAAA;EACA,kBAAA;EACA,SAAA;EACA,SAAA;EACA,QAAA;EACA,OAAA;EACA,gBAAA;EACA,iBAAA;EACA,YAAA;AAPR;;AAWA;EACI,eAAA;AARJ;;AAWA;EACI,uBAAA;EACA,uBAAA;EACA,UAAA;EACA,eAAA;AARJ;;AAWA;EACI,iDAAA;AARJ", sourcesContent: [`.port-container {\r margin-top: 6px;\r margin-bottom: 6px;\r margin-left: 4px;\r margin-right: 4px;\r color: white;\r grid-row: 2;\r grid-column: 1;\r display: grid;\r grid-template-rows: 100%;\r grid-template-columns: 50% 50%;\r z-index: 2;\r }\r \r .frame-box-border {\r grid-row: 1 / span 2;\r grid-column: 1;\r width: 100%;\r height: 100%;\r border: transparent solid 4px;\r pointer-events: none;\r box-sizing: border-box;\r }\r \r .frame-comments {\r display: grid;\r grid-row: 2;\r grid-column: 1;\r padding: 0 10px;\r font-style: italic;\r word-wrap: break-word;\r }\r \r .selected.frame-box-border {\r border-color: white;\r }\r \r .frame-box-header {\r grid-row: 1;\r grid-column: 1;\r background: rgba(72, 72, 72, 1);\r color: white;\r text-align: center;\r display: grid;\r grid-template-rows: 100%;\r grid-template-columns: calc(100% - 74px) 30px 7px 30px 7px;\r align-content: center;\r overflow: hidden;\r }\r \r .frame-box-header-button {\r cursor: pointer;\r align-self: center;\r transform-origin: 50% 50%;\r transform: scale(1);\r stroke: transparent;\r fill: white;\r display: grid;\r \r &.down {\r transform: scale(0.9);\r }\r }\r \r .frame-box-header-collapse {\r grid-column: 2;\r grid-row: 1;\r }\r \r .frame-box-header-close {\r grid-column: 4;\r grid-row: 1;\r }\r \r .frame-box-header-title {\r grid-column: 1;\r grid-row: 1;\r display: grid;\r height: 100%;\r width: 100%;\r align-self: stretch;\r align-items: center;\r margin-top: -2px;\r }\r \r .right-handle {\r grid-area: 1 / 2 / 3 / 2;\r width: 4px;\r background-color: transparent;\r cursor: ew-resize;\r \r &::after {\r content: "";\r width: 8px;\r position: absolute;\r top: 0;\r bottom: 0;\r margin-left: -4px;\r cursor: ew-resize;\r }\r }\r \r .top-right-corner-handle {\r background-color: transparent;\r height: 4px;\r z-index: 21;\r cursor: ne-resize;\r width: 4px;\r margin-left: -6px;\r \r &::after {\r background-color: transparent;\r cursor: ne-resize;\r margin-left: unset;\r top: -4px;\r height: 10px;\r width: 10px;\r }\r }\r \r .bottom-right-corner-handle {\r background-color: transparent;\r height: 0px;\r z-index: 21;\r cursor: nw-resize;\r grid-area: 4 / 2 / 4 / 2;\r margin-left: -2px;\r \r &::after {\r background-color: transparent;\r height: 10px;\r cursor: nw-resize;\r top: unset;\r bottom: -4px;\r width: 10px;\r }\r }\r \r .left-handle {\r grid-area: 1 / 1 / 3 / 1;\r width: 4px;\r background-color: transparent;\r cursor: ew-resize;\r \r &::before {\r content: "";\r width: 8px;\r position: absolute;\r top: 0;\r bottom: 0;\r margin-left: -4px;\r }\r }\r \r .top-left-corner-handle {\r background-color: transparent;\r height: 4px;\r z-index: 21;\r cursor: nw-resize;\r width: 4px;\r margin-left: -4px;\r \r &::before {\r background-color: transparent;\r cursor: nw-resize;\r margin-left: unset;\r top: -4px;\r height: 10px;\r width: 10px;\r }\r }\r \r .bottom-left-corner-handle {\r background-color: transparent;\r height: 0px;\r z-index: 21;\r cursor: sw-resize;\r grid-area: 4 / 1 / 4 / 1;\r \r &::before {\r background-color: transparent;\r height: 10px;\r cursor: sw-resize;\r top: unset;\r bottom: -4px;\r width: 10px;\r }\r }\r \r .top-handle {\r grid-area: 1 / 1 / 1 / 1;\r background-color: transparent;\r height: 4px;\r cursor: ns-resize;\r \r &::before {\r content: "";\r width: 100%;\r position: absolute;\r top: -4px;\r bottom: 100%;\r right: 0;\r left: 0;\r margin-bottom: -8px;\r cursor: ns-resize;\r height: 8px;\r }\r }\r \r .bottom-handle {\r grid-area: 3 / 1 / 3 / 1;\r background-color: transparent;\r height: 4px;\r cursor: ns-resize;\r \r &::after {\r content: "";\r width: 100%;\r position: absolute;\r top: 100%;\r bottom: 0;\r right: 0;\r left: 0;\r margin-top: -8px;\r cursor: ns-resize;\r height: 12px;\r }\r }\r \r .expanded {\r font-size: 24px;\r }\r \r .collapsed {\r height: auto !important;\r width: 200px !important;\r z-index: 3;\r font-size: 16px;\r }\r \r .collapsedHeader {\r grid-template-columns: calc(100% - 37px) 30px 7px;\r }\r `], sourceRoot: "" }]), P.locals = { "port-container": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__port-container", "frame-box-border": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-box-border", "frame-comments": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-comments", selected: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__selected", "frame-box-header": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-box-header", "frame-box-header-button": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-box-header-button", down: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__down", "frame-box-header-collapse": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-box-header-collapse", "frame-box-header-close": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-box-header-close", "frame-box-header-title": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__frame-box-header-title", "right-handle": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__right-handle", "top-right-corner-handle": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__top-right-corner-handle", "bottom-right-corner-handle": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__bottom-right-corner-handle", "left-handle": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__left-handle", "top-left-corner-handle": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__top-left-corner-handle", "bottom-left-corner-handle": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__bottom-left-corner-handle", "top-handle": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__top-handle", "bottom-handle": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__bottom-handle", expanded: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__expanded", collapsed: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__collapsed", collapsedHeader: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphFrame-modules__collapsedHeader" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../dev/sharedUiComponents/dist/nodeGraphSystem/graphNode.modules.scss": ( /*!********************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../dev/sharedUiComponents/dist/nodeGraphSystem/graphNode.modules.scss ***! \********************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__visual { z-index: 4; width: 200px; position: absolute; left: 0; top: 0; background: gray; border: 4px solid black; border-radius: 12px; display: grid; grid-template-rows: 30px auto; grid-template-columns: 100%; color: white; box-shadow: 7px 6px 2px rgba(0, 0, 0, 0.2); } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__highlighted { animation: \\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__glow 0.5s infinite alternate; } @keyframes \\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__glow { to { border-color: white; } } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__header-container { grid-row: 1; grid-column: 1; position: relative; border: 4px solid black; border-top-right-radius: 7px; border-top-left-radius: 7px; background: black; color: white; transform: scaleX(1.01) translateY(-0.5px); transform-origin: center; display: grid; grid-template-columns: 1fr auto; grid-template-rows: 100%; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__headerIcon { z-index: 10; align-self: center; user-select: none; pointer-events: none; width: 20px; display: grid; margin-left: 6px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__header { font-size: 16px; text-align: center; white-space: nowrap; text-overflow: ellipsis; overflow: hidden; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__headerWithIcon { margin-left: calc(20px + 6px); } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__connections { grid-row: 2; grid-column: 1; display: grid; grid-template-columns: 50% 50%; transform: scale(1); } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__content { min-height: 20px; grid-row: 3; grid-column: 1; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__comments { position: absolute; top: -50px; width: 200px; height: 45px; overflow: hidden; font-style: italic; opacity: 0.8; display: grid; align-items: flex-end; pointer-events: none; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__executionTime { position: absolute; bottom: 1px; width: 195px; height: 20px; overflow: hidden; font-size: 11px; opacity: 0.5; display: grid; align-items: flex-end; justify-content: end; pointer-events: none; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__selected { border-color: white !important; }`, "", { version: 3, sources: ["webpack://./../../../dev/sharedUiComponents/dist/nodeGraphSystem/graphNode.modules.scss"], names: [], mappings: "AAAA;EACI,UAAA;EACA,YAAA;EACA,kBAAA;EACA,OAAA;EACA,MAAA;EACA,gBAAA;EACA,uBAAA;EACA,mBAAA;EACA,aAAA;EACA,6BAAA;EACA,2BAAA;EACA,YAAA;EACA,0CAAA;AACJ;;AAEA;EAMI,gHAAA;AAJJ;AADI;EACI;IACI,mBAAA;EAGV;AACF;;AAEA;EACI,WAAA;EACA,cAAA;EACA,kBAAA;EACA,uBAAA;EACA,4BAAA;EACA,2BAAA;EACA,iBAAA;EACA,YAAA;EACA,0CAAA;EACA,wBAAA;EACA,aAAA;EACA,+BAAA;EACA,wBAAA;AACJ;;AAKA;EACI,WAAA;EACA,kBAAA;EACA,iBAAA;EACA,oBAAA;EACA,WARO;EASP,aAAA;EACA,gBATS;AAOb;;AAKA;EACI,eAAA;EACA,kBAAA;EACA,mBAAA;EACA,uBAAA;EACA,gBAAA;AAFJ;;AAKA;EACI,6BAAA;AAFJ;;AAKA;EACI,WAAA;EACA,cAAA;EAEA,aAAA;EACA,8BAAA;EACA,mBAAA;AAHJ;;AAMA;EACI,gBAAA;EACA,WAAA;EACA,cAAA;AAHJ;;AAMA;EACI,kBAAA;EACA,UAAA;EACA,YAAA;EACA,YAAA;EACA,gBAAA;EACA,kBAAA;EACA,YAAA;EACA,aAAA;EACA,qBAAA;EACA,oBAAA;AAHJ;;AAMA;EACI,kBAAA;EACA,WAAA;EACA,YAAA;EACA,YAAA;EACA,gBAAA;EACA,eAAA;EACA,YAAA;EACA,aAAA;EACA,qBAAA;EACA,oBAAA;EACA,oBAAA;AAHJ;;AAMA;EACI,8BAAA;AAHJ", sourcesContent: [`.visual {\r z-index: 4;\r width: 200px;\r position: absolute;\r left: 0;\r top: 0;\r background: gray;\r border: 4px solid black;\r border-radius: 12px;\r display: grid;\r grid-template-rows: 30px auto;\r grid-template-columns: 100%;\r color: white;\r box-shadow: 7px 6px 2px rgba(0, 0, 0, 0.2);\r }\r \r .highlighted {\r @keyframes glow {\r to {\r border-color: white;\r }\r }\r animation: glow 0.5s infinite alternate;\r }\r \r .header-container {\r grid-row: 1;\r grid-column: 1;\r position: relative;\r border: 4px solid black;\r border-top-right-radius: 7px;\r border-top-left-radius: 7px;\r background: black;\r color: white;\r transform: scaleX(1.01) translateY(-0.5px);\r transform-origin: center;\r display: grid;\r grid-template-columns: 1fr auto;\r grid-template-rows: 100%;\r }\r \r $iconSize: 20px;\r $iconMargin: 6px;\r \r .headerIcon {\r z-index: 10;\r align-self: center;\r user-select: none;\r pointer-events: none;\r width: $iconSize;\r display: grid;\r margin-left: $iconMargin;\r }\r \r .header {\r font-size: 16px;\r text-align: center;\r white-space: nowrap;\r text-overflow: ellipsis;\r overflow: hidden;\r }\r \r .headerWithIcon {\r margin-left: calc(#{$iconSize} + #{$iconMargin});\r }\r \r .connections {\r grid-row: 2;\r grid-column: 1;\r \r display: grid;\r grid-template-columns: 50% 50%;\r transform: scale(1);\r }\r \r .content {\r min-height: 20px;\r grid-row: 3;\r grid-column: 1;\r }\r \r .comments {\r position: absolute;\r top: -50px;\r width: 200px;\r height: 45px;\r overflow: hidden;\r font-style: italic;\r opacity: 0.8;\r display: grid;\r align-items: flex-end;\r pointer-events: none;\r }\r \r .executionTime {\r position: absolute;\r bottom: 1px;\r width: 195px;\r height: 20px;\r overflow: hidden;\r font-size: 11px;\r opacity: 0.5;\r display: grid;\r align-items: flex-end;\r justify-content: end;\r pointer-events: none;\r }\r \r .selected {\r border-color: white !important;\r }\r `], sourceRoot: "" }]), P.locals = { visual: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__visual", highlighted: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__highlighted", glow: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__glow", "header-container": "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__header-container", headerIcon: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__headerIcon", header: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__header", headerWithIcon: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__headerWithIcon", connections: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__connections", content: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__content", comments: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__comments", executionTime: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__executionTime", selected: "---------dev-sharedUiComponents-dist-nodeGraphSystem-graphNode-modules__selected" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../dev/sharedUiComponents/dist/nodeGraphSystem/nodeLink.modules.scss": ( /*!*******************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../dev/sharedUiComponents/dist/nodeGraphSystem/nodeLink.modules.scss ***! \*******************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------dev-sharedUiComponents-dist-nodeGraphSystem-nodeLink-modules__link { stroke-width: 4px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-nodeLink-modules__link.\\---------dev-sharedUiComponents-dist-nodeGraphSystem-nodeLink-modules__selected { stroke: white !important; stroke-dasharray: 10, 2; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-nodeLink-modules__target-candidate { filter: brightness(1.5); stroke-width: 6px; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-nodeLink-modules__selection-link { pointer-events: all; stroke-width: 16px; opacity: 0; transition: opacity 75ms; stroke: transparent; cursor: pointer; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-nodeLink-modules__selection-link:hover, .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-nodeLink-modules__selection-link.\\---------dev-sharedUiComponents-dist-nodeGraphSystem-nodeLink-modules__selected { stroke: white !important; opacity: 0.4; }`, "", { version: 3, sources: ["webpack://./../../../dev/sharedUiComponents/dist/nodeGraphSystem/nodeLink.modules.scss"], names: [], mappings: "AAAA;EACI,iBAAA;AACJ;AAAI;EACI,wBAAA;EACA,uBAAA;AAER;;AAEA;EACI,uBAAA;EACA,iBAAA;AACJ;;AAEA;EACI,mBAAA;EACA,kBAAA;EACA,UAAA;EACA,wBAAA;EACA,mBAAA;EACA,eAAA;AACJ;AACI;EAEI,wBAAA;EACA,YAAA;AAAR", sourcesContent: [`.link {\r stroke-width: 4px;\r &.selected {\r stroke: white !important;\r stroke-dasharray: 10, 2;\r }\r }\r \r .target-candidate {\r filter: brightness(1.5);\r stroke-width: 6px;\r }\r \r .selection-link {\r pointer-events: all;\r stroke-width: 16px;\r opacity: 0;\r transition: opacity 75ms;\r stroke: transparent;\r cursor: pointer;\r \r &:hover,\r &.selected {\r stroke: white !important;\r opacity: 0.4;\r }\r }\r `], sourceRoot: "" }]), P.locals = { link: "---------dev-sharedUiComponents-dist-nodeGraphSystem-nodeLink-modules__link", selected: "---------dev-sharedUiComponents-dist-nodeGraphSystem-nodeLink-modules__selected", "target-candidate": "---------dev-sharedUiComponents-dist-nodeGraphSystem-nodeLink-modules__target-candidate", "selection-link": "---------dev-sharedUiComponents-dist-nodeGraphSystem-nodeLink-modules__selection-link" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../dev/sharedUiComponents/dist/nodeGraphSystem/nodePort.modules.scss": ( /*!*******************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../dev/sharedUiComponents/dist/nodeGraphSystem/nodePort.modules.scss ***! \*******************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------dev-sharedUiComponents-dist-nodeGraphSystem-nodePort-modules__selected { filter: brightness(2); } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-nodePort-modules__pip { background: green; width: 6px; height: 6px; grid-row: 1; grid-column: 1; align-self: center; justify-self: center; border-radius: 7px; border: 2px solid black; } .\\---------dev-sharedUiComponents-dist-nodeGraphSystem-nodePort-modules__pip.\\---------dev-sharedUiComponents-dist-nodeGraphSystem-nodePort-modules__hidden { display: none; } img { grid-row: 1; grid-column: 1; } img.\\---------dev-sharedUiComponents-dist-nodeGraphSystem-nodePort-modules__selected { box-shadow: 0 0 0 2px; border-radius: 50%; }`, "", { version: 3, sources: ["webpack://./../../../dev/sharedUiComponents/dist/nodeGraphSystem/nodePort.modules.scss"], names: [], mappings: "AAAA;EACI,qBAAA;AACJ;;AAEA;EACI,iBAAA;EACA,UAAA;EACA,WAAA;EACA,WAAA;EACA,cAAA;EACA,kBAAA;EACA,oBAAA;EACA,kBAAA;EACA,uBAAA;AACJ;AACI;EACI,aAAA;AACR;;AAGA;EACI,WAAA;EACA,cAAA;AAAJ;;AAGA;EACI,qBAAA;EACA,kBAAA;AAAJ", sourcesContent: [`.selected {\r filter: brightness(2);\r }\r \r .pip {\r background: green;\r width: 6px;\r height: 6px;\r grid-row: 1;\r grid-column: 1;\r align-self: center;\r justify-self: center;\r border-radius: 7px;\r border: 2px solid black;\r \r &.hidden {\r display: none;\r }\r }\r \r img {\r grid-row: 1;\r grid-column: 1;\r }\r \r img.selected {\r box-shadow: 0 0 0 2px;\r border-radius: 50%;\r }\r `], sourceRoot: "" }]), P.locals = { selected: "---------dev-sharedUiComponents-dist-nodeGraphSystem-nodePort-modules__selected", pip: "---------dev-sharedUiComponents-dist-nodeGraphSystem-nodePort-modules__pip", hidden: "---------dev-sharedUiComponents-dist-nodeGraphSystem-nodePort-modules__hidden" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/blockNodeData.modules.scss": ( /*!**************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/blockNodeData.modules.scss ***! \**************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------tools-nodeEditor-dist-graphSystem-blockNodeData-modules__hidden { display: none !important; }`, "", { version: 3, sources: ["webpack://./../../../tools/nodeEditor/dist/graphSystem/blockNodeData.modules.scss"], names: [], mappings: "AAAA;EACI,wBAAA;AACJ", sourcesContent: [`.hidden {\r display: none !important;\r }\r `], sourceRoot: "" }]), P.locals = { hidden: "---------tools-nodeEditor-dist-graphSystem-blockNodeData-modules__hidden" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/clampDisplayManager.modules.scss": ( /*!****************************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/clampDisplayManager.modules.scss ***! \****************************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------tools-nodeEditor-dist-graphSystem-display-clampDisplayManager-modules__clampBlock { grid-row: 2; height: 34px; text-align: center; font-size: 18px; font-weight: bold; margin: 0 10px; }`, "", { version: 3, sources: ["webpack://./../../../tools/nodeEditor/dist/graphSystem/display/clampDisplayManager.modules.scss"], names: [], mappings: "AAAA;EACI,WAAA;EACA,YAAA;EACA,kBAAA;EACA,eAAA;EACA,iBAAA;EACA,cAAA;AACJ", sourcesContent: [`.clampBlock {\r grid-row: 2;\r height: 34px;\r text-align: center;\r font-size: 18px;\r font-weight: bold;\r margin: 0 10px;\r }\r `], sourceRoot: "" }]), P.locals = { clampBlock: "---------tools-nodeEditor-dist-graphSystem-display-clampDisplayManager-modules__clampBlock" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/common.modules.scss": ( /*!***************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/common.modules.scss ***! \***************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------tools-nodeEditor-dist-graphSystem-display-common-modules__texture-block { grid-row: 2; height: 140px; width: 140px; overflow: hidden; border-bottom-left-radius: 7px; border: black 4px solid; border-left: 0px; border-bottom: 0px; } .\\---------tools-nodeEditor-dist-graphSystem-display-common-modules__texture-block img { width: 100%; height: 100%; pointer-events: none; } .\\---------tools-nodeEditor-dist-graphSystem-display-common-modules__texture-block img.\\---------tools-nodeEditor-dist-graphSystem-display-common-modules__empty { display: none; } .\\---------tools-nodeEditor-dist-graphSystem-display-common-modules__empty { display: none; }`, "", { version: 3, sources: ["webpack://./../../../tools/nodeEditor/dist/graphSystem/display/common.modules.scss"], names: [], mappings: "AAAA;EACI,WAAA;EACA,aAAA;EACA,YAAA;EACA,gBAAA;EACA,8BAAA;EACA,uBAAA;EACA,gBAAA;EACA,kBAAA;AACJ;AACI;EACI,WAAA;EACA,YAAA;EACA,oBAAA;AACR;AACQ;EACI,aAAA;AACZ;;AAIA;EACI,aAAA;AADJ", sourcesContent: [`.texture-block {\r grid-row: 2;\r height: 140px;\r width: 140px;\r overflow: hidden;\r border-bottom-left-radius: 7px;\r border: black 4px solid;\r border-left: 0px;\r border-bottom: 0px;\r \r img {\r width: 100%;\r height: 100%;\r pointer-events: none;\r \r &.empty {\r display: none;\r }\r }\r }\r \r .empty {\r display: none;\r }\r `], sourceRoot: "" }]), P.locals = { "texture-block": "---------tools-nodeEditor-dist-graphSystem-display-common-modules__texture-block", empty: "---------tools-nodeEditor-dist-graphSystem-display-common-modules__empty" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/curveDisplayManager.modules.scss": ( /*!****************************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/curveDisplayManager.modules.scss ***! \****************************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------tools-nodeEditor-dist-graphSystem-display-curveDisplayManager-modules__curve-block { grid-row: 2; height: 34px; text-align: center; font-size: 18px; font-weight: bold; margin: 0 10px; }`, "", { version: 3, sources: ["webpack://./../../../tools/nodeEditor/dist/graphSystem/display/curveDisplayManager.modules.scss"], names: [], mappings: "AAAA;EACI,WAAA;EACA,YAAA;EACA,kBAAA;EACA,eAAA;EACA,iBAAA;EACA,cAAA;AACJ", sourcesContent: [`.curve-block {\r grid-row: 2;\r height: 34px;\r text-align: center;\r font-size: 18px;\r font-weight: bold;\r margin: 0 10px;\r }\r `], sourceRoot: "" }]), P.locals = { "curve-block": "---------tools-nodeEditor-dist-graphSystem-display-curveDisplayManager-modules__curve-block" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/elbowDisplayManager.modules.scss": ( /*!****************************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/elbowDisplayManager.modules.scss ***! \****************************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------tools-nodeEditor-dist-graphSystem-display-elbowDisplayManager-modules__elbowBlock { width: 40px; grid-template-rows: 0px 40px 0px; border-radius: 40px; transform: translateY(-7px); } .\\---------tools-nodeEditor-dist-graphSystem-display-elbowDisplayManager-modules__hidden { display: none; } .\\---------tools-nodeEditor-dist-graphSystem-display-elbowDisplayManager-modules__translatedConnections { transform: translateY(7px); } .\\---------tools-nodeEditor-dist-graphSystem-display-elbowDisplayManager-modules__roundSelectionBorder { border-radius: 40px; }`, "", { version: 3, sources: ["webpack://./../../../tools/nodeEditor/dist/graphSystem/display/elbowDisplayManager.modules.scss"], names: [], mappings: "AAAA;EACI,WAAA;EACA,gCAAA;EACA,mBAAA;EACA,2BAAA;AACJ;;AAEA;EACI,aAAA;AACJ;;AAEA;EACI,0BAAA;AACJ;;AAEA;EACI,mBAAA;AACJ", sourcesContent: [`.elbowBlock {\r width: 40px;\r grid-template-rows: 0px 40px 0px;\r border-radius: 40px;\r transform: translateY(-7px);\r }\r \r .hidden {\r display: none;\r }\r \r .translatedConnections {\r transform: translateY(7px);\r }\r \r .roundSelectionBorder {\r border-radius: 40px;\r }\r `], sourceRoot: "" }]), P.locals = { elbowBlock: "---------tools-nodeEditor-dist-graphSystem-display-elbowDisplayManager-modules__elbowBlock", hidden: "---------tools-nodeEditor-dist-graphSystem-display-elbowDisplayManager-modules__hidden", translatedConnections: "---------tools-nodeEditor-dist-graphSystem-display-elbowDisplayManager-modules__translatedConnections", roundSelectionBorder: "---------tools-nodeEditor-dist-graphSystem-display-elbowDisplayManager-modules__roundSelectionBorder" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/gradientDisplayManager.modules.scss": ( /*!*******************************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/gradientDisplayManager.modules.scss ***! \*******************************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------tools-nodeEditor-dist-graphSystem-display-gradientDisplayManager-modules__gradientBlock { grid-row: 2; height: 34px; }`, "", { version: 3, sources: ["webpack://./../../../tools/nodeEditor/dist/graphSystem/display/gradientDisplayManager.modules.scss"], names: [], mappings: "AAAA;EACI,WAAA;EACA,YAAA;AACJ", sourcesContent: [`.gradientBlock {\r grid-row: 2;\r height: 34px;\r }\r `], sourceRoot: "" }]), P.locals = { gradientBlock: "---------tools-nodeEditor-dist-graphSystem-display-gradientDisplayManager-modules__gradientBlock" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/imageSourceDisplayManager.modules.scss": ( /*!**********************************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/imageSourceDisplayManager.modules.scss ***! \**********************************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------tools-nodeEditor-dist-graphSystem-display-imageSourceDisplayManager-modules__image-source-block { margin-top: 5px; }`, "", { version: 3, sources: ["webpack://./../../../tools/nodeEditor/dist/graphSystem/display/imageSourceDisplayManager.modules.scss"], names: [], mappings: "AAAA;EACI,eAAA;AACJ", sourcesContent: [`.image-source-block {\r margin-top: 5px;\r }\r `], sourceRoot: "" }]), P.locals = { "image-source-block": "---------tools-nodeEditor-dist-graphSystem-display-imageSourceDisplayManager-modules__image-source-block" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/inputDisplayManager.modules.scss": ( /*!****************************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/inputDisplayManager.modules.scss ***! \****************************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------tools-nodeEditor-dist-graphSystem-display-inputDisplayManager-modules__input-block { grid-row: 2; min-height: 34px; text-align: center; font-size: 18px; font-weight: bold; margin: 0 10px 5px; display: grid; align-content: center; } .\\---------tools-nodeEditor-dist-graphSystem-display-inputDisplayManager-modules__input-block.\\---------tools-nodeEditor-dist-graphSystem-display-inputDisplayManager-modules__small-font { font-size: 17px; } .\\---------tools-nodeEditor-dist-graphSystem-display-inputDisplayManager-modules__constant { border-color: #464348; background: #464348; } .\\---------tools-nodeEditor-dist-graphSystem-display-inputDisplayManager-modules__inspector { border-color: #66491b; background: #66491b; }`, "", { version: 3, sources: ["webpack://./../../../tools/nodeEditor/dist/graphSystem/display/inputDisplayManager.modules.scss"], names: [], mappings: "AAAA;EACI,WAAA;EACA,gBAAA;EACA,kBAAA;EACA,eAAA;EACA,iBAAA;EACA,kBAAA;EACA,aAAA;EACA,qBAAA;AACJ;AACI;EACI,eAAA;AACR;;AAGA;EACI,qBAAA;EACA,mBAAA;AAAJ;;AAGA;EACI,qBAAA;EACA,mBAAA;AAAJ", sourcesContent: [`.input-block {\r grid-row: 2;\r min-height: 34px;\r text-align: center;\r font-size: 18px;\r font-weight: bold;\r margin: 0 10px 5px;\r display: grid;\r align-content: center;\r \r &.small-font {\r font-size: 17px;\r }\r }\r \r .constant {\r border-color: #464348;\r background: #464348;\r }\r \r .inspector {\r border-color: #66491b;\r background: #66491b;\r }\r `], sourceRoot: "" }]), P.locals = { "input-block": "---------tools-nodeEditor-dist-graphSystem-display-inputDisplayManager-modules__input-block", "small-font": "---------tools-nodeEditor-dist-graphSystem-display-inputDisplayManager-modules__small-font", constant: "---------tools-nodeEditor-dist-graphSystem-display-inputDisplayManager-modules__constant", inspector: "---------tools-nodeEditor-dist-graphSystem-display-inputDisplayManager-modules__inspector" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/remapDisplayManager.modules.scss": ( /*!****************************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/remapDisplayManager.modules.scss ***! \****************************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------tools-nodeEditor-dist-graphSystem-display-remapDisplayManager-modules__remap-block { height: 34px; text-align: center; font-size: 18px; font-weight: bold; margin: 0 10px; }`, "", { version: 3, sources: ["webpack://./../../../tools/nodeEditor/dist/graphSystem/display/remapDisplayManager.modules.scss"], names: [], mappings: "AAAA;EACI,YAAA;EACA,kBAAA;EACA,eAAA;EACA,iBAAA;EACA,cAAA;AACJ", sourcesContent: [`.remap-block {\r height: 34px;\r text-align: center;\r font-size: 18px;\r font-weight: bold;\r margin: 0 10px;\r }\r `], sourceRoot: "" }]), P.locals = { "remap-block": "---------tools-nodeEditor-dist-graphSystem-display-remapDisplayManager-modules__remap-block" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/textureDisplayManager.modules.scss": ( /*!******************************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/textureDisplayManager.modules.scss ***! \******************************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------tools-nodeEditor-dist-graphSystem-display-textureDisplayManager-modules__regular-texture-block { margin-top: 80px; } .\\---------tools-nodeEditor-dist-graphSystem-display-textureDisplayManager-modules__reduced-texture-block { margin-top: 30px; } .\\---------tools-nodeEditor-dist-graphSystem-display-textureDisplayManager-modules__reflection-block { grid-row: 3 !important; } .\\---------tools-nodeEditor-dist-graphSystem-display-textureDisplayManager-modules__triplanar-texture-block { margin-top: 155px; } .\\---------tools-nodeEditor-dist-graphSystem-display-textureDisplayManager-modules__refraction-texture-block { margin-top: 80px; }`, "", { version: 3, sources: ["webpack://./../../../tools/nodeEditor/dist/graphSystem/display/textureDisplayManager.modules.scss"], names: [], mappings: "AAAA;EACI,gBAAA;AACJ;;AAEA;EACI,gBAAA;AACJ;;AAEA;EACI,sBAAA;AACJ;;AAEA;EACI,iBAAA;AACJ;;AAEA;EACI,gBAAA;AACJ", sourcesContent: [`.regular-texture-block {\r margin-top: 80px;\r }\r \r .reduced-texture-block {\r margin-top: 30px;\r }\r \r .reflection-block {\r grid-row: 3 !important;\r }\r \r .triplanar-texture-block {\r margin-top: 155px;\r }\r \r .refraction-texture-block {\r margin-top: 80px;\r }\r `], sourceRoot: "" }]), P.locals = { "regular-texture-block": "---------tools-nodeEditor-dist-graphSystem-display-textureDisplayManager-modules__regular-texture-block", "reduced-texture-block": "---------tools-nodeEditor-dist-graphSystem-display-textureDisplayManager-modules__reduced-texture-block", "reflection-block": "---------tools-nodeEditor-dist-graphSystem-display-textureDisplayManager-modules__reflection-block", "triplanar-texture-block": "---------tools-nodeEditor-dist-graphSystem-display-textureDisplayManager-modules__triplanar-texture-block", "refraction-texture-block": "---------tools-nodeEditor-dist-graphSystem-display-textureDisplayManager-modules__refraction-texture-block" }; const p = P; } ), /***/ "../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/trigonometryDisplayManager.modules.scss": ( /*!***********************************************************************************************************************************************************************************************************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[4].use[1]!../../../../node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[4].use[2]!../../../tools/nodeEditor/dist/graphSystem/display/trigonometryDisplayManager.modules.scss ***! \***********************************************************************************************************************************************************************************************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ default: () => p /* harmony export */ }); var d = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/sourceMaps.js */ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js" ), v = /* @__PURE__ */ o.n(d), u = o( /*! ../../../../../../node_modules/css-loader/dist/runtime/api.js */ "../../../../node_modules/css-loader/dist/runtime/api.js" ), l = /* @__PURE__ */ o.n(u), P = l()(v()); P.push([a.id, `.\\---------tools-nodeEditor-dist-graphSystem-display-trigonometryDisplayManager-modules__trigonometry-block { grid-row: 2; height: 34px; text-align: center; font-size: 18px; font-weight: bold; margin: 0 10px; }`, "", { version: 3, sources: ["webpack://./../../../tools/nodeEditor/dist/graphSystem/display/trigonometryDisplayManager.modules.scss"], names: [], mappings: "AAAA;EACI,WAAA;EACA,YAAA;EACA,kBAAA;EACA,eAAA;EACA,iBAAA;EACA,cAAA;AACJ", sourcesContent: [`.trigonometry-block {\r grid-row: 2;\r height: 34px;\r text-align: center;\r font-size: 18px;\r font-weight: bold;\r margin: 0 10px;\r }\r `], sourceRoot: "" }]), P.locals = { "trigonometry-block": "---------tools-nodeEditor-dist-graphSystem-display-trigonometryDisplayManager-modules__trigonometry-block" }; const p = P; } ), /***/ "../../../../node_modules/@fortawesome/react-fontawesome/index.es.js": ( /*!***************************************************************************!*\ !*** ../../../../node_modules/@fortawesome/react-fontawesome/index.es.js ***! \***************************************************************************/ /***/ (a, f, o) => { o.r(f), o.d(f, { /* harmony export */ FontAwesomeIcon: () => ( /* binding */ L ) /* harmony export */ }); var d = o( /*! @fortawesome/fontawesome-svg-core */ "../../../../node_modules/@fortawesome/fontawesome-svg-core/index.mjs" ), v = o( /*! prop-types */ "../../../../node_modules/prop-types/index.js" ), u = /* @__PURE__ */ o.n(v), l = o( /*! react */ "../../../../node_modules/react/index.js" ); function P(ae, Pe) { var ge = Object.keys(ae); if (Object.getOwnPropertySymbols) { var me = Object.getOwnPropertySymbols(ae); Pe && (me = me.filter(function(Xe) { return Object.getOwnPropertyDescriptor(ae, Xe).enumerable; })), ge.push.apply(ge, me); } return ge; } function p(ae) { for (var Pe = 1; Pe < arguments.length; Pe++) { var ge = arguments[Pe] != null ? arguments[Pe] : {}; Pe % 2 ? P(Object(ge), !0).forEach(function(me) { H(ae, me, ge[me]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(ae, Object.getOwnPropertyDescriptors(ge)) : P(Object(ge)).forEach(function(me) { Object.defineProperty(ae, me, Object.getOwnPropertyDescriptor(ge, me)); }); } return ae; } function c(ae) { "@babel/helpers - typeof"; return c = typeof Symbol == "function" && typeof Symbol.iterator == "symbol" ? function(Pe) { return typeof Pe; } : function(Pe) { return Pe && typeof Symbol == "function" && Pe.constructor === Symbol && Pe !== Symbol.prototype ? "symbol" : typeof Pe; }, c(ae); } function H(ae, Pe, ge) { return Pe in ae ? Object.defineProperty(ae, Pe, { value: ge, enumerable: !0, configurable: !0, writable: !0 }) : ae[Pe] = ge, ae; } function T(ae, Pe) { if (ae == null) return {}; var ge = {}, me = Object.keys(ae), Xe, De; for (De = 0; De < me.length; De++) Xe = me[De], !(Pe.indexOf(Xe) >= 0) && (ge[Xe] = ae[Xe]); return ge; } function q(ae, Pe) { if (ae == null) return {}; var ge = T(ae, Pe), me, Xe; if (Object.getOwnPropertySymbols) { var De = Object.getOwnPropertySymbols(ae); for (Xe = 0; Xe < De.length; Xe++) me = De[Xe], !(Pe.indexOf(me) >= 0) && Object.prototype.propertyIsEnumerable.call(ae, me) && (ge[me] = ae[me]); } return ge; } function b(ae) { return j(ae) || w(ae) || m(ae) || N(); } function j(ae) { if (Array.isArray(ae)) return I(ae); } function w(ae) { if (typeof Symbol < "u" && ae[Symbol.iterator] != null || ae["@@iterator"] != null) return Array.from(ae); } function m(ae, Pe) { if (ae) { if (typeof ae == "string") return I(ae, Pe); var ge = Object.prototype.toString.call(ae).slice(8, -1); if (ge === "Object" && ae.constructor && (ge = ae.constructor.name), ge === "Map" || ge === "Set") return Array.from(ae); if (ge === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(ge)) return I(ae, Pe); } } function I(ae, Pe) { (Pe == null || Pe > ae.length) && (Pe = ae.length); for (var ge = 0, me = new Array(Pe); ge < Pe; ge++) me[ge] = ae[ge]; return me; } function N() { throw new TypeError(`Invalid attempt to spread non-iterable instance. In order to be iterable, non-array objects must have a [Symbol.iterator]() method.`); } function k(ae) { var Pe, ge = ae.beat, me = ae.fade, Xe = ae.beatFade, De = ae.bounce, ne = ae.shake, re = ae.flash, ve = ae.spin, qe = ae.spinPulse, ke = ae.spinReverse, be = ae.pulse, Fe = ae.fixedWidth, Ke = ae.inverse, nt = ae.border, ut = ae.listItem, bt = ae.flip, wt = ae.size, Tt = ae.rotation, lr = ae.pull, Qt = (Pe = { "fa-beat": ge, "fa-fade": me, "fa-beat-fade": Xe, "fa-bounce": De, "fa-shake": ne, "fa-flash": re, "fa-spin": ve, "fa-spin-reverse": ke, "fa-spin-pulse": qe, "fa-pulse": be, "fa-fw": Fe, "fa-inverse": Ke, "fa-border": nt, "fa-li": ut, "fa-flip": bt === !0, "fa-flip-horizontal": bt === "horizontal" || bt === "both", "fa-flip-vertical": bt === "vertical" || bt === "both" }, H(Pe, "fa-".concat(wt), typeof wt < "u" && wt !== null), H(Pe, "fa-rotate-".concat(Tt), typeof Tt < "u" && Tt !== null && Tt !== 0), H(Pe, "fa-pull-".concat(lr), typeof lr < "u" && lr !== null), H(Pe, "fa-swap-opacity", ae.swapOpacity), Pe); return Object.keys(Qt).map(function(tr) { return Qt[tr] ? tr : null; }).filter(function(tr) { return tr; }); } function R(ae) { return ae = ae - 0, ae === ae; } function y(ae) { return R(ae) ? ae : (ae = ae.replace(/[\-_\s]+(.)?/g, function(Pe, ge) { return ge ? ge.toUpperCase() : ""; }), ae.substr(0, 1).toLowerCase() + ae.substr(1)); } var O = ["style"]; function Y(ae) { return ae.charAt(0).toUpperCase() + ae.slice(1); } function ee(ae) { return ae.split(";").map(function(Pe) { return Pe.trim(); }).filter(function(Pe) { return Pe; }).reduce(function(Pe, ge) { var me = ge.indexOf(":"), Xe = y(ge.slice(0, me)), De = ge.slice(me + 1).trim(); return Xe.startsWith("webkit") ? Pe[Y(Xe)] = De : Pe[Xe] = De, Pe; }, {}); } function Z(ae, Pe) { var ge = arguments.length > 2 && arguments[2] !== void 0 ? arguments[2] : {}; if (typeof Pe == "string") return Pe; var me = (Pe.children || []).map(function(ve) { return Z(ae, ve); }), Xe = Object.keys(Pe.attributes || {}).reduce(function(ve, qe) { var ke = Pe.attributes[qe]; switch (qe) { case "class": ve.attrs.className = ke, delete Pe.attributes.class; break; case "style": ve.attrs.style = ee(ke); break; default: qe.indexOf("aria-") === 0 || qe.indexOf("data-") === 0 ? ve.attrs[qe.toLowerCase()] = ke : ve.attrs[y(qe)] = ke; } return ve; }, { attrs: {} }), De = ge.style, ne = De === void 0 ? {} : De, re = q(ge, O); return Xe.attrs.style = p(p({}, Xe.attrs.style), ne), ae.apply(void 0, [Pe.tag, p(p({}, Xe.attrs), re)].concat(b(me))); } var te = !1; try { te = !1; } catch { } function fe() { if (!te && console && typeof console.error == "function") { var ae; (ae = console).error.apply(ae, arguments); } } function _(ae) { if (ae && c(ae) === "object" && ae.prefix && ae.iconName && ae.icon) return ae; if (d.parse.icon) return d.parse.icon(ae); if (ae === null) return null; if (ae && c(ae) === "object" && ae.prefix && ae.iconName) return ae; if (Array.isArray(ae) && ae.length === 2) return { prefix: ae[0], iconName: ae[1] }; if (typeof ae == "string") return { prefix: "fas", iconName: ae }; } function G(ae, Pe) { return Array.isArray(Pe) && Pe.length > 0 || !Array.isArray(Pe) && Pe ? H({}, ae, Pe) : {}; } var L = /* @__PURE__ */ l.forwardRef(function(ae, Pe) { var ge = ae.icon, me = ae.mask, Xe = ae.symbol, De = ae.className, ne = ae.title, re = ae.titleId, ve = ae.maskId, qe = _(ge), ke = G("classes", [].concat(b(k(ae)), b(De.split(" ")))), be = G("transform", typeof ae.transform == "string" ? d.parse.transform(ae.transform) : ae.transform), Fe = G("mask", _(me)), Ke = (0, d.icon)(qe, p(p(p(p({}, ke), be), Fe), {}, { symbol: Xe, title: ne, titleId: re, maskId: ve })); if (!Ke) return fe("Could not find icon", qe), null; var nt = Ke.abstract, ut = { ref: Pe }; return Object.keys(ae).forEach(function(bt) { L.defaultProps.hasOwnProperty(bt) || (ut[bt] = ae[bt]); }), $(nt[0], ut); }); L.displayName = "FontAwesomeIcon", L.propTypes = { beat: u().bool, border: u().bool, beatFade: u().bool, bounce: u().bool, className: u().string, fade: u().bool, flash: u().bool, mask: u().oneOfType([u().object, u().array, u().string]), maskId: u().string, fixedWidth: u().bool, inverse: u().bool, flip: u().oneOf([!0, !1, "horizontal", "vertical", "both"]), icon: u().oneOfType([u().object, u().array, u().string]), listItem: u().bool, pull: u().oneOf(["right", "left"]), pulse: u().bool, rotation: u().oneOf([0, 90, 180, 270]), shake: u().bool, size: u().oneOf(["2xs", "xs", "sm", "lg", "xl", "2xl", "1x", "2x", "3x", "4x", "5x", "6x", "7x", "8x", "9x", "10x"]), spin: u().bool, spinPulse: u().bool, spinReverse: u().bool, symbol: u().oneOfType([u().bool, u().string]), title: u().string, titleId: u().string, transform: u().oneOfType([u().string, u().object]), swapOpacity: u().bool }, L.defaultProps = { border: !1, className: "", mask: null, maskId: null, fixedWidth: !1, inverse: !1, flip: !1, icon: null, listItem: !1, pull: null, pulse: !1, rotation: null, size: null, spin: !1, spinPulse: !1, spinReverse: !1, beat: !1, fade: !1, beatFade: !1, bounce: !1, shake: !1, symbol: !1, title: "", titleId: null, transform: null, swapOpacity: !1 }; var $ = Z.bind(null, l.createElement); } ), /***/ "../../../../node_modules/css-loader/dist/runtime/api.js": ( /*!***************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/runtime/api.js ***! \***************************************************************/ /***/ (a) => { a.exports = function(f) { var o = []; return o.toString = function() { return this.map(function(v) { var u = "", l = typeof v[5] < "u"; return v[4] && (u += "@supports (".concat(v[4], ") {")), v[2] && (u += "@media ".concat(v[2], " {")), l && (u += "@layer".concat(v[5].length > 0 ? " ".concat(v[5]) : "", " {")), u += f(v), l && (u += "}"), v[2] && (u += "}"), v[4] && (u += "}"), u; }).join(""); }, o.i = function(v, u, l, P, p) { typeof v == "string" && (v = [[null, v, void 0]]); var c = {}; if (l) for (var H = 0; H < this.length; H++) { var T = this[H][0]; T != null && (c[T] = !0); } for (var q = 0; q < v.length; q++) { var b = [].concat(v[q]); l && c[b[0]] || (typeof p < "u" && (typeof b[5] > "u" || (b[1] = "@layer".concat(b[5].length > 0 ? " ".concat(b[5]) : "", " {").concat(b[1], "}")), b[5] = p), u && (b[2] && (b[1] = "@media ".concat(b[2], " {").concat(b[1], "}")), b[2] = u), P && (b[4] ? (b[1] = "@supports (".concat(b[4], ") {").concat(b[1], "}"), b[4] = P) : b[4] = "".concat(P)), o.push(b)); } }, o; }; } ), /***/ "../../../../node_modules/css-loader/dist/runtime/sourceMaps.js": ( /*!**********************************************************************!*\ !*** ../../../../node_modules/css-loader/dist/runtime/sourceMaps.js ***! \**********************************************************************/ /***/ (a) => { a.exports = function(f) { var o = f[1], d = f[3]; if (!d) return o; if (typeof btoa == "function") { var v = btoa(unescape(encodeURIComponent(JSON.stringify(d)))), u = "sourceMappingURL=data:application/json;charset=utf-8;base64,".concat(v), l = "/*# ".concat(u, " */"); return [o].concat([l]).join(` `); } return [o].join(` `); }; } ), /***/ "../../../../node_modules/dagre/index.js": ( /*!***********************************************!*\ !*** ../../../../node_modules/dagre/index.js ***! \***********************************************/ /***/ (a, f, o) => { a.exports = { graphlib: o( /*! ./lib/graphlib */ "../../../../node_modules/dagre/lib/graphlib.js" ), layout: o( /*! ./lib/layout */ "../../../../node_modules/dagre/lib/layout.js" ), debug: o( /*! ./lib/debug */ "../../../../node_modules/dagre/lib/debug.js" ), util: { time: o( /*! ./lib/util */ "../../../../node_modules/dagre/lib/util.js" ).time, notime: o( /*! ./lib/util */ "../../../../node_modules/dagre/lib/util.js" ).notime }, version: o( /*! ./lib/version */ "../../../../node_modules/dagre/lib/version.js" ) }; } ), /***/ "../../../../node_modules/dagre/lib/acyclic.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/dagre/lib/acyclic.js ***! \*****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ./greedy-fas */ "../../../../node_modules/dagre/lib/greedy-fas.js" ); a.exports = { run: u, undo: P }; function u(p) { var c = p.graph().acyclicer === "greedy" ? v(p, H(p)) : l(p); d.forEach(c, function(T) { var q = p.edge(T); p.removeEdge(T), q.forwardName = T.name, q.reversed = !0, p.setEdge(T.w, T.v, q, d.uniqueId("rev")); }); function H(T) { return function(q) { return T.edge(q).weight; }; } } function l(p) { var c = [], H = {}, T = {}; function q(b) { d.has(T, b) || (T[b] = !0, H[b] = !0, d.forEach(p.outEdges(b), function(j) { d.has(H, j.w) ? c.push(j) : q(j.w); }), delete H[b]); } return d.forEach(p.nodes(), q), c; } function P(p) { d.forEach(p.edges(), function(c) { var H = p.edge(c); if (H.reversed) { p.removeEdge(c); var T = H.forwardName; delete H.reversed, delete H.forwardName, p.setEdge(c.w, c.v, H, T); } }); } } ), /***/ "../../../../node_modules/dagre/lib/add-border-segments.js": ( /*!*****************************************************************!*\ !*** ../../../../node_modules/dagre/lib/add-border-segments.js ***! \*****************************************************************/ /***/ (a, f, o) => { var d = o( /*! ./lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ./util */ "../../../../node_modules/dagre/lib/util.js" ); a.exports = u; function u(P) { function p(c) { var H = P.children(c), T = P.node(c); if (H.length && d.forEach(H, p), d.has(T, "minRank")) { T.borderLeft = [], T.borderRight = []; for (var q = T.minRank, b = T.maxRank + 1; q < b; ++q) l(P, "borderLeft", "_bl", c, T, q), l(P, "borderRight", "_br", c, T, q); } } d.forEach(P.children(), p); } function l(P, p, c, H, T, q) { var b = { width: 0, height: 0, rank: q, borderType: p }, j = T[p][q - 1], w = v.addDummyNode(P, "border", b, c); T[p][q] = w, P.setParent(w, H), j && P.setEdge(j, w, { weight: 1 }); } } ), /***/ "../../../../node_modules/dagre/lib/coordinate-system.js": ( /*!***************************************************************!*\ !*** ../../../../node_modules/dagre/lib/coordinate-system.js ***! \***************************************************************/ /***/ (a, f, o) => { var d = o( /*! ./lodash */ "../../../../node_modules/dagre/lib/lodash.js" ); a.exports = { adjust: v, undo: u }; function v(q) { var b = q.graph().rankdir.toLowerCase(); (b === "lr" || b === "rl") && l(q); } function u(q) { var b = q.graph().rankdir.toLowerCase(); (b === "bt" || b === "rl") && p(q), (b === "lr" || b === "rl") && (H(q), l(q)); } function l(q) { d.forEach(q.nodes(), function(b) { P(q.node(b)); }), d.forEach(q.edges(), function(b) { P(q.edge(b)); }); } function P(q) { var b = q.width; q.width = q.height, q.height = b; } function p(q) { d.forEach(q.nodes(), function(b) { c(q.node(b)); }), d.forEach(q.edges(), function(b) { var j = q.edge(b); d.forEach(j.points, c), d.has(j, "y") && c(j); }); } function c(q) { q.y = -q.y; } function H(q) { d.forEach(q.nodes(), function(b) { T(q.node(b)); }), d.forEach(q.edges(), function(b) { var j = q.edge(b); d.forEach(j.points, T), d.has(j, "x") && T(j); }); } function T(q) { var b = q.x; q.x = q.y, q.y = b; } } ), /***/ "../../../../node_modules/dagre/lib/data/list.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/dagre/lib/data/list.js ***! \*******************************************************/ /***/ (a) => { a.exports = f; function f() { var v = {}; v._next = v._prev = v, this._sentinel = v; } f.prototype.dequeue = function() { var v = this._sentinel, u = v._prev; if (u !== v) return o(u), u; }, f.prototype.enqueue = function(v) { var u = this._sentinel; v._prev && v._next && o(v), v._next = u._next, u._next._prev = v, u._next = v, v._prev = u; }, f.prototype.toString = function() { for (var v = [], u = this._sentinel, l = u._prev; l !== u; ) v.push(JSON.stringify(l, d)), l = l._prev; return "[" + v.join(", ") + "]"; }; function o(v) { v._prev._next = v._next, v._next._prev = v._prev, delete v._next, delete v._prev; } function d(v, u) { if (v !== "_next" && v !== "_prev") return u; } } ), /***/ "../../../../node_modules/dagre/lib/debug.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/dagre/lib/debug.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ./util */ "../../../../node_modules/dagre/lib/util.js" ), u = o( /*! ./graphlib */ "../../../../node_modules/dagre/lib/graphlib.js" ).Graph; a.exports = { debugOrdering: l }; function l(P) { var p = v.buildLayerMatrix(P), c = new u({ compound: !0, multigraph: !0 }).setGraph({}); return d.forEach(P.nodes(), function(H) { c.setNode(H, { label: H }), c.setParent(H, "layer" + P.node(H).rank); }), d.forEach(P.edges(), function(H) { c.setEdge(H.v, H.w, {}, H.name); }), d.forEach(p, function(H, T) { var q = "layer" + T; c.setNode(q, { rank: "same" }), d.reduce(H, function(b, j) { return c.setEdge(b, j, { style: "invis" }), j; }); }), c; } } ), /***/ "../../../../node_modules/dagre/lib/graphlib.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/dagre/lib/graphlib.js ***! \******************************************************/ /***/ (a, f, o) => { var d; try { d = o( /*! graphlib */ "../../../../node_modules/graphlib/index.js" ); } catch { } d || (d = window.graphlib), a.exports = d; } ), /***/ "../../../../node_modules/dagre/lib/greedy-fas.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/dagre/lib/greedy-fas.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ./graphlib */ "../../../../node_modules/dagre/lib/graphlib.js" ).Graph, u = o( /*! ./data/list */ "../../../../node_modules/dagre/lib/data/list.js" ); a.exports = P; var l = d.constant(1); function P(q, b) { if (q.nodeCount() <= 1) return []; var j = H(q, b || l), w = p(j.graph, j.buckets, j.zeroIdx); return d.flatten(d.map(w, function(m) { return q.outEdges(m.v, m.w); }), !0); } function p(q, b, j) { for (var w = [], m = b[b.length - 1], I = b[0], N; q.nodeCount(); ) { for (; N = I.dequeue(); ) c(q, b, j, N); for (; N = m.dequeue(); ) c(q, b, j, N); if (q.nodeCount()) { for (var k = b.length - 2; k > 0; --k) if (N = b[k].dequeue(), N) { w = w.concat(c(q, b, j, N, !0)); break; } } } return w; } function c(q, b, j, w, m) { var I = m ? [] : void 0; return d.forEach(q.inEdges(w.v), function(N) { var k = q.edge(N), R = q.node(N.v); m && I.push({ v: N.v, w: N.w }), R.out -= k, T(b, j, R); }), d.forEach(q.outEdges(w.v), function(N) { var k = q.edge(N), R = N.w, y = q.node(R); y.in -= k, T(b, j, y); }), q.removeNode(w.v), I; } function H(q, b) { var j = new v(), w = 0, m = 0; d.forEach(q.nodes(), function(k) { j.setNode(k, { v: k, in: 0, out: 0 }); }), d.forEach(q.edges(), function(k) { var R = j.edge(k.v, k.w) || 0, y = b(k), O = R + y; j.setEdge(k.v, k.w, O), m = Math.max(m, j.node(k.v).out += y), w = Math.max(w, j.node(k.w).in += y); }); var I = d.range(m + w + 3).map(function() { return new u(); }), N = w + 1; return d.forEach(j.nodes(), function(k) { T(I, N, j.node(k)); }), { graph: j, buckets: I, zeroIdx: N }; } function T(q, b, j) { j.out ? j.in ? q[j.out - j.in + b].enqueue(j) : q[q.length - 1].enqueue(j) : q[0].enqueue(j); } } ), /***/ "../../../../node_modules/dagre/lib/layout.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/dagre/lib/layout.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ./acyclic */ "../../../../node_modules/dagre/lib/acyclic.js" ), u = o( /*! ./normalize */ "../../../../node_modules/dagre/lib/normalize.js" ), l = o( /*! ./rank */ "../../../../node_modules/dagre/lib/rank/index.js" ), P = o( /*! ./util */ "../../../../node_modules/dagre/lib/util.js" ).normalizeRanks, p = o( /*! ./parent-dummy-chains */ "../../../../node_modules/dagre/lib/parent-dummy-chains.js" ), c = o( /*! ./util */ "../../../../node_modules/dagre/lib/util.js" ).removeEmptyRanks, H = o( /*! ./nesting-graph */ "../../../../node_modules/dagre/lib/nesting-graph.js" ), T = o( /*! ./add-border-segments */ "../../../../node_modules/dagre/lib/add-border-segments.js" ), q = o( /*! ./coordinate-system */ "../../../../node_modules/dagre/lib/coordinate-system.js" ), b = o( /*! ./order */ "../../../../node_modules/dagre/lib/order/index.js" ), j = o( /*! ./position */ "../../../../node_modules/dagre/lib/position/index.js" ), w = o( /*! ./util */ "../../../../node_modules/dagre/lib/util.js" ), m = o( /*! ./graphlib */ "../../../../node_modules/dagre/lib/graphlib.js" ).Graph; a.exports = I; function I(be, Fe) { var Ke = Fe && Fe.debugTiming ? w.time : w.notime; Ke("layout", function() { var nt = Ke(" buildLayoutGraph", function() { return _(be); }); Ke(" runLayout", function() { N(nt, Ke); }), Ke(" updateInputGraph", function() { k(be, nt); }); }); } function N(be, Fe) { Fe(" makeSpaceForEdgeLabels", function() { G(be); }), Fe(" removeSelfEdges", function() { ne(be); }), Fe(" acyclic", function() { v.run(be); }), Fe(" nestingGraph.run", function() { H.run(be); }), Fe(" rank", function() { l(w.asNonCompoundGraph(be)); }), Fe(" injectEdgeLabelProxies", function() { L(be); }), Fe(" removeEmptyRanks", function() { c(be); }), Fe(" nestingGraph.cleanup", function() { H.cleanup(be); }), Fe(" normalizeRanks", function() { P(be); }), Fe(" assignRankMinMax", function() { $(be); }), Fe(" removeEdgeLabelProxies", function() { ae(be); }), Fe(" normalize.run", function() { u.run(be); }), Fe(" parentDummyChains", function() { p(be); }), Fe(" addBorderSegments", function() { T(be); }), Fe(" order", function() { b(be); }), Fe(" insertSelfEdges", function() { re(be); }), Fe(" adjustCoordinateSystem", function() { q.adjust(be); }), Fe(" position", function() { j(be); }), Fe(" positionSelfEdges", function() { ve(be); }), Fe(" removeBorderNodes", function() { De(be); }), Fe(" normalize.undo", function() { u.undo(be); }), Fe(" fixupEdgeLabelCoords", function() { me(be); }), Fe(" undoCoordinateSystem", function() { q.undo(be); }), Fe(" translateGraph", function() { Pe(be); }), Fe(" assignNodeIntersects", function() { ge(be); }), Fe(" reversePoints", function() { Xe(be); }), Fe(" acyclic.undo", function() { v.undo(be); }); } function k(be, Fe) { d.forEach(be.nodes(), function(Ke) { var nt = be.node(Ke), ut = Fe.node(Ke); nt && (nt.x = ut.x, nt.y = ut.y, Fe.children(Ke).length && (nt.width = ut.width, nt.height = ut.height)); }), d.forEach(be.edges(), function(Ke) { var nt = be.edge(Ke), ut = Fe.edge(Ke); nt.points = ut.points, d.has(ut, "x") && (nt.x = ut.x, nt.y = ut.y); }), be.graph().width = Fe.graph().width, be.graph().height = Fe.graph().height; } var R = ["nodesep", "edgesep", "ranksep", "marginx", "marginy"], y = { ranksep: 50, edgesep: 20, nodesep: 50, rankdir: "tb" }, O = ["acyclicer", "ranker", "rankdir", "align"], Y = ["width", "height"], ee = { width: 0, height: 0 }, Z = ["minlen", "weight", "width", "height", "labeloffset"], te = { minlen: 1, weight: 1, width: 0, height: 0, labeloffset: 10, labelpos: "r" }, fe = ["labelpos"]; function _(be) { var Fe = new m({ multigraph: !0, compound: !0 }), Ke = ke(be.graph()); return Fe.setGraph(d.merge( {}, y, qe(Ke, R), d.pick(Ke, O) )), d.forEach(be.nodes(), function(nt) { var ut = ke(be.node(nt)); Fe.setNode(nt, d.defaults(qe(ut, Y), ee)), Fe.setParent(nt, be.parent(nt)); }), d.forEach(be.edges(), function(nt) { var ut = ke(be.edge(nt)); Fe.setEdge(nt, d.merge( {}, te, qe(ut, Z), d.pick(ut, fe) )); }), Fe; } function G(be) { var Fe = be.graph(); Fe.ranksep /= 2, d.forEach(be.edges(), function(Ke) { var nt = be.edge(Ke); nt.minlen *= 2, nt.labelpos.toLowerCase() !== "c" && (Fe.rankdir === "TB" || Fe.rankdir === "BT" ? nt.width += nt.labeloffset : nt.height += nt.labeloffset); }); } function L(be) { d.forEach(be.edges(), function(Fe) { var Ke = be.edge(Fe); if (Ke.width && Ke.height) { var nt = be.node(Fe.v), ut = be.node(Fe.w), bt = { rank: (ut.rank - nt.rank) / 2 + nt.rank, e: Fe }; w.addDummyNode(be, "edge-proxy", bt, "_ep"); } }); } function $(be) { var Fe = 0; d.forEach(be.nodes(), function(Ke) { var nt = be.node(Ke); nt.borderTop && (nt.minRank = be.node(nt.borderTop).rank, nt.maxRank = be.node(nt.borderBottom).rank, Fe = d.max(Fe, nt.maxRank)); }), be.graph().maxRank = Fe; } function ae(be) { d.forEach(be.nodes(), function(Fe) { var Ke = be.node(Fe); Ke.dummy === "edge-proxy" && (be.edge(Ke.e).labelRank = Ke.rank, be.removeNode(Fe)); }); } function Pe(be) { var Fe = Number.POSITIVE_INFINITY, Ke = 0, nt = Number.POSITIVE_INFINITY, ut = 0, bt = be.graph(), wt = bt.marginx || 0, Tt = bt.marginy || 0; function lr(Qt) { var tr = Qt.x, br = Qt.y, Xn = Qt.width, qr = Qt.height; Fe = Math.min(Fe, tr - Xn / 2), Ke = Math.max(Ke, tr + Xn / 2), nt = Math.min(nt, br - qr / 2), ut = Math.max(ut, br + qr / 2); } d.forEach(be.nodes(), function(Qt) { lr(be.node(Qt)); }), d.forEach(be.edges(), function(Qt) { var tr = be.edge(Qt); d.has(tr, "x") && lr(tr); }), Fe -= wt, nt -= Tt, d.forEach(be.nodes(), function(Qt) { var tr = be.node(Qt); tr.x -= Fe, tr.y -= nt; }), d.forEach(be.edges(), function(Qt) { var tr = be.edge(Qt); d.forEach(tr.points, function(br) { br.x -= Fe, br.y -= nt; }), d.has(tr, "x") && (tr.x -= Fe), d.has(tr, "y") && (tr.y -= nt); }), bt.width = Ke - Fe + wt, bt.height = ut - nt + Tt; } function ge(be) { d.forEach(be.edges(), function(Fe) { var Ke = be.edge(Fe), nt = be.node(Fe.v), ut = be.node(Fe.w), bt, wt; Ke.points ? (bt = Ke.points[0], wt = Ke.points[Ke.points.length - 1]) : (Ke.points = [], bt = ut, wt = nt), Ke.points.unshift(w.intersectRect(nt, bt)), Ke.points.push(w.intersectRect(ut, wt)); }); } function me(be) { d.forEach(be.edges(), function(Fe) { var Ke = be.edge(Fe); if (d.has(Ke, "x")) switch ((Ke.labelpos === "l" || Ke.labelpos === "r") && (Ke.width -= Ke.labeloffset), Ke.labelpos) { case "l": Ke.x -= Ke.width / 2 + Ke.labeloffset; break; case "r": Ke.x += Ke.width / 2 + Ke.labeloffset; break; } }); } function Xe(be) { d.forEach(be.edges(), function(Fe) { var Ke = be.edge(Fe); Ke.reversed && Ke.points.reverse(); }); } function De(be) { d.forEach(be.nodes(), function(Fe) { if (be.children(Fe).length) { var Ke = be.node(Fe), nt = be.node(Ke.borderTop), ut = be.node(Ke.borderBottom), bt = be.node(d.last(Ke.borderLeft)), wt = be.node(d.last(Ke.borderRight)); Ke.width = Math.abs(wt.x - bt.x), Ke.height = Math.abs(ut.y - nt.y), Ke.x = bt.x + Ke.width / 2, Ke.y = nt.y + Ke.height / 2; } }), d.forEach(be.nodes(), function(Fe) { be.node(Fe).dummy === "border" && be.removeNode(Fe); }); } function ne(be) { d.forEach(be.edges(), function(Fe) { if (Fe.v === Fe.w) { var Ke = be.node(Fe.v); Ke.selfEdges || (Ke.selfEdges = []), Ke.selfEdges.push({ e: Fe, label: be.edge(Fe) }), be.removeEdge(Fe); } }); } function re(be) { var Fe = w.buildLayerMatrix(be); d.forEach(Fe, function(Ke) { var nt = 0; d.forEach(Ke, function(ut, bt) { var wt = be.node(ut); wt.order = bt + nt, d.forEach(wt.selfEdges, function(Tt) { w.addDummyNode(be, "selfedge", { width: Tt.label.width, height: Tt.label.height, rank: wt.rank, order: bt + ++nt, e: Tt.e, label: Tt.label }, "_se"); }), delete wt.selfEdges; }); }); } function ve(be) { d.forEach(be.nodes(), function(Fe) { var Ke = be.node(Fe); if (Ke.dummy === "selfedge") { var nt = be.node(Ke.e.v), ut = nt.x + nt.width / 2, bt = nt.y, wt = Ke.x - ut, Tt = nt.height / 2; be.setEdge(Ke.e, Ke.label), be.removeNode(Fe), Ke.label.points = [ { x: ut + 2 * wt / 3, y: bt - Tt }, { x: ut + 5 * wt / 6, y: bt - Tt }, { x: ut + wt, y: bt }, { x: ut + 5 * wt / 6, y: bt + Tt }, { x: ut + 2 * wt / 3, y: bt + Tt } ], Ke.label.x = Ke.x, Ke.label.y = Ke.y; } }); } function qe(be, Fe) { return d.mapValues(d.pick(be, Fe), Number); } function ke(be) { var Fe = {}; return d.forEach(be, function(Ke, nt) { Fe[nt.toLowerCase()] = Ke; }), Fe; } } ), /***/ "../../../../node_modules/dagre/lib/lodash.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/dagre/lib/lodash.js ***! \****************************************************/ /***/ (a, f, o) => { var d; try { d = { cloneDeep: o( /*! lodash/cloneDeep */ "../../../../node_modules/lodash/cloneDeep.js" ), constant: o( /*! lodash/constant */ "../../../../node_modules/lodash/constant.js" ), defaults: o( /*! lodash/defaults */ "../../../../node_modules/lodash/defaults.js" ), each: o( /*! lodash/each */ "../../../../node_modules/lodash/each.js" ), filter: o( /*! lodash/filter */ "../../../../node_modules/lodash/filter.js" ), find: o( /*! lodash/find */ "../../../../node_modules/lodash/find.js" ), flatten: o( /*! lodash/flatten */ "../../../../node_modules/lodash/flatten.js" ), forEach: o( /*! lodash/forEach */ "../../../../node_modules/lodash/forEach.js" ), forIn: o( /*! lodash/forIn */ "../../../../node_modules/lodash/forIn.js" ), has: o( /*! lodash/has */ "../../../../node_modules/lodash/has.js" ), isUndefined: o( /*! lodash/isUndefined */ "../../../../node_modules/lodash/isUndefined.js" ), last: o( /*! lodash/last */ "../../../../node_modules/lodash/last.js" ), map: o( /*! lodash/map */ "../../../../node_modules/lodash/map.js" ), mapValues: o( /*! lodash/mapValues */ "../../../../node_modules/lodash/mapValues.js" ), max: o( /*! lodash/max */ "../../../../node_modules/lodash/max.js" ), merge: o( /*! lodash/merge */ "../../../../node_modules/lodash/merge.js" ), min: o( /*! lodash/min */ "../../../../node_modules/lodash/min.js" ), minBy: o( /*! lodash/minBy */ "../../../../node_modules/lodash/minBy.js" ), now: o( /*! lodash/now */ "../../../../node_modules/lodash/now.js" ), pick: o( /*! lodash/pick */ "../../../../node_modules/lodash/pick.js" ), range: o( /*! lodash/range */ "../../../../node_modules/lodash/range.js" ), reduce: o( /*! lodash/reduce */ "../../../../node_modules/lodash/reduce.js" ), sortBy: o( /*! lodash/sortBy */ "../../../../node_modules/lodash/sortBy.js" ), uniqueId: o( /*! lodash/uniqueId */ "../../../../node_modules/lodash/uniqueId.js" ), values: o( /*! lodash/values */ "../../../../node_modules/lodash/values.js" ), zipObject: o( /*! lodash/zipObject */ "../../../../node_modules/lodash/zipObject.js" ) }; } catch { } d || (d = window._), a.exports = d; } ), /***/ "../../../../node_modules/dagre/lib/nesting-graph.js": ( /*!***********************************************************!*\ !*** ../../../../node_modules/dagre/lib/nesting-graph.js ***! \***********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ./util */ "../../../../node_modules/dagre/lib/util.js" ); a.exports = { run: u, cleanup: c }; function u(H) { var T = v.addDummyNode(H, "root", {}, "_root"), q = P(H), b = d.max(d.values(q)) - 1, j = 2 * b + 1; H.graph().nestingRoot = T, d.forEach(H.edges(), function(m) { H.edge(m).minlen *= j; }); var w = p(H) + 1; d.forEach(H.children(), function(m) { l(H, T, j, w, b, q, m); }), H.graph().nodeRankFactor = j; } function l(H, T, q, b, j, w, m) { var I = H.children(m); if (!I.length) { m !== T && H.setEdge(T, m, { weight: 0, minlen: q }); return; } var N = v.addBorderNode(H, "_bt"), k = v.addBorderNode(H, "_bb"), R = H.node(m); H.setParent(N, m), R.borderTop = N, H.setParent(k, m), R.borderBottom = k, d.forEach(I, function(y) { l(H, T, q, b, j, w, y); var O = H.node(y), Y = O.borderTop ? O.borderTop : y, ee = O.borderBottom ? O.borderBottom : y, Z = O.borderTop ? b : 2 * b, te = Y !== ee ? 1 : j - w[m] + 1; H.setEdge(N, Y, { weight: Z, minlen: te, nestingEdge: !0 }), H.setEdge(ee, k, { weight: Z, minlen: te, nestingEdge: !0 }); }), H.parent(m) || H.setEdge(T, N, { weight: 0, minlen: j + w[m] }); } function P(H) { var T = {}; function q(b, j) { var w = H.children(b); w && w.length && d.forEach(w, function(m) { q(m, j + 1); }), T[b] = j; } return d.forEach(H.children(), function(b) { q(b, 1); }), T; } function p(H) { return d.reduce(H.edges(), function(T, q) { return T + H.edge(q).weight; }, 0); } function c(H) { var T = H.graph(); H.removeNode(T.nestingRoot), delete T.nestingRoot, d.forEach(H.edges(), function(q) { var b = H.edge(q); b.nestingEdge && H.removeEdge(q); }); } } ), /***/ "../../../../node_modules/dagre/lib/normalize.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/dagre/lib/normalize.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ./util */ "../../../../node_modules/dagre/lib/util.js" ); a.exports = { run: u, undo: P }; function u(p) { p.graph().dummyChains = [], d.forEach(p.edges(), function(c) { l(p, c); }); } function l(p, c) { var H = c.v, T = p.node(H).rank, q = c.w, b = p.node(q).rank, j = c.name, w = p.edge(c), m = w.labelRank; if (b !== T + 1) { p.removeEdge(c); var I, N, k; for (k = 0, ++T; T < b; ++k, ++T) w.points = [], N = { width: 0, height: 0, edgeLabel: w, edgeObj: c, rank: T }, I = v.addDummyNode(p, "edge", N, "_d"), T === m && (N.width = w.width, N.height = w.height, N.dummy = "edge-label", N.labelpos = w.labelpos), p.setEdge(H, I, { weight: w.weight }, j), k === 0 && p.graph().dummyChains.push(I), H = I; p.setEdge(H, q, { weight: w.weight }, j); } } function P(p) { d.forEach(p.graph().dummyChains, function(c) { var H = p.node(c), T = H.edgeLabel, q; for (p.setEdge(H.edgeObj, T); H.dummy; ) q = p.successors(c)[0], p.removeNode(c), T.points.push({ x: H.x, y: H.y }), H.dummy === "edge-label" && (T.x = H.x, T.y = H.y, T.width = H.width, T.height = H.height), c = q, H = p.node(c); }); } } ), /***/ "../../../../node_modules/dagre/lib/order/add-subgraph-constraints.js": ( /*!****************************************************************************!*\ !*** ../../../../node_modules/dagre/lib/order/add-subgraph-constraints.js ***! \****************************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/dagre/lib/lodash.js" ); a.exports = v; function v(u, l, P) { var p = {}, c; d.forEach(P, function(H) { for (var T = u.parent(H), q, b; T; ) { if (q = u.parent(T), q ? (b = p[q], p[q] = T) : (b = c, c = T), b && b !== T) { l.setEdge(b, T); return; } T = q; } }); } } ), /***/ "../../../../node_modules/dagre/lib/order/barycenter.js": ( /*!**************************************************************!*\ !*** ../../../../node_modules/dagre/lib/order/barycenter.js ***! \**************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/dagre/lib/lodash.js" ); a.exports = v; function v(u, l) { return d.map(l, function(P) { var p = u.inEdges(P); if (p.length) { var c = d.reduce(p, function(H, T) { var q = u.edge(T), b = u.node(T.v); return { sum: H.sum + q.weight * b.order, weight: H.weight + q.weight }; }, { sum: 0, weight: 0 }); return { v: P, barycenter: c.sum / c.weight, weight: c.weight }; } else return { v: P }; }); } } ), /***/ "../../../../node_modules/dagre/lib/order/build-layer-graph.js": ( /*!*********************************************************************!*\ !*** ../../../../node_modules/dagre/lib/order/build-layer-graph.js ***! \*********************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ../graphlib */ "../../../../node_modules/dagre/lib/graphlib.js" ).Graph; a.exports = u; function u(P, p, c) { var H = l(P), T = new v({ compound: !0 }).setGraph({ root: H }).setDefaultNodeLabel(function(q) { return P.node(q); }); return d.forEach(P.nodes(), function(q) { var b = P.node(q), j = P.parent(q); (b.rank === p || b.minRank <= p && p <= b.maxRank) && (T.setNode(q), T.setParent(q, j || H), d.forEach(P[c](q), function(w) { var m = w.v === q ? w.w : w.v, I = T.edge(m, q), N = d.isUndefined(I) ? 0 : I.weight; T.setEdge(m, q, { weight: P.edge(w).weight + N }); }), d.has(b, "minRank") && T.setNode(q, { borderLeft: b.borderLeft[p], borderRight: b.borderRight[p] })); }), T; } function l(P) { for (var p; P.hasNode(p = d.uniqueId("_root")); ) ; return p; } } ), /***/ "../../../../node_modules/dagre/lib/order/cross-count.js": ( /*!***************************************************************!*\ !*** ../../../../node_modules/dagre/lib/order/cross-count.js ***! \***************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/dagre/lib/lodash.js" ); a.exports = v; function v(l, P) { for (var p = 0, c = 1; c < P.length; ++c) p += u(l, P[c - 1], P[c]); return p; } function u(l, P, p) { for (var c = d.zipObject( p, d.map(p, function(w, m) { return m; }) ), H = d.flatten(d.map(P, function(w) { return d.sortBy(d.map(l.outEdges(w), function(m) { return { pos: c[m.w], weight: l.edge(m).weight }; }), "pos"); }), !0), T = 1; T < p.length; ) T <<= 1; var q = 2 * T - 1; T -= 1; var b = d.map(new Array(q), function() { return 0; }), j = 0; return d.forEach(H.forEach(function(w) { var m = w.pos + T; b[m] += w.weight; for (var I = 0; m > 0; ) m % 2 && (I += b[m + 1]), m = m - 1 >> 1, b[m] += w.weight; j += w.weight * I; })), j; } } ), /***/ "../../../../node_modules/dagre/lib/order/index.js": ( /*!*********************************************************!*\ !*** ../../../../node_modules/dagre/lib/order/index.js ***! \*********************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ./init-order */ "../../../../node_modules/dagre/lib/order/init-order.js" ), u = o( /*! ./cross-count */ "../../../../node_modules/dagre/lib/order/cross-count.js" ), l = o( /*! ./sort-subgraph */ "../../../../node_modules/dagre/lib/order/sort-subgraph.js" ), P = o( /*! ./build-layer-graph */ "../../../../node_modules/dagre/lib/order/build-layer-graph.js" ), p = o( /*! ./add-subgraph-constraints */ "../../../../node_modules/dagre/lib/order/add-subgraph-constraints.js" ), c = o( /*! ../graphlib */ "../../../../node_modules/dagre/lib/graphlib.js" ).Graph, H = o( /*! ../util */ "../../../../node_modules/dagre/lib/util.js" ); a.exports = T; function T(w) { var m = H.maxRank(w), I = q(w, d.range(1, m + 1), "inEdges"), N = q(w, d.range(m - 1, -1, -1), "outEdges"), k = v(w); j(w, k); for (var R = Number.POSITIVE_INFINITY, y, O = 0, Y = 0; Y < 4; ++O, ++Y) { b(O % 2 ? I : N, O % 4 >= 2), k = H.buildLayerMatrix(w); var ee = u(w, k); ee < R && (Y = 0, y = d.cloneDeep(k), R = ee); } j(w, y); } function q(w, m, I) { return d.map(m, function(N) { return P(w, N, I); }); } function b(w, m) { var I = new c(); d.forEach(w, function(N) { var k = N.graph().root, R = l(N, k, I, m); d.forEach(R.vs, function(y, O) { N.node(y).order = O; }), p(N, I, R.vs); }); } function j(w, m) { d.forEach(m, function(I) { d.forEach(I, function(N, k) { w.node(N).order = k; }); }); } } ), /***/ "../../../../node_modules/dagre/lib/order/init-order.js": ( /*!**************************************************************!*\ !*** ../../../../node_modules/dagre/lib/order/init-order.js ***! \**************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/dagre/lib/lodash.js" ); a.exports = v; function v(u) { var l = {}, P = d.filter(u.nodes(), function(q) { return !u.children(q).length; }), p = d.max(d.map(P, function(q) { return u.node(q).rank; })), c = d.map(d.range(p + 1), function() { return []; }); function H(q) { if (!d.has(l, q)) { l[q] = !0; var b = u.node(q); c[b.rank].push(q), d.forEach(u.successors(q), H); } } var T = d.sortBy(P, function(q) { return u.node(q).rank; }); return d.forEach(T, H), c; } } ), /***/ "../../../../node_modules/dagre/lib/order/resolve-conflicts.js": ( /*!*********************************************************************!*\ !*** ../../../../node_modules/dagre/lib/order/resolve-conflicts.js ***! \*********************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/dagre/lib/lodash.js" ); a.exports = v; function v(P, p) { var c = {}; d.forEach(P, function(T, q) { var b = c[T.v] = { indegree: 0, in: [], out: [], vs: [T.v], i: q }; d.isUndefined(T.barycenter) || (b.barycenter = T.barycenter, b.weight = T.weight); }), d.forEach(p.edges(), function(T) { var q = c[T.v], b = c[T.w]; !d.isUndefined(q) && !d.isUndefined(b) && (b.indegree++, q.out.push(c[T.w])); }); var H = d.filter(c, function(T) { return !T.indegree; }); return u(H); } function u(P) { var p = []; function c(q) { return function(b) { b.merged || (d.isUndefined(b.barycenter) || d.isUndefined(q.barycenter) || b.barycenter >= q.barycenter) && l(q, b); }; } function H(q) { return function(b) { b.in.push(q), --b.indegree === 0 && P.push(b); }; } for (; P.length; ) { var T = P.pop(); p.push(T), d.forEach(T.in.reverse(), c(T)), d.forEach(T.out, H(T)); } return d.map( d.filter(p, function(q) { return !q.merged; }), function(q) { return d.pick(q, ["vs", "i", "barycenter", "weight"]); } ); } function l(P, p) { var c = 0, H = 0; P.weight && (c += P.barycenter * P.weight, H += P.weight), p.weight && (c += p.barycenter * p.weight, H += p.weight), P.vs = p.vs.concat(P.vs), P.barycenter = c / H, P.weight = H, P.i = Math.min(p.i, P.i), p.merged = !0; } } ), /***/ "../../../../node_modules/dagre/lib/order/sort-subgraph.js": ( /*!*****************************************************************!*\ !*** ../../../../node_modules/dagre/lib/order/sort-subgraph.js ***! \*****************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ./barycenter */ "../../../../node_modules/dagre/lib/order/barycenter.js" ), u = o( /*! ./resolve-conflicts */ "../../../../node_modules/dagre/lib/order/resolve-conflicts.js" ), l = o( /*! ./sort */ "../../../../node_modules/dagre/lib/order/sort.js" ); a.exports = P; function P(H, T, q, b) { var j = H.children(T), w = H.node(T), m = w ? w.borderLeft : void 0, I = w ? w.borderRight : void 0, N = {}; m && (j = d.filter(j, function(ee) { return ee !== m && ee !== I; })); var k = v(H, j); d.forEach(k, function(ee) { if (H.children(ee.v).length) { var Z = P(H, ee.v, q, b); N[ee.v] = Z, d.has(Z, "barycenter") && c(ee, Z); } }); var R = u(k, q); p(R, N); var y = l(R, b); if (m && (y.vs = d.flatten([m, y.vs, I], !0), H.predecessors(m).length)) { var O = H.node(H.predecessors(m)[0]), Y = H.node(H.predecessors(I)[0]); d.has(y, "barycenter") || (y.barycenter = 0, y.weight = 0), y.barycenter = (y.barycenter * y.weight + O.order + Y.order) / (y.weight + 2), y.weight += 2; } return y; } function p(H, T) { d.forEach(H, function(q) { q.vs = d.flatten(q.vs.map(function(b) { return T[b] ? T[b].vs : b; }), !0); }); } function c(H, T) { d.isUndefined(H.barycenter) ? (H.barycenter = T.barycenter, H.weight = T.weight) : (H.barycenter = (H.barycenter * H.weight + T.barycenter * T.weight) / (H.weight + T.weight), H.weight += T.weight); } } ), /***/ "../../../../node_modules/dagre/lib/order/sort.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/dagre/lib/order/sort.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ../util */ "../../../../node_modules/dagre/lib/util.js" ); a.exports = u; function u(p, c) { var H = v.partition(p, function(N) { return d.has(N, "barycenter"); }), T = H.lhs, q = d.sortBy(H.rhs, function(N) { return -N.i; }), b = [], j = 0, w = 0, m = 0; T.sort(P(!!c)), m = l(b, q, m), d.forEach(T, function(N) { m += N.vs.length, b.push(N.vs), j += N.barycenter * N.weight, w += N.weight, m = l(b, q, m); }); var I = { vs: d.flatten(b, !0) }; return w && (I.barycenter = j / w, I.weight = w), I; } function l(p, c, H) { for (var T; c.length && (T = d.last(c)).i <= H; ) c.pop(), p.push(T.vs), H++; return H; } function P(p) { return function(c, H) { return c.barycenter < H.barycenter ? -1 : c.barycenter > H.barycenter ? 1 : p ? H.i - c.i : c.i - H.i; }; } } ), /***/ "../../../../node_modules/dagre/lib/parent-dummy-chains.js": ( /*!*****************************************************************!*\ !*** ../../../../node_modules/dagre/lib/parent-dummy-chains.js ***! \*****************************************************************/ /***/ (a, f, o) => { var d = o( /*! ./lodash */ "../../../../node_modules/dagre/lib/lodash.js" ); a.exports = v; function v(P) { var p = l(P); d.forEach(P.graph().dummyChains, function(c) { for (var H = P.node(c), T = H.edgeObj, q = u(P, p, T.v, T.w), b = q.path, j = q.lca, w = 0, m = b[w], I = !0; c !== T.w; ) { if (H = P.node(c), I) { for (; (m = b[w]) !== j && P.node(m).maxRank < H.rank; ) w++; m === j && (I = !1); } if (!I) { for (; w < b.length - 1 && P.node(m = b[w + 1]).minRank <= H.rank; ) w++; m = b[w]; } P.setParent(c, m), c = P.successors(c)[0]; } }); } function u(P, p, c, H) { var T = [], q = [], b = Math.min(p[c].low, p[H].low), j = Math.max(p[c].lim, p[H].lim), w, m; w = c; do w = P.parent(w), T.push(w); while (w && (p[w].low > b || j > p[w].lim)); for (m = w, w = H; (w = P.parent(w)) !== m; ) q.push(w); return { path: T.concat(q.reverse()), lca: m }; } function l(P) { var p = {}, c = 0; function H(T) { var q = c; d.forEach(P.children(T), H), p[T] = { low: q, lim: c++ }; } return d.forEach(P.children(), H), p; } } ), /***/ "../../../../node_modules/dagre/lib/position/bk.js": ( /*!*********************************************************!*\ !*** ../../../../node_modules/dagre/lib/position/bk.js ***! \*********************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ../graphlib */ "../../../../node_modules/dagre/lib/graphlib.js" ).Graph, u = o( /*! ../util */ "../../../../node_modules/dagre/lib/util.js" ); a.exports = { positionX: I, findType1Conflicts: l, findType2Conflicts: P, addConflict: c, hasConflict: H, verticalAlignment: T, horizontalCompaction: q, alignCoordinates: w, findSmallestWidthAlignment: j, balance: m }; function l(R, y) { var O = {}; function Y(ee, Z) { var te = 0, fe = 0, _ = ee.length, G = d.last(Z); return d.forEach(Z, function(L, $) { var ae = p(R, L), Pe = ae ? R.node(ae).order : _; (ae || L === G) && (d.forEach(Z.slice(fe, $ + 1), function(ge) { d.forEach(R.predecessors(ge), function(me) { var Xe = R.node(me), De = Xe.order; (De < te || Pe < De) && !(Xe.dummy && R.node(ge).dummy) && c(O, me, ge); }); }), fe = $ + 1, te = Pe); }), Z; } return d.reduce(y, Y), O; } function P(R, y) { var O = {}; function Y(Z, te, fe, _, G) { var L; d.forEach(d.range(te, fe), function($) { L = Z[$], R.node(L).dummy && d.forEach(R.predecessors(L), function(ae) { var Pe = R.node(ae); Pe.dummy && (Pe.order < _ || Pe.order > G) && c(O, ae, L); }); }); } function ee(Z, te) { var fe = -1, _, G = 0; return d.forEach(te, function(L, $) { if (R.node(L).dummy === "border") { var ae = R.predecessors(L); ae.length && (_ = R.node(ae[0]).order, Y(te, G, $, fe, _), G = $, fe = _); } Y(te, G, te.length, _, Z.length); }), te; } return d.reduce(y, ee), O; } function p(R, y) { if (R.node(y).dummy) return d.find(R.predecessors(y), function(O) { return R.node(O).dummy; }); } function c(R, y, O) { if (y > O) { var Y = y; y = O, O = Y; } var ee = R[y]; ee || (R[y] = ee = {}), ee[O] = !0; } function H(R, y, O) { if (y > O) { var Y = y; y = O, O = Y; } return d.has(R[y], O); } function T(R, y, O, Y) { var ee = {}, Z = {}, te = {}; return d.forEach(y, function(fe) { d.forEach(fe, function(_, G) { ee[_] = _, Z[_] = _, te[_] = G; }); }), d.forEach(y, function(fe) { var _ = -1; d.forEach(fe, function(G) { var L = Y(G); if (L.length) { L = d.sortBy(L, function(me) { return te[me]; }); for (var $ = (L.length - 1) / 2, ae = Math.floor($), Pe = Math.ceil($); ae <= Pe; ++ae) { var ge = L[ae]; Z[G] === G && _ < te[ge] && !H(O, G, ge) && (Z[ge] = G, Z[G] = ee[G] = ee[ge], _ = te[ge]); } } }); }), { root: ee, align: Z }; } function q(R, y, O, Y, ee) { var Z = {}, te = b(R, y, O, ee), fe = ee ? "borderLeft" : "borderRight"; function _($, ae) { for (var Pe = te.nodes(), ge = Pe.pop(), me = {}; ge; ) me[ge] ? $(ge) : (me[ge] = !0, Pe.push(ge), Pe = Pe.concat(ae(ge))), ge = Pe.pop(); } function G($) { Z[$] = te.inEdges($).reduce(function(ae, Pe) { return Math.max(ae, Z[Pe.v] + te.edge(Pe)); }, 0); } function L($) { var ae = te.outEdges($).reduce(function(ge, me) { return Math.min(ge, Z[me.w] - te.edge(me)); }, Number.POSITIVE_INFINITY), Pe = R.node($); ae !== Number.POSITIVE_INFINITY && Pe.borderType !== fe && (Z[$] = Math.max(Z[$], ae)); } return _(G, te.predecessors.bind(te)), _(L, te.successors.bind(te)), d.forEach(Y, function($) { Z[$] = Z[O[$]]; }), Z; } function b(R, y, O, Y) { var ee = new v(), Z = R.graph(), te = N(Z.nodesep, Z.edgesep, Y); return d.forEach(y, function(fe) { var _; d.forEach(fe, function(G) { var L = O[G]; if (ee.setNode(L), _) { var $ = O[_], ae = ee.edge($, L); ee.setEdge($, L, Math.max(te(R, G, _), ae || 0)); } _ = G; }); }), ee; } function j(R, y) { return d.minBy(d.values(y), function(O) { var Y = Number.NEGATIVE_INFINITY, ee = Number.POSITIVE_INFINITY; return d.forIn(O, function(Z, te) { var fe = k(R, te) / 2; Y = Math.max(Z + fe, Y), ee = Math.min(Z - fe, ee); }), Y - ee; }); } function w(R, y) { var O = d.values(y), Y = d.min(O), ee = d.max(O); d.forEach(["u", "d"], function(Z) { d.forEach(["l", "r"], function(te) { var fe = Z + te, _ = R[fe], G; if (_ !== y) { var L = d.values(_); G = te === "l" ? Y - d.min(L) : ee - d.max(L), G && (R[fe] = d.mapValues(_, function($) { return $ + G; })); } }); }); } function m(R, y) { return d.mapValues(R.ul, function(O, Y) { if (y) return R[y.toLowerCase()][Y]; var ee = d.sortBy(d.map(R, Y)); return (ee[1] + ee[2]) / 2; }); } function I(R) { var y = u.buildLayerMatrix(R), O = d.merge( l(R, y), P(R, y) ), Y = {}, ee; d.forEach(["u", "d"], function(te) { ee = te === "u" ? y : d.values(y).reverse(), d.forEach(["l", "r"], function(fe) { fe === "r" && (ee = d.map(ee, function($) { return d.values($).reverse(); })); var _ = (te === "u" ? R.predecessors : R.successors).bind(R), G = T(R, ee, O, _), L = q( R, ee, G.root, G.align, fe === "r" ); fe === "r" && (L = d.mapValues(L, function($) { return -$; })), Y[te + fe] = L; }); }); var Z = j(R, Y); return w(Y, Z), m(Y, R.graph().align); } function N(R, y, O) { return function(Y, ee, Z) { var te = Y.node(ee), fe = Y.node(Z), _ = 0, G; if (_ += te.width / 2, d.has(te, "labelpos")) switch (te.labelpos.toLowerCase()) { case "l": G = -te.width / 2; break; case "r": G = te.width / 2; break; } if (G && (_ += O ? G : -G), G = 0, _ += (te.dummy ? y : R) / 2, _ += (fe.dummy ? y : R) / 2, _ += fe.width / 2, d.has(fe, "labelpos")) switch (fe.labelpos.toLowerCase()) { case "l": G = fe.width / 2; break; case "r": G = -fe.width / 2; break; } return G && (_ += O ? G : -G), G = 0, _; }; } function k(R, y) { return R.node(y).width; } } ), /***/ "../../../../node_modules/dagre/lib/position/index.js": ( /*!************************************************************!*\ !*** ../../../../node_modules/dagre/lib/position/index.js ***! \************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ../util */ "../../../../node_modules/dagre/lib/util.js" ), u = o( /*! ./bk */ "../../../../node_modules/dagre/lib/position/bk.js" ).positionX; a.exports = l; function l(p) { p = v.asNonCompoundGraph(p), P(p), d.forEach(u(p), function(c, H) { p.node(H).x = c; }); } function P(p) { var c = v.buildLayerMatrix(p), H = p.graph().ranksep, T = 0; d.forEach(c, function(q) { var b = d.max(d.map(q, function(j) { return p.node(j).height; })); d.forEach(q, function(j) { p.node(j).y = T + b / 2; }), T += b + H; }); } } ), /***/ "../../../../node_modules/dagre/lib/rank/feasible-tree.js": ( /*!****************************************************************!*\ !*** ../../../../node_modules/dagre/lib/rank/feasible-tree.js ***! \****************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ../graphlib */ "../../../../node_modules/dagre/lib/graphlib.js" ).Graph, u = o( /*! ./util */ "../../../../node_modules/dagre/lib/rank/util.js" ).slack; a.exports = l; function l(H) { var T = new v({ directed: !1 }), q = H.nodes()[0], b = H.nodeCount(); T.setNode(q, {}); for (var j, w; P(T, H) < b; ) j = p(T, H), w = T.hasNode(j.v) ? u(H, j) : -u(H, j), c(T, H, w); return T; } function P(H, T) { function q(b) { d.forEach(T.nodeEdges(b), function(j) { var w = j.v, m = b === w ? j.w : w; !H.hasNode(m) && !u(T, j) && (H.setNode(m, {}), H.setEdge(b, m, {}), q(m)); }); } return d.forEach(H.nodes(), q), H.nodeCount(); } function p(H, T) { return d.minBy(T.edges(), function(q) { if (H.hasNode(q.v) !== H.hasNode(q.w)) return u(T, q); }); } function c(H, T, q) { d.forEach(H.nodes(), function(b) { T.node(b).rank += q; }); } } ), /***/ "../../../../node_modules/dagre/lib/rank/index.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/dagre/lib/rank/index.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./util */ "../../../../node_modules/dagre/lib/rank/util.js" ), v = d.longestPath, u = o( /*! ./feasible-tree */ "../../../../node_modules/dagre/lib/rank/feasible-tree.js" ), l = o( /*! ./network-simplex */ "../../../../node_modules/dagre/lib/rank/network-simplex.js" ); a.exports = P; function P(T) { switch (T.graph().ranker) { case "network-simplex": H(T); break; case "tight-tree": c(T); break; case "longest-path": p(T); break; default: H(T); } } var p = v; function c(T) { v(T), u(T); } function H(T) { l(T); } } ), /***/ "../../../../node_modules/dagre/lib/rank/network-simplex.js": ( /*!******************************************************************!*\ !*** ../../../../node_modules/dagre/lib/rank/network-simplex.js ***! \******************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ./feasible-tree */ "../../../../node_modules/dagre/lib/rank/feasible-tree.js" ), u = o( /*! ./util */ "../../../../node_modules/dagre/lib/rank/util.js" ).slack, l = o( /*! ./util */ "../../../../node_modules/dagre/lib/rank/util.js" ).longestPath, P = o( /*! ../graphlib */ "../../../../node_modules/dagre/lib/graphlib.js" ).alg.preorder, p = o( /*! ../graphlib */ "../../../../node_modules/dagre/lib/graphlib.js" ).alg.postorder, c = o( /*! ../util */ "../../../../node_modules/dagre/lib/util.js" ).simplify; a.exports = H, H.initLowLimValues = j, H.initCutValues = T, H.calcCutValue = b, H.leaveEdge = m, H.enterEdge = I, H.exchangeEdges = N; function H(O) { O = c(O), l(O); var Y = v(O); j(Y), T(Y, O); for (var ee, Z; ee = m(Y); ) Z = I(Y, O, ee), N(Y, O, ee, Z); } function T(O, Y) { var ee = p(O, O.nodes()); ee = ee.slice(0, ee.length - 1), d.forEach(ee, function(Z) { q(O, Y, Z); }); } function q(O, Y, ee) { var Z = O.node(ee), te = Z.parent; O.edge(ee, te).cutvalue = b(O, Y, ee); } function b(O, Y, ee) { var Z = O.node(ee), te = Z.parent, fe = !0, _ = Y.edge(ee, te), G = 0; return _ || (fe = !1, _ = Y.edge(te, ee)), G = _.weight, d.forEach(Y.nodeEdges(ee), function(L) { var $ = L.v === ee, ae = $ ? L.w : L.v; if (ae !== te) { var Pe = $ === fe, ge = Y.edge(L).weight; if (G += Pe ? ge : -ge, R(O, ee, ae)) { var me = O.edge(ee, ae).cutvalue; G += Pe ? -me : me; } } }), G; } function j(O, Y) { arguments.length < 2 && (Y = O.nodes()[0]), w(O, {}, 1, Y); } function w(O, Y, ee, Z, te) { var fe = ee, _ = O.node(Z); return Y[Z] = !0, d.forEach(O.neighbors(Z), function(G) { d.has(Y, G) || (ee = w(O, Y, ee, G, Z)); }), _.low = fe, _.lim = ee++, te ? _.parent = te : delete _.parent, ee; } function m(O) { return d.find(O.edges(), function(Y) { return O.edge(Y).cutvalue < 0; }); } function I(O, Y, ee) { var Z = ee.v, te = ee.w; Y.hasEdge(Z, te) || (Z = ee.w, te = ee.v); var fe = O.node(Z), _ = O.node(te), G = fe, L = !1; fe.lim > _.lim && (G = _, L = !0); var $ = d.filter(Y.edges(), function(ae) { return L === y(O, O.node(ae.v), G) && L !== y(O, O.node(ae.w), G); }); return d.minBy($, function(ae) { return u(Y, ae); }); } function N(O, Y, ee, Z) { var te = ee.v, fe = ee.w; O.removeEdge(te, fe), O.setEdge(Z.v, Z.w, {}), j(O), T(O, Y), k(O, Y); } function k(O, Y) { var ee = d.find(O.nodes(), function(te) { return !Y.node(te).parent; }), Z = P(O, ee); Z = Z.slice(1), d.forEach(Z, function(te) { var fe = O.node(te).parent, _ = Y.edge(te, fe), G = !1; _ || (_ = Y.edge(fe, te), G = !0), Y.node(te).rank = Y.node(fe).rank + (G ? _.minlen : -_.minlen); }); } function R(O, Y, ee) { return O.hasEdge(Y, ee); } function y(O, Y, ee) { return ee.low <= Y.lim && Y.lim <= ee.lim; } } ), /***/ "../../../../node_modules/dagre/lib/rank/util.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/dagre/lib/rank/util.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/dagre/lib/lodash.js" ); a.exports = { longestPath: v, slack: u }; function v(l) { var P = {}; function p(c) { var H = l.node(c); if (d.has(P, c)) return H.rank; P[c] = !0; var T = d.min(d.map(l.outEdges(c), function(q) { return p(q.w) - l.edge(q).minlen; })); return (T === Number.POSITIVE_INFINITY || // return value of _.map([]) for Lodash 3 T === void 0 || // return value of _.map([]) for Lodash 4 T === null) && (T = 0), H.rank = T; } d.forEach(l.sources(), p); } function u(l, P) { return l.node(P.w).rank - l.node(P.v).rank - l.edge(P).minlen; } } ), /***/ "../../../../node_modules/dagre/lib/util.js": ( /*!**************************************************!*\ !*** ../../../../node_modules/dagre/lib/util.js ***! \**************************************************/ /***/ (a, f, o) => { var d = o( /*! ./lodash */ "../../../../node_modules/dagre/lib/lodash.js" ), v = o( /*! ./graphlib */ "../../../../node_modules/dagre/lib/graphlib.js" ).Graph; a.exports = { addDummyNode: u, simplify: l, asNonCompoundGraph: P, successorWeights: p, predecessorWeights: c, intersectRect: H, buildLayerMatrix: T, normalizeRanks: q, removeEmptyRanks: b, addBorderNode: j, maxRank: w, partition: m, time: I, notime: N }; function u(k, R, y, O) { var Y; do Y = d.uniqueId(O); while (k.hasNode(Y)); return y.dummy = R, k.setNode(Y, y), Y; } function l(k) { var R = new v().setGraph(k.graph()); return d.forEach(k.nodes(), function(y) { R.setNode(y, k.node(y)); }), d.forEach(k.edges(), function(y) { var O = R.edge(y.v, y.w) || { weight: 0, minlen: 1 }, Y = k.edge(y); R.setEdge(y.v, y.w, { weight: O.weight + Y.weight, minlen: Math.max(O.minlen, Y.minlen) }); }), R; } function P(k) { var R = new v({ multigraph: k.isMultigraph() }).setGraph(k.graph()); return d.forEach(k.nodes(), function(y) { k.children(y).length || R.setNode(y, k.node(y)); }), d.forEach(k.edges(), function(y) { R.setEdge(y, k.edge(y)); }), R; } function p(k) { var R = d.map(k.nodes(), function(y) { var O = {}; return d.forEach(k.outEdges(y), function(Y) { O[Y.w] = (O[Y.w] || 0) + k.edge(Y).weight; }), O; }); return d.zipObject(k.nodes(), R); } function c(k) { var R = d.map(k.nodes(), function(y) { var O = {}; return d.forEach(k.inEdges(y), function(Y) { O[Y.v] = (O[Y.v] || 0) + k.edge(Y).weight; }), O; }); return d.zipObject(k.nodes(), R); } function H(k, R) { var y = k.x, O = k.y, Y = R.x - y, ee = R.y - O, Z = k.width / 2, te = k.height / 2; if (!Y && !ee) throw new Error("Not possible to find intersection inside of the rectangle"); var fe, _; return Math.abs(ee) * Z > Math.abs(Y) * te ? (ee < 0 && (te = -te), fe = te * Y / ee, _ = te) : (Y < 0 && (Z = -Z), fe = Z, _ = Z * ee / Y), { x: y + fe, y: O + _ }; } function T(k) { var R = d.map(d.range(w(k) + 1), function() { return []; }); return d.forEach(k.nodes(), function(y) { var O = k.node(y), Y = O.rank; d.isUndefined(Y) || (R[Y][O.order] = y); }), R; } function q(k) { var R = d.min(d.map(k.nodes(), function(y) { return k.node(y).rank; })); d.forEach(k.nodes(), function(y) { var O = k.node(y); d.has(O, "rank") && (O.rank -= R); }); } function b(k) { var R = d.min(d.map(k.nodes(), function(ee) { return k.node(ee).rank; })), y = []; d.forEach(k.nodes(), function(ee) { var Z = k.node(ee).rank - R; y[Z] || (y[Z] = []), y[Z].push(ee); }); var O = 0, Y = k.graph().nodeRankFactor; d.forEach(y, function(ee, Z) { d.isUndefined(ee) && Z % Y !== 0 ? --O : O && d.forEach(ee, function(te) { k.node(te).rank += O; }); }); } function j(k, R, y, O) { var Y = { width: 0, height: 0 }; return arguments.length >= 4 && (Y.rank = y, Y.order = O), u(k, "border", Y, R); } function w(k) { return d.max(d.map(k.nodes(), function(R) { var y = k.node(R).rank; if (!d.isUndefined(y)) return y; })); } function m(k, R) { var y = { lhs: [], rhs: [] }; return d.forEach(k, function(O) { R(O) ? y.lhs.push(O) : y.rhs.push(O); }), y; } function I(k, R) { var y = d.now(); try { return R(); } finally { console.log(k + " time: " + (d.now() - y) + "ms"); } } function N(k, R) { return R(); } } ), /***/ "../../../../node_modules/dagre/lib/version.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/dagre/lib/version.js ***! \*****************************************************/ /***/ (a) => { a.exports = "0.8.5"; } ), /***/ "../../../../node_modules/graphlib/index.js": ( /*!**************************************************!*\ !*** ../../../../node_modules/graphlib/index.js ***! \**************************************************/ /***/ (a, f, o) => { var d = o( /*! ./lib */ "../../../../node_modules/graphlib/lib/index.js" ); a.exports = { Graph: d.Graph, json: o( /*! ./lib/json */ "../../../../node_modules/graphlib/lib/json.js" ), alg: o( /*! ./lib/alg */ "../../../../node_modules/graphlib/lib/alg/index.js" ), version: d.version }; } ), /***/ "../../../../node_modules/graphlib/lib/alg/components.js": ( /*!***************************************************************!*\ !*** ../../../../node_modules/graphlib/lib/alg/components.js ***! \***************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/graphlib/lib/lodash.js" ); a.exports = v; function v(u) { var l = {}, P = [], p; function c(H) { d.has(l, H) || (l[H] = !0, p.push(H), d.each(u.successors(H), c), d.each(u.predecessors(H), c)); } return d.each(u.nodes(), function(H) { p = [], c(H), p.length && P.push(p); }), P; } } ), /***/ "../../../../node_modules/graphlib/lib/alg/dfs.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/graphlib/lib/alg/dfs.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/graphlib/lib/lodash.js" ); a.exports = v; function v(l, P, p) { d.isArray(P) || (P = [P]); var c = (l.isDirected() ? l.successors : l.neighbors).bind(l), H = [], T = {}; return d.each(P, function(q) { if (!l.hasNode(q)) throw new Error("Graph does not have node: " + q); u(l, q, p === "post", T, c, H); }), H; } function u(l, P, p, c, H, T) { d.has(c, P) || (c[P] = !0, p || T.push(P), d.each(H(P), function(q) { u(l, q, p, c, H, T); }), p && T.push(P)); } } ), /***/ "../../../../node_modules/graphlib/lib/alg/dijkstra-all.js": ( /*!*****************************************************************!*\ !*** ../../../../node_modules/graphlib/lib/alg/dijkstra-all.js ***! \*****************************************************************/ /***/ (a, f, o) => { var d = o( /*! ./dijkstra */ "../../../../node_modules/graphlib/lib/alg/dijkstra.js" ), v = o( /*! ../lodash */ "../../../../node_modules/graphlib/lib/lodash.js" ); a.exports = u; function u(l, P, p) { return v.transform(l.nodes(), function(c, H) { c[H] = d(l, H, P, p); }, {}); } } ), /***/ "../../../../node_modules/graphlib/lib/alg/dijkstra.js": ( /*!*************************************************************!*\ !*** ../../../../node_modules/graphlib/lib/alg/dijkstra.js ***! \*************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/graphlib/lib/lodash.js" ), v = o( /*! ../data/priority-queue */ "../../../../node_modules/graphlib/lib/data/priority-queue.js" ); a.exports = l; var u = d.constant(1); function l(p, c, H, T) { return P( p, String(c), H || u, T || function(q) { return p.outEdges(q); } ); } function P(p, c, H, T) { var q = {}, b = new v(), j, w, m = function(I) { var N = I.v !== j ? I.v : I.w, k = q[N], R = H(I), y = w.distance + R; if (R < 0) throw new Error("dijkstra does not allow negative edge weights. Bad edge: " + I + " Weight: " + R); y < k.distance && (k.distance = y, k.predecessor = j, b.decrease(N, y)); }; for (p.nodes().forEach(function(I) { var N = I === c ? 0 : Number.POSITIVE_INFINITY; q[I] = { distance: N }, b.add(I, N); }); b.size() > 0 && (j = b.removeMin(), w = q[j], w.distance !== Number.POSITIVE_INFINITY); ) T(j).forEach(m); return q; } } ), /***/ "../../../../node_modules/graphlib/lib/alg/find-cycles.js": ( /*!****************************************************************!*\ !*** ../../../../node_modules/graphlib/lib/alg/find-cycles.js ***! \****************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/graphlib/lib/lodash.js" ), v = o( /*! ./tarjan */ "../../../../node_modules/graphlib/lib/alg/tarjan.js" ); a.exports = u; function u(l) { return d.filter(v(l), function(P) { return P.length > 1 || P.length === 1 && l.hasEdge(P[0], P[0]); }); } } ), /***/ "../../../../node_modules/graphlib/lib/alg/floyd-warshall.js": ( /*!*******************************************************************!*\ !*** ../../../../node_modules/graphlib/lib/alg/floyd-warshall.js ***! \*******************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/graphlib/lib/lodash.js" ); a.exports = u; var v = d.constant(1); function u(P, p, c) { return l( P, p || v, c || function(H) { return P.outEdges(H); } ); } function l(P, p, c) { var H = {}, T = P.nodes(); return T.forEach(function(q) { H[q] = {}, H[q][q] = { distance: 0 }, T.forEach(function(b) { q !== b && (H[q][b] = { distance: Number.POSITIVE_INFINITY }); }), c(q).forEach(function(b) { var j = b.v === q ? b.w : b.v, w = p(b); H[q][j] = { distance: w, predecessor: q }; }); }), T.forEach(function(q) { var b = H[q]; T.forEach(function(j) { var w = H[j]; T.forEach(function(m) { var I = w[q], N = b[m], k = w[m], R = I.distance + N.distance; R < k.distance && (k.distance = R, k.predecessor = N.predecessor); }); }); }), H; } } ), /***/ "../../../../node_modules/graphlib/lib/alg/index.js": ( /*!**********************************************************!*\ !*** ../../../../node_modules/graphlib/lib/alg/index.js ***! \**********************************************************/ /***/ (a, f, o) => { a.exports = { components: o( /*! ./components */ "../../../../node_modules/graphlib/lib/alg/components.js" ), dijkstra: o( /*! ./dijkstra */ "../../../../node_modules/graphlib/lib/alg/dijkstra.js" ), dijkstraAll: o( /*! ./dijkstra-all */ "../../../../node_modules/graphlib/lib/alg/dijkstra-all.js" ), findCycles: o( /*! ./find-cycles */ "../../../../node_modules/graphlib/lib/alg/find-cycles.js" ), floydWarshall: o( /*! ./floyd-warshall */ "../../../../node_modules/graphlib/lib/alg/floyd-warshall.js" ), isAcyclic: o( /*! ./is-acyclic */ "../../../../node_modules/graphlib/lib/alg/is-acyclic.js" ), postorder: o( /*! ./postorder */ "../../../../node_modules/graphlib/lib/alg/postorder.js" ), preorder: o( /*! ./preorder */ "../../../../node_modules/graphlib/lib/alg/preorder.js" ), prim: o( /*! ./prim */ "../../../../node_modules/graphlib/lib/alg/prim.js" ), tarjan: o( /*! ./tarjan */ "../../../../node_modules/graphlib/lib/alg/tarjan.js" ), topsort: o( /*! ./topsort */ "../../../../node_modules/graphlib/lib/alg/topsort.js" ) }; } ), /***/ "../../../../node_modules/graphlib/lib/alg/is-acyclic.js": ( /*!***************************************************************!*\ !*** ../../../../node_modules/graphlib/lib/alg/is-acyclic.js ***! \***************************************************************/ /***/ (a, f, o) => { var d = o( /*! ./topsort */ "../../../../node_modules/graphlib/lib/alg/topsort.js" ); a.exports = v; function v(u) { try { d(u); } catch (l) { if (l instanceof d.CycleException) return !1; throw l; } return !0; } } ), /***/ "../../../../node_modules/graphlib/lib/alg/postorder.js": ( /*!**************************************************************!*\ !*** ../../../../node_modules/graphlib/lib/alg/postorder.js ***! \**************************************************************/ /***/ (a, f, o) => { var d = o( /*! ./dfs */ "../../../../node_modules/graphlib/lib/alg/dfs.js" ); a.exports = v; function v(u, l) { return d(u, l, "post"); } } ), /***/ "../../../../node_modules/graphlib/lib/alg/preorder.js": ( /*!*************************************************************!*\ !*** ../../../../node_modules/graphlib/lib/alg/preorder.js ***! \*************************************************************/ /***/ (a, f, o) => { var d = o( /*! ./dfs */ "../../../../node_modules/graphlib/lib/alg/dfs.js" ); a.exports = v; function v(u, l) { return d(u, l, "pre"); } } ), /***/ "../../../../node_modules/graphlib/lib/alg/prim.js": ( /*!*********************************************************!*\ !*** ../../../../node_modules/graphlib/lib/alg/prim.js ***! \*********************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/graphlib/lib/lodash.js" ), v = o( /*! ../graph */ "../../../../node_modules/graphlib/lib/graph.js" ), u = o( /*! ../data/priority-queue */ "../../../../node_modules/graphlib/lib/data/priority-queue.js" ); a.exports = l; function l(P, p) { var c = new v(), H = {}, T = new u(), q; function b(w) { var m = w.v === q ? w.w : w.v, I = T.priority(m); if (I !== void 0) { var N = p(w); N < I && (H[m] = q, T.decrease(m, N)); } } if (P.nodeCount() === 0) return c; d.each(P.nodes(), function(w) { T.add(w, Number.POSITIVE_INFINITY), c.setNode(w); }), T.decrease(P.nodes()[0], 0); for (var j = !1; T.size() > 0; ) { if (q = T.removeMin(), d.has(H, q)) c.setEdge(q, H[q]); else { if (j) throw new Error("Input graph is not connected: " + P); j = !0; } P.nodeEdges(q).forEach(b); } return c; } } ), /***/ "../../../../node_modules/graphlib/lib/alg/tarjan.js": ( /*!***********************************************************!*\ !*** ../../../../node_modules/graphlib/lib/alg/tarjan.js ***! \***********************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/graphlib/lib/lodash.js" ); a.exports = v; function v(u) { var l = 0, P = [], p = {}, c = []; function H(T) { var q = p[T] = { onStack: !0, lowlink: l, index: l++ }; if (P.push(T), u.successors(T).forEach(function(w) { d.has(p, w) ? p[w].onStack && (q.lowlink = Math.min(q.lowlink, p[w].index)) : (H(w), q.lowlink = Math.min(q.lowlink, p[w].lowlink)); }), q.lowlink === q.index) { var b = [], j; do j = P.pop(), p[j].onStack = !1, b.push(j); while (T !== j); c.push(b); } } return u.nodes().forEach(function(T) { d.has(p, T) || H(T); }), c; } } ), /***/ "../../../../node_modules/graphlib/lib/alg/topsort.js": ( /*!************************************************************!*\ !*** ../../../../node_modules/graphlib/lib/alg/topsort.js ***! \************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/graphlib/lib/lodash.js" ); a.exports = v, v.CycleException = u; function v(l) { var P = {}, p = {}, c = []; function H(T) { if (d.has(p, T)) throw new u(); d.has(P, T) || (p[T] = !0, P[T] = !0, d.each(l.predecessors(T), H), delete p[T], c.push(T)); } if (d.each(l.sinks(), H), d.size(P) !== l.nodeCount()) throw new u(); return c; } function u() { } u.prototype = new Error(); } ), /***/ "../../../../node_modules/graphlib/lib/data/priority-queue.js": ( /*!********************************************************************!*\ !*** ../../../../node_modules/graphlib/lib/data/priority-queue.js ***! \********************************************************************/ /***/ (a, f, o) => { var d = o( /*! ../lodash */ "../../../../node_modules/graphlib/lib/lodash.js" ); a.exports = v; function v() { this._arr = [], this._keyIndices = {}; } v.prototype.size = function() { return this._arr.length; }, v.prototype.keys = function() { return this._arr.map(function(u) { return u.key; }); }, v.prototype.has = function(u) { return d.has(this._keyIndices, u); }, v.prototype.priority = function(u) { var l = this._keyIndices[u]; if (l !== void 0) return this._arr[l].priority; }, v.prototype.min = function() { if (this.size() === 0) throw new Error("Queue underflow"); return this._arr[0].key; }, v.prototype.add = function(u, l) { var P = this._keyIndices; if (u = String(u), !d.has(P, u)) { var p = this._arr, c = p.length; return P[u] = c, p.push({ key: u, priority: l }), this._decrease(c), !0; } return !1; }, v.prototype.removeMin = function() { this._swap(0, this._arr.length - 1); var u = this._arr.pop(); return delete this._keyIndices[u.key], this._heapify(0), u.key; }, v.prototype.decrease = function(u, l) { var P = this._keyIndices[u]; if (l > this._arr[P].priority) throw new Error("New priority is greater than current priority. Key: " + u + " Old: " + this._arr[P].priority + " New: " + l); this._arr[P].priority = l, this._decrease(P); }, v.prototype._heapify = function(u) { var l = this._arr, P = 2 * u, p = P + 1, c = u; P < l.length && (c = l[P].priority < l[c].priority ? P : c, p < l.length && (c = l[p].priority < l[c].priority ? p : c), c !== u && (this._swap(u, c), this._heapify(c))); }, v.prototype._decrease = function(u) { for (var l = this._arr, P = l[u].priority, p; u !== 0 && (p = u >> 1, !(l[p].priority < P)); ) this._swap(u, p), u = p; }, v.prototype._swap = function(u, l) { var P = this._arr, p = this._keyIndices, c = P[u], H = P[l]; P[u] = H, P[l] = c, p[H.key] = u, p[c.key] = l; }; } ), /***/ "../../../../node_modules/graphlib/lib/graph.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/graphlib/lib/graph.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./lodash */ "../../../../node_modules/graphlib/lib/lodash.js" ); a.exports = P; var v = "\0", u = "\0", l = ""; function P(b) { this._isDirected = d.has(b, "directed") ? b.directed : !0, this._isMultigraph = d.has(b, "multigraph") ? b.multigraph : !1, this._isCompound = d.has(b, "compound") ? b.compound : !1, this._label = void 0, this._defaultNodeLabelFn = d.constant(void 0), this._defaultEdgeLabelFn = d.constant(void 0), this._nodes = {}, this._isCompound && (this._parent = {}, this._children = {}, this._children[u] = {}), this._in = {}, this._preds = {}, this._out = {}, this._sucs = {}, this._edgeObjs = {}, this._edgeLabels = {}; } P.prototype._nodeCount = 0, P.prototype._edgeCount = 0, P.prototype.isDirected = function() { return this._isDirected; }, P.prototype.isMultigraph = function() { return this._isMultigraph; }, P.prototype.isCompound = function() { return this._isCompound; }, P.prototype.setGraph = function(b) { return this._label = b, this; }, P.prototype.graph = function() { return this._label; }, P.prototype.setDefaultNodeLabel = function(b) { return d.isFunction(b) || (b = d.constant(b)), this._defaultNodeLabelFn = b, this; }, P.prototype.nodeCount = function() { return this._nodeCount; }, P.prototype.nodes = function() { return d.keys(this._nodes); }, P.prototype.sources = function() { var b = this; return d.filter(this.nodes(), function(j) { return d.isEmpty(b._in[j]); }); }, P.prototype.sinks = function() { var b = this; return d.filter(this.nodes(), function(j) { return d.isEmpty(b._out[j]); }); }, P.prototype.setNodes = function(b, j) { var w = arguments, m = this; return d.each(b, function(I) { w.length > 1 ? m.setNode(I, j) : m.setNode(I); }), this; }, P.prototype.setNode = function(b, j) { return d.has(this._nodes, b) ? (arguments.length > 1 && (this._nodes[b] = j), this) : (this._nodes[b] = arguments.length > 1 ? j : this._defaultNodeLabelFn(b), this._isCompound && (this._parent[b] = u, this._children[b] = {}, this._children[u][b] = !0), this._in[b] = {}, this._preds[b] = {}, this._out[b] = {}, this._sucs[b] = {}, ++this._nodeCount, this); }, P.prototype.node = function(b) { return this._nodes[b]; }, P.prototype.hasNode = function(b) { return d.has(this._nodes, b); }, P.prototype.removeNode = function(b) { var j = this; if (d.has(this._nodes, b)) { var w = function(m) { j.removeEdge(j._edgeObjs[m]); }; delete this._nodes[b], this._isCompound && (this._removeFromParentsChildList(b), delete this._parent[b], d.each(this.children(b), function(m) { j.setParent(m); }), delete this._children[b]), d.each(d.keys(this._in[b]), w), delete this._in[b], delete this._preds[b], d.each(d.keys(this._out[b]), w), delete this._out[b], delete this._sucs[b], --this._nodeCount; } return this; }, P.prototype.setParent = function(b, j) { if (!this._isCompound) throw new Error("Cannot set parent in a non-compound graph"); if (d.isUndefined(j)) j = u; else { j += ""; for (var w = j; !d.isUndefined(w); w = this.parent(w)) if (w === b) throw new Error("Setting " + j + " as parent of " + b + " would create a cycle"); this.setNode(j); } return this.setNode(b), this._removeFromParentsChildList(b), this._parent[b] = j, this._children[j][b] = !0, this; }, P.prototype._removeFromParentsChildList = function(b) { delete this._children[this._parent[b]][b]; }, P.prototype.parent = function(b) { if (this._isCompound) { var j = this._parent[b]; if (j !== u) return j; } }, P.prototype.children = function(b) { if (d.isUndefined(b) && (b = u), this._isCompound) { var j = this._children[b]; if (j) return d.keys(j); } else { if (b === u) return this.nodes(); if (this.hasNode(b)) return []; } }, P.prototype.predecessors = function(b) { var j = this._preds[b]; if (j) return d.keys(j); }, P.prototype.successors = function(b) { var j = this._sucs[b]; if (j) return d.keys(j); }, P.prototype.neighbors = function(b) { var j = this.predecessors(b); if (j) return d.union(j, this.successors(b)); }, P.prototype.isLeaf = function(b) { var j; return this.isDirected() ? j = this.successors(b) : j = this.neighbors(b), j.length === 0; }, P.prototype.filterNodes = function(b) { var j = new this.constructor({ directed: this._isDirected, multigraph: this._isMultigraph, compound: this._isCompound }); j.setGraph(this.graph()); var w = this; d.each(this._nodes, function(N, k) { b(k) && j.setNode(k, N); }), d.each(this._edgeObjs, function(N) { j.hasNode(N.v) && j.hasNode(N.w) && j.setEdge(N, w.edge(N)); }); var m = {}; function I(N) { var k = w.parent(N); return k === void 0 || j.hasNode(k) ? (m[N] = k, k) : k in m ? m[k] : I(k); } return this._isCompound && d.each(j.nodes(), function(N) { j.setParent(N, I(N)); }), j; }, P.prototype.setDefaultEdgeLabel = function(b) { return d.isFunction(b) || (b = d.constant(b)), this._defaultEdgeLabelFn = b, this; }, P.prototype.edgeCount = function() { return this._edgeCount; }, P.prototype.edges = function() { return d.values(this._edgeObjs); }, P.prototype.setPath = function(b, j) { var w = this, m = arguments; return d.reduce(b, function(I, N) { return m.length > 1 ? w.setEdge(I, N, j) : w.setEdge(I, N), N; }), this; }, P.prototype.setEdge = function() { var b, j, w, m, I = !1, N = arguments[0]; typeof N == "object" && N !== null && "v" in N ? (b = N.v, j = N.w, w = N.name, arguments.length === 2 && (m = arguments[1], I = !0)) : (b = N, j = arguments[1], w = arguments[3], arguments.length > 2 && (m = arguments[2], I = !0)), b = "" + b, j = "" + j, d.isUndefined(w) || (w = "" + w); var k = H(this._isDirected, b, j, w); if (d.has(this._edgeLabels, k)) return I && (this._edgeLabels[k] = m), this; if (!d.isUndefined(w) && !this._isMultigraph) throw new Error("Cannot set a named edge when isMultigraph = false"); this.setNode(b), this.setNode(j), this._edgeLabels[k] = I ? m : this._defaultEdgeLabelFn(b, j, w); var R = T(this._isDirected, b, j, w); return b = R.v, j = R.w, Object.freeze(R), this._edgeObjs[k] = R, p(this._preds[j], b), p(this._sucs[b], j), this._in[j][k] = R, this._out[b][k] = R, this._edgeCount++, this; }, P.prototype.edge = function(b, j, w) { var m = arguments.length === 1 ? q(this._isDirected, arguments[0]) : H(this._isDirected, b, j, w); return this._edgeLabels[m]; }, P.prototype.hasEdge = function(b, j, w) { var m = arguments.length === 1 ? q(this._isDirected, arguments[0]) : H(this._isDirected, b, j, w); return d.has(this._edgeLabels, m); }, P.prototype.removeEdge = function(b, j, w) { var m = arguments.length === 1 ? q(this._isDirected, arguments[0]) : H(this._isDirected, b, j, w), I = this._edgeObjs[m]; return I && (b = I.v, j = I.w, delete this._edgeLabels[m], delete this._edgeObjs[m], c(this._preds[j], b), c(this._sucs[b], j), delete this._in[j][m], delete this._out[b][m], this._edgeCount--), this; }, P.prototype.inEdges = function(b, j) { var w = this._in[b]; if (w) { var m = d.values(w); return j ? d.filter(m, function(I) { return I.v === j; }) : m; } }, P.prototype.outEdges = function(b, j) { var w = this._out[b]; if (w) { var m = d.values(w); return j ? d.filter(m, function(I) { return I.w === j; }) : m; } }, P.prototype.nodeEdges = function(b, j) { var w = this.inEdges(b, j); if (w) return w.concat(this.outEdges(b, j)); }; function p(b, j) { b[j] ? b[j]++ : b[j] = 1; } function c(b, j) { --b[j] || delete b[j]; } function H(b, j, w, m) { var I = "" + j, N = "" + w; if (!b && I > N) { var k = I; I = N, N = k; } return I + l + N + l + (d.isUndefined(m) ? v : m); } function T(b, j, w, m) { var I = "" + j, N = "" + w; if (!b && I > N) { var k = I; I = N, N = k; } var R = { v: I, w: N }; return m && (R.name = m), R; } function q(b, j) { return H(b, j.v, j.w, j.name); } } ), /***/ "../../../../node_modules/graphlib/lib/index.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/graphlib/lib/index.js ***! \******************************************************/ /***/ (a, f, o) => { a.exports = { Graph: o( /*! ./graph */ "../../../../node_modules/graphlib/lib/graph.js" ), version: o( /*! ./version */ "../../../../node_modules/graphlib/lib/version.js" ) }; } ), /***/ "../../../../node_modules/graphlib/lib/json.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/graphlib/lib/json.js ***! \*****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./lodash */ "../../../../node_modules/graphlib/lib/lodash.js" ), v = o( /*! ./graph */ "../../../../node_modules/graphlib/lib/graph.js" ); a.exports = { write: u, read: p }; function u(c) { var H = { options: { directed: c.isDirected(), multigraph: c.isMultigraph(), compound: c.isCompound() }, nodes: l(c), edges: P(c) }; return d.isUndefined(c.graph()) || (H.value = d.clone(c.graph())), H; } function l(c) { return d.map(c.nodes(), function(H) { var T = c.node(H), q = c.parent(H), b = { v: H }; return d.isUndefined(T) || (b.value = T), d.isUndefined(q) || (b.parent = q), b; }); } function P(c) { return d.map(c.edges(), function(H) { var T = c.edge(H), q = { v: H.v, w: H.w }; return d.isUndefined(H.name) || (q.name = H.name), d.isUndefined(T) || (q.value = T), q; }); } function p(c) { var H = new v(c.options).setGraph(c.value); return d.each(c.nodes, function(T) { H.setNode(T.v, T.value), T.parent && H.setParent(T.v, T.parent); }), d.each(c.edges, function(T) { H.setEdge({ v: T.v, w: T.w, name: T.name }, T.value); }), H; } } ), /***/ "../../../../node_modules/graphlib/lib/lodash.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/graphlib/lib/lodash.js ***! \*******************************************************/ /***/ (a, f, o) => { var d; try { d = { clone: o( /*! lodash/clone */ "../../../../node_modules/lodash/clone.js" ), constant: o( /*! lodash/constant */ "../../../../node_modules/lodash/constant.js" ), each: o( /*! lodash/each */ "../../../../node_modules/lodash/each.js" ), filter: o( /*! lodash/filter */ "../../../../node_modules/lodash/filter.js" ), has: o( /*! lodash/has */ "../../../../node_modules/lodash/has.js" ), isArray: o( /*! lodash/isArray */ "../../../../node_modules/lodash/isArray.js" ), isEmpty: o( /*! lodash/isEmpty */ "../../../../node_modules/lodash/isEmpty.js" ), isFunction: o( /*! lodash/isFunction */ "../../../../node_modules/lodash/isFunction.js" ), isUndefined: o( /*! lodash/isUndefined */ "../../../../node_modules/lodash/isUndefined.js" ), keys: o( /*! lodash/keys */ "../../../../node_modules/lodash/keys.js" ), map: o( /*! lodash/map */ "../../../../node_modules/lodash/map.js" ), reduce: o( /*! lodash/reduce */ "../../../../node_modules/lodash/reduce.js" ), size: o( /*! lodash/size */ "../../../../node_modules/lodash/size.js" ), transform: o( /*! lodash/transform */ "../../../../node_modules/lodash/transform.js" ), union: o( /*! lodash/union */ "../../../../node_modules/lodash/union.js" ), values: o( /*! lodash/values */ "../../../../node_modules/lodash/values.js" ) }; } catch { } d || (d = window._), a.exports = d; } ), /***/ "../../../../node_modules/graphlib/lib/version.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/graphlib/lib/version.js ***! \********************************************************/ /***/ (a) => { a.exports = "2.1.8"; } ), /***/ "../../../../node_modules/lodash/_DataView.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_DataView.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_getNative */ "../../../../node_modules/lodash/_getNative.js" ), v = o( /*! ./_root */ "../../../../node_modules/lodash/_root.js" ), u = d(v, "DataView"); a.exports = u; } ), /***/ "../../../../node_modules/lodash/_Hash.js": ( /*!************************************************!*\ !*** ../../../../node_modules/lodash/_Hash.js ***! \************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_hashClear */ "../../../../node_modules/lodash/_hashClear.js" ), v = o( /*! ./_hashDelete */ "../../../../node_modules/lodash/_hashDelete.js" ), u = o( /*! ./_hashGet */ "../../../../node_modules/lodash/_hashGet.js" ), l = o( /*! ./_hashHas */ "../../../../node_modules/lodash/_hashHas.js" ), P = o( /*! ./_hashSet */ "../../../../node_modules/lodash/_hashSet.js" ); function p(c) { var H = -1, T = c == null ? 0 : c.length; for (this.clear(); ++H < T; ) { var q = c[H]; this.set(q[0], q[1]); } } p.prototype.clear = d, p.prototype.delete = v, p.prototype.get = u, p.prototype.has = l, p.prototype.set = P, a.exports = p; } ), /***/ "../../../../node_modules/lodash/_ListCache.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_ListCache.js ***! \*****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_listCacheClear */ "../../../../node_modules/lodash/_listCacheClear.js" ), v = o( /*! ./_listCacheDelete */ "../../../../node_modules/lodash/_listCacheDelete.js" ), u = o( /*! ./_listCacheGet */ "../../../../node_modules/lodash/_listCacheGet.js" ), l = o( /*! ./_listCacheHas */ "../../../../node_modules/lodash/_listCacheHas.js" ), P = o( /*! ./_listCacheSet */ "../../../../node_modules/lodash/_listCacheSet.js" ); function p(c) { var H = -1, T = c == null ? 0 : c.length; for (this.clear(); ++H < T; ) { var q = c[H]; this.set(q[0], q[1]); } } p.prototype.clear = d, p.prototype.delete = v, p.prototype.get = u, p.prototype.has = l, p.prototype.set = P, a.exports = p; } ), /***/ "../../../../node_modules/lodash/_Map.js": ( /*!***********************************************!*\ !*** ../../../../node_modules/lodash/_Map.js ***! \***********************************************/ /***/ (a, f, o) => { var d = o( /*! ./_getNative */ "../../../../node_modules/lodash/_getNative.js" ), v = o( /*! ./_root */ "../../../../node_modules/lodash/_root.js" ), u = d(v, "Map"); a.exports = u; } ), /***/ "../../../../node_modules/lodash/_MapCache.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_MapCache.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_mapCacheClear */ "../../../../node_modules/lodash/_mapCacheClear.js" ), v = o( /*! ./_mapCacheDelete */ "../../../../node_modules/lodash/_mapCacheDelete.js" ), u = o( /*! ./_mapCacheGet */ "../../../../node_modules/lodash/_mapCacheGet.js" ), l = o( /*! ./_mapCacheHas */ "../../../../node_modules/lodash/_mapCacheHas.js" ), P = o( /*! ./_mapCacheSet */ "../../../../node_modules/lodash/_mapCacheSet.js" ); function p(c) { var H = -1, T = c == null ? 0 : c.length; for (this.clear(); ++H < T; ) { var q = c[H]; this.set(q[0], q[1]); } } p.prototype.clear = d, p.prototype.delete = v, p.prototype.get = u, p.prototype.has = l, p.prototype.set = P, a.exports = p; } ), /***/ "../../../../node_modules/lodash/_Promise.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/_Promise.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_getNative */ "../../../../node_modules/lodash/_getNative.js" ), v = o( /*! ./_root */ "../../../../node_modules/lodash/_root.js" ), u = d(v, "Promise"); a.exports = u; } ), /***/ "../../../../node_modules/lodash/_Set.js": ( /*!***********************************************!*\ !*** ../../../../node_modules/lodash/_Set.js ***! \***********************************************/ /***/ (a, f, o) => { var d = o( /*! ./_getNative */ "../../../../node_modules/lodash/_getNative.js" ), v = o( /*! ./_root */ "../../../../node_modules/lodash/_root.js" ), u = d(v, "Set"); a.exports = u; } ), /***/ "../../../../node_modules/lodash/_SetCache.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_SetCache.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_MapCache */ "../../../../node_modules/lodash/_MapCache.js" ), v = o( /*! ./_setCacheAdd */ "../../../../node_modules/lodash/_setCacheAdd.js" ), u = o( /*! ./_setCacheHas */ "../../../../node_modules/lodash/_setCacheHas.js" ); function l(P) { var p = -1, c = P == null ? 0 : P.length; for (this.__data__ = new d(); ++p < c; ) this.add(P[p]); } l.prototype.add = l.prototype.push = v, l.prototype.has = u, a.exports = l; } ), /***/ "../../../../node_modules/lodash/_Stack.js": ( /*!*************************************************!*\ !*** ../../../../node_modules/lodash/_Stack.js ***! \*************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_ListCache */ "../../../../node_modules/lodash/_ListCache.js" ), v = o( /*! ./_stackClear */ "../../../../node_modules/lodash/_stackClear.js" ), u = o( /*! ./_stackDelete */ "../../../../node_modules/lodash/_stackDelete.js" ), l = o( /*! ./_stackGet */ "../../../../node_modules/lodash/_stackGet.js" ), P = o( /*! ./_stackHas */ "../../../../node_modules/lodash/_stackHas.js" ), p = o( /*! ./_stackSet */ "../../../../node_modules/lodash/_stackSet.js" ); function c(H) { var T = this.__data__ = new d(H); this.size = T.size; } c.prototype.clear = v, c.prototype.delete = u, c.prototype.get = l, c.prototype.has = P, c.prototype.set = p, a.exports = c; } ), /***/ "../../../../node_modules/lodash/_Symbol.js": ( /*!**************************************************!*\ !*** ../../../../node_modules/lodash/_Symbol.js ***! \**************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_root */ "../../../../node_modules/lodash/_root.js" ), v = d.Symbol; a.exports = v; } ), /***/ "../../../../node_modules/lodash/_Uint8Array.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_Uint8Array.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_root */ "../../../../node_modules/lodash/_root.js" ), v = d.Uint8Array; a.exports = v; } ), /***/ "../../../../node_modules/lodash/_WeakMap.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/_WeakMap.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_getNative */ "../../../../node_modules/lodash/_getNative.js" ), v = o( /*! ./_root */ "../../../../node_modules/lodash/_root.js" ), u = d(v, "WeakMap"); a.exports = u; } ), /***/ "../../../../node_modules/lodash/_apply.js": ( /*!*************************************************!*\ !*** ../../../../node_modules/lodash/_apply.js ***! \*************************************************/ /***/ (a) => { function f(o, d, v) { switch (v.length) { case 0: return o.call(d); case 1: return o.call(d, v[0]); case 2: return o.call(d, v[0], v[1]); case 3: return o.call(d, v[0], v[1], v[2]); } return o.apply(d, v); } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_arrayEach.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_arrayEach.js ***! \*****************************************************/ /***/ (a) => { function f(o, d) { for (var v = -1, u = o == null ? 0 : o.length; ++v < u && d(o[v], v, o) !== !1; ) ; return o; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_arrayFilter.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_arrayFilter.js ***! \*******************************************************/ /***/ (a) => { function f(o, d) { for (var v = -1, u = o == null ? 0 : o.length, l = 0, P = []; ++v < u; ) { var p = o[v]; d(p, v, o) && (P[l++] = p); } return P; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_arrayIncludes.js": ( /*!*********************************************************!*\ !*** ../../../../node_modules/lodash/_arrayIncludes.js ***! \*********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseIndexOf */ "../../../../node_modules/lodash/_baseIndexOf.js" ); function v(u, l) { var P = u == null ? 0 : u.length; return !!P && d(u, l, 0) > -1; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_arrayIncludesWith.js": ( /*!*************************************************************!*\ !*** ../../../../node_modules/lodash/_arrayIncludesWith.js ***! \*************************************************************/ /***/ (a) => { function f(o, d, v) { for (var u = -1, l = o == null ? 0 : o.length; ++u < l; ) if (v(d, o[u])) return !0; return !1; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_arrayLikeKeys.js": ( /*!*********************************************************!*\ !*** ../../../../node_modules/lodash/_arrayLikeKeys.js ***! \*********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseTimes */ "../../../../node_modules/lodash/_baseTimes.js" ), v = o( /*! ./isArguments */ "../../../../node_modules/lodash/isArguments.js" ), u = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ), l = o( /*! ./isBuffer */ "../../../../node_modules/lodash/isBuffer.js" ), P = o( /*! ./_isIndex */ "../../../../node_modules/lodash/_isIndex.js" ), p = o( /*! ./isTypedArray */ "../../../../node_modules/lodash/isTypedArray.js" ), c = Object.prototype, H = c.hasOwnProperty; function T(q, b) { var j = u(q), w = !j && v(q), m = !j && !w && l(q), I = !j && !w && !m && p(q), N = j || w || m || I, k = N ? d(q.length, String) : [], R = k.length; for (var y in q) (b || H.call(q, y)) && !(N && // Safari 9 has enumerable `arguments.length` in strict mode. (y == "length" || // Node.js 0.10 has enumerable non-index properties on buffers. m && (y == "offset" || y == "parent") || // PhantomJS 2 has enumerable non-index properties on typed arrays. I && (y == "buffer" || y == "byteLength" || y == "byteOffset") || // Skip index properties. P(y, R))) && k.push(y); return k; } a.exports = T; } ), /***/ "../../../../node_modules/lodash/_arrayMap.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_arrayMap.js ***! \****************************************************/ /***/ (a) => { function f(o, d) { for (var v = -1, u = o == null ? 0 : o.length, l = Array(u); ++v < u; ) l[v] = d(o[v], v, o); return l; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_arrayPush.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_arrayPush.js ***! \*****************************************************/ /***/ (a) => { function f(o, d) { for (var v = -1, u = d.length, l = o.length; ++v < u; ) o[l + v] = d[v]; return o; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_arrayReduce.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_arrayReduce.js ***! \*******************************************************/ /***/ (a) => { function f(o, d, v, u) { var l = -1, P = o == null ? 0 : o.length; for (u && P && (v = o[++l]); ++l < P; ) v = d(v, o[l], l, o); return v; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_arraySome.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_arraySome.js ***! \*****************************************************/ /***/ (a) => { function f(o, d) { for (var v = -1, u = o == null ? 0 : o.length; ++v < u; ) if (d(o[v], v, o)) return !0; return !1; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_asciiSize.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_asciiSize.js ***! \*****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseProperty */ "../../../../node_modules/lodash/_baseProperty.js" ), v = d("length"); a.exports = v; } ), /***/ "../../../../node_modules/lodash/_assignMergeValue.js": ( /*!************************************************************!*\ !*** ../../../../node_modules/lodash/_assignMergeValue.js ***! \************************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseAssignValue */ "../../../../node_modules/lodash/_baseAssignValue.js" ), v = o( /*! ./eq */ "../../../../node_modules/lodash/eq.js" ); function u(l, P, p) { (p !== void 0 && !v(l[P], p) || p === void 0 && !(P in l)) && d(l, P, p); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_assignValue.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_assignValue.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseAssignValue */ "../../../../node_modules/lodash/_baseAssignValue.js" ), v = o( /*! ./eq */ "../../../../node_modules/lodash/eq.js" ), u = Object.prototype, l = u.hasOwnProperty; function P(p, c, H) { var T = p[c]; (!(l.call(p, c) && v(T, H)) || H === void 0 && !(c in p)) && d(p, c, H); } a.exports = P; } ), /***/ "../../../../node_modules/lodash/_assocIndexOf.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_assocIndexOf.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./eq */ "../../../../node_modules/lodash/eq.js" ); function v(u, l) { for (var P = u.length; P--; ) if (d(u[P][0], l)) return P; return -1; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_baseAssign.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_baseAssign.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_copyObject */ "../../../../node_modules/lodash/_copyObject.js" ), v = o( /*! ./keys */ "../../../../node_modules/lodash/keys.js" ); function u(l, P) { return l && d(P, v(P), l); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_baseAssignIn.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_baseAssignIn.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_copyObject */ "../../../../node_modules/lodash/_copyObject.js" ), v = o( /*! ./keysIn */ "../../../../node_modules/lodash/keysIn.js" ); function u(l, P) { return l && d(P, v(P), l); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_baseAssignValue.js": ( /*!***********************************************************!*\ !*** ../../../../node_modules/lodash/_baseAssignValue.js ***! \***********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_defineProperty */ "../../../../node_modules/lodash/_defineProperty.js" ); function v(u, l, P) { l == "__proto__" && d ? d(u, l, { configurable: !0, enumerable: !0, value: P, writable: !0 }) : u[l] = P; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_baseClone.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_baseClone.js ***! \*****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_Stack */ "../../../../node_modules/lodash/_Stack.js" ), v = o( /*! ./_arrayEach */ "../../../../node_modules/lodash/_arrayEach.js" ), u = o( /*! ./_assignValue */ "../../../../node_modules/lodash/_assignValue.js" ), l = o( /*! ./_baseAssign */ "../../../../node_modules/lodash/_baseAssign.js" ), P = o( /*! ./_baseAssignIn */ "../../../../node_modules/lodash/_baseAssignIn.js" ), p = o( /*! ./_cloneBuffer */ "../../../../node_modules/lodash/_cloneBuffer.js" ), c = o( /*! ./_copyArray */ "../../../../node_modules/lodash/_copyArray.js" ), H = o( /*! ./_copySymbols */ "../../../../node_modules/lodash/_copySymbols.js" ), T = o( /*! ./_copySymbolsIn */ "../../../../node_modules/lodash/_copySymbolsIn.js" ), q = o( /*! ./_getAllKeys */ "../../../../node_modules/lodash/_getAllKeys.js" ), b = o( /*! ./_getAllKeysIn */ "../../../../node_modules/lodash/_getAllKeysIn.js" ), j = o( /*! ./_getTag */ "../../../../node_modules/lodash/_getTag.js" ), w = o( /*! ./_initCloneArray */ "../../../../node_modules/lodash/_initCloneArray.js" ), m = o( /*! ./_initCloneByTag */ "../../../../node_modules/lodash/_initCloneByTag.js" ), I = o( /*! ./_initCloneObject */ "../../../../node_modules/lodash/_initCloneObject.js" ), N = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ), k = o( /*! ./isBuffer */ "../../../../node_modules/lodash/isBuffer.js" ), R = o( /*! ./isMap */ "../../../../node_modules/lodash/isMap.js" ), y = o( /*! ./isObject */ "../../../../node_modules/lodash/isObject.js" ), O = o( /*! ./isSet */ "../../../../node_modules/lodash/isSet.js" ), Y = o( /*! ./keys */ "../../../../node_modules/lodash/keys.js" ), ee = o( /*! ./keysIn */ "../../../../node_modules/lodash/keysIn.js" ), Z = 1, te = 2, fe = 4, _ = "[object Arguments]", G = "[object Array]", L = "[object Boolean]", $ = "[object Date]", ae = "[object Error]", Pe = "[object Function]", ge = "[object GeneratorFunction]", me = "[object Map]", Xe = "[object Number]", De = "[object Object]", ne = "[object RegExp]", re = "[object Set]", ve = "[object String]", qe = "[object Symbol]", ke = "[object WeakMap]", be = "[object ArrayBuffer]", Fe = "[object DataView]", Ke = "[object Float32Array]", nt = "[object Float64Array]", ut = "[object Int8Array]", bt = "[object Int16Array]", wt = "[object Int32Array]", Tt = "[object Uint8Array]", lr = "[object Uint8ClampedArray]", Qt = "[object Uint16Array]", tr = "[object Uint32Array]", br = {}; br[_] = br[G] = br[be] = br[Fe] = br[L] = br[$] = br[Ke] = br[nt] = br[ut] = br[bt] = br[wt] = br[me] = br[Xe] = br[De] = br[ne] = br[re] = br[ve] = br[qe] = br[Tt] = br[lr] = br[Qt] = br[tr] = !0, br[ae] = br[Pe] = br[ke] = !1; function Xn(qr, En, Bn, Fi, Gt, xr) { var Br, nn = En & Z, sn = En & te, Pn = En & fe; if (Bn && (Br = Gt ? Bn(qr, Fi, Gt, xr) : Bn(qr)), Br !== void 0) return Br; if (!y(qr)) return qr; var an = N(qr); if (an) { if (Br = w(qr), !nn) return c(qr, Br); } else { var Wn = j(qr), rr = Wn == Pe || Wn == ge; if (k(qr)) return p(qr, nn); if (Wn == De || Wn == _ || rr && !Gt) { if (Br = sn || rr ? {} : I(qr), !nn) return sn ? T(qr, P(Br, qr)) : H(qr, l(Br, qr)); } else { if (!br[Wn]) return Gt ? qr : {}; Br = m(qr, Wn, nn); } } xr || (xr = new d()); var Sr = xr.get(qr); if (Sr) return Sr; xr.set(qr, Br), O(qr) ? qr.forEach(function(Rn) { Br.add(Xn(Rn, En, Bn, Rn, qr, xr)); }) : R(qr) && qr.forEach(function(Rn, Vn) { Br.set(Vn, Xn(Rn, En, Bn, Vn, qr, xr)); }); var nr = Pn ? sn ? b : q : sn ? ee : Y, Er = an ? void 0 : nr(qr); return v(Er || qr, function(Rn, Vn) { Er && (Vn = Rn, Rn = qr[Vn]), u(Br, Vn, Xn(Rn, En, Bn, Vn, qr, xr)); }), Br; } a.exports = Xn; } ), /***/ "../../../../node_modules/lodash/_baseCreate.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_baseCreate.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./isObject */ "../../../../node_modules/lodash/isObject.js" ), v = Object.create, u = /* @__PURE__ */ function() { function l() { } return function(P) { if (!d(P)) return {}; if (v) return v(P); l.prototype = P; var p = new l(); return l.prototype = void 0, p; }; }(); a.exports = u; } ), /***/ "../../../../node_modules/lodash/_baseEach.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_baseEach.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseForOwn */ "../../../../node_modules/lodash/_baseForOwn.js" ), v = o( /*! ./_createBaseEach */ "../../../../node_modules/lodash/_createBaseEach.js" ), u = v(d); a.exports = u; } ), /***/ "../../../../node_modules/lodash/_baseExtremum.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_baseExtremum.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./isSymbol */ "../../../../node_modules/lodash/isSymbol.js" ); function v(u, l, P) { for (var p = -1, c = u.length; ++p < c; ) { var H = u[p], T = l(H); if (T != null && (q === void 0 ? T === T && !d(T) : P(T, q))) var q = T, b = H; } return b; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_baseFilter.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_baseFilter.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseEach */ "../../../../node_modules/lodash/_baseEach.js" ); function v(u, l) { var P = []; return d(u, function(p, c, H) { l(p, c, H) && P.push(p); }), P; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_baseFindIndex.js": ( /*!*********************************************************!*\ !*** ../../../../node_modules/lodash/_baseFindIndex.js ***! \*********************************************************/ /***/ (a) => { function f(o, d, v, u) { for (var l = o.length, P = v + (u ? 1 : -1); u ? P-- : ++P < l; ) if (d(o[P], P, o)) return P; return -1; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_baseFlatten.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_baseFlatten.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_arrayPush */ "../../../../node_modules/lodash/_arrayPush.js" ), v = o( /*! ./_isFlattenable */ "../../../../node_modules/lodash/_isFlattenable.js" ); function u(l, P, p, c, H) { var T = -1, q = l.length; for (p || (p = v), H || (H = []); ++T < q; ) { var b = l[T]; P > 0 && p(b) ? P > 1 ? u(b, P - 1, p, c, H) : d(H, b) : c || (H[H.length] = b); } return H; } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_baseFor.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/_baseFor.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_createBaseFor */ "../../../../node_modules/lodash/_createBaseFor.js" ), v = d(); a.exports = v; } ), /***/ "../../../../node_modules/lodash/_baseForOwn.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_baseForOwn.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseFor */ "../../../../node_modules/lodash/_baseFor.js" ), v = o( /*! ./keys */ "../../../../node_modules/lodash/keys.js" ); function u(l, P) { return l && d(l, P, v); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_baseGet.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/_baseGet.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_castPath */ "../../../../node_modules/lodash/_castPath.js" ), v = o( /*! ./_toKey */ "../../../../node_modules/lodash/_toKey.js" ); function u(l, P) { P = d(P, l); for (var p = 0, c = P.length; l != null && p < c; ) l = l[v(P[p++])]; return p && p == c ? l : void 0; } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_baseGetAllKeys.js": ( /*!**********************************************************!*\ !*** ../../../../node_modules/lodash/_baseGetAllKeys.js ***! \**********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_arrayPush */ "../../../../node_modules/lodash/_arrayPush.js" ), v = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ); function u(l, P, p) { var c = P(l); return v(l) ? c : d(c, p(l)); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_baseGetTag.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_baseGetTag.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_Symbol */ "../../../../node_modules/lodash/_Symbol.js" ), v = o( /*! ./_getRawTag */ "../../../../node_modules/lodash/_getRawTag.js" ), u = o( /*! ./_objectToString */ "../../../../node_modules/lodash/_objectToString.js" ), l = "[object Null]", P = "[object Undefined]", p = d ? d.toStringTag : void 0; function c(H) { return H == null ? H === void 0 ? P : l : p && p in Object(H) ? v(H) : u(H); } a.exports = c; } ), /***/ "../../../../node_modules/lodash/_baseGt.js": ( /*!**************************************************!*\ !*** ../../../../node_modules/lodash/_baseGt.js ***! \**************************************************/ /***/ (a) => { function f(o, d) { return o > d; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_baseHas.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/_baseHas.js ***! \***************************************************/ /***/ (a) => { var f = Object.prototype, o = f.hasOwnProperty; function d(v, u) { return v != null && o.call(v, u); } a.exports = d; } ), /***/ "../../../../node_modules/lodash/_baseHasIn.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_baseHasIn.js ***! \*****************************************************/ /***/ (a) => { function f(o, d) { return o != null && d in Object(o); } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_baseIndexOf.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_baseIndexOf.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseFindIndex */ "../../../../node_modules/lodash/_baseFindIndex.js" ), v = o( /*! ./_baseIsNaN */ "../../../../node_modules/lodash/_baseIsNaN.js" ), u = o( /*! ./_strictIndexOf */ "../../../../node_modules/lodash/_strictIndexOf.js" ); function l(P, p, c) { return p === p ? u(P, p, c) : d(P, v, c); } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_baseIsArguments.js": ( /*!***********************************************************!*\ !*** ../../../../node_modules/lodash/_baseIsArguments.js ***! \***********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseGetTag */ "../../../../node_modules/lodash/_baseGetTag.js" ), v = o( /*! ./isObjectLike */ "../../../../node_modules/lodash/isObjectLike.js" ), u = "[object Arguments]"; function l(P) { return v(P) && d(P) == u; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_baseIsEqual.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_baseIsEqual.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseIsEqualDeep */ "../../../../node_modules/lodash/_baseIsEqualDeep.js" ), v = o( /*! ./isObjectLike */ "../../../../node_modules/lodash/isObjectLike.js" ); function u(l, P, p, c, H) { return l === P ? !0 : l == null || P == null || !v(l) && !v(P) ? l !== l && P !== P : d(l, P, p, c, u, H); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_baseIsEqualDeep.js": ( /*!***********************************************************!*\ !*** ../../../../node_modules/lodash/_baseIsEqualDeep.js ***! \***********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_Stack */ "../../../../node_modules/lodash/_Stack.js" ), v = o( /*! ./_equalArrays */ "../../../../node_modules/lodash/_equalArrays.js" ), u = o( /*! ./_equalByTag */ "../../../../node_modules/lodash/_equalByTag.js" ), l = o( /*! ./_equalObjects */ "../../../../node_modules/lodash/_equalObjects.js" ), P = o( /*! ./_getTag */ "../../../../node_modules/lodash/_getTag.js" ), p = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ), c = o( /*! ./isBuffer */ "../../../../node_modules/lodash/isBuffer.js" ), H = o( /*! ./isTypedArray */ "../../../../node_modules/lodash/isTypedArray.js" ), T = 1, q = "[object Arguments]", b = "[object Array]", j = "[object Object]", w = Object.prototype, m = w.hasOwnProperty; function I(N, k, R, y, O, Y) { var ee = p(N), Z = p(k), te = ee ? b : P(N), fe = Z ? b : P(k); te = te == q ? j : te, fe = fe == q ? j : fe; var _ = te == j, G = fe == j, L = te == fe; if (L && c(N)) { if (!c(k)) return !1; ee = !0, _ = !1; } if (L && !_) return Y || (Y = new d()), ee || H(N) ? v(N, k, R, y, O, Y) : u(N, k, te, R, y, O, Y); if (!(R & T)) { var $ = _ && m.call(N, "__wrapped__"), ae = G && m.call(k, "__wrapped__"); if ($ || ae) { var Pe = $ ? N.value() : N, ge = ae ? k.value() : k; return Y || (Y = new d()), O(Pe, ge, R, y, Y); } } return L ? (Y || (Y = new d()), l(N, k, R, y, O, Y)) : !1; } a.exports = I; } ), /***/ "../../../../node_modules/lodash/_baseIsMap.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_baseIsMap.js ***! \*****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_getTag */ "../../../../node_modules/lodash/_getTag.js" ), v = o( /*! ./isObjectLike */ "../../../../node_modules/lodash/isObjectLike.js" ), u = "[object Map]"; function l(P) { return v(P) && d(P) == u; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_baseIsMatch.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_baseIsMatch.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_Stack */ "../../../../node_modules/lodash/_Stack.js" ), v = o( /*! ./_baseIsEqual */ "../../../../node_modules/lodash/_baseIsEqual.js" ), u = 1, l = 2; function P(p, c, H, T) { var q = H.length, b = q, j = !T; if (p == null) return !b; for (p = Object(p); q--; ) { var w = H[q]; if (j && w[2] ? w[1] !== p[w[0]] : !(w[0] in p)) return !1; } for (; ++q < b; ) { w = H[q]; var m = w[0], I = p[m], N = w[1]; if (j && w[2]) { if (I === void 0 && !(m in p)) return !1; } else { var k = new d(); if (T) var R = T(I, N, m, p, c, k); if (!(R === void 0 ? v(N, I, u | l, T, k) : R)) return !1; } } return !0; } a.exports = P; } ), /***/ "../../../../node_modules/lodash/_baseIsNaN.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_baseIsNaN.js ***! \*****************************************************/ /***/ (a) => { function f(o) { return o !== o; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_baseIsNative.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_baseIsNative.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./isFunction */ "../../../../node_modules/lodash/isFunction.js" ), v = o( /*! ./_isMasked */ "../../../../node_modules/lodash/_isMasked.js" ), u = o( /*! ./isObject */ "../../../../node_modules/lodash/isObject.js" ), l = o( /*! ./_toSource */ "../../../../node_modules/lodash/_toSource.js" ), P = /[\\^$.*+?()[\]{}|]/g, p = /^\[object .+?Constructor\]$/, c = Function.prototype, H = Object.prototype, T = c.toString, q = H.hasOwnProperty, b = RegExp( "^" + T.call(q).replace(P, "\\$&").replace(/hasOwnProperty|(function).*?(?=\\\()| for .+?(?=\\\])/g, "$1.*?") + "$" ); function j(w) { if (!u(w) || v(w)) return !1; var m = d(w) ? b : p; return m.test(l(w)); } a.exports = j; } ), /***/ "../../../../node_modules/lodash/_baseIsSet.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_baseIsSet.js ***! \*****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_getTag */ "../../../../node_modules/lodash/_getTag.js" ), v = o( /*! ./isObjectLike */ "../../../../node_modules/lodash/isObjectLike.js" ), u = "[object Set]"; function l(P) { return v(P) && d(P) == u; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_baseIsTypedArray.js": ( /*!************************************************************!*\ !*** ../../../../node_modules/lodash/_baseIsTypedArray.js ***! \************************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseGetTag */ "../../../../node_modules/lodash/_baseGetTag.js" ), v = o( /*! ./isLength */ "../../../../node_modules/lodash/isLength.js" ), u = o( /*! ./isObjectLike */ "../../../../node_modules/lodash/isObjectLike.js" ), l = "[object Arguments]", P = "[object Array]", p = "[object Boolean]", c = "[object Date]", H = "[object Error]", T = "[object Function]", q = "[object Map]", b = "[object Number]", j = "[object Object]", w = "[object RegExp]", m = "[object Set]", I = "[object String]", N = "[object WeakMap]", k = "[object ArrayBuffer]", R = "[object DataView]", y = "[object Float32Array]", O = "[object Float64Array]", Y = "[object Int8Array]", ee = "[object Int16Array]", Z = "[object Int32Array]", te = "[object Uint8Array]", fe = "[object Uint8ClampedArray]", _ = "[object Uint16Array]", G = "[object Uint32Array]", L = {}; L[y] = L[O] = L[Y] = L[ee] = L[Z] = L[te] = L[fe] = L[_] = L[G] = !0, L[l] = L[P] = L[k] = L[p] = L[R] = L[c] = L[H] = L[T] = L[q] = L[b] = L[j] = L[w] = L[m] = L[I] = L[N] = !1; function $(ae) { return u(ae) && v(ae.length) && !!L[d(ae)]; } a.exports = $; } ), /***/ "../../../../node_modules/lodash/_baseIteratee.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_baseIteratee.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseMatches */ "../../../../node_modules/lodash/_baseMatches.js" ), v = o( /*! ./_baseMatchesProperty */ "../../../../node_modules/lodash/_baseMatchesProperty.js" ), u = o( /*! ./identity */ "../../../../node_modules/lodash/identity.js" ), l = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ), P = o( /*! ./property */ "../../../../node_modules/lodash/property.js" ); function p(c) { return typeof c == "function" ? c : c == null ? u : typeof c == "object" ? l(c) ? v(c[0], c[1]) : d(c) : P(c); } a.exports = p; } ), /***/ "../../../../node_modules/lodash/_baseKeys.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_baseKeys.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_isPrototype */ "../../../../node_modules/lodash/_isPrototype.js" ), v = o( /*! ./_nativeKeys */ "../../../../node_modules/lodash/_nativeKeys.js" ), u = Object.prototype, l = u.hasOwnProperty; function P(p) { if (!d(p)) return v(p); var c = []; for (var H in Object(p)) l.call(p, H) && H != "constructor" && c.push(H); return c; } a.exports = P; } ), /***/ "../../../../node_modules/lodash/_baseKeysIn.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_baseKeysIn.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./isObject */ "../../../../node_modules/lodash/isObject.js" ), v = o( /*! ./_isPrototype */ "../../../../node_modules/lodash/_isPrototype.js" ), u = o( /*! ./_nativeKeysIn */ "../../../../node_modules/lodash/_nativeKeysIn.js" ), l = Object.prototype, P = l.hasOwnProperty; function p(c) { if (!d(c)) return u(c); var H = v(c), T = []; for (var q in c) q == "constructor" && (H || !P.call(c, q)) || T.push(q); return T; } a.exports = p; } ), /***/ "../../../../node_modules/lodash/_baseLt.js": ( /*!**************************************************!*\ !*** ../../../../node_modules/lodash/_baseLt.js ***! \**************************************************/ /***/ (a) => { function f(o, d) { return o < d; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_baseMap.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/_baseMap.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseEach */ "../../../../node_modules/lodash/_baseEach.js" ), v = o( /*! ./isArrayLike */ "../../../../node_modules/lodash/isArrayLike.js" ); function u(l, P) { var p = -1, c = v(l) ? Array(l.length) : []; return d(l, function(H, T, q) { c[++p] = P(H, T, q); }), c; } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_baseMatches.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_baseMatches.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseIsMatch */ "../../../../node_modules/lodash/_baseIsMatch.js" ), v = o( /*! ./_getMatchData */ "../../../../node_modules/lodash/_getMatchData.js" ), u = o( /*! ./_matchesStrictComparable */ "../../../../node_modules/lodash/_matchesStrictComparable.js" ); function l(P) { var p = v(P); return p.length == 1 && p[0][2] ? u(p[0][0], p[0][1]) : function(c) { return c === P || d(c, P, p); }; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_baseMatchesProperty.js": ( /*!***************************************************************!*\ !*** ../../../../node_modules/lodash/_baseMatchesProperty.js ***! \***************************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseIsEqual */ "../../../../node_modules/lodash/_baseIsEqual.js" ), v = o( /*! ./get */ "../../../../node_modules/lodash/get.js" ), u = o( /*! ./hasIn */ "../../../../node_modules/lodash/hasIn.js" ), l = o( /*! ./_isKey */ "../../../../node_modules/lodash/_isKey.js" ), P = o( /*! ./_isStrictComparable */ "../../../../node_modules/lodash/_isStrictComparable.js" ), p = o( /*! ./_matchesStrictComparable */ "../../../../node_modules/lodash/_matchesStrictComparable.js" ), c = o( /*! ./_toKey */ "../../../../node_modules/lodash/_toKey.js" ), H = 1, T = 2; function q(b, j) { return l(b) && P(j) ? p(c(b), j) : function(w) { var m = v(w, b); return m === void 0 && m === j ? u(w, b) : d(j, m, H | T); }; } a.exports = q; } ), /***/ "../../../../node_modules/lodash/_baseMerge.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_baseMerge.js ***! \*****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_Stack */ "../../../../node_modules/lodash/_Stack.js" ), v = o( /*! ./_assignMergeValue */ "../../../../node_modules/lodash/_assignMergeValue.js" ), u = o( /*! ./_baseFor */ "../../../../node_modules/lodash/_baseFor.js" ), l = o( /*! ./_baseMergeDeep */ "../../../../node_modules/lodash/_baseMergeDeep.js" ), P = o( /*! ./isObject */ "../../../../node_modules/lodash/isObject.js" ), p = o( /*! ./keysIn */ "../../../../node_modules/lodash/keysIn.js" ), c = o( /*! ./_safeGet */ "../../../../node_modules/lodash/_safeGet.js" ); function H(T, q, b, j, w) { T !== q && u(q, function(m, I) { if (w || (w = new d()), P(m)) l(T, q, I, b, H, j, w); else { var N = j ? j(c(T, I), m, I + "", T, q, w) : void 0; N === void 0 && (N = m), v(T, I, N); } }, p); } a.exports = H; } ), /***/ "../../../../node_modules/lodash/_baseMergeDeep.js": ( /*!*********************************************************!*\ !*** ../../../../node_modules/lodash/_baseMergeDeep.js ***! \*********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_assignMergeValue */ "../../../../node_modules/lodash/_assignMergeValue.js" ), v = o( /*! ./_cloneBuffer */ "../../../../node_modules/lodash/_cloneBuffer.js" ), u = o( /*! ./_cloneTypedArray */ "../../../../node_modules/lodash/_cloneTypedArray.js" ), l = o( /*! ./_copyArray */ "../../../../node_modules/lodash/_copyArray.js" ), P = o( /*! ./_initCloneObject */ "../../../../node_modules/lodash/_initCloneObject.js" ), p = o( /*! ./isArguments */ "../../../../node_modules/lodash/isArguments.js" ), c = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ), H = o( /*! ./isArrayLikeObject */ "../../../../node_modules/lodash/isArrayLikeObject.js" ), T = o( /*! ./isBuffer */ "../../../../node_modules/lodash/isBuffer.js" ), q = o( /*! ./isFunction */ "../../../../node_modules/lodash/isFunction.js" ), b = o( /*! ./isObject */ "../../../../node_modules/lodash/isObject.js" ), j = o( /*! ./isPlainObject */ "../../../../node_modules/lodash/isPlainObject.js" ), w = o( /*! ./isTypedArray */ "../../../../node_modules/lodash/isTypedArray.js" ), m = o( /*! ./_safeGet */ "../../../../node_modules/lodash/_safeGet.js" ), I = o( /*! ./toPlainObject */ "../../../../node_modules/lodash/toPlainObject.js" ); function N(k, R, y, O, Y, ee, Z) { var te = m(k, y), fe = m(R, y), _ = Z.get(fe); if (_) { d(k, y, _); return; } var G = ee ? ee(te, fe, y + "", k, R, Z) : void 0, L = G === void 0; if (L) { var $ = c(fe), ae = !$ && T(fe), Pe = !$ && !ae && w(fe); G = fe, $ || ae || Pe ? c(te) ? G = te : H(te) ? G = l(te) : ae ? (L = !1, G = v(fe, !0)) : Pe ? (L = !1, G = u(fe, !0)) : G = [] : j(fe) || p(fe) ? (G = te, p(te) ? G = I(te) : (!b(te) || q(te)) && (G = P(fe))) : L = !1; } L && (Z.set(fe, G), Y(G, fe, O, ee, Z), Z.delete(fe)), d(k, y, G); } a.exports = N; } ), /***/ "../../../../node_modules/lodash/_baseOrderBy.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_baseOrderBy.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_arrayMap */ "../../../../node_modules/lodash/_arrayMap.js" ), v = o( /*! ./_baseGet */ "../../../../node_modules/lodash/_baseGet.js" ), u = o( /*! ./_baseIteratee */ "../../../../node_modules/lodash/_baseIteratee.js" ), l = o( /*! ./_baseMap */ "../../../../node_modules/lodash/_baseMap.js" ), P = o( /*! ./_baseSortBy */ "../../../../node_modules/lodash/_baseSortBy.js" ), p = o( /*! ./_baseUnary */ "../../../../node_modules/lodash/_baseUnary.js" ), c = o( /*! ./_compareMultiple */ "../../../../node_modules/lodash/_compareMultiple.js" ), H = o( /*! ./identity */ "../../../../node_modules/lodash/identity.js" ), T = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ); function q(b, j, w) { j.length ? j = d(j, function(N) { return T(N) ? function(k) { return v(k, N.length === 1 ? N[0] : N); } : N; }) : j = [H]; var m = -1; j = d(j, p(u)); var I = l(b, function(N, k, R) { var y = d(j, function(O) { return O(N); }); return { criteria: y, index: ++m, value: N }; }); return P(I, function(N, k) { return c(N, k, w); }); } a.exports = q; } ), /***/ "../../../../node_modules/lodash/_basePick.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_basePick.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_basePickBy */ "../../../../node_modules/lodash/_basePickBy.js" ), v = o( /*! ./hasIn */ "../../../../node_modules/lodash/hasIn.js" ); function u(l, P) { return d(l, P, function(p, c) { return v(l, c); }); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_basePickBy.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_basePickBy.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseGet */ "../../../../node_modules/lodash/_baseGet.js" ), v = o( /*! ./_baseSet */ "../../../../node_modules/lodash/_baseSet.js" ), u = o( /*! ./_castPath */ "../../../../node_modules/lodash/_castPath.js" ); function l(P, p, c) { for (var H = -1, T = p.length, q = {}; ++H < T; ) { var b = p[H], j = d(P, b); c(j, b) && v(q, u(b, P), j); } return q; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_baseProperty.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_baseProperty.js ***! \********************************************************/ /***/ (a) => { function f(o) { return function(d) { return d == null ? void 0 : d[o]; }; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_basePropertyDeep.js": ( /*!************************************************************!*\ !*** ../../../../node_modules/lodash/_basePropertyDeep.js ***! \************************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseGet */ "../../../../node_modules/lodash/_baseGet.js" ); function v(u) { return function(l) { return d(l, u); }; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_baseRange.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_baseRange.js ***! \*****************************************************/ /***/ (a) => { var f = Math.ceil, o = Math.max; function d(v, u, l, P) { for (var p = -1, c = o(f((u - v) / (l || 1)), 0), H = Array(c); c--; ) H[P ? c : ++p] = v, v += l; return H; } a.exports = d; } ), /***/ "../../../../node_modules/lodash/_baseReduce.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_baseReduce.js ***! \******************************************************/ /***/ (a) => { function f(o, d, v, u, l) { return l(o, function(P, p, c) { v = u ? (u = !1, P) : d(v, P, p, c); }), v; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_baseRest.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_baseRest.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./identity */ "../../../../node_modules/lodash/identity.js" ), v = o( /*! ./_overRest */ "../../../../node_modules/lodash/_overRest.js" ), u = o( /*! ./_setToString */ "../../../../node_modules/lodash/_setToString.js" ); function l(P, p) { return u(v(P, p, d), P + ""); } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_baseSet.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/_baseSet.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_assignValue */ "../../../../node_modules/lodash/_assignValue.js" ), v = o( /*! ./_castPath */ "../../../../node_modules/lodash/_castPath.js" ), u = o( /*! ./_isIndex */ "../../../../node_modules/lodash/_isIndex.js" ), l = o( /*! ./isObject */ "../../../../node_modules/lodash/isObject.js" ), P = o( /*! ./_toKey */ "../../../../node_modules/lodash/_toKey.js" ); function p(c, H, T, q) { if (!l(c)) return c; H = v(H, c); for (var b = -1, j = H.length, w = j - 1, m = c; m != null && ++b < j; ) { var I = P(H[b]), N = T; if (I === "__proto__" || I === "constructor" || I === "prototype") return c; if (b != w) { var k = m[I]; N = q ? q(k, I, m) : void 0, N === void 0 && (N = l(k) ? k : u(H[b + 1]) ? [] : {}); } d(m, I, N), m = m[I]; } return c; } a.exports = p; } ), /***/ "../../../../node_modules/lodash/_baseSetToString.js": ( /*!***********************************************************!*\ !*** ../../../../node_modules/lodash/_baseSetToString.js ***! \***********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./constant */ "../../../../node_modules/lodash/constant.js" ), v = o( /*! ./_defineProperty */ "../../../../node_modules/lodash/_defineProperty.js" ), u = o( /*! ./identity */ "../../../../node_modules/lodash/identity.js" ), l = v ? function(P, p) { return v(P, "toString", { configurable: !0, enumerable: !1, value: d(p), writable: !0 }); } : u; a.exports = l; } ), /***/ "../../../../node_modules/lodash/_baseSortBy.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_baseSortBy.js ***! \******************************************************/ /***/ (a) => { function f(o, d) { var v = o.length; for (o.sort(d); v--; ) o[v] = o[v].value; return o; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_baseTimes.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_baseTimes.js ***! \*****************************************************/ /***/ (a) => { function f(o, d) { for (var v = -1, u = Array(o); ++v < o; ) u[v] = d(v); return u; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_baseToString.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_baseToString.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_Symbol */ "../../../../node_modules/lodash/_Symbol.js" ), v = o( /*! ./_arrayMap */ "../../../../node_modules/lodash/_arrayMap.js" ), u = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ), l = o( /*! ./isSymbol */ "../../../../node_modules/lodash/isSymbol.js" ), P = 1 / 0, p = d ? d.prototype : void 0, c = p ? p.toString : void 0; function H(T) { if (typeof T == "string") return T; if (u(T)) return v(T, H) + ""; if (l(T)) return c ? c.call(T) : ""; var q = T + ""; return q == "0" && 1 / T == -P ? "-0" : q; } a.exports = H; } ), /***/ "../../../../node_modules/lodash/_baseTrim.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_baseTrim.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_trimmedEndIndex */ "../../../../node_modules/lodash/_trimmedEndIndex.js" ), v = /^\s+/; function u(l) { return l && l.slice(0, d(l) + 1).replace(v, ""); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_baseUnary.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_baseUnary.js ***! \*****************************************************/ /***/ (a) => { function f(o) { return function(d) { return o(d); }; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_baseUniq.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_baseUniq.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_SetCache */ "../../../../node_modules/lodash/_SetCache.js" ), v = o( /*! ./_arrayIncludes */ "../../../../node_modules/lodash/_arrayIncludes.js" ), u = o( /*! ./_arrayIncludesWith */ "../../../../node_modules/lodash/_arrayIncludesWith.js" ), l = o( /*! ./_cacheHas */ "../../../../node_modules/lodash/_cacheHas.js" ), P = o( /*! ./_createSet */ "../../../../node_modules/lodash/_createSet.js" ), p = o( /*! ./_setToArray */ "../../../../node_modules/lodash/_setToArray.js" ), c = 200; function H(T, q, b) { var j = -1, w = v, m = T.length, I = !0, N = [], k = N; if (b) I = !1, w = u; else if (m >= c) { var R = q ? null : P(T); if (R) return p(R); I = !1, w = l, k = new d(); } else k = q ? [] : N; e: for (; ++j < m; ) { var y = T[j], O = q ? q(y) : y; if (y = b || y !== 0 ? y : 0, I && O === O) { for (var Y = k.length; Y--; ) if (k[Y] === O) continue e; q && k.push(O), N.push(y); } else w(k, O, b) || (k !== N && k.push(O), N.push(y)); } return N; } a.exports = H; } ), /***/ "../../../../node_modules/lodash/_baseValues.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_baseValues.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_arrayMap */ "../../../../node_modules/lodash/_arrayMap.js" ); function v(u, l) { return d(l, function(P) { return u[P]; }); } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_baseZipObject.js": ( /*!*********************************************************!*\ !*** ../../../../node_modules/lodash/_baseZipObject.js ***! \*********************************************************/ /***/ (a) => { function f(o, d, v) { for (var u = -1, l = o.length, P = d.length, p = {}; ++u < l; ) { var c = u < P ? d[u] : void 0; v(p, o[u], c); } return p; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_cacheHas.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_cacheHas.js ***! \****************************************************/ /***/ (a) => { function f(o, d) { return o.has(d); } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_castFunction.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_castFunction.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./identity */ "../../../../node_modules/lodash/identity.js" ); function v(u) { return typeof u == "function" ? u : d; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_castPath.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_castPath.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ), v = o( /*! ./_isKey */ "../../../../node_modules/lodash/_isKey.js" ), u = o( /*! ./_stringToPath */ "../../../../node_modules/lodash/_stringToPath.js" ), l = o( /*! ./toString */ "../../../../node_modules/lodash/toString.js" ); function P(p, c) { return d(p) ? p : v(p, c) ? [p] : u(l(p)); } a.exports = P; } ), /***/ "../../../../node_modules/lodash/_cloneArrayBuffer.js": ( /*!************************************************************!*\ !*** ../../../../node_modules/lodash/_cloneArrayBuffer.js ***! \************************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_Uint8Array */ "../../../../node_modules/lodash/_Uint8Array.js" ); function v(u) { var l = new u.constructor(u.byteLength); return new d(l).set(new d(u)), l; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_cloneBuffer.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_cloneBuffer.js ***! \*******************************************************/ /***/ (a, f, o) => { a = o.nmd(a); var d = o( /*! ./_root */ "../../../../node_modules/lodash/_root.js" ), v = f && !f.nodeType && f, u = v && !0 && a && !a.nodeType && a, l = u && u.exports === v, P = l ? d.Buffer : void 0, p = P ? P.allocUnsafe : void 0; function c(H, T) { if (T) return H.slice(); var q = H.length, b = p ? p(q) : new H.constructor(q); return H.copy(b), b; } a.exports = c; } ), /***/ "../../../../node_modules/lodash/_cloneDataView.js": ( /*!*********************************************************!*\ !*** ../../../../node_modules/lodash/_cloneDataView.js ***! \*********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_cloneArrayBuffer */ "../../../../node_modules/lodash/_cloneArrayBuffer.js" ); function v(u, l) { var P = l ? d(u.buffer) : u.buffer; return new u.constructor(P, u.byteOffset, u.byteLength); } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_cloneRegExp.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_cloneRegExp.js ***! \*******************************************************/ /***/ (a) => { var f = /\w*$/; function o(d) { var v = new d.constructor(d.source, f.exec(d)); return v.lastIndex = d.lastIndex, v; } a.exports = o; } ), /***/ "../../../../node_modules/lodash/_cloneSymbol.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_cloneSymbol.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_Symbol */ "../../../../node_modules/lodash/_Symbol.js" ), v = d ? d.prototype : void 0, u = v ? v.valueOf : void 0; function l(P) { return u ? Object(u.call(P)) : {}; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_cloneTypedArray.js": ( /*!***********************************************************!*\ !*** ../../../../node_modules/lodash/_cloneTypedArray.js ***! \***********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_cloneArrayBuffer */ "../../../../node_modules/lodash/_cloneArrayBuffer.js" ); function v(u, l) { var P = l ? d(u.buffer) : u.buffer; return new u.constructor(P, u.byteOffset, u.length); } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_compareAscending.js": ( /*!************************************************************!*\ !*** ../../../../node_modules/lodash/_compareAscending.js ***! \************************************************************/ /***/ (a, f, o) => { var d = o( /*! ./isSymbol */ "../../../../node_modules/lodash/isSymbol.js" ); function v(u, l) { if (u !== l) { var P = u !== void 0, p = u === null, c = u === u, H = d(u), T = l !== void 0, q = l === null, b = l === l, j = d(l); if (!q && !j && !H && u > l || H && T && b && !q && !j || p && T && b || !P && b || !c) return 1; if (!p && !H && !j && u < l || j && P && c && !p && !H || q && P && c || !T && c || !b) return -1; } return 0; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_compareMultiple.js": ( /*!***********************************************************!*\ !*** ../../../../node_modules/lodash/_compareMultiple.js ***! \***********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_compareAscending */ "../../../../node_modules/lodash/_compareAscending.js" ); function v(u, l, P) { for (var p = -1, c = u.criteria, H = l.criteria, T = c.length, q = P.length; ++p < T; ) { var b = d(c[p], H[p]); if (b) { if (p >= q) return b; var j = P[p]; return b * (j == "desc" ? -1 : 1); } } return u.index - l.index; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_copyArray.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_copyArray.js ***! \*****************************************************/ /***/ (a) => { function f(o, d) { var v = -1, u = o.length; for (d || (d = Array(u)); ++v < u; ) d[v] = o[v]; return d; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_copyObject.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_copyObject.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_assignValue */ "../../../../node_modules/lodash/_assignValue.js" ), v = o( /*! ./_baseAssignValue */ "../../../../node_modules/lodash/_baseAssignValue.js" ); function u(l, P, p, c) { var H = !p; p || (p = {}); for (var T = -1, q = P.length; ++T < q; ) { var b = P[T], j = c ? c(p[b], l[b], b, p, l) : void 0; j === void 0 && (j = l[b]), H ? v(p, b, j) : d(p, b, j); } return p; } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_copySymbols.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_copySymbols.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_copyObject */ "../../../../node_modules/lodash/_copyObject.js" ), v = o( /*! ./_getSymbols */ "../../../../node_modules/lodash/_getSymbols.js" ); function u(l, P) { return d(l, v(l), P); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_copySymbolsIn.js": ( /*!*********************************************************!*\ !*** ../../../../node_modules/lodash/_copySymbolsIn.js ***! \*********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_copyObject */ "../../../../node_modules/lodash/_copyObject.js" ), v = o( /*! ./_getSymbolsIn */ "../../../../node_modules/lodash/_getSymbolsIn.js" ); function u(l, P) { return d(l, v(l), P); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_coreJsData.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_coreJsData.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_root */ "../../../../node_modules/lodash/_root.js" ), v = d["__core-js_shared__"]; a.exports = v; } ), /***/ "../../../../node_modules/lodash/_createAssigner.js": ( /*!**********************************************************!*\ !*** ../../../../node_modules/lodash/_createAssigner.js ***! \**********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseRest */ "../../../../node_modules/lodash/_baseRest.js" ), v = o( /*! ./_isIterateeCall */ "../../../../node_modules/lodash/_isIterateeCall.js" ); function u(l) { return d(function(P, p) { var c = -1, H = p.length, T = H > 1 ? p[H - 1] : void 0, q = H > 2 ? p[2] : void 0; for (T = l.length > 3 && typeof T == "function" ? (H--, T) : void 0, q && v(p[0], p[1], q) && (T = H < 3 ? void 0 : T, H = 1), P = Object(P); ++c < H; ) { var b = p[c]; b && l(P, b, c, T); } return P; }); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_createBaseEach.js": ( /*!**********************************************************!*\ !*** ../../../../node_modules/lodash/_createBaseEach.js ***! \**********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./isArrayLike */ "../../../../node_modules/lodash/isArrayLike.js" ); function v(u, l) { return function(P, p) { if (P == null) return P; if (!d(P)) return u(P, p); for (var c = P.length, H = l ? c : -1, T = Object(P); (l ? H-- : ++H < c) && p(T[H], H, T) !== !1; ) ; return P; }; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_createBaseFor.js": ( /*!*********************************************************!*\ !*** ../../../../node_modules/lodash/_createBaseFor.js ***! \*********************************************************/ /***/ (a) => { function f(o) { return function(d, v, u) { for (var l = -1, P = Object(d), p = u(d), c = p.length; c--; ) { var H = p[o ? c : ++l]; if (v(P[H], H, P) === !1) break; } return d; }; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_createFind.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_createFind.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseIteratee */ "../../../../node_modules/lodash/_baseIteratee.js" ), v = o( /*! ./isArrayLike */ "../../../../node_modules/lodash/isArrayLike.js" ), u = o( /*! ./keys */ "../../../../node_modules/lodash/keys.js" ); function l(P) { return function(p, c, H) { var T = Object(p); if (!v(p)) { var q = d(c, 3); p = u(p), c = function(j) { return q(T[j], j, T); }; } var b = P(p, c, H); return b > -1 ? T[q ? p[b] : b] : void 0; }; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_createRange.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_createRange.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseRange */ "../../../../node_modules/lodash/_baseRange.js" ), v = o( /*! ./_isIterateeCall */ "../../../../node_modules/lodash/_isIterateeCall.js" ), u = o( /*! ./toFinite */ "../../../../node_modules/lodash/toFinite.js" ); function l(P) { return function(p, c, H) { return H && typeof H != "number" && v(p, c, H) && (c = H = void 0), p = u(p), c === void 0 ? (c = p, p = 0) : c = u(c), H = H === void 0 ? p < c ? 1 : -1 : u(H), d(p, c, H, P); }; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_createSet.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_createSet.js ***! \*****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_Set */ "../../../../node_modules/lodash/_Set.js" ), v = o( /*! ./noop */ "../../../../node_modules/lodash/noop.js" ), u = o( /*! ./_setToArray */ "../../../../node_modules/lodash/_setToArray.js" ), l = 1 / 0, P = d && 1 / u(new d([, -0]))[1] == l ? function(p) { return new d(p); } : v; a.exports = P; } ), /***/ "../../../../node_modules/lodash/_defineProperty.js": ( /*!**********************************************************!*\ !*** ../../../../node_modules/lodash/_defineProperty.js ***! \**********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_getNative */ "../../../../node_modules/lodash/_getNative.js" ), v = function() { try { var u = d(Object, "defineProperty"); return u({}, "", {}), u; } catch { } }(); a.exports = v; } ), /***/ "../../../../node_modules/lodash/_equalArrays.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_equalArrays.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_SetCache */ "../../../../node_modules/lodash/_SetCache.js" ), v = o( /*! ./_arraySome */ "../../../../node_modules/lodash/_arraySome.js" ), u = o( /*! ./_cacheHas */ "../../../../node_modules/lodash/_cacheHas.js" ), l = 1, P = 2; function p(c, H, T, q, b, j) { var w = T & l, m = c.length, I = H.length; if (m != I && !(w && I > m)) return !1; var N = j.get(c), k = j.get(H); if (N && k) return N == H && k == c; var R = -1, y = !0, O = T & P ? new d() : void 0; for (j.set(c, H), j.set(H, c); ++R < m; ) { var Y = c[R], ee = H[R]; if (q) var Z = w ? q(ee, Y, R, H, c, j) : q(Y, ee, R, c, H, j); if (Z !== void 0) { if (Z) continue; y = !1; break; } if (O) { if (!v(H, function(te, fe) { if (!u(O, fe) && (Y === te || b(Y, te, T, q, j))) return O.push(fe); })) { y = !1; break; } } else if (!(Y === ee || b(Y, ee, T, q, j))) { y = !1; break; } } return j.delete(c), j.delete(H), y; } a.exports = p; } ), /***/ "../../../../node_modules/lodash/_equalByTag.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_equalByTag.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_Symbol */ "../../../../node_modules/lodash/_Symbol.js" ), v = o( /*! ./_Uint8Array */ "../../../../node_modules/lodash/_Uint8Array.js" ), u = o( /*! ./eq */ "../../../../node_modules/lodash/eq.js" ), l = o( /*! ./_equalArrays */ "../../../../node_modules/lodash/_equalArrays.js" ), P = o( /*! ./_mapToArray */ "../../../../node_modules/lodash/_mapToArray.js" ), p = o( /*! ./_setToArray */ "../../../../node_modules/lodash/_setToArray.js" ), c = 1, H = 2, T = "[object Boolean]", q = "[object Date]", b = "[object Error]", j = "[object Map]", w = "[object Number]", m = "[object RegExp]", I = "[object Set]", N = "[object String]", k = "[object Symbol]", R = "[object ArrayBuffer]", y = "[object DataView]", O = d ? d.prototype : void 0, Y = O ? O.valueOf : void 0; function ee(Z, te, fe, _, G, L, $) { switch (fe) { case y: if (Z.byteLength != te.byteLength || Z.byteOffset != te.byteOffset) return !1; Z = Z.buffer, te = te.buffer; case R: return !(Z.byteLength != te.byteLength || !L(new v(Z), new v(te))); case T: case q: case w: return u(+Z, +te); case b: return Z.name == te.name && Z.message == te.message; case m: case N: return Z == te + ""; case j: var ae = P; case I: var Pe = _ & c; if (ae || (ae = p), Z.size != te.size && !Pe) return !1; var ge = $.get(Z); if (ge) return ge == te; _ |= H, $.set(Z, te); var me = l(ae(Z), ae(te), _, G, L, $); return $.delete(Z), me; case k: if (Y) return Y.call(Z) == Y.call(te); } return !1; } a.exports = ee; } ), /***/ "../../../../node_modules/lodash/_equalObjects.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_equalObjects.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_getAllKeys */ "../../../../node_modules/lodash/_getAllKeys.js" ), v = 1, u = Object.prototype, l = u.hasOwnProperty; function P(p, c, H, T, q, b) { var j = H & v, w = d(p), m = w.length, I = d(c), N = I.length; if (m != N && !j) return !1; for (var k = m; k--; ) { var R = w[k]; if (!(j ? R in c : l.call(c, R))) return !1; } var y = b.get(p), O = b.get(c); if (y && O) return y == c && O == p; var Y = !0; b.set(p, c), b.set(c, p); for (var ee = j; ++k < m; ) { R = w[k]; var Z = p[R], te = c[R]; if (T) var fe = j ? T(te, Z, R, c, p, b) : T(Z, te, R, p, c, b); if (!(fe === void 0 ? Z === te || q(Z, te, H, T, b) : fe)) { Y = !1; break; } ee || (ee = R == "constructor"); } if (Y && !ee) { var _ = p.constructor, G = c.constructor; _ != G && "constructor" in p && "constructor" in c && !(typeof _ == "function" && _ instanceof _ && typeof G == "function" && G instanceof G) && (Y = !1); } return b.delete(p), b.delete(c), Y; } a.exports = P; } ), /***/ "../../../../node_modules/lodash/_flatRest.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_flatRest.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./flatten */ "../../../../node_modules/lodash/flatten.js" ), v = o( /*! ./_overRest */ "../../../../node_modules/lodash/_overRest.js" ), u = o( /*! ./_setToString */ "../../../../node_modules/lodash/_setToString.js" ); function l(P) { return u(v(P, void 0, d), P + ""); } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_freeGlobal.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_freeGlobal.js ***! \******************************************************/ /***/ (a, f, o) => { var d = typeof o.g == "object" && o.g && o.g.Object === Object && o.g; a.exports = d; } ), /***/ "../../../../node_modules/lodash/_getAllKeys.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_getAllKeys.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseGetAllKeys */ "../../../../node_modules/lodash/_baseGetAllKeys.js" ), v = o( /*! ./_getSymbols */ "../../../../node_modules/lodash/_getSymbols.js" ), u = o( /*! ./keys */ "../../../../node_modules/lodash/keys.js" ); function l(P) { return d(P, u, v); } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_getAllKeysIn.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_getAllKeysIn.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseGetAllKeys */ "../../../../node_modules/lodash/_baseGetAllKeys.js" ), v = o( /*! ./_getSymbolsIn */ "../../../../node_modules/lodash/_getSymbolsIn.js" ), u = o( /*! ./keysIn */ "../../../../node_modules/lodash/keysIn.js" ); function l(P) { return d(P, u, v); } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_getMapData.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_getMapData.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_isKeyable */ "../../../../node_modules/lodash/_isKeyable.js" ); function v(u, l) { var P = u.__data__; return d(l) ? P[typeof l == "string" ? "string" : "hash"] : P.map; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_getMatchData.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_getMatchData.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_isStrictComparable */ "../../../../node_modules/lodash/_isStrictComparable.js" ), v = o( /*! ./keys */ "../../../../node_modules/lodash/keys.js" ); function u(l) { for (var P = v(l), p = P.length; p--; ) { var c = P[p], H = l[c]; P[p] = [c, H, d(H)]; } return P; } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_getNative.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_getNative.js ***! \*****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseIsNative */ "../../../../node_modules/lodash/_baseIsNative.js" ), v = o( /*! ./_getValue */ "../../../../node_modules/lodash/_getValue.js" ); function u(l, P) { var p = v(l, P); return d(p) ? p : void 0; } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_getPrototype.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_getPrototype.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_overArg */ "../../../../node_modules/lodash/_overArg.js" ), v = d(Object.getPrototypeOf, Object); a.exports = v; } ), /***/ "../../../../node_modules/lodash/_getRawTag.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_getRawTag.js ***! \*****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_Symbol */ "../../../../node_modules/lodash/_Symbol.js" ), v = Object.prototype, u = v.hasOwnProperty, l = v.toString, P = d ? d.toStringTag : void 0; function p(c) { var H = u.call(c, P), T = c[P]; try { c[P] = void 0; var q = !0; } catch { } var b = l.call(c); return q && (H ? c[P] = T : delete c[P]), b; } a.exports = p; } ), /***/ "../../../../node_modules/lodash/_getSymbols.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_getSymbols.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_arrayFilter */ "../../../../node_modules/lodash/_arrayFilter.js" ), v = o( /*! ./stubArray */ "../../../../node_modules/lodash/stubArray.js" ), u = Object.prototype, l = u.propertyIsEnumerable, P = Object.getOwnPropertySymbols, p = P ? function(c) { return c == null ? [] : (c = Object(c), d(P(c), function(H) { return l.call(c, H); })); } : v; a.exports = p; } ), /***/ "../../../../node_modules/lodash/_getSymbolsIn.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_getSymbolsIn.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_arrayPush */ "../../../../node_modules/lodash/_arrayPush.js" ), v = o( /*! ./_getPrototype */ "../../../../node_modules/lodash/_getPrototype.js" ), u = o( /*! ./_getSymbols */ "../../../../node_modules/lodash/_getSymbols.js" ), l = o( /*! ./stubArray */ "../../../../node_modules/lodash/stubArray.js" ), P = Object.getOwnPropertySymbols, p = P ? function(c) { for (var H = []; c; ) d(H, u(c)), c = v(c); return H; } : l; a.exports = p; } ), /***/ "../../../../node_modules/lodash/_getTag.js": ( /*!**************************************************!*\ !*** ../../../../node_modules/lodash/_getTag.js ***! \**************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_DataView */ "../../../../node_modules/lodash/_DataView.js" ), v = o( /*! ./_Map */ "../../../../node_modules/lodash/_Map.js" ), u = o( /*! ./_Promise */ "../../../../node_modules/lodash/_Promise.js" ), l = o( /*! ./_Set */ "../../../../node_modules/lodash/_Set.js" ), P = o( /*! ./_WeakMap */ "../../../../node_modules/lodash/_WeakMap.js" ), p = o( /*! ./_baseGetTag */ "../../../../node_modules/lodash/_baseGetTag.js" ), c = o( /*! ./_toSource */ "../../../../node_modules/lodash/_toSource.js" ), H = "[object Map]", T = "[object Object]", q = "[object Promise]", b = "[object Set]", j = "[object WeakMap]", w = "[object DataView]", m = c(d), I = c(v), N = c(u), k = c(l), R = c(P), y = p; (d && y(new d(new ArrayBuffer(1))) != w || v && y(new v()) != H || u && y(u.resolve()) != q || l && y(new l()) != b || P && y(new P()) != j) && (y = function(O) { var Y = p(O), ee = Y == T ? O.constructor : void 0, Z = ee ? c(ee) : ""; if (Z) switch (Z) { case m: return w; case I: return H; case N: return q; case k: return b; case R: return j; } return Y; }), a.exports = y; } ), /***/ "../../../../node_modules/lodash/_getValue.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_getValue.js ***! \****************************************************/ /***/ (a) => { function f(o, d) { return o == null ? void 0 : o[d]; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_hasPath.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/_hasPath.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_castPath */ "../../../../node_modules/lodash/_castPath.js" ), v = o( /*! ./isArguments */ "../../../../node_modules/lodash/isArguments.js" ), u = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ), l = o( /*! ./_isIndex */ "../../../../node_modules/lodash/_isIndex.js" ), P = o( /*! ./isLength */ "../../../../node_modules/lodash/isLength.js" ), p = o( /*! ./_toKey */ "../../../../node_modules/lodash/_toKey.js" ); function c(H, T, q) { T = d(T, H); for (var b = -1, j = T.length, w = !1; ++b < j; ) { var m = p(T[b]); if (!(w = H != null && q(H, m))) break; H = H[m]; } return w || ++b != j ? w : (j = H == null ? 0 : H.length, !!j && P(j) && l(m, j) && (u(H) || v(H))); } a.exports = c; } ), /***/ "../../../../node_modules/lodash/_hasUnicode.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_hasUnicode.js ***! \******************************************************/ /***/ (a) => { var f = "\\ud800-\\udfff", o = "\\u0300-\\u036f", d = "\\ufe20-\\ufe2f", v = "\\u20d0-\\u20ff", u = o + d + v, l = "\\ufe0e\\ufe0f", P = "\\u200d", p = RegExp("[" + P + f + u + l + "]"); function c(H) { return p.test(H); } a.exports = c; } ), /***/ "../../../../node_modules/lodash/_hashClear.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_hashClear.js ***! \*****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_nativeCreate */ "../../../../node_modules/lodash/_nativeCreate.js" ); function v() { this.__data__ = d ? d(null) : {}, this.size = 0; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_hashDelete.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_hashDelete.js ***! \******************************************************/ /***/ (a) => { function f(o) { var d = this.has(o) && delete this.__data__[o]; return this.size -= d ? 1 : 0, d; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_hashGet.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/_hashGet.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_nativeCreate */ "../../../../node_modules/lodash/_nativeCreate.js" ), v = "__lodash_hash_undefined__", u = Object.prototype, l = u.hasOwnProperty; function P(p) { var c = this.__data__; if (d) { var H = c[p]; return H === v ? void 0 : H; } return l.call(c, p) ? c[p] : void 0; } a.exports = P; } ), /***/ "../../../../node_modules/lodash/_hashHas.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/_hashHas.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_nativeCreate */ "../../../../node_modules/lodash/_nativeCreate.js" ), v = Object.prototype, u = v.hasOwnProperty; function l(P) { var p = this.__data__; return d ? p[P] !== void 0 : u.call(p, P); } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_hashSet.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/_hashSet.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_nativeCreate */ "../../../../node_modules/lodash/_nativeCreate.js" ), v = "__lodash_hash_undefined__"; function u(l, P) { var p = this.__data__; return this.size += this.has(l) ? 0 : 1, p[l] = d && P === void 0 ? v : P, this; } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_initCloneArray.js": ( /*!**********************************************************!*\ !*** ../../../../node_modules/lodash/_initCloneArray.js ***! \**********************************************************/ /***/ (a) => { var f = Object.prototype, o = f.hasOwnProperty; function d(v) { var u = v.length, l = new v.constructor(u); return u && typeof v[0] == "string" && o.call(v, "index") && (l.index = v.index, l.input = v.input), l; } a.exports = d; } ), /***/ "../../../../node_modules/lodash/_initCloneByTag.js": ( /*!**********************************************************!*\ !*** ../../../../node_modules/lodash/_initCloneByTag.js ***! \**********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_cloneArrayBuffer */ "../../../../node_modules/lodash/_cloneArrayBuffer.js" ), v = o( /*! ./_cloneDataView */ "../../../../node_modules/lodash/_cloneDataView.js" ), u = o( /*! ./_cloneRegExp */ "../../../../node_modules/lodash/_cloneRegExp.js" ), l = o( /*! ./_cloneSymbol */ "../../../../node_modules/lodash/_cloneSymbol.js" ), P = o( /*! ./_cloneTypedArray */ "../../../../node_modules/lodash/_cloneTypedArray.js" ), p = "[object Boolean]", c = "[object Date]", H = "[object Map]", T = "[object Number]", q = "[object RegExp]", b = "[object Set]", j = "[object String]", w = "[object Symbol]", m = "[object ArrayBuffer]", I = "[object DataView]", N = "[object Float32Array]", k = "[object Float64Array]", R = "[object Int8Array]", y = "[object Int16Array]", O = "[object Int32Array]", Y = "[object Uint8Array]", ee = "[object Uint8ClampedArray]", Z = "[object Uint16Array]", te = "[object Uint32Array]"; function fe(_, G, L) { var $ = _.constructor; switch (G) { case m: return d(_); case p: case c: return new $(+_); case I: return v(_, L); case N: case k: case R: case y: case O: case Y: case ee: case Z: case te: return P(_, L); case H: return new $(); case T: case j: return new $(_); case q: return u(_); case b: return new $(); case w: return l(_); } } a.exports = fe; } ), /***/ "../../../../node_modules/lodash/_initCloneObject.js": ( /*!***********************************************************!*\ !*** ../../../../node_modules/lodash/_initCloneObject.js ***! \***********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseCreate */ "../../../../node_modules/lodash/_baseCreate.js" ), v = o( /*! ./_getPrototype */ "../../../../node_modules/lodash/_getPrototype.js" ), u = o( /*! ./_isPrototype */ "../../../../node_modules/lodash/_isPrototype.js" ); function l(P) { return typeof P.constructor == "function" && !u(P) ? d(v(P)) : {}; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_isFlattenable.js": ( /*!*********************************************************!*\ !*** ../../../../node_modules/lodash/_isFlattenable.js ***! \*********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_Symbol */ "../../../../node_modules/lodash/_Symbol.js" ), v = o( /*! ./isArguments */ "../../../../node_modules/lodash/isArguments.js" ), u = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ), l = d ? d.isConcatSpreadable : void 0; function P(p) { return u(p) || v(p) || !!(l && p && p[l]); } a.exports = P; } ), /***/ "../../../../node_modules/lodash/_isIndex.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/_isIndex.js ***! \***************************************************/ /***/ (a) => { var f = 9007199254740991, o = /^(?:0|[1-9]\d*)$/; function d(v, u) { var l = typeof v; return u = u ?? f, !!u && (l == "number" || l != "symbol" && o.test(v)) && v > -1 && v % 1 == 0 && v < u; } a.exports = d; } ), /***/ "../../../../node_modules/lodash/_isIterateeCall.js": ( /*!**********************************************************!*\ !*** ../../../../node_modules/lodash/_isIterateeCall.js ***! \**********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./eq */ "../../../../node_modules/lodash/eq.js" ), v = o( /*! ./isArrayLike */ "../../../../node_modules/lodash/isArrayLike.js" ), u = o( /*! ./_isIndex */ "../../../../node_modules/lodash/_isIndex.js" ), l = o( /*! ./isObject */ "../../../../node_modules/lodash/isObject.js" ); function P(p, c, H) { if (!l(H)) return !1; var T = typeof c; return (T == "number" ? v(H) && u(c, H.length) : T == "string" && c in H) ? d(H[c], p) : !1; } a.exports = P; } ), /***/ "../../../../node_modules/lodash/_isKey.js": ( /*!*************************************************!*\ !*** ../../../../node_modules/lodash/_isKey.js ***! \*************************************************/ /***/ (a, f, o) => { var d = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ), v = o( /*! ./isSymbol */ "../../../../node_modules/lodash/isSymbol.js" ), u = /\.|\[(?:[^[\]]*|(["'])(?:(?!\1)[^\\]|\\.)*?\1)\]/, l = /^\w*$/; function P(p, c) { if (d(p)) return !1; var H = typeof p; return H == "number" || H == "symbol" || H == "boolean" || p == null || v(p) ? !0 : l.test(p) || !u.test(p) || c != null && p in Object(c); } a.exports = P; } ), /***/ "../../../../node_modules/lodash/_isKeyable.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/_isKeyable.js ***! \*****************************************************/ /***/ (a) => { function f(o) { var d = typeof o; return d == "string" || d == "number" || d == "symbol" || d == "boolean" ? o !== "__proto__" : o === null; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_isMasked.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_isMasked.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_coreJsData */ "../../../../node_modules/lodash/_coreJsData.js" ), v = function() { var l = /[^.]+$/.exec(d && d.keys && d.keys.IE_PROTO || ""); return l ? "Symbol(src)_1." + l : ""; }(); function u(l) { return !!v && v in l; } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_isPrototype.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_isPrototype.js ***! \*******************************************************/ /***/ (a) => { var f = Object.prototype; function o(d) { var v = d && d.constructor, u = typeof v == "function" && v.prototype || f; return d === u; } a.exports = o; } ), /***/ "../../../../node_modules/lodash/_isStrictComparable.js": ( /*!**************************************************************!*\ !*** ../../../../node_modules/lodash/_isStrictComparable.js ***! \**************************************************************/ /***/ (a, f, o) => { var d = o( /*! ./isObject */ "../../../../node_modules/lodash/isObject.js" ); function v(u) { return u === u && !d(u); } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_listCacheClear.js": ( /*!**********************************************************!*\ !*** ../../../../node_modules/lodash/_listCacheClear.js ***! \**********************************************************/ /***/ (a) => { function f() { this.__data__ = [], this.size = 0; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_listCacheDelete.js": ( /*!***********************************************************!*\ !*** ../../../../node_modules/lodash/_listCacheDelete.js ***! \***********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_assocIndexOf */ "../../../../node_modules/lodash/_assocIndexOf.js" ), v = Array.prototype, u = v.splice; function l(P) { var p = this.__data__, c = d(p, P); if (c < 0) return !1; var H = p.length - 1; return c == H ? p.pop() : u.call(p, c, 1), --this.size, !0; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_listCacheGet.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_listCacheGet.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_assocIndexOf */ "../../../../node_modules/lodash/_assocIndexOf.js" ); function v(u) { var l = this.__data__, P = d(l, u); return P < 0 ? void 0 : l[P][1]; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_listCacheHas.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_listCacheHas.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_assocIndexOf */ "../../../../node_modules/lodash/_assocIndexOf.js" ); function v(u) { return d(this.__data__, u) > -1; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_listCacheSet.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_listCacheSet.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_assocIndexOf */ "../../../../node_modules/lodash/_assocIndexOf.js" ); function v(u, l) { var P = this.__data__, p = d(P, u); return p < 0 ? (++this.size, P.push([u, l])) : P[p][1] = l, this; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_mapCacheClear.js": ( /*!*********************************************************!*\ !*** ../../../../node_modules/lodash/_mapCacheClear.js ***! \*********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_Hash */ "../../../../node_modules/lodash/_Hash.js" ), v = o( /*! ./_ListCache */ "../../../../node_modules/lodash/_ListCache.js" ), u = o( /*! ./_Map */ "../../../../node_modules/lodash/_Map.js" ); function l() { this.size = 0, this.__data__ = { hash: new d(), map: new (u || v)(), string: new d() }; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_mapCacheDelete.js": ( /*!**********************************************************!*\ !*** ../../../../node_modules/lodash/_mapCacheDelete.js ***! \**********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_getMapData */ "../../../../node_modules/lodash/_getMapData.js" ); function v(u) { var l = d(this, u).delete(u); return this.size -= l ? 1 : 0, l; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_mapCacheGet.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_mapCacheGet.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_getMapData */ "../../../../node_modules/lodash/_getMapData.js" ); function v(u) { return d(this, u).get(u); } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_mapCacheHas.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_mapCacheHas.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_getMapData */ "../../../../node_modules/lodash/_getMapData.js" ); function v(u) { return d(this, u).has(u); } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_mapCacheSet.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_mapCacheSet.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_getMapData */ "../../../../node_modules/lodash/_getMapData.js" ); function v(u, l) { var P = d(this, u), p = P.size; return P.set(u, l), this.size += P.size == p ? 0 : 1, this; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_mapToArray.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_mapToArray.js ***! \******************************************************/ /***/ (a) => { function f(o) { var d = -1, v = Array(o.size); return o.forEach(function(u, l) { v[++d] = [l, u]; }), v; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_matchesStrictComparable.js": ( /*!*******************************************************************!*\ !*** ../../../../node_modules/lodash/_matchesStrictComparable.js ***! \*******************************************************************/ /***/ (a) => { function f(o, d) { return function(v) { return v == null ? !1 : v[o] === d && (d !== void 0 || o in Object(v)); }; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_memoizeCapped.js": ( /*!*********************************************************!*\ !*** ../../../../node_modules/lodash/_memoizeCapped.js ***! \*********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./memoize */ "../../../../node_modules/lodash/memoize.js" ), v = 500; function u(l) { var P = d(l, function(c) { return p.size === v && p.clear(), c; }), p = P.cache; return P; } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_nativeCreate.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_nativeCreate.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_getNative */ "../../../../node_modules/lodash/_getNative.js" ), v = d(Object, "create"); a.exports = v; } ), /***/ "../../../../node_modules/lodash/_nativeKeys.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_nativeKeys.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_overArg */ "../../../../node_modules/lodash/_overArg.js" ), v = d(Object.keys, Object); a.exports = v; } ), /***/ "../../../../node_modules/lodash/_nativeKeysIn.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_nativeKeysIn.js ***! \********************************************************/ /***/ (a) => { function f(o) { var d = []; if (o != null) for (var v in Object(o)) d.push(v); return d; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_nodeUtil.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_nodeUtil.js ***! \****************************************************/ /***/ (a, f, o) => { a = o.nmd(a); var d = o( /*! ./_freeGlobal */ "../../../../node_modules/lodash/_freeGlobal.js" ), v = f && !f.nodeType && f, u = v && !0 && a && !a.nodeType && a, l = u && u.exports === v, P = l && d.process, p = function() { try { var c = u && u.require && u.require("util").types; return c || P && P.binding && P.binding("util"); } catch { } }(); a.exports = p; } ), /***/ "../../../../node_modules/lodash/_objectToString.js": ( /*!**********************************************************!*\ !*** ../../../../node_modules/lodash/_objectToString.js ***! \**********************************************************/ /***/ (a) => { var f = Object.prototype, o = f.toString; function d(v) { return o.call(v); } a.exports = d; } ), /***/ "../../../../node_modules/lodash/_overArg.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/_overArg.js ***! \***************************************************/ /***/ (a) => { function f(o, d) { return function(v) { return o(d(v)); }; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_overRest.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_overRest.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_apply */ "../../../../node_modules/lodash/_apply.js" ), v = Math.max; function u(l, P, p) { return P = v(P === void 0 ? l.length - 1 : P, 0), function() { for (var c = arguments, H = -1, T = v(c.length - P, 0), q = Array(T); ++H < T; ) q[H] = c[P + H]; H = -1; for (var b = Array(P + 1); ++H < P; ) b[H] = c[H]; return b[P] = p(q), d(l, this, b); }; } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_root.js": ( /*!************************************************!*\ !*** ../../../../node_modules/lodash/_root.js ***! \************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_freeGlobal */ "../../../../node_modules/lodash/_freeGlobal.js" ), v = typeof self == "object" && self && self.Object === Object && self, u = d || v || Function("return this")(); a.exports = u; } ), /***/ "../../../../node_modules/lodash/_safeGet.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/_safeGet.js ***! \***************************************************/ /***/ (a) => { function f(o, d) { if (!(d === "constructor" && typeof o[d] == "function") && d != "__proto__") return o[d]; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_setCacheAdd.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_setCacheAdd.js ***! \*******************************************************/ /***/ (a) => { var f = "__lodash_hash_undefined__"; function o(d) { return this.__data__.set(d, f), this; } a.exports = o; } ), /***/ "../../../../node_modules/lodash/_setCacheHas.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_setCacheHas.js ***! \*******************************************************/ /***/ (a) => { function f(o) { return this.__data__.has(o); } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_setToArray.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_setToArray.js ***! \******************************************************/ /***/ (a) => { function f(o) { var d = -1, v = Array(o.size); return o.forEach(function(u) { v[++d] = u; }), v; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_setToString.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_setToString.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseSetToString */ "../../../../node_modules/lodash/_baseSetToString.js" ), v = o( /*! ./_shortOut */ "../../../../node_modules/lodash/_shortOut.js" ), u = v(d); a.exports = u; } ), /***/ "../../../../node_modules/lodash/_shortOut.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_shortOut.js ***! \****************************************************/ /***/ (a) => { var f = 800, o = 16, d = Date.now; function v(u) { var l = 0, P = 0; return function() { var p = d(), c = o - (p - P); if (P = p, c > 0) { if (++l >= f) return arguments[0]; } else l = 0; return u.apply(void 0, arguments); }; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_stackClear.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_stackClear.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_ListCache */ "../../../../node_modules/lodash/_ListCache.js" ); function v() { this.__data__ = new d(), this.size = 0; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/_stackDelete.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_stackDelete.js ***! \*******************************************************/ /***/ (a) => { function f(o) { var d = this.__data__, v = d.delete(o); return this.size = d.size, v; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_stackGet.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_stackGet.js ***! \****************************************************/ /***/ (a) => { function f(o) { return this.__data__.get(o); } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_stackHas.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_stackHas.js ***! \****************************************************/ /***/ (a) => { function f(o) { return this.__data__.has(o); } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_stackSet.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_stackSet.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_ListCache */ "../../../../node_modules/lodash/_ListCache.js" ), v = o( /*! ./_Map */ "../../../../node_modules/lodash/_Map.js" ), u = o( /*! ./_MapCache */ "../../../../node_modules/lodash/_MapCache.js" ), l = 200; function P(p, c) { var H = this.__data__; if (H instanceof d) { var T = H.__data__; if (!v || T.length < l - 1) return T.push([p, c]), this.size = ++H.size, this; H = this.__data__ = new u(T); } return H.set(p, c), this.size = H.size, this; } a.exports = P; } ), /***/ "../../../../node_modules/lodash/_strictIndexOf.js": ( /*!*********************************************************!*\ !*** ../../../../node_modules/lodash/_strictIndexOf.js ***! \*********************************************************/ /***/ (a) => { function f(o, d, v) { for (var u = v - 1, l = o.length; ++u < l; ) if (o[u] === d) return u; return -1; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/_stringSize.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/_stringSize.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_asciiSize */ "../../../../node_modules/lodash/_asciiSize.js" ), v = o( /*! ./_hasUnicode */ "../../../../node_modules/lodash/_hasUnicode.js" ), u = o( /*! ./_unicodeSize */ "../../../../node_modules/lodash/_unicodeSize.js" ); function l(P) { return v(P) ? u(P) : d(P); } a.exports = l; } ), /***/ "../../../../node_modules/lodash/_stringToPath.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/_stringToPath.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_memoizeCapped */ "../../../../node_modules/lodash/_memoizeCapped.js" ), v = /[^.[\]]+|\[(?:(-?\d+(?:\.\d+)?)|(["'])((?:(?!\2)[^\\]|\\.)*?)\2)\]|(?=(?:\.|\[\])(?:\.|\[\]|$))/g, u = /\\(\\)?/g, l = d(function(P) { var p = []; return P.charCodeAt(0) === 46 && p.push(""), P.replace(v, function(c, H, T, q) { p.push(T ? q.replace(u, "$1") : H || c); }), p; }); a.exports = l; } ), /***/ "../../../../node_modules/lodash/_toKey.js": ( /*!*************************************************!*\ !*** ../../../../node_modules/lodash/_toKey.js ***! \*************************************************/ /***/ (a, f, o) => { var d = o( /*! ./isSymbol */ "../../../../node_modules/lodash/isSymbol.js" ), v = 1 / 0; function u(l) { if (typeof l == "string" || d(l)) return l; var P = l + ""; return P == "0" && 1 / l == -v ? "-0" : P; } a.exports = u; } ), /***/ "../../../../node_modules/lodash/_toSource.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/_toSource.js ***! \****************************************************/ /***/ (a) => { var f = Function.prototype, o = f.toString; function d(v) { if (v != null) { try { return o.call(v); } catch { } try { return v + ""; } catch { } } return ""; } a.exports = d; } ), /***/ "../../../../node_modules/lodash/_trimmedEndIndex.js": ( /*!***********************************************************!*\ !*** ../../../../node_modules/lodash/_trimmedEndIndex.js ***! \***********************************************************/ /***/ (a) => { var f = /\s/; function o(d) { for (var v = d.length; v-- && f.test(d.charAt(v)); ) ; return v; } a.exports = o; } ), /***/ "../../../../node_modules/lodash/_unicodeSize.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/_unicodeSize.js ***! \*******************************************************/ /***/ (a) => { var f = "\\ud800-\\udfff", o = "\\u0300-\\u036f", d = "\\ufe20-\\ufe2f", v = "\\u20d0-\\u20ff", u = o + d + v, l = "\\ufe0e\\ufe0f", P = "[" + f + "]", p = "[" + u + "]", c = "\\ud83c[\\udffb-\\udfff]", H = "(?:" + p + "|" + c + ")", T = "[^" + f + "]", q = "(?:\\ud83c[\\udde6-\\uddff]){2}", b = "[\\ud800-\\udbff][\\udc00-\\udfff]", j = "\\u200d", w = H + "?", m = "[" + l + "]?", I = "(?:" + j + "(?:" + [T, q, b].join("|") + ")" + m + w + ")*", N = m + w + I, k = "(?:" + [T + p + "?", p, q, b, P].join("|") + ")", R = RegExp(c + "(?=" + c + ")|" + k + N, "g"); function y(O) { for (var Y = R.lastIndex = 0; R.test(O); ) ++Y; return Y; } a.exports = y; } ), /***/ "../../../../node_modules/lodash/clone.js": ( /*!************************************************!*\ !*** ../../../../node_modules/lodash/clone.js ***! \************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseClone */ "../../../../node_modules/lodash/_baseClone.js" ), v = 4; function u(l) { return d(l, v); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/cloneDeep.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/cloneDeep.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseClone */ "../../../../node_modules/lodash/_baseClone.js" ), v = 1, u = 4; function l(P) { return d(P, v | u); } a.exports = l; } ), /***/ "../../../../node_modules/lodash/constant.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/constant.js ***! \***************************************************/ /***/ (a) => { function f(o) { return function() { return o; }; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/defaults.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/defaults.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseRest */ "../../../../node_modules/lodash/_baseRest.js" ), v = o( /*! ./eq */ "../../../../node_modules/lodash/eq.js" ), u = o( /*! ./_isIterateeCall */ "../../../../node_modules/lodash/_isIterateeCall.js" ), l = o( /*! ./keysIn */ "../../../../node_modules/lodash/keysIn.js" ), P = Object.prototype, p = P.hasOwnProperty, c = d(function(H, T) { H = Object(H); var q = -1, b = T.length, j = b > 2 ? T[2] : void 0; for (j && u(T[0], T[1], j) && (b = 1); ++q < b; ) for (var w = T[q], m = l(w), I = -1, N = m.length; ++I < N; ) { var k = m[I], R = H[k]; (R === void 0 || v(R, P[k]) && !p.call(H, k)) && (H[k] = w[k]); } return H; }); a.exports = c; } ), /***/ "../../../../node_modules/lodash/each.js": ( /*!***********************************************!*\ !*** ../../../../node_modules/lodash/each.js ***! \***********************************************/ /***/ (a, f, o) => { a.exports = o( /*! ./forEach */ "../../../../node_modules/lodash/forEach.js" ); } ), /***/ "../../../../node_modules/lodash/eq.js": ( /*!*********************************************!*\ !*** ../../../../node_modules/lodash/eq.js ***! \*********************************************/ /***/ (a) => { function f(o, d) { return o === d || o !== o && d !== d; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/filter.js": ( /*!*************************************************!*\ !*** ../../../../node_modules/lodash/filter.js ***! \*************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_arrayFilter */ "../../../../node_modules/lodash/_arrayFilter.js" ), v = o( /*! ./_baseFilter */ "../../../../node_modules/lodash/_baseFilter.js" ), u = o( /*! ./_baseIteratee */ "../../../../node_modules/lodash/_baseIteratee.js" ), l = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ); function P(p, c) { var H = l(p) ? d : v; return H(p, u(c, 3)); } a.exports = P; } ), /***/ "../../../../node_modules/lodash/find.js": ( /*!***********************************************!*\ !*** ../../../../node_modules/lodash/find.js ***! \***********************************************/ /***/ (a, f, o) => { var d = o( /*! ./_createFind */ "../../../../node_modules/lodash/_createFind.js" ), v = o( /*! ./findIndex */ "../../../../node_modules/lodash/findIndex.js" ), u = d(v); a.exports = u; } ), /***/ "../../../../node_modules/lodash/findIndex.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/findIndex.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseFindIndex */ "../../../../node_modules/lodash/_baseFindIndex.js" ), v = o( /*! ./_baseIteratee */ "../../../../node_modules/lodash/_baseIteratee.js" ), u = o( /*! ./toInteger */ "../../../../node_modules/lodash/toInteger.js" ), l = Math.max; function P(p, c, H) { var T = p == null ? 0 : p.length; if (!T) return -1; var q = H == null ? 0 : u(H); return q < 0 && (q = l(T + q, 0)), d(p, v(c, 3), q); } a.exports = P; } ), /***/ "../../../../node_modules/lodash/flatten.js": ( /*!**************************************************!*\ !*** ../../../../node_modules/lodash/flatten.js ***! \**************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseFlatten */ "../../../../node_modules/lodash/_baseFlatten.js" ); function v(u) { var l = u == null ? 0 : u.length; return l ? d(u, 1) : []; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/forEach.js": ( /*!**************************************************!*\ !*** ../../../../node_modules/lodash/forEach.js ***! \**************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_arrayEach */ "../../../../node_modules/lodash/_arrayEach.js" ), v = o( /*! ./_baseEach */ "../../../../node_modules/lodash/_baseEach.js" ), u = o( /*! ./_castFunction */ "../../../../node_modules/lodash/_castFunction.js" ), l = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ); function P(p, c) { var H = l(p) ? d : v; return H(p, u(c)); } a.exports = P; } ), /***/ "../../../../node_modules/lodash/forIn.js": ( /*!************************************************!*\ !*** ../../../../node_modules/lodash/forIn.js ***! \************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseFor */ "../../../../node_modules/lodash/_baseFor.js" ), v = o( /*! ./_castFunction */ "../../../../node_modules/lodash/_castFunction.js" ), u = o( /*! ./keysIn */ "../../../../node_modules/lodash/keysIn.js" ); function l(P, p) { return P == null ? P : d(P, v(p), u); } a.exports = l; } ), /***/ "../../../../node_modules/lodash/get.js": ( /*!**********************************************!*\ !*** ../../../../node_modules/lodash/get.js ***! \**********************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseGet */ "../../../../node_modules/lodash/_baseGet.js" ); function v(u, l, P) { var p = u == null ? void 0 : d(u, l); return p === void 0 ? P : p; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/has.js": ( /*!**********************************************!*\ !*** ../../../../node_modules/lodash/has.js ***! \**********************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseHas */ "../../../../node_modules/lodash/_baseHas.js" ), v = o( /*! ./_hasPath */ "../../../../node_modules/lodash/_hasPath.js" ); function u(l, P) { return l != null && v(l, P, d); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/hasIn.js": ( /*!************************************************!*\ !*** ../../../../node_modules/lodash/hasIn.js ***! \************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseHasIn */ "../../../../node_modules/lodash/_baseHasIn.js" ), v = o( /*! ./_hasPath */ "../../../../node_modules/lodash/_hasPath.js" ); function u(l, P) { return l != null && v(l, P, d); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/identity.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/identity.js ***! \***************************************************/ /***/ (a) => { function f(o) { return o; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/isArguments.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/isArguments.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseIsArguments */ "../../../../node_modules/lodash/_baseIsArguments.js" ), v = o( /*! ./isObjectLike */ "../../../../node_modules/lodash/isObjectLike.js" ), u = Object.prototype, l = u.hasOwnProperty, P = u.propertyIsEnumerable, p = d(/* @__PURE__ */ function() { return arguments; }()) ? d : function(c) { return v(c) && l.call(c, "callee") && !P.call(c, "callee"); }; a.exports = p; } ), /***/ "../../../../node_modules/lodash/isArray.js": ( /*!**************************************************!*\ !*** ../../../../node_modules/lodash/isArray.js ***! \**************************************************/ /***/ (a) => { var f = Array.isArray; a.exports = f; } ), /***/ "../../../../node_modules/lodash/isArrayLike.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/isArrayLike.js ***! \******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./isFunction */ "../../../../node_modules/lodash/isFunction.js" ), v = o( /*! ./isLength */ "../../../../node_modules/lodash/isLength.js" ); function u(l) { return l != null && v(l.length) && !d(l); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/isArrayLikeObject.js": ( /*!************************************************************!*\ !*** ../../../../node_modules/lodash/isArrayLikeObject.js ***! \************************************************************/ /***/ (a, f, o) => { var d = o( /*! ./isArrayLike */ "../../../../node_modules/lodash/isArrayLike.js" ), v = o( /*! ./isObjectLike */ "../../../../node_modules/lodash/isObjectLike.js" ); function u(l) { return v(l) && d(l); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/isBuffer.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/isBuffer.js ***! \***************************************************/ /***/ (a, f, o) => { a = o.nmd(a); var d = o( /*! ./_root */ "../../../../node_modules/lodash/_root.js" ), v = o( /*! ./stubFalse */ "../../../../node_modules/lodash/stubFalse.js" ), u = f && !f.nodeType && f, l = u && !0 && a && !a.nodeType && a, P = l && l.exports === u, p = P ? d.Buffer : void 0, c = p ? p.isBuffer : void 0, H = c || v; a.exports = H; } ), /***/ "../../../../node_modules/lodash/isEmpty.js": ( /*!**************************************************!*\ !*** ../../../../node_modules/lodash/isEmpty.js ***! \**************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseKeys */ "../../../../node_modules/lodash/_baseKeys.js" ), v = o( /*! ./_getTag */ "../../../../node_modules/lodash/_getTag.js" ), u = o( /*! ./isArguments */ "../../../../node_modules/lodash/isArguments.js" ), l = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ), P = o( /*! ./isArrayLike */ "../../../../node_modules/lodash/isArrayLike.js" ), p = o( /*! ./isBuffer */ "../../../../node_modules/lodash/isBuffer.js" ), c = o( /*! ./_isPrototype */ "../../../../node_modules/lodash/_isPrototype.js" ), H = o( /*! ./isTypedArray */ "../../../../node_modules/lodash/isTypedArray.js" ), T = "[object Map]", q = "[object Set]", b = Object.prototype, j = b.hasOwnProperty; function w(m) { if (m == null) return !0; if (P(m) && (l(m) || typeof m == "string" || typeof m.splice == "function" || p(m) || H(m) || u(m))) return !m.length; var I = v(m); if (I == T || I == q) return !m.size; if (c(m)) return !d(m).length; for (var N in m) if (j.call(m, N)) return !1; return !0; } a.exports = w; } ), /***/ "../../../../node_modules/lodash/isFunction.js": ( /*!*****************************************************!*\ !*** ../../../../node_modules/lodash/isFunction.js ***! \*****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseGetTag */ "../../../../node_modules/lodash/_baseGetTag.js" ), v = o( /*! ./isObject */ "../../../../node_modules/lodash/isObject.js" ), u = "[object AsyncFunction]", l = "[object Function]", P = "[object GeneratorFunction]", p = "[object Proxy]"; function c(H) { if (!v(H)) return !1; var T = d(H); return T == l || T == P || T == u || T == p; } a.exports = c; } ), /***/ "../../../../node_modules/lodash/isLength.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/isLength.js ***! \***************************************************/ /***/ (a) => { var f = 9007199254740991; function o(d) { return typeof d == "number" && d > -1 && d % 1 == 0 && d <= f; } a.exports = o; } ), /***/ "../../../../node_modules/lodash/isMap.js": ( /*!************************************************!*\ !*** ../../../../node_modules/lodash/isMap.js ***! \************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseIsMap */ "../../../../node_modules/lodash/_baseIsMap.js" ), v = o( /*! ./_baseUnary */ "../../../../node_modules/lodash/_baseUnary.js" ), u = o( /*! ./_nodeUtil */ "../../../../node_modules/lodash/_nodeUtil.js" ), l = u && u.isMap, P = l ? v(l) : d; a.exports = P; } ), /***/ "../../../../node_modules/lodash/isObject.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/isObject.js ***! \***************************************************/ /***/ (a) => { function f(o) { var d = typeof o; return o != null && (d == "object" || d == "function"); } a.exports = f; } ), /***/ "../../../../node_modules/lodash/isObjectLike.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/isObjectLike.js ***! \*******************************************************/ /***/ (a) => { function f(o) { return o != null && typeof o == "object"; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/isPlainObject.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/isPlainObject.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseGetTag */ "../../../../node_modules/lodash/_baseGetTag.js" ), v = o( /*! ./_getPrototype */ "../../../../node_modules/lodash/_getPrototype.js" ), u = o( /*! ./isObjectLike */ "../../../../node_modules/lodash/isObjectLike.js" ), l = "[object Object]", P = Function.prototype, p = Object.prototype, c = P.toString, H = p.hasOwnProperty, T = c.call(Object); function q(b) { if (!u(b) || d(b) != l) return !1; var j = v(b); if (j === null) return !0; var w = H.call(j, "constructor") && j.constructor; return typeof w == "function" && w instanceof w && c.call(w) == T; } a.exports = q; } ), /***/ "../../../../node_modules/lodash/isSet.js": ( /*!************************************************!*\ !*** ../../../../node_modules/lodash/isSet.js ***! \************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseIsSet */ "../../../../node_modules/lodash/_baseIsSet.js" ), v = o( /*! ./_baseUnary */ "../../../../node_modules/lodash/_baseUnary.js" ), u = o( /*! ./_nodeUtil */ "../../../../node_modules/lodash/_nodeUtil.js" ), l = u && u.isSet, P = l ? v(l) : d; a.exports = P; } ), /***/ "../../../../node_modules/lodash/isString.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/isString.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseGetTag */ "../../../../node_modules/lodash/_baseGetTag.js" ), v = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ), u = o( /*! ./isObjectLike */ "../../../../node_modules/lodash/isObjectLike.js" ), l = "[object String]"; function P(p) { return typeof p == "string" || !v(p) && u(p) && d(p) == l; } a.exports = P; } ), /***/ "../../../../node_modules/lodash/isSymbol.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/isSymbol.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseGetTag */ "../../../../node_modules/lodash/_baseGetTag.js" ), v = o( /*! ./isObjectLike */ "../../../../node_modules/lodash/isObjectLike.js" ), u = "[object Symbol]"; function l(P) { return typeof P == "symbol" || v(P) && d(P) == u; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/isTypedArray.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/lodash/isTypedArray.js ***! \*******************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseIsTypedArray */ "../../../../node_modules/lodash/_baseIsTypedArray.js" ), v = o( /*! ./_baseUnary */ "../../../../node_modules/lodash/_baseUnary.js" ), u = o( /*! ./_nodeUtil */ "../../../../node_modules/lodash/_nodeUtil.js" ), l = u && u.isTypedArray, P = l ? v(l) : d; a.exports = P; } ), /***/ "../../../../node_modules/lodash/isUndefined.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/lodash/isUndefined.js ***! \******************************************************/ /***/ (a) => { function f(o) { return o === void 0; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/keys.js": ( /*!***********************************************!*\ !*** ../../../../node_modules/lodash/keys.js ***! \***********************************************/ /***/ (a, f, o) => { var d = o( /*! ./_arrayLikeKeys */ "../../../../node_modules/lodash/_arrayLikeKeys.js" ), v = o( /*! ./_baseKeys */ "../../../../node_modules/lodash/_baseKeys.js" ), u = o( /*! ./isArrayLike */ "../../../../node_modules/lodash/isArrayLike.js" ); function l(P) { return u(P) ? d(P) : v(P); } a.exports = l; } ), /***/ "../../../../node_modules/lodash/keysIn.js": ( /*!*************************************************!*\ !*** ../../../../node_modules/lodash/keysIn.js ***! \*************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_arrayLikeKeys */ "../../../../node_modules/lodash/_arrayLikeKeys.js" ), v = o( /*! ./_baseKeysIn */ "../../../../node_modules/lodash/_baseKeysIn.js" ), u = o( /*! ./isArrayLike */ "../../../../node_modules/lodash/isArrayLike.js" ); function l(P) { return u(P) ? d(P, !0) : v(P); } a.exports = l; } ), /***/ "../../../../node_modules/lodash/last.js": ( /*!***********************************************!*\ !*** ../../../../node_modules/lodash/last.js ***! \***********************************************/ /***/ (a) => { function f(o) { var d = o == null ? 0 : o.length; return d ? o[d - 1] : void 0; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/map.js": ( /*!**********************************************!*\ !*** ../../../../node_modules/lodash/map.js ***! \**********************************************/ /***/ (a, f, o) => { var d = o( /*! ./_arrayMap */ "../../../../node_modules/lodash/_arrayMap.js" ), v = o( /*! ./_baseIteratee */ "../../../../node_modules/lodash/_baseIteratee.js" ), u = o( /*! ./_baseMap */ "../../../../node_modules/lodash/_baseMap.js" ), l = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ); function P(p, c) { var H = l(p) ? d : u; return H(p, v(c, 3)); } a.exports = P; } ), /***/ "../../../../node_modules/lodash/mapValues.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/mapValues.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseAssignValue */ "../../../../node_modules/lodash/_baseAssignValue.js" ), v = o( /*! ./_baseForOwn */ "../../../../node_modules/lodash/_baseForOwn.js" ), u = o( /*! ./_baseIteratee */ "../../../../node_modules/lodash/_baseIteratee.js" ); function l(P, p) { var c = {}; return p = u(p, 3), v(P, function(H, T, q) { d(c, T, p(H, T, q)); }), c; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/max.js": ( /*!**********************************************!*\ !*** ../../../../node_modules/lodash/max.js ***! \**********************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseExtremum */ "../../../../node_modules/lodash/_baseExtremum.js" ), v = o( /*! ./_baseGt */ "../../../../node_modules/lodash/_baseGt.js" ), u = o( /*! ./identity */ "../../../../node_modules/lodash/identity.js" ); function l(P) { return P && P.length ? d(P, u, v) : void 0; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/memoize.js": ( /*!**************************************************!*\ !*** ../../../../node_modules/lodash/memoize.js ***! \**************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_MapCache */ "../../../../node_modules/lodash/_MapCache.js" ), v = "Expected a function"; function u(l, P) { if (typeof l != "function" || P != null && typeof P != "function") throw new TypeError(v); var p = function() { var c = arguments, H = P ? P.apply(this, c) : c[0], T = p.cache; if (T.has(H)) return T.get(H); var q = l.apply(this, c); return p.cache = T.set(H, q) || T, q; }; return p.cache = new (u.Cache || d)(), p; } u.Cache = d, a.exports = u; } ), /***/ "../../../../node_modules/lodash/merge.js": ( /*!************************************************!*\ !*** ../../../../node_modules/lodash/merge.js ***! \************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseMerge */ "../../../../node_modules/lodash/_baseMerge.js" ), v = o( /*! ./_createAssigner */ "../../../../node_modules/lodash/_createAssigner.js" ), u = v(function(l, P, p) { d(l, P, p); }); a.exports = u; } ), /***/ "../../../../node_modules/lodash/min.js": ( /*!**********************************************!*\ !*** ../../../../node_modules/lodash/min.js ***! \**********************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseExtremum */ "../../../../node_modules/lodash/_baseExtremum.js" ), v = o( /*! ./_baseLt */ "../../../../node_modules/lodash/_baseLt.js" ), u = o( /*! ./identity */ "../../../../node_modules/lodash/identity.js" ); function l(P) { return P && P.length ? d(P, u, v) : void 0; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/minBy.js": ( /*!************************************************!*\ !*** ../../../../node_modules/lodash/minBy.js ***! \************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseExtremum */ "../../../../node_modules/lodash/_baseExtremum.js" ), v = o( /*! ./_baseIteratee */ "../../../../node_modules/lodash/_baseIteratee.js" ), u = o( /*! ./_baseLt */ "../../../../node_modules/lodash/_baseLt.js" ); function l(P, p) { return P && P.length ? d(P, v(p, 2), u) : void 0; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/noop.js": ( /*!***********************************************!*\ !*** ../../../../node_modules/lodash/noop.js ***! \***********************************************/ /***/ (a) => { function f() { } a.exports = f; } ), /***/ "../../../../node_modules/lodash/now.js": ( /*!**********************************************!*\ !*** ../../../../node_modules/lodash/now.js ***! \**********************************************/ /***/ (a, f, o) => { var d = o( /*! ./_root */ "../../../../node_modules/lodash/_root.js" ), v = function() { return d.Date.now(); }; a.exports = v; } ), /***/ "../../../../node_modules/lodash/pick.js": ( /*!***********************************************!*\ !*** ../../../../node_modules/lodash/pick.js ***! \***********************************************/ /***/ (a, f, o) => { var d = o( /*! ./_basePick */ "../../../../node_modules/lodash/_basePick.js" ), v = o( /*! ./_flatRest */ "../../../../node_modules/lodash/_flatRest.js" ), u = v(function(l, P) { return l == null ? {} : d(l, P); }); a.exports = u; } ), /***/ "../../../../node_modules/lodash/property.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/property.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseProperty */ "../../../../node_modules/lodash/_baseProperty.js" ), v = o( /*! ./_basePropertyDeep */ "../../../../node_modules/lodash/_basePropertyDeep.js" ), u = o( /*! ./_isKey */ "../../../../node_modules/lodash/_isKey.js" ), l = o( /*! ./_toKey */ "../../../../node_modules/lodash/_toKey.js" ); function P(p) { return u(p) ? d(l(p)) : v(p); } a.exports = P; } ), /***/ "../../../../node_modules/lodash/range.js": ( /*!************************************************!*\ !*** ../../../../node_modules/lodash/range.js ***! \************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_createRange */ "../../../../node_modules/lodash/_createRange.js" ), v = d(); a.exports = v; } ), /***/ "../../../../node_modules/lodash/reduce.js": ( /*!*************************************************!*\ !*** ../../../../node_modules/lodash/reduce.js ***! \*************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_arrayReduce */ "../../../../node_modules/lodash/_arrayReduce.js" ), v = o( /*! ./_baseEach */ "../../../../node_modules/lodash/_baseEach.js" ), u = o( /*! ./_baseIteratee */ "../../../../node_modules/lodash/_baseIteratee.js" ), l = o( /*! ./_baseReduce */ "../../../../node_modules/lodash/_baseReduce.js" ), P = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ); function p(c, H, T) { var q = P(c) ? d : l, b = arguments.length < 3; return q(c, u(H, 4), T, b, v); } a.exports = p; } ), /***/ "../../../../node_modules/lodash/size.js": ( /*!***********************************************!*\ !*** ../../../../node_modules/lodash/size.js ***! \***********************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseKeys */ "../../../../node_modules/lodash/_baseKeys.js" ), v = o( /*! ./_getTag */ "../../../../node_modules/lodash/_getTag.js" ), u = o( /*! ./isArrayLike */ "../../../../node_modules/lodash/isArrayLike.js" ), l = o( /*! ./isString */ "../../../../node_modules/lodash/isString.js" ), P = o( /*! ./_stringSize */ "../../../../node_modules/lodash/_stringSize.js" ), p = "[object Map]", c = "[object Set]"; function H(T) { if (T == null) return 0; if (u(T)) return l(T) ? P(T) : T.length; var q = v(T); return q == p || q == c ? T.size : d(T).length; } a.exports = H; } ), /***/ "../../../../node_modules/lodash/sortBy.js": ( /*!*************************************************!*\ !*** ../../../../node_modules/lodash/sortBy.js ***! \*************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseFlatten */ "../../../../node_modules/lodash/_baseFlatten.js" ), v = o( /*! ./_baseOrderBy */ "../../../../node_modules/lodash/_baseOrderBy.js" ), u = o( /*! ./_baseRest */ "../../../../node_modules/lodash/_baseRest.js" ), l = o( /*! ./_isIterateeCall */ "../../../../node_modules/lodash/_isIterateeCall.js" ), P = u(function(p, c) { if (p == null) return []; var H = c.length; return H > 1 && l(p, c[0], c[1]) ? c = [] : H > 2 && l(c[0], c[1], c[2]) && (c = [c[0]]), v(p, d(c, 1), []); }); a.exports = P; } ), /***/ "../../../../node_modules/lodash/stubArray.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/stubArray.js ***! \****************************************************/ /***/ (a) => { function f() { return []; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/stubFalse.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/stubFalse.js ***! \****************************************************/ /***/ (a) => { function f() { return !1; } a.exports = f; } ), /***/ "../../../../node_modules/lodash/toFinite.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/toFinite.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./toNumber */ "../../../../node_modules/lodash/toNumber.js" ), v = 1 / 0, u = 17976931348623157e292; function l(P) { if (!P) return P === 0 ? P : 0; if (P = d(P), P === v || P === -v) { var p = P < 0 ? -1 : 1; return p * u; } return P === P ? P : 0; } a.exports = l; } ), /***/ "../../../../node_modules/lodash/toInteger.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/toInteger.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./toFinite */ "../../../../node_modules/lodash/toFinite.js" ); function v(u) { var l = d(u), P = l % 1; return l === l ? P ? l - P : l : 0; } a.exports = v; } ), /***/ "../../../../node_modules/lodash/toNumber.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/toNumber.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseTrim */ "../../../../node_modules/lodash/_baseTrim.js" ), v = o( /*! ./isObject */ "../../../../node_modules/lodash/isObject.js" ), u = o( /*! ./isSymbol */ "../../../../node_modules/lodash/isSymbol.js" ), l = NaN, P = /^[-+]0x[0-9a-f]+$/i, p = /^0b[01]+$/i, c = /^0o[0-7]+$/i, H = parseInt; function T(q) { if (typeof q == "number") return q; if (u(q)) return l; if (v(q)) { var b = typeof q.valueOf == "function" ? q.valueOf() : q; q = v(b) ? b + "" : b; } if (typeof q != "string") return q === 0 ? q : +q; q = d(q); var j = p.test(q); return j || c.test(q) ? H(q.slice(2), j ? 2 : 8) : P.test(q) ? l : +q; } a.exports = T; } ), /***/ "../../../../node_modules/lodash/toPlainObject.js": ( /*!********************************************************!*\ !*** ../../../../node_modules/lodash/toPlainObject.js ***! \********************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_copyObject */ "../../../../node_modules/lodash/_copyObject.js" ), v = o( /*! ./keysIn */ "../../../../node_modules/lodash/keysIn.js" ); function u(l) { return d(l, v(l)); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/toString.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/toString.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseToString */ "../../../../node_modules/lodash/_baseToString.js" ); function v(u) { return u == null ? "" : d(u); } a.exports = v; } ), /***/ "../../../../node_modules/lodash/transform.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/transform.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_arrayEach */ "../../../../node_modules/lodash/_arrayEach.js" ), v = o( /*! ./_baseCreate */ "../../../../node_modules/lodash/_baseCreate.js" ), u = o( /*! ./_baseForOwn */ "../../../../node_modules/lodash/_baseForOwn.js" ), l = o( /*! ./_baseIteratee */ "../../../../node_modules/lodash/_baseIteratee.js" ), P = o( /*! ./_getPrototype */ "../../../../node_modules/lodash/_getPrototype.js" ), p = o( /*! ./isArray */ "../../../../node_modules/lodash/isArray.js" ), c = o( /*! ./isBuffer */ "../../../../node_modules/lodash/isBuffer.js" ), H = o( /*! ./isFunction */ "../../../../node_modules/lodash/isFunction.js" ), T = o( /*! ./isObject */ "../../../../node_modules/lodash/isObject.js" ), q = o( /*! ./isTypedArray */ "../../../../node_modules/lodash/isTypedArray.js" ); function b(j, w, m) { var I = p(j), N = I || c(j) || q(j); if (w = l(w, 4), m == null) { var k = j && j.constructor; N ? m = I ? new k() : [] : T(j) ? m = H(k) ? v(P(j)) : {} : m = {}; } return (N ? d : u)(j, function(R, y, O) { return w(m, R, y, O); }), m; } a.exports = b; } ), /***/ "../../../../node_modules/lodash/union.js": ( /*!************************************************!*\ !*** ../../../../node_modules/lodash/union.js ***! \************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseFlatten */ "../../../../node_modules/lodash/_baseFlatten.js" ), v = o( /*! ./_baseRest */ "../../../../node_modules/lodash/_baseRest.js" ), u = o( /*! ./_baseUniq */ "../../../../node_modules/lodash/_baseUniq.js" ), l = o( /*! ./isArrayLikeObject */ "../../../../node_modules/lodash/isArrayLikeObject.js" ), P = v(function(p) { return u(d(p, 1, l, !0)); }); a.exports = P; } ), /***/ "../../../../node_modules/lodash/uniqueId.js": ( /*!***************************************************!*\ !*** ../../../../node_modules/lodash/uniqueId.js ***! \***************************************************/ /***/ (a, f, o) => { var d = o( /*! ./toString */ "../../../../node_modules/lodash/toString.js" ), v = 0; function u(l) { var P = ++v; return d(l) + P; } a.exports = u; } ), /***/ "../../../../node_modules/lodash/values.js": ( /*!*************************************************!*\ !*** ../../../../node_modules/lodash/values.js ***! \*************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_baseValues */ "../../../../node_modules/lodash/_baseValues.js" ), v = o( /*! ./keys */ "../../../../node_modules/lodash/keys.js" ); function u(l) { return l == null ? [] : d(l, v(l)); } a.exports = u; } ), /***/ "../../../../node_modules/lodash/zipObject.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/lodash/zipObject.js ***! \****************************************************/ /***/ (a, f, o) => { var d = o( /*! ./_assignValue */ "../../../../node_modules/lodash/_assignValue.js" ), v = o( /*! ./_baseZipObject */ "../../../../node_modules/lodash/_baseZipObject.js" ); function u(l, P) { return v(l || [], P || [], d); } a.exports = u; } ), /***/ "../../../../node_modules/object-assign/index.js": ( /*!*******************************************************!*\ !*** ../../../../node_modules/object-assign/index.js ***! \*******************************************************/ /***/ (a) => { /* object-assign (c) Sindre Sorhus @license MIT */ var f = Object.getOwnPropertySymbols, o = Object.prototype.hasOwnProperty, d = Object.prototype.propertyIsEnumerable; function v(l) { if (l == null) throw new TypeError("Object.assign cannot be called with null or undefined"); return Object(l); } function u() { try { if (!Object.assign) return !1; var l = new String("abc"); if (l[5] = "de", Object.getOwnPropertyNames(l)[0] === "5") return !1; for (var P = {}, p = 0; p < 10; p++) P["_" + String.fromCharCode(p)] = p; var c = Object.getOwnPropertyNames(P).map(function(T) { return P[T]; }); if (c.join("") !== "0123456789") return !1; var H = {}; return "abcdefghijklmnopqrst".split("").forEach(function(T) { H[T] = T; }), Object.keys(Object.assign({}, H)).join("") === "abcdefghijklmnopqrst"; } catch { return !1; } } a.exports = u() ? Object.assign : function(l, P) { for (var p, c = v(l), H, T = 1; T < arguments.length; T++) { p = Object(arguments[T]); for (var q in p) o.call(p, q) && (c[q] = p[q]); if (f) { H = f(p); for (var b = 0; b < H.length; b++) d.call(p, H[b]) && (c[H[b]] = p[H[b]]); } } return c; }; } ), /***/ "../../../../node_modules/prop-types/checkPropTypes.js": ( /*!*************************************************************!*\ !*** ../../../../node_modules/prop-types/checkPropTypes.js ***! \*************************************************************/ /***/ (a, f, o) => { var d = function() { }; { var v = o( /*! ./lib/ReactPropTypesSecret */ "../../../../node_modules/prop-types/lib/ReactPropTypesSecret.js" ), u = {}, l = o( /*! ./lib/has */ "../../../../node_modules/prop-types/lib/has.js" ); d = function(p) { var c = "Warning: " + p; typeof console < "u" && console.error(c); try { throw new Error(c); } catch { } }; } function P(p, c, H, T, q) { for (var b in p) if (l(p, b)) { var j; try { if (typeof p[b] != "function") { var w = Error( (T || "React class") + ": " + H + " type `" + b + "` is invalid; it must be a function, usually from the `prop-types` package, but received `" + typeof p[b] + "`.This often happens because of typos such as `PropTypes.function` instead of `PropTypes.func`." ); throw w.name = "Invariant Violation", w; } j = p[b](c, b, T, H, null, v); } catch (I) { j = I; } if (j && !(j instanceof Error) && d( (T || "React class") + ": type specification of " + H + " `" + b + "` is invalid; the type checker function must return `null` or an `Error` but returned a " + typeof j + ". You may have forgotten to pass an argument to the type checker creator (arrayOf, instanceOf, objectOf, oneOf, oneOfType, and shape all require an argument)." ), j instanceof Error && !(j.message in u)) { u[j.message] = !0; var m = q ? q() : ""; d( "Failed " + H + " type: " + j.message + (m ?? "") ); } } } P.resetWarningCache = function() { u = {}; }, a.exports = P; } ), /***/ "../../../../node_modules/prop-types/factoryWithTypeCheckers.js": ( /*!**********************************************************************!*\ !*** ../../../../node_modules/prop-types/factoryWithTypeCheckers.js ***! \**********************************************************************/ /***/ (a, f, o) => { var d = o( /*! react-is */ "../../../../node_modules/prop-types/node_modules/react-is/index.js" ), v = o( /*! object-assign */ "../../../../node_modules/object-assign/index.js" ), u = o( /*! ./lib/ReactPropTypesSecret */ "../../../../node_modules/prop-types/lib/ReactPropTypesSecret.js" ), l = o( /*! ./lib/has */ "../../../../node_modules/prop-types/lib/has.js" ), P = o( /*! ./checkPropTypes */ "../../../../node_modules/prop-types/checkPropTypes.js" ), p = function() { }; p = function(H) { var T = "Warning: " + H; typeof console < "u" && console.error(T); try { throw new Error(T); } catch { } }; function c() { return null; } a.exports = function(H, T) { var q = typeof Symbol == "function" && Symbol.iterator, b = "@@iterator"; function j(re) { var ve = re && (q && re[q] || re[b]); if (typeof ve == "function") return ve; } var w = "<>", m = { array: R("array"), bigint: R("bigint"), bool: R("boolean"), func: R("function"), number: R("number"), object: R("object"), string: R("string"), symbol: R("symbol"), any: y(), arrayOf: O, element: Y(), elementType: ee(), instanceOf: Z, node: G(), objectOf: fe, oneOf: te, oneOfType: _, shape: $, exact: ae }; function I(re, ve) { return re === ve ? re !== 0 || 1 / re === 1 / ve : re !== re && ve !== ve; } function N(re, ve) { this.message = re, this.data = ve && typeof ve == "object" ? ve : {}, this.stack = ""; } N.prototype = Error.prototype; function k(re) { var ve = {}, qe = 0; function ke(Fe, Ke, nt, ut, bt, wt, Tt) { if (ut = ut || w, wt = wt || nt, Tt !== u) { if (T) { var lr = new Error( "Calling PropTypes validators directly is not supported by the `prop-types` package. Use `PropTypes.checkPropTypes()` to call them. Read more at http://fb.me/use-check-prop-types" ); throw lr.name = "Invariant Violation", lr; } else if (typeof console < "u") { var Qt = ut + ":" + nt; !ve[Qt] && // Avoid spamming the console because they are often not actionable except for lib authors qe < 3 && (p( "You are manually calling a React.PropTypes validation function for the `" + wt + "` prop on `" + ut + "`. This is deprecated and will throw in the standalone `prop-types` package. You may be seeing this warning due to a third-party PropTypes library. See https://fb.me/react-warning-dont-call-proptypes for details." ), ve[Qt] = !0, qe++); } } return Ke[nt] == null ? Fe ? Ke[nt] === null ? new N("The " + bt + " `" + wt + "` is marked as required " + ("in `" + ut + "`, but its value is `null`.")) : new N("The " + bt + " `" + wt + "` is marked as required in " + ("`" + ut + "`, but its value is `undefined`.")) : null : re(Ke, nt, ut, bt, wt); } var be = ke.bind(null, !1); return be.isRequired = ke.bind(null, !0), be; } function R(re) { function ve(qe, ke, be, Fe, Ke, nt) { var ut = qe[ke], bt = me(ut); if (bt !== re) { var wt = Xe(ut); return new N( "Invalid " + Fe + " `" + Ke + "` of type " + ("`" + wt + "` supplied to `" + be + "`, expected ") + ("`" + re + "`."), { expectedType: re } ); } return null; } return k(ve); } function y() { return k(c); } function O(re) { function ve(qe, ke, be, Fe, Ke) { if (typeof re != "function") return new N("Property `" + Ke + "` of component `" + be + "` has invalid PropType notation inside arrayOf."); var nt = qe[ke]; if (!Array.isArray(nt)) { var ut = me(nt); return new N("Invalid " + Fe + " `" + Ke + "` of type " + ("`" + ut + "` supplied to `" + be + "`, expected an array.")); } for (var bt = 0; bt < nt.length; bt++) { var wt = re(nt, bt, be, Fe, Ke + "[" + bt + "]", u); if (wt instanceof Error) return wt; } return null; } return k(ve); } function Y() { function re(ve, qe, ke, be, Fe) { var Ke = ve[qe]; if (!H(Ke)) { var nt = me(Ke); return new N("Invalid " + be + " `" + Fe + "` of type " + ("`" + nt + "` supplied to `" + ke + "`, expected a single ReactElement.")); } return null; } return k(re); } function ee() { function re(ve, qe, ke, be, Fe) { var Ke = ve[qe]; if (!d.isValidElementType(Ke)) { var nt = me(Ke); return new N("Invalid " + be + " `" + Fe + "` of type " + ("`" + nt + "` supplied to `" + ke + "`, expected a single ReactElement type.")); } return null; } return k(re); } function Z(re) { function ve(qe, ke, be, Fe, Ke) { if (!(qe[ke] instanceof re)) { var nt = re.name || w, ut = ne(qe[ke]); return new N("Invalid " + Fe + " `" + Ke + "` of type " + ("`" + ut + "` supplied to `" + be + "`, expected ") + ("instance of `" + nt + "`.")); } return null; } return k(ve); } function te(re) { if (!Array.isArray(re)) return arguments.length > 1 ? p( "Invalid arguments supplied to oneOf, expected an array, got " + arguments.length + " arguments. A common mistake is to write oneOf(x, y, z) instead of oneOf([x, y, z])." ) : p("Invalid argument supplied to oneOf, expected an array."), c; function ve(qe, ke, be, Fe, Ke) { for (var nt = qe[ke], ut = 0; ut < re.length; ut++) if (I(nt, re[ut])) return null; var bt = JSON.stringify(re, function(Tt, lr) { var Qt = Xe(lr); return Qt === "symbol" ? String(lr) : lr; }); return new N("Invalid " + Fe + " `" + Ke + "` of value `" + String(nt) + "` " + ("supplied to `" + be + "`, expected one of " + bt + ".")); } return k(ve); } function fe(re) { function ve(qe, ke, be, Fe, Ke) { if (typeof re != "function") return new N("Property `" + Ke + "` of component `" + be + "` has invalid PropType notation inside objectOf."); var nt = qe[ke], ut = me(nt); if (ut !== "object") return new N("Invalid " + Fe + " `" + Ke + "` of type " + ("`" + ut + "` supplied to `" + be + "`, expected an object.")); for (var bt in nt) if (l(nt, bt)) { var wt = re(nt, bt, be, Fe, Ke + "." + bt, u); if (wt instanceof Error) return wt; } return null; } return k(ve); } function _(re) { if (!Array.isArray(re)) return p("Invalid argument supplied to oneOfType, expected an instance of array."), c; for (var ve = 0; ve < re.length; ve++) { var qe = re[ve]; if (typeof qe != "function") return p( "Invalid argument supplied to oneOfType. Expected an array of check functions, but received " + De(qe) + " at index " + ve + "." ), c; } function ke(be, Fe, Ke, nt, ut) { for (var bt = [], wt = 0; wt < re.length; wt++) { var Tt = re[wt], lr = Tt(be, Fe, Ke, nt, ut, u); if (lr == null) return null; lr.data && l(lr.data, "expectedType") && bt.push(lr.data.expectedType); } var Qt = bt.length > 0 ? ", expected one of type [" + bt.join(", ") + "]" : ""; return new N("Invalid " + nt + " `" + ut + "` supplied to " + ("`" + Ke + "`" + Qt + ".")); } return k(ke); } function G() { function re(ve, qe, ke, be, Fe) { return Pe(ve[qe]) ? null : new N("Invalid " + be + " `" + Fe + "` supplied to " + ("`" + ke + "`, expected a ReactNode.")); } return k(re); } function L(re, ve, qe, ke, be) { return new N( (re || "React class") + ": " + ve + " type `" + qe + "." + ke + "` is invalid; it must be a function, usually from the `prop-types` package, but received `" + be + "`." ); } function $(re) { function ve(qe, ke, be, Fe, Ke) { var nt = qe[ke], ut = me(nt); if (ut !== "object") return new N("Invalid " + Fe + " `" + Ke + "` of type `" + ut + "` " + ("supplied to `" + be + "`, expected `object`.")); for (var bt in re) { var wt = re[bt]; if (typeof wt != "function") return L(be, Fe, Ke, bt, Xe(wt)); var Tt = wt(nt, bt, be, Fe, Ke + "." + bt, u); if (Tt) return Tt; } return null; } return k(ve); } function ae(re) { function ve(qe, ke, be, Fe, Ke) { var nt = qe[ke], ut = me(nt); if (ut !== "object") return new N("Invalid " + Fe + " `" + Ke + "` of type `" + ut + "` " + ("supplied to `" + be + "`, expected `object`.")); var bt = v({}, qe[ke], re); for (var wt in bt) { var Tt = re[wt]; if (l(re, wt) && typeof Tt != "function") return L(be, Fe, Ke, wt, Xe(Tt)); if (!Tt) return new N( "Invalid " + Fe + " `" + Ke + "` key `" + wt + "` supplied to `" + be + "`.\nBad object: " + JSON.stringify(qe[ke], null, " ") + ` Valid keys: ` + JSON.stringify(Object.keys(re), null, " ") ); var lr = Tt(nt, wt, be, Fe, Ke + "." + wt, u); if (lr) return lr; } return null; } return k(ve); } function Pe(re) { switch (typeof re) { case "number": case "string": case "undefined": return !0; case "boolean": return !re; case "object": if (Array.isArray(re)) return re.every(Pe); if (re === null || H(re)) return !0; var ve = j(re); if (ve) { var qe = ve.call(re), ke; if (ve !== re.entries) { for (; !(ke = qe.next()).done; ) if (!Pe(ke.value)) return !1; } else for (; !(ke = qe.next()).done; ) { var be = ke.value; if (be && !Pe(be[1])) return !1; } } else return !1; return !0; default: return !1; } } function ge(re, ve) { return re === "symbol" ? !0 : ve ? ve["@@toStringTag"] === "Symbol" || typeof Symbol == "function" && ve instanceof Symbol : !1; } function me(re) { var ve = typeof re; return Array.isArray(re) ? "array" : re instanceof RegExp ? "object" : ge(ve, re) ? "symbol" : ve; } function Xe(re) { if (typeof re > "u" || re === null) return "" + re; var ve = me(re); if (ve === "object") { if (re instanceof Date) return "date"; if (re instanceof RegExp) return "regexp"; } return ve; } function De(re) { var ve = Xe(re); switch (ve) { case "array": case "object": return "an " + ve; case "boolean": case "date": case "regexp": return "a " + ve; default: return ve; } } function ne(re) { return !re.constructor || !re.constructor.name ? w : re.constructor.name; } return m.checkPropTypes = P, m.resetWarningCache = P.resetWarningCache, m.PropTypes = m, m; }; } ), /***/ "../../../../node_modules/prop-types/index.js": ( /*!****************************************************!*\ !*** ../../../../node_modules/prop-types/index.js ***! \****************************************************/ /***/ (a, f, o) => { { var d = o( /*! react-is */ "../../../../node_modules/prop-types/node_modules/react-is/index.js" ), v = !0; a.exports = o( /*! ./factoryWithTypeCheckers */ "../../../../node_modules/prop-types/factoryWithTypeCheckers.js" )(d.isElement, v); } } ), /***/ "../../../../node_modules/prop-types/lib/ReactPropTypesSecret.js": ( /*!***********************************************************************!*\ !*** ../../../../node_modules/prop-types/lib/ReactPropTypesSecret.js ***! \***********************************************************************/ /***/ (a) => { var f = "SECRET_DO_NOT_PASS_THIS_OR_YOU_WILL_BE_FIRED"; a.exports = f; } ), /***/ "../../../../node_modules/prop-types/lib/has.js": ( /*!******************************************************!*\ !*** ../../../../node_modules/prop-types/lib/has.js ***! \******************************************************/ /***/ (a) => { a.exports = Function.call.bind(Object.prototype.hasOwnProperty); } ), /***/ "../../../../node_modules/prop-types/node_modules/react-is/cjs/react-is.development.js": ( /*!*********************************************************************************************!*\ !*** ../../../../node_modules/prop-types/node_modules/react-is/cjs/react-is.development.js ***! \*********************************************************************************************/ /***/ (a, f) => { /** @license React v16.13.1 * react-is.development.js * * Copyright (c) Facebook, Inc. and its affiliates. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. */ (function() { var o = typeof Symbol == "function" && Symbol.for, d = o ? Symbol.for("react.element") : 60103, v = o ? Symbol.for("react.portal") : 60106, u = o ? Symbol.for("react.fragment") : 60107, l = o ? Symbol.for("react.strict_mode") : 60108, P = o ? Symbol.for("react.profiler") : 60114, p = o ? Symbol.for("react.provider") : 60109, c = o ? Symbol.for("react.context") : 60110, H = o ? Symbol.for("react.async_mode") : 60111, T = o ? Symbol.for("react.concurrent_mode") : 60111, q = o ? Symbol.for("react.forward_ref") : 60112, b = o ? Symbol.for("react.suspense") : 60113, j = o ? Symbol.for("react.suspense_list") : 60120, w = o ? Symbol.for("react.memo") : 60115, m = o ? Symbol.for("react.lazy") : 60116, I = o ? Symbol.for("react.block") : 60121, N = o ? Symbol.for("react.fundamental") : 60117, k = o ? Symbol.for("react.responder") : 60118, R = o ? Symbol.for("react.scope") : 60119; function y(Tt) { return typeof Tt == "string" || typeof Tt == "function" || // Note: its typeof might be other than 'symbol' or 'number' if it's a polyfill. Tt === u || Tt === T || Tt === P || Tt === l || Tt === b || Tt === j || typeof Tt == "object" && Tt !== null && (Tt.$$typeof === m || Tt.$$typeof === w || Tt.$$typeof === p || Tt.$$typeof === c || Tt.$$typeof === q || Tt.$$typeof === N || Tt.$$typeof === k || Tt.$$typeof === R || Tt.$$typeof === I); } function O(Tt) { if (typeof Tt == "object" && Tt !== null) { var lr = Tt.$$typeof; switch (lr) { case d: var Qt = Tt.type; switch (Qt) { case H: case T: case u: case P: case l: case b: return Qt; default: var tr = Qt && Qt.$$typeof; switch (tr) { case c: case q: case m: case w: case p: return tr; default: return lr; } } case v: return lr; } } } var Y = H, ee = T, Z = c, te = p, fe = d, _ = q, G = u, L = m, $ = w, ae = v, Pe = P, ge = l, me = b, Xe = !1; function De(Tt) { return Xe || (Xe = !0, console.warn("The ReactIs.isAsyncMode() alias has been deprecated, and will be removed in React 17+. Update your code to use ReactIs.isConcurrentMode() instead. It has the exact same API.")), ne(Tt) || O(Tt) === H; } function ne(Tt) { return O(Tt) === T; } function re(Tt) { return O(Tt) === c; } function ve(Tt) { return O(Tt) === p; } function qe(Tt) { return typeof Tt == "object" && Tt !== null && Tt.$$typeof === d; } function ke(Tt) { return O(Tt) === q; } function be(Tt) { return O(Tt) === u; } function Fe(Tt) { return O(Tt) === m; } function Ke(Tt) { return O(Tt) === w; } function nt(Tt) { return O(Tt) === v; } function ut(Tt) { return O(Tt) === P; } function bt(Tt) { return O(Tt) === l; } function wt(Tt) { return O(Tt) === b; } f.AsyncMode = Y, f.ConcurrentMode = ee, f.ContextConsumer = Z, f.ContextProvider = te, f.Element = fe, f.ForwardRef = _, f.Fragment = G, f.Lazy = L, f.Memo = $, f.Portal = ae, f.Profiler = Pe, f.StrictMode = ge, f.Suspense = me, f.isAsyncMode = De, f.isConcurrentMode = ne, f.isContextConsumer = re, f.isContextProvider = ve, f.isElement = qe, f.isForwardRef = ke, f.isFragment = be, f.isLazy = Fe, f.isMemo = Ke, f.isPortal = nt, f.isProfiler = ut, f.isStrictMode = bt, f.isSuspense = wt, f.isValidElementType = y, f.typeOf = O; })(); } ), /***/ "../../../../node_modules/prop-types/node_modules/react-is/index.js": ( /*!**************************************************************************!*\ !*** ../../../../node_modules/prop-types/node_modules/react-is/index.js ***! \**************************************************************************/ /***/ (a, f, o) => { a.exports = o( /*! ./cjs/react-is.development.js */ "../../../../node_modules/prop-types/node_modules/react-is/cjs/react-is.development.js" ); } ), /***/ "../../../../node_modules/react-dom/cjs/react-dom.development.js": ( /*!***********************************************************************!*\ !*** ../../../../node_modules/react-dom/cjs/react-dom.development.js ***! \***********************************************************************/ /***/ (a, f, o) => { /** @license React v17.0.2 * react-dom.development.js * * Copyright (c) Facebook, Inc. and its affiliates. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. */ (function() { var d = o( /*! react */ "../../../../node_modules/react/index.js" ), v = o( /*! object-assign */ "../../../../node_modules/object-assign/index.js" ), u = o( /*! scheduler */ "../../../../node_modules/scheduler/index.js" ), l = o( /*! scheduler/tracing */ "../../../../node_modules/scheduler/tracing.js" ), P = d.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED; function p(g) { { for (var x = arguments.length, U = new Array(x > 1 ? x - 1 : 0), F = 1; F < x; F++) U[F - 1] = arguments[F]; H("warn", g, U); } } function c(g) { { for (var x = arguments.length, U = new Array(x > 1 ? x - 1 : 0), F = 1; F < x; F++) U[F - 1] = arguments[F]; H("error", g, U); } } function H(g, x, U) { { var F = P.ReactDebugCurrentFrame, K = F.getStackAddendum(); K !== "" && (x += "%s", U = U.concat([K])); var oe = U.map(function(ce) { return "" + ce; }); oe.unshift("Warning: " + x), Function.prototype.apply.call(console[g], console, oe); } } if (!d) throw Error("ReactDOM was loaded before React. Make sure you load the React package before loading ReactDOM."); var T = 0, q = 1, b = 2, j = 3, w = 4, m = 5, I = 6, N = 7, k = 8, R = 9, y = 10, O = 11, Y = 12, ee = 13, Z = 14, te = 15, fe = 16, _ = 17, G = 18, L = 19, $ = 20, ae = 21, Pe = 22, ge = 23, me = 24, Xe = !0, De = !1, ne = !1, re = !1, ve = /* @__PURE__ */ new Set(), qe = {}, ke = {}; function be(g, x) { Fe(g, x), Fe(g + "Capture", x); } function Fe(g, x) { qe[g] && c("EventRegistry: More than one plugin attempted to publish the same registration name, `%s`.", g), qe[g] = x; { var U = g.toLowerCase(); ke[U] = g, g === "onDoubleClick" && (ke.ondblclick = g); } for (var F = 0; F < x.length; F++) ve.add(x[F]); } var Ke = typeof window < "u" && typeof window.document < "u" && typeof window.document.createElement < "u", nt = 0, ut = 1, bt = 2, wt = 3, Tt = 4, lr = 5, Qt = 6, tr = ":A-Z_a-z\\u00C0-\\u00D6\\u00D8-\\u00F6\\u00F8-\\u02FF\\u0370-\\u037D\\u037F-\\u1FFF\\u200C-\\u200D\\u2070-\\u218F\\u2C00-\\u2FEF\\u3001-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFFD", br = tr + "\\-.0-9\\u00B7\\u0300-\\u036F\\u203F-\\u2040", Xn = "data-reactroot", qr = new RegExp("^[" + tr + "][" + br + "]*$"), En = Object.prototype.hasOwnProperty, Bn = {}, Fi = {}; function Gt(g) { return En.call(Fi, g) ? !0 : En.call(Bn, g) ? !1 : qr.test(g) ? (Fi[g] = !0, !0) : (Bn[g] = !0, c("Invalid attribute name: `%s`", g), !1); } function xr(g, x, U) { return x !== null ? x.type === nt : U ? !1 : g.length > 2 && (g[0] === "o" || g[0] === "O") && (g[1] === "n" || g[1] === "N"); } function Br(g, x, U, F) { if (U !== null && U.type === nt) return !1; switch (typeof x) { case "function": case "symbol": return !0; case "boolean": { if (F) return !1; if (U !== null) return !U.acceptsBooleans; var K = g.toLowerCase().slice(0, 5); return K !== "data-" && K !== "aria-"; } default: return !1; } } function nn(g, x, U, F) { if (x === null || typeof x > "u" || Br(g, x, U, F)) return !0; if (F) return !1; if (U !== null) switch (U.type) { case wt: return !x; case Tt: return x === !1; case lr: return isNaN(x); case Qt: return isNaN(x) || x < 1; } return !1; } function sn(g) { return an.hasOwnProperty(g) ? an[g] : null; } function Pn(g, x, U, F, K, oe, ce) { this.acceptsBooleans = x === bt || x === wt || x === Tt, this.attributeName = F, this.attributeNamespace = K, this.mustUseProperty = U, this.propertyName = g, this.type = x, this.sanitizeURL = oe, this.removeEmptyString = ce; } var an = {}, Wn = [ "children", "dangerouslySetInnerHTML", // TODO: This prevents the assignment of defaultValue to regular // elements (not just inputs). Now that ReactDOMInput assigns to the // defaultValue property -- do we need this? "defaultValue", "defaultChecked", "innerHTML", "suppressContentEditableWarning", "suppressHydrationWarning", "style" ]; Wn.forEach(function(g) { an[g] = new Pn( g, nt, !1, // mustUseProperty g, // attributeName null, // attributeNamespace !1, // sanitizeURL !1 ); }), [["acceptCharset", "accept-charset"], ["className", "class"], ["htmlFor", "for"], ["httpEquiv", "http-equiv"]].forEach(function(g) { var x = g[0], U = g[1]; an[x] = new Pn( x, ut, !1, // mustUseProperty U, // attributeName null, // attributeNamespace !1, // sanitizeURL !1 ); }), ["contentEditable", "draggable", "spellCheck", "value"].forEach(function(g) { an[g] = new Pn( g, bt, !1, // mustUseProperty g.toLowerCase(), // attributeName null, // attributeNamespace !1, // sanitizeURL !1 ); }), ["autoReverse", "externalResourcesRequired", "focusable", "preserveAlpha"].forEach(function(g) { an[g] = new Pn( g, bt, !1, // mustUseProperty g, // attributeName null, // attributeNamespace !1, // sanitizeURL !1 ); }), [ "allowFullScreen", "async", // Note: there is a special case that prevents it from being written to the DOM // on the client side because the browsers are inconsistent. Instead we call focus(). "autoFocus", "autoPlay", "controls", "default", "defer", "disabled", "disablePictureInPicture", "disableRemotePlayback", "formNoValidate", "hidden", "loop", "noModule", "noValidate", "open", "playsInline", "readOnly", "required", "reversed", "scoped", "seamless", // Microdata "itemScope" ].forEach(function(g) { an[g] = new Pn( g, wt, !1, // mustUseProperty g.toLowerCase(), // attributeName null, // attributeNamespace !1, // sanitizeURL !1 ); }), [ "checked", // Note: `option.selected` is not updated if `select.multiple` is // disabled with `removeAttribute`. We have special logic for handling this. "multiple", "muted", "selected" // NOTE: if you add a camelCased prop to this list, // you'll need to set attributeName to name.toLowerCase() // instead in the assignment below. ].forEach(function(g) { an[g] = new Pn( g, wt, !0, // mustUseProperty g, // attributeName null, // attributeNamespace !1, // sanitizeURL !1 ); }), [ "capture", "download" // NOTE: if you add a camelCased prop to this list, // you'll need to set attributeName to name.toLowerCase() // instead in the assignment below. ].forEach(function(g) { an[g] = new Pn( g, Tt, !1, // mustUseProperty g, // attributeName null, // attributeNamespace !1, // sanitizeURL !1 ); }), [ "cols", "rows", "size", "span" // NOTE: if you add a camelCased prop to this list, // you'll need to set attributeName to name.toLowerCase() // instead in the assignment below. ].forEach(function(g) { an[g] = new Pn( g, Qt, !1, // mustUseProperty g, // attributeName null, // attributeNamespace !1, // sanitizeURL !1 ); }), ["rowSpan", "start"].forEach(function(g) { an[g] = new Pn( g, lr, !1, // mustUseProperty g.toLowerCase(), // attributeName null, // attributeNamespace !1, // sanitizeURL !1 ); }); var rr = /[\-\:]([a-z])/g, Sr = function(g) { return g[1].toUpperCase(); }; [ "accent-height", "alignment-baseline", "arabic-form", "baseline-shift", "cap-height", "clip-path", "clip-rule", "color-interpolation", "color-interpolation-filters", "color-profile", "color-rendering", "dominant-baseline", "enable-background", "fill-opacity", "fill-rule", "flood-color", "flood-opacity", "font-family", "font-size", "font-size-adjust", "font-stretch", "font-style", "font-variant", "font-weight", "glyph-name", "glyph-orientation-horizontal", "glyph-orientation-vertical", "horiz-adv-x", "horiz-origin-x", "image-rendering", "letter-spacing", "lighting-color", "marker-end", "marker-mid", "marker-start", "overline-position", "overline-thickness", "paint-order", "panose-1", "pointer-events", "rendering-intent", "shape-rendering", "stop-color", "stop-opacity", "strikethrough-position", "strikethrough-thickness", "stroke-dasharray", "stroke-dashoffset", "stroke-linecap", "stroke-linejoin", "stroke-miterlimit", "stroke-opacity", "stroke-width", "text-anchor", "text-decoration", "text-rendering", "underline-position", "underline-thickness", "unicode-bidi", "unicode-range", "units-per-em", "v-alphabetic", "v-hanging", "v-ideographic", "v-mathematical", "vector-effect", "vert-adv-y", "vert-origin-x", "vert-origin-y", "word-spacing", "writing-mode", "xmlns:xlink", "x-height" // NOTE: if you add a camelCased prop to this list, // you'll need to set attributeName to name.toLowerCase() // instead in the assignment below. ].forEach(function(g) { var x = g.replace(rr, Sr); an[x] = new Pn( x, ut, !1, // mustUseProperty g, null, // attributeNamespace !1, // sanitizeURL !1 ); }), [ "xlink:actuate", "xlink:arcrole", "xlink:role", "xlink:show", "xlink:title", "xlink:type" // NOTE: if you add a camelCased prop to this list, // you'll need to set attributeName to name.toLowerCase() // instead in the assignment below. ].forEach(function(g) { var x = g.replace(rr, Sr); an[x] = new Pn( x, ut, !1, // mustUseProperty g, "http://www.w3.org/1999/xlink", !1, // sanitizeURL !1 ); }), [ "xml:base", "xml:lang", "xml:space" // NOTE: if you add a camelCased prop to this list, // you'll need to set attributeName to name.toLowerCase() // instead in the assignment below. ].forEach(function(g) { var x = g.replace(rr, Sr); an[x] = new Pn( x, ut, !1, // mustUseProperty g, "http://www.w3.org/XML/1998/namespace", !1, // sanitizeURL !1 ); }), ["tabIndex", "crossOrigin"].forEach(function(g) { an[g] = new Pn( g, ut, !1, // mustUseProperty g.toLowerCase(), // attributeName null, // attributeNamespace !1, // sanitizeURL !1 ); }); var nr = "xlinkHref"; an[nr] = new Pn( "xlinkHref", ut, !1, // mustUseProperty "xlink:href", "http://www.w3.org/1999/xlink", !0, // sanitizeURL !1 ), ["src", "href", "action", "formAction"].forEach(function(g) { an[g] = new Pn( g, ut, !1, // mustUseProperty g.toLowerCase(), // attributeName null, // attributeNamespace !0, // sanitizeURL !0 ); }); var Er = /^[\u0000-\u001F ]*j[\r\n\t]*a[\r\n\t]*v[\r\n\t]*a[\r\n\t]*s[\r\n\t]*c[\r\n\t]*r[\r\n\t]*i[\r\n\t]*p[\r\n\t]*t[\r\n\t]*\:/i, Rn = !1; function Vn(g) { !Rn && Er.test(g) && (Rn = !0, c("A future version of React will block javascript: URLs as a security precaution. Use event handlers instead if you can. If you need to generate unsafe HTML try using dangerouslySetInnerHTML instead. React was passed %s.", JSON.stringify(g))); } function ki(g, x, U, F) { if (F.mustUseProperty) { var K = F.propertyName; return g[K]; } else { F.sanitizeURL && Vn("" + U); var oe = F.attributeName, ce = null; if (F.type === Tt) { if (g.hasAttribute(oe)) { var Te = g.getAttribute(oe); return Te === "" ? !0 : nn(x, U, F, !1) ? Te : Te === "" + U ? U : Te; } } else if (g.hasAttribute(oe)) { if (nn(x, U, F, !1)) return g.getAttribute(oe); if (F.type === wt) return U; ce = g.getAttribute(oe); } return nn(x, U, F, !1) ? ce === null ? U : ce : ce === "" + U ? U : ce; } } function as(g, x, U) { { if (!Gt(x)) return; if (AX(U)) return U; if (!g.hasAttribute(x)) return U === void 0 ? void 0 : null; var F = g.getAttribute(x); return F === "" + U ? U : F; } } function Rr(g, x, U, F) { var K = sn(x); if (!xr(x, K, F)) { if (nn(x, U, K, F) && (U = null), F || K === null) { if (Gt(x)) { var oe = x; U === null ? g.removeAttribute(oe) : g.setAttribute(oe, "" + U); } return; } var ce = K.mustUseProperty; if (ce) { var Te = K.propertyName; if (U === null) { var Ce = K.type; g[Te] = Ce === wt ? !1 : ""; } else g[Te] = U; return; } var tt = K.attributeName, ft = K.attributeNamespace; if (U === null) g.removeAttribute(tt); else { var Rt = K.type, ct; Rt === wt || Rt === Tt && U === !0 ? ct = "" : (ct = "" + U, K.sanitizeURL && Vn(ct.toString())), ft ? g.setAttributeNS(ft, tt, ct) : g.setAttribute(tt, ct); } } } var Ii = 60103, is = 60106, ji = 60107, _i = 60108, oi = 60114, vr = 60109, Fr = 60110, pr = 60112, ds = 60113, it = 60120, $t = 60115, Ur = 60116, $e = 60121, Kt = 60119, cr = 60128, Vr = 60129, Lr = 60130, pi = 60131; if (typeof Symbol == "function" && Symbol.for) { var zr = Symbol.for; Ii = zr("react.element"), is = zr("react.portal"), ji = zr("react.fragment"), _i = zr("react.strict_mode"), oi = zr("react.profiler"), vr = zr("react.provider"), Fr = zr("react.context"), pr = zr("react.forward_ref"), ds = zr("react.suspense"), it = zr("react.suspense_list"), $t = zr("react.memo"), Ur = zr("react.lazy"), $e = zr("react.block"), zr("react.server.block"), zr("react.fundamental"), Kt = zr("react.scope"), cr = zr("react.opaque.id"), Vr = zr("react.debug_trace_mode"), Lr = zr("react.offscreen"), pi = zr("react.legacy_hidden"); } var Cn = typeof Symbol == "function" && Symbol.iterator, Mi = "@@iterator"; function li(g) { if (g === null || typeof g != "object") return null; var x = Cn && g[Cn] || g[Mi]; return typeof x == "function" ? x : null; } var Xs = 0, Ri, j1, w1, di, jf, B6, W6; function S6() { } S6.__reactDisabledLog = !0; function qa() { { if (Xs === 0) { Ri = console.log, j1 = console.info, w1 = console.warn, di = console.error, jf = console.group, B6 = console.groupCollapsed, W6 = console.groupEnd; var g = { configurable: !0, enumerable: !0, value: S6, writable: !0 }; Object.defineProperties(console, { info: g, log: g, warn: g, error: g, group: g, groupCollapsed: g, groupEnd: g }); } Xs++; } } function m1() { { if (Xs--, Xs === 0) { var g = { configurable: !0, enumerable: !0, writable: !0 }; Object.defineProperties(console, { log: v({}, g, { value: Ri }), info: v({}, g, { value: j1 }), warn: v({}, g, { value: w1 }), error: v({}, g, { value: di }), group: v({}, g, { value: jf }), groupCollapsed: v({}, g, { value: B6 }), groupEnd: v({}, g, { value: W6 }) }); } Xs < 0 && c("disabledDepth fell below zero. This is a bug in React. Please file an issue."); } } var Ho = P.ReactCurrentDispatcher, X9; function $o(g, x, U) { { if (X9 === void 0) try { throw Error(); } catch (K) { var F = K.stack.trim().match(/\n( *(at )?)/); X9 = F && F[1] || ""; } return ` ` + X9 + g; } } var T9 = !1, q9; { var wf = typeof WeakMap == "function" ? WeakMap : Map; q9 = new wf(); } function mf(g, x) { if (!g || T9) return ""; { var U = q9.get(g); if (U !== void 0) return U; } var F; T9 = !0; var K = Error.prepareStackTrace; Error.prepareStackTrace = void 0; var oe; oe = Ho.current, Ho.current = null, qa(); try { if (x) { var ce = function() { throw Error(); }; if (Object.defineProperty(ce.prototype, "props", { set: function() { throw Error(); } }), typeof Reflect == "object" && Reflect.construct) { try { Reflect.construct(ce, []); } catch (wr) { F = wr; } Reflect.construct(g, [], ce); } else { try { ce.call(); } catch (wr) { F = wr; } g.call(ce.prototype); } } else { try { throw Error(); } catch (wr) { F = wr; } g(); } } catch (wr) { if (wr && F && typeof wr.stack == "string") { for (var Te = wr.stack.split(` `), Ce = F.stack.split(` `), tt = Te.length - 1, ft = Ce.length - 1; tt >= 1 && ft >= 0 && Te[tt] !== Ce[ft]; ) ft--; for (; tt >= 1 && ft >= 0; tt--, ft--) if (Te[tt] !== Ce[ft]) { if (tt !== 1 || ft !== 1) do if (tt--, ft--, ft < 0 || Te[tt] !== Ce[ft]) { var Rt = ` ` + Te[tt].replace(" at new ", " at "); return typeof g == "function" && q9.set(g, Rt), Rt; } while (tt >= 1 && ft >= 0); break; } } } finally { T9 = !1, Ho.current = oe, m1(), Error.prepareStackTrace = K; } var ct = g ? g.displayName || g.name : "", Ft = ct ? $o(ct) : ""; return typeof g == "function" && q9.set(g, Ft), Ft; } function B1(g, x, U) { return mf(g, !0); } function h1(g, x, U) { return mf(g, !1); } function U6(g) { var x = g.prototype; return !!(x && x.isReactComponent); } function b9(g, x, U) { if (g == null) return ""; if (typeof g == "function") return mf(g, U6(g)); if (typeof g == "string") return $o(g); switch (g) { case ds: return $o("Suspense"); case it: return $o("SuspenseList"); } if (typeof g == "object") switch (g.$$typeof) { case pr: return h1(g.render); case $t: return b9(g.type, x, U); case $e: return h1(g._render); case Ur: { var F = g, K = F._payload, oe = F._init; try { return b9(oe(K), x, U); } catch { } } } return ""; } function J9(g) { switch (g._debugOwner && g._debugOwner.type, g._debugSource, g.tag) { case m: return $o(g.type); case fe: return $o("Lazy"); case ee: return $o("Suspense"); case L: return $o("SuspenseList"); case T: case b: case te: return h1(g.type); case O: return h1(g.type.render); case Pe: return h1(g.type._render); case q: return B1(g.type); default: return ""; } } function OA(g) { try { var x = "", U = g; do x += J9(U), U = U.return; while (U); return x; } catch (F) { return ` Error generating stack: ` + F.message + ` ` + F.stack; } } function z1(g, x, U) { var F = x.displayName || x.name || ""; return g.displayName || (F !== "" ? U + "(" + F + ")" : U); } function Zd(g) { return g.displayName || "Context"; } function Zr(g) { if (g == null) return null; if (typeof g.tag == "number" && c("Received an unexpected object in getComponentName(). This is likely a bug in React. Please file an issue."), typeof g == "function") return g.displayName || g.name || null; if (typeof g == "string") return g; switch (g) { case ji: return "Fragment"; case is: return "Portal"; case oi: return "Profiler"; case _i: return "StrictMode"; case ds: return "Suspense"; case it: return "SuspenseList"; } if (typeof g == "object") switch (g.$$typeof) { case Fr: var x = g; return Zd(x) + ".Consumer"; case vr: var U = g; return Zd(U._context) + ".Provider"; case pr: return z1(g, g.render, "ForwardRef"); case $t: return Zr(g.type); case $e: return Zr(g._render); case Ur: { var F = g, K = F._payload, oe = F._init; try { return Zr(oe(K)); } catch { return null; } } } return null; } var Ts = P.ReactDebugCurrentFrame, go = null, G1 = !1; function z9() { { if (go === null) return null; var g = go._debugOwner; if (g !== null && typeof g < "u") return Zr(g.type); } return null; } function Li() { return go === null ? "" : OA(go); } function Fa() { Ts.getCurrentStack = null, go = null, G1 = !1; } function Sa(g) { Ts.getCurrentStack = Li, go = g, G1 = !1; } function Fn(g) { G1 = g; } function I6() { return G1; } function $i(g) { return "" + g; } function Ga(g) { switch (typeof g) { case "boolean": case "number": case "object": case "string": case "undefined": return g; default: return ""; } } var e1 = { button: !0, checkbox: !0, image: !0, hidden: !0, radio: !0, reset: !0, submit: !0 }; function Lf(g, x) { e1[x.type] || x.onChange || x.onInput || x.readOnly || x.disabled || x.value == null || c("You provided a `value` prop to a form field without an `onChange` handler. This will render a read-only field. If the field should be mutable use `defaultValue`. Otherwise, set either `onChange` or `readOnly`."), x.onChange || x.readOnly || x.disabled || x.checked == null || c("You provided a `checked` prop to a form field without an `onChange` handler. This will render a read-only field. If the field should be mutable use `defaultChecked`. Otherwise, set either `onChange` or `readOnly`."); } function Ua(g) { var x = g.type, U = g.nodeName; return U && U.toLowerCase() === "input" && (x === "checkbox" || x === "radio"); } function vs(g) { return g._valueTracker; } function R6(g) { g._valueTracker = null; } function Bf(g) { var x = ""; return g && (Ua(g) ? x = g.checked ? "true" : "false" : x = g.value), x; } function Ie(g) { var x = Ua(g) ? "checked" : "value", U = Object.getOwnPropertyDescriptor(g.constructor.prototype, x), F = "" + g[x]; if (!(g.hasOwnProperty(x) || typeof U > "u" || typeof U.get != "function" || typeof U.set != "function")) { var K = U.get, oe = U.set; Object.defineProperty(g, x, { configurable: !0, get: function() { return K.call(this); }, set: function(Te) { F = "" + Te, oe.call(this, Te); } }), Object.defineProperty(g, x, { enumerable: U.enumerable }); var ce = { getValue: function() { return F; }, setValue: function(Te) { F = "" + Te; }, stopTracking: function() { R6(g), delete g[x]; } }; return ce; } } function Pt(g) { vs(g) || (g._valueTracker = Ie(g)); } function Et(g) { if (!g) return !1; var x = vs(g); if (!x) return !0; var U = x.getValue(), F = Bf(g); return F !== U ? (x.setValue(F), !0) : !1; } function er(g) { if (g = g || (typeof document < "u" ? document : void 0), typeof g > "u") return null; try { return g.activeElement || g.body; } catch { return g.body; } } var _r = !1, Xi = !1, Mn = !1, _n = !1; function Ps(g) { var x = g.type === "checkbox" || g.type === "radio"; return x ? g.checked != null : g.value != null; } function qi(g, x) { var U = g, F = x.checked, K = v({}, x, { defaultChecked: void 0, defaultValue: void 0, value: void 0, checked: F ?? U._wrapperState.initialChecked }); return K; } function Gs(g, x) { Lf("input", x), x.checked !== void 0 && x.defaultChecked !== void 0 && !Xi && (c("%s contains an input of type %s with both checked and defaultChecked props. Input elements must be either controlled or uncontrolled (specify either the checked prop, or the defaultChecked prop, but not both). Decide between using a controlled or uncontrolled input element and remove one of these props. More info: https://reactjs.org/link/controlled-components", z9() || "A component", x.type), Xi = !0), x.value !== void 0 && x.defaultValue !== void 0 && !_r && (c("%s contains an input of type %s with both value and defaultValue props. Input elements must be either controlled or uncontrolled (specify either the value prop, or the defaultValue prop, but not both). Decide between using a controlled or uncontrolled input element and remove one of these props. More info: https://reactjs.org/link/controlled-components", z9() || "A component", x.type), _r = !0); var U = g, F = x.defaultValue == null ? "" : x.defaultValue; U._wrapperState = { initialChecked: x.checked != null ? x.checked : x.defaultChecked, initialValue: Ga(x.value != null ? x.value : F), controlled: Ps(x) }; } function ks(g, x) { var U = g, F = x.checked; F != null && Rr(U, "checked", F, !1); } function Na(g, x) { var U = g; { var F = Ps(x); !U._wrapperState.controlled && F && !_n && (c("A component is changing an uncontrolled input to be controlled. This is likely caused by the value changing from undefined to a defined value, which should not happen. Decide between using a controlled or uncontrolled input element for the lifetime of the component. More info: https://reactjs.org/link/controlled-components"), _n = !0), U._wrapperState.controlled && !F && !Mn && (c("A component is changing a controlled input to be uncontrolled. This is likely caused by the value changing from a defined to undefined, which should not happen. Decide between using a controlled or uncontrolled input element for the lifetime of the component. More info: https://reactjs.org/link/controlled-components"), Mn = !0); } ks(g, x); var K = Ga(x.value), oe = x.type; if (K != null) oe === "number" ? (K === 0 && U.value === "" || // We explicitly want to coerce to number here if possible. // eslint-disable-next-line U.value != K) && (U.value = $i(K)) : U.value !== $i(K) && (U.value = $i(K)); else if (oe === "submit" || oe === "reset") { U.removeAttribute("value"); return; } x.hasOwnProperty("value") ? W1(U, x.type, K) : x.hasOwnProperty("defaultValue") && W1(U, x.type, Ga(x.defaultValue)), x.checked == null && x.defaultChecked != null && (U.defaultChecked = !!x.defaultChecked); } function Ao(g, x, U) { var F = g; if (x.hasOwnProperty("value") || x.hasOwnProperty("defaultValue")) { var K = x.type, oe = K === "submit" || K === "reset"; if (oe && (x.value === void 0 || x.value === null)) return; var ce = $i(F._wrapperState.initialValue); U || ce !== F.value && (F.value = ce), F.defaultValue = ce; } var Te = F.name; Te !== "" && (F.name = ""), F.defaultChecked = !F.defaultChecked, F.defaultChecked = !!F._wrapperState.initialChecked, Te !== "" && (F.name = Te); } function Ro(g, x) { var U = g; Na(U, x), oA(U, x); } function oA(g, x) { var U = x.name; if (x.type === "radio" && U != null) { for (var F = g; F.parentNode; ) F = F.parentNode; for (var K = F.querySelectorAll("input[name=" + JSON.stringify("" + U) + '][type="radio"]'), oe = 0; oe < K.length; oe++) { var ce = K[oe]; if (!(ce === g || ce.form !== g.form)) { var Te = lX(ce); if (!Te) throw Error("ReactDOMInput: Mixing React and non-React radio inputs with the same `name` is not supported."); Et(ce), Na(ce, Te); } } } } function W1(g, x, U) { // Focused number inputs synchronize on blur. See ChangeEventPlugin.js (x !== "number" || er(g.ownerDocument) !== g) && (U == null ? g.defaultValue = $i(g._wrapperState.initialValue) : g.defaultValue !== $i(U) && (g.defaultValue = $i(U))); } var je = !1, dt = !1; function kt(g) { var x = ""; return d.Children.forEach(g, function(U) { U != null && (x += U); }), x; } function _t(g, x) { typeof x.children == "object" && x.children !== null && d.Children.forEach(x.children, function(U) { U != null && (typeof U == "string" || typeof U == "number" || typeof U.type == "string" && (dt || (dt = !0, c("Only strings and numbers are supported as